@rljson/io 0.0.43 → 0.0.46
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/conformance-tests/goldens/rawTableCfgs.json +168 -0
- package/dist/conformance-tests/io-conformance.setup.ts +41 -6
- package/dist/conformance-tests/io-conformance.spec.ts +210 -5
- package/dist/io-mem.d.ts +9 -1
- package/dist/io-peer.d.ts +100 -0
- package/dist/io-server.d.ts +34 -0
- package/dist/io-tools.d.ts +33 -0
- package/dist/io.d.ts +12 -1
- package/dist/io.js +85 -1
- package/dist/peer-server-socket-mock.d.ts +18 -0
- package/dist/peer-socket-mock.d.ts +73 -0
- package/dist/socket.d.ts +15 -0
- package/package.json +13 -13
|
@@ -0,0 +1,168 @@
|
|
|
1
|
+
[
|
|
2
|
+
{
|
|
3
|
+
"key": "revisions",
|
|
4
|
+
"type": "revisions",
|
|
5
|
+
"isHead": true,
|
|
6
|
+
"isRoot": true,
|
|
7
|
+
"isShared": false,
|
|
8
|
+
"columns": [
|
|
9
|
+
{
|
|
10
|
+
"key": "_hash",
|
|
11
|
+
"type": "string",
|
|
12
|
+
"_hash": "df4oYftB-71Njv9FprRCeg"
|
|
13
|
+
},
|
|
14
|
+
{
|
|
15
|
+
"key": "table",
|
|
16
|
+
"type": "string",
|
|
17
|
+
"_hash": "FmsYjsit04XJi02Sihdwl-"
|
|
18
|
+
},
|
|
19
|
+
{
|
|
20
|
+
"key": "predecessor",
|
|
21
|
+
"type": "string",
|
|
22
|
+
"_hash": "RlAk7Lj-lZf2KDlUk5fekZ"
|
|
23
|
+
},
|
|
24
|
+
{
|
|
25
|
+
"key": "successor",
|
|
26
|
+
"type": "string",
|
|
27
|
+
"_hash": "MJlFtQVty5oUJKVzzj5hyB"
|
|
28
|
+
},
|
|
29
|
+
{
|
|
30
|
+
"key": "timestamp",
|
|
31
|
+
"type": "number",
|
|
32
|
+
"_hash": "XpKYGh23thW6vONgQuoMHf"
|
|
33
|
+
},
|
|
34
|
+
{
|
|
35
|
+
"key": "id",
|
|
36
|
+
"type": "string",
|
|
37
|
+
"_hash": "eq2P3RIaSgy81i7PomNUvn"
|
|
38
|
+
}
|
|
39
|
+
],
|
|
40
|
+
"_hash": "2ztQLyMtsFFIs0z_rvT6UL"
|
|
41
|
+
},
|
|
42
|
+
{
|
|
43
|
+
"key": "table0",
|
|
44
|
+
"type": "components",
|
|
45
|
+
"isHead": false,
|
|
46
|
+
"isRoot": false,
|
|
47
|
+
"isShared": true,
|
|
48
|
+
"columns": [
|
|
49
|
+
{
|
|
50
|
+
"key": "_hash",
|
|
51
|
+
"type": "string",
|
|
52
|
+
"_hash": "df4oYftB-71Njv9FprRCeg"
|
|
53
|
+
},
|
|
54
|
+
{
|
|
55
|
+
"key": "col0",
|
|
56
|
+
"type": "string",
|
|
57
|
+
"_hash": "LyqWPAOnRnSAofi4I1aMIv"
|
|
58
|
+
}
|
|
59
|
+
],
|
|
60
|
+
"_hash": "SkSdXUd0JvYYu_AtvQ2sOZ"
|
|
61
|
+
},
|
|
62
|
+
{
|
|
63
|
+
"_hash": "er5aXXmEqihTA7Y1glILb4",
|
|
64
|
+
"key": "tableCfgs",
|
|
65
|
+
"type": "tableCfgs",
|
|
66
|
+
"isHead": false,
|
|
67
|
+
"isRoot": false,
|
|
68
|
+
"isShared": true,
|
|
69
|
+
"previous": "",
|
|
70
|
+
"columns": [
|
|
71
|
+
{
|
|
72
|
+
"key": "_hash",
|
|
73
|
+
"type": "string",
|
|
74
|
+
"_hash": "df4oYftB-71Njv9FprRCeg"
|
|
75
|
+
},
|
|
76
|
+
{
|
|
77
|
+
"key": "key",
|
|
78
|
+
"type": "string",
|
|
79
|
+
"_hash": "tapEY-QdaJwWhz1PPZoG35"
|
|
80
|
+
},
|
|
81
|
+
{
|
|
82
|
+
"key": "type",
|
|
83
|
+
"type": "string",
|
|
84
|
+
"_hash": "zE7ALsNzu5JPgDD4AxeRCh"
|
|
85
|
+
},
|
|
86
|
+
{
|
|
87
|
+
"key": "isHead",
|
|
88
|
+
"type": "boolean",
|
|
89
|
+
"_hash": "ar8Hajt2UZDn-F21QLMyC_"
|
|
90
|
+
},
|
|
91
|
+
{
|
|
92
|
+
"key": "isRoot",
|
|
93
|
+
"type": "boolean",
|
|
94
|
+
"_hash": "hedsCm_HtiiTuzJyTeMqhm"
|
|
95
|
+
},
|
|
96
|
+
{
|
|
97
|
+
"key": "isShared",
|
|
98
|
+
"type": "boolean",
|
|
99
|
+
"_hash": "pxXOf3-3qCo9Z3FzC9JXSH"
|
|
100
|
+
},
|
|
101
|
+
{
|
|
102
|
+
"key": "previous",
|
|
103
|
+
"type": "string",
|
|
104
|
+
"_hash": "e0TEeDmSvJcLEMtrVxc0oW"
|
|
105
|
+
},
|
|
106
|
+
{
|
|
107
|
+
"key": "columns",
|
|
108
|
+
"type": "jsonArray",
|
|
109
|
+
"_hash": "aPmGTcNng9E1RvlEj1LtxB"
|
|
110
|
+
}
|
|
111
|
+
]
|
|
112
|
+
},
|
|
113
|
+
{
|
|
114
|
+
"key": "table0",
|
|
115
|
+
"type": "components",
|
|
116
|
+
"isHead": false,
|
|
117
|
+
"isRoot": false,
|
|
118
|
+
"isShared": true,
|
|
119
|
+
"columns": [
|
|
120
|
+
{
|
|
121
|
+
"key": "_hash",
|
|
122
|
+
"type": "string",
|
|
123
|
+
"_hash": "df4oYftB-71Njv9FprRCeg"
|
|
124
|
+
},
|
|
125
|
+
{
|
|
126
|
+
"key": "col0",
|
|
127
|
+
"type": "string",
|
|
128
|
+
"_hash": "LyqWPAOnRnSAofi4I1aMIv"
|
|
129
|
+
},
|
|
130
|
+
{
|
|
131
|
+
"key": "col1",
|
|
132
|
+
"type": "string",
|
|
133
|
+
"_hash": "aD2OJziZ76JeLyPuV4ioKe"
|
|
134
|
+
}
|
|
135
|
+
],
|
|
136
|
+
"_hash": "vP2jpAaUyT_Xo3bvmTgfpo"
|
|
137
|
+
},
|
|
138
|
+
{
|
|
139
|
+
"key": "table0",
|
|
140
|
+
"type": "components",
|
|
141
|
+
"isHead": false,
|
|
142
|
+
"isRoot": false,
|
|
143
|
+
"isShared": true,
|
|
144
|
+
"columns": [
|
|
145
|
+
{
|
|
146
|
+
"key": "_hash",
|
|
147
|
+
"type": "string",
|
|
148
|
+
"_hash": "df4oYftB-71Njv9FprRCeg"
|
|
149
|
+
},
|
|
150
|
+
{
|
|
151
|
+
"key": "col0",
|
|
152
|
+
"type": "string",
|
|
153
|
+
"_hash": "LyqWPAOnRnSAofi4I1aMIv"
|
|
154
|
+
},
|
|
155
|
+
{
|
|
156
|
+
"key": "col1",
|
|
157
|
+
"type": "string",
|
|
158
|
+
"_hash": "aD2OJziZ76JeLyPuV4ioKe"
|
|
159
|
+
},
|
|
160
|
+
{
|
|
161
|
+
"key": "col2",
|
|
162
|
+
"type": "string",
|
|
163
|
+
"_hash": "yd4AdprM2Z0vqKieWOibX5"
|
|
164
|
+
}
|
|
165
|
+
],
|
|
166
|
+
"_hash": "wfRMrOPA2C9ogANQmzsI7K"
|
|
167
|
+
}
|
|
168
|
+
]
|
|
@@ -5,16 +5,17 @@
|
|
|
5
5
|
// found in the LICENSE file in the root of this package.
|
|
6
6
|
|
|
7
7
|
import { Io, IoMem, IoTestSetup } from '@rljson/io';
|
|
8
|
+
import { IoPeer } from '../src/io-peer';
|
|
9
|
+
import { IoServer } from '../src/io-server';
|
|
10
|
+
import { PeerServerSocketMock } from '../src/peer-server-socket-mock';
|
|
8
11
|
|
|
9
12
|
// ..............................................................................
|
|
10
|
-
class
|
|
13
|
+
abstract class GenericIoTestSetup implements IoTestSetup {
|
|
11
14
|
async beforeAll(): Promise<void> {
|
|
12
15
|
// This method can be used for any additional setup required before init.
|
|
13
16
|
// Currently, it does nothing.
|
|
14
17
|
}
|
|
15
|
-
async beforeEach(): Promise<void> {
|
|
16
|
-
this._io = await IoMem.example();
|
|
17
|
-
}
|
|
18
|
+
async beforeEach(): Promise<void> {}
|
|
18
19
|
|
|
19
20
|
async afterEach(): Promise<void> {
|
|
20
21
|
await this.io.close();
|
|
@@ -31,8 +32,42 @@ class MyIoTestSetup implements IoTestSetup {
|
|
|
31
32
|
return this._io;
|
|
32
33
|
}
|
|
33
34
|
|
|
34
|
-
|
|
35
|
+
protected _io: Io | null = null;
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
class IoMemTestSetup extends GenericIoTestSetup {
|
|
39
|
+
async beforeEach(): Promise<void> {
|
|
40
|
+
this._io = await IoMem.example();
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
class IoPeerTestSetup extends GenericIoTestSetup {
|
|
45
|
+
async beforeEach(): Promise<void> {
|
|
46
|
+
this._io = await IoPeer.example();
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
class IoPeerServerTestSetup extends GenericIoTestSetup {
|
|
51
|
+
async beforeEach(): Promise<void> {
|
|
52
|
+
//Io of Server --> IoMem
|
|
53
|
+
const ioMemServer = await IoMem.example();
|
|
54
|
+
|
|
55
|
+
//Socket between Server and Peer
|
|
56
|
+
const socket = new PeerServerSocketMock();
|
|
57
|
+
|
|
58
|
+
//IoPeer of Peer --> Socket
|
|
59
|
+
const ioServer = new IoServer(ioMemServer);
|
|
60
|
+
ioServer.addSocket(socket);
|
|
61
|
+
|
|
62
|
+
//IoPeer of Peer --> Socket
|
|
63
|
+
const io = new IoPeer(socket);
|
|
64
|
+
await io.init();
|
|
65
|
+
|
|
66
|
+
this._io = io;
|
|
67
|
+
}
|
|
35
68
|
}
|
|
36
69
|
|
|
37
70
|
// .............................................................................
|
|
38
|
-
export const
|
|
71
|
+
export const testMemSetup = () => new IoMemTestSetup();
|
|
72
|
+
export const testPeerSetup = () => new IoPeerTestSetup();
|
|
73
|
+
export const testPeerServerSetup = () => new IoPeerServerTestSetup();
|
|
@@ -32,25 +32,39 @@ import {
|
|
|
32
32
|
describe,
|
|
33
33
|
expect,
|
|
34
34
|
it,
|
|
35
|
+
vi,
|
|
35
36
|
} from 'vitest';
|
|
36
37
|
|
|
37
38
|
import { Io, IoTestSetup, IoTools } from '@rljson/io';
|
|
38
39
|
|
|
39
|
-
import {
|
|
40
|
+
import {
|
|
41
|
+
testMemSetup,
|
|
42
|
+
testPeerServerSetup,
|
|
43
|
+
testPeerSetup,
|
|
44
|
+
} from './io-conformance.setup.ts';
|
|
40
45
|
import { expectGolden, ExpectGoldenOptions } from './setup/goldens.ts';
|
|
41
46
|
|
|
42
47
|
const ego: ExpectGoldenOptions = {
|
|
43
48
|
npmUpdateGoldensEnabled: false,
|
|
44
49
|
};
|
|
45
50
|
|
|
46
|
-
|
|
47
|
-
|
|
51
|
+
const setups: Record<string, IoTestSetup> = {
|
|
52
|
+
IoMem: testMemSetup(),
|
|
53
|
+
IoPeer: testPeerSetup(),
|
|
54
|
+
IoPeerServer: testPeerServerSetup(),
|
|
55
|
+
};
|
|
56
|
+
|
|
57
|
+
export const runIoConformanceTests = (
|
|
58
|
+
setupName: string,
|
|
59
|
+
ioTestSetup: IoTestSetup,
|
|
60
|
+
) => {
|
|
61
|
+
return describe('Io Conformance - ' + setupName, async () => {
|
|
48
62
|
let io: Io;
|
|
49
63
|
let ioTools: IoTools;
|
|
50
64
|
let setup: IoTestSetup;
|
|
51
65
|
|
|
52
66
|
beforeAll(async () => {
|
|
53
|
-
setup =
|
|
67
|
+
setup = ioTestSetup;
|
|
54
68
|
await setup.beforeAll();
|
|
55
69
|
});
|
|
56
70
|
|
|
@@ -139,11 +153,54 @@ export const runIoConformanceTests = () => {
|
|
|
139
153
|
});
|
|
140
154
|
});
|
|
141
155
|
|
|
156
|
+
describe('rawTableCfgs()', () => {
|
|
157
|
+
it('returns an array of all table configurations', async () => {
|
|
158
|
+
//create four tables with two versions each
|
|
159
|
+
const tableV0: TableCfg = {
|
|
160
|
+
key: 'table0',
|
|
161
|
+
type: 'components',
|
|
162
|
+
isHead: false,
|
|
163
|
+
isRoot: false,
|
|
164
|
+
isShared: true,
|
|
165
|
+
columns: [
|
|
166
|
+
{ key: '_hash', type: 'string' },
|
|
167
|
+
{ key: 'col0', type: 'string' },
|
|
168
|
+
],
|
|
169
|
+
};
|
|
170
|
+
|
|
171
|
+
const tableV1 = addColumnsToTableCfg(tableV0, [
|
|
172
|
+
{ key: 'col1', type: 'string' },
|
|
173
|
+
]);
|
|
174
|
+
|
|
175
|
+
const tableV2 = addColumnsToTableCfg(tableV1, [
|
|
176
|
+
{ key: 'col2', type: 'string' },
|
|
177
|
+
]);
|
|
178
|
+
|
|
179
|
+
await io.createOrExtendTable({ tableCfg: tableV0 });
|
|
180
|
+
await io.createOrExtendTable({ tableCfg: tableV1 });
|
|
181
|
+
await io.createOrExtendTable({ tableCfg: tableV2 });
|
|
182
|
+
|
|
183
|
+
// Check the tableCfgs
|
|
184
|
+
const actualTableCfgs = await io.rawTableCfgs();
|
|
185
|
+
await expectGolden('io-conformance/rawTableCfgs.json', ego).toBe(
|
|
186
|
+
actualTableCfgs,
|
|
187
|
+
);
|
|
188
|
+
});
|
|
189
|
+
});
|
|
190
|
+
|
|
142
191
|
describe('throws an error', async () => {
|
|
143
192
|
it('if the hashes in the tableCfg are wrong', async () => {
|
|
144
193
|
const tableCfg: TableCfg = hip(exampleTableCfg({ key: 'table1' }));
|
|
145
194
|
tableCfg._hash = 'wrongHash';
|
|
146
195
|
let message: string = '';
|
|
196
|
+
|
|
197
|
+
await expect(
|
|
198
|
+
io.createOrExtendTable({ tableCfg: tableCfg }),
|
|
199
|
+
).rejects.toThrow(
|
|
200
|
+
'Hash "wrongHash" does not match the newly calculated one "uX24nHRtwkXRsq8l46cNRZ". ' +
|
|
201
|
+
'Please make sure that all systems are producing the same hashes.',
|
|
202
|
+
);
|
|
203
|
+
|
|
147
204
|
try {
|
|
148
205
|
await io.createOrExtendTable({ tableCfg: tableCfg });
|
|
149
206
|
} catch (err: any) {
|
|
@@ -947,7 +1004,155 @@ export const runIoConformanceTests = () => {
|
|
|
947
1004
|
).rejects.toThrow('Table "nonexistentTable" not found');
|
|
948
1005
|
});
|
|
949
1006
|
});
|
|
1007
|
+
|
|
1008
|
+
describe('contentType()', () => {
|
|
1009
|
+
it('returns the content type of the given table', async () => {
|
|
1010
|
+
await createExampleTable('table1');
|
|
1011
|
+
|
|
1012
|
+
await io.write({
|
|
1013
|
+
data: {
|
|
1014
|
+
table1: {
|
|
1015
|
+
_type: 'components',
|
|
1016
|
+
_data: [{ a: 'a2' }],
|
|
1017
|
+
},
|
|
1018
|
+
},
|
|
1019
|
+
});
|
|
1020
|
+
|
|
1021
|
+
const contentType = await io.contentType({ table: 'table1' });
|
|
1022
|
+
expect(contentType).toBe('components');
|
|
1023
|
+
});
|
|
1024
|
+
|
|
1025
|
+
it('returns error if table is not existing', async () => {
|
|
1026
|
+
await expect(io.contentType({ table: 'unknown' })).rejects.toThrow(
|
|
1027
|
+
'Table "unknown" not found',
|
|
1028
|
+
);
|
|
1029
|
+
});
|
|
1030
|
+
});
|
|
1031
|
+
|
|
1032
|
+
describe('observeTable(table, callback)', () => {
|
|
1033
|
+
it('should call listener on table changes', async () => {
|
|
1034
|
+
//Create example table and add initial data
|
|
1035
|
+
await createExampleTable('table1');
|
|
1036
|
+
await io.write({
|
|
1037
|
+
data: {
|
|
1038
|
+
table1: {
|
|
1039
|
+
_type: 'components',
|
|
1040
|
+
_data: [{ a: 'a1' }],
|
|
1041
|
+
},
|
|
1042
|
+
},
|
|
1043
|
+
});
|
|
1044
|
+
|
|
1045
|
+
//Create listener
|
|
1046
|
+
const cb = vi.fn();
|
|
1047
|
+
|
|
1048
|
+
//Data to write
|
|
1049
|
+
const data = {
|
|
1050
|
+
table1: {
|
|
1051
|
+
_type: 'components',
|
|
1052
|
+
_data: [{ a: 'a2' }],
|
|
1053
|
+
},
|
|
1054
|
+
} as Rljson;
|
|
1055
|
+
|
|
1056
|
+
//Subscribe to changes
|
|
1057
|
+
io.observeTable('table1', cb);
|
|
1058
|
+
|
|
1059
|
+
//Write new data triggering the listener
|
|
1060
|
+
await io.write({
|
|
1061
|
+
data,
|
|
1062
|
+
});
|
|
1063
|
+
|
|
1064
|
+
//Check that the listener was called with the latest data only
|
|
1065
|
+
expect(cb).toHaveBeenCalledTimes(1);
|
|
1066
|
+
expect(cb).toHaveBeenCalledWith({ table1: hsh(data.table1) });
|
|
1067
|
+
});
|
|
1068
|
+
});
|
|
1069
|
+
|
|
1070
|
+
describe('unobserveTable(table, callback) and unobserveAll(table)', () => {
|
|
1071
|
+
it('should not call listener after unobserve', async () => {
|
|
1072
|
+
await createExampleTable('table1');
|
|
1073
|
+
|
|
1074
|
+
const cb = vi.fn();
|
|
1075
|
+
const data = {
|
|
1076
|
+
table1: {
|
|
1077
|
+
_type: 'components',
|
|
1078
|
+
_data: [{ a: 'a2' }],
|
|
1079
|
+
},
|
|
1080
|
+
} as Rljson;
|
|
1081
|
+
|
|
1082
|
+
//Subscribe to changes
|
|
1083
|
+
io.observeTable('table1', cb);
|
|
1084
|
+
|
|
1085
|
+
await io.write({
|
|
1086
|
+
data,
|
|
1087
|
+
});
|
|
1088
|
+
|
|
1089
|
+
//Unsubscribe
|
|
1090
|
+
io.unobserveTable('table1', cb);
|
|
1091
|
+
|
|
1092
|
+
await io.write({
|
|
1093
|
+
data,
|
|
1094
|
+
});
|
|
1095
|
+
|
|
1096
|
+
expect(cb).toHaveBeenCalledTimes(1);
|
|
1097
|
+
});
|
|
1098
|
+
});
|
|
1099
|
+
describe('unobserveAll(table)', () => {
|
|
1100
|
+
it('should not call listener after unobserve all', async () => {
|
|
1101
|
+
await createExampleTable('table1');
|
|
1102
|
+
|
|
1103
|
+
const cb = vi.fn();
|
|
1104
|
+
const data = {
|
|
1105
|
+
table1: {
|
|
1106
|
+
_type: 'components',
|
|
1107
|
+
_data: [{ a: 'a2' }],
|
|
1108
|
+
},
|
|
1109
|
+
} as Rljson;
|
|
1110
|
+
|
|
1111
|
+
//Subscribe to changes
|
|
1112
|
+
io.observeTable('table1', cb);
|
|
1113
|
+
|
|
1114
|
+
await io.write({
|
|
1115
|
+
data,
|
|
1116
|
+
});
|
|
1117
|
+
|
|
1118
|
+
//Unsubscribe all listeners from table1
|
|
1119
|
+
io.unobserveAll('table1');
|
|
1120
|
+
|
|
1121
|
+
await io.write({
|
|
1122
|
+
data,
|
|
1123
|
+
});
|
|
1124
|
+
|
|
1125
|
+
expect(cb).toHaveBeenCalledTimes(1);
|
|
1126
|
+
});
|
|
1127
|
+
});
|
|
1128
|
+
|
|
1129
|
+
describe('observers(table)', () => {
|
|
1130
|
+
it('get a list of all observers', async () => {
|
|
1131
|
+
await createExampleTable('table1');
|
|
1132
|
+
await createExampleTable('table2');
|
|
1133
|
+
|
|
1134
|
+
const cb1 = vi.fn();
|
|
1135
|
+
const cb2 = vi.fn();
|
|
1136
|
+
|
|
1137
|
+
//Subscribe to changes
|
|
1138
|
+
io.observeTable('table1', cb1);
|
|
1139
|
+
io.observeTable('table1', cb2);
|
|
1140
|
+
io.observeTable('table2', cb2);
|
|
1141
|
+
|
|
1142
|
+
const observers = io.observers('table1');
|
|
1143
|
+
expect(observers.length).toBe(2);
|
|
1144
|
+
expect(observers).toContain(cb1);
|
|
1145
|
+
expect(observers).toContain(cb2);
|
|
1146
|
+
|
|
1147
|
+
const observers2 = io.observers('table2');
|
|
1148
|
+
expect(observers2.length).toBe(1);
|
|
1149
|
+
expect(observers2).toContain(cb2);
|
|
1150
|
+
});
|
|
1151
|
+
});
|
|
950
1152
|
});
|
|
951
1153
|
};
|
|
952
1154
|
|
|
953
|
-
|
|
1155
|
+
// Run the tests for all setups
|
|
1156
|
+
for (const [setupName, setup] of Object.entries(setups)) {
|
|
1157
|
+
runIoConformanceTests(setupName, setup);
|
|
1158
|
+
}
|
package/dist/io-mem.d.ts
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { JsonValue } from '@rljson/json';
|
|
2
|
-
import { Rljson, TableCfg, TableKey } from '@rljson/rljson';
|
|
2
|
+
import { ContentType, Rljson, TableCfg, TableKey } from '@rljson/rljson';
|
|
3
3
|
import { Io } from './io.ts';
|
|
4
4
|
/**
|
|
5
5
|
* In-Memory implementation of the Rljson Io interface.
|
|
@@ -14,6 +14,9 @@ export declare class IoMem implements Io {
|
|
|
14
14
|
dumpTable(request: {
|
|
15
15
|
table: string;
|
|
16
16
|
}): Promise<Rljson>;
|
|
17
|
+
contentType(request: {
|
|
18
|
+
table: string;
|
|
19
|
+
}): Promise<ContentType>;
|
|
17
20
|
readRows(request: {
|
|
18
21
|
table: string;
|
|
19
22
|
where: {
|
|
@@ -42,7 +45,12 @@ export declare class IoMem implements Io {
|
|
|
42
45
|
private _extendTable;
|
|
43
46
|
private _dump;
|
|
44
47
|
private _dumpTable;
|
|
48
|
+
private _contentType;
|
|
45
49
|
private _write;
|
|
46
50
|
private _readRows;
|
|
47
51
|
_removeNullValues(rljson: Rljson): void;
|
|
52
|
+
observeTable(table: string, callback: (data: Rljson) => void): void;
|
|
53
|
+
unobserveTable(table: string, callback: (data: Rljson) => void): void;
|
|
54
|
+
unobserveAll(table: string): void;
|
|
55
|
+
observers(table: string): ((data: Rljson) => void)[];
|
|
48
56
|
}
|
|
@@ -0,0 +1,100 @@
|
|
|
1
|
+
import { JsonValue } from '@rljson/json';
|
|
2
|
+
import { ContentType, Rljson, TableCfg, TableKey } from '@rljson/rljson';
|
|
3
|
+
import { Io } from './io.ts';
|
|
4
|
+
import { Socket } from './socket.ts';
|
|
5
|
+
export declare class IoPeer implements Io {
|
|
6
|
+
private _socket;
|
|
7
|
+
isOpen: boolean;
|
|
8
|
+
private _ioTools;
|
|
9
|
+
constructor(_socket: Socket);
|
|
10
|
+
/**
|
|
11
|
+
*
|
|
12
|
+
* Initializes the Peer connection.
|
|
13
|
+
* @returns
|
|
14
|
+
*/
|
|
15
|
+
init(): Promise<void>;
|
|
16
|
+
/**
|
|
17
|
+
* Closes the Peer connection.
|
|
18
|
+
* @returns
|
|
19
|
+
*/
|
|
20
|
+
close(): Promise<void>;
|
|
21
|
+
/**
|
|
22
|
+
* Returns a promise that resolves once the Peer connection is ready.
|
|
23
|
+
* @returns
|
|
24
|
+
*/
|
|
25
|
+
isReady(): Promise<void>;
|
|
26
|
+
/**
|
|
27
|
+
* Dumps the entire database content.
|
|
28
|
+
* @returns A promise that resolves to the dumped database content.
|
|
29
|
+
*/
|
|
30
|
+
dump(): Promise<Rljson>;
|
|
31
|
+
/**
|
|
32
|
+
* Dumps a specific table from the database.
|
|
33
|
+
* @param request An object containing the table name to dump.
|
|
34
|
+
* @returns A promise that resolves to the dumped table data.
|
|
35
|
+
*/
|
|
36
|
+
dumpTable(request: {
|
|
37
|
+
table: string;
|
|
38
|
+
}): Promise<Rljson>;
|
|
39
|
+
/**
|
|
40
|
+
* Gets the content type of a specific table.
|
|
41
|
+
* @param request An object containing the table name to get the content type for.
|
|
42
|
+
* @returns A promise that resolves to the content type of the specified table.
|
|
43
|
+
*/
|
|
44
|
+
contentType(request: {
|
|
45
|
+
table: string;
|
|
46
|
+
}): Promise<ContentType>;
|
|
47
|
+
/**
|
|
48
|
+
* Checks if a specific table exists in the database.
|
|
49
|
+
* @param tableKey The key of the table to check for existence.
|
|
50
|
+
* @returns A promise that resolves to true if the table exists, false otherwise.
|
|
51
|
+
*/
|
|
52
|
+
tableExists(tableKey: TableKey): Promise<boolean>;
|
|
53
|
+
/**
|
|
54
|
+
* Creates or extends a table with the given configuration.
|
|
55
|
+
* @param request An object containing the table configuration.
|
|
56
|
+
* @returns A promise that resolves once the table is created or extended.
|
|
57
|
+
*/
|
|
58
|
+
createOrExtendTable(request: {
|
|
59
|
+
tableCfg: TableCfg;
|
|
60
|
+
}): Promise<void>;
|
|
61
|
+
/**
|
|
62
|
+
* Retrieves the raw table configurations from the database.
|
|
63
|
+
* @returns A promise that resolves to an array of table configurations.
|
|
64
|
+
*/
|
|
65
|
+
rawTableCfgs(): Promise<TableCfg[]>;
|
|
66
|
+
/**
|
|
67
|
+
* Writes data to the database.
|
|
68
|
+
* @param request An object containing the data to write.
|
|
69
|
+
* @returns A promise that resolves once the data is written.
|
|
70
|
+
*/
|
|
71
|
+
write(request: {
|
|
72
|
+
data: Rljson;
|
|
73
|
+
}): Promise<void>;
|
|
74
|
+
/**
|
|
75
|
+
* Reads rows from a specific table that match the given criteria.
|
|
76
|
+
* @param request An object containing the table name and the criteria for selecting rows.
|
|
77
|
+
* @returns A promise that resolves to the selected rows.
|
|
78
|
+
*/
|
|
79
|
+
readRows(request: {
|
|
80
|
+
table: string;
|
|
81
|
+
where: {
|
|
82
|
+
[column: string]: JsonValue | null;
|
|
83
|
+
};
|
|
84
|
+
}): Promise<Rljson>;
|
|
85
|
+
/**
|
|
86
|
+
* Retrieves the number of rows in a specific table.
|
|
87
|
+
* @param table The name of the table to count rows in.
|
|
88
|
+
* @returns A promise that resolves to the number of rows in the specified table.
|
|
89
|
+
*/
|
|
90
|
+
rowCount(table: string): Promise<number>;
|
|
91
|
+
/** Start observing changes on a specific table */
|
|
92
|
+
observeTable(table: string, callback: (data: Rljson) => void): Promise<void>;
|
|
93
|
+
/** Stop observing changes on a specific table */
|
|
94
|
+
unobserveTable(table: string, callback: (data: Rljson) => void): void;
|
|
95
|
+
/** Stop observing all changes on a specific table */
|
|
96
|
+
unobserveAll(table: string): void;
|
|
97
|
+
/** Returns all observers for a specific table */
|
|
98
|
+
observers(table: string): ((data: Rljson) => void)[];
|
|
99
|
+
static example: () => Promise<IoPeer>;
|
|
100
|
+
}
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
import { Io } from './io.ts';
|
|
2
|
+
import { Socket } from './socket.ts';
|
|
3
|
+
export declare class IoServer {
|
|
4
|
+
private readonly _io;
|
|
5
|
+
private _sockets;
|
|
6
|
+
constructor(_io: Io);
|
|
7
|
+
/**
|
|
8
|
+
* Adds a socket to the IoServer instance.
|
|
9
|
+
* @param socket - The socket to add.
|
|
10
|
+
*/
|
|
11
|
+
addSocket(socket: Socket): Promise<void>;
|
|
12
|
+
/**
|
|
13
|
+
* Removes a transport layer from the given socket.
|
|
14
|
+
* @param socket - The socket to remove the transport layer from.
|
|
15
|
+
*/
|
|
16
|
+
removeSocket(socket: Socket): void;
|
|
17
|
+
/**
|
|
18
|
+
* Adds a transport layer to the given socket.
|
|
19
|
+
* @param socket - The socket to add the transport layer to.
|
|
20
|
+
*/
|
|
21
|
+
private _addTransportLayer;
|
|
22
|
+
/**
|
|
23
|
+
* Creates or extends a table with the given configuration.
|
|
24
|
+
* Also sets up observation for the new or extended table on all connected sockets.
|
|
25
|
+
* @param request - An object containing the table configuration.
|
|
26
|
+
*/
|
|
27
|
+
private createOrExtendTable;
|
|
28
|
+
/**
|
|
29
|
+
* Generates a transport layer object for the given Io instance.
|
|
30
|
+
* @param io - The Io instance to generate the transport layer for.
|
|
31
|
+
* @returns An object containing methods that correspond to the Io interface.
|
|
32
|
+
*/
|
|
33
|
+
private _generateTransportLayerCRUD;
|
|
34
|
+
}
|
package/dist/io-tools.d.ts
CHANGED
|
@@ -1,10 +1,12 @@
|
|
|
1
1
|
import { Rljson, TableCfg, TableKey, TableType } from '@rljson/rljson';
|
|
2
2
|
import { Io } from './io.ts';
|
|
3
|
+
export type IoObserver = (data: Rljson) => void;
|
|
3
4
|
/**
|
|
4
5
|
* Provides utility functions for the Io interface.
|
|
5
6
|
*/
|
|
6
7
|
export declare class IoTools {
|
|
7
8
|
readonly io: Io;
|
|
9
|
+
private _observers;
|
|
8
10
|
/**
|
|
9
11
|
* Constructor
|
|
10
12
|
* @param io The Io interface to use
|
|
@@ -54,6 +56,37 @@ export declare class IoTools {
|
|
|
54
56
|
* Returns a list of all column names of a given table
|
|
55
57
|
*/
|
|
56
58
|
allColumnKeys(table: TableKey): Promise<string[]>;
|
|
59
|
+
/**
|
|
60
|
+
* Observes changes to a given table
|
|
61
|
+
* @param table - The table to observe
|
|
62
|
+
* @param observer - The observer function to call on changes
|
|
63
|
+
* @returns
|
|
64
|
+
*/
|
|
65
|
+
observeTable(table: string, observer: IoObserver): void;
|
|
66
|
+
/**
|
|
67
|
+
* Stops observing changes to a given table
|
|
68
|
+
* @param table - The table to stop observing
|
|
69
|
+
* @param observer - The observer function to remove
|
|
70
|
+
* @returns
|
|
71
|
+
*/
|
|
72
|
+
unobserveTable(table: string, observer: IoObserver): void;
|
|
73
|
+
/**
|
|
74
|
+
* Stops observing all tables or all changes on a specific table
|
|
75
|
+
* @param table - The table to stop observing or undefined to stop observing all tables
|
|
76
|
+
*/
|
|
77
|
+
unobserveAll(table?: string): void;
|
|
78
|
+
/**
|
|
79
|
+
* Returns all observers of a given table
|
|
80
|
+
* @param table - The table to get observers for
|
|
81
|
+
* @returns A list of observers for the given table
|
|
82
|
+
*/
|
|
83
|
+
observers(table: string): IoObserver[];
|
|
84
|
+
/**
|
|
85
|
+
* Notifies all observers of a given table
|
|
86
|
+
* @param table - The table to notify observers of
|
|
87
|
+
* @param data - The data to pass to the observers
|
|
88
|
+
*/
|
|
89
|
+
notifyObservers(table: string, data: Rljson): void;
|
|
57
90
|
/**
|
|
58
91
|
* Throws when a column does not exist in a given table
|
|
59
92
|
* @param table - The table to check
|
package/dist/io.d.ts
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { JsonValue } from '@rljson/json';
|
|
2
|
-
import { Rljson, TableCfg, TableKey } from '@rljson/rljson';
|
|
2
|
+
import { ContentType, Rljson, TableCfg, TableKey } from '@rljson/rljson';
|
|
3
3
|
export interface Io {
|
|
4
4
|
/** Starts the initialization */
|
|
5
5
|
init(): Promise<void>;
|
|
@@ -18,6 +18,9 @@ export interface Io {
|
|
|
18
18
|
dumpTable(request: {
|
|
19
19
|
table: string;
|
|
20
20
|
}): Promise<Rljson>;
|
|
21
|
+
contentType(request: {
|
|
22
|
+
table: string;
|
|
23
|
+
}): Promise<ContentType>;
|
|
21
24
|
/**
|
|
22
25
|
* Returns true if the table exists
|
|
23
26
|
*/
|
|
@@ -49,5 +52,13 @@ export interface Io {
|
|
|
49
52
|
}): Promise<Rljson>;
|
|
50
53
|
/** Returns the number of rows in the given table */
|
|
51
54
|
rowCount(table: string): Promise<number>;
|
|
55
|
+
/** Observe changes on a specific table */
|
|
56
|
+
observeTable(table: string, callback: (data: Rljson) => void): void;
|
|
57
|
+
/** Stop observing changes on a specific table */
|
|
58
|
+
unobserveTable(table: string, callback: (data: Rljson) => void): void;
|
|
59
|
+
/** Stop observing all changes on a specific table */
|
|
60
|
+
unobserveAll(table?: string): void;
|
|
61
|
+
/** Returns all observers for a specific table */
|
|
62
|
+
observers(table: string): ((data: Rljson) => void)[];
|
|
52
63
|
}
|
|
53
64
|
export declare const exampleIo = "Checkout @rljson/io-mem for an example implementation";
|
package/dist/io.js
CHANGED
|
@@ -11,6 +11,7 @@ class IoTools {
|
|
|
11
11
|
constructor(io) {
|
|
12
12
|
this.io = io;
|
|
13
13
|
}
|
|
14
|
+
_observers = /* @__PURE__ */ new Map();
|
|
14
15
|
/**
|
|
15
16
|
* Returns the table configuration of the tableCfgs table.
|
|
16
17
|
*/
|
|
@@ -154,6 +155,66 @@ class IoTools {
|
|
|
154
155
|
const result = tableCfg.columns.map((column) => column.key);
|
|
155
156
|
return result;
|
|
156
157
|
}
|
|
158
|
+
/**
|
|
159
|
+
* Observes changes to a given table
|
|
160
|
+
* @param table - The table to observe
|
|
161
|
+
* @param observer - The observer function to call on changes
|
|
162
|
+
* @returns
|
|
163
|
+
*/
|
|
164
|
+
observeTable(table, observer) {
|
|
165
|
+
if (!this._observers.has(table)) {
|
|
166
|
+
this._observers.set(table, []);
|
|
167
|
+
}
|
|
168
|
+
this._observers.get(table).push(observer);
|
|
169
|
+
}
|
|
170
|
+
/**
|
|
171
|
+
* Stops observing changes to a given table
|
|
172
|
+
* @param table - The table to stop observing
|
|
173
|
+
* @param observer - The observer function to remove
|
|
174
|
+
* @returns
|
|
175
|
+
*/
|
|
176
|
+
unobserveTable(table, observer) {
|
|
177
|
+
if (!this._observers.has(table)) return;
|
|
178
|
+
const observers = this._observers.get(table);
|
|
179
|
+
const index = observers.indexOf(observer);
|
|
180
|
+
if (index !== -1) {
|
|
181
|
+
observers.splice(index, 1);
|
|
182
|
+
}
|
|
183
|
+
if (observers.length === 0) {
|
|
184
|
+
this._observers.delete(table);
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
/**
|
|
188
|
+
* Stops observing all tables or all changes on a specific table
|
|
189
|
+
* @param table - The table to stop observing or undefined to stop observing all tables
|
|
190
|
+
*/
|
|
191
|
+
unobserveAll(table) {
|
|
192
|
+
if (table) {
|
|
193
|
+
this._observers.delete(table);
|
|
194
|
+
return;
|
|
195
|
+
}
|
|
196
|
+
this._observers.clear();
|
|
197
|
+
}
|
|
198
|
+
/**
|
|
199
|
+
* Returns all observers of a given table
|
|
200
|
+
* @param table - The table to get observers for
|
|
201
|
+
* @returns A list of observers for the given table
|
|
202
|
+
*/
|
|
203
|
+
observers(table) {
|
|
204
|
+
return this._observers.get(table) ?? [];
|
|
205
|
+
}
|
|
206
|
+
/**
|
|
207
|
+
* Notifies all observers of a given table
|
|
208
|
+
* @param table - The table to notify observers of
|
|
209
|
+
* @param data - The data to pass to the observers
|
|
210
|
+
*/
|
|
211
|
+
notifyObservers(table, data) {
|
|
212
|
+
if (!this._observers.has(table)) return;
|
|
213
|
+
const observers = this._observers.get(table);
|
|
214
|
+
for (const observer of observers) {
|
|
215
|
+
observer(data);
|
|
216
|
+
}
|
|
217
|
+
}
|
|
157
218
|
/**
|
|
158
219
|
* Throws when a column does not exist in a given table
|
|
159
220
|
* @param table - The table to check
|
|
@@ -290,6 +351,11 @@ class IoMem {
|
|
|
290
351
|
return this._dumpTable(request);
|
|
291
352
|
}
|
|
292
353
|
// ...........................................................................
|
|
354
|
+
// Meta Data
|
|
355
|
+
async contentType(request) {
|
|
356
|
+
return this._contentType(request);
|
|
357
|
+
}
|
|
358
|
+
// ...........................................................................
|
|
293
359
|
// Rows
|
|
294
360
|
readRows(request) {
|
|
295
361
|
return this._readRows(request);
|
|
@@ -410,6 +476,11 @@ class IoMem {
|
|
|
410
476
|
};
|
|
411
477
|
}
|
|
412
478
|
// ...........................................................................
|
|
479
|
+
async _contentType(request) {
|
|
480
|
+
await this._ioTools.throwWhenTableDoesNotExist(request.table);
|
|
481
|
+
return this._mem[request.table]._type;
|
|
482
|
+
}
|
|
483
|
+
// ...........................................................................
|
|
413
484
|
async _write(request) {
|
|
414
485
|
const addedData = hsh(request.data);
|
|
415
486
|
this._removeNullValues(addedData);
|
|
@@ -431,6 +502,7 @@ class IoMem {
|
|
|
431
502
|
}
|
|
432
503
|
}
|
|
433
504
|
this._ioTools.sortTableDataAndUpdateHash(oldTable);
|
|
505
|
+
this._ioTools.notifyObservers(table, { [table]: newTable });
|
|
434
506
|
}
|
|
435
507
|
this._updateGlobalHash();
|
|
436
508
|
}
|
|
@@ -477,6 +549,18 @@ class IoMem {
|
|
|
477
549
|
}
|
|
478
550
|
});
|
|
479
551
|
}
|
|
552
|
+
observeTable(table, callback) {
|
|
553
|
+
this._ioTools.observeTable(table, callback);
|
|
554
|
+
}
|
|
555
|
+
unobserveTable(table, callback) {
|
|
556
|
+
this._ioTools.unobserveTable(table, callback);
|
|
557
|
+
}
|
|
558
|
+
unobserveAll(table) {
|
|
559
|
+
this._ioTools.unobserveAll(table);
|
|
560
|
+
}
|
|
561
|
+
observers(table) {
|
|
562
|
+
return this._ioTools.observers(table);
|
|
563
|
+
}
|
|
480
564
|
}
|
|
481
565
|
// @license
|
|
482
566
|
const exampleTestSetup = () => {
|
|
@@ -602,4 +686,4 @@ export {
|
|
|
602
686
|
exampleIo,
|
|
603
687
|
exampleTestSetup
|
|
604
688
|
};
|
|
605
|
-
//# 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\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        { key: '_hash', type: 'string' },\n        { key: 'key', type: 'string' },\n        { key: 'type', type: 'string' },\n        { key: 'isHead', type: 'boolean' },\n        { key: 'isRoot', type: 'boolean' },\n        { key: 'isShared', type: 'boolean' },\n        { key: 'previous', type: 'string' },\n        { key: 'columns', type: 'jsonArray' },\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        { key: '_hash', type: 'string' },\n        { key: 'table', type: 'string' },\n        { key: 'predecessor', type: 'string' },\n        { key: 'successor', type: 'string' },\n        { key: 'timestamp', type: 'number' },\n        { key: 'id', type: 'string' },\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      /* v8 ignore start */\n      if (!existing || existing.columns.length < table.columns.length) {\n        newestVersion[table.key] = table;\n      }\n      /* v8 ignore end */\n    }\n\n    // Sort the tables by key\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        /* v8 ignore start */\n      }\n\n      return 0;\n      /* v8 ignore end */\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        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      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      if (hashA < hashB) {\n        return -1;\n      }\n      if (hashA > hashB) {\n        return 1;\n        /* v8 ignore start */\n      }\n\n      return 0;\n      /* v8 ignore end */\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 { iterateTablesSync, Rljson, TableCfg, TableKey, TableType } from '@rljson/rljson';\n\nimport { IoTools } from './io-tools.ts';\nimport { Io } from './io.ts';\n\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  // 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 _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 { Rljson, TableCfg, TableKey } from '@rljson/rljson';\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  // 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","// @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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"}
|
|
689
|
+
//# 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  private _observers: Map<string, IoObserver[]> = new Map();\n\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        { key: '_hash', type: 'string' },\n        { key: 'key', type: 'string' },\n        { key: 'type', type: 'string' },\n        { key: 'isHead', type: 'boolean' },\n        { key: 'isRoot', type: 'boolean' },\n        { key: 'isShared', type: 'boolean' },\n        { key: 'previous', type: 'string' },\n        { key: 'columns', type: 'jsonArray' },\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        { key: '_hash', type: 'string' },\n        { key: 'table', type: 'string' },\n        { key: 'predecessor', type: 'string' },\n        { key: 'successor', type: 'string' },\n        { key: 'timestamp', type: 'number' },\n        { key: 'id', type: 'string' },\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      /* v8 ignore start */\n      if (!existing || existing.columns.length < table.columns.length) {\n        newestVersion[table.key] = table;\n      }\n      /* v8 ignore end */\n    }\n\n    // Sort the tables by key\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        /* v8 ignore start */\n      }\n\n      return 0;\n      /* v8 ignore end */\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   * Observes changes to a given table\n   * @param table - The table to observe\n   * @param observer - The observer function to call on changes\n   * @returns\n   */\n  observeTable(table: string, observer: IoObserver): void {\n    if (!this._observers.has(table)) {\n      this._observers.set(table, []);\n    }\n    this._observers.get(table)!.push(observer);\n  }\n\n  /**\n   * Stops observing changes to a given table\n   * @param table - The table to stop observing\n   * @param observer - The observer function to remove\n   * @returns\n   */\n  unobserveTable(table: string, observer: IoObserver): void {\n    if (!this._observers.has(table)) return;\n    const observers = this._observers.get(table)!;\n    const index = observers.indexOf(observer);\n    if (index !== -1) {\n      observers.splice(index, 1);\n    }\n    if (observers.length === 0) {\n      this._observers.delete(table);\n    }\n  }\n\n  /**\n   * Stops observing all tables or all changes on a specific table\n   * @param table - The table to stop observing or undefined to stop observing all tables\n   */\n  unobserveAll(table?: string) {\n    if (table) {\n      this._observers.delete(table);\n      return;\n    }\n    this._observers.clear();\n  }\n\n  /**\n   * Returns all observers of a given table\n   * @param table - The table to get observers for\n   * @returns A list of observers for the given table\n   */\n\n  observers(table: string): IoObserver[] {\n    return this._observers.get(table) ?? [];\n  }\n\n  /**\n   * Notifies all observers of a given table\n   * @param table - The table to notify observers of\n   * @param data - The data to pass to the observers\n   */\n  notifyObservers(table: string, data: Rljson): void {\n    if (!this._observers.has(table)) return;\n    const observers = this._observers.get(table)!;\n    for (const observer of observers) {\n      observer(data);\n    }\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        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      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      if (hashA < hashB) {\n        return -1;\n      }\n      if (hashA > hashB) {\n        return 1;\n        /* v8 ignore start */\n      }\n\n      return 0;\n      /* v8 ignore end */\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      // Notify observers\n      this._ioTools.notifyObservers(table, { [table]: newTable } as Rljson);\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  observeTable(table: string, callback: (data: Rljson) => void) {\n    this._ioTools.observeTable(table, callback);\n  }\n\n  unobserveTable(table: string, callback: (data: Rljson) => void) {\n    this._ioTools.unobserveTable(table, callback);\n  }\n\n  unobserveAll(table: string) {\n    this._ioTools.unobserveAll(table);\n  }\n\n  observers(table: string): ((data: Rljson) => void)[] {\n    return this._ioTools.observers(table);\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// .............................................................................\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  // Observing\n\n  /** Observe changes on a specific table */\n  observeTable(table: string, callback: (data: Rljson) => void): void;\n\n  /** Stop observing changes on a specific table */\n  unobserveTable(table: string, callback: (data: Rljson) => void): void;\n\n  /** Stop observing all changes on a specific table */\n  unobserveAll(table?: string): void;\n\n  /** Returns all observers for a specific table */\n  observers(table: string): ((data: Rljson) => void)[];\n}\n\n// .............................................................................\nexport const exampleIo =\n  'Checkout @rljson/io-mem for an example implementation';\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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"}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import { default as EventEmitter } from 'node:events';
|
|
2
|
+
export declare class PeerServerSocketMock extends EventEmitter {
|
|
3
|
+
connected: boolean;
|
|
4
|
+
disconnected: boolean;
|
|
5
|
+
constructor();
|
|
6
|
+
/**
|
|
7
|
+
* Simulates a connection event.
|
|
8
|
+
*
|
|
9
|
+
* Emits the 'connect' event to all registered listeners.
|
|
10
|
+
*/
|
|
11
|
+
connect(): void;
|
|
12
|
+
/**
|
|
13
|
+
* Simulates a disconnection event.
|
|
14
|
+
*
|
|
15
|
+
* Emits the 'disconnect' event to all registered listeners.
|
|
16
|
+
*/
|
|
17
|
+
disconnect(): void;
|
|
18
|
+
}
|
|
@@ -0,0 +1,73 @@
|
|
|
1
|
+
import { Io } from './io.ts';
|
|
2
|
+
import { Socket } from './socket.ts';
|
|
3
|
+
export declare class PeerSocketMock implements Socket {
|
|
4
|
+
private _io;
|
|
5
|
+
private _onceListeners;
|
|
6
|
+
connected: boolean;
|
|
7
|
+
disconnected: boolean;
|
|
8
|
+
constructor(_io: Io);
|
|
9
|
+
/**
|
|
10
|
+
* Simulates a connection event.
|
|
11
|
+
*
|
|
12
|
+
* Emits the 'connect' event to all registered listeners.
|
|
13
|
+
*/
|
|
14
|
+
connect(): void;
|
|
15
|
+
/**
|
|
16
|
+
* Simulates a disconnection event.
|
|
17
|
+
*
|
|
18
|
+
* Emits the 'disconnect' event to all registered listeners.
|
|
19
|
+
*/
|
|
20
|
+
disconnect(): void;
|
|
21
|
+
/**
|
|
22
|
+
* Adds a listener for the specified event.
|
|
23
|
+
* @param eventName - The name of the event to listen for.
|
|
24
|
+
* @param listener - The callback function to invoke when the event is emitted.
|
|
25
|
+
* @returns The current PeerSocketMock instance.
|
|
26
|
+
*/
|
|
27
|
+
addListener(eventName: string | symbol, listener: (...args: any[]) => void): this;
|
|
28
|
+
/**
|
|
29
|
+
* Alias for addListener method.
|
|
30
|
+
* @param eventName - The name of the event to listen for.
|
|
31
|
+
* @param listener - The callback function to invoke when the event is emitted.
|
|
32
|
+
* @returns The current PeerSocketMock instance.
|
|
33
|
+
*/
|
|
34
|
+
on(eventName: string | symbol, listener: (...args: any[]) => void): this;
|
|
35
|
+
/**
|
|
36
|
+
* Adds a one-time listener for the specified event.
|
|
37
|
+
* The listener is invoked only the next time the event is emitted, after which it is removed.
|
|
38
|
+
* @param eventName - The name of the event to listen for.
|
|
39
|
+
* @param listener - The callback function to invoke when the event is emitted.
|
|
40
|
+
* @returns The current PeerSocketMock instance.
|
|
41
|
+
*/
|
|
42
|
+
once(eventName: string | symbol, listener: (...args: any[]) => void): this;
|
|
43
|
+
/**
|
|
44
|
+
* Removes a listener for the specified event.
|
|
45
|
+
* @param eventName - The name of the event to stop listening for.
|
|
46
|
+
* @param listener - The callback function to remove.
|
|
47
|
+
* @returns The current PeerSocketMock instance.
|
|
48
|
+
*/
|
|
49
|
+
removeListener(eventName: string | symbol, listener: (...args: any[]) => void): this;
|
|
50
|
+
/**
|
|
51
|
+
* Alias for removeListener method.
|
|
52
|
+
* @param eventName - The name of the event to stop listening for.
|
|
53
|
+
* @param listener - The callback function to remove.
|
|
54
|
+
* @returns The current PeerSocketMock instance.
|
|
55
|
+
*/
|
|
56
|
+
off(eventName: string | symbol, listener: (...args: any[]) => void): this;
|
|
57
|
+
/**
|
|
58
|
+
* Removes all listeners for the specified event.
|
|
59
|
+
* If no event is specified, all listeners for all events are removed.
|
|
60
|
+
* @param eventName - The name of the event to stop listening for (optional).
|
|
61
|
+
* @returns The current PeerSocketMock instance.
|
|
62
|
+
*/
|
|
63
|
+
removeAllListeners(eventName?: string | symbol | undefined): this;
|
|
64
|
+
listeners(eventName: string | symbol): ((data: import('@rljson/rljson').Rljson) => void)[];
|
|
65
|
+
rawListeners(eventName: string | symbol): ((data: import('@rljson/rljson').Rljson) => void)[];
|
|
66
|
+
/**
|
|
67
|
+
* Emits an event, invoking the corresponding method on the Io instance.
|
|
68
|
+
* @param eventName - The name of the event to emit.
|
|
69
|
+
* @param args - The arguments to pass to the event listener.
|
|
70
|
+
* @returns
|
|
71
|
+
*/
|
|
72
|
+
emit(eventName: string | symbol, ...args: any[]): boolean;
|
|
73
|
+
}
|
package/dist/socket.d.ts
ADDED
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
export interface Socket {
|
|
2
|
+
connected: boolean;
|
|
3
|
+
disconnected: boolean;
|
|
4
|
+
connect(): void;
|
|
5
|
+
disconnect(): void;
|
|
6
|
+
addListener(eventName: string | symbol, listener: (...args: any[]) => void): this;
|
|
7
|
+
on(eventName: string | symbol, listener: (...args: any[]) => void): this;
|
|
8
|
+
once(eventName: string | symbol, listener: (...args: any[]) => void): this;
|
|
9
|
+
removeListener(eventName: string | symbol, listener: (...args: any[]) => void): this;
|
|
10
|
+
off(eventName: string | symbol, listener: (...args: any[]) => void): this;
|
|
11
|
+
removeAllListeners(event?: string | symbol): this;
|
|
12
|
+
listeners(eventName: string | symbol): ((...args: any[]) => void)[];
|
|
13
|
+
rawListeners(eventName: string | symbol): ((...args: any[]) => void)[];
|
|
14
|
+
emit(eventName: string | symbol, ...args: any[]): boolean;
|
|
15
|
+
}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@rljson/io",
|
|
3
|
-
"version": "0.0.
|
|
3
|
+
"version": "0.0.46",
|
|
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",
|
|
@@ -20,20 +20,20 @@
|
|
|
20
20
|
],
|
|
21
21
|
"type": "module",
|
|
22
22
|
"devDependencies": {
|
|
23
|
-
"@types/node": "^24.
|
|
24
|
-
"@typescript-eslint/eslint-plugin": "^8.
|
|
25
|
-
"@typescript-eslint/parser": "^8.
|
|
23
|
+
"@types/node": "^24.8.1",
|
|
24
|
+
"@typescript-eslint/eslint-plugin": "^8.46.1",
|
|
25
|
+
"@typescript-eslint/parser": "^8.46.1",
|
|
26
26
|
"@vitest/coverage-v8": "^3.2.4",
|
|
27
|
-
"cross-env": "^10.
|
|
28
|
-
"eslint": "^9.
|
|
29
|
-
"eslint-plugin-jsdoc": "^
|
|
27
|
+
"cross-env": "^10.1.0",
|
|
28
|
+
"eslint": "^9.38.0",
|
|
29
|
+
"eslint-plugin-jsdoc": "^61.1.4",
|
|
30
30
|
"eslint-plugin-tsdoc": "^0.4.0",
|
|
31
31
|
"globals": "^16.4.0",
|
|
32
|
-
"jsdoc": "^4.0.
|
|
32
|
+
"jsdoc": "^4.0.5",
|
|
33
33
|
"read-pkg": "^9.0.1",
|
|
34
|
-
"typescript": "~5.9.
|
|
35
|
-
"typescript-eslint": "^8.
|
|
36
|
-
"vite": "^7.1.
|
|
34
|
+
"typescript": "~5.9.3",
|
|
35
|
+
"typescript-eslint": "^8.46.1",
|
|
36
|
+
"vite": "^7.1.11",
|
|
37
37
|
"vite-node": "^3.2.4",
|
|
38
38
|
"vite-plugin-dts": "^4.5.4",
|
|
39
39
|
"vite-tsconfig-paths": "^5.1.4",
|
|
@@ -43,8 +43,8 @@
|
|
|
43
43
|
"dependencies": {
|
|
44
44
|
"@rljson/hash": "^0.0.16",
|
|
45
45
|
"@rljson/is-ready": "^0.0.17",
|
|
46
|
-
"@rljson/json": "^0.0.
|
|
47
|
-
"@rljson/rljson": "^0.0.
|
|
46
|
+
"@rljson/json": "^0.0.22",
|
|
47
|
+
"@rljson/rljson": "^0.0.62",
|
|
48
48
|
"@rljson/validate": "^0.0.11"
|
|
49
49
|
},
|
|
50
50
|
"scripts": {
|