mirror of
https://github.com/fluencelabs/js-libp2p
synced 2025-04-25 10:32:14 +00:00
148 lines
3.8 KiB
JavaScript
148 lines
3.8 KiB
JavaScript
'use strict'
|
|
|
|
const errcode = require('err-code')
|
|
const debug = require('debug')
|
|
const log = debug('libp2p:peer-store')
|
|
log.error = debug('libp2p:peer-store:error')
|
|
|
|
const { EventEmitter } = require('events')
|
|
const PeerId = require('peer-id')
|
|
|
|
const AddressBook = require('./address-book')
|
|
const ProtoBook = require('./proto-book')
|
|
|
|
const {
|
|
ERR_INVALID_PARAMETERS
|
|
} = require('../errors')
|
|
|
|
/**
|
|
* Responsible for managing known peers, as well as their addresses, protocols and metadata.
|
|
* @fires PeerStore#peer Emitted when a new peer is added.
|
|
* @fires PeerStore#change:protocols Emitted when a known peer supports a different set of protocols.
|
|
* @fires PeerStore#change:multiaddrs Emitted when a known peer has a different set of multiaddrs.
|
|
*/
|
|
class PeerStore extends EventEmitter {
|
|
/**
|
|
* Peer object
|
|
* @typedef {Object} Peer
|
|
* @property {PeerId} id peer's peer-id instance.
|
|
* @property {Array<Address>} addresses peer's addresses containing its multiaddrs and metadata.
|
|
* @property {Array<string>} protocols peer's supported protocols.
|
|
*/
|
|
|
|
/**
|
|
* @constructor
|
|
* @param {Object} properties
|
|
* @param {Datastore} [properties.datastore] Datastore to persist data.
|
|
* @param {boolean} [properties.persistance = true] Persist peerstore data.
|
|
*/
|
|
constructor ({ datastore, persistance = true } = {}) {
|
|
super()
|
|
|
|
/**
|
|
* Backend datastore used to persist data.
|
|
*/
|
|
this._datastore = datastore
|
|
|
|
/**
|
|
* AddressBook containing a map of peerIdStr to Address.
|
|
*/
|
|
this.addressBook = new AddressBook(this)
|
|
|
|
/**
|
|
* ProtoBook containing a map of peerIdStr to supported protocols.
|
|
*/
|
|
this.protoBook = new ProtoBook(this)
|
|
|
|
/**
|
|
* TODO: this should only exist until we have the key-book
|
|
* Map known peers to their peer-id.
|
|
* @type {Map<string, Array<PeerId>}
|
|
*/
|
|
this.peerIds = new Map()
|
|
|
|
this._enabledPersistance = persistance
|
|
}
|
|
|
|
/**
|
|
* Load data from the datastore to populate the PeerStore.
|
|
*/
|
|
async load () {
|
|
if (this._enabledPersistance) {
|
|
await this.addressBook._loadData()
|
|
await this.protoBook._loadData()
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get all the stored information of every peer.
|
|
* @returns {Map<string, Peer>}
|
|
*/
|
|
get peers () {
|
|
const peersData = new Map()
|
|
|
|
// AddressBook
|
|
for (const [idStr, addresses] of this.addressBook.data.entries()) {
|
|
const id = PeerId.createFromCID(idStr)
|
|
peersData.set(idStr, {
|
|
id,
|
|
addresses,
|
|
protocols: this.protoBook.get(id) || []
|
|
})
|
|
}
|
|
|
|
// ProtoBook
|
|
for (const [idStr, protocols] of this.protoBook.data.entries()) {
|
|
const pData = peersData.get(idStr)
|
|
|
|
if (!pData) {
|
|
peersData.set(idStr, {
|
|
id: PeerId.createFromCID(idStr),
|
|
addresses: [],
|
|
protocols: Array.from(protocols)
|
|
})
|
|
}
|
|
}
|
|
|
|
return peersData
|
|
}
|
|
|
|
/**
|
|
* Delete the information of the given peer in every book.
|
|
* @param {PeerId} peerId
|
|
* @returns {boolean} true if found and removed
|
|
*/
|
|
delete (peerId) {
|
|
const addressesDeleted = this.addressBook.delete(peerId)
|
|
const protocolsDeleted = this.protoBook.delete(peerId)
|
|
return addressesDeleted || protocolsDeleted
|
|
}
|
|
|
|
/**
|
|
* Get the stored information of a given peer.
|
|
* @param {PeerId} peerId
|
|
* @returns {Peer}
|
|
*/
|
|
get (peerId) {
|
|
if (!PeerId.isPeerId(peerId)) {
|
|
throw errcode(new Error('peerId must be an instance of peer-id'), ERR_INVALID_PARAMETERS)
|
|
}
|
|
|
|
const id = this.peerIds.get(peerId.toB58String())
|
|
const addresses = this.addressBook.get(peerId)
|
|
const protocols = this.protoBook.get(peerId)
|
|
|
|
if (!addresses && !protocols) {
|
|
return undefined
|
|
}
|
|
|
|
return {
|
|
id: id || peerId,
|
|
addresses: addresses || [],
|
|
protocols: protocols || []
|
|
}
|
|
}
|
|
}
|
|
|
|
module.exports = PeerStore
|