@syncular/relay 0.0.1-60

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 (68) hide show
  1. package/dist/client-role/forward-engine.d.ts +63 -0
  2. package/dist/client-role/forward-engine.d.ts.map +1 -0
  3. package/dist/client-role/forward-engine.js +263 -0
  4. package/dist/client-role/forward-engine.js.map +1 -0
  5. package/dist/client-role/index.d.ts +9 -0
  6. package/dist/client-role/index.d.ts.map +1 -0
  7. package/dist/client-role/index.js +9 -0
  8. package/dist/client-role/index.js.map +1 -0
  9. package/dist/client-role/pull-engine.d.ts +70 -0
  10. package/dist/client-role/pull-engine.d.ts.map +1 -0
  11. package/dist/client-role/pull-engine.js +233 -0
  12. package/dist/client-role/pull-engine.js.map +1 -0
  13. package/dist/client-role/sequence-mapper.d.ts +65 -0
  14. package/dist/client-role/sequence-mapper.d.ts.map +1 -0
  15. package/dist/client-role/sequence-mapper.js +161 -0
  16. package/dist/client-role/sequence-mapper.js.map +1 -0
  17. package/dist/index.d.ts +37 -0
  18. package/dist/index.d.ts.map +1 -0
  19. package/dist/index.js +44 -0
  20. package/dist/index.js.map +1 -0
  21. package/dist/migrate.d.ts +18 -0
  22. package/dist/migrate.d.ts.map +1 -0
  23. package/dist/migrate.js +99 -0
  24. package/dist/migrate.js.map +1 -0
  25. package/dist/mode-manager.d.ts +60 -0
  26. package/dist/mode-manager.d.ts.map +1 -0
  27. package/dist/mode-manager.js +114 -0
  28. package/dist/mode-manager.js.map +1 -0
  29. package/dist/realtime.d.ts +102 -0
  30. package/dist/realtime.d.ts.map +1 -0
  31. package/dist/realtime.js +305 -0
  32. package/dist/realtime.js.map +1 -0
  33. package/dist/relay.d.ts +188 -0
  34. package/dist/relay.d.ts.map +1 -0
  35. package/dist/relay.js +315 -0
  36. package/dist/relay.js.map +1 -0
  37. package/dist/schema.d.ts +158 -0
  38. package/dist/schema.d.ts.map +1 -0
  39. package/dist/schema.js +7 -0
  40. package/dist/schema.js.map +1 -0
  41. package/dist/server-role/index.d.ts +54 -0
  42. package/dist/server-role/index.d.ts.map +1 -0
  43. package/dist/server-role/index.js +198 -0
  44. package/dist/server-role/index.js.map +1 -0
  45. package/dist/server-role/pull.d.ts +25 -0
  46. package/dist/server-role/pull.d.ts.map +1 -0
  47. package/dist/server-role/pull.js +24 -0
  48. package/dist/server-role/pull.js.map +1 -0
  49. package/dist/server-role/push.d.ts +27 -0
  50. package/dist/server-role/push.d.ts.map +1 -0
  51. package/dist/server-role/push.js +98 -0
  52. package/dist/server-role/push.js.map +1 -0
  53. package/package.json +61 -0
  54. package/src/__tests__/relay.test.ts +464 -0
  55. package/src/bun-types.d.ts +50 -0
  56. package/src/client-role/forward-engine.ts +352 -0
  57. package/src/client-role/index.ts +9 -0
  58. package/src/client-role/pull-engine.ts +301 -0
  59. package/src/client-role/sequence-mapper.ts +201 -0
  60. package/src/index.ts +50 -0
  61. package/src/migrate.ts +113 -0
  62. package/src/mode-manager.ts +142 -0
  63. package/src/realtime.ts +370 -0
  64. package/src/relay.ts +421 -0
  65. package/src/schema.ts +171 -0
  66. package/src/server-role/index.ts +342 -0
  67. package/src/server-role/pull.ts +37 -0
  68. package/src/server-role/push.ts +130 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pull-engine.js","sourceRoot":"","sources":["../../src/client-role/pull-engine.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,65 @@
1
+ /**
2
+ * @syncular/relay - Sequence Mapper
3
+ *
4
+ * Tracks the mapping between relay's local commit_seq
5
+ * and the main server's global commit_seq.
6
+ */
7
+ import { type Kysely } from 'kysely';
8
+ import type { RelayDatabase, RelaySequenceMapStatus } from '../schema';
9
+ /**
10
+ * Sequence mapping entry.
11
+ */
12
+ export interface SequenceMapping {
13
+ localCommitSeq: number;
14
+ mainCommitSeq: number | null;
15
+ status: RelaySequenceMapStatus;
16
+ }
17
+ /**
18
+ * Sequence mapper for tracking local to main commit sequence mappings.
19
+ */
20
+ export declare class SequenceMapper<DB extends RelayDatabase = RelayDatabase> {
21
+ private readonly db;
22
+ constructor(options: {
23
+ db: Kysely<DB>;
24
+ });
25
+ /**
26
+ * Create a pending mapping for a local commit that will be forwarded.
27
+ */
28
+ createPendingMapping(localCommitSeq: number): Promise<void>;
29
+ /**
30
+ * Mark a mapping as forwarded with the main server's commit_seq.
31
+ */
32
+ markForwarded(localCommitSeq: number, mainCommitSeq: number): Promise<void>;
33
+ /**
34
+ * Mark a mapping as confirmed (main server acknowledged).
35
+ */
36
+ markConfirmed(localCommitSeq: number): Promise<void>;
37
+ /**
38
+ * Get the mapping for a local commit sequence.
39
+ */
40
+ getMapping(localCommitSeq: number): Promise<SequenceMapping | null>;
41
+ /**
42
+ * Get the local commit sequence for a main server commit sequence.
43
+ */
44
+ getLocalCommitSeq(mainCommitSeq: number): Promise<number | null>;
45
+ /**
46
+ * Get all pending mappings (commits not yet forwarded).
47
+ */
48
+ getPendingMappings(): Promise<SequenceMapping[]>;
49
+ /**
50
+ * Create a mapping for commits pulled from main (assigned new local commit_seq).
51
+ *
52
+ * These mappings go directly to 'confirmed' status since they came from main.
53
+ */
54
+ createConfirmedMapping(localCommitSeq: number, mainCommitSeq: number): Promise<void>;
55
+ /**
56
+ * Delete confirmed/forwarded sequence mappings older than the given age.
57
+ * Keeps pending mappings (they haven't been forwarded yet).
58
+ */
59
+ pruneOldMappings(maxAgeMs: number): Promise<number>;
60
+ /**
61
+ * Get the highest main_commit_seq we've seen (for tracking pull cursor).
62
+ */
63
+ getHighestMainCommitSeq(): Promise<number>;
64
+ }
65
+ //# sourceMappingURL=sequence-mapper.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sequence-mapper.d.ts","sourceRoot":"","sources":["../../src/client-role/sequence-mapper.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAEH,OAAO,EAAE,KAAK,MAAM,EAAO,MAAM,QAAQ,CAAC;AAC1C,OAAO,KAAK,EAAE,aAAa,EAAE,sBAAsB,EAAE,MAAM,WAAW,CAAC;AAEvE;;GAEG;AACH,MAAM,WAAW,eAAe;IAC9B,cAAc,EAAE,MAAM,CAAC;IACvB,aAAa,EAAE,MAAM,GAAG,IAAI,CAAC;IAC7B,MAAM,EAAE,sBAAsB,CAAC;CAChC;AAED;;GAEG;AACH,qBAAa,cAAc,CAAC,EAAE,SAAS,aAAa,GAAG,aAAa;IAClE,OAAO,CAAC,QAAQ,CAAC,EAAE,CAAa;IAEhC,YAAY,OAAO,EAAE;QAAE,EAAE,EAAE,MAAM,CAAC,EAAE,CAAC,CAAA;KAAE,EAEtC;IAED;;OAEG;IACG,oBAAoB,CAAC,cAAc,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAahE;IAED;;OAEG;IACG,aAAa,CACjB,cAAc,EAAE,MAAM,EACtB,aAAa,EAAE,MAAM,GACpB,OAAO,CAAC,IAAI,CAAC,CAUf;IAED;;OAEG;IACG,aAAa,CAAC,cAAc,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAOzD;IAED;;OAEG;IACG,UAAU,CAAC,cAAc,EAAE,MAAM,GAAG,OAAO,CAAC,eAAe,GAAG,IAAI,CAAC,CAoBxE;IAED;;OAEG;IACG,iBAAiB,CAAC,aAAa,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,CASrE;IAED;;OAEG;IACG,kBAAkB,IAAI,OAAO,CAAC,eAAe,EAAE,CAAC,CAkBrD;IAED;;;;OAIG;IACG,sBAAsB,CAC1B,cAAc,EAAE,MAAM,EACtB,aAAa,EAAE,MAAM,GACpB,OAAO,CAAC,IAAI,CAAC,CAuBf;IAED;;;OAGG;IACG,gBAAgB,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC,CASxD;IAED;;OAEG;IACG,uBAAuB,IAAI,OAAO,CAAC,MAAM,CAAC,CAS/C;CACF"}
@@ -0,0 +1,161 @@
1
+ /**
2
+ * @syncular/relay - Sequence Mapper
3
+ *
4
+ * Tracks the mapping between relay's local commit_seq
5
+ * and the main server's global commit_seq.
6
+ */
7
+ import { sql } from 'kysely';
8
+ /**
9
+ * Sequence mapper for tracking local to main commit sequence mappings.
10
+ */
11
+ export class SequenceMapper {
12
+ db;
13
+ constructor(options) {
14
+ this.db = options.db;
15
+ }
16
+ /**
17
+ * Create a pending mapping for a local commit that will be forwarded.
18
+ */
19
+ async createPendingMapping(localCommitSeq) {
20
+ const now = Date.now();
21
+ await sql `
22
+ insert into ${sql.table('relay_sequence_map')} (
23
+ local_commit_seq,
24
+ main_commit_seq,
25
+ status,
26
+ created_at,
27
+ updated_at
28
+ )
29
+ values (${localCommitSeq}, ${null}, 'pending', ${now}, ${now})
30
+ on conflict (local_commit_seq) do nothing
31
+ `.execute(this.db);
32
+ }
33
+ /**
34
+ * Mark a mapping as forwarded with the main server's commit_seq.
35
+ */
36
+ async markForwarded(localCommitSeq, mainCommitSeq) {
37
+ const now = Date.now();
38
+ await sql `
39
+ update ${sql.table('relay_sequence_map')}
40
+ set
41
+ main_commit_seq = ${mainCommitSeq},
42
+ status = 'forwarded',
43
+ updated_at = ${now}
44
+ where local_commit_seq = ${localCommitSeq}
45
+ `.execute(this.db);
46
+ }
47
+ /**
48
+ * Mark a mapping as confirmed (main server acknowledged).
49
+ */
50
+ async markConfirmed(localCommitSeq) {
51
+ const now = Date.now();
52
+ await sql `
53
+ update ${sql.table('relay_sequence_map')}
54
+ set status = 'confirmed', updated_at = ${now}
55
+ where local_commit_seq = ${localCommitSeq}
56
+ `.execute(this.db);
57
+ }
58
+ /**
59
+ * Get the mapping for a local commit sequence.
60
+ */
61
+ async getMapping(localCommitSeq) {
62
+ const rowResult = await sql `
63
+ select local_commit_seq, main_commit_seq, status
64
+ from ${sql.table('relay_sequence_map')}
65
+ where local_commit_seq = ${localCommitSeq}
66
+ limit 1
67
+ `.execute(this.db);
68
+ const row = rowResult.rows[0];
69
+ if (!row)
70
+ return null;
71
+ return {
72
+ localCommitSeq: row.local_commit_seq,
73
+ mainCommitSeq: row.main_commit_seq,
74
+ status: row.status,
75
+ };
76
+ }
77
+ /**
78
+ * Get the local commit sequence for a main server commit sequence.
79
+ */
80
+ async getLocalCommitSeq(mainCommitSeq) {
81
+ const rowResult = await sql `
82
+ select local_commit_seq
83
+ from ${sql.table('relay_sequence_map')}
84
+ where main_commit_seq = ${mainCommitSeq}
85
+ limit 1
86
+ `.execute(this.db);
87
+ return rowResult.rows[0]?.local_commit_seq ?? null;
88
+ }
89
+ /**
90
+ * Get all pending mappings (commits not yet forwarded).
91
+ */
92
+ async getPendingMappings() {
93
+ const rowsResult = await sql `
94
+ select local_commit_seq, main_commit_seq, status
95
+ from ${sql.table('relay_sequence_map')}
96
+ where status = 'pending'
97
+ order by local_commit_seq asc
98
+ `.execute(this.db);
99
+ const rows = rowsResult.rows;
100
+ return rows.map((row) => ({
101
+ localCommitSeq: row.local_commit_seq,
102
+ mainCommitSeq: row.main_commit_seq,
103
+ status: row.status,
104
+ }));
105
+ }
106
+ /**
107
+ * Create a mapping for commits pulled from main (assigned new local commit_seq).
108
+ *
109
+ * These mappings go directly to 'confirmed' status since they came from main.
110
+ */
111
+ async createConfirmedMapping(localCommitSeq, mainCommitSeq) {
112
+ const now = Date.now();
113
+ await sql `
114
+ insert into ${sql.table('relay_sequence_map')} (
115
+ local_commit_seq,
116
+ main_commit_seq,
117
+ status,
118
+ created_at,
119
+ updated_at
120
+ )
121
+ values (
122
+ ${localCommitSeq},
123
+ ${mainCommitSeq},
124
+ 'confirmed',
125
+ ${now},
126
+ ${now}
127
+ )
128
+ on conflict (local_commit_seq)
129
+ do update set
130
+ main_commit_seq = ${mainCommitSeq},
131
+ status = 'confirmed',
132
+ updated_at = ${now}
133
+ `.execute(this.db);
134
+ }
135
+ /**
136
+ * Delete confirmed/forwarded sequence mappings older than the given age.
137
+ * Keeps pending mappings (they haven't been forwarded yet).
138
+ */
139
+ async pruneOldMappings(maxAgeMs) {
140
+ const threshold = Date.now() - maxAgeMs;
141
+ const result = await sql `
142
+ delete from ${sql.table('relay_sequence_map')}
143
+ where status in ('confirmed', 'forwarded')
144
+ and updated_at < ${threshold}
145
+ `.execute(this.db);
146
+ return Number(result.numAffectedRows ?? 0);
147
+ }
148
+ /**
149
+ * Get the highest main_commit_seq we've seen (for tracking pull cursor).
150
+ */
151
+ async getHighestMainCommitSeq() {
152
+ const rowResult = await sql `
153
+ select max(main_commit_seq) as max_seq
154
+ from ${sql.table('relay_sequence_map')}
155
+ where main_commit_seq is not null
156
+ limit 1
157
+ `.execute(this.db);
158
+ return rowResult.rows[0]?.max_seq ?? 0;
159
+ }
160
+ }
161
+ //# sourceMappingURL=sequence-mapper.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sequence-mapper.js","sourceRoot":"","sources":["../../src/client-role/sequence-mapper.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,37 @@
1
+ /**
2
+ * @syncular/relay - Edge Relay Server
3
+ *
4
+ * An edge relay server that acts as a local server to nearby clients
5
+ * while simultaneously acting as a client to the main server.
6
+ *
7
+ * @example
8
+ * ```typescript
9
+ * import { createRelayServer } from '@syncular/relay';
10
+ * import { createSqliteServerDialect } from '@syncular/server-dialect-sqlite';
11
+ *
12
+ * const relay = createRelayServer({
13
+ * db: sqliteDb,
14
+ * dialect: createSqliteServerDialect(),
15
+ * mainServerTransport: createHttpTransport({ baseUrl: 'https://main.example.com/sync' }),
16
+ * mainServerClientId: 'relay-branch-001',
17
+ * mainServerActorId: 'relay-service',
18
+ * scopeKeys: ['client:acme'],
19
+ * shapes: shapeRegistry,
20
+ * subscriptions: subscriptionRegistry,
21
+ * });
22
+ *
23
+ * // Mount routes for local clients
24
+ * app.route('/sync', relay.getRoutes());
25
+ *
26
+ * // Start background sync with main
27
+ * await relay.start();
28
+ * ```
29
+ */
30
+ export * from './client-role';
31
+ export * from './migrate';
32
+ export * from './mode-manager';
33
+ export * from './realtime';
34
+ export * from './relay';
35
+ export * from './schema';
36
+ export * from './server-role';
37
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA4BG;AAGH,cAAc,eAAe,CAAC;AAG9B,cAAc,WAAW,CAAC;AAG1B,cAAc,gBAAgB,CAAC;AAG/B,cAAc,YAAY,CAAC;AAG3B,cAAc,SAAS,CAAC;AAGxB,cAAc,UAAU,CAAC;AAGzB,cAAc,eAAe,CAAC"}
package/dist/index.js ADDED
@@ -0,0 +1,44 @@
1
+ /**
2
+ * @syncular/relay - Edge Relay Server
3
+ *
4
+ * An edge relay server that acts as a local server to nearby clients
5
+ * while simultaneously acting as a client to the main server.
6
+ *
7
+ * @example
8
+ * ```typescript
9
+ * import { createRelayServer } from '@syncular/relay';
10
+ * import { createSqliteServerDialect } from '@syncular/server-dialect-sqlite';
11
+ *
12
+ * const relay = createRelayServer({
13
+ * db: sqliteDb,
14
+ * dialect: createSqliteServerDialect(),
15
+ * mainServerTransport: createHttpTransport({ baseUrl: 'https://main.example.com/sync' }),
16
+ * mainServerClientId: 'relay-branch-001',
17
+ * mainServerActorId: 'relay-service',
18
+ * scopeKeys: ['client:acme'],
19
+ * shapes: shapeRegistry,
20
+ * subscriptions: subscriptionRegistry,
21
+ * });
22
+ *
23
+ * // Mount routes for local clients
24
+ * app.route('/sync', relay.getRoutes());
25
+ *
26
+ * // Start background sync with main
27
+ * await relay.start();
28
+ * ```
29
+ */
30
+ // Client role (syncing with main server)
31
+ export * from './client-role';
32
+ // Migration
33
+ export * from './migrate';
34
+ // Mode manager
35
+ export * from './mode-manager';
36
+ // Realtime WebSocket manager
37
+ export * from './realtime';
38
+ // Main exports
39
+ export * from './relay';
40
+ // Schema types
41
+ export * from './schema';
42
+ // Server role (serving local clients)
43
+ export * from './server-role';
44
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA4BG;AAEH,yCAAyC;AACzC,cAAc,eAAe,CAAC;AAE9B,YAAY;AACZ,cAAc,WAAW,CAAC;AAE1B,eAAe;AACf,cAAc,gBAAgB,CAAC;AAE/B,6BAA6B;AAC7B,cAAc,YAAY,CAAC;AAE3B,eAAe;AACf,cAAc,SAAS,CAAC;AAExB,eAAe;AACf,cAAc,UAAU,CAAC;AAEzB,sCAAsC;AACtC,cAAc,eAAe,CAAC"}
@@ -0,0 +1,18 @@
1
+ /**
2
+ * @syncular/relay - Schema setup
3
+ *
4
+ * Creates relay-specific tables for edge relay functionality.
5
+ * Uses Kysely for dialect-agnostic schema creation.
6
+ */
7
+ import type { ServerSyncDialect } from '@syncular/server';
8
+ import type { Kysely } from 'kysely';
9
+ import type { RelayDatabase } from './schema';
10
+ /**
11
+ * Ensures the relay schema exists in the database.
12
+ * Safe to call multiple times (idempotent).
13
+ *
14
+ * This creates relay-specific tables on top of the base sync schema.
15
+ * Call `ensureSyncSchema()` from @syncular/server first to create base tables.
16
+ */
17
+ export declare function ensureRelaySchema<DB extends RelayDatabase = RelayDatabase>(db: Kysely<DB>, dialect: ServerSyncDialect): Promise<void>;
18
+ //# sourceMappingURL=migrate.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"migrate.d.ts","sourceRoot":"","sources":["../src/migrate.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAEH,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,kBAAkB,CAAC;AAC1D,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,QAAQ,CAAC;AAErC,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,UAAU,CAAC;AAE9C;;;;;;GAMG;AACH,wBAAsB,iBAAiB,CACrC,EAAE,SAAS,aAAa,GAAG,aAAa,EACxC,EAAE,EAAE,MAAM,CAAC,EAAE,CAAC,EAAE,OAAO,EAAE,iBAAiB,GAAG,OAAO,CAAC,IAAI,CAAC,CA2F3D"}
@@ -0,0 +1,99 @@
1
+ /**
2
+ * @syncular/relay - Schema setup
3
+ *
4
+ * Creates relay-specific tables for edge relay functionality.
5
+ * Uses Kysely for dialect-agnostic schema creation.
6
+ */
7
+ import { sql } from 'kysely';
8
+ /**
9
+ * Ensures the relay schema exists in the database.
10
+ * Safe to call multiple times (idempotent).
11
+ *
12
+ * This creates relay-specific tables on top of the base sync schema.
13
+ * Call `ensureSyncSchema()` from @syncular/server first to create base tables.
14
+ */
15
+ export async function ensureRelaySchema(db, dialect) {
16
+ // Ensure base sync schema exists first
17
+ await dialect.ensureSyncSchema(db);
18
+ // Create relay-specific tables
19
+ const isSqlite = dialect.name === 'sqlite';
20
+ // Forward outbox table
21
+ await sql
22
+ .raw(`
23
+ CREATE TABLE IF NOT EXISTS relay_forward_outbox (
24
+ id TEXT PRIMARY KEY,
25
+ local_commit_seq INTEGER NOT NULL,
26
+ client_id TEXT NOT NULL,
27
+ client_commit_id TEXT NOT NULL,
28
+ operations_json TEXT NOT NULL,
29
+ schema_version INTEGER NOT NULL DEFAULT 1,
30
+ status TEXT NOT NULL DEFAULT 'pending',
31
+ main_commit_seq INTEGER,
32
+ error TEXT,
33
+ last_response_json TEXT,
34
+ created_at INTEGER NOT NULL DEFAULT (${isSqlite ? "strftime('%s','now') * 1000" : 'EXTRACT(EPOCH FROM NOW()) * 1000'}),
35
+ updated_at INTEGER NOT NULL DEFAULT (${isSqlite ? "strftime('%s','now') * 1000" : 'EXTRACT(EPOCH FROM NOW()) * 1000'}),
36
+ attempt_count INTEGER NOT NULL DEFAULT 0
37
+ )
38
+ `)
39
+ .execute(db);
40
+ // Index for finding next sendable outbox entry
41
+ await sql
42
+ .raw(`
43
+ CREATE INDEX IF NOT EXISTS idx_relay_forward_outbox_status
44
+ ON relay_forward_outbox (status, created_at)
45
+ `)
46
+ .execute(db);
47
+ // Sequence map table
48
+ await sql
49
+ .raw(`
50
+ CREATE TABLE IF NOT EXISTS relay_sequence_map (
51
+ local_commit_seq INTEGER PRIMARY KEY,
52
+ main_commit_seq INTEGER,
53
+ status TEXT NOT NULL DEFAULT 'pending',
54
+ created_at INTEGER NOT NULL DEFAULT (${isSqlite ? "strftime('%s','now') * 1000" : 'EXTRACT(EPOCH FROM NOW()) * 1000'}),
55
+ updated_at INTEGER NOT NULL DEFAULT (${isSqlite ? "strftime('%s','now') * 1000" : 'EXTRACT(EPOCH FROM NOW()) * 1000'})
56
+ )
57
+ `)
58
+ .execute(db);
59
+ // Index for looking up main_commit_seq
60
+ await sql
61
+ .raw(`
62
+ CREATE INDEX IF NOT EXISTS idx_relay_sequence_map_main
63
+ ON relay_sequence_map (main_commit_seq)
64
+ WHERE main_commit_seq IS NOT NULL
65
+ `)
66
+ .execute(db);
67
+ // Forward conflicts table
68
+ await sql
69
+ .raw(`
70
+ CREATE TABLE IF NOT EXISTS relay_forward_conflicts (
71
+ id TEXT PRIMARY KEY,
72
+ local_commit_seq INTEGER NOT NULL,
73
+ client_id TEXT NOT NULL,
74
+ client_commit_id TEXT NOT NULL,
75
+ response_json TEXT NOT NULL,
76
+ created_at INTEGER NOT NULL,
77
+ resolved_at INTEGER
78
+ )
79
+ `)
80
+ .execute(db);
81
+ // Index for finding unresolved conflicts
82
+ await sql
83
+ .raw(`
84
+ CREATE INDEX IF NOT EXISTS idx_relay_forward_conflicts_unresolved
85
+ ON relay_forward_conflicts (resolved_at)
86
+ WHERE resolved_at IS NULL
87
+ `)
88
+ .execute(db);
89
+ // Config table
90
+ await sql
91
+ .raw(`
92
+ CREATE TABLE IF NOT EXISTS relay_config (
93
+ key TEXT PRIMARY KEY,
94
+ value_json TEXT NOT NULL
95
+ )
96
+ `)
97
+ .execute(db);
98
+ }
99
+ //# sourceMappingURL=migrate.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"migrate.js","sourceRoot":"","sources":["../src/migrate.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAIH,OAAO,EAAE,GAAG,EAAE,MAAM,QAAQ,CAAC;AAG7B;;;;;;GAMG;AACH,MAAM,CAAC,KAAK,UAAU,iBAAiB,CAErC,EAAc,EAAE,OAA0B,EAAiB;IAC3D,uCAAuC;IACvC,MAAM,OAAO,CAAC,gBAAgB,CAAC,EAAE,CAAC,CAAC;IAEnC,+BAA+B;IAC/B,MAAM,QAAQ,GAAG,OAAO,CAAC,IAAI,KAAK,QAAQ,CAAC;IAE3C,uBAAuB;IACvB,MAAM,GAAG;SACN,GAAG,CAAC;;;;;;;;;;;;6CAYoC,QAAQ,CAAC,CAAC,CAAC,6BAA6B,CAAC,CAAC,CAAC,kCAAkC;6CAC7E,QAAQ,CAAC,CAAC,CAAC,6BAA6B,CAAC,CAAC,CAAC,kCAAkC;;;GAGvH,CAAC;SACC,OAAO,CAAC,EAAE,CAAC,CAAC;IAEf,+CAA+C;IAC/C,MAAM,GAAG;SACN,GAAG,CAAC;;;GAGN,CAAC;SACC,OAAO,CAAC,EAAE,CAAC,CAAC;IAEf,qBAAqB;IACrB,MAAM,GAAG;SACN,GAAG,CAAC;;;;;6CAKoC,QAAQ,CAAC,CAAC,CAAC,6BAA6B,CAAC,CAAC,CAAC,kCAAkC;6CAC7E,QAAQ,CAAC,CAAC,CAAC,6BAA6B,CAAC,CAAC,CAAC,kCAAkC;;GAEvH,CAAC;SACC,OAAO,CAAC,EAAE,CAAC,CAAC;IAEf,uCAAuC;IACvC,MAAM,GAAG;SACN,GAAG,CAAC;;;;GAIN,CAAC;SACC,OAAO,CAAC,EAAE,CAAC,CAAC;IAEf,0BAA0B;IAC1B,MAAM,GAAG;SACN,GAAG,CAAC;;;;;;;;;;GAUN,CAAC;SACC,OAAO,CAAC,EAAE,CAAC,CAAC;IAEf,yCAAyC;IACzC,MAAM,GAAG;SACN,GAAG,CAAC;;;;GAIN,CAAC;SACC,OAAO,CAAC,EAAE,CAAC,CAAC;IAEf,eAAe;IACf,MAAM,GAAG;SACN,GAAG,CAAC;;;;;GAKN,CAAC;SACC,OAAO,CAAC,EAAE,CAAC,CAAC;AAAA,CAChB"}
@@ -0,0 +1,60 @@
1
+ /**
2
+ * @syncular/relay - Mode Manager
3
+ *
4
+ * State machine for tracking relay online/offline status.
5
+ */
6
+ /**
7
+ * Relay operating modes.
8
+ */
9
+ export type RelayMode = 'online' | 'offline' | 'reconnecting';
10
+ /**
11
+ * Mode manager options.
12
+ */
13
+ export interface ModeManagerOptions {
14
+ healthCheckIntervalMs?: number;
15
+ reconnectBackoffMs?: number;
16
+ maxReconnectBackoffMs?: number;
17
+ onModeChange?: (mode: RelayMode) => void;
18
+ }
19
+ /**
20
+ * Mode manager for tracking relay online/offline state.
21
+ *
22
+ * Uses health checks to detect connectivity to the main server
23
+ * and manages reconnection with exponential backoff.
24
+ */
25
+ export declare class ModeManager {
26
+ private mode;
27
+ private healthCheckIntervalMs;
28
+ private reconnectBackoffMs;
29
+ private maxReconnectBackoffMs;
30
+ private currentBackoffMs;
31
+ private onModeChange?;
32
+ private running;
33
+ private timer;
34
+ private healthCheckFn;
35
+ private consecutiveFailures;
36
+ constructor(options?: ModeManagerOptions);
37
+ /**
38
+ * Get the current mode.
39
+ */
40
+ getMode(): RelayMode;
41
+ /**
42
+ * Start the mode manager with a health check function.
43
+ */
44
+ start(healthCheckFn: () => Promise<boolean>): void;
45
+ /**
46
+ * Stop the mode manager.
47
+ */
48
+ stop(): void;
49
+ /**
50
+ * Manually report a successful operation (resets backoff).
51
+ */
52
+ reportSuccess(): void;
53
+ /**
54
+ * Manually report a failed operation.
55
+ */
56
+ reportFailure(): void;
57
+ private setMode;
58
+ private scheduleHealthCheck;
59
+ }
60
+ //# sourceMappingURL=mode-manager.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mode-manager.d.ts","sourceRoot":"","sources":["../src/mode-manager.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH;;GAEG;AACH,MAAM,MAAM,SAAS,GAAG,QAAQ,GAAG,SAAS,GAAG,cAAc,CAAC;AAE9D;;GAEG;AACH,MAAM,WAAW,kBAAkB;IACjC,qBAAqB,CAAC,EAAE,MAAM,CAAC;IAC/B,kBAAkB,CAAC,EAAE,MAAM,CAAC;IAC5B,qBAAqB,CAAC,EAAE,MAAM,CAAC;IAC/B,YAAY,CAAC,EAAE,CAAC,IAAI,EAAE,SAAS,KAAK,IAAI,CAAC;CAC1C;AAED;;;;;GAKG;AACH,qBAAa,WAAW;IACtB,OAAO,CAAC,IAAI,CAAwB;IACpC,OAAO,CAAC,qBAAqB,CAAS;IACtC,OAAO,CAAC,kBAAkB,CAAS;IACnC,OAAO,CAAC,qBAAqB,CAAS;IACtC,OAAO,CAAC,gBAAgB,CAAS;IACjC,OAAO,CAAC,YAAY,CAAC,CAA4B;IAEjD,OAAO,CAAC,OAAO,CAAS;IACxB,OAAO,CAAC,KAAK,CAA8C;IAC3D,OAAO,CAAC,aAAa,CAAyC;IAE9D,OAAO,CAAC,mBAAmB,CAAK;IAEhC,YAAY,OAAO,GAAE,kBAAuB,EAM3C;IAED;;OAEG;IACH,OAAO,IAAI,SAAS,CAEnB;IAED;;OAEG;IACH,KAAK,CAAC,aAAa,EAAE,MAAM,OAAO,CAAC,OAAO,CAAC,GAAG,IAAI,CAOjD;IAED;;OAEG;IACH,IAAI,IAAI,IAAI,CAMX;IAED;;OAEG;IACH,aAAa,IAAI,IAAI,CAMpB;IAED;;OAEG;IACH,aAAa,IAAI,IAAI,CAYpB;IAED,OAAO,CAAC,OAAO;IAMf,OAAO,CAAC,mBAAmB;CA4B5B"}
@@ -0,0 +1,114 @@
1
+ /**
2
+ * @syncular/relay - Mode Manager
3
+ *
4
+ * State machine for tracking relay online/offline status.
5
+ */
6
+ /**
7
+ * Mode manager for tracking relay online/offline state.
8
+ *
9
+ * Uses health checks to detect connectivity to the main server
10
+ * and manages reconnection with exponential backoff.
11
+ */
12
+ export class ModeManager {
13
+ mode = 'offline';
14
+ healthCheckIntervalMs;
15
+ reconnectBackoffMs;
16
+ maxReconnectBackoffMs;
17
+ currentBackoffMs;
18
+ onModeChange;
19
+ running = false;
20
+ timer = null;
21
+ healthCheckFn = null;
22
+ // biome-ignore lint/correctness/noUnusedPrivateClassMembers: assigned and incremented in healthCheck
23
+ consecutiveFailures = 0;
24
+ constructor(options = {}) {
25
+ this.healthCheckIntervalMs = options.healthCheckIntervalMs ?? 30000;
26
+ this.reconnectBackoffMs = options.reconnectBackoffMs ?? 1000;
27
+ this.maxReconnectBackoffMs = options.maxReconnectBackoffMs ?? 60000;
28
+ this.currentBackoffMs = this.reconnectBackoffMs;
29
+ this.onModeChange = options.onModeChange;
30
+ }
31
+ /**
32
+ * Get the current mode.
33
+ */
34
+ getMode() {
35
+ return this.mode;
36
+ }
37
+ /**
38
+ * Start the mode manager with a health check function.
39
+ */
40
+ start(healthCheckFn) {
41
+ if (this.running)
42
+ return;
43
+ this.running = true;
44
+ this.healthCheckFn = healthCheckFn;
45
+ // Start with immediate health check
46
+ this.scheduleHealthCheck(0);
47
+ }
48
+ /**
49
+ * Stop the mode manager.
50
+ */
51
+ stop() {
52
+ this.running = false;
53
+ if (this.timer) {
54
+ clearTimeout(this.timer);
55
+ this.timer = null;
56
+ }
57
+ }
58
+ /**
59
+ * Manually report a successful operation (resets backoff).
60
+ */
61
+ reportSuccess() {
62
+ if (this.mode !== 'online') {
63
+ this.setMode('online');
64
+ }
65
+ this.consecutiveFailures = 0;
66
+ this.currentBackoffMs = this.reconnectBackoffMs;
67
+ }
68
+ /**
69
+ * Manually report a failed operation.
70
+ */
71
+ reportFailure() {
72
+ this.consecutiveFailures++;
73
+ if (this.mode === 'online') {
74
+ this.setMode('reconnecting');
75
+ }
76
+ // Increase backoff for next attempt
77
+ this.currentBackoffMs = Math.min(this.currentBackoffMs * 2, this.maxReconnectBackoffMs);
78
+ }
79
+ setMode(newMode) {
80
+ if (this.mode === newMode)
81
+ return;
82
+ this.mode = newMode;
83
+ this.onModeChange?.(newMode);
84
+ }
85
+ scheduleHealthCheck(delayMs) {
86
+ if (!this.running)
87
+ return;
88
+ if (this.timer)
89
+ return;
90
+ this.timer = setTimeout(async () => {
91
+ this.timer = null;
92
+ if (!this.healthCheckFn) {
93
+ this.scheduleHealthCheck(this.healthCheckIntervalMs);
94
+ return;
95
+ }
96
+ try {
97
+ const healthy = await this.healthCheckFn();
98
+ if (healthy) {
99
+ this.reportSuccess();
100
+ this.scheduleHealthCheck(this.healthCheckIntervalMs);
101
+ }
102
+ else {
103
+ this.reportFailure();
104
+ this.scheduleHealthCheck(this.currentBackoffMs);
105
+ }
106
+ }
107
+ catch {
108
+ this.reportFailure();
109
+ this.scheduleHealthCheck(this.currentBackoffMs);
110
+ }
111
+ }, delayMs);
112
+ }
113
+ }
114
+ //# sourceMappingURL=mode-manager.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mode-manager.js","sourceRoot":"","sources":["../src/mode-manager.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAiBH;;;;;GAKG;AACH,MAAM,OAAO,WAAW;IACd,IAAI,GAAc,SAAS,CAAC;IAC5B,qBAAqB,CAAS;IAC9B,kBAAkB,CAAS;IAC3B,qBAAqB,CAAS;IAC9B,gBAAgB,CAAS;IACzB,YAAY,CAA6B;IAEzC,OAAO,GAAG,KAAK,CAAC;IAChB,KAAK,GAAyC,IAAI,CAAC;IACnD,aAAa,GAAoC,IAAI,CAAC;IAC9D,qGAAqG;IAC7F,mBAAmB,GAAG,CAAC,CAAC;IAEhC,YAAY,OAAO,GAAuB,EAAE,EAAE;QAC5C,IAAI,CAAC,qBAAqB,GAAG,OAAO,CAAC,qBAAqB,IAAI,KAAK,CAAC;QACpE,IAAI,CAAC,kBAAkB,GAAG,OAAO,CAAC,kBAAkB,IAAI,IAAI,CAAC;QAC7D,IAAI,CAAC,qBAAqB,GAAG,OAAO,CAAC,qBAAqB,IAAI,KAAK,CAAC;QACpE,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,kBAAkB,CAAC;QAChD,IAAI,CAAC,YAAY,GAAG,OAAO,CAAC,YAAY,CAAC;IAAA,CAC1C;IAED;;OAEG;IACH,OAAO,GAAc;QACnB,OAAO,IAAI,CAAC,IAAI,CAAC;IAAA,CAClB;IAED;;OAEG;IACH,KAAK,CAAC,aAAqC,EAAQ;QACjD,IAAI,IAAI,CAAC,OAAO;YAAE,OAAO;QACzB,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC;QACpB,IAAI,CAAC,aAAa,GAAG,aAAa,CAAC;QAEnC,oCAAoC;QACpC,IAAI,CAAC,mBAAmB,CAAC,CAAC,CAAC,CAAC;IAAA,CAC7B;IAED;;OAEG;IACH,IAAI,GAAS;QACX,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC;QACrB,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;YACf,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YACzB,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC;QACpB,CAAC;IAAA,CACF;IAED;;OAEG;IACH,aAAa,GAAS;QACpB,IAAI,IAAI,CAAC,IAAI,KAAK,QAAQ,EAAE,CAAC;YAC3B,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;QACzB,CAAC;QACD,IAAI,CAAC,mBAAmB,GAAG,CAAC,CAAC;QAC7B,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,kBAAkB,CAAC;IAAA,CACjD;IAED;;OAEG;IACH,aAAa,GAAS;QACpB,IAAI,CAAC,mBAAmB,EAAE,CAAC;QAE3B,IAAI,IAAI,CAAC,IAAI,KAAK,QAAQ,EAAE,CAAC;YAC3B,IAAI,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;QAC/B,CAAC;QAED,oCAAoC;QACpC,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,GAAG,CAC9B,IAAI,CAAC,gBAAgB,GAAG,CAAC,EACzB,IAAI,CAAC,qBAAqB,CAC3B,CAAC;IAAA,CACH;IAEO,OAAO,CAAC,OAAkB,EAAQ;QACxC,IAAI,IAAI,CAAC,IAAI,KAAK,OAAO;YAAE,OAAO;QAClC,IAAI,CAAC,IAAI,GAAG,OAAO,CAAC;QACpB,IAAI,CAAC,YAAY,EAAE,CAAC,OAAO,CAAC,CAAC;IAAA,CAC9B;IAEO,mBAAmB,CAAC,OAAe,EAAQ;QACjD,IAAI,CAAC,IAAI,CAAC,OAAO;YAAE,OAAO;QAC1B,IAAI,IAAI,CAAC,KAAK;YAAE,OAAO;QAEvB,IAAI,CAAC,KAAK,GAAG,UAAU,CAAC,KAAK,IAAI,EAAE,CAAC;YAClC,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC;YAElB,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,CAAC;gBACxB,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC;gBACrD,OAAO;YACT,CAAC;YAED,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,aAAa,EAAE,CAAC;gBAE3C,IAAI,OAAO,EAAE,CAAC;oBACZ,IAAI,CAAC,aAAa,EAAE,CAAC;oBACrB,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC;gBACvD,CAAC;qBAAM,CAAC;oBACN,IAAI,CAAC,aAAa,EAAE,CAAC;oBACrB,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;gBAClD,CAAC;YACH,CAAC;YAAC,MAAM,CAAC;gBACP,IAAI,CAAC,aAAa,EAAE,CAAC;gBACrB,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;YAClD,CAAC;QAAA,CACF,EAAE,OAAO,CAAC,CAAC;IAAA,CACb;CACF"}