15 "github.com/cespare/cp"
16 "github.com/davecgh/go-spew/spew"
17 "github.com/vapor/crypto/ed25519/chainkd"
21 cachetestDir, _ = filepath.Abs(filepath.Join("testdata", "keystore"))
22 cachetestKeys = []XPub{
25 File: filepath.Join(cachetestDir, "UTC--2017-09-13T07-11-07.863320100Z--bm1pktmny6q69dlqulja2p2ja28k2vd6wvqpk5r76a"),
29 File: filepath.Join(cachetestDir, "aaa"),
33 File: filepath.Join(cachetestDir, "zzz"),
38 func TestWatchNewFile(t *testing.T) {
41 dir, kc := tmpManager(t)
42 // defer os.RemoveAll(dir)
44 // Ensure the watcher is started before adding any files.
46 time.Sleep(200 * time.Millisecond)
48 wantKeystores := make([]XPub, len(cachetestKeys))
49 for i := range cachetestKeys {
51 a.File = filepath.Join(dir, filepath.Base(a.File))
53 if err := cp.CopyFile(a.File, cachetestKeys[i].File); err != nil {
58 // kc should see the keys.
60 for d := 200 * time.Millisecond; d < 5*time.Second; d *= 2 {
62 if reflect.DeepEqual(list, wantKeystores) {
67 t.Errorf("got %s, want %s", spew.Sdump(list), spew.Sdump(wantKeystores))
70 func TestWatchNoDir(t *testing.T) {
73 // Create am but not the directory that it watches.
74 rand.Seed(time.Now().UnixNano())
75 dir := filepath.Join(os.TempDir(), fmt.Sprintf("bytom-keystore-watch-test-%d-%d", os.Getpid(), rand.Int()))
76 kc := newKeyCache(dir)
79 t.Error("initial account list not empty:", list)
81 time.Sleep(100 * time.Millisecond)
83 // Create the directory and copy a key file into it.
84 os.MkdirAll(dir, 0700)
85 defer os.RemoveAll(dir)
86 file := filepath.Join(dir, "aaa")
87 if err := cp.CopyFile(file, cachetestKeys[0].File); err != nil {
91 // am should see the account.
92 wantKeys := []XPub{cachetestKeys[0]}
93 wantKeys[0].File = file
94 for d := 200 * time.Millisecond; d < 8*time.Second; d *= 2 {
96 if reflect.DeepEqual(list, wantKeys) {
101 t.Errorf("\ngot %v\nwant %v", list, wantKeys)
104 func TestCacheInitialReload(t *testing.T) {
105 cache := newKeyCache(cachetestDir)
107 if !reflect.DeepEqual(keys, cachetestKeys) {
108 t.Fatalf("got initial accounts: %swant %s", spew.Sdump(keys), spew.Sdump(cachetestKeys))
112 func TestCacheAddDeleteOrder(t *testing.T) {
113 cache := newKeyCache("testdata/no-such-dir")
114 cache.watcher.running = true // prevent unexpected reloads
115 r := rand.New(rand.NewSource(time.Now().UnixNano()))
119 Alias: "bm1pvheagygs9d72stp79u9vduhmdyjpnvud0y89y7",
121 XPub: tmpPubkeys(t, r),
124 Alias: "bm1pyk3qny8gzem6p4fx8t5d344tnldguv8lvx2aww",
126 XPub: tmpPubkeys(t, r),
129 Alias: "bm1p6s0ckxrudy7hqht4n5fhcs4gp69krv3c84jn9x",
130 File: "zzzzzz-the-very-last-one.keyXXX",
131 XPub: tmpPubkeys(t, r),
134 Alias: "bm1p7xkfhsw50y44t63mk0dfxxkvuyg6t3s0r6xs54",
135 File: "SOMETHING.key",
136 XPub: tmpPubkeys(t, r),
139 Alias: "bm1peu9ql7x8c7aeca60j40sg5w4kylpf7l3jmau0g",
140 File: "UTC--2016-03-22T12-57-55.920751759Z--bm1peu9ql7x8c7aeca60j40sg5w4kylpf7l3jmau0g",
141 XPub: tmpPubkeys(t, r),
144 Alias: "bm1p0s68e4ggp0vy5ue2lztsxvl2smpnqp9al8jyvh",
146 XPub: tmpPubkeys(t, r),
149 Alias: "bm1pjq8ttfl7ppqtcc5qqff0s45p7ew9l9pjmlu5xw",
151 XPub: tmpPubkeys(t, r),
154 for _, a := range keys {
157 // Add some of them twice to check that they don't get reinserted.
161 // Check that the account list is sorted by filename.
162 wantKeys := make([]XPub, len(keys))
164 sort.Sort(keysByFile(wantKeys))
167 if !reflect.DeepEqual(list, wantKeys) {
168 t.Fatalf("got keys: %s\nwant %s", spew.Sdump(keys), spew.Sdump(wantKeys))
171 for _, a := range keys {
172 if !cache.hasKey(a.XPub) {
173 t.Errorf("expected hashKey(%x) to return true", a.XPub)
176 // Delete a few keys from the cache.
177 for i := 0; i < len(keys); i += 2 {
178 cache.delete(wantKeys[i])
180 cache.delete(XPub{Alias: "bm1pug2xpcvpzepdf0paulnndhpxtpjvre8ypd0jtj", File: "something", XPub: tmpPubkeys(t, r)})
182 // Check content again after deletion.
183 wantKeysAfterDelete := []XPub{
189 if !reflect.DeepEqual(list, wantKeysAfterDelete) {
190 t.Fatalf("got keys after delete: %s\nwant %s", spew.Sdump(list), spew.Sdump(wantKeysAfterDelete))
192 for _, a := range wantKeysAfterDelete {
193 if !cache.hasKey(a.XPub) {
194 t.Errorf("expected hasKey(%x) to return true", a.XPub)
197 if cache.hasKey(wantKeys[0].XPub) {
198 t.Errorf("expected hasKey(%x) to return false", wantKeys[0].XPub)
202 func TestCacheFind(t *testing.T) {
203 dir := filepath.Join("testdata", "dir")
204 cache := newKeyCache(dir)
205 cache.watcher.running = true // prevent unexpected reloads
206 r := rand.New(rand.NewSource(time.Now().UnixNano()))
208 dup := tmpPubkeys(t, r)
211 Alias: "bm1pmv9kg68j3edvqrv62lxllev4ugjv0zf6g5pwf6",
212 File: filepath.Join(dir, "a.key"),
213 XPub: tmpPubkeys(t, r),
216 Alias: "bm1ptspg4x6kjjp642gdpzan0ynq9zr7z4m34nqpet",
217 File: filepath.Join(dir, "b.key"),
218 XPub: tmpPubkeys(t, r),
221 Alias: "bm1pmlpy0946zsvdg29v80gw0mkq2n0ghkg0fpmhav",
222 File: filepath.Join(dir, "c.key"),
226 Alias: "bm1pmlpy0946zsvdg29v80gw0mkq2n0ghkg0fpmhav",
227 File: filepath.Join(dir, "c2.key"),
231 for _, a := range keys {
236 Alias: "bm1pu2vmgps4d9e3mrsuzp58w777apky4rjgn5rn9e",
237 File: filepath.Join(dir, "something"),
238 XPub: tmpPubkeys(t, r),
246 {Query: XPub{XPub: keys[0].XPub}, WantResult: keys[0]},
248 {Query: XPub{File: keys[0].File}, WantResult: keys[0]},
250 {Query: XPub{File: filepath.Base(keys[0].File)}, WantResult: keys[0]},
252 {Query: keys[0], WantResult: keys[0]},
253 // ambiguous xpub, tie resolved by file
254 {Query: keys[2], WantResult: keys[2]},
255 // ambiguous xpub error
257 Query: XPub{XPub: keys[2].XPub},
258 WantError: &AmbiguousKeyError{
259 Pubkey: hex.EncodeToString(keys[2].XPub[:]),
260 Matches: []XPub{keys[2], keys[3]},
264 {Query: nomatchKey, WantError: ErrLoadKey},
265 {Query: XPub{File: nomatchKey.File}, WantError: ErrLoadKey},
266 {Query: XPub{File: filepath.Base(nomatchKey.File)}, WantError: ErrLoadKey},
267 {Query: XPub{XPub: nomatchKey.XPub}, WantError: ErrLoadKey},
269 for i, test := range tests {
270 a, err := cache.find(test.Query)
271 if !reflect.DeepEqual(err, test.WantError) {
272 t.Errorf("test %d: error mismatch for query %v\ngot %q\nwant %q", i, test.Query, err, test.WantError)
275 if a != test.WantResult {
276 t.Errorf("test %d: result mismatch for query %v\ngot %v\nwant %v", i, test.Query, a, test.WantResult)
282 func tmpManager(t *testing.T) (string, *keyCache) {
283 d, err := ioutil.TempDir("", "bytom-keystore-test")
287 return d, newKeyCache(d)
290 func tmpPubkeys(t *testing.T, r *rand.Rand) chainkd.XPub {
292 var xpub chainkd.XPub
293 pick := "0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"
294 bytes := []byte(pick)
297 for i := 0; i < 64; i++ {
298 result = append(result, bytes[r.Intn(len(bytes))])
300 copy(xpub[:], result[:])