@syncular/client 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 (176) hide show
  1. package/dist/blobs/index.d.ts +7 -0
  2. package/dist/blobs/index.d.ts.map +1 -0
  3. package/dist/blobs/index.js +7 -0
  4. package/dist/blobs/index.js.map +1 -0
  5. package/dist/blobs/manager.d.ts +345 -0
  6. package/dist/blobs/manager.d.ts.map +1 -0
  7. package/dist/blobs/manager.js +749 -0
  8. package/dist/blobs/manager.js.map +1 -0
  9. package/dist/blobs/migrate.d.ts +14 -0
  10. package/dist/blobs/migrate.d.ts.map +1 -0
  11. package/dist/blobs/migrate.js +59 -0
  12. package/dist/blobs/migrate.js.map +1 -0
  13. package/dist/blobs/types.d.ts +62 -0
  14. package/dist/blobs/types.d.ts.map +1 -0
  15. package/dist/blobs/types.js +5 -0
  16. package/dist/blobs/types.js.map +1 -0
  17. package/dist/client.d.ts +338 -0
  18. package/dist/client.d.ts.map +1 -0
  19. package/dist/client.js +834 -0
  20. package/dist/client.js.map +1 -0
  21. package/dist/conflicts.d.ts +31 -0
  22. package/dist/conflicts.d.ts.map +1 -0
  23. package/dist/conflicts.js +118 -0
  24. package/dist/conflicts.js.map +1 -0
  25. package/dist/create-client.d.ts +115 -0
  26. package/dist/create-client.d.ts.map +1 -0
  27. package/dist/create-client.js +162 -0
  28. package/dist/create-client.js.map +1 -0
  29. package/dist/engine/SyncEngine.d.ts +215 -0
  30. package/dist/engine/SyncEngine.d.ts.map +1 -0
  31. package/dist/engine/SyncEngine.js +1066 -0
  32. package/dist/engine/SyncEngine.js.map +1 -0
  33. package/dist/engine/index.d.ts +6 -0
  34. package/dist/engine/index.d.ts.map +1 -0
  35. package/dist/engine/index.js +6 -0
  36. package/dist/engine/index.js.map +1 -0
  37. package/dist/engine/types.d.ts +230 -0
  38. package/dist/engine/types.d.ts.map +1 -0
  39. package/dist/engine/types.js +7 -0
  40. package/dist/engine/types.js.map +1 -0
  41. package/dist/handlers/create-handler.d.ts +110 -0
  42. package/dist/handlers/create-handler.d.ts.map +1 -0
  43. package/dist/handlers/create-handler.js +140 -0
  44. package/dist/handlers/create-handler.js.map +1 -0
  45. package/dist/handlers/registry.d.ts +15 -0
  46. package/dist/handlers/registry.d.ts.map +1 -0
  47. package/dist/handlers/registry.js +29 -0
  48. package/dist/handlers/registry.js.map +1 -0
  49. package/dist/handlers/types.d.ts +83 -0
  50. package/dist/handlers/types.d.ts.map +1 -0
  51. package/dist/handlers/types.js +5 -0
  52. package/dist/handlers/types.js.map +1 -0
  53. package/dist/index.d.ts +24 -0
  54. package/dist/index.d.ts.map +1 -0
  55. package/dist/index.js +24 -0
  56. package/dist/index.js.map +1 -0
  57. package/dist/migrate.d.ts +19 -0
  58. package/dist/migrate.d.ts.map +1 -0
  59. package/dist/migrate.js +106 -0
  60. package/dist/migrate.js.map +1 -0
  61. package/dist/mutations.d.ts +138 -0
  62. package/dist/mutations.d.ts.map +1 -0
  63. package/dist/mutations.js +611 -0
  64. package/dist/mutations.js.map +1 -0
  65. package/dist/outbox.d.ts +112 -0
  66. package/dist/outbox.d.ts.map +1 -0
  67. package/dist/outbox.js +304 -0
  68. package/dist/outbox.js.map +1 -0
  69. package/dist/plugins/incrementing-version.d.ts +34 -0
  70. package/dist/plugins/incrementing-version.d.ts.map +1 -0
  71. package/dist/plugins/incrementing-version.js +83 -0
  72. package/dist/plugins/incrementing-version.js.map +1 -0
  73. package/dist/plugins/index.d.ts +3 -0
  74. package/dist/plugins/index.d.ts.map +1 -0
  75. package/dist/plugins/index.js +3 -0
  76. package/dist/plugins/index.js.map +1 -0
  77. package/dist/plugins/types.d.ts +49 -0
  78. package/dist/plugins/types.d.ts.map +1 -0
  79. package/dist/plugins/types.js +15 -0
  80. package/dist/plugins/types.js.map +1 -0
  81. package/dist/proxy/connection.d.ts +33 -0
  82. package/dist/proxy/connection.d.ts.map +1 -0
  83. package/dist/proxy/connection.js +153 -0
  84. package/dist/proxy/connection.js.map +1 -0
  85. package/dist/proxy/dialect.d.ts +46 -0
  86. package/dist/proxy/dialect.d.ts.map +1 -0
  87. package/dist/proxy/dialect.js +58 -0
  88. package/dist/proxy/dialect.js.map +1 -0
  89. package/dist/proxy/driver.d.ts +42 -0
  90. package/dist/proxy/driver.d.ts.map +1 -0
  91. package/dist/proxy/driver.js +78 -0
  92. package/dist/proxy/driver.js.map +1 -0
  93. package/dist/proxy/index.d.ts +10 -0
  94. package/dist/proxy/index.d.ts.map +1 -0
  95. package/dist/proxy/index.js +10 -0
  96. package/dist/proxy/index.js.map +1 -0
  97. package/dist/proxy/mutations.d.ts +9 -0
  98. package/dist/proxy/mutations.d.ts.map +1 -0
  99. package/dist/proxy/mutations.js +11 -0
  100. package/dist/proxy/mutations.js.map +1 -0
  101. package/dist/pull-engine.d.ts +45 -0
  102. package/dist/pull-engine.d.ts.map +1 -0
  103. package/dist/pull-engine.js +391 -0
  104. package/dist/pull-engine.js.map +1 -0
  105. package/dist/push-engine.d.ts +18 -0
  106. package/dist/push-engine.d.ts.map +1 -0
  107. package/dist/push-engine.js +155 -0
  108. package/dist/push-engine.js.map +1 -0
  109. package/dist/query/FingerprintCollector.d.ts +18 -0
  110. package/dist/query/FingerprintCollector.d.ts.map +1 -0
  111. package/dist/query/FingerprintCollector.js +28 -0
  112. package/dist/query/FingerprintCollector.js.map +1 -0
  113. package/dist/query/QueryContext.d.ts +33 -0
  114. package/dist/query/QueryContext.d.ts.map +1 -0
  115. package/dist/query/QueryContext.js +16 -0
  116. package/dist/query/QueryContext.js.map +1 -0
  117. package/dist/query/fingerprint.d.ts +61 -0
  118. package/dist/query/fingerprint.d.ts.map +1 -0
  119. package/dist/query/fingerprint.js +91 -0
  120. package/dist/query/fingerprint.js.map +1 -0
  121. package/dist/query/index.d.ts +7 -0
  122. package/dist/query/index.d.ts.map +1 -0
  123. package/dist/query/index.js +7 -0
  124. package/dist/query/index.js.map +1 -0
  125. package/dist/query/tracked-select.d.ts +18 -0
  126. package/dist/query/tracked-select.d.ts.map +1 -0
  127. package/dist/query/tracked-select.js +90 -0
  128. package/dist/query/tracked-select.js.map +1 -0
  129. package/dist/schema.d.ts +83 -0
  130. package/dist/schema.d.ts.map +1 -0
  131. package/dist/schema.js +7 -0
  132. package/dist/schema.js.map +1 -0
  133. package/dist/sync-loop.d.ts +32 -0
  134. package/dist/sync-loop.d.ts.map +1 -0
  135. package/dist/sync-loop.js +249 -0
  136. package/dist/sync-loop.js.map +1 -0
  137. package/dist/utils/id.d.ts +8 -0
  138. package/dist/utils/id.d.ts.map +1 -0
  139. package/dist/utils/id.js +19 -0
  140. package/dist/utils/id.js.map +1 -0
  141. package/package.json +58 -0
  142. package/src/blobs/index.ts +7 -0
  143. package/src/blobs/manager.ts +1027 -0
  144. package/src/blobs/migrate.ts +67 -0
  145. package/src/blobs/types.ts +84 -0
  146. package/src/client.ts +1222 -0
  147. package/src/conflicts.ts +180 -0
  148. package/src/create-client.ts +297 -0
  149. package/src/engine/SyncEngine.ts +1337 -0
  150. package/src/engine/index.ts +6 -0
  151. package/src/engine/types.ts +268 -0
  152. package/src/handlers/create-handler.ts +287 -0
  153. package/src/handlers/registry.ts +36 -0
  154. package/src/handlers/types.ts +102 -0
  155. package/src/index.ts +25 -0
  156. package/src/migrate.ts +122 -0
  157. package/src/mutations.ts +926 -0
  158. package/src/outbox.ts +397 -0
  159. package/src/plugins/incrementing-version.ts +133 -0
  160. package/src/plugins/index.ts +2 -0
  161. package/src/plugins/types.ts +63 -0
  162. package/src/proxy/connection.ts +191 -0
  163. package/src/proxy/dialect.ts +76 -0
  164. package/src/proxy/driver.ts +126 -0
  165. package/src/proxy/index.ts +10 -0
  166. package/src/proxy/mutations.ts +18 -0
  167. package/src/pull-engine.ts +518 -0
  168. package/src/push-engine.ts +201 -0
  169. package/src/query/FingerprintCollector.ts +29 -0
  170. package/src/query/QueryContext.ts +54 -0
  171. package/src/query/fingerprint.ts +109 -0
  172. package/src/query/index.ts +10 -0
  173. package/src/query/tracked-select.ts +139 -0
  174. package/src/schema.ts +94 -0
  175. package/src/sync-loop.ts +368 -0
  176. package/src/utils/id.ts +20 -0
@@ -0,0 +1,29 @@
1
+ /**
2
+ * @syncular/client - Fingerprint Collector
3
+ *
4
+ * Accumulates fingerprint data from multiple queries.
5
+ */
6
+
7
+ /**
8
+ * Fingerprint collector accumulates fingerprint data from multiple queries
9
+ */
10
+ export class FingerprintCollector {
11
+ private fingerprints: string[] = [];
12
+
13
+ add(fingerprint: string): void {
14
+ this.fingerprints.push(fingerprint);
15
+ }
16
+
17
+ /**
18
+ * Get combined fingerprint from all collected fingerprints
19
+ */
20
+ getCombined(): string {
21
+ if (this.fingerprints.length === 0) return '';
22
+ if (this.fingerprints.length === 1) return this.fingerprints[0] ?? '';
23
+ return this.fingerprints.join('|');
24
+ }
25
+
26
+ clear(): void {
27
+ this.fingerprints = [];
28
+ }
29
+ }
@@ -0,0 +1,54 @@
1
+ /**
2
+ * @syncular/client - Query Context
3
+ *
4
+ * Provides a query context with tracked selectFrom for scope tracking
5
+ * and automatic fingerprint generation.
6
+ */
7
+
8
+ import type { Kysely } from 'kysely';
9
+ import type { SyncClientDb } from '../schema';
10
+ import type { FingerprintCollector } from './FingerprintCollector';
11
+ import type { MutationTimestampSource } from './fingerprint';
12
+ import { createTrackedSelectFrom } from './tracked-select';
13
+
14
+ export type TrackedSelectFrom<DB> = ReturnType<
15
+ typeof createTrackedSelectFrom<DB>
16
+ >;
17
+
18
+ /**
19
+ * Query context provided to query functions.
20
+ *
21
+ * Only `selectFrom` is exposed to ensure proper scope tracking and fingerprinting.
22
+ * If you need raw database access, use the db directly outside the query function.
23
+ */
24
+ export interface QueryContext<DB extends SyncClientDb = SyncClientDb> {
25
+ /**
26
+ * Wrapped selectFrom that:
27
+ * 1. Registers table as watched scope
28
+ * 2. Intercepts .execute() to auto-detect fingerprinting mode:
29
+ * - Result has keyField (default: 'id')? -> row-level fingerprinting
30
+ * - No keyField? -> value-based fingerprinting (for aggregates)
31
+ */
32
+ selectFrom: TrackedSelectFrom<DB>;
33
+ }
34
+
35
+ /**
36
+ * Create a query context with tracked selectFrom.
37
+ */
38
+ export function createQueryContext<DB extends SyncClientDb>(
39
+ db: Kysely<DB>,
40
+ scopeCollector: Set<string>,
41
+ fingerprintCollector: FingerprintCollector,
42
+ engine: MutationTimestampSource,
43
+ keyField = 'id'
44
+ ): QueryContext<DB> {
45
+ return {
46
+ selectFrom: createTrackedSelectFrom(
47
+ db,
48
+ scopeCollector,
49
+ fingerprintCollector,
50
+ engine,
51
+ keyField
52
+ ),
53
+ };
54
+ }
@@ -0,0 +1,109 @@
1
+ /**
2
+ * @syncular/client - Fingerprint-based rerender optimization utilities
3
+ *
4
+ * Provides efficient fingerprint computation for query results to avoid
5
+ * expensive deep equality checks. Uses mutation timestamps from the SyncEngine.
6
+ */
7
+
8
+ export interface MutationTimestampSource {
9
+ getMutationTimestamp(table: string, rowId: string): number;
10
+ }
11
+
12
+ /**
13
+ * Compute a fingerprint for query results based on length + ids + mutation timestamps.
14
+ * Much faster than deep equality for large datasets.
15
+ *
16
+ * Fingerprint format: `length:id1@ts1,id2@ts2,...`
17
+ *
18
+ * @param rows - Query result rows (must have an id-like field)
19
+ * @param engine - SyncEngine to look up mutation timestamps
20
+ * @param table - Table name for timestamp lookup
21
+ * @param keyField - Field name to use as row identifier (default: 'id')
22
+ * @returns Fingerprint string for comparison
23
+ *
24
+ * @example
25
+ * ```ts
26
+ * const fingerprint = computeFingerprint(tasks, engine, 'tasks', 'id');
27
+ * // Returns: "3:abc@1706123456789,def@1706123456790,ghi@0"
28
+ * ```
29
+ */
30
+ export function computeFingerprint<T extends Record<string, unknown>>(
31
+ rows: T[],
32
+ engine: MutationTimestampSource,
33
+ table: string,
34
+ keyField = 'id'
35
+ ): string {
36
+ if (rows.length === 0) return '0:';
37
+
38
+ const parts: string[] = [];
39
+ for (const row of rows) {
40
+ const id = String(row[keyField] ?? '');
41
+ const ts = engine.getMutationTimestamp(table, id);
42
+ parts.push(`${id}@${ts}`);
43
+ }
44
+
45
+ return `${rows.length}:${parts.join(',')}`;
46
+ }
47
+
48
+ /**
49
+ * Check if rows have the required key field for fingerprinting.
50
+ * Returns true for empty arrays (no data to fingerprint).
51
+ *
52
+ * @param rows - Query result rows to check
53
+ * @param keyField - Field name to check for (default: 'id')
54
+ * @returns true if rows can be fingerprinted, false otherwise
55
+ *
56
+ * @example
57
+ * ```ts
58
+ * // Can fingerprint - rows have 'id' field
59
+ * canFingerprint([{ id: '1', name: 'foo' }]); // true
60
+ *
61
+ * // Cannot fingerprint - rows lack 'id' field (aggregates, etc.)
62
+ * canFingerprint([{ count: 42 }]); // false
63
+ * ```
64
+ */
65
+ export function canFingerprint<T>(rows: T[], keyField = 'id'): boolean {
66
+ if (rows.length === 0) return true;
67
+ return keyField in (rows[0] as Record<string, unknown>);
68
+ }
69
+
70
+ /**
71
+ * Compute row-level fingerprint from query results.
72
+ * Format: `table:count:id1@ts1,id2@ts2,...`
73
+ */
74
+ export function computeRowFingerprint(
75
+ rows: unknown[],
76
+ table: string,
77
+ engine: MutationTimestampSource,
78
+ keyField: string
79
+ ): string {
80
+ if (rows.length === 0) return `${table}:0:`;
81
+
82
+ const parts: string[] = [];
83
+ for (const row of rows) {
84
+ const r = row as Record<string, unknown>;
85
+ const id = String(r[keyField] ?? '');
86
+ const ts = engine.getMutationTimestamp(table, id);
87
+ parts.push(`${id}@${ts}`);
88
+ }
89
+
90
+ return `${table}:${rows.length}:${parts.join(',')}`;
91
+ }
92
+
93
+ /**
94
+ * Compute value-based fingerprint for aggregate/scalar queries.
95
+ * Format: `table:hash(value)`
96
+ */
97
+ export function computeValueFingerprint(table: string, value: unknown): string {
98
+ // Simple hash of the JSON representation
99
+ const json = JSON.stringify(value);
100
+ return `${table}:${json}`;
101
+ }
102
+
103
+ /**
104
+ * Check if result rows have the key field for row-level fingerprinting.
105
+ */
106
+ export function hasKeyField(rows: unknown[], keyField: string): boolean {
107
+ if (rows.length === 0) return false;
108
+ return keyField in (rows[0] as Record<string, unknown>);
109
+ }
@@ -0,0 +1,10 @@
1
+ /**
2
+ * @syncular/client - Query utilities exports
3
+ */
4
+
5
+ export { FingerprintCollector } from './FingerprintCollector';
6
+ export {
7
+ canFingerprint,
8
+ computeFingerprint,
9
+ } from './fingerprint';
10
+ export { createQueryContext, type QueryContext } from './QueryContext';
@@ -0,0 +1,139 @@
1
+ /**
2
+ * @syncular/client - Tracked SelectFrom
3
+ *
4
+ * Provides a wrapped selectFrom that:
5
+ * 1. Tracks tables as watched scopes when called
6
+ * 2. Intercepts .execute() to auto-generate fingerprints
7
+ */
8
+
9
+ import type { Kysely } from 'kysely';
10
+ import type { FingerprintCollector } from './FingerprintCollector';
11
+
12
+ /** Portable type alias for Kysely's selectFrom method signature */
13
+ type TrackedSelectFrom<DB> = Kysely<DB>['selectFrom'];
14
+
15
+ import {
16
+ computeRowFingerprint,
17
+ computeValueFingerprint,
18
+ hasKeyField,
19
+ type MutationTimestampSource,
20
+ } from './fingerprint';
21
+
22
+ /**
23
+ * Create a proxy that intercepts execute() to compute fingerprints.
24
+ */
25
+
26
+ type ExecutableQuery = {
27
+ execute: () => Promise<unknown>;
28
+ executeTakeFirst: () => Promise<unknown>;
29
+ executeTakeFirstOrThrow: () => Promise<unknown>;
30
+ };
31
+
32
+ function isExecutableQuery(value: unknown): value is ExecutableQuery {
33
+ if (typeof value !== 'object' || value === null) return false;
34
+ return (
35
+ typeof Reflect.get(value, 'execute') === 'function' &&
36
+ typeof Reflect.get(value, 'executeTakeFirst') === 'function' &&
37
+ typeof Reflect.get(value, 'executeTakeFirstOrThrow') === 'function'
38
+ );
39
+ }
40
+
41
+ function createExecuteProxy<B extends ExecutableQuery>(
42
+ builder: B,
43
+ table: string,
44
+ collector: FingerprintCollector,
45
+ engine: MutationTimestampSource,
46
+ keyField: string
47
+ ): B {
48
+ return new Proxy(builder, {
49
+ get(target, prop: string | symbol) {
50
+ if (prop === 'execute') {
51
+ return async () => {
52
+ const rows = await target.execute();
53
+ // Auto-detect fingerprint mode based on result shape
54
+ if (Array.isArray(rows)) {
55
+ if (hasKeyField(rows, keyField)) {
56
+ // Row-level fingerprinting - result has keyField
57
+ const fp = computeRowFingerprint(rows, table, engine, keyField);
58
+ collector.add(fp);
59
+ } else {
60
+ // Value-based fingerprinting - for aggregates/scalars
61
+ const fp = computeValueFingerprint(table, rows);
62
+ collector.add(fp);
63
+ }
64
+ } else {
65
+ // Unexpected, but keep rerender behavior deterministic.
66
+ const fp = computeValueFingerprint(table, rows);
67
+ collector.add(fp);
68
+ }
69
+ return rows;
70
+ };
71
+ }
72
+ if (prop === 'executeTakeFirst') {
73
+ return async () => {
74
+ const row = await target.executeTakeFirst();
75
+ // Value-based fingerprinting for single-row queries
76
+ const fp = computeValueFingerprint(table, row);
77
+ collector.add(fp);
78
+ return row;
79
+ };
80
+ }
81
+ if (prop === 'executeTakeFirstOrThrow') {
82
+ return async () => {
83
+ const row = await target.executeTakeFirstOrThrow();
84
+ // Value-based fingerprinting for single-row queries
85
+ const fp = computeValueFingerprint(table, row);
86
+ collector.add(fp);
87
+ return row;
88
+ };
89
+ }
90
+ // For other methods, return wrapped builder for chaining
91
+ const value = Reflect.get(target, prop, target);
92
+ if (typeof value === 'function') {
93
+ return (...args: unknown[]) => {
94
+ const result = Reflect.apply(value, target, args);
95
+ if (isExecutableQuery(result)) {
96
+ return createExecuteProxy(
97
+ result,
98
+ table,
99
+ collector,
100
+ engine,
101
+ keyField
102
+ );
103
+ }
104
+ return result;
105
+ };
106
+ }
107
+ return value;
108
+ },
109
+ });
110
+ }
111
+
112
+ /**
113
+ * Create a tracked selectFrom that registers scopes and generates fingerprints.
114
+ */
115
+ export function createTrackedSelectFrom<DB>(
116
+ db: Kysely<DB>,
117
+ scopeCollector: Set<string>,
118
+ fingerprintCollector: FingerprintCollector,
119
+ engine: MutationTimestampSource,
120
+ keyField = 'id'
121
+ ): TrackedSelectFrom<DB> {
122
+ const selectFrom = <TB extends keyof DB & string>(table: TB) => {
123
+ // 1. Register this table as a watched scope
124
+ scopeCollector.add(table);
125
+
126
+ // 2. Get the real query builder
127
+ const builder = db.selectFrom(table);
128
+
129
+ // 3. Return a proxy that intercepts .execute()
130
+ return createExecuteProxy(
131
+ builder,
132
+ table,
133
+ fingerprintCollector,
134
+ engine,
135
+ keyField
136
+ );
137
+ };
138
+ return selectFrom as TrackedSelectFrom<DB>;
139
+ }
package/src/schema.ts ADDED
@@ -0,0 +1,94 @@
1
+ /**
2
+ * @syncular/client - Sync client schema types (SQLite reference)
3
+ *
4
+ * These tables are sync-internal and should not collide with app tables.
5
+ */
6
+
7
+ import type { Generated, Kysely } from 'kysely';
8
+ import type { SyncBlobClientDb } from './blobs/types';
9
+
10
+ /**
11
+ * Database executor type that both Kysely and Transaction satisfy.
12
+ * Use this when a function needs to work with either a raw Kysely instance
13
+ * or within a transaction context.
14
+ */
15
+ export type SyncClientExecutor<DB extends SyncClientDb = SyncClientDb> = Pick<
16
+ Kysely<DB>,
17
+ 'selectFrom' | 'insertInto' | 'updateTable' | 'deleteFrom'
18
+ >;
19
+
20
+ export type OutboxCommitStatus = 'pending' | 'sending' | 'acked' | 'failed';
21
+
22
+ export type SubscriptionStatus = 'active' | 'revoked';
23
+
24
+ export interface SyncSubscriptionStateTable {
25
+ /** Logical profile id (default: 'default') */
26
+ state_id: string;
27
+ /** Subscription identifier (client-chosen) */
28
+ subscription_id: string;
29
+ /** Shape name (table) for this subscription */
30
+ shape: string;
31
+ /** JSON string of ScopeValues for this subscription */
32
+ scopes_json: string;
33
+ /** JSON string of params */
34
+ params_json: string;
35
+ /** Per-subscription cursor (last applied commit_seq) */
36
+ cursor: number;
37
+ /**
38
+ * JSON string of SyncBootstrapState (or null).
39
+ * When set, indicates a paginated bootstrap is in progress for this subscription.
40
+ */
41
+ bootstrap_state_json: string | null;
42
+ /** active | revoked */
43
+ status: SubscriptionStatus;
44
+ created_at: number;
45
+ updated_at: number;
46
+ }
47
+
48
+ export interface SyncOutboxCommitsTable {
49
+ /** Local row id (uuid) */
50
+ id: string;
51
+ /** Client-provided idempotency key (uuid) */
52
+ client_commit_id: string;
53
+ /** pending | sending | acked | failed */
54
+ status: OutboxCommitStatus;
55
+ /** JSON string of SyncOperation[] */
56
+ operations_json: string;
57
+ /** JSON string of the last SyncPushResponse (optional) */
58
+ last_response_json: string | null;
59
+ /** Last error (if any) */
60
+ error: string | null;
61
+ /** Created timestamp (ms since epoch) */
62
+ created_at: number;
63
+ /** Updated timestamp (ms since epoch) */
64
+ updated_at: number;
65
+ /** How many send attempts have been made */
66
+ attempt_count: Generated<number>;
67
+ /** Server commit_seq if acked (optional) */
68
+ acked_commit_seq: number | null;
69
+ /** Client schema version when commit was created (default: 1 for legacy) */
70
+ schema_version: Generated<number>;
71
+ }
72
+
73
+ export interface SyncClientDb extends SyncBlobClientDb {
74
+ sync_subscription_state: SyncSubscriptionStateTable;
75
+ sync_outbox_commits: SyncOutboxCommitsTable;
76
+ sync_conflicts: SyncConflictsTable;
77
+ }
78
+
79
+ export type ConflictResultStatus = 'conflict' | 'error';
80
+
81
+ export interface SyncConflictsTable {
82
+ id: string;
83
+ outbox_commit_id: string;
84
+ client_commit_id: string;
85
+ op_index: number;
86
+ result_status: ConflictResultStatus;
87
+ message: string;
88
+ code: string | null;
89
+ server_version: number | null;
90
+ server_row_json: string | null;
91
+ created_at: number;
92
+ resolved_at: number | null;
93
+ resolution: string | null;
94
+ }