15 "github.com/bytom/crypto/ed25519/chainkd"
16 "github.com/cespare/cp"
17 "github.com/davecgh/go-spew/spew"
22 cachetestDir, _ = filepath.Abs(filepath.Join("testdata", "keystore"))
23 cachetestKeys = []XPub{
26 File: filepath.Join(cachetestDir, "UTC--2017-09-13T07-11-07.863320100Z--bm1pktmny6q69dlqulja2p2ja28k2vd6wvqpk5r76a"),
30 File: filepath.Join(cachetestDir, "aaa"),
34 File: filepath.Join(cachetestDir, "zzz"),
39 func TestWatchNewFile(t *testing.T) {
42 dir, kc := tmpManager(t)
43 // defer os.RemoveAll(dir)
45 // Ensure the watcher is started before adding any files.
47 time.Sleep(200 * time.Millisecond)
49 wantKeystores:= make([]XPub, len(cachetestKeys))
50 for i := range cachetestKeys {
52 a.File = filepath.Join(dir, filepath.Base(a.File))
54 if err := cp.CopyFile(a.File, cachetestKeys[i].File); err != nil {
59 // kc should see the keys.
61 for d := 200 * time.Millisecond; d < 5*time.Second; d *= 2 {
63 if reflect.DeepEqual(list, wantKeystores) {
68 t.Errorf("got %s, want %s", spew.Sdump(list), spew.Sdump(wantKeystores))
72 func TestWatchNoDir(t *testing.T) {
75 // Create am but not the directory that it watches.
76 rand.Seed(time.Now().UnixNano())
77 dir := filepath.Join(os.TempDir(), fmt.Sprintf("bytom-keystore-watch-test-%d-%d", os.Getpid(), rand.Int()))
78 kc := newKeyCache(dir)
81 t.Error("initial account list not empty:", list)
83 time.Sleep(100 * time.Millisecond)
85 // Create the directory and copy a key file into it.
86 os.MkdirAll(dir, 0700)
87 defer os.RemoveAll(dir)
88 file := filepath.Join(dir, "aaa")
89 if err := cp.CopyFile(file, cachetestKeys[0].File); err != nil {
93 // am should see the account.
94 wantKeys := []XPub{cachetestKeys[0]}
95 wantKeys[0].File = file
96 for d := 200 * time.Millisecond; d < 8*time.Second; d *= 2 {
98 if reflect.DeepEqual(list, wantKeys) {
103 t.Errorf("\ngot %v\nwant %v", list, wantKeys)
106 func TestCacheInitialReload(t *testing.T) {
107 cache := newKeyCache(cachetestDir)
109 if !reflect.DeepEqual(keys, cachetestKeys) {
110 t.Fatalf("got initial accounts: %swant %s", spew.Sdump(keys), spew.Sdump(cachetestKeys))
115 func TestCacheAddDeleteOrder(t *testing.T) {
116 cache := newKeyCache("testdata/no-such-dir")
117 cache.watcher.running = true // prevent unexpected reloads
118 r := rand.New(rand.NewSource(time.Now().UnixNano()))
122 Alias: "bm1pvheagygs9d72stp79u9vduhmdyjpnvud0y89y7",
124 XPub: tmpPubkeys(t, r),
127 Alias: "bm1pyk3qny8gzem6p4fx8t5d344tnldguv8lvx2aww",
129 XPub: tmpPubkeys(t, r),
132 Alias: "bm1p6s0ckxrudy7hqht4n5fhcs4gp69krv3c84jn9x",
133 File: "zzzzzz-the-very-last-one.keyXXX",
134 XPub: tmpPubkeys(t, r),
138 Alias: "bm1p7xkfhsw50y44t63mk0dfxxkvuyg6t3s0r6xs54",
139 File: "SOMETHING.key",
140 XPub: tmpPubkeys(t, r),
144 Alias: "bm1peu9ql7x8c7aeca60j40sg5w4kylpf7l3jmau0g",
145 File: "UTC--2016-03-22T12-57-55.920751759Z--bm1peu9ql7x8c7aeca60j40sg5w4kylpf7l3jmau0g",
146 XPub: tmpPubkeys(t, r),
150 Alias: "bm1p0s68e4ggp0vy5ue2lztsxvl2smpnqp9al8jyvh",
152 XPub: tmpPubkeys(t, r),
156 Alias: "bm1pjq8ttfl7ppqtcc5qqff0s45p7ew9l9pjmlu5xw",
158 XPub: tmpPubkeys(t, r),
161 for _, a := range keys {
164 // Add some of them twice to check that they don't get reinserted.
168 // Check that the account list is sorted by filename.
169 wantKeys := make([]XPub, len(keys))
170 copy(wantKeys , keys)
171 sort.Sort(keysByFile(wantKeys))
174 if !reflect.DeepEqual(list, wantKeys) {
175 t.Fatalf("got keys: %s\nwant %s", spew.Sdump(keys), spew.Sdump(wantKeys))
178 for _, a := range keys {
179 if !cache.hasKey(a.XPub) {
180 t.Errorf("expected hashKey(%x) to return true", a.XPub)
183 // Delete a few keys from the cache.
184 for i := 0; i < len(keys); i += 2 {
185 cache.delete(wantKeys[i])
187 cache.delete(XPub{Alias: "bm1pug2xpcvpzepdf0paulnndhpxtpjvre8ypd0jtj", File: "something", XPub:tmpPubkeys(t, r)})
189 // Check content again after deletion.
190 wantKeysAfterDelete := []XPub{
196 if !reflect.DeepEqual(list, wantKeysAfterDelete) {
197 t.Fatalf("got keys after delete: %s\nwant %s", spew.Sdump(list), spew.Sdump(wantKeysAfterDelete))
199 for _, a := range wantKeysAfterDelete {
200 if !cache.hasKey(a.XPub) {
201 t.Errorf("expected hasKey(%x) to return true", a.XPub)
204 if cache.hasKey(wantKeys[0].XPub) {
205 t.Errorf("expected hasKey(%x) to return false", wantKeys[0].XPub)
210 func TestCacheFind(t *testing.T) {
211 dir := filepath.Join("testdata", "dir")
212 cache := newKeyCache(dir)
213 cache.watcher.running = true // prevent unexpected reloads
214 r := rand.New(rand.NewSource(time.Now().UnixNano()))
216 dup := tmpPubkeys(t, r)
219 Alias: "bm1pmv9kg68j3edvqrv62lxllev4ugjv0zf6g5pwf6",
220 File: filepath.Join(dir, "a.key"),
221 XPub: tmpPubkeys(t, r),
225 Alias: "bm1ptspg4x6kjjp642gdpzan0ynq9zr7z4m34nqpet",
226 File: filepath.Join(dir, "b.key"),
227 XPub: tmpPubkeys(t, r),
230 Alias: "bm1pmlpy0946zsvdg29v80gw0mkq2n0ghkg0fpmhav",
231 File: filepath.Join(dir, "c.key"),
235 Alias: "bm1pmlpy0946zsvdg29v80gw0mkq2n0ghkg0fpmhav",
236 File: filepath.Join(dir, "c2.key"),
240 for _, a := range keys {
245 Alias: "bm1pu2vmgps4d9e3mrsuzp58w777apky4rjgn5rn9e",
246 File: filepath.Join(dir, "something"),
247 XPub: tmpPubkeys(t, r),
255 {Query: XPub{XPub: keys[0].XPub}, WantResult: keys[0]},
257 {Query: XPub{File: keys[0].File}, WantResult: keys[0]},
259 {Query: XPub{File: filepath.Base(keys[0].File)}, WantResult: keys[0]},
261 {Query: keys[0], WantResult: keys[0]},
262 // ambiguous xpub, tie resolved by file
263 {Query: keys[2], WantResult: keys[2]},
264 // ambiguous xpub error
266 Query: XPub{XPub: keys[2].XPub},
267 WantError: &AmbiguousKeyError{
268 Pubkey: hex.EncodeToString(keys[2].XPub[:]),
269 Matches: []XPub{keys[2], keys[3]},
273 {Query: nomatchKey, WantError: ErrNoKey},
274 {Query: XPub{File: nomatchKey.File}, WantError: ErrNoKey},
275 {Query: XPub{File: filepath.Base(nomatchKey.File)}, WantError: ErrNoKey},
276 {Query: XPub{XPub: nomatchKey.XPub}, WantError: ErrNoKey},
278 for i, test := range tests {
279 a, err := cache.find(test.Query)
280 if !reflect.DeepEqual(err, test.WantError) {
281 t.Errorf("test %d: error mismatch for query %v\ngot %q\nwant %q", i, test.Query, err, test.WantError)
284 if a != test.WantResult {
285 t.Errorf("test %d: result mismatch for query %v\ngot %v\nwant %v", i, test.Query, a, test.WantResult)
292 func tmpManager(t *testing.T) (string, *keyCache) {
293 d, err := ioutil.TempDir("", "bytom-keystore-test")
297 return d, newKeyCache(d)
301 func tmpPubkeys(t *testing.T, r *rand.Rand) chainkd.XPub {
303 var xpub chainkd.XPub
304 pick := "0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"
305 bytes := []byte(pick)
308 for i := 0; i < 64; i++ {
309 result = append(result, bytes[r.Intn(len(bytes))])
311 copy(xpub[:], result[:])