Skip to content

Commit 557234d

Browse files
authored
fix: add browser shim (#18)
To allow importing the module in browsers, add a shim that will throw if it's actually used. Simplifies writing isomorphic code where we may configure libp2p in different ways based on the environment, but we don't have to worry about `import`ing this but not `import`ing that.
1 parent c22f429 commit 557234d

8 files changed

+90
-19
lines changed

.gitignore

+2-1
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@ lib-cov
2525
# Coverage directory used by tools like istanbul
2626
coverage
2727
*.lcov
28+
.coverage
2829

2930
# nyc test coverage
3031
.nyc_output
@@ -121,7 +122,7 @@ dist
121122
.AppleDouble
122123
.LSOverride
123124

124-
# Icon must end with two
125+
# Icon must end with two
125126
Icon
126127

127128

package.json

+14-1
Original file line numberDiff line numberDiff line change
@@ -58,6 +58,12 @@
5858
"lint": "aegir lint",
5959
"prepublishOnly": "napi prepublish -t npm",
6060
"test": "aegir test -t node",
61+
"test:chrome": "aegir test -t browser -f ./dist/test/browser.js --cov",
62+
"test:chrome-webworker": "aegir test -t webworker -f ./dist/test/browser.js",
63+
"test:firefox": "aegir test -t browser -f ./dist/test/browser.js -- --browser firefox",
64+
"test:firefox-webworker": "aegir test -t webworker -f ./dist/test/browser.js -- --browser firefox",
65+
"test:node": "aegir test -t node --cov",
66+
"test:electron-main": "aegir test -t electron-main",
6167
"universal": "napi universal",
6268
"version": "napi version"
6369
},
@@ -76,5 +82,12 @@
7682
"aegir": "^45.1.3",
7783
"sinon-ts": "^2.0.0"
7884
},
79-
"packageManager": "[email protected]"
85+
"packageManager": "[email protected]",
86+
"browser": {
87+
"./dist/src/transport.js": "./dist/src/transport.browser.js"
88+
},
89+
"react-native": {
90+
"./dist/src/transport.js": "./dist/src/transport.js"
91+
},
92+
"sideEffects": false
8093
}

src/index.ts

+13-2
Original file line numberDiff line numberDiff line change
@@ -27,8 +27,19 @@
2727
* ```
2828
*/
2929

30-
import { QuicTransport, type QuicComponents, type QuicOptions } from './transport.js'
31-
import type { Transport } from '@libp2p/interface'
30+
import { QuicTransport } from './transport.js'
31+
import type { ComponentLogger, DialTransportOptions, Metrics, PrivateKey, Transport } from '@libp2p/interface'
32+
import type * as napi from './napi.js'
33+
34+
export type QuicOptions = Omit<napi.Config, 'privateKeyProto'>
35+
36+
export interface QuicComponents {
37+
metrics?: Metrics
38+
logger: ComponentLogger
39+
privateKey: PrivateKey
40+
}
41+
42+
export type QuicDialOptions = DialTransportOptions
3243

3344
export function quic (options?: Partial<QuicOptions>): (components: QuicComponents) => Transport {
3445
return (components) => new QuicTransport(components, { ...defaultOptions, ...options })

src/transport.browser.ts

+25
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
import { serviceCapabilities, transportSymbol } from '@libp2p/interface'
2+
import type { Connection, Listener, MultiaddrFilter, Transport } from '@libp2p/interface'
3+
4+
export class QuicTransport implements Transport {
5+
readonly [Symbol.toStringTag]: string = 'quic'
6+
readonly [transportSymbol] = true
7+
readonly [serviceCapabilities]: string[] = [
8+
'@libp2p/transport'
9+
]
10+
11+
readonly listenFilter: MultiaddrFilter
12+
readonly dialFilter: MultiaddrFilter
13+
14+
constructor () {
15+
throw new Error('QUIC connections are not possible in browsers')
16+
}
17+
18+
async dial (): Promise<Connection> {
19+
throw new Error('QUIC connections are not possible in browsers')
20+
}
21+
22+
createListener (): Listener {
23+
throw new Error('QUIC connections are not possible in browsers')
24+
}
25+
}

src/transport.ts

+7-13
Original file line numberDiff line numberDiff line change
@@ -1,30 +1,24 @@
11
import { privateKeyToProtobuf } from '@libp2p/crypto/keys'
2-
import { AbortError, transportSymbol } from '@libp2p/interface'
2+
import { AbortError, serviceCapabilities, transportSymbol } from '@libp2p/interface'
33
import { QuicConnection } from './connection.js'
44
import { dialFilter, listenFilter } from './filter.js'
55
import { QuicListener, type QuicCreateListenerOptions } from './listener.js'
66
import * as napi from './napi.js'
77
import { QuicStreamMuxerFactory } from './stream-muxer.js'
8-
import type { ComponentLogger, Connection, CounterGroup, DialTransportOptions, Listener, Logger, Metrics, MultiaddrFilter, PrivateKey, Transport } from '@libp2p/interface'
8+
import type { Connection, CounterGroup, Listener, Logger, MultiaddrFilter, Transport } from '@libp2p/interface'
99
import type { Multiaddr } from '@multiformats/multiaddr'
10+
import type { QuicComponents, QuicDialOptions, QuicOptions } from './index.js'
1011

11-
export type QuicOptions = Omit<napi.Config, 'privateKeyProto'>
12-
13-
export interface QuicComponents {
14-
metrics?: Metrics
15-
logger: ComponentLogger
16-
privateKey: PrivateKey
17-
}
18-
19-
export type QuicDialOptions = DialTransportOptions
20-
21-
export interface QuicTransportMetrics {
12+
interface QuicTransportMetrics {
2213
events: CounterGroup
2314
}
2415

2516
export class QuicTransport implements Transport {
2617
readonly [Symbol.toStringTag]: string = 'quic'
2718
readonly [transportSymbol] = true
19+
readonly [serviceCapabilities]: string[] = [
20+
'@libp2p/transport'
21+
]
2822

2923
readonly log: Logger
3024
readonly components: QuicComponents

test/browser.ts

+27
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
import { generateKeyPair } from '@libp2p/crypto/keys'
2+
import { defaultLogger } from '@libp2p/logger'
3+
import { expect } from 'aegir/chai'
4+
import { isBrowser, isWebWorker } from 'wherearewe'
5+
import { quic } from '../src/index.js'
6+
import type { PrivateKey } from '@libp2p/interface'
7+
8+
describe('browser non-support', () => {
9+
let privateKey: PrivateKey
10+
11+
beforeEach(async () => {
12+
privateKey = await generateKeyPair('Ed25519')
13+
})
14+
15+
it('should throw in browsers', async function () {
16+
if (!isBrowser && !isWebWorker) {
17+
return this.skip()
18+
}
19+
20+
expect(() => {
21+
quic()({
22+
logger: defaultLogger(),
23+
privateKey
24+
})
25+
}).to.be.throw()
26+
})
27+
})

test/transport.spec.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ import { pEvent } from 'p-event'
99
import { stubInterface } from 'sinon-ts'
1010
import { quic } from '../src/index.js'
1111
import { createComponents } from './util.js'
12-
import type { QuicComponents } from '../src/transport.js'
12+
import type { QuicComponents } from '../src/index.js'
1313
import type { Listener, Upgrader } from '@libp2p/interface'
1414

1515
describe('Quic Transport', () => {

test/util.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
import { generateKeyPair } from '@libp2p/crypto/keys'
22
import { defaultLogger } from '@libp2p/logger'
3-
import type { QuicComponents } from '../src/transport.js'
3+
import type { QuicComponents } from '../src/index.js'
44

55
export async function createComponents (): Promise<QuicComponents> {
66
return {

0 commit comments

Comments
 (0)