@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.
- package/dist/client-role/forward-engine.d.ts +63 -0
- package/dist/client-role/forward-engine.d.ts.map +1 -0
- package/dist/client-role/forward-engine.js +263 -0
- package/dist/client-role/forward-engine.js.map +1 -0
- package/dist/client-role/index.d.ts +9 -0
- package/dist/client-role/index.d.ts.map +1 -0
- package/dist/client-role/index.js +9 -0
- package/dist/client-role/index.js.map +1 -0
- package/dist/client-role/pull-engine.d.ts +70 -0
- package/dist/client-role/pull-engine.d.ts.map +1 -0
- package/dist/client-role/pull-engine.js +233 -0
- package/dist/client-role/pull-engine.js.map +1 -0
- package/dist/client-role/sequence-mapper.d.ts +65 -0
- package/dist/client-role/sequence-mapper.d.ts.map +1 -0
- package/dist/client-role/sequence-mapper.js +161 -0
- package/dist/client-role/sequence-mapper.js.map +1 -0
- package/dist/index.d.ts +37 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +44 -0
- package/dist/index.js.map +1 -0
- package/dist/migrate.d.ts +18 -0
- package/dist/migrate.d.ts.map +1 -0
- package/dist/migrate.js +99 -0
- package/dist/migrate.js.map +1 -0
- package/dist/mode-manager.d.ts +60 -0
- package/dist/mode-manager.d.ts.map +1 -0
- package/dist/mode-manager.js +114 -0
- package/dist/mode-manager.js.map +1 -0
- package/dist/realtime.d.ts +102 -0
- package/dist/realtime.d.ts.map +1 -0
- package/dist/realtime.js +305 -0
- package/dist/realtime.js.map +1 -0
- package/dist/relay.d.ts +188 -0
- package/dist/relay.d.ts.map +1 -0
- package/dist/relay.js +315 -0
- package/dist/relay.js.map +1 -0
- package/dist/schema.d.ts +158 -0
- package/dist/schema.d.ts.map +1 -0
- package/dist/schema.js +7 -0
- package/dist/schema.js.map +1 -0
- package/dist/server-role/index.d.ts +54 -0
- package/dist/server-role/index.d.ts.map +1 -0
- package/dist/server-role/index.js +198 -0
- package/dist/server-role/index.js.map +1 -0
- package/dist/server-role/pull.d.ts +25 -0
- package/dist/server-role/pull.d.ts.map +1 -0
- package/dist/server-role/pull.js +24 -0
- package/dist/server-role/pull.js.map +1 -0
- package/dist/server-role/push.d.ts +27 -0
- package/dist/server-role/push.d.ts.map +1 -0
- package/dist/server-role/push.js +98 -0
- package/dist/server-role/push.js.map +1 -0
- package/package.json +61 -0
- package/src/__tests__/relay.test.ts +464 -0
- package/src/bun-types.d.ts +50 -0
- package/src/client-role/forward-engine.ts +352 -0
- package/src/client-role/index.ts +9 -0
- package/src/client-role/pull-engine.ts +301 -0
- package/src/client-role/sequence-mapper.ts +201 -0
- package/src/index.ts +50 -0
- package/src/migrate.ts +113 -0
- package/src/mode-manager.ts +142 -0
- package/src/realtime.ts +370 -0
- package/src/relay.ts +421 -0
- package/src/schema.ts +171 -0
- package/src/server-role/index.ts +342 -0
- package/src/server-role/pull.ts +37 -0
- 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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"}
|
package/dist/index.d.ts
ADDED
|
@@ -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"}
|
package/dist/migrate.js
ADDED
|
@@ -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":"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"}
|