@kyro-cms/core 0.6.0 → 0.7.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (145) hide show
  1. package/dist/api-handler.cjs +75 -35
  2. package/dist/api-handler.cjs.map +1 -1
  3. package/dist/api-handler.d.cts +2 -5
  4. package/dist/api-handler.d.ts +2 -5
  5. package/dist/api-handler.js +75 -36
  6. package/dist/api-handler.js.map +1 -1
  7. package/dist/bootstrap-AKAUP6F6.cjs +32 -0
  8. package/dist/{bootstrap-EE6BJZWL.cjs.map → bootstrap-AKAUP6F6.cjs.map} +1 -1
  9. package/dist/bootstrap-JCML6NFO.js +7 -0
  10. package/dist/{bootstrap-4MH44YKG.js.map → bootstrap-JCML6NFO.js.map} +1 -1
  11. package/dist/{chunk-WVPOPOEQ.cjs → chunk-2KVHZE6O.cjs} +286 -126
  12. package/dist/chunk-2KVHZE6O.cjs.map +1 -0
  13. package/dist/{chunk-RALQO47U.cjs → chunk-2OL4O2TH.cjs} +55 -2
  14. package/dist/chunk-2OL4O2TH.cjs.map +1 -0
  15. package/dist/{chunk-XU7AFF6V.js → chunk-35U3FROB.js} +982 -4
  16. package/dist/chunk-35U3FROB.js.map +1 -0
  17. package/dist/{chunk-WSCJQI2B.js → chunk-3J4MFTI3.js} +27 -11
  18. package/dist/chunk-3J4MFTI3.js.map +1 -0
  19. package/dist/chunk-3ZFYL34R.js +391 -0
  20. package/dist/chunk-3ZFYL34R.js.map +1 -0
  21. package/dist/chunk-4DA7QPLA.cjs +356 -0
  22. package/dist/chunk-4DA7QPLA.cjs.map +1 -0
  23. package/dist/{chunk-TP5YQFIX.js → chunk-57P6MJKC.js} +3 -715
  24. package/dist/chunk-57P6MJKC.js.map +1 -0
  25. package/dist/{chunk-R2YHJN6W.cjs → chunk-5KVM3WEY.cjs} +34 -208
  26. package/dist/chunk-5KVM3WEY.cjs.map +1 -0
  27. package/dist/{chunk-Z2OVHWHB.cjs → chunk-6IMPH6WV.cjs} +28 -11
  28. package/dist/chunk-6IMPH6WV.cjs.map +1 -0
  29. package/dist/{chunk-QKVA2SOG.js → chunk-DXHRBMGB.js} +27 -284
  30. package/dist/chunk-DXHRBMGB.js.map +1 -0
  31. package/dist/{chunk-E3BZLMX6.js → chunk-ES5HNFFT.js} +43 -2
  32. package/dist/chunk-ES5HNFFT.js.map +1 -0
  33. package/dist/{chunk-QYZKIPSD.js → chunk-FXYP2HA6.js} +34 -3
  34. package/dist/chunk-FXYP2HA6.js.map +1 -0
  35. package/dist/chunk-H727JIG7.js +809 -0
  36. package/dist/chunk-H727JIG7.js.map +1 -0
  37. package/dist/{chunk-AM4JKIPP.js → chunk-HXRD4B37.js} +9 -183
  38. package/dist/chunk-HXRD4B37.js.map +1 -0
  39. package/dist/chunk-I7HHI6QV.cjs +816 -0
  40. package/dist/chunk-I7HHI6QV.cjs.map +1 -0
  41. package/dist/{chunk-RDRJVCL5.cjs → chunk-IA6AU5PI.cjs} +2 -720
  42. package/dist/chunk-IA6AU5PI.cjs.map +1 -0
  43. package/dist/{chunk-55BNRTLW.cjs → chunk-LINKCEG4.cjs} +985 -4
  44. package/dist/chunk-LINKCEG4.cjs.map +1 -0
  45. package/dist/{chunk-TVVYZ2TH.js → chunk-OHVB4AJ7.js} +56 -3
  46. package/dist/chunk-OHVB4AJ7.js.map +1 -0
  47. package/dist/{chunk-XAEBVZTI.cjs → chunk-PDYFVNUX.cjs} +26 -289
  48. package/dist/chunk-PDYFVNUX.cjs.map +1 -0
  49. package/dist/{chunk-6WXQRYTW.js → chunk-QPPDLRNR.js} +286 -126
  50. package/dist/chunk-QPPDLRNR.js.map +1 -0
  51. package/dist/{chunk-WBCIEYHC.cjs → chunk-QUW2RZTM.cjs} +35 -4
  52. package/dist/chunk-QUW2RZTM.cjs.map +1 -0
  53. package/dist/chunk-SA7NSSIQ.cjs +397 -0
  54. package/dist/chunk-SA7NSSIQ.cjs.map +1 -0
  55. package/dist/{chunk-H4XCAPA6.cjs → chunk-V3LKPM3O.cjs} +43 -2
  56. package/dist/chunk-V3LKPM3O.cjs.map +1 -0
  57. package/dist/chunk-Y3N7UUDO.js +349 -0
  58. package/dist/chunk-Y3N7UUDO.js.map +1 -0
  59. package/dist/{chunk-S3FG2NY7.js → chunk-Y3QQN7PN.js} +4 -3
  60. package/dist/chunk-Y3QQN7PN.js.map +1 -0
  61. package/dist/{chunk-5HA5OMFH.cjs → chunk-YVUJBEXE.cjs} +7 -6
  62. package/dist/chunk-YVUJBEXE.cjs.map +1 -0
  63. package/dist/cli/index.cjs +103 -20
  64. package/dist/cli/index.cjs.map +1 -1
  65. package/dist/cli/index.js +103 -20
  66. package/dist/cli/index.js.map +1 -1
  67. package/dist/client.d.cts +1 -1
  68. package/dist/client.d.ts +1 -1
  69. package/dist/drizzle/index.cjs +12 -12
  70. package/dist/drizzle/index.d.cts +23 -2
  71. package/dist/drizzle/index.d.ts +23 -2
  72. package/dist/drizzle/index.js +3 -3
  73. package/dist/index.cjs +174 -1054
  74. package/dist/index.cjs.map +1 -1
  75. package/dist/index.d.cts +85 -7
  76. package/dist/index.d.ts +85 -7
  77. package/dist/index.js +91 -980
  78. package/dist/index.js.map +1 -1
  79. package/dist/integration.cjs +2 -2
  80. package/dist/integration.d.cts +3 -16
  81. package/dist/integration.d.ts +3 -16
  82. package/dist/integration.js +1 -1
  83. package/dist/mongo-auth-adapter-NHHUJHVH.cjs +17 -0
  84. package/dist/mongo-auth-adapter-NHHUJHVH.cjs.map +1 -0
  85. package/dist/mongo-auth-adapter-NJQUUCTP.js +4 -0
  86. package/dist/mongo-auth-adapter-NJQUUCTP.js.map +1 -0
  87. package/dist/mongodb/index.cjs +9 -8
  88. package/dist/mongodb/index.d.cts +86 -5
  89. package/dist/mongodb/index.d.ts +86 -5
  90. package/dist/mongodb/index.js +3 -2
  91. package/dist/postgres-auth-adapter-3T2NKTSE.js +5 -0
  92. package/dist/{postgres-auth-adapter-B65BULNS.js.map → postgres-auth-adapter-3T2NKTSE.js.map} +1 -1
  93. package/dist/postgres-auth-adapter-7IEENCKQ.cjs +14 -0
  94. package/dist/{postgres-auth-adapter-6742WDCF.cjs.map → postgres-auth-adapter-7IEENCKQ.cjs.map} +1 -1
  95. package/dist/redis-adapter-D2E2S3GB.cjs +13 -0
  96. package/dist/{redis-adapter-LPUWLE4Y.cjs.map → redis-adapter-D2E2S3GB.cjs.map} +1 -1
  97. package/dist/redis-adapter-VQXD7ESY.js +4 -0
  98. package/dist/{redis-adapter-THYDCGQR.js.map → redis-adapter-VQXD7ESY.js.map} +1 -1
  99. package/dist/rest/index.cjs +10 -8
  100. package/dist/rest/index.js +8 -6
  101. package/dist/sqlite-adapter-LVK5PS4T.cjs +13 -0
  102. package/dist/sqlite-adapter-LVK5PS4T.cjs.map +1 -0
  103. package/dist/sqlite-adapter-TR3U3W6Q.js +4 -0
  104. package/dist/sqlite-adapter-TR3U3W6Q.js.map +1 -0
  105. package/dist/templates/index.cjs +31 -27
  106. package/dist/templates/index.d.cts +8 -5
  107. package/dist/templates/index.d.ts +8 -5
  108. package/dist/templates/index.js +1 -1
  109. package/dist/{base-eVegJ_Pr.d.ts → tenant-B1YB0Jy8.d.ts} +10 -1
  110. package/dist/{base-DvvNqnM-.d.cts → tenant-Cpeveji6.d.cts} +10 -1
  111. package/dist/{types-DqN4ckOC.d.cts → types-D6ZLRGbH.d.cts} +19 -1
  112. package/dist/{types-DqN4ckOC.d.ts → types-D6ZLRGbH.d.ts} +19 -1
  113. package/package.json +56 -9
  114. package/dist/adapter-BSvBudTG.d.cts +0 -65
  115. package/dist/adapter-CXGB2Elb.d.ts +0 -65
  116. package/dist/bootstrap-4MH44YKG.js +0 -6
  117. package/dist/bootstrap-EE6BJZWL.cjs +0 -31
  118. package/dist/chunk-55BNRTLW.cjs.map +0 -1
  119. package/dist/chunk-5HA5OMFH.cjs.map +0 -1
  120. package/dist/chunk-6WXQRYTW.js.map +0 -1
  121. package/dist/chunk-A4USRVTQ.js +0 -115
  122. package/dist/chunk-A4USRVTQ.js.map +0 -1
  123. package/dist/chunk-AM4JKIPP.js.map +0 -1
  124. package/dist/chunk-E3BZLMX6.js.map +0 -1
  125. package/dist/chunk-H4XCAPA6.cjs.map +0 -1
  126. package/dist/chunk-KOCTZKPV.cjs +0 -117
  127. package/dist/chunk-KOCTZKPV.cjs.map +0 -1
  128. package/dist/chunk-QKVA2SOG.js.map +0 -1
  129. package/dist/chunk-QYZKIPSD.js.map +0 -1
  130. package/dist/chunk-R2YHJN6W.cjs.map +0 -1
  131. package/dist/chunk-RALQO47U.cjs.map +0 -1
  132. package/dist/chunk-RDRJVCL5.cjs.map +0 -1
  133. package/dist/chunk-S3FG2NY7.js.map +0 -1
  134. package/dist/chunk-TP5YQFIX.js.map +0 -1
  135. package/dist/chunk-TVVYZ2TH.js.map +0 -1
  136. package/dist/chunk-WBCIEYHC.cjs.map +0 -1
  137. package/dist/chunk-WSCJQI2B.js.map +0 -1
  138. package/dist/chunk-WVPOPOEQ.cjs.map +0 -1
  139. package/dist/chunk-XAEBVZTI.cjs.map +0 -1
  140. package/dist/chunk-XU7AFF6V.js.map +0 -1
  141. package/dist/chunk-Z2OVHWHB.cjs.map +0 -1
  142. package/dist/postgres-auth-adapter-6742WDCF.cjs +0 -14
  143. package/dist/postgres-auth-adapter-B65BULNS.js +0 -5
  144. package/dist/redis-adapter-LPUWLE4Y.cjs +0 -13
  145. package/dist/redis-adapter-THYDCGQR.js +0 -4
@@ -1,12 +1,16 @@
1
1
  'use strict';
2
2
 
3
3
  var chunkK7JPTH3G_cjs = require('./chunk-K7JPTH3G.cjs');
4
- var chunkR2YHJN6W_cjs = require('./chunk-R2YHJN6W.cjs');
4
+ var chunk5KVM3WEY_cjs = require('./chunk-5KVM3WEY.cjs');
5
5
  var chunkIBG6V56E_cjs = require('./chunk-IBG6V56E.cjs');
6
6
  var chunkVJT6P4N6_cjs = require('./chunk-VJT6P4N6.cjs');
7
7
  var chunkDVD5P72E_cjs = require('./chunk-DVD5P72E.cjs');
8
+ var chunkSA7NSSIQ_cjs = require('./chunk-SA7NSSIQ.cjs');
8
9
  var zod = require('zod');
10
+ var module$1 = require('module');
11
+ var crypto = require('crypto');
9
12
 
13
+ var _documentCurrentScript = typeof document !== 'undefined' ? document.currentScript : null;
10
14
  // src/registry/validator.ts
11
15
  function findFieldByName(fields, name) {
12
16
  for (const field of fields) {
@@ -1196,7 +1200,7 @@ var Kyro = class {
1196
1200
  const authObj = typeof this.config.auth === "object" ? this.config.auth : null;
1197
1201
  const authSecret = authObj?.secret;
1198
1202
  const checkSession = authObj?.checkSession !== false;
1199
- return chunkR2YHJN6W_cjs.createHonoApp({
1203
+ return chunk5KVM3WEY_cjs.createHonoApp({
1200
1204
  registry: this.registry,
1201
1205
  db: this.db,
1202
1206
  authSecret,
@@ -1254,15 +1258,992 @@ var Kyro = class {
1254
1258
  function createKyro(config) {
1255
1259
  return new Kyro(config);
1256
1260
  }
1261
+ var _require = module$1.createRequire((typeof document === 'undefined' ? require('u' + 'rl').pathToFileURL(__filename).href : (_documentCurrentScript && _documentCurrentScript.tagName.toUpperCase() === 'SCRIPT' && _documentCurrentScript.src || new URL('chunk-LINKCEG4.cjs', document.baseURI).href)));
1262
+ var modPath = "node:sqlite";
1263
+ var { DatabaseSync } = _require(modPath);
1264
+ function flattenFields(fields) {
1265
+ const result = [];
1266
+ for (const field of fields) {
1267
+ if (field.type === "tabs" && "tabs" in field) {
1268
+ for (const tab of field.tabs) {
1269
+ result.push(...flattenFields(tab.fields));
1270
+ }
1271
+ } else if (field.type === "row" && "fields" in field) {
1272
+ result.push(...flattenFields(field.fields));
1273
+ } else if (field.type === "collapsible" && "fields" in field) {
1274
+ result.push(...flattenFields(field.fields));
1275
+ } else {
1276
+ result.push(field);
1277
+ }
1278
+ }
1279
+ return result;
1280
+ }
1281
+ function processFieldValue(row, field) {
1282
+ const f = field;
1283
+ let value = row[f.name];
1284
+ if (f.type === "json" || f.type === "richtext" || f.type === "array" || f.type === "group" || f.type === "blocks" || f.type === "list" || f.type === "relationship-block") {
1285
+ try {
1286
+ value = value ? JSON.parse(value) : null;
1287
+ } catch {
1288
+ value = null;
1289
+ }
1290
+ }
1291
+ if (f.type === "checkbox") {
1292
+ value = Boolean(value);
1293
+ }
1294
+ if (f.type === "date" && value) {
1295
+ try {
1296
+ const d = new Date(value);
1297
+ if (isNaN(d.getTime())) {
1298
+ value = null;
1299
+ } else {
1300
+ value = d.toISOString();
1301
+ }
1302
+ } catch {
1303
+ value = null;
1304
+ }
1305
+ }
1306
+ if ((f.type === "upload" || f.type === "image") && value) {
1307
+ try {
1308
+ const parsed = JSON.parse(value);
1309
+ if (Array.isArray(parsed)) {
1310
+ value = parsed.map((item) => {
1311
+ if (typeof item === "object" && item !== null) {
1312
+ return item;
1313
+ }
1314
+ return { id: item };
1315
+ });
1316
+ } else {
1317
+ value = typeof parsed === "object" ? parsed : { id: parsed };
1318
+ }
1319
+ } catch {
1320
+ value = { id: value };
1321
+ }
1322
+ }
1323
+ if (f.type === "relationship" && value) {
1324
+ try {
1325
+ const parsed = JSON.parse(value);
1326
+ if (Array.isArray(parsed)) {
1327
+ value = parsed;
1328
+ } else {
1329
+ value = parsed;
1330
+ }
1331
+ } catch {
1332
+ value = { relationTo: Array.isArray(f.relationTo) ? f.relationTo[0] : f.relationTo, value };
1333
+ }
1334
+ }
1335
+ if (f.type === "list" && value) {
1336
+ try {
1337
+ const parsed = JSON.parse(value);
1338
+ value = Array.isArray(parsed) ? parsed : [];
1339
+ } catch {
1340
+ value = [];
1341
+ }
1342
+ }
1343
+ if (f.type === "relationship-block" && value) {
1344
+ try {
1345
+ const parsed = JSON.parse(value);
1346
+ value = Array.isArray(parsed) ? parsed : [];
1347
+ } catch {
1348
+ value = [];
1349
+ }
1350
+ }
1351
+ return value;
1352
+ }
1353
+ function getTableColumns(db, tableName) {
1354
+ try {
1355
+ const rows = db.prepare(`PRAGMA table_info(${tableName})`).all();
1356
+ return rows.map((r) => r.name);
1357
+ } catch {
1358
+ return [];
1359
+ }
1360
+ }
1361
+ var LocalAdapter = class extends chunkSA7NSSIQ_cjs.AbstractBaseAdapter {
1362
+ db;
1363
+ path;
1364
+ migrations = /* @__PURE__ */ new Map();
1365
+ draftsTableName = "kyro_drafts";
1366
+ versionsTableName = "kyro_versions";
1367
+ tenantContext;
1368
+ setTenantContext(context) {
1369
+ this.tenantContext = context;
1370
+ }
1371
+ getTenantContext() {
1372
+ return this.tenantContext;
1373
+ }
1374
+ constructor(options) {
1375
+ super();
1376
+ this.path = options.path;
1377
+ if (options.db) {
1378
+ this.db = options.db;
1379
+ } else {
1380
+ this.db = null;
1381
+ }
1382
+ }
1383
+ async connect() {
1384
+ if (!this.db) {
1385
+ this.db = new DatabaseSync(this.path || ":memory:");
1386
+ }
1387
+ this.db.exec("PRAGMA journal_mode = WAL");
1388
+ this.db.exec("PRAGMA foreign_keys = ON");
1389
+ this.connected = true;
1390
+ console.log(
1391
+ `[LocalAdapter] Connected to SQLite (${this.path || "memory"})`
1392
+ );
1393
+ }
1394
+ async disconnect() {
1395
+ if (this.db) {
1396
+ this.db.close();
1397
+ }
1398
+ this.connected = false;
1399
+ console.log("[LocalAdapter] Disconnected");
1400
+ }
1401
+ // ========================================================================
1402
+ // Schema Management
1403
+ // ========================================================================
1404
+ ensureTable(config, tableName) {
1405
+ const name = tableName || this.getTableNameFor(config.slug);
1406
+ const columns = [`id TEXT PRIMARY KEY`];
1407
+ for (const field of flattenFields(config.fields)) {
1408
+ if (!field.name || field.name === "id") continue;
1409
+ const colDef = this.fieldToSQL(field);
1410
+ if (colDef) columns.push(colDef);
1411
+ }
1412
+ columns.push(`${this.col("createdAt")} TEXT DEFAULT (datetime('now'))`);
1413
+ columns.push(`${this.col("updatedAt")} TEXT DEFAULT (datetime('now'))`);
1414
+ columns.push(`_status TEXT DEFAULT 'published'`);
1415
+ columns.push(`_has_draft INTEGER DEFAULT 0`);
1416
+ if (config.tenantScoped) {
1417
+ columns.push(`tenant_id TEXT NOT NULL`);
1418
+ }
1419
+ const existingColumns = getTableColumns(this.db, name);
1420
+ if (existingColumns.length === 0) {
1421
+ const createSQL = `CREATE TABLE IF NOT EXISTS ${name} (${columns.join(", ")})`;
1422
+ this.db.exec(createSQL);
1423
+ this.db.exec(`CREATE INDEX IF NOT EXISTS idx_${name}__status ON ${name}(_status)`);
1424
+ for (const field of flattenFields(config.fields)) {
1425
+ if (field.name && field.indexed) {
1426
+ this.db.exec(
1427
+ `CREATE INDEX IF NOT EXISTS idx_${name}_${field.name} ON ${name}(${this.col(field.name)})`
1428
+ );
1429
+ }
1430
+ if (field.name && field.unique) {
1431
+ this.db.exec(
1432
+ `CREATE UNIQUE INDEX IF NOT EXISTS idx_${name}_${field.name}_unique ON ${name}(${this.col(field.name)})`
1433
+ );
1434
+ }
1435
+ }
1436
+ } else {
1437
+ const existingSet = new Set(existingColumns);
1438
+ for (const colDef of columns) {
1439
+ const colName = colDef.split(" ")[0].replace(/^"/, "").replace(/"$/, "");
1440
+ if (!existingSet.has(colName) && colName !== "id") {
1441
+ try {
1442
+ if (colName === "_status") {
1443
+ this.db.exec(`ALTER TABLE ${name} ADD COLUMN ${this.col(colName)} TEXT DEFAULT 'published'`);
1444
+ } else if (colName === "_has_draft") {
1445
+ this.db.exec(`ALTER TABLE ${name} ADD COLUMN ${this.col(colName)} INTEGER DEFAULT 0`);
1446
+ } else {
1447
+ this.db.exec(`ALTER TABLE ${name} ADD COLUMN ${this.col(colName)} TEXT`);
1448
+ }
1449
+ } catch {
1450
+ }
1451
+ }
1452
+ }
1453
+ }
1454
+ this.migrations.set(name, true);
1455
+ }
1456
+ ensureVersionsTable() {
1457
+ this.db.exec(`
1458
+ CREATE TABLE IF NOT EXISTS ${this.versionsTableName} (
1459
+ id TEXT PRIMARY KEY,
1460
+ collection_slug TEXT NOT NULL,
1461
+ document_id TEXT NOT NULL,
1462
+ tenant_id TEXT,
1463
+ version INTEGER NOT NULL,
1464
+ status TEXT NOT NULL DEFAULT 'draft',
1465
+ data TEXT NOT NULL,
1466
+ created_by TEXT,
1467
+ change_description TEXT,
1468
+ published_at TEXT,
1469
+ created_at TEXT DEFAULT (datetime('now')),
1470
+ updated_at TEXT DEFAULT (datetime('now'))
1471
+ )
1472
+ `);
1473
+ this.db.exec(
1474
+ `CREATE INDEX IF NOT EXISTS idx_${this.versionsTableName}_doc ON ${this.versionsTableName}(collection_slug, document_id)`
1475
+ );
1476
+ this.db.exec(
1477
+ `CREATE INDEX IF NOT EXISTS idx_${this.versionsTableName}_status ON ${this.versionsTableName}(status)`
1478
+ );
1479
+ }
1480
+ ensureDraftsTable() {
1481
+ this.db.exec(`
1482
+ CREATE TABLE IF NOT EXISTS ${this.draftsTableName} (
1483
+ id TEXT PRIMARY KEY,
1484
+ collection_slug TEXT NOT NULL,
1485
+ document_id TEXT NOT NULL,
1486
+ tenant_id TEXT,
1487
+ data TEXT NOT NULL,
1488
+ base_updated_at TEXT,
1489
+ draft_updated_at TEXT NOT NULL,
1490
+ created_at TEXT DEFAULT (datetime('now')),
1491
+ updated_at TEXT DEFAULT (datetime('now'))
1492
+ )
1493
+ `);
1494
+ this.db.exec(
1495
+ `CREATE UNIQUE INDEX IF NOT EXISTS idx_${this.draftsTableName}_document ON ${this.draftsTableName}(collection_slug, document_id, tenant_id)`
1496
+ );
1497
+ }
1498
+ // ========================================================================
1499
+ // SQL Quoting
1500
+ // ========================================================================
1501
+ col(name) {
1502
+ return `"${name}"`;
1503
+ }
1504
+ fieldToSQL(field) {
1505
+ switch (field.type) {
1506
+ case "text":
1507
+ case "email":
1508
+ case "password":
1509
+ case "textarea":
1510
+ case "color":
1511
+ case "code":
1512
+ case "markdown":
1513
+ case "url":
1514
+ return this.col(field.name) + " TEXT";
1515
+ case "number":
1516
+ return this.col(field.name) + " REAL";
1517
+ case "checkbox":
1518
+ return this.col(field.name) + " INTEGER DEFAULT 0";
1519
+ case "date":
1520
+ return this.col(field.name) + " TEXT";
1521
+ case "select":
1522
+ case "radio":
1523
+ return this.col(field.name) + " TEXT";
1524
+ case "relationship":
1525
+ case "upload":
1526
+ return this.col(field.name) + " TEXT";
1527
+ case "json":
1528
+ case "richtext":
1529
+ case "array":
1530
+ case "group":
1531
+ case "blocks":
1532
+ return this.col(field.name) + " TEXT";
1533
+ default:
1534
+ return null;
1535
+ }
1536
+ }
1537
+ // ========================================================================
1538
+ // CRUD Operations
1539
+ // ========================================================================
1540
+ parseGlobalsSlug(slug) {
1541
+ if (slug.startsWith("_globals_")) {
1542
+ const globalSlug = slug.replace("_globals_", "");
1543
+ return {
1544
+ isGlobal: true,
1545
+ globalSlug,
1546
+ tableName: `global_${globalSlug.replace(/-/g, "_")}`
1547
+ };
1548
+ }
1549
+ return {
1550
+ isGlobal: false,
1551
+ globalSlug: "",
1552
+ tableName: this.getTableNameFor(slug)
1553
+ };
1554
+ }
1555
+ async find(args) {
1556
+ const {
1557
+ collection: slug,
1558
+ where = {},
1559
+ sort,
1560
+ limit = 10,
1561
+ page = 1,
1562
+ tenantID,
1563
+ draft = false
1564
+ } = args;
1565
+ const parsed = this.parseGlobalsSlug(slug);
1566
+ const config = parsed.isGlobal ? this.globals.get(parsed.globalSlug) : this.getCollection(slug);
1567
+ this.ensureTable(config, parsed.tableName);
1568
+ const tableName = parsed.tableName;
1569
+ let sql = `SELECT * FROM ${tableName}`;
1570
+ const params = [];
1571
+ const conditions = [];
1572
+ let effectiveWhere = { ...where };
1573
+ if (this.tenantContext && config.tenantScoped) {
1574
+ const rlsQuery = chunkSA7NSSIQ_cjs.applyRLS({ where: effectiveWhere }, slug, this.tenantContext, chunkSA7NSSIQ_cjs.DEFAULT_RLS_CONFIG);
1575
+ effectiveWhere = rlsQuery.where || {};
1576
+ }
1577
+ if (!draft && config.versions?.drafts) {
1578
+ conditions.push(`_status = ?`);
1579
+ params.push("published");
1580
+ }
1581
+ if (tenantID && config.tenantScoped) {
1582
+ conditions.push(`tenant_id = ?`);
1583
+ params.push(tenantID);
1584
+ }
1585
+ for (const [key, value] of Object.entries(effectiveWhere)) {
1586
+ if (key === "AND" || key === "OR") continue;
1587
+ if (typeof value === "object" && value !== null) {
1588
+ if (value.equals !== void 0) {
1589
+ conditions.push(`${this.col(key)} = ?`);
1590
+ params.push(value.equals);
1591
+ }
1592
+ if (value.in !== void 0) {
1593
+ conditions.push(`${this.col(key)} IN (${value.in.map(() => "?").join(", ")})`);
1594
+ params.push(...value.in);
1595
+ }
1596
+ if (value.not_equals !== void 0) {
1597
+ conditions.push(`${this.col(key)} != ?`);
1598
+ params.push(value.not_equals);
1599
+ }
1600
+ } else {
1601
+ conditions.push(`${this.col(key)} = ?`);
1602
+ params.push(value);
1603
+ }
1604
+ }
1605
+ if (conditions.length > 0) {
1606
+ sql += ` WHERE ${conditions.join(" AND ")}`;
1607
+ }
1608
+ const sortField = this.col(sort?.replace("-", "") || "createdAt");
1609
+ const sortDir = sort?.startsWith("-") ? "DESC" : "ASC";
1610
+ sql += ` ORDER BY ${sortField} ${sortDir}`;
1611
+ const countSql = sql.replace("SELECT *", "SELECT COUNT(*) as count");
1612
+ const countResult = this.db.prepare(countSql).get(...params);
1613
+ const totalDocs = countResult?.count || 0;
1614
+ sql += ` LIMIT ? OFFSET ?`;
1615
+ params.push(limit, (page - 1) * limit);
1616
+ const rows = this.db.prepare(sql).all(...params);
1617
+ let docs = rows.map((row) => this.rowToDoc(row, config));
1618
+ if (this.tenantContext && !this.tenantContext.isSuperAdmin) {
1619
+ docs = docs.filter((doc) => chunkSA7NSSIQ_cjs.canAccessDocument(doc, slug, this.tenantContext, chunkSA7NSSIQ_cjs.DEFAULT_RLS_CONFIG));
1620
+ }
1621
+ if (draft) {
1622
+ docs = await Promise.all(docs.map(async (doc) => {
1623
+ if (doc._has_draft) {
1624
+ const versions = await this.findVersions({
1625
+ collection: slug,
1626
+ documentId: doc.id,
1627
+ limit: 1,
1628
+ sort: "-createdAt"
1629
+ });
1630
+ if (versions.docs.length > 0 && versions.docs[0].status === "draft") {
1631
+ return { ...doc, ...versions.docs[0].data, _has_draft: true, _status: doc._status };
1632
+ }
1633
+ }
1634
+ return doc;
1635
+ }));
1636
+ }
1637
+ return {
1638
+ docs,
1639
+ totalDocs,
1640
+ limit,
1641
+ totalPages: Math.ceil(totalDocs / limit),
1642
+ page,
1643
+ pagingCounter: (page - 1) * limit + 1,
1644
+ hasPrevPage: page > 1,
1645
+ hasNextPage: page < Math.ceil(totalDocs / limit),
1646
+ prevPage: page > 1 ? page - 1 : null,
1647
+ nextPage: page < Math.ceil(totalDocs / limit) ? page + 1 : null
1648
+ };
1649
+ }
1650
+ async findByID(args) {
1651
+ const { collection: slug, id, tenantID, draft = false } = args;
1652
+ const parsed = this.parseGlobalsSlug(slug);
1653
+ const config = parsed.isGlobal ? this.globals.get(parsed.globalSlug) : this.getCollection(slug);
1654
+ this.ensureTable(config, parsed.tableName);
1655
+ const tableName = parsed.tableName;
1656
+ let sql = `SELECT * FROM ${tableName} WHERE id = ?`;
1657
+ const params = [id];
1658
+ if (this.tenantContext && config.tenantScoped) {
1659
+ const tempDoc = { id, tenant_id: this.tenantContext.tenantId };
1660
+ if (!chunkSA7NSSIQ_cjs.canAccessDocument(tempDoc, slug, this.tenantContext, chunkSA7NSSIQ_cjs.DEFAULT_RLS_CONFIG)) {
1661
+ return null;
1662
+ }
1663
+ }
1664
+ if (!draft && config.versions?.drafts) {
1665
+ sql += ` AND _status = ?`;
1666
+ params.push("published");
1667
+ }
1668
+ if (tenantID && config.tenantScoped) {
1669
+ sql += ` AND tenant_id = ?`;
1670
+ params.push(tenantID);
1671
+ }
1672
+ const row = this.db.prepare(sql).get(...params);
1673
+ if (!row) return null;
1674
+ let doc = this.rowToDoc(row, config);
1675
+ if (draft && doc._has_draft) {
1676
+ const versions = await this.findVersions({
1677
+ collection: slug,
1678
+ documentId: doc.id,
1679
+ limit: 1,
1680
+ sort: "-createdAt"
1681
+ });
1682
+ if (versions.docs.length > 0 && versions.docs[0].status === "draft") {
1683
+ doc = { ...doc, ...versions.docs[0].data, _has_draft: true, _status: doc._status };
1684
+ }
1685
+ }
1686
+ return doc;
1687
+ }
1688
+ async create(args) {
1689
+ const { collection: slug, data, tenantID } = args;
1690
+ const parsed = this.parseGlobalsSlug(slug);
1691
+ const config = parsed.isGlobal ? this.globals.get(parsed.globalSlug) : this.getCollection(slug);
1692
+ this.ensureTable(config, parsed.tableName);
1693
+ const tableName = parsed.tableName;
1694
+ const id = parsed.isGlobal ? parsed.globalSlug : data.id || this.generateId();
1695
+ const insertData = this.prepareData(data, config);
1696
+ insertData.id = id;
1697
+ insertData.created_at = (/* @__PURE__ */ new Date()).toISOString();
1698
+ insertData.updated_at = (/* @__PURE__ */ new Date()).toISOString();
1699
+ if (tenantID && config.tenantScoped) {
1700
+ insertData.tenant_id = tenantID;
1701
+ }
1702
+ const columns = Object.keys(insertData);
1703
+ const validColumns = getTableColumns(this.db, tableName);
1704
+ const filteredData = {};
1705
+ for (const key of columns) {
1706
+ if (validColumns.includes(key)) {
1707
+ filteredData[key] = insertData[key];
1708
+ }
1709
+ }
1710
+ const filteredColumns = Object.keys(filteredData);
1711
+ const quotedColumns = filteredColumns.map((c) => this.col(c));
1712
+ const placeholders = filteredColumns.map(() => "?").join(", ");
1713
+ const values = Object.values(filteredData).map(
1714
+ (v) => typeof v === "object" ? JSON.stringify(v) : v
1715
+ );
1716
+ this.db.prepare(
1717
+ `INSERT OR REPLACE INTO ${tableName} (${quotedColumns.join(", ")}) VALUES (${placeholders})`
1718
+ ).run(...values);
1719
+ return this.findByID({ collection: slug, id, tenantID });
1720
+ }
1721
+ async update(args) {
1722
+ const { collection: slug, id, data, tenantID } = args;
1723
+ const parsed = this.parseGlobalsSlug(slug);
1724
+ const config = parsed.isGlobal ? this.globals.get(parsed.globalSlug) : this.getCollection(slug);
1725
+ this.ensureTable(config, parsed.tableName);
1726
+ const tableName = parsed.tableName;
1727
+ const updateData = this.prepareData(data, config);
1728
+ updateData.updated_at = (/* @__PURE__ */ new Date()).toISOString();
1729
+ const validColumns = getTableColumns(this.db, tableName);
1730
+ const filteredData = {};
1731
+ for (const key of Object.keys(updateData)) {
1732
+ if (validColumns.includes(key)) {
1733
+ filteredData[key] = updateData[key];
1734
+ }
1735
+ }
1736
+ const columns = Object.keys(filteredData);
1737
+ const setClause = columns.map((c) => `${this.col(c)} = ?`).join(", ");
1738
+ const values = Object.values(filteredData).map(
1739
+ (v) => v !== null && typeof v === "object" ? JSON.stringify(v) : v
1740
+ );
1741
+ let sql = `UPDATE ${tableName} SET ${setClause} WHERE id = ?`;
1742
+ const params = [...values, id];
1743
+ if (tenantID && config.tenantScoped) {
1744
+ sql += ` AND tenant_id = ?`;
1745
+ params.push(tenantID);
1746
+ }
1747
+ this.db.prepare(sql).run(...params);
1748
+ return this.findByID({ collection: slug, id, tenantID, draft: true });
1749
+ }
1750
+ async delete(args) {
1751
+ const { collection: slug, id, tenantID } = args;
1752
+ const parsed = this.parseGlobalsSlug(slug);
1753
+ const config = parsed.isGlobal ? this.globals.get(parsed.globalSlug) : this.getCollection(slug);
1754
+ this.ensureTable(config, parsed.tableName);
1755
+ const doc = await this.findByID({ collection: slug, id, tenantID });
1756
+ if (!doc) throw new Error(`Document not found: ${slug}/${id}`);
1757
+ const tableName = parsed.tableName;
1758
+ let sql = `DELETE FROM ${tableName} WHERE id = ?`;
1759
+ const params = [id];
1760
+ if (tenantID && config.tenantScoped) {
1761
+ sql += ` AND tenant_id = ?`;
1762
+ params.push(tenantID);
1763
+ }
1764
+ this.db.prepare(sql).run(...params);
1765
+ return doc;
1766
+ }
1767
+ async count(args) {
1768
+ const { collection: slug, tenantID } = args;
1769
+ const parsed = this.parseGlobalsSlug(slug);
1770
+ const config = parsed.isGlobal ? this.globals.get(parsed.globalSlug) : this.getCollection(slug);
1771
+ this.ensureTable(config, parsed.tableName);
1772
+ const tableName = parsed.tableName;
1773
+ let sql = `SELECT COUNT(*) as count FROM ${tableName}`;
1774
+ const params = [];
1775
+ if (tenantID && config.tenantScoped) {
1776
+ sql += ` WHERE tenant_id = ?`;
1777
+ params.push(tenantID);
1778
+ }
1779
+ const result = this.db.prepare(sql).get(...params);
1780
+ return result?.count || 0;
1781
+ }
1782
+ async findOne(args) {
1783
+ const parsed = this.parseGlobalsSlug(args.collection);
1784
+ if (parsed.isGlobal) {
1785
+ const globalConfig = this.globals.get(parsed.globalSlug);
1786
+ if (!globalConfig) {
1787
+ throw new Error(`Global "${parsed.globalSlug}" not found in adapter`);
1788
+ }
1789
+ this.ensureTable(globalConfig, parsed.tableName);
1790
+ let sql = `SELECT * FROM ${parsed.tableName}`;
1791
+ const conditions = [];
1792
+ const params = [];
1793
+ if (!args.draft && globalConfig.versions) {
1794
+ conditions.push("_status = 'published'");
1795
+ }
1796
+ if (conditions.length > 0) {
1797
+ sql += ` WHERE ${conditions.join(" AND ")}`;
1798
+ }
1799
+ sql += " LIMIT 1";
1800
+ const result2 = this.db.prepare(sql).get(...params);
1801
+ if (result2) {
1802
+ let doc = this.rowToDoc(result2, globalConfig);
1803
+ if (args.draft && doc._has_draft) {
1804
+ const versions = await this.findVersions({
1805
+ collection: args.collection,
1806
+ documentId: parsed.globalSlug,
1807
+ limit: 1,
1808
+ sort: "-createdAt"
1809
+ });
1810
+ if (versions.docs.length > 0 && versions.docs[0].status === "draft") {
1811
+ doc = { ...doc, ...versions.docs[0].data, _has_draft: true, _status: doc._status };
1812
+ }
1813
+ }
1814
+ return doc;
1815
+ }
1816
+ return null;
1817
+ }
1818
+ const result = await this.find({ ...args, limit: 1 });
1819
+ return result.docs[0] || null;
1820
+ }
1821
+ // ========================================================================
1822
+ // Version History
1823
+ // ========================================================================
1824
+ async findVersions(args) {
1825
+ this.ensureVersionsTable();
1826
+ const { collection, documentId, tenantID, limit = 20, page = 1 } = args;
1827
+ const conditions = [`collection_slug = ?`, `document_id = ?`];
1828
+ const params = [collection, documentId];
1829
+ if (tenantID) {
1830
+ conditions.push(`tenant_id = ?`);
1831
+ params.push(tenantID);
1832
+ } else {
1833
+ conditions.push(`tenant_id IS NULL`);
1834
+ }
1835
+ const where = `WHERE ${conditions.join(" AND ")}`;
1836
+ const countResult = this.db.prepare(`SELECT COUNT(*) as count FROM ${this.versionsTableName} ${where}`).get(...params);
1837
+ const totalDocs = countResult?.count || 0;
1838
+ const offset = (page - 1) * limit;
1839
+ const rows = this.db.prepare(`SELECT * FROM ${this.versionsTableName} ${where} ORDER BY version DESC LIMIT ? OFFSET ?`).all(...params, limit, offset);
1840
+ const docs = rows.map((r) => this.rowToVersion(r));
1841
+ return {
1842
+ docs,
1843
+ totalDocs,
1844
+ limit,
1845
+ totalPages: Math.ceil(totalDocs / limit),
1846
+ page,
1847
+ pagingCounter: (page - 1) * limit + 1,
1848
+ hasPrevPage: page > 1,
1849
+ hasNextPage: page < Math.ceil(totalDocs / limit),
1850
+ prevPage: page > 1 ? page - 1 : null,
1851
+ nextPage: page < Math.ceil(totalDocs / limit) ? page + 1 : null
1852
+ };
1853
+ }
1854
+ async findVersionByID(args) {
1855
+ this.ensureVersionsTable();
1856
+ const row = this.db.prepare(`SELECT * FROM ${this.versionsTableName} WHERE id = ? AND collection_slug = ? LIMIT 1`).get(args.versionId, args.collection);
1857
+ return row ? this.rowToVersion(row) : null;
1858
+ }
1859
+ async createVersion(args) {
1860
+ this.ensureVersionsTable();
1861
+ const now = (/* @__PURE__ */ new Date()).toISOString();
1862
+ const id = this.generateId();
1863
+ const latestRow = this.db.prepare(`SELECT version FROM ${this.versionsTableName} WHERE collection_slug = ? AND document_id = ? ORDER BY version DESC LIMIT 1`).get(args.collection, args.documentId);
1864
+ const nextVersion = (latestRow?.version ?? 0) + 1;
1865
+ this.db.prepare(
1866
+ `INSERT INTO ${this.versionsTableName} (
1867
+ id, collection_slug, document_id, tenant_id, version, status, data, created_by, change_description, published_at, created_at, updated_at
1868
+ ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`
1869
+ ).run(
1870
+ id,
1871
+ args.collection,
1872
+ args.documentId,
1873
+ args.tenantID ?? null,
1874
+ nextVersion,
1875
+ args.status,
1876
+ JSON.stringify(args.data),
1877
+ args.createdBy ?? null,
1878
+ args.changeDescription ?? null,
1879
+ args.status === "published" ? now : null,
1880
+ now,
1881
+ now
1882
+ );
1883
+ const collectionConfig = this.collections.get(args.collection);
1884
+ const maxPerDoc = collectionConfig?.versions?.maxPerDoc;
1885
+ if (maxPerDoc && maxPerDoc > 0) {
1886
+ await this.deleteVersions({ collection: args.collection, documentId: args.documentId, keepLatest: maxPerDoc, tenantID: args.tenantID });
1887
+ }
1888
+ const saved = await this.findVersionByID({ collection: args.collection, versionId: id });
1889
+ return saved;
1890
+ }
1891
+ async deleteVersions(args) {
1892
+ this.ensureVersionsTable();
1893
+ const { collection, documentId, keepLatest, tenantID } = args;
1894
+ if (keepLatest && keepLatest > 0) {
1895
+ const rows = this.db.prepare(`SELECT id, status FROM ${this.versionsTableName} WHERE collection_slug = ? AND document_id = ? ORDER BY version DESC`).all(collection, documentId);
1896
+ let draftCount = 0;
1897
+ const toDelete = [];
1898
+ for (const row of rows) {
1899
+ if (row.status === "published") continue;
1900
+ draftCount++;
1901
+ if (draftCount > keepLatest) toDelete.push(row.id);
1902
+ }
1903
+ for (const vid of toDelete) {
1904
+ this.db.prepare(`DELETE FROM ${this.versionsTableName} WHERE id = ?`).run(vid);
1905
+ }
1906
+ } else {
1907
+ let sql = `DELETE FROM ${this.versionsTableName} WHERE collection_slug = ? AND document_id = ?`;
1908
+ const params = [collection, documentId];
1909
+ if (tenantID) {
1910
+ sql += ` AND tenant_id = ?`;
1911
+ params.push(tenantID);
1912
+ }
1913
+ this.db.prepare(sql).run(...params);
1914
+ }
1915
+ }
1916
+ rowToVersion(row) {
1917
+ return {
1918
+ id: String(row.id),
1919
+ collection: row.collection_slug,
1920
+ documentId: row.document_id,
1921
+ version: row.version,
1922
+ status: row.status,
1923
+ data: row.data ? JSON.parse(row.data) : {},
1924
+ createdBy: row.created_by ?? void 0,
1925
+ changeDescription: row.change_description ?? void 0,
1926
+ publishedAt: row.published_at ?? null,
1927
+ createdAt: row.created_at,
1928
+ updatedAt: row.updated_at
1929
+ };
1930
+ }
1931
+ async findDraft(args) {
1932
+ this.ensureDraftsTable();
1933
+ let sql = `SELECT * FROM ${this.draftsTableName} WHERE collection_slug = ? AND document_id = ?`;
1934
+ const params = [args.collection, args.documentId];
1935
+ if (args.tenantID) {
1936
+ sql += ` AND tenant_id = ?`;
1937
+ params.push(args.tenantID);
1938
+ } else {
1939
+ sql += ` AND tenant_id IS NULL`;
1940
+ }
1941
+ sql += ` LIMIT 1`;
1942
+ const row = this.db.prepare(sql).get(...params);
1943
+ if (!row) return null;
1944
+ return this.rowToDraft(row);
1945
+ }
1946
+ async upsertDraft(args) {
1947
+ this.ensureDraftsTable();
1948
+ const existing = await this.findDraft(args);
1949
+ const now = (/* @__PURE__ */ new Date()).toISOString();
1950
+ const draftUpdatedAt = args.draftUpdatedAt || now;
1951
+ const id = existing?.id || this.generateId();
1952
+ this.db.prepare(
1953
+ `INSERT OR REPLACE INTO ${this.draftsTableName} (
1954
+ id, collection_slug, document_id, tenant_id, data, base_updated_at, draft_updated_at, created_at, updated_at
1955
+ ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)`
1956
+ ).run(
1957
+ id,
1958
+ args.collection,
1959
+ args.documentId,
1960
+ args.tenantID ?? null,
1961
+ JSON.stringify(args.data),
1962
+ args.baseUpdatedAt ?? null,
1963
+ draftUpdatedAt,
1964
+ existing?.createdAt || now,
1965
+ now
1966
+ );
1967
+ const saved = await this.findDraft(args);
1968
+ if (!saved) {
1969
+ throw new Error("Failed to persist draft snapshot");
1970
+ }
1971
+ return saved;
1972
+ }
1973
+ async deleteDraft(args) {
1974
+ this.ensureDraftsTable();
1975
+ let sql = `DELETE FROM ${this.draftsTableName} WHERE collection_slug = ? AND document_id = ?`;
1976
+ const params = [args.collection, args.documentId];
1977
+ if (args.tenantID) {
1978
+ sql += ` AND tenant_id = ?`;
1979
+ params.push(args.tenantID);
1980
+ } else {
1981
+ sql += ` AND tenant_id IS NULL`;
1982
+ }
1983
+ this.db.prepare(sql).run(...params);
1984
+ }
1985
+ // ========================================================================
1986
+ // Helpers
1987
+ // ========================================================================
1988
+ prepareData(data, config) {
1989
+ const result = {};
1990
+ const fields = flattenFields(config.fields);
1991
+ const processValue = (field, value) => {
1992
+ const f = field;
1993
+ if (f.type === "json" || f.type === "richtext" || f.type === "array" || f.type === "group" || f.type === "blocks" || f.type === "list" || f.type === "relationship-block") {
1994
+ return value !== null && value !== void 0 ? JSON.stringify(value) : null;
1995
+ } else if (f.type === "checkbox") {
1996
+ return value ? 1 : 0;
1997
+ } else if (f.type === "number") {
1998
+ return value !== null && value !== "" ? Number(value) : null;
1999
+ } else if (f.type === "upload" || f.type === "image") {
2000
+ if (value === null || value === void 0) return null;
2001
+ if (Array.isArray(value)) {
2002
+ const items = value.map((v) => {
2003
+ if (typeof v === "string") return v;
2004
+ if (typeof v === "object") return v.id || v._id || v;
2005
+ return String(v);
2006
+ });
2007
+ return JSON.stringify(items);
2008
+ }
2009
+ if (typeof value === "string") return value;
2010
+ if (typeof value === "object") return JSON.stringify(value);
2011
+ return String(value);
2012
+ } else if (field.type === "relationship") {
2013
+ if (value === null || value === void 0) return null;
2014
+ if (Array.isArray(value)) {
2015
+ const rels = value.map((v) => {
2016
+ if (typeof v === "string") return v;
2017
+ if (typeof v === "object") return JSON.stringify({ relationTo: field.relationTo, value: v.id || v });
2018
+ return String(v);
2019
+ });
2020
+ return JSON.stringify(rels);
2021
+ }
2022
+ if (typeof value === "string") return value;
2023
+ if (typeof value === "object") return JSON.stringify({ relationTo: field.relationTo, value: value.id || value });
2024
+ return String(value);
2025
+ }
2026
+ return value;
2027
+ };
2028
+ for (const field of fields) {
2029
+ if (!field.name || field.name === "id") continue;
2030
+ const isInTab = config.fields.some(
2031
+ (f) => f.type === "tabs" && "tabs" in f && f.tabs.some((t) => t.fields.some((tf) => tf.name === field.name))
2032
+ );
2033
+ if (isInTab) continue;
2034
+ const value = data[field.name];
2035
+ if (value !== void 0) {
2036
+ result[field.name] = processValue(field, value);
2037
+ }
2038
+ }
2039
+ for (const field of config.fields) {
2040
+ if (field.type === "tabs" && "tabs" in field && field.name) {
2041
+ const tabData = data[field.name];
2042
+ if (tabData && typeof tabData === "object") {
2043
+ for (const tab of field.tabs) {
2044
+ for (const tabField of tab.fields) {
2045
+ if (tabField.name && tabField.name !== "id") {
2046
+ const value = tabData[tabField.name];
2047
+ if (value !== void 0) {
2048
+ result[tabField.name] = processValue(tabField, value);
2049
+ }
2050
+ }
2051
+ }
2052
+ }
2053
+ }
2054
+ }
2055
+ }
2056
+ return result;
2057
+ }
2058
+ rowToDoc(row, config) {
2059
+ const doc = { id: row.id };
2060
+ for (const field of flattenFields(config.fields)) {
2061
+ if (!field.name || field.name === "id") continue;
2062
+ const f = field;
2063
+ let value = row[f.name];
2064
+ if (f.type === "json" || f.type === "richtext" || f.type === "array" || f.type === "group" || f.type === "blocks" || f.type === "list" || f.type === "relationship-block") {
2065
+ try {
2066
+ value = value ? JSON.parse(value) : null;
2067
+ } catch {
2068
+ value = null;
2069
+ }
2070
+ }
2071
+ if (field.type === "checkbox") {
2072
+ value = Boolean(value);
2073
+ }
2074
+ if (field.type === "date" && value) {
2075
+ try {
2076
+ const d = new Date(value);
2077
+ if (isNaN(d.getTime())) {
2078
+ console.warn(`[LocalAdapter] Invalid date value for field "${field.name}":`, value);
2079
+ value = null;
2080
+ } else {
2081
+ value = d.toISOString();
2082
+ }
2083
+ } catch {
2084
+ value = null;
2085
+ }
2086
+ }
2087
+ if ((field.type === "upload" || field.type === "image") && value) {
2088
+ try {
2089
+ const parsed = JSON.parse(value);
2090
+ if (Array.isArray(parsed)) {
2091
+ value = parsed.map((item) => {
2092
+ if (typeof item === "object" && item !== null) {
2093
+ return item;
2094
+ }
2095
+ return { id: item };
2096
+ });
2097
+ } else {
2098
+ value = typeof parsed === "object" ? parsed : { id: parsed };
2099
+ }
2100
+ } catch {
2101
+ value = { id: value };
2102
+ }
2103
+ }
2104
+ if (field.type === "relationship" && value) {
2105
+ try {
2106
+ const parsed = JSON.parse(value);
2107
+ if (Array.isArray(parsed)) {
2108
+ value = parsed;
2109
+ } else {
2110
+ value = parsed;
2111
+ }
2112
+ } catch {
2113
+ value = { relationTo: Array.isArray(field.relationTo) ? field.relationTo[0] : field.relationTo, value };
2114
+ }
2115
+ }
2116
+ doc[field.name] = value;
2117
+ }
2118
+ for (const field of config.fields) {
2119
+ if (!field.name || field.name === "id" || field.name === "row" || field.name === "collapsible") continue;
2120
+ if (field.type === "tabs" && "tabs" in field) {
2121
+ const tabData = {};
2122
+ for (const tab of field.tabs) {
2123
+ for (const tabField of tab.fields) {
2124
+ if (tabField.name && tabField.name !== "id") {
2125
+ tabData[tabField.name] = processFieldValue(row, tabField);
2126
+ }
2127
+ }
2128
+ }
2129
+ doc[field.name] = tabData;
2130
+ }
2131
+ }
2132
+ if (config.timestamps) {
2133
+ const cAt = row.createdAt || row.created_at;
2134
+ const uAt = row.updatedAt || row.updated_at;
2135
+ if (cAt) {
2136
+ try {
2137
+ const d = new Date(cAt);
2138
+ if (!isNaN(d.getTime())) doc.createdAt = d.toISOString();
2139
+ } catch {
2140
+ }
2141
+ }
2142
+ if (uAt) {
2143
+ try {
2144
+ const d = new Date(uAt);
2145
+ if (!isNaN(d.getTime())) doc.updatedAt = d.toISOString();
2146
+ } catch {
2147
+ }
2148
+ }
2149
+ }
2150
+ if (config.tenantScoped) {
2151
+ doc.tenantID = row.tenant_id;
2152
+ }
2153
+ return doc;
2154
+ }
2155
+ generateId() {
2156
+ const timestamp = Date.now().toString(16).padStart(12, "0");
2157
+ const random = crypto.randomBytes(6).toString("hex");
2158
+ return timestamp + random;
2159
+ }
2160
+ getMediaById(mediaId) {
2161
+ try {
2162
+ const tableName = this.getTableNameFor("media");
2163
+ const row = this.db.prepare(`SELECT id, url, thumbnail_url FROM ${tableName} WHERE id = ?`).get(mediaId);
2164
+ if (row) {
2165
+ return {
2166
+ id: row.id,
2167
+ url: row.url,
2168
+ thumbnailUrl: row.thumbnail_url || row.url
2169
+ };
2170
+ }
2171
+ } catch (err) {
2172
+ }
2173
+ return null;
2174
+ }
2175
+ getTableNameFor(slug) {
2176
+ return slug.replace(/-/g, "_");
2177
+ }
2178
+ rowToDraft(row) {
2179
+ return {
2180
+ id: row.id,
2181
+ collection: row.collection_slug,
2182
+ documentId: row.document_id,
2183
+ tenantID: row.tenant_id ?? void 0,
2184
+ data: row.data ? JSON.parse(row.data) : {},
2185
+ baseUpdatedAt: row.base_updated_at ?? null,
2186
+ draftUpdatedAt: row.draft_updated_at,
2187
+ createdAt: row.created_at,
2188
+ updatedAt: row.updated_at
2189
+ };
2190
+ }
2191
+ // ========================================================================
2192
+ // Migrations
2193
+ // ========================================================================
2194
+ async migrate() {
2195
+ for (const config of this.collections.values()) {
2196
+ this.ensureTable(config);
2197
+ }
2198
+ this.ensureDraftsTable();
2199
+ console.log("[LocalAdapter] Migrations complete");
2200
+ }
2201
+ async rollback() {
2202
+ console.log("[LocalAdapter] Rollback not supported for schema changes");
2203
+ }
2204
+ // ========================================================================
2205
+ // Transaction Support
2206
+ // ========================================================================
2207
+ async transaction(fn) {
2208
+ return new Promise((resolve, reject) => {
2209
+ const tx = this.db.transaction(async () => {
2210
+ return fn({ db: this.db });
2211
+ });
2212
+ try {
2213
+ const result = tx();
2214
+ resolve(result);
2215
+ } catch (error) {
2216
+ reject(error);
2217
+ }
2218
+ });
2219
+ }
2220
+ // ========================================================================
2221
+ // Direct DB Access
2222
+ // ========================================================================
2223
+ getDatabase() {
2224
+ return this.db;
2225
+ }
2226
+ exec(sql) {
2227
+ this.db.exec(sql);
2228
+ }
2229
+ prepare(sql) {
2230
+ return this.db.prepare(sql);
2231
+ }
2232
+ };
2233
+ function createLocalAdapter(options) {
2234
+ return new LocalAdapter(options || {});
2235
+ }
1257
2236
 
1258
2237
  exports.ConfigValidationError = ConfigValidationError;
1259
2238
  exports.Kyro = Kyro;
2239
+ exports.LocalAdapter = LocalAdapter;
1260
2240
  exports.Registry = Registry;
1261
2241
  exports.collectionToCreateZod = collectionToCreateZod;
1262
2242
  exports.collectionToUpdateZod = collectionToUpdateZod;
1263
2243
  exports.collectionToWhereZod = collectionToWhereZod;
1264
2244
  exports.collectionToZod = collectionToZod;
1265
2245
  exports.createKyro = createKyro;
2246
+ exports.createLocalAdapter = createLocalAdapter;
1266
2247
  exports.createRegistry = createRegistry;
1267
2248
  exports.fieldToZod = fieldToZod;
1268
2249
  exports.getRegistry = getRegistry;
@@ -1272,5 +2253,5 @@ exports.validateCollection = validateCollection;
1272
2253
  exports.validateConfig = validateConfig;
1273
2254
  exports.validateFields = validateFields;
1274
2255
  exports.validateGlobal = validateGlobal;
1275
- //# sourceMappingURL=chunk-55BNRTLW.cjs.map
1276
- //# sourceMappingURL=chunk-55BNRTLW.cjs.map
2256
+ //# sourceMappingURL=chunk-LINKCEG4.cjs.map
2257
+ //# sourceMappingURL=chunk-LINKCEG4.cjs.map