@livestore/webmesh 0.0.0-snapshot-1d99fea7d2ce2c7a5d9ed0a3752f8a7bda6bc3db
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/README.md +5 -0
- package/dist/.tsbuildinfo +1 -0
- package/dist/channel/message-channel.d.ts +20 -0
- package/dist/channel/message-channel.d.ts.map +1 -0
- package/dist/channel/message-channel.js +183 -0
- package/dist/channel/message-channel.js.map +1 -0
- package/dist/channel/proxy-channel.d.ts +19 -0
- package/dist/channel/proxy-channel.d.ts.map +1 -0
- package/dist/channel/proxy-channel.js +179 -0
- package/dist/channel/proxy-channel.js.map +1 -0
- package/dist/common.d.ts +83 -0
- package/dist/common.d.ts.map +1 -0
- package/dist/common.js +13 -0
- package/dist/common.js.map +1 -0
- package/dist/mesh-schema.d.ts +104 -0
- package/dist/mesh-schema.d.ts.map +1 -0
- package/dist/mesh-schema.js +77 -0
- package/dist/mesh-schema.js.map +1 -0
- package/dist/mod.d.ts +5 -0
- package/dist/mod.d.ts.map +1 -0
- package/dist/mod.js +5 -0
- package/dist/mod.js.map +1 -0
- package/dist/node.d.ts +65 -0
- package/dist/node.d.ts.map +1 -0
- package/dist/node.js +216 -0
- package/dist/node.js.map +1 -0
- package/dist/node.test.d.ts +2 -0
- package/dist/node.test.d.ts.map +1 -0
- package/dist/node.test.js +351 -0
- package/dist/node.test.js.map +1 -0
- package/dist/utils.d.ts +19 -0
- package/dist/utils.d.ts.map +1 -0
- package/dist/utils.js +41 -0
- package/dist/utils.js.map +1 -0
- package/dist/websocket-connection.d.ts +51 -0
- package/dist/websocket-connection.d.ts.map +1 -0
- package/dist/websocket-connection.js +74 -0
- package/dist/websocket-connection.js.map +1 -0
- package/dist/websocket-server.d.ts +7 -0
- package/dist/websocket-server.d.ts.map +1 -0
- package/dist/websocket-server.js +24 -0
- package/dist/websocket-server.js.map +1 -0
- package/package.json +32 -0
- package/src/channel/message-channel.ts +354 -0
- package/src/channel/proxy-channel.ts +332 -0
- package/src/common.ts +36 -0
- package/src/mesh-schema.ts +94 -0
- package/src/mod.ts +4 -0
- package/src/node.test.ts +533 -0
- package/src/node.ts +408 -0
- package/src/utils.ts +47 -0
- package/src/websocket-connection.ts +158 -0
- package/src/websocket-server.ts +40 -0
- package/tsconfig.json +11 -0
|
@@ -0,0 +1,351 @@
|
|
|
1
|
+
import { IS_CI } from '@livestore/utils';
|
|
2
|
+
import { Chunk, Deferred, Effect, identity, Layer, Logger, Schema, Stream, WebChannel } from '@livestore/utils/effect';
|
|
3
|
+
import { OtelLiveHttp } from '@livestore/utils/node';
|
|
4
|
+
import { Vitest } from '@livestore/utils/node-vitest';
|
|
5
|
+
import { expect } from 'vitest';
|
|
6
|
+
import { Packet } from './mesh-schema.js';
|
|
7
|
+
import { makeMeshNode } from './node.js';
|
|
8
|
+
// TODO test cases where in-between node only comes online later
|
|
9
|
+
// TODO test cases where other side tries to reconnect
|
|
10
|
+
// TODO test combination of connection types (message, proxy)
|
|
11
|
+
// TODO test "diamond shape" topology (A <> B1, A <> B2, B1 <> C, B2 <> C)
|
|
12
|
+
// TODO test cases where multiple entities try to claim to be the same channel end (e.g. A,B,B)
|
|
13
|
+
// TODO write tests with worker threads
|
|
14
|
+
const ExampleSchema = Schema.Struct({ message: Schema.String });
|
|
15
|
+
const connectNodesViaMessageChannel = (nodeA, nodeB) => Effect.gen(function* () {
|
|
16
|
+
const mc = new MessageChannel();
|
|
17
|
+
const meshChannelAToB = yield* WebChannel.messagePortChannel({ port: mc.port1, schema: Packet });
|
|
18
|
+
const meshChannelBToA = yield* WebChannel.messagePortChannel({ port: mc.port2, schema: Packet });
|
|
19
|
+
yield* nodeA.addConnection({ target: nodeB.nodeName, connectionChannel: meshChannelAToB });
|
|
20
|
+
yield* nodeB.addConnection({ target: nodeA.nodeName, connectionChannel: meshChannelBToA });
|
|
21
|
+
return mc;
|
|
22
|
+
}).pipe(Effect.withSpan(`connectNodesViaMessageChannel:${nodeA.nodeName}↔${nodeB.nodeName}`));
|
|
23
|
+
const connectNodesViaBroadcastChannel = (nodeA, nodeB) => Effect.gen(function* () {
|
|
24
|
+
// Need to instantiate two different channels because they filter out messages they sent themselves
|
|
25
|
+
const broadcastWebChannelA = yield* WebChannel.broadcastChannelWithAck({
|
|
26
|
+
channelName: `${nodeA.nodeName}↔${nodeB.nodeName}`,
|
|
27
|
+
listenSchema: Packet,
|
|
28
|
+
sendSchema: Packet,
|
|
29
|
+
});
|
|
30
|
+
const broadcastWebChannelB = yield* WebChannel.broadcastChannelWithAck({
|
|
31
|
+
channelName: `${nodeA.nodeName}↔${nodeB.nodeName}`,
|
|
32
|
+
listenSchema: Packet,
|
|
33
|
+
sendSchema: Packet,
|
|
34
|
+
});
|
|
35
|
+
yield* nodeA.addConnection({ target: nodeB.nodeName, connectionChannel: broadcastWebChannelA });
|
|
36
|
+
yield* nodeB.addConnection({ target: nodeA.nodeName, connectionChannel: broadcastWebChannelB });
|
|
37
|
+
}).pipe(Effect.withSpan(`connectNodesViaBroadcastChannel:${nodeA.nodeName}↔${nodeB.nodeName}`));
|
|
38
|
+
const createChannel = (source, target, options) => source.makeChannel({
|
|
39
|
+
target,
|
|
40
|
+
channelName: options?.channelName ?? 'test',
|
|
41
|
+
schema: ExampleSchema,
|
|
42
|
+
// transferables: options?.transferables ?? 'prefer',
|
|
43
|
+
mode: options?.mode ?? 'messagechannel',
|
|
44
|
+
timeout: options?.timeout ?? 200,
|
|
45
|
+
});
|
|
46
|
+
const getFirstMessage = (channel) => channel.listen.pipe(Stream.flatten(), Stream.take(1), Stream.runCollect, Effect.map(([message]) => message));
|
|
47
|
+
// NOTE we distinguish between undefined and 0 delays as it changes the fiber execution
|
|
48
|
+
const maybeDelay = (delay, label) => (effect) => delay === undefined
|
|
49
|
+
? effect
|
|
50
|
+
: Effect.sleep(delay).pipe(Effect.withSpan(`${label}:delay(${delay})`), Effect.andThen(effect));
|
|
51
|
+
const testTimeout = IS_CI ? 30_000 : 500;
|
|
52
|
+
// TODO also make work without `Vitest.scopedLive` (i.e. with `Vitest.scoped`)
|
|
53
|
+
// probably requires controlling the clocks
|
|
54
|
+
Vitest.describe('webmesh node', { timeout: testTimeout }, () => {
|
|
55
|
+
Vitest.describe('A <> B', () => {
|
|
56
|
+
Vitest.describe('prop tests', () => {
|
|
57
|
+
const Delay = Schema.UndefinedOr(Schema.Literal(0, 1, 10, 50));
|
|
58
|
+
// NOTE for message channels, we test both with and without transferables (i.e. proxying)
|
|
59
|
+
const ChannelType = Schema.Literal('messagechannel', 'messagechannel.proxy', 'proxy');
|
|
60
|
+
const fromChannelType = (channelType) => {
|
|
61
|
+
switch (channelType) {
|
|
62
|
+
case 'proxy': {
|
|
63
|
+
return { mode: 'proxy', connectNodes: connectNodesViaBroadcastChannel };
|
|
64
|
+
}
|
|
65
|
+
case 'messagechannel': {
|
|
66
|
+
return { mode: 'messagechannel', connectNodes: connectNodesViaMessageChannel };
|
|
67
|
+
}
|
|
68
|
+
case 'messagechannel.proxy': {
|
|
69
|
+
return { mode: 'proxy', connectNodes: connectNodesViaMessageChannel };
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
};
|
|
73
|
+
Vitest.scopedLive.prop(
|
|
74
|
+
// Vitest.scopedLive.only(
|
|
75
|
+
'a / b connect at different times with different channel types', [Delay, Delay, Delay, ChannelType], ([delayA, delayB, connectDelay, channelType], test) =>
|
|
76
|
+
// (test) =>
|
|
77
|
+
Effect.gen(function* () {
|
|
78
|
+
// const delayA = 1
|
|
79
|
+
// const delayB = 10
|
|
80
|
+
// const connectDelay = 10
|
|
81
|
+
// const channelType = 'message.prefer'
|
|
82
|
+
// console.log('delayA', delayA, 'delayB', delayB, 'connectDelay', connectDelay, 'channelType', channelType)
|
|
83
|
+
const nodeA = yield* makeMeshNode('A');
|
|
84
|
+
const nodeB = yield* makeMeshNode('B');
|
|
85
|
+
const { mode, connectNodes } = fromChannelType(channelType);
|
|
86
|
+
const nodeACode = Effect.gen(function* () {
|
|
87
|
+
const channelAToB = yield* createChannel(nodeA, 'B', { mode });
|
|
88
|
+
yield* channelAToB.send({ message: 'A1' });
|
|
89
|
+
expect(yield* getFirstMessage(channelAToB)).toEqual({ message: 'A2' });
|
|
90
|
+
});
|
|
91
|
+
const nodeBCode = Effect.gen(function* () {
|
|
92
|
+
const channelBToA = yield* createChannel(nodeB, 'A', { mode });
|
|
93
|
+
yield* channelBToA.send({ message: 'A2' });
|
|
94
|
+
expect(yield* getFirstMessage(channelBToA)).toEqual({ message: 'A1' });
|
|
95
|
+
});
|
|
96
|
+
yield* Effect.all([
|
|
97
|
+
connectNodes(nodeA, nodeB).pipe(maybeDelay(connectDelay, 'connectNodes')),
|
|
98
|
+
nodeACode.pipe(maybeDelay(delayA, 'nodeACode')),
|
|
99
|
+
nodeBCode.pipe(maybeDelay(delayB, 'nodeBCode')),
|
|
100
|
+
], { concurrency: 'unbounded' });
|
|
101
|
+
}).pipe(withCtx(test, { skipOtel: true, suffix: `delayA=${delayA} delayB=${delayB} channelType=${channelType}` })));
|
|
102
|
+
// Vitest.scopedLive.only(
|
|
103
|
+
// 'reconnects',
|
|
104
|
+
// (test) =>
|
|
105
|
+
Vitest.scopedLive.prop('b reconnects', [Delay, Delay, ChannelType], ([waitForOfflineDelay, sleepDelay, channelType], test) => Effect.gen(function* () {
|
|
106
|
+
// const waitForOfflineDelay = 0
|
|
107
|
+
// const sleepDelay = 10
|
|
108
|
+
// const channelType = 'proxy'
|
|
109
|
+
// console.log(
|
|
110
|
+
// 'waitForOfflineDelay',
|
|
111
|
+
// waitForOfflineDelay,
|
|
112
|
+
// 'sleepDelay',
|
|
113
|
+
// sleepDelay,
|
|
114
|
+
// 'channelType',
|
|
115
|
+
// channelType,
|
|
116
|
+
// )
|
|
117
|
+
const nodeA = yield* makeMeshNode('A');
|
|
118
|
+
const nodeB = yield* makeMeshNode('B');
|
|
119
|
+
const { mode, connectNodes } = fromChannelType(channelType);
|
|
120
|
+
// TODO also optionally delay the connection
|
|
121
|
+
yield* connectNodes(nodeA, nodeB);
|
|
122
|
+
const waitForBToBeOffline = waitForOfflineDelay === undefined ? undefined : yield* Deferred.make();
|
|
123
|
+
const nodeACode = Effect.gen(function* () {
|
|
124
|
+
const channelAToB = yield* createChannel(nodeA, 'B', { mode });
|
|
125
|
+
yield* channelAToB.send({ message: 'A1' });
|
|
126
|
+
expect(yield* getFirstMessage(channelAToB)).toEqual({ message: 'B1' });
|
|
127
|
+
if (waitForBToBeOffline !== undefined) {
|
|
128
|
+
yield* waitForBToBeOffline;
|
|
129
|
+
}
|
|
130
|
+
yield* channelAToB.send({ message: 'A2' });
|
|
131
|
+
expect(yield* getFirstMessage(channelAToB)).toEqual({ message: 'B2' });
|
|
132
|
+
});
|
|
133
|
+
// Simulating node b going offline and then coming back online
|
|
134
|
+
const nodeBCode = Effect.gen(function* () {
|
|
135
|
+
yield* Effect.gen(function* () {
|
|
136
|
+
const channelBToA = yield* createChannel(nodeB, 'A', { mode });
|
|
137
|
+
yield* channelBToA.send({ message: 'B1' });
|
|
138
|
+
expect(yield* getFirstMessage(channelBToA)).toEqual({ message: 'A1' });
|
|
139
|
+
}).pipe(Effect.scoped);
|
|
140
|
+
if (waitForBToBeOffline !== undefined) {
|
|
141
|
+
yield* Deferred.succeed(waitForBToBeOffline, void 0);
|
|
142
|
+
}
|
|
143
|
+
if (sleepDelay !== undefined) {
|
|
144
|
+
yield* Effect.sleep(sleepDelay).pipe(Effect.withSpan(`B:sleep(${sleepDelay})`));
|
|
145
|
+
}
|
|
146
|
+
yield* Effect.gen(function* () {
|
|
147
|
+
const channelBToA = yield* createChannel(nodeB, 'A', { mode });
|
|
148
|
+
yield* channelBToA.send({ message: 'B2' });
|
|
149
|
+
expect(yield* getFirstMessage(channelBToA)).toEqual({ message: 'A2' });
|
|
150
|
+
}).pipe(Effect.scoped);
|
|
151
|
+
});
|
|
152
|
+
yield* Effect.all([nodeACode, nodeBCode], { concurrency: 'unbounded' });
|
|
153
|
+
}).pipe(withCtx(test, {
|
|
154
|
+
skipOtel: true,
|
|
155
|
+
suffix: `waitForOfflineDelay=${waitForOfflineDelay} sleepDelay=${sleepDelay} channelType=${channelType}`,
|
|
156
|
+
})));
|
|
157
|
+
const ChannelTypeWithoutMessageChannelProxy = Schema.Literal('proxy', 'messagechannel');
|
|
158
|
+
Vitest.scopedLive.prop('replace connection while keeping the channel', [ChannelTypeWithoutMessageChannelProxy], ([channelType], test) => Effect.gen(function* () {
|
|
159
|
+
const nodeA = yield* makeMeshNode('A');
|
|
160
|
+
const nodeB = yield* makeMeshNode('B');
|
|
161
|
+
const { mode, connectNodes } = fromChannelType(channelType);
|
|
162
|
+
yield* connectNodes(nodeA, nodeB);
|
|
163
|
+
const waitForConnectionReplacement = yield* Deferred.make();
|
|
164
|
+
const nodeACode = Effect.gen(function* () {
|
|
165
|
+
const channelAToB = yield* createChannel(nodeA, 'B', { mode });
|
|
166
|
+
yield* channelAToB.send({ message: 'A1' });
|
|
167
|
+
expect(yield* getFirstMessage(channelAToB)).toEqual({ message: 'B1' });
|
|
168
|
+
yield* waitForConnectionReplacement;
|
|
169
|
+
yield* channelAToB.send({ message: 'A2' });
|
|
170
|
+
expect(yield* getFirstMessage(channelAToB)).toEqual({ message: 'B2' });
|
|
171
|
+
});
|
|
172
|
+
const nodeBCode = Effect.gen(function* () {
|
|
173
|
+
const channelBToA = yield* createChannel(nodeB, 'A', { mode });
|
|
174
|
+
yield* channelBToA.send({ message: 'B1' });
|
|
175
|
+
expect(yield* getFirstMessage(channelBToA)).toEqual({ message: 'A1' });
|
|
176
|
+
// Switch out connection while keeping the channel
|
|
177
|
+
yield* nodeA.removeConnection('B');
|
|
178
|
+
yield* nodeB.removeConnection('A');
|
|
179
|
+
yield* connectNodes(nodeA, nodeB);
|
|
180
|
+
yield* Deferred.succeed(waitForConnectionReplacement, void 0);
|
|
181
|
+
yield* channelBToA.send({ message: 'B2' });
|
|
182
|
+
expect(yield* getFirstMessage(channelBToA)).toEqual({ message: 'A2' });
|
|
183
|
+
});
|
|
184
|
+
yield* Effect.all([nodeACode, nodeBCode], { concurrency: 'unbounded' });
|
|
185
|
+
}).pipe(withCtx(test, { skipOtel: true, suffix: `channelType=${channelType}` })));
|
|
186
|
+
Vitest.describe.todo('TODO improve latency', () => {
|
|
187
|
+
// TODO we need to improve latency when sending messages concurrently
|
|
188
|
+
Vitest.scopedLive.prop('concurrent messages', [ChannelType, Schema.Int.pipe(Schema.between(1, 50))], ([channelType, count], test) => Effect.gen(function* () {
|
|
189
|
+
const nodeA = yield* makeMeshNode('A');
|
|
190
|
+
const nodeB = yield* makeMeshNode('B');
|
|
191
|
+
const { mode, connectNodes } = fromChannelType(channelType);
|
|
192
|
+
console.log('channelType', channelType, 'mode', mode);
|
|
193
|
+
const nodeACode = Effect.gen(function* () {
|
|
194
|
+
const channelAToB = yield* createChannel(nodeA, 'B', { mode });
|
|
195
|
+
// send 10 times A1
|
|
196
|
+
yield* Effect.forEach(Chunk.makeBy(count, (i) => ({ message: `A${i}` })), channelAToB.send, { concurrency: 'unbounded' });
|
|
197
|
+
expect(yield* channelAToB.listen.pipe(Stream.flatten(), Stream.take(count), Stream.runCollect)).toEqual(Chunk.makeBy(count, (i) => ({ message: `B${i}` })));
|
|
198
|
+
// expect(yield* getFirstMessage(channelAToB)).toEqual({ message: 'A2' })
|
|
199
|
+
});
|
|
200
|
+
const nodeBCode = Effect.gen(function* () {
|
|
201
|
+
const channelBToA = yield* createChannel(nodeB, 'A', { mode });
|
|
202
|
+
// send 10 times B1
|
|
203
|
+
yield* Effect.forEach(Chunk.makeBy(count, (i) => ({ message: `B${i}` })), channelBToA.send, { concurrency: 'unbounded' });
|
|
204
|
+
expect(yield* channelBToA.listen.pipe(Stream.flatten(), Stream.take(count), Stream.runCollect)).toEqual(Chunk.makeBy(count, (i) => ({ message: `A${i}` })));
|
|
205
|
+
});
|
|
206
|
+
yield* Effect.all([nodeACode, nodeBCode, connectNodes(nodeA, nodeB).pipe(Effect.delay(100))], {
|
|
207
|
+
concurrency: 'unbounded',
|
|
208
|
+
});
|
|
209
|
+
}).pipe(withCtx(test, { skipOtel: false, suffix: `channelType=${channelType} count=${count}` })));
|
|
210
|
+
});
|
|
211
|
+
});
|
|
212
|
+
Vitest.scopedLive('manual debug test', (test) => Effect.gen(function* () {
|
|
213
|
+
const nodeA = yield* makeMeshNode('A');
|
|
214
|
+
const nodeB = yield* makeMeshNode('B');
|
|
215
|
+
// const connectNodes = connectNodesViaBroadcastChannel
|
|
216
|
+
const connectNodes = connectNodesViaMessageChannel;
|
|
217
|
+
const nodeACode = Effect.gen(function* () {
|
|
218
|
+
const channelAToB = yield* createChannel(nodeA, 'B');
|
|
219
|
+
yield* channelAToB.send({ message: 'A1' });
|
|
220
|
+
expect(yield* getFirstMessage(channelAToB)).toEqual({ message: 'A2' });
|
|
221
|
+
});
|
|
222
|
+
const nodeBCode = Effect.gen(function* () {
|
|
223
|
+
const channelBToA = yield* createChannel(nodeB, 'A');
|
|
224
|
+
yield* channelBToA.send({ message: 'A2' });
|
|
225
|
+
expect(yield* getFirstMessage(channelBToA)).toEqual({ message: 'A1' });
|
|
226
|
+
});
|
|
227
|
+
yield* Effect.all([nodeACode, nodeBCode, connectNodes(nodeA, nodeB).pipe(Effect.delay(100))], {
|
|
228
|
+
concurrency: 'unbounded',
|
|
229
|
+
});
|
|
230
|
+
}).pipe(withCtx(test)));
|
|
231
|
+
Vitest.scopedLive('broadcast connection with message channel', (test) => Effect.gen(function* () {
|
|
232
|
+
const nodeA = yield* makeMeshNode('A');
|
|
233
|
+
const nodeB = yield* makeMeshNode('B');
|
|
234
|
+
yield* connectNodesViaBroadcastChannel(nodeA, nodeB);
|
|
235
|
+
const err = yield* createChannel(nodeA, 'B', { mode: 'messagechannel' }).pipe(Effect.timeout(200), Effect.flip);
|
|
236
|
+
expect(err._tag).toBe('TimeoutException');
|
|
237
|
+
}).pipe(withCtx(test)));
|
|
238
|
+
});
|
|
239
|
+
Vitest.describe('A <> B <> C', () => {
|
|
240
|
+
Vitest.scopedLive('should work', (test) => Effect.gen(function* () {
|
|
241
|
+
const nodeA = yield* makeMeshNode('A');
|
|
242
|
+
const nodeB = yield* makeMeshNode('B');
|
|
243
|
+
const nodeC = yield* makeMeshNode('C');
|
|
244
|
+
yield* connectNodesViaMessageChannel(nodeA, nodeB);
|
|
245
|
+
yield* connectNodesViaMessageChannel(nodeB, nodeC);
|
|
246
|
+
const nodeACode = Effect.gen(function* () {
|
|
247
|
+
const channelAToC = yield* createChannel(nodeA, 'C');
|
|
248
|
+
yield* channelAToC.send({ message: 'A1' });
|
|
249
|
+
expect(yield* getFirstMessage(channelAToC)).toEqual({ message: 'C1' });
|
|
250
|
+
expect(yield* getFirstMessage(channelAToC)).toEqual({ message: 'C2' });
|
|
251
|
+
});
|
|
252
|
+
const nodeCCode = Effect.gen(function* () {
|
|
253
|
+
const channelCToA = yield* createChannel(nodeC, 'A');
|
|
254
|
+
yield* channelCToA.send({ message: 'C1' });
|
|
255
|
+
yield* channelCToA.send({ message: 'C2' });
|
|
256
|
+
yield* channelCToA.send({ message: 'C3' });
|
|
257
|
+
expect(yield* getFirstMessage(channelCToA)).toEqual({ message: 'A1' });
|
|
258
|
+
});
|
|
259
|
+
yield* Effect.all([nodeACode, nodeCCode], { concurrency: 'unbounded' });
|
|
260
|
+
}).pipe(withCtx(test)));
|
|
261
|
+
Vitest.scopedLive('should work - delayed connection', (test) => Effect.gen(function* () {
|
|
262
|
+
const nodeA = yield* makeMeshNode('A');
|
|
263
|
+
const nodeB = yield* makeMeshNode('B');
|
|
264
|
+
const nodeC = yield* makeMeshNode('C');
|
|
265
|
+
const connectNodes = connectNodesViaMessageChannel;
|
|
266
|
+
// const connectNodes = connectNodesViaBroadcastChannel
|
|
267
|
+
yield* connectNodes(nodeA, nodeB);
|
|
268
|
+
// yield* connectNodes(nodeB, nodeC)
|
|
269
|
+
const nodeACode = Effect.gen(function* () {
|
|
270
|
+
const channelAToC = yield* createChannel(nodeA, 'C');
|
|
271
|
+
yield* channelAToC.send({ message: 'A1' });
|
|
272
|
+
expect(yield* getFirstMessage(channelAToC)).toEqual({ message: 'C1' });
|
|
273
|
+
});
|
|
274
|
+
const nodeCCode = Effect.gen(function* () {
|
|
275
|
+
const channelCToA = yield* createChannel(nodeC, 'A');
|
|
276
|
+
yield* channelCToA.send({ message: 'C1' });
|
|
277
|
+
expect(yield* getFirstMessage(channelCToA)).toEqual({ message: 'A1' });
|
|
278
|
+
});
|
|
279
|
+
yield* Effect.all([nodeACode, nodeCCode, connectNodes(nodeB, nodeC).pipe(Effect.delay(100))], {
|
|
280
|
+
concurrency: 'unbounded',
|
|
281
|
+
});
|
|
282
|
+
}).pipe(withCtx(test)));
|
|
283
|
+
Vitest.scopedLive('proxy channel', (test) => Effect.gen(function* () {
|
|
284
|
+
const nodeA = yield* makeMeshNode('A');
|
|
285
|
+
const nodeB = yield* makeMeshNode('B');
|
|
286
|
+
const nodeC = yield* makeMeshNode('C');
|
|
287
|
+
yield* connectNodesViaBroadcastChannel(nodeA, nodeB);
|
|
288
|
+
yield* connectNodesViaBroadcastChannel(nodeB, nodeC);
|
|
289
|
+
const nodeACode = Effect.gen(function* () {
|
|
290
|
+
const channelAToC = yield* createChannel(nodeA, 'C', { mode: 'proxy' });
|
|
291
|
+
yield* channelAToC.send({ message: 'A1' });
|
|
292
|
+
expect(yield* getFirstMessage(channelAToC)).toEqual({ message: 'hello from nodeC' });
|
|
293
|
+
});
|
|
294
|
+
const nodeCCode = Effect.gen(function* () {
|
|
295
|
+
const channelCToA = yield* createChannel(nodeC, 'A', { mode: 'proxy' });
|
|
296
|
+
yield* channelCToA.send({ message: 'hello from nodeC' });
|
|
297
|
+
expect(yield* getFirstMessage(channelCToA)).toEqual({ message: 'A1' });
|
|
298
|
+
});
|
|
299
|
+
yield* Effect.all([nodeACode, nodeCCode], { concurrency: 'unbounded' });
|
|
300
|
+
}).pipe(withCtx(test)));
|
|
301
|
+
Vitest.scopedLive('should fail', (test) => Effect.gen(function* () {
|
|
302
|
+
const nodeA = yield* makeMeshNode('A');
|
|
303
|
+
const nodeB = yield* makeMeshNode('B');
|
|
304
|
+
const nodeC = yield* makeMeshNode('C');
|
|
305
|
+
yield* connectNodesViaMessageChannel(nodeA, nodeB);
|
|
306
|
+
// We're not connecting nodeB and nodeC, so this should fail
|
|
307
|
+
const nodeACode = Effect.gen(function* () {
|
|
308
|
+
const err = yield* createChannel(nodeA, 'C').pipe(Effect.timeout(200), Effect.flip);
|
|
309
|
+
expect(err._tag).toBe('TimeoutException');
|
|
310
|
+
});
|
|
311
|
+
const nodeCCode = Effect.gen(function* () {
|
|
312
|
+
const err = yield* createChannel(nodeC, 'A').pipe(Effect.timeout(200), Effect.flip);
|
|
313
|
+
expect(err._tag).toBe('TimeoutException');
|
|
314
|
+
});
|
|
315
|
+
yield* Effect.all([nodeACode, nodeCCode], { concurrency: 'unbounded' });
|
|
316
|
+
}).pipe(withCtx(test)));
|
|
317
|
+
});
|
|
318
|
+
Vitest.describe('mixture of messagechannel and proxy connections', () => {
|
|
319
|
+
// TODO test case to better guard against case where side A tries to create a proxy channel to B
|
|
320
|
+
// and side B tries to create a messagechannel to A
|
|
321
|
+
Vitest.scopedLive('should work for proxy channels', (test) => Effect.gen(function* () {
|
|
322
|
+
const nodeA = yield* makeMeshNode('A');
|
|
323
|
+
const nodeB = yield* makeMeshNode('B');
|
|
324
|
+
yield* connectNodesViaMessageChannel(nodeB, nodeA);
|
|
325
|
+
const err = yield* connectNodesViaBroadcastChannel(nodeA, nodeB).pipe(Effect.flip);
|
|
326
|
+
expect(err._tag).toBe('ConnectionAlreadyExistsError');
|
|
327
|
+
}).pipe(withCtx(test)));
|
|
328
|
+
// TODO this currently fails but should work. probably needs some more guarding internally.
|
|
329
|
+
Vitest.scopedLive.skip('should work for messagechannels', (test) => Effect.gen(function* () {
|
|
330
|
+
const nodeA = yield* makeMeshNode('A');
|
|
331
|
+
const nodeB = yield* makeMeshNode('B');
|
|
332
|
+
yield* connectNodesViaMessageChannel(nodeB, nodeA);
|
|
333
|
+
yield* connectNodesViaBroadcastChannel(nodeA, nodeB);
|
|
334
|
+
const nodeACode = Effect.gen(function* () {
|
|
335
|
+
const channelAToB = yield* createChannel(nodeA, 'B', { mode: 'messagechannel' });
|
|
336
|
+
yield* channelAToB.send({ message: 'A1' });
|
|
337
|
+
expect(yield* getFirstMessage(channelAToB)).toEqual({ message: 'B1' });
|
|
338
|
+
});
|
|
339
|
+
const nodeBCode = Effect.gen(function* () {
|
|
340
|
+
const channelBToA = yield* createChannel(nodeB, 'A', { mode: 'messagechannel' });
|
|
341
|
+
yield* channelBToA.send({ message: 'B1' });
|
|
342
|
+
expect(yield* getFirstMessage(channelBToA)).toEqual({ message: 'A1' });
|
|
343
|
+
});
|
|
344
|
+
yield* Effect.all([nodeACode, nodeBCode], { concurrency: 'unbounded' });
|
|
345
|
+
}).pipe(withCtx(test)));
|
|
346
|
+
});
|
|
347
|
+
});
|
|
348
|
+
const otelLayer = IS_CI ? Layer.empty : OtelLiveHttp({ serviceName: 'webmesh-node-test', skipLogUrl: false });
|
|
349
|
+
const withCtx = (testContext, { suffix, skipOtel = false } = {}) => (self) => self.pipe(Effect.timeout(testTimeout), Effect.provide(Logger.pretty), Effect.scoped, // We need to scope the effect manually here because otherwise the span is not closed
|
|
350
|
+
Effect.withSpan(`${testContext.task.suite?.name}:${testContext.task.name}${suffix ? `:${suffix}` : ''}`), skipOtel ? identity : Effect.provide(otelLayer));
|
|
351
|
+
//# sourceMappingURL=node.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"node.test.js","sourceRoot":"","sources":["../src/node.test.ts"],"names":[],"mappings":"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"}
|
package/dist/utils.d.ts
ADDED
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import { Duration } from '@livestore/utils/effect';
|
|
2
|
+
/**
|
|
3
|
+
* A set of values that expire after a given timeout
|
|
4
|
+
* The timeout cleanup is performed in a batched way to avoid excessive setTimeout calls
|
|
5
|
+
*/
|
|
6
|
+
export declare class TimeoutSet<V> {
|
|
7
|
+
private values;
|
|
8
|
+
private timeoutHandle;
|
|
9
|
+
private readonly timeoutMs;
|
|
10
|
+
constructor({ timeout }: {
|
|
11
|
+
timeout: Duration.DurationInput;
|
|
12
|
+
});
|
|
13
|
+
add(value: V): void;
|
|
14
|
+
has(value: V): boolean;
|
|
15
|
+
delete(value: V): void;
|
|
16
|
+
private scheduleCleanup;
|
|
17
|
+
private cleanup;
|
|
18
|
+
}
|
|
19
|
+
//# sourceMappingURL=utils.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["../src/utils.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,QAAQ,EAAE,MAAM,yBAAyB,CAAA;AAElD;;;GAGG;AACH,qBAAa,UAAU,CAAC,CAAC;IACvB,OAAO,CAAC,MAAM,CAAuB;IACrC,OAAO,CAAC,aAAa,CAA4B;IACjD,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAQ;gBAEtB,EAAE,OAAO,EAAE,EAAE;QAAE,OAAO,EAAE,QAAQ,CAAC,aAAa,CAAA;KAAE;IAI5D,GAAG,CAAC,KAAK,EAAE,CAAC,GAAG,IAAI;IAKnB,GAAG,CAAC,KAAK,EAAE,CAAC,GAAG,OAAO;IAItB,MAAM,CAAC,KAAK,EAAE,CAAC,GAAG,IAAI;IAItB,OAAO,CAAC,eAAe;IASvB,OAAO,CAAC,OAAO;CAQhB"}
|
package/dist/utils.js
ADDED
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
/* eslint-disable prefer-arrow/prefer-arrow-functions */
|
|
2
|
+
import { Duration } from '@livestore/utils/effect';
|
|
3
|
+
/**
|
|
4
|
+
* A set of values that expire after a given timeout
|
|
5
|
+
* The timeout cleanup is performed in a batched way to avoid excessive setTimeout calls
|
|
6
|
+
*/
|
|
7
|
+
export class TimeoutSet {
|
|
8
|
+
values = new Map();
|
|
9
|
+
timeoutHandle;
|
|
10
|
+
timeoutMs;
|
|
11
|
+
constructor({ timeout }) {
|
|
12
|
+
this.timeoutMs = Duration.toMillis(timeout);
|
|
13
|
+
}
|
|
14
|
+
add(value) {
|
|
15
|
+
this.values.set(value, Date.now());
|
|
16
|
+
this.scheduleCleanup();
|
|
17
|
+
}
|
|
18
|
+
has(value) {
|
|
19
|
+
return this.values.has(value);
|
|
20
|
+
}
|
|
21
|
+
delete(value) {
|
|
22
|
+
this.values.delete(value);
|
|
23
|
+
}
|
|
24
|
+
scheduleCleanup() {
|
|
25
|
+
if (this.timeoutHandle === undefined) {
|
|
26
|
+
this.timeoutHandle = setTimeout(() => {
|
|
27
|
+
this.cleanup();
|
|
28
|
+
this.timeoutHandle = undefined;
|
|
29
|
+
}, this.timeoutMs);
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
cleanup() {
|
|
33
|
+
const now = Date.now();
|
|
34
|
+
for (const [value, timestamp] of this.values.entries()) {
|
|
35
|
+
if (now - timestamp >= this.timeoutMs) {
|
|
36
|
+
this.values.delete(value);
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
//# sourceMappingURL=utils.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"utils.js","sourceRoot":"","sources":["../src/utils.ts"],"names":[],"mappings":"AAAA,wDAAwD;AACxD,OAAO,EAAE,QAAQ,EAAE,MAAM,yBAAyB,CAAA;AAElD;;;GAGG;AACH,MAAM,OAAO,UAAU;IACb,MAAM,GAAG,IAAI,GAAG,EAAa,CAAA;IAC7B,aAAa,CAA4B;IAChC,SAAS,CAAQ;IAElC,YAAY,EAAE,OAAO,EAAuC;QAC1D,IAAI,CAAC,SAAS,GAAG,QAAQ,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAA;IAC7C,CAAC;IAED,GAAG,CAAC,KAAQ;QACV,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,EAAE,IAAI,CAAC,GAAG,EAAE,CAAC,CAAA;QAClC,IAAI,CAAC,eAAe,EAAE,CAAA;IACxB,CAAC;IAED,GAAG,CAAC,KAAQ;QACV,OAAO,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,CAAA;IAC/B,CAAC;IAED,MAAM,CAAC,KAAQ;QACb,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAA;IAC3B,CAAC;IAEO,eAAe;QACrB,IAAI,IAAI,CAAC,aAAa,KAAK,SAAS,EAAE,CAAC;YACrC,IAAI,CAAC,aAAa,GAAG,UAAU,CAAC,GAAG,EAAE;gBACnC,IAAI,CAAC,OAAO,EAAE,CAAA;gBACd,IAAI,CAAC,aAAa,GAAG,SAAS,CAAA;YAChC,CAAC,EAAE,IAAI,CAAC,SAAS,CAAC,CAAA;QACpB,CAAC;IACH,CAAC;IAEO,OAAO;QACb,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAA;QACtB,KAAK,MAAM,CAAC,KAAK,EAAE,SAAS,CAAC,IAAI,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,EAAE,CAAC;YACvD,IAAI,GAAG,GAAG,SAAS,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;gBACtC,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAA;YAC3B,CAAC;QACH,CAAC;IACH,CAAC;CACF"}
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
import type { Scope } from '@livestore/utils/effect';
|
|
2
|
+
import { Effect, Schedule, Schema, WebChannel } from '@livestore/utils/effect';
|
|
3
|
+
import type NodeWebSocket from 'ws';
|
|
4
|
+
import * as MeshSchema from './mesh-schema.js';
|
|
5
|
+
import type { MeshNode } from './node.js';
|
|
6
|
+
declare const WSConnectionInit_base: Schema.TaggedStruct<"WSConnectionInit", {
|
|
7
|
+
from: typeof Schema.String;
|
|
8
|
+
}>;
|
|
9
|
+
export declare class WSConnectionInit extends WSConnectionInit_base {
|
|
10
|
+
}
|
|
11
|
+
declare const WSConnectionPayload_base: Schema.TaggedStruct<"WSConnectionPayload", {
|
|
12
|
+
from: typeof Schema.String;
|
|
13
|
+
payload: typeof Schema.Any;
|
|
14
|
+
}>;
|
|
15
|
+
export declare class WSConnectionPayload extends WSConnectionPayload_base {
|
|
16
|
+
}
|
|
17
|
+
declare const WSConnectionMessage_base: Schema.Union<[typeof WSConnectionInit, typeof WSConnectionPayload]>;
|
|
18
|
+
export declare class WSConnectionMessage extends WSConnectionMessage_base {
|
|
19
|
+
}
|
|
20
|
+
export declare const MessageMsgPack: Schema.transform<Schema.Schema<Uint8Array<ArrayBufferLike>, Uint8Array<ArrayBufferLike>, never>, Schema.Schema<{
|
|
21
|
+
readonly _tag: "WSConnectionInit";
|
|
22
|
+
readonly from: string;
|
|
23
|
+
} | {
|
|
24
|
+
readonly _tag: "WSConnectionPayload";
|
|
25
|
+
readonly payload: any;
|
|
26
|
+
readonly from: string;
|
|
27
|
+
}, {
|
|
28
|
+
readonly _tag: "WSConnectionInit";
|
|
29
|
+
readonly from: string;
|
|
30
|
+
} | {
|
|
31
|
+
readonly _tag: "WSConnectionPayload";
|
|
32
|
+
readonly payload: any;
|
|
33
|
+
readonly from: string;
|
|
34
|
+
}, never>>;
|
|
35
|
+
export type SocketType = {
|
|
36
|
+
_tag: 'leaf';
|
|
37
|
+
from: string;
|
|
38
|
+
} | {
|
|
39
|
+
_tag: 'relay';
|
|
40
|
+
};
|
|
41
|
+
export declare const connectViaWebSocket: ({ node, url, reconnect, }: {
|
|
42
|
+
node: MeshNode;
|
|
43
|
+
url: string;
|
|
44
|
+
reconnect?: Schedule.Schedule<unknown> | false;
|
|
45
|
+
}) => Effect.Effect<void, never, Scope.Scope>;
|
|
46
|
+
export declare const makeWebSocketConnection: (socket: globalThis.WebSocket | NodeWebSocket.WebSocket, socketType: SocketType) => Effect.Effect<{
|
|
47
|
+
webChannel: WebChannel.WebChannel<typeof MeshSchema.Packet.Type, typeof MeshSchema.Packet.Type>;
|
|
48
|
+
from: string;
|
|
49
|
+
}, never, Scope.Scope>;
|
|
50
|
+
export {};
|
|
51
|
+
//# sourceMappingURL=websocket-connection.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"websocket-connection.d.ts","sourceRoot":"","sources":["../src/websocket-connection.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,KAAK,EAAE,MAAM,yBAAyB,CAAA;AACpD,OAAO,EAEL,MAAM,EAIN,QAAQ,EACR,MAAM,EAEN,UAAU,EAEX,MAAM,yBAAyB,CAAA;AAChC,OAAO,KAAK,aAAa,MAAM,IAAI,CAAA;AAEnC,OAAO,KAAK,UAAU,MAAM,kBAAkB,CAAA;AAC9C,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,WAAW,CAAA;;;;AAEzC,qBAAa,gBAAiB,SAAQ,qBAEpC;CAAG;;;;;AAEL,qBAAa,mBAAoB,SAAQ,wBAGvC;CAAG;;AAEL,qBAAa,mBAAoB,SAAQ,wBAAmD;CAAG;AAE/F,eAAO,MAAM,cAAc;;;;;;;;;;;;;;UAAsC,CAAA;AAEjE,MAAM,MAAM,UAAU,GAClB;IACE,IAAI,EAAE,MAAM,CAAA;IACZ,IAAI,EAAE,MAAM,CAAA;CACb,GACD;IACE,IAAI,EAAE,OAAO,CAAA;CACd,CAAA;AAEL,eAAO,MAAM,mBAAmB,8BAI7B;IACD,IAAI,EAAE,QAAQ,CAAA;IACd,GAAG,EAAE,MAAM,CAAA;IACX,SAAS,CAAC,EAAE,QAAQ,CAAC,QAAQ,CAAC,OAAO,CAAC,GAAG,KAAK,CAAA;CAC/C,KAAG,MAAM,CAAC,MAAM,CAAC,IAAI,EAAE,KAAK,EAAE,KAAK,CAAC,KAAK,CAkBtC,CAAA;AAEJ,eAAO,MAAM,uBAAuB,WAC1B,UAAU,CAAC,SAAS,GAAG,aAAa,CAAC,SAAS,cAC1C,UAAU,KACrB,MAAM,CAAC,MAAM,CACd;IACE,UAAU,EAAE,UAAU,CAAC,UAAU,CAAC,OAAO,UAAU,CAAC,MAAM,CAAC,IAAI,EAAE,OAAO,UAAU,CAAC,MAAM,CAAC,IAAI,CAAC,CAAA;IAC/F,IAAI,EAAE,MAAM,CAAA;CACb,EACD,KAAK,EACL,KAAK,CAAC,KAAK,CAgFT,CAAA"}
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
import { Deferred, Effect, Either, FiberHandle, Queue, Schedule, Schema, Stream, WebChannel, WebSocket, } from '@livestore/utils/effect';
|
|
2
|
+
import * as MeshSchema from './mesh-schema.js';
|
|
3
|
+
export class WSConnectionInit extends Schema.TaggedStruct('WSConnectionInit', {
|
|
4
|
+
from: Schema.String,
|
|
5
|
+
}) {
|
|
6
|
+
}
|
|
7
|
+
export class WSConnectionPayload extends Schema.TaggedStruct('WSConnectionPayload', {
|
|
8
|
+
from: Schema.String,
|
|
9
|
+
payload: Schema.Any,
|
|
10
|
+
}) {
|
|
11
|
+
}
|
|
12
|
+
export class WSConnectionMessage extends Schema.Union(WSConnectionInit, WSConnectionPayload) {
|
|
13
|
+
}
|
|
14
|
+
export const MessageMsgPack = Schema.MsgPack(WSConnectionMessage);
|
|
15
|
+
export const connectViaWebSocket = ({ node, url, reconnect = Schedule.exponential(100), }) => Effect.gen(function* () {
|
|
16
|
+
const fiberHandle = yield* FiberHandle.make();
|
|
17
|
+
const connect = Effect.gen(function* () {
|
|
18
|
+
const socket = yield* WebSocket.makeWebSocket({ url, reconnect });
|
|
19
|
+
// NOTE we want to use `runFork` here so this Effect is not part of the fiber that will be interrupted
|
|
20
|
+
socket.addEventListener('close', () => FiberHandle.run(fiberHandle, connect).pipe(Effect.runFork));
|
|
21
|
+
const connection = yield* makeWebSocketConnection(socket, { _tag: 'leaf', from: node.nodeName });
|
|
22
|
+
yield* node.addConnection({ target: 'ws', connectionChannel: connection.webChannel, replaceIfExists: true });
|
|
23
|
+
yield* Effect.never;
|
|
24
|
+
}).pipe(Effect.scoped, Effect.withSpan('@livestore/webmesh:websocket-connection:connect'));
|
|
25
|
+
yield* FiberHandle.run(fiberHandle, connect);
|
|
26
|
+
});
|
|
27
|
+
export const makeWebSocketConnection = (socket, socketType) => Effect.gen(function* () {
|
|
28
|
+
socket.binaryType = 'arraybuffer';
|
|
29
|
+
const fromDeferred = yield* Deferred.make();
|
|
30
|
+
yield* Stream.fromEventListener(socket, 'message').pipe(Stream.map((msg) => Schema.decodeUnknownEither(MessageMsgPack)(new Uint8Array(msg.data))), Stream.flatten(), Stream.tap((msg) => Effect.gen(function* () {
|
|
31
|
+
if (msg._tag === 'WSConnectionInit') {
|
|
32
|
+
yield* Deferred.succeed(fromDeferred, msg.from);
|
|
33
|
+
}
|
|
34
|
+
else {
|
|
35
|
+
const decodedPayload = yield* Schema.decode(MeshSchema.Packet)(msg.payload);
|
|
36
|
+
yield* Queue.offer(listenQueue, decodedPayload);
|
|
37
|
+
}
|
|
38
|
+
})), Stream.runDrain, Effect.tapCauseLogPretty, Effect.forkScoped);
|
|
39
|
+
const listenQueue = yield* Queue.unbounded().pipe(Effect.acquireRelease(Queue.shutdown));
|
|
40
|
+
const initHandshake = (from) => socket.send(Schema.encodeSync(MessageMsgPack)({ _tag: 'WSConnectionInit', from }));
|
|
41
|
+
if (socketType._tag === 'leaf') {
|
|
42
|
+
initHandshake(socketType.from);
|
|
43
|
+
}
|
|
44
|
+
const deferredResult = yield* fromDeferred;
|
|
45
|
+
const from = socketType._tag === 'leaf' ? socketType.from : deferredResult;
|
|
46
|
+
if (socketType._tag === 'relay') {
|
|
47
|
+
initHandshake(from);
|
|
48
|
+
}
|
|
49
|
+
const isConnectedLatch = yield* Effect.makeLatch(true);
|
|
50
|
+
const closedDeferred = yield* Deferred.make();
|
|
51
|
+
yield* Effect.eventListener(socket, 'close', () => Effect.gen(function* () {
|
|
52
|
+
yield* isConnectedLatch.close;
|
|
53
|
+
yield* Deferred.succeed(closedDeferred, undefined);
|
|
54
|
+
}), { once: true });
|
|
55
|
+
const send = (message) => Effect.gen(function* () {
|
|
56
|
+
yield* isConnectedLatch.await;
|
|
57
|
+
const payload = yield* Schema.encode(MeshSchema.Packet)(message);
|
|
58
|
+
socket.send(Schema.encodeSync(MessageMsgPack)({ _tag: 'WSConnectionPayload', payload, from }));
|
|
59
|
+
});
|
|
60
|
+
const listen = Stream.fromQueue(listenQueue).pipe(Stream.map(Either.right));
|
|
61
|
+
const webChannel = {
|
|
62
|
+
[WebChannel.WebChannelSymbol]: WebChannel.WebChannelSymbol,
|
|
63
|
+
send,
|
|
64
|
+
listen,
|
|
65
|
+
closedDeferred,
|
|
66
|
+
schema: { listen: MeshSchema.Packet, send: MeshSchema.Packet },
|
|
67
|
+
supportsTransferables: false,
|
|
68
|
+
};
|
|
69
|
+
return {
|
|
70
|
+
webChannel: webChannel,
|
|
71
|
+
from,
|
|
72
|
+
};
|
|
73
|
+
});
|
|
74
|
+
//# sourceMappingURL=websocket-connection.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"websocket-connection.js","sourceRoot":"","sources":["../src/websocket-connection.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import type { Scope } from '@livestore/utils/effect';
|
|
2
|
+
import { Effect } from '@livestore/utils/effect';
|
|
3
|
+
import * as WebSocket from 'ws';
|
|
4
|
+
export declare const makeWebSocketServer: ({ relayNodeName, }: {
|
|
5
|
+
relayNodeName: string;
|
|
6
|
+
}) => Effect.Effect<WebSocket.WebSocketServer, never, Scope.Scope>;
|
|
7
|
+
//# sourceMappingURL=websocket-server.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"websocket-server.d.ts","sourceRoot":"","sources":["../src/websocket-server.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,KAAK,EAAE,MAAM,yBAAyB,CAAA;AACpD,OAAO,EAAE,MAAM,EAAE,MAAM,yBAAyB,CAAA;AAChD,OAAO,KAAK,SAAS,MAAM,IAAI,CAAA;AAK/B,eAAO,MAAM,mBAAmB,uBAE7B;IACD,aAAa,EAAE,MAAM,CAAA;CACtB,KAAG,MAAM,CAAC,MAAM,CAAC,SAAS,CAAC,eAAe,EAAE,KAAK,EAAE,KAAK,CAAC,KAAK,CA4B3D,CAAA"}
|