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