@rljson/io 0.0.62 → 0.0.63
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/tableCfgs-1.json +3 -157
- package/dist/conformance-tests/io-conformance.spec.ts +5 -4
- package/dist/index.d.ts +2 -0
- package/dist/io-multi.d.ts +5 -4
- package/dist/io-peer-bridge.d.ts +75 -0
- package/dist/io.js +391 -52
- package/dist/io.js.map +1 -0
- package/dist/peer-socket-mock.d.ts +13 -0
- package/dist/socket.d.ts +3 -1
- package/package.json +10 -10
package/dist/io.js
CHANGED
|
@@ -608,6 +608,36 @@ class PeerSocketMock {
|
|
|
608
608
|
connected = false;
|
|
609
609
|
disconnected = true;
|
|
610
610
|
// ............................................................................
|
|
611
|
+
/**
|
|
612
|
+
* Removes a specific listener for the specified event.
|
|
613
|
+
* @param eventName - The name of the event.
|
|
614
|
+
* @param listener - The callback function to remove.
|
|
615
|
+
* @returns The PeerSocketMock instance for chaining.
|
|
616
|
+
*/
|
|
617
|
+
off(eventName, listener) {
|
|
618
|
+
const listeners = this._listenersMap.get(eventName) || [];
|
|
619
|
+
const index = listeners.indexOf(listener);
|
|
620
|
+
if (index !== -1) {
|
|
621
|
+
listeners.splice(index, 1);
|
|
622
|
+
this._listenersMap.set(eventName, listeners);
|
|
623
|
+
}
|
|
624
|
+
return this;
|
|
625
|
+
}
|
|
626
|
+
// ............................................................................
|
|
627
|
+
/**
|
|
628
|
+
* Removes all listeners for the specified event, or all listeners if no event is specified.
|
|
629
|
+
* @param eventName - (Optional) The name of the event.
|
|
630
|
+
* @returns The PeerSocketMock instance for chaining.
|
|
631
|
+
*/
|
|
632
|
+
removeAllListeners(eventName) {
|
|
633
|
+
if (eventName) {
|
|
634
|
+
this._listenersMap.delete(eventName);
|
|
635
|
+
} else {
|
|
636
|
+
this._listenersMap.clear();
|
|
637
|
+
}
|
|
638
|
+
return this;
|
|
639
|
+
}
|
|
640
|
+
// ............................................................................
|
|
611
641
|
/**
|
|
612
642
|
* Registers an event listener for the specified event.
|
|
613
643
|
* @param eventName - The name of the event to listen for.
|
|
@@ -689,7 +719,16 @@ class IoPeer {
|
|
|
689
719
|
this.isOpen = false;
|
|
690
720
|
});
|
|
691
721
|
this._socket.connect();
|
|
692
|
-
return
|
|
722
|
+
return new Promise((resolve) => {
|
|
723
|
+
if (this._socket.connected) {
|
|
724
|
+
this.isOpen = true;
|
|
725
|
+
resolve();
|
|
726
|
+
} else {
|
|
727
|
+
this._socket.on("connect", () => {
|
|
728
|
+
resolve();
|
|
729
|
+
});
|
|
730
|
+
}
|
|
731
|
+
});
|
|
693
732
|
}
|
|
694
733
|
// ...........................................................................
|
|
695
734
|
/**
|
|
@@ -698,8 +737,12 @@ class IoPeer {
|
|
|
698
737
|
*/
|
|
699
738
|
async close() {
|
|
700
739
|
if (!this._socket.connected) return;
|
|
701
|
-
|
|
702
|
-
|
|
740
|
+
return new Promise((resolve) => {
|
|
741
|
+
this._socket.on("disconnect", () => {
|
|
742
|
+
resolve();
|
|
743
|
+
});
|
|
744
|
+
this._socket.disconnect();
|
|
745
|
+
});
|
|
703
746
|
}
|
|
704
747
|
// ...........................................................................
|
|
705
748
|
/**
|
|
@@ -707,7 +750,9 @@ class IoPeer {
|
|
|
707
750
|
* @returns
|
|
708
751
|
*/
|
|
709
752
|
async isReady() {
|
|
710
|
-
|
|
753
|
+
if (!!this._socket && this._socket.connected === true) this.isOpen = true;
|
|
754
|
+
else this.isOpen = false;
|
|
755
|
+
return !!this.isOpen ? Promise.resolve() : Promise.reject();
|
|
711
756
|
}
|
|
712
757
|
// ...........................................................................
|
|
713
758
|
/**
|
|
@@ -863,12 +908,14 @@ class IoMulti {
|
|
|
863
908
|
* @returns
|
|
864
909
|
*/
|
|
865
910
|
async init() {
|
|
866
|
-
for (
|
|
867
|
-
|
|
911
|
+
for (let idx = 0; idx < this._ios.length; idx++) {
|
|
912
|
+
const { io } = this._ios[idx];
|
|
913
|
+
if (io.isOpen === false) {
|
|
868
914
|
throw new Error(
|
|
869
915
|
"All underlying Io instances must be initialized before initializing IoMulti"
|
|
870
916
|
);
|
|
871
917
|
}
|
|
918
|
+
this._ios[idx] = { ...this._ios[idx], id: `io-${idx}` };
|
|
872
919
|
}
|
|
873
920
|
this.isOpen = true;
|
|
874
921
|
return Promise.resolve();
|
|
@@ -900,10 +947,10 @@ class IoMulti {
|
|
|
900
947
|
*/
|
|
901
948
|
async dump() {
|
|
902
949
|
if (this.dumpables.length === 0) {
|
|
903
|
-
|
|
950
|
+
throw new Error("No dumpable Io available");
|
|
904
951
|
}
|
|
905
952
|
const dumps = await Promise.all(
|
|
906
|
-
this.dumpables.map((dumpable) => dumpable.dump())
|
|
953
|
+
this.dumpables.map(({ io: dumpable }) => dumpable.dump())
|
|
907
954
|
);
|
|
908
955
|
return merge(...dumps);
|
|
909
956
|
}
|
|
@@ -915,21 +962,21 @@ class IoMulti {
|
|
|
915
962
|
*/
|
|
916
963
|
async dumpTable(request) {
|
|
917
964
|
if (this.dumpables.length === 0) {
|
|
918
|
-
|
|
965
|
+
throw new Error("No dumpable Io available");
|
|
919
966
|
}
|
|
920
967
|
const dumps = [];
|
|
921
|
-
for (const dumpable of this.dumpables) {
|
|
922
|
-
|
|
923
|
-
if (tableExists) {
|
|
968
|
+
for (const { io: dumpable } of this.dumpables) {
|
|
969
|
+
try {
|
|
924
970
|
const dump = await dumpable.dumpTable(request);
|
|
925
971
|
dumps.push(dump);
|
|
972
|
+
} catch {
|
|
973
|
+
continue;
|
|
926
974
|
}
|
|
927
975
|
}
|
|
928
|
-
if (dumps.length
|
|
929
|
-
|
|
930
|
-
} else {
|
|
931
|
-
return Promise.reject(new Error(`Table "${request.table}" not found`));
|
|
976
|
+
if (dumps.length === 0) {
|
|
977
|
+
throw new Error(`Table "${request.table}" not found`);
|
|
932
978
|
}
|
|
979
|
+
return merge(...dumps);
|
|
933
980
|
}
|
|
934
981
|
// ...........................................................................
|
|
935
982
|
/**
|
|
@@ -939,15 +986,12 @@ class IoMulti {
|
|
|
939
986
|
*/
|
|
940
987
|
async contentType(request) {
|
|
941
988
|
if (this.readables.length === 0) {
|
|
942
|
-
|
|
989
|
+
throw new Error("No readable Io available");
|
|
943
990
|
}
|
|
944
|
-
for (const readable of this.readables) {
|
|
945
|
-
|
|
946
|
-
if (tableExists) {
|
|
947
|
-
return readable.contentType(request);
|
|
948
|
-
}
|
|
991
|
+
for (const { io: readable } of this.readables) {
|
|
992
|
+
return readable.contentType(request);
|
|
949
993
|
}
|
|
950
|
-
|
|
994
|
+
throw new Error(`Table "${request.table}" not found`);
|
|
951
995
|
}
|
|
952
996
|
// ...........................................................................
|
|
953
997
|
/**
|
|
@@ -957,9 +1001,10 @@ class IoMulti {
|
|
|
957
1001
|
*/
|
|
958
1002
|
async tableExists(tableKey) {
|
|
959
1003
|
if (this.readables.length === 0) {
|
|
960
|
-
|
|
1004
|
+
throw new Error("No readable Io available");
|
|
961
1005
|
}
|
|
962
|
-
for (
|
|
1006
|
+
for (let i = 0; i < this.readables.length; i++) {
|
|
1007
|
+
const readable = this.readables[i].io;
|
|
963
1008
|
const exists = await readable.tableExists(tableKey);
|
|
964
1009
|
if (exists) {
|
|
965
1010
|
return true;
|
|
@@ -975,10 +1020,10 @@ class IoMulti {
|
|
|
975
1020
|
*/
|
|
976
1021
|
createOrExtendTable(request) {
|
|
977
1022
|
if (this.writables.length === 0) {
|
|
978
|
-
|
|
1023
|
+
throw new Error("No writable Io available");
|
|
979
1024
|
}
|
|
980
1025
|
const creations = this.writables.map(
|
|
981
|
-
(writable) => writable.createOrExtendTable(request)
|
|
1026
|
+
({ io: writable }) => writable.createOrExtendTable(request)
|
|
982
1027
|
);
|
|
983
1028
|
return Promise.all(creations).then(() => Promise.resolve());
|
|
984
1029
|
}
|
|
@@ -987,12 +1032,22 @@ class IoMulti {
|
|
|
987
1032
|
* Retrieves the raw table configurations from the highest priority underlying readable Io instance.
|
|
988
1033
|
* @returns A promise that resolves to an array of table configurations.
|
|
989
1034
|
*/
|
|
990
|
-
rawTableCfgs() {
|
|
1035
|
+
async rawTableCfgs() {
|
|
991
1036
|
if (this.readables.length === 0) {
|
|
992
|
-
|
|
1037
|
+
throw new Error("No readable Io available");
|
|
1038
|
+
}
|
|
1039
|
+
const rawTableCfgs = /* @__PURE__ */ new Map();
|
|
1040
|
+
for (const { io: readable } of this.readables) {
|
|
1041
|
+
const cfgs = await readable.rawTableCfgs();
|
|
1042
|
+
if (cfgs.length > 0) {
|
|
1043
|
+
for (const tableCfg of cfgs) {
|
|
1044
|
+
if (!rawTableCfgs.has(tableCfg.key)) {
|
|
1045
|
+
rawTableCfgs.set(tableCfg.key, tableCfg);
|
|
1046
|
+
}
|
|
1047
|
+
}
|
|
1048
|
+
}
|
|
993
1049
|
}
|
|
994
|
-
|
|
995
|
-
return readable.rawTableCfgs();
|
|
1050
|
+
return Array.from(rawTableCfgs.values());
|
|
996
1051
|
}
|
|
997
1052
|
// ...........................................................................
|
|
998
1053
|
/**
|
|
@@ -1002,38 +1057,85 @@ class IoMulti {
|
|
|
1002
1057
|
*/
|
|
1003
1058
|
write(request) {
|
|
1004
1059
|
if (this.writables.length === 0) {
|
|
1005
|
-
|
|
1060
|
+
throw new Error("No writable Io available");
|
|
1006
1061
|
}
|
|
1007
|
-
const writes = this.writables.map(
|
|
1062
|
+
const writes = this.writables.map(
|
|
1063
|
+
({ io: writable }) => writable.write(request)
|
|
1064
|
+
);
|
|
1008
1065
|
return Promise.all(writes).then(() => Promise.resolve());
|
|
1009
1066
|
}
|
|
1010
1067
|
// ...........................................................................
|
|
1011
1068
|
/**
|
|
1012
|
-
* Reads rows from
|
|
1069
|
+
* Reads rows from a specific table by merging rows from all underlying readable Io instances.
|
|
1013
1070
|
* @param request An object containing the table name and where clause.
|
|
1014
1071
|
* @returns A promise that resolves to the read rows.
|
|
1015
1072
|
*/
|
|
1016
1073
|
async readRows(request) {
|
|
1017
1074
|
if (this.readables.length === 0) {
|
|
1018
|
-
|
|
1075
|
+
throw new Error("No readable Io available");
|
|
1019
1076
|
}
|
|
1020
1077
|
let tableExistsAny = false;
|
|
1021
|
-
const rows =
|
|
1078
|
+
const rows = /* @__PURE__ */ new Map();
|
|
1079
|
+
let type = void 0;
|
|
1080
|
+
let readFrom = "";
|
|
1081
|
+
const errors = [];
|
|
1022
1082
|
for (const readable of this.readables) {
|
|
1023
|
-
|
|
1024
|
-
|
|
1025
|
-
|
|
1026
|
-
const
|
|
1027
|
-
|
|
1028
|
-
|
|
1029
|
-
|
|
1030
|
-
|
|
1083
|
+
let tableRows = [];
|
|
1084
|
+
let tableType;
|
|
1085
|
+
try {
|
|
1086
|
+
const { [request.table]: tableData } = await readable.io.readRows(
|
|
1087
|
+
request
|
|
1088
|
+
);
|
|
1089
|
+
tableRows = tableData._data;
|
|
1090
|
+
tableType = tableData._type;
|
|
1091
|
+
tableExistsAny = true;
|
|
1092
|
+
readFrom = readable.id ?? "";
|
|
1093
|
+
} catch (e) {
|
|
1094
|
+
errors.push(e);
|
|
1095
|
+
continue;
|
|
1096
|
+
}
|
|
1097
|
+
type ??= tableType;
|
|
1098
|
+
if (tableRows.length === 0) {
|
|
1099
|
+
continue;
|
|
1031
1100
|
}
|
|
1101
|
+
for (const tableRow of tableRows) {
|
|
1102
|
+
const ref = tableRow._hash;
|
|
1103
|
+
rows.set(ref, tableRow);
|
|
1104
|
+
}
|
|
1105
|
+
break;
|
|
1032
1106
|
}
|
|
1033
1107
|
if (!tableExistsAny) {
|
|
1034
|
-
|
|
1108
|
+
if (errors.length === 0) {
|
|
1109
|
+
throw new Error(`Table "${request.table}" not found`);
|
|
1110
|
+
} else {
|
|
1111
|
+
const preciseErrors = errors.filter(
|
|
1112
|
+
(err) => !err.message.includes(`Table "${request.table}" not found`)
|
|
1113
|
+
);
|
|
1114
|
+
if (preciseErrors.length > 0) {
|
|
1115
|
+
throw preciseErrors[0];
|
|
1116
|
+
} else {
|
|
1117
|
+
throw errors[0];
|
|
1118
|
+
}
|
|
1119
|
+
}
|
|
1035
1120
|
} else {
|
|
1036
|
-
|
|
1121
|
+
const rljson = {
|
|
1122
|
+
[request.table]: hip({ _data: Array.from(rows.values()), _type: type })
|
|
1123
|
+
};
|
|
1124
|
+
if (this.writables.length > 0 && rows.size > 0) {
|
|
1125
|
+
for (const writeable of this.writables) {
|
|
1126
|
+
if (writeable.id === readFrom) {
|
|
1127
|
+
continue;
|
|
1128
|
+
}
|
|
1129
|
+
try {
|
|
1130
|
+
await writeable.io.write({
|
|
1131
|
+
data: rljson
|
|
1132
|
+
});
|
|
1133
|
+
} catch {
|
|
1134
|
+
continue;
|
|
1135
|
+
}
|
|
1136
|
+
}
|
|
1137
|
+
}
|
|
1138
|
+
return rljson;
|
|
1037
1139
|
}
|
|
1038
1140
|
}
|
|
1039
1141
|
// ...........................................................................
|
|
@@ -1044,12 +1146,12 @@ class IoMulti {
|
|
|
1044
1146
|
*/
|
|
1045
1147
|
async rowCount(table) {
|
|
1046
1148
|
if (this.dumpables.length === 0) {
|
|
1047
|
-
|
|
1149
|
+
throw new Error("No dumpable Io available");
|
|
1048
1150
|
}
|
|
1049
1151
|
const dumpTable = await this.dumpTable({ table });
|
|
1050
1152
|
const tableData = dumpTable[table];
|
|
1051
1153
|
if (!tableData) {
|
|
1052
|
-
|
|
1154
|
+
throw new Error(`Table "${table}" not found`);
|
|
1053
1155
|
}
|
|
1054
1156
|
return Promise.resolve(tableData._data.length);
|
|
1055
1157
|
}
|
|
@@ -1058,21 +1160,21 @@ class IoMulti {
|
|
|
1058
1160
|
* Gets the list of underlying readable Io instances, sorted by priority.
|
|
1059
1161
|
*/
|
|
1060
1162
|
get readables() {
|
|
1061
|
-
return this._ios.filter((ioMultiIo) => ioMultiIo.read).sort((a, b) => a.priority - b.priority)
|
|
1163
|
+
return this._ios.filter((ioMultiIo) => ioMultiIo.read).sort((a, b) => a.priority - b.priority);
|
|
1062
1164
|
}
|
|
1063
1165
|
// ...........................................................................
|
|
1064
1166
|
/**
|
|
1065
1167
|
* Gets the list of underlying writable Io instances, sorted by priority.
|
|
1066
1168
|
*/
|
|
1067
1169
|
get writables() {
|
|
1068
|
-
return this._ios.filter((ioMultiIo) => ioMultiIo.write).sort((a, b) => a.priority - b.priority)
|
|
1170
|
+
return this._ios.filter((ioMultiIo) => ioMultiIo.write).sort((a, b) => a.priority - b.priority);
|
|
1069
1171
|
}
|
|
1070
1172
|
// ...........................................................................
|
|
1071
1173
|
/**
|
|
1072
1174
|
* Gets the list of underlying dumpable Io instances, sorted by priority.
|
|
1073
1175
|
*/
|
|
1074
1176
|
get dumpables() {
|
|
1075
|
-
return this._ios.filter((ioMultiIo) => ioMultiIo.dump).sort((a, b) => a.priority - b.priority)
|
|
1177
|
+
return this._ios.filter((ioMultiIo) => ioMultiIo.dump).sort((a, b) => a.priority - b.priority);
|
|
1076
1178
|
}
|
|
1077
1179
|
// ...........................................................................
|
|
1078
1180
|
static example = async () => {
|
|
@@ -1092,6 +1194,233 @@ class IoMulti {
|
|
|
1092
1194
|
return ioMulti;
|
|
1093
1195
|
};
|
|
1094
1196
|
}
|
|
1197
|
+
class IoPeerBridge {
|
|
1198
|
+
constructor(_io, _socket) {
|
|
1199
|
+
this._io = _io;
|
|
1200
|
+
this._socket = _socket;
|
|
1201
|
+
}
|
|
1202
|
+
_eventHandlers = /* @__PURE__ */ new Map();
|
|
1203
|
+
/**
|
|
1204
|
+
* Starts the bridge by setting up connection event handlers and
|
|
1205
|
+
* automatically registering all Io methods.
|
|
1206
|
+
*/
|
|
1207
|
+
start() {
|
|
1208
|
+
this._socket.on("connect", () => this._handleConnect());
|
|
1209
|
+
this._socket.on("disconnect", () => this._handleDisconnect());
|
|
1210
|
+
this._registerIoMethods();
|
|
1211
|
+
}
|
|
1212
|
+
/**
|
|
1213
|
+
* Stops the bridge by removing all event handlers.
|
|
1214
|
+
*/
|
|
1215
|
+
stop() {
|
|
1216
|
+
this._socket.off("connect", () => this._handleConnect());
|
|
1217
|
+
this._socket.off("disconnect", () => this._handleDisconnect());
|
|
1218
|
+
for (const [eventName, handler] of this._eventHandlers) {
|
|
1219
|
+
this._socket.off(eventName, handler);
|
|
1220
|
+
}
|
|
1221
|
+
this._eventHandlers.clear();
|
|
1222
|
+
}
|
|
1223
|
+
/**
|
|
1224
|
+
* Automatically registers all Io interface methods as socket event handlers.
|
|
1225
|
+
*/
|
|
1226
|
+
_registerIoMethods() {
|
|
1227
|
+
const ioMethods = [
|
|
1228
|
+
"init",
|
|
1229
|
+
"isReady",
|
|
1230
|
+
"close",
|
|
1231
|
+
"tableExists",
|
|
1232
|
+
"createOrExtendTable",
|
|
1233
|
+
"write",
|
|
1234
|
+
"readRows",
|
|
1235
|
+
"rowCount",
|
|
1236
|
+
"dumpTable",
|
|
1237
|
+
"dump",
|
|
1238
|
+
"contentType",
|
|
1239
|
+
"rawTableCfgs"
|
|
1240
|
+
];
|
|
1241
|
+
for (const methodName of ioMethods) {
|
|
1242
|
+
this.registerEvent(methodName);
|
|
1243
|
+
}
|
|
1244
|
+
}
|
|
1245
|
+
/**
|
|
1246
|
+
* Registers a socket event to be translated to an Io method call.
|
|
1247
|
+
*
|
|
1248
|
+
* @param eventName - The socket event name (should match an Io method name)
|
|
1249
|
+
* @param ioMethodName - (Optional) The Io method name if different from eventName
|
|
1250
|
+
*/
|
|
1251
|
+
registerEvent(eventName, ioMethodName) {
|
|
1252
|
+
const methodName = ioMethodName || eventName;
|
|
1253
|
+
const handler = (...args) => {
|
|
1254
|
+
const callback = args[args.length - 1];
|
|
1255
|
+
const methodArgs = args.slice(0, -1);
|
|
1256
|
+
const ioMethod = this._io[methodName];
|
|
1257
|
+
if (typeof ioMethod !== "function") {
|
|
1258
|
+
const error = new Error(
|
|
1259
|
+
`Method "${methodName}" not found on Io instance`
|
|
1260
|
+
);
|
|
1261
|
+
if (typeof callback === "function") {
|
|
1262
|
+
callback(null, error);
|
|
1263
|
+
}
|
|
1264
|
+
return;
|
|
1265
|
+
}
|
|
1266
|
+
ioMethod.apply(this._io, methodArgs).then((result) => {
|
|
1267
|
+
if (typeof callback === "function") {
|
|
1268
|
+
callback(result, null);
|
|
1269
|
+
}
|
|
1270
|
+
}).catch((error) => {
|
|
1271
|
+
if (typeof callback === "function") {
|
|
1272
|
+
callback(null, error);
|
|
1273
|
+
}
|
|
1274
|
+
});
|
|
1275
|
+
};
|
|
1276
|
+
this._eventHandlers.set(eventName, handler);
|
|
1277
|
+
this._socket.on(eventName, handler);
|
|
1278
|
+
}
|
|
1279
|
+
/**
|
|
1280
|
+
* Registers multiple socket events at once.
|
|
1281
|
+
*
|
|
1282
|
+
* @param eventNames - Array of event names to register
|
|
1283
|
+
*/
|
|
1284
|
+
registerEvents(eventNames) {
|
|
1285
|
+
for (const eventName of eventNames) {
|
|
1286
|
+
this.registerEvent(eventName);
|
|
1287
|
+
}
|
|
1288
|
+
}
|
|
1289
|
+
/**
|
|
1290
|
+
* Unregisters a socket event handler.
|
|
1291
|
+
*
|
|
1292
|
+
* @param eventName - The event name to unregister
|
|
1293
|
+
*/
|
|
1294
|
+
unregisterEvent(eventName) {
|
|
1295
|
+
const handler = this._eventHandlers.get(eventName);
|
|
1296
|
+
if (handler) {
|
|
1297
|
+
this._socket.off(eventName, handler);
|
|
1298
|
+
this._eventHandlers.delete(eventName);
|
|
1299
|
+
}
|
|
1300
|
+
}
|
|
1301
|
+
/**
|
|
1302
|
+
* Emits a result back through the socket.
|
|
1303
|
+
*
|
|
1304
|
+
* @param eventName - The event name to emit
|
|
1305
|
+
* @param data - The data to send
|
|
1306
|
+
*/
|
|
1307
|
+
emitToSocket(eventName, ...data) {
|
|
1308
|
+
this._socket.emit(eventName, ...data);
|
|
1309
|
+
}
|
|
1310
|
+
/**
|
|
1311
|
+
* Calls an Io method directly and emits the result through the socket.
|
|
1312
|
+
*
|
|
1313
|
+
* @param ioMethodName - The Io method to call
|
|
1314
|
+
* @param socketEventName - The socket event to emit with the result
|
|
1315
|
+
* @param args - Arguments to pass to the Io method
|
|
1316
|
+
*/
|
|
1317
|
+
async callIoAndEmit(ioMethodName, socketEventName, ...args) {
|
|
1318
|
+
try {
|
|
1319
|
+
const ioMethod = this._io[ioMethodName];
|
|
1320
|
+
if (typeof ioMethod !== "function") {
|
|
1321
|
+
throw new Error(`Method "${ioMethodName}" not found on Io instance`);
|
|
1322
|
+
}
|
|
1323
|
+
const result = await ioMethod.apply(this._io, args);
|
|
1324
|
+
this._socket.emit(socketEventName, result, null);
|
|
1325
|
+
} catch (error) {
|
|
1326
|
+
this._socket.emit(socketEventName, null, error);
|
|
1327
|
+
}
|
|
1328
|
+
}
|
|
1329
|
+
/* v8 ignore next -- @preserve */
|
|
1330
|
+
_handleConnect() {
|
|
1331
|
+
}
|
|
1332
|
+
/* v8 ignore next -- @preserve */
|
|
1333
|
+
_handleDisconnect() {
|
|
1334
|
+
}
|
|
1335
|
+
/**
|
|
1336
|
+
* Gets the current socket instance.
|
|
1337
|
+
*/
|
|
1338
|
+
get socket() {
|
|
1339
|
+
return this._socket;
|
|
1340
|
+
}
|
|
1341
|
+
/**
|
|
1342
|
+
* Gets the current Io instance.
|
|
1343
|
+
*/
|
|
1344
|
+
get io() {
|
|
1345
|
+
return this._io;
|
|
1346
|
+
}
|
|
1347
|
+
/**
|
|
1348
|
+
* Returns whether the socket is currently connected.
|
|
1349
|
+
*/
|
|
1350
|
+
get isConnected() {
|
|
1351
|
+
return this._socket.connected;
|
|
1352
|
+
}
|
|
1353
|
+
}
|
|
1354
|
+
class IoServer {
|
|
1355
|
+
constructor(_io) {
|
|
1356
|
+
this._io = _io;
|
|
1357
|
+
}
|
|
1358
|
+
_sockets = [];
|
|
1359
|
+
// ...........................................................................
|
|
1360
|
+
/**
|
|
1361
|
+
* Adds a socket to the IoServer instance.
|
|
1362
|
+
* @param socket - The socket to add.
|
|
1363
|
+
*/
|
|
1364
|
+
async addSocket(socket) {
|
|
1365
|
+
await this._addTransportLayer(socket);
|
|
1366
|
+
this._sockets.push(socket);
|
|
1367
|
+
}
|
|
1368
|
+
// ...........................................................................
|
|
1369
|
+
/**
|
|
1370
|
+
* Removes a transport layer from the given socket.
|
|
1371
|
+
* @param socket - The socket to remove the transport layer from.
|
|
1372
|
+
*/
|
|
1373
|
+
removeSocket(socket) {
|
|
1374
|
+
this._sockets = this._sockets.filter((s) => s !== socket);
|
|
1375
|
+
}
|
|
1376
|
+
// ...........................................................................
|
|
1377
|
+
/**
|
|
1378
|
+
* Adds a transport layer to the given socket.
|
|
1379
|
+
* @param socket - The socket to add the transport layer to.
|
|
1380
|
+
*/
|
|
1381
|
+
async _addTransportLayer(socket) {
|
|
1382
|
+
const crud = this._generateTransportLayerCRUD(this._io);
|
|
1383
|
+
for (const [key, fn] of Object.entries(crud)) {
|
|
1384
|
+
socket.on(key, (...args) => {
|
|
1385
|
+
const cb = args[args.length - 1];
|
|
1386
|
+
fn.apply(this, args.slice(0, -1)).then((result) => {
|
|
1387
|
+
cb(result, null);
|
|
1388
|
+
}).catch((err) => {
|
|
1389
|
+
cb(null, err);
|
|
1390
|
+
});
|
|
1391
|
+
});
|
|
1392
|
+
}
|
|
1393
|
+
}
|
|
1394
|
+
// ...........................................................................
|
|
1395
|
+
/**
|
|
1396
|
+
* Creates or extends a table with the given configuration.
|
|
1397
|
+
* @param request - An object containing the table configuration.
|
|
1398
|
+
*/
|
|
1399
|
+
async createOrExtendTable(request) {
|
|
1400
|
+
return this._io.createOrExtendTable(request);
|
|
1401
|
+
}
|
|
1402
|
+
// ...........................................................................
|
|
1403
|
+
/**
|
|
1404
|
+
* Generates a transport layer object for the given Io instance.
|
|
1405
|
+
* @param io - The Io instance to generate the transport layer for.
|
|
1406
|
+
* @returns An object containing methods that correspond to the Io interface.
|
|
1407
|
+
*/
|
|
1408
|
+
_generateTransportLayerCRUD = (io) => ({
|
|
1409
|
+
init: () => io.init(),
|
|
1410
|
+
close: () => io.close(),
|
|
1411
|
+
isOpen: () => new Promise((resolve) => resolve(io.isOpen)),
|
|
1412
|
+
isReady: () => io.isReady(),
|
|
1413
|
+
dump: () => io.dump(),
|
|
1414
|
+
dumpTable: (request) => io.dumpTable(request),
|
|
1415
|
+
contentType: (request) => io.contentType(request),
|
|
1416
|
+
tableExists: (tableKey) => io.tableExists(tableKey),
|
|
1417
|
+
createOrExtendTable: (request) => this.createOrExtendTable(request),
|
|
1418
|
+
rawTableCfgs: () => io.rawTableCfgs(),
|
|
1419
|
+
write: (request) => io.write(request),
|
|
1420
|
+
readRows: (request) => io.readRows(request),
|
|
1421
|
+
rowCount: (table) => io.rowCount(table)
|
|
1422
|
+
});
|
|
1423
|
+
}
|
|
1095
1424
|
const exampleTestSetup = () => {
|
|
1096
1425
|
return {
|
|
1097
1426
|
io: new IoMem(),
|
|
@@ -1362,6 +1691,14 @@ const socketExample = () => ({
|
|
|
1362
1691
|
/* v8 ignore next -- @preserve */
|
|
1363
1692
|
emit() {
|
|
1364
1693
|
return true;
|
|
1694
|
+
},
|
|
1695
|
+
/* v8 ignore next -- @preserve */
|
|
1696
|
+
off() {
|
|
1697
|
+
return this;
|
|
1698
|
+
},
|
|
1699
|
+
/* v8 ignore next -- @preserve */
|
|
1700
|
+
removeAllListeners() {
|
|
1701
|
+
return this;
|
|
1365
1702
|
}
|
|
1366
1703
|
});
|
|
1367
1704
|
export {
|
|
@@ -1369,6 +1706,8 @@ export {
|
|
|
1369
1706
|
IoMem,
|
|
1370
1707
|
IoMulti,
|
|
1371
1708
|
IoPeer,
|
|
1709
|
+
IoPeerBridge,
|
|
1710
|
+
IoServer,
|
|
1372
1711
|
IoTools,
|
|
1373
1712
|
PeerSocketMock,
|
|
1374
1713
|
SocketMock,
|
|
@@ -1377,4 +1716,4 @@ export {
|
|
|
1377
1716
|
exampleTestSetup,
|
|
1378
1717
|
socketExample
|
|
1379
1718
|
};
|
|
1380
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"io.js","sources":["../src/io-db-name-mapping.ts","../src/io-tools.ts","../src/io-mem.ts","../src/peer-socket-mock.ts","../src/io-peer.ts","../src/io-multi.ts","../src/io-test-setup.ts","../src/io.ts","../src/reverse-ref.ts","../src/socket-mock.ts","../src/socket.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\nexport class IoDbNameMapping {\n  // The primary key column is always named '_hash'\n  public primaryKeyColumn: string = '_hash';\n  public dataSection: string = '_data';\n  public typeColumn: string = 'type';\n  public keyColumn: string = 'key';\n\n  // Names for the main tables in the database\n  public tableNames: { [key: string]: string } = {\n    main: 'tableCfgs',\n    revision: 'revisions',\n  };\n\n  /// Suffix handling for the database\n  private _suffix: { [key: string]: string } = {\n    col: '_col',\n    tbl: '_tbl',\n    tmp: '_tmp',\n  };\n\n  // ********************************************************************\n  // add and remove suffixes for use in SQL statements\n  private _addFix(name: string, fix: string): string {\n    return name.endsWith(fix) ? name : name + fix;\n  }\n\n  public addTableSuffix(name: string): string {\n    return this._addFix(name, this._suffix.tbl);\n  }\n\n  public addColumnSuffix(name: string): string {\n    return this._addFix(name, this._suffix.col);\n  }\n\n  public addTmpSuffix(name: string): string {\n    return this._addFix(name, this._suffix.tmp);\n  }\n\n  private _removeFix(name: string, fix: string): string {\n    return name.endsWith(fix) ? name.slice(0, -fix.length) : name;\n  }\n\n  public removeTableSuffix(name: string): string {\n    return this._removeFix(name, this._suffix.tbl);\n  }\n\n  public removeColumnSuffix(name: string): string {\n    return this._removeFix(name, this._suffix.col);\n  }\n\n  public removeTmpSuffix(name: string): string {\n    return this._removeFix(name, this._suffix.tmp);\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 } from '@rljson/hash';\nimport {\n  iterateTables,\n  Rljson,\n  TableCfg,\n  TableKey,\n  TableType,\n  throwOnInvalidTableCfg,\n  validateRljsonAgainstTableCfg,\n} from '@rljson/rljson';\n\nimport { IoMem } from './io-mem.ts';\nimport { Io } from './io.ts';\n\nexport type IoObserver = (data: Rljson) => void;\n\n/**\n * Provides utility functions for the Io interface.\n */\nexport class IoTools {\n  /**\n   * Constructor\n   * @param io The Io interface to use\n   */\n  constructor(public readonly io: Io) {}\n\n  /**\n   * Returns the table configuration of the tableCfgs table.\n   */\n  static get tableCfgsTableCfg() {\n    const tableCfg = hip<TableCfg>({\n      _hash: '',\n      key: 'tableCfgs',\n      type: 'tableCfgs',\n      isHead: false,\n      isRoot: false,\n      isShared: true,\n      previous: '',\n\n      columns: [\n        {\n          key: '_hash',\n          type: 'string',\n          titleShort: 'Hash',\n          titleLong: 'Row Hash',\n        },\n        {\n          key: 'key',\n          type: 'string',\n          titleShort: 'Key',\n          titleLong: 'Table Key',\n        },\n        {\n          key: 'type',\n          type: 'string',\n          titleShort: 'Type',\n          titleLong: 'Content Type',\n        },\n        {\n          key: 'isHead',\n          type: 'boolean',\n          titleShort: 'Is Head',\n          titleLong: 'Is Head Table',\n        },\n        {\n          key: 'isRoot',\n          type: 'boolean',\n          titleShort: 'Is Root',\n          titleLong: 'Is Root Table',\n        },\n        {\n          key: 'isShared',\n          type: 'boolean',\n          titleShort: 'Is Shared',\n          titleLong: 'Is Shared Table',\n        },\n        {\n          key: 'previous',\n          type: 'string',\n          titleShort: 'Previous',\n          titleLong: 'Previous Table Configuration Hash',\n        },\n        {\n          key: 'columns',\n          type: 'jsonArray',\n          titleShort: 'Columns',\n          titleLong: 'Column Configurations',\n        },\n      ],\n    });\n\n    return tableCfg;\n  }\n\n  /**\n   * Initializes the revisions table.\n   */\n  initRevisionsTable = async () => {\n    const tableCfg: TableCfg = {\n      key: 'revisions',\n      type: 'revisions',\n      isHead: true,\n      isRoot: true,\n      isShared: false,\n\n      columns: [\n        {\n          key: '_hash',\n          type: 'string',\n          titleShort: 'Hash',\n          titleLong: 'Row Hash',\n        },\n        {\n          key: 'table',\n          type: 'string',\n          titleShort: 'Table',\n          titleLong: 'Table Key',\n        },\n        {\n          key: 'predecessor',\n          type: 'string',\n          titleShort: 'Predecessor',\n          titleLong: 'Predecessor Revision Hash',\n        },\n        {\n          key: 'successor',\n          type: 'string',\n          titleShort: 'Successor',\n          titleLong: 'Successor Revision Hash',\n        },\n        {\n          key: 'timestamp',\n          type: 'number',\n          titleShort: 'Timestamp',\n          titleLong: 'Revision Timestamp',\n        },\n        {\n          key: 'id',\n          type: 'string',\n          titleShort: 'ID',\n          titleLong: 'Revision ID',\n        },\n      ],\n    };\n\n    await this.io.createOrExtendTable({ tableCfg });\n  };\n\n  /**\n   * Example object for test purposes\n   * @returns An instance of io tools\n   */\n  static example = async () => {\n    const io = await IoMem.example();\n    await io.init();\n    await io.isReady();\n    return new IoTools(io);\n  };\n\n  /**\n   * Throws if the table does not exist\n   */\n  async throwWhenTableDoesNotExist(table: TableKey): Promise<void> {\n    const exists = await this.io.tableExists(table);\n    if (!exists) {\n      throw new Error(`Table \"${table}\" not found`);\n    }\n  }\n\n  /**\n   * Throws if any of the tables in rljson do not exist\n   * @param rljson - The Rljson object to check\n   */\n  async throwWhenTablesDoNotExist(rljson: Rljson): Promise<void> {\n    try {\n      await iterateTables(rljson, async (tableKey) => {\n        const exists = await this.io.tableExists(tableKey);\n        if (!exists) {\n          throw new Error(`Table \"${tableKey}\" not found`);\n        }\n      });\n    } catch (e) {\n      const missingTables = (e as Array<any>).map((e) => e.tableKey);\n\n      throw new Error(\n        `The following tables do not exist: ${missingTables.join(', ')}`,\n      );\n    }\n  }\n\n  /**\n   * Returns the current table cfgs of all tables\n   * @returns The table configuration of all tables\n   */\n  async tableCfgs(): Promise<TableCfg[]> {\n    const tables = await this.io.rawTableCfgs();\n\n    // Take the latest version of each type key\n    const newestVersion: Record<TableKey, TableCfg> = {};\n    for (let i = tables.length - 1; i >= 0; i--) {\n      const table = tables[i];\n      const existing = newestVersion[table.key];\n      if (!existing || existing.columns.length < table.columns.length) {\n        newestVersion[table.key] = table;\n      }\n    }\n\n    // Sort the tables by key\n    /* v8 ignore next -- @preserve */\n    const resultData = Object.values(newestVersion).sort((a, b) => {\n      if (a.key < b.key) {\n        return -1;\n      }\n      if (a.key > b.key) {\n        return 1;\n      }\n\n      return 0;\n    });\n    return resultData;\n  }\n\n  /**\n   * Returns a list with all table names\n   */\n  async allTableKeys(): Promise<string[]> {\n    const result = (await this.tableCfgs()).map((e) => e.key);\n    return result;\n  }\n\n  /**\n   * Returns the configuration of a given table\n   */\n  async tableCfg(table: TableKey): Promise<TableCfg> {\n    const tableCfg = await this.tableCfgOrNull(table);\n    if (!tableCfg) {\n      throw new Error(`Table \"${table}\" not found`);\n    }\n\n    return tableCfg!;\n  }\n\n  /**\n   * Returns the configuration of a given table or null if it does not exist.\n\n   */\n  async tableCfgOrNull(table: TableKey): Promise<TableCfg | null> {\n    const tableCfgs = await this.tableCfgs();\n    const tableCfg = tableCfgs.find((e) => e.key === table);\n    return tableCfg ?? null;\n  }\n\n  /**\n   * Returns a list of all column names of a given table\n   */\n  async allColumnKeys(table: TableKey): Promise<string[]> {\n    const tableCfg = await this.tableCfg(table);\n    const result = tableCfg.columns.map((column) => column.key);\n    return result;\n  }\n\n  /**\n   * Throws when a column does not exist in a given table\n   * @param table - The table to check\n   * @param columns - The column to check\n   */\n  async throwWhenColumnDoesNotExist(\n    table: TableKey,\n    columns: string[],\n  ): Promise<void> {\n    const tableCfg = await this.tableCfg(table);\n    const columnKeys = tableCfg.columns.map((column) => column.key);\n    const missingColumns = columns.filter(\n      (column) => !columnKeys.includes(column),\n    );\n    if (missingColumns.length > 0) {\n      throw new Error(\n        `The following columns do not exist in table \"${table}\": ${missingColumns.join(\n          ', ',\n        )}.`,\n      );\n    }\n  }\n\n  /**\n   * Throws when a table update is not compatible with the current table\n   * configuration.\n   */\n  async throwWhenTableIsNotCompatible(update: TableCfg): Promise<void> {\n    const prefix = `Invalid update of table able \"${update.key}\"`;\n\n    throwOnInvalidTableCfg(update);\n\n    // Check compatibility with existing table\n    const existing = await this.tableCfgOrNull(update.key);\n    if (existing) {\n      // Have columns been deleted?\n      if (existing.columns.length > update.columns.length) {\n        const deletedColumnKeys = existing.columns\n          .map((column) => column.key)\n          .filter(\n            (key) => !update.columns.some((column) => column.key === key),\n          );\n        /* v8 ignore next -- @preserve */\n        if (deletedColumnKeys.length > 0) {\n          const deletedColumns = deletedColumnKeys.join(', ');\n          throw new Error(\n            `${prefix}: Columns must not be deleted. ` +\n              `Deleted columns: ${deletedColumns}}`,\n          );\n        }\n      }\n\n      // Have column keys changed?\n      for (let i = 0; i < existing.columns.length; i++) {\n        const before = existing.columns[i].key;\n        const after = update.columns[i].key;\n        if (before !== after) {\n          throw new Error(\n            `${prefix}: ` +\n              `Column keys must not change! ` +\n              `Column \"${before}\" was renamed into \"${after}\".`,\n          );\n        }\n      }\n\n      // Have column types changed?\n      for (let i = 0; i < existing.columns.length; i++) {\n        const column = existing.columns[i].key;\n        const before = existing.columns[i].type;\n        const after = update.columns[i].type;\n        if (before !== after) {\n          throw new Error(\n            `${prefix}: ` +\n              `Column types must not change! ` +\n              `Type of column \"${column}\" was changed from \"${before}\" to ${after}.`,\n          );\n        }\n      }\n    }\n  }\n\n  /**\n   * Throws if the data in the table do not match the table configuration\n   */\n  async throwWhenTableDataDoesNotMatchCfg(data: Rljson) {\n    const errors: string[] = [];\n\n    await iterateTables(data, async (tableKey) => {\n      const tableCfg = await this.tableCfg(tableKey);\n      const table = data[tableKey];\n\n      // Ignore tableCfgs table\n      /* v8 ignore next -- @preserve */\n      if (table._type === 'tableCfgs') return;\n\n      errors.push(...validateRljsonAgainstTableCfg(table._data, tableCfg));\n    });\n\n    if (errors.length > 0) {\n      throw new Error(\n        `Table data does not match the configuration.\\n\\nErrors:\\n${errors\n          .map((e) => `- ${e}`)\n          .join('\\n')}`,\n      );\n    }\n  }\n\n  /**\n   * Sorts the data of a table by the hash and updates the table hash in place\n   */\n  sortTableDataAndUpdateHash(table: TableType): void {\n    table._data.sort((a, b) => {\n      const hashA = a._hash as string;\n      const hashB = b._hash as string;\n      /* v8 ignore next -- @preserve */\n      if (hashA < hashB) {\n        return -1;\n      }\n      /* v8 ignore next -- @preserve */\n      if (hashA > hashB) {\n        return 1;\n      }\n\n      /* v8 ignore next -- @preserve */\n      return 0;\n    });\n\n    table._hash = '';\n    hip(table, {\n      updateExistingHashes: false,\n      throwOnWrongHashes: false,\n    });\n  }\n}\n","// @license\n// Copyright (c) 2025 Rljson\n//\n// Use of this source code is governed by terms that can be\n// found in the LICENSE file in the root of this package.\n\nimport { hip, hsh } from '@rljson/hash';\nimport { IsReady } from '@rljson/is-ready';\nimport { copy, equals, JsonValue } from '@rljson/json';\nimport {\n  ContentType,\n  iterateTablesSync,\n  Rljson,\n  TableCfg,\n  TableKey,\n  TableType,\n} from '@rljson/rljson';\n\nimport { IoTools } from './io-tools.ts';\nimport { Io } from './io.ts';\n\n/**\n * In-Memory implementation of the Rljson Io interface.\n */\nexport class IoMem implements Io {\n  // ...........................................................................\n  // Constructor & example\n\n  init(): Promise<void> {\n    this._isOpen = true;\n    return this._init();\n  }\n\n  close(): Promise<void> {\n    this._isOpen = false;\n    return Promise.resolve();\n  }\n\n  get isOpen(): boolean {\n    return this._isOpen;\n  }\n\n  static example = async () => {\n    const io = new IoMem();\n    await io.init();\n    return io;\n  };\n\n  // ...........................................................................\n  // General\n  isReady() {\n    return this._isReady.promise;\n  }\n\n  // ...........................................................................\n  // Dump\n\n  dump(): Promise<Rljson> {\n    return this._dump();\n  }\n\n  async dumpTable(request: { table: string }): Promise<Rljson> {\n    return this._dumpTable(request);\n  }\n\n  // ...........................................................................\n  // Meta Data\n\n  async contentType(request: { table: string }): Promise<ContentType> {\n    return this._contentType(request);\n  }\n\n  // ...........................................................................\n  // Rows\n\n  readRows(request: {\n    table: string;\n    where: { [column: string]: JsonValue };\n  }): Promise<Rljson> {\n    return this._readRows(request);\n  }\n\n  async rowCount(table: string): Promise<number> {\n    const tableData = this._mem[table] as TableType;\n    if (!tableData) {\n      throw new Error(`Table \"${table}\" not found`);\n    }\n    return Promise.resolve(tableData._data.length);\n  }\n\n  // ...........................................................................\n  // Write\n\n  write(request: { data: Rljson }): Promise<void> {\n    return this._write(request);\n  }\n\n  // ...........................................................................\n  // Table management\n  async tableExists(tableKey: TableKey): Promise<boolean> {\n    const table = this._mem[tableKey] as TableType;\n    return table ? true : false;\n  }\n\n  createOrExtendTable(request: { tableCfg: TableCfg }): Promise<void> {\n    return this._createOrExtendTable(request);\n  }\n\n  async rawTableCfgs(): Promise<TableCfg[]> {\n    const tables = this._mem.tableCfgs._data as TableCfg[];\n    return tables;\n  }\n\n  // ######################\n  // Private\n  // ######################\n\n  private _ioTools!: IoTools;\n\n  private _isReady = new IsReady();\n  private _isOpen = false;\n\n  private _mem: Rljson = hip({} as Rljson);\n\n  // ...........................................................................\n  private async _init() {\n    this._ioTools = new IoTools(this);\n    this._initTableCfgs();\n    this._updateGlobalHash();\n    await this._ioTools.initRevisionsTable();\n    hsh(this._mem);\n\n    this._isReady.resolve();\n  }\n\n  // ...........................................................................\n  private _initTableCfgs = () => {\n    const tableCfg = IoTools.tableCfgsTableCfg;\n\n    this._mem.tableCfgs = hip({\n      _type: 'tableCfgs',\n      _data: [tableCfg],\n      _tableCfg: tableCfg._hash as string,\n    });\n  };\n\n  // ...........................................................................\n  private _updateGlobalHash() {\n    (this._mem as any)._hash = '';\n    hip(this._mem, {\n      updateExistingHashes: false,\n    });\n  }\n\n  // ...........................................................................\n  private _updateTableHash(tableKey: TableKey) {\n    const table = this._mem[tableKey] as TableType;\n    table._hash = '';\n    hip(table, { updateExistingHashes: false });\n  }\n\n  // ...........................................................................\n  private async _createOrExtendTable(request: {\n    tableCfg: TableCfg;\n  }): Promise<void> {\n    // Make sure that the table config is compatible\n    // with an potential existing table\n    const tableCfg = request.tableCfg;\n    await this._ioTools.throwWhenTableIsNotCompatible(tableCfg);\n\n    const { key } = tableCfg;\n\n    // Recreate hashes in the case the existing hashes are wrong\n    const newConfig = hsh(tableCfg);\n\n    // Find an existing table config with the same hash\n    const existingConfig = await this._ioTools.tableCfgOrNull(key);\n\n    // Write the new config into the database\n    if (!existingConfig) {\n      this._createTable(newConfig, key);\n    } else {\n      this._extendTable(existingConfig, newConfig);\n    }\n  }\n\n  // ...........................................................................\n  private _createTable(newConfig: TableCfg, tableKey: TableKey) {\n    // Write the table config into the database\n    newConfig = hsh(newConfig);\n    this._mem.tableCfgs._data.push(newConfig);\n    this._ioTools.sortTableDataAndUpdateHash(this._mem.tableCfgs);\n\n    // Create a table and write it into the database\n    const table: TableType = {\n      _type: newConfig.type,\n      _data: [],\n      _tableCfg: newConfig._hash as string,\n    };\n\n    this._mem[tableKey] ??= hip(table);\n\n    // Update hashes\n    this._updateTableHash(tableKey);\n    this._updateGlobalHash();\n  }\n\n  // ...........................................................................\n  private _extendTable(existingConfig: TableCfg, newConfig: TableCfg) {\n    // No columns added? Return.\n    if (existingConfig.columns.length === newConfig.columns.length) {\n      return;\n    }\n\n    // Write the new table config into the database\n    newConfig = hsh(newConfig);\n    this._mem.tableCfgs._data.push(newConfig);\n    this._ioTools.sortTableDataAndUpdateHash(this._mem.tableCfgs);\n\n    // Update the config of the existing table\n    const table = this._mem[newConfig.key] as TableType;\n    table._tableCfg = newConfig._hash as string;\n\n    // Update the hashes\n    this._updateTableHash(newConfig.key);\n    this._updateGlobalHash();\n  }\n\n  // ...........................................................................\n\n  private async _dump(): Promise<Rljson> {\n    return copy(this._mem);\n  }\n\n  // ...........................................................................\n  private async _dumpTable(request: { table: string }): Promise<Rljson> {\n    await this._ioTools.throwWhenTableDoesNotExist(request.table);\n\n    const table = this._mem[request.table] as TableType;\n\n    return {\n      [request.table]: copy(table),\n    };\n  }\n\n  // ...........................................................................\n  private async _contentType(request: { table: string }): Promise<ContentType> {\n    await this._ioTools.throwWhenTableDoesNotExist(request.table);\n\n    return (this._mem[request.table] as TableType)._type;\n  }\n\n  // ...........................................................................\n  private async _write(request: { data: Rljson }): Promise<void> {\n    const addedData = hsh(request.data);\n    this._removeNullValues(addedData);\n    const tables = Object.keys(addedData);\n    hsh(addedData);\n\n    await this._ioTools.throwWhenTablesDoNotExist(request.data);\n    await this._ioTools.throwWhenTableDataDoesNotMatchCfg(request.data);\n\n    for (const table of tables) {\n      if (table.startsWith('_')) {\n        continue;\n      }\n\n      const oldTable = this._mem[table] as TableType;\n      const newTable = addedData[table] as TableType;\n\n      // Table exists. Merge data\n      for (const item of newTable._data) {\n        const hash = item._hash;\n        const exists = oldTable._data.find((i) => i._hash === hash);\n        if (!exists) {\n          oldTable._data.push(item as any);\n        }\n      }\n\n      this._ioTools.sortTableDataAndUpdateHash(oldTable);\n    }\n\n    // Recalc main hashes\n    this._updateGlobalHash();\n  }\n\n  // ...........................................................................\n  private async _readRows(request: {\n    table: string;\n    where: { [column: string]: JsonValue };\n  }): Promise<Rljson> {\n    await this._ioTools.throwWhenTableDoesNotExist(request.table);\n    await this._ioTools.throwWhenColumnDoesNotExist(\n      request.table,\n      Object.keys(request.where),\n    );\n\n    // Read table from data\n    const table = this._mem[request.table] as TableType;\n\n    // Filter table data\n    const tableDataFiltered = table._data.filter((row) => {\n      for (const column in request.where) {\n        const a = row[column];\n        const b = request.where[column];\n        if (b === null && a === undefined) {\n          return true;\n        }\n\n        if (!equals(a, b)) {\n          return false;\n        }\n      }\n      return true;\n    });\n\n    // Create an table\n    const tableFiltered: TableType = {\n      _type: table._type,\n      _data: tableDataFiltered,\n    };\n\n    this._ioTools.sortTableDataAndUpdateHash(tableFiltered);\n\n    const result: Rljson = {\n      [request.table]: tableFiltered,\n    };\n\n    return result;\n  }\n\n  _removeNullValues(rljson: Rljson) {\n    iterateTablesSync(rljson, (table) => {\n      const data = rljson[table]._data;\n\n      for (const row of data) {\n        for (const key in row) {\n          if (row[key] === null) {\n            delete row[key];\n          }\n        }\n      }\n    });\n  }\n}\n","// @license\n// Copyright (c) 2025 Rljson\n//\n// Use of this source code is governed by terms that can be\n// found in the LICENSE file in the root of this package.\n\nimport { Io } from './io.ts';\nimport { Socket } from './socket.ts';\n\n\nexport class PeerSocketMock implements Socket {\n  private _listenersMap: Map<string | symbol, Array<(...args: any[]) => void>> =\n    new Map();\n\n  connected: boolean = false;\n  disconnected: boolean = true;\n\n  constructor(private _io: Io) {}\n\n  // ............................................................................\n  /**\n   * Registers an event listener for the specified event.\n   * @param eventName - The name of the event to listen for.\n   * @param listener - The callback function to invoke when the event is emitted.\n   * @returns The PeerSocketMock instance for chaining.\n   */\n  on(eventName: string | symbol, listener: (...args: any[]) => void): this {\n    if (!this._listenersMap.has(eventName)) {\n      this._listenersMap.set(eventName, []);\n    }\n    this._listenersMap.get(eventName)!.push(listener);\n    return this;\n  }\n\n  // ...........................................................................\n  /**\n   * Simulates a connection event.\n   *\n   * Emits the 'connect' event to all registered listeners.\n   */\n  connect(): void {\n    this.connected = true;\n    this.disconnected = false;\n\n    const listeners = this._listenersMap.get('connect') || [];\n    for (const cb of listeners) {\n      cb({});\n    }\n  }\n\n  // ...........................................................................\n  /**\n   * Simulates a disconnection event.\n   *\n   * Emits the 'disconnect' event to all registered listeners.\n   */\n  disconnect(): void {\n    this.connected = false;\n    this.disconnected = true;\n\n    const listeners = this._listenersMap.get('disconnect') || [];\n    for (const cb of listeners) {\n      cb({});\n    }\n  }\n\n  // ............................................................................\n  /**\n   * Emits an event, invoking the corresponding method on the Io instance.\n   * @param eventName - The name of the event to emit.\n   * @param args - The arguments to pass to the event listener.\n   * @returns\n   */\n  emit(eventName: string | symbol, ...args: any[]): boolean {\n    const fn = (this._io as any)[eventName] as (...args: any[]) => Promise<any>;\n    if (typeof fn !== 'function') {\n      throw new Error(`Event ${eventName.toString()} not supported`);\n    }\n    const cb = args[args.length - 1];\n    fn.apply(this._io, args.slice(0, -1))\n      .then((result) => {\n        cb(result, null);\n      })\n      .catch((err) => {\n        cb(null, err);\n      });\n\n    return true;\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\nimport { IoMem } from './io-mem.ts';\nimport { Io } from './io.ts';\nimport { PeerSocketMock } from './peer-socket-mock.ts';\nimport { Socket } from './socket.ts';\n\nexport class IoPeer implements Io {\n  isOpen: boolean = false;\n\n  constructor(private _socket: Socket) {}\n\n  // ...........................................................................\n  /**\n   *\n   * Initializes the Peer connection.\n   * @returns\n   */\n  async init(): Promise<void> {\n    // Update isOpen on connect/disconnect\n    this._socket.on('connect', () => {\n      this.isOpen = true;\n    });\n    this._socket.on('disconnect', () => {\n      this.isOpen = false;\n    });\n\n    // Connect the socket\n    this._socket.connect();\n\n    return;\n  }\n\n  // ...........................................................................\n  /**\n   * Closes the Peer connection.\n   * @returns\n   */\n\n  async close(): Promise<void> {\n    // Disconnect the socket\n    if (!this._socket.connected) return;\n    this._socket.disconnect();\n    return;\n  }\n\n  // ...........................................................................\n  /**\n   * Returns a promise that resolves once the Peer connection is ready.\n   * @returns\n   */\n  async isReady(): Promise<void> {\n    return this._socket.connected ? Promise.resolve() : Promise.reject();\n  }\n\n  // ...........................................................................\n  /**\n   * Dumps the entire database content.\n   * @returns A promise that resolves to the dumped database content.\n   */\n  async dump(): Promise<Rljson> {\n    return new Promise((resolve) => {\n      // Request dump, resolve once the data is received (ack)\n      this._socket.emit('dump', (data: Rljson) => {\n        resolve(data);\n      });\n    });\n  }\n\n  // ...........................................................................\n  /**\n   * Dumps a specific table from the database.\n   * @param request An object containing the table name to dump.\n   * @returns A promise that resolves to the dumped table data.\n   */\n  dumpTable(request: { table: string }): Promise<Rljson> {\n    return new Promise((resolve, reject) => {\n      // Request dumpTable, resolve once the data is received (ack)\n      this._socket.emit('dumpTable', request, (data: Rljson, error?: Error) => {\n        if (error) reject(error);\n        resolve(data);\n      });\n    });\n  }\n\n  // ...........................................................................\n  /**\n   * Gets the content type of a specific table.\n   * @param request An object containing the table name to get the content type for.\n   * @returns A promise that resolves to the content type of the specified table.\n   */\n  contentType(request: { table: string }): Promise<ContentType> {\n    return new Promise((resolve, reject) => {\n      // Request contentType, resolve once the data is received (ack)\n      this._socket.emit(\n        'contentType',\n        request,\n        (data: ContentType, error?: Error) => {\n          /* v8 ignore next -- @preserve */\n          if (error) reject(error);\n          resolve(data);\n        },\n      );\n    });\n  }\n\n  // ...........................................................................\n  /**\n   * Checks if a specific table exists in the database.\n   * @param tableKey The key of the table to check for existence.\n   * @returns A promise that resolves to true if the table exists, false otherwise.\n   */\n  tableExists(tableKey: TableKey): Promise<boolean> {\n    return new Promise((resolve) => {\n      // Request tableExists, resolve once the data is received (ack)\n      this._socket.emit('tableExists', tableKey, (exists: boolean) => {\n        resolve(exists);\n      });\n    });\n  }\n\n  // ...........................................................................\n  /**\n   * Creates or extends a table with the given configuration.\n   * @param request An object containing the table configuration.\n   * @returns A promise that resolves once the table is created or extended.\n   */\n  createOrExtendTable(request: { tableCfg: TableCfg }): Promise<void> {\n    return new Promise((resolve, reject) => {\n      // Request createOrExtendTable, resolve once the data is received (ack)\n      this._socket.emit(\n        'createOrExtendTable',\n        request,\n        (_?: boolean, error?: Error) => {\n          if (error) reject(error);\n          resolve();\n        },\n      );\n    });\n  }\n\n  // ...........................................................................\n  /**\n   * Retrieves the raw table configurations from the database.\n   * @returns A promise that resolves to an array of table configurations.\n   */\n  rawTableCfgs(): Promise<TableCfg[]> {\n    return new Promise((resolve) => {\n      // Request rawTableCfgs, resolve once the data is received (ack)\n      this._socket.emit('rawTableCfgs', (data: TableCfg[]) => {\n        resolve(data);\n      });\n    });\n  }\n\n  // ...........................................................................\n  /**\n   * Writes data to the database.\n   * @param request An object containing the data to write.\n   * @returns A promise that resolves once the data is written.\n   */\n  write(request: { data: Rljson }): Promise<void> {\n    return new Promise((resolve, reject) => {\n      // Request write, resolve once the data is received (ack)\n      this._socket.emit('write', request, (_?: boolean, error?: Error) => {\n        if (error) reject(error);\n        resolve();\n      });\n    });\n  }\n\n  // ...........................................................................\n  /**\n   * Reads rows from a specific table that match the given criteria.\n   * @param request An object containing the table name and the criteria for selecting rows.\n   * @returns A promise that resolves to the selected rows.\n   */\n  readRows(request: {\n    table: string;\n    where: { [column: string]: JsonValue | null };\n  }): Promise<Rljson> {\n    return new Promise((resolve, reject) => {\n      // Request readRows, resolve once the data is received (ack)\n      this._socket.emit(\n        'readRows',\n        request,\n        (result?: Rljson, error?: Error) => {\n          if (error) reject(error);\n          resolve(result!);\n        },\n      );\n    });\n  }\n\n  // ...........................................................................\n  /**\n   * Retrieves the number of rows in a specific table.\n   * @param table The name of the table to count rows in.\n   * @returns A promise that resolves to the number of rows in the specified table.\n   */\n  rowCount(table: string): Promise<number> {\n    return new Promise((resolve, reject) => {\n      // Request rowCount, resolve once the data is received (ack)\n      this._socket.emit('rowCount', table, (count?: number, error?: Error) => {\n        if (error) reject(error);\n        resolve(count!);\n      });\n    });\n  }\n\n  // ...........................................................................\n  static example = async () => {\n    const ioMem = await IoMem.example();\n    const socket = new PeerSocketMock(ioMem);\n    const io = new IoPeer(socket);\n    await io.init();\n    return io;\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, merge } from '@rljson/json';\nimport {\n  ContentType,\n  Rljson,\n  TableCfg,\n  TableKey,\n  TableType,\n} from '@rljson/rljson';\n\nimport { IoMem } from './io-mem.ts';\nimport { IoPeer } from './io-peer.ts';\nimport { Io } from './io.ts';\nimport { PeerSocketMock } from './peer-socket-mock.ts';\n\n// ...........................................................................\n/**\n * Type representing an Io instance along with its capabilities and priority.\n */\nexport type IoMultiIo = {\n  io: Io;\n  priority: number;\n  read: boolean;\n  write: boolean;\n  dump: boolean;\n};\n\n// ...........................................................................\n/**\n * Multi Io implementation that combines multiple underlying Io instances\n * with different capabilities (read, write, dump) and priorities.\n */\nexport class IoMulti implements Io {\n  isOpen: boolean = false;\n\n  constructor(private _ios: Array<IoMultiIo>) {}\n\n  // ...........................................................................\n  /**\n   *\n   * Initializes all underlying Io instances.\n   * @returns\n   */\n  async init(): Promise<void> {\n    for (const ioMultiIo of this._ios) {\n      if (!ioMultiIo.io.isOpen) {\n        throw new Error(\n          'All underlying Io instances must be initialized before initializing IoMulti',\n        );\n      }\n    }\n\n    this.isOpen = true;\n    return Promise.resolve();\n  }\n\n  // ...........................................................................\n  /**\n   * Closes all underlying Io instances.\n   * @returns\n   */\n  async close(): Promise<void> {\n    await Promise.all(this._ios.map((ioMultiIo) => ioMultiIo.io.close()));\n\n    this.isOpen = false;\n\n    return Promise.resolve();\n  }\n\n  // ...........................................................................\n  /**\n   * Returns a promise that resolves once all underlying Io instances are ready.\n   * @returns\n   */\n  isReady(): Promise<void> {\n    return Promise.all(\n      this._ios.map((ioMultiIo) => ioMultiIo.io.isReady()),\n    ).then(() => Promise.resolve());\n  }\n\n  // ...........................................................................\n  /**\n   * Dumps the entire database content by merging dumps from all dumpable underlying Io instances.\n   * @returns\n   */\n  async dump(): Promise<Rljson> {\n    /* v8 ignore next -- @preserve */\n    if (this.dumpables.length === 0) {\n      return Promise.reject(new Error('No dumpable Io available'));\n    }\n\n    const dumps = await Promise.all(\n      this.dumpables.map((dumpable) => dumpable.dump()),\n    );\n\n    return merge(...dumps) as Rljson;\n  }\n\n  // ...........................................................................\n  /**\n   * Dumps a specific table by merging dumps from all dumpable underlying Io instances that contain the table.\n   * @param request An object containing the table name to dump.\n   * @returns A promise that resolves to the dumped table data.\n   */\n  async dumpTable(request: { table: string }): Promise<Rljson> {\n    /* v8 ignore next -- @preserve */\n    if (this.dumpables.length === 0) {\n      return Promise.reject(new Error('No dumpable Io available'));\n    }\n\n    const dumps: Rljson[] = [];\n\n    for (const dumpable of this.dumpables) {\n      const tableExists = await dumpable.tableExists(request.table);\n      if (tableExists) {\n        const dump = await dumpable.dumpTable(request);\n        dumps.push(dump);\n      }\n    }\n\n    if (dumps.length > 0) {\n      return merge(...dumps) as Rljson;\n    } else {\n      return Promise.reject(new Error(`Table \"${request.table}\" not found`));\n    }\n  }\n\n  // ...........................................................................\n  /**\n   * Retrieves the content type of a specific table from the first underlying readable Io instance that contains the table.\n   * @param request An object containing the table name.\n   * @returns A promise that resolves to the content type of the table.\n   */\n  async contentType(request: { table: string }): Promise<ContentType> {\n    /* v8 ignore next -- @preserve */\n    if (this.readables.length === 0) {\n      return Promise.reject(new Error('No readable Io available'));\n    }\n\n    for (const readable of this.readables) {\n      const tableExists = await readable.tableExists(request.table);\n      /* v8 ignore else -- @preserve */\n      if (tableExists) {\n        return readable.contentType(request);\n      }\n    }\n    return Promise.reject(new Error(`Table \"${request.table}\" not found`));\n  }\n\n  // ...........................................................................\n  /**\n   * Checks if a specific table exists in any of the underlying readable Io instances.\n   * @param tableKey The key of the table to check.\n   * @returns A promise that resolves to true if the table exists in any readable Io, false otherwise.\n   */\n  async tableExists(tableKey: TableKey): Promise<boolean> {\n    /* v8 ignore next -- @preserve */\n    if (this.readables.length === 0) {\n      return Promise.reject(new Error('No readable Io available'));\n    }\n\n    for (const readable of this.readables) {\n      const exists = await readable.tableExists(tableKey);\n      if (exists) {\n        return true;\n      }\n    }\n    return false;\n  }\n\n  // ...........................................................................\n  /**\n   * Creates or extends a table in all underlying writable Io instances.\n   * @param request An object containing the table configuration.\n   * @returns A promise that resolves once the table has been created or extended in all writable Io instances.\n   */\n  createOrExtendTable(request: { tableCfg: TableCfg }): Promise<void> {\n    /* v8 ignore next -- @preserve */\n    if (this.writables.length === 0) {\n      return Promise.reject(new Error('No writable Io available'));\n    }\n    //Create or extend table in all writables in parallel and resolve when all have completed\n    const creations = this.writables.map((writable) =>\n      writable.createOrExtendTable(request),\n    );\n    return Promise.all(creations).then(() => Promise.resolve());\n  }\n\n  // ...........................................................................\n  /**\n   * Retrieves the raw table configurations from the highest priority underlying readable Io instance.\n   * @returns A promise that resolves to an array of table configurations.\n   */\n  rawTableCfgs(): Promise<TableCfg[]> {\n    /* v8 ignore next -- @preserve */\n    if (this.readables.length === 0) {\n      return Promise.reject(new Error('No readable Io available'));\n    }\n\n    //Simple strategy: use the highest priority readable Io\n    const readable = this.readables[0];\n\n    return readable.rawTableCfgs();\n  }\n\n  // ...........................................................................\n  /**\n   * Writes data to all underlying writable Io instances.\n   * @param request - An object containing the data to write.\n   * @returns A promise that resolves once the data has been written to all writable Io instances.\n   */\n  write(request: { data: Rljson }): Promise<void> {\n    /* v8 ignore next -- @preserve */\n    if (this.writables.length === 0) {\n      return Promise.reject(new Error('No writable Io available'));\n    }\n\n    // Write to all writables in parallel and resolve when all have completed\n    const writes = this.writables.map((writable) => writable.write(request));\n    return Promise.all(writes).then(() => Promise.resolve());\n  }\n\n  // ...........................................................................\n  /**\n   * Reads rows from the first underlying readable Io instance that contains the requested table and has matching rows.\n   * @param request An object containing the table name and where clause.\n   * @returns A promise that resolves to the read rows.\n   */\n  async readRows(request: {\n    table: string;\n    where: { [column: string]: JsonValue | null };\n  }): Promise<Rljson> {\n    /* v8 ignore next -- @preserve */\n    if (this.readables.length === 0) {\n      return Promise.reject(new Error('No readable Io available'));\n    }\n\n    let tableExistsAny = false;\n    const rows: Rljson[] = [];\n    for (const readable of this.readables) {\n      // Check if table exists in this readable Io\n      const tableExists = await readable.tableExists(request.table);\n      tableExistsAny = tableExistsAny || tableExists;\n\n      if (tableExists) {\n        // Read rows from this readable Io\n        const readRows = await readable.readRows(request);\n        const tableData: TableType = readRows[request.table];\n        /* v8 ignore else -- @preserve */\n        if (tableData) {\n          rows.push(readRows);\n        }\n      }\n    }\n\n    if (!tableExistsAny) {\n      return Promise.reject(new Error(`Table \"${request.table}\" not found`));\n    } else {\n      return merge(...rows) as Rljson;\n    }\n  }\n\n  // ...........................................................................\n  /**\n   * Retrieves the row count of a specific table by aggregating row counts from all dumpable underlying Io instances.\n   * @param table The name of the table.\n   * @returns A promise that resolves to the row count of the table.\n   */\n  async rowCount(table: string): Promise<number> {\n    /* v8 ignore next -- @preserve */\n    if (this.dumpables.length === 0) {\n      return Promise.reject(new Error('No dumpable Io available'));\n    }\n\n    const dumpTable = await this.dumpTable({ table });\n    const tableData: TableType = dumpTable[table];\n    /* v8 ignore next -- @preserve */\n    if (!tableData) {\n      return Promise.reject(new Error(`Table \"${table}\" not found`));\n    }\n    return Promise.resolve(tableData._data.length);\n  }\n\n  // ...........................................................................\n  /**\n   * Gets the list of underlying readable Io instances, sorted by priority.\n   */\n  get readables(): Array<Io> {\n    return this._ios\n      .filter((ioMultiIo) => ioMultiIo.read)\n      .sort((a, b) => a.priority - b.priority)\n      .map((ioMultiIo) => ioMultiIo.io);\n  }\n\n  // ...........................................................................\n  /**\n   * Gets the list of underlying writable Io instances, sorted by priority.\n   */\n  get writables(): Array<Io> {\n    return this._ios\n      .filter((ioMultiIo) => ioMultiIo.write)\n      .sort((a, b) => a.priority - b.priority)\n      .map((ioMultiIo) => ioMultiIo.io);\n  }\n\n  // ...........................................................................\n  /**\n   * Gets the list of underlying dumpable Io instances, sorted by priority.\n   */\n  get dumpables(): Array<Io> {\n    return this._ios\n      .filter((ioMultiIo) => ioMultiIo.dump)\n      .sort((a, b) => a.priority - b.priority)\n      .map((ioMultiIo) => ioMultiIo.io);\n  }\n\n  // ...........................................................................\n  static example = async () => {\n    const ioPeerMem = await IoMem.example();\n    await ioPeerMem.init();\n\n    const ioPeerSocket = new PeerSocketMock(ioPeerMem);\n    const ioPeer = new IoPeer(ioPeerSocket);\n    await ioPeer.init();\n\n    const ioMem = await IoMem.example();\n    await ioMem.init();\n\n    const ios: Array<IoMultiIo> = [\n      { io: ioPeer, priority: 1, read: true, write: false, dump: false },\n      { io: ioMem, priority: 0, read: true, write: true, dump: true },\n    ];\n\n    const ioMulti = new IoMulti(ios);\n    await ioMulti.init();\n\n    return ioMulti;\n  };\n}\n","// @license\n// Copyright (c) 2025 Rljson\n//\n// Use of this source code is governed by terms that can be\n// found in the LICENSE file in the root of this package.\n\nimport { Io, IoMem } from './index.ts';\n\n// .............................................................................\n/**\n * Io implementation need to implement this interface to be used in\n * conformance tests.\n */\nexport interface IoTestSetup {\n  /** setup before the single setups */\n  beforeAll: () => Promise<void>;\n\n  /**\n   * Initializes the io implementation.\n   * @returns The io implementation.\n   */\n  beforeEach: () => Promise<void>;\n\n  /**\n   * Tears down the io implementation.\n   * @returns The io implementation.\n   */\n  afterEach: () => Promise<void>;\n\n  /** cleanup after all tests */\n  afterAll: () => Promise<void>;\n\n  /**\n   * The io implementation to be used in the conformance tests.\n   */\n  io: Io;\n}\n\n// .............................................................................\n// Example implementation of the IoTestSetup interface\nexport const exampleTestSetup = (): IoTestSetup => {\n  return {\n    io: new IoMem(),\n    beforeAll: async () => {\n      // This method can be used for any additional setup required before init.\n      // Currently, it does nothing.\n    },\n\n    beforeEach: async () => {\n      // Initialize the io implementation\n    },\n    afterEach: async () => {\n      // Tear down the io implementation\n    },\n\n    afterAll: async () => {\n      // This method can be used for any additional cleanup after tearDown.\n    },\n  };\n};\n","// @license\n// Copyright (c) 2025 Rljson\n//\n// Use of this source code is governed by terms that can be\n// found in the LICENSE file in the root of this package.\n\nimport { JsonValue } from '@rljson/json';\nimport { ContentType, Rljson, TableCfg, TableKey } from '@rljson/rljson';\n\n\n// .............................................................................\nexport interface Io {\n  // ...........................................................................\n  // General\n\n  /** Starts the initialization */\n  init(): Promise<void>;\n\n  /** Closes the io */\n  close(): Promise<void>;\n\n  /** Returns true if io is opened */\n  isOpen: boolean;\n\n  /** A promise resolving once the Io interface is ready\n   *\n   * 💡 Use @rljson/is-ready\n   */\n  isReady(): Promise<void>;\n\n  // ...........................................................................\n  // Dump\n\n  /** Returns the complete db content as Rljson */\n  dump(): Promise<Rljson>;\n\n  /** Returns the dump of a complete table */\n  dumpTable(request: { table: string }): Promise<Rljson>;\n\n  // ...........................................................................\n  // Meta Data\n  contentType(request: { table: string }): Promise<ContentType>;\n\n  // ...........................................................................\n  // Tables\n\n  /**\n   * Returns true if the table exists\n   */\n  tableExists(tableKey: TableKey): Promise<boolean>;\n\n  /**\n   * Creates a table with a given config.\n   * If the table already exists, new columns are added to the existing table.\n   * If the table does not exist, it is created with the given config.\n   * If the table exists and columns are removed, an error is thrown.\n   * If the table exists and the column type is changed, an error is thrown.\n   */\n  createOrExtendTable(request: { tableCfg: TableCfg }): Promise<void>;\n\n  /**\n   * Returns a json structure returning current table configurations\n   */\n  rawTableCfgs(): Promise<TableCfg[]>;\n\n  // ...........................................................................\n  // Write\n\n  /** Writes Rljson data into the database */\n  write(request: { data: Rljson }): Promise<void>;\n\n  // ...........................................................................\n  // Read rows\n\n  /** Queries a list of rows */\n  readRows(request: {\n    table: string;\n    where: { [column: string]: JsonValue | null };\n  }): Promise<Rljson>;\n\n  /** Returns the number of rows in the given table */\n  rowCount(table: string): Promise<number>;\n}\n\n// .............................................................................\nexport const exampleIo =\n  'Checkout @rljson/io-mem for an example implementation';\n","/* v8 ignore file -- @preserve */\n// @license\n// Copyright (c) 2025 Rljson\n//\n// Use of this source code is governed by terms that can be\n// found in the LICENSE file in the root of this package.\n\nimport { Json } from '@rljson/json';\nimport {\n  Buffet,\n  Cake,\n  iterateTablesSync,\n  Layer,\n  Ref,\n  Rljson,\n  TableKey,\n} from '@rljson/rljson';\n\n// .............................................................................\n/**\n * Describes a row that references a child table row\n */\nexport interface ParentRef {\n  /**\n   * The parent table that references the child table\n   */\n  [parentTable: TableKey]: {\n    /**\n     * The parent row that references the child row\n     */\n    [parentRow: Ref]: {\n      /**\n       * Details about the reference, e.g. an array index etc.\n       */\n      details?: Json;\n    };\n  };\n}\n\n// .............................................................................\n/**\n * Describes the parent table rows referencing a child table row\n */\nexport interface ReverseRefs {\n  /**\n   * The child table we need the referencing rows for\n   */\n  [childTable: TableKey]: {\n    /**\n     * The row hashwe need the referencing rows for\n     */\n    [childRow: Ref]: ParentRef;\n  };\n}\n\n/* v8 ignore start */\n\n// .............................................................................\n/**\n * Calculates the reverse references for a given rljson object\n */\nexport const calcReverseRefs = (rljson: Rljson): ReverseRefs => {\n  const result: ReverseRefs = {};\n\n  // ......................\n  // Prepare data structure\n  iterateTablesSync(rljson, (childTableKey, table) => {\n    const childTable: { [childRowHash: string]: ParentRef } = {};\n    result[childTableKey] = childTable;\n    for (const childRow of table._data) {\n      childTable[childRow._hash] = {};\n    }\n  });\n\n  // ............................\n  // Generate reverse references\n  iterateTablesSync(rljson, (parentTableKey, parentTable) => {\n    // Iterate all rows of each table\n    for (const parentTableRow of parentTable._data) {\n      // Find out whe other tables & rows are referenced by this row\n      // Write these information intto result\n      switch (parentTable._type) {\n        case 'components':\n          _writeComponentRefs(parentTableKey, parentTableRow, result);\n          break;\n\n        case 'layers': {\n          _writeLayerRefs(parentTableKey, parentTableRow, result);\n          break;\n        }\n\n        case 'sliceIds': {\n          // Slice ids do not reference other tables\n          break;\n        }\n\n        case 'cakes': {\n          _writeCakeRefs(parentTableKey, parentTableRow, result);\n          break;\n        }\n\n        case 'buffets': {\n          _writeBuffetRefs(parentTableKey, parentTableRow, result);\n          break;\n        }\n      }\n    }\n  });\n\n  return result;\n};\n\n/* v8 ignore end */\n\n// .............................................................................\nconst _writeComponentRefs = (\n  parentTableName: TableKey,\n  parentRow: Json,\n  result: ReverseRefs,\n) => {\n  const parentRowHash = parentRow._hash as string;\n\n  for (const parentColumnName in parentRow) {\n    if (parentColumnName.startsWith('_')) {\n      continue;\n    }\n\n    if (!parentColumnName.endsWith('Ref')) {\n      continue;\n    }\n\n    const childTableName = parentColumnName.slice(0, -3);\n    const childRowHash = parentRow[parentColumnName] as string;\n\n    _write(\n      result,\n      childTableName,\n      childRowHash,\n      parentTableName,\n      parentRowHash,\n    );\n  }\n};\n\n// .............................................................................\nconst _writeLayerRefs = (\n  parentTableName: TableKey,\n  parentRow: Layer,\n  result: ReverseRefs,\n) => {\n  const childTableName = parentRow.componentsTable;\n  const parentRowHash = parentRow._hash as string;\n\n  for (const sliceId in parentRow.add) {\n    if (sliceId.startsWith('_')) {\n      continue;\n    }\n\n    const sliceHash = parentRow.add[sliceId] as string;\n\n    _write(result, childTableName, sliceHash, parentTableName, parentRowHash);\n  }\n};\n\n// .............................................................................\nconst _writeCakeRefs = (\n  parentTableName: TableKey,\n  parentRow: Cake,\n  result: ReverseRefs,\n) => {\n  const parentRowHash = parentRow._hash as string;\n\n  for (const layer in parentRow.layers) {\n    const childTableName = layer;\n    const childRowHash = parentRow.layers[layer] as string;\n    _write(\n      result,\n      childTableName,\n      childRowHash,\n      parentTableName,\n      parentRowHash,\n    );\n  }\n};\n\n// .............................................................................\nconst _writeBuffetRefs = (\n  parentTableName: TableKey,\n  parentRow: Buffet,\n  result: ReverseRefs,\n) => {\n  const parentRowHash = parentRow._hash as string;\n\n  for (const item of parentRow.items) {\n    const childTableName = item.table;\n    const childRowHash = item.ref;\n    _write(\n      result,\n      childTableName,\n      childRowHash,\n      parentTableName,\n      parentRowHash,\n    );\n  }\n};\n\n// .............................................................................\nconst _write = (\n  result: ReverseRefs,\n  childTableName: string,\n  childRowHash: string,\n  parentTableName: string,\n  parentRowHash: string,\n) => {\n  const referencesForChildTable = (result[childTableName] ??= {});\n  const referencesForChildTableRow = (referencesForChildTable[childRowHash] ??=\n    {});\n\n  referencesForChildTableRow[parentTableName] ??= {};\n  referencesForChildTableRow[parentTableName][parentRowHash] ??= {};\n};\n","// @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/* v8 ignore file -- @preserve */\nimport { Socket } from './socket.ts';\n\nexport class SocketMock implements Socket {\n  public connected: boolean = false;\n  public disconnected: boolean = true;\n\n  private _listeners: Map<string | symbol, Array<(...args: any[]) => void>> =\n    new Map();\n  private _onceListeners: Map<\n    string | symbol,\n    Array<(...args: any[]) => void>\n  > = new Map();\n\n  connect(): void {\n    if (!this.connected) {\n      this.connected = true;\n      this.disconnected = false;\n      this.emit('connect');\n    }\n  }\n\n  disconnect(): void {\n    if (this.connected) {\n      this.connected = false;\n      this.disconnected = true;\n      this.emit('disconnect');\n    }\n  }\n\n  on(eventName: string | symbol, listener: (...args: any[]) => void): this {\n    if (!this._listeners.has(eventName)) {\n      this._listeners.set(eventName, []);\n    }\n    this._listeners.get(eventName)!.push(listener);\n    return this;\n  }\n\n  once(eventName: string | symbol, listener: (...args: any[]) => void): this {\n    if (!this._onceListeners.has(eventName)) {\n      this._onceListeners.set(eventName, []);\n    }\n    this._onceListeners.get(eventName)!.push(listener);\n    return this;\n  }\n\n  off(eventName: string | symbol, listener?: (...args: any[]) => void): this {\n    if (listener) {\n      // Remove specific listener\n      const regularListeners = this._listeners.get(eventName);\n      if (regularListeners) {\n        const index = regularListeners.indexOf(listener);\n        if (index > -1) {\n          regularListeners.splice(index, 1);\n        }\n      }\n\n      const onceListeners = this._onceListeners.get(eventName);\n      if (onceListeners) {\n        const index = onceListeners.indexOf(listener);\n        if (index > -1) {\n          onceListeners.splice(index, 1);\n        }\n      }\n    } else {\n      // Remove all listeners for the event\n      this._listeners.delete(eventName);\n      this._onceListeners.delete(eventName);\n    }\n    return this;\n  }\n\n  emit(eventName: string | symbol, ...args: any[]): boolean {\n    let hasListeners = false;\n\n    // Emit to regular listeners\n    const regularListeners = this._listeners.get(eventName);\n    if (regularListeners && regularListeners.length > 0) {\n      hasListeners = true;\n      // Create a copy to avoid issues if listeners are removed during emission\n      [...regularListeners].forEach((listener) => {\n        try {\n          listener(...args);\n        } catch (error) {\n          // In a real EventEmitter, this would be handled differently\n          console.error(\n            `Error in listener for event ${String(eventName)}:`,\n            error,\n          );\n        }\n      });\n    }\n\n    // Emit to once listeners and remove them\n    const onceListeners = this._onceListeners.get(eventName);\n    if (onceListeners && onceListeners.length > 0) {\n      hasListeners = true;\n      // Create a copy and clear the original array\n      const listenersToCall = [...onceListeners];\n      this._onceListeners.delete(eventName);\n\n      listenersToCall.forEach((listener) => {\n        try {\n          listener(...args);\n        } catch (error) {\n          console.error(\n            `Error in once listener for event ${String(eventName)}:`,\n            error,\n          );\n        }\n      });\n    }\n\n    return hasListeners;\n  }\n\n  removeAllListeners(eventName?: string | symbol): this {\n    if (eventName !== undefined) {\n      this._listeners.delete(eventName);\n      this._onceListeners.delete(eventName);\n    } else {\n      this._listeners.clear();\n      this._onceListeners.clear();\n    }\n    return this;\n  }\n\n  listenerCount(eventName: string | symbol): number {\n    const regularCount = this._listeners.get(eventName)?.length || 0;\n    const onceCount = this._onceListeners.get(eventName)?.length || 0;\n    return regularCount + onceCount;\n  }\n\n  listeners(eventName: string | symbol): Array<(...args: any[]) => void> {\n    const regularListeners = this._listeners.get(eventName) || [];\n    const onceListeners = this._onceListeners.get(eventName) || [];\n    return [...regularListeners, ...onceListeners];\n  }\n\n  eventNames(): Array<string | symbol> {\n    const allEvents = new Set([\n      ...this._listeners.keys(),\n      ...this._onceListeners.keys(),\n    ]);\n    return Array.from(allEvents);\n  }\n\n  // Test helper methods\n  reset(): void {\n    this.connected = false;\n    this.disconnected = true;\n    this.removeAllListeners();\n  }\n\n  simulateError(error: Error): void {\n    this.emit('error', error);\n  }\n\n  simulateMessage(message: any): void {\n    this.emit('message', message);\n  }\n\n  // Get internal state for testing\n  getListeners(): Map<string | symbol, Array<(...args: any[]) => void>> {\n    return new Map(this._listeners);\n  }\n\n  getOnceListeners(): Map<string | symbol, Array<(...args: any[]) => void>> {\n    return new Map(this._onceListeners);\n  }\n}\n\n// ...existing code...\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\n//Interface for a generic Socket, similar to Node.js EventEmitter\n//This is a simplified version and may not cover all features of a full EventEmitter\nexport interface Socket {\n  connected: boolean;\n  disconnected: boolean;\n  connect(): void;\n  disconnect(): void;\n  on(eventName: string | symbol, listener: (...args: any[]) => void): this;\n  emit(eventName: string | symbol, ...args: any[]): boolean;\n}\n\nexport const socketExample = (): Socket => ({\n  connected: false,\n  disconnected: true,\n  connect() {\n    this.connected = true;\n    this.disconnected = false;\n    this.emit('connect');\n  },\n  disconnect() {\n    this.connected = false;\n    this.disconnected = true;\n    this.emit('disconnect');\n  },\n  /* v8 ignore next -- @preserve */\n  on() {\n    // Implementation of event listener registration\n    return this;\n  },\n  /* v8 ignore next -- @preserve */\n  emit() {\n    // Implementation of event emission\n    return true;\n  },\n});\n"],"names":["e"],"mappings":"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"}
|
|
1719
|
+
//# sourceMappingURL=io.js.map
|