| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // Copyright 2015 The go-ethereum Authors | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // This file is part of the go-ethereum library. | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // | 
					
						
							| 
									
										
										
										
											2015-07-23 18:35:11 +02:00
										 |  |  | // The go-ethereum library is free software: you can redistribute it and/or modify | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // it under the terms of the GNU Lesser General Public License as published by | 
					
						
							|  |  |  | // the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  | // (at your option) any later version. | 
					
						
							|  |  |  | // | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // The go-ethereum library is distributed in the hope that it will be useful, | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // GNU Lesser General Public License for more details. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // You should have received a copy of the GNU Lesser General Public License | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | package p2p | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"encoding/binary" | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | 	"net" | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 	"reflect" | 
					
						
							|  |  |  | 	"testing" | 
					
						
							|  |  |  | 	"time" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/davecgh/go-spew/spew" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/p2p/discover" | 
					
						
							| 
									
										
										
										
											2016-11-22 20:51:59 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/p2p/netutil" | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func init() { | 
					
						
							|  |  |  | 	spew.Config.Indent = "\t" | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type dialtest struct { | 
					
						
							|  |  |  | 	init   *dialstate // state before and after the test. | 
					
						
							|  |  |  | 	rounds []round | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type round struct { | 
					
						
							|  |  |  | 	peers []*Peer // current peer set | 
					
						
							|  |  |  | 	done  []task  // tasks that got done this round | 
					
						
							|  |  |  | 	new   []task  // the result must match this one | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func runDialTest(t *testing.T, test dialtest) { | 
					
						
							|  |  |  | 	var ( | 
					
						
							|  |  |  | 		vtime   time.Time | 
					
						
							|  |  |  | 		running int | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 	pm := func(ps []*Peer) map[discover.NodeID]*Peer { | 
					
						
							|  |  |  | 		m := make(map[discover.NodeID]*Peer) | 
					
						
							|  |  |  | 		for _, p := range ps { | 
					
						
							|  |  |  | 			m[p.rw.id] = p | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return m | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	for i, round := range test.rounds { | 
					
						
							|  |  |  | 		for _, task := range round.done { | 
					
						
							|  |  |  | 			running-- | 
					
						
							|  |  |  | 			if running < 0 { | 
					
						
							|  |  |  | 				panic("running task counter underflow") | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			test.init.taskDone(task, vtime) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		new := test.init.newTasks(running, pm(round.peers), vtime) | 
					
						
							|  |  |  | 		if !sametasks(new, round.new) { | 
					
						
							|  |  |  | 			t.Errorf("round %d: new tasks mismatch:\ngot %v\nwant %v\nstate: %v\nrunning: %v\n", | 
					
						
							|  |  |  | 				i, spew.Sdump(new), spew.Sdump(round.new), spew.Sdump(test.init), spew.Sdump(running)) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Time advances by 16 seconds on every round. | 
					
						
							|  |  |  | 		vtime = vtime.Add(16 * time.Second) | 
					
						
							|  |  |  | 		running += len(new) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type fakeTable []*discover.Node | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-07 12:06:49 +01:00
										 |  |  | func (t fakeTable) Self() *discover.Node                     { return new(discover.Node) } | 
					
						
							|  |  |  | func (t fakeTable) Close()                                   {} | 
					
						
							|  |  |  | func (t fakeTable) Lookup(discover.NodeID) []*discover.Node  { return nil } | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | func (t fakeTable) Resolve(discover.NodeID) *discover.Node   { return nil } | 
					
						
							| 
									
										
										
										
											2015-12-07 12:06:49 +01:00
										 |  |  | func (t fakeTable) ReadRandomNodes(buf []*discover.Node) int { return copy(buf, t) } | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | // This test checks that dynamic dials are launched from discovery results. | 
					
						
							|  |  |  | func TestDialStateDynDial(t *testing.T) { | 
					
						
							|  |  |  | 	runDialTest(t, dialtest{ | 
					
						
							| 
									
										
										
										
											2017-04-10 19:33:41 +03:00
										 |  |  | 		init: newDialState(nil, nil, fakeTable{}, 5, nil), | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 		rounds: []round{ | 
					
						
							|  |  |  | 			// A discovery query is launched. | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				peers: []*Peer{ | 
					
						
							|  |  |  | 					{rw: &conn{flags: staticDialedConn, id: uintID(0)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(1)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(2)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							| 
									
										
										
										
											2015-12-07 12:06:49 +01:00
										 |  |  | 				new: []task{&discoverTask{}}, | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 			}, | 
					
						
							|  |  |  | 			// Dynamic dials are launched when it completes. | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				peers: []*Peer{ | 
					
						
							|  |  |  | 					{rw: &conn{flags: staticDialedConn, id: uintID(0)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(1)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(2)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				done: []task{ | 
					
						
							| 
									
										
										
										
											2015-12-07 12:06:49 +01:00
										 |  |  | 					&discoverTask{results: []*discover.Node{ | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 						{ID: uintID(2)}, // this one is already connected and not dialed. | 
					
						
							|  |  |  | 						{ID: uintID(3)}, | 
					
						
							|  |  |  | 						{ID: uintID(4)}, | 
					
						
							|  |  |  | 						{ID: uintID(5)}, | 
					
						
							|  |  |  | 						{ID: uintID(6)}, // these are not tried because max dyn dials is 5 | 
					
						
							|  |  |  | 						{ID: uintID(7)}, // ... | 
					
						
							|  |  |  | 					}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				new: []task{ | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(3)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(4)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(5)}}, | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			// Some of the dials complete but no new ones are launched yet because | 
					
						
							|  |  |  | 			// the sum of active dial count and dynamic peer count is == maxDynDials. | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				peers: []*Peer{ | 
					
						
							|  |  |  | 					{rw: &conn{flags: staticDialedConn, id: uintID(0)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(1)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(2)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(3)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(4)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				done: []task{ | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(3)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(4)}}, | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			// No new dial tasks are launched in the this round because | 
					
						
							|  |  |  | 			// maxDynDials has been reached. | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				peers: []*Peer{ | 
					
						
							|  |  |  | 					{rw: &conn{flags: staticDialedConn, id: uintID(0)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(1)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(2)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(3)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(4)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(5)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				done: []task{ | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(5)}}, | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 				}, | 
					
						
							|  |  |  | 				new: []task{ | 
					
						
							|  |  |  | 					&waitExpireTask{Duration: 14 * time.Second}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			// In this round, the peer with id 2 drops off. The query | 
					
						
							|  |  |  | 			// results from last discovery lookup are reused. | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				peers: []*Peer{ | 
					
						
							|  |  |  | 					{rw: &conn{flags: staticDialedConn, id: uintID(0)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(1)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(3)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(4)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(5)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				new: []task{ | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(6)}}, | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			// More peers (3,4) drop off and dial for ID 6 completes. | 
					
						
							|  |  |  | 			// The last query result from the discovery lookup is reused | 
					
						
							|  |  |  | 			// and a new one is spawned because more candidates are needed. | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				peers: []*Peer{ | 
					
						
							|  |  |  | 					{rw: &conn{flags: staticDialedConn, id: uintID(0)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(1)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(5)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				done: []task{ | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(6)}}, | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 				}, | 
					
						
							|  |  |  | 				new: []task{ | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(7)}}, | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 					&discoverTask{}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			// Peer 7 is connected, but there still aren't enough dynamic peers | 
					
						
							|  |  |  | 			// (4 out of 5). However, a discovery is already running, so ensure | 
					
						
							|  |  |  | 			// no new is started. | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				peers: []*Peer{ | 
					
						
							|  |  |  | 					{rw: &conn{flags: staticDialedConn, id: uintID(0)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(1)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(5)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(7)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				done: []task{ | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(7)}}, | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			// Finish the running node discovery with an empty set. A new lookup | 
					
						
							|  |  |  | 			// should be immediately requested. | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				peers: []*Peer{ | 
					
						
							|  |  |  | 					{rw: &conn{flags: staticDialedConn, id: uintID(0)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(1)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(5)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(7)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				done: []task{ | 
					
						
							|  |  |  | 					&discoverTask{}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				new: []task{ | 
					
						
							|  |  |  | 					&discoverTask{}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-10 19:33:41 +03:00
										 |  |  | // Tests that bootnodes are dialed if no peers are connectd, but not otherwise. | 
					
						
							|  |  |  | func TestDialStateDynDialBootnode(t *testing.T) { | 
					
						
							|  |  |  | 	bootnodes := []*discover.Node{ | 
					
						
							|  |  |  | 		{ID: uintID(1)}, | 
					
						
							|  |  |  | 		{ID: uintID(2)}, | 
					
						
							|  |  |  | 		{ID: uintID(3)}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	table := fakeTable{ | 
					
						
							|  |  |  | 		{ID: uintID(4)}, | 
					
						
							|  |  |  | 		{ID: uintID(5)}, | 
					
						
							|  |  |  | 		{ID: uintID(6)}, | 
					
						
							|  |  |  | 		{ID: uintID(7)}, | 
					
						
							|  |  |  | 		{ID: uintID(8)}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	runDialTest(t, dialtest{ | 
					
						
							|  |  |  | 		init: newDialState(nil, bootnodes, table, 5, nil), | 
					
						
							|  |  |  | 		rounds: []round{ | 
					
						
							|  |  |  | 			// 2 dynamic dials attempted, bootnodes pending fallback interval | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				new: []task{ | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(4)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(5)}}, | 
					
						
							|  |  |  | 					&discoverTask{}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			// No dials succeed, bootnodes still pending fallback interval | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				done: []task{ | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(4)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(5)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			// No dials succeed, bootnodes still pending fallback interval | 
					
						
							|  |  |  | 			{}, | 
					
						
							|  |  |  | 			// No dials succeed, 2 dynamic dials attempted and 1 bootnode too as fallback interval was reached | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				new: []task{ | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(1)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(4)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(5)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			// No dials succeed, 2nd bootnode is attempted | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				done: []task{ | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(1)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(4)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(5)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				new: []task{ | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(2)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			// No dials succeed, 3rd bootnode is attempted | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				done: []task{ | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(2)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				new: []task{ | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(3)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			// No dials succeed, 1st bootnode is attempted again, expired random nodes retried | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				done: []task{ | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(3)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				new: []task{ | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(1)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(4)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(5)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			// Random dial succeeds, no more bootnodes are attempted | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				peers: []*Peer{ | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(4)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				done: []task{ | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(1)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(4)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(5)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | func TestDialStateDynDialFromTable(t *testing.T) { | 
					
						
							|  |  |  | 	// This table always returns the same random nodes | 
					
						
							|  |  |  | 	// in the order given below. | 
					
						
							|  |  |  | 	table := fakeTable{ | 
					
						
							|  |  |  | 		{ID: uintID(1)}, | 
					
						
							|  |  |  | 		{ID: uintID(2)}, | 
					
						
							|  |  |  | 		{ID: uintID(3)}, | 
					
						
							|  |  |  | 		{ID: uintID(4)}, | 
					
						
							|  |  |  | 		{ID: uintID(5)}, | 
					
						
							|  |  |  | 		{ID: uintID(6)}, | 
					
						
							|  |  |  | 		{ID: uintID(7)}, | 
					
						
							|  |  |  | 		{ID: uintID(8)}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	runDialTest(t, dialtest{ | 
					
						
							| 
									
										
										
										
											2017-04-10 19:33:41 +03:00
										 |  |  | 		init: newDialState(nil, nil, table, 10, nil), | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 		rounds: []round{ | 
					
						
							|  |  |  | 			// 5 out of 8 of the nodes returned by ReadRandomNodes are dialed. | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				new: []task{ | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(1)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(2)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(3)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(4)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(5)}}, | 
					
						
							| 
									
										
										
										
											2015-12-07 12:06:49 +01:00
										 |  |  | 					&discoverTask{}, | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			// Dialing nodes 1,2 succeeds. Dials from the lookup are launched. | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				peers: []*Peer{ | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(1)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(2)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				done: []task{ | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(1)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(2)}}, | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 					&discoverTask{results: []*discover.Node{ | 
					
						
							|  |  |  | 						{ID: uintID(10)}, | 
					
						
							|  |  |  | 						{ID: uintID(11)}, | 
					
						
							|  |  |  | 						{ID: uintID(12)}, | 
					
						
							|  |  |  | 					}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				new: []task{ | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(10)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(11)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(12)}}, | 
					
						
							| 
									
										
										
										
											2015-12-07 12:06:49 +01:00
										 |  |  | 					&discoverTask{}, | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			// Dialing nodes 3,4,5 fails. The dials from the lookup succeed. | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				peers: []*Peer{ | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(1)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(2)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(10)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(11)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(12)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				done: []task{ | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(3)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(4)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(5)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(10)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(11)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: &discover.Node{ID: uintID(12)}}, | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			// Waiting for expiry. No waitExpireTask is launched because the | 
					
						
							|  |  |  | 			// discovery query is still running. | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				peers: []*Peer{ | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(1)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(2)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(10)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(11)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(12)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			// Nodes 3,4 are not tried again because only the first two | 
					
						
							|  |  |  | 			// returned random nodes (nodes 1,2) are tried and they're | 
					
						
							|  |  |  | 			// already connected. | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				peers: []*Peer{ | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(1)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(2)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(10)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(11)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(12)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-22 20:51:59 +01:00
										 |  |  | // This test checks that candidates that do not match the netrestrict list are not dialed. | 
					
						
							|  |  |  | func TestDialStateNetRestrict(t *testing.T) { | 
					
						
							|  |  |  | 	// This table always returns the same random nodes | 
					
						
							|  |  |  | 	// in the order given below. | 
					
						
							|  |  |  | 	table := fakeTable{ | 
					
						
							|  |  |  | 		{ID: uintID(1), IP: net.ParseIP("127.0.0.1")}, | 
					
						
							|  |  |  | 		{ID: uintID(2), IP: net.ParseIP("127.0.0.2")}, | 
					
						
							|  |  |  | 		{ID: uintID(3), IP: net.ParseIP("127.0.0.3")}, | 
					
						
							|  |  |  | 		{ID: uintID(4), IP: net.ParseIP("127.0.0.4")}, | 
					
						
							|  |  |  | 		{ID: uintID(5), IP: net.ParseIP("127.0.2.5")}, | 
					
						
							|  |  |  | 		{ID: uintID(6), IP: net.ParseIP("127.0.2.6")}, | 
					
						
							|  |  |  | 		{ID: uintID(7), IP: net.ParseIP("127.0.2.7")}, | 
					
						
							|  |  |  | 		{ID: uintID(8), IP: net.ParseIP("127.0.2.8")}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	restrict := new(netutil.Netlist) | 
					
						
							|  |  |  | 	restrict.Add("127.0.2.0/24") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	runDialTest(t, dialtest{ | 
					
						
							| 
									
										
										
										
											2017-04-10 19:33:41 +03:00
										 |  |  | 		init: newDialState(nil, nil, table, 10, restrict), | 
					
						
							| 
									
										
										
										
											2016-11-22 20:51:59 +01:00
										 |  |  | 		rounds: []round{ | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				new: []task{ | 
					
						
							|  |  |  | 					&dialTask{flags: dynDialedConn, dest: table[4]}, | 
					
						
							|  |  |  | 					&discoverTask{}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | // This test checks that static dials are launched. | 
					
						
							|  |  |  | func TestDialStateStaticDial(t *testing.T) { | 
					
						
							|  |  |  | 	wantStatic := []*discover.Node{ | 
					
						
							|  |  |  | 		{ID: uintID(1)}, | 
					
						
							|  |  |  | 		{ID: uintID(2)}, | 
					
						
							|  |  |  | 		{ID: uintID(3)}, | 
					
						
							|  |  |  | 		{ID: uintID(4)}, | 
					
						
							|  |  |  | 		{ID: uintID(5)}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	runDialTest(t, dialtest{ | 
					
						
							| 
									
										
										
										
											2017-04-10 19:33:41 +03:00
										 |  |  | 		init: newDialState(wantStatic, nil, fakeTable{}, 0, nil), | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 		rounds: []round{ | 
					
						
							|  |  |  | 			// Static dials are launched for the nodes that | 
					
						
							|  |  |  | 			// aren't yet connected. | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				peers: []*Peer{ | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(1)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(2)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				new: []task{ | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | 					&dialTask{flags: staticDialedConn, dest: &discover.Node{ID: uintID(3)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: staticDialedConn, dest: &discover.Node{ID: uintID(4)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: staticDialedConn, dest: &discover.Node{ID: uintID(5)}}, | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			// No new tasks are launched in this round because all static | 
					
						
							|  |  |  | 			// nodes are either connected or still being dialed. | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				peers: []*Peer{ | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(1)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(2)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: staticDialedConn, id: uintID(3)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				done: []task{ | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | 					&dialTask{flags: staticDialedConn, dest: &discover.Node{ID: uintID(3)}}, | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			// No new dial tasks are launched because all static | 
					
						
							|  |  |  | 			// nodes are now connected. | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				peers: []*Peer{ | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(1)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(2)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: staticDialedConn, id: uintID(3)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: staticDialedConn, id: uintID(4)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: staticDialedConn, id: uintID(5)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				done: []task{ | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | 					&dialTask{flags: staticDialedConn, dest: &discover.Node{ID: uintID(4)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: staticDialedConn, dest: &discover.Node{ID: uintID(5)}}, | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 				}, | 
					
						
							|  |  |  | 				new: []task{ | 
					
						
							|  |  |  | 					&waitExpireTask{Duration: 14 * time.Second}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			// Wait a round for dial history to expire, no new tasks should spawn. | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				peers: []*Peer{ | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(1)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(2)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: staticDialedConn, id: uintID(3)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: staticDialedConn, id: uintID(4)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: staticDialedConn, id: uintID(5)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			// If a static node is dropped, it should be immediately redialed, | 
					
						
							|  |  |  | 			// irrespective whether it was originally static or dynamic. | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				peers: []*Peer{ | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(1)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: staticDialedConn, id: uintID(3)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: staticDialedConn, id: uintID(5)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				new: []task{ | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | 					&dialTask{flags: staticDialedConn, dest: &discover.Node{ID: uintID(2)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: staticDialedConn, dest: &discover.Node{ID: uintID(4)}}, | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // This test checks that past dials are not retried for some time. | 
					
						
							|  |  |  | func TestDialStateCache(t *testing.T) { | 
					
						
							|  |  |  | 	wantStatic := []*discover.Node{ | 
					
						
							|  |  |  | 		{ID: uintID(1)}, | 
					
						
							|  |  |  | 		{ID: uintID(2)}, | 
					
						
							|  |  |  | 		{ID: uintID(3)}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	runDialTest(t, dialtest{ | 
					
						
							| 
									
										
										
										
											2017-04-10 19:33:41 +03:00
										 |  |  | 		init: newDialState(wantStatic, nil, fakeTable{}, 0, nil), | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 		rounds: []round{ | 
					
						
							|  |  |  | 			// Static dials are launched for the nodes that | 
					
						
							|  |  |  | 			// aren't yet connected. | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				peers: nil, | 
					
						
							|  |  |  | 				new: []task{ | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | 					&dialTask{flags: staticDialedConn, dest: &discover.Node{ID: uintID(1)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: staticDialedConn, dest: &discover.Node{ID: uintID(2)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: staticDialedConn, dest: &discover.Node{ID: uintID(3)}}, | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			// No new tasks are launched in this round because all static | 
					
						
							|  |  |  | 			// nodes are either connected or still being dialed. | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				peers: []*Peer{ | 
					
						
							|  |  |  | 					{rw: &conn{flags: staticDialedConn, id: uintID(1)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: staticDialedConn, id: uintID(2)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				done: []task{ | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | 					&dialTask{flags: staticDialedConn, dest: &discover.Node{ID: uintID(1)}}, | 
					
						
							|  |  |  | 					&dialTask{flags: staticDialedConn, dest: &discover.Node{ID: uintID(2)}}, | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			// A salvage task is launched to wait for node 3's history | 
					
						
							|  |  |  | 			// entry to expire. | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				peers: []*Peer{ | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(1)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(2)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				done: []task{ | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | 					&dialTask{flags: staticDialedConn, dest: &discover.Node{ID: uintID(3)}}, | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 				}, | 
					
						
							|  |  |  | 				new: []task{ | 
					
						
							|  |  |  | 					&waitExpireTask{Duration: 14 * time.Second}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			// Still waiting for node 3's entry to expire in the cache. | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				peers: []*Peer{ | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(1)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(2)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			// The cache entry for node 3 has expired and is retried. | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				peers: []*Peer{ | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(1)}}, | 
					
						
							|  |  |  | 					{rw: &conn{flags: dynDialedConn, id: uintID(2)}}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				new: []task{ | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | 					&dialTask{flags: staticDialedConn, dest: &discover.Node{ID: uintID(3)}}, | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | func TestDialResolve(t *testing.T) { | 
					
						
							|  |  |  | 	resolved := discover.NewNode(uintID(1), net.IP{127, 0, 55, 234}, 3333, 4444) | 
					
						
							|  |  |  | 	table := &resolveMock{answer: resolved} | 
					
						
							| 
									
										
										
										
											2017-04-10 19:33:41 +03:00
										 |  |  | 	state := newDialState(nil, nil, table, 0, nil) | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Check that the task is generated with an incomplete ID. | 
					
						
							|  |  |  | 	dest := discover.NewNode(uintID(1), nil, 0, 0) | 
					
						
							|  |  |  | 	state.addStatic(dest) | 
					
						
							|  |  |  | 	tasks := state.newTasks(0, nil, time.Time{}) | 
					
						
							|  |  |  | 	if !reflect.DeepEqual(tasks, []task{&dialTask{flags: staticDialedConn, dest: dest}}) { | 
					
						
							|  |  |  | 		t.Fatalf("expected dial task, got %#v", tasks) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Now run the task, it should resolve the ID once. | 
					
						
							| 
									
										
										
										
											2017-09-25 09:08:07 +01:00
										 |  |  | 	config := Config{Dialer: TCPDialer{&net.Dialer{Deadline: time.Now().Add(-5 * time.Minute)}}} | 
					
						
							| 
									
										
										
										
											2016-05-18 11:31:00 +02:00
										 |  |  | 	srv := &Server{ntab: table, Config: config} | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | 	tasks[0].Do(srv) | 
					
						
							|  |  |  | 	if !reflect.DeepEqual(table.resolveCalls, []discover.NodeID{dest.ID}) { | 
					
						
							|  |  |  | 		t.Fatalf("wrong resolve calls, got %v", table.resolveCalls) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Report it as done to the dialer, which should update the static node record. | 
					
						
							|  |  |  | 	state.taskDone(tasks[0], time.Now()) | 
					
						
							|  |  |  | 	if state.static[uintID(1)].dest != resolved { | 
					
						
							|  |  |  | 		t.Fatalf("state.dest not updated") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | // compares task lists but doesn't care about the order. | 
					
						
							|  |  |  | func sametasks(a, b []task) bool { | 
					
						
							|  |  |  | 	if len(a) != len(b) { | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | next: | 
					
						
							|  |  |  | 	for _, ta := range a { | 
					
						
							|  |  |  | 		for _, tb := range b { | 
					
						
							|  |  |  | 			if reflect.DeepEqual(ta, tb) { | 
					
						
							|  |  |  | 				continue next | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return true | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func uintID(i uint32) discover.NodeID { | 
					
						
							|  |  |  | 	var id discover.NodeID | 
					
						
							|  |  |  | 	binary.BigEndian.PutUint32(id[:], i) | 
					
						
							|  |  |  | 	return id | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-10-22 23:46:28 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | // implements discoverTable for TestDialResolve | 
					
						
							|  |  |  | type resolveMock struct { | 
					
						
							|  |  |  | 	resolveCalls []discover.NodeID | 
					
						
							|  |  |  | 	answer       *discover.Node | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (t *resolveMock) Resolve(id discover.NodeID) *discover.Node { | 
					
						
							|  |  |  | 	t.resolveCalls = append(t.resolveCalls, id) | 
					
						
							|  |  |  | 	return t.answer | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (t *resolveMock) Self() *discover.Node                     { return new(discover.Node) } | 
					
						
							|  |  |  | func (t *resolveMock) Close()                                   {} | 
					
						
							|  |  |  | func (t *resolveMock) Bootstrap([]*discover.Node)               {} | 
					
						
							|  |  |  | func (t *resolveMock) Lookup(discover.NodeID) []*discover.Node  { return nil } | 
					
						
							|  |  |  | func (t *resolveMock) ReadRandomNodes(buf []*discover.Node) int { return 0 } |