mirror of
https://github.com/ipfs/kubo.git
synced 2026-02-22 19:07:48 +08:00
humanize bandwidth output instrument conn.Conn for bandwidth metrics add poll command for continuous bandwidth reporting move bandwidth tracking onto multiaddr net connections another mild refactor of recording locations address concerns from PR lower mock nodes in race test due to increased goroutines per connection
238 lines
6.4 KiB
Go
238 lines
6.4 KiB
Go
package basichost
|
|
|
|
import (
|
|
ma "github.com/ipfs/go-ipfs/Godeps/_workspace/src/github.com/jbenet/go-multiaddr"
|
|
goprocess "github.com/ipfs/go-ipfs/Godeps/_workspace/src/github.com/jbenet/goprocess"
|
|
context "github.com/ipfs/go-ipfs/Godeps/_workspace/src/golang.org/x/net/context"
|
|
metrics "github.com/ipfs/go-ipfs/metrics"
|
|
mstream "github.com/ipfs/go-ipfs/metrics/stream"
|
|
eventlog "github.com/ipfs/go-ipfs/thirdparty/eventlog"
|
|
|
|
inet "github.com/ipfs/go-ipfs/p2p/net"
|
|
peer "github.com/ipfs/go-ipfs/p2p/peer"
|
|
protocol "github.com/ipfs/go-ipfs/p2p/protocol"
|
|
identify "github.com/ipfs/go-ipfs/p2p/protocol/identify"
|
|
relay "github.com/ipfs/go-ipfs/p2p/protocol/relay"
|
|
)
|
|
|
|
var log = eventlog.Logger("p2p/host/basic")
|
|
|
|
// Option is a type used to pass in options to the host.
|
|
type Option int
|
|
|
|
const (
|
|
// NATPortMap makes the host attempt to open port-mapping in NAT devices
|
|
// for all its listeners. Pass in this option in the constructor to
|
|
// asynchronously a) find a gateway, b) open port mappings, c) republish
|
|
// port mappings periodically. The NATed addresses are included in the
|
|
// Host's Addrs() list.
|
|
NATPortMap Option = iota
|
|
)
|
|
|
|
// BasicHost is the basic implementation of the host.Host interface. This
|
|
// particular host implementation:
|
|
// * uses a protocol muxer to mux per-protocol streams
|
|
// * uses an identity service to send + receive node information
|
|
// * uses a relay service to allow hosts to relay conns for each other
|
|
// * uses a nat service to establish NAT port mappings
|
|
type BasicHost struct {
|
|
network inet.Network
|
|
mux *protocol.Mux
|
|
ids *identify.IDService
|
|
relay *relay.RelayService
|
|
natmgr *natManager
|
|
|
|
proc goprocess.Process
|
|
|
|
bwc metrics.Reporter
|
|
}
|
|
|
|
// New constructs and sets up a new *BasicHost with given Network
|
|
func New(net inet.Network, opts ...interface{}) *BasicHost {
|
|
h := &BasicHost{
|
|
network: net,
|
|
mux: protocol.NewMux(),
|
|
bwc: metrics.NewBandwidthCounter(),
|
|
}
|
|
|
|
h.proc = goprocess.WithTeardown(func() error {
|
|
if h.natmgr != nil {
|
|
h.natmgr.Close()
|
|
}
|
|
|
|
return h.Network().Close()
|
|
})
|
|
|
|
// setup host services
|
|
h.ids = identify.NewIDService(h)
|
|
h.relay = relay.NewRelayService(h, h.Mux().HandleSync)
|
|
|
|
for _, o := range opts {
|
|
switch o := o.(type) {
|
|
case Option:
|
|
switch o {
|
|
case NATPortMap:
|
|
h.natmgr = newNatManager(h)
|
|
}
|
|
case metrics.Reporter:
|
|
h.bwc = o
|
|
}
|
|
}
|
|
|
|
net.SetConnHandler(h.newConnHandler)
|
|
net.SetStreamHandler(h.newStreamHandler)
|
|
|
|
return h
|
|
}
|
|
|
|
// newConnHandler is the remote-opened conn handler for inet.Network
|
|
func (h *BasicHost) newConnHandler(c inet.Conn) {
|
|
h.ids.IdentifyConn(c)
|
|
}
|
|
|
|
// newStreamHandler is the remote-opened stream handler for inet.Network
|
|
// TODO: this feels a bit wonky
|
|
func (h *BasicHost) newStreamHandler(s inet.Stream) {
|
|
protoID, handle, err := h.Mux().ReadHeader(s)
|
|
if err != nil {
|
|
log.Error("protocol mux failed: %s", err)
|
|
return
|
|
}
|
|
|
|
logStream := mstream.WrapStream(s, protoID, h.bwc)
|
|
|
|
go handle(logStream)
|
|
}
|
|
|
|
// ID returns the (local) peer.ID associated with this Host
|
|
func (h *BasicHost) ID() peer.ID {
|
|
return h.Network().LocalPeer()
|
|
}
|
|
|
|
// Peerstore returns the Host's repository of Peer Addresses and Keys.
|
|
func (h *BasicHost) Peerstore() peer.Peerstore {
|
|
return h.Network().Peerstore()
|
|
}
|
|
|
|
// Networks returns the Network interface of the Host
|
|
func (h *BasicHost) Network() inet.Network {
|
|
return h.network
|
|
}
|
|
|
|
// Mux returns the Mux multiplexing incoming streams to protocol handlers
|
|
func (h *BasicHost) Mux() *protocol.Mux {
|
|
return h.mux
|
|
}
|
|
|
|
func (h *BasicHost) IDService() *identify.IDService {
|
|
return h.ids
|
|
}
|
|
|
|
// SetStreamHandler sets the protocol handler on the Host's Mux.
|
|
// This is equivalent to:
|
|
// host.Mux().SetHandler(proto, handler)
|
|
// (Threadsafe)
|
|
func (h *BasicHost) SetStreamHandler(pid protocol.ID, handler inet.StreamHandler) {
|
|
h.Mux().SetHandler(pid, handler)
|
|
}
|
|
|
|
func (h *BasicHost) RemoveStreamHandler(pid protocol.ID) {
|
|
h.Mux().RemoveHandler(pid)
|
|
}
|
|
|
|
// NewStream opens a new stream to given peer p, and writes a p2p/protocol
|
|
// header with given protocol.ID. If there is no connection to p, attempts
|
|
// to create one. If ProtocolID is "", writes no header.
|
|
// (Threadsafe)
|
|
func (h *BasicHost) NewStream(pid protocol.ID, p peer.ID) (inet.Stream, error) {
|
|
s, err := h.Network().NewStream(p)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
logStream := mstream.WrapStream(s, pid, h.bwc)
|
|
|
|
if err := protocol.WriteHeader(logStream, pid); err != nil {
|
|
logStream.Close()
|
|
return nil, err
|
|
}
|
|
|
|
return logStream, nil
|
|
}
|
|
|
|
// Connect ensures there is a connection between this host and the peer with
|
|
// given peer.ID. Connect will absorb the addresses in pi into its internal
|
|
// peerstore. If there is not an active connection, Connect will issue a
|
|
// h.Network.Dial, and block until a connection is open, or an error is
|
|
// returned. // TODO: Relay + NAT.
|
|
func (h *BasicHost) Connect(ctx context.Context, pi peer.PeerInfo) error {
|
|
|
|
// absorb addresses into peerstore
|
|
h.Peerstore().AddAddrs(pi.ID, pi.Addrs, peer.TempAddrTTL)
|
|
|
|
cs := h.Network().ConnsToPeer(pi.ID)
|
|
if len(cs) > 0 {
|
|
return nil
|
|
}
|
|
|
|
return h.dialPeer(ctx, pi.ID)
|
|
}
|
|
|
|
// dialPeer opens a connection to peer, and makes sure to identify
|
|
// the connection once it has been opened.
|
|
func (h *BasicHost) dialPeer(ctx context.Context, p peer.ID) error {
|
|
log.Debugf("host %s dialing %s", h.ID, p)
|
|
c, err := h.Network().DialPeer(ctx, p)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// identify the connection before returning.
|
|
done := make(chan struct{})
|
|
go func() {
|
|
h.ids.IdentifyConn(c)
|
|
close(done)
|
|
}()
|
|
|
|
// respect don contexteone
|
|
select {
|
|
case <-done:
|
|
case <-ctx.Done():
|
|
return ctx.Err()
|
|
}
|
|
|
|
log.Debugf("host %s finished dialing %s", h.ID, p)
|
|
return nil
|
|
}
|
|
|
|
// Addrs returns all the addresses of BasicHost at this moment in time.
|
|
// It's ok to not include addresses if they're not available to be used now.
|
|
func (h *BasicHost) Addrs() []ma.Multiaddr {
|
|
addrs, err := h.Network().InterfaceListenAddresses()
|
|
if err != nil {
|
|
log.Debug("error retrieving network interface addrs")
|
|
}
|
|
|
|
if h.ids != nil { // add external observed addresses
|
|
addrs = append(addrs, h.ids.OwnObservedAddrs()...)
|
|
}
|
|
|
|
if h.natmgr != nil { // natmgr is nil if we do not use nat option.
|
|
nat := h.natmgr.NAT()
|
|
if nat != nil { // nat is nil if not ready, or no nat is available.
|
|
addrs = append(addrs, nat.ExternalAddrs()...)
|
|
}
|
|
}
|
|
|
|
return addrs
|
|
}
|
|
|
|
// Close shuts down the Host's services (network, etc).
|
|
func (h *BasicHost) Close() error {
|
|
return h.proc.Close()
|
|
}
|
|
|
|
func (h *BasicHost) GetBandwidthReporter() metrics.Reporter {
|
|
return h.bwc
|
|
}
|