mirror of
https://github.com/fluencelabs/js-libp2p
synced 2025-04-01 23:31:04 +00:00
* fix: yield final peers from dht getClosestPeers `PEER_RESPONSE` is an intermediate event, we should only yield from `FINAL_PEER` events as we'll only get `K` of those. * chore: fix test
698 lines
20 KiB
TypeScript
698 lines
20 KiB
TypeScript
/* eslint-env mocha */
|
|
|
|
import { expect } from 'aegir/chai'
|
|
import sinon from 'sinon'
|
|
import delay from 'delay'
|
|
import pDefer from 'p-defer'
|
|
import pWaitFor from 'p-wait-for'
|
|
import drain from 'it-drain'
|
|
import all from 'it-all'
|
|
import { multiaddr } from '@multiformats/multiaddr'
|
|
import { createNode, createPeerId, populateAddressBooks } from '../utils/creators/peer.js'
|
|
import type { Libp2pNode } from '../../src/libp2p.js'
|
|
import { createBaseOptions } from '../utils/base-options.js'
|
|
import { createRoutingOptions } from './utils.js'
|
|
import type { PeerId } from '@libp2p/interface-peer-id'
|
|
import { createEd25519PeerId } from '@libp2p/peer-id-factory'
|
|
import { EventTypes, MessageType } from '@libp2p/interface-dht'
|
|
import type { PeerInfo } from '@libp2p/interface-peer-info'
|
|
import { KadDHT } from '@libp2p/kad-dht'
|
|
import type { PeerRouting } from '@libp2p/interface-peer-routing'
|
|
import { StubbedInstance, stubInterface } from 'ts-sinon'
|
|
|
|
describe('peer-routing', () => {
|
|
let peerId: PeerId
|
|
|
|
beforeEach(async () => {
|
|
peerId = await createEd25519PeerId()
|
|
})
|
|
|
|
describe('no routers', () => {
|
|
let node: Libp2pNode
|
|
|
|
before(async () => {
|
|
node = await createNode({
|
|
config: createBaseOptions()
|
|
})
|
|
})
|
|
|
|
after(async () => await node.stop())
|
|
|
|
it('.findPeer should return an error', async () => {
|
|
await expect(node.peerRouting.findPeer(peerId))
|
|
.to.eventually.be.rejected()
|
|
.and.to.have.property('code', 'ERR_NO_ROUTERS_AVAILABLE')
|
|
})
|
|
|
|
it('.getClosestPeers should return an error', async () => {
|
|
try {
|
|
for await (const _ of node.peerRouting.getClosestPeers(peerId.toBytes())) { } // eslint-disable-line
|
|
throw new Error('.getClosestPeers should return an error')
|
|
} catch (err: any) {
|
|
expect(err).to.exist()
|
|
expect(err.code).to.equal('ERR_NO_ROUTERS_AVAILABLE')
|
|
}
|
|
})
|
|
})
|
|
|
|
describe('via dht router', () => {
|
|
let nodes: Libp2pNode[]
|
|
|
|
before(async () => {
|
|
nodes = await Promise.all([
|
|
createNode({ config: createRoutingOptions() }),
|
|
createNode({ config: createRoutingOptions() }),
|
|
createNode({ config: createRoutingOptions() }),
|
|
createNode({ config: createRoutingOptions() }),
|
|
createNode({ config: createRoutingOptions() })
|
|
])
|
|
await populateAddressBooks(nodes)
|
|
|
|
// Ring dial
|
|
await Promise.all(
|
|
nodes.map(async (peer, i) => await peer.dial(nodes[(i + 1) % nodes.length].peerId))
|
|
)
|
|
})
|
|
|
|
after(() => {
|
|
sinon.restore()
|
|
})
|
|
|
|
after(async () => await Promise.all(nodes.map(async (n) => await n.stop())))
|
|
|
|
it('should use the nodes dht', async () => {
|
|
if (nodes[0].dht == null) {
|
|
throw new Error('DHT not configured')
|
|
}
|
|
|
|
const dhtFindPeerStub = sinon.stub(nodes[0].dht, 'findPeer').callsFake(async function * () {
|
|
yield {
|
|
from: nodes[2].peerId,
|
|
type: EventTypes.FINAL_PEER,
|
|
name: 'FINAL_PEER',
|
|
peer: {
|
|
id: nodes[1].peerId,
|
|
multiaddrs: [],
|
|
protocols: []
|
|
}
|
|
}
|
|
})
|
|
|
|
expect(dhtFindPeerStub.called).to.be.false()
|
|
await nodes[0].peerRouting.findPeer(nodes[1].peerId)
|
|
expect(dhtFindPeerStub.called).to.be.true()
|
|
dhtFindPeerStub.restore()
|
|
})
|
|
|
|
it('should use the nodes dht to get the closest peers', async () => {
|
|
if (nodes[0].dht == null) {
|
|
throw new Error('DHT not configured')
|
|
}
|
|
|
|
const dhtGetClosestPeersStub = sinon.stub(nodes[0].dht, 'getClosestPeers').callsFake(async function * () {
|
|
yield {
|
|
from: nodes[2].peerId,
|
|
type: EventTypes.FINAL_PEER,
|
|
name: 'FINAL_PEER',
|
|
messageName: 'FIND_NODE',
|
|
messageType: MessageType.FIND_NODE,
|
|
peer: {
|
|
id: nodes[1].peerId,
|
|
multiaddrs: [],
|
|
protocols: []
|
|
}
|
|
}
|
|
})
|
|
|
|
expect(dhtGetClosestPeersStub.called).to.be.false()
|
|
await drain(nodes[0].peerRouting.getClosestPeers(nodes[1].peerId.toBytes()))
|
|
expect(dhtGetClosestPeersStub.called).to.be.true()
|
|
dhtGetClosestPeersStub.restore()
|
|
})
|
|
|
|
it('should error when peer tries to find itself', async () => {
|
|
await expect(nodes[0].peerRouting.findPeer(nodes[0].peerId))
|
|
.to.eventually.be.rejected()
|
|
.and.to.have.property('code', 'ERR_FIND_SELF')
|
|
})
|
|
|
|
it('should handle error thrown synchronously during find peer', async () => {
|
|
const unknownPeer = await createPeerId()
|
|
|
|
// @ts-expect-error private field
|
|
nodes[0].peerRouting.routers = [{
|
|
findPeer () {
|
|
throw new Error('Thrown sync')
|
|
}
|
|
}]
|
|
|
|
await expect(nodes[0].peerRouting.findPeer(unknownPeer))
|
|
.to.eventually.be.rejected()
|
|
.and.to.have.property('code', 'ERR_NOT_FOUND')
|
|
})
|
|
|
|
it('should handle error thrown asynchronously during find peer', async () => {
|
|
const unknownPeer = await createPeerId()
|
|
|
|
// @ts-expect-error private field
|
|
nodes[0].peerRouting.routers = [{
|
|
async findPeer () {
|
|
throw new Error('Thrown async')
|
|
}
|
|
}]
|
|
|
|
await expect(nodes[0].peerRouting.findPeer(unknownPeer))
|
|
.to.eventually.be.rejected()
|
|
.and.to.have.property('code', 'ERR_NOT_FOUND')
|
|
})
|
|
|
|
it('should handle error thrown asynchronously after delay during find peer', async () => {
|
|
const unknownPeer = await createPeerId()
|
|
|
|
// @ts-expect-error private field
|
|
nodes[0].peerRouting.routers = [{
|
|
async findPeer () {
|
|
await delay(100)
|
|
throw new Error('Thrown async after delay')
|
|
}
|
|
}]
|
|
|
|
await expect(nodes[0].peerRouting.findPeer(unknownPeer))
|
|
.to.eventually.be.rejected()
|
|
.and.to.have.property('code', 'ERR_NOT_FOUND')
|
|
})
|
|
|
|
it('should return value when one router errors synchronously and another returns a value', async () => {
|
|
const peer = await createPeerId()
|
|
|
|
// @ts-expect-error private field
|
|
nodes[0].peerRouting.routers = [{
|
|
findPeer () {
|
|
throw new Error('Thrown sync')
|
|
}
|
|
}, {
|
|
async findPeer () {
|
|
return await Promise.resolve({
|
|
id: peer,
|
|
multiaddrs: []
|
|
})
|
|
}
|
|
}]
|
|
|
|
await expect(nodes[0].peerRouting.findPeer(peer))
|
|
.to.eventually.deep.equal({
|
|
id: peer,
|
|
multiaddrs: []
|
|
})
|
|
})
|
|
|
|
it('should return value when one router errors asynchronously and another returns a value', async () => {
|
|
const peer = await createPeerId()
|
|
|
|
// @ts-expect-error private field
|
|
nodes[0].peerRouting.routers = [{
|
|
async findPeer () {
|
|
throw new Error('Thrown sync')
|
|
}
|
|
}, {
|
|
async findPeer () {
|
|
return await Promise.resolve({
|
|
id: peer,
|
|
multiaddrs: []
|
|
})
|
|
}
|
|
}]
|
|
|
|
await expect(nodes[0].peerRouting.findPeer(peer))
|
|
.to.eventually.deep.equal({
|
|
id: peer,
|
|
multiaddrs: []
|
|
})
|
|
})
|
|
})
|
|
|
|
describe('via delegate router', () => {
|
|
let node: Libp2pNode
|
|
let delegate: StubbedInstance<PeerRouting>
|
|
|
|
beforeEach(async () => {
|
|
delegate = stubInterface<PeerRouting>()
|
|
delegate.findPeer.returns(Promise.reject(new Error('Could not find peer')))
|
|
delegate.getClosestPeers.returns(async function * () {}())
|
|
|
|
node = await createNode({
|
|
config: createBaseOptions({
|
|
peerRouters: [delegate]
|
|
})
|
|
})
|
|
})
|
|
|
|
afterEach(() => {
|
|
sinon.restore()
|
|
})
|
|
|
|
afterEach(async () => await node.stop())
|
|
|
|
it('should only have one router', () => {
|
|
// @ts-expect-error private field
|
|
expect(node.peerRouting.routers).to.have.lengthOf(1)
|
|
})
|
|
|
|
it('should use the delegate router to find peers', async () => {
|
|
const remotePeerId = await createPeerId()
|
|
|
|
delegate.findPeer.callsFake(async function () {
|
|
return {
|
|
id: remotePeerId,
|
|
multiaddrs: [],
|
|
protocols: []
|
|
}
|
|
})
|
|
|
|
expect(delegate.findPeer.called).to.be.false()
|
|
await node.peerRouting.findPeer(remotePeerId)
|
|
expect(delegate.findPeer.called).to.be.true()
|
|
})
|
|
|
|
it('should use the delegate router to get the closest peers', async () => {
|
|
const remotePeerId = await createPeerId()
|
|
|
|
delegate.getClosestPeers.callsFake(async function * () {
|
|
yield {
|
|
id: remotePeerId,
|
|
multiaddrs: [],
|
|
protocols: []
|
|
}
|
|
})
|
|
|
|
expect(delegate.getClosestPeers.called).to.be.false()
|
|
await drain(node.peerRouting.getClosestPeers(remotePeerId.toBytes()))
|
|
expect(delegate.getClosestPeers.called).to.be.true()
|
|
})
|
|
|
|
it('should error when peer tries to find itself', async () => {
|
|
await expect(node.peerRouting.findPeer(node.peerId))
|
|
.to.eventually.be.rejected()
|
|
.and.to.have.property('code', 'ERR_FIND_SELF')
|
|
})
|
|
|
|
it('should handle errors from the delegate when finding closest peers', async () => {
|
|
const remotePeerId = await createPeerId()
|
|
|
|
delegate.getClosestPeers.callsFake(async function * () { // eslint-disable-line require-yield
|
|
throw new Error('Could not find closer peers')
|
|
})
|
|
|
|
expect(delegate.getClosestPeers.called).to.be.false()
|
|
await expect(drain(node.peerRouting.getClosestPeers(remotePeerId.toBytes())))
|
|
.to.eventually.be.rejectedWith('Could not find closer peers')
|
|
})
|
|
})
|
|
|
|
describe('via dht and delegate routers', () => {
|
|
let node: Libp2pNode
|
|
let delegate: StubbedInstance<PeerRouting>
|
|
|
|
beforeEach(async () => {
|
|
delegate = stubInterface<PeerRouting>()
|
|
delegate.findPeer.throws(new Error('Could not find peer'))
|
|
delegate.getClosestPeers.returns(async function * () {}())
|
|
|
|
node = await createNode({
|
|
config: createRoutingOptions({
|
|
peerRouters: [delegate],
|
|
dht: new KadDHT()
|
|
})
|
|
})
|
|
})
|
|
|
|
afterEach(() => {
|
|
sinon.restore()
|
|
})
|
|
|
|
afterEach(async () => await node.stop())
|
|
|
|
it('should use the delegate if the dht fails to find the peer', async () => {
|
|
const remotePeerId = await createPeerId()
|
|
const results = {
|
|
id: remotePeerId,
|
|
multiaddrs: [],
|
|
protocols: []
|
|
}
|
|
|
|
if (node.dht == null) {
|
|
throw new Error('DHT not configured')
|
|
}
|
|
|
|
sinon.stub(node.dht, 'findPeer').callsFake(async function * () {})
|
|
delegate.findPeer.reset()
|
|
delegate.findPeer.callsFake(async () => {
|
|
return results
|
|
})
|
|
|
|
const peer = await node.peerRouting.findPeer(remotePeerId)
|
|
expect(peer).to.eql(results)
|
|
})
|
|
|
|
it('should not wait for the dht to return if the delegate does first', async () => {
|
|
const remotePeerId = await createPeerId()
|
|
const results = {
|
|
id: remotePeerId,
|
|
multiaddrs: [],
|
|
protocols: []
|
|
}
|
|
|
|
if (node.dht == null) {
|
|
throw new Error('DHT not configured')
|
|
}
|
|
|
|
const defer = pDefer()
|
|
|
|
sinon.stub(node.dht, 'findPeer').callsFake(async function * () {
|
|
yield {
|
|
name: 'SENDING_QUERY',
|
|
type: EventTypes.SENDING_QUERY,
|
|
to: remotePeerId,
|
|
messageName: 'FIND_NODE',
|
|
messageType: MessageType.FIND_NODE
|
|
}
|
|
await defer.promise
|
|
})
|
|
delegate.findPeer.reset()
|
|
delegate.findPeer.callsFake(async () => {
|
|
return results
|
|
})
|
|
|
|
const peer = await node.peerRouting.findPeer(remotePeerId)
|
|
expect(peer).to.eql(results)
|
|
|
|
defer.resolve()
|
|
})
|
|
|
|
it('should not wait for the delegate to return if the dht does first', async () => {
|
|
const remotePeerId = await createPeerId()
|
|
const result = {
|
|
id: remotePeerId,
|
|
multiaddrs: [],
|
|
protocols: []
|
|
}
|
|
|
|
if (node.dht == null) {
|
|
throw new Error('DHT not configured')
|
|
}
|
|
|
|
const defer = pDefer<PeerInfo>()
|
|
|
|
sinon.stub(node.dht, 'findPeer').callsFake(async function * () {
|
|
yield {
|
|
from: remotePeerId,
|
|
name: 'FINAL_PEER',
|
|
type: EventTypes.FINAL_PEER,
|
|
peer: result
|
|
}
|
|
})
|
|
delegate.findPeer.reset()
|
|
delegate.findPeer.callsFake(async () => {
|
|
return await defer.promise
|
|
})
|
|
|
|
const peer = await node.peerRouting.findPeer(remotePeerId)
|
|
expect(peer).to.eql(result)
|
|
|
|
defer.resolve(result)
|
|
})
|
|
|
|
it('should store the addresses of the found peer', async () => {
|
|
const remotePeerId = await createPeerId()
|
|
const result = {
|
|
id: remotePeerId,
|
|
multiaddrs: [
|
|
multiaddr('/ip4/123.123.123.123/tcp/38982')
|
|
],
|
|
protocols: []
|
|
}
|
|
|
|
if (node.dht == null) {
|
|
throw new Error('DHT not configured')
|
|
}
|
|
|
|
const spy = sinon.spy(node.peerStore.addressBook, 'add')
|
|
|
|
sinon.stub(node.dht, 'findPeer').callsFake(async function * () {
|
|
yield {
|
|
from: remotePeerId,
|
|
name: 'FINAL_PEER',
|
|
type: EventTypes.FINAL_PEER,
|
|
peer: result
|
|
}
|
|
})
|
|
delegate.findPeer.reset()
|
|
delegate.findPeer.callsFake(async () => {
|
|
const deferred = pDefer<PeerInfo>()
|
|
|
|
return await deferred.promise
|
|
})
|
|
|
|
await node.peerRouting.findPeer(remotePeerId)
|
|
|
|
expect(spy.calledWith(result.id, result.multiaddrs)).to.be.true()
|
|
})
|
|
|
|
it('should use the delegate if the dht fails to get the closest peer', async () => {
|
|
const remotePeerId = await createPeerId()
|
|
const results = [{
|
|
id: remotePeerId,
|
|
multiaddrs: [],
|
|
protocols: []
|
|
}]
|
|
|
|
if (node.dht == null) {
|
|
throw new Error('DHT not configured')
|
|
}
|
|
|
|
sinon.stub(node.dht, 'getClosestPeers').callsFake(async function * () { })
|
|
|
|
delegate.getClosestPeers.callsFake(async function * () {
|
|
yield results[0]
|
|
})
|
|
|
|
const closest = await all(node.peerRouting.getClosestPeers(remotePeerId.toBytes()))
|
|
|
|
expect(closest).to.have.length.above(0)
|
|
expect(closest).to.eql(results)
|
|
})
|
|
|
|
it('should store the addresses of the closest peer', async () => {
|
|
const remotePeerId = await createPeerId()
|
|
const result = {
|
|
id: remotePeerId,
|
|
multiaddrs: [
|
|
multiaddr('/ip4/123.123.123.123/tcp/38982')
|
|
],
|
|
protocols: []
|
|
}
|
|
|
|
if (node.dht == null) {
|
|
throw new Error('DHT not configured')
|
|
}
|
|
|
|
const spy = sinon.spy(node.peerStore.addressBook, 'add')
|
|
|
|
sinon.stub(node.dht, 'getClosestPeers').callsFake(async function * () { })
|
|
|
|
delegate.getClosestPeers.callsFake(async function * () {
|
|
yield result
|
|
})
|
|
|
|
await drain(node.peerRouting.getClosestPeers(remotePeerId.toBytes()))
|
|
|
|
expect(spy.calledWith(result.id, result.multiaddrs)).to.be.true()
|
|
})
|
|
|
|
it('should dedupe closest peers', async () => {
|
|
const remotePeerId = await createPeerId()
|
|
const results = [{
|
|
id: remotePeerId,
|
|
multiaddrs: [
|
|
multiaddr('/ip4/123.123.123.123/tcp/38982')
|
|
],
|
|
protocols: []
|
|
}]
|
|
|
|
if (node.dht == null) {
|
|
throw new Error('DHT not configured')
|
|
}
|
|
|
|
sinon.stub(node.dht, 'getClosestPeers').callsFake(async function * () {
|
|
for (const peer of results) {
|
|
yield {
|
|
from: remotePeerId,
|
|
name: 'FINAL_PEER',
|
|
type: EventTypes.FINAL_PEER,
|
|
peer
|
|
}
|
|
}
|
|
})
|
|
|
|
delegate.getClosestPeers.callsFake(async function * () {
|
|
yield * results
|
|
})
|
|
|
|
const peers = await all(node.peerRouting.getClosestPeers(remotePeerId.toBytes()))
|
|
|
|
expect(peers).to.be.an('array').with.a.lengthOf(1).that.deep.equals(results)
|
|
})
|
|
})
|
|
|
|
describe('peer routing refresh manager service', () => {
|
|
let node: Libp2pNode
|
|
let peerIds: PeerId[]
|
|
|
|
before(async () => {
|
|
peerIds = await Promise.all([
|
|
createPeerId(),
|
|
createPeerId()
|
|
])
|
|
})
|
|
|
|
afterEach(async () => {
|
|
sinon.restore()
|
|
|
|
if (node != null) {
|
|
await node.stop()
|
|
}
|
|
})
|
|
|
|
it('should be enabled and start by default', async () => {
|
|
const results: PeerInfo[] = [
|
|
{ id: peerIds[0], multiaddrs: [multiaddr('/ip4/30.0.0.1/tcp/2000')], protocols: [] },
|
|
{ id: peerIds[1], multiaddrs: [multiaddr('/ip4/32.0.0.1/tcp/2000')], protocols: [] }
|
|
]
|
|
|
|
node = await createNode({
|
|
config: createRoutingOptions({
|
|
peerRouting: {
|
|
refreshManager: {
|
|
enabled: true,
|
|
bootDelay: 100
|
|
}
|
|
}
|
|
}),
|
|
started: false
|
|
})
|
|
|
|
if (node.dht == null) {
|
|
throw new Error('DHT not configured')
|
|
}
|
|
|
|
const peerStoreAddressBookAddStub = sinon.spy(node.peerStore.addressBook, 'add')
|
|
const dhtGetClosestPeersStub = sinon.stub(node.dht, 'getClosestPeers').callsFake(async function * () {
|
|
yield {
|
|
name: 'FINAL_PEER',
|
|
type: EventTypes.FINAL_PEER,
|
|
messageName: 'FIND_NODE',
|
|
messageType: MessageType.FIND_NODE,
|
|
from: peerIds[0],
|
|
peer: results[0]
|
|
}
|
|
yield {
|
|
name: 'FINAL_PEER',
|
|
type: EventTypes.FINAL_PEER,
|
|
messageName: 'FIND_NODE',
|
|
messageType: MessageType.FIND_NODE,
|
|
from: peerIds[0],
|
|
peer: results[1]
|
|
}
|
|
})
|
|
|
|
await node.start()
|
|
|
|
await pWaitFor(() => dhtGetClosestPeersStub.callCount === 1)
|
|
await pWaitFor(() => peerStoreAddressBookAddStub.callCount === results.length)
|
|
|
|
const call0 = peerStoreAddressBookAddStub.getCall(0)
|
|
expect(call0.args[0].equals(results[0].id))
|
|
call0.args[1].forEach((m, index) => {
|
|
expect(m.equals(results[0].multiaddrs[index]))
|
|
})
|
|
|
|
const call1 = peerStoreAddressBookAddStub.getCall(1)
|
|
expect(call1.args[0].equals(results[1].id))
|
|
call0.args[1].forEach((m, index) => {
|
|
expect(m.equals(results[1].multiaddrs[index]))
|
|
})
|
|
})
|
|
|
|
it('should support being disabled', async () => {
|
|
node = await createNode({
|
|
config: createRoutingOptions({
|
|
peerRouting: {
|
|
refreshManager: {
|
|
bootDelay: 100,
|
|
enabled: false
|
|
}
|
|
}
|
|
}),
|
|
started: false
|
|
})
|
|
|
|
if (node.dht == null) {
|
|
throw new Error('DHT not configured')
|
|
}
|
|
|
|
const dhtGetClosestPeersStub = sinon.stub(node.dht, 'getClosestPeers').callsFake(async function * () {
|
|
yield {
|
|
name: 'SENDING_QUERY',
|
|
type: EventTypes.SENDING_QUERY,
|
|
to: peerIds[0],
|
|
messageName: 'FIND_NODE',
|
|
messageType: MessageType.FIND_NODE
|
|
}
|
|
throw new Error('should not be called')
|
|
})
|
|
|
|
await node.start()
|
|
await delay(100)
|
|
|
|
expect(dhtGetClosestPeersStub.callCount === 0)
|
|
})
|
|
|
|
it('should start and run on interval', async () => {
|
|
node = await createNode({
|
|
config: createRoutingOptions({
|
|
peerRouting: {
|
|
refreshManager: {
|
|
interval: 500,
|
|
bootDelay: 200
|
|
}
|
|
}
|
|
}),
|
|
started: false
|
|
})
|
|
|
|
if (node.dht == null) {
|
|
throw new Error('DHT not configured')
|
|
}
|
|
|
|
const dhtGetClosestPeersStub = sinon.stub(node.dht, 'getClosestPeers').callsFake(async function * () {
|
|
yield {
|
|
name: 'PEER_RESPONSE',
|
|
type: EventTypes.PEER_RESPONSE,
|
|
messageName: 'FIND_NODE',
|
|
messageType: MessageType.FIND_NODE,
|
|
from: peerIds[0],
|
|
closer: [
|
|
{ id: peerIds[0], multiaddrs: [multiaddr('/ip4/30.0.0.1/tcp/2000')], protocols: [] }
|
|
],
|
|
providers: []
|
|
}
|
|
})
|
|
|
|
await node.start()
|
|
|
|
// should run more than once
|
|
await pWaitFor(() => dhtGetClosestPeersStub.callCount === 2)
|
|
})
|
|
})
|
|
})
|