@libp2p/interface-compliance-tests 3.0.5 → 3.0.7-05abd49f

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 (138) hide show
  1. package/README.md +14 -5
  2. package/dist/src/connection/index.d.ts +5 -0
  3. package/dist/src/connection/index.d.ts.map +1 -0
  4. package/dist/src/connection/index.js +151 -0
  5. package/dist/src/connection/index.js.map +1 -0
  6. package/dist/src/connection-encryption/index.d.ts +5 -0
  7. package/dist/src/connection-encryption/index.d.ts.map +1 -0
  8. package/dist/src/connection-encryption/index.js +71 -0
  9. package/dist/src/connection-encryption/index.js.map +1 -0
  10. package/dist/src/connection-encryption/utils/index.d.ts +3 -0
  11. package/dist/src/connection-encryption/utils/index.d.ts.map +1 -0
  12. package/dist/src/connection-encryption/utils/index.js +18 -0
  13. package/dist/src/connection-encryption/utils/index.js.map +1 -0
  14. package/dist/src/index.d.ts +1 -1
  15. package/dist/src/index.d.ts.map +1 -1
  16. package/dist/src/is-valid-tick.d.ts.map +1 -1
  17. package/dist/src/mocks/connection-encrypter.d.ts +3 -0
  18. package/dist/src/mocks/connection-encrypter.d.ts.map +1 -0
  19. package/dist/src/mocks/connection-encrypter.js +98 -0
  20. package/dist/src/mocks/connection-encrypter.js.map +1 -0
  21. package/dist/src/mocks/connection-gater.d.ts +3 -0
  22. package/dist/src/mocks/connection-gater.d.ts.map +1 -0
  23. package/dist/src/mocks/connection-gater.js +17 -0
  24. package/dist/src/mocks/connection-gater.js.map +1 -0
  25. package/dist/src/mocks/connection-manager.d.ts +27 -0
  26. package/dist/src/mocks/connection-manager.d.ts.map +1 -0
  27. package/dist/src/mocks/connection-manager.js +145 -0
  28. package/dist/src/mocks/connection-manager.js.map +1 -0
  29. package/dist/src/mocks/connection.d.ts +32 -0
  30. package/dist/src/mocks/connection.d.ts.map +1 -0
  31. package/dist/src/mocks/connection.js +162 -0
  32. package/dist/src/mocks/connection.js.map +1 -0
  33. package/dist/src/mocks/duplex.d.ts +3 -0
  34. package/dist/src/mocks/duplex.d.ts.map +1 -0
  35. package/dist/src/mocks/duplex.js +9 -0
  36. package/dist/src/mocks/duplex.js.map +1 -0
  37. package/dist/src/mocks/index.d.ts +13 -0
  38. package/dist/src/mocks/index.d.ts.map +1 -0
  39. package/dist/src/mocks/index.js +11 -0
  40. package/dist/src/mocks/index.js.map +1 -0
  41. package/dist/src/mocks/metrics.d.ts +3 -0
  42. package/dist/src/mocks/metrics.d.ts.map +1 -0
  43. package/dist/src/mocks/metrics.js +115 -0
  44. package/dist/src/mocks/metrics.js.map +1 -0
  45. package/dist/src/mocks/multiaddr-connection.d.ts +17 -0
  46. package/dist/src/mocks/multiaddr-connection.d.ts.map +1 -0
  47. package/dist/src/mocks/multiaddr-connection.js +51 -0
  48. package/dist/src/mocks/multiaddr-connection.js.map +1 -0
  49. package/dist/src/mocks/muxer.d.ts +8 -0
  50. package/dist/src/mocks/muxer.d.ts.map +1 -0
  51. package/dist/src/mocks/muxer.js +341 -0
  52. package/dist/src/mocks/muxer.js.map +1 -0
  53. package/dist/src/mocks/peer-discovery.d.ts +22 -0
  54. package/dist/src/mocks/peer-discovery.d.ts.map +1 -0
  55. package/dist/src/mocks/peer-discovery.js +47 -0
  56. package/dist/src/mocks/peer-discovery.js.map +1 -0
  57. package/dist/src/mocks/registrar.d.ts +18 -0
  58. package/dist/src/mocks/registrar.d.ts.map +1 -0
  59. package/dist/src/mocks/registrar.js +66 -0
  60. package/dist/src/mocks/registrar.js.map +1 -0
  61. package/dist/src/mocks/upgrader.d.ts +10 -0
  62. package/dist/src/mocks/upgrader.d.ts.map +1 -0
  63. package/dist/src/mocks/upgrader.js +31 -0
  64. package/dist/src/mocks/upgrader.js.map +1 -0
  65. package/dist/src/peer-discovery/index.d.ts +5 -0
  66. package/dist/src/peer-discovery/index.d.ts.map +1 -0
  67. package/dist/src/peer-discovery/index.js +66 -0
  68. package/dist/src/peer-discovery/index.js.map +1 -0
  69. package/dist/src/stream-muxer/base-test.d.ts +5 -0
  70. package/dist/src/stream-muxer/base-test.d.ts.map +1 -0
  71. package/dist/src/stream-muxer/base-test.js +153 -0
  72. package/dist/src/stream-muxer/base-test.js.map +1 -0
  73. package/dist/src/stream-muxer/close-test.d.ts +5 -0
  74. package/dist/src/stream-muxer/close-test.d.ts.map +1 -0
  75. package/dist/src/stream-muxer/close-test.js +269 -0
  76. package/dist/src/stream-muxer/close-test.js.map +1 -0
  77. package/dist/src/stream-muxer/index.d.ts +5 -0
  78. package/dist/src/stream-muxer/index.d.ts.map +1 -0
  79. package/dist/src/stream-muxer/index.js +13 -0
  80. package/dist/src/stream-muxer/index.js.map +1 -0
  81. package/dist/src/stream-muxer/mega-stress-test.d.ts +5 -0
  82. package/dist/src/stream-muxer/mega-stress-test.d.ts.map +1 -0
  83. package/dist/src/stream-muxer/mega-stress-test.js +11 -0
  84. package/dist/src/stream-muxer/mega-stress-test.js.map +1 -0
  85. package/dist/src/stream-muxer/spawner.d.ts +4 -0
  86. package/dist/src/stream-muxer/spawner.d.ts.map +1 -0
  87. package/dist/src/stream-muxer/spawner.js +36 -0
  88. package/dist/src/stream-muxer/spawner.js.map +1 -0
  89. package/dist/src/stream-muxer/stress-test.d.ts +5 -0
  90. package/dist/src/stream-muxer/stress-test.d.ts.map +1 -0
  91. package/dist/src/stream-muxer/stress-test.js +23 -0
  92. package/dist/src/stream-muxer/stress-test.js.map +1 -0
  93. package/dist/src/transport/dial-test.d.ts +5 -0
  94. package/dist/src/transport/dial-test.d.ts.map +1 -0
  95. package/dist/src/transport/dial-test.js +98 -0
  96. package/dist/src/transport/dial-test.js.map +1 -0
  97. package/dist/src/transport/filter-test.d.ts +5 -0
  98. package/dist/src/transport/filter-test.d.ts.map +1 -0
  99. package/dist/src/transport/filter-test.js +18 -0
  100. package/dist/src/transport/filter-test.js.map +1 -0
  101. package/dist/src/transport/index.d.ts +15 -0
  102. package/dist/src/transport/index.d.ts.map +1 -0
  103. package/dist/src/transport/index.js +11 -0
  104. package/dist/src/transport/index.js.map +1 -0
  105. package/dist/src/transport/listen-test.d.ts +5 -0
  106. package/dist/src/transport/listen-test.d.ts.map +1 -0
  107. package/dist/src/transport/listen-test.js +152 -0
  108. package/dist/src/transport/listen-test.js.map +1 -0
  109. package/package.json +66 -95
  110. package/src/connection/index.ts +184 -0
  111. package/src/connection-encryption/index.ts +97 -0
  112. package/src/connection-encryption/utils/index.ts +23 -0
  113. package/src/index.ts +1 -1
  114. package/src/is-valid-tick.ts +1 -1
  115. package/src/mocks/connection-encrypter.ts +113 -0
  116. package/src/mocks/connection-gater.ts +18 -0
  117. package/src/mocks/connection-manager.ts +209 -0
  118. package/src/mocks/connection.ts +218 -0
  119. package/src/mocks/duplex.ts +10 -0
  120. package/src/mocks/index.ts +12 -0
  121. package/src/mocks/metrics.ts +162 -0
  122. package/src/mocks/multiaddr-connection.ts +67 -0
  123. package/src/mocks/muxer.ts +447 -0
  124. package/src/mocks/peer-discovery.ts +60 -0
  125. package/src/mocks/registrar.ts +88 -0
  126. package/src/mocks/upgrader.ts +49 -0
  127. package/src/peer-discovery/index.ts +90 -0
  128. package/src/stream-muxer/base-test.ts +196 -0
  129. package/src/stream-muxer/close-test.ts +346 -0
  130. package/src/stream-muxer/index.ts +15 -0
  131. package/src/stream-muxer/mega-stress-test.ts +14 -0
  132. package/src/stream-muxer/spawner.ts +54 -0
  133. package/src/stream-muxer/stress-test.ts +27 -0
  134. package/src/transport/dial-test.ts +124 -0
  135. package/src/transport/filter-test.ts +25 -0
  136. package/src/transport/index.ts +25 -0
  137. package/src/transport/listen-test.ts +191 -0
  138. package/dist/typedoc-urls.json +0 -3
@@ -0,0 +1,90 @@
1
+ import { start, stop } from '@libp2p/interface/startable'
2
+ import { isMultiaddr } from '@multiformats/multiaddr'
3
+ import { expect } from 'aegir/chai'
4
+ import delay from 'delay'
5
+ import pDefer from 'p-defer'
6
+ import type { TestSetup } from '../index.js'
7
+ import type { PeerDiscovery } from '@libp2p/interface/peer-discovery'
8
+
9
+ export default (common: TestSetup<PeerDiscovery>): void => {
10
+ describe('interface-peer-discovery compliance tests', () => {
11
+ let discovery: PeerDiscovery
12
+
13
+ beforeEach(async () => {
14
+ discovery = await common.setup()
15
+ })
16
+
17
+ afterEach('ensure discovery was stopped', async () => {
18
+ await stop(discovery)
19
+
20
+ await common.teardown()
21
+ })
22
+
23
+ it('can start the service', async () => {
24
+ await start(discovery)
25
+ })
26
+
27
+ it('can start and stop the service', async () => {
28
+ await start(discovery)
29
+ await stop(discovery)
30
+ })
31
+
32
+ it('should not fail to stop the service if it was not started', async () => {
33
+ await stop(discovery)
34
+ })
35
+
36
+ it('should not fail to start the service if it is already started', async () => {
37
+ await start(discovery)
38
+ await start(discovery)
39
+ })
40
+
41
+ it('should emit a peer event after start', async () => {
42
+ const defer = pDefer()
43
+
44
+ discovery.addEventListener('peer', (evt) => {
45
+ const { id, multiaddrs } = evt.detail
46
+ expect(id).to.exist()
47
+ expect(id)
48
+ .to.have.property('type')
49
+ .that.is.oneOf(['RSA', 'Ed25519', 'secp256k1'])
50
+ expect(multiaddrs).to.exist()
51
+
52
+ multiaddrs.forEach((m) => expect(isMultiaddr(m)).to.eql(true))
53
+
54
+ defer.resolve()
55
+ })
56
+
57
+ await start(discovery)
58
+
59
+ await defer.promise
60
+ })
61
+
62
+ it('should not receive a peer event before start', async () => {
63
+ discovery.addEventListener('peer', () => {
64
+ throw new Error('should not receive a peer event before start')
65
+ })
66
+
67
+ await delay(2000)
68
+ })
69
+
70
+ it('should not receive a peer event after stop', async () => {
71
+ const deferStart = pDefer()
72
+
73
+ discovery.addEventListener('peer', () => {
74
+ deferStart.resolve()
75
+ })
76
+
77
+ await start(discovery)
78
+
79
+ await deferStart.promise
80
+
81
+ await stop(discovery)
82
+
83
+ discovery.addEventListener('peer', () => {
84
+ throw new Error('should not receive a peer event after stop')
85
+ })
86
+
87
+ await delay(2000)
88
+ })
89
+ })
90
+ }
@@ -0,0 +1,196 @@
1
+ import { expect } from 'aegir/chai'
2
+ import all from 'it-all'
3
+ import drain from 'it-drain'
4
+ import map from 'it-map'
5
+ import { duplexPair } from 'it-pair/duplex'
6
+ import { pipe } from 'it-pipe'
7
+ import defer from 'p-defer'
8
+ import { Uint8ArrayList } from 'uint8arraylist'
9
+ import { fromString as uint8ArrayFromString } from 'uint8arrays/from-string'
10
+ import { toString as uint8ArrayToString } from 'uint8arrays/to-string'
11
+ import { isValidTick } from '../is-valid-tick.js'
12
+ import type { TestSetup } from '../index.js'
13
+ import type { Stream } from '@libp2p/interface/connection'
14
+ import type { StreamMuxerFactory } from '@libp2p/interface/stream-muxer'
15
+ import type { Source, Duplex } from 'it-stream-types'
16
+ import type { DeferredPromise } from 'p-defer'
17
+
18
+ async function drainAndClose (stream: Duplex<any>): Promise<void> {
19
+ await pipe([], stream, drain)
20
+ }
21
+
22
+ export default (common: TestSetup<StreamMuxerFactory>): void => {
23
+ describe('base', () => {
24
+ it('Open a stream from the dialer', async () => {
25
+ const p = duplexPair<Uint8Array>()
26
+ const dialerFactory = await common.setup()
27
+ const dialer = dialerFactory.createStreamMuxer({ direction: 'outbound' })
28
+ const onStreamPromise: DeferredPromise<Stream> = defer()
29
+ const onStreamEndPromise: DeferredPromise<Stream> = defer()
30
+
31
+ const listenerFactory = await common.setup()
32
+ const listener = listenerFactory.createStreamMuxer({
33
+ direction: 'inbound',
34
+ onIncomingStream: (stream) => {
35
+ onStreamPromise.resolve(stream)
36
+ },
37
+ onStreamEnd: (stream) => {
38
+ onStreamEndPromise.resolve(stream)
39
+ }
40
+ })
41
+
42
+ void pipe(p[0], dialer, p[0])
43
+ void pipe(p[1], listener, p[1])
44
+
45
+ const conn = await dialer.newStream()
46
+ expect(dialer.streams).to.include(conn)
47
+ expect(isValidTick(conn.stat.timeline.open)).to.equal(true)
48
+
49
+ void drainAndClose(conn)
50
+
51
+ const stream = await onStreamPromise.promise
52
+ expect(isValidTick(stream.stat.timeline.open)).to.equal(true)
53
+ // Make sure the stream is being tracked
54
+ expect(listener.streams).to.include(stream)
55
+
56
+ void drainAndClose(stream)
57
+
58
+ // Make sure stream is closed properly
59
+ const endedStream = await onStreamEndPromise.promise
60
+ expect(listener.streams).to.not.include(endedStream)
61
+
62
+ if (endedStream.stat.timeline.close == null) {
63
+ throw new Error('timeline had no close time')
64
+ }
65
+
66
+ // Make sure the stream is removed from tracking
67
+ expect(isValidTick(endedStream.stat.timeline.close)).to.equal(true)
68
+
69
+ await drainAndClose(dialer)
70
+ await drainAndClose(listener)
71
+
72
+ // ensure we have no streams left
73
+ expect(dialer.streams).to.have.length(0)
74
+ expect(listener.streams).to.have.length(0)
75
+ })
76
+
77
+ it('Open a stream from the listener', async () => {
78
+ const p = duplexPair<Uint8Array>()
79
+ const onStreamPromise: DeferredPromise<Stream> = defer()
80
+ const dialerFactory = await common.setup()
81
+ const dialer = dialerFactory.createStreamMuxer({
82
+ direction: 'outbound',
83
+ onIncomingStream: (stream: Stream) => {
84
+ onStreamPromise.resolve(stream)
85
+ }
86
+ })
87
+
88
+ const listenerFactory = await common.setup()
89
+ const listener = listenerFactory.createStreamMuxer({ direction: 'inbound' })
90
+
91
+ void pipe(p[0], dialer, p[0])
92
+ void pipe(p[1], listener, p[1])
93
+
94
+ const conn = await listener.newStream()
95
+
96
+ void drainAndClose(conn)
97
+
98
+ const stream = await onStreamPromise.promise
99
+ expect(isValidTick(stream.stat.timeline.open)).to.equal(true)
100
+ expect(listener.streams).to.include(conn)
101
+ expect(isValidTick(conn.stat.timeline.open)).to.equal(true)
102
+ void drainAndClose(stream)
103
+
104
+ await drainAndClose(dialer)
105
+ await drainAndClose(listener)
106
+ })
107
+
108
+ it('Open a stream on both sides', async () => {
109
+ const p = duplexPair<Uint8Array>()
110
+ const onDialerStreamPromise: DeferredPromise<Stream> = defer()
111
+ const onListenerStreamPromise: DeferredPromise<Stream> = defer()
112
+ const dialerFactory = await common.setup()
113
+ const dialer = dialerFactory.createStreamMuxer({
114
+ direction: 'outbound',
115
+ onIncomingStream: (stream) => {
116
+ onDialerStreamPromise.resolve(stream)
117
+ }
118
+ })
119
+
120
+ const listenerFactory = await common.setup()
121
+ const listener = listenerFactory.createStreamMuxer({
122
+ direction: 'inbound',
123
+ onIncomingStream: (stream) => {
124
+ onListenerStreamPromise.resolve(stream)
125
+ }
126
+ })
127
+
128
+ void pipe(p[0], dialer, p[0])
129
+ void pipe(p[1], listener, p[1])
130
+
131
+ const dialerInitiatorStream = await dialer.newStream()
132
+ const listenerInitiatorStream = await listener.newStream()
133
+
134
+ await Promise.all([
135
+ drainAndClose(dialerInitiatorStream),
136
+ drainAndClose(listenerInitiatorStream),
137
+ onDialerStreamPromise.promise.then(async stream => { await drainAndClose(stream) }),
138
+ onListenerStreamPromise.promise.then(async stream => { await drainAndClose(stream) })
139
+ ])
140
+
141
+ await Promise.all([
142
+ drainAndClose(dialer),
143
+ drainAndClose(listener)
144
+ ])
145
+ })
146
+
147
+ it('Open a stream on one side, write, open a stream on the other side', async () => {
148
+ const toString = (source: Source<Uint8ArrayList>): AsyncGenerator<string> => map(source, (u) => uint8ArrayToString(u.subarray()))
149
+ const p = duplexPair<Uint8Array>()
150
+ const onDialerStreamPromise: DeferredPromise<Stream> = defer()
151
+ const onListenerStreamPromise: DeferredPromise<Stream> = defer()
152
+ const dialerFactory = await common.setup()
153
+ const dialer = dialerFactory.createStreamMuxer({
154
+ direction: 'outbound',
155
+ onIncomingStream: (stream) => {
156
+ onDialerStreamPromise.resolve(stream)
157
+ }
158
+ })
159
+ const listenerFactory = await common.setup()
160
+ const listener = listenerFactory.createStreamMuxer({
161
+ direction: 'inbound',
162
+ onIncomingStream: (stream) => {
163
+ onListenerStreamPromise.resolve(stream)
164
+ }
165
+ })
166
+
167
+ void pipe(p[0], dialer, p[0])
168
+ void pipe(p[1], listener, p[1])
169
+
170
+ const dialerConn = await dialer.newStream()
171
+ const listenerConn = await listener.newStream()
172
+
173
+ void pipe([new Uint8ArrayList(uint8ArrayFromString('hey'))], dialerConn)
174
+ void pipe([new Uint8ArrayList(uint8ArrayFromString('hello'))], listenerConn)
175
+
176
+ const [
177
+ dialerStream,
178
+ listenerStream
179
+ ] = await Promise.all([
180
+ onDialerStreamPromise.promise,
181
+ onListenerStreamPromise.promise
182
+ ])
183
+
184
+ const [
185
+ listenerChunks,
186
+ dialerChunks
187
+ ] = await Promise.all([
188
+ pipe(listenerStream, toString, async (source) => all(source)),
189
+ pipe(dialerStream, toString, async (source) => all(source))
190
+ ])
191
+
192
+ expect(listenerChunks).to.be.eql(['hey'])
193
+ expect(dialerChunks).to.be.eql(['hello'])
194
+ })
195
+ })
196
+ }
@@ -0,0 +1,346 @@
1
+ /* eslint max-nested-callbacks: ["error", 8] */
2
+ import { abortableSource } from 'abortable-iterator'
3
+ import { expect } from 'aegir/chai'
4
+ import delay from 'delay'
5
+ import all from 'it-all'
6
+ import drain from 'it-drain'
7
+ import { duplexPair } from 'it-pair/duplex'
8
+ import { pipe } from 'it-pipe'
9
+ import pDefer from 'p-defer'
10
+ import { Uint8ArrayList } from 'uint8arraylist'
11
+ import { fromString as uint8ArrayFromString } from 'uint8arrays/from-string'
12
+ import type { TestSetup } from '../index.js'
13
+ import type { StreamMuxerFactory } from '@libp2p/interface/stream-muxer'
14
+
15
+ function randomBuffer (): Uint8Array {
16
+ return uint8ArrayFromString(Math.random().toString())
17
+ }
18
+
19
+ const infiniteRandom = {
20
+ [Symbol.asyncIterator]: async function * () {
21
+ while (true) {
22
+ yield new Uint8ArrayList(randomBuffer())
23
+ await delay(50)
24
+ }
25
+ }
26
+ }
27
+
28
+ export default (common: TestSetup<StreamMuxerFactory>): void => {
29
+ describe('close', () => {
30
+ it('closing underlying socket closes streams', async () => {
31
+ let openedStreams = 0
32
+ const expectedStreams = 5
33
+ const dialerFactory = await common.setup()
34
+ const dialer = dialerFactory.createStreamMuxer({ direction: 'outbound' })
35
+
36
+ // Listener is echo server :)
37
+ const listenerFactory = await common.setup()
38
+ const listener = listenerFactory.createStreamMuxer({
39
+ direction: 'inbound',
40
+ onIncomingStream: (stream) => {
41
+ openedStreams++
42
+ void pipe(stream, stream)
43
+ }
44
+ })
45
+
46
+ const p = duplexPair<Uint8Array>()
47
+ void pipe(p[0], dialer, p[0])
48
+ void pipe(p[1], listener, p[1])
49
+
50
+ const streams = await Promise.all(Array(expectedStreams).fill(0).map(async () => dialer.newStream()))
51
+
52
+ void Promise.all(
53
+ streams.map(async stream => {
54
+ await pipe(
55
+ infiniteRandom,
56
+ stream,
57
+ drain
58
+ )
59
+ })
60
+ )
61
+
62
+ expect(dialer.streams).to.have.lengthOf(expectedStreams)
63
+
64
+ // Pause, and then send some data and close the dialer
65
+ await delay(50)
66
+ await pipe([randomBuffer()], dialer, drain)
67
+
68
+ expect(openedStreams).to.have.equal(expectedStreams)
69
+ expect(dialer.streams).to.have.lengthOf(0)
70
+ })
71
+
72
+ it('calling close closes streams', async () => {
73
+ let openedStreams = 0
74
+ const expectedStreams = 5
75
+ const dialerFactory = await common.setup()
76
+ const dialer = dialerFactory.createStreamMuxer({ direction: 'outbound' })
77
+
78
+ // Listener is echo server :)
79
+ const listenerFactory = await common.setup()
80
+ const listener = listenerFactory.createStreamMuxer({
81
+ direction: 'inbound',
82
+ onIncomingStream: (stream) => {
83
+ openedStreams++
84
+ void pipe(stream, stream).catch(() => {})
85
+ }
86
+ })
87
+
88
+ const p = duplexPair<Uint8Array>()
89
+ void pipe(p[0], dialer, p[0])
90
+ void pipe(p[1], listener, p[1])
91
+
92
+ const streams = await Promise.all(Array(expectedStreams).fill(0).map(async () => dialer.newStream()))
93
+
94
+ void Promise.all(
95
+ streams.map(async stream => {
96
+ await pipe(
97
+ infiniteRandom,
98
+ stream,
99
+ drain
100
+ )
101
+ })
102
+ )
103
+
104
+ expect(dialer.streams, 'dialer - number of opened streams should match number of calls to newStream').to.have.lengthOf(expectedStreams)
105
+
106
+ // Pause, and then close the dialer
107
+ await delay(50)
108
+
109
+ dialer.close()
110
+
111
+ expect(openedStreams, 'listener - number of opened streams should match number of calls to newStream').to.have.equal(expectedStreams)
112
+ expect(dialer.streams, 'all tracked streams should be deleted after the muxer has called close').to.have.lengthOf(0)
113
+ })
114
+
115
+ it('calling close with an error aborts streams', async () => {
116
+ let openedStreams = 0
117
+ const expectedStreams = 5
118
+ const dialerFactory = await common.setup()
119
+ const dialer = dialerFactory.createStreamMuxer({ direction: 'outbound' })
120
+
121
+ // Listener is echo server :)
122
+ const listenerFactory = await common.setup()
123
+ const listener = listenerFactory.createStreamMuxer({
124
+ direction: 'inbound',
125
+ onIncomingStream: (stream) => {
126
+ openedStreams++
127
+ void pipe(stream, stream).catch(() => {})
128
+ }
129
+ })
130
+
131
+ const p = duplexPair<Uint8Array>()
132
+ void pipe(p[0], dialer, p[0])
133
+ void pipe(p[1], listener, p[1])
134
+
135
+ const streams = await Promise.all(Array(expectedStreams).fill(0).map(async () => dialer.newStream()))
136
+
137
+ const streamPipes = streams.map(async stream => {
138
+ await pipe(
139
+ infiniteRandom,
140
+ stream,
141
+ drain
142
+ )
143
+ })
144
+
145
+ expect(dialer.streams, 'dialer - number of opened streams should match number of calls to newStream').to.have.lengthOf(expectedStreams)
146
+
147
+ // Pause, and then close the dialer
148
+ await delay(50)
149
+
150
+ // close _with an error_
151
+ dialer.close(new Error())
152
+
153
+ const timeoutError = new Error('timeout')
154
+ for (const pipe of streamPipes) {
155
+ try {
156
+ await Promise.race([
157
+ pipe,
158
+ new Promise((_resolve, reject) => setTimeout(() => { reject(timeoutError) }, 20))
159
+ ])
160
+ expect.fail('stream pipe with infinite source should never return')
161
+ } catch (e) {
162
+ if (e === timeoutError) {
163
+ expect.fail('expected stream pipe to throw an error after muxer closed with error')
164
+ }
165
+ }
166
+ }
167
+
168
+ expect(openedStreams, 'listener - number of opened streams should match number of calls to newStream').to.have.equal(expectedStreams)
169
+ expect(dialer.streams, 'all tracked streams should be deleted after the muxer has called close').to.have.lengthOf(0)
170
+ })
171
+
172
+ it('calling newStream after close throws an error', async () => {
173
+ const dialerFactory = await common.setup()
174
+ const dialer = dialerFactory.createStreamMuxer({ direction: 'outbound' })
175
+
176
+ dialer.close()
177
+
178
+ try {
179
+ await dialer.newStream()
180
+ expect.fail('newStream should throw if called after close')
181
+ } catch (e) {
182
+ expect(dialer.streams, 'closed muxer should have no streams').to.have.lengthOf(0)
183
+ }
184
+ })
185
+
186
+ it('closing one of the muxed streams doesn\'t close others', async () => {
187
+ const p = duplexPair<Uint8Array>()
188
+ const dialerFactory = await common.setup()
189
+ const dialer = dialerFactory.createStreamMuxer({ direction: 'outbound' })
190
+
191
+ // Listener is echo server :)
192
+ const listenerFactory = await common.setup()
193
+ const listener = listenerFactory.createStreamMuxer({
194
+ direction: 'inbound',
195
+ onIncomingStream: (stream) => {
196
+ void pipe(stream, stream).catch(() => {})
197
+ }
198
+ })
199
+
200
+ void pipe(p[0], dialer, p[0])
201
+ void pipe(p[1], listener, p[1])
202
+
203
+ const stream = await dialer.newStream()
204
+ const streams = await Promise.all(Array.from(Array(5), async () => dialer.newStream()))
205
+ let closed = false
206
+ const controllers: AbortController[] = []
207
+
208
+ const streamResults = streams.map(async stream => {
209
+ const controller = new AbortController()
210
+ controllers.push(controller)
211
+
212
+ try {
213
+ const abortableRand = abortableSource(infiniteRandom, controller.signal, { abortCode: 'ERR_TEST_ABORT' })
214
+ await pipe(abortableRand, stream, drain)
215
+ } catch (err: any) {
216
+ if (err.code !== 'ERR_TEST_ABORT') throw err
217
+ }
218
+
219
+ if (!closed) throw new Error('stream should not have ended yet!')
220
+ })
221
+
222
+ // Pause, and then send some data and close the first stream
223
+ await delay(50)
224
+ await pipe([new Uint8ArrayList(randomBuffer())], stream, drain)
225
+ closed = true
226
+
227
+ // Abort all the other streams later
228
+ await delay(50)
229
+ controllers.forEach(c => { c.abort() })
230
+
231
+ // These should now all resolve without error
232
+ await Promise.all(streamResults)
233
+ })
234
+
235
+ it('can close a stream for writing', async () => {
236
+ const deferred = pDefer<Error>()
237
+
238
+ const p = duplexPair<Uint8Array>()
239
+ const dialerFactory = await common.setup()
240
+ const dialer = dialerFactory.createStreamMuxer({ direction: 'outbound' })
241
+ const data = [randomBuffer(), randomBuffer()]
242
+
243
+ const listenerFactory = await common.setup()
244
+ const listener = listenerFactory.createStreamMuxer({
245
+ direction: 'inbound',
246
+ onIncomingStream: (stream) => {
247
+ void Promise.resolve().then(async () => {
248
+ // Immediate close for write
249
+ stream.closeWrite()
250
+
251
+ const results = await pipe(stream, async (source) => {
252
+ const data = []
253
+ for await (const chunk of source) {
254
+ data.push(chunk.slice())
255
+ }
256
+ return data
257
+ })
258
+ expect(results).to.eql(data)
259
+
260
+ try {
261
+ await stream.sink([new Uint8ArrayList(randomBuffer())])
262
+ } catch (err: any) {
263
+ deferred.resolve(err)
264
+ }
265
+
266
+ deferred.reject(new Error('should not support writing to closed writer'))
267
+ })
268
+ }
269
+ })
270
+
271
+ void pipe(p[0], dialer, p[0])
272
+ void pipe(p[1], listener, p[1])
273
+
274
+ const stream = await dialer.newStream()
275
+ await stream.sink(data)
276
+
277
+ const err = await deferred.promise
278
+ expect(err).to.have.property('message').that.matches(/stream closed for writing/)
279
+ })
280
+
281
+ it('can close a stream for reading', async () => {
282
+ const deferred = pDefer<any>()
283
+
284
+ const p = duplexPair<Uint8Array>()
285
+ const dialerFactory = await common.setup()
286
+ const dialer = dialerFactory.createStreamMuxer({ direction: 'outbound' })
287
+ const data = [randomBuffer(), randomBuffer()].map(d => new Uint8ArrayList(d))
288
+
289
+ const listenerFactory = await common.setup()
290
+ const listener = listenerFactory.createStreamMuxer({
291
+ direction: 'inbound',
292
+ onIncomingStream: (stream) => {
293
+ void all(stream.source).then(deferred.resolve, deferred.reject)
294
+ }
295
+ })
296
+
297
+ void pipe(p[0], dialer, p[0])
298
+ void pipe(p[1], listener, p[1])
299
+
300
+ const stream = await dialer.newStream()
301
+ stream.closeRead()
302
+
303
+ // Source should be done
304
+ void Promise.resolve().then(async () => {
305
+ expect(await stream.source.next()).to.have.property('done', true)
306
+ await stream.sink(data)
307
+ })
308
+
309
+ const results = await deferred.promise
310
+ expect(results).to.eql(data)
311
+ })
312
+
313
+ it('calls onStreamEnd for closed streams not previously written', async () => {
314
+ const deferred = pDefer()
315
+
316
+ const onStreamEnd = (): void => { deferred.resolve() }
317
+ const dialerFactory = await common.setup()
318
+ const dialer = dialerFactory.createStreamMuxer({
319
+ direction: 'outbound',
320
+ onStreamEnd
321
+ })
322
+
323
+ const stream = await dialer.newStream()
324
+
325
+ stream.close()
326
+ await deferred.promise
327
+ })
328
+
329
+ it('calls onStreamEnd for read and write closed streams not previously written', async () => {
330
+ const deferred = pDefer()
331
+
332
+ const onStreamEnd = (): void => { deferred.resolve() }
333
+ const dialerFactory = await common.setup()
334
+ const dialer = dialerFactory.createStreamMuxer({
335
+ direction: 'outbound',
336
+ onStreamEnd
337
+ })
338
+
339
+ const stream = await dialer.newStream()
340
+
341
+ stream.closeWrite()
342
+ stream.closeRead()
343
+ await deferred.promise
344
+ })
345
+ })
346
+ }
@@ -0,0 +1,15 @@
1
+ import baseTest from './base-test.js'
2
+ import closeTest from './close-test.js'
3
+ import megaStressTest from './mega-stress-test.js'
4
+ import stressTest from './stress-test.js'
5
+ import type { TestSetup } from '../index.js'
6
+ import type { StreamMuxerFactory } from '@libp2p/interface/stream-muxer'
7
+
8
+ export default (common: TestSetup<StreamMuxerFactory>): void => {
9
+ describe('interface-stream-muxer', () => {
10
+ baseTest(common)
11
+ closeTest(common)
12
+ stressTest(common)
13
+ megaStressTest(common)
14
+ })
15
+ }
@@ -0,0 +1,14 @@
1
+ import spawn from './spawner.js'
2
+ import type { TestSetup } from '../index.js'
3
+ import type { StreamMuxer, StreamMuxerFactory, StreamMuxerInit } from '@libp2p/interface/stream-muxer'
4
+
5
+ export default (common: TestSetup<StreamMuxerFactory>): void => {
6
+ const createMuxer = async (init?: StreamMuxerInit): Promise<StreamMuxer> => {
7
+ const factory = await common.setup()
8
+ return factory.createStreamMuxer(init)
9
+ }
10
+
11
+ describe.skip('mega stress test', function () {
12
+ it('10,000 streams with 10,000 msg', async () => { await spawn(createMuxer, 10000, 10000, 5000) })
13
+ })
14
+ }