dht_net.go 3.41 KB
Newer Older
1 2 3 4 5 6
package dht

import (
	"errors"
	"time"

7 8 9 10
	inet "github.com/ipfs/go-ipfs/p2p/net"
	peer "github.com/ipfs/go-ipfs/p2p/peer"
	pb "github.com/ipfs/go-ipfs/routing/dht/pb"
	ctxutil "github.com/ipfs/go-ipfs/util/ctx"
11

12 13
	ggio "github.com/ipfs/go-ipfs/Godeps/_workspace/src/code.google.com/p/gogoprotobuf/io"
	context "github.com/ipfs/go-ipfs/Godeps/_workspace/src/golang.org/x/net/context"
14 15 16 17 18 19 20 21 22 23 24
)

// handleNewStream implements the inet.StreamHandler
func (dht *IpfsDHT) handleNewStream(s inet.Stream) {
	go dht.handleNewMessage(s)
}

func (dht *IpfsDHT) handleNewMessage(s inet.Stream) {
	defer s.Close()

	ctx := dht.Context()
25 26 27 28
	cr := ctxutil.NewReader(ctx, s) // ok to use. we defer close stream in this func
	cw := ctxutil.NewWriter(ctx, s) // ok to use. we defer close stream in this func
	r := ggio.NewDelimitedReader(cr, inet.MessageSizeMax)
	w := ggio.NewDelimitedWriter(cw)
29 30 31 32 33
	mPeer := s.Conn().RemotePeer()

	// receive msg
	pmes := new(pb.Message)
	if err := r.ReadMsg(pmes); err != nil {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
34
		log.Debugf("Error unmarshaling data: %s", err)
35 36
		return
	}
37

38
	// update the peer (on valid msgs only)
39
	dht.updateFromMessage(ctx, mPeer, pmes)
40 41 42 43

	// get handler for this msg type.
	handler := dht.handlerForMsgType(pmes.GetType())
	if handler == nil {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
44
		log.Debug("got back nil handler from handlerForMsgType")
45 46 47 48 49 50
		return
	}

	// dispatch handler.
	rpmes, err := handler(ctx, mPeer, pmes)
	if err != nil {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
51
		log.Debugf("handle message error: %s", err)
52 53 54 55 56 57 58 59 60 61 62
		return
	}

	// if nil response, return it before serializing
	if rpmes == nil {
		log.Warning("Got back nil response from request.")
		return
	}

	// send out response msg
	if err := w.WriteMsg(rpmes); err != nil {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
63
		log.Debugf("send response error: %s", err)
64 65 66 67 68 69 70 71
		return
	}

	return
}

// sendRequest sends out a request, but also makes sure to
// measure the RTT for latency measurements.
72
func (dht *IpfsDHT) sendRequest(ctx context.Context, p peer.ID, pmes *pb.Message) (*pb.Message, error) {
73 74

	log.Debugf("%s dht starting stream", dht.self)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
75
	s, err := dht.host.NewStream(ProtocolDHT, p)
76 77 78 79 80
	if err != nil {
		return nil, err
	}
	defer s.Close()

81 82 83 84
	cr := ctxutil.NewReader(ctx, s) // ok to use. we defer close stream in this func
	cw := ctxutil.NewWriter(ctx, s) // ok to use. we defer close stream in this func
	r := ggio.NewDelimitedReader(cr, inet.MessageSizeMax)
	w := ggio.NewDelimitedWriter(cw)
85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100

	start := time.Now()

	if err := w.WriteMsg(pmes); err != nil {
		return nil, err
	}
	log.Event(ctx, "dhtSentMessage", dht.self, p, pmes)

	rpmes := new(pb.Message)
	if err := r.ReadMsg(rpmes); err != nil {
		return nil, err
	}
	if rpmes == nil {
		return nil, errors.New("no response to request")
	}

101 102 103
	// update the peer (on valid msgs only)
	dht.updateFromMessage(ctx, p, rpmes)

104
	dht.peerstore.RecordLatency(p, time.Since(start))
105 106 107
	log.Event(ctx, "dhtReceivedMessage", dht.self, p, rpmes)
	return rpmes, nil
}
108 109

// sendMessage sends out a message
110
func (dht *IpfsDHT) sendMessage(ctx context.Context, p peer.ID, pmes *pb.Message) error {
111 112

	log.Debugf("%s dht starting stream", dht.self)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
113
	s, err := dht.host.NewStream(ProtocolDHT, p)
114 115 116 117 118
	if err != nil {
		return err
	}
	defer s.Close()

119 120
	cw := ctxutil.NewWriter(ctx, s) // ok to use. we defer close stream in this func
	w := ggio.NewDelimitedWriter(cw)
121 122 123 124 125 126 127

	if err := w.WriteMsg(pmes); err != nil {
		return err
	}
	log.Event(ctx, "dhtSentMessage", dht.self, p, pmes)
	return nil
}
128 129 130 131 132

func (dht *IpfsDHT) updateFromMessage(ctx context.Context, p peer.ID, mes *pb.Message) error {
	dht.Update(ctx, p)
	return nil
}