Skip to content

Commit 4f9024e

Browse files
authored
fix: pass interface compliance tests (#15)
- Awaits connection close promise so we can close the libp2p version - Extend AbstractStream to handle the (approx) 3k ways there are to close a stream - Ensure closed streams are removed from the muxer - Enable interface compliance tests - Listen on wildcard addresses properly
1 parent 4382173 commit 4f9024e

18 files changed

+477
-780
lines changed

.github/workflows/CI.yml

+12-12
Original file line numberDiff line numberDiff line change
@@ -123,8 +123,7 @@ jobs:
123123
target: x86_64-pc-windows-msvc
124124
architecture: x64
125125
node:
126-
- '18'
127-
- '22'
126+
- 22
128127
runs-on: ${{ matrix.settings.host }}
129128
steps:
130129
- uses: actions/checkout@v4
@@ -157,8 +156,7 @@ jobs:
157156
fail-fast: false
158157
matrix:
159158
node:
160-
- '18'
161-
- '22'
159+
- 22
162160
runs-on: ubuntu-latest
163161
steps:
164162
- uses: actions/checkout@v4
@@ -173,12 +171,12 @@ jobs:
173171
uses: actions/download-artifact@v4
174172
with:
175173
name: bindings-x86_64-unknown-linux-gnu
176-
path: src
174+
path: src
177175
- name: List packages
178176
run: ls -R .
179177
shell: bash
180178
- name: Test bindings
181-
run: docker run --rm -v $(pwd):/build -w /build node:${{ matrix.node }}-slim corepack enable && yarn run test
179+
run: docker run --rm -v "$(pwd):/build" -w /build node:${{ matrix.node }}-slim corepack enable && yarn run test
182180
test-linux-x64-musl-binding:
183181
name: Test bindings on x86_64-unknown-linux-musl - node@${{ matrix.node }}
184182
needs:
@@ -187,8 +185,7 @@ jobs:
187185
fail-fast: false
188186
matrix:
189187
node:
190-
- '18'
191-
- '22'
188+
- 22
192189
runs-on: ubuntu-latest
193190
steps:
194191
- uses: actions/checkout@v4
@@ -223,7 +220,7 @@ jobs:
223220
fail-fast: false
224221
matrix:
225222
node:
226-
- '22'
223+
- 22
227224
runs-on: ubuntu-latest
228225
steps:
229226
- uses: actions/checkout@v4
@@ -260,8 +257,7 @@ jobs:
260257
fail-fast: false
261258
matrix:
262259
node:
263-
- '18'
264-
- '22'
260+
- 22
265261
runs-on: ubuntu-latest
266262
steps:
267263
- uses: actions/checkout@v4
@@ -294,7 +290,11 @@ jobs:
294290
name: Publish
295291
runs-on: ubuntu-latest
296292
needs:
297-
- build
293+
- test-macOS-windows-binding
294+
- test-linux-x64-gnu-binding
295+
- test-linux-x64-musl-binding
296+
- test-linux-aarch64-gnu-binding
297+
- test-linux-aarch64-musl-binding
298298
steps:
299299
- uses: googleapis/release-please-action@v4
300300
id: release

package.json

+9-1
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,13 @@
4040
"url": "https://github.com/ChainSafe/js-libp2p-quic"
4141
},
4242
"homepage": "https://github.com/ChainSafe/js-libp2p-quic",
43+
"eslintConfig": {
44+
"extends": "ipfs",
45+
"parserOptions": {
46+
"project": true,
47+
"sourceType": "module"
48+
}
49+
},
4350
"scripts": {
4451
"artifacts": "napi artifacts",
4552
"build": "yarn build:ts",
@@ -66,7 +73,8 @@
6673
"@libp2p/interface-compliance-tests": "^6.3.2",
6774
"@libp2p/logger": "^5.0.1",
6875
"@napi-rs/cli": "^3.0.0-alpha.70",
69-
"aegir": "^44.1.1"
76+
"aegir": "^45.1.3",
77+
"sinon-ts": "^2.0.0"
7078
},
7179
"packageManager": "[email protected]"
7280
}

rust/lib.rs

+5
Original file line numberDiff line numberDiff line change
@@ -45,6 +45,11 @@ impl Server {
4545
Ok(Self { socket, endpoint })
4646
}
4747

48+
#[napi]
49+
pub fn port(&self) -> u16 {
50+
return self.endpoint.local_addr().unwrap().port();
51+
}
52+
4853
#[napi]
4954
pub async fn inbound_connection(&self) -> Result<Connection> {
5055
let incoming = match self.endpoint.accept().await {

src/connection.ts

+35-18
Original file line numberDiff line numberDiff line change
@@ -1,20 +1,18 @@
11
import { TypedEventEmitter } from '@libp2p/interface'
22
import { multiaddr, type Multiaddr } from '@multiformats/multiaddr'
3-
import { Uint8ArrayList } from 'uint8arraylist'
4-
3+
import { type Uint8ArrayList } from 'uint8arraylist'
4+
import type * as napi from './napi.js'
55
import type { AbortOptions, ComponentLogger, CounterGroup, Direction, Logger, MultiaddrConnection, MultiaddrConnectionTimeline } from '@libp2p/interface'
66
import type { Sink } from 'it-stream-types'
77

8-
import * as napi from './napi.js'
9-
10-
type QuicConnectionInit = {
8+
interface QuicConnectionInit {
119
connection: napi.Connection
1210
logger: ComponentLogger
1311
direction: Direction
1412
metrics?: CounterGroup
1513
}
1614

17-
type QuicConnectionEvents = {
15+
interface QuicConnectionEvents {
1816
'close': CustomEvent
1917
}
2018

@@ -26,35 +24,54 @@ export class QuicConnection extends TypedEventEmitter<QuicConnectionEvents> impl
2624
readonly metrics?: CounterGroup
2725

2826
timeline: MultiaddrConnectionTimeline = {
29-
open: Date.now(),
27+
open: Date.now()
3028
}
31-
source: AsyncGenerator<Uint8Array | Uint8ArrayList> = (async function* () {})()
32-
sink: Sink<AsyncGenerator<Uint8Array | Uint8ArrayList>> = async function* () {}
3329

34-
constructor(init: QuicConnectionInit) {
30+
source: AsyncGenerator<Uint8Array | Uint8ArrayList> = (async function * () {})()
31+
sink: Sink<AsyncGenerator<Uint8Array | Uint8ArrayList>> = async function * () {}
32+
33+
constructor (init: QuicConnectionInit) {
3534
super()
3635

3736
this.#connection = init.connection
38-
this.log = init.logger.forComponent('libp2p:quic:connection')
37+
this.log = init.logger.forComponent(`libp2p:quic:connection:${this.#connection.id()}:${init.direction}`)
3938
this.remoteAddr = multiaddr(this.#connection.remoteMultiaddr())
4039
this.metrics = init.metrics
4140

42-
this.log('new', init.direction, this.#connection.id())
41+
// close maconn when connection is closed by remote
42+
this.#connection.closed().then(() => {
43+
this.close()
44+
.catch(err => {
45+
this.abort(err)
46+
})
47+
}, (err) => {
48+
this.abort(err)
49+
})
4350
}
44-
async close(options?: AbortOptions): Promise<void> {
51+
52+
async close (options?: AbortOptions): Promise<void> {
53+
if (this.timeline.close != null) {
54+
return
55+
}
56+
4557
this.#connection.abort()
4658

4759
this.timeline.close = Date.now()
48-
this.log('%s closed', this.#connection.id())
49-
this.metrics?.increment({close: true})
60+
this.log('closed')
61+
this.metrics?.increment({ close: true })
5062
this.safeDispatchEvent('close')
5163
}
52-
abort(err: Error): void {
64+
65+
abort (err: Error): void {
66+
if (this.timeline.close != null) {
67+
return
68+
}
69+
5370
this.#connection.abort()
5471

5572
this.timeline.close = Date.now()
56-
this.log('%s aborted', this.#connection.id())
57-
this.metrics?.increment({abort: true})
73+
this.log('aborted - %e', err)
74+
this.metrics?.increment({ abort: true })
5875
this.safeDispatchEvent('close')
5976
}
6077
}

src/filter.ts

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,17 +1,17 @@
1-
import type { Multiaddr } from "@multiformats/multiaddr"
21
import { QUICV1 } from '@multiformats/multiaddr-matcher'
2+
import type { Multiaddr } from '@multiformats/multiaddr'
33

44
// p2p multi-address code
55
export const CODE_P2P = 421
66
export const CODE_CIRCUIT = 290
77
export const CODE_UNIX = 400
88

9-
export function listenFilter(multiaddrs: Multiaddr[]): Multiaddr[] {
9+
export function listenFilter (multiaddrs: Multiaddr[]): Multiaddr[] {
1010
return multiaddrs.filter((ma) => {
1111
return QUICV1.exactMatch(ma)
1212
})
1313
}
1414

15-
export function dialFilter(multiaddrs: Multiaddr[]): Multiaddr[] {
15+
export function dialFilter (multiaddrs: Multiaddr[]): Multiaddr[] {
1616
return listenFilter(multiaddrs)
1717
}

src/index.ts

+4-4
Original file line numberDiff line numberDiff line change
@@ -27,11 +27,11 @@
2727
* ```
2828
*/
2929

30-
import type { Transport } from '@libp2p/interface'
3130
import { QuicTransport, type QuicComponents, type QuicOptions } from './transport.js'
31+
import type { Transport } from '@libp2p/interface'
3232

33-
export function quic(options?: Partial<QuicOptions>): (components: QuicComponents) => Transport {
34-
return (components) => new QuicTransport(components, {...defaultOptions, ...options})
33+
export function quic (options?: Partial<QuicOptions>): (components: QuicComponents) => Transport {
34+
return (components) => new QuicTransport(components, { ...defaultOptions, ...options })
3535
}
3636

3737
export const defaultOptions: QuicOptions = {
@@ -40,5 +40,5 @@ export const defaultOptions: QuicOptions = {
4040
keepAliveInterval: 5_000,
4141
maxConcurrentStreamLimit: 256,
4242
maxStreamData: 10_000_000,
43-
maxConnectionData: 15_000_000,
43+
maxConnectionData: 15_000_000
4444
}

0 commit comments

Comments
 (0)