github.com/klaytn/klaytn@v1.10.2/networks/p2p/discover/database_test.go (about) 1 // Modifications Copyright 2018 The klaytn Authors 2 // Copyright 2015 The go-ethereum Authors 3 // This file is part of the go-ethereum library. 4 // 5 // The go-ethereum library is free software: you can redistribute it and/or modify 6 // it under the terms of the GNU Lesser General Public License as published by 7 // the Free Software Foundation, either version 3 of the License, or 8 // (at your option) any later version. 9 // 10 // The go-ethereum library is distributed in the hope that it will be useful, 11 // but WITHOUT ANY WARRANTY; without even the implied warranty of 12 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 13 // GNU Lesser General Public License for more details. 14 // 15 // You should have received a copy of the GNU Lesser General Public License 16 // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. 17 // 18 // This file is derived from p2p/discover/database_test.go (2018/06/04). 19 // Modified and improved for the klaytn development. 20 21 package discover 22 23 import ( 24 "bytes" 25 "io/ioutil" 26 "net" 27 "os" 28 "path/filepath" 29 "reflect" 30 "testing" 31 "time" 32 ) 33 34 var nodeDBKeyTests = []struct { 35 id NodeID 36 field string 37 key []byte 38 }{ 39 { 40 id: NodeID{}, 41 field: "version", 42 key: []byte{0x76, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e}, // field 43 }, 44 { 45 id: MustHexID("0x1dd9d65c4552b5eb43d5ad55a2ee3f56c6cbc1c64a5c8d659f51fcd51bace24351232b8d7821617d2b29b54b81cdefb9b3e9c37d7fd5f63270bcc9e1a6f6a439"), 46 field: ":discover", 47 key: []byte{ 48 0x6e, 0x3a, // prefix 49 0x1d, 0xd9, 0xd6, 0x5c, 0x45, 0x52, 0xb5, 0xeb, // node id 50 0x43, 0xd5, 0xad, 0x55, 0xa2, 0xee, 0x3f, 0x56, // 51 0xc6, 0xcb, 0xc1, 0xc6, 0x4a, 0x5c, 0x8d, 0x65, // 52 0x9f, 0x51, 0xfc, 0xd5, 0x1b, 0xac, 0xe2, 0x43, // 53 0x51, 0x23, 0x2b, 0x8d, 0x78, 0x21, 0x61, 0x7d, // 54 0x2b, 0x29, 0xb5, 0x4b, 0x81, 0xcd, 0xef, 0xb9, // 55 0xb3, 0xe9, 0xc3, 0x7d, 0x7f, 0xd5, 0xf6, 0x32, // 56 0x70, 0xbc, 0xc9, 0xe1, 0xa6, 0xf6, 0xa4, 0x39, // 57 0x3a, 0x64, 0x69, 0x73, 0x63, 0x6f, 0x76, 0x65, 0x72, // field 58 }, 59 }, 60 } 61 62 func TestNodeDBKeys(t *testing.T) { 63 for i, tt := range nodeDBKeyTests { 64 if key := makeKey(tt.id, tt.field); !bytes.Equal(key, tt.key) { 65 t.Errorf("make test %d: key mismatch: have 0x%x, want 0x%x", i, key, tt.key) 66 } 67 id, field := splitKey(tt.key) 68 if !bytes.Equal(id[:], tt.id[:]) { 69 t.Errorf("split test %d: id mismatch: have 0x%x, want 0x%x", i, id, tt.id) 70 } 71 if field != tt.field { 72 t.Errorf("split test %d: field mismatch: have 0x%x, want 0x%x", i, field, tt.field) 73 } 74 } 75 } 76 77 var nodeDBInt64Tests = []struct { 78 key []byte 79 value int64 80 }{ 81 {key: []byte{0x01}, value: 1}, 82 {key: []byte{0x02}, value: 2}, 83 {key: []byte{0x03}, value: 3}, 84 } 85 86 func TestNodeDBInt64(t *testing.T) { 87 db, _ := newNodeDB("", Version, NodeID{}) 88 defer db.close() 89 90 tests := nodeDBInt64Tests 91 for i := 0; i < len(tests); i++ { 92 // Insert the next value 93 if err := db.storeInt64(tests[i].key, tests[i].value); err != nil { 94 t.Errorf("test %d: failed to store value: %v", i, err) 95 } 96 // Check all existing and non existing values 97 for j := 0; j < len(tests); j++ { 98 num := db.fetchInt64(tests[j].key) 99 switch { 100 case j <= i && num != tests[j].value: 101 t.Errorf("test %d, item %d: value mismatch: have %v, want %v", i, j, num, tests[j].value) 102 case j > i && num != 0: 103 t.Errorf("test %d, item %d: value mismatch: have %v, want %v", i, j, num, 0) 104 } 105 } 106 } 107 } 108 109 func TestNodeDBFetchStore(t *testing.T) { 110 node := NewNode( 111 MustHexID("0x1dd9d65c4552b5eb43d5ad55a2ee3f56c6cbc1c64a5c8d659f51fcd51bace24351232b8d7821617d2b29b54b81cdefb9b3e9c37d7fd5f63270bcc9e1a6f6a439"), 112 net.IP{192, 168, 0, 1}, 113 30303, 114 30303, 115 nil, 116 NodeTypeUnknown, 117 ) 118 inst := time.Now() 119 num := 314 120 121 db, _ := newNodeDB("", Version, NodeID{}) 122 defer db.close() 123 124 // Check fetch/store operations on a node ping object 125 if stored := db.lastPing(node.ID); stored.Unix() != 0 { 126 t.Errorf("ping: non-existing object: %v", stored) 127 } 128 if err := db.updateLastPing(node.ID, inst); err != nil { 129 t.Errorf("ping: failed to update: %v", err) 130 } 131 if stored := db.lastPing(node.ID); stored.Unix() != inst.Unix() { 132 t.Errorf("ping: value mismatch: have %v, want %v", stored, inst) 133 } 134 // Check fetch/store operations on a node pong object 135 if stored := db.bondTime(node.ID); stored.Unix() != 0 { 136 t.Errorf("pong: non-existing object: %v", stored) 137 } 138 if err := db.updateBondTime(node.ID, inst); err != nil { 139 t.Errorf("pong: failed to update: %v", err) 140 } 141 if stored := db.bondTime(node.ID); stored.Unix() != inst.Unix() { 142 t.Errorf("pong: value mismatch: have %v, want %v", stored, inst) 143 } 144 // Check fetch/store operations on a node findnode-failure object 145 if stored := db.findFails(node.ID); stored != 0 { 146 t.Errorf("find-node fails: non-existing object: %v", stored) 147 } 148 if err := db.updateFindFails(node.ID, num); err != nil { 149 t.Errorf("find-node fails: failed to update: %v", err) 150 } 151 if stored := db.findFails(node.ID); stored != num { 152 t.Errorf("find-node fails: value mismatch: have %v, want %v", stored, num) 153 } 154 // Check fetch/store operations on an actual node object 155 if stored := db.node(node.ID); stored != nil { 156 t.Errorf("node: non-existing object: %v", stored) 157 } 158 if err := db.updateNode(node); err != nil { 159 t.Errorf("node: failed to update: %v", err) 160 } 161 if stored := db.node(node.ID); stored == nil { 162 t.Errorf("node: not found") 163 } else if !reflect.DeepEqual(stored, node) { 164 t.Errorf("node: data mismatch: have %v, want %v", stored, node) 165 } 166 } 167 168 var nodeDBSeedQueryNodes = []struct { 169 node *Node 170 pong time.Time 171 }{ 172 // This one should not be in the result set because its last 173 // pong time is too far in the past. 174 { 175 node: NewNode( 176 MustHexID("0x84d9d65c4552b5eb43d5ad55a2ee3f56c6cbc1c64a5c8d659f51fcd51bace24351232b8d7821617d2b29b54b81cdefb9b3e9c37d7fd5f63270bcc9e1a6f6a439"), 177 net.IP{127, 0, 0, 3}, 178 30303, 179 30303, 180 nil, 181 NodeTypeUnknown, 182 ), 183 pong: time.Now().Add(-3 * time.Hour), 184 }, 185 // This one shouldn't be in the result set because its 186 // nodeID is the local node's ID. 187 { 188 node: NewNode( 189 MustHexID("0x57d9d65c4552b5eb43d5ad55a2ee3f56c6cbc1c64a5c8d659f51fcd51bace24351232b8d7821617d2b29b54b81cdefb9b3e9c37d7fd5f63270bcc9e1a6f6a439"), 190 net.IP{127, 0, 0, 3}, 191 30303, 192 30303, 193 nil, 194 NodeTypeUnknown, 195 ), 196 pong: time.Now().Add(-4 * time.Second), 197 }, 198 199 // These should be in the result set. 200 { 201 node: NewNode( 202 MustHexID("0x22d9d65c4552b5eb43d5ad55a2ee3f56c6cbc1c64a5c8d659f51fcd51bace24351232b8d7821617d2b29b54b81cdefb9b3e9c37d7fd5f63270bcc9e1a6f6a439"), 203 net.IP{127, 0, 0, 1}, 204 30303, 205 30303, 206 nil, 207 NodeTypeUnknown, 208 ), 209 pong: time.Now().Add(-2 * time.Second), 210 }, 211 { 212 node: NewNode( 213 MustHexID("0x44d9d65c4552b5eb43d5ad55a2ee3f56c6cbc1c64a5c8d659f51fcd51bace24351232b8d7821617d2b29b54b81cdefb9b3e9c37d7fd5f63270bcc9e1a6f6a439"), 214 net.IP{127, 0, 0, 2}, 215 30303, 216 30303, 217 nil, 218 NodeTypeUnknown, 219 ), 220 pong: time.Now().Add(-3 * time.Second), 221 }, 222 { 223 node: NewNode( 224 MustHexID("0xe2d9d65c4552b5eb43d5ad55a2ee3f56c6cbc1c64a5c8d659f51fcd51bace24351232b8d7821617d2b29b54b81cdefb9b3e9c37d7fd5f63270bcc9e1a6f6a439"), 225 net.IP{127, 0, 0, 3}, 226 30303, 227 30303, 228 nil, 229 NodeTypeUnknown, 230 ), 231 pong: time.Now().Add(-1 * time.Second), 232 }, 233 } 234 235 func TestNodeDBSeedQuery(t *testing.T) { 236 db, _ := newNodeDB("", Version, nodeDBSeedQueryNodes[1].node.ID) 237 defer db.close() 238 239 // Insert a batch of nodes for querying 240 for i, seed := range nodeDBSeedQueryNodes { 241 if err := db.updateNode(seed.node); err != nil { 242 t.Fatalf("node %d: failed to insert: %v", i, err) 243 } 244 if err := db.updateBondTime(seed.node.ID, seed.pong); err != nil { 245 t.Fatalf("node %d: failed to insert bondTime: %v", i, err) 246 } 247 } 248 249 // Retrieve the entire batch and check for duplicates 250 seeds := db.querySeeds(len(nodeDBSeedQueryNodes)*2, time.Hour) 251 have := make(map[NodeID]struct{}) 252 for _, seed := range seeds { 253 have[seed.ID] = struct{}{} 254 } 255 want := make(map[NodeID]struct{}) 256 for _, seed := range nodeDBSeedQueryNodes[2:] { 257 want[seed.node.ID] = struct{}{} 258 } 259 if len(seeds) != len(want) { 260 t.Errorf("seed count mismatch: have %v, want %v", len(seeds), len(want)) 261 } 262 for id := range have { 263 if _, ok := want[id]; !ok { 264 t.Errorf("extra seed: %v", id) 265 } 266 } 267 for id := range want { 268 if _, ok := have[id]; !ok { 269 t.Errorf("missing seed: %v", id) 270 } 271 } 272 } 273 274 func TestNodeDBPersistency(t *testing.T) { 275 root, err := ioutil.TempDir("", "nodedb-") 276 if err != nil { 277 t.Fatalf("failed to create temporary data folder: %v", err) 278 } 279 defer os.RemoveAll(root) 280 281 var ( 282 testKey = []byte("somekey") 283 testInt = int64(314) 284 ) 285 286 // Create a persistent database and store some values 287 db, err := newNodeDB(filepath.Join(root, "database"), Version, NodeID{}) 288 if err != nil { 289 t.Fatalf("failed to create persistent database: %v", err) 290 } 291 if err := db.storeInt64(testKey, testInt); err != nil { 292 t.Fatalf("failed to store value: %v.", err) 293 } 294 db.close() 295 296 // Reopen the database and check the value 297 db, err = newNodeDB(filepath.Join(root, "database"), Version, NodeID{}) 298 if err != nil { 299 t.Fatalf("failed to open persistent database: %v", err) 300 } 301 if val := db.fetchInt64(testKey); val != testInt { 302 t.Fatalf("value mismatch: have %v, want %v", val, testInt) 303 } 304 db.close() 305 306 // Change the database version and check flush 307 db, err = newNodeDB(filepath.Join(root, "database"), Version+1, NodeID{}) 308 if err != nil { 309 t.Fatalf("failed to open persistent database: %v", err) 310 } 311 if val := db.fetchInt64(testKey); val != 0 { 312 t.Fatalf("value mismatch: have %v, want %v", val, 0) 313 } 314 db.close() 315 } 316 317 var nodeDBExpirationNodes = []struct { 318 node *Node 319 pong time.Time 320 exp bool 321 }{ 322 { 323 node: NewNode( 324 MustHexID("0x01d9d65c4552b5eb43d5ad55a2ee3f56c6cbc1c64a5c8d659f51fcd51bace24351232b8d7821617d2b29b54b81cdefb9b3e9c37d7fd5f63270bcc9e1a6f6a439"), 325 net.IP{127, 0, 0, 1}, 326 30303, 327 30303, 328 nil, 329 NodeTypeUnknown, 330 ), 331 pong: time.Now().Add(-nodeDBNodeExpiration + time.Minute), 332 exp: false, 333 }, { 334 node: NewNode( 335 MustHexID("0x02d9d65c4552b5eb43d5ad55a2ee3f56c6cbc1c64a5c8d659f51fcd51bace24351232b8d7821617d2b29b54b81cdefb9b3e9c37d7fd5f63270bcc9e1a6f6a439"), 336 net.IP{127, 0, 0, 2}, 337 30303, 338 30303, 339 nil, 340 NodeTypeUnknown, 341 ), 342 pong: time.Now().Add(-nodeDBNodeExpiration - time.Minute), 343 exp: true, 344 }, 345 } 346 347 func TestNodeDBExpiration(t *testing.T) { 348 db, _ := newNodeDB("", Version, NodeID{}) 349 defer db.close() 350 351 // Add all the test nodes and set their last pong time 352 for i, seed := range nodeDBExpirationNodes { 353 if err := db.updateNode(seed.node); err != nil { 354 t.Fatalf("node %d: failed to insert: %v", i, err) 355 } 356 if err := db.updateBondTime(seed.node.ID, seed.pong); err != nil { 357 t.Fatalf("node %d: failed to update bondTime: %v", i, err) 358 } 359 } 360 // Expire some of them, and check the rest 361 if err := db.expireNodes(); err != nil { 362 t.Fatalf("failed to expire nodes: %v", err) 363 } 364 for i, seed := range nodeDBExpirationNodes { 365 node := db.node(seed.node.ID) 366 if (node == nil && !seed.exp) || (node != nil && seed.exp) { 367 t.Errorf("node %d: expiration mismatch: have %v, want %v", i, node, seed.exp) 368 } 369 } 370 } 371 372 func TestNodeDBSelfExpiration(t *testing.T) { 373 // Find a node in the tests that shouldn't expire, and assign it as self 374 var self NodeID 375 for _, node := range nodeDBExpirationNodes { 376 if !node.exp { 377 self = node.node.ID 378 break 379 } 380 } 381 db, _ := newNodeDB("", Version, self) 382 defer db.close() 383 384 // Add all the test nodes and set their last pong time 385 for i, seed := range nodeDBExpirationNodes { 386 if err := db.updateNode(seed.node); err != nil { 387 t.Fatalf("node %d: failed to insert: %v", i, err) 388 } 389 if err := db.updateBondTime(seed.node.ID, seed.pong); err != nil { 390 t.Fatalf("node %d: failed to update bondTime: %v", i, err) 391 } 392 } 393 // Expire the nodes and make sure self has been evacuated too 394 if err := db.expireNodes(); err != nil { 395 t.Fatalf("failed to expire nodes: %v", err) 396 } 397 node := db.node(self) 398 if node != nil { 399 t.Errorf("self not evacuated") 400 } 401 }