mirror of
https://github.com/fluencelabs/js-libp2p
synced 2025-04-04 00:31:04 +00:00
123 lines
3.9 KiB
TypeScript
123 lines
3.9 KiB
TypeScript
|
/* eslint-env mocha */
|
||
|
|
||
|
import { expect } from 'aegir/chai'
|
||
|
import sinon from 'sinon'
|
||
|
import { PingService } from '../../src/ping/index.js'
|
||
|
import Peers from '../fixtures/peers.js'
|
||
|
import { mockRegistrar, mockUpgrader, connectionPair } from '@libp2p/interface-compliance-tests/mocks'
|
||
|
import { createFromJSON } from '@libp2p/peer-id-factory'
|
||
|
import { Components } from '@libp2p/interfaces/components'
|
||
|
import { DefaultConnectionManager } from '../../src/connection-manager/index.js'
|
||
|
import { start, stop } from '@libp2p/interfaces/startable'
|
||
|
import { CustomEvent } from '@libp2p/interfaces/events'
|
||
|
import { TimeoutController } from 'timeout-abort-controller'
|
||
|
import delay from 'delay'
|
||
|
import { pipe } from 'it-pipe'
|
||
|
|
||
|
const defaultInit = {
|
||
|
protocolPrefix: 'ipfs'
|
||
|
}
|
||
|
|
||
|
async function createComponents (index: number) {
|
||
|
const peerId = await createFromJSON(Peers[index])
|
||
|
|
||
|
const components = new Components({
|
||
|
peerId,
|
||
|
registrar: mockRegistrar(),
|
||
|
upgrader: mockUpgrader(),
|
||
|
connectionManager: new DefaultConnectionManager({
|
||
|
minConnections: 50,
|
||
|
maxConnections: 1000,
|
||
|
autoDialInterval: 1000
|
||
|
})
|
||
|
})
|
||
|
|
||
|
return components
|
||
|
}
|
||
|
|
||
|
describe('ping', () => {
|
||
|
let localComponents: Components
|
||
|
let remoteComponents: Components
|
||
|
|
||
|
beforeEach(async () => {
|
||
|
localComponents = await createComponents(0)
|
||
|
remoteComponents = await createComponents(1)
|
||
|
|
||
|
await Promise.all([
|
||
|
start(localComponents),
|
||
|
start(remoteComponents)
|
||
|
])
|
||
|
})
|
||
|
|
||
|
afterEach(async () => {
|
||
|
sinon.restore()
|
||
|
|
||
|
await Promise.all([
|
||
|
stop(localComponents),
|
||
|
stop(remoteComponents)
|
||
|
])
|
||
|
})
|
||
|
|
||
|
it('should be able to ping another peer', async () => {
|
||
|
const localPing = new PingService(localComponents, defaultInit)
|
||
|
const remotePing = new PingService(remoteComponents, defaultInit)
|
||
|
|
||
|
await start(localPing)
|
||
|
await start(remotePing)
|
||
|
|
||
|
// simulate connection between nodes
|
||
|
const [localToRemote, remoteToLocal] = connectionPair(localComponents, remoteComponents)
|
||
|
localComponents.getUpgrader().dispatchEvent(new CustomEvent('connection', { detail: localToRemote }))
|
||
|
remoteComponents.getUpgrader().dispatchEvent(new CustomEvent('connection', { detail: remoteToLocal }))
|
||
|
|
||
|
// Run ping
|
||
|
await expect(localPing.ping(remoteComponents.getPeerId())).to.eventually.be.gte(0)
|
||
|
})
|
||
|
|
||
|
it('should time out pinging another peer when waiting for a pong', async () => {
|
||
|
const localPing = new PingService(localComponents, defaultInit)
|
||
|
const remotePing = new PingService(remoteComponents, defaultInit)
|
||
|
|
||
|
await start(localPing)
|
||
|
await start(remotePing)
|
||
|
|
||
|
// simulate connection between nodes
|
||
|
const [localToRemote, remoteToLocal] = connectionPair(localComponents, remoteComponents)
|
||
|
localComponents.getUpgrader().dispatchEvent(new CustomEvent('connection', { detail: localToRemote }))
|
||
|
remoteComponents.getUpgrader().dispatchEvent(new CustomEvent('connection', { detail: remoteToLocal }))
|
||
|
|
||
|
// replace existing handler with a really slow one
|
||
|
await remoteComponents.getRegistrar().unhandle(remotePing.protocol)
|
||
|
await remoteComponents.getRegistrar().handle(remotePing.protocol, ({ stream }) => {
|
||
|
void pipe(
|
||
|
stream,
|
||
|
async function * (source) {
|
||
|
for await (const chunk of source) {
|
||
|
// longer than the timeout
|
||
|
await delay(1000)
|
||
|
|
||
|
yield chunk
|
||
|
}
|
||
|
},
|
||
|
stream
|
||
|
)
|
||
|
})
|
||
|
|
||
|
const newStreamSpy = sinon.spy(localToRemote, 'newStream')
|
||
|
|
||
|
// 10 ms timeout
|
||
|
const timeoutController = new TimeoutController(10)
|
||
|
|
||
|
// Run ping, should time out
|
||
|
await expect(localPing.ping(remoteComponents.getPeerId(), {
|
||
|
signal: timeoutController.signal
|
||
|
}))
|
||
|
.to.eventually.be.rejected.with.property('code', 'ABORT_ERR')
|
||
|
|
||
|
// should have closed stream
|
||
|
expect(newStreamSpy).to.have.property('callCount', 1)
|
||
|
const { stream } = await newStreamSpy.getCall(0).returnValue
|
||
|
expect(stream).to.have.nested.property('timeline.close')
|
||
|
})
|
||
|
})
|