@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.
Files changed (47) hide show
  1. package/dist/index.min.js +3 -3
  2. package/dist/src/index.d.ts +12 -1
  3. package/dist/src/index.d.ts.map +1 -1
  4. package/dist/src/index.js +55 -94
  5. package/dist/src/index.js.map +1 -1
  6. package/dist/src/listener.browser.d.ts +3 -0
  7. package/dist/src/listener.browser.d.ts.map +1 -0
  8. package/dist/src/listener.browser.js +4 -0
  9. package/dist/src/listener.browser.js.map +1 -0
  10. package/dist/src/listener.d.ts +15 -0
  11. package/dist/src/listener.d.ts.map +1 -0
  12. package/dist/src/listener.js +4 -0
  13. package/dist/src/listener.js.map +1 -0
  14. package/dist/src/muxer.d.ts +7 -0
  15. package/dist/src/muxer.d.ts.map +1 -0
  16. package/dist/src/muxer.js +70 -0
  17. package/dist/src/muxer.js.map +1 -0
  18. package/dist/src/stream.d.ts.map +1 -1
  19. package/dist/src/stream.js +79 -151
  20. package/dist/src/stream.js.map +1 -1
  21. package/dist/src/utils/generate-certificates.browser.d.ts +2 -0
  22. package/dist/src/utils/generate-certificates.browser.d.ts.map +1 -0
  23. package/dist/src/utils/generate-certificates.browser.js +4 -0
  24. package/dist/src/utils/generate-certificates.browser.js.map +1 -0
  25. package/dist/src/utils/generate-certificates.d.ts +8 -0
  26. package/dist/src/utils/generate-certificates.d.ts.map +1 -0
  27. package/dist/src/utils/generate-certificates.js +4 -0
  28. package/dist/src/utils/generate-certificates.js.map +1 -0
  29. package/dist/src/webtransport.browser.d.ts +2 -0
  30. package/dist/src/webtransport.browser.d.ts.map +1 -0
  31. package/dist/src/webtransport.browser.js +2 -0
  32. package/dist/src/webtransport.browser.js.map +1 -0
  33. package/dist/src/webtransport.d.ts +9 -0
  34. package/dist/src/webtransport.d.ts.map +1 -0
  35. package/dist/src/webtransport.js +15 -0
  36. package/dist/src/webtransport.js.map +1 -0
  37. package/dist/typedoc-urls.json +2 -0
  38. package/package.json +18 -9
  39. package/src/index.ts +74 -122
  40. package/src/listener.browser.ts +5 -0
  41. package/src/listener.ts +19 -0
  42. package/src/muxer.ts +95 -0
  43. package/src/stream.ts +90 -164
  44. package/src/utils/generate-certificates.browser.ts +3 -0
  45. package/src/utils/generate-certificates.ts +11 -0
  46. package/src/webtransport.browser.ts +1 -0
  47. 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
- export async function webtransportBiDiStreamToStream (bidiStream: WebTransportBidirectionalStream, streamId: string, direction: Direction, activeStreams: Stream[], onStreamEnd: undefined | ((s: Stream) => void), logger: ComponentLogger): Promise<Stream> {
6
- const log = logger.forComponent(`libp2p:webtransport:stream:${direction}:${streamId}`)
7
- const writer = bidiStream.writable.getWriter()
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
- let writerClosed = false
21
- let readerClosed = false;
22
- (async function () {
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
- let sinkSunk = false
52
- const stream: Stream = {
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
- this.timeline.closeRead = Date.now()
105
- this.readStatus = 'closed'
17
+ this.writer = init.bidiStream.writable.getWriter()
18
+ this.reader = init.bidiStream.readable.getReader()
106
19
 
107
- readerClosed = true
108
- }
20
+ Promise.resolve().then(async () => {
21
+ while (true) {
22
+ const result = await this.reader.read()
109
23
 
110
- if (writerClosed) {
111
- cleanupStreamFromActiveStreams()
112
- }
113
- },
24
+ if (result.done) {
25
+ init.log('remote closed write')
26
+ return
27
+ }
114
28
 
115
- async closeWrite (options?: AbortOptions) {
116
- if (!writerClosed) {
117
- writerClosed = true
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
- this.writeStatus = 'closing'
54
+ sendNewStream (options?: AbortOptions | undefined): void {
55
+ // this is a no-op
56
+ }
120
57
 
121
- try {
122
- await writer.close()
123
- } catch (err: any) {
124
- if (err.toString().includes('RESET_STREAM') === true) {
125
- readerClosed = true
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
- this.timeline.closeWrite = Date.now()
130
- this.writeStatus = 'closed'
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
- if (readerClosed) {
134
- cleanupStreamFromActiveStreams()
135
- }
136
- },
137
- direction,
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
- yield new Uint8ArrayList(val.value)
152
- }
153
- })(),
154
- sink: async function (source: Source<Uint8Array | Uint8ArrayList>) {
155
- if (sinkSunk) {
156
- throw new Error('sink already called on stream')
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
- this.writeStatus = 'done'
173
- } finally {
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
- await stream.closeWrite()
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
- log
181
- }
104
+
105
+ onStreamEnd?.(stream)
106
+ }
107
+ })
182
108
 
183
109
  return stream
184
110
  }
@@ -0,0 +1,3 @@
1
+ export async function generateWebTransportCertificates (): Promise<any> {
2
+ throw new Error('Not implemented')
3
+ }
@@ -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
+ }