github.1485827954.workers.dev/ethereum/go-ethereum@v1.14.3/p2p/discover/v5_udp_test.go (about) 1 // Copyright 2020 The go-ethereum Authors 2 // This file is part of the go-ethereum library. 3 // 4 // The go-ethereum library is free software: you can redistribute it and/or modify 5 // it under the terms of the GNU Lesser General Public License as published by 6 // the Free Software Foundation, either version 3 of the License, or 7 // (at your option) any later version. 8 // 9 // The go-ethereum library is distributed in the hope that it will be useful, 10 // but WITHOUT ANY WARRANTY; without even the implied warranty of 11 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 12 // GNU Lesser General Public License for more details. 13 // 14 // You should have received a copy of the GNU Lesser General Public License 15 // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. 16 17 package discover 18 19 import ( 20 "bytes" 21 "crypto/ecdsa" 22 "encoding/binary" 23 "fmt" 24 "math/rand" 25 "net" 26 "reflect" 27 "slices" 28 "testing" 29 "time" 30 31 "github.com/ethereum/go-ethereum/internal/testlog" 32 "github.com/ethereum/go-ethereum/log" 33 "github.com/ethereum/go-ethereum/p2p/discover/v5wire" 34 "github.com/ethereum/go-ethereum/p2p/enode" 35 "github.com/ethereum/go-ethereum/p2p/enr" 36 "github.com/ethereum/go-ethereum/rlp" 37 "github.com/stretchr/testify/require" 38 ) 39 40 // Real sockets, real crypto: this test checks end-to-end connectivity for UDPv5. 41 func TestUDPv5_lookupE2E(t *testing.T) { 42 t.Parallel() 43 44 const N = 5 45 var nodes []*UDPv5 46 for i := 0; i < N; i++ { 47 var cfg Config 48 if len(nodes) > 0 { 49 bn := nodes[0].Self() 50 cfg.Bootnodes = []*enode.Node{bn} 51 } 52 node := startLocalhostV5(t, cfg) 53 nodes = append(nodes, node) 54 defer node.Close() 55 } 56 last := nodes[N-1] 57 target := nodes[rand.Intn(N-2)].Self() 58 59 // It is expected that all nodes can be found. 60 expectedResult := make([]*enode.Node, len(nodes)) 61 for i := range nodes { 62 expectedResult[i] = nodes[i].Self() 63 } 64 slices.SortFunc(expectedResult, func(a, b *enode.Node) int { 65 return enode.DistCmp(target.ID(), a.ID(), b.ID()) 66 }) 67 68 // Do the lookup. 69 results := last.Lookup(target.ID()) 70 if err := checkNodesEqual(results, expectedResult); err != nil { 71 t.Fatalf("lookup returned wrong results: %v", err) 72 } 73 } 74 75 func startLocalhostV5(t *testing.T, cfg Config) *UDPv5 { 76 cfg.PrivateKey = newkey() 77 db, _ := enode.OpenDB("") 78 ln := enode.NewLocalNode(db, cfg.PrivateKey) 79 80 // Prefix logs with node ID. 81 lprefix := fmt.Sprintf("(%s)", ln.ID().TerminalString()) 82 cfg.Log = testlog.Logger(t, log.LevelTrace).With("node-id", lprefix) 83 84 // Listen. 85 socket, err := net.ListenUDP("udp4", &net.UDPAddr{IP: net.IP{127, 0, 0, 1}}) 86 if err != nil { 87 t.Fatal(err) 88 } 89 realaddr := socket.LocalAddr().(*net.UDPAddr) 90 ln.SetStaticIP(realaddr.IP) 91 ln.Set(enr.UDP(realaddr.Port)) 92 udp, err := ListenV5(socket, ln, cfg) 93 if err != nil { 94 t.Fatal(err) 95 } 96 return udp 97 } 98 99 // This test checks that incoming PING calls are handled correctly. 100 func TestUDPv5_pingHandling(t *testing.T) { 101 t.Parallel() 102 test := newUDPV5Test(t) 103 defer test.close() 104 105 test.packetIn(&v5wire.Ping{ReqID: []byte("foo")}) 106 test.waitPacketOut(func(p *v5wire.Pong, addr *net.UDPAddr, _ v5wire.Nonce) { 107 if !bytes.Equal(p.ReqID, []byte("foo")) { 108 t.Error("wrong request ID in response:", p.ReqID) 109 } 110 if p.ENRSeq != test.table.self().Seq() { 111 t.Error("wrong ENR sequence number in response:", p.ENRSeq) 112 } 113 }) 114 } 115 116 // This test checks that incoming 'unknown' packets trigger the handshake. 117 func TestUDPv5_unknownPacket(t *testing.T) { 118 t.Parallel() 119 test := newUDPV5Test(t) 120 defer test.close() 121 122 nonce := v5wire.Nonce{1, 2, 3} 123 check := func(p *v5wire.Whoareyou, wantSeq uint64) { 124 t.Helper() 125 if p.Nonce != nonce { 126 t.Error("wrong nonce in WHOAREYOU:", p.Nonce, nonce) 127 } 128 if p.IDNonce == ([16]byte{}) { 129 t.Error("all zero ID nonce") 130 } 131 if p.RecordSeq != wantSeq { 132 t.Errorf("wrong record seq %d in WHOAREYOU, want %d", p.RecordSeq, wantSeq) 133 } 134 } 135 136 // Unknown packet from unknown node. 137 test.packetIn(&v5wire.Unknown{Nonce: nonce}) 138 test.waitPacketOut(func(p *v5wire.Whoareyou, addr *net.UDPAddr, _ v5wire.Nonce) { 139 check(p, 0) 140 }) 141 142 // Make node known. 143 n := test.getNode(test.remotekey, test.remoteaddr).Node() 144 test.table.addSeenNode(wrapNode(n)) 145 146 test.packetIn(&v5wire.Unknown{Nonce: nonce}) 147 test.waitPacketOut(func(p *v5wire.Whoareyou, addr *net.UDPAddr, _ v5wire.Nonce) { 148 check(p, n.Seq()) 149 }) 150 } 151 152 // This test checks that incoming FINDNODE calls are handled correctly. 153 func TestUDPv5_findnodeHandling(t *testing.T) { 154 t.Parallel() 155 test := newUDPV5Test(t) 156 defer test.close() 157 158 // Create test nodes and insert them into the table. 159 nodes253 := nodesAtDistance(test.table.self().ID(), 253, 16) 160 nodes249 := nodesAtDistance(test.table.self().ID(), 249, 4) 161 nodes248 := nodesAtDistance(test.table.self().ID(), 248, 10) 162 fillTable(test.table, wrapNodes(nodes253), true) 163 fillTable(test.table, wrapNodes(nodes249), true) 164 fillTable(test.table, wrapNodes(nodes248), true) 165 166 // Requesting with distance zero should return the node's own record. 167 test.packetIn(&v5wire.Findnode{ReqID: []byte{0}, Distances: []uint{0}}) 168 test.expectNodes([]byte{0}, 1, []*enode.Node{test.udp.Self()}) 169 170 // Requesting with distance > 256 shouldn't crash. 171 test.packetIn(&v5wire.Findnode{ReqID: []byte{1}, Distances: []uint{4234098}}) 172 test.expectNodes([]byte{1}, 1, nil) 173 174 // Requesting with empty distance list shouldn't crash either. 175 test.packetIn(&v5wire.Findnode{ReqID: []byte{2}, Distances: []uint{}}) 176 test.expectNodes([]byte{2}, 1, nil) 177 178 // This request gets no nodes because the corresponding bucket is empty. 179 test.packetIn(&v5wire.Findnode{ReqID: []byte{3}, Distances: []uint{254}}) 180 test.expectNodes([]byte{3}, 1, nil) 181 182 // This request gets all the distance-253 nodes. 183 test.packetIn(&v5wire.Findnode{ReqID: []byte{4}, Distances: []uint{253}}) 184 test.expectNodes([]byte{4}, 2, nodes253) 185 186 // This request gets all the distance-249 nodes and some more at 248 because 187 // the bucket at 249 is not full. 188 test.packetIn(&v5wire.Findnode{ReqID: []byte{5}, Distances: []uint{249, 248}}) 189 var nodes []*enode.Node 190 nodes = append(nodes, nodes249...) 191 nodes = append(nodes, nodes248[:10]...) 192 test.expectNodes([]byte{5}, 1, nodes) 193 } 194 195 func (test *udpV5Test) expectNodes(wantReqID []byte, wantTotal uint8, wantNodes []*enode.Node) { 196 nodeSet := make(map[enode.ID]*enr.Record, len(wantNodes)) 197 for _, n := range wantNodes { 198 nodeSet[n.ID()] = n.Record() 199 } 200 201 for { 202 test.waitPacketOut(func(p *v5wire.Nodes, addr *net.UDPAddr, _ v5wire.Nonce) { 203 if !bytes.Equal(p.ReqID, wantReqID) { 204 test.t.Fatalf("wrong request ID %v in response, want %v", p.ReqID, wantReqID) 205 } 206 if p.RespCount != wantTotal { 207 test.t.Fatalf("wrong total response count %d, want %d", p.RespCount, wantTotal) 208 } 209 for _, record := range p.Nodes { 210 n, _ := enode.New(enode.ValidSchemesForTesting, record) 211 want := nodeSet[n.ID()] 212 if want == nil { 213 test.t.Fatalf("unexpected node in response: %v", n) 214 } 215 if !reflect.DeepEqual(record, want) { 216 test.t.Fatalf("wrong record in response: %v", n) 217 } 218 delete(nodeSet, n.ID()) 219 } 220 }) 221 if len(nodeSet) == 0 { 222 return 223 } 224 } 225 } 226 227 // This test checks that outgoing PING calls work. 228 func TestUDPv5_pingCall(t *testing.T) { 229 t.Parallel() 230 test := newUDPV5Test(t) 231 defer test.close() 232 233 remote := test.getNode(test.remotekey, test.remoteaddr).Node() 234 done := make(chan error, 1) 235 236 // This ping times out. 237 go func() { 238 _, err := test.udp.ping(remote) 239 done <- err 240 }() 241 test.waitPacketOut(func(p *v5wire.Ping, addr *net.UDPAddr, _ v5wire.Nonce) {}) 242 if err := <-done; err != errTimeout { 243 t.Fatalf("want errTimeout, got %q", err) 244 } 245 246 // This ping works. 247 go func() { 248 _, err := test.udp.ping(remote) 249 done <- err 250 }() 251 test.waitPacketOut(func(p *v5wire.Ping, addr *net.UDPAddr, _ v5wire.Nonce) { 252 test.packetInFrom(test.remotekey, test.remoteaddr, &v5wire.Pong{ReqID: p.ReqID}) 253 }) 254 if err := <-done; err != nil { 255 t.Fatal(err) 256 } 257 258 // This ping gets a reply from the wrong endpoint. 259 go func() { 260 _, err := test.udp.ping(remote) 261 done <- err 262 }() 263 test.waitPacketOut(func(p *v5wire.Ping, addr *net.UDPAddr, _ v5wire.Nonce) { 264 wrongAddr := &net.UDPAddr{IP: net.IP{33, 44, 55, 22}, Port: 10101} 265 test.packetInFrom(test.remotekey, wrongAddr, &v5wire.Pong{ReqID: p.ReqID}) 266 }) 267 if err := <-done; err != errTimeout { 268 t.Fatalf("want errTimeout for reply from wrong IP, got %q", err) 269 } 270 } 271 272 // This test checks that outgoing FINDNODE calls work and multiple NODES 273 // replies are aggregated. 274 func TestUDPv5_findnodeCall(t *testing.T) { 275 t.Parallel() 276 test := newUDPV5Test(t) 277 defer test.close() 278 279 // Launch the request: 280 var ( 281 distances = []uint{230} 282 remote = test.getNode(test.remotekey, test.remoteaddr).Node() 283 nodes = nodesAtDistance(remote.ID(), int(distances[0]), 8) 284 done = make(chan error, 1) 285 response []*enode.Node 286 ) 287 go func() { 288 var err error 289 response, err = test.udp.findnode(remote, distances) 290 done <- err 291 }() 292 293 // Serve the responses: 294 test.waitPacketOut(func(p *v5wire.Findnode, addr *net.UDPAddr, _ v5wire.Nonce) { 295 if !reflect.DeepEqual(p.Distances, distances) { 296 t.Fatalf("wrong distances in request: %v", p.Distances) 297 } 298 test.packetIn(&v5wire.Nodes{ 299 ReqID: p.ReqID, 300 RespCount: 2, 301 Nodes: nodesToRecords(nodes[:4]), 302 }) 303 test.packetIn(&v5wire.Nodes{ 304 ReqID: p.ReqID, 305 RespCount: 2, 306 Nodes: nodesToRecords(nodes[4:]), 307 }) 308 }) 309 310 // Check results: 311 if err := <-done; err != nil { 312 t.Fatalf("unexpected error: %v", err) 313 } 314 if !reflect.DeepEqual(response, nodes) { 315 t.Fatalf("wrong nodes in response") 316 } 317 318 // TODO: check invalid IPs 319 // TODO: check invalid/unsigned record 320 } 321 322 // This test checks that pending calls are re-sent when a handshake happens. 323 func TestUDPv5_callResend(t *testing.T) { 324 t.Parallel() 325 test := newUDPV5Test(t) 326 defer test.close() 327 328 remote := test.getNode(test.remotekey, test.remoteaddr).Node() 329 done := make(chan error, 2) 330 go func() { 331 _, err := test.udp.ping(remote) 332 done <- err 333 }() 334 go func() { 335 _, err := test.udp.ping(remote) 336 done <- err 337 }() 338 339 // Ping answered by WHOAREYOU. 340 test.waitPacketOut(func(p *v5wire.Ping, addr *net.UDPAddr, nonce v5wire.Nonce) { 341 test.packetIn(&v5wire.Whoareyou{Nonce: nonce}) 342 }) 343 // Ping should be re-sent. 344 test.waitPacketOut(func(p *v5wire.Ping, addr *net.UDPAddr, _ v5wire.Nonce) { 345 test.packetIn(&v5wire.Pong{ReqID: p.ReqID}) 346 }) 347 // Answer the other ping. 348 test.waitPacketOut(func(p *v5wire.Ping, addr *net.UDPAddr, _ v5wire.Nonce) { 349 test.packetIn(&v5wire.Pong{ReqID: p.ReqID}) 350 }) 351 if err := <-done; err != nil { 352 t.Fatalf("unexpected ping error: %v", err) 353 } 354 if err := <-done; err != nil { 355 t.Fatalf("unexpected ping error: %v", err) 356 } 357 } 358 359 // This test ensures we don't allow multiple rounds of WHOAREYOU for a single call. 360 func TestUDPv5_multipleHandshakeRounds(t *testing.T) { 361 t.Parallel() 362 test := newUDPV5Test(t) 363 defer test.close() 364 365 remote := test.getNode(test.remotekey, test.remoteaddr).Node() 366 done := make(chan error, 1) 367 go func() { 368 _, err := test.udp.ping(remote) 369 done <- err 370 }() 371 372 // Ping answered by WHOAREYOU. 373 test.waitPacketOut(func(p *v5wire.Ping, addr *net.UDPAddr, nonce v5wire.Nonce) { 374 test.packetIn(&v5wire.Whoareyou{Nonce: nonce}) 375 }) 376 // Ping answered by WHOAREYOU again. 377 test.waitPacketOut(func(p *v5wire.Ping, addr *net.UDPAddr, nonce v5wire.Nonce) { 378 test.packetIn(&v5wire.Whoareyou{Nonce: nonce}) 379 }) 380 if err := <-done; err != errTimeout { 381 t.Fatalf("unexpected ping error: %q", err) 382 } 383 } 384 385 // This test checks that calls with n replies may take up to n * respTimeout. 386 func TestUDPv5_callTimeoutReset(t *testing.T) { 387 t.Parallel() 388 test := newUDPV5Test(t) 389 defer test.close() 390 391 // Launch the request: 392 var ( 393 distance = uint(230) 394 remote = test.getNode(test.remotekey, test.remoteaddr).Node() 395 nodes = nodesAtDistance(remote.ID(), int(distance), 8) 396 done = make(chan error, 1) 397 ) 398 go func() { 399 _, err := test.udp.findnode(remote, []uint{distance}) 400 done <- err 401 }() 402 403 // Serve two responses, slowly. 404 test.waitPacketOut(func(p *v5wire.Findnode, addr *net.UDPAddr, _ v5wire.Nonce) { 405 time.Sleep(respTimeout - 50*time.Millisecond) 406 test.packetIn(&v5wire.Nodes{ 407 ReqID: p.ReqID, 408 RespCount: 2, 409 Nodes: nodesToRecords(nodes[:4]), 410 }) 411 412 time.Sleep(respTimeout - 50*time.Millisecond) 413 test.packetIn(&v5wire.Nodes{ 414 ReqID: p.ReqID, 415 RespCount: 2, 416 Nodes: nodesToRecords(nodes[4:]), 417 }) 418 }) 419 if err := <-done; err != nil { 420 t.Fatalf("unexpected error: %q", err) 421 } 422 } 423 424 // This test checks that TALKREQ calls the registered handler function. 425 func TestUDPv5_talkHandling(t *testing.T) { 426 t.Parallel() 427 test := newUDPV5Test(t) 428 defer test.close() 429 430 var recvMessage []byte 431 test.udp.RegisterTalkHandler("test", func(id enode.ID, addr *net.UDPAddr, message []byte) []byte { 432 recvMessage = message 433 return []byte("test response") 434 }) 435 436 // Successful case: 437 test.packetIn(&v5wire.TalkRequest{ 438 ReqID: []byte("foo"), 439 Protocol: "test", 440 Message: []byte("test request"), 441 }) 442 test.waitPacketOut(func(p *v5wire.TalkResponse, addr *net.UDPAddr, _ v5wire.Nonce) { 443 if !bytes.Equal(p.ReqID, []byte("foo")) { 444 t.Error("wrong request ID in response:", p.ReqID) 445 } 446 if string(p.Message) != "test response" { 447 t.Errorf("wrong talk response message: %q", p.Message) 448 } 449 if string(recvMessage) != "test request" { 450 t.Errorf("wrong message received in handler: %q", recvMessage) 451 } 452 }) 453 454 // Check that empty response is returned for unregistered protocols. 455 recvMessage = nil 456 test.packetIn(&v5wire.TalkRequest{ 457 ReqID: []byte("2"), 458 Protocol: "wrong", 459 Message: []byte("test request"), 460 }) 461 test.waitPacketOut(func(p *v5wire.TalkResponse, addr *net.UDPAddr, _ v5wire.Nonce) { 462 if !bytes.Equal(p.ReqID, []byte("2")) { 463 t.Error("wrong request ID in response:", p.ReqID) 464 } 465 if string(p.Message) != "" { 466 t.Errorf("wrong talk response message: %q", p.Message) 467 } 468 if recvMessage != nil { 469 t.Errorf("handler was called for wrong protocol: %q", recvMessage) 470 } 471 }) 472 } 473 474 // This test checks that outgoing TALKREQ calls work. 475 func TestUDPv5_talkRequest(t *testing.T) { 476 t.Parallel() 477 test := newUDPV5Test(t) 478 defer test.close() 479 480 remote := test.getNode(test.remotekey, test.remoteaddr).Node() 481 done := make(chan error, 1) 482 483 // This request times out. 484 go func() { 485 _, err := test.udp.TalkRequest(remote, "test", []byte("test request")) 486 done <- err 487 }() 488 test.waitPacketOut(func(p *v5wire.TalkRequest, addr *net.UDPAddr, _ v5wire.Nonce) {}) 489 if err := <-done; err != errTimeout { 490 t.Fatalf("want errTimeout, got %q", err) 491 } 492 493 // This request works. 494 go func() { 495 _, err := test.udp.TalkRequest(remote, "test", []byte("test request")) 496 done <- err 497 }() 498 test.waitPacketOut(func(p *v5wire.TalkRequest, addr *net.UDPAddr, _ v5wire.Nonce) { 499 if p.Protocol != "test" { 500 t.Errorf("wrong protocol ID in talk request: %q", p.Protocol) 501 } 502 if string(p.Message) != "test request" { 503 t.Errorf("wrong message talk request: %q", p.Message) 504 } 505 test.packetInFrom(test.remotekey, test.remoteaddr, &v5wire.TalkResponse{ 506 ReqID: p.ReqID, 507 Message: []byte("test response"), 508 }) 509 }) 510 if err := <-done; err != nil { 511 t.Fatal(err) 512 } 513 514 // Also check requesting without ENR. 515 go func() { 516 _, err := test.udp.TalkRequestToID(remote.ID(), test.remoteaddr, "test", []byte("test request 2")) 517 done <- err 518 }() 519 test.waitPacketOut(func(p *v5wire.TalkRequest, addr *net.UDPAddr, _ v5wire.Nonce) { 520 if p.Protocol != "test" { 521 t.Errorf("wrong protocol ID in talk request: %q", p.Protocol) 522 } 523 if string(p.Message) != "test request 2" { 524 t.Errorf("wrong message talk request: %q", p.Message) 525 } 526 test.packetInFrom(test.remotekey, test.remoteaddr, &v5wire.TalkResponse{ 527 ReqID: p.ReqID, 528 Message: []byte("test response 2"), 529 }) 530 }) 531 if err := <-done; err != nil { 532 t.Fatal(err) 533 } 534 } 535 536 // This test checks that lookupDistances works. 537 func TestUDPv5_lookupDistances(t *testing.T) { 538 test := newUDPV5Test(t) 539 lnID := test.table.self().ID() 540 541 t.Run("target distance of 1", func(t *testing.T) { 542 node := nodeAtDistance(lnID, 1, intIP(0)) 543 dists := lookupDistances(lnID, node.ID()) 544 require.Equal(t, []uint{1, 2, 3}, dists) 545 }) 546 547 t.Run("target distance of 2", func(t *testing.T) { 548 node := nodeAtDistance(lnID, 2, intIP(0)) 549 dists := lookupDistances(lnID, node.ID()) 550 require.Equal(t, []uint{2, 3, 1}, dists) 551 }) 552 553 t.Run("target distance of 128", func(t *testing.T) { 554 node := nodeAtDistance(lnID, 128, intIP(0)) 555 dists := lookupDistances(lnID, node.ID()) 556 require.Equal(t, []uint{128, 129, 127}, dists) 557 }) 558 559 t.Run("target distance of 255", func(t *testing.T) { 560 node := nodeAtDistance(lnID, 255, intIP(0)) 561 dists := lookupDistances(lnID, node.ID()) 562 require.Equal(t, []uint{255, 256, 254}, dists) 563 }) 564 565 t.Run("target distance of 256", func(t *testing.T) { 566 node := nodeAtDistance(lnID, 256, intIP(0)) 567 dists := lookupDistances(lnID, node.ID()) 568 require.Equal(t, []uint{256, 255, 254}, dists) 569 }) 570 } 571 572 // This test checks that lookup works. 573 func TestUDPv5_lookup(t *testing.T) { 574 t.Parallel() 575 test := newUDPV5Test(t) 576 577 // Lookup on empty table returns no nodes. 578 if results := test.udp.Lookup(lookupTestnet.target.id()); len(results) > 0 { 579 t.Fatalf("lookup on empty table returned %d results: %#v", len(results), results) 580 } 581 582 // Ensure the tester knows all nodes in lookupTestnet by IP. 583 for d, nn := range lookupTestnet.dists { 584 for i, key := range nn { 585 n := lookupTestnet.node(d, i) 586 test.getNode(key, &net.UDPAddr{IP: n.IP(), Port: n.UDP()}) 587 } 588 } 589 590 // Seed table with initial node. 591 initialNode := lookupTestnet.node(256, 0) 592 fillTable(test.table, []*node{wrapNode(initialNode)}, true) 593 594 // Start the lookup. 595 resultC := make(chan []*enode.Node, 1) 596 go func() { 597 resultC <- test.udp.Lookup(lookupTestnet.target.id()) 598 test.close() 599 }() 600 601 // Answer lookup packets. 602 asked := make(map[enode.ID]bool) 603 for done := false; !done; { 604 done = test.waitPacketOut(func(p v5wire.Packet, to *net.UDPAddr, _ v5wire.Nonce) { 605 recipient, key := lookupTestnet.nodeByAddr(to) 606 switch p := p.(type) { 607 case *v5wire.Ping: 608 test.packetInFrom(key, to, &v5wire.Pong{ReqID: p.ReqID}) 609 case *v5wire.Findnode: 610 if asked[recipient.ID()] { 611 t.Error("Asked node", recipient.ID(), "twice") 612 } 613 asked[recipient.ID()] = true 614 nodes := lookupTestnet.neighborsAtDistances(recipient, p.Distances, 16) 615 t.Logf("Got FINDNODE for %v, returning %d nodes", p.Distances, len(nodes)) 616 for _, resp := range packNodes(p.ReqID, nodes) { 617 test.packetInFrom(key, to, resp) 618 } 619 } 620 }) 621 } 622 623 // Verify result nodes. 624 results := <-resultC 625 checkLookupResults(t, lookupTestnet, results) 626 } 627 628 // This test checks the local node can be utilised to set key-values. 629 func TestUDPv5_LocalNode(t *testing.T) { 630 t.Parallel() 631 var cfg Config 632 node := startLocalhostV5(t, cfg) 633 defer node.Close() 634 localNd := node.LocalNode() 635 636 // set value in node's local record 637 testVal := [4]byte{'A', 'B', 'C', 'D'} 638 localNd.Set(enr.WithEntry("testing", &testVal)) 639 640 // retrieve the value from self to make sure it matches. 641 outputVal := [4]byte{} 642 if err := node.Self().Load(enr.WithEntry("testing", &outputVal)); err != nil { 643 t.Errorf("Could not load value from record: %v", err) 644 } 645 if testVal != outputVal { 646 t.Errorf("Wanted %#x to be retrieved from the record but instead got %#x", testVal, outputVal) 647 } 648 } 649 650 func TestUDPv5_PingWithIPV4MappedAddress(t *testing.T) { 651 t.Parallel() 652 test := newUDPV5Test(t) 653 defer test.close() 654 655 rawIP := net.IPv4(0xFF, 0x12, 0x33, 0xE5) 656 test.remoteaddr = &net.UDPAddr{ 657 IP: rawIP.To16(), 658 Port: 0, 659 } 660 remote := test.getNode(test.remotekey, test.remoteaddr).Node() 661 done := make(chan struct{}, 1) 662 663 // This handler will truncate the ipv4-mapped in ipv6 address. 664 go func() { 665 test.udp.handlePing(&v5wire.Ping{ENRSeq: 1}, remote.ID(), test.remoteaddr) 666 done <- struct{}{} 667 }() 668 test.waitPacketOut(func(p *v5wire.Pong, addr *net.UDPAddr, _ v5wire.Nonce) { 669 if len(p.ToIP) == net.IPv6len { 670 t.Error("Received untruncated ip address") 671 } 672 if len(p.ToIP) != net.IPv4len { 673 t.Errorf("Received ip address with incorrect length: %d", len(p.ToIP)) 674 } 675 if !p.ToIP.Equal(rawIP) { 676 t.Errorf("Received incorrect ip address: wanted %s but received %s", rawIP.String(), p.ToIP.String()) 677 } 678 }) 679 <-done 680 } 681 682 // udpV5Test is the framework for all tests above. 683 // It runs the UDPv5 transport on a virtual socket and allows testing outgoing packets. 684 type udpV5Test struct { 685 t *testing.T 686 pipe *dgramPipe 687 table *Table 688 db *enode.DB 689 udp *UDPv5 690 localkey, remotekey *ecdsa.PrivateKey 691 remoteaddr *net.UDPAddr 692 nodesByID map[enode.ID]*enode.LocalNode 693 nodesByIP map[string]*enode.LocalNode 694 } 695 696 // testCodec is the packet encoding used by protocol tests. This codec does not perform encryption. 697 type testCodec struct { 698 test *udpV5Test 699 id enode.ID 700 ctr uint64 701 } 702 703 type testCodecFrame struct { 704 NodeID enode.ID 705 AuthTag v5wire.Nonce 706 Ptype byte 707 Packet rlp.RawValue 708 } 709 710 func (c *testCodec) Encode(toID enode.ID, addr string, p v5wire.Packet, _ *v5wire.Whoareyou) ([]byte, v5wire.Nonce, error) { 711 c.ctr++ 712 var authTag v5wire.Nonce 713 binary.BigEndian.PutUint64(authTag[:], c.ctr) 714 715 penc, _ := rlp.EncodeToBytes(p) 716 frame, err := rlp.EncodeToBytes(testCodecFrame{c.id, authTag, p.Kind(), penc}) 717 return frame, authTag, err 718 } 719 720 func (c *testCodec) Decode(input []byte, addr string) (enode.ID, *enode.Node, v5wire.Packet, error) { 721 frame, p, err := c.decodeFrame(input) 722 if err != nil { 723 return enode.ID{}, nil, nil, err 724 } 725 return frame.NodeID, nil, p, nil 726 } 727 728 func (c *testCodec) decodeFrame(input []byte) (frame testCodecFrame, p v5wire.Packet, err error) { 729 if err = rlp.DecodeBytes(input, &frame); err != nil { 730 return frame, nil, fmt.Errorf("invalid frame: %v", err) 731 } 732 switch frame.Ptype { 733 case v5wire.UnknownPacket: 734 dec := new(v5wire.Unknown) 735 err = rlp.DecodeBytes(frame.Packet, &dec) 736 p = dec 737 case v5wire.WhoareyouPacket: 738 dec := new(v5wire.Whoareyou) 739 err = rlp.DecodeBytes(frame.Packet, &dec) 740 p = dec 741 default: 742 p, err = v5wire.DecodeMessage(frame.Ptype, frame.Packet) 743 } 744 return frame, p, err 745 } 746 747 func newUDPV5Test(t *testing.T) *udpV5Test { 748 test := &udpV5Test{ 749 t: t, 750 pipe: newpipe(), 751 localkey: newkey(), 752 remotekey: newkey(), 753 remoteaddr: &net.UDPAddr{IP: net.IP{10, 0, 1, 99}, Port: 30303}, 754 nodesByID: make(map[enode.ID]*enode.LocalNode), 755 nodesByIP: make(map[string]*enode.LocalNode), 756 } 757 test.db, _ = enode.OpenDB("") 758 ln := enode.NewLocalNode(test.db, test.localkey) 759 ln.SetStaticIP(net.IP{10, 0, 0, 1}) 760 ln.Set(enr.UDP(30303)) 761 test.udp, _ = ListenV5(test.pipe, ln, Config{ 762 PrivateKey: test.localkey, 763 Log: testlog.Logger(t, log.LvlTrace), 764 ValidSchemes: enode.ValidSchemesForTesting, 765 }) 766 test.udp.codec = &testCodec{test: test, id: ln.ID()} 767 test.table = test.udp.tab 768 test.nodesByID[ln.ID()] = ln 769 // Wait for initial refresh so the table doesn't send unexpected findnode. 770 <-test.table.initDone 771 return test 772 } 773 774 // handles a packet as if it had been sent to the transport. 775 func (test *udpV5Test) packetIn(packet v5wire.Packet) { 776 test.t.Helper() 777 test.packetInFrom(test.remotekey, test.remoteaddr, packet) 778 } 779 780 // handles a packet as if it had been sent to the transport by the key/endpoint. 781 func (test *udpV5Test) packetInFrom(key *ecdsa.PrivateKey, addr *net.UDPAddr, packet v5wire.Packet) { 782 test.t.Helper() 783 784 ln := test.getNode(key, addr) 785 codec := &testCodec{test: test, id: ln.ID()} 786 enc, _, err := codec.Encode(test.udp.Self().ID(), addr.String(), packet, nil) 787 if err != nil { 788 test.t.Errorf("%s encode error: %v", packet.Name(), err) 789 } 790 if test.udp.dispatchReadPacket(addr, enc) { 791 <-test.udp.readNextCh // unblock UDPv5.dispatch 792 } 793 } 794 795 // getNode ensures the test knows about a node at the given endpoint. 796 func (test *udpV5Test) getNode(key *ecdsa.PrivateKey, addr *net.UDPAddr) *enode.LocalNode { 797 id := encodePubkey(&key.PublicKey).id() 798 ln := test.nodesByID[id] 799 if ln == nil { 800 db, _ := enode.OpenDB("") 801 ln = enode.NewLocalNode(db, key) 802 ln.SetStaticIP(addr.IP) 803 ln.Set(enr.UDP(addr.Port)) 804 test.nodesByID[id] = ln 805 } 806 test.nodesByIP[string(addr.IP)] = ln 807 return ln 808 } 809 810 // waitPacketOut waits for the next output packet and handles it using the given 'validate' 811 // function. The function must be of type func (X, *net.UDPAddr, v5wire.Nonce) where X is 812 // assignable to packetV5. 813 func (test *udpV5Test) waitPacketOut(validate interface{}) (closed bool) { 814 test.t.Helper() 815 816 fn := reflect.ValueOf(validate) 817 exptype := fn.Type().In(0) 818 819 dgram, err := test.pipe.receive() 820 if err == errClosed { 821 return true 822 } 823 if err == errTimeout { 824 test.t.Fatalf("timed out waiting for %v", exptype) 825 return false 826 } 827 ln := test.nodesByIP[string(dgram.to.IP)] 828 if ln == nil { 829 test.t.Fatalf("attempt to send to non-existing node %v", &dgram.to) 830 return false 831 } 832 codec := &testCodec{test: test, id: ln.ID()} 833 frame, p, err := codec.decodeFrame(dgram.data) 834 if err != nil { 835 test.t.Errorf("sent packet decode error: %v", err) 836 return false 837 } 838 if !reflect.TypeOf(p).AssignableTo(exptype) { 839 test.t.Errorf("sent packet type mismatch, got: %v, want: %v", reflect.TypeOf(p), exptype) 840 return false 841 } 842 fn.Call([]reflect.Value{reflect.ValueOf(p), reflect.ValueOf(&dgram.to), reflect.ValueOf(frame.AuthTag)}) 843 return false 844 } 845 846 func (test *udpV5Test) close() { 847 test.t.Helper() 848 849 test.udp.Close() 850 test.db.Close() 851 for id, n := range test.nodesByID { 852 if id != test.udp.Self().ID() { 853 n.Database().Close() 854 } 855 } 856 if len(test.pipe.queue) != 0 { 857 test.t.Fatalf("%d unmatched UDP packets in queue", len(test.pipe.queue)) 858 } 859 }