| 
									
										
										
										
											2014-10-23 16:57:54 +01:00
										 |  |  | package p2p | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-21 21:48:49 +01:00
										 |  |  | import ( | 
					
						
							| 
									
										
										
										
											2015-04-08 17:37:11 +02:00
										 |  |  | 	"errors" | 
					
						
							| 
									
										
										
										
											2015-02-05 03:07:58 +01:00
										 |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2015-04-08 17:37:11 +02:00
										 |  |  | 	"math/rand" | 
					
						
							| 
									
										
										
										
											2014-11-21 21:48:49 +01:00
										 |  |  | 	"net" | 
					
						
							| 
									
										
										
										
											2014-11-26 22:49:40 +01:00
										 |  |  | 	"reflect" | 
					
						
							| 
									
										
										
										
											2014-11-21 21:48:49 +01:00
										 |  |  | 	"testing" | 
					
						
							|  |  |  | 	"time" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var discard = Protocol{ | 
					
						
							|  |  |  | 	Name:   "discard", | 
					
						
							|  |  |  | 	Length: 1, | 
					
						
							|  |  |  | 	Run: func(p *Peer, rw MsgReadWriter) error { | 
					
						
							|  |  |  | 		for { | 
					
						
							|  |  |  | 			msg, err := rw.ReadMsg() | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				return err | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2015-02-19 01:52:03 +01:00
										 |  |  | 			fmt.Printf("discarding %d\n", msg.Code) | 
					
						
							| 
									
										
										
										
											2014-11-21 21:48:49 +01:00
										 |  |  | 			if err = msg.Discard(); err != nil { | 
					
						
							|  |  |  | 				return err | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-08 17:37:11 +02:00
										 |  |  | func testPeer(protos []Protocol) (func(), *conn, *Peer, <-chan DiscReason) { | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 	fd1, fd2 := net.Pipe() | 
					
						
							|  |  |  | 	c1 := &conn{fd: fd1, transport: newTestTransport(randomID(), fd1)} | 
					
						
							|  |  |  | 	c2 := &conn{fd: fd2, transport: newTestTransport(randomID(), fd2)} | 
					
						
							| 
									
										
										
										
											2015-02-19 01:52:03 +01:00
										 |  |  | 	for _, p := range protos { | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 		c1.caps = append(c1.caps, p.cap()) | 
					
						
							|  |  |  | 		c2.caps = append(c2.caps, p.cap()) | 
					
						
							| 
									
										
										
										
											2015-02-19 01:52:03 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 	peer := newPeer(c1, protos) | 
					
						
							| 
									
										
										
										
											2015-02-05 03:07:58 +01:00
										 |  |  | 	errc := make(chan DiscReason, 1) | 
					
						
							|  |  |  | 	go func() { errc <- peer.run() }() | 
					
						
							| 
									
										
										
										
											2015-02-19 01:52:03 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-16 00:38:28 +02:00
										 |  |  | 	closer := func() { c2.close(errors.New("close func called")) } | 
					
						
							|  |  |  | 	return closer, c2, peer, errc | 
					
						
							| 
									
										
										
										
											2014-11-21 21:48:49 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestPeerProtoReadMsg(t *testing.T) { | 
					
						
							|  |  |  | 	done := make(chan struct{}) | 
					
						
							|  |  |  | 	proto := Protocol{ | 
					
						
							|  |  |  | 		Name:   "a", | 
					
						
							|  |  |  | 		Length: 5, | 
					
						
							|  |  |  | 		Run: func(peer *Peer, rw MsgReadWriter) error { | 
					
						
							| 
									
										
										
										
											2015-03-19 15:11:02 +01:00
										 |  |  | 			if err := ExpectMsg(rw, 2, []uint{1}); err != nil { | 
					
						
							| 
									
										
										
										
											2015-02-05 03:07:58 +01:00
										 |  |  | 				t.Error(err) | 
					
						
							| 
									
										
										
										
											2014-11-21 21:48:49 +01:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2015-03-19 15:11:02 +01:00
										 |  |  | 			if err := ExpectMsg(rw, 3, []uint{2}); err != nil { | 
					
						
							| 
									
										
										
										
											2015-02-05 03:07:58 +01:00
										 |  |  | 				t.Error(err) | 
					
						
							| 
									
										
										
										
											2014-11-21 21:48:49 +01:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2015-03-19 15:11:02 +01:00
										 |  |  | 			if err := ExpectMsg(rw, 4, []uint{3}); err != nil { | 
					
						
							| 
									
										
										
										
											2015-02-05 03:07:58 +01:00
										 |  |  | 				t.Error(err) | 
					
						
							| 
									
										
										
										
											2014-11-21 21:48:49 +01:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			close(done) | 
					
						
							|  |  |  | 			return nil | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-27 03:06:55 +00:00
										 |  |  | 	closer, rw, _, errc := testPeer([]Protocol{proto}) | 
					
						
							| 
									
										
										
										
											2015-04-08 17:37:11 +02:00
										 |  |  | 	defer closer() | 
					
						
							| 
									
										
										
										
											2014-11-21 21:48:49 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-19 15:11:02 +01:00
										 |  |  | 	Send(rw, baseProtocolLength+2, []uint{1}) | 
					
						
							|  |  |  | 	Send(rw, baseProtocolLength+3, []uint{2}) | 
					
						
							|  |  |  | 	Send(rw, baseProtocolLength+4, []uint{3}) | 
					
						
							| 
									
										
										
										
											2015-02-05 03:07:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-21 21:48:49 +01:00
										 |  |  | 	select { | 
					
						
							|  |  |  | 	case <-done: | 
					
						
							|  |  |  | 	case err := <-errc: | 
					
						
							|  |  |  | 		t.Errorf("peer returned: %v", err) | 
					
						
							|  |  |  | 	case <-time.After(2 * time.Second): | 
					
						
							|  |  |  | 		t.Errorf("receive timeout") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestPeerProtoEncodeMsg(t *testing.T) { | 
					
						
							|  |  |  | 	proto := Protocol{ | 
					
						
							|  |  |  | 		Name:   "a", | 
					
						
							|  |  |  | 		Length: 2, | 
					
						
							|  |  |  | 		Run: func(peer *Peer, rw MsgReadWriter) error { | 
					
						
							| 
									
										
										
										
											2015-03-19 15:11:02 +01:00
										 |  |  | 			if err := SendItems(rw, 2); err == nil { | 
					
						
							| 
									
										
										
										
											2014-11-21 21:48:49 +01:00
										 |  |  | 				t.Error("expected error for out-of-range msg code, got nil") | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2015-03-19 15:11:02 +01:00
										 |  |  | 			if err := SendItems(rw, 1, "foo", "bar"); err != nil { | 
					
						
							| 
									
										
										
										
											2014-11-21 21:48:49 +01:00
										 |  |  | 				t.Errorf("write error: %v", err) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			return nil | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-02-27 03:06:55 +00:00
										 |  |  | 	closer, rw, _, _ := testPeer([]Protocol{proto}) | 
					
						
							| 
									
										
										
										
											2015-04-08 17:37:11 +02:00
										 |  |  | 	defer closer() | 
					
						
							| 
									
										
										
										
											2014-11-21 21:48:49 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-19 15:11:02 +01:00
										 |  |  | 	if err := ExpectMsg(rw, 17, []string{"foo", "bar"}); err != nil { | 
					
						
							| 
									
										
										
										
											2015-02-05 03:07:58 +01:00
										 |  |  | 		t.Error(err) | 
					
						
							| 
									
										
										
										
											2015-01-05 17:10:42 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-11-21 21:48:49 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-05 03:07:58 +01:00
										 |  |  | func TestPeerPing(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2015-02-27 03:06:55 +00:00
										 |  |  | 	closer, rw, _, _ := testPeer(nil) | 
					
						
							| 
									
										
										
										
											2015-04-08 17:37:11 +02:00
										 |  |  | 	defer closer() | 
					
						
							| 
									
										
										
										
											2015-03-19 15:11:02 +01:00
										 |  |  | 	if err := SendItems(rw, pingMsg); err != nil { | 
					
						
							| 
									
										
										
										
											2015-02-05 03:07:58 +01:00
										 |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-03-19 15:11:02 +01:00
										 |  |  | 	if err := ExpectMsg(rw, pongMsg, nil); err != nil { | 
					
						
							| 
									
										
										
										
											2015-02-05 03:07:58 +01:00
										 |  |  | 		t.Error(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-11-21 21:48:49 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-05 03:07:58 +01:00
										 |  |  | func TestPeerDisconnect(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2015-02-27 03:06:55 +00:00
										 |  |  | 	closer, rw, _, disc := testPeer(nil) | 
					
						
							| 
									
										
										
										
											2015-04-08 17:37:11 +02:00
										 |  |  | 	defer closer() | 
					
						
							| 
									
										
										
										
											2015-03-19 15:11:02 +01:00
										 |  |  | 	if err := SendItems(rw, discMsg, DiscQuitting); err != nil { | 
					
						
							| 
									
										
										
										
											2015-02-05 03:07:58 +01:00
										 |  |  | 		t.Fatal(err) | 
					
						
							| 
									
										
										
										
											2014-11-21 21:48:49 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-05-14 03:04:04 +02:00
										 |  |  | 	select { | 
					
						
							|  |  |  | 	case reason := <-disc: | 
					
						
							|  |  |  | 		if reason != DiscQuitting { | 
					
						
							|  |  |  | 			t.Errorf("run returned wrong reason: got %v, want %v", reason, DiscRequested) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	case <-time.After(500 * time.Millisecond): | 
					
						
							|  |  |  | 		t.Error("peer did not return") | 
					
						
							| 
									
										
										
										
											2014-11-21 21:48:49 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-11-26 22:49:40 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-08 17:37:11 +02:00
										 |  |  | // This test is supposed to verify that Peer can reliably handle | 
					
						
							|  |  |  | // multiple causes of disconnection occurring at the same time. | 
					
						
							|  |  |  | func TestPeerDisconnectRace(t *testing.T) { | 
					
						
							|  |  |  | 	maybe := func() bool { return rand.Intn(1) == 1 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for i := 0; i < 1000; i++ { | 
					
						
							|  |  |  | 		protoclose := make(chan error) | 
					
						
							|  |  |  | 		protodisc := make(chan DiscReason) | 
					
						
							|  |  |  | 		closer, rw, p, disc := testPeer([]Protocol{ | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				Name:   "closereq", | 
					
						
							|  |  |  | 				Run:    func(p *Peer, rw MsgReadWriter) error { return <-protoclose }, | 
					
						
							|  |  |  | 				Length: 1, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				Name:   "disconnect", | 
					
						
							|  |  |  | 				Run:    func(p *Peer, rw MsgReadWriter) error { p.Disconnect(<-protodisc); return nil }, | 
					
						
							|  |  |  | 				Length: 1, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Simulate incoming messages. | 
					
						
							|  |  |  | 		go SendItems(rw, baseProtocolLength+1) | 
					
						
							|  |  |  | 		go SendItems(rw, baseProtocolLength+2) | 
					
						
							|  |  |  | 		// Close the network connection. | 
					
						
							|  |  |  | 		go closer() | 
					
						
							|  |  |  | 		// Make protocol "closereq" return. | 
					
						
							|  |  |  | 		protoclose <- errors.New("protocol closed") | 
					
						
							|  |  |  | 		// Make protocol "disconnect" call peer.Disconnect | 
					
						
							|  |  |  | 		protodisc <- DiscAlreadyConnected | 
					
						
							|  |  |  | 		// In some cases, simulate something else calling peer.Disconnect. | 
					
						
							|  |  |  | 		if maybe() { | 
					
						
							|  |  |  | 			go p.Disconnect(DiscInvalidIdentity) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		// In some cases, simulate remote requesting a disconnect. | 
					
						
							|  |  |  | 		if maybe() { | 
					
						
							|  |  |  | 			go SendItems(rw, discMsg, DiscQuitting) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		select { | 
					
						
							|  |  |  | 		case <-disc: | 
					
						
							|  |  |  | 		case <-time.After(2 * time.Second): | 
					
						
							|  |  |  | 			// Peer.run should return quickly. If it doesn't the Peer | 
					
						
							|  |  |  | 			// goroutines are probably deadlocked. Call panic in order to | 
					
						
							|  |  |  | 			// show the stacks. | 
					
						
							|  |  |  | 			panic("Peer.run took to long to return.") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-05 03:07:58 +01:00
										 |  |  | func TestNewPeer(t *testing.T) { | 
					
						
							|  |  |  | 	name := "nodename" | 
					
						
							|  |  |  | 	caps := []Cap{{"foo", 2}, {"bar", 3}} | 
					
						
							|  |  |  | 	id := randomID() | 
					
						
							|  |  |  | 	p := NewPeer(id, name, caps) | 
					
						
							|  |  |  | 	if p.ID() != id { | 
					
						
							|  |  |  | 		t.Errorf("ID mismatch: got %v, expected %v", p.ID(), id) | 
					
						
							| 
									
										
										
										
											2014-12-12 11:38:42 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-02-05 03:07:58 +01:00
										 |  |  | 	if p.Name() != name { | 
					
						
							|  |  |  | 		t.Errorf("Name mismatch: got %v, expected %v", p.Name(), name) | 
					
						
							| 
									
										
										
										
											2014-12-12 11:38:42 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-02-05 03:07:58 +01:00
										 |  |  | 	if !reflect.DeepEqual(p.Caps(), caps) { | 
					
						
							|  |  |  | 		t.Errorf("Caps mismatch: got %v, expected %v", p.Caps(), caps) | 
					
						
							| 
									
										
										
										
											2014-12-12 11:38:42 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-05 03:07:58 +01:00
										 |  |  | 	p.Disconnect(DiscAlreadyConnected) // Should not hang | 
					
						
							|  |  |  | } |