js-libp2p/src/index.js

217 lines
5.8 KiB
JavaScript
Raw Normal View History

2016-03-23 15:30:14 +01:00
'use strict'
2016-11-26 03:07:52 +01:00
const Swarm = require('libp2p-swarm')
const PeerId = require('peer-id')
const PeerInfo = require('peer-info')
const PeerBook = require('peer-book')
const multiaddr = require('multiaddr')
2017-03-27 12:26:34 +01:00
const EventEmitter = require('events').EventEmitter
2016-11-26 03:07:52 +01:00
const assert = require('assert')
2016-12-01 11:53:27 +00:00
const Ping = require('libp2p-ping')
2017-02-10 19:28:34 -08:00
const setImmediate = require('async/setImmediate')
2016-11-26 03:07:52 +01:00
exports = module.exports
2016-11-26 03:07:52 +01:00
const OFFLINE_ERROR_MESSAGE = 'The libp2p node is not started yet'
2017-03-27 12:26:34 +01:00
class Node extends EventEmitter {
2016-12-11 12:03:21 -08:00
constructor (_modules, _peerInfo, _peerBook, _options) {
2017-03-27 12:26:34 +01:00
super()
2016-11-26 03:07:52 +01:00
assert(_modules, 'requires modules to equip libp2p with features')
assert(_peerInfo, 'requires a PeerInfo instance')
this.modules = _modules
this.peerInfo = _peerInfo
this.peerBook = _peerBook || new PeerBook()
this.isOnline = false
this.swarm = new Swarm(this.peerInfo)
// Attach stream multiplexers
if (this.modules.connection.muxer) {
let muxers = this.modules.connection.muxer
muxers = Array.isArray(muxers) ? muxers : [muxers]
muxers.forEach((muxer) => {
this.swarm.connection.addStreamMuxer(muxer)
})
// If muxer exists, we can use Identify
this.swarm.connection.reuse()
// Received incommind dial and muxer upgrade happened, reuse this
// muxed connection
this.swarm.on('peer-mux-established', (peerInfo) => {
this.peerBook.put(peerInfo)
})
this.swarm.on('peer-mux-closed', (peerInfo) => {
this.peerBook.removeByB58String(peerInfo.id.toB58String())
})
}
// Attach crypto channels
if (this.modules.connection.crypto) {
let cryptos = this.modules.connection.crypto
cryptos = Array.isArray(cryptos) ? cryptos : [cryptos]
cryptos.forEach((crypto) => {
this.swarm.connection.crypto(crypto.tag, crypto.encrypt)
})
}
// Attach discovery mechanisms
2017-01-28 20:59:47 +00:00
if (this.modules.discovery) {
2016-11-26 03:07:52 +01:00
let discoveries = this.modules.discovery
discoveries = Array.isArray(discoveries) ? discoveries : [discoveries]
discoveries.forEach((discovery) => {
2017-03-27 12:26:34 +01:00
discovery.on('peer', (peerInfo) => this.emit('peer', peerInfo))
2016-11-26 03:07:52 +01:00
})
}
2016-12-01 11:53:27 +00:00
// Mount default protocols
Ping.mount(this.swarm)
2016-11-26 03:07:52 +01:00
// Not fully implemented in js-libp2p yet
this.routing = undefined
this.records = undefined
}
/*
* Start the libp2p node
* - create listeners on the multiaddrs the Peer wants to listen
*/
start (callback) {
if (!this.modules.transport) {
return callback(new Error('no transports were present'))
}
let ws
2016-11-26 03:07:52 +01:00
let transports = this.modules.transport
2016-11-26 03:07:52 +01:00
transports = Array.isArray(transports) ? transports : [transports]
const multiaddrs = this.peerInfo.multiaddrs
2016-11-26 03:07:52 +01:00
transports.forEach((transport) => {
if (transport.filter(multiaddrs).length > 0) {
this.swarm.transport.add(
transport.tag || transport.constructor.name, transport)
} else if (transport.constructor &&
transport.constructor.name === 'WebSockets') {
// TODO find a cleaner way to signal that a transport is always
// used for dialing, even if no listener
ws = transport
2016-11-26 03:07:52 +01:00
}
})
this.swarm.listen((err) => {
if (err) {
return callback(err)
}
if (ws) {
this.swarm.transport.add(ws.tag || ws.constructor.name, ws)
}
2016-11-26 03:07:52 +01:00
this.isOnline = true
2017-01-28 20:59:47 +00:00
if (this.modules.discovery) {
this.modules.discovery.forEach((discovery) => {
setImmediate(() => discovery.start(() => {}))
})
}
2016-11-26 03:07:52 +01:00
callback()
})
}
2016-11-26 03:07:52 +01:00
/*
* Stop the libp2p node by closing its listeners and open connections
*/
stop (callback) {
this.isOnline = false
2017-01-28 20:59:47 +00:00
if (this.modules.discovery) {
this.modules.discovery.forEach((discovery) => {
setImmediate(() => discovery.stop(() => {}))
})
}
2016-11-26 03:07:52 +01:00
this.swarm.close(callback)
}
2017-03-27 12:26:34 +01:00
isOn () {
return this.isOnline
2016-12-01 11:53:27 +00:00
}
2017-03-27 12:26:34 +01:00
ping (peer, callback) {
assert(this.isOn, OFFLINE_ERROR_MESSAGE)
const peerInfo = this._getPeerInfo(peer)
2016-12-01 11:53:27 +00:00
callback(null, new Ping(this.swarm, peerInfo))
}
2017-03-27 12:26:34 +01:00
dial (peer, protocol, callback) {
2016-11-26 03:07:52 +01:00
assert(this.isOnline, OFFLINE_ERROR_MESSAGE)
2017-03-27 12:26:34 +01:00
const peerInfo = this._getPeerInfo(peer)
2016-11-26 03:07:52 +01:00
if (typeof protocol === 'function') {
callback = protocol
protocol = undefined
}
2017-03-27 12:26:34 +01:00
this.swarm.dial(peerInfo, protocol, (err, conn) => {
2016-11-26 03:07:52 +01:00
if (err) {
return callback(err)
}
this.peerBook.put(peer)
callback(null, conn)
})
}
2017-03-27 12:26:34 +01:00
hangUp (peer, callback) {
2016-11-26 03:07:52 +01:00
assert(this.isOnline, OFFLINE_ERROR_MESSAGE)
2017-03-27 12:26:34 +01:00
const peerInfo = this._getPeerInfo(peer)
2016-11-26 03:07:52 +01:00
2017-03-27 12:26:34 +01:00
this.peerBook.removeByB58String(peerInfo.id.toB58String())
2016-11-26 03:07:52 +01:00
this.swarm.hangUp(peer, callback)
}
handle (protocol, handlerFunc, matchFunc) {
this.swarm.handle(protocol, handlerFunc, matchFunc)
}
unhandle (protocol) {
this.swarm.unhandle(protocol)
}
2017-03-27 12:26:34 +01:00
/*
* Helper method to check the data type of peer and convert it to PeerInfo
*/
_getPeerInfo (peer) {
let p
switch (peer) {
case PeerInfo.isPeerInfo(peer): p = peer; break
case multiaddr.isMultiaddr(peer): {
const peerIdB58Str = multiaddr.getPeerId(peer)
try {
p = this.peerBook.getByB58String(peerIdB58Str)
} catch (err) {
p = new PeerInfo(PeerId.createFromB58String(peerIdB58Str))
}
p.multiaddr.add(peer)
} break
case PeerId.isPeerId(peer): {
const peerIdB58Str = peer.toB58String()
try {
p = this.peerBook.getByB58String(peerIdB58Str)
} catch (err) {
// TODO this is where PeerRouting comes into place
throw new Error('No knowledge about: ' + peerIdB58Str)
}
} break
default: throw new Error('Peer type not recognized')
}
return p
}
2016-11-26 03:07:52 +01:00
}
module.exports = Node