syncorejs 0.2.1 → 0.2.3

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 (169) hide show
  1. package/README.md +2 -1
  2. package/dist/_vendor/cli/app.d.mts.map +1 -1
  3. package/dist/_vendor/cli/app.mjs +330 -46
  4. package/dist/_vendor/cli/app.mjs.map +1 -1
  5. package/dist/_vendor/cli/context.mjs +27 -9
  6. package/dist/_vendor/cli/context.mjs.map +1 -1
  7. package/dist/_vendor/cli/dev-session.mjs.map +1 -1
  8. package/dist/_vendor/cli/doctor.mjs +513 -46
  9. package/dist/_vendor/cli/doctor.mjs.map +1 -1
  10. package/dist/_vendor/cli/errors.mjs.map +1 -1
  11. package/dist/_vendor/cli/help.mjs.map +1 -1
  12. package/dist/_vendor/cli/index.mjs +9 -2
  13. package/dist/_vendor/cli/index.mjs.map +1 -1
  14. package/dist/_vendor/cli/messages.mjs +5 -4
  15. package/dist/_vendor/cli/messages.mjs.map +1 -1
  16. package/dist/_vendor/cli/preflight.mjs.map +1 -1
  17. package/dist/_vendor/cli/project.mjs +125 -27
  18. package/dist/_vendor/cli/project.mjs.map +1 -1
  19. package/dist/_vendor/cli/render.mjs +57 -9
  20. package/dist/_vendor/cli/render.mjs.map +1 -1
  21. package/dist/_vendor/cli/targets.mjs +4 -3
  22. package/dist/_vendor/cli/targets.mjs.map +1 -1
  23. package/dist/_vendor/core/cli.d.mts +20 -4
  24. package/dist/_vendor/core/cli.d.mts.map +1 -1
  25. package/dist/_vendor/core/cli.mjs +458 -133
  26. package/dist/_vendor/core/cli.mjs.map +1 -1
  27. package/dist/_vendor/core/devtools-auth.mjs +60 -0
  28. package/dist/_vendor/core/devtools-auth.mjs.map +1 -0
  29. package/dist/_vendor/core/index.d.mts +5 -3
  30. package/dist/_vendor/core/index.mjs +22 -2
  31. package/dist/_vendor/core/index.mjs.map +1 -1
  32. package/dist/_vendor/core/runtime/components.d.mts +111 -0
  33. package/dist/_vendor/core/runtime/components.d.mts.map +1 -0
  34. package/dist/_vendor/core/runtime/components.mjs +186 -0
  35. package/dist/_vendor/core/runtime/components.mjs.map +1 -0
  36. package/dist/_vendor/core/runtime/devtools.d.mts +4 -4
  37. package/dist/_vendor/core/runtime/devtools.d.mts.map +1 -1
  38. package/dist/_vendor/core/runtime/devtools.mjs +178 -60
  39. package/dist/_vendor/core/runtime/devtools.mjs.map +1 -1
  40. package/dist/_vendor/core/runtime/functions.d.mts +398 -16
  41. package/dist/_vendor/core/runtime/functions.d.mts.map +1 -1
  42. package/dist/_vendor/core/runtime/functions.mjs +74 -3
  43. package/dist/_vendor/core/runtime/functions.mjs.map +1 -1
  44. package/dist/_vendor/core/runtime/id.d.mts.map +1 -1
  45. package/dist/_vendor/core/runtime/id.mjs.map +1 -1
  46. package/dist/_vendor/core/runtime/internal/engines/devtoolsEngine.mjs +83 -0
  47. package/dist/_vendor/core/runtime/internal/engines/devtoolsEngine.mjs.map +1 -0
  48. package/dist/_vendor/core/runtime/internal/engines/executionEngine.mjs +720 -0
  49. package/dist/_vendor/core/runtime/internal/engines/executionEngine.mjs.map +1 -0
  50. package/dist/_vendor/core/runtime/internal/engines/reactivityEngine.mjs +234 -0
  51. package/dist/_vendor/core/runtime/internal/engines/reactivityEngine.mjs.map +1 -0
  52. package/dist/_vendor/core/runtime/internal/engines/schedulerEngine.mjs +255 -0
  53. package/dist/_vendor/core/runtime/internal/engines/schedulerEngine.mjs.map +1 -0
  54. package/dist/_vendor/core/runtime/internal/engines/schemaEngine.mjs +200 -0
  55. package/dist/_vendor/core/runtime/internal/engines/schemaEngine.mjs.map +1 -0
  56. package/dist/_vendor/core/runtime/internal/engines/shared.mjs +252 -0
  57. package/dist/_vendor/core/runtime/internal/engines/shared.mjs.map +1 -0
  58. package/dist/_vendor/core/runtime/internal/engines/storageEngine.mjs +145 -0
  59. package/dist/_vendor/core/runtime/internal/engines/storageEngine.mjs.map +1 -0
  60. package/dist/_vendor/core/runtime/internal/runtimeKernel.mjs +221 -0
  61. package/dist/_vendor/core/runtime/internal/runtimeKernel.mjs.map +1 -0
  62. package/dist/_vendor/core/runtime/internal/runtimeStatus.mjs +32 -0
  63. package/dist/_vendor/core/runtime/internal/runtimeStatus.mjs.map +1 -0
  64. package/dist/_vendor/core/runtime/internal/systemMeta.mjs +61 -0
  65. package/dist/_vendor/core/runtime/internal/systemMeta.mjs.map +1 -0
  66. package/dist/_vendor/core/runtime/internal/transactionCoordinator.mjs +41 -0
  67. package/dist/_vendor/core/runtime/internal/transactionCoordinator.mjs.map +1 -0
  68. package/dist/_vendor/core/runtime/runtime.d.mts +1187 -202
  69. package/dist/_vendor/core/runtime/runtime.d.mts.map +1 -1
  70. package/dist/_vendor/core/runtime/runtime.mjs +73 -1365
  71. package/dist/_vendor/core/runtime/runtime.mjs.map +1 -1
  72. package/dist/_vendor/core/transport.d.mts +113 -0
  73. package/dist/_vendor/core/transport.d.mts.map +1 -0
  74. package/dist/_vendor/core/transport.mjs +428 -0
  75. package/dist/_vendor/core/transport.mjs.map +1 -0
  76. package/dist/_vendor/devtools-protocol/index.d.ts +187 -4
  77. package/dist/_vendor/devtools-protocol/index.d.ts.map +1 -1
  78. package/dist/_vendor/devtools-protocol/index.js +25 -9
  79. package/dist/_vendor/devtools-protocol/index.js.map +1 -1
  80. package/dist/_vendor/next/config.d.ts +3 -4
  81. package/dist/_vendor/next/config.d.ts.map +1 -1
  82. package/dist/_vendor/next/config.js +37 -19
  83. package/dist/_vendor/next/config.js.map +1 -1
  84. package/dist/_vendor/next/index.d.ts +109 -29
  85. package/dist/_vendor/next/index.d.ts.map +1 -1
  86. package/dist/_vendor/next/index.js +104 -26
  87. package/dist/_vendor/next/index.js.map +1 -1
  88. package/dist/_vendor/platform-expo/index.d.ts +156 -37
  89. package/dist/_vendor/platform-expo/index.d.ts.map +1 -1
  90. package/dist/_vendor/platform-expo/index.js +80 -12
  91. package/dist/_vendor/platform-expo/index.js.map +1 -1
  92. package/dist/_vendor/platform-expo/react.d.ts.map +1 -1
  93. package/dist/_vendor/platform-expo/react.js +11 -10
  94. package/dist/_vendor/platform-expo/react.js.map +1 -1
  95. package/dist/_vendor/platform-expo/web-sqljs-wasm.js +16 -0
  96. package/dist/_vendor/platform-expo/web-sqljs-wasm.js.map +1 -0
  97. package/dist/_vendor/platform-node/index.d.mts +192 -24
  98. package/dist/_vendor/platform-node/index.d.mts.map +1 -1
  99. package/dist/_vendor/platform-node/index.mjs +236 -97
  100. package/dist/_vendor/platform-node/index.mjs.map +1 -1
  101. package/dist/_vendor/platform-node/ipc-react.d.mts.map +1 -1
  102. package/dist/_vendor/platform-node/ipc-react.mjs +15 -2
  103. package/dist/_vendor/platform-node/ipc-react.mjs.map +1 -1
  104. package/dist/_vendor/platform-node/ipc.d.mts +11 -35
  105. package/dist/_vendor/platform-node/ipc.d.mts.map +1 -1
  106. package/dist/_vendor/platform-node/ipc.mjs +3 -273
  107. package/dist/_vendor/platform-node/ipc.mjs.map +1 -1
  108. package/dist/_vendor/platform-web/external-change.d.ts +43 -1
  109. package/dist/_vendor/platform-web/external-change.d.ts.map +1 -1
  110. package/dist/_vendor/platform-web/external-change.js +32 -1
  111. package/dist/_vendor/platform-web/external-change.js.map +1 -1
  112. package/dist/_vendor/platform-web/index.d.ts +323 -51
  113. package/dist/_vendor/platform-web/index.d.ts.map +1 -1
  114. package/dist/_vendor/platform-web/index.js +233 -30
  115. package/dist/_vendor/platform-web/index.js.map +1 -1
  116. package/dist/_vendor/platform-web/indexeddb.d.ts +12 -0
  117. package/dist/_vendor/platform-web/indexeddb.d.ts.map +1 -1
  118. package/dist/_vendor/platform-web/indexeddb.js +10 -0
  119. package/dist/_vendor/platform-web/indexeddb.js.map +1 -1
  120. package/dist/_vendor/platform-web/opfs.d.ts +13 -0
  121. package/dist/_vendor/platform-web/opfs.d.ts.map +1 -1
  122. package/dist/_vendor/platform-web/opfs.js +12 -0
  123. package/dist/_vendor/platform-web/opfs.js.map +1 -1
  124. package/dist/_vendor/platform-web/persistence.d.ts +54 -0
  125. package/dist/_vendor/platform-web/persistence.d.ts.map +1 -1
  126. package/dist/_vendor/platform-web/persistence.js +15 -0
  127. package/dist/_vendor/platform-web/persistence.js.map +1 -1
  128. package/dist/_vendor/platform-web/react.d.ts +1 -2
  129. package/dist/_vendor/platform-web/react.d.ts.map +1 -1
  130. package/dist/_vendor/platform-web/react.js +27 -13
  131. package/dist/_vendor/platform-web/react.js.map +1 -1
  132. package/dist/_vendor/platform-web/sqljs.js +10 -1
  133. package/dist/_vendor/platform-web/sqljs.js.map +1 -1
  134. package/dist/_vendor/platform-web/web-sqljs-wasm.js +8 -0
  135. package/dist/_vendor/platform-web/web-sqljs-wasm.js.map +1 -0
  136. package/dist/_vendor/platform-web/worker.d.ts +71 -44
  137. package/dist/_vendor/platform-web/worker.d.ts.map +1 -1
  138. package/dist/_vendor/platform-web/worker.js +40 -271
  139. package/dist/_vendor/platform-web/worker.js.map +1 -1
  140. package/dist/_vendor/react/index.d.ts +222 -23
  141. package/dist/_vendor/react/index.d.ts.map +1 -1
  142. package/dist/_vendor/react/index.js +476 -63
  143. package/dist/_vendor/react/index.js.map +1 -1
  144. package/dist/_vendor/schema/definition.d.ts +151 -37
  145. package/dist/_vendor/schema/definition.d.ts.map +1 -1
  146. package/dist/_vendor/schema/definition.js +102 -20
  147. package/dist/_vendor/schema/definition.js.map +1 -1
  148. package/dist/_vendor/schema/index.d.ts +4 -4
  149. package/dist/_vendor/schema/index.js +2 -2
  150. package/dist/_vendor/schema/planner.d.ts +19 -2
  151. package/dist/_vendor/schema/planner.d.ts.map +1 -1
  152. package/dist/_vendor/schema/planner.js +79 -3
  153. package/dist/_vendor/schema/planner.js.map +1 -1
  154. package/dist/_vendor/schema/validators.d.ts +279 -83
  155. package/dist/_vendor/schema/validators.d.ts.map +1 -1
  156. package/dist/_vendor/schema/validators.js +330 -38
  157. package/dist/_vendor/schema/validators.js.map +1 -1
  158. package/dist/_vendor/svelte/index.d.ts +245 -19
  159. package/dist/_vendor/svelte/index.d.ts.map +1 -1
  160. package/dist/_vendor/svelte/index.js +443 -20
  161. package/dist/_vendor/svelte/index.js.map +1 -1
  162. package/dist/browser.d.ts.map +1 -1
  163. package/dist/cli.js +3 -1
  164. package/dist/cli.js.map +1 -1
  165. package/dist/components.d.ts +2 -0
  166. package/dist/components.js +2 -0
  167. package/dist/index.d.ts +3 -2
  168. package/dist/index.js +2 -1
  169. package/package.json +29 -21
@@ -0,0 +1,252 @@
1
+ import { searchIndexTableName } from "../../../../schema/index.js";
2
+ import { fromZonedTime, toZonedTime } from "date-fns-tz";
3
+ //#region src/runtime/internal/engines/shared.ts
4
+ const PREVIEW_MAX_DEPTH = 5;
5
+ const PREVIEW_MAX_ARRAY_ITEMS = 50;
6
+ const PREVIEW_MAX_OBJECT_KEYS = 80;
7
+ const PREVIEW_MAX_STRING_LENGTH = 4e3;
8
+ function createDevtoolsPreview(value) {
9
+ const seen = /* @__PURE__ */ new WeakSet();
10
+ let truncated = false;
11
+ const preview = (input, depth) => {
12
+ if (typeof input === "string") {
13
+ if (input.length > PREVIEW_MAX_STRING_LENGTH) {
14
+ truncated = true;
15
+ return `${input.slice(0, PREVIEW_MAX_STRING_LENGTH)}...`;
16
+ }
17
+ return input;
18
+ }
19
+ if (input === null || typeof input === "number" || typeof input === "boolean") return input;
20
+ if (typeof input === "bigint") return `${input.toString()}n`;
21
+ if (input === void 0) return "[undefined]";
22
+ if (typeof input === "function") {
23
+ truncated = true;
24
+ return "[function]";
25
+ }
26
+ if (typeof input === "symbol") {
27
+ truncated = true;
28
+ return input.toString();
29
+ }
30
+ if (input instanceof Error) return {
31
+ name: input.name,
32
+ message: input.message,
33
+ ...input.stack ? { stack: input.stack } : {}
34
+ };
35
+ if (input instanceof Date) return input.toISOString();
36
+ if (typeof input !== "object") return String(input);
37
+ if (seen.has(input)) {
38
+ truncated = true;
39
+ return "[circular]";
40
+ }
41
+ if (depth >= PREVIEW_MAX_DEPTH) {
42
+ truncated = true;
43
+ return Array.isArray(input) ? "[array]" : "[object]";
44
+ }
45
+ seen.add(input);
46
+ if (Array.isArray(input)) {
47
+ const items = input.slice(0, PREVIEW_MAX_ARRAY_ITEMS).map((item) => preview(item, depth + 1));
48
+ if (input.length > PREVIEW_MAX_ARRAY_ITEMS) {
49
+ truncated = true;
50
+ items.push(`[${input.length - PREVIEW_MAX_ARRAY_ITEMS} more items]`);
51
+ }
52
+ seen.delete(input);
53
+ return items;
54
+ }
55
+ const entries = Object.entries(input);
56
+ const result = {};
57
+ for (const [key, nested] of entries.slice(0, PREVIEW_MAX_OBJECT_KEYS)) result[key] = preview(nested, depth + 1);
58
+ if (entries.length > PREVIEW_MAX_OBJECT_KEYS) {
59
+ truncated = true;
60
+ result.__truncatedKeys = entries.length - PREVIEW_MAX_OBJECT_KEYS;
61
+ }
62
+ seen.delete(input);
63
+ return result;
64
+ };
65
+ try {
66
+ return {
67
+ kind: "value",
68
+ value: preview(value, 0),
69
+ ...truncated ? { truncated: true } : {}
70
+ };
71
+ } catch (error) {
72
+ return {
73
+ kind: "error",
74
+ message: error instanceof Error ? error.message : String(error),
75
+ truncated: true
76
+ };
77
+ }
78
+ }
79
+ function fieldExpression(tableAlias, field) {
80
+ return `json_extract(${tableAlias ? `${tableAlias}.` : ""}_json, '$.${field}')`;
81
+ }
82
+ function quoteIdentifier(identifier) {
83
+ return `"${identifier.replaceAll("\"", "\"\"")}"`;
84
+ }
85
+ function stableStringify(value) {
86
+ return JSON.stringify(sortValue(value));
87
+ }
88
+ function sortValue(value) {
89
+ if (Array.isArray(value)) return value.map(sortValue);
90
+ if (value && typeof value === "object") return Object.fromEntries(Object.entries(value).sort(([left], [right]) => left.localeCompare(right)).map(([key, nested]) => [key, sortValue(nested)]));
91
+ return value;
92
+ }
93
+ function omitSystemFields(document) {
94
+ const clone = { ...document };
95
+ delete clone._id;
96
+ delete clone._creationTime;
97
+ return clone;
98
+ }
99
+ function toSearchValue(value) {
100
+ if (typeof value === "string") return value;
101
+ if (value === null || value === void 0) return "";
102
+ if (typeof value === "number" || typeof value === "boolean" || typeof value === "bigint") return String(value);
103
+ return stableStringify(value);
104
+ }
105
+ function parseMisfirePolicy(type, windowMs) {
106
+ if (type === "windowed") return {
107
+ type,
108
+ windowMs: windowMs ?? 0
109
+ };
110
+ if (type === "skip" || type === "run_once_if_missed") return { type };
111
+ return { type: "catch_up" };
112
+ }
113
+ function shouldRunMissedJob(scheduledAt, now, policy) {
114
+ if (scheduledAt >= now) return true;
115
+ switch (policy.type) {
116
+ case "catch_up": return true;
117
+ case "run_once_if_missed": return true;
118
+ case "skip": return false;
119
+ case "windowed": return now - scheduledAt <= policy.windowMs;
120
+ }
121
+ }
122
+ function computeNextRun(schedule, fromTimestamp) {
123
+ switch (schedule.type) {
124
+ case "interval": return fromTimestamp + intervalToMs(schedule);
125
+ case "daily": return nextDailyOccurrence(fromTimestamp, schedule);
126
+ case "weekly": return nextWeeklyOccurrence(fromTimestamp, schedule);
127
+ }
128
+ }
129
+ function safeReadRecurringSchedule(scheduleJson) {
130
+ if (!scheduleJson) return;
131
+ try {
132
+ const parsed = JSON.parse(scheduleJson);
133
+ return isRecurringSchedule(parsed) ? parsed : void 0;
134
+ } catch {
135
+ return;
136
+ }
137
+ }
138
+ function normalizeOptionalArgs(args) {
139
+ return args[0] ?? {};
140
+ }
141
+ function splitSchedulerArgs(args) {
142
+ if (args.length === 0) return [{}, void 0];
143
+ if (args.length === 1) {
144
+ const [first] = args;
145
+ if (isMisfirePolicy(first)) return [{}, first];
146
+ return [first ?? {}, void 0];
147
+ }
148
+ return [args[0] ?? {}, args[1]];
149
+ }
150
+ function inferDriverDatabasePath(driver) {
151
+ return driver.databasePath ?? driver.filename;
152
+ }
153
+ function parseCanonicalComponentFunctionName(functionName) {
154
+ const match = /^components\/(.+)\/(public|internal)\/(.+)$/.exec(functionName);
155
+ if (!match) return;
156
+ return {
157
+ componentPath: match[1] ?? "",
158
+ visibility: match[2] ?? "public",
159
+ localName: match[3] ?? ""
160
+ };
161
+ }
162
+ function parseComponentScopedIdentifier(value) {
163
+ const match = /^component:([^:]+):(.+)$/.exec(value);
164
+ if (!match) return;
165
+ return {
166
+ componentPath: match[1] ?? "",
167
+ localId: match[2] ?? ""
168
+ };
169
+ }
170
+ function devtoolsScopesForEvent(event) {
171
+ switch (event.type) {
172
+ case "runtime.connected":
173
+ case "runtime.disconnected": return new Set(["runtime.summary", "runtime.activeQueries"]);
174
+ case "query.executed":
175
+ case "query.invalidated": return new Set(["runtime.summary", "runtime.activeQueries"]);
176
+ case "mutation.committed": return new Set(["runtime.summary", ...event.changedTables.map((table) => `table:${table}`)]);
177
+ case "scheduler.tick": return new Set(["scheduler.jobs", "runtime.summary"]);
178
+ case "storage.updated": return new Set(["runtime.summary"]);
179
+ case "action.completed":
180
+ case "log": return new Set(["runtime.summary"]);
181
+ default: return new Set(["runtime.summary"]);
182
+ }
183
+ }
184
+ function getTableDefinition(schema, tableName) {
185
+ return schema.getTable(tableName);
186
+ }
187
+ function searchIndexKey(tableName, indexName) {
188
+ return `${tableName}:${indexName}`;
189
+ }
190
+ function resolveSearchIndexTableName(tableName, indexName) {
191
+ return searchIndexTableName(tableName, indexName);
192
+ }
193
+ function intervalToMs(schedule) {
194
+ if (schedule.seconds) return schedule.seconds * 1e3;
195
+ if (schedule.minutes) return schedule.minutes * 60 * 1e3;
196
+ return (schedule.hours ?? 1) * 60 * 60 * 1e3;
197
+ }
198
+ function nextDailyOccurrence(fromTimestamp, schedule) {
199
+ const timezone = schedule.timezone ?? "UTC";
200
+ const zonedNow = toZonedTime(new Date(fromTimestamp), timezone);
201
+ const zoned = new Date(zonedNow.getTime());
202
+ zoned.setHours(schedule.hour, schedule.minute, 0, 0);
203
+ if (zoned.getTime() <= zonedNow.getTime()) zoned.setDate(zoned.getDate() + 1);
204
+ return fromZonedTime(zoned, timezone).getTime();
205
+ }
206
+ function nextWeeklyOccurrence(fromTimestamp, schedule) {
207
+ const timezone = schedule.timezone ?? "UTC";
208
+ const zonedNow = toZonedTime(new Date(fromTimestamp), timezone);
209
+ const targetDay = [
210
+ "sunday",
211
+ "monday",
212
+ "tuesday",
213
+ "wednesday",
214
+ "thursday",
215
+ "friday",
216
+ "saturday"
217
+ ].indexOf(schedule.dayOfWeek);
218
+ const zoned = new Date(zonedNow.getTime());
219
+ const delta = (targetDay - zonedNow.getDay() + 7) % 7;
220
+ zoned.setDate(zoned.getDate() + delta);
221
+ zoned.setHours(schedule.hour, schedule.minute, 0, 0);
222
+ if (zoned.getTime() <= zonedNow.getTime()) zoned.setDate(zoned.getDate() + 7);
223
+ return fromZonedTime(zoned, timezone).getTime();
224
+ }
225
+ function isRecurringSchedule(value) {
226
+ if (!isRecord(value) || typeof value.type !== "string") return false;
227
+ switch (value.type) {
228
+ case "interval": return isOptionalNumber(value.seconds) && isOptionalNumber(value.minutes) && isOptionalNumber(value.hours);
229
+ case "daily": return typeof value.hour === "number" && typeof value.minute === "number" && isOptionalString(value.timezone);
230
+ case "weekly": return isDayOfWeek(value.dayOfWeek) && typeof value.hour === "number" && typeof value.minute === "number" && isOptionalString(value.timezone);
231
+ default: return false;
232
+ }
233
+ }
234
+ function isMisfirePolicy(value) {
235
+ return typeof value === "object" && value !== null && "type" in value && typeof value.type === "string";
236
+ }
237
+ function isRecord(value) {
238
+ return typeof value === "object" && value !== null;
239
+ }
240
+ function isOptionalNumber(value) {
241
+ return value === void 0 || typeof value === "number";
242
+ }
243
+ function isOptionalString(value) {
244
+ return value === void 0 || typeof value === "string";
245
+ }
246
+ function isDayOfWeek(value) {
247
+ return value === "sunday" || value === "monday" || value === "tuesday" || value === "wednesday" || value === "thursday" || value === "friday" || value === "saturday";
248
+ }
249
+ //#endregion
250
+ export { computeNextRun, createDevtoolsPreview, devtoolsScopesForEvent, fieldExpression, getTableDefinition, inferDriverDatabasePath, normalizeOptionalArgs, omitSystemFields, parseCanonicalComponentFunctionName, parseComponentScopedIdentifier, parseMisfirePolicy, quoteIdentifier, resolveSearchIndexTableName, safeReadRecurringSchedule, searchIndexKey, shouldRunMissedJob, splitSchedulerArgs, stableStringify, toSearchValue };
251
+
252
+ //# sourceMappingURL=shared.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"shared.mjs","names":[],"sources":["../../../../src/runtime/internal/engines/shared.ts"],"sourcesContent":["import { fromZonedTime, toZonedTime } from \"date-fns-tz\";\nimport type {\n DevtoolsPreview,\n DocumentChangePreview,\n SyncoreDevtoolsEvent,\n SyncoreDevtoolsEventOrigin\n} from \"@syncore/devtools-protocol\";\nimport {\n searchIndexTableName,\n type TableDefinition,\n type Validator\n} from \"@syncore/schema\";\nimport type {\n MisfirePolicy,\n RecurringSchedule,\n SyncoreFunctionKind\n} from \"../../functions.js\";\nimport type { SyncoreComponentFunctionMetadata } from \"../../components.js\";\nimport type {\n DevtoolsLiveQueryScope,\n JsonObject,\n QueryCondition,\n QueryExpression,\n SearchQuery,\n SyncoreDataModel,\n SyncoreExternalChangeReason\n} from \"../../runtime.js\";\n\nexport type DatabaseRow = {\n _id: string;\n _creationTime: number;\n _json: string;\n};\n\nexport type DependencyKey = string;\n\nexport type ActiveQueryRecord = {\n id: string;\n functionName: string;\n args: JsonObject;\n listeners: Set<() => void>;\n consumers: number;\n dependencyKeys: Set<DependencyKey>;\n lastResult: unknown;\n lastError: Error | undefined;\n lastRunAt: number;\n};\n\nexport type DevtoolsEventMeta = {\n origin?: SyncoreDevtoolsEventOrigin;\n executionId?: string;\n parentExecutionId?: string;\n schedulerJobId?: string;\n schedulerRun?: boolean;\n};\n\nexport type ScheduledJobRow = {\n id: string;\n function_name: string;\n function_kind: SyncoreFunctionKind;\n args_json: string;\n status: \"scheduled\" | \"completed\" | \"failed\" | \"cancelled\" | \"skipped\";\n run_at: number;\n created_at: number;\n updated_at: number;\n recurring_name: string | null;\n schedule_json: string | null;\n timezone: string | null;\n misfire_policy: string;\n last_run_at: number | null;\n window_ms: number | null;\n};\n\nexport type StorageMetadataRow = {\n _id: string;\n _creationTime: number;\n file_name: string | null;\n content_type: string | null;\n size: number;\n path: string;\n};\n\nexport type StoragePendingRow = {\n _id: string;\n _creationTime: number;\n file_name: string | null;\n content_type: string | null;\n};\n\nexport type QuerySource =\n | { type: \"table\" }\n | { type: \"index\"; name: string; range: QueryCondition[] }\n | { type: \"search\"; name: string; query: SearchQuery };\n\nexport type ExecuteQueryBuilderOptions = {\n tableName: string;\n source: QuerySource;\n filterExpression: QueryExpression | undefined;\n orderDirection: \"asc\" | \"desc\";\n dependencyCollector?: Set<DependencyKey>;\n limit?: number;\n offset?: number;\n};\n\nexport type RuntimeExecutionState = {\n executionId?: string;\n mutationDepth: number;\n changedTables: Set<string>;\n documentChanges: DocumentChangePreview[];\n storageChanges: Array<{\n storageId: string;\n reason: Extract<\n SyncoreExternalChangeReason,\n \"storage-put\" | \"storage-delete\"\n >;\n }>;\n dependencyCollector?: Set<DependencyKey>;\n componentMetadata?: SyncoreComponentFunctionMetadata | undefined;\n};\n\nconst PREVIEW_MAX_DEPTH = 5;\nconst PREVIEW_MAX_ARRAY_ITEMS = 50;\nconst PREVIEW_MAX_OBJECT_KEYS = 80;\nconst PREVIEW_MAX_STRING_LENGTH = 4000;\n\nexport function createDevtoolsPreview(value: unknown): DevtoolsPreview {\n const seen = new WeakSet<object>();\n let truncated = false;\n\n const preview = (input: unknown, depth: number): unknown => {\n if (typeof input === \"string\") {\n if (input.length > PREVIEW_MAX_STRING_LENGTH) {\n truncated = true;\n return `${input.slice(0, PREVIEW_MAX_STRING_LENGTH)}...`;\n }\n return input;\n }\n if (\n input === null ||\n typeof input === \"number\" ||\n typeof input === \"boolean\"\n ) {\n return input;\n }\n if (typeof input === \"bigint\") {\n return `${input.toString()}n`;\n }\n if (input === undefined) {\n return \"[undefined]\";\n }\n if (typeof input === \"function\") {\n truncated = true;\n return \"[function]\";\n }\n if (typeof input === \"symbol\") {\n truncated = true;\n return input.toString();\n }\n if (input instanceof Error) {\n return {\n name: input.name,\n message: input.message,\n ...(input.stack ? { stack: input.stack } : {})\n };\n }\n if (input instanceof Date) {\n return input.toISOString();\n }\n if (typeof input !== \"object\") {\n return String(input);\n }\n if (seen.has(input)) {\n truncated = true;\n return \"[circular]\";\n }\n if (depth >= PREVIEW_MAX_DEPTH) {\n truncated = true;\n return Array.isArray(input) ? \"[array]\" : \"[object]\";\n }\n\n seen.add(input);\n if (Array.isArray(input)) {\n const items = input\n .slice(0, PREVIEW_MAX_ARRAY_ITEMS)\n .map((item) => preview(item, depth + 1));\n if (input.length > PREVIEW_MAX_ARRAY_ITEMS) {\n truncated = true;\n items.push(`[${input.length - PREVIEW_MAX_ARRAY_ITEMS} more items]`);\n }\n seen.delete(input);\n return items;\n }\n\n const entries = Object.entries(input as Record<string, unknown>);\n const result: Record<string, unknown> = {};\n for (const [key, nested] of entries.slice(0, PREVIEW_MAX_OBJECT_KEYS)) {\n result[key] = preview(nested, depth + 1);\n }\n if (entries.length > PREVIEW_MAX_OBJECT_KEYS) {\n truncated = true;\n result.__truncatedKeys = entries.length - PREVIEW_MAX_OBJECT_KEYS;\n }\n seen.delete(input);\n return result;\n };\n\n try {\n return {\n kind: \"value\",\n value: preview(value, 0),\n ...(truncated ? { truncated: true } : {})\n };\n } catch (error) {\n return {\n kind: \"error\",\n message: error instanceof Error ? error.message : String(error),\n truncated: true\n };\n }\n}\n\nexport function fieldExpression(tableAlias: string, field: string): string {\n const prefix = tableAlias ? `${tableAlias}.` : \"\";\n return `json_extract(${prefix}_json, '$.${field}')`;\n}\n\nexport function quoteIdentifier(identifier: string): string {\n return `\"${identifier.replaceAll('\"', '\"\"')}\"`;\n}\n\nexport function stableStringify(value: unknown): string {\n return JSON.stringify(sortValue(value));\n}\n\nexport function sortValue(value: unknown): unknown {\n if (Array.isArray(value)) {\n return value.map(sortValue);\n }\n if (value && typeof value === \"object\") {\n return Object.fromEntries(\n Object.entries(value as Record<string, unknown>)\n .sort(([left], [right]) => left.localeCompare(right))\n .map(([key, nested]) => [key, sortValue(nested)])\n );\n }\n return value;\n}\n\nexport function omitSystemFields<TDocument extends object>(\n document: TDocument\n): JsonObject {\n const clone = { ...(document as Record<string, unknown>) };\n delete clone._id;\n delete clone._creationTime;\n return clone;\n}\n\nexport function toSearchValue(value: unknown): string {\n if (typeof value === \"string\") {\n return value;\n }\n if (value === null || value === undefined) {\n return \"\";\n }\n if (\n typeof value === \"number\" ||\n typeof value === \"boolean\" ||\n typeof value === \"bigint\"\n ) {\n return String(value);\n }\n return stableStringify(value);\n}\n\nexport function parseMisfirePolicy(\n type: string,\n windowMs: number | null\n): MisfirePolicy {\n if (type === \"windowed\") {\n return { type, windowMs: windowMs ?? 0 };\n }\n if (type === \"skip\" || type === \"run_once_if_missed\") {\n return { type };\n }\n return { type: \"catch_up\" };\n}\n\nexport function shouldRunMissedJob(\n scheduledAt: number,\n now: number,\n policy: MisfirePolicy\n): boolean {\n if (scheduledAt >= now) {\n return true;\n }\n switch (policy.type) {\n case \"catch_up\":\n return true;\n case \"run_once_if_missed\":\n return true;\n case \"skip\":\n return false;\n case \"windowed\":\n return now - scheduledAt <= policy.windowMs;\n }\n}\n\nexport function computeNextRun(\n schedule: RecurringSchedule,\n fromTimestamp: number\n): number {\n switch (schedule.type) {\n case \"interval\":\n return fromTimestamp + intervalToMs(schedule);\n case \"daily\":\n return nextDailyOccurrence(fromTimestamp, schedule);\n case \"weekly\":\n return nextWeeklyOccurrence(fromTimestamp, schedule);\n }\n}\n\nexport function safeReadRecurringSchedule(\n scheduleJson: string | null\n): RecurringSchedule | undefined {\n if (!scheduleJson) {\n return undefined;\n }\n try {\n const parsed = JSON.parse(scheduleJson) as unknown;\n return isRecurringSchedule(parsed) ? parsed : undefined;\n } catch {\n return undefined;\n }\n}\n\nexport function normalizeOptionalArgs<TArgs>(\n args: [] | [TArgs] | readonly unknown[]\n): TArgs {\n return (args[0] ?? {}) as TArgs;\n}\n\nexport function splitSchedulerArgs(\n args: readonly unknown[]\n): [JsonObject, MisfirePolicy | undefined] {\n if (args.length === 0) {\n return [{}, undefined];\n }\n if (args.length === 1) {\n const [first] = args;\n if (isMisfirePolicy(first)) {\n return [{}, first];\n }\n return [(first ?? {}) as JsonObject, undefined];\n }\n return [(args[0] ?? {}) as JsonObject, args[1] as MisfirePolicy | undefined];\n}\n\nexport function inferDriverDatabasePath(driver: {\n filename?: string;\n databasePath?: string;\n}): string | undefined {\n return driver.databasePath ?? driver.filename;\n}\n\nexport function parseCanonicalComponentFunctionName(functionName: string):\n | {\n componentPath: string;\n visibility: \"public\" | \"internal\";\n localName: string;\n }\n | undefined {\n const match = /^components\\/(.+)\\/(public|internal)\\/(.+)$/.exec(functionName);\n if (!match) {\n return undefined;\n }\n return {\n componentPath: match[1] ?? \"\",\n visibility: (match[2] as \"public\" | \"internal\") ?? \"public\",\n localName: match[3] ?? \"\"\n };\n}\n\nexport function parseComponentScopedIdentifier(\n value: string\n): { componentPath: string; localId: string } | undefined {\n const match = /^component:([^:]+):(.+)$/.exec(value);\n if (!match) {\n return undefined;\n }\n return {\n componentPath: match[1] ?? \"\",\n localId: match[2] ?? \"\"\n };\n}\n\nexport function devtoolsScopesForEvent(\n event: SyncoreDevtoolsEvent\n): Set<DevtoolsLiveQueryScope> {\n switch (event.type) {\n case \"runtime.connected\":\n case \"runtime.disconnected\":\n return new Set([\"runtime.summary\", \"runtime.activeQueries\"]);\n case \"query.executed\":\n case \"query.invalidated\":\n return new Set([\"runtime.summary\", \"runtime.activeQueries\"]);\n case \"mutation.committed\":\n return new Set([\n \"runtime.summary\",\n ...event.changedTables.map((table: string) => `table:${table}` as const)\n ]);\n case \"scheduler.tick\":\n return new Set([\"scheduler.jobs\", \"runtime.summary\"]);\n case \"storage.updated\":\n return new Set([\"runtime.summary\"]);\n case \"action.completed\":\n case \"log\":\n return new Set([\"runtime.summary\"]);\n default:\n return new Set([\"runtime.summary\"]);\n }\n}\n\nexport function getTableDefinition<\n TSchema extends SyncoreDataModel\n>(\n schema: TSchema,\n tableName: string\n): TableDefinition<\n Validator<Record<string, unknown>, Record<string, unknown>, string>\n> {\n return schema.getTable(tableName as never) as TableDefinition<\n Validator<Record<string, unknown>, Record<string, unknown>, string>\n >;\n}\n\nexport function searchIndexKey(tableName: string, indexName: string): string {\n return `${tableName}:${indexName}`;\n}\n\nexport function resolveSearchIndexTableName(\n tableName: string,\n indexName: string\n): string {\n return searchIndexTableName(tableName, indexName);\n}\n\nfunction intervalToMs(schedule: {\n seconds?: number;\n minutes?: number;\n hours?: number;\n}): number {\n if (schedule.seconds) {\n return schedule.seconds * 1000;\n }\n if (schedule.minutes) {\n return schedule.minutes * 60 * 1000;\n }\n return (schedule.hours ?? 1) * 60 * 60 * 1000;\n}\n\nfunction nextDailyOccurrence(\n fromTimestamp: number,\n schedule: Extract<RecurringSchedule, { type: \"daily\" }>\n): number {\n const timezone = schedule.timezone ?? \"UTC\";\n const now = new Date(fromTimestamp);\n const zonedNow = toZonedTime(now, timezone);\n const zoned = new Date(zonedNow.getTime());\n zoned.setHours(schedule.hour, schedule.minute, 0, 0);\n if (zoned.getTime() <= zonedNow.getTime()) {\n zoned.setDate(zoned.getDate() + 1);\n }\n return fromZonedTime(zoned, timezone).getTime();\n}\n\nfunction nextWeeklyOccurrence(\n fromTimestamp: number,\n schedule: Extract<RecurringSchedule, { type: \"weekly\" }>\n): number {\n const timezone = schedule.timezone ?? \"UTC\";\n const now = new Date(fromTimestamp);\n const zonedNow = toZonedTime(now, timezone);\n const targetDay = [\n \"sunday\",\n \"monday\",\n \"tuesday\",\n \"wednesday\",\n \"thursday\",\n \"friday\",\n \"saturday\"\n ].indexOf(schedule.dayOfWeek);\n const zoned = new Date(zonedNow.getTime());\n const delta = (targetDay - zonedNow.getDay() + 7) % 7;\n zoned.setDate(zoned.getDate() + delta);\n zoned.setHours(schedule.hour, schedule.minute, 0, 0);\n if (zoned.getTime() <= zonedNow.getTime()) {\n zoned.setDate(zoned.getDate() + 7);\n }\n return fromZonedTime(zoned, timezone).getTime();\n}\n\nfunction isRecurringSchedule(value: unknown): value is RecurringSchedule {\n if (!isRecord(value) || typeof value.type !== \"string\") {\n return false;\n }\n switch (value.type) {\n case \"interval\":\n return (\n isOptionalNumber(value.seconds) &&\n isOptionalNumber(value.minutes) &&\n isOptionalNumber(value.hours)\n );\n case \"daily\":\n return (\n typeof value.hour === \"number\" &&\n typeof value.minute === \"number\" &&\n isOptionalString(value.timezone)\n );\n case \"weekly\":\n return (\n isDayOfWeek(value.dayOfWeek) &&\n typeof value.hour === \"number\" &&\n typeof value.minute === \"number\" &&\n isOptionalString(value.timezone)\n );\n default:\n return false;\n }\n}\n\nfunction isMisfirePolicy(value: unknown): value is MisfirePolicy {\n return (\n typeof value === \"object\" &&\n value !== null &&\n \"type\" in value &&\n typeof (value as { type?: unknown }).type === \"string\"\n );\n}\n\nfunction isRecord(value: unknown): value is Record<string, unknown> {\n return typeof value === \"object\" && value !== null;\n}\n\nfunction isOptionalNumber(value: unknown): value is number | undefined {\n return value === undefined || typeof value === \"number\";\n}\n\nfunction isOptionalString(value: unknown): value is string | undefined {\n return value === undefined || typeof value === \"string\";\n}\n\nfunction isDayOfWeek(\n value: unknown\n): value is Extract<RecurringSchedule, { type: \"weekly\" }>[\"dayOfWeek\"] {\n return (\n value === \"sunday\" ||\n value === \"monday\" ||\n value === \"tuesday\" ||\n value === \"wednesday\" ||\n value === \"thursday\" ||\n value === \"friday\" ||\n value === \"saturday\"\n );\n}\n"],"mappings":"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"}
@@ -0,0 +1,145 @@
1
+ import { generateId } from "../../id.mjs";
2
+ //#region src/runtime/internal/engines/storageEngine.ts
3
+ var StorageEngine = class {
4
+ deps;
5
+ constructor(deps) {
6
+ this.deps = deps;
7
+ }
8
+ async prepare() {
9
+ await this.deps.driver.exec(`
10
+ CREATE TABLE IF NOT EXISTS "_storage" (
11
+ _id TEXT PRIMARY KEY,
12
+ _creationTime INTEGER NOT NULL,
13
+ file_name TEXT,
14
+ content_type TEXT,
15
+ size INTEGER NOT NULL,
16
+ path TEXT NOT NULL
17
+ );
18
+ CREATE TABLE IF NOT EXISTS "_storage_pending" (
19
+ _id TEXT PRIMARY KEY,
20
+ _creationTime INTEGER NOT NULL,
21
+ file_name TEXT,
22
+ content_type TEXT
23
+ );
24
+ `);
25
+ }
26
+ async reconcile() {
27
+ const pendingRows = await this.deps.driver.all(`SELECT _id, _creationTime, file_name, content_type FROM "_storage_pending"`);
28
+ for (const pendingRow of pendingRows) {
29
+ if (!await this.deps.driver.get(`SELECT _id FROM "_storage" WHERE _id = ?`, [pendingRow._id])) {
30
+ await this.deps.storage.delete(pendingRow._id);
31
+ this.deps.devtools.emit({
32
+ type: "log",
33
+ runtimeId: this.deps.runtimeId,
34
+ level: "warn",
35
+ message: `Recovered interrupted storage write ${pendingRow._id}.`,
36
+ timestamp: Date.now()
37
+ });
38
+ }
39
+ await this.deps.driver.run(`DELETE FROM "_storage_pending" WHERE _id = ?`, [pendingRow._id]);
40
+ }
41
+ if (!this.deps.storage.list) return;
42
+ const storedRows = await this.deps.driver.all(`SELECT _id FROM "_storage"`);
43
+ const knownIds = new Set(storedRows.map((row) => row._id));
44
+ const physicalObjects = await this.deps.storage.list();
45
+ for (const object of physicalObjects) {
46
+ if (knownIds.has(object.id)) continue;
47
+ await this.deps.storage.delete(object.id);
48
+ this.deps.devtools.emit({
49
+ type: "log",
50
+ runtimeId: this.deps.runtimeId,
51
+ level: "warn",
52
+ message: `Removed orphaned storage object ${object.id}.`,
53
+ timestamp: Date.now()
54
+ });
55
+ }
56
+ }
57
+ createStorageApi(state) {
58
+ const componentMetadata = state.componentMetadata;
59
+ const namespacePrefix = componentMetadata ? `component:${componentMetadata.componentPath}:` : "";
60
+ const ensureStorageCapability = () => {
61
+ if (componentMetadata && !componentMetadata.grantedCapabilities.includes("storage")) throw new Error(`Component ${JSON.stringify(componentMetadata.componentPath)} is not allowed to use storage.`);
62
+ };
63
+ const scopedId = (id) => `${namespacePrefix}${id}`;
64
+ return {
65
+ put: async (input) => {
66
+ ensureStorageCapability();
67
+ const id = scopedId(generateId());
68
+ const createdAt = Date.now();
69
+ await this.deps.driver.run(`INSERT OR REPLACE INTO "_storage_pending" (_id, _creationTime, file_name, content_type) VALUES (?, ?, ?, ?)`, [
70
+ id,
71
+ createdAt,
72
+ input.fileName ?? null,
73
+ input.contentType ?? null
74
+ ]);
75
+ const object = await this.deps.storage.put(id, input);
76
+ await this.deps.driver.withTransaction(async () => {
77
+ await this.deps.driver.run(`INSERT OR REPLACE INTO "_storage" (_id, _creationTime, file_name, content_type, size, path) VALUES (?, ?, ?, ?, ?, ?)`, [
78
+ id,
79
+ createdAt,
80
+ input.fileName ?? null,
81
+ object.contentType,
82
+ object.size,
83
+ object.path
84
+ ]);
85
+ await this.deps.driver.run(`DELETE FROM "_storage_pending" WHERE _id = ?`, [id]);
86
+ });
87
+ this.deps.devtools.emit({
88
+ type: "storage.updated",
89
+ runtimeId: this.deps.runtimeId,
90
+ storageId: id,
91
+ ...componentMetadata ? { componentPath: componentMetadata.componentPath } : {},
92
+ operation: "put",
93
+ timestamp: Date.now()
94
+ });
95
+ state.storageChanges.push({
96
+ storageId: id,
97
+ reason: "storage-put"
98
+ });
99
+ return id;
100
+ },
101
+ get: async (id) => {
102
+ ensureStorageCapability();
103
+ state.dependencyCollector?.add(`storage:${id}`);
104
+ const row = await this.deps.driver.get(`SELECT _id, _creationTime, file_name, content_type, size, path FROM "_storage" WHERE _id = ?`, [id]);
105
+ if (!row) return null;
106
+ return {
107
+ id: row._id,
108
+ path: row.path,
109
+ size: row.size,
110
+ contentType: row.content_type
111
+ };
112
+ },
113
+ read: async (id) => {
114
+ ensureStorageCapability();
115
+ state.dependencyCollector?.add(`storage:${id}`);
116
+ if (!await this.deps.driver.get(`SELECT _id FROM "_storage" WHERE _id = ?`, [id])) return null;
117
+ return this.deps.storage.read(id);
118
+ },
119
+ delete: async (id) => {
120
+ ensureStorageCapability();
121
+ await this.deps.storage.delete(id);
122
+ await this.deps.driver.withTransaction(async () => {
123
+ await this.deps.driver.run(`DELETE FROM "_storage" WHERE _id = ?`, [id]);
124
+ await this.deps.driver.run(`DELETE FROM "_storage_pending" WHERE _id = ?`, [id]);
125
+ });
126
+ this.deps.devtools.emit({
127
+ type: "storage.updated",
128
+ runtimeId: this.deps.runtimeId,
129
+ storageId: id,
130
+ ...componentMetadata ? { componentPath: componentMetadata.componentPath } : {},
131
+ operation: "delete",
132
+ timestamp: Date.now()
133
+ });
134
+ state.storageChanges.push({
135
+ storageId: id,
136
+ reason: "storage-delete"
137
+ });
138
+ }
139
+ };
140
+ }
141
+ };
142
+ //#endregion
143
+ export { StorageEngine };
144
+
145
+ //# sourceMappingURL=storageEngine.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"storageEngine.mjs","names":[],"sources":["../../../../src/runtime/internal/engines/storageEngine.ts"],"sourcesContent":["import type {\n StorageObject,\n SyncoreSqlDriver,\n SyncoreStorageAdapter,\n SyncoreStorageApi,\n StorageWriteInput\n} from \"../../runtime.js\";\nimport {\n type RuntimeExecutionState,\n type StorageMetadataRow,\n type StoragePendingRow\n} from \"./shared.js\";\nimport { DevtoolsEngine } from \"./devtoolsEngine.js\";\nimport { generateId } from \"../../id.js\";\n\ntype StorageEngineDeps = {\n driver: SyncoreSqlDriver;\n storage: SyncoreStorageAdapter;\n runtimeId: string;\n devtools: DevtoolsEngine;\n};\n\nexport class StorageEngine {\n constructor(private readonly deps: StorageEngineDeps) {}\n\n async prepare(): Promise<void> {\n await this.deps.driver.exec(`\n CREATE TABLE IF NOT EXISTS \"_storage\" (\n _id TEXT PRIMARY KEY,\n _creationTime INTEGER NOT NULL,\n file_name TEXT,\n content_type TEXT,\n size INTEGER NOT NULL,\n path TEXT NOT NULL\n );\n CREATE TABLE IF NOT EXISTS \"_storage_pending\" (\n _id TEXT PRIMARY KEY,\n _creationTime INTEGER NOT NULL,\n file_name TEXT,\n content_type TEXT\n );\n `);\n }\n\n async reconcile(): Promise<void> {\n const pendingRows = await this.deps.driver.all<StoragePendingRow>(\n `SELECT _id, _creationTime, file_name, content_type FROM \"_storage_pending\"`\n );\n\n for (const pendingRow of pendingRows) {\n const committed = await this.deps.driver.get<\n Pick<StorageMetadataRow, \"_id\">\n >(`SELECT _id FROM \"_storage\" WHERE _id = ?`, [pendingRow._id]);\n if (!committed) {\n await this.deps.storage.delete(pendingRow._id);\n this.deps.devtools.emit({\n type: \"log\",\n runtimeId: this.deps.runtimeId,\n level: \"warn\",\n message: `Recovered interrupted storage write ${pendingRow._id}.`,\n timestamp: Date.now()\n });\n }\n await this.deps.driver.run(\n `DELETE FROM \"_storage_pending\" WHERE _id = ?`,\n [pendingRow._id]\n );\n }\n\n if (!this.deps.storage.list) {\n return;\n }\n\n const storedRows = await this.deps.driver.all<Pick<StorageMetadataRow, \"_id\">>(\n `SELECT _id FROM \"_storage\"`\n );\n const knownIds = new Set(storedRows.map((row) => row._id));\n const physicalObjects = await this.deps.storage.list();\n for (const object of physicalObjects) {\n if (knownIds.has(object.id)) {\n continue;\n }\n await this.deps.storage.delete(object.id);\n this.deps.devtools.emit({\n type: \"log\",\n runtimeId: this.deps.runtimeId,\n level: \"warn\",\n message: `Removed orphaned storage object ${object.id}.`,\n timestamp: Date.now()\n });\n }\n }\n\n createStorageApi(state: RuntimeExecutionState): SyncoreStorageApi {\n const componentMetadata = state.componentMetadata;\n const namespacePrefix = componentMetadata\n ? `component:${componentMetadata.componentPath}:`\n : \"\";\n const ensureStorageCapability = () => {\n if (\n componentMetadata &&\n !componentMetadata.grantedCapabilities.includes(\"storage\")\n ) {\n throw new Error(\n `Component ${JSON.stringify(componentMetadata.componentPath)} is not allowed to use storage.`\n );\n }\n };\n const scopedId = (id: string) => `${namespacePrefix}${id}`;\n\n return {\n put: async (input: StorageWriteInput) => {\n ensureStorageCapability();\n const id = scopedId(generateId());\n const createdAt = Date.now();\n await this.deps.driver.run(\n `INSERT OR REPLACE INTO \"_storage_pending\" (_id, _creationTime, file_name, content_type) VALUES (?, ?, ?, ?)`,\n [id, createdAt, input.fileName ?? null, input.contentType ?? null]\n );\n const object = await this.deps.storage.put(id, input);\n await this.deps.driver.withTransaction(async () => {\n await this.deps.driver.run(\n `INSERT OR REPLACE INTO \"_storage\" (_id, _creationTime, file_name, content_type, size, path) VALUES (?, ?, ?, ?, ?, ?)`,\n [\n id,\n createdAt,\n input.fileName ?? null,\n object.contentType,\n object.size,\n object.path\n ]\n );\n await this.deps.driver.run(\n `DELETE FROM \"_storage_pending\" WHERE _id = ?`,\n [id]\n );\n });\n this.deps.devtools.emit({\n type: \"storage.updated\",\n runtimeId: this.deps.runtimeId,\n storageId: id,\n ...(componentMetadata\n ? { componentPath: componentMetadata.componentPath }\n : {}),\n operation: \"put\",\n timestamp: Date.now()\n });\n state.storageChanges.push({\n storageId: id,\n reason: \"storage-put\"\n });\n return id;\n },\n get: async (id: string): Promise<StorageObject | null> => {\n ensureStorageCapability();\n state.dependencyCollector?.add(`storage:${id}`);\n const row = await this.deps.driver.get<StorageMetadataRow>(\n `SELECT _id, _creationTime, file_name, content_type, size, path FROM \"_storage\" WHERE _id = ?`,\n [id]\n );\n if (!row) {\n return null;\n }\n return {\n id: row._id,\n path: row.path,\n size: row.size,\n contentType: row.content_type\n };\n },\n read: async (id: string) => {\n ensureStorageCapability();\n state.dependencyCollector?.add(`storage:${id}`);\n const row = await this.deps.driver.get<Pick<StorageMetadataRow, \"_id\">>(\n `SELECT _id FROM \"_storage\" WHERE _id = ?`,\n [id]\n );\n if (!row) {\n return null;\n }\n return this.deps.storage.read(id);\n },\n delete: async (id: string) => {\n ensureStorageCapability();\n await this.deps.storage.delete(id);\n await this.deps.driver.withTransaction(async () => {\n await this.deps.driver.run(\n `DELETE FROM \"_storage\" WHERE _id = ?`,\n [id]\n );\n await this.deps.driver.run(\n `DELETE FROM \"_storage_pending\" WHERE _id = ?`,\n [id]\n );\n });\n this.deps.devtools.emit({\n type: \"storage.updated\",\n runtimeId: this.deps.runtimeId,\n storageId: id,\n ...(componentMetadata\n ? { componentPath: componentMetadata.componentPath }\n : {}),\n operation: \"delete\",\n timestamp: Date.now()\n });\n state.storageChanges.push({\n storageId: id,\n reason: \"storage-delete\"\n });\n }\n };\n }\n}\n"],"mappings":"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"}