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
}