'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 { /** * PeerData object * @typedef {Object} PeerData * @property {PeerId} id peer's peer-id instance. * @property {Array} multiaddrsInfos peer's information of the multiaddrs. * @property {Array} protocols peer's supported protocols. */ constructor () { super() /** * AddressBook containing a map of peerIdStr to multiaddrsInfo */ 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} */ this.peerIds = new Map() } /** * Get all the stored information of every peer. * @returns {Map} */ get peers () { const peersData = new Map() // AddressBook for (const [idStr, multiaddrInfos] of this.addressBook.data.entries()) { const id = PeerId.createFromCID(idStr) peersData.set(idStr, { id, multiaddrInfos, 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), multiaddrInfos: [], 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 {PeerData} */ 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 multiaddrInfos = this.addressBook.get(peerId) const protocols = this.protoBook.get(peerId) if (!multiaddrInfos && !protocols) { return undefined } return { id: id || peerId, multiaddrInfos: multiaddrInfos || [], protocols: protocols || [] } } } module.exports = PeerStore