@rljson/io 0.0.56 → 0.0.58

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.
@@ -1,103 +1,103 @@
1
- /*
2
- * @license
3
- * Copyright (c) 2025 Rljson
4
- *
5
- * Use of this source code is governed by terms that can be
6
- * found in the LICENSE file in the root of this package.
7
- */
8
-
9
- // Create a folder dist/conformance-tests if it does not exist
10
- import { existsSync } from 'fs';
11
- import fs from 'fs/promises';
12
- import path from 'path';
13
- import { blue, gray, green, red, yellow } from './functions/colors.js';
14
- import {
15
- nodeModulesDir,
16
- scriptsDir,
17
- testDir,
18
- } from './functions/directories.js';
19
- import { syncFolders } from './functions/sync-folders.js';
20
-
21
- // .............................................................................
22
- async function _copyGoldens(srcDir) {
23
- const from = path.join(srcDir, 'goldens');
24
- const to = path.join(testDir, 'goldens', 'io-conformance');
25
- if (!existsSync(to)) {
26
- await fs.mkdir(to, { recursive: true });
27
- }
28
-
29
- console.log(gray(`cp -r ${from} ${to}`));
30
- await syncFolders(from, to, { excludeHidden: true });
31
- }
32
-
33
- // .............................................................................
34
- async function _copyConformanceTests(srcDir) {
35
- const from = path.join(srcDir, 'io-conformance.spec.ts');
36
- const to = path.join(testDir, 'io-conformance.spec.ts');
37
- console.log(gray(`cp ${from} ${to}`));
38
- await fs.copyFile(from, to);
39
- }
40
-
41
- // .............................................................................
42
- async function _srcDir() {
43
- const targetDir = path.join(
44
- nodeModulesDir,
45
- '@rljson',
46
- 'io',
47
- 'dist',
48
- 'conformance-tests',
49
- );
50
- if (!existsSync(targetDir)) {
51
- await fs.mkdir(targetDir, { recursive: true });
52
- }
53
- return targetDir;
54
- }
55
-
56
- // .............................................................................
57
- async function _copyInstallScript(srcDir) {
58
- const from = path.join(srcDir, 'scripts', 'install-conformance-tests.js');
59
- const to = path.join(scriptsDir, 'install-conformance-tests.js');
60
- console.log(gray(`cp ${from} ${to}`));
61
- await fs.copyFile(from, to);
62
- }
63
-
64
- // .............................................................................
65
- async function _copySetupScript(srcDir) {
66
- const from = path.join(srcDir, 'io-conformance.setup.ts');
67
- const to = path.join(testDir, 'io-conformance.setup.ts');
68
-
69
- // Only setup file if not exists
70
- if (existsSync(to)) {
71
- return;
72
- }
73
-
74
- // Update exports of Io, IoTestSetup and IoTools
75
- console.log(gray(`cp ${from} ${to}`));
76
- await fs.copyFile(from, to);
77
-
78
- console.log(
79
- [
80
- yellow('Please open'),
81
- blue('test/io-conformance.setup.ts'),
82
- yellow('and modify'),
83
- green('MyIoTestSetup'),
84
- yellow('to instantiate your Io implementation.'),
85
- ].join(' '),
86
- );
87
- }
88
-
89
- // .............................................................................
90
- try {
91
- // Create target directory if it doesn't exist
92
- console.log(blue('Update conformance tests...'));
93
- const srcDir = await _srcDir();
94
- await _copyConformanceTests(srcDir);
95
- await _copyGoldens(srcDir);
96
- await _copyInstallScript(srcDir);
97
- await _copySetupScript(srcDir);
98
- } catch (err) {
99
- console.error(
100
- red('❌ Error while deploying conformance tests:', err.message),
101
- );
102
- process.exit(1);
103
- }
1
+ /*
2
+ * @license
3
+ * Copyright (c) 2025 Rljson
4
+ *
5
+ * Use of this source code is governed by terms that can be
6
+ * found in the LICENSE file in the root of this package.
7
+ */
8
+
9
+ // Create a folder dist/conformance-tests if it does not exist
10
+ import { existsSync } from 'fs';
11
+ import fs from 'fs/promises';
12
+ import path from 'path';
13
+ import { blue, gray, green, red, yellow } from './functions/colors.js';
14
+ import {
15
+ nodeModulesDir,
16
+ scriptsDir,
17
+ testDir,
18
+ } from './functions/directories.js';
19
+ import { syncFolders } from './functions/sync-folders.js';
20
+
21
+ // .............................................................................
22
+ async function _copyGoldens(srcDir) {
23
+ const from = path.join(srcDir, 'goldens');
24
+ const to = path.join(testDir, 'goldens', 'io-conformance');
25
+ if (!existsSync(to)) {
26
+ await fs.mkdir(to, { recursive: true });
27
+ }
28
+
29
+ console.log(gray(`cp -r ${from} ${to}`));
30
+ await syncFolders(from, to, { excludeHidden: true });
31
+ }
32
+
33
+ // .............................................................................
34
+ async function _copyConformanceTests(srcDir) {
35
+ const from = path.join(srcDir, 'io-conformance.spec.ts');
36
+ const to = path.join(testDir, 'io-conformance.spec.ts');
37
+ console.log(gray(`cp ${from} ${to}`));
38
+ await fs.copyFile(from, to);
39
+ }
40
+
41
+ // .............................................................................
42
+ async function _srcDir() {
43
+ const targetDir = path.join(
44
+ nodeModulesDir,
45
+ '@rljson',
46
+ 'io',
47
+ 'dist',
48
+ 'conformance-tests',
49
+ );
50
+ if (!existsSync(targetDir)) {
51
+ await fs.mkdir(targetDir, { recursive: true });
52
+ }
53
+ return targetDir;
54
+ }
55
+
56
+ // .............................................................................
57
+ async function _copyInstallScript(srcDir) {
58
+ const from = path.join(srcDir, 'scripts', 'install-conformance-tests.js');
59
+ const to = path.join(scriptsDir, 'install-conformance-tests.js');
60
+ console.log(gray(`cp ${from} ${to}`));
61
+ await fs.copyFile(from, to);
62
+ }
63
+
64
+ // .............................................................................
65
+ async function _copySetupScript(srcDir) {
66
+ const from = path.join(srcDir, 'io-conformance.setup.ts');
67
+ const to = path.join(testDir, 'io-conformance.setup.ts');
68
+
69
+ // Only setup file if not exists
70
+ if (existsSync(to)) {
71
+ return;
72
+ }
73
+
74
+ // Update exports of Io, IoTestSetup and IoTools
75
+ console.log(gray(`cp ${from} ${to}`));
76
+ await fs.copyFile(from, to);
77
+
78
+ console.log(
79
+ [
80
+ yellow('Please open'),
81
+ blue('test/io-conformance.setup.ts'),
82
+ yellow('and modify'),
83
+ green('MyIoTestSetup'),
84
+ yellow('to instantiate your Io implementation.'),
85
+ ].join(' '),
86
+ );
87
+ }
88
+
89
+ // .............................................................................
90
+ try {
91
+ // Create target directory if it doesn't exist
92
+ console.log(blue('Update conformance tests...'));
93
+ const srcDir = await _srcDir();
94
+ await _copyConformanceTests(srcDir);
95
+ await _copyGoldens(srcDir);
96
+ await _copyInstallScript(srcDir);
97
+ await _copySetupScript(srcDir);
98
+ } catch (err) {
99
+ console.error(
100
+ red('❌ Error while deploying conformance tests:', err.message),
101
+ );
102
+ process.exit(1);
103
+ }
package/dist/index.d.ts CHANGED
@@ -1,3 +1,4 @@
1
+ export * from './io-db-name-mapping.ts';
1
2
  export * from './io-mem.ts';
2
3
  export * from './io-test-setup.ts';
3
4
  export * from './io-tools.ts';
@@ -0,0 +1,15 @@
1
+ export declare class IoDbNameMapping {
2
+ primaryKeyColumn: string;
3
+ dataSection: string;
4
+ typeColumn: string;
5
+ tableNames: {
6
+ [key: string]: string;
7
+ };
8
+ private _suffix;
9
+ private _addFix;
10
+ addTableSuffix(name: string): string;
11
+ addColumnSuffix(name: string): string;
12
+ private _removeFix;
13
+ removeTableSuffix(name: string): string;
14
+ removeColumnSuffix(name: string): string;
15
+ }
package/dist/io.js CHANGED
@@ -2,6 +2,43 @@ import { hip, hsh } from "@rljson/hash";
2
2
  import { IsReady } from "@rljson/is-ready";
3
3
  import { copy, equals } from "@rljson/json";
4
4
  import { iterateTables, throwOnInvalidTableCfg, validateRljsonAgainstTableCfg, iterateTablesSync } from "@rljson/rljson";
5
+ class IoDbNameMapping {
6
+ // The primary key column is always named '_hash'
7
+ primaryKeyColumn = "_hash";
8
+ dataSection = "_data";
9
+ typeColumn = "type";
10
+ // Names for the main tables in the database
11
+ tableNames = {
12
+ main: "tableCfgs",
13
+ revision: "revisions"
14
+ };
15
+ /// Suffix handling for the database
16
+ _suffix = {
17
+ col: "_col",
18
+ tbl: "_tbl",
19
+ tmp: "_tmp"
20
+ };
21
+ // ********************************************************************
22
+ // add and remove suffixes for use in SQL statements
23
+ _addFix(name, fix) {
24
+ return name.endsWith(fix) ? name : name + fix;
25
+ }
26
+ addTableSuffix(name) {
27
+ return this._addFix(name, this._suffix.tbl);
28
+ }
29
+ addColumnSuffix(name) {
30
+ return this._addFix(name, this._suffix.col);
31
+ }
32
+ _removeFix(name, fix) {
33
+ return name.endsWith(fix) ? name.slice(0, -fix.length) : name;
34
+ }
35
+ removeTableSuffix(name) {
36
+ return this._removeFix(name, this._suffix.tbl);
37
+ }
38
+ removeColumnSuffix(name) {
39
+ return this._removeFix(name, this._suffix.col);
40
+ }
41
+ }
5
42
  class IoTools {
6
43
  /**
7
44
  * Constructor
@@ -671,10 +708,11 @@ const _write = (result, childTableName, childRowHash, parentTableName, parentRow
671
708
  referencesForChildTableRow[parentTableName][parentRowHash] ??= {};
672
709
  };
673
710
  export {
711
+ IoDbNameMapping,
674
712
  IoMem,
675
713
  IoTools,
676
714
  calcReverseRefs,
677
715
  exampleIo,
678
716
  exampleTestSetup
679
717
  };
680
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"io.js","sources":["../src/io-tools.ts","../src/io-mem.ts","../src/io-test-setup.ts","../src/io.ts","../src/reverse-ref.ts"],"sourcesContent":["// @license\n// Copyright (c) 2025 Rljson\n//\n// Use of this source code is governed by terms that can be\n// found in the LICENSE file in the root of this package.\n\nimport { hip } from '@rljson/hash';\nimport {\n  iterateTables,\n  Rljson,\n  TableCfg,\n  TableKey,\n  TableType,\n  throwOnInvalidTableCfg,\n  validateRljsonAgainstTableCfg,\n} from '@rljson/rljson';\n\nimport { IoMem } from './io-mem.ts';\nimport { Io } from './io.ts';\n\nexport type IoObserver = (data: Rljson) => void;\n\n/**\n * Provides utility functions for the Io interface.\n */\nexport class IoTools {\n  /**\n   * Constructor\n   * @param io The Io interface to use\n   */\n  constructor(public readonly io: Io) {}\n\n  /**\n   * Returns the table configuration of the tableCfgs table.\n   */\n  static get tableCfgsTableCfg() {\n    const tableCfg = hip<TableCfg>({\n      _hash: '',\n      key: 'tableCfgs',\n      type: 'tableCfgs',\n      isHead: false,\n      isRoot: false,\n      isShared: true,\n      previous: '',\n\n      columns: [\n        {\n          key: '_hash',\n          type: 'string',\n          titleShort: 'Hash',\n          titleLong: 'Row Hash',\n        },\n        {\n          key: 'key',\n          type: 'string',\n          titleShort: 'Key',\n          titleLong: 'Table Key',\n        },\n        {\n          key: 'type',\n          type: 'string',\n          titleShort: 'Type',\n          titleLong: 'Content Type',\n        },\n        {\n          key: 'isHead',\n          type: 'boolean',\n          titleShort: 'Is Head',\n          titleLong: 'Is Head Table',\n        },\n        {\n          key: 'isRoot',\n          type: 'boolean',\n          titleShort: 'Is Root',\n          titleLong: 'Is Root Table',\n        },\n        {\n          key: 'isShared',\n          type: 'boolean',\n          titleShort: 'Is Shared',\n          titleLong: 'Is Shared Table',\n        },\n        {\n          key: 'previous',\n          type: 'string',\n          titleShort: 'Previous',\n          titleLong: 'Previous Table Configuration Hash',\n        },\n        {\n          key: 'columns',\n          type: 'jsonArray',\n          titleShort: 'Columns',\n          titleLong: 'Column Configurations',\n        },\n      ],\n    });\n\n    return tableCfg;\n  }\n\n  /**\n   * Initializes the revisions table.\n   */\n  initRevisionsTable = async () => {\n    const tableCfg: TableCfg = {\n      key: 'revisions',\n      type: 'revisions',\n      isHead: true,\n      isRoot: true,\n      isShared: false,\n\n      columns: [\n        {\n          key: '_hash',\n          type: 'string',\n          titleShort: 'Hash',\n          titleLong: 'Row Hash',\n        },\n        {\n          key: 'table',\n          type: 'string',\n          titleShort: 'Table',\n          titleLong: 'Table Key',\n        },\n        {\n          key: 'predecessor',\n          type: 'string',\n          titleShort: 'Predecessor',\n          titleLong: 'Predecessor Revision Hash',\n        },\n        {\n          key: 'successor',\n          type: 'string',\n          titleShort: 'Successor',\n          titleLong: 'Successor Revision Hash',\n        },\n        {\n          key: 'timestamp',\n          type: 'number',\n          titleShort: 'Timestamp',\n          titleLong: 'Revision Timestamp',\n        },\n        {\n          key: 'id',\n          type: 'string',\n          titleShort: 'ID',\n          titleLong: 'Revision ID',\n        },\n      ],\n    };\n\n    await this.io.createOrExtendTable({ tableCfg });\n  };\n\n  /**\n   * Example object for test purposes\n   * @returns An instance of io tools\n   */\n  static example = async () => {\n    const io = await IoMem.example();\n    await io.init();\n    await io.isReady();\n    return new IoTools(io);\n  };\n\n  /**\n   * Throws if the table does not exist\n   */\n  async throwWhenTableDoesNotExist(table: TableKey): Promise<void> {\n    const exists = await this.io.tableExists(table);\n    if (!exists) {\n      throw new Error(`Table \"${table}\" not found`);\n    }\n  }\n\n  /**\n   * Throws if any of the tables in rljson do not exist\n   * @param rljson - The Rljson object to check\n   */\n  async throwWhenTablesDoNotExist(rljson: Rljson): Promise<void> {\n    try {\n      await iterateTables(rljson, async (tableKey) => {\n        const exists = await this.io.tableExists(tableKey);\n        if (!exists) {\n          throw new Error(`Table \"${tableKey}\" not found`);\n        }\n      });\n    } catch (e) {\n      const missingTables = (e as Array<any>).map((e) => e.tableKey);\n\n      throw new Error(\n        `The following tables do not exist: ${missingTables.join(', ')}`,\n      );\n    }\n  }\n\n  /**\n   * Returns the current table cfgs of all tables\n   * @returns The table configuration of all tables\n   */\n  async tableCfgs(): Promise<TableCfg[]> {\n    const tables = await this.io.rawTableCfgs();\n\n    // Take the latest version of each type key\n    const newestVersion: Record<TableKey, TableCfg> = {};\n    for (let i = tables.length - 1; i >= 0; i--) {\n      const table = tables[i];\n      const existing = newestVersion[table.key];\n      if (!existing || existing.columns.length < table.columns.length) {\n        newestVersion[table.key] = table;\n      }\n    }\n\n    // Sort the tables by key\n    /* v8 ignore next -- @preserve */\n    const resultData = Object.values(newestVersion).sort((a, b) => {\n      if (a.key < b.key) {\n        return -1;\n      }\n      if (a.key > b.key) {\n        return 1;\n      }\n\n      return 0;\n    });\n    return resultData;\n  }\n\n  /**\n   * Returns a list with all table names\n   */\n  async allTableKeys(): Promise<string[]> {\n    const result = (await this.tableCfgs()).map((e) => e.key);\n    return result;\n  }\n\n  /**\n   * Returns the configuration of a given table\n   */\n  async tableCfg(table: TableKey): Promise<TableCfg> {\n    const tableCfg = await this.tableCfgOrNull(table);\n    if (!tableCfg) {\n      throw new Error(`Table \"${table}\" not found`);\n    }\n\n    return tableCfg!;\n  }\n\n  /**\n   * Returns the configuration of a given table or null if it does not exist.\n\n   */\n  async tableCfgOrNull(table: TableKey): Promise<TableCfg | null> {\n    const tableCfgs = await this.tableCfgs();\n    const tableCfg = tableCfgs.find((e) => e.key === table);\n    return tableCfg ?? null;\n  }\n\n  /**\n   * Returns a list of all column names of a given table\n   */\n  async allColumnKeys(table: TableKey): Promise<string[]> {\n    const tableCfg = await this.tableCfg(table);\n    const result = tableCfg.columns.map((column) => column.key);\n    return result;\n  }\n\n  /**\n   * Throws when a column does not exist in a given table\n   * @param table - The table to check\n   * @param columns - The column to check\n   */\n  async throwWhenColumnDoesNotExist(\n    table: TableKey,\n    columns: string[],\n  ): Promise<void> {\n    const tableCfg = await this.tableCfg(table);\n    const columnKeys = tableCfg.columns.map((column) => column.key);\n    const missingColumns = columns.filter(\n      (column) => !columnKeys.includes(column),\n    );\n    if (missingColumns.length > 0) {\n      throw new Error(\n        `The following columns do not exist in table \"${table}\": ${missingColumns.join(\n          ', ',\n        )}.`,\n      );\n    }\n  }\n\n  /**\n   * Throws when a table update is not compatible with the current table\n   * configuration.\n   */\n  async throwWhenTableIsNotCompatible(update: TableCfg): Promise<void> {\n    const prefix = `Invalid update of table able \"${update.key}\"`;\n\n    throwOnInvalidTableCfg(update);\n\n    // Check compatibility with existing table\n    const existing = await this.tableCfgOrNull(update.key);\n    if (existing) {\n      // Have columns been deleted?\n      if (existing.columns.length > update.columns.length) {\n        const deletedColumnKeys = existing.columns\n          .map((column) => column.key)\n          .filter(\n            (key) => !update.columns.some((column) => column.key === key),\n          );\n        /* v8 ignore next -- @preserve */\n        if (deletedColumnKeys.length > 0) {\n          const deletedColumns = deletedColumnKeys.join(', ');\n          throw new Error(\n            `${prefix}: Columns must not be deleted. ` +\n              `Deleted columns: ${deletedColumns}}`,\n          );\n        }\n      }\n\n      // Have column keys changed?\n      for (let i = 0; i < existing.columns.length; i++) {\n        const before = existing.columns[i].key;\n        const after = update.columns[i].key;\n        if (before !== after) {\n          throw new Error(\n            `${prefix}: ` +\n              `Column keys must not change! ` +\n              `Column \"${before}\" was renamed into \"${after}\".`,\n          );\n        }\n      }\n\n      // Have column types changed?\n      for (let i = 0; i < existing.columns.length; i++) {\n        const column = existing.columns[i].key;\n        const before = existing.columns[i].type;\n        const after = update.columns[i].type;\n        if (before !== after) {\n          throw new Error(\n            `${prefix}: ` +\n              `Column types must not change! ` +\n              `Type of column \"${column}\" was changed from \"${before}\" to ${after}.`,\n          );\n        }\n      }\n    }\n  }\n\n  /**\n   * Throws if the data in the table do not match the table configuration\n   */\n  async throwWhenTableDataDoesNotMatchCfg(data: Rljson) {\n    const errors: string[] = [];\n\n    await iterateTables(data, async (tableKey) => {\n      const tableCfg = await this.tableCfg(tableKey);\n      const table = data[tableKey];\n\n      // Ignore tableCfgs table\n      /* v8 ignore next -- @preserve */\n      if (table._type === 'tableCfgs') return;\n\n      errors.push(...validateRljsonAgainstTableCfg(table._data, tableCfg));\n    });\n\n    if (errors.length > 0) {\n      throw new Error(\n        `Table data does not match the configuration.\\n\\nErrors:\\n${errors\n          .map((e) => `- ${e}`)\n          .join('\\n')}`,\n      );\n    }\n  }\n\n  /**\n   * Sorts the data of a table by the hash and updates the table hash in place\n   */\n  sortTableDataAndUpdateHash(table: TableType): void {\n    table._data.sort((a, b) => {\n      const hashA = a._hash as string;\n      const hashB = b._hash as string;\n      /* v8 ignore next -- @preserve */\n      if (hashA < hashB) {\n        return -1;\n      }\n      /* v8 ignore next -- @preserve */\n      if (hashA > hashB) {\n        return 1;\n      }\n\n      /* v8 ignore next -- @preserve */\n      return 0;\n    });\n\n    table._hash = '';\n    hip(table, {\n      updateExistingHashes: false,\n      throwOnWrongHashes: false,\n    });\n  }\n}\n","// @license\n// Copyright (c) 2025 Rljson\n//\n// Use of this source code is governed by terms that can be\n// found in the LICENSE file in the root of this package.\n\nimport { hip, hsh } from '@rljson/hash';\nimport { IsReady } from '@rljson/is-ready';\nimport { copy, equals, JsonValue } from '@rljson/json';\nimport {\n  ContentType,\n  iterateTablesSync,\n  Rljson,\n  TableCfg,\n  TableKey,\n  TableType,\n} from '@rljson/rljson';\n\nimport { IoTools } from './io-tools.ts';\nimport { Io } from './io.ts';\n\n/**\n * In-Memory implementation of the Rljson Io interface.\n */\nexport class IoMem implements Io {\n  // ...........................................................................\n  // Constructor & example\n\n  init(): Promise<void> {\n    this._isOpen = true;\n    return this._init();\n  }\n\n  close(): Promise<void> {\n    this._isOpen = false;\n    return Promise.resolve();\n  }\n\n  get isOpen(): boolean {\n    return this._isOpen;\n  }\n\n  static example = async () => {\n    const io = new IoMem();\n    await io.init();\n    return io;\n  };\n\n  // ...........................................................................\n  // General\n  isReady() {\n    return this._isReady.promise;\n  }\n\n  // ...........................................................................\n  // Dump\n\n  dump(): Promise<Rljson> {\n    return this._dump();\n  }\n\n  async dumpTable(request: { table: string }): Promise<Rljson> {\n    return this._dumpTable(request);\n  }\n\n  // ...........................................................................\n  // Meta Data\n\n  async contentType(request: { table: string }): Promise<ContentType> {\n    return this._contentType(request);\n  }\n\n  // ...........................................................................\n  // Rows\n\n  readRows(request: {\n    table: string;\n    where: { [column: string]: JsonValue };\n  }): Promise<Rljson> {\n    return this._readRows(request);\n  }\n\n  async rowCount(table: string): Promise<number> {\n    const tableData = this._mem[table] as TableType;\n    if (!tableData) {\n      throw new Error(`Table \"${table}\" not found`);\n    }\n    return Promise.resolve(tableData._data.length);\n  }\n\n  // ...........................................................................\n  // Write\n\n  write(request: { data: Rljson }): Promise<void> {\n    return this._write(request);\n  }\n\n  // ...........................................................................\n  // Table management\n  async tableExists(tableKey: TableKey): Promise<boolean> {\n    const table = this._mem[tableKey] as TableType;\n    return table ? true : false;\n  }\n\n  createOrExtendTable(request: { tableCfg: TableCfg }): Promise<void> {\n    return this._createOrExtendTable(request);\n  }\n\n  async rawTableCfgs(): Promise<TableCfg[]> {\n    const tables = this._mem.tableCfgs._data as TableCfg[];\n    return tables;\n  }\n\n  // ######################\n  // Private\n  // ######################\n\n  private _ioTools!: IoTools;\n\n  private _isReady = new IsReady();\n  private _isOpen = false;\n\n  private _mem: Rljson = hip({} as Rljson);\n\n  // ...........................................................................\n  private async _init() {\n    this._ioTools = new IoTools(this);\n    this._initTableCfgs();\n    this._updateGlobalHash();\n    await this._ioTools.initRevisionsTable();\n    hsh(this._mem);\n\n    this._isReady.resolve();\n  }\n\n  // ...........................................................................\n  private _initTableCfgs = () => {\n    const tableCfg = IoTools.tableCfgsTableCfg;\n\n    this._mem.tableCfgs = hip({\n      _type: 'tableCfgs',\n      _data: [tableCfg],\n      _tableCfg: tableCfg._hash as string,\n    });\n  };\n\n  // ...........................................................................\n  private _updateGlobalHash() {\n    (this._mem as any)._hash = '';\n    hip(this._mem, {\n      updateExistingHashes: false,\n    });\n  }\n\n  // ...........................................................................\n  private _updateTableHash(tableKey: TableKey) {\n    const table = this._mem[tableKey] as TableType;\n    table._hash = '';\n    hip(table, { updateExistingHashes: false });\n  }\n\n  // ...........................................................................\n  private async _createOrExtendTable(request: {\n    tableCfg: TableCfg;\n  }): Promise<void> {\n    // Make sure that the table config is compatible\n    // with an potential existing table\n    const tableCfg = request.tableCfg;\n    await this._ioTools.throwWhenTableIsNotCompatible(tableCfg);\n\n    const { key } = tableCfg;\n\n    // Recreate hashes in the case the existing hashes are wrong\n    const newConfig = hsh(tableCfg);\n\n    // Find an existing table config with the same hash\n    const existingConfig = await this._ioTools.tableCfgOrNull(key);\n\n    // Write the new config into the database\n    if (!existingConfig) {\n      this._createTable(newConfig, key);\n    } else {\n      this._extendTable(existingConfig, newConfig);\n    }\n  }\n\n  // ...........................................................................\n  private _createTable(newConfig: TableCfg, tableKey: TableKey) {\n    // Write the table config into the database\n    newConfig = hsh(newConfig);\n    this._mem.tableCfgs._data.push(newConfig);\n    this._ioTools.sortTableDataAndUpdateHash(this._mem.tableCfgs);\n\n    // Create a table and write it into the database\n    const table: TableType = {\n      _type: newConfig.type,\n      _data: [],\n      _tableCfg: newConfig._hash as string,\n    };\n\n    this._mem[tableKey] ??= hip(table);\n\n    // Update hashes\n    this._updateTableHash(tableKey);\n    this._updateGlobalHash();\n  }\n\n  // ...........................................................................\n  private _extendTable(existingConfig: TableCfg, newConfig: TableCfg) {\n    // No columns added? Return.\n    if (existingConfig.columns.length === newConfig.columns.length) {\n      return;\n    }\n\n    // Write the new table config into the database\n    newConfig = hsh(newConfig);\n    this._mem.tableCfgs._data.push(newConfig);\n    this._ioTools.sortTableDataAndUpdateHash(this._mem.tableCfgs);\n\n    // Update the config of the existing table\n    const table = this._mem[newConfig.key] as TableType;\n    table._tableCfg = newConfig._hash as string;\n\n    // Update the hashes\n    this._updateTableHash(newConfig.key);\n    this._updateGlobalHash();\n  }\n\n  // ...........................................................................\n\n  private async _dump(): Promise<Rljson> {\n    return copy(this._mem);\n  }\n\n  // ...........................................................................\n  private async _dumpTable(request: { table: string }): Promise<Rljson> {\n    await this._ioTools.throwWhenTableDoesNotExist(request.table);\n\n    const table = this._mem[request.table] as TableType;\n\n    return {\n      [request.table]: copy(table),\n    };\n  }\n\n  // ...........................................................................\n  private async _contentType(request: { table: string }): Promise<ContentType> {\n    await this._ioTools.throwWhenTableDoesNotExist(request.table);\n\n    return (this._mem[request.table] as TableType)._type;\n  }\n\n  // ...........................................................................\n  private async _write(request: { data: Rljson }): Promise<void> {\n    const addedData = hsh(request.data);\n    this._removeNullValues(addedData);\n    const tables = Object.keys(addedData);\n    hsh(addedData);\n\n    await this._ioTools.throwWhenTablesDoNotExist(request.data);\n    await this._ioTools.throwWhenTableDataDoesNotMatchCfg(request.data);\n\n    for (const table of tables) {\n      if (table.startsWith('_')) {\n        continue;\n      }\n\n      const oldTable = this._mem[table] as TableType;\n      const newTable = addedData[table] as TableType;\n\n      // Table exists. Merge data\n      for (const item of newTable._data) {\n        const hash = item._hash;\n        const exists = oldTable._data.find((i) => i._hash === hash);\n        if (!exists) {\n          oldTable._data.push(item as any);\n        }\n      }\n\n      this._ioTools.sortTableDataAndUpdateHash(oldTable);\n    }\n\n    // Recalc main hashes\n    this._updateGlobalHash();\n  }\n\n  // ...........................................................................\n  private async _readRows(request: {\n    table: string;\n    where: { [column: string]: JsonValue };\n  }): Promise<Rljson> {\n    await this._ioTools.throwWhenTableDoesNotExist(request.table);\n    await this._ioTools.throwWhenColumnDoesNotExist(\n      request.table,\n      Object.keys(request.where),\n    );\n\n    // Read table from data\n    const table = this._mem[request.table] as TableType;\n\n    // Filter table data\n    const tableDataFiltered = table._data.filter((row) => {\n      for (const column in request.where) {\n        const a = row[column];\n        const b = request.where[column];\n        if (b === null && a === undefined) {\n          return true;\n        }\n\n        if (!equals(a, b)) {\n          return false;\n        }\n      }\n      return true;\n    });\n\n    // Create an table\n    const tableFiltered: TableType = {\n      _type: table._type,\n      _data: tableDataFiltered,\n    };\n\n    this._ioTools.sortTableDataAndUpdateHash(tableFiltered);\n\n    const result: Rljson = {\n      [request.table]: tableFiltered,\n    };\n\n    return result;\n  }\n\n  _removeNullValues(rljson: Rljson) {\n    iterateTablesSync(rljson, (table) => {\n      const data = rljson[table]._data;\n\n      for (const row of data) {\n        for (const key in row) {\n          if (row[key] === null) {\n            delete row[key];\n          }\n        }\n      }\n    });\n  }\n}\n","// @license\n// Copyright (c) 2025 Rljson\n//\n// Use of this source code is governed by terms that can be\n// found in the LICENSE file in the root of this package.\n\nimport { Io, IoMem } from './index.ts';\n\n// .............................................................................\n/**\n * Io implementation need to implement this interface to be used in\n * conformance tests.\n */\nexport interface IoTestSetup {\n  /** setup before the single setups */\n  beforeAll: () => Promise<void>;\n\n  /**\n   * Initializes the io implementation.\n   * @returns The io implementation.\n   */\n  beforeEach: () => Promise<void>;\n\n  /**\n   * Tears down the io implementation.\n   * @returns The io implementation.\n   */\n  afterEach: () => Promise<void>;\n\n  /** cleanup after all tests */\n  afterAll: () => Promise<void>;\n\n  /**\n   * The io implementation to be used in the conformance tests.\n   */\n  io: Io;\n}\n\n// .............................................................................\n// Example implementation of the IoTestSetup interface\nexport const exampleTestSetup = (): IoTestSetup => {\n  return {\n    io: new IoMem(),\n    beforeAll: async () => {\n      // This method can be used for any additional setup required before init.\n      // Currently, it does nothing.\n    },\n\n    beforeEach: async () => {\n      // Initialize the io implementation\n    },\n    afterEach: async () => {\n      // Tear down the io implementation\n    },\n\n    afterAll: async () => {\n      // This method can be used for any additional cleanup after tearDown.\n    },\n  };\n};\n","// @license\n// Copyright (c) 2025 Rljson\n//\n// Use of this source code is governed by terms that can be\n// found in the LICENSE file in the root of this package.\n\nimport { JsonValue } from '@rljson/json';\nimport { ContentType, Rljson, TableCfg, TableKey } from '@rljson/rljson';\n\n\n// .............................................................................\nexport interface Io {\n  // ...........................................................................\n  // General\n\n  /** Starts the initialization */\n  init(): Promise<void>;\n\n  /** Closes the io */\n  close(): Promise<void>;\n\n  /** Returns true if io is opened */\n  isOpen: boolean;\n\n  /** A promise resolving once the Io interface is ready\n   *\n   * 💡 Use @rljson/is-ready\n   */\n  isReady(): Promise<void>;\n\n  // ...........................................................................\n  // Dump\n\n  /** Returns the complete db content as Rljson */\n  dump(): Promise<Rljson>;\n\n  /** Returns the dump of a complete table */\n  dumpTable(request: { table: string }): Promise<Rljson>;\n\n  // ...........................................................................\n  // Meta Data\n  contentType(request: { table: string }): Promise<ContentType>;\n\n  // ...........................................................................\n  // Tables\n\n  /**\n   * Returns true if the table exists\n   */\n  tableExists(tableKey: TableKey): Promise<boolean>;\n\n  /**\n   * Creates a table with a given config.\n   * If the table already exists, new columns are added to the existing table.\n   * If the table does not exist, it is created with the given config.\n   * If the table exists and columns are removed, an error is thrown.\n   * If the table exists and the column type is changed, an error is thrown.\n   */\n  createOrExtendTable(request: { tableCfg: TableCfg }): Promise<void>;\n\n  /**\n   * Returns a json structure returning current table configurations\n   */\n  rawTableCfgs(): Promise<TableCfg[]>;\n\n  // ...........................................................................\n  // Write\n\n  /** Writes Rljson data into the database */\n  write(request: { data: Rljson }): Promise<void>;\n\n  // ...........................................................................\n  // Read rows\n\n  /** Queries a list of rows */\n  readRows(request: {\n    table: string;\n    where: { [column: string]: JsonValue | null };\n  }): Promise<Rljson>;\n\n  /** Returns the number of rows in the given table */\n  rowCount(table: string): Promise<number>;\n}\n\n// .............................................................................\nexport const exampleIo =\n  'Checkout @rljson/io-mem for an example implementation';\n","/* v8 ignore file -- @preserve */\n// @license\n// Copyright (c) 2025 Rljson\n//\n// Use of this source code is governed by terms that can be\n// found in the LICENSE file in the root of this package.\n\nimport { Json } from '@rljson/json';\nimport {\n  Buffet,\n  Cake,\n  iterateTablesSync,\n  Layer,\n  Ref,\n  Rljson,\n  TableKey,\n} from '@rljson/rljson';\n\n// .............................................................................\n/**\n * Describes a row that references a child table row\n */\nexport interface ParentRef {\n  /**\n   * The parent table that references the child table\n   */\n  [parentTable: TableKey]: {\n    /**\n     * The parent row that references the child row\n     */\n    [parentRow: Ref]: {\n      /**\n       * Details about the reference, e.g. an array index etc.\n       */\n      details?: Json;\n    };\n  };\n}\n\n// .............................................................................\n/**\n * Describes the parent table rows referencing a child table row\n */\nexport interface ReverseRefs {\n  /**\n   * The child table we need the referencing rows for\n   */\n  [childTable: TableKey]: {\n    /**\n     * The row hashwe need the referencing rows for\n     */\n    [childRow: Ref]: ParentRef;\n  };\n}\n\n/* v8 ignore start */\n\n// .............................................................................\n/**\n * Calculates the reverse references for a given rljson object\n */\nexport const calcReverseRefs = (rljson: Rljson): ReverseRefs => {\n  const result: ReverseRefs = {};\n\n  // ......................\n  // Prepare data structure\n  iterateTablesSync(rljson, (childTableKey, table) => {\n    const childTable: { [childRowHash: string]: ParentRef } = {};\n    result[childTableKey] = childTable;\n    for (const childRow of table._data) {\n      childTable[childRow._hash] = {};\n    }\n  });\n\n  // ............................\n  // Generate reverse references\n  iterateTablesSync(rljson, (parentTableKey, parentTable) => {\n    // Iterate all rows of each table\n    for (const parentTableRow of parentTable._data) {\n      // Find out whe other tables & rows are referenced by this row\n      // Write these information intto result\n      switch (parentTable._type) {\n        case 'components':\n          _writeComponentRefs(parentTableKey, parentTableRow, result);\n          break;\n\n        case 'layers': {\n          _writeLayerRefs(parentTableKey, parentTableRow, result);\n          break;\n        }\n\n        case 'sliceIds': {\n          // Slice ids do not reference other tables\n          break;\n        }\n\n        case 'cakes': {\n          _writeCakeRefs(parentTableKey, parentTableRow, result);\n          break;\n        }\n\n        case 'buffets': {\n          _writeBuffetRefs(parentTableKey, parentTableRow, result);\n          break;\n        }\n      }\n    }\n  });\n\n  return result;\n};\n\n/* v8 ignore end */\n\n// .............................................................................\nconst _writeComponentRefs = (\n  parentTableName: TableKey,\n  parentRow: Json,\n  result: ReverseRefs,\n) => {\n  const parentRowHash = parentRow._hash as string;\n\n  for (const parentColumnName in parentRow) {\n    if (parentColumnName.startsWith('_')) {\n      continue;\n    }\n\n    if (!parentColumnName.endsWith('Ref')) {\n      continue;\n    }\n\n    const childTableName = parentColumnName.slice(0, -3);\n    const childRowHash = parentRow[parentColumnName] as string;\n\n    _write(\n      result,\n      childTableName,\n      childRowHash,\n      parentTableName,\n      parentRowHash,\n    );\n  }\n};\n\n// .............................................................................\nconst _writeLayerRefs = (\n  parentTableName: TableKey,\n  parentRow: Layer,\n  result: ReverseRefs,\n) => {\n  const childTableName = parentRow.componentsTable;\n  const parentRowHash = parentRow._hash as string;\n\n  for (const sliceId in parentRow.add) {\n    if (sliceId.startsWith('_')) {\n      continue;\n    }\n\n    const sliceHash = parentRow.add[sliceId] as string;\n\n    _write(result, childTableName, sliceHash, parentTableName, parentRowHash);\n  }\n};\n\n// .............................................................................\nconst _writeCakeRefs = (\n  parentTableName: TableKey,\n  parentRow: Cake,\n  result: ReverseRefs,\n) => {\n  const parentRowHash = parentRow._hash as string;\n\n  for (const layer in parentRow.layers) {\n    const childTableName = layer;\n    const childRowHash = parentRow.layers[layer] as string;\n    _write(\n      result,\n      childTableName,\n      childRowHash,\n      parentTableName,\n      parentRowHash,\n    );\n  }\n};\n\n// .............................................................................\nconst _writeBuffetRefs = (\n  parentTableName: TableKey,\n  parentRow: Buffet,\n  result: ReverseRefs,\n) => {\n  const parentRowHash = parentRow._hash as string;\n\n  for (const item of parentRow.items) {\n    const childTableName = item.table;\n    const childRowHash = item.ref;\n    _write(\n      result,\n      childTableName,\n      childRowHash,\n      parentTableName,\n      parentRowHash,\n    );\n  }\n};\n\n// .............................................................................\nconst _write = (\n  result: ReverseRefs,\n  childTableName: string,\n  childRowHash: string,\n  parentTableName: string,\n  parentRowHash: string,\n) => {\n  const referencesForChildTable = (result[childTableName] ??= {});\n  const referencesForChildTableRow = (referencesForChildTable[childRowHash] ??=\n    {});\n\n  referencesForChildTableRow[parentTableName] ??= {};\n  referencesForChildTableRow[parentTableName][parentRowHash] ??= {};\n};\n"],"names":["e"],"mappings":"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"}
718
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"io.js","sources":["../src/io-db-name-mapping.ts","../src/io-tools.ts","../src/io-mem.ts","../src/io-test-setup.ts","../src/io.ts","../src/reverse-ref.ts"],"sourcesContent":["// @license\r\n// Copyright (c) 2025 Rljson\r\n//\r\n// Use of this source code is governed by terms that can be\r\n// found in the LICENSE file in the root of this package.\r\n\r\nexport class IoDbNameMapping {\r\n  // The primary key column is always named '_hash'\r\n  public primaryKeyColumn: string = '_hash';\r\n  public dataSection: string = '_data';\r\n  public typeColumn: string = 'type';\r\n\r\n  // Names for the main tables in the database\r\n  public tableNames: { [key: string]: string } = {\r\n    main: 'tableCfgs',\r\n    revision: 'revisions',\r\n  };\r\n\r\n  /// Suffix handling for the database\r\n  private _suffix: { [key: string]: string } = {\r\n    col: '_col',\r\n    tbl: '_tbl',\r\n    tmp: '_tmp',\r\n  };\r\n\r\n  // ********************************************************************\r\n  // add and remove suffixes for use in SQL statements\r\n  private _addFix(name: string, fix: string): string {\r\n    return name.endsWith(fix) ? name : name + fix;\r\n  }\r\n\r\n  public addTableSuffix(name: string): string {\r\n    return this._addFix(name, this._suffix.tbl);\r\n  }\r\n\r\n  public addColumnSuffix(name: string): string {\r\n    return this._addFix(name, this._suffix.col);\r\n  }\r\n\r\n  private _removeFix(name: string, fix: string): string {\r\n    return name.endsWith(fix) ? name.slice(0, -fix.length) : name;\r\n  }\r\n\r\n  public removeTableSuffix(name: string): string {\r\n    return this._removeFix(name, this._suffix.tbl);\r\n  }\r\n\r\n  public removeColumnSuffix(name: string): string {\r\n    return this._removeFix(name, this._suffix.col);\r\n  }\r\n}\r\n","// @license\r\n// Copyright (c) 2025 Rljson\r\n//\r\n// Use of this source code is governed by terms that can be\r\n// found in the LICENSE file in the root of this package.\r\n\r\nimport { hip } from '@rljson/hash';\r\nimport {\r\n  iterateTables,\r\n  Rljson,\r\n  TableCfg,\r\n  TableKey,\r\n  TableType,\r\n  throwOnInvalidTableCfg,\r\n  validateRljsonAgainstTableCfg,\r\n} from '@rljson/rljson';\r\n\r\nimport { IoMem } from './io-mem.ts';\r\nimport { Io } from './io.ts';\r\n\r\nexport type IoObserver = (data: Rljson) => void;\r\n\r\n/**\r\n * Provides utility functions for the Io interface.\r\n */\r\nexport class IoTools {\r\n  /**\r\n   * Constructor\r\n   * @param io The Io interface to use\r\n   */\r\n  constructor(public readonly io: Io) {}\r\n\r\n  /**\r\n   * Returns the table configuration of the tableCfgs table.\r\n   */\r\n  static get tableCfgsTableCfg() {\r\n    const tableCfg = hip<TableCfg>({\r\n      _hash: '',\r\n      key: 'tableCfgs',\r\n      type: 'tableCfgs',\r\n      isHead: false,\r\n      isRoot: false,\r\n      isShared: true,\r\n      previous: '',\r\n\r\n      columns: [\r\n        {\r\n          key: '_hash',\r\n          type: 'string',\r\n          titleShort: 'Hash',\r\n          titleLong: 'Row Hash',\r\n        },\r\n        {\r\n          key: 'key',\r\n          type: 'string',\r\n          titleShort: 'Key',\r\n          titleLong: 'Table Key',\r\n        },\r\n        {\r\n          key: 'type',\r\n          type: 'string',\r\n          titleShort: 'Type',\r\n          titleLong: 'Content Type',\r\n        },\r\n        {\r\n          key: 'isHead',\r\n          type: 'boolean',\r\n          titleShort: 'Is Head',\r\n          titleLong: 'Is Head Table',\r\n        },\r\n        {\r\n          key: 'isRoot',\r\n          type: 'boolean',\r\n          titleShort: 'Is Root',\r\n          titleLong: 'Is Root Table',\r\n        },\r\n        {\r\n          key: 'isShared',\r\n          type: 'boolean',\r\n          titleShort: 'Is Shared',\r\n          titleLong: 'Is Shared Table',\r\n        },\r\n        {\r\n          key: 'previous',\r\n          type: 'string',\r\n          titleShort: 'Previous',\r\n          titleLong: 'Previous Table Configuration Hash',\r\n        },\r\n        {\r\n          key: 'columns',\r\n          type: 'jsonArray',\r\n          titleShort: 'Columns',\r\n          titleLong: 'Column Configurations',\r\n        },\r\n      ],\r\n    });\r\n\r\n    return tableCfg;\r\n  }\r\n\r\n  /**\r\n   * Initializes the revisions table.\r\n   */\r\n  initRevisionsTable = async () => {\r\n    const tableCfg: TableCfg = {\r\n      key: 'revisions',\r\n      type: 'revisions',\r\n      isHead: true,\r\n      isRoot: true,\r\n      isShared: false,\r\n\r\n      columns: [\r\n        {\r\n          key: '_hash',\r\n          type: 'string',\r\n          titleShort: 'Hash',\r\n          titleLong: 'Row Hash',\r\n        },\r\n        {\r\n          key: 'table',\r\n          type: 'string',\r\n          titleShort: 'Table',\r\n          titleLong: 'Table Key',\r\n        },\r\n        {\r\n          key: 'predecessor',\r\n          type: 'string',\r\n          titleShort: 'Predecessor',\r\n          titleLong: 'Predecessor Revision Hash',\r\n        },\r\n        {\r\n          key: 'successor',\r\n          type: 'string',\r\n          titleShort: 'Successor',\r\n          titleLong: 'Successor Revision Hash',\r\n        },\r\n        {\r\n          key: 'timestamp',\r\n          type: 'number',\r\n          titleShort: 'Timestamp',\r\n          titleLong: 'Revision Timestamp',\r\n        },\r\n        {\r\n          key: 'id',\r\n          type: 'string',\r\n          titleShort: 'ID',\r\n          titleLong: 'Revision ID',\r\n        },\r\n      ],\r\n    };\r\n\r\n    await this.io.createOrExtendTable({ tableCfg });\r\n  };\r\n\r\n  /**\r\n   * Example object for test purposes\r\n   * @returns An instance of io tools\r\n   */\r\n  static example = async () => {\r\n    const io = await IoMem.example();\r\n    await io.init();\r\n    await io.isReady();\r\n    return new IoTools(io);\r\n  };\r\n\r\n  /**\r\n   * Throws if the table does not exist\r\n   */\r\n  async throwWhenTableDoesNotExist(table: TableKey): Promise<void> {\r\n    const exists = await this.io.tableExists(table);\r\n    if (!exists) {\r\n      throw new Error(`Table \"${table}\" not found`);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Throws if any of the tables in rljson do not exist\r\n   * @param rljson - The Rljson object to check\r\n   */\r\n  async throwWhenTablesDoNotExist(rljson: Rljson): Promise<void> {\r\n    try {\r\n      await iterateTables(rljson, async (tableKey) => {\r\n        const exists = await this.io.tableExists(tableKey);\r\n        if (!exists) {\r\n          throw new Error(`Table \"${tableKey}\" not found`);\r\n        }\r\n      });\r\n    } catch (e) {\r\n      const missingTables = (e as Array<any>).map((e) => e.tableKey);\r\n\r\n      throw new Error(\r\n        `The following tables do not exist: ${missingTables.join(', ')}`,\r\n      );\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Returns the current table cfgs of all tables\r\n   * @returns The table configuration of all tables\r\n   */\r\n  async tableCfgs(): Promise<TableCfg[]> {\r\n    const tables = await this.io.rawTableCfgs();\r\n\r\n    // Take the latest version of each type key\r\n    const newestVersion: Record<TableKey, TableCfg> = {};\r\n    for (let i = tables.length - 1; i >= 0; i--) {\r\n      const table = tables[i];\r\n      const existing = newestVersion[table.key];\r\n      if (!existing || existing.columns.length < table.columns.length) {\r\n        newestVersion[table.key] = table;\r\n      }\r\n    }\r\n\r\n    // Sort the tables by key\r\n    /* v8 ignore next -- @preserve */\r\n    const resultData = Object.values(newestVersion).sort((a, b) => {\r\n      if (a.key < b.key) {\r\n        return -1;\r\n      }\r\n      if (a.key > b.key) {\r\n        return 1;\r\n      }\r\n\r\n      return 0;\r\n    });\r\n    return resultData;\r\n  }\r\n\r\n  /**\r\n   * Returns a list with all table names\r\n   */\r\n  async allTableKeys(): Promise<string[]> {\r\n    const result = (await this.tableCfgs()).map((e) => e.key);\r\n    return result;\r\n  }\r\n\r\n  /**\r\n   * Returns the configuration of a given table\r\n   */\r\n  async tableCfg(table: TableKey): Promise<TableCfg> {\r\n    const tableCfg = await this.tableCfgOrNull(table);\r\n    if (!tableCfg) {\r\n      throw new Error(`Table \"${table}\" not found`);\r\n    }\r\n\r\n    return tableCfg!;\r\n  }\r\n\r\n  /**\r\n   * Returns the configuration of a given table or null if it does not exist.\r\n\r\n   */\r\n  async tableCfgOrNull(table: TableKey): Promise<TableCfg | null> {\r\n    const tableCfgs = await this.tableCfgs();\r\n    const tableCfg = tableCfgs.find((e) => e.key === table);\r\n    return tableCfg ?? null;\r\n  }\r\n\r\n  /**\r\n   * Returns a list of all column names of a given table\r\n   */\r\n  async allColumnKeys(table: TableKey): Promise<string[]> {\r\n    const tableCfg = await this.tableCfg(table);\r\n    const result = tableCfg.columns.map((column) => column.key);\r\n    return result;\r\n  }\r\n\r\n  /**\r\n   * Throws when a column does not exist in a given table\r\n   * @param table - The table to check\r\n   * @param columns - The column to check\r\n   */\r\n  async throwWhenColumnDoesNotExist(\r\n    table: TableKey,\r\n    columns: string[],\r\n  ): Promise<void> {\r\n    const tableCfg = await this.tableCfg(table);\r\n    const columnKeys = tableCfg.columns.map((column) => column.key);\r\n    const missingColumns = columns.filter(\r\n      (column) => !columnKeys.includes(column),\r\n    );\r\n    if (missingColumns.length > 0) {\r\n      throw new Error(\r\n        `The following columns do not exist in table \"${table}\": ${missingColumns.join(\r\n          ', ',\r\n        )}.`,\r\n      );\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Throws when a table update is not compatible with the current table\r\n   * configuration.\r\n   */\r\n  async throwWhenTableIsNotCompatible(update: TableCfg): Promise<void> {\r\n    const prefix = `Invalid update of table able \"${update.key}\"`;\r\n\r\n    throwOnInvalidTableCfg(update);\r\n\r\n    // Check compatibility with existing table\r\n    const existing = await this.tableCfgOrNull(update.key);\r\n    if (existing) {\r\n      // Have columns been deleted?\r\n      if (existing.columns.length > update.columns.length) {\r\n        const deletedColumnKeys = existing.columns\r\n          .map((column) => column.key)\r\n          .filter(\r\n            (key) => !update.columns.some((column) => column.key === key),\r\n          );\r\n        /* v8 ignore next -- @preserve */\r\n        if (deletedColumnKeys.length > 0) {\r\n          const deletedColumns = deletedColumnKeys.join(', ');\r\n          throw new Error(\r\n            `${prefix}: Columns must not be deleted. ` +\r\n              `Deleted columns: ${deletedColumns}}`,\r\n          );\r\n        }\r\n      }\r\n\r\n      // Have column keys changed?\r\n      for (let i = 0; i < existing.columns.length; i++) {\r\n        const before = existing.columns[i].key;\r\n        const after = update.columns[i].key;\r\n        if (before !== after) {\r\n          throw new Error(\r\n            `${prefix}: ` +\r\n              `Column keys must not change! ` +\r\n              `Column \"${before}\" was renamed into \"${after}\".`,\r\n          );\r\n        }\r\n      }\r\n\r\n      // Have column types changed?\r\n      for (let i = 0; i < existing.columns.length; i++) {\r\n        const column = existing.columns[i].key;\r\n        const before = existing.columns[i].type;\r\n        const after = update.columns[i].type;\r\n        if (before !== after) {\r\n          throw new Error(\r\n            `${prefix}: ` +\r\n              `Column types must not change! ` +\r\n              `Type of column \"${column}\" was changed from \"${before}\" to ${after}.`,\r\n          );\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Throws if the data in the table do not match the table configuration\r\n   */\r\n  async throwWhenTableDataDoesNotMatchCfg(data: Rljson) {\r\n    const errors: string[] = [];\r\n\r\n    await iterateTables(data, async (tableKey) => {\r\n      const tableCfg = await this.tableCfg(tableKey);\r\n      const table = data[tableKey];\r\n\r\n      // Ignore tableCfgs table\r\n      /* v8 ignore next -- @preserve */\r\n      if (table._type === 'tableCfgs') return;\r\n\r\n      errors.push(...validateRljsonAgainstTableCfg(table._data, tableCfg));\r\n    });\r\n\r\n    if (errors.length > 0) {\r\n      throw new Error(\r\n        `Table data does not match the configuration.\\n\\nErrors:\\n${errors\r\n          .map((e) => `- ${e}`)\r\n          .join('\\n')}`,\r\n      );\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Sorts the data of a table by the hash and updates the table hash in place\r\n   */\r\n  sortTableDataAndUpdateHash(table: TableType): void {\r\n    table._data.sort((a, b) => {\r\n      const hashA = a._hash as string;\r\n      const hashB = b._hash as string;\r\n      /* v8 ignore next -- @preserve */\r\n      if (hashA < hashB) {\r\n        return -1;\r\n      }\r\n      /* v8 ignore next -- @preserve */\r\n      if (hashA > hashB) {\r\n        return 1;\r\n      }\r\n\r\n      /* v8 ignore next -- @preserve */\r\n      return 0;\r\n    });\r\n\r\n    table._hash = '';\r\n    hip(table, {\r\n      updateExistingHashes: false,\r\n      throwOnWrongHashes: false,\r\n    });\r\n  }\r\n}\r\n","// @license\r\n// Copyright (c) 2025 Rljson\r\n//\r\n// Use of this source code is governed by terms that can be\r\n// found in the LICENSE file in the root of this package.\r\n\r\nimport { hip, hsh } from '@rljson/hash';\r\nimport { IsReady } from '@rljson/is-ready';\r\nimport { copy, equals, JsonValue } from '@rljson/json';\r\nimport {\r\n  ContentType,\r\n  iterateTablesSync,\r\n  Rljson,\r\n  TableCfg,\r\n  TableKey,\r\n  TableType,\r\n} from '@rljson/rljson';\r\n\r\nimport { IoTools } from './io-tools.ts';\r\nimport { Io } from './io.ts';\r\n\r\n/**\r\n * In-Memory implementation of the Rljson Io interface.\r\n */\r\nexport class IoMem implements Io {\r\n  // ...........................................................................\r\n  // Constructor & example\r\n\r\n  init(): Promise<void> {\r\n    this._isOpen = true;\r\n    return this._init();\r\n  }\r\n\r\n  close(): Promise<void> {\r\n    this._isOpen = false;\r\n    return Promise.resolve();\r\n  }\r\n\r\n  get isOpen(): boolean {\r\n    return this._isOpen;\r\n  }\r\n\r\n  static example = async () => {\r\n    const io = new IoMem();\r\n    await io.init();\r\n    return io;\r\n  };\r\n\r\n  // ...........................................................................\r\n  // General\r\n  isReady() {\r\n    return this._isReady.promise;\r\n  }\r\n\r\n  // ...........................................................................\r\n  // Dump\r\n\r\n  dump(): Promise<Rljson> {\r\n    return this._dump();\r\n  }\r\n\r\n  async dumpTable(request: { table: string }): Promise<Rljson> {\r\n    return this._dumpTable(request);\r\n  }\r\n\r\n  // ...........................................................................\r\n  // Meta Data\r\n\r\n  async contentType(request: { table: string }): Promise<ContentType> {\r\n    return this._contentType(request);\r\n  }\r\n\r\n  // ...........................................................................\r\n  // Rows\r\n\r\n  readRows(request: {\r\n    table: string;\r\n    where: { [column: string]: JsonValue };\r\n  }): Promise<Rljson> {\r\n    return this._readRows(request);\r\n  }\r\n\r\n  async rowCount(table: string): Promise<number> {\r\n    const tableData = this._mem[table] as TableType;\r\n    if (!tableData) {\r\n      throw new Error(`Table \"${table}\" not found`);\r\n    }\r\n    return Promise.resolve(tableData._data.length);\r\n  }\r\n\r\n  // ...........................................................................\r\n  // Write\r\n\r\n  write(request: { data: Rljson }): Promise<void> {\r\n    return this._write(request);\r\n  }\r\n\r\n  // ...........................................................................\r\n  // Table management\r\n  async tableExists(tableKey: TableKey): Promise<boolean> {\r\n    const table = this._mem[tableKey] as TableType;\r\n    return table ? true : false;\r\n  }\r\n\r\n  createOrExtendTable(request: { tableCfg: TableCfg }): Promise<void> {\r\n    return this._createOrExtendTable(request);\r\n  }\r\n\r\n  async rawTableCfgs(): Promise<TableCfg[]> {\r\n    const tables = this._mem.tableCfgs._data as TableCfg[];\r\n    return tables;\r\n  }\r\n\r\n  // ######################\r\n  // Private\r\n  // ######################\r\n\r\n  private _ioTools!: IoTools;\r\n\r\n  private _isReady = new IsReady();\r\n  private _isOpen = false;\r\n\r\n  private _mem: Rljson = hip({} as Rljson);\r\n\r\n  // ...........................................................................\r\n  private async _init() {\r\n    this._ioTools = new IoTools(this);\r\n    this._initTableCfgs();\r\n    this._updateGlobalHash();\r\n    await this._ioTools.initRevisionsTable();\r\n    hsh(this._mem);\r\n\r\n    this._isReady.resolve();\r\n  }\r\n\r\n  // ...........................................................................\r\n  private _initTableCfgs = () => {\r\n    const tableCfg = IoTools.tableCfgsTableCfg;\r\n\r\n    this._mem.tableCfgs = hip({\r\n      _type: 'tableCfgs',\r\n      _data: [tableCfg],\r\n      _tableCfg: tableCfg._hash as string,\r\n    });\r\n  };\r\n\r\n  // ...........................................................................\r\n  private _updateGlobalHash() {\r\n    (this._mem as any)._hash = '';\r\n    hip(this._mem, {\r\n      updateExistingHashes: false,\r\n    });\r\n  }\r\n\r\n  // ...........................................................................\r\n  private _updateTableHash(tableKey: TableKey) {\r\n    const table = this._mem[tableKey] as TableType;\r\n    table._hash = '';\r\n    hip(table, { updateExistingHashes: false });\r\n  }\r\n\r\n  // ...........................................................................\r\n  private async _createOrExtendTable(request: {\r\n    tableCfg: TableCfg;\r\n  }): Promise<void> {\r\n    // Make sure that the table config is compatible\r\n    // with an potential existing table\r\n    const tableCfg = request.tableCfg;\r\n    await this._ioTools.throwWhenTableIsNotCompatible(tableCfg);\r\n\r\n    const { key } = tableCfg;\r\n\r\n    // Recreate hashes in the case the existing hashes are wrong\r\n    const newConfig = hsh(tableCfg);\r\n\r\n    // Find an existing table config with the same hash\r\n    const existingConfig = await this._ioTools.tableCfgOrNull(key);\r\n\r\n    // Write the new config into the database\r\n    if (!existingConfig) {\r\n      this._createTable(newConfig, key);\r\n    } else {\r\n      this._extendTable(existingConfig, newConfig);\r\n    }\r\n  }\r\n\r\n  // ...........................................................................\r\n  private _createTable(newConfig: TableCfg, tableKey: TableKey) {\r\n    // Write the table config into the database\r\n    newConfig = hsh(newConfig);\r\n    this._mem.tableCfgs._data.push(newConfig);\r\n    this._ioTools.sortTableDataAndUpdateHash(this._mem.tableCfgs);\r\n\r\n    // Create a table and write it into the database\r\n    const table: TableType = {\r\n      _type: newConfig.type,\r\n      _data: [],\r\n      _tableCfg: newConfig._hash as string,\r\n    };\r\n\r\n    this._mem[tableKey] ??= hip(table);\r\n\r\n    // Update hashes\r\n    this._updateTableHash(tableKey);\r\n    this._updateGlobalHash();\r\n  }\r\n\r\n  // ...........................................................................\r\n  private _extendTable(existingConfig: TableCfg, newConfig: TableCfg) {\r\n    // No columns added? Return.\r\n    if (existingConfig.columns.length === newConfig.columns.length) {\r\n      return;\r\n    }\r\n\r\n    // Write the new table config into the database\r\n    newConfig = hsh(newConfig);\r\n    this._mem.tableCfgs._data.push(newConfig);\r\n    this._ioTools.sortTableDataAndUpdateHash(this._mem.tableCfgs);\r\n\r\n    // Update the config of the existing table\r\n    const table = this._mem[newConfig.key] as TableType;\r\n    table._tableCfg = newConfig._hash as string;\r\n\r\n    // Update the hashes\r\n    this._updateTableHash(newConfig.key);\r\n    this._updateGlobalHash();\r\n  }\r\n\r\n  // ...........................................................................\r\n\r\n  private async _dump(): Promise<Rljson> {\r\n    return copy(this._mem);\r\n  }\r\n\r\n  // ...........................................................................\r\n  private async _dumpTable(request: { table: string }): Promise<Rljson> {\r\n    await this._ioTools.throwWhenTableDoesNotExist(request.table);\r\n\r\n    const table = this._mem[request.table] as TableType;\r\n\r\n    return {\r\n      [request.table]: copy(table),\r\n    };\r\n  }\r\n\r\n  // ...........................................................................\r\n  private async _contentType(request: { table: string }): Promise<ContentType> {\r\n    await this._ioTools.throwWhenTableDoesNotExist(request.table);\r\n\r\n    return (this._mem[request.table] as TableType)._type;\r\n  }\r\n\r\n  // ...........................................................................\r\n  private async _write(request: { data: Rljson }): Promise<void> {\r\n    const addedData = hsh(request.data);\r\n    this._removeNullValues(addedData);\r\n    const tables = Object.keys(addedData);\r\n    hsh(addedData);\r\n\r\n    await this._ioTools.throwWhenTablesDoNotExist(request.data);\r\n    await this._ioTools.throwWhenTableDataDoesNotMatchCfg(request.data);\r\n\r\n    for (const table of tables) {\r\n      if (table.startsWith('_')) {\r\n        continue;\r\n      }\r\n\r\n      const oldTable = this._mem[table] as TableType;\r\n      const newTable = addedData[table] as TableType;\r\n\r\n      // Table exists. Merge data\r\n      for (const item of newTable._data) {\r\n        const hash = item._hash;\r\n        const exists = oldTable._data.find((i) => i._hash === hash);\r\n        if (!exists) {\r\n          oldTable._data.push(item as any);\r\n        }\r\n      }\r\n\r\n      this._ioTools.sortTableDataAndUpdateHash(oldTable);\r\n    }\r\n\r\n    // Recalc main hashes\r\n    this._updateGlobalHash();\r\n  }\r\n\r\n  // ...........................................................................\r\n  private async _readRows(request: {\r\n    table: string;\r\n    where: { [column: string]: JsonValue };\r\n  }): Promise<Rljson> {\r\n    await this._ioTools.throwWhenTableDoesNotExist(request.table);\r\n    await this._ioTools.throwWhenColumnDoesNotExist(\r\n      request.table,\r\n      Object.keys(request.where),\r\n    );\r\n\r\n    // Read table from data\r\n    const table = this._mem[request.table] as TableType;\r\n\r\n    // Filter table data\r\n    const tableDataFiltered = table._data.filter((row) => {\r\n      for (const column in request.where) {\r\n        const a = row[column];\r\n        const b = request.where[column];\r\n        if (b === null && a === undefined) {\r\n          return true;\r\n        }\r\n\r\n        if (!equals(a, b)) {\r\n          return false;\r\n        }\r\n      }\r\n      return true;\r\n    });\r\n\r\n    // Create an table\r\n    const tableFiltered: TableType = {\r\n      _type: table._type,\r\n      _data: tableDataFiltered,\r\n    };\r\n\r\n    this._ioTools.sortTableDataAndUpdateHash(tableFiltered);\r\n\r\n    const result: Rljson = {\r\n      [request.table]: tableFiltered,\r\n    };\r\n\r\n    return result;\r\n  }\r\n\r\n  _removeNullValues(rljson: Rljson) {\r\n    iterateTablesSync(rljson, (table) => {\r\n      const data = rljson[table]._data;\r\n\r\n      for (const row of data) {\r\n        for (const key in row) {\r\n          if (row[key] === null) {\r\n            delete row[key];\r\n          }\r\n        }\r\n      }\r\n    });\r\n  }\r\n}\r\n","// @license\r\n// Copyright (c) 2025 Rljson\r\n//\r\n// Use of this source code is governed by terms that can be\r\n// found in the LICENSE file in the root of this package.\r\n\r\nimport { Io, IoMem } from './index.ts';\r\n\r\n// .............................................................................\r\n/**\r\n * Io implementation need to implement this interface to be used in\r\n * conformance tests.\r\n */\r\nexport interface IoTestSetup {\r\n  /** setup before the single setups */\r\n  beforeAll: () => Promise<void>;\r\n\r\n  /**\r\n   * Initializes the io implementation.\r\n   * @returns The io implementation.\r\n   */\r\n  beforeEach: () => Promise<void>;\r\n\r\n  /**\r\n   * Tears down the io implementation.\r\n   * @returns The io implementation.\r\n   */\r\n  afterEach: () => Promise<void>;\r\n\r\n  /** cleanup after all tests */\r\n  afterAll: () => Promise<void>;\r\n\r\n  /**\r\n   * The io implementation to be used in the conformance tests.\r\n   */\r\n  io: Io;\r\n}\r\n\r\n// .............................................................................\r\n// Example implementation of the IoTestSetup interface\r\nexport const exampleTestSetup = (): IoTestSetup => {\r\n  return {\r\n    io: new IoMem(),\r\n    beforeAll: async () => {\r\n      // This method can be used for any additional setup required before init.\r\n      // Currently, it does nothing.\r\n    },\r\n\r\n    beforeEach: async () => {\r\n      // Initialize the io implementation\r\n    },\r\n    afterEach: async () => {\r\n      // Tear down the io implementation\r\n    },\r\n\r\n    afterAll: async () => {\r\n      // This method can be used for any additional cleanup after tearDown.\r\n    },\r\n  };\r\n};\r\n","// @license\r\n// Copyright (c) 2025 Rljson\r\n//\r\n// Use of this source code is governed by terms that can be\r\n// found in the LICENSE file in the root of this package.\r\n\r\nimport { JsonValue } from '@rljson/json';\r\nimport { ContentType, Rljson, TableCfg, TableKey } from '@rljson/rljson';\r\n\r\n\r\n// .............................................................................\r\nexport interface Io {\r\n  // ...........................................................................\r\n  // General\r\n\r\n  /** Starts the initialization */\r\n  init(): Promise<void>;\r\n\r\n  /** Closes the io */\r\n  close(): Promise<void>;\r\n\r\n  /** Returns true if io is opened */\r\n  isOpen: boolean;\r\n\r\n  /** A promise resolving once the Io interface is ready\r\n   *\r\n   * 💡 Use @rljson/is-ready\r\n   */\r\n  isReady(): Promise<void>;\r\n\r\n  // ...........................................................................\r\n  // Dump\r\n\r\n  /** Returns the complete db content as Rljson */\r\n  dump(): Promise<Rljson>;\r\n\r\n  /** Returns the dump of a complete table */\r\n  dumpTable(request: { table: string }): Promise<Rljson>;\r\n\r\n  // ...........................................................................\r\n  // Meta Data\r\n  contentType(request: { table: string }): Promise<ContentType>;\r\n\r\n  // ...........................................................................\r\n  // Tables\r\n\r\n  /**\r\n   * Returns true if the table exists\r\n   */\r\n  tableExists(tableKey: TableKey): Promise<boolean>;\r\n\r\n  /**\r\n   * Creates a table with a given config.\r\n   * If the table already exists, new columns are added to the existing table.\r\n   * If the table does not exist, it is created with the given config.\r\n   * If the table exists and columns are removed, an error is thrown.\r\n   * If the table exists and the column type is changed, an error is thrown.\r\n   */\r\n  createOrExtendTable(request: { tableCfg: TableCfg }): Promise<void>;\r\n\r\n  /**\r\n   * Returns a json structure returning current table configurations\r\n   */\r\n  rawTableCfgs(): Promise<TableCfg[]>;\r\n\r\n  // ...........................................................................\r\n  // Write\r\n\r\n  /** Writes Rljson data into the database */\r\n  write(request: { data: Rljson }): Promise<void>;\r\n\r\n  // ...........................................................................\r\n  // Read rows\r\n\r\n  /** Queries a list of rows */\r\n  readRows(request: {\r\n    table: string;\r\n    where: { [column: string]: JsonValue | null };\r\n  }): Promise<Rljson>;\r\n\r\n  /** Returns the number of rows in the given table */\r\n  rowCount(table: string): Promise<number>;\r\n}\r\n\r\n// .............................................................................\r\nexport const exampleIo =\r\n  'Checkout @rljson/io-mem for an example implementation';\r\n","/* v8 ignore file -- @preserve */\r\n// @license\r\n// Copyright (c) 2025 Rljson\r\n//\r\n// Use of this source code is governed by terms that can be\r\n// found in the LICENSE file in the root of this package.\r\n\r\nimport { Json } from '@rljson/json';\r\nimport {\r\n  Buffet,\r\n  Cake,\r\n  iterateTablesSync,\r\n  Layer,\r\n  Ref,\r\n  Rljson,\r\n  TableKey,\r\n} from '@rljson/rljson';\r\n\r\n// .............................................................................\r\n/**\r\n * Describes a row that references a child table row\r\n */\r\nexport interface ParentRef {\r\n  /**\r\n   * The parent table that references the child table\r\n   */\r\n  [parentTable: TableKey]: {\r\n    /**\r\n     * The parent row that references the child row\r\n     */\r\n    [parentRow: Ref]: {\r\n      /**\r\n       * Details about the reference, e.g. an array index etc.\r\n       */\r\n      details?: Json;\r\n    };\r\n  };\r\n}\r\n\r\n// .............................................................................\r\n/**\r\n * Describes the parent table rows referencing a child table row\r\n */\r\nexport interface ReverseRefs {\r\n  /**\r\n   * The child table we need the referencing rows for\r\n   */\r\n  [childTable: TableKey]: {\r\n    /**\r\n     * The row hashwe need the referencing rows for\r\n     */\r\n    [childRow: Ref]: ParentRef;\r\n  };\r\n}\r\n\r\n/* v8 ignore start */\r\n\r\n// .............................................................................\r\n/**\r\n * Calculates the reverse references for a given rljson object\r\n */\r\nexport const calcReverseRefs = (rljson: Rljson): ReverseRefs => {\r\n  const result: ReverseRefs = {};\r\n\r\n  // ......................\r\n  // Prepare data structure\r\n  iterateTablesSync(rljson, (childTableKey, table) => {\r\n    const childTable: { [childRowHash: string]: ParentRef } = {};\r\n    result[childTableKey] = childTable;\r\n    for (const childRow of table._data) {\r\n      childTable[childRow._hash] = {};\r\n    }\r\n  });\r\n\r\n  // ............................\r\n  // Generate reverse references\r\n  iterateTablesSync(rljson, (parentTableKey, parentTable) => {\r\n    // Iterate all rows of each table\r\n    for (const parentTableRow of parentTable._data) {\r\n      // Find out whe other tables & rows are referenced by this row\r\n      // Write these information intto result\r\n      switch (parentTable._type) {\r\n        case 'components':\r\n          _writeComponentRefs(parentTableKey, parentTableRow, result);\r\n          break;\r\n\r\n        case 'layers': {\r\n          _writeLayerRefs(parentTableKey, parentTableRow, result);\r\n          break;\r\n        }\r\n\r\n        case 'sliceIds': {\r\n          // Slice ids do not reference other tables\r\n          break;\r\n        }\r\n\r\n        case 'cakes': {\r\n          _writeCakeRefs(parentTableKey, parentTableRow, result);\r\n          break;\r\n        }\r\n\r\n        case 'buffets': {\r\n          _writeBuffetRefs(parentTableKey, parentTableRow, result);\r\n          break;\r\n        }\r\n      }\r\n    }\r\n  });\r\n\r\n  return result;\r\n};\r\n\r\n/* v8 ignore end */\r\n\r\n// .............................................................................\r\nconst _writeComponentRefs = (\r\n  parentTableName: TableKey,\r\n  parentRow: Json,\r\n  result: ReverseRefs,\r\n) => {\r\n  const parentRowHash = parentRow._hash as string;\r\n\r\n  for (const parentColumnName in parentRow) {\r\n    if (parentColumnName.startsWith('_')) {\r\n      continue;\r\n    }\r\n\r\n    if (!parentColumnName.endsWith('Ref')) {\r\n      continue;\r\n    }\r\n\r\n    const childTableName = parentColumnName.slice(0, -3);\r\n    const childRowHash = parentRow[parentColumnName] as string;\r\n\r\n    _write(\r\n      result,\r\n      childTableName,\r\n      childRowHash,\r\n      parentTableName,\r\n      parentRowHash,\r\n    );\r\n  }\r\n};\r\n\r\n// .............................................................................\r\nconst _writeLayerRefs = (\r\n  parentTableName: TableKey,\r\n  parentRow: Layer,\r\n  result: ReverseRefs,\r\n) => {\r\n  const childTableName = parentRow.componentsTable;\r\n  const parentRowHash = parentRow._hash as string;\r\n\r\n  for (const sliceId in parentRow.add) {\r\n    if (sliceId.startsWith('_')) {\r\n      continue;\r\n    }\r\n\r\n    const sliceHash = parentRow.add[sliceId] as string;\r\n\r\n    _write(result, childTableName, sliceHash, parentTableName, parentRowHash);\r\n  }\r\n};\r\n\r\n// .............................................................................\r\nconst _writeCakeRefs = (\r\n  parentTableName: TableKey,\r\n  parentRow: Cake,\r\n  result: ReverseRefs,\r\n) => {\r\n  const parentRowHash = parentRow._hash as string;\r\n\r\n  for (const layer in parentRow.layers) {\r\n    const childTableName = layer;\r\n    const childRowHash = parentRow.layers[layer] as string;\r\n    _write(\r\n      result,\r\n      childTableName,\r\n      childRowHash,\r\n      parentTableName,\r\n      parentRowHash,\r\n    );\r\n  }\r\n};\r\n\r\n// .............................................................................\r\nconst _writeBuffetRefs = (\r\n  parentTableName: TableKey,\r\n  parentRow: Buffet,\r\n  result: ReverseRefs,\r\n) => {\r\n  const parentRowHash = parentRow._hash as string;\r\n\r\n  for (const item of parentRow.items) {\r\n    const childTableName = item.table;\r\n    const childRowHash = item.ref;\r\n    _write(\r\n      result,\r\n      childTableName,\r\n      childRowHash,\r\n      parentTableName,\r\n      parentRowHash,\r\n    );\r\n  }\r\n};\r\n\r\n// .............................................................................\r\nconst _write = (\r\n  result: ReverseRefs,\r\n  childTableName: string,\r\n  childRowHash: string,\r\n  parentTableName: string,\r\n  parentRowHash: string,\r\n) => {\r\n  const referencesForChildTable = (result[childTableName] ??= {});\r\n  const referencesForChildTableRow = (referencesForChildTable[childRowHash] ??=\r\n    {});\r\n\r\n  referencesForChildTableRow[parentTableName] ??= {};\r\n  referencesForChildTableRow[parentTableName][parentRowHash] ??= {};\r\n};\r\n"],"names":["e"],"mappings":"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"}
@@ -1,10 +1,10 @@
1
- // @license
2
- // Copyright (c) 2025 Rljson
3
- //
4
- // Use of this source code is governed by terms that can be
5
- // found in the LICENSE file in the root of this package.
6
-
7
- // Run »pnpm updateGoldens« when you change this file
8
- export const example = async () => {};
9
-
10
- // example();
1
+ // @license
2
+ // Copyright (c) 2025 Rljson
3
+ //
4
+ // Use of this source code is governed by terms that can be
5
+ // found in the LICENSE file in the root of this package.
6
+
7
+ // Run »pnpm updateGoldens« when you change this file
8
+ export const example = async () => {};
9
+
10
+ // example();
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@rljson/io",
3
- "version": "0.0.56",
3
+ "version": "0.0.58",
4
4
  "description": "Low level interface for reading and writing RLJSON data",
5
5
  "homepage": "https://github.com/rljson/io",
6
6
  "bugs": "https://github.com/rljson/io/issues",
@@ -21,23 +21,23 @@
21
21
  "type": "module",
22
22
  "devDependencies": {
23
23
  "@types/node": "^24.10.1",
24
- "@typescript-eslint/eslint-plugin": "^8.46.4",
25
- "@typescript-eslint/parser": "^8.46.4",
26
- "@vitest/coverage-v8": "^4.0.9",
24
+ "@typescript-eslint/eslint-plugin": "^8.47.0",
25
+ "@typescript-eslint/parser": "^8.47.0",
26
+ "@vitest/coverage-v8": "^4.0.12",
27
27
  "cross-env": "^10.1.0",
28
28
  "eslint": "^9.39.1",
29
- "eslint-plugin-jsdoc": "^61.2.1",
29
+ "eslint-plugin-jsdoc": "^61.4.0",
30
30
  "eslint-plugin-tsdoc": "^0.5.0",
31
31
  "globals": "^16.5.0",
32
32
  "jsdoc": "^4.0.5",
33
33
  "read-pkg": "^10.0.0",
34
34
  "typescript": "~5.9.3",
35
- "typescript-eslint": "^8.46.4",
36
- "vite": "^7.2.2",
37
- "vite-node": "^5.1.0",
35
+ "typescript-eslint": "^8.47.0",
36
+ "vite": "^7.2.4",
37
+ "vite-node": "^5.2.0",
38
38
  "vite-plugin-dts": "^4.5.4",
39
39
  "vite-tsconfig-paths": "^5.1.4",
40
- "vitest": "^4.0.9",
40
+ "vitest": "^4.0.12",
41
41
  "vitest-dom": "^0.1.1"
42
42
  },
43
43
  "dependencies": {