@privateclaw/privateclaw-relay 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (42) hide show
  1. package/.env.example +13 -0
  2. package/README.md +86 -0
  3. package/dist/cli-error.d.ts +3 -0
  4. package/dist/cli-error.js +7 -0
  5. package/dist/cli-error.js.map +1 -0
  6. package/dist/cli.d.ts +2 -0
  7. package/dist/cli.js +14 -0
  8. package/dist/cli.js.map +1 -0
  9. package/dist/config.d.ts +13 -0
  10. package/dist/config.js +36 -0
  11. package/dist/config.js.map +1 -0
  12. package/dist/frame-cache.d.ts +77 -0
  13. package/dist/frame-cache.js +127 -0
  14. package/dist/frame-cache.js.map +1 -0
  15. package/dist/index.d.ts +4 -0
  16. package/dist/index.js +5 -0
  17. package/dist/index.js.map +1 -0
  18. package/dist/push-notifier.d.ts +36 -0
  19. package/dist/push-notifier.js +198 -0
  20. package/dist/push-notifier.js.map +1 -0
  21. package/dist/push-registration-store.d.ts +39 -0
  22. package/dist/push-registration-store.js +98 -0
  23. package/dist/push-registration-store.js.map +1 -0
  24. package/dist/relay-cli.d.ts +28 -0
  25. package/dist/relay-cli.js +256 -0
  26. package/dist/relay-cli.js.map +1 -0
  27. package/dist/relay-cluster.d.ts +144 -0
  28. package/dist/relay-cluster.js +436 -0
  29. package/dist/relay-cluster.js.map +1 -0
  30. package/dist/relay-server.d.ts +25 -0
  31. package/dist/relay-server.js +1090 -0
  32. package/dist/relay-server.js.map +1 -0
  33. package/dist/session-store.d.ts +40 -0
  34. package/dist/session-store.js +159 -0
  35. package/dist/session-store.js.map +1 -0
  36. package/dist/tunnel-installer.d.ts +36 -0
  37. package/dist/tunnel-installer.js +402 -0
  38. package/dist/tunnel-installer.js.map +1 -0
  39. package/dist/tunnel.d.ts +35 -0
  40. package/dist/tunnel.js +334 -0
  41. package/dist/tunnel.js.map +1 -0
  42. package/package.json +45 -0
@@ -0,0 +1,436 @@
1
+ import { Redis } from "ioredis";
2
+ const BUS_PREFIX = "privateclaw:bus:v1";
3
+ const PRESENCE_PREFIX = "privateclaw:presence:v1";
4
+ const LEASE_MS = 45_000;
5
+ function appBroadcastChannel(sessionId) {
6
+ return `${BUS_PREFIX}:session:${sessionId}:app:broadcast`;
7
+ }
8
+ function appTargetChannel(sessionId, appId) {
9
+ return `${BUS_PREFIX}:session:${sessionId}:app:${appId}`;
10
+ }
11
+ function providerChannel(sessionId) {
12
+ return `${BUS_PREFIX}:session:${sessionId}:provider`;
13
+ }
14
+ function sessionControlChannel(sessionId) {
15
+ return `${BUS_PREFIX}:session:${sessionId}:control`;
16
+ }
17
+ function providerControlChannel(providerId) {
18
+ return `${BUS_PREFIX}:provider:${providerId}:control`;
19
+ }
20
+ function providerPresenceKey(providerId) {
21
+ return `${PRESENCE_PREFIX}:provider:${providerId}`;
22
+ }
23
+ function appPresenceKey(sessionId, appId) {
24
+ return `${PRESENCE_PREFIX}:session:${sessionId}:app:${appId}`;
25
+ }
26
+ function occupantKey(sessionId) {
27
+ return `${PRESENCE_PREFIX}:session:${sessionId}:occupant`;
28
+ }
29
+ function occupantValue(nodeId, appId) {
30
+ return JSON.stringify({ nodeId, appId });
31
+ }
32
+ function parseOccupant(raw) {
33
+ return JSON.parse(raw);
34
+ }
35
+ async function compareAndDelete(redis, key, expectedValue) {
36
+ await redis.eval(`
37
+ if redis.call("get", KEYS[1]) == ARGV[1] then
38
+ return redis.call("del", KEYS[1])
39
+ end
40
+ return 0
41
+ `, 1, key, expectedValue);
42
+ }
43
+ export class RelayClaimConflictError extends Error {
44
+ constructor(message) {
45
+ super(message);
46
+ this.name = "RelayClaimConflictError";
47
+ }
48
+ }
49
+ class BaseRelayClusterClient {
50
+ nodeId;
51
+ callbacks;
52
+ persistent = true;
53
+ channelRefs = new Map();
54
+ constructor(nodeId, callbacks) {
55
+ this.nodeId = nodeId;
56
+ this.callbacks = callbacks;
57
+ }
58
+ async subscribeProvider(providerId, sessionId) {
59
+ await this.retainChannel(providerChannel(sessionId));
60
+ await this.retainChannel(sessionControlChannel(sessionId));
61
+ await this.retainChannel(providerControlChannel(providerId));
62
+ }
63
+ async unsubscribeProvider(providerId, sessionId) {
64
+ await this.releaseChannel(providerChannel(sessionId));
65
+ await this.releaseChannel(sessionControlChannel(sessionId));
66
+ await this.releaseChannel(providerControlChannel(providerId));
67
+ }
68
+ async subscribeApp(sessionId, appId) {
69
+ await this.retainChannel(appBroadcastChannel(sessionId));
70
+ await this.retainChannel(appTargetChannel(sessionId, appId));
71
+ await this.retainChannel(sessionControlChannel(sessionId));
72
+ }
73
+ async unsubscribeApp(sessionId, appId) {
74
+ await this.releaseChannel(appBroadcastChannel(sessionId));
75
+ await this.releaseChannel(appTargetChannel(sessionId, appId));
76
+ await this.releaseChannel(sessionControlChannel(sessionId));
77
+ }
78
+ async publishFrameToApp(sessionId, envelope, targetAppId) {
79
+ return this.publish(targetAppId
80
+ ? appTargetChannel(sessionId, targetAppId)
81
+ : appBroadcastChannel(sessionId), JSON.stringify({
82
+ originNodeId: this.nodeId,
83
+ sessionId,
84
+ envelope,
85
+ }));
86
+ }
87
+ async publishFrameToProvider(sessionId, envelope) {
88
+ return this.publish(providerChannel(sessionId), JSON.stringify({
89
+ originNodeId: this.nodeId,
90
+ sessionId,
91
+ envelope,
92
+ }));
93
+ }
94
+ async publishSessionClosed(sessionId, reason) {
95
+ await this.publish(sessionControlChannel(sessionId), JSON.stringify({
96
+ kind: "session_closed",
97
+ originNodeId: this.nodeId,
98
+ sessionId,
99
+ reason,
100
+ }));
101
+ }
102
+ async publishAppClosed(sessionId, appId, reason) {
103
+ await this.publish(sessionControlChannel(sessionId), JSON.stringify({
104
+ kind: "app_closed",
105
+ originNodeId: this.nodeId,
106
+ sessionId,
107
+ appId,
108
+ reason,
109
+ }));
110
+ }
111
+ async publishAppReconnected(sessionId, appId, targetNodeId) {
112
+ await this.publish(sessionControlChannel(sessionId), JSON.stringify({
113
+ kind: "app_reconnected",
114
+ originNodeId: this.nodeId,
115
+ targetNodeId,
116
+ sessionId,
117
+ appId,
118
+ }));
119
+ }
120
+ async publishProviderReconnected(providerId, targetNodeId) {
121
+ await this.publish(providerControlChannel(providerId), JSON.stringify({
122
+ kind: "provider_reconnected",
123
+ originNodeId: this.nodeId,
124
+ targetNodeId,
125
+ providerId,
126
+ }));
127
+ }
128
+ async handleChannelMessage(channel, payload) {
129
+ if (channel.startsWith(`${BUS_PREFIX}:provider:`)) {
130
+ const providerMatch = channel.match(/^privateclaw:bus:v1:provider:([^:]+):control$/);
131
+ if (!providerMatch) {
132
+ return;
133
+ }
134
+ const message = JSON.parse(payload);
135
+ if (message.originNodeId === this.nodeId ||
136
+ message.targetNodeId !== this.nodeId) {
137
+ return;
138
+ }
139
+ await this.callbacks.onRemoteProviderReconnected(message.providerId);
140
+ return;
141
+ }
142
+ const broadcastMatch = channel.match(/^privateclaw:bus:v1:session:([^:]+):app:broadcast$/);
143
+ if (broadcastMatch) {
144
+ const message = JSON.parse(payload);
145
+ if (message.originNodeId === this.nodeId) {
146
+ return;
147
+ }
148
+ await this.callbacks.onRemoteAppFrame(message.sessionId, message.envelope);
149
+ return;
150
+ }
151
+ const targetedAppMatch = channel.match(/^privateclaw:bus:v1:session:([^:]+):app:([^:]+)$/);
152
+ if (targetedAppMatch) {
153
+ const message = JSON.parse(payload);
154
+ if (message.originNodeId === this.nodeId) {
155
+ return;
156
+ }
157
+ await this.callbacks.onRemoteAppFrame(message.sessionId, message.envelope, targetedAppMatch[2]);
158
+ return;
159
+ }
160
+ const providerFrameMatch = channel.match(/^privateclaw:bus:v1:session:([^:]+):provider$/);
161
+ if (providerFrameMatch) {
162
+ const message = JSON.parse(payload);
163
+ if (message.originNodeId === this.nodeId) {
164
+ return;
165
+ }
166
+ await this.callbacks.onRemoteProviderFrame(message.sessionId, message.envelope);
167
+ return;
168
+ }
169
+ const controlMatch = channel.match(/^privateclaw:bus:v1:session:([^:]+):control$/);
170
+ if (!controlMatch) {
171
+ return;
172
+ }
173
+ const message = JSON.parse(payload);
174
+ if (message.originNodeId === this.nodeId) {
175
+ return;
176
+ }
177
+ if (message.kind === "session_closed") {
178
+ await this.callbacks.onRemoteSessionClosed(message.sessionId, message.reason);
179
+ return;
180
+ }
181
+ if (message.kind === "app_closed") {
182
+ await this.callbacks.onRemoteAppClosed(message.sessionId, message.appId, message.reason);
183
+ return;
184
+ }
185
+ if (message.targetNodeId !== this.nodeId) {
186
+ return;
187
+ }
188
+ await this.callbacks.onRemoteAppReconnected(message.sessionId, message.appId);
189
+ }
190
+ async retainChannel(channel) {
191
+ const current = this.channelRefs.get(channel) ?? 0;
192
+ this.channelRefs.set(channel, current + 1);
193
+ if (current > 0) {
194
+ return;
195
+ }
196
+ await this.subscribeChannel(channel);
197
+ }
198
+ async releaseChannel(channel) {
199
+ const current = this.channelRefs.get(channel);
200
+ if (!current) {
201
+ return;
202
+ }
203
+ if (current > 1) {
204
+ this.channelRefs.set(channel, current - 1);
205
+ return;
206
+ }
207
+ this.channelRefs.delete(channel);
208
+ await this.unsubscribeChannel(channel);
209
+ }
210
+ }
211
+ export function createInMemoryRelayClusterSharedState() {
212
+ return {
213
+ channelSubscribers: new Map(),
214
+ providerPresence: new Map(),
215
+ appPresence: new Map(),
216
+ singleSessionOccupants: new Map(),
217
+ };
218
+ }
219
+ export class InMemoryRelayClusterClient extends BaseRelayClusterClient {
220
+ shared;
221
+ constructor(shared, params) {
222
+ super(params.nodeId, params.callbacks);
223
+ this.shared = shared;
224
+ }
225
+ async claimProvider(providerId) {
226
+ const previousNodeId = this.shared.providerPresence.get(providerId);
227
+ this.shared.providerPresence.set(providerId, this.nodeId);
228
+ return previousNodeId && previousNodeId !== this.nodeId
229
+ ? { previousNodeId }
230
+ : {};
231
+ }
232
+ async releaseProvider(providerId) {
233
+ if (this.shared.providerPresence.get(providerId) === this.nodeId) {
234
+ this.shared.providerPresence.delete(providerId);
235
+ }
236
+ }
237
+ async claimApp(binding) {
238
+ if (!binding.groupMode) {
239
+ const currentOccupant = this.shared.singleSessionOccupants.get(binding.sessionId);
240
+ if (currentOccupant) {
241
+ const current = parseOccupant(currentOccupant);
242
+ if (current.appId !== binding.appId) {
243
+ throw new RelayClaimConflictError("This PrivateClaw session is already attached to another app.");
244
+ }
245
+ }
246
+ this.shared.singleSessionOccupants.set(binding.sessionId, occupantValue(this.nodeId, binding.appId));
247
+ }
248
+ const key = `${binding.sessionId}:${binding.appId}`;
249
+ const previousNodeId = this.shared.appPresence.get(key);
250
+ this.shared.appPresence.set(key, this.nodeId);
251
+ return previousNodeId && previousNodeId !== this.nodeId
252
+ ? { previousNodeId }
253
+ : {};
254
+ }
255
+ async releaseApp(binding) {
256
+ const key = `${binding.sessionId}:${binding.appId}`;
257
+ if (this.shared.appPresence.get(key) === this.nodeId) {
258
+ this.shared.appPresence.delete(key);
259
+ }
260
+ if (!binding.groupMode) {
261
+ const currentOccupant = this.shared.singleSessionOccupants.get(binding.sessionId);
262
+ if (currentOccupant === occupantValue(this.nodeId, binding.appId)) {
263
+ this.shared.singleSessionOccupants.delete(binding.sessionId);
264
+ }
265
+ }
266
+ }
267
+ async refreshPresence(params) {
268
+ for (const providerId of params.providerIds) {
269
+ this.shared.providerPresence.set(providerId, this.nodeId);
270
+ }
271
+ for (const binding of params.appBindings) {
272
+ this.shared.appPresence.set(`${binding.sessionId}:${binding.appId}`, this.nodeId);
273
+ if (!binding.groupMode) {
274
+ this.shared.singleSessionOccupants.set(binding.sessionId, occupantValue(this.nodeId, binding.appId));
275
+ }
276
+ }
277
+ }
278
+ async subscribeChannel(channel) {
279
+ const subscribers = this.shared.channelSubscribers.get(channel) ??
280
+ new Set();
281
+ subscribers.add(this);
282
+ this.shared.channelSubscribers.set(channel, subscribers);
283
+ }
284
+ async unsubscribeChannel(channel) {
285
+ const subscribers = this.shared.channelSubscribers.get(channel);
286
+ if (!subscribers) {
287
+ return;
288
+ }
289
+ subscribers.delete(this);
290
+ if (subscribers.size === 0) {
291
+ this.shared.channelSubscribers.delete(channel);
292
+ }
293
+ }
294
+ async publish(channel, payload) {
295
+ const subscribers = [...(this.shared.channelSubscribers.get(channel) ?? [])];
296
+ await Promise.all(subscribers.map((subscriber) => subscriber.handleChannelMessage(channel, payload)));
297
+ return subscribers.length;
298
+ }
299
+ async close() {
300
+ for (const [channel, subscribers] of this.shared.channelSubscribers.entries()) {
301
+ if (!subscribers.delete(this)) {
302
+ continue;
303
+ }
304
+ if (subscribers.size === 0) {
305
+ this.shared.channelSubscribers.delete(channel);
306
+ }
307
+ }
308
+ }
309
+ async hasAppBinding(sessionId, appId) {
310
+ return this.shared.appPresence.has(`${sessionId}:${appId}`);
311
+ }
312
+ async hasProviderSessionSubscriber(sessionId) {
313
+ return (this.shared.channelSubscribers.get(providerChannel(sessionId))?.size ?? 0) > 0;
314
+ }
315
+ }
316
+ export class RedisRelayClusterClient extends BaseRelayClusterClient {
317
+ redis;
318
+ subscriber;
319
+ constructor(redisUrl, params) {
320
+ super(params.nodeId, params.callbacks);
321
+ this.redis = new Redis(redisUrl, {
322
+ lazyConnect: false,
323
+ maxRetriesPerRequest: 1,
324
+ });
325
+ this.subscriber = this.redis.duplicate({
326
+ lazyConnect: false,
327
+ maxRetriesPerRequest: 1,
328
+ });
329
+ this.redis.on("error", (error) => {
330
+ console.error("[privateclaw-relay] relay cluster Redis error", error);
331
+ });
332
+ this.subscriber.on("error", (error) => {
333
+ console.error("[privateclaw-relay] relay cluster subscriber error", error);
334
+ });
335
+ this.subscriber.on("message", (channel, payload) => {
336
+ void this.handleChannelMessage(channel, payload).catch((error) => {
337
+ console.error("[privateclaw-relay] failed to process cluster message", error);
338
+ });
339
+ });
340
+ }
341
+ async claimProvider(providerId) {
342
+ const key = providerPresenceKey(providerId);
343
+ const previousNodeId = await this.redis.get(key);
344
+ await this.redis.set(key, this.nodeId, "PX", LEASE_MS);
345
+ return previousNodeId && previousNodeId !== this.nodeId
346
+ ? { previousNodeId }
347
+ : {};
348
+ }
349
+ async releaseProvider(providerId) {
350
+ await compareAndDelete(this.redis, providerPresenceKey(providerId), this.nodeId);
351
+ }
352
+ async claimApp(binding) {
353
+ const presenceKey = appPresenceKey(binding.sessionId, binding.appId);
354
+ const previousNodeId = await this.redis.get(presenceKey);
355
+ if (!binding.groupMode) {
356
+ const occupancyKey = occupantKey(binding.sessionId);
357
+ const expectedOccupantValue = occupantValue(this.nodeId, binding.appId);
358
+ for (let attempt = 0; attempt < 2; attempt += 1) {
359
+ const created = await this.redis.set(occupancyKey, expectedOccupantValue, "PX", LEASE_MS, "NX");
360
+ if (created !== null) {
361
+ break;
362
+ }
363
+ const currentOccupant = await this.redis.get(occupancyKey);
364
+ if (!currentOccupant) {
365
+ continue;
366
+ }
367
+ const parsed = parseOccupant(currentOccupant);
368
+ if (parsed.appId !== binding.appId) {
369
+ throw new RelayClaimConflictError("This PrivateClaw session is already attached to another app.");
370
+ }
371
+ await this.redis.set(occupancyKey, expectedOccupantValue, "PX", LEASE_MS, "XX");
372
+ break;
373
+ }
374
+ }
375
+ await this.redis.set(presenceKey, this.nodeId, "PX", LEASE_MS);
376
+ return previousNodeId && previousNodeId !== this.nodeId
377
+ ? { previousNodeId }
378
+ : {};
379
+ }
380
+ async releaseApp(binding) {
381
+ await compareAndDelete(this.redis, appPresenceKey(binding.sessionId, binding.appId), this.nodeId);
382
+ if (!binding.groupMode) {
383
+ await compareAndDelete(this.redis, occupantKey(binding.sessionId), occupantValue(this.nodeId, binding.appId));
384
+ }
385
+ }
386
+ async refreshPresence(params) {
387
+ const pipeline = this.redis.multi();
388
+ for (const providerId of params.providerIds) {
389
+ pipeline.set(providerPresenceKey(providerId), this.nodeId, "PX", LEASE_MS);
390
+ }
391
+ for (const binding of params.appBindings) {
392
+ pipeline.set(appPresenceKey(binding.sessionId, binding.appId), this.nodeId, "PX", LEASE_MS);
393
+ if (!binding.groupMode) {
394
+ pipeline.set(occupantKey(binding.sessionId), occupantValue(this.nodeId, binding.appId), "PX", LEASE_MS);
395
+ }
396
+ }
397
+ await pipeline.exec();
398
+ }
399
+ async subscribeChannel(channel) {
400
+ await this.subscriber.subscribe(channel);
401
+ }
402
+ async unsubscribeChannel(channel) {
403
+ await this.subscriber.unsubscribe(channel);
404
+ }
405
+ async publish(channel, payload) {
406
+ return this.redis.publish(channel, payload);
407
+ }
408
+ async close() {
409
+ await this.subscriber.quit();
410
+ await this.redis.quit();
411
+ }
412
+ async hasAppBinding(sessionId, appId) {
413
+ return (await this.redis.exists(appPresenceKey(sessionId, appId))) === 1;
414
+ }
415
+ async hasProviderSessionSubscriber(sessionId) {
416
+ const result = await this.redis.call("PUBSUB", "NUMSUB", providerChannel(sessionId));
417
+ if (!Array.isArray(result)) {
418
+ return false;
419
+ }
420
+ const subscriberCount = Number.parseInt(String(result[1] ?? "0"), 10);
421
+ return Number.isFinite(subscriberCount) && subscriberCount > 0;
422
+ }
423
+ }
424
+ export function createRedisRelayClusterClient(params) {
425
+ return new RedisRelayClusterClient(params.redisUrl, {
426
+ nodeId: params.nodeId,
427
+ callbacks: params.callbacks,
428
+ });
429
+ }
430
+ export function createInMemoryRelayClusterClient(params) {
431
+ return new InMemoryRelayClusterClient(params.shared, {
432
+ nodeId: params.nodeId,
433
+ callbacks: params.callbacks,
434
+ });
435
+ }
436
+ //# sourceMappingURL=relay-cluster.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"relay-cluster.js","sourceRoot":"","sources":["../src/relay-cluster.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,25 @@
1
+ import type { RelayServerConfig } from "./config.js";
2
+ import { type EncryptedFrameCache } from "./frame-cache.js";
3
+ import { type RelayPushRegistrationStore } from "./push-registration-store.js";
4
+ import { type RelayPushNotifier } from "./push-notifier.js";
5
+ import { type RelayClusterCallbacks, type RelayClusterClient } from "./relay-cluster.js";
6
+ import { type RelaySessionStore } from "./session-store.js";
7
+ export interface RelayServerInstance {
8
+ readonly port: number;
9
+ readonly url: string;
10
+ start(): Promise<{
11
+ port: number;
12
+ url: string;
13
+ }>;
14
+ stop(): Promise<void>;
15
+ }
16
+ export interface RelayServerDependencies {
17
+ frameCache?: EncryptedFrameCache;
18
+ sessionStore?: RelaySessionStore;
19
+ pushRegistrationStore?: RelayPushRegistrationStore;
20
+ pushNotifier?: RelayPushNotifier;
21
+ cluster?: RelayClusterClient;
22
+ clusterFactory?: (callbacks: RelayClusterCallbacks) => RelayClusterClient;
23
+ now?: () => number;
24
+ }
25
+ export declare function createRelayServer(config: RelayServerConfig, deps?: RelayServerDependencies): RelayServerInstance;