basic_host.go 5.29 KB
Newer Older
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
1 2 3
package basichost

import (
4 5
	"sync"

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
6
	context "github.com/jbenet/go-ipfs/Godeps/_workspace/src/code.google.com/p/go.net/context"
7 8
	ma "github.com/jbenet/go-ipfs/Godeps/_workspace/src/github.com/jbenet/go-multiaddr"
	goprocess "github.com/jbenet/go-ipfs/Godeps/_workspace/src/github.com/jbenet/goprocess"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
9

10
	eventlog "github.com/jbenet/go-ipfs/thirdparty/eventlog"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
11

12
	inat "github.com/jbenet/go-ipfs/p2p/nat"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
13
	inet "github.com/jbenet/go-ipfs/p2p/net"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
14 15 16 17 18 19 20 21
	peer "github.com/jbenet/go-ipfs/p2p/peer"
	protocol "github.com/jbenet/go-ipfs/p2p/protocol"
	identify "github.com/jbenet/go-ipfs/p2p/protocol/identify"
	relay "github.com/jbenet/go-ipfs/p2p/protocol/relay"
)

var log = eventlog.Logger("p2p/host/basic")

22 23 24 25 26 27
type Option int

const (
	NATPortMap Option = iota
)

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
28 29
type BasicHost struct {
	network inet.Network
Brian Tiger Chow's avatar
Brian Tiger Chow committed
30
	mux     *protocol.Mux
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
31 32
	ids     *identify.IDService
	relay   *relay.RelayService
33 34 35 36 37

	natmu sync.Mutex
	nat   *inat.NAT

	proc goprocess.Process
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
38 39 40
}

// New constructs and sets up a new *BasicHost with given Network
41
func New(net inet.Network, opts ...Option) *BasicHost {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
42 43
	h := &BasicHost{
		network: net,
Brian Tiger Chow's avatar
Brian Tiger Chow committed
44
		mux:     protocol.NewMux(),
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
45 46
	}

47 48 49 50
	h.proc = goprocess.WithTeardown(func() error {
		return h.Network().Close()
	})

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
51 52 53 54 55 56 57
	// setup host services
	h.ids = identify.NewIDService(h)
	h.relay = relay.NewRelayService(h, h.Mux().HandleSync)

	net.SetConnHandler(h.newConnHandler)
	net.SetStreamHandler(h.newStreamHandler)

58 59 60 61 62 63 64
	for _, o := range opts {
		switch o {
		case NATPortMap:
			h.setupNATPortMap()
		}
	}

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
65 66 67
	return h
}

68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99
func (h *BasicHost) setupNATPortMap() {
	// do this asynchronously to avoid blocking daemon startup

	h.proc.Go(func(worker goprocess.Process) {
		nat := inat.DiscoverNAT()
		if nat == nil { // no nat, or failed to get it.
			return
		}

		select {
		case <-worker.Closing():
			nat.Close()
			return
		default:
		}

		// wire up the nat to close when proc closes.
		h.proc.AddChild(nat.Process())

		h.natmu.Lock()
		h.nat = nat
		h.natmu.Unlock()

		addrs := h.Network().ListenAddresses()
		nat.PortMapAddrs(addrs)
		mapAddrs := nat.ExternalAddrs()
		if len(mapAddrs) > 0 {
			log.Infof("NAT mapping addrs: %s", mapAddrs)
		}
	})
}

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126
// 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
func (h *BasicHost) newStreamHandler(s inet.Stream) {
	h.Mux().Handle(s)
}

// 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 {
Brian Tiger Chow's avatar
Brian Tiger Chow committed
127
	return h.mux
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204
}

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)
}

// 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
	}

	if err := protocol.WriteHeader(s, pid); err != nil {
		s.Close()
		return nil, err
	}

	return s, 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().AddPeerInfo(pi)

	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
}

205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220
func (h *BasicHost) Addrs() []ma.Multiaddr {
	addrs, err := h.Network().InterfaceListenAddresses()
	if err != nil {
		log.Debug("error retrieving network interface addrs")
	}

	h.natmu.Lock()
	nat := h.nat
	h.natmu.Unlock()
	if nat != nil {
		addrs = append(addrs, nat.ExternalAddrs()...)
	}

	return addrs
}

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
221 222
// Close shuts down the Host's services (network, etc).
func (h *BasicHost) Close() error {
223
	return h.proc.Close()
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
224
}