core.go 24 KB
Newer Older
1 2 3 4
/*
Package core implements the IpfsNode object and related methods.

Packages underneath core/ provide a (relatively) stable, low-level API
5 6 7 8
to carry out most IPFS-related tasks.  For more details on the other
interfaces and how core/... fits into the bigger IPFS picture, see:

  $ godoc github.com/ipfs/go-ipfs
9
*/
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
10 11
package core

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
12
import (
Jakub Sztandera's avatar
Jakub Sztandera committed
13
	"bytes"
14
	"context"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
15
	"errors"
Juan Batiz-Benet's avatar
go fmt  
Juan Batiz-Benet committed
16
	"fmt"
17
	"io"
18
	"io/ioutil"
Jeromy's avatar
Jeromy committed
19
	"net"
20 21
	"os"
	"strings"
22
	"time"
23

24 25 26 27 28 29
	bstore "github.com/ipfs/go-ipfs/blocks/blockstore"
	bserv "github.com/ipfs/go-ipfs/blockservice"
	exchange "github.com/ipfs/go-ipfs/exchange"
	bitswap "github.com/ipfs/go-ipfs/exchange/bitswap"
	bsnet "github.com/ipfs/go-ipfs/exchange/bitswap/network"
	rp "github.com/ipfs/go-ipfs/exchange/reprovide"
30
	filestore "github.com/ipfs/go-ipfs/filestore"
31 32 33 34 35
	mount "github.com/ipfs/go-ipfs/fuse/mount"
	merkledag "github.com/ipfs/go-ipfs/merkledag"
	mfs "github.com/ipfs/go-ipfs/mfs"
	namesys "github.com/ipfs/go-ipfs/namesys"
	ipnsrp "github.com/ipfs/go-ipfs/namesys/republisher"
Łukasz Magiera's avatar
Łukasz Magiera committed
36
	p2p "github.com/ipfs/go-ipfs/p2p"
37 38 39 40 41 42 43
	path "github.com/ipfs/go-ipfs/path"
	pin "github.com/ipfs/go-ipfs/pin"
	repo "github.com/ipfs/go-ipfs/repo"
	config "github.com/ipfs/go-ipfs/repo/config"
	nilrouting "github.com/ipfs/go-ipfs/routing/none"
	offroute "github.com/ipfs/go-ipfs/routing/offline"
	ft "github.com/ipfs/go-ipfs/unixfs"
Jeromy's avatar
Jeromy committed
44

Steven Allen's avatar
Steven Allen committed
45
	pnet "gx/ipfs/QmNMCAuxnQFHLGWcvay3DmVFrKuY6Y2nsc9vzsf4gVouJV/go-libp2p-pnet"
46 47
	pstore "gx/ipfs/QmPgDWmTmuzvP7QE5zwo1TmjbJme9pmZHNujB2453jkCTr/go-libp2p-peerstore"
	routing "gx/ipfs/QmPjTrrSfE6TzLv6ya6VWhGcCgPrUAdcgrDcQyRDX2VyW1/go-libp2p-routing"
48
	mafilter "gx/ipfs/QmQBB2dQLmQHJgs2gqZ3iqL2XiuCtUCvXzWt5kMXDf5Zcr/go-maddr-filter"
Steven Allen's avatar
Steven Allen committed
49
	floodsub "gx/ipfs/QmQCm4CY8hXwYnaEZ5E2iRWFmWWX9xGCTABySyLbfyqXNh/floodsub"
50
	ipnet "gx/ipfs/QmQq9YzmdFdWNTDdArueGyD7L5yyiRQigrRHJnTGkxcEjT/go-libp2p-interface-pnet"
Steven Allen's avatar
Steven Allen committed
51
	p2phost "gx/ipfs/QmRNyPNJGNCaZyYonJj7owciWTsMd9gRfEKmZY3o6xwN3h/go-libp2p-host"
52
	mssmux "gx/ipfs/QmRVYfZ7tWNHPBzWiG6KWGzvT2hcGems8srihsQE29x1U5/go-smux-multistream"
53
	goprocess "gx/ipfs/QmSF8fPo3jgVBAy8fpdjjYqgG87dkJgUprRBHRd2tmfgpP/goprocess"
Jeromy's avatar
Jeromy committed
54
	mamask "gx/ipfs/QmSMZwvs3n4GBikZ7hKzT17c3bk65FmyZo2JqtJ16swqCv/multiaddr-filter"
55
	u "gx/ipfs/QmSU6eubNdhXjFBJBSksTp8kv8YRub8mGAPv8tVJHmL2EU/go-ipfs-util"
Jeromy's avatar
Jeromy committed
56
	logging "gx/ipfs/QmSpJByNKFX1sCsHBEp3R73FL4NF6FnQTEGyNAXHm2GS52/go-log"
Jeromy's avatar
Jeromy committed
57
	b58 "gx/ipfs/QmT8rehPR3F6bmwL6zjUN8XpiDBFFpMP2myPdC6ApsWfJf/go-base58"
58
	cid "gx/ipfs/QmTprEaAA2A9bst5XH7exuyi5KzNMK3SEDNN8rBDnKWcUS/go-cid"
vyzo's avatar
vyzo committed
59
	circuit "gx/ipfs/QmVEPsD9h95ToAC7NJpYopFcXyjVJm35xV4tw43J5JdCnL/go-libp2p-circuit"
60
	ds "gx/ipfs/QmVSase1JP7cq9QkPT46oNwdp9pT6kBkG3oqS14y3QcZjG/go-datastore"
61 62 63
	metrics "gx/ipfs/QmVjRAPfRtResCMCE4eBqr4Beoa6A89P1YweG9wUS6RqUL/go-libp2p-metrics"
	ma "gx/ipfs/QmXY77cVe7rVRQXZZQRioukUM7aRW3BTcAgJe12MCtb3Ji/go-multiaddr"
	peer "gx/ipfs/QmXYjuNuxVzXKJCfWasQk1RqkhVLDM9jtUKhqc2WPQmFSB/go-libp2p-peer"
Steven Allen's avatar
Steven Allen committed
64
	mplex "gx/ipfs/QmYBCDHS74pfbhMhtLeDpZyBa4Z9X76JLyACSKum1K39eP/go-smux-multiplex"
Steven Allen's avatar
Steven Allen committed
65 66 67 68 69
	discovery "gx/ipfs/QmZPBrKq6S1fdYaRAzYZivJL12QkUqHwnNzF9wC8VXC4bo/go-libp2p/p2p/discovery"
	p2pbhost "gx/ipfs/QmZPBrKq6S1fdYaRAzYZivJL12QkUqHwnNzF9wC8VXC4bo/go-libp2p/p2p/host/basic"
	rhost "gx/ipfs/QmZPBrKq6S1fdYaRAzYZivJL12QkUqHwnNzF9wC8VXC4bo/go-libp2p/p2p/host/routed"
	identify "gx/ipfs/QmZPBrKq6S1fdYaRAzYZivJL12QkUqHwnNzF9wC8VXC4bo/go-libp2p/p2p/protocol/identify"
	ping "gx/ipfs/QmZPBrKq6S1fdYaRAzYZivJL12QkUqHwnNzF9wC8VXC4bo/go-libp2p/p2p/protocol/ping"
70
	ic "gx/ipfs/QmaPbCnUMBohSGo3KnxEa2bHqyJVVeEEcwtqJAYxerieBo/go-libp2p-crypto"
Steven Allen's avatar
Steven Allen committed
71
	swarm "gx/ipfs/Qmb9GA1fMGfXdLBwJA9h3SRpr7C4AXUhpWgZzVmqcDJi1P/go-libp2p-swarm"
72
	yamux "gx/ipfs/Qmbn7RYyWzBVXiUp9jZ1dA4VADHy9DtS7iZLwfhEUQvm3U/go-smux-yamux"
Steven Allen's avatar
Steven Allen committed
73
	dht "gx/ipfs/QmcKxeQomXUjo54VwisTiXeic5FFBknwUPtT7yRWvmPD1D/go-libp2p-kad-dht"
Steven Allen's avatar
Steven Allen committed
74
	addrutil "gx/ipfs/Qmcx4KoZ91XS6izLjdkujAMAunAS1YuTgfgASgYaZF5GkR/go-addr-util"
75
	smux "gx/ipfs/QmeZBgYBHvxMukGK5ojg28BCNLB9SeXqT7XXg6o7r2GbJy/go-stream-muxer"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
76 77
)

Jeromy's avatar
Jeromy committed
78
const IpnsValidatorTag = "ipns"
79

80
const kReprovideFrequency = time.Hour * 12
81
const discoveryConnTimeout = time.Second * 30
Jeromy's avatar
Jeromy committed
82

Jeromy's avatar
Jeromy committed
83
var log = logging.Logger("core")
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
84

85 86 87 88
type mode int

const (
	// zero value is not a valid mode, must be explicitly set
89
	localMode mode = iota
90 91 92 93
	offlineMode
	onlineMode
)

94 95 96 97
func init() {
	identify.ClientVersion = "go-ipfs/" + config.CurrentVersionNumber + "/" + config.CurrentCommit
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
98
// IpfsNode is IPFS Core module. It represents an IPFS instance.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
99 100
type IpfsNode struct {

101
	// Self
102
	Identity peer.ID // the local node's identity
103

104
	Repo repo.Repo
105 106

	// Local node
Jakub Sztandera's avatar
Jakub Sztandera committed
107 108 109 110
	Pinning        pin.Pinner // the pinning manager
	Mounts         Mounts     // current mount state, if any.
	PrivateKey     ic.PrivKey // the local node's private Key
	PNetFingerpint []byte     // fingerprint of private network
111 112

	// Services
Jeromy's avatar
Jeromy committed
113
	Peerstore  pstore.Peerstore     // storage for other Peer instances
114
	Blockstore bstore.GCBlockstore  // the block store (lower level)
115 116 117
	Filestore  *filestore.Filestore // the filestore blockstore
	BaseBlocks bstore.Blockstore    // the raw blockstore, no filestore wrapping
	GCLocker   bstore.GCLocker      // the locker used to protect the blockstore during gc
118
	Blocks     bserv.BlockService   // the block service, get/add blocks.
119 120
	DAG        merkledag.DAGService // the merkle dag service, get/add objects.
	Resolver   *path.Resolver       // the path resolution system
Jeromy's avatar
Jeromy committed
121
	Reporter   metrics.Reporter
122
	Discovery  discovery.Service
Jeromy's avatar
Jeromy committed
123
	FilesRoot  *mfs.Root
124 125

	// Online
126 127 128 129 130
	PeerHost     p2phost.Host        // the network host (server+client)
	Bootstrapper io.Closer           // the periodic bootstrapper
	Routing      routing.IpfsRouting // the routing system. recommend ipfs-dht
	Exchange     exchange.Interface  // the block exchange + strategy (bitswap)
	Namesys      namesys.NameSystem  // the name system, resolves paths to hashes
Jeromy's avatar
Jeromy committed
131 132
	Ping         *ping.PingService
	Reprovider   *rp.Reprovider // the value reprovider system
Jeromy's avatar
Jeromy committed
133
	IpnsRepub    *ipnsrp.Republisher
134

Jeromy's avatar
Jeromy committed
135
	Floodsub *floodsub.PubSub
Łukasz Magiera's avatar
Łukasz Magiera committed
136
	P2P      *p2p.P2P
Jeromy's avatar
Jeromy committed
137

138 139
	proc goprocess.Process
	ctx  context.Context
140

Jeromy's avatar
Jeromy committed
141
	mode         mode
142
	localModeSet bool
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
143 144
}

145 146 147 148 149 150 151 152
// Mounts defines what the node's mount state is. This should
// perhaps be moved to the daemon or mount. It's here because
// it needs to be accessible across daemon requests.
type Mounts struct {
	Ipfs mount.Mount
	Ipns mount.Mount
}

153
func (n *IpfsNode) startOnlineServices(ctx context.Context, routingOption RoutingOption, hostOption HostOption, do DiscoveryOption, pubsub, mplex bool) error {
154 155

	if n.PeerHost != nil { // already online.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
156
		return errors.New("node already online")
157 158 159
	}

	// load private key
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
160
	if err := n.LoadPrivateKey(); err != nil {
161 162 163
		return err
	}

Jeromy's avatar
Jeromy committed
164
	// get undialable addrs from config
165 166 167 168
	cfg, err := n.Repo.Config()
	if err != nil {
		return err
	}
Jeromy's avatar
Jeromy committed
169
	var addrfilter []*net.IPNet
170
	for _, s := range cfg.Swarm.AddrFilters {
Jeromy's avatar
Jeromy committed
171 172
		f, err := mamask.NewMask(s)
		if err != nil {
173
			return fmt.Errorf("incorrectly formatted address filter in config: %s", s)
Jeromy's avatar
Jeromy committed
174 175 176 177
		}
		addrfilter = append(addrfilter, f)
	}

178 179 180 181 182
	if !cfg.Swarm.DisableBandwidthMetrics {
		// Set reporter
		n.Reporter = metrics.NewBandwidthCounter()
	}

183 184
	tpt := makeSmuxTransport(mplex)

Jakub Sztandera's avatar
Jakub Sztandera committed
185 186 187 188 189 190 191 192 193 194 195 196
	swarmkey, err := n.Repo.SwarmKey()
	if err != nil {
		return err
	}

	var protec ipnet.Protector
	if swarmkey != nil {
		protec, err = pnet.NewProtector(bytes.NewReader(swarmkey))
		if err != nil {
			return err
		}
		n.PNetFingerpint = protec.Fingerprint()
197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214
		go func() {
			t := time.NewTicker(30 * time.Second)
			<-t.C // swallow one tick
			for {
				select {
				case <-t.C:
					if ph := n.PeerHost; ph != nil {
						if len(ph.Network().Peers()) == 0 {
							log.Warning("We are in private network and have no peers.")
							log.Warning("This might be configuration mistake.")
						}
					}
				case <-n.Process().Closing():
					t.Stop()
					return
				}
			}
		}()
Jakub Sztandera's avatar
Jakub Sztandera committed
215 216
	}

217 218 219 220 221 222 223 224
	addrsFactory, err := makeAddrsFactory(cfg.Addresses)
	if err != nil {
		return err
	}

	hostopts := &ConstructPeerHostOpts{
		AddrsFactory:      addrsFactory,
		DisableNatPortMap: cfg.Swarm.DisableNatPortMap,
vyzo's avatar
vyzo committed
225 226
		DisableRelay:      cfg.Swarm.DisableRelay,
		EnableRelayHop:    cfg.Swarm.EnableRelayHop,
227
	}
Jakub Sztandera's avatar
Jakub Sztandera committed
228
	peerhost, err := hostOption(ctx, n.Identity, n.Peerstore, n.Reporter,
229
		addrfilter, tpt, protec, hostopts)
vyzo's avatar
vyzo committed
230

231
	if err != nil {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
232
		return err
233 234
	}

235
	if err := n.startOnlineServicesWithHost(ctx, peerhost, routingOption); err != nil {
236
		return err
237 238 239
	}

	// Ok, now we're ready to listen.
240
	if err := startListening(ctx, n.PeerHost, cfg); err != nil {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
241
		return err
242
	}
243

Jeromy's avatar
Jeromy committed
244 245 246
	if pubsub {
		n.Floodsub = floodsub.NewFloodSub(ctx, peerhost)
	}
Jeromy's avatar
Jeromy committed
247

Łukasz Magiera's avatar
Łukasz Magiera committed
248
	n.P2P = p2p.NewP2P(n.Identity, n.PeerHost, n.Peerstore)
249

250
	// setup local discovery
Jeromy's avatar
Jeromy committed
251
	if do != nil {
Jeromy's avatar
Jeromy committed
252
		service, err := do(ctx, n.PeerHost)
Jeromy's avatar
Jeromy committed
253
		if err != nil {
Jeromy's avatar
Jeromy committed
254 255 256 257
			log.Error("mdns error: ", err)
		} else {
			service.RegisterNotifee(n)
			n.Discovery = service
Jeromy's avatar
Jeromy committed
258
		}
259 260
	}

261
	return n.Bootstrap(DefaultBootstrapConfig)
262 263
}

Łukasz Magiera's avatar
Łukasz Magiera committed
264 265 266 267 268 269
func (n *IpfsNode) startLateOnlineServices(ctx context.Context) error {
	cfg, err := n.Repo.Config()
	if err != nil {
		return err
	}

270
	var keyProvider rp.KeyChanFunc
Łukasz Magiera's avatar
Łukasz Magiera committed
271 272 273 274 275 276 277 278 279 280 281

	switch cfg.Reprovider.Strategy {
	case "all":
		fallthrough
	case "":
		keyProvider = rp.NewBlockstoreProvider(n.Blockstore)
	case "roots":
		keyProvider = rp.NewPinnedProvider(n.Pinning, n.DAG, true)
	case "pinned":
		keyProvider = rp.NewPinnedProvider(n.Pinning, n.DAG, false)
	default:
282
		return fmt.Errorf("unknown reprovider strategy '%s'", cfg.Reprovider.Strategy)
Łukasz Magiera's avatar
Łukasz Magiera committed
283
	}
284
	n.Reprovider = rp.NewReprovider(ctx, n.Routing, keyProvider)
Łukasz Magiera's avatar
Łukasz Magiera committed
285 286 287 288 289 290 291 292 293 294 295 296

	if cfg.Reprovider.Interval != "0" {
		interval := kReprovideFrequency
		if cfg.Reprovider.Interval != "" {
			dur, err := time.ParseDuration(cfg.Reprovider.Interval)
			if err != nil {
				return err
			}

			interval = dur
		}

297
		go n.Reprovider.ProvideEvery(interval)
Łukasz Magiera's avatar
Łukasz Magiera committed
298 299 300 301 302
	}

	return nil
}

303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348
func makeAddrsFactory(cfg config.Addresses) (p2pbhost.AddrsFactory, error) {
	var annAddrs []ma.Multiaddr
	for _, addr := range cfg.Announce {
		maddr, err := ma.NewMultiaddr(addr)
		if err != nil {
			return nil, err
		}
		annAddrs = append(annAddrs, maddr)
	}

	filters := mafilter.NewFilters()
	noAnnAddrs := map[string]bool{}
	for _, addr := range cfg.NoAnnounce {
		f, err := mamask.NewMask(addr)
		if err == nil {
			filters.AddDialFilter(f)
			continue
		}
		maddr, err := ma.NewMultiaddr(addr)
		if err != nil {
			return nil, err
		}
		noAnnAddrs[maddr.String()] = true
	}

	return func(allAddrs []ma.Multiaddr) []ma.Multiaddr {
		var addrs []ma.Multiaddr
		if len(annAddrs) > 0 {
			addrs = annAddrs
		} else {
			addrs = allAddrs
		}

		var out []ma.Multiaddr
		for _, maddr := range addrs {
			// check for exact matches
			ok, _ := noAnnAddrs[maddr.String()]
			// check for /ipcidr matches
			if !ok && !filters.AddrBlocked(maddr) {
				out = append(out, maddr)
			}
		}
		return out
	}, nil
}

349 350 351 352 353 354 355 356 357 358 359 360
func makeSmuxTransport(mplexExp bool) smux.Transport {
	mstpt := mssmux.NewBlankTransport()

	ymxtpt := &yamux.Transport{
		AcceptBacklog:          8192,
		ConnectionWriteTimeout: time.Second * 10,
		KeepAliveInterval:      time.Second * 30,
		EnableKeepAlive:        true,
		MaxStreamWindowSize:    uint32(1024 * 512),
		LogOutput:              ioutil.Discard,
	}

361 362 363 364
	if os.Getenv("YAMUX_DEBUG") != "" {
		ymxtpt.LogOutput = os.Stderr
	}

365 366 367 368 369 370 371 372 373 374 375 376 377 378
	mstpt.AddTransport("/yamux/1.0.0", ymxtpt)

	if mplexExp {
		mstpt.AddTransport("/mplex/6.7.0", mplex.DefaultTransport)
	}

	// Allow muxer preference order overriding
	if prefs := os.Getenv("LIBP2P_MUX_PREFS"); prefs != "" {
		mstpt.OrderPreference = strings.Fields(prefs)
	}

	return mstpt
}

Jeromy's avatar
Jeromy committed
379 380
func setupDiscoveryOption(d config.Discovery) DiscoveryOption {
	if d.MDNS.Enabled {
Jeromy's avatar
Jeromy committed
381
		return func(ctx context.Context, h p2phost.Host) (discovery.Service, error) {
Jeromy's avatar
Jeromy committed
382 383 384
			if d.MDNS.Interval == 0 {
				d.MDNS.Interval = 5
			}
Jeromy's avatar
Jeromy committed
385
			return discovery.NewMdnsService(ctx, h, time.Duration(d.MDNS.Interval)*time.Second)
Jeromy's avatar
Jeromy committed
386 387 388 389 390
		}
	}
	return nil
}

Jeromy's avatar
Jeromy committed
391
func (n *IpfsNode) HandlePeerFound(p pstore.PeerInfo) {
392
	log.Warning("trying peer info: ", p)
393
	ctx, cancel := context.WithTimeout(n.Context(), discoveryConnTimeout)
rht's avatar
rht committed
394
	defer cancel()
395
	if err := n.PeerHost.Connect(ctx, p); err != nil {
396 397 398 399
		log.Warning("Failed to connect to peer found by discovery: ", err)
	}
}

400 401
// startOnlineServicesWithHost  is the set of services which need to be
// initialized with the host and _before_ we start listening.
402
func (n *IpfsNode) startOnlineServicesWithHost(ctx context.Context, host p2phost.Host, routingOption RoutingOption) error {
403
	// setup diagnostics service
Jeromy's avatar
Jeromy committed
404
	n.Ping = ping.NewPingService(host)
405 406

	// setup routing service
407
	r, err := routingOption(ctx, host, n.Repo.Datastore())
Jeromy's avatar
Jeromy committed
408
	if err != nil {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
409
		return err
410
	}
Jeromy's avatar
Jeromy committed
411
	n.Routing = r
412

413 414 415
	// Wrap standard peer host with routing system to allow unknown peer lookups
	n.PeerHost = rhost.Wrap(host, n.Routing)

416 417
	// setup exchange service
	const alwaysSendToPeer = true // use YesManStrategy
418
	bitswapNetwork := bsnet.NewFromIpfsHost(n.PeerHost, n.Routing)
419 420
	n.Exchange = bitswap.New(ctx, n.Identity, bitswapNetwork, n.Blockstore, alwaysSendToPeer)

421 422 423 424 425
	size, err := n.getCacheSize()
	if err != nil {
		return err
	}

426
	// setup name system
427
	n.Namesys = namesys.NewNameSystem(n.Routing, n.Repo.Datastore(), size)
428

Jeromy's avatar
Jeromy committed
429
	// setup ipns republishing
430
	return n.setupIpnsRepublisher()
431 432
}

433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449
// getCacheSize returns cache life and cache size
func (n *IpfsNode) getCacheSize() (int, error) {
	cfg, err := n.Repo.Config()
	if err != nil {
		return 0, err
	}

	cs := cfg.Ipns.ResolveCacheSize
	if cs == 0 {
		cs = 128
	}
	if cs < 0 {
		return 0, fmt.Errorf("cannot specify negative resolve cache size")
	}
	return cs, nil
}

450
func (n *IpfsNode) setupIpnsRepublisher() error {
Jeromy's avatar
Jeromy committed
451 452 453 454
	cfg, err := n.Repo.Config()
	if err != nil {
		return err
	}
455

456
	n.IpnsRepub = ipnsrp.NewRepublisher(n.Routing, n.Repo.Datastore(), n.PrivateKey, n.Repo.Keystore())
457

Jeromy's avatar
Jeromy committed
458 459 460 461 462 463
	if cfg.Ipns.RepublishPeriod != "" {
		d, err := time.ParseDuration(cfg.Ipns.RepublishPeriod)
		if err != nil {
			return fmt.Errorf("failure to parse config setting IPNS.RepublishPeriod: %s", err)
		}

464
		if !u.Debug && (d < time.Minute || d > (time.Hour*24)) {
Jeromy's avatar
Jeromy committed
465 466 467 468 469 470
			return fmt.Errorf("config setting IPNS.RepublishPeriod is not between 1min and 1day: %s", d)
		}

		n.IpnsRepub.Interval = d
	}

471 472 473 474 475 476 477 478 479
	if cfg.Ipns.RecordLifetime != "" {
		d, err := time.ParseDuration(cfg.Ipns.RepublishPeriod)
		if err != nil {
			return fmt.Errorf("failure to parse config setting IPNS.RecordLifetime: %s", err)
		}

		n.IpnsRepub.RecordLifetime = d
	}

Jeromy's avatar
Jeromy committed
480 481
	n.Process().Go(n.IpnsRepub.Run)

482 483 484
	return nil
}

485 486 487 488 489 490 491 492 493 494 495 496
// Process returns the Process object
func (n *IpfsNode) Process() goprocess.Process {
	return n.proc
}

// Close calls Close() on the Process object
func (n *IpfsNode) Close() error {
	return n.proc.Close()
}

// Context returns the IpfsNode context
func (n *IpfsNode) Context() context.Context {
497 498 499
	if n.ctx == nil {
		n.ctx = context.TODO()
	}
500 501 502
	return n.ctx
}

503 504
// teardown closes owned children. If any errors occur, this function returns
// the first error.
Brian Tiger Chow's avatar
Brian Tiger Chow committed
505
func (n *IpfsNode) teardown() error {
506
	log.Debug("core is shutting down...")
507 508
	// owned objects are closed in this teardown to ensure that they're closed
	// regardless of which constructor was used to add them to the node.
Jeromy's avatar
Jeromy committed
509 510
	var closers []io.Closer

511
	// NOTE: The order that objects are added(closed) matters, if an object
Jeromy's avatar
Jeromy committed
512 513 514 515 516
	// needs to use another during its shutdown/cleanup process, it should be
	// closed before that other object

	if n.FilesRoot != nil {
		closers = append(closers, n.FilesRoot)
Jeromy's avatar
Jeromy committed
517
	}
518

519 520 521 522
	if n.Exchange != nil {
		closers = append(closers, n.Exchange)
	}

523
	if n.Mounts.Ipfs != nil && !n.Mounts.Ipfs.IsActive() {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
524 525
		closers = append(closers, mount.Closer(n.Mounts.Ipfs))
	}
526
	if n.Mounts.Ipns != nil && !n.Mounts.Ipns.IsActive() {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
527 528 529
		closers = append(closers, mount.Closer(n.Mounts.Ipns))
	}

Jeromy's avatar
Jeromy committed
530 531 532 533
	if dht, ok := n.Routing.(*dht.IpfsDHT); ok {
		closers = append(closers, dht.Process())
	}

Jeromy's avatar
Jeromy committed
534 535 536 537
	if n.Blocks != nil {
		closers = append(closers, n.Blocks)
	}

Jeromy's avatar
Jeromy committed
538 539
	if n.Bootstrapper != nil {
		closers = append(closers, n.Bootstrapper)
540 541
	}

Jeromy's avatar
Jeromy committed
542 543
	if n.PeerHost != nil {
		closers = append(closers, n.PeerHost)
544
	}
545

Jeromy's avatar
Jeromy committed
546 547 548
	// Repo closed last, most things need to preserve state here
	closers = append(closers, n.Repo)

549
	var errs []error
550
	for _, closer := range closers {
551 552
		if err := closer.Close(); err != nil {
			errs = append(errs, err)
553 554 555 556
		}
	}
	if len(errs) > 0 {
		return errs[0]
Brian Tiger Chow's avatar
Brian Tiger Chow committed
557 558
	}
	return nil
Brian Tiger Chow's avatar
Brian Tiger Chow committed
559 560
}

Brian Tiger Chow's avatar
Brian Tiger Chow committed
561
func (n *IpfsNode) OnlineMode() bool {
562 563 564 565 566 567
	switch n.mode {
	case onlineMode:
		return true
	default:
		return false
	}
Brian Tiger Chow's avatar
Brian Tiger Chow committed
568 569
}

570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589
func (n *IpfsNode) SetLocal(isLocal bool) {
	if isLocal {
		n.mode = localMode
	}
	n.localModeSet = true
}

func (n *IpfsNode) LocalMode() bool {
	if !n.localModeSet {
		// programmer error should not happen
		panic("local mode not set")
	}
	switch n.mode {
	case localMode:
		return true
	default:
		return false
	}
}

590
func (n *IpfsNode) Bootstrap(cfg BootstrapConfig) error {
591 592

	// TODO what should return value be when in offlineMode?
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
593 594 595 596
	if n.Routing == nil {
		return nil
	}

597 598 599 600 601 602 603
	if n.Bootstrapper != nil {
		n.Bootstrapper.Close() // stop previous bootstrap process.
	}

	// if the caller did not specify a bootstrap peer function, get the
	// freshest bootstrap peers from config. this responds to live changes.
	if cfg.BootstrapPeers == nil {
Jeromy's avatar
Jeromy committed
604
		cfg.BootstrapPeers = func() []pstore.PeerInfo {
605
			ps, err := n.loadBootstrapPeers()
606
			if err != nil {
607
				log.Warning("failed to parse bootstrap peers from config")
608 609 610 611 612 613 614 615 616
				return nil
			}
			return ps
		}
	}

	var err error
	n.Bootstrapper, err = Bootstrap(n, cfg)
	return err
617 618
}

619 620
func (n *IpfsNode) loadID() error {
	if n.Identity != "" {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
621
		return errors.New("identity already loaded")
622 623
	}

624 625 626 627 628 629
	cfg, err := n.Repo.Config()
	if err != nil {
		return err
	}

	cid := cfg.Identity.PeerID
630
	if cid == "" {
631
		return errors.New("identity was not set in config (was 'ipfs init' run?)")
632 633
	}
	if len(cid) == 0 {
634
		return errors.New("no peer ID in config! (was 'ipfs init' run?)")
635 636
	}

637 638 639
	n.Identity = peer.ID(b58.Decode(cid))
	return nil
}
640

641 642 643 644 645 646 647 648
func (n *IpfsNode) GetKey(name string) (ic.PrivKey, error) {
	if name == "self" {
		return n.PrivateKey, nil
	} else {
		return n.Repo.Keystore().Get(name)
	}
}

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
649
func (n *IpfsNode) LoadPrivateKey() error {
650
	if n.Identity == "" || n.Peerstore == nil {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
651
		return errors.New("loaded private key out of order.")
652 653
	}

654
	if n.PrivateKey != nil {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
655
		return errors.New("private key already loaded")
656 657
	}

658 659 660 661 662 663
	cfg, err := n.Repo.Config()
	if err != nil {
		return err
	}

	sk, err := loadPrivateKey(&cfg.Identity, n.Identity)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
664
	if err != nil {
665
		return err
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
666
	}
667

668 669
	n.PrivateKey = sk
	n.Peerstore.AddPrivKey(n.Identity, n.PrivateKey)
Jeromy's avatar
Jeromy committed
670 671 672 673
	n.Peerstore.AddPubKey(n.Identity, sk.GetPublic())
	return nil
}

Jeromy's avatar
Jeromy committed
674
func (n *IpfsNode) loadBootstrapPeers() ([]pstore.PeerInfo, error) {
675 676 677 678 679 680
	cfg, err := n.Repo.Config()
	if err != nil {
		return nil, err
	}

	parsed, err := cfg.BootstrapPeers()
681 682 683 684 685 686
	if err != nil {
		return nil, err
	}
	return toPeerInfos(parsed), nil
}

Jeromy's avatar
Jeromy committed
687
func (n *IpfsNode) loadFilesRoot() error {
Jeromy's avatar
Jeromy committed
688
	dsk := ds.NewKey("/local/filesroot")
Jeromy's avatar
Jeromy committed
689 690
	pf := func(ctx context.Context, c *cid.Cid) error {
		return n.Repo.Datastore().Put(dsk, c.Bytes())
Jeromy's avatar
Jeromy committed
691 692
	}

693
	var nd *merkledag.ProtoNode
Jeromy's avatar
Jeromy committed
694 695 696 697
	val, err := n.Repo.Datastore().Get(dsk)

	switch {
	case err == ds.ErrNotFound || val == nil:
698
		nd = ft.EmptyDirNode()
Jeromy's avatar
Jeromy committed
699 700 701 702 703
		_, err := n.DAG.Add(nd)
		if err != nil {
			return fmt.Errorf("failure writing to dagstore: %s", err)
		}
	case err == nil:
Jeromy's avatar
Jeromy committed
704 705 706 707 708
		c, err := cid.Cast(val.([]byte))
		if err != nil {
			return err
		}

709
		rnd, err := n.DAG.Get(n.Context(), c)
Jeromy's avatar
Jeromy committed
710 711 712
		if err != nil {
			return fmt.Errorf("error loading filesroot from DAG: %s", err)
		}
713 714 715 716 717 718 719

		pbnd, ok := rnd.(*merkledag.ProtoNode)
		if !ok {
			return merkledag.ErrNotProtobuf
		}

		nd = pbnd
Jeromy's avatar
Jeromy committed
720 721 722 723 724 725 726 727 728 729 730 731 732
	default:
		return err
	}

	mr, err := mfs.NewRoot(n.Context(), n.DAG, nd, pf)
	if err != nil {
		return err
	}

	n.FilesRoot = mr
	return nil
}

Jeromy's avatar
Jeromy committed
733 734 735
// SetupOfflineRouting loads the local nodes private key and
// uses it to instantiate a routing system in offline mode.
// This is primarily used for offline ipns modifications.
Jeromy's avatar
Jeromy committed
736
func (n *IpfsNode) SetupOfflineRouting() error {
737 738 739 740
	if n.Routing != nil {
		// Routing was already set up
		return nil
	}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
741
	err := n.LoadPrivateKey()
Jeromy's avatar
Jeromy committed
742 743 744 745 746
	if err != nil {
		return err
	}

	n.Routing = offroute.NewOfflineRouter(n.Repo.Datastore(), n.PrivateKey)
747

748 749 750 751 752 753
	size, err := n.getCacheSize()
	if err != nil {
		return err
	}

	n.Namesys = namesys.NewNameSystem(n.Routing, n.Repo.Datastore(), size)
754

755
	return nil
756 757 758 759
}

func loadPrivateKey(cfg *config.Identity, id peer.ID) (ic.PrivKey, error) {
	sk, err := cfg.DecodePrivateKey("passphrase todo!")
760 761 762
	if err != nil {
		return nil, err
	}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
763

764 765 766 767
	id2, err := peer.IDFromPrivateKey(sk)
	if err != nil {
		return nil, err
	}
768

769 770
	if id2 != id {
		return nil, fmt.Errorf("private key in config does not match id: %s != %s", id, id2)
771 772
	}

773
	return sk, nil
774
}
775

776
func listenAddresses(cfg *config.Config) ([]ma.Multiaddr, error) {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
777 778 779
	var listen []ma.Multiaddr
	for _, addr := range cfg.Addresses.Swarm {
		maddr, err := ma.NewMultiaddr(addr)
780
		if err != nil {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
781
			return nil, fmt.Errorf("Failure to parse config.Addresses.Swarm: %s", cfg.Addresses.Swarm)
782
		}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
783
		listen = append(listen, maddr)
784 785 786 787
	}

	return listen, nil
}
788

Kevin Atkinson's avatar
Kevin Atkinson committed
789
type ConstructPeerHostOpts struct {
790
	AddrsFactory      p2pbhost.AddrsFactory
vyzo's avatar
vyzo committed
791 792 793
	DisableNatPortMap bool
	DisableRelay      bool
	EnableRelayHop    bool
Kevin Atkinson's avatar
Kevin Atkinson committed
794 795 796
}

type HostOption func(ctx context.Context, id peer.ID, ps pstore.Peerstore, bwr metrics.Reporter, fs []*net.IPNet, tpt smux.Transport, protc ipnet.Protector, opts *ConstructPeerHostOpts) (p2phost.Host, error)
Jeromy's avatar
Jeromy committed
797 798 799

var DefaultHostOption HostOption = constructPeerHost

800
// isolates the complex initialization steps
Kevin Atkinson's avatar
Kevin Atkinson committed
801
func constructPeerHost(ctx context.Context, id peer.ID, ps pstore.Peerstore, bwr metrics.Reporter, fs []*net.IPNet, tpt smux.Transport, protec ipnet.Protector, opts *ConstructPeerHostOpts) (p2phost.Host, error) {
802 803

	// no addresses to begin with. we'll start later.
Jakub Sztandera's avatar
Jakub Sztandera committed
804
	swrm, err := swarm.NewSwarmWithProtector(ctx, nil, id, ps, protec, tpt, bwr)
805
	if err != nil {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
806
		return nil, err
807
	}
808

809 810
	network := (*swarm.Network)(swrm)

811 812 813 814
	for _, f := range fs {
		network.Swarm().Filters.AddDialFilter(f)
	}

Kevin Atkinson's avatar
Kevin Atkinson committed
815 816 817 818
	hostOpts := []interface{}{bwr}
	if !opts.DisableNatPortMap {
		hostOpts = append(hostOpts, p2pbhost.NATPortMap)
	}
819 820 821
	if opts.AddrsFactory != nil {
		hostOpts = append(hostOpts, opts.AddrsFactory)
	}
Kevin Atkinson's avatar
Kevin Atkinson committed
822

vyzo's avatar
vyzo committed
823 824 825 826 827 828 829 830 831 832 833 834 835 836 837
	if !opts.DisableRelay {
		filterRelayAddr := func(addrs []ma.Multiaddr) []ma.Multiaddr {
			var raddrs []ma.Multiaddr
			for _, addr := range addrs {
				_, err := addr.ValueForProtocol(circuit.P_CIRCUIT)
				if err == nil {
					continue
				}
				raddrs = append(raddrs, addr)
			}
			return raddrs
		}
		hostOpts = append(hostOpts, p2pbhost.AddrsFactory(filterRelayAddr))
	}

Kevin Atkinson's avatar
Kevin Atkinson committed
838
	host := p2pbhost.New(network, hostOpts...)
Jeromy's avatar
Jeromy committed
839

vyzo's avatar
vyzo committed
840 841 842 843 844 845 846 847 848 849 850 851
	if !opts.DisableRelay {
		var relayOpts []circuit.RelayOpt
		if opts.EnableRelayHop {
			relayOpts = append(relayOpts, circuit.OptHop)
		}

		err := circuit.AddRelayTransport(ctx, host, relayOpts...)
		if err != nil {
			return nil, err
		}
	}

852 853 854 855 856 857 858
	return host, nil
}

// startListening on the network addresses
func startListening(ctx context.Context, host p2phost.Host, cfg *config.Config) error {
	listenAddrs, err := listenAddresses(cfg)
	if err != nil {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
859
		return err
860 861
	}

862
	// make sure we error out if our config does not have addresses we can use
863
	log.Debugf("Config.Addresses.Swarm:%s", listenAddrs)
864
	filteredAddrs := addrutil.FilterUsableAddrs(listenAddrs)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
865
	log.Debugf("Config.Addresses.Swarm:%s (filtered)", filteredAddrs)
866
	if len(filteredAddrs) < 1 {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
867
		return fmt.Errorf("addresses in config not usable: %s", listenAddrs)
868 869
	}

870 871 872
	// Actually start listening:
	if err := host.Network().Listen(filteredAddrs...); err != nil {
		return err
873 874
	}

875
	// list out our addresses
876
	addrs, err := host.Network().InterfaceListenAddresses()
877
	if err != nil {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
878
		return err
879
	}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
880
	log.Infof("Swarm listening at: %s", addrs)
881
	return nil
882
}
883

Jeromy's avatar
Jeromy committed
884
func constructDHTRouting(ctx context.Context, host p2phost.Host, dstore repo.Datastore) (routing.IpfsRouting, error) {
885
	dhtRouting := dht.NewDHT(ctx, host, dstore)
886
	dhtRouting.Validator[IpnsValidatorTag] = namesys.IpnsRecordValidator
887
	dhtRouting.Selector[IpnsValidatorTag] = namesys.IpnsSelectorFunc
888 889
	return dhtRouting, nil
}
Jeromy's avatar
Jeromy committed
890

Jeromy's avatar
Jeromy committed
891 892 893 894 895 896 897
func constructClientDHTRouting(ctx context.Context, host p2phost.Host, dstore repo.Datastore) (routing.IpfsRouting, error) {
	dhtRouting := dht.NewDHTClient(ctx, host, dstore)
	dhtRouting.Validator[IpnsValidatorTag] = namesys.IpnsRecordValidator
	dhtRouting.Selector[IpnsValidatorTag] = namesys.IpnsSelectorFunc
	return dhtRouting, nil
}

Jeromy's avatar
Jeromy committed
898
type RoutingOption func(context.Context, p2phost.Host, repo.Datastore) (routing.IpfsRouting, error)
Jeromy's avatar
Jeromy committed
899

Jeromy's avatar
Jeromy committed
900
type DiscoveryOption func(context.Context, p2phost.Host) (discovery.Service, error)
Jeromy's avatar
Jeromy committed
901

902
var DHTOption RoutingOption = constructDHTRouting
Jeromy's avatar
Jeromy committed
903
var DHTClientOption RoutingOption = constructClientDHTRouting
904
var NilRouterOption RoutingOption = nilrouting.ConstructNilRouting