@auxiora/bridge 1.0.0 → 1.3.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/package.json +8 -2
- package/src/__tests__/bridge.test.ts +0 -411
- package/src/index.ts +0 -22
- package/src/pairing.ts +0 -129
- package/src/registry.ts +0 -108
- package/src/server.ts +0 -338
- package/src/types.ts +0 -115
- package/tsconfig.json +0 -11
- package/tsconfig.tsbuildinfo +0 -1
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@auxiora/bridge",
|
|
3
|
-
"version": "1.
|
|
3
|
+
"version": "1.3.0",
|
|
4
4
|
"description": "Bridge protocol for device pairing — connects mobile and desktop nodes to the server",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"main": "dist/index.js",
|
|
@@ -12,11 +12,17 @@
|
|
|
12
12
|
}
|
|
13
13
|
},
|
|
14
14
|
"dependencies": {
|
|
15
|
-
"@auxiora/logger": "1.
|
|
15
|
+
"@auxiora/logger": "1.3.0"
|
|
16
16
|
},
|
|
17
17
|
"engines": {
|
|
18
18
|
"node": ">=22.0.0"
|
|
19
19
|
},
|
|
20
|
+
"publishConfig": {
|
|
21
|
+
"access": "public"
|
|
22
|
+
},
|
|
23
|
+
"files": [
|
|
24
|
+
"dist/"
|
|
25
|
+
],
|
|
20
26
|
"scripts": {
|
|
21
27
|
"build": "tsc",
|
|
22
28
|
"clean": "rm -rf dist",
|
|
@@ -1,411 +0,0 @@
|
|
|
1
|
-
import { describe, it, expect, beforeEach, afterEach, vi } from 'vitest';
|
|
2
|
-
import { DeviceRegistry } from '../registry.js';
|
|
3
|
-
import { PairingFlow } from '../pairing.js';
|
|
4
|
-
import { BridgeServer, WS_OPEN, type BridgeSocket } from '../server.js';
|
|
5
|
-
import type { DeviceInfo, BridgeMessage } from '../types.js';
|
|
6
|
-
|
|
7
|
-
function makeMockSocket(): BridgeSocket & { sent: string[]; closed: boolean } {
|
|
8
|
-
return {
|
|
9
|
-
sent: [],
|
|
10
|
-
closed: false,
|
|
11
|
-
readyState: WS_OPEN,
|
|
12
|
-
send(data: string) { this.sent.push(data); },
|
|
13
|
-
close() { this.closed = true; this.readyState = 3; },
|
|
14
|
-
};
|
|
15
|
-
}
|
|
16
|
-
|
|
17
|
-
function parseMessage(raw: string): BridgeMessage {
|
|
18
|
-
return JSON.parse(raw) as BridgeMessage;
|
|
19
|
-
}
|
|
20
|
-
|
|
21
|
-
describe('DeviceRegistry', () => {
|
|
22
|
-
let registry: DeviceRegistry;
|
|
23
|
-
|
|
24
|
-
const device1: DeviceInfo = {
|
|
25
|
-
id: 'dev-1',
|
|
26
|
-
name: 'iPhone',
|
|
27
|
-
platform: 'ios',
|
|
28
|
-
capabilities: ['camera', 'location', 'notifications'],
|
|
29
|
-
state: 'online',
|
|
30
|
-
pairedAt: Date.now(),
|
|
31
|
-
lastSeen: Date.now(),
|
|
32
|
-
};
|
|
33
|
-
|
|
34
|
-
const device2: DeviceInfo = {
|
|
35
|
-
id: 'dev-2',
|
|
36
|
-
name: 'MacBook',
|
|
37
|
-
platform: 'macos',
|
|
38
|
-
capabilities: ['screen', 'clipboard', 'notifications'],
|
|
39
|
-
state: 'online',
|
|
40
|
-
pairedAt: Date.now(),
|
|
41
|
-
lastSeen: Date.now(),
|
|
42
|
-
};
|
|
43
|
-
|
|
44
|
-
beforeEach(() => {
|
|
45
|
-
registry = new DeviceRegistry(5);
|
|
46
|
-
});
|
|
47
|
-
|
|
48
|
-
it('registers and retrieves a device', () => {
|
|
49
|
-
registry.register(device1);
|
|
50
|
-
const found = registry.get('dev-1');
|
|
51
|
-
expect(found).toBeDefined();
|
|
52
|
-
expect(found!.name).toBe('iPhone');
|
|
53
|
-
expect(found!.platform).toBe('ios');
|
|
54
|
-
});
|
|
55
|
-
|
|
56
|
-
it('returns a copy of device info', () => {
|
|
57
|
-
registry.register(device1);
|
|
58
|
-
const found = registry.get('dev-1')!;
|
|
59
|
-
found.name = 'Modified';
|
|
60
|
-
expect(registry.get('dev-1')!.name).toBe('iPhone');
|
|
61
|
-
});
|
|
62
|
-
|
|
63
|
-
it('returns undefined for unknown device', () => {
|
|
64
|
-
expect(registry.get('nonexistent')).toBeUndefined();
|
|
65
|
-
});
|
|
66
|
-
|
|
67
|
-
it('lists all devices', () => {
|
|
68
|
-
registry.register(device1);
|
|
69
|
-
registry.register(device2);
|
|
70
|
-
expect(registry.getAll()).toHaveLength(2);
|
|
71
|
-
});
|
|
72
|
-
|
|
73
|
-
it('filters by capability', () => {
|
|
74
|
-
registry.register(device1);
|
|
75
|
-
registry.register(device2);
|
|
76
|
-
const withCamera = registry.getByCapability('camera');
|
|
77
|
-
expect(withCamera).toHaveLength(1);
|
|
78
|
-
expect(withCamera[0].id).toBe('dev-1');
|
|
79
|
-
});
|
|
80
|
-
|
|
81
|
-
it('filters online devices', () => {
|
|
82
|
-
registry.register(device1);
|
|
83
|
-
registry.register({ ...device2, state: 'offline' });
|
|
84
|
-
const online = registry.getOnline();
|
|
85
|
-
expect(online).toHaveLength(1);
|
|
86
|
-
});
|
|
87
|
-
|
|
88
|
-
it('unregisters a device', () => {
|
|
89
|
-
registry.register(device1);
|
|
90
|
-
expect(registry.unregister('dev-1')).toBe(true);
|
|
91
|
-
expect(registry.get('dev-1')).toBeUndefined();
|
|
92
|
-
expect(registry.size).toBe(0);
|
|
93
|
-
});
|
|
94
|
-
|
|
95
|
-
it('enforces max device limit', () => {
|
|
96
|
-
const small = new DeviceRegistry(1);
|
|
97
|
-
small.register(device1);
|
|
98
|
-
expect(() => small.register(device2)).toThrow('Maximum device limit (1) reached');
|
|
99
|
-
});
|
|
100
|
-
|
|
101
|
-
it('allows re-registering existing device', () => {
|
|
102
|
-
const small = new DeviceRegistry(1);
|
|
103
|
-
small.register(device1);
|
|
104
|
-
small.register({ ...device1, name: 'Updated iPhone' });
|
|
105
|
-
expect(small.get('dev-1')!.name).toBe('Updated iPhone');
|
|
106
|
-
});
|
|
107
|
-
|
|
108
|
-
it('updates connection state', () => {
|
|
109
|
-
registry.register(device1);
|
|
110
|
-
registry.setState('dev-1', 'offline');
|
|
111
|
-
expect(registry.get('dev-1')!.state).toBe('offline');
|
|
112
|
-
});
|
|
113
|
-
|
|
114
|
-
it('updates lastSeen on heartbeat', () => {
|
|
115
|
-
registry.register({ ...device1, lastSeen: 1000 });
|
|
116
|
-
registry.heartbeat('dev-1');
|
|
117
|
-
expect(registry.get('dev-1')!.lastSeen).toBeGreaterThan(1000);
|
|
118
|
-
});
|
|
119
|
-
|
|
120
|
-
it('detects timed-out devices', () => {
|
|
121
|
-
registry.register({ ...device1, lastSeen: Date.now() - 100_000 });
|
|
122
|
-
const timedOut = registry.checkTimeouts(60_000);
|
|
123
|
-
expect(timedOut).toEqual(['dev-1']);
|
|
124
|
-
expect(registry.get('dev-1')!.state).toBe('offline');
|
|
125
|
-
});
|
|
126
|
-
|
|
127
|
-
it('reports capacity', () => {
|
|
128
|
-
const small = new DeviceRegistry(1);
|
|
129
|
-
expect(small.hasCapacity()).toBe(true);
|
|
130
|
-
small.register(device1);
|
|
131
|
-
expect(small.hasCapacity()).toBe(false);
|
|
132
|
-
});
|
|
133
|
-
});
|
|
134
|
-
|
|
135
|
-
describe('PairingFlow', () => {
|
|
136
|
-
let pairing: PairingFlow;
|
|
137
|
-
|
|
138
|
-
beforeEach(() => {
|
|
139
|
-
pairing = new PairingFlow({ codeLength: 6, codeExpirySeconds: 300 });
|
|
140
|
-
});
|
|
141
|
-
|
|
142
|
-
afterEach(() => {
|
|
143
|
-
pairing.destroy();
|
|
144
|
-
});
|
|
145
|
-
|
|
146
|
-
it('generates a code of correct length', () => {
|
|
147
|
-
const code = pairing.generateCode();
|
|
148
|
-
expect(code.code).toHaveLength(6);
|
|
149
|
-
expect(code.used).toBe(false);
|
|
150
|
-
expect(code.expiresAt).toBeGreaterThan(Date.now());
|
|
151
|
-
});
|
|
152
|
-
|
|
153
|
-
it('validates a fresh code', () => {
|
|
154
|
-
const code = pairing.generateCode();
|
|
155
|
-
expect(pairing.validate(code.code)).toBe(true);
|
|
156
|
-
});
|
|
157
|
-
|
|
158
|
-
it('rejects unknown code', () => {
|
|
159
|
-
expect(pairing.validate('000000')).toBe(false);
|
|
160
|
-
});
|
|
161
|
-
|
|
162
|
-
it('consumes a code', () => {
|
|
163
|
-
const code = pairing.generateCode();
|
|
164
|
-
expect(pairing.consume(code.code)).toBe(true);
|
|
165
|
-
// Cannot consume again
|
|
166
|
-
expect(pairing.consume(code.code)).toBe(false);
|
|
167
|
-
// Cannot validate consumed code
|
|
168
|
-
expect(pairing.validate(code.code)).toBe(false);
|
|
169
|
-
});
|
|
170
|
-
|
|
171
|
-
it('revokes a code', () => {
|
|
172
|
-
const code = pairing.generateCode();
|
|
173
|
-
expect(pairing.revoke(code.code)).toBe(true);
|
|
174
|
-
expect(pairing.validate(code.code)).toBe(false);
|
|
175
|
-
});
|
|
176
|
-
|
|
177
|
-
it('rejects expired codes', () => {
|
|
178
|
-
const code = pairing.generateCode();
|
|
179
|
-
// Manually expire
|
|
180
|
-
vi.spyOn(Date, 'now').mockReturnValue(Date.now() + 400_000);
|
|
181
|
-
expect(pairing.validate(code.code)).toBe(false);
|
|
182
|
-
vi.restoreAllMocks();
|
|
183
|
-
});
|
|
184
|
-
|
|
185
|
-
it('lists active codes', () => {
|
|
186
|
-
pairing.generateCode();
|
|
187
|
-
pairing.generateCode();
|
|
188
|
-
const consumed = pairing.generateCode();
|
|
189
|
-
pairing.consume(consumed.code);
|
|
190
|
-
|
|
191
|
-
const active = pairing.getActiveCodes();
|
|
192
|
-
expect(active).toHaveLength(2);
|
|
193
|
-
});
|
|
194
|
-
|
|
195
|
-
it('cleans up expired and consumed codes', () => {
|
|
196
|
-
const c1 = pairing.generateCode();
|
|
197
|
-
pairing.consume(c1.code);
|
|
198
|
-
pairing.generateCode(); // still active
|
|
199
|
-
|
|
200
|
-
const removed = pairing.cleanup();
|
|
201
|
-
expect(removed).toBe(1);
|
|
202
|
-
});
|
|
203
|
-
});
|
|
204
|
-
|
|
205
|
-
describe('BridgeServer', () => {
|
|
206
|
-
let server: BridgeServer;
|
|
207
|
-
|
|
208
|
-
beforeEach(() => {
|
|
209
|
-
server = new BridgeServer({ maxDevices: 5, codeLength: 6, codeExpirySeconds: 300 });
|
|
210
|
-
});
|
|
211
|
-
|
|
212
|
-
afterEach(() => {
|
|
213
|
-
server.stop();
|
|
214
|
-
});
|
|
215
|
-
|
|
216
|
-
it('generates pairing codes', () => {
|
|
217
|
-
const code = server.generatePairingCode();
|
|
218
|
-
expect(code).toHaveLength(6);
|
|
219
|
-
});
|
|
220
|
-
|
|
221
|
-
it('handles device connection', () => {
|
|
222
|
-
const socket = makeMockSocket();
|
|
223
|
-
server.handleConnection(socket, 'conn-1');
|
|
224
|
-
expect(server.getConnectionCount()).toBe(1);
|
|
225
|
-
});
|
|
226
|
-
|
|
227
|
-
it('handles full pairing flow', async () => {
|
|
228
|
-
const socket = makeMockSocket();
|
|
229
|
-
server.handleConnection(socket, 'conn-1');
|
|
230
|
-
|
|
231
|
-
const code = server.generatePairingCode();
|
|
232
|
-
|
|
233
|
-
await server.handleMessage('conn-1', JSON.stringify({
|
|
234
|
-
type: 'pair_request',
|
|
235
|
-
id: 'req-1',
|
|
236
|
-
payload: {
|
|
237
|
-
code,
|
|
238
|
-
deviceName: 'Test Phone',
|
|
239
|
-
platform: 'android',
|
|
240
|
-
capabilities: ['camera', 'location'],
|
|
241
|
-
},
|
|
242
|
-
timestamp: Date.now(),
|
|
243
|
-
}));
|
|
244
|
-
|
|
245
|
-
expect(socket.sent).toHaveLength(1);
|
|
246
|
-
const response = parseMessage(socket.sent[0]);
|
|
247
|
-
expect(response.type).toBe('pair_accepted');
|
|
248
|
-
expect(response.deviceId).toBeDefined();
|
|
249
|
-
|
|
250
|
-
// Device should be in registry
|
|
251
|
-
const devices = server.registry.getAll();
|
|
252
|
-
expect(devices).toHaveLength(1);
|
|
253
|
-
expect(devices[0].name).toBe('Test Phone');
|
|
254
|
-
expect(devices[0].platform).toBe('android');
|
|
255
|
-
expect(devices[0].capabilities).toEqual(['camera', 'location']);
|
|
256
|
-
});
|
|
257
|
-
|
|
258
|
-
it('rejects invalid pairing code', async () => {
|
|
259
|
-
const socket = makeMockSocket();
|
|
260
|
-
server.handleConnection(socket, 'conn-1');
|
|
261
|
-
|
|
262
|
-
await server.handleMessage('conn-1', JSON.stringify({
|
|
263
|
-
type: 'pair_request',
|
|
264
|
-
id: 'req-1',
|
|
265
|
-
payload: {
|
|
266
|
-
code: '999999',
|
|
267
|
-
deviceName: 'Test Phone',
|
|
268
|
-
platform: 'android',
|
|
269
|
-
capabilities: [],
|
|
270
|
-
},
|
|
271
|
-
timestamp: Date.now(),
|
|
272
|
-
}));
|
|
273
|
-
|
|
274
|
-
const response = parseMessage(socket.sent[0]);
|
|
275
|
-
expect(response.type).toBe('pair_rejected');
|
|
276
|
-
});
|
|
277
|
-
|
|
278
|
-
it('rejects incomplete pair request', async () => {
|
|
279
|
-
const socket = makeMockSocket();
|
|
280
|
-
server.handleConnection(socket, 'conn-1');
|
|
281
|
-
|
|
282
|
-
await server.handleMessage('conn-1', JSON.stringify({
|
|
283
|
-
type: 'pair_request',
|
|
284
|
-
id: 'req-1',
|
|
285
|
-
payload: { code: '123456' },
|
|
286
|
-
timestamp: Date.now(),
|
|
287
|
-
}));
|
|
288
|
-
|
|
289
|
-
const response = parseMessage(socket.sent[0]);
|
|
290
|
-
expect(response.type).toBe('error');
|
|
291
|
-
});
|
|
292
|
-
|
|
293
|
-
it('handles heartbeat from paired device', async () => {
|
|
294
|
-
const socket = makeMockSocket();
|
|
295
|
-
server.handleConnection(socket, 'conn-1');
|
|
296
|
-
|
|
297
|
-
// Pair the device first
|
|
298
|
-
const code = server.generatePairingCode();
|
|
299
|
-
await server.handleMessage('conn-1', JSON.stringify({
|
|
300
|
-
type: 'pair_request',
|
|
301
|
-
id: 'req-1',
|
|
302
|
-
payload: {
|
|
303
|
-
code,
|
|
304
|
-
deviceName: 'Phone',
|
|
305
|
-
platform: 'ios',
|
|
306
|
-
capabilities: [],
|
|
307
|
-
},
|
|
308
|
-
timestamp: Date.now(),
|
|
309
|
-
}));
|
|
310
|
-
|
|
311
|
-
socket.sent.length = 0;
|
|
312
|
-
|
|
313
|
-
await server.handleMessage('conn-1', JSON.stringify({
|
|
314
|
-
type: 'heartbeat',
|
|
315
|
-
id: 'hb-1',
|
|
316
|
-
timestamp: Date.now(),
|
|
317
|
-
}));
|
|
318
|
-
|
|
319
|
-
expect(socket.sent).toHaveLength(1);
|
|
320
|
-
const response = parseMessage(socket.sent[0]);
|
|
321
|
-
expect(response.type).toBe('heartbeat_ack');
|
|
322
|
-
});
|
|
323
|
-
|
|
324
|
-
it('handles device disconnection', async () => {
|
|
325
|
-
const socket = makeMockSocket();
|
|
326
|
-
server.handleConnection(socket, 'conn-1');
|
|
327
|
-
|
|
328
|
-
// Pair device
|
|
329
|
-
const code = server.generatePairingCode();
|
|
330
|
-
await server.handleMessage('conn-1', JSON.stringify({
|
|
331
|
-
type: 'pair_request',
|
|
332
|
-
id: 'req-1',
|
|
333
|
-
payload: {
|
|
334
|
-
code,
|
|
335
|
-
deviceName: 'Phone',
|
|
336
|
-
platform: 'ios',
|
|
337
|
-
capabilities: ['camera'],
|
|
338
|
-
},
|
|
339
|
-
timestamp: Date.now(),
|
|
340
|
-
}));
|
|
341
|
-
|
|
342
|
-
const deviceId = server.registry.getAll()[0].id;
|
|
343
|
-
server.handleDisconnection('conn-1');
|
|
344
|
-
|
|
345
|
-
expect(server.registry.get(deviceId)!.state).toBe('offline');
|
|
346
|
-
expect(server.getConnectionCount()).toBe(0);
|
|
347
|
-
});
|
|
348
|
-
|
|
349
|
-
it('handles invalid JSON gracefully', async () => {
|
|
350
|
-
const socket = makeMockSocket();
|
|
351
|
-
server.handleConnection(socket, 'conn-1');
|
|
352
|
-
|
|
353
|
-
await server.handleMessage('conn-1', 'not-json{');
|
|
354
|
-
const response = parseMessage(socket.sent[0]);
|
|
355
|
-
expect(response.type).toBe('error');
|
|
356
|
-
});
|
|
357
|
-
|
|
358
|
-
it('handles unknown message type', async () => {
|
|
359
|
-
const socket = makeMockSocket();
|
|
360
|
-
server.handleConnection(socket, 'conn-1');
|
|
361
|
-
|
|
362
|
-
await server.handleMessage('conn-1', JSON.stringify({
|
|
363
|
-
type: 'unknown_type',
|
|
364
|
-
timestamp: Date.now(),
|
|
365
|
-
}));
|
|
366
|
-
|
|
367
|
-
const response = parseMessage(socket.sent[0]);
|
|
368
|
-
expect(response.type).toBe('error');
|
|
369
|
-
expect((response.payload as { message: string }).message).toContain('Unknown message type');
|
|
370
|
-
});
|
|
371
|
-
|
|
372
|
-
it('fires event on device paired', async () => {
|
|
373
|
-
const paired: DeviceInfo[] = [];
|
|
374
|
-
server.onEvent({ onDevicePaired: (d) => paired.push(d) });
|
|
375
|
-
|
|
376
|
-
const socket = makeMockSocket();
|
|
377
|
-
server.handleConnection(socket, 'conn-1');
|
|
378
|
-
const code = server.generatePairingCode();
|
|
379
|
-
|
|
380
|
-
await server.handleMessage('conn-1', JSON.stringify({
|
|
381
|
-
type: 'pair_request',
|
|
382
|
-
id: 'req-1',
|
|
383
|
-
payload: {
|
|
384
|
-
code,
|
|
385
|
-
deviceName: 'Phone',
|
|
386
|
-
platform: 'android',
|
|
387
|
-
capabilities: [],
|
|
388
|
-
},
|
|
389
|
-
timestamp: Date.now(),
|
|
390
|
-
}));
|
|
391
|
-
|
|
392
|
-
expect(paired).toHaveLength(1);
|
|
393
|
-
expect(paired[0].name).toBe('Phone');
|
|
394
|
-
});
|
|
395
|
-
|
|
396
|
-
it('rejects capability request for unknown device', async () => {
|
|
397
|
-
await expect(
|
|
398
|
-
server.requestCapability('unknown', 'camera', 'capture'),
|
|
399
|
-
).rejects.toThrow('Device not found');
|
|
400
|
-
});
|
|
401
|
-
|
|
402
|
-
it('stops cleanly', () => {
|
|
403
|
-
server.start();
|
|
404
|
-
const socket = makeMockSocket();
|
|
405
|
-
server.handleConnection(socket, 'conn-1');
|
|
406
|
-
|
|
407
|
-
server.stop();
|
|
408
|
-
expect(socket.closed).toBe(true);
|
|
409
|
-
expect(server.getConnectionCount()).toBe(0);
|
|
410
|
-
});
|
|
411
|
-
});
|
package/src/index.ts
DELETED
|
@@ -1,22 +0,0 @@
|
|
|
1
|
-
export type {
|
|
2
|
-
DeviceCapability,
|
|
3
|
-
DevicePlatform,
|
|
4
|
-
DeviceConnectionState,
|
|
5
|
-
DeviceInfo,
|
|
6
|
-
PairingCode,
|
|
7
|
-
BridgeConfig,
|
|
8
|
-
BridgeMessageType,
|
|
9
|
-
BridgeMessage,
|
|
10
|
-
PairRequestPayload,
|
|
11
|
-
CapabilityRequestPayload,
|
|
12
|
-
CapabilityResponsePayload,
|
|
13
|
-
} from './types.js';
|
|
14
|
-
export { DEFAULT_BRIDGE_CONFIG } from './types.js';
|
|
15
|
-
export { DeviceRegistry } from './registry.js';
|
|
16
|
-
export { PairingFlow } from './pairing.js';
|
|
17
|
-
export {
|
|
18
|
-
BridgeServer,
|
|
19
|
-
WS_OPEN,
|
|
20
|
-
type BridgeSocket,
|
|
21
|
-
type BridgeEventHandler,
|
|
22
|
-
} from './server.js';
|
package/src/pairing.ts
DELETED
|
@@ -1,129 +0,0 @@
|
|
|
1
|
-
import { getLogger } from '@auxiora/logger';
|
|
2
|
-
import * as crypto from 'node:crypto';
|
|
3
|
-
import type { PairingCode, BridgeConfig } from './types.js';
|
|
4
|
-
import { DEFAULT_BRIDGE_CONFIG } from './types.js';
|
|
5
|
-
|
|
6
|
-
const logger = getLogger('bridge:pairing');
|
|
7
|
-
|
|
8
|
-
/**
|
|
9
|
-
* Manages pairing code generation, validation, and expiry
|
|
10
|
-
* for the device pairing flow.
|
|
11
|
-
*/
|
|
12
|
-
export class PairingFlow {
|
|
13
|
-
private activeCodes = new Map<string, PairingCode>();
|
|
14
|
-
private config: BridgeConfig;
|
|
15
|
-
private cleanupTimer: ReturnType<typeof setInterval> | null = null;
|
|
16
|
-
|
|
17
|
-
constructor(config?: Partial<BridgeConfig>) {
|
|
18
|
-
this.config = { ...DEFAULT_BRIDGE_CONFIG, ...config };
|
|
19
|
-
}
|
|
20
|
-
|
|
21
|
-
/** Generate a new pairing code. */
|
|
22
|
-
generateCode(): PairingCode {
|
|
23
|
-
const code = this.makeCode(this.config.codeLength);
|
|
24
|
-
const pairingCode: PairingCode = {
|
|
25
|
-
code,
|
|
26
|
-
expiresAt: Date.now() + this.config.codeExpirySeconds * 1000,
|
|
27
|
-
used: false,
|
|
28
|
-
};
|
|
29
|
-
|
|
30
|
-
this.activeCodes.set(code, pairingCode);
|
|
31
|
-
logger.info('Pairing code generated', { code });
|
|
32
|
-
return { ...pairingCode };
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
/** Validate a pairing code. Returns true if the code is valid and not expired. */
|
|
36
|
-
validate(code: string): boolean {
|
|
37
|
-
const pairingCode = this.activeCodes.get(code);
|
|
38
|
-
if (!pairingCode) {
|
|
39
|
-
return false;
|
|
40
|
-
}
|
|
41
|
-
if (pairingCode.used) {
|
|
42
|
-
return false;
|
|
43
|
-
}
|
|
44
|
-
if (Date.now() > pairingCode.expiresAt) {
|
|
45
|
-
this.activeCodes.delete(code);
|
|
46
|
-
return false;
|
|
47
|
-
}
|
|
48
|
-
return true;
|
|
49
|
-
}
|
|
50
|
-
|
|
51
|
-
/** Consume a pairing code, marking it as used. Returns true if successful. */
|
|
52
|
-
consume(code: string): boolean {
|
|
53
|
-
if (!this.validate(code)) {
|
|
54
|
-
return false;
|
|
55
|
-
}
|
|
56
|
-
const pairingCode = this.activeCodes.get(code)!;
|
|
57
|
-
pairingCode.used = true;
|
|
58
|
-
logger.info('Pairing code consumed', { code });
|
|
59
|
-
return true;
|
|
60
|
-
}
|
|
61
|
-
|
|
62
|
-
/** Revoke an active pairing code. */
|
|
63
|
-
revoke(code: string): boolean {
|
|
64
|
-
const removed = this.activeCodes.delete(code);
|
|
65
|
-
if (removed) {
|
|
66
|
-
logger.info('Pairing code revoked', { code });
|
|
67
|
-
}
|
|
68
|
-
return removed;
|
|
69
|
-
}
|
|
70
|
-
|
|
71
|
-
/** Get all active (non-expired, non-used) codes. */
|
|
72
|
-
getActiveCodes(): PairingCode[] {
|
|
73
|
-
const now = Date.now();
|
|
74
|
-
const active: PairingCode[] = [];
|
|
75
|
-
|
|
76
|
-
for (const pc of this.activeCodes.values()) {
|
|
77
|
-
if (!pc.used && now <= pc.expiresAt) {
|
|
78
|
-
active.push({ ...pc });
|
|
79
|
-
}
|
|
80
|
-
}
|
|
81
|
-
|
|
82
|
-
return active;
|
|
83
|
-
}
|
|
84
|
-
|
|
85
|
-
/** Remove expired codes. */
|
|
86
|
-
cleanup(): number {
|
|
87
|
-
const now = Date.now();
|
|
88
|
-
let removed = 0;
|
|
89
|
-
|
|
90
|
-
for (const [code, pc] of this.activeCodes) {
|
|
91
|
-
if (pc.used || now > pc.expiresAt) {
|
|
92
|
-
this.activeCodes.delete(code);
|
|
93
|
-
removed++;
|
|
94
|
-
}
|
|
95
|
-
}
|
|
96
|
-
|
|
97
|
-
return removed;
|
|
98
|
-
}
|
|
99
|
-
|
|
100
|
-
/** Start automatic cleanup timer. */
|
|
101
|
-
startCleanup(intervalMs = 60_000): void {
|
|
102
|
-
this.stopCleanup();
|
|
103
|
-
this.cleanupTimer = setInterval(() => this.cleanup(), intervalMs);
|
|
104
|
-
}
|
|
105
|
-
|
|
106
|
-
/** Stop automatic cleanup timer. */
|
|
107
|
-
stopCleanup(): void {
|
|
108
|
-
if (this.cleanupTimer) {
|
|
109
|
-
clearInterval(this.cleanupTimer);
|
|
110
|
-
this.cleanupTimer = null;
|
|
111
|
-
}
|
|
112
|
-
}
|
|
113
|
-
|
|
114
|
-
/** Destroy the pairing flow, clearing all codes and stopping timers. */
|
|
115
|
-
destroy(): void {
|
|
116
|
-
this.stopCleanup();
|
|
117
|
-
this.activeCodes.clear();
|
|
118
|
-
}
|
|
119
|
-
|
|
120
|
-
/** Generate a random numeric code of the given length. */
|
|
121
|
-
private makeCode(length: number): string {
|
|
122
|
-
const bytes = crypto.randomBytes(length);
|
|
123
|
-
let code = '';
|
|
124
|
-
for (let i = 0; i < length; i++) {
|
|
125
|
-
code += (bytes[i] % 10).toString();
|
|
126
|
-
}
|
|
127
|
-
return code;
|
|
128
|
-
}
|
|
129
|
-
}
|
package/src/registry.ts
DELETED
|
@@ -1,108 +0,0 @@
|
|
|
1
|
-
import { getLogger } from '@auxiora/logger';
|
|
2
|
-
import type { DeviceInfo, DeviceCapability, DeviceConnectionState } from './types.js';
|
|
3
|
-
|
|
4
|
-
const logger = getLogger('bridge:registry');
|
|
5
|
-
|
|
6
|
-
/**
|
|
7
|
-
* Tracks paired devices, their capabilities, and online status.
|
|
8
|
-
*/
|
|
9
|
-
export class DeviceRegistry {
|
|
10
|
-
private devices = new Map<string, DeviceInfo>();
|
|
11
|
-
private maxDevices: number;
|
|
12
|
-
|
|
13
|
-
constructor(maxDevices = 10) {
|
|
14
|
-
this.maxDevices = maxDevices;
|
|
15
|
-
}
|
|
16
|
-
|
|
17
|
-
/** Register a newly paired device. */
|
|
18
|
-
register(device: DeviceInfo): void {
|
|
19
|
-
if (this.devices.size >= this.maxDevices && !this.devices.has(device.id)) {
|
|
20
|
-
throw new Error(`Maximum device limit (${this.maxDevices}) reached`);
|
|
21
|
-
}
|
|
22
|
-
this.devices.set(device.id, { ...device });
|
|
23
|
-
logger.info('Device registered', { deviceId: device.id, name: device.name, platform: device.platform });
|
|
24
|
-
}
|
|
25
|
-
|
|
26
|
-
/** Remove a device from the registry. */
|
|
27
|
-
unregister(deviceId: string): boolean {
|
|
28
|
-
const removed = this.devices.delete(deviceId);
|
|
29
|
-
if (removed) {
|
|
30
|
-
logger.info('Device unregistered', { deviceId });
|
|
31
|
-
}
|
|
32
|
-
return removed;
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
/** Get a device by ID. */
|
|
36
|
-
get(deviceId: string): DeviceInfo | undefined {
|
|
37
|
-
const device = this.devices.get(deviceId);
|
|
38
|
-
return device ? { ...device } : undefined;
|
|
39
|
-
}
|
|
40
|
-
|
|
41
|
-
/** Get all registered devices. */
|
|
42
|
-
getAll(): DeviceInfo[] {
|
|
43
|
-
return Array.from(this.devices.values()).map((d) => ({ ...d }));
|
|
44
|
-
}
|
|
45
|
-
|
|
46
|
-
/** Get devices that are currently online. */
|
|
47
|
-
getOnline(): DeviceInfo[] {
|
|
48
|
-
return this.getAll().filter((d) => d.state === 'online');
|
|
49
|
-
}
|
|
50
|
-
|
|
51
|
-
/** Get devices that have a specific capability. */
|
|
52
|
-
getByCapability(capability: DeviceCapability): DeviceInfo[] {
|
|
53
|
-
return this.getAll().filter((d) => d.capabilities.includes(capability));
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
/** Update a device's connection state. */
|
|
57
|
-
setState(deviceId: string, state: DeviceConnectionState): void {
|
|
58
|
-
const device = this.devices.get(deviceId);
|
|
59
|
-
if (device) {
|
|
60
|
-
device.state = state;
|
|
61
|
-
if (state === 'online') {
|
|
62
|
-
device.lastSeen = Date.now();
|
|
63
|
-
}
|
|
64
|
-
}
|
|
65
|
-
}
|
|
66
|
-
|
|
67
|
-
/** Record that a heartbeat was received from a device. */
|
|
68
|
-
heartbeat(deviceId: string): void {
|
|
69
|
-
const device = this.devices.get(deviceId);
|
|
70
|
-
if (device) {
|
|
71
|
-
device.lastSeen = Date.now();
|
|
72
|
-
if (device.state !== 'online') {
|
|
73
|
-
device.state = 'online';
|
|
74
|
-
}
|
|
75
|
-
}
|
|
76
|
-
}
|
|
77
|
-
|
|
78
|
-
/** Check for devices that have gone offline based on heartbeat timeout. */
|
|
79
|
-
checkTimeouts(timeoutMs: number): string[] {
|
|
80
|
-
const now = Date.now();
|
|
81
|
-
const timedOut: string[] = [];
|
|
82
|
-
|
|
83
|
-
for (const device of this.devices.values()) {
|
|
84
|
-
if (device.state === 'online' && now - device.lastSeen > timeoutMs) {
|
|
85
|
-
device.state = 'offline';
|
|
86
|
-
timedOut.push(device.id);
|
|
87
|
-
logger.info('Device timed out', { deviceId: device.id });
|
|
88
|
-
}
|
|
89
|
-
}
|
|
90
|
-
|
|
91
|
-
return timedOut;
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
/** Get the total count of devices. */
|
|
95
|
-
get size(): number {
|
|
96
|
-
return this.devices.size;
|
|
97
|
-
}
|
|
98
|
-
|
|
99
|
-
/** Check if registry has space for more devices. */
|
|
100
|
-
hasCapacity(): boolean {
|
|
101
|
-
return this.devices.size < this.maxDevices;
|
|
102
|
-
}
|
|
103
|
-
|
|
104
|
-
/** Clear all devices. */
|
|
105
|
-
clear(): void {
|
|
106
|
-
this.devices.clear();
|
|
107
|
-
}
|
|
108
|
-
}
|
package/src/server.ts
DELETED
|
@@ -1,338 +0,0 @@
|
|
|
1
|
-
import { getLogger } from '@auxiora/logger';
|
|
2
|
-
import * as crypto from 'node:crypto';
|
|
3
|
-
import type {
|
|
4
|
-
BridgeConfig,
|
|
5
|
-
BridgeMessage,
|
|
6
|
-
DeviceCapability,
|
|
7
|
-
DeviceInfo,
|
|
8
|
-
DevicePlatform,
|
|
9
|
-
PairRequestPayload,
|
|
10
|
-
CapabilityRequestPayload,
|
|
11
|
-
CapabilityResponsePayload,
|
|
12
|
-
} from './types.js';
|
|
13
|
-
import { DEFAULT_BRIDGE_CONFIG } from './types.js';
|
|
14
|
-
import { DeviceRegistry } from './registry.js';
|
|
15
|
-
import { PairingFlow } from './pairing.js';
|
|
16
|
-
|
|
17
|
-
const logger = getLogger('bridge:server');
|
|
18
|
-
|
|
19
|
-
/** Minimal WebSocket interface for dependency injection (no tight coupling to 'ws'). */
|
|
20
|
-
export interface BridgeSocket {
|
|
21
|
-
send(data: string): void;
|
|
22
|
-
close(code?: number, reason?: string): void;
|
|
23
|
-
readyState: number;
|
|
24
|
-
}
|
|
25
|
-
|
|
26
|
-
/** WebSocket readyState constants. */
|
|
27
|
-
export const WS_OPEN = 1;
|
|
28
|
-
|
|
29
|
-
/** Event handler for Bridge server events. */
|
|
30
|
-
export interface BridgeEventHandler {
|
|
31
|
-
onDevicePaired?(device: DeviceInfo): void;
|
|
32
|
-
onDeviceDisconnected?(deviceId: string): void;
|
|
33
|
-
onCapabilityResponse?(deviceId: string, response: CapabilityResponsePayload): void;
|
|
34
|
-
}
|
|
35
|
-
|
|
36
|
-
/**
|
|
37
|
-
* Bridge server that manages device connections and the pairing protocol.
|
|
38
|
-
* Designed to be mounted on an existing WebSocket server (e.g., the gateway).
|
|
39
|
-
*/
|
|
40
|
-
export class BridgeServer {
|
|
41
|
-
private config: BridgeConfig;
|
|
42
|
-
readonly registry: DeviceRegistry;
|
|
43
|
-
readonly pairing: PairingFlow;
|
|
44
|
-
private connections = new Map<string, BridgeSocket>();
|
|
45
|
-
private pendingRequests = new Map<string, {
|
|
46
|
-
resolve: (response: CapabilityResponsePayload) => void;
|
|
47
|
-
reject: (error: Error) => void;
|
|
48
|
-
timer: ReturnType<typeof setTimeout>;
|
|
49
|
-
}>();
|
|
50
|
-
private heartbeatTimer: ReturnType<typeof setInterval> | null = null;
|
|
51
|
-
private eventHandler: BridgeEventHandler = {};
|
|
52
|
-
|
|
53
|
-
constructor(config?: Partial<BridgeConfig>) {
|
|
54
|
-
this.config = { ...DEFAULT_BRIDGE_CONFIG, ...config };
|
|
55
|
-
this.registry = new DeviceRegistry(this.config.maxDevices);
|
|
56
|
-
this.pairing = new PairingFlow(this.config);
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
/** Set event handler. */
|
|
60
|
-
onEvent(handler: BridgeEventHandler): void {
|
|
61
|
-
this.eventHandler = handler;
|
|
62
|
-
}
|
|
63
|
-
|
|
64
|
-
/** Handle a new WebSocket connection from a device. */
|
|
65
|
-
handleConnection(socket: BridgeSocket, connectionId: string): void {
|
|
66
|
-
this.connections.set(connectionId, socket);
|
|
67
|
-
logger.info('New bridge connection', { connectionId });
|
|
68
|
-
}
|
|
69
|
-
|
|
70
|
-
/** Handle a message from a connected device. */
|
|
71
|
-
async handleMessage(connectionId: string, raw: string): Promise<void> {
|
|
72
|
-
let message: BridgeMessage;
|
|
73
|
-
try {
|
|
74
|
-
message = JSON.parse(raw) as BridgeMessage;
|
|
75
|
-
} catch {
|
|
76
|
-
this.sendError(connectionId, 'Invalid message format');
|
|
77
|
-
return;
|
|
78
|
-
}
|
|
79
|
-
|
|
80
|
-
switch (message.type) {
|
|
81
|
-
case 'pair_request':
|
|
82
|
-
await this.handlePairRequest(connectionId, message);
|
|
83
|
-
break;
|
|
84
|
-
case 'heartbeat':
|
|
85
|
-
this.handleHeartbeat(connectionId, message);
|
|
86
|
-
break;
|
|
87
|
-
case 'capability_response':
|
|
88
|
-
this.handleCapabilityResponse(message);
|
|
89
|
-
break;
|
|
90
|
-
case 'device_info':
|
|
91
|
-
this.handleDeviceInfo(connectionId, message);
|
|
92
|
-
break;
|
|
93
|
-
case 'disconnect':
|
|
94
|
-
this.handleDisconnect(connectionId);
|
|
95
|
-
break;
|
|
96
|
-
default:
|
|
97
|
-
this.sendError(connectionId, `Unknown message type: ${message.type}`);
|
|
98
|
-
}
|
|
99
|
-
}
|
|
100
|
-
|
|
101
|
-
/** Handle device disconnection. */
|
|
102
|
-
handleDisconnection(connectionId: string): void {
|
|
103
|
-
this.connections.delete(connectionId);
|
|
104
|
-
|
|
105
|
-
// Find device for this connection
|
|
106
|
-
const device = this.findDeviceByConnection(connectionId);
|
|
107
|
-
if (device) {
|
|
108
|
-
this.registry.setState(device.id, 'offline');
|
|
109
|
-
this.eventHandler.onDeviceDisconnected?.(device.id);
|
|
110
|
-
logger.info('Device disconnected', { deviceId: device.id });
|
|
111
|
-
}
|
|
112
|
-
}
|
|
113
|
-
|
|
114
|
-
/** Request a capability from a specific device. */
|
|
115
|
-
async requestCapability(
|
|
116
|
-
deviceId: string,
|
|
117
|
-
capability: DeviceCapability,
|
|
118
|
-
action: string,
|
|
119
|
-
params?: Record<string, unknown>,
|
|
120
|
-
timeoutMs = 30_000,
|
|
121
|
-
): Promise<CapabilityResponsePayload> {
|
|
122
|
-
const device = this.registry.get(deviceId);
|
|
123
|
-
if (!device) {
|
|
124
|
-
throw new Error(`Device not found: ${deviceId}`);
|
|
125
|
-
}
|
|
126
|
-
if (device.state !== 'online') {
|
|
127
|
-
throw new Error(`Device is not online: ${deviceId}`);
|
|
128
|
-
}
|
|
129
|
-
if (!device.capabilities.includes(capability)) {
|
|
130
|
-
throw new Error(`Device ${deviceId} does not have capability: ${capability}`);
|
|
131
|
-
}
|
|
132
|
-
|
|
133
|
-
const requestId = crypto.randomUUID();
|
|
134
|
-
const request: BridgeMessage = {
|
|
135
|
-
type: 'capability_request',
|
|
136
|
-
id: requestId,
|
|
137
|
-
deviceId,
|
|
138
|
-
payload: { capability, action, params } satisfies CapabilityRequestPayload,
|
|
139
|
-
timestamp: Date.now(),
|
|
140
|
-
};
|
|
141
|
-
|
|
142
|
-
return new Promise<CapabilityResponsePayload>((resolve, reject) => {
|
|
143
|
-
const timer = setTimeout(() => {
|
|
144
|
-
this.pendingRequests.delete(requestId);
|
|
145
|
-
reject(new Error(`Capability request timed out: ${capability}/${action}`));
|
|
146
|
-
}, timeoutMs);
|
|
147
|
-
|
|
148
|
-
this.pendingRequests.set(requestId, { resolve, reject, timer });
|
|
149
|
-
this.sendToDevice(deviceId, request);
|
|
150
|
-
});
|
|
151
|
-
}
|
|
152
|
-
|
|
153
|
-
/** Generate a new pairing code. */
|
|
154
|
-
generatePairingCode(): string {
|
|
155
|
-
const pc = this.pairing.generateCode();
|
|
156
|
-
return pc.code;
|
|
157
|
-
}
|
|
158
|
-
|
|
159
|
-
/** Start heartbeat checking. */
|
|
160
|
-
start(): void {
|
|
161
|
-
this.pairing.startCleanup();
|
|
162
|
-
const interval = this.config.heartbeatIntervalMs;
|
|
163
|
-
const timeout = interval * this.config.offlineAfterMissedHeartbeats;
|
|
164
|
-
|
|
165
|
-
this.heartbeatTimer = setInterval(() => {
|
|
166
|
-
const timedOut = this.registry.checkTimeouts(timeout);
|
|
167
|
-
for (const deviceId of timedOut) {
|
|
168
|
-
this.eventHandler.onDeviceDisconnected?.(deviceId);
|
|
169
|
-
}
|
|
170
|
-
}, interval);
|
|
171
|
-
|
|
172
|
-
logger.info('Bridge server started');
|
|
173
|
-
}
|
|
174
|
-
|
|
175
|
-
/** Stop the bridge server. */
|
|
176
|
-
stop(): void {
|
|
177
|
-
if (this.heartbeatTimer) {
|
|
178
|
-
clearInterval(this.heartbeatTimer);
|
|
179
|
-
this.heartbeatTimer = null;
|
|
180
|
-
}
|
|
181
|
-
this.pairing.destroy();
|
|
182
|
-
|
|
183
|
-
// Reject all pending requests
|
|
184
|
-
for (const [id, pending] of this.pendingRequests) {
|
|
185
|
-
clearTimeout(pending.timer);
|
|
186
|
-
pending.reject(new Error('Bridge server stopped'));
|
|
187
|
-
this.pendingRequests.delete(id);
|
|
188
|
-
}
|
|
189
|
-
|
|
190
|
-
// Close all connections
|
|
191
|
-
for (const [id, socket] of this.connections) {
|
|
192
|
-
try {
|
|
193
|
-
socket.close(1001, 'Bridge server stopping');
|
|
194
|
-
} catch {
|
|
195
|
-
// Ignore close errors
|
|
196
|
-
}
|
|
197
|
-
}
|
|
198
|
-
this.connections.clear();
|
|
199
|
-
|
|
200
|
-
logger.info('Bridge server stopped');
|
|
201
|
-
}
|
|
202
|
-
|
|
203
|
-
/** Get the number of active connections. */
|
|
204
|
-
getConnectionCount(): number {
|
|
205
|
-
return this.connections.size;
|
|
206
|
-
}
|
|
207
|
-
|
|
208
|
-
// --- Private helpers ---
|
|
209
|
-
|
|
210
|
-
private async handlePairRequest(connectionId: string, message: BridgeMessage): Promise<void> {
|
|
211
|
-
const payload = message.payload as PairRequestPayload;
|
|
212
|
-
if (!payload?.code || !payload?.deviceName || !payload?.platform) {
|
|
213
|
-
this.sendError(connectionId, 'Invalid pair request: missing required fields');
|
|
214
|
-
return;
|
|
215
|
-
}
|
|
216
|
-
|
|
217
|
-
const valid = this.pairing.consume(payload.code);
|
|
218
|
-
if (!valid) {
|
|
219
|
-
this.sendTo(connectionId, {
|
|
220
|
-
type: 'pair_rejected',
|
|
221
|
-
id: message.id,
|
|
222
|
-
payload: { reason: 'Invalid or expired pairing code' },
|
|
223
|
-
timestamp: Date.now(),
|
|
224
|
-
});
|
|
225
|
-
return;
|
|
226
|
-
}
|
|
227
|
-
|
|
228
|
-
const deviceId = crypto.randomUUID();
|
|
229
|
-
const device: DeviceInfo = {
|
|
230
|
-
id: deviceId,
|
|
231
|
-
name: payload.deviceName,
|
|
232
|
-
platform: payload.platform,
|
|
233
|
-
capabilities: payload.capabilities ?? [],
|
|
234
|
-
state: 'online',
|
|
235
|
-
pairedAt: Date.now(),
|
|
236
|
-
lastSeen: Date.now(),
|
|
237
|
-
};
|
|
238
|
-
|
|
239
|
-
this.registry.register(device);
|
|
240
|
-
|
|
241
|
-
// Associate this connection with the device
|
|
242
|
-
this.setDeviceConnection(deviceId, connectionId);
|
|
243
|
-
|
|
244
|
-
this.sendTo(connectionId, {
|
|
245
|
-
type: 'pair_accepted',
|
|
246
|
-
id: message.id,
|
|
247
|
-
deviceId,
|
|
248
|
-
payload: { deviceId, name: device.name },
|
|
249
|
-
timestamp: Date.now(),
|
|
250
|
-
});
|
|
251
|
-
|
|
252
|
-
this.eventHandler.onDevicePaired?.(device);
|
|
253
|
-
logger.info('Device paired', { deviceId, name: device.name, platform: device.platform });
|
|
254
|
-
}
|
|
255
|
-
|
|
256
|
-
private handleHeartbeat(connectionId: string, message: BridgeMessage): void {
|
|
257
|
-
const device = this.findDeviceByConnection(connectionId);
|
|
258
|
-
if (device) {
|
|
259
|
-
this.registry.heartbeat(device.id);
|
|
260
|
-
this.sendTo(connectionId, {
|
|
261
|
-
type: 'heartbeat_ack',
|
|
262
|
-
id: message.id,
|
|
263
|
-
deviceId: device.id,
|
|
264
|
-
timestamp: Date.now(),
|
|
265
|
-
});
|
|
266
|
-
}
|
|
267
|
-
}
|
|
268
|
-
|
|
269
|
-
private handleCapabilityResponse(message: BridgeMessage): void {
|
|
270
|
-
if (!message.id) return;
|
|
271
|
-
const pending = this.pendingRequests.get(message.id);
|
|
272
|
-
if (pending) {
|
|
273
|
-
clearTimeout(pending.timer);
|
|
274
|
-
this.pendingRequests.delete(message.id);
|
|
275
|
-
pending.resolve(message.payload as CapabilityResponsePayload);
|
|
276
|
-
this.eventHandler.onCapabilityResponse?.(
|
|
277
|
-
message.deviceId ?? '',
|
|
278
|
-
message.payload as CapabilityResponsePayload,
|
|
279
|
-
);
|
|
280
|
-
}
|
|
281
|
-
}
|
|
282
|
-
|
|
283
|
-
private handleDeviceInfo(connectionId: string, message: BridgeMessage): void {
|
|
284
|
-
const device = this.findDeviceByConnection(connectionId);
|
|
285
|
-
if (!device) return;
|
|
286
|
-
|
|
287
|
-
const info = message.payload as Partial<{ name: string; capabilities: DeviceCapability[] }>;
|
|
288
|
-
if (info?.capabilities) {
|
|
289
|
-
// Re-register with updated capabilities
|
|
290
|
-
const existing = this.registry.get(device.id);
|
|
291
|
-
if (existing) {
|
|
292
|
-
existing.capabilities = info.capabilities;
|
|
293
|
-
if (info.name) existing.name = info.name;
|
|
294
|
-
this.registry.register(existing);
|
|
295
|
-
}
|
|
296
|
-
}
|
|
297
|
-
}
|
|
298
|
-
|
|
299
|
-
private handleDisconnect(connectionId: string): void {
|
|
300
|
-
this.handleDisconnection(connectionId);
|
|
301
|
-
}
|
|
302
|
-
|
|
303
|
-
private sendTo(connectionId: string, message: BridgeMessage): void {
|
|
304
|
-
const socket = this.connections.get(connectionId);
|
|
305
|
-
if (socket && socket.readyState === WS_OPEN) {
|
|
306
|
-
socket.send(JSON.stringify(message));
|
|
307
|
-
}
|
|
308
|
-
}
|
|
309
|
-
|
|
310
|
-
private sendToDevice(deviceId: string, message: BridgeMessage): void {
|
|
311
|
-
const connectionId = this.deviceConnections.get(deviceId);
|
|
312
|
-
if (connectionId) {
|
|
313
|
-
this.sendTo(connectionId, message);
|
|
314
|
-
}
|
|
315
|
-
}
|
|
316
|
-
|
|
317
|
-
private sendError(connectionId: string, errorMessage: string): void {
|
|
318
|
-
this.sendTo(connectionId, {
|
|
319
|
-
type: 'error',
|
|
320
|
-
payload: { message: errorMessage },
|
|
321
|
-
timestamp: Date.now(),
|
|
322
|
-
});
|
|
323
|
-
}
|
|
324
|
-
|
|
325
|
-
// Device-to-connection mapping
|
|
326
|
-
private deviceConnections = new Map<string, string>();
|
|
327
|
-
private connectionDevices = new Map<string, string>();
|
|
328
|
-
|
|
329
|
-
private setDeviceConnection(deviceId: string, connectionId: string): void {
|
|
330
|
-
this.deviceConnections.set(deviceId, connectionId);
|
|
331
|
-
this.connectionDevices.set(connectionId, deviceId);
|
|
332
|
-
}
|
|
333
|
-
|
|
334
|
-
private findDeviceByConnection(connectionId: string): DeviceInfo | undefined {
|
|
335
|
-
const deviceId = this.connectionDevices.get(connectionId);
|
|
336
|
-
return deviceId ? this.registry.get(deviceId) : undefined;
|
|
337
|
-
}
|
|
338
|
-
}
|
package/src/types.ts
DELETED
|
@@ -1,115 +0,0 @@
|
|
|
1
|
-
/** Capabilities a device node can provide. */
|
|
2
|
-
export type DeviceCapability =
|
|
3
|
-
| 'camera'
|
|
4
|
-
| 'screen'
|
|
5
|
-
| 'microphone'
|
|
6
|
-
| 'location'
|
|
7
|
-
| 'notifications'
|
|
8
|
-
| 'clipboard'
|
|
9
|
-
| 'sensors';
|
|
10
|
-
|
|
11
|
-
/** Platform of the device. */
|
|
12
|
-
export type DevicePlatform = 'ios' | 'android' | 'macos' | 'windows' | 'linux' | 'web';
|
|
13
|
-
|
|
14
|
-
/** Connection state for a device. */
|
|
15
|
-
export type DeviceConnectionState = 'connecting' | 'paired' | 'online' | 'offline';
|
|
16
|
-
|
|
17
|
-
/** Information about a paired device. */
|
|
18
|
-
export interface DeviceInfo {
|
|
19
|
-
/** Unique device identifier. */
|
|
20
|
-
id: string;
|
|
21
|
-
/** User-facing device name. */
|
|
22
|
-
name: string;
|
|
23
|
-
/** Device platform. */
|
|
24
|
-
platform: DevicePlatform;
|
|
25
|
-
/** Capabilities this device supports. */
|
|
26
|
-
capabilities: DeviceCapability[];
|
|
27
|
-
/** Current connection state. */
|
|
28
|
-
state: DeviceConnectionState;
|
|
29
|
-
/** When the device was first paired. */
|
|
30
|
-
pairedAt: number;
|
|
31
|
-
/** When the device was last seen online. */
|
|
32
|
-
lastSeen: number;
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
/** Pairing code issued by the server. */
|
|
36
|
-
export interface PairingCode {
|
|
37
|
-
/** The short code the user enters on the device. */
|
|
38
|
-
code: string;
|
|
39
|
-
/** When the code expires (unix ms). */
|
|
40
|
-
expiresAt: number;
|
|
41
|
-
/** Whether the code has been used. */
|
|
42
|
-
used: boolean;
|
|
43
|
-
}
|
|
44
|
-
|
|
45
|
-
/** Configuration for the Bridge server. */
|
|
46
|
-
export interface BridgeConfig {
|
|
47
|
-
/** Maximum number of paired devices. */
|
|
48
|
-
maxDevices: number;
|
|
49
|
-
/** Pairing code length (digits). */
|
|
50
|
-
codeLength: number;
|
|
51
|
-
/** Pairing code expiry in seconds. */
|
|
52
|
-
codeExpirySeconds: number;
|
|
53
|
-
/** Heartbeat interval in milliseconds. */
|
|
54
|
-
heartbeatIntervalMs: number;
|
|
55
|
-
/** Consider device offline after this many missed heartbeats. */
|
|
56
|
-
offlineAfterMissedHeartbeats: number;
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
export const DEFAULT_BRIDGE_CONFIG: BridgeConfig = {
|
|
60
|
-
maxDevices: 10,
|
|
61
|
-
codeLength: 6,
|
|
62
|
-
codeExpirySeconds: 300,
|
|
63
|
-
heartbeatIntervalMs: 30_000,
|
|
64
|
-
offlineAfterMissedHeartbeats: 3,
|
|
65
|
-
};
|
|
66
|
-
|
|
67
|
-
/** Messages sent between Bridge server and device nodes. */
|
|
68
|
-
export type BridgeMessageType =
|
|
69
|
-
| 'pair_request'
|
|
70
|
-
| 'pair_accepted'
|
|
71
|
-
| 'pair_rejected'
|
|
72
|
-
| 'heartbeat'
|
|
73
|
-
| 'heartbeat_ack'
|
|
74
|
-
| 'capability_request'
|
|
75
|
-
| 'capability_response'
|
|
76
|
-
| 'device_info'
|
|
77
|
-
| 'disconnect'
|
|
78
|
-
| 'error';
|
|
79
|
-
|
|
80
|
-
/** A message in the Bridge protocol. */
|
|
81
|
-
export interface BridgeMessage {
|
|
82
|
-
type: BridgeMessageType;
|
|
83
|
-
/** Correlation ID for request/response matching. */
|
|
84
|
-
id?: string;
|
|
85
|
-
/** The sending device ID (set by server for forwarded messages). */
|
|
86
|
-
deviceId?: string;
|
|
87
|
-
/** Message-specific payload. */
|
|
88
|
-
payload?: unknown;
|
|
89
|
-
/** Timestamp (unix ms). */
|
|
90
|
-
timestamp: number;
|
|
91
|
-
}
|
|
92
|
-
|
|
93
|
-
/** Payload for pair_request. */
|
|
94
|
-
export interface PairRequestPayload {
|
|
95
|
-
code: string;
|
|
96
|
-
deviceName: string;
|
|
97
|
-
platform: DevicePlatform;
|
|
98
|
-
capabilities: DeviceCapability[];
|
|
99
|
-
}
|
|
100
|
-
|
|
101
|
-
/** Payload for capability_request. */
|
|
102
|
-
export interface CapabilityRequestPayload {
|
|
103
|
-
capability: DeviceCapability;
|
|
104
|
-
action: string;
|
|
105
|
-
params?: Record<string, unknown>;
|
|
106
|
-
}
|
|
107
|
-
|
|
108
|
-
/** Payload for capability_response. */
|
|
109
|
-
export interface CapabilityResponsePayload {
|
|
110
|
-
capability: DeviceCapability;
|
|
111
|
-
action: string;
|
|
112
|
-
success: boolean;
|
|
113
|
-
data?: unknown;
|
|
114
|
-
error?: string;
|
|
115
|
-
}
|
package/tsconfig.json
DELETED
package/tsconfig.tsbuildinfo
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"fileNames":["../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es5.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2015.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2016.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2017.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2018.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2019.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2020.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2021.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2022.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2023.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2015.core.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2015.collection.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2015.generator.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2015.iterable.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2015.promise.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2015.proxy.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2015.reflect.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2015.symbol.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2015.symbol.wellknown.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2016.array.include.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2016.intl.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2017.arraybuffer.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2017.date.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2017.object.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2017.sharedmemory.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2017.string.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2017.intl.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2017.typedarrays.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2018.asyncgenerator.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2018.asynciterable.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2018.intl.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2018.promise.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2018.regexp.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2019.array.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2019.object.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2019.string.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2019.symbol.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2019.intl.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2020.bigint.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2020.date.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2020.promise.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2020.sharedmemory.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2020.string.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2020.symbol.wellknown.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2020.intl.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2020.number.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2021.promise.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2021.string.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2021.weakref.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2021.intl.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2022.array.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2022.error.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2022.intl.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2022.object.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2022.string.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2022.regexp.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2023.array.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2023.collection.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.es2023.intl.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.esnext.disposable.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.esnext.float16.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.decorators.d.ts","../../node_modules/.pnpm/typescript@5.9.3/node_modules/typescript/lib/lib.decorators.legacy.d.ts","./src/types.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/compatibility/iterators.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/globals.typedarray.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/buffer.buffer.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/globals.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/web-globals/abortcontroller.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/web-globals/blob.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/web-globals/console.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/web-globals/crypto.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/web-globals/domexception.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/web-globals/encoding.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/web-globals/events.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/utility.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/header.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/readable.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/fetch.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/formdata.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/connector.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/client-stats.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/client.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/errors.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/dispatcher.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/global-dispatcher.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/global-origin.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/pool-stats.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/pool.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/handlers.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/balanced-pool.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/h2c-client.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/agent.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/mock-interceptor.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/mock-call-history.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/mock-agent.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/mock-client.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/mock-pool.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/snapshot-agent.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/mock-errors.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/proxy-agent.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/env-http-proxy-agent.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/retry-handler.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/retry-agent.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/api.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/cache-interceptor.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/interceptors.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/util.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/cookies.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/patch.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/websocket.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/eventsource.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/diagnostics-channel.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/content-type.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/cache.d.ts","../../node_modules/.pnpm/undici-types@7.16.0/node_modules/undici-types/index.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/web-globals/fetch.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/web-globals/importmeta.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/web-globals/messaging.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/web-globals/navigator.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/web-globals/performance.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/web-globals/storage.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/web-globals/streams.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/web-globals/timers.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/web-globals/url.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/assert.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/assert/strict.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/async_hooks.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/buffer.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/child_process.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/cluster.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/console.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/constants.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/crypto.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/dgram.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/diagnostics_channel.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/dns.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/dns/promises.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/domain.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/events.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/fs.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/fs/promises.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/http.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/http2.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/https.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/inspector.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/inspector.generated.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/inspector/promises.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/module.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/net.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/os.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/path.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/path/posix.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/path/win32.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/perf_hooks.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/process.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/punycode.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/querystring.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/quic.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/readline.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/readline/promises.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/repl.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/sea.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/sqlite.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/stream.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/stream/consumers.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/stream/promises.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/stream/web.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/string_decoder.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/test.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/test/reporters.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/timers.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/timers/promises.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/tls.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/trace_events.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/tty.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/url.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/util.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/util/types.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/v8.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/vm.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/wasi.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/worker_threads.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/zlib.d.ts","../../node_modules/.pnpm/@types+node@25.1.0/node_modules/@types/node/index.d.ts","../../node_modules/.pnpm/pino-std-serializers@7.1.0/node_modules/pino-std-serializers/index.d.ts","../../node_modules/.pnpm/sonic-boom@4.2.0/node_modules/sonic-boom/types/index.d.ts","../../node_modules/.pnpm/pino@9.14.0/node_modules/pino/pino.d.ts","../logger/dist/index.d.ts","./src/registry.ts","./src/pairing.ts","./src/server.ts","./src/index.ts","../../node_modules/.pnpm/@vitest+pretty-format@4.0.18/node_modules/@vitest/pretty-format/dist/index.d.ts","../../node_modules/.pnpm/@vitest+utils@4.0.18/node_modules/@vitest/utils/dist/display.d.ts","../../node_modules/.pnpm/@vitest+utils@4.0.18/node_modules/@vitest/utils/dist/types.d.ts","../../node_modules/.pnpm/@vitest+utils@4.0.18/node_modules/@vitest/utils/dist/helpers.d.ts","../../node_modules/.pnpm/@vitest+utils@4.0.18/node_modules/@vitest/utils/dist/timers.d.ts","../../node_modules/.pnpm/@vitest+utils@4.0.18/node_modules/@vitest/utils/dist/index.d.ts","../../node_modules/.pnpm/@vitest+runner@4.0.18/node_modules/@vitest/runner/dist/tasks.d-C7UxawJ9.d.ts","../../node_modules/.pnpm/@vitest+utils@4.0.18/node_modules/@vitest/utils/dist/types.d-BCElaP-c.d.ts","../../node_modules/.pnpm/@vitest+utils@4.0.18/node_modules/@vitest/utils/dist/diff.d.ts","../../node_modules/.pnpm/@vitest+runner@4.0.18/node_modules/@vitest/runner/dist/types.d.ts","../../node_modules/.pnpm/@vitest+runner@4.0.18/node_modules/@vitest/runner/dist/index.d.ts","../../node_modules/.pnpm/vitest@4.0.18_@types+node@25.1.0/node_modules/vitest/dist/chunks/traces.d.402V_yFI.d.ts","../../node_modules/.pnpm/vite@7.3.1_@types+node@25.1.0/node_modules/vite/types/hmrPayload.d.ts","../../node_modules/.pnpm/vite@7.3.1_@types+node@25.1.0/node_modules/vite/dist/node/chunks/moduleRunnerTransport.d.ts","../../node_modules/.pnpm/vite@7.3.1_@types+node@25.1.0/node_modules/vite/types/customEvent.d.ts","../../node_modules/.pnpm/vite@7.3.1_@types+node@25.1.0/node_modules/vite/types/hot.d.ts","../../node_modules/.pnpm/vite@7.3.1_@types+node@25.1.0/node_modules/vite/dist/node/module-runner.d.ts","../../node_modules/.pnpm/@vitest+snapshot@4.0.18/node_modules/@vitest/snapshot/dist/environment.d-DHdQ1Csl.d.ts","../../node_modules/.pnpm/@vitest+snapshot@4.0.18/node_modules/@vitest/snapshot/dist/rawSnapshot.d-lFsMJFUd.d.ts","../../node_modules/.pnpm/@vitest+snapshot@4.0.18/node_modules/@vitest/snapshot/dist/index.d.ts","../../node_modules/.pnpm/vitest@4.0.18_@types+node@25.1.0/node_modules/vitest/dist/chunks/config.d.Cy95HiCx.d.ts","../../node_modules/.pnpm/vitest@4.0.18_@types+node@25.1.0/node_modules/vitest/dist/chunks/environment.d.CrsxCzP1.d.ts","../../node_modules/.pnpm/vitest@4.0.18_@types+node@25.1.0/node_modules/vitest/dist/chunks/rpc.d.RH3apGEf.d.ts","../../node_modules/.pnpm/vitest@4.0.18_@types+node@25.1.0/node_modules/vitest/dist/chunks/worker.d.Dyxm8DEL.d.ts","../../node_modules/.pnpm/vitest@4.0.18_@types+node@25.1.0/node_modules/vitest/dist/chunks/browser.d.ChKACdzH.d.ts","../../node_modules/.pnpm/@vitest+spy@4.0.18/node_modules/@vitest/spy/dist/index.d.ts","../../node_modules/.pnpm/tinyrainbow@3.0.3/node_modules/tinyrainbow/dist/index.d.ts","../../node_modules/.pnpm/@standard-schema+spec@1.1.0/node_modules/@standard-schema/spec/dist/index.d.ts","../../node_modules/.pnpm/@types+deep-eql@4.0.2/node_modules/@types/deep-eql/index.d.ts","../../node_modules/.pnpm/assertion-error@2.0.1/node_modules/assertion-error/index.d.ts","../../node_modules/.pnpm/@types+chai@5.2.3/node_modules/@types/chai/index.d.ts","../../node_modules/.pnpm/@vitest+expect@4.0.18/node_modules/@vitest/expect/dist/index.d.ts","../../node_modules/.pnpm/@vitest+runner@4.0.18/node_modules/@vitest/runner/dist/utils.d.ts","../../node_modules/.pnpm/tinybench@2.9.0/node_modules/tinybench/dist/index.d.ts","../../node_modules/.pnpm/vitest@4.0.18_@types+node@25.1.0/node_modules/vitest/dist/chunks/benchmark.d.DAaHLpsq.d.ts","../../node_modules/.pnpm/vitest@4.0.18_@types+node@25.1.0/node_modules/vitest/dist/chunks/global.d.B15mdLcR.d.ts","../../node_modules/.pnpm/vitest@4.0.18_@types+node@25.1.0/node_modules/vitest/dist/chunks/suite.d.BJWk38HB.d.ts","../../node_modules/.pnpm/vitest@4.0.18_@types+node@25.1.0/node_modules/vitest/dist/chunks/evaluatedModules.d.BxJ5omdx.d.ts","../../node_modules/.pnpm/expect-type@1.3.0/node_modules/expect-type/dist/utils.d.ts","../../node_modules/.pnpm/expect-type@1.3.0/node_modules/expect-type/dist/overloads.d.ts","../../node_modules/.pnpm/expect-type@1.3.0/node_modules/expect-type/dist/branding.d.ts","../../node_modules/.pnpm/expect-type@1.3.0/node_modules/expect-type/dist/messages.d.ts","../../node_modules/.pnpm/expect-type@1.3.0/node_modules/expect-type/dist/index.d.ts","../../node_modules/.pnpm/vitest@4.0.18_@types+node@25.1.0/node_modules/vitest/dist/index.d.ts","./src/__tests__/bridge.test.ts"],"fileIdsList":[[67,129,137,141,144,146,147,148,160],[67,129,137,141,144,146,147,148,160,222,223],[67,126,127,129,137,141,144,146,147,148,160],[67,128,129,137,141,144,146,147,148,160],[129,137,141,144,146,147,148,160],[67,129,137,141,144,146,147,148,160,168],[67,129,130,135,137,140,141,144,146,147,148,150,160,165,177],[67,129,130,131,137,140,141,144,146,147,148,160],[67,129,132,137,141,144,146,147,148,160,178],[67,129,133,134,137,141,144,146,147,148,151,160],[67,129,134,137,141,144,146,147,148,160,165,174],[67,129,135,137,140,141,144,146,147,148,150,160],[67,128,129,136,137,141,144,146,147,148,160],[67,129,137,138,141,144,146,147,148,160],[67,129,137,139,140,141,144,146,147,148,160],[67,128,129,137,140,141,144,146,147,148,160],[67,129,137,140,141,142,144,146,147,148,160,165,177],[67,129,137,140,141,142,144,146,147,148,160,165,168],[67,116,129,137,140,141,143,144,146,147,148,150,160,165,177],[67,129,137,140,141,143,144,146,147,148,150,160,165,174,177],[67,129,137,141,143,144,145,146,147,148,160,165,174,177],[65,66,67,68,69,70,71,72,73,74,75,117,118,119,120,121,122,123,124,125,126,127,128,129,130,131,132,133,134,135,136,137,138,139,140,141,142,143,144,145,146,147,148,149,150,151,152,153,154,155,156,157,158,159,160,161,162,163,164,165,166,167,168,169,170,171,172,173,174,175,176,177,178,179,180,181,182,183,184],[67,129,137,140,141,144,146,147,148,160],[67,129,137,141,144,146,148,160],[67,129,137,141,144,146,147,148,149,160,177],[67,129,137,140,141,144,146,147,148,150,160,165],[67,129,137,141,144,146,147,148,151,160],[67,129,137,141,144,146,147,148,152,160],[67,129,137,140,141,144,146,147,148,155,160],[67,126,127,128,129,130,131,132,133,134,135,136,137,138,139,140,141,142,143,144,145,146,147,148,149,150,151,152,153,154,155,156,157,158,159,160,161,162,163,164,165,166,167,168,169,170,171,172,173,174,175,176,177,178,179,180,181,182,183,184],[67,129,137,141,144,146,147,148,157,160],[67,129,137,141,144,146,147,148,158,160],[67,129,134,137,141,144,146,147,148,150,160,168],[67,129,137,140,141,144,146,147,148,160,161],[67,129,137,141,144,146,147,148,160,162,178,181],[67,129,137,140,141,144,146,147,148,160,165,167,168],[67,129,137,141,144,146,147,148,160,166,168],[67,129,137,141,144,146,147,148,160,168,178],[67,129,137,141,144,146,147,148,160,169],[67,126,129,137,141,144,146,147,148,160,165,171],[67,129,137,141,144,146,147,148,160,165,170],[67,129,137,140,141,144,146,147,148,160,172,173],[67,129,137,141,144,146,147,148,160,172,173],[67,129,134,137,141,144,146,147,148,150,160,165,174],[67,129,137,141,144,146,147,148,160,175],[67,129,137,141,144,146,147,148,150,160,176],[67,129,137,141,143,144,146,147,148,158,160,177],[67,129,137,141,144,146,147,148,160,178,179],[67,129,134,137,141,144,146,147,148,160,179],[67,129,137,141,144,146,147,148,160,165,180],[67,129,137,141,144,146,147,148,149,160,181],[67,129,137,141,144,146,147,148,160,182],[67,129,132,137,141,144,146,147,148,160],[67,129,134,137,141,144,146,147,148,160],[67,129,137,141,144,146,147,148,160,178],[67,116,129,137,141,144,146,147,148,160],[67,129,137,141,144,146,147,148,160,177],[67,129,137,141,144,146,147,148,160,183],[67,129,137,141,144,146,147,148,155,160],[67,129,137,141,144,146,147,148,160,173],[67,116,129,137,140,141,142,144,146,147,148,155,160,165,168,177,180,181,183],[67,129,137,141,144,146,147,148,160,165,184],[67,129,137,141,144,146,147,148,160,195,199,202,204,219,220,221,224,229],[67,129,137,141,144,146,147,148,160,199,200,202,203],[67,129,137,141,144,146,147,148,160,199],[67,129,137,141,144,146,147,148,160,199,200,202],[67,129,137,141,144,146,147,148,160,199,200],[67,129,137,141,144,146,147,148,160,194,211,212],[67,129,137,141,144,146,147,148,160,194,211],[67,129,137,141,144,146,147,148,160,194,201],[67,129,137,141,144,146,147,148,160,194],[67,129,137,141,144,146,147,148,160,196],[67,129,137,141,144,146,147,148,160,194,195,196,197,198],[67,129,137,141,144,146,147,148,160,232,233],[67,129,137,141,144,146,147,148,160,232,233,234,235],[67,129,137,141,144,146,147,148,160,232,234],[67,129,137,141,144,146,147,148,160,232],[67,129,137,141,143,144,146,147,148,160,185],[67,129,137,140,141,144,146,147,148,160,183,186,187],[67,129,137,140,141,144,146,147,148,160,185],[67,82,85,88,89,129,137,141,144,146,147,148,160,177],[67,85,129,137,141,144,146,147,148,160,165,177],[67,85,89,129,137,141,144,146,147,148,160,177],[67,129,137,141,144,146,147,148,160,165],[67,79,129,137,141,144,146,147,148,160],[67,83,129,137,141,144,146,147,148,160],[67,81,82,85,129,137,141,144,146,147,148,160,177],[67,129,137,141,144,146,147,148,150,160,174],[67,129,137,141,144,146,147,148,160,185],[67,79,129,137,141,144,146,147,148,160,185],[67,81,85,129,137,141,144,146,147,148,150,160,177],[67,76,77,78,80,84,129,137,140,141,144,146,147,148,160,165,177],[67,85,93,101,129,137,141,144,146,147,148,160],[67,77,83,129,137,141,144,146,147,148,160],[67,85,110,111,129,137,141,144,146,147,148,160],[67,77,80,85,129,137,141,144,146,147,148,160,168,177,185],[67,85,129,137,141,144,146,147,148,160],[67,81,85,129,137,141,144,146,147,148,160,177],[67,76,129,137,141,144,146,147,148,160],[67,79,80,81,83,84,85,86,87,89,90,91,92,93,94,95,96,97,98,99,100,101,102,103,104,105,106,107,108,109,111,112,113,114,115,129,137,141,144,146,147,148,160],[67,85,103,106,129,137,141,144,146,147,148,160],[67,85,93,94,95,129,137,141,144,146,147,148,160],[67,83,85,94,96,129,137,141,144,146,147,148,160],[67,84,129,137,141,144,146,147,148,160],[67,77,79,85,129,137,141,144,146,147,148,160],[67,85,89,94,96,129,137,141,144,146,147,148,160],[67,89,129,137,141,144,146,147,148,160],[67,83,85,88,129,137,141,144,146,147,148,160,177],[67,77,81,85,93,129,137,141,144,146,147,148,160],[67,85,103,129,137,141,144,146,147,148,160],[67,96,129,137,141,144,146,147,148,160],[67,79,85,110,129,137,141,144,146,147,148,160,168,183,185],[67,129,137,141,144,146,147,148,160,206],[67,129,137,141,144,146,147,148,160,206,207,208,209],[67,129,137,141,144,146,147,148,160,208],[67,129,137,141,144,146,147,148,160,204,226,227,229],[67,129,137,141,144,146,147,148,160,204,205,217,229],[67,129,137,141,144,146,147,148,160,194,202,204,213,229],[67,129,137,141,144,146,147,148,160,210],[67,129,137,141,144,146,147,148,160,194,204,213,216,225,228,229],[67,129,137,141,144,146,147,148,160,204,205,210,213,229],[67,129,137,141,144,146,147,148,160,204,226,227,228,229],[67,129,137,141,144,146,147,148,160,204,210,214,215,216,229],[67,129,137,141,144,146,147,148,160,194,199,202,204,205,210,213,214,215,216,217,218,219,225,226,227,228,229,230,231,236],[64,67,129,137,141,144,146,147,148,160,190,191,192,237],[64,67,129,137,141,144,146,147,148,160,190,191,192],[64,67,129,134,137,141,144,146,147,148,160,189],[64,67,129,137,141,144,146,147,148,160,189],[64,67,129,134,137,141,144,146,147,148,160,189,190,191],[67,129,137,141,144,146,147,148,160,188]],"fileInfos":[{"version":"c430d44666289dae81f30fa7b2edebf186ecc91a2d4c71266ea6ae76388792e1","affectsGlobalScope":true,"impliedFormat":1},{"version":"45b7ab580deca34ae9729e97c13cfd999df04416a79116c3bfb483804f85ded4","impliedFormat":1},{"version":"3facaf05f0c5fc569c5649dd359892c98a85557e3e0c847964caeb67076f4d75","impliedFormat":1},{"version":"e44bb8bbac7f10ecc786703fe0a6a4b952189f908707980ba8f3c8975a760962","impliedFormat":1},{"version":"5e1c4c362065a6b95ff952c0eab010f04dcd2c3494e813b493ecfd4fcb9fc0d8","impliedFormat":1},{"version":"68d73b4a11549f9c0b7d352d10e91e5dca8faa3322bfb77b661839c42b1ddec7","impliedFormat":1},{"version":"5efce4fc3c29ea84e8928f97adec086e3dc876365e0982cc8479a07954a3efd4","impliedFormat":1},{"version":"feecb1be483ed332fad555aff858affd90a48ab19ba7272ee084704eb7167569","impliedFormat":1},{"version":"ee7bad0c15b58988daa84371e0b89d313b762ab83cb5b31b8a2d1162e8eb41c2","impliedFormat":1},{"version":"27bdc30a0e32783366a5abeda841bc22757c1797de8681bbe81fbc735eeb1c10","impliedFormat":1},{"version":"c57796738e7f83dbc4b8e65132f11a377649c00dd3eee333f672b8f0a6bea671","affectsGlobalScope":true,"impliedFormat":1},{"version":"dc2df20b1bcdc8c2d34af4926e2c3ab15ffe1160a63e58b7e09833f616efff44","affectsGlobalScope":true,"impliedFormat":1},{"version":"515d0b7b9bea2e31ea4ec968e9edd2c39d3eebf4a2d5cbd04e88639819ae3b71","affectsGlobalScope":true,"impliedFormat":1},{"version":"0559b1f683ac7505ae451f9a96ce4c3c92bdc71411651ca6ddb0e88baaaad6a3","affectsGlobalScope":true,"impliedFormat":1},{"version":"0dc1e7ceda9b8b9b455c3a2d67b0412feab00bd2f66656cd8850e8831b08b537","affectsGlobalScope":true,"impliedFormat":1},{"version":"ce691fb9e5c64efb9547083e4a34091bcbe5bdb41027e310ebba8f7d96a98671","affectsGlobalScope":true,"impliedFormat":1},{"version":"8d697a2a929a5fcb38b7a65594020fcef05ec1630804a33748829c5ff53640d0","affectsGlobalScope":true,"impliedFormat":1},{"version":"4ff2a353abf8a80ee399af572debb8faab2d33ad38c4b4474cff7f26e7653b8d","affectsGlobalScope":true,"impliedFormat":1},{"version":"fb0f136d372979348d59b3f5020b4cdb81b5504192b1cacff5d1fbba29378aa1","affectsGlobalScope":true,"impliedFormat":1},{"version":"d15bea3d62cbbdb9797079416b8ac375ae99162a7fba5de2c6c505446486ac0a","affectsGlobalScope":true,"impliedFormat":1},{"version":"68d18b664c9d32a7336a70235958b8997ebc1c3b8505f4f1ae2b7e7753b87618","affectsGlobalScope":true,"impliedFormat":1},{"version":"eb3d66c8327153d8fa7dd03f9c58d351107fe824c79e9b56b462935176cdf12a","affectsGlobalScope":true,"impliedFormat":1},{"version":"38f0219c9e23c915ef9790ab1d680440d95419ad264816fa15009a8851e79119","affectsGlobalScope":true,"impliedFormat":1},{"version":"69ab18c3b76cd9b1be3d188eaf8bba06112ebbe2f47f6c322b5105a6fbc45a2e","affectsGlobalScope":true,"impliedFormat":1},{"version":"a680117f487a4d2f30ea46f1b4b7f58bef1480456e18ba53ee85c2746eeca012","affectsGlobalScope":true,"impliedFormat":1},{"version":"2f11ff796926e0832f9ae148008138ad583bd181899ab7dd768a2666700b1893","affectsGlobalScope":true,"impliedFormat":1},{"version":"4de680d5bb41c17f7f68e0419412ca23c98d5749dcaaea1896172f06435891fc","affectsGlobalScope":true,"impliedFormat":1},{"version":"954296b30da6d508a104a3a0b5d96b76495c709785c1d11610908e63481ee667","affectsGlobalScope":true,"impliedFormat":1},{"version":"ac9538681b19688c8eae65811b329d3744af679e0bdfa5d842d0e32524c73e1c","affectsGlobalScope":true,"impliedFormat":1},{"version":"0a969edff4bd52585473d24995c5ef223f6652d6ef46193309b3921d65dd4376","affectsGlobalScope":true,"impliedFormat":1},{"version":"9e9fbd7030c440b33d021da145d3232984c8bb7916f277e8ffd3dc2e3eae2bdb","affectsGlobalScope":true,"impliedFormat":1},{"version":"811ec78f7fefcabbda4bfa93b3eb67d9ae166ef95f9bff989d964061cbf81a0c","affectsGlobalScope":true,"impliedFormat":1},{"version":"717937616a17072082152a2ef351cb51f98802fb4b2fdabd32399843875974ca","affectsGlobalScope":true,"impliedFormat":1},{"version":"d7e7d9b7b50e5f22c915b525acc5a49a7a6584cf8f62d0569e557c5cfc4b2ac2","affectsGlobalScope":true,"impliedFormat":1},{"version":"71c37f4c9543f31dfced6c7840e068c5a5aacb7b89111a4364b1d5276b852557","affectsGlobalScope":true,"impliedFormat":1},{"version":"576711e016cf4f1804676043e6a0a5414252560eb57de9faceee34d79798c850","affectsGlobalScope":true,"impliedFormat":1},{"version":"89c1b1281ba7b8a96efc676b11b264de7a8374c5ea1e6617f11880a13fc56dc6","affectsGlobalScope":true,"impliedFormat":1},{"version":"74f7fa2d027d5b33eb0471c8e82a6c87216223181ec31247c357a3e8e2fddc5b","affectsGlobalScope":true,"impliedFormat":1},{"version":"d6d7ae4d1f1f3772e2a3cde568ed08991a8ae34a080ff1151af28b7f798e22ca","affectsGlobalScope":true,"impliedFormat":1},{"version":"063600664504610fe3e99b717a1223f8b1900087fab0b4cad1496a114744f8df","affectsGlobalScope":true,"impliedFormat":1},{"version":"934019d7e3c81950f9a8426d093458b65d5aff2c7c1511233c0fd5b941e608ab","affectsGlobalScope":true,"impliedFormat":1},{"version":"52ada8e0b6e0482b728070b7639ee42e83a9b1c22d205992756fe020fd9f4a47","affectsGlobalScope":true,"impliedFormat":1},{"version":"3bdefe1bfd4d6dee0e26f928f93ccc128f1b64d5d501ff4a8cf3c6371200e5e6","affectsGlobalScope":true,"impliedFormat":1},{"version":"59fb2c069260b4ba00b5643b907ef5d5341b167e7d1dbf58dfd895658bda2867","affectsGlobalScope":true,"impliedFormat":1},{"version":"639e512c0dfc3fad96a84caad71b8834d66329a1f28dc95e3946c9b58176c73a","affectsGlobalScope":true,"impliedFormat":1},{"version":"368af93f74c9c932edd84c58883e736c9e3d53cec1fe24c0b0ff451f529ceab1","affectsGlobalScope":true,"impliedFormat":1},{"version":"af3dd424cf267428f30ccfc376f47a2c0114546b55c44d8c0f1d57d841e28d74","affectsGlobalScope":true,"impliedFormat":1},{"version":"995c005ab91a498455ea8dfb63aa9f83fa2ea793c3d8aa344be4a1678d06d399","affectsGlobalScope":true,"impliedFormat":1},{"version":"959d36cddf5e7d572a65045b876f2956c973a586da58e5d26cde519184fd9b8a","affectsGlobalScope":true,"impliedFormat":1},{"version":"965f36eae237dd74e6cca203a43e9ca801ce38824ead814728a2807b1910117d","affectsGlobalScope":true,"impliedFormat":1},{"version":"3925a6c820dcb1a06506c90b1577db1fdbf7705d65b62b99dce4be75c637e26b","affectsGlobalScope":true,"impliedFormat":1},{"version":"0a3d63ef2b853447ec4f749d3f368ce642264246e02911fcb1590d8c161b8005","affectsGlobalScope":true,"impliedFormat":1},{"version":"8cdf8847677ac7d20486e54dd3fcf09eda95812ac8ace44b4418da1bbbab6eb8","affectsGlobalScope":true,"impliedFormat":1},{"version":"8444af78980e3b20b49324f4a16ba35024fef3ee069a0eb67616ea6ca821c47a","affectsGlobalScope":true,"impliedFormat":1},{"version":"3287d9d085fbd618c3971944b65b4be57859f5415f495b33a6adc994edd2f004","affectsGlobalScope":true,"impliedFormat":1},{"version":"b4b67b1a91182421f5df999988c690f14d813b9850b40acd06ed44691f6727ad","affectsGlobalScope":true,"impliedFormat":1},{"version":"df83c2a6c73228b625b0beb6669c7ee2a09c914637e2d35170723ad49c0f5cd4","affectsGlobalScope":true,"impliedFormat":1},{"version":"436aaf437562f276ec2ddbee2f2cdedac7664c1e4c1d2c36839ddd582eeb3d0a","affectsGlobalScope":true,"impliedFormat":1},{"version":"8e3c06ea092138bf9fa5e874a1fdbc9d54805d074bee1de31b99a11e2fec239d","affectsGlobalScope":true,"impliedFormat":1},{"version":"51ad4c928303041605b4d7ae32e0c1ee387d43a24cd6f1ebf4a2699e1076d4fa","affectsGlobalScope":true,"impliedFormat":1},{"version":"196cb558a13d4533a5163286f30b0509ce0210e4b316c56c38d4c0fd2fb38405","affectsGlobalScope":true,"impliedFormat":1},{"version":"8e7f8264d0fb4c5339605a15daadb037bf238c10b654bb3eee14208f860a32ea","affectsGlobalScope":true,"impliedFormat":1},{"version":"782dec38049b92d4e85c1585fbea5474a219c6984a35b004963b00beb1aab538","affectsGlobalScope":true,"impliedFormat":1},{"version":"2ece231470aabb97e90c322fc9f29755d14139b37cde6462e44bb1b1aa44f8e2","signature":"d0c1ca528444d9ec8875751413d0bd6b17eb8d69ba1da304bace7b0561014589","impliedFormat":99},{"version":"d153a11543fd884b596587ccd97aebbeed950b26933ee000f94009f1ab142848","affectsGlobalScope":true,"impliedFormat":1},{"version":"0ccdaa19852d25ecd84eec365c3bfa16e7859cadecf6e9ca6d0dbbbee439743f","affectsGlobalScope":true,"impliedFormat":1},{"version":"438b41419b1df9f1fbe33b5e1b18f5853432be205991d1b19f5b7f351675541e","affectsGlobalScope":true,"impliedFormat":1},{"version":"096116f8fedc1765d5bd6ef360c257b4a9048e5415054b3bf3c41b07f8951b0b","affectsGlobalScope":true,"impliedFormat":1},{"version":"e5e01375c9e124a83b52ee4b3244ed1a4d214a6cfb54ac73e164a823a4a7860a","affectsGlobalScope":true,"impliedFormat":1},{"version":"f90ae2bbce1505e67f2f6502392e318f5714bae82d2d969185c4a6cecc8af2fc","affectsGlobalScope":true,"impliedFormat":1},{"version":"4b58e207b93a8f1c88bbf2a95ddc686ac83962b13830fe8ad3f404ffc7051fb4","affectsGlobalScope":true,"impliedFormat":1},{"version":"1fefabcb2b06736a66d2904074d56268753654805e829989a46a0161cd8412c5","affectsGlobalScope":true,"impliedFormat":1},{"version":"9798340ffb0d067d69b1ae5b32faa17ab31b82466a3fc00d8f2f2df0c8554aaa","affectsGlobalScope":true,"impliedFormat":1},{"version":"c18a99f01eb788d849ad032b31cafd49de0b19e083fe775370834c5675d7df8e","affectsGlobalScope":true,"impliedFormat":1},{"version":"5247874c2a23b9a62d178ae84f2db6a1d54e6c9a2e7e057e178cc5eea13757fc","affectsGlobalScope":true,"impliedFormat":1},{"version":"cdcf9ea426ad970f96ac930cd176d5c69c6c24eebd9fc580e1572d6c6a88f62c","impliedFormat":1},{"version":"23cd712e2ce083d68afe69224587438e5914b457b8acf87073c22494d706a3d0","impliedFormat":1},{"version":"487b694c3de27ddf4ad107d4007ad304d29effccf9800c8ae23c2093638d906a","impliedFormat":1},{"version":"3a80bc85f38526ca3b08007ee80712e7bb0601df178b23fbf0bf87036fce40ce","impliedFormat":1},{"version":"ccf4552357ce3c159ef75f0f0114e80401702228f1898bdc9402214c9499e8c0","impliedFormat":1},{"version":"c6fd2c5a395f2432786c9cb8deb870b9b0e8ff7e22c029954fabdd692bff6195","impliedFormat":1},{"version":"68834d631c8838c715f225509cfc3927913b9cc7a4870460b5b60c8dbdb99baf","impliedFormat":1},{"version":"2931540c47ee0ff8a62860e61782eb17b155615db61e36986e54645ec67f67c2","impliedFormat":1},{"version":"ccab02f3920fc75c01174c47fcf67882a11daf16baf9e81701d0a94636e94556","impliedFormat":1},{"version":"f6faf5f74e4c4cc309a6c6a6c4da02dbb840be5d3e92905a23dcd7b2b0bd1986","impliedFormat":1},{"version":"ea6bc8de8b59f90a7a3960005fd01988f98fd0784e14bc6922dde2e93305ec7d","impliedFormat":1},{"version":"36107995674b29284a115e21a0618c4c2751b32a8766dd4cb3ba740308b16d59","impliedFormat":1},{"version":"914a0ae30d96d71915fc519ccb4efbf2b62c0ddfb3a3fc6129151076bc01dc60","impliedFormat":1},{"version":"33e981bf6376e939f99bd7f89abec757c64897d33c005036b9a10d9587d80187","impliedFormat":1},{"version":"7fd1b31fd35876b0aa650811c25ec2c97a3c6387e5473eb18004bed86cdd76b6","impliedFormat":1},{"version":"b41767d372275c154c7ea6c9d5449d9a741b8ce080f640155cc88ba1763e35b3","impliedFormat":1},{"version":"3bacf516d686d08682751a3bd2519ea3b8041a164bfb4f1d35728993e70a2426","impliedFormat":1},{"version":"7fb266686238369442bd1719bc0d7edd0199da4fb8540354e1ff7f16669b4323","impliedFormat":1},{"version":"0a60a292b89ca7218b8616f78e5bbd1c96b87e048849469cccb4355e98af959a","impliedFormat":1},{"version":"0b6e25234b4eec6ed96ab138d96eb70b135690d7dd01f3dd8a8ab291c35a683a","impliedFormat":1},{"version":"9666f2f84b985b62400d2e5ab0adae9ff44de9b2a34803c2c5bd3c8325b17dc0","impliedFormat":1},{"version":"40cd35c95e9cf22cfa5bd84e96408b6fcbca55295f4ff822390abb11afbc3dca","impliedFormat":1},{"version":"b1616b8959bf557feb16369c6124a97a0e74ed6f49d1df73bb4b9ddf68acf3f3","impliedFormat":1},{"version":"5b03a034c72146b61573aab280f295b015b9168470f2df05f6080a2122f9b4df","impliedFormat":1},{"version":"40b463c6766ca1b689bfcc46d26b5e295954f32ad43e37ee6953c0a677e4ae2b","impliedFormat":1},{"version":"249b9cab7f5d628b71308c7d9bb0a808b50b091e640ba3ed6e2d0516f4a8d91d","impliedFormat":1},{"version":"80aae6afc67faa5ac0b32b5b8bc8cc9f7fa299cff15cf09cc2e11fd28c6ae29e","impliedFormat":1},{"version":"f473cd2288991ff3221165dcf73cd5d24da30391f87e85b3dd4d0450c787a391","impliedFormat":1},{"version":"499e5b055a5aba1e1998f7311a6c441a369831c70905cc565ceac93c28083d53","impliedFormat":1},{"version":"54c3e2371e3d016469ad959697fd257e5621e16296fa67082c2575d0bf8eced0","impliedFormat":1},{"version":"beb8233b2c220cfa0feea31fbe9218d89fa02faa81ef744be8dce5acb89bb1fd","impliedFormat":1},{"version":"c183b931b68ad184bc8e8372bf663f3d33304772fb482f29fb91b3c391031f3e","impliedFormat":1},{"version":"5d0375ca7310efb77e3ef18d068d53784faf62705e0ad04569597ae0e755c401","impliedFormat":1},{"version":"59af37caec41ecf7b2e76059c9672a49e682c1a2aa6f9d7dc78878f53aa284d6","impliedFormat":1},{"version":"addf417b9eb3f938fddf8d81e96393a165e4be0d4a8b6402292f9c634b1cb00d","impliedFormat":1},{"version":"48cc3ec153b50985fb95153258a710782b25975b10dd4ac8a4f3920632d10790","impliedFormat":1},{"version":"adf27937dba6af9f08a68c5b1d3fce0ca7d4b960c57e6d6c844e7d1a8e53adae","impliedFormat":1},{"version":"e1528ca65ac90f6fa0e4a247eb656b4263c470bb22d9033e466463e13395e599","impliedFormat":1},{"version":"2e85db9e6fd73cfa3d7f28e0ab6b55417ea18931423bd47b409a96e4a169e8e6","impliedFormat":1},{"version":"c46e079fe54c76f95c67fb89081b3e399da2c7d109e7dca8e4b58d83e332e605","impliedFormat":1},{"version":"866078923a56d026e39243b4392e282c1c63159723996fa89243140e1388a98d","impliedFormat":1},{"version":"dd0109710de4cd93e245121ab86d8c66d20f3ead80074b68e9c3e349c4f53342","affectsGlobalScope":true,"impliedFormat":1},{"version":"b3275d55fac10b799c9546804126239baf020d220136163f763b55a74e50e750","affectsGlobalScope":true,"impliedFormat":1},{"version":"fa68a0a3b7cb32c00e39ee3cd31f8f15b80cac97dce51b6ee7fc14a1e8deb30b","affectsGlobalScope":true,"impliedFormat":1},{"version":"1cf059eaf468efcc649f8cf6075d3cb98e9a35a0fe9c44419ec3d2f5428d7123","affectsGlobalScope":true,"impliedFormat":1},{"version":"6c36e755bced82df7fb6ce8169265d0a7bb046ab4e2cb6d0da0cb72b22033e89","affectsGlobalScope":true,"impliedFormat":1},{"version":"e7721c4f69f93c91360c26a0a84ee885997d748237ef78ef665b153e622b36c1","affectsGlobalScope":true,"impliedFormat":1},{"version":"7a93de4ff8a63bafe62ba86b89af1df0ccb5e40bb85b0c67d6bbcfdcf96bf3d4","affectsGlobalScope":true,"impliedFormat":1},{"version":"90e85f9bc549dfe2b5749b45fe734144e96cd5d04b38eae244028794e142a77e","affectsGlobalScope":true,"impliedFormat":1},{"version":"e0a5deeb610b2a50a6350bd23df6490036a1773a8a71d70f2f9549ab009e67ee","affectsGlobalScope":true,"impliedFormat":1},{"version":"435b3711465425770ed2ee2f1cf00ce071835265e0851a7dc4600ab4b007550e","impliedFormat":1},{"version":"7e49f52a159435fc8df4de9dc377ef5860732ca2dc9efec1640531d3cf5da7a3","impliedFormat":1},{"version":"dd4bde4bdc2e5394aed6855e98cf135dfdf5dd6468cad842e03116d31bbcc9bc","impliedFormat":1},{"version":"4d4e879009a84a47c05350b8dca823036ba3a29a3038efed1be76c9f81e45edf","affectsGlobalScope":true,"impliedFormat":1},{"version":"cf83d90d5faf27b994c2e79af02e32b555dbfe42cd9bd1571445f2168d1f4e2d","impliedFormat":1},{"version":"9ba13b47cb450a438e3076c4a3f6afb9dc85e17eae50f26d4b2d72c0688c9251","impliedFormat":1},{"version":"b64cd4401633ea4ecadfd700ddc8323a13b63b106ac7127c1d2726f32424622c","impliedFormat":1},{"version":"37c6e5fe5715814412b43cc9b50b24c67a63c4e04e753e0d1305970d65417a60","impliedFormat":1},{"version":"0e28335ac43f4d94dd2fe6d9e6fa6813570640839addd10d309d7985f33a6308","impliedFormat":1},{"version":"ee0e4946247f842c6dd483cbb60a5e6b484fee07996e3a7bc7343dfb68a04c5d","impliedFormat":1},{"version":"ef051f42b7e0ef5ca04552f54c4552eac84099d64b6c5ad0ef4033574b6035b8","impliedFormat":1},{"version":"853a43154f1d01b0173d9cbd74063507ece57170bad7a3b68f3fa1229ad0a92f","impliedFormat":1},{"version":"56231e3c39a031bfb0afb797690b20ed4537670c93c0318b72d5180833d98b72","impliedFormat":1},{"version":"5cc7c39031bfd8b00ad58f32143d59eb6ffc24f5d41a20931269011dccd36c5e","impliedFormat":1},{"version":"b0b69c61b0f0ec8ca15db4c8c41f6e77f4cacb784d42bca948f42dea33e8757e","affectsGlobalScope":true,"impliedFormat":1},{"version":"f96a48183254c00d24575401f1a761b4ce4927d927407e7862a83e06ce5d6964","impliedFormat":1},{"version":"cc25940cfb27aa538e60d465f98bb5068d4d7d33131861ace43f04fe6947d68f","impliedFormat":1},{"version":"e6f370b5c1d52edabb93ef055d0c0c396a98be77db8aa022fcc3670787b8b5f5","impliedFormat":1},{"version":"01ff95aa1443e3f7248974e5a771f513cb2ac158c8898f470a1792f817bee497","impliedFormat":1},{"version":"9d96a7ce809392ff2cb99691acf7c62e632fe56897356ba013b689277aca3619","impliedFormat":1},{"version":"42a05d8f239f74587d4926aba8cc54792eed8e8a442c7adc9b38b516642aadfe","impliedFormat":1},{"version":"5d21b58d60383cc6ab9ad3d3e265d7d25af24a2c9b506247e0e50b0a884920be","impliedFormat":1},{"version":"101f482fd48cb4c7c0468dcc6d62c843d842977aea6235644b1edd05e81fbf22","impliedFormat":1},{"version":"ae6757460f37078884b1571a3de3ebaf724d827d7e1d53626c02b3c2a408ac63","affectsGlobalScope":true,"impliedFormat":1},{"version":"27c0a08e343c6a0ae17bd13ba6d44a9758236dc904cd5e4b43456996cd51f520","impliedFormat":1},{"version":"3ef397f12387eff17f550bc484ea7c27d21d43816bbe609d495107f44b97e933","impliedFormat":1},{"version":"1023282e2ba810bc07905d3668349fbd37a26411f0c8f94a70ef3c05fe523fcf","impliedFormat":1},{"version":"b214ebcf76c51b115453f69729ee8aa7b7f8eccdae2a922b568a45c2d7ff52f7","impliedFormat":1},{"version":"429c9cdfa7d126255779efd7e6d9057ced2d69c81859bbab32073bad52e9ba76","impliedFormat":1},{"version":"6f80e51ba310608cd71bcdc09a171d7bbfb3b316048601c9ec215ce16a8dcfbc","impliedFormat":1},{"version":"10947bb49601aeec9ea1dddf61ef6e4f8442f949bd40a8008e12b129deb037be","affectsGlobalScope":true,"impliedFormat":1},{"version":"7f2c62938251b45715fd2a9887060ec4fbc8724727029d1cbce373747252bdd7","impliedFormat":1},{"version":"e3ace08b6bbd84655d41e244677b474fd995923ffef7149ddb68af8848b60b05","impliedFormat":1},{"version":"132580b0e86c48fab152bab850fc57a4b74fe915c8958d2ccb052b809a44b61c","impliedFormat":1},{"version":"af4ab0aa8908fc9a655bb833d3bc28e117c4f0e1038c5a891546158beb25accb","impliedFormat":1},{"version":"69c9a5a9392e8564bd81116e1ed93b13205201fb44cb35a7fde8c9f9e21c4b23","impliedFormat":1},{"version":"5f8fc37f8434691ffac1bfd8fc2634647da2c0e84253ab5d2dd19a7718915b35","impliedFormat":1},{"version":"5981c2340fd8b076cae8efbae818d42c11ffc615994cb060b1cd390795f1be2b","impliedFormat":1},{"version":"1641d32611fc7aa82cdd9fa38ff18349aac4eda9e032ced76b21943673887f9a","impliedFormat":1},{"version":"ed4f674fc8c0c993cc7e145069ac44129e03519b910c62be206a0cc777bdc60b","affectsGlobalScope":true,"impliedFormat":1},{"version":"0250da3eb85c99624f974e77ef355cdf86f43980251bc371475c2b397ba55bcd","impliedFormat":1},{"version":"f1c93e046fb3d9b7f8249629f4b63dc068dd839b824dd0aa39a5e68476dc9420","impliedFormat":1},{"version":"3d3a5f27ffbc06c885dd4d5f9ee20de61faf877fe2c3a7051c4825903d9a7fdc","impliedFormat":1},{"version":"12806f9f085598ef930edaf2467a5fa1789a878fba077cd27e85dc5851e11834","impliedFormat":1},{"version":"17d06eb5709839c7ce719f0c38ada6f308fb433f2cd6d8c87b35856e07400950","impliedFormat":1},{"version":"a43fe41c33d0a192a0ecaf9b92e87bef3709c9972e6d53c42c49251ccb962d69","impliedFormat":1},{"version":"a177959203c017fad3ecc4f3d96c8757a840957a4959a3ae00dab9d35961ca6c","affectsGlobalScope":true,"impliedFormat":1},{"version":"6fc727ccf9b36e257ff982ea0badeffbfc2c151802f741bddff00c6af3b784cf","impliedFormat":1},{"version":"2a00d005e3af99cd1cfa75220e60c61b04bfb6be7ca7453bfe2ef6cca37cc03c","impliedFormat":1},{"version":"4844a4c9b4b1e812b257676ed8a80b3f3be0e29bf05e742cc2ea9c3c6865e6c6","impliedFormat":1},{"version":"064878a60367e0407c42fb7ba02a2ea4d83257357dc20088e549bd4d89433e9c","impliedFormat":1},{"version":"14d4bd22d1b05824971b98f7e91b2484c90f1a684805c330476641417c3d9735","impliedFormat":1},{"version":"586eaf66bace2e731cee0ddfbfac326ad74a83c1acfeac4afb2db85ad23226c7","impliedFormat":1},{"version":"b484ec11ba00e3a2235562a41898d55372ccabe607986c6fa4f4aba72093749f","impliedFormat":1},{"version":"d1a14d87cedcf4f0b8173720d6eb29cc02878bf2b6dabf9c9d9cee742f275368","impliedFormat":1},{"version":"41ef7992c555671a8fe54db302788adefa191ded810a50329b79d20a6772d14c","impliedFormat":1},{"version":"041a7781b9127ab568d2cdcce62c58fdea7c7407f40b8c50045d7866a2727130","impliedFormat":1},{"version":"b37f83e7deea729aa9ce5593f78905afb45b7532fdff63041d374f60059e7852","impliedFormat":1},{"version":"e1cb68f3ef3a8dd7b2a9dfb3de482ed6c0f1586ba0db4e7d73c1d2147b6ffc51","impliedFormat":1},{"version":"55cdbeebe76a1fa18bbd7e7bf73350a2173926bd3085bb050cf5a5397025ee4e","impliedFormat":1},{"version":"4fe80f12b1d5189384a219095c2eabadbb389c2d3703aae7c5376dbaa56061df","impliedFormat":1},{"version":"9eb1d2dceae65d1c82fc6be7e9b6b19cf3ca93c364678611107362b6ad4d2d41","impliedFormat":1},{"version":"8065e20ac0ad0536d4f1c8d4c2303272a4d25c450bea8d25deb25697d19300e5","impliedFormat":1},{"version":"cdf4f3357ce01800bffbdf998e6be6acc14f49050de04c9f28878336b0ce3230","impliedFormat":99},{"version":"9480e232e5fba552cb88529379a7d42a7c7ea9cec08000d0267d634d89f0c718","signature":"b342a41f2f9bc333ce0f9aa4e96f6ca819aa556f0dfccff8aa22ff7a864b5954","impliedFormat":99},{"version":"f036a10e83eaac4b74b5363e41abe1b72454a9f58f472e768b46d9d43343ec71","signature":"0a7dd5d4ec5a8405a6f96f2170731b14a1ce4d4f79f71c6b4996acc4fc947435","impliedFormat":99},{"version":"2a6fba1d3c1ce0ddb0bf9ed67a36caa511c65f58ea160f1387700a6c6e85de69","signature":"5da401fbe96adcc40c81037ccf9a7296df94f9ccb2f2ab4edc756e685cd80994","impliedFormat":99},{"version":"1c51c060c46192d5c6825da5b42cc7facd60b2f93a28670279931c44d5f48392","signature":"1f8f2d7a142a46fe78a71e70aa45f7c9bc6fa811c1dfda1461e089f9fcd8e05a","impliedFormat":99},{"version":"acfb723d81eda39156251aed414c553294870bf53062429ebfcfba8a68cb4753","impliedFormat":99},{"version":"fa69a90381c2f85889722a911a732a5ee3596dc3acecda8a9aa2fa89b9615d8d","impliedFormat":99},{"version":"b5ce343886d23392be9c8280e9f24a87f1d7d3667f6672c2fe4aa61fa4ece7d4","impliedFormat":99},{"version":"57e9e1b0911874c62d743af24b5d56032759846533641d550b12a45ff404bf07","impliedFormat":99},{"version":"b0857bb28fd5236ace84280f79a25093f919fd0eff13e47cc26ea03de60a7294","impliedFormat":99},{"version":"5e43e0824f10cd8c48e7a8c5c673638488925a12c31f0f9e0957965c290eb14c","impliedFormat":99},{"version":"854cd3a3375ffc4e7a92b2168dd065d7ff2614b43341038a65cca865a44c00c5","impliedFormat":99},{"version":"ef13c73d6157a32933c612d476c1524dd674cf5b9a88571d7d6a0d147544d529","impliedFormat":99},{"version":"3b0a56d056d81a011e484b9c05d5e430711aaecd561a788bad1d0498aad782c7","impliedFormat":99},{"version":"2f863ee9b873a65d9c3338ea7aaddbdb41a9673f062f06983d712bd01c25dc6b","impliedFormat":99},{"version":"67aa128c2bc170b93794f191feffc65a4b33e878db211cfcb7658c4b72f7a1f5","impliedFormat":99},{"version":"ac3d263474022e9a14c43f588f485d549641d839b159ecc971978b90f34bdf6b","impliedFormat":99},{"version":"a7ca8df4f2931bef2aa4118078584d84a0b16539598eaadf7dce9104dfaa381c","impliedFormat":1},{"version":"10073cdcf56982064c5337787cc59b79586131e1b28c106ede5bff362f912b70","impliedFormat":99},{"version":"72950913f4900b680f44d8cab6dd1ea0311698fc1eefb014eb9cdfc37ac4a734","impliedFormat":1},{"version":"36977c14a7f7bfc8c0426ae4343875689949fb699f3f84ecbe5b300ebf9a2c55","impliedFormat":1},{"version":"ff0a83c9a0489a627e264ffcb63f2264b935b20a502afa3a018848139e3d8575","impliedFormat":99},{"version":"324ac98294dab54fbd580c7d0e707d94506d7b2c3d5efe981a8495f02cf9ad96","impliedFormat":99},{"version":"9ec72eb493ff209b470467e24264116b6a8616484bca438091433a545dfba17e","impliedFormat":99},{"version":"c35b8117804c639c53c87f2c23e0c786df61d552e513bd5179f5b88e29964838","impliedFormat":99},{"version":"c609331c6ed4ad4af54e101088c6a4dcb48f8db7b0b97e44a6efeb130f4331bd","impliedFormat":99},{"version":"bcbd3becd08b4515225880abea0dbfbbf0d1181ce3af8f18f72f61edbe4febfb","impliedFormat":99},{"version":"67acaedb46832d66c15f1b09fb7b6a0b7f41bdbf8eaa586ec70459b3e8896eb9","impliedFormat":99},{"version":"4535ab977ee871e956eb7bebe2db5de79f5d5ec7dfbbf1d35e08f4a2d6630dac","impliedFormat":99},{"version":"b79b5ed99f26ffb2f8ae4bdcc4b34a9542197dc3fa96cfb425c2a81e618cff28","impliedFormat":99},{"version":"31fd7c12f6e27154efb52a916b872509a771880f3b20f2dfd045785c13aa813f","impliedFormat":99},{"version":"b481de4ab5379bd481ca12fc0b255cdc47341629a22c240a89cdb4e209522be2","impliedFormat":99},{"version":"bdd14f07b4eca0b4b5203b85b8dbc4d084c749fa590bee5ea613e1641dcd3b29","impliedFormat":99},{"version":"427fe2004642504828c1476d0af4270e6ad4db6de78c0b5da3e4c5ca95052a99","impliedFormat":1},{"version":"2eeffcee5c1661ddca53353929558037b8cf305ffb86a803512982f99bcab50d","impliedFormat":99},{"version":"9afb4cb864d297e4092a79ee2871b5d3143ea14153f62ef0bb04ede25f432030","affectsGlobalScope":true,"impliedFormat":99},{"version":"4e258d11c899cb9ff36b4b5c53df59cf4a5ccae9a9931529686e77431e0a3518","affectsGlobalScope":true,"impliedFormat":99},{"version":"a5ae67a67f786ffe92d34b55467a40fb50fb0093e92388cadce6168fa42690fd","impliedFormat":99},{"version":"69bf2422313487956e4dacf049f30cb91b34968912058d244cb19e4baa24da97","impliedFormat":99},{"version":"6987dfb4b0c4e02112cc4e548e7a77b3d9ddfeffa8c8a2db13ceac361a4567d9","impliedFormat":99},{"version":"a534e61c2f06a147d97aebad720db97dffd8066b7142212e46bcbcdcb640b81a","impliedFormat":99},{"version":"ddf569d04470a4d629090d43a16735185001f3fcf0ae036ead99f2ceab62be48","impliedFormat":99},{"version":"b413fbc6658fe2774f8bf9a15cf4c53e586fc38a2d5256b3b9647da242c14389","impliedFormat":99},{"version":"c30a41267fc04c6518b17e55dcb2b810f267af4314b0b6d7df1c33a76ce1b330","impliedFormat":1},{"version":"72422d0bac4076912385d0c10911b82e4694fc106e2d70added091f88f0824ba","impliedFormat":1},{"version":"da251b82c25bee1d93f9fd80c5a61d945da4f708ca21285541d7aff83ecb8200","impliedFormat":1},{"version":"64db14db2bf37ac089766fdb3c7e1160fabc10e9929bc2deeede7237e4419fc8","impliedFormat":1},{"version":"98b94085c9f78eba36d3d2314affe973e8994f99864b8708122750788825c771","impliedFormat":1},{"version":"53c448183c7177c83d3eb0b40824cf8952721a6584cf22052adc24f778986732","impliedFormat":99},{"version":"9d95b66b0f6b76afd843902c64692dd86dcec59a08268cdbfb77470718aed277","signature":"8e609bb71c20b858c77f0e9f90bb1319db8477b13f9f965f1a1e18524bf50881","impliedFormat":99}],"root":[64,[190,193],238],"options":{"composite":true,"declaration":true,"declarationMap":true,"esModuleInterop":true,"module":199,"outDir":"./dist","rootDir":"./src","skipLibCheck":true,"sourceMap":true,"strict":true,"target":10},"referencedMap":[[221,1],[224,2],[222,1],[126,3],[127,3],[128,4],[67,5],[129,6],[130,7],[131,8],[65,1],[132,9],[133,10],[134,11],[135,12],[136,13],[137,14],[138,14],[139,15],[140,16],[141,17],[142,18],[68,1],[66,1],[143,19],[144,20],[145,21],[185,22],[146,23],[147,24],[148,23],[149,25],[150,26],[151,27],[152,28],[153,28],[154,28],[155,29],[156,30],[157,31],[158,32],[159,33],[160,34],[161,34],[162,35],[163,1],[164,1],[165,36],[166,37],[167,36],[168,38],[169,39],[170,40],[171,41],[172,42],[173,43],[174,44],[175,45],[176,46],[177,47],[178,48],[179,49],[180,50],[181,51],[182,52],[69,23],[70,1],[71,53],[72,54],[73,1],[74,55],[75,1],[117,56],[118,57],[119,58],[120,58],[121,59],[122,1],[123,6],[124,60],[125,57],[183,61],[184,62],[225,63],[194,1],[204,64],[200,65],[203,66],[226,67],[211,1],[213,68],[212,69],[219,1],[202,70],[195,71],[197,72],[199,73],[198,1],[201,71],[196,1],[223,1],[234,74],[236,75],[235,76],[233,77],[232,1],[186,78],[188,79],[187,80],[227,1],[220,1],[62,1],[63,1],[12,1],[11,1],[2,1],[13,1],[14,1],[15,1],[16,1],[17,1],[18,1],[19,1],[20,1],[3,1],[21,1],[22,1],[4,1],[23,1],[27,1],[24,1],[25,1],[26,1],[28,1],[29,1],[30,1],[5,1],[31,1],[32,1],[33,1],[34,1],[6,1],[38,1],[35,1],[36,1],[37,1],[39,1],[7,1],[40,1],[45,1],[46,1],[41,1],[42,1],[43,1],[44,1],[8,1],[50,1],[47,1],[48,1],[49,1],[51,1],[9,1],[52,1],[53,1],[54,1],[56,1],[55,1],[57,1],[58,1],[10,1],[59,1],[1,1],[60,1],[61,1],[93,81],[105,82],[91,83],[106,84],[115,85],[82,86],[83,87],[81,88],[114,89],[109,90],[113,91],[85,92],[102,93],[84,94],[112,95],[79,96],[80,90],[86,97],[87,1],[92,98],[90,97],[77,99],[116,100],[107,101],[96,102],[95,97],[97,103],[100,104],[94,105],[98,106],[110,89],[88,107],[89,108],[101,109],[78,84],[104,110],[103,97],[99,111],[108,1],[76,1],[111,112],[207,113],[210,114],[208,113],[206,1],[209,115],[228,116],[218,117],[214,118],[215,65],[231,119],[229,120],[216,121],[230,122],[205,1],[217,123],[237,124],[238,125],[193,126],[191,127],[190,128],[192,129],[64,1],[189,130]],"latestChangedDtsFile":"./dist/__tests__/bridge.test.d.ts","version":"5.9.3"}
|