mirror of
https://github.com/ipfs/kubo.git
synced 2026-02-24 20:07:45 +08:00
181 lines
4.7 KiB
Go
181 lines
4.7 KiB
Go
package net
|
|
|
|
import (
|
|
"sync"
|
|
|
|
handshake "github.com/jbenet/go-ipfs/net/handshake"
|
|
pb "github.com/jbenet/go-ipfs/net/handshake/pb"
|
|
|
|
ggio "github.com/jbenet/go-ipfs/Godeps/_workspace/src/code.google.com/p/gogoprotobuf/io"
|
|
ma "github.com/jbenet/go-ipfs/Godeps/_workspace/src/github.com/jbenet/go-multiaddr"
|
|
)
|
|
|
|
// IDService is a structure that implements ProtocolIdentify.
|
|
// It is a trivial service that gives the other peer some
|
|
// useful information about the local peer. A sort of hello.
|
|
//
|
|
// The IDService sends:
|
|
// * Our IPFS Protocol Version
|
|
// * Our IPFS Agent Version
|
|
// * Our public Listen Addresses
|
|
type IDService struct {
|
|
Network Network
|
|
|
|
// connections undergoing identification
|
|
// for wait purposes
|
|
currid map[Conn]chan struct{}
|
|
currmu sync.RWMutex
|
|
}
|
|
|
|
func NewIDService(n Network) *IDService {
|
|
s := &IDService{
|
|
Network: n,
|
|
currid: make(map[Conn]chan struct{}),
|
|
}
|
|
n.SetHandler(ProtocolIdentify, s.RequestHandler)
|
|
return s
|
|
}
|
|
|
|
func (ids *IDService) IdentifyConn(c Conn) {
|
|
ids.currmu.Lock()
|
|
if wait, found := ids.currid[c]; found {
|
|
ids.currmu.Unlock()
|
|
log.Debugf("IdentifyConn called twice on: %s", c)
|
|
<-wait // already identifying it. wait for it.
|
|
return
|
|
}
|
|
ids.currid[c] = make(chan struct{})
|
|
ids.currmu.Unlock()
|
|
|
|
s, err := c.NewStreamWithProtocol(ProtocolIdentify)
|
|
if err != nil {
|
|
log.Error("network: unable to open initial stream for %s", ProtocolIdentify)
|
|
log.Event(ids.Network.CtxGroup().Context(), "IdentifyOpenFailed", c.RemotePeer())
|
|
} else {
|
|
|
|
// ok give the response to our handler.
|
|
ids.ResponseHandler(s)
|
|
}
|
|
|
|
ids.currmu.Lock()
|
|
ch, found := ids.currid[c]
|
|
delete(ids.currid, c)
|
|
ids.currmu.Unlock()
|
|
|
|
if !found {
|
|
log.Errorf("IdentifyConn failed to find channel (programmer error) for %s", c)
|
|
return
|
|
}
|
|
|
|
close(ch) // release everyone waiting.
|
|
}
|
|
|
|
func (ids *IDService) RequestHandler(s Stream) {
|
|
defer s.Close()
|
|
c := s.Conn()
|
|
|
|
w := ggio.NewDelimitedWriter(s)
|
|
mes := pb.Handshake3{}
|
|
ids.populateMessage(&mes, s.Conn())
|
|
w.WriteMsg(&mes)
|
|
|
|
log.Debugf("%s sent message to %s %s", ProtocolIdentify,
|
|
c.RemotePeer(), c.RemoteMultiaddr())
|
|
}
|
|
|
|
func (ids *IDService) ResponseHandler(s Stream) {
|
|
defer s.Close()
|
|
c := s.Conn()
|
|
|
|
r := ggio.NewDelimitedReader(s, 2048)
|
|
mes := pb.Handshake3{}
|
|
if err := r.ReadMsg(&mes); err != nil {
|
|
log.Errorf("%s error receiving message from %s %s", ProtocolIdentify,
|
|
c.RemotePeer(), c.RemoteMultiaddr())
|
|
return
|
|
}
|
|
ids.consumeMessage(&mes, c)
|
|
|
|
log.Debugf("%s received message from %s %s", ProtocolIdentify,
|
|
c.RemotePeer(), c.RemoteMultiaddr())
|
|
}
|
|
|
|
func (ids *IDService) populateMessage(mes *pb.Handshake3, c Conn) {
|
|
|
|
// set protocols this node is currently handling
|
|
protos := ids.Network.Protocols()
|
|
mes.Protocols = make([]string, len(protos))
|
|
for i, p := range protos {
|
|
mes.Protocols[i] = string(p)
|
|
}
|
|
|
|
// observed address so other side is informed of their
|
|
// "public" address, at least in relation to us.
|
|
mes.ObservedAddr = c.RemoteMultiaddr().Bytes()
|
|
|
|
// set listen addrs
|
|
laddrs, err := ids.Network.InterfaceListenAddresses()
|
|
if err != nil {
|
|
log.Error(err)
|
|
} else {
|
|
mes.ListenAddrs = make([][]byte, len(laddrs))
|
|
for i, addr := range laddrs {
|
|
mes.ListenAddrs[i] = addr.Bytes()
|
|
}
|
|
log.Debugf("%s sent listen addrs to %s: %s", c.LocalPeer(), c.RemotePeer(), laddrs)
|
|
}
|
|
|
|
// set protocol versions
|
|
mes.H1 = handshake.NewHandshake1("", "")
|
|
}
|
|
|
|
func (ids *IDService) consumeMessage(mes *pb.Handshake3, c Conn) {
|
|
p := c.RemotePeer()
|
|
|
|
// mes.Protocols
|
|
// mes.ObservedAddr
|
|
|
|
// mes.ListenAddrs
|
|
laddrs := mes.GetListenAddrs()
|
|
lmaddrs := make([]ma.Multiaddr, 0, len(laddrs))
|
|
for _, addr := range laddrs {
|
|
maddr, err := ma.NewMultiaddrBytes(addr)
|
|
if err != nil {
|
|
log.Errorf("%s failed to parse multiaddr from %s %s", ProtocolIdentify, p,
|
|
c.RemoteMultiaddr())
|
|
continue
|
|
}
|
|
lmaddrs = append(lmaddrs, maddr)
|
|
}
|
|
|
|
// update our peerstore with the addresses.
|
|
ids.Network.Peerstore().AddAddresses(p, lmaddrs)
|
|
log.Debugf("%s received listen addrs for %s: %s", c.LocalPeer(), c.RemotePeer(), lmaddrs)
|
|
|
|
// get protocol versions
|
|
pv := *mes.H1.ProtocolVersion
|
|
av := *mes.H1.AgentVersion
|
|
ids.Network.Peerstore().Put(p, "ProtocolVersion", pv)
|
|
ids.Network.Peerstore().Put(p, "AgentVersion", av)
|
|
}
|
|
|
|
// IdentifyWait returns a channel which will be closed once
|
|
// "ProtocolIdentify" (handshake3) finishes on given conn.
|
|
// This happens async so the connection can start to be used
|
|
// even if handshake3 knowledge is not necesary.
|
|
// Users **MUST** call IdentifyWait _after_ IdentifyConn
|
|
func (ids *IDService) IdentifyWait(c Conn) <-chan struct{} {
|
|
ids.currmu.Lock()
|
|
ch, found := ids.currid[c]
|
|
ids.currmu.Unlock()
|
|
if found {
|
|
return ch
|
|
}
|
|
|
|
// if not found, it means we are already done identifying it, or
|
|
// haven't even started. either way, return a new channel closed.
|
|
ch = make(chan struct{})
|
|
close(ch)
|
|
return ch
|
|
}
|