lastriko 0.1.0
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/CHANGELOG.md +7 -0
- package/README.md +58 -0
- package/dist/__tests__/integration/ws-flow.integration.test.d.ts +1 -0
- package/dist/__tests__/integration/ws-flow.integration.test.js +249 -0
- package/dist/__tests__/integration/ws-flow.integration.test.js.map +1 -0
- package/dist/__tests__/integration/ws-flow.test.d.ts +1 -0
- package/dist/__tests__/integration/ws-flow.test.js +70 -0
- package/dist/__tests__/integration/ws-flow.test.js.map +1 -0
- package/dist/client/events.d.ts +6 -0
- package/dist/client/events.js +114 -0
- package/dist/client/events.js.map +1 -0
- package/dist/client/index.d.ts +2 -0
- package/dist/client/index.js +5 -0
- package/dist/client/index.js.map +1 -0
- package/dist/client/swap.d.ts +4 -0
- package/dist/client/swap.js +47 -0
- package/dist/client/swap.js.map +1 -0
- package/dist/client/ws.d.ts +11 -0
- package/dist/client/ws.js +111 -0
- package/dist/client/ws.js.map +1 -0
- package/dist/components/context.d.ts +51 -0
- package/dist/components/context.js +595 -0
- package/dist/components/context.js.map +1 -0
- package/dist/components/context.test.d.ts +1 -0
- package/dist/components/context.test.js +69 -0
- package/dist/components/context.test.js.map +1 -0
- package/dist/components/id.d.ts +3 -0
- package/dist/components/id.js +9 -0
- package/dist/components/id.js.map +1 -0
- package/dist/components/registry.d.ts +2 -0
- package/dist/components/registry.js +97 -0
- package/dist/components/registry.js.map +1 -0
- package/dist/components/types.d.ts +390 -0
- package/dist/components/types.js +2 -0
- package/dist/components/types.js.map +1 -0
- package/dist/engine/executor.d.ts +20 -0
- package/dist/engine/executor.js +84 -0
- package/dist/engine/executor.js.map +1 -0
- package/dist/engine/executor.test.d.ts +1 -0
- package/dist/engine/executor.test.js +79 -0
- package/dist/engine/executor.test.js.map +1 -0
- package/dist/engine/index.d.ts +6 -0
- package/dist/engine/index.js +7 -0
- package/dist/engine/index.js.map +1 -0
- package/dist/engine/lifecycle.d.ts +16 -0
- package/dist/engine/lifecycle.js +44 -0
- package/dist/engine/lifecycle.js.map +1 -0
- package/dist/engine/lifecycle.test.d.ts +1 -0
- package/dist/engine/lifecycle.test.js +15 -0
- package/dist/engine/lifecycle.test.js.map +1 -0
- package/dist/engine/messages.d.ts +95 -0
- package/dist/engine/messages.js +21 -0
- package/dist/engine/messages.js.map +1 -0
- package/dist/engine/renderer.components.test.d.ts +1 -0
- package/dist/engine/renderer.components.test.js +302 -0
- package/dist/engine/renderer.components.test.js.map +1 -0
- package/dist/engine/renderer.d.ts +4 -0
- package/dist/engine/renderer.js +408 -0
- package/dist/engine/renderer.js.map +1 -0
- package/dist/engine/renderer.test.d.ts +1 -0
- package/dist/engine/renderer.test.js +80 -0
- package/dist/engine/renderer.test.js.map +1 -0
- package/dist/engine/server.d.ts +43 -0
- package/dist/engine/server.js +402 -0
- package/dist/engine/server.js.map +1 -0
- package/dist/engine/server.test.d.ts +1 -0
- package/dist/engine/server.test.js +149 -0
- package/dist/engine/server.test.js.map +1 -0
- package/dist/engine/shell.d.ts +7 -0
- package/dist/engine/shell.js +25 -0
- package/dist/engine/shell.js.map +1 -0
- package/dist/engine/theme-path.d.ts +11 -0
- package/dist/engine/theme-path.js +32 -0
- package/dist/engine/theme-path.js.map +1 -0
- package/dist/engine/theme-path.test.d.ts +1 -0
- package/dist/engine/theme-path.test.js +52 -0
- package/dist/engine/theme-path.test.js.map +1 -0
- package/dist/engine/watcher.d.ts +8 -0
- package/dist/engine/watcher.js +27 -0
- package/dist/engine/watcher.js.map +1 -0
- package/dist/engine/websocket.d.ts +24 -0
- package/dist/engine/websocket.hub.test.d.ts +1 -0
- package/dist/engine/websocket.hub.test.js +75 -0
- package/dist/engine/websocket.hub.test.js.map +1 -0
- package/dist/engine/websocket.js +119 -0
- package/dist/engine/websocket.js.map +1 -0
- package/dist/index.d.ts +21 -0
- package/dist/index.js +26 -0
- package/dist/index.js.map +1 -0
- package/dist/plugins/registry.d.ts +11 -0
- package/dist/plugins/registry.js +61 -0
- package/dist/plugins/registry.js.map +1 -0
- package/dist/plugins/registry.test.d.ts +1 -0
- package/dist/plugins/registry.test.js +44 -0
- package/dist/plugins/registry.test.js.map +1 -0
- package/dist/plugins/types.d.ts +30 -0
- package/dist/plugins/types.js +2 -0
- package/dist/plugins/types.js.map +1 -0
- package/dist/style.css +134 -0
- package/dist/theme/lastriko.css +134 -0
- package/package.json +60 -0
package/CHANGELOG.md
ADDED
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
# Changelog
|
|
2
|
+
|
|
3
|
+
All notable changes to the **`lastriko`** npm package are documented here. The monorepo roadmap and design history live in the repository root [MANIFEST.md](https://github.com/aixpose/lastriko/blob/main/MANIFEST.md).
|
|
4
|
+
|
|
5
|
+
## 0.1.0 — 2026-04-08
|
|
6
|
+
|
|
7
|
+
Initial published MVP (Phase 2 scope): `app()`, WebSocket + HTML fragments, MVP component set, `lastriko/style.css`, Node.js 22+.
|
package/README.md
ADDED
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
# lastriko
|
|
2
|
+
|
|
3
|
+
Server-driven UI for AI demos and rapid prototypes: your TypeScript `app()` builds handles, the engine renders HTML fragments, and the browser swaps `outerHTML` over a WebSocket.
|
|
4
|
+
|
|
5
|
+
## Install
|
|
6
|
+
|
|
7
|
+
```bash
|
|
8
|
+
npm install lastriko
|
|
9
|
+
```
|
|
10
|
+
|
|
11
|
+
Requires **Node.js 22+**. [Bun](https://bun.sh) is also supported.
|
|
12
|
+
|
|
13
|
+
## Quick start
|
|
14
|
+
|
|
15
|
+
```typescript
|
|
16
|
+
import { app } from 'lastriko';
|
|
17
|
+
|
|
18
|
+
await app('Hello', (ui) => {
|
|
19
|
+
const name = ui.textInput('Name');
|
|
20
|
+
const out = ui.text('Hi!');
|
|
21
|
+
ui.button('Greet', () => {
|
|
22
|
+
out.update(`Hello, ${name.value || 'stranger'}!`);
|
|
23
|
+
});
|
|
24
|
+
});
|
|
25
|
+
```
|
|
26
|
+
|
|
27
|
+
Run with `tsx` or `bun`; open the URL printed in the console (default port **3500**).
|
|
28
|
+
|
|
29
|
+
## Exports
|
|
30
|
+
|
|
31
|
+
- **JS API** — `import { app, defineConfig } from 'lastriko'`
|
|
32
|
+
- **Styles** — `import 'lastriko/style.css'` (or load `/style.css` from the dev server)
|
|
33
|
+
|
|
34
|
+
## Docs
|
|
35
|
+
|
|
36
|
+
- [Project manifest & roadmap](https://github.com/aixpose/lastriko/blob/main/MANIFEST.md)
|
|
37
|
+
- [API reference](https://github.com/aixpose/lastriko/blob/main/docs/API-REFERENCE.md)
|
|
38
|
+
- [Phase 2 (MVP) scope](https://github.com/aixpose/lastriko/blob/main/docs/phases/PHASE-2.md)
|
|
39
|
+
|
|
40
|
+
## Examples in this repo
|
|
41
|
+
|
|
42
|
+
- `examples/component-gallery` — all MVP components on one page
|
|
43
|
+
- `examples/experiment-monitor`, `examples/image-viewer` — larger demos
|
|
44
|
+
|
|
45
|
+
## Publishing (maintainers)
|
|
46
|
+
|
|
47
|
+
From the monorepo root, after tests pass:
|
|
48
|
+
|
|
49
|
+
```bash
|
|
50
|
+
npm run build -w lastriko
|
|
51
|
+
npm publish -w lastriko --access public
|
|
52
|
+
```
|
|
53
|
+
|
|
54
|
+
`prepack` runs `build` automatically on `npm publish`.
|
|
55
|
+
|
|
56
|
+
## License
|
|
57
|
+
|
|
58
|
+
MIT — see [LICENSE](https://github.com/aixpose/lastriko/blob/main/LICENSE) in the repository root.
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,249 @@
|
|
|
1
|
+
import { describe, expect, it } from 'vitest';
|
|
2
|
+
import { createWebSocketHub } from '../../engine/websocket';
|
|
3
|
+
function createSocket() {
|
|
4
|
+
const sent = [];
|
|
5
|
+
return {
|
|
6
|
+
socket: {
|
|
7
|
+
send(data) {
|
|
8
|
+
sent.push(data);
|
|
9
|
+
},
|
|
10
|
+
},
|
|
11
|
+
sent,
|
|
12
|
+
};
|
|
13
|
+
}
|
|
14
|
+
function byType(sent, type) {
|
|
15
|
+
return sent
|
|
16
|
+
.map((item) => JSON.parse(item))
|
|
17
|
+
.filter((msg) => msg.type === type);
|
|
18
|
+
}
|
|
19
|
+
describe('websocket integration flow', () => {
|
|
20
|
+
it('ready emits RENDER and button click emits FRAGMENT', () => {
|
|
21
|
+
const { socket, sent } = createSocket();
|
|
22
|
+
const runtime = createWebSocketHub({
|
|
23
|
+
title: 'Phase2',
|
|
24
|
+
callback: (ui) => {
|
|
25
|
+
const text = ui.text('before');
|
|
26
|
+
ui.button('go', () => {
|
|
27
|
+
text.update('after');
|
|
28
|
+
});
|
|
29
|
+
},
|
|
30
|
+
});
|
|
31
|
+
runtime.addConnection(socket);
|
|
32
|
+
runtime.handleRawMessage(socket, JSON.stringify({
|
|
33
|
+
type: 'READY',
|
|
34
|
+
payload: { viewport: { width: 100, height: 100 }, theme: 'dark' },
|
|
35
|
+
}));
|
|
36
|
+
const render = byType(sent, 'RENDER')[0];
|
|
37
|
+
expect(render).toBeDefined();
|
|
38
|
+
expect(render?.payload.html).toContain('before');
|
|
39
|
+
const buttonId = render?.payload.html.match(/data-lk-id="(button-[^"]+)"/)?.[1];
|
|
40
|
+
expect(buttonId).toBeDefined();
|
|
41
|
+
runtime.handleRawMessage(socket, JSON.stringify({
|
|
42
|
+
type: 'EVENT',
|
|
43
|
+
payload: { id: buttonId, event: 'click' },
|
|
44
|
+
}));
|
|
45
|
+
const fragment = byType(sent, 'FRAGMENT').at(-1);
|
|
46
|
+
expect(fragment?.payload.html).toContain('after');
|
|
47
|
+
});
|
|
48
|
+
it('change EVENT updates input value and pushes FRAGMENT', () => {
|
|
49
|
+
const { socket, sent } = createSocket();
|
|
50
|
+
const runtime = createWebSocketHub({
|
|
51
|
+
title: 'Inputs',
|
|
52
|
+
callback: (ui) => {
|
|
53
|
+
ui.textInput('Prompt', { default: 'a' });
|
|
54
|
+
},
|
|
55
|
+
});
|
|
56
|
+
runtime.addConnection(socket);
|
|
57
|
+
runtime.handleRawMessage(socket, JSON.stringify({
|
|
58
|
+
type: 'READY',
|
|
59
|
+
payload: { viewport: { width: 100, height: 100 }, theme: null },
|
|
60
|
+
}));
|
|
61
|
+
const render = byType(sent, 'RENDER')[0];
|
|
62
|
+
const inputId = render.payload.html.match(/data-lk-id="(textInput-[^"]+)"/)?.[1];
|
|
63
|
+
expect(inputId).toBeDefined();
|
|
64
|
+
runtime.handleRawMessage(socket, JSON.stringify({
|
|
65
|
+
type: 'EVENT',
|
|
66
|
+
payload: { id: inputId, event: 'change', value: 'updated' },
|
|
67
|
+
}));
|
|
68
|
+
const fragment = byType(sent, 'FRAGMENT').at(-1);
|
|
69
|
+
expect(fragment?.payload.id).toBe(inputId);
|
|
70
|
+
expect(fragment?.payload.html).toContain('updated');
|
|
71
|
+
});
|
|
72
|
+
it('streamText append emits STREAM_CHUNK messages in order', () => {
|
|
73
|
+
const { socket, sent } = createSocket();
|
|
74
|
+
const runtime = createWebSocketHub({
|
|
75
|
+
title: 'Streaming',
|
|
76
|
+
callback: (ui) => {
|
|
77
|
+
const stream = ui.streamText({ format: 'plain' });
|
|
78
|
+
ui.button('go', () => {
|
|
79
|
+
stream.append('a');
|
|
80
|
+
stream.append('b');
|
|
81
|
+
stream.done();
|
|
82
|
+
});
|
|
83
|
+
},
|
|
84
|
+
});
|
|
85
|
+
runtime.addConnection(socket);
|
|
86
|
+
runtime.handleRawMessage(socket, JSON.stringify({
|
|
87
|
+
type: 'READY',
|
|
88
|
+
payload: { viewport: { width: 100, height: 100 }, theme: null },
|
|
89
|
+
}));
|
|
90
|
+
const render = byType(sent, 'RENDER')[0];
|
|
91
|
+
const buttonId = render.payload.html.match(/data-lk-id="(button-[^"]+)"/)?.[1];
|
|
92
|
+
runtime.handleRawMessage(socket, JSON.stringify({
|
|
93
|
+
type: 'EVENT',
|
|
94
|
+
payload: { id: buttonId, event: 'click' },
|
|
95
|
+
}));
|
|
96
|
+
const chunks = byType(sent, 'STREAM_CHUNK');
|
|
97
|
+
expect(chunks).toHaveLength(3);
|
|
98
|
+
expect(chunks[0]?.payload.chunk).toBe('a');
|
|
99
|
+
expect(chunks[1]?.payload.chunk).toBe('b');
|
|
100
|
+
expect(chunks[2]?.payload.done).toBe(true);
|
|
101
|
+
});
|
|
102
|
+
it('table row click EVENT dispatches onRowClick handler', () => {
|
|
103
|
+
const { socket, sent } = createSocket();
|
|
104
|
+
const clicks = [];
|
|
105
|
+
const runtime = createWebSocketHub({
|
|
106
|
+
title: 'Table',
|
|
107
|
+
callback: (ui) => {
|
|
108
|
+
const table = ui.table([{ name: 'row-1' }], { columns: ['name'] });
|
|
109
|
+
table.onRowClick((row) => {
|
|
110
|
+
clicks.push(String(row.name));
|
|
111
|
+
});
|
|
112
|
+
},
|
|
113
|
+
});
|
|
114
|
+
runtime.addConnection(socket);
|
|
115
|
+
runtime.handleRawMessage(socket, JSON.stringify({
|
|
116
|
+
type: 'READY',
|
|
117
|
+
payload: { viewport: { width: 120, height: 80 }, theme: null },
|
|
118
|
+
}));
|
|
119
|
+
const render = byType(sent, 'RENDER')[0];
|
|
120
|
+
const tableId = render.payload.html.match(/data-lk-id="(table-[^"]+)"/)?.[1];
|
|
121
|
+
const rowId = render.payload.html.match(/data-lk-table-row-id="([^"]+)"/)?.[1];
|
|
122
|
+
expect(tableId).toBeDefined();
|
|
123
|
+
expect(rowId).toBeDefined();
|
|
124
|
+
runtime.handleRawMessage(socket, JSON.stringify({
|
|
125
|
+
type: 'EVENT',
|
|
126
|
+
payload: { id: tableId, event: 'click', value: rowId },
|
|
127
|
+
}));
|
|
128
|
+
expect(clicks).toEqual(['row-1']);
|
|
129
|
+
});
|
|
130
|
+
it('tracks READY viewport/theme and handles THEME_CHANGE', () => {
|
|
131
|
+
const { socket, sent } = createSocket();
|
|
132
|
+
const runtime = createWebSocketHub({
|
|
133
|
+
title: 'Scope',
|
|
134
|
+
callback: (ui) => {
|
|
135
|
+
ui.text(`viewport-${ui.viewport.width}x${ui.viewport.height}-${ui.theme}`);
|
|
136
|
+
},
|
|
137
|
+
});
|
|
138
|
+
const scope = runtime.addConnection(socket);
|
|
139
|
+
runtime.handleRawMessage(socket, JSON.stringify({
|
|
140
|
+
type: 'READY',
|
|
141
|
+
payload: { viewport: { width: 10, height: 20 }, theme: 'dark' },
|
|
142
|
+
}));
|
|
143
|
+
expect(scope.viewport).toEqual({ width: 10, height: 20 });
|
|
144
|
+
expect(scope.theme).toBe('dark');
|
|
145
|
+
runtime.handleRawMessage(socket, JSON.stringify({
|
|
146
|
+
type: 'THEME_CHANGE',
|
|
147
|
+
payload: { mode: 'light' },
|
|
148
|
+
}));
|
|
149
|
+
const themeMsg = byType(sent, 'THEME').at(-1);
|
|
150
|
+
expect(themeMsg?.payload.mode).toBe('light');
|
|
151
|
+
});
|
|
152
|
+
it('numberInput and toggle change EVENT update values and emit FRAGMENT', () => {
|
|
153
|
+
const { socket, sent } = createSocket();
|
|
154
|
+
const runtime = createWebSocketHub({
|
|
155
|
+
title: 'More inputs',
|
|
156
|
+
callback: (ui) => {
|
|
157
|
+
ui.numberInput('N', { default: 1 });
|
|
158
|
+
ui.toggle('T', { default: false });
|
|
159
|
+
},
|
|
160
|
+
});
|
|
161
|
+
runtime.addConnection(socket);
|
|
162
|
+
runtime.handleRawMessage(socket, JSON.stringify({
|
|
163
|
+
type: 'READY',
|
|
164
|
+
payload: { viewport: { width: 100, height: 100 }, theme: null },
|
|
165
|
+
}));
|
|
166
|
+
const render = byType(sent, 'RENDER')[0];
|
|
167
|
+
const numId = render.payload.html.match(/data-lk-id="(numberInput-[^"]+)"/)?.[1];
|
|
168
|
+
const toggleId = render.payload.html.match(/data-lk-id="(toggle-[^"]+)"/)?.[1];
|
|
169
|
+
expect(numId).toBeDefined();
|
|
170
|
+
expect(toggleId).toBeDefined();
|
|
171
|
+
runtime.handleRawMessage(socket, JSON.stringify({
|
|
172
|
+
type: 'EVENT',
|
|
173
|
+
payload: { id: numId, event: 'change', value: 42 },
|
|
174
|
+
}));
|
|
175
|
+
const fragNum = byType(sent, 'FRAGMENT').at(-1);
|
|
176
|
+
expect(fragNum?.payload.id).toBe(numId);
|
|
177
|
+
expect(fragNum?.payload.html).toContain('value="42"');
|
|
178
|
+
runtime.handleRawMessage(socket, JSON.stringify({
|
|
179
|
+
type: 'EVENT',
|
|
180
|
+
payload: { id: toggleId, event: 'change', value: true },
|
|
181
|
+
}));
|
|
182
|
+
const fragToggle = byType(sent, 'FRAGMENT').at(-1);
|
|
183
|
+
expect(fragToggle?.payload.id).toBe(toggleId);
|
|
184
|
+
expect(fragToggle?.payload.html).toContain('checked');
|
|
185
|
+
});
|
|
186
|
+
it('cross-region: sidebar button updates header metric via shared scope', () => {
|
|
187
|
+
const { socket, sent } = createSocket();
|
|
188
|
+
const runtime = createWebSocketHub({
|
|
189
|
+
title: 'Shell',
|
|
190
|
+
callback: (ui) => {
|
|
191
|
+
let metric;
|
|
192
|
+
ui.shell({
|
|
193
|
+
header: (h) => {
|
|
194
|
+
metric = h.metric('Count', '0');
|
|
195
|
+
},
|
|
196
|
+
main: () => { },
|
|
197
|
+
sidebar: (s) => {
|
|
198
|
+
s.button('inc', () => {
|
|
199
|
+
metric?.update('1');
|
|
200
|
+
});
|
|
201
|
+
},
|
|
202
|
+
});
|
|
203
|
+
},
|
|
204
|
+
});
|
|
205
|
+
runtime.addConnection(socket);
|
|
206
|
+
runtime.handleRawMessage(socket, JSON.stringify({
|
|
207
|
+
type: 'READY',
|
|
208
|
+
payload: { viewport: { width: 200, height: 200 }, theme: null },
|
|
209
|
+
}));
|
|
210
|
+
const render = byType(sent, 'RENDER')[0];
|
|
211
|
+
const buttonId = render.payload.html.match(/data-lk-id="(button-[^"]+)"/)?.[1];
|
|
212
|
+
expect(buttonId).toBeDefined();
|
|
213
|
+
expect(render.payload.html).toContain('lk-metric-value');
|
|
214
|
+
runtime.handleRawMessage(socket, JSON.stringify({
|
|
215
|
+
type: 'EVENT',
|
|
216
|
+
payload: { id: buttonId, event: 'click' },
|
|
217
|
+
}));
|
|
218
|
+
const lastMetricFrag = [...byType(sent, 'FRAGMENT')]
|
|
219
|
+
.filter((f) => f.payload.html.includes('lk-metric'))
|
|
220
|
+
.at(-1);
|
|
221
|
+
expect(lastMetricFrag?.payload.html).toContain('1');
|
|
222
|
+
});
|
|
223
|
+
it('keeps scopes isolated across connections', () => {
|
|
224
|
+
const a = createSocket();
|
|
225
|
+
const b = createSocket();
|
|
226
|
+
const runtime = createWebSocketHub({
|
|
227
|
+
title: 'Isolation',
|
|
228
|
+
callback: (ui) => {
|
|
229
|
+
ui.text(`scope-${ui.scope.id}`);
|
|
230
|
+
},
|
|
231
|
+
});
|
|
232
|
+
const scopeA = runtime.addConnection(a.socket);
|
|
233
|
+
const scopeB = runtime.addConnection(b.socket);
|
|
234
|
+
runtime.handleRawMessage(a.socket, JSON.stringify({
|
|
235
|
+
type: 'READY',
|
|
236
|
+
payload: { viewport: { width: 10, height: 10 }, theme: null },
|
|
237
|
+
}));
|
|
238
|
+
runtime.handleRawMessage(b.socket, JSON.stringify({
|
|
239
|
+
type: 'READY',
|
|
240
|
+
payload: { viewport: { width: 20, height: 20 }, theme: null },
|
|
241
|
+
}));
|
|
242
|
+
const aRender = byType(a.sent, 'RENDER')[0];
|
|
243
|
+
const bRender = byType(b.sent, 'RENDER')[0];
|
|
244
|
+
expect(aRender.payload.html).toContain(`scope-${scopeA.id}`);
|
|
245
|
+
expect(bRender.payload.html).toContain(`scope-${scopeB.id}`);
|
|
246
|
+
expect(scopeA.id).not.toBe(scopeB.id);
|
|
247
|
+
});
|
|
248
|
+
});
|
|
249
|
+
//# sourceMappingURL=ws-flow.integration.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ws-flow.integration.test.js","sourceRoot":"","sources":["../../../src/__tests__/integration/ws-flow.integration.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
import { describe, expect, it } from 'vitest';
|
|
2
|
+
import { createWebSocketHub } from '../../engine/websocket';
|
|
3
|
+
function createSocket() {
|
|
4
|
+
const sent = [];
|
|
5
|
+
return {
|
|
6
|
+
socket: {
|
|
7
|
+
send(data) {
|
|
8
|
+
sent.push(data);
|
|
9
|
+
},
|
|
10
|
+
},
|
|
11
|
+
sent,
|
|
12
|
+
};
|
|
13
|
+
}
|
|
14
|
+
describe('websocket integration flow', () => {
|
|
15
|
+
it('rEADY emits RENDER and click EVENT emits FRAGMENT', async () => {
|
|
16
|
+
const { socket, sent } = createSocket();
|
|
17
|
+
const runtime = createWebSocketHub({
|
|
18
|
+
title: 'Phase1',
|
|
19
|
+
callback: (ui) => {
|
|
20
|
+
const text = ui.text('before');
|
|
21
|
+
ui.button('go', () => {
|
|
22
|
+
text.update('after');
|
|
23
|
+
});
|
|
24
|
+
},
|
|
25
|
+
});
|
|
26
|
+
runtime.addConnection(socket);
|
|
27
|
+
runtime.handleRawMessage(socket, JSON.stringify({
|
|
28
|
+
type: 'READY',
|
|
29
|
+
payload: { viewport: { width: 100, height: 100 }, theme: null },
|
|
30
|
+
}));
|
|
31
|
+
const renderRaw = sent.find((item) => JSON.parse(item).type === 'RENDER');
|
|
32
|
+
expect(renderRaw).toBeDefined();
|
|
33
|
+
const render = JSON.parse(renderRaw ?? '{}');
|
|
34
|
+
expect(render.payload.html).toContain('before');
|
|
35
|
+
const buttonId = render.payload.html.match(/data-lk-id="(button-[^"]+)"/)?.[1];
|
|
36
|
+
runtime.handleRawMessage(socket, JSON.stringify({
|
|
37
|
+
type: 'EVENT',
|
|
38
|
+
payload: { id: buttonId, event: 'click' },
|
|
39
|
+
}));
|
|
40
|
+
const fragmentRaw = sent.find((item) => JSON.parse(item).type === 'FRAGMENT');
|
|
41
|
+
expect(fragmentRaw).toBeDefined();
|
|
42
|
+
const fragment = JSON.parse(fragmentRaw ?? '{}');
|
|
43
|
+
expect(fragment.payload.html).toContain('after');
|
|
44
|
+
});
|
|
45
|
+
it('two connections render independently', () => {
|
|
46
|
+
const a = createSocket();
|
|
47
|
+
const b = createSocket();
|
|
48
|
+
const runtime = createWebSocketHub({
|
|
49
|
+
title: 'Scope',
|
|
50
|
+
callback: (ui) => {
|
|
51
|
+
ui.text(`scope-${ui.scope.id}`);
|
|
52
|
+
},
|
|
53
|
+
});
|
|
54
|
+
const scopeA = runtime.addConnection(a.socket);
|
|
55
|
+
const scopeB = runtime.addConnection(b.socket);
|
|
56
|
+
runtime.handleRawMessage(a.socket, JSON.stringify({
|
|
57
|
+
type: 'READY',
|
|
58
|
+
payload: { viewport: { width: 10, height: 10 }, theme: null },
|
|
59
|
+
}));
|
|
60
|
+
runtime.handleRawMessage(b.socket, JSON.stringify({
|
|
61
|
+
type: 'READY',
|
|
62
|
+
payload: { viewport: { width: 10, height: 10 }, theme: null },
|
|
63
|
+
}));
|
|
64
|
+
const aRender = JSON.parse(a.sent.find((item) => JSON.parse(item).type === 'RENDER') ?? '{}');
|
|
65
|
+
const bRender = JSON.parse(b.sent.find((item) => JSON.parse(item).type === 'RENDER') ?? '{}');
|
|
66
|
+
expect(aRender.payload.html).toContain(`scope-${scopeA.id}`);
|
|
67
|
+
expect(bRender.payload.html).toContain(`scope-${scopeB.id}`);
|
|
68
|
+
});
|
|
69
|
+
});
|
|
70
|
+
//# sourceMappingURL=ws-flow.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ws-flow.test.js","sourceRoot":"","sources":["../../../src/__tests__/integration/ws-flow.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,QAAQ,CAAC;AAC9C,OAAO,EAAE,kBAAkB,EAAE,MAAM,wBAAwB,CAAC;AAE5D,SAAS,YAAY;IACnB,MAAM,IAAI,GAAa,EAAE,CAAC;IAC1B,OAAO;QACL,MAAM,EAAE;YACN,IAAI,CAAC,IAAY;gBACf,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAClB,CAAC;SACF;QACD,IAAI;KACL,CAAC;AACJ,CAAC;AAED,QAAQ,CAAC,4BAA4B,EAAE,GAAG,EAAE;IAC1C,EAAE,CAAC,mDAAmD,EAAE,KAAK,IAAI,EAAE;QACjE,MAAM,EAAE,MAAM,EAAE,IAAI,EAAE,GAAG,YAAY,EAAE,CAAC;QACxC,MAAM,OAAO,GAAG,kBAAkB,CAAC;YACjC,KAAK,EAAE,QAAQ;YACf,QAAQ,EAAE,CAAC,EAAE,EAAE,EAAE;gBACf,MAAM,IAAI,GAAG,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;gBAC/B,EAAE,CAAC,MAAM,CAAC,IAAI,EAAE,GAAG,EAAE;oBACnB,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;gBACvB,CAAC,CAAC,CAAC;YACL,CAAC;SACF,CAAC,CAAC;QAEH,OAAO,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC;QAC9B,OAAO,CAAC,gBAAgB,CACtB,MAAM,EACN,IAAI,CAAC,SAAS,CAAC;YACb,IAAI,EAAE,OAAO;YACb,OAAO,EAAE,EAAE,QAAQ,EAAE,EAAE,KAAK,EAAE,GAAG,EAAE,MAAM,EAAE,GAAG,EAAE,EAAE,KAAK,EAAE,IAAI,EAAE;SAChE,CAAC,CACH,CAAC;QAEF,MAAM,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,IAAI,KAAK,QAAQ,CAAC,CAAC;QAC1E,MAAM,CAAC,SAAS,CAAC,CAAC,WAAW,EAAE,CAAC;QAChC,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,IAAI,IAAI,CAAC,CAAC;QAC7C,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;QAEhD,MAAM,QAAQ,GAAG,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,6BAA6B,CAAC,EAAE,CAAC,CAAC,CAAW,CAAC;QACzF,OAAO,CAAC,gBAAgB,CACtB,MAAM,EACN,IAAI,CAAC,SAAS,CAAC;YACb,IAAI,EAAE,OAAO;YACb,OAAO,EAAE,EAAE,EAAE,EAAE,QAAQ,EAAE,KAAK,EAAE,OAAO,EAAE;SAC1C,CAAC,CACH,CAAC;QAEF,MAAM,WAAW,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,IAAI,KAAK,UAAU,CAAC,CAAC;QAC9E,MAAM,CAAC,WAAW,CAAC,CAAC,WAAW,EAAE,CAAC;QAClC,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,IAAI,IAAI,CAAC,CAAC;QACjD,MAAM,CAAC,QAAQ,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC;IACnD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,sCAAsC,EAAE,GAAG,EAAE;QAC9C,MAAM,CAAC,GAAG,YAAY,EAAE,CAAC;QACzB,MAAM,CAAC,GAAG,YAAY,EAAE,CAAC;QACzB,MAAM,OAAO,GAAG,kBAAkB,CAAC;YACjC,KAAK,EAAE,OAAO;YACd,QAAQ,EAAE,CAAC,EAAE,EAAE,EAAE;gBACf,EAAE,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC,KAAK,CAAC,EAAE,EAAE,CAAC,CAAC;YAClC,CAAC;SACF,CAAC,CAAC;QAEH,MAAM,MAAM,GAAG,OAAO,CAAC,aAAa,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC;QAC/C,MAAM,MAAM,GAAG,OAAO,CAAC,aAAa,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC;QAE/C,OAAO,CAAC,gBAAgB,CACtB,CAAC,CAAC,MAAM,EACR,IAAI,CAAC,SAAS,CAAC;YACb,IAAI,EAAE,OAAO;YACb,OAAO,EAAE,EAAE,QAAQ,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,EAAE,EAAE,KAAK,EAAE,IAAI,EAAE;SAC9D,CAAC,CACH,CAAC;QACF,OAAO,CAAC,gBAAgB,CACtB,CAAC,CAAC,MAAM,EACR,IAAI,CAAC,SAAS,CAAC;YACb,IAAI,EAAE,OAAO;YACb,OAAO,EAAE,EAAE,QAAQ,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,EAAE,EAAE,KAAK,EAAE,IAAI,EAAE;SAC9D,CAAC,CACH,CAAC;QAEF,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,IAAI,KAAK,QAAQ,CAAC,IAAI,IAAI,CAAC,CAAC;QAC9F,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,IAAI,KAAK,QAAQ,CAAC,IAAI,IAAI,CAAC,CAAC;QAC9F,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,SAAS,CAAC,SAAS,MAAM,CAAC,EAAE,EAAE,CAAC,CAAC;QAC7D,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,SAAS,CAAC,SAAS,MAAM,CAAC,EAAE,EAAE,CAAC,CAAC;IAC/D,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
import type { ClientToServerMessage } from '../engine/messages';
|
|
2
|
+
export interface EventChannel {
|
|
3
|
+
send: (message: ClientToServerMessage) => void;
|
|
4
|
+
}
|
|
5
|
+
export declare function bindEventDelegation(root: Document, channel: EventChannel): void;
|
|
6
|
+
export declare function bindThemeToggle(root: Document, channel: EventChannel): void;
|
|
@@ -0,0 +1,114 @@
|
|
|
1
|
+
function coerceValue(target) {
|
|
2
|
+
if (!(target instanceof HTMLElement))
|
|
3
|
+
return undefined;
|
|
4
|
+
if (target instanceof HTMLInputElement) {
|
|
5
|
+
if (target.type === 'checkbox')
|
|
6
|
+
return target.checked;
|
|
7
|
+
if (target.type === 'number' || target.type === 'range')
|
|
8
|
+
return Number(target.value);
|
|
9
|
+
return target.value;
|
|
10
|
+
}
|
|
11
|
+
if (target instanceof HTMLSelectElement || target instanceof HTMLTextAreaElement) {
|
|
12
|
+
return target.value;
|
|
13
|
+
}
|
|
14
|
+
return undefined;
|
|
15
|
+
}
|
|
16
|
+
function resolveComponentId(target) {
|
|
17
|
+
if (!(target instanceof Element))
|
|
18
|
+
return null;
|
|
19
|
+
const holder = target.closest('[data-lk-id]');
|
|
20
|
+
return holder?.dataset.lkId ?? null;
|
|
21
|
+
}
|
|
22
|
+
export function bindEventDelegation(root, channel) {
|
|
23
|
+
root.addEventListener('click', (event) => {
|
|
24
|
+
if (!(event.target instanceof Element))
|
|
25
|
+
return;
|
|
26
|
+
const clickable = event.target.closest('[data-lk-event="click"]');
|
|
27
|
+
if (!clickable)
|
|
28
|
+
return;
|
|
29
|
+
const id = resolveComponentId(clickable);
|
|
30
|
+
if (!id)
|
|
31
|
+
return;
|
|
32
|
+
channel.send({ type: 'EVENT', payload: { id, event: 'click' } });
|
|
33
|
+
});
|
|
34
|
+
root.addEventListener('click', (event) => {
|
|
35
|
+
if (!(event.target instanceof Element))
|
|
36
|
+
return;
|
|
37
|
+
const tableRow = event.target.closest('[data-lk-table-row-id]');
|
|
38
|
+
if (!tableRow)
|
|
39
|
+
return;
|
|
40
|
+
const table = tableRow.closest('[data-lk-id][data-lk-kind="table"]');
|
|
41
|
+
const id = table?.dataset.lkId;
|
|
42
|
+
const rowId = tableRow.dataset.lkTableRowId;
|
|
43
|
+
if (!id || !rowId)
|
|
44
|
+
return;
|
|
45
|
+
channel.send({ type: 'EVENT', payload: { id, event: 'click', value: rowId } });
|
|
46
|
+
});
|
|
47
|
+
root.addEventListener('change', (event) => {
|
|
48
|
+
if (!(event.target instanceof Element))
|
|
49
|
+
return;
|
|
50
|
+
const changeable = event.target.closest('[data-lk-event="change"]');
|
|
51
|
+
if (!changeable)
|
|
52
|
+
return;
|
|
53
|
+
const id = resolveComponentId(changeable);
|
|
54
|
+
if (!id)
|
|
55
|
+
return;
|
|
56
|
+
channel.send({
|
|
57
|
+
type: 'EVENT',
|
|
58
|
+
payload: { id, event: 'change', value: coerceValue(event.target) },
|
|
59
|
+
});
|
|
60
|
+
});
|
|
61
|
+
root.addEventListener('change', async (event) => {
|
|
62
|
+
if (!(event.target instanceof HTMLInputElement))
|
|
63
|
+
return;
|
|
64
|
+
if (event.target.type !== 'file')
|
|
65
|
+
return;
|
|
66
|
+
const holder = event.target.closest('[data-lk-id][data-lk-kind="fileUpload"]');
|
|
67
|
+
const id = holder?.dataset.lkId;
|
|
68
|
+
if (!id)
|
|
69
|
+
return;
|
|
70
|
+
const files = event.target.files ? Array.from(event.target.files) : [];
|
|
71
|
+
if (files.length === 0) {
|
|
72
|
+
channel.send({ type: 'EVENT', payload: { id, event: 'change', value: null } });
|
|
73
|
+
return;
|
|
74
|
+
}
|
|
75
|
+
const formData = new FormData();
|
|
76
|
+
for (const file of files) {
|
|
77
|
+
formData.append('file', file, file.name);
|
|
78
|
+
}
|
|
79
|
+
try {
|
|
80
|
+
const connectionId = String(window.localStorage.getItem('lk-connection-id') ?? '');
|
|
81
|
+
const search = connectionId ? `?connectionId=${encodeURIComponent(connectionId)}` : '';
|
|
82
|
+
const response = await fetch(`/upload${search}`, { method: 'POST', body: formData });
|
|
83
|
+
if (!response.ok) {
|
|
84
|
+
return;
|
|
85
|
+
}
|
|
86
|
+
const payload = await response.json();
|
|
87
|
+
const uploaded = Array.isArray(payload)
|
|
88
|
+
? payload
|
|
89
|
+
: payload.file;
|
|
90
|
+
channel.send({
|
|
91
|
+
type: 'EVENT',
|
|
92
|
+
payload: {
|
|
93
|
+
id,
|
|
94
|
+
event: 'change',
|
|
95
|
+
value: uploaded ?? null,
|
|
96
|
+
},
|
|
97
|
+
});
|
|
98
|
+
}
|
|
99
|
+
catch {
|
|
100
|
+
// best effort
|
|
101
|
+
}
|
|
102
|
+
});
|
|
103
|
+
}
|
|
104
|
+
export function bindThemeToggle(root, channel) {
|
|
105
|
+
const toggle = root.getElementById('lk-theme-toggle');
|
|
106
|
+
if (!toggle)
|
|
107
|
+
return;
|
|
108
|
+
toggle.addEventListener('click', () => {
|
|
109
|
+
const current = root.documentElement.getAttribute('data-theme') === 'dark' ? 'dark' : 'light';
|
|
110
|
+
const mode = current === 'dark' ? 'light' : 'dark';
|
|
111
|
+
channel.send({ type: 'THEME_CHANGE', payload: { mode } });
|
|
112
|
+
});
|
|
113
|
+
}
|
|
114
|
+
//# sourceMappingURL=events.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"events.js","sourceRoot":"","sources":["../../src/client/events.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/client/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,eAAe,EAAE,MAAM,MAAM,CAAC;AAEvC,MAAM,OAAO,GAAG,eAAe,EAAE,CAAC;AAClC,OAAO,CAAC,OAAO,EAAE,CAAC;AAElB,OAAO,EAAE,eAAe,EAAE,CAAC"}
|
|
@@ -0,0 +1,4 @@
|
|
|
1
|
+
import type { RenderMessage } from '../engine/messages';
|
|
2
|
+
export declare function applyRender(payload: RenderMessage['payload']): void;
|
|
3
|
+
export declare function applyFragmentSwap(id: string, html: string): void;
|
|
4
|
+
export declare function applyStreamChunk(id: string, chunk: string, done: boolean, format: 'plain' | 'markdown'): void;
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
export function applyRender(payload) {
|
|
2
|
+
const root = document.getElementById('lk-root');
|
|
3
|
+
if (!root) {
|
|
4
|
+
return;
|
|
5
|
+
}
|
|
6
|
+
root.innerHTML = payload.html;
|
|
7
|
+
document.title = payload.title;
|
|
8
|
+
document.documentElement.setAttribute('data-theme', payload.theme);
|
|
9
|
+
}
|
|
10
|
+
export function applyFragmentSwap(id, html) {
|
|
11
|
+
const target = document.querySelector(`[data-lk-id="${id}"]`);
|
|
12
|
+
if (!target) {
|
|
13
|
+
return;
|
|
14
|
+
}
|
|
15
|
+
target.outerHTML = html;
|
|
16
|
+
}
|
|
17
|
+
function markdownToHtml(input) {
|
|
18
|
+
const escaped = input
|
|
19
|
+
.replaceAll('&', '&')
|
|
20
|
+
.replaceAll('<', '<')
|
|
21
|
+
.replaceAll('>', '>');
|
|
22
|
+
return escaped.replaceAll('\n', '<br />');
|
|
23
|
+
}
|
|
24
|
+
export function applyStreamChunk(id, chunk, done, format) {
|
|
25
|
+
const container = document.querySelector(`[data-lk-id="${id}"]`);
|
|
26
|
+
if (!container) {
|
|
27
|
+
return;
|
|
28
|
+
}
|
|
29
|
+
const body = container.querySelector('[data-lk-stream-body]');
|
|
30
|
+
if (!body) {
|
|
31
|
+
return;
|
|
32
|
+
}
|
|
33
|
+
const current = body.dataset.lkText ?? '';
|
|
34
|
+
const next = `${current}${chunk}`;
|
|
35
|
+
body.dataset.lkText = next;
|
|
36
|
+
if (format === 'markdown') {
|
|
37
|
+
body.innerHTML = markdownToHtml(next);
|
|
38
|
+
}
|
|
39
|
+
else {
|
|
40
|
+
body.textContent = next;
|
|
41
|
+
}
|
|
42
|
+
const cursor = container.querySelector('[data-lk-stream-cursor]');
|
|
43
|
+
if (cursor) {
|
|
44
|
+
cursor.style.display = done ? 'none' : '';
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
//# sourceMappingURL=swap.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"swap.js","sourceRoot":"","sources":["../../src/client/swap.ts"],"names":[],"mappings":"AAEA,MAAM,UAAU,WAAW,CAAC,OAAiC;IAC3D,MAAM,IAAI,GAAG,QAAQ,CAAC,cAAc,CAAC,SAAS,CAAC,CAAC;IAChD,IAAI,CAAC,IAAI,EAAE,CAAC;QACV,OAAO;IACT,CAAC;IACD,IAAI,CAAC,SAAS,GAAG,OAAO,CAAC,IAAI,CAAC;IAC9B,QAAQ,CAAC,KAAK,GAAG,OAAO,CAAC,KAAK,CAAC;IAC/B,QAAQ,CAAC,eAAe,CAAC,YAAY,CAAC,YAAY,EAAE,OAAO,CAAC,KAAK,CAAC,CAAC;AACrE,CAAC;AAED,MAAM,UAAU,iBAAiB,CAAC,EAAU,EAAE,IAAY;IACxD,MAAM,MAAM,GAAG,QAAQ,CAAC,aAAa,CAAc,gBAAgB,EAAE,IAAI,CAAC,CAAC;IAC3E,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,OAAO;IACT,CAAC;IACD,MAAM,CAAC,SAAS,GAAG,IAAI,CAAC;AAC1B,CAAC;AAED,SAAS,cAAc,CAAC,KAAa;IACnC,MAAM,OAAO,GAAG,KAAK;SAClB,UAAU,CAAC,GAAG,EAAE,OAAO,CAAC;SACxB,UAAU,CAAC,GAAG,EAAE,MAAM,CAAC;SACvB,UAAU,CAAC,GAAG,EAAE,MAAM,CAAC,CAAC;IAC3B,OAAO,OAAO,CAAC,UAAU,CAAC,IAAI,EAAE,QAAQ,CAAC,CAAC;AAC5C,CAAC;AAED,MAAM,UAAU,gBAAgB,CAC9B,EAAU,EACV,KAAa,EACb,IAAa,EACb,MAA4B;IAE5B,MAAM,SAAS,GAAG,QAAQ,CAAC,aAAa,CAAc,gBAAgB,EAAE,IAAI,CAAC,CAAC;IAC9E,IAAI,CAAC,SAAS,EAAE,CAAC;QACf,OAAO;IACT,CAAC;IACD,MAAM,IAAI,GAAG,SAAS,CAAC,aAAa,CAAc,uBAAuB,CAAC,CAAC;IAC3E,IAAI,CAAC,IAAI,EAAE,CAAC;QACV,OAAO;IACT,CAAC;IAED,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,IAAI,EAAE,CAAC;IAC1C,MAAM,IAAI,GAAG,GAAG,OAAO,GAAG,KAAK,EAAE,CAAC;IAClC,IAAI,CAAC,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC;IAC3B,IAAI,MAAM,KAAK,UAAU,EAAE,CAAC;QAC1B,IAAI,CAAC,SAAS,GAAG,cAAc,CAAC,IAAI,CAAC,CAAC;IACxC,CAAC;SAAM,CAAC;QACN,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC;IAC1B,CAAC;IAED,MAAM,MAAM,GAAG,SAAS,CAAC,aAAa,CAAc,yBAAyB,CAAC,CAAC;IAC/E,IAAI,MAAM,EAAE,CAAC;QACX,MAAM,CAAC,KAAK,CAAC,OAAO,GAAG,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC;IAC5C,CAAC;AACH,CAAC"}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
export interface WSManagerOptions {
|
|
2
|
+
wsFactory?: (url: string) => WebSocket;
|
|
3
|
+
url?: string;
|
|
4
|
+
maxRetries?: number;
|
|
5
|
+
initialDelayMs?: number;
|
|
6
|
+
}
|
|
7
|
+
export interface WSManager {
|
|
8
|
+
connect(): void;
|
|
9
|
+
disconnect(): void;
|
|
10
|
+
}
|
|
11
|
+
export declare function createWSManager(opts?: WSManagerOptions): WSManager;
|