@libp2p/webtransport 4.0.28 → 4.0.29
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.min.js +3 -3
- package/dist/src/index.d.ts +12 -1
- package/dist/src/index.d.ts.map +1 -1
- package/dist/src/index.js +55 -94
- package/dist/src/index.js.map +1 -1
- package/dist/src/listener.browser.d.ts +3 -0
- package/dist/src/listener.browser.d.ts.map +1 -0
- package/dist/src/listener.browser.js +4 -0
- package/dist/src/listener.browser.js.map +1 -0
- package/dist/src/listener.d.ts +15 -0
- package/dist/src/listener.d.ts.map +1 -0
- package/dist/src/listener.js +4 -0
- package/dist/src/listener.js.map +1 -0
- package/dist/src/muxer.d.ts +7 -0
- package/dist/src/muxer.d.ts.map +1 -0
- package/dist/src/muxer.js +70 -0
- package/dist/src/muxer.js.map +1 -0
- package/dist/src/stream.d.ts.map +1 -1
- package/dist/src/stream.js +79 -151
- package/dist/src/stream.js.map +1 -1
- package/dist/src/utils/generate-certificates.browser.d.ts +2 -0
- package/dist/src/utils/generate-certificates.browser.d.ts.map +1 -0
- package/dist/src/utils/generate-certificates.browser.js +4 -0
- package/dist/src/utils/generate-certificates.browser.js.map +1 -0
- package/dist/src/utils/generate-certificates.d.ts +8 -0
- package/dist/src/utils/generate-certificates.d.ts.map +1 -0
- package/dist/src/utils/generate-certificates.js +4 -0
- package/dist/src/utils/generate-certificates.js.map +1 -0
- package/dist/src/webtransport.browser.d.ts +2 -0
- package/dist/src/webtransport.browser.d.ts.map +1 -0
- package/dist/src/webtransport.browser.js +2 -0
- package/dist/src/webtransport.browser.js.map +1 -0
- package/dist/src/webtransport.d.ts +9 -0
- package/dist/src/webtransport.d.ts.map +1 -0
- package/dist/src/webtransport.js +15 -0
- package/dist/src/webtransport.js.map +1 -0
- package/dist/typedoc-urls.json +2 -0
- package/package.json +18 -9
- package/src/index.ts +74 -122
- package/src/listener.browser.ts +5 -0
- package/src/listener.ts +19 -0
- package/src/muxer.ts +95 -0
- package/src/stream.ts +90 -164
- package/src/utils/generate-certificates.browser.ts +3 -0
- package/src/utils/generate-certificates.ts +11 -0
- package/src/webtransport.browser.ts +1 -0
- package/src/webtransport.ts +17 -0
package/src/stream.ts
CHANGED
|
@@ -1,184 +1,110 @@
|
|
|
1
|
+
import { AbstractStream, type AbstractStreamInit } from '@libp2p/utils/abstract-stream'
|
|
2
|
+
import { raceSignal } from 'race-signal'
|
|
1
3
|
import { Uint8ArrayList } from 'uint8arraylist'
|
|
2
4
|
import type { AbortOptions, ComponentLogger, Direction, Stream } from '@libp2p/interface'
|
|
3
|
-
import type { Source } from 'it-stream-types'
|
|
4
5
|
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
const reader = bidiStream.readable.getReader()
|
|
9
|
-
await writer.ready
|
|
10
|
-
|
|
11
|
-
function cleanupStreamFromActiveStreams (): void {
|
|
12
|
-
const index = activeStreams.findIndex(s => s === stream)
|
|
13
|
-
if (index !== -1) {
|
|
14
|
-
activeStreams.splice(index, 1)
|
|
15
|
-
stream.timeline.close = Date.now()
|
|
16
|
-
onStreamEnd?.(stream)
|
|
17
|
-
}
|
|
18
|
-
}
|
|
6
|
+
interface WebTransportStreamInit extends AbstractStreamInit {
|
|
7
|
+
bidiStream: WebTransportBidirectionalStream
|
|
8
|
+
}
|
|
19
9
|
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
const err: Error | undefined = await writer.closed.catch((err: Error) => err)
|
|
24
|
-
if (err != null) {
|
|
25
|
-
const msg = err.message
|
|
26
|
-
if (!(msg.includes('aborted by the remote server') || msg.includes('STOP_SENDING'))) {
|
|
27
|
-
log.error(`WebTransport writer closed unexpectedly: streamId=${streamId} err=${err.message}`)
|
|
28
|
-
}
|
|
29
|
-
}
|
|
30
|
-
writerClosed = true
|
|
31
|
-
if (writerClosed && readerClosed) {
|
|
32
|
-
cleanupStreamFromActiveStreams()
|
|
33
|
-
}
|
|
34
|
-
})().catch(() => {
|
|
35
|
-
log.error('WebTransport failed to cleanup closed stream')
|
|
36
|
-
});
|
|
37
|
-
|
|
38
|
-
(async function () {
|
|
39
|
-
const err: Error | undefined = await reader.closed.catch((err: Error) => err)
|
|
40
|
-
if (err != null) {
|
|
41
|
-
log.error(`WebTransport reader closed unexpectedly: streamId=${streamId} err=${err.message}`)
|
|
42
|
-
}
|
|
43
|
-
readerClosed = true
|
|
44
|
-
if (writerClosed && readerClosed) {
|
|
45
|
-
cleanupStreamFromActiveStreams()
|
|
46
|
-
}
|
|
47
|
-
})().catch(() => {
|
|
48
|
-
log.error('WebTransport failed to cleanup closed stream')
|
|
49
|
-
})
|
|
10
|
+
class WebTransportStream extends AbstractStream {
|
|
11
|
+
private readonly writer: WritableStreamDefaultWriter<Uint8Array>
|
|
12
|
+
private readonly reader: ReadableStreamDefaultReader<Uint8Array>
|
|
50
13
|
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
id: streamId,
|
|
54
|
-
status: 'open',
|
|
55
|
-
writeStatus: 'ready',
|
|
56
|
-
readStatus: 'ready',
|
|
57
|
-
abort (err: Error) {
|
|
58
|
-
if (!writerClosed) {
|
|
59
|
-
writer.abort(err)
|
|
60
|
-
.catch(err => {
|
|
61
|
-
log.error('could not abort stream', err)
|
|
62
|
-
})
|
|
63
|
-
writerClosed = true
|
|
64
|
-
}
|
|
65
|
-
readerClosed = true
|
|
66
|
-
|
|
67
|
-
this.status = 'aborted'
|
|
68
|
-
this.writeStatus = 'closed'
|
|
69
|
-
this.readStatus = 'closed'
|
|
70
|
-
|
|
71
|
-
this.timeline.reset =
|
|
72
|
-
this.timeline.close =
|
|
73
|
-
this.timeline.closeRead =
|
|
74
|
-
this.timeline.closeWrite = Date.now()
|
|
75
|
-
|
|
76
|
-
cleanupStreamFromActiveStreams()
|
|
77
|
-
},
|
|
78
|
-
async close (options?: AbortOptions) {
|
|
79
|
-
this.status = 'closing'
|
|
80
|
-
|
|
81
|
-
await Promise.all([
|
|
82
|
-
stream.closeRead(options),
|
|
83
|
-
stream.closeWrite(options)
|
|
84
|
-
])
|
|
85
|
-
|
|
86
|
-
cleanupStreamFromActiveStreams()
|
|
87
|
-
|
|
88
|
-
this.status = 'closed'
|
|
89
|
-
this.timeline.close = Date.now()
|
|
90
|
-
},
|
|
91
|
-
|
|
92
|
-
async closeRead (options?: AbortOptions) {
|
|
93
|
-
if (!readerClosed) {
|
|
94
|
-
this.readStatus = 'closing'
|
|
95
|
-
|
|
96
|
-
try {
|
|
97
|
-
await reader.cancel()
|
|
98
|
-
} catch (err: any) {
|
|
99
|
-
if (err.toString().includes('RESET_STREAM') === true) {
|
|
100
|
-
writerClosed = true
|
|
101
|
-
}
|
|
102
|
-
}
|
|
14
|
+
constructor (init: WebTransportStreamInit) {
|
|
15
|
+
super(init)
|
|
103
16
|
|
|
104
|
-
|
|
105
|
-
|
|
17
|
+
this.writer = init.bidiStream.writable.getWriter()
|
|
18
|
+
this.reader = init.bidiStream.readable.getReader()
|
|
106
19
|
|
|
107
|
-
|
|
108
|
-
|
|
20
|
+
Promise.resolve().then(async () => {
|
|
21
|
+
while (true) {
|
|
22
|
+
const result = await this.reader.read()
|
|
109
23
|
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
24
|
+
if (result.done) {
|
|
25
|
+
init.log('remote closed write')
|
|
26
|
+
return
|
|
27
|
+
}
|
|
114
28
|
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
29
|
+
if (result.value != null) {
|
|
30
|
+
this.sourcePush(new Uint8ArrayList(result.value))
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
})
|
|
34
|
+
.catch(err => {
|
|
35
|
+
init.log.error('error reading from stream', err)
|
|
36
|
+
this.abort(err)
|
|
37
|
+
})
|
|
38
|
+
.finally(() => {
|
|
39
|
+
this.remoteCloseWrite()
|
|
40
|
+
})
|
|
41
|
+
|
|
42
|
+
void this.writer.closed
|
|
43
|
+
.then(() => {
|
|
44
|
+
init.log('writer closed')
|
|
45
|
+
})
|
|
46
|
+
.catch((err) => {
|
|
47
|
+
init.log('writer close promise rejected', err)
|
|
48
|
+
})
|
|
49
|
+
.finally(() => {
|
|
50
|
+
this.remoteCloseRead()
|
|
51
|
+
})
|
|
52
|
+
}
|
|
118
53
|
|
|
119
|
-
|
|
54
|
+
sendNewStream (options?: AbortOptions | undefined): void {
|
|
55
|
+
// this is a no-op
|
|
56
|
+
}
|
|
120
57
|
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
58
|
+
async sendData (buf: Uint8ArrayList, options?: AbortOptions): Promise<void> {
|
|
59
|
+
for await (const chunk of buf) {
|
|
60
|
+
this.log('sendData waiting for writer to be ready')
|
|
61
|
+
await raceSignal(this.writer.ready, options?.signal)
|
|
62
|
+
|
|
63
|
+
// the streams spec recommends not waiting for data to be sent
|
|
64
|
+
// https://streams.spec.whatwg.org/#example-manual-write-dont-await
|
|
65
|
+
this.writer.write(chunk)
|
|
66
|
+
.catch(err => {
|
|
67
|
+
this.log.error('error sending stream data', err)
|
|
68
|
+
})
|
|
69
|
+
}
|
|
70
|
+
}
|
|
128
71
|
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
72
|
+
async sendReset (options?: AbortOptions): Promise<void> {
|
|
73
|
+
this.log('sendReset aborting writer')
|
|
74
|
+
await raceSignal(this.writer.abort(), options?.signal)
|
|
75
|
+
this.log('sendReset aborted writer')
|
|
76
|
+
}
|
|
132
77
|
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
timeline: { open: Date.now() },
|
|
139
|
-
metadata: {},
|
|
140
|
-
source: (async function * () {
|
|
141
|
-
while (true) {
|
|
142
|
-
const val = await reader.read()
|
|
143
|
-
if (val.done) {
|
|
144
|
-
readerClosed = true
|
|
145
|
-
if (writerClosed) {
|
|
146
|
-
cleanupStreamFromActiveStreams()
|
|
147
|
-
}
|
|
148
|
-
return
|
|
149
|
-
}
|
|
78
|
+
async sendCloseWrite (options?: AbortOptions): Promise<void> {
|
|
79
|
+
this.log('sendCloseWrite closing writer')
|
|
80
|
+
await raceSignal(this.writer.close(), options?.signal)
|
|
81
|
+
this.log('sendCloseWrite closed writer')
|
|
82
|
+
}
|
|
150
83
|
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
}
|
|
158
|
-
sinkSunk = true
|
|
159
|
-
try {
|
|
160
|
-
this.writeStatus = 'writing'
|
|
161
|
-
|
|
162
|
-
for await (const chunks of source) {
|
|
163
|
-
if (chunks instanceof Uint8Array) {
|
|
164
|
-
await writer.write(chunks)
|
|
165
|
-
} else {
|
|
166
|
-
for (const buf of chunks) {
|
|
167
|
-
await writer.write(buf)
|
|
168
|
-
}
|
|
169
|
-
}
|
|
170
|
-
}
|
|
84
|
+
async sendCloseRead (options?: AbortOptions): Promise<void> {
|
|
85
|
+
this.log('sendCloseRead cancelling reader')
|
|
86
|
+
await raceSignal(this.reader.cancel(), options?.signal)
|
|
87
|
+
this.log('sendCloseRead cancelled reader')
|
|
88
|
+
}
|
|
89
|
+
}
|
|
171
90
|
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
this.timeline.closeWrite = Date.now()
|
|
175
|
-
this.writeStatus = 'closed'
|
|
91
|
+
export async function webtransportBiDiStreamToStream (bidiStream: WebTransportBidirectionalStream, streamId: string, direction: Direction, activeStreams: Stream[], onStreamEnd: undefined | ((s: Stream) => void), logger: ComponentLogger): Promise<Stream> {
|
|
92
|
+
const log = logger.forComponent(`libp2p:webtransport:stream:${direction}:${streamId}`)
|
|
176
93
|
|
|
177
|
-
|
|
94
|
+
const stream = new WebTransportStream({
|
|
95
|
+
bidiStream,
|
|
96
|
+
id: streamId,
|
|
97
|
+
direction,
|
|
98
|
+
log,
|
|
99
|
+
onEnd: () => {
|
|
100
|
+
const index = activeStreams.findIndex(s => s === stream)
|
|
101
|
+
if (index !== -1) {
|
|
102
|
+
activeStreams.splice(index, 1)
|
|
178
103
|
}
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
104
|
+
|
|
105
|
+
onStreamEnd?.(stream)
|
|
106
|
+
}
|
|
107
|
+
})
|
|
182
108
|
|
|
183
109
|
return stream
|
|
184
110
|
}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import type { WebTransportCertificate } from '../../src/index.js'
|
|
2
|
+
|
|
3
|
+
export interface GenerateWebTransportCertificateOptions {
|
|
4
|
+
days: number
|
|
5
|
+
start?: Date
|
|
6
|
+
extensions?: any[]
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
export async function generateWebTransportCertificates (options: GenerateWebTransportCertificateOptions[] = []): Promise<WebTransportCertificate[]> {
|
|
10
|
+
throw new Error('Not implemented')
|
|
11
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export default WebTransport
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
export default class WebTransport {
|
|
2
|
+
constructor (url: string | URL, options?: WebTransportOptions) {
|
|
3
|
+
throw new Error('Only supported in browsers')
|
|
4
|
+
}
|
|
5
|
+
|
|
6
|
+
close (): void {
|
|
7
|
+
throw new Error('Only supported in browsers')
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
async createBidirectionalStream (): Promise<WebTransportBidirectionalStream> {
|
|
11
|
+
throw new Error('Only supported in browsers')
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
public closed = Promise.reject(new Error('Only supported in browsers'))
|
|
15
|
+
public ready = Promise.reject(new Error('Only supported in browsers'))
|
|
16
|
+
public incomingBidirectionalStreams: ReadableStream
|
|
17
|
+
}
|