@rljson/io 0.0.34 → 0.0.36
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/dump/empty.json +42 -42
- package/dist/conformance-tests/goldens/dump/two-tables.json +57 -57
- package/dist/conformance-tests/goldens/tableCfgs-1.json +123 -0
- package/dist/conformance-tests/goldens/tableCfgs.json +41 -41
- package/dist/conformance-tests/io-conformance.spec.ts +45 -34
- package/dist/io-mem.d.ts +2 -2
- package/dist/io-tools.d.ts +10 -1
- package/dist/io.d.ts +2 -4
- package/dist/io.js +62 -28
- package/package.json +2 -2
|
@@ -1,7 +1,47 @@
|
|
|
1
1
|
{
|
|
2
|
-
"_hash": "
|
|
2
|
+
"_hash": "O6ctI717FuhDeecsczUYLn",
|
|
3
3
|
"tableCfgs": {
|
|
4
4
|
"_data": [
|
|
5
|
+
{
|
|
6
|
+
"key": "revisions",
|
|
7
|
+
"type": "ingredients",
|
|
8
|
+
"isHead": true,
|
|
9
|
+
"isRoot": true,
|
|
10
|
+
"isShared": false,
|
|
11
|
+
"columns": [
|
|
12
|
+
{
|
|
13
|
+
"key": "_hash",
|
|
14
|
+
"type": "string",
|
|
15
|
+
"_hash": "df4oYftB-71Njv9FprRCeg"
|
|
16
|
+
},
|
|
17
|
+
{
|
|
18
|
+
"key": "table",
|
|
19
|
+
"type": "string",
|
|
20
|
+
"_hash": "FmsYjsit04XJi02Sihdwl-"
|
|
21
|
+
},
|
|
22
|
+
{
|
|
23
|
+
"key": "predecessor",
|
|
24
|
+
"type": "string",
|
|
25
|
+
"_hash": "RlAk7Lj-lZf2KDlUk5fekZ"
|
|
26
|
+
},
|
|
27
|
+
{
|
|
28
|
+
"key": "successor",
|
|
29
|
+
"type": "string",
|
|
30
|
+
"_hash": "MJlFtQVty5oUJKVzzj5hyB"
|
|
31
|
+
},
|
|
32
|
+
{
|
|
33
|
+
"key": "timestamp",
|
|
34
|
+
"type": "number",
|
|
35
|
+
"_hash": "XpKYGh23thW6vONgQuoMHf"
|
|
36
|
+
},
|
|
37
|
+
{
|
|
38
|
+
"key": "id",
|
|
39
|
+
"type": "string",
|
|
40
|
+
"_hash": "eq2P3RIaSgy81i7PomNUvn"
|
|
41
|
+
}
|
|
42
|
+
],
|
|
43
|
+
"_hash": "CubBZQUGTMLa5wMqqzHLXz"
|
|
44
|
+
},
|
|
5
45
|
{
|
|
6
46
|
"_hash": "oS4O57K1hbZY9hQzRID9nO",
|
|
7
47
|
"key": "tableCfgs",
|
|
@@ -52,50 +92,10 @@
|
|
|
52
92
|
"_hash": "aPmGTcNng9E1RvlEj1LtxB"
|
|
53
93
|
}
|
|
54
94
|
]
|
|
55
|
-
},
|
|
56
|
-
{
|
|
57
|
-
"key": "revisions",
|
|
58
|
-
"type": "ingredients",
|
|
59
|
-
"isHead": true,
|
|
60
|
-
"isRoot": true,
|
|
61
|
-
"isShared": false,
|
|
62
|
-
"columns": [
|
|
63
|
-
{
|
|
64
|
-
"key": "_hash",
|
|
65
|
-
"type": "string",
|
|
66
|
-
"_hash": "df4oYftB-71Njv9FprRCeg"
|
|
67
|
-
},
|
|
68
|
-
{
|
|
69
|
-
"key": "table",
|
|
70
|
-
"type": "string",
|
|
71
|
-
"_hash": "FmsYjsit04XJi02Sihdwl-"
|
|
72
|
-
},
|
|
73
|
-
{
|
|
74
|
-
"key": "predecessor",
|
|
75
|
-
"type": "string",
|
|
76
|
-
"_hash": "RlAk7Lj-lZf2KDlUk5fekZ"
|
|
77
|
-
},
|
|
78
|
-
{
|
|
79
|
-
"key": "successor",
|
|
80
|
-
"type": "string",
|
|
81
|
-
"_hash": "MJlFtQVty5oUJKVzzj5hyB"
|
|
82
|
-
},
|
|
83
|
-
{
|
|
84
|
-
"key": "timestamp",
|
|
85
|
-
"type": "number",
|
|
86
|
-
"_hash": "XpKYGh23thW6vONgQuoMHf"
|
|
87
|
-
},
|
|
88
|
-
{
|
|
89
|
-
"key": "id",
|
|
90
|
-
"type": "string",
|
|
91
|
-
"_hash": "eq2P3RIaSgy81i7PomNUvn"
|
|
92
|
-
}
|
|
93
|
-
],
|
|
94
|
-
"_hash": "CubBZQUGTMLa5wMqqzHLXz"
|
|
95
95
|
}
|
|
96
96
|
],
|
|
97
97
|
"_tableCfg": "oS4O57K1hbZY9hQzRID9nO",
|
|
98
|
-
"_hash": "
|
|
98
|
+
"_hash": "Rq-oDdFCeBCeW3HTiFMhhN"
|
|
99
99
|
},
|
|
100
100
|
"revisions": {
|
|
101
101
|
"_data": [],
|
|
@@ -1,58 +1,7 @@
|
|
|
1
1
|
{
|
|
2
|
-
"_hash": "
|
|
2
|
+
"_hash": "YqwSqooi7GRv1Fnl7zWaV9",
|
|
3
3
|
"tableCfgs": {
|
|
4
4
|
"_data": [
|
|
5
|
-
{
|
|
6
|
-
"_hash": "oS4O57K1hbZY9hQzRID9nO",
|
|
7
|
-
"key": "tableCfgs",
|
|
8
|
-
"type": "ingredients",
|
|
9
|
-
"isHead": false,
|
|
10
|
-
"isRoot": false,
|
|
11
|
-
"isShared": true,
|
|
12
|
-
"previous": "",
|
|
13
|
-
"columns": [
|
|
14
|
-
{
|
|
15
|
-
"key": "_hash",
|
|
16
|
-
"type": "string",
|
|
17
|
-
"_hash": "df4oYftB-71Njv9FprRCeg"
|
|
18
|
-
},
|
|
19
|
-
{
|
|
20
|
-
"key": "key",
|
|
21
|
-
"type": "string",
|
|
22
|
-
"_hash": "tapEY-QdaJwWhz1PPZoG35"
|
|
23
|
-
},
|
|
24
|
-
{
|
|
25
|
-
"key": "type",
|
|
26
|
-
"type": "string",
|
|
27
|
-
"_hash": "zE7ALsNzu5JPgDD4AxeRCh"
|
|
28
|
-
},
|
|
29
|
-
{
|
|
30
|
-
"key": "isHead",
|
|
31
|
-
"type": "boolean",
|
|
32
|
-
"_hash": "ar8Hajt2UZDn-F21QLMyC_"
|
|
33
|
-
},
|
|
34
|
-
{
|
|
35
|
-
"key": "isRoot",
|
|
36
|
-
"type": "boolean",
|
|
37
|
-
"_hash": "hedsCm_HtiiTuzJyTeMqhm"
|
|
38
|
-
},
|
|
39
|
-
{
|
|
40
|
-
"key": "isShared",
|
|
41
|
-
"type": "boolean",
|
|
42
|
-
"_hash": "pxXOf3-3qCo9Z3FzC9JXSH"
|
|
43
|
-
},
|
|
44
|
-
{
|
|
45
|
-
"key": "previous",
|
|
46
|
-
"type": "string",
|
|
47
|
-
"_hash": "e0TEeDmSvJcLEMtrVxc0oW"
|
|
48
|
-
},
|
|
49
|
-
{
|
|
50
|
-
"key": "columns",
|
|
51
|
-
"type": "jsonArray",
|
|
52
|
-
"_hash": "aPmGTcNng9E1RvlEj1LtxB"
|
|
53
|
-
}
|
|
54
|
-
]
|
|
55
|
-
},
|
|
56
5
|
{
|
|
57
6
|
"key": "revisions",
|
|
58
7
|
"type": "ingredients",
|
|
@@ -94,7 +43,7 @@
|
|
|
94
43
|
"_hash": "CubBZQUGTMLa5wMqqzHLXz"
|
|
95
44
|
},
|
|
96
45
|
{
|
|
97
|
-
"key": "
|
|
46
|
+
"key": "table2",
|
|
98
47
|
"columns": [
|
|
99
48
|
{
|
|
100
49
|
"key": "_hash",
|
|
@@ -116,10 +65,10 @@
|
|
|
116
65
|
"isHead": true,
|
|
117
66
|
"isRoot": true,
|
|
118
67
|
"isShared": false,
|
|
119
|
-
"_hash": "
|
|
68
|
+
"_hash": "GGPEyV5H-4C3fZr_emK8iC"
|
|
120
69
|
},
|
|
121
70
|
{
|
|
122
|
-
"key": "
|
|
71
|
+
"key": "table1",
|
|
123
72
|
"columns": [
|
|
124
73
|
{
|
|
125
74
|
"key": "_hash",
|
|
@@ -141,11 +90,62 @@
|
|
|
141
90
|
"isHead": true,
|
|
142
91
|
"isRoot": true,
|
|
143
92
|
"isShared": false,
|
|
144
|
-
"_hash": "
|
|
93
|
+
"_hash": "LM5fm8eNChH3kE3D38X0Fa"
|
|
94
|
+
},
|
|
95
|
+
{
|
|
96
|
+
"_hash": "oS4O57K1hbZY9hQzRID9nO",
|
|
97
|
+
"key": "tableCfgs",
|
|
98
|
+
"type": "ingredients",
|
|
99
|
+
"isHead": false,
|
|
100
|
+
"isRoot": false,
|
|
101
|
+
"isShared": true,
|
|
102
|
+
"previous": "",
|
|
103
|
+
"columns": [
|
|
104
|
+
{
|
|
105
|
+
"key": "_hash",
|
|
106
|
+
"type": "string",
|
|
107
|
+
"_hash": "df4oYftB-71Njv9FprRCeg"
|
|
108
|
+
},
|
|
109
|
+
{
|
|
110
|
+
"key": "key",
|
|
111
|
+
"type": "string",
|
|
112
|
+
"_hash": "tapEY-QdaJwWhz1PPZoG35"
|
|
113
|
+
},
|
|
114
|
+
{
|
|
115
|
+
"key": "type",
|
|
116
|
+
"type": "string",
|
|
117
|
+
"_hash": "zE7ALsNzu5JPgDD4AxeRCh"
|
|
118
|
+
},
|
|
119
|
+
{
|
|
120
|
+
"key": "isHead",
|
|
121
|
+
"type": "boolean",
|
|
122
|
+
"_hash": "ar8Hajt2UZDn-F21QLMyC_"
|
|
123
|
+
},
|
|
124
|
+
{
|
|
125
|
+
"key": "isRoot",
|
|
126
|
+
"type": "boolean",
|
|
127
|
+
"_hash": "hedsCm_HtiiTuzJyTeMqhm"
|
|
128
|
+
},
|
|
129
|
+
{
|
|
130
|
+
"key": "isShared",
|
|
131
|
+
"type": "boolean",
|
|
132
|
+
"_hash": "pxXOf3-3qCo9Z3FzC9JXSH"
|
|
133
|
+
},
|
|
134
|
+
{
|
|
135
|
+
"key": "previous",
|
|
136
|
+
"type": "string",
|
|
137
|
+
"_hash": "e0TEeDmSvJcLEMtrVxc0oW"
|
|
138
|
+
},
|
|
139
|
+
{
|
|
140
|
+
"key": "columns",
|
|
141
|
+
"type": "jsonArray",
|
|
142
|
+
"_hash": "aPmGTcNng9E1RvlEj1LtxB"
|
|
143
|
+
}
|
|
144
|
+
]
|
|
145
145
|
}
|
|
146
146
|
],
|
|
147
147
|
"_tableCfg": "oS4O57K1hbZY9hQzRID9nO",
|
|
148
|
-
"_hash": "
|
|
148
|
+
"_hash": "hiWhWYx-PreD9EvDuG64rT"
|
|
149
149
|
},
|
|
150
150
|
"revisions": {
|
|
151
151
|
"_data": [],
|
|
@@ -0,0 +1,123 @@
|
|
|
1
|
+
[
|
|
2
|
+
{
|
|
3
|
+
"key": "revisions",
|
|
4
|
+
"type": "ingredients",
|
|
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": "CubBZQUGTMLa5wMqqzHLXz"
|
|
41
|
+
},
|
|
42
|
+
{
|
|
43
|
+
"key": "table0",
|
|
44
|
+
"type": "ingredients",
|
|
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
|
+
"key": "col1",
|
|
61
|
+
"type": "string",
|
|
62
|
+
"_hash": "aD2OJziZ76JeLyPuV4ioKe"
|
|
63
|
+
},
|
|
64
|
+
{
|
|
65
|
+
"key": "col2",
|
|
66
|
+
"type": "string",
|
|
67
|
+
"_hash": "yd4AdprM2Z0vqKieWOibX5"
|
|
68
|
+
}
|
|
69
|
+
],
|
|
70
|
+
"_hash": "SmI9Z47zQXdHKsZYDCLRVN"
|
|
71
|
+
},
|
|
72
|
+
{
|
|
73
|
+
"_hash": "oS4O57K1hbZY9hQzRID9nO",
|
|
74
|
+
"key": "tableCfgs",
|
|
75
|
+
"type": "ingredients",
|
|
76
|
+
"isHead": false,
|
|
77
|
+
"isRoot": false,
|
|
78
|
+
"isShared": true,
|
|
79
|
+
"previous": "",
|
|
80
|
+
"columns": [
|
|
81
|
+
{
|
|
82
|
+
"key": "_hash",
|
|
83
|
+
"type": "string",
|
|
84
|
+
"_hash": "df4oYftB-71Njv9FprRCeg"
|
|
85
|
+
},
|
|
86
|
+
{
|
|
87
|
+
"key": "key",
|
|
88
|
+
"type": "string",
|
|
89
|
+
"_hash": "tapEY-QdaJwWhz1PPZoG35"
|
|
90
|
+
},
|
|
91
|
+
{
|
|
92
|
+
"key": "type",
|
|
93
|
+
"type": "string",
|
|
94
|
+
"_hash": "zE7ALsNzu5JPgDD4AxeRCh"
|
|
95
|
+
},
|
|
96
|
+
{
|
|
97
|
+
"key": "isHead",
|
|
98
|
+
"type": "boolean",
|
|
99
|
+
"_hash": "ar8Hajt2UZDn-F21QLMyC_"
|
|
100
|
+
},
|
|
101
|
+
{
|
|
102
|
+
"key": "isRoot",
|
|
103
|
+
"type": "boolean",
|
|
104
|
+
"_hash": "hedsCm_HtiiTuzJyTeMqhm"
|
|
105
|
+
},
|
|
106
|
+
{
|
|
107
|
+
"key": "isShared",
|
|
108
|
+
"type": "boolean",
|
|
109
|
+
"_hash": "pxXOf3-3qCo9Z3FzC9JXSH"
|
|
110
|
+
},
|
|
111
|
+
{
|
|
112
|
+
"key": "previous",
|
|
113
|
+
"type": "string",
|
|
114
|
+
"_hash": "e0TEeDmSvJcLEMtrVxc0oW"
|
|
115
|
+
},
|
|
116
|
+
{
|
|
117
|
+
"key": "columns",
|
|
118
|
+
"type": "jsonArray",
|
|
119
|
+
"_hash": "aPmGTcNng9E1RvlEj1LtxB"
|
|
120
|
+
}
|
|
121
|
+
]
|
|
122
|
+
}
|
|
123
|
+
]
|
|
@@ -1,6 +1,46 @@
|
|
|
1
1
|
{
|
|
2
2
|
"tableCfgs": {
|
|
3
3
|
"_data": [
|
|
4
|
+
{
|
|
5
|
+
"key": "revisions",
|
|
6
|
+
"type": "ingredients",
|
|
7
|
+
"isHead": true,
|
|
8
|
+
"isRoot": true,
|
|
9
|
+
"isShared": false,
|
|
10
|
+
"columns": [
|
|
11
|
+
{
|
|
12
|
+
"key": "_hash",
|
|
13
|
+
"type": "string",
|
|
14
|
+
"_hash": "df4oYftB-71Njv9FprRCeg"
|
|
15
|
+
},
|
|
16
|
+
{
|
|
17
|
+
"key": "table",
|
|
18
|
+
"type": "string",
|
|
19
|
+
"_hash": "FmsYjsit04XJi02Sihdwl-"
|
|
20
|
+
},
|
|
21
|
+
{
|
|
22
|
+
"key": "predecessor",
|
|
23
|
+
"type": "string",
|
|
24
|
+
"_hash": "RlAk7Lj-lZf2KDlUk5fekZ"
|
|
25
|
+
},
|
|
26
|
+
{
|
|
27
|
+
"key": "successor",
|
|
28
|
+
"type": "string",
|
|
29
|
+
"_hash": "MJlFtQVty5oUJKVzzj5hyB"
|
|
30
|
+
},
|
|
31
|
+
{
|
|
32
|
+
"key": "timestamp",
|
|
33
|
+
"type": "number",
|
|
34
|
+
"_hash": "XpKYGh23thW6vONgQuoMHf"
|
|
35
|
+
},
|
|
36
|
+
{
|
|
37
|
+
"key": "id",
|
|
38
|
+
"type": "string",
|
|
39
|
+
"_hash": "eq2P3RIaSgy81i7PomNUvn"
|
|
40
|
+
}
|
|
41
|
+
],
|
|
42
|
+
"_hash": "CubBZQUGTMLa5wMqqzHLXz"
|
|
43
|
+
},
|
|
4
44
|
{
|
|
5
45
|
"_hash": "oS4O57K1hbZY9hQzRID9nO",
|
|
6
46
|
"key": "tableCfgs",
|
|
@@ -51,49 +91,9 @@
|
|
|
51
91
|
"_hash": "aPmGTcNng9E1RvlEj1LtxB"
|
|
52
92
|
}
|
|
53
93
|
]
|
|
54
|
-
},
|
|
55
|
-
{
|
|
56
|
-
"key": "revisions",
|
|
57
|
-
"type": "ingredients",
|
|
58
|
-
"isHead": true,
|
|
59
|
-
"isRoot": true,
|
|
60
|
-
"isShared": false,
|
|
61
|
-
"columns": [
|
|
62
|
-
{
|
|
63
|
-
"key": "_hash",
|
|
64
|
-
"type": "string",
|
|
65
|
-
"_hash": "df4oYftB-71Njv9FprRCeg"
|
|
66
|
-
},
|
|
67
|
-
{
|
|
68
|
-
"key": "table",
|
|
69
|
-
"type": "string",
|
|
70
|
-
"_hash": "FmsYjsit04XJi02Sihdwl-"
|
|
71
|
-
},
|
|
72
|
-
{
|
|
73
|
-
"key": "predecessor",
|
|
74
|
-
"type": "string",
|
|
75
|
-
"_hash": "RlAk7Lj-lZf2KDlUk5fekZ"
|
|
76
|
-
},
|
|
77
|
-
{
|
|
78
|
-
"key": "successor",
|
|
79
|
-
"type": "string",
|
|
80
|
-
"_hash": "MJlFtQVty5oUJKVzzj5hyB"
|
|
81
|
-
},
|
|
82
|
-
{
|
|
83
|
-
"key": "timestamp",
|
|
84
|
-
"type": "number",
|
|
85
|
-
"_hash": "XpKYGh23thW6vONgQuoMHf"
|
|
86
|
-
},
|
|
87
|
-
{
|
|
88
|
-
"key": "id",
|
|
89
|
-
"type": "string",
|
|
90
|
-
"_hash": "eq2P3RIaSgy81i7PomNUvn"
|
|
91
|
-
}
|
|
92
|
-
],
|
|
93
|
-
"_hash": "CubBZQUGTMLa5wMqqzHLXz"
|
|
94
94
|
}
|
|
95
95
|
],
|
|
96
96
|
"_tableCfg": "oS4O57K1hbZY9hQzRID9nO",
|
|
97
|
-
"_hash": "
|
|
97
|
+
"_hash": "Rq-oDdFCeBCeW3HTiFMhhN"
|
|
98
98
|
}
|
|
99
99
|
}
|
|
@@ -19,7 +19,6 @@ import { hip, hsh, rmhsh } from '@rljson/hash';
|
|
|
19
19
|
import {
|
|
20
20
|
addColumnsToTableCfg,
|
|
21
21
|
exampleTableCfg,
|
|
22
|
-
IngredientsTable,
|
|
23
22
|
Rljson,
|
|
24
23
|
TableCfg,
|
|
25
24
|
TableType,
|
|
@@ -62,6 +61,22 @@ export const runIoConformanceTests = () => {
|
|
|
62
61
|
});
|
|
63
62
|
});
|
|
64
63
|
|
|
64
|
+
describe('isOpen()', () => {
|
|
65
|
+
it('should return false before init, true after and false after close', async () => {
|
|
66
|
+
const setup = testSetup();
|
|
67
|
+
await setup.init();
|
|
68
|
+
|
|
69
|
+
const io = setup.io;
|
|
70
|
+
expect(io.isOpen).toBe(false);
|
|
71
|
+
|
|
72
|
+
await io.init();
|
|
73
|
+
expect(io.isOpen).toBe(true);
|
|
74
|
+
|
|
75
|
+
await io.close();
|
|
76
|
+
expect(io.isOpen).toBe(false);
|
|
77
|
+
});
|
|
78
|
+
});
|
|
79
|
+
|
|
65
80
|
const createExampleTable = async (key: string) => {
|
|
66
81
|
// Register a new table config and generate the table
|
|
67
82
|
const tableCfg: TableCfg = exampleTableCfg({ key });
|
|
@@ -107,13 +122,11 @@ export const runIoConformanceTests = () => {
|
|
|
107
122
|
await io.createOrExtendTable({ tableCfg: tableV2 });
|
|
108
123
|
|
|
109
124
|
// Check the tableCfgs
|
|
110
|
-
const actualTableCfgs =
|
|
111
|
-
._data as unknown as TableCfg[];
|
|
125
|
+
const actualTableCfgs = await ioTools.tableCfgs();
|
|
112
126
|
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
expect((actualTableCfgs[2] as TableCfg).key).toBe('table0');
|
|
127
|
+
await expectGolden('io-conformance/tableCfgs-1.json', ego).toBe(
|
|
128
|
+
actualTableCfgs,
|
|
129
|
+
);
|
|
117
130
|
});
|
|
118
131
|
});
|
|
119
132
|
|
|
@@ -243,10 +256,8 @@ export const runIoConformanceTests = () => {
|
|
|
243
256
|
|
|
244
257
|
it('should add a table and a table config', async () => {
|
|
245
258
|
const tablesFromConfig = async () => {
|
|
246
|
-
const tables =
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
return tables._data.map((e) => e.key);
|
|
259
|
+
const tables = await ioTools.tableCfgs();
|
|
260
|
+
return tables.map((e) => e.key);
|
|
250
261
|
};
|
|
251
262
|
|
|
252
263
|
const physicalTables = async () => await ioTools.allTableKeys();
|
|
@@ -255,33 +266,33 @@ export const runIoConformanceTests = () => {
|
|
|
255
266
|
await createExampleTable('table1');
|
|
256
267
|
|
|
257
268
|
expect(await tablesFromConfig()).toEqual([
|
|
258
|
-
'tableCfgs',
|
|
259
269
|
'revisions',
|
|
260
270
|
'table',
|
|
261
271
|
'table1',
|
|
272
|
+
'tableCfgs',
|
|
262
273
|
]);
|
|
263
274
|
expect(await physicalTables()).toEqual([
|
|
264
|
-
'tableCfgs',
|
|
265
275
|
'revisions',
|
|
266
276
|
'table',
|
|
267
277
|
'table1',
|
|
278
|
+
'tableCfgs',
|
|
268
279
|
]);
|
|
269
280
|
|
|
270
281
|
// Create a second table
|
|
271
282
|
await createExampleTable('table2');
|
|
272
283
|
expect(await tablesFromConfig()).toEqual([
|
|
273
|
-
'tableCfgs',
|
|
274
284
|
'revisions',
|
|
275
285
|
'table',
|
|
276
286
|
'table1',
|
|
277
287
|
'table2',
|
|
288
|
+
'tableCfgs',
|
|
278
289
|
]);
|
|
279
290
|
expect(await physicalTables()).toEqual([
|
|
280
|
-
'tableCfgs',
|
|
281
291
|
'revisions',
|
|
282
292
|
'table',
|
|
283
293
|
'table1',
|
|
284
294
|
'table2',
|
|
295
|
+
'tableCfgs',
|
|
285
296
|
]);
|
|
286
297
|
});
|
|
287
298
|
|
|
@@ -508,7 +519,7 @@ export const runIoConformanceTests = () => {
|
|
|
508
519
|
expect(rowCountAfterFirstWrite).toEqual(2);
|
|
509
520
|
|
|
510
521
|
// Write the same item again
|
|
511
|
-
|
|
522
|
+
await io.write({ data: testData });
|
|
512
523
|
|
|
513
524
|
// Nothing changes because the data is the same
|
|
514
525
|
const rowCountAfterSecondWrite = await io.rowCount(tableName);
|
|
@@ -648,16 +659,16 @@ export const runIoConformanceTests = () => {
|
|
|
648
659
|
{
|
|
649
660
|
array: [1, 2, { a: 10 }],
|
|
650
661
|
boolean: true,
|
|
651
|
-
number:
|
|
652
|
-
object: { a: 1, b:
|
|
653
|
-
string: '
|
|
662
|
+
number: 6,
|
|
663
|
+
object: { a: 1, b: 2 },
|
|
664
|
+
string: 'world',
|
|
654
665
|
},
|
|
655
666
|
{
|
|
656
667
|
array: [1, 2, { a: 10 }],
|
|
657
668
|
boolean: true,
|
|
658
|
-
number:
|
|
659
|
-
object: { a: 1, b:
|
|
660
|
-
string: '
|
|
669
|
+
number: 5,
|
|
670
|
+
object: { a: 1, b: { c: 3 } },
|
|
671
|
+
string: 'hello',
|
|
661
672
|
},
|
|
662
673
|
],
|
|
663
674
|
},
|
|
@@ -678,16 +689,16 @@ export const runIoConformanceTests = () => {
|
|
|
678
689
|
{
|
|
679
690
|
array: [1, 2, { a: 10 }],
|
|
680
691
|
boolean: true,
|
|
681
|
-
number:
|
|
682
|
-
object: { a: 1, b:
|
|
683
|
-
string: '
|
|
692
|
+
number: 6,
|
|
693
|
+
object: { a: 1, b: 2 },
|
|
694
|
+
string: 'world',
|
|
684
695
|
},
|
|
685
696
|
{
|
|
686
697
|
array: [1, 2, { a: 10 }],
|
|
687
698
|
boolean: true,
|
|
688
|
-
number:
|
|
689
|
-
object: { a: 1, b:
|
|
690
|
-
string: '
|
|
699
|
+
number: 5,
|
|
700
|
+
object: { a: 1, b: { c: 3 } },
|
|
701
|
+
string: 'hello',
|
|
691
702
|
},
|
|
692
703
|
],
|
|
693
704
|
},
|
|
@@ -711,16 +722,16 @@ export const runIoConformanceTests = () => {
|
|
|
711
722
|
{
|
|
712
723
|
array: [1, 2, { a: 10 }],
|
|
713
724
|
boolean: true,
|
|
714
|
-
number:
|
|
715
|
-
object: { a: 1, b:
|
|
716
|
-
string: '
|
|
725
|
+
number: 6,
|
|
726
|
+
object: { a: 1, b: 2 },
|
|
727
|
+
string: 'world',
|
|
717
728
|
},
|
|
718
729
|
{
|
|
719
730
|
array: [1, 2, { a: 10 }],
|
|
720
731
|
boolean: true,
|
|
721
|
-
number:
|
|
722
|
-
object: { a: 1, b:
|
|
723
|
-
string: '
|
|
732
|
+
number: 5,
|
|
733
|
+
object: { a: 1, b: { c: 3 } },
|
|
734
|
+
string: 'hello',
|
|
724
735
|
},
|
|
725
736
|
],
|
|
726
737
|
},
|
package/dist/io-mem.d.ts
CHANGED
|
@@ -5,9 +5,9 @@ import { Io } from './io.ts';
|
|
|
5
5
|
* In-Memory implementation of the Rljson Io interface.
|
|
6
6
|
*/
|
|
7
7
|
export declare class IoMem implements Io {
|
|
8
|
-
constructor();
|
|
9
8
|
init(): Promise<void>;
|
|
10
9
|
close(): Promise<void>;
|
|
10
|
+
get isOpen(): boolean;
|
|
11
11
|
static example: () => Promise<IoMem>;
|
|
12
12
|
isReady(): Promise<void>;
|
|
13
13
|
dump(): Promise<Rljson>;
|
|
@@ -28,9 +28,9 @@ export declare class IoMem implements Io {
|
|
|
28
28
|
createOrExtendTable(request: {
|
|
29
29
|
tableCfg: TableCfg;
|
|
30
30
|
}): Promise<void>;
|
|
31
|
-
tableCfgs(): Promise<Rljson>;
|
|
32
31
|
private _ioTools;
|
|
33
32
|
private _isReady;
|
|
33
|
+
private _isOpen;
|
|
34
34
|
private _mem;
|
|
35
35
|
private _init;
|
|
36
36
|
private _initTableCfgs;
|
package/dist/io-tools.d.ts
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { Rljson, TableCfg, TableKey } from '@rljson/rljson';
|
|
1
|
+
import { Rljson, TableCfg, TableKey, TableType } from '@rljson/rljson';
|
|
2
2
|
import { Io } from './io.ts';
|
|
3
3
|
/**
|
|
4
4
|
* Provides utility functions for the Io interface.
|
|
@@ -32,6 +32,11 @@ export declare class IoTools {
|
|
|
32
32
|
* @param rljson - The Rljson object to check
|
|
33
33
|
*/
|
|
34
34
|
throwWhenTablesDoNotExist(rljson: Rljson): Promise<void>;
|
|
35
|
+
/**
|
|
36
|
+
* Returns the current table cfgs of all tables
|
|
37
|
+
* @returns The table configuration of all tables
|
|
38
|
+
*/
|
|
39
|
+
tableCfgs(): Promise<TableCfg[]>;
|
|
35
40
|
/**
|
|
36
41
|
* Returns a list with all table names
|
|
37
42
|
*/
|
|
@@ -64,4 +69,8 @@ export declare class IoTools {
|
|
|
64
69
|
* Throws if the data in the table do not match the table configuration
|
|
65
70
|
*/
|
|
66
71
|
throwWhenTableDataDoesNotMatchCfg(data: Rljson): Promise<void>;
|
|
72
|
+
/**
|
|
73
|
+
* Sorts the data of a table by the hash and updates the table hash in place
|
|
74
|
+
*/
|
|
75
|
+
sortTableDataAndUpdateHash(table: TableType): void;
|
|
67
76
|
}
|
package/dist/io.d.ts
CHANGED
|
@@ -5,6 +5,8 @@ export interface Io {
|
|
|
5
5
|
init(): Promise<void>;
|
|
6
6
|
/** Closes the io */
|
|
7
7
|
close(): Promise<void>;
|
|
8
|
+
/** Returns true if io is opened */
|
|
9
|
+
isOpen: boolean;
|
|
8
10
|
/** A promise resolving once the Io interface is ready
|
|
9
11
|
*
|
|
10
12
|
* 💡 Use @rljson/is-ready
|
|
@@ -30,10 +32,6 @@ export interface Io {
|
|
|
30
32
|
createOrExtendTable(request: {
|
|
31
33
|
tableCfg: TableCfg;
|
|
32
34
|
}): Promise<void>;
|
|
33
|
-
/**
|
|
34
|
-
* Returns a json structure returning current table configurations
|
|
35
|
-
*/
|
|
36
|
-
tableCfgs(): Promise<Rljson>;
|
|
37
35
|
/** Writes Rljson data into the database */
|
|
38
36
|
write(request: {
|
|
39
37
|
data: Rljson;
|
package/dist/io.js
CHANGED
|
@@ -88,13 +88,37 @@ const _IoTools = class _IoTools {
|
|
|
88
88
|
);
|
|
89
89
|
}
|
|
90
90
|
}
|
|
91
|
+
/**
|
|
92
|
+
* Returns the current table cfgs of all tables
|
|
93
|
+
* @returns The table configuration of all tables
|
|
94
|
+
*/
|
|
95
|
+
async tableCfgs() {
|
|
96
|
+
const tableCfgDump = await this.io.dumpTable({ table: "tableCfgs" });
|
|
97
|
+
const tables = tableCfgDump.tableCfgs._data;
|
|
98
|
+
const newestVersion = {};
|
|
99
|
+
for (let i = tables.length - 1; i >= 0; i--) {
|
|
100
|
+
const table = tables[i];
|
|
101
|
+
const existing = newestVersion[table.key];
|
|
102
|
+
if (!existing || existing.columns.length < table.columns.length) {
|
|
103
|
+
newestVersion[table.key] = table;
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
const resultData = Object.values(newestVersion).sort((a, b) => {
|
|
107
|
+
if (a.key < b.key) {
|
|
108
|
+
return -1;
|
|
109
|
+
}
|
|
110
|
+
if (a.key > b.key) {
|
|
111
|
+
return 1;
|
|
112
|
+
}
|
|
113
|
+
return 0;
|
|
114
|
+
});
|
|
115
|
+
return resultData;
|
|
116
|
+
}
|
|
91
117
|
/**
|
|
92
118
|
* Returns a list with all table names
|
|
93
119
|
*/
|
|
94
120
|
async allTableKeys() {
|
|
95
|
-
const result = (await this.
|
|
96
|
-
(e) => e.key
|
|
97
|
-
);
|
|
121
|
+
const result = (await this.tableCfgs()).map((e) => e.key);
|
|
98
122
|
return result;
|
|
99
123
|
}
|
|
100
124
|
/**
|
|
@@ -112,8 +136,8 @@ const _IoTools = class _IoTools {
|
|
|
112
136
|
|
|
113
137
|
*/
|
|
114
138
|
async tableCfgOrNull(table) {
|
|
115
|
-
const tableCfgs = await this.
|
|
116
|
-
const tableCfg = tableCfgs.
|
|
139
|
+
const tableCfgs = await this.tableCfgs();
|
|
140
|
+
const tableCfg = tableCfgs.find((e) => e.key === table);
|
|
117
141
|
return tableCfg ?? null;
|
|
118
142
|
}
|
|
119
143
|
/**
|
|
@@ -203,6 +227,27 @@ ${errors.map((e) => `- ${e}`).join("\n")}`
|
|
|
203
227
|
);
|
|
204
228
|
}
|
|
205
229
|
}
|
|
230
|
+
/**
|
|
231
|
+
* Sorts the data of a table by the hash and updates the table hash in place
|
|
232
|
+
*/
|
|
233
|
+
sortTableDataAndUpdateHash(table) {
|
|
234
|
+
table._data.sort((a, b) => {
|
|
235
|
+
const hashA = a._hash;
|
|
236
|
+
const hashB = b._hash;
|
|
237
|
+
if (hashA < hashB) {
|
|
238
|
+
return -1;
|
|
239
|
+
}
|
|
240
|
+
if (hashA > hashB) {
|
|
241
|
+
return 1;
|
|
242
|
+
}
|
|
243
|
+
return 0;
|
|
244
|
+
});
|
|
245
|
+
table._hash = "";
|
|
246
|
+
hip(table, {
|
|
247
|
+
updateExistingHashes: false,
|
|
248
|
+
throwOnWrongHashes: false
|
|
249
|
+
});
|
|
250
|
+
}
|
|
206
251
|
};
|
|
207
252
|
/**
|
|
208
253
|
* Example object for test purposes
|
|
@@ -217,14 +262,13 @@ __publicField(_IoTools, "example", async () => {
|
|
|
217
262
|
let IoTools = _IoTools;
|
|
218
263
|
// @license
|
|
219
264
|
const _IoMem = class _IoMem {
|
|
220
|
-
// ...........................................................................
|
|
221
|
-
// Constructor & example
|
|
222
265
|
constructor() {
|
|
223
266
|
// ######################
|
|
224
267
|
// Private
|
|
225
268
|
// ######################
|
|
226
269
|
__publicField(this, "_ioTools");
|
|
227
270
|
__publicField(this, "_isReady", new IsReady());
|
|
271
|
+
__publicField(this, "_isOpen", false);
|
|
228
272
|
__publicField(this, "_mem", hip({}));
|
|
229
273
|
// ...........................................................................
|
|
230
274
|
__publicField(this, "_initTableCfgs", () => {
|
|
@@ -235,12 +279,19 @@ const _IoMem = class _IoMem {
|
|
|
235
279
|
});
|
|
236
280
|
});
|
|
237
281
|
}
|
|
282
|
+
// ...........................................................................
|
|
283
|
+
// Constructor & example
|
|
238
284
|
init() {
|
|
285
|
+
this._isOpen = true;
|
|
239
286
|
return this._init();
|
|
240
287
|
}
|
|
241
288
|
close() {
|
|
289
|
+
this._isOpen = false;
|
|
242
290
|
return Promise.resolve();
|
|
243
291
|
}
|
|
292
|
+
get isOpen() {
|
|
293
|
+
return this._isOpen;
|
|
294
|
+
}
|
|
244
295
|
// ...........................................................................
|
|
245
296
|
// General
|
|
246
297
|
isReady() {
|
|
@@ -280,23 +331,6 @@ const _IoMem = class _IoMem {
|
|
|
280
331
|
createOrExtendTable(request) {
|
|
281
332
|
return this._createOrExtendTable(request);
|
|
282
333
|
}
|
|
283
|
-
async tableCfgs() {
|
|
284
|
-
const tables = this._mem.tableCfgs._data;
|
|
285
|
-
const newestVersion = {};
|
|
286
|
-
for (let i = tables.length - 1; i >= 0; i--) {
|
|
287
|
-
const table = tables[i];
|
|
288
|
-
const existing = newestVersion[table.key];
|
|
289
|
-
if (!existing) {
|
|
290
|
-
newestVersion[table.key] = table;
|
|
291
|
-
}
|
|
292
|
-
}
|
|
293
|
-
const resultData = Object.values(newestVersion).reverse();
|
|
294
|
-
return hip({
|
|
295
|
-
tableCfgs: {
|
|
296
|
-
_data: resultData
|
|
297
|
-
}
|
|
298
|
-
});
|
|
299
|
-
}
|
|
300
334
|
// ...........................................................................
|
|
301
335
|
async _init() {
|
|
302
336
|
this._ioTools = new IoTools(this);
|
|
@@ -337,7 +371,7 @@ const _IoMem = class _IoMem {
|
|
|
337
371
|
var _a;
|
|
338
372
|
newConfig = hsh(newConfig);
|
|
339
373
|
this._mem.tableCfgs._data.push(newConfig);
|
|
340
|
-
this.
|
|
374
|
+
this._ioTools.sortTableDataAndUpdateHash(this._mem.tableCfgs);
|
|
341
375
|
const table = {
|
|
342
376
|
_data: [],
|
|
343
377
|
_tableCfg: newConfig._hash
|
|
@@ -353,9 +387,9 @@ const _IoMem = class _IoMem {
|
|
|
353
387
|
}
|
|
354
388
|
newConfig = hsh(newConfig);
|
|
355
389
|
this._mem.tableCfgs._data.push(newConfig);
|
|
390
|
+
this._ioTools.sortTableDataAndUpdateHash(this._mem.tableCfgs);
|
|
356
391
|
const table = this._mem[newConfig.key];
|
|
357
392
|
table._tableCfg = newConfig._hash;
|
|
358
|
-
this._updateTableHash("tableCfgs");
|
|
359
393
|
this._updateTableHash(newConfig.key);
|
|
360
394
|
this._updateGlobalHash();
|
|
361
395
|
}
|
|
@@ -392,7 +426,7 @@ const _IoMem = class _IoMem {
|
|
|
392
426
|
oldTable._data.push(item);
|
|
393
427
|
}
|
|
394
428
|
}
|
|
395
|
-
this.
|
|
429
|
+
this._ioTools.sortTableDataAndUpdateHash(oldTable);
|
|
396
430
|
}
|
|
397
431
|
this._updateGlobalHash();
|
|
398
432
|
}
|
|
@@ -561,4 +595,4 @@ export {
|
|
|
561
595
|
exampleIo,
|
|
562
596
|
exampleTestSetup
|
|
563
597
|
};
|
|
564
|
-
//# 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  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: 'ingredients',\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: 'ingredients',\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 a list with all table names\n   */\n  async allTableKeys(): Promise<string[]> {\n    const result = (await this.io.tableCfgs()).tableCfgs._data.map(\n      (e) => e.key,\n    );\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.io.tableCfgs();\n    const tableCfg = tableCfgs.tableCfgs._data.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      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","// @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  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  constructor() {}\n\n  init(): Promise<void> {\n    return this._init();\n  }\n\n  close(): Promise<void> {\n    return Promise.resolve();\n  }\n\n  static example = async () => {\n    return new IoMem();\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 tableCfgs(): Promise<Rljson> {\n    const tables = this._mem.tableCfgs._data as TableCfg[];\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) {\n        newestVersion[table.key] = table;\n      }\n    }\n\n    const resultData = Object.values(newestVersion).reverse();\n\n    return hip({\n      tableCfgs: {\n        _data: resultData,\n      },\n    } as Rljson);\n  }\n\n  // ######################\n  // Private\n  // ######################\n\n  private _ioTools!: IoTools;\n\n  private _isReady = new IsReady();\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      _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._updateTableHash('tableCfgs');\n\n    // Create a table and write it into the database\n    const table: TableType = {\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\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('tableCfgs');\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._updateTableHash(table);\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    const table = this._mem[request.table] as TableType;\n\n    const result: Rljson = {\n      [request.table]: {\n        _data: 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    } as any;\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  /**\n   * Initializes the io implementation.\n   * @returns The io implementation.\n   */\n  init: () => Promise<void>;\n\n  /**\n   * Tears down the io implementation.\n   * @returns The io implementation.\n   */\n  tearDown: () => 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    init: async () => {\n      // Initialize the io implementation\n    },\n    tearDown: async () => {\n      // Tear down the io implementation\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  /** 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  tableCfgs(): Promise<Rljson>;\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 'ingredients':\n          _writeIngredientRefs(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 _writeIngredientRefs = (\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.ingredientsTable;\n  const parentRowHash = parentRow._hash as string;\n\n  for (const sliceId in parentRow.assign) {\n    if (sliceId.startsWith('_')) {\n      continue;\n    }\n\n    const sliceHash = parentRow.assign[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 = parentRow.layersTable;\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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"}
|
|
598
|
+
//# 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: 'ingredients',\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: 'ingredients',\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 tableCfgDump = await this.io.dumpTable({ table: 'tableCfgs' });\n    const tables = tableCfgDump.tableCfgs._data as TableCfg[];\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    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      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  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    return new IoMem();\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  // ######################\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      _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      _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    const table = this._mem[request.table] as TableType;\n\n    const result: Rljson = {\n      [request.table]: {\n        _data: 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    } as any;\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  /**\n   * Initializes the io implementation.\n   * @returns The io implementation.\n   */\n  init: () => Promise<void>;\n\n  /**\n   * Tears down the io implementation.\n   * @returns The io implementation.\n   */\n  tearDown: () => 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    init: async () => {\n      // Initialize the io implementation\n    },\n    tearDown: async () => {\n      // Tear down the io implementation\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  // 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 'ingredients':\n          _writeIngredientRefs(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 _writeIngredientRefs = (\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.ingredientsTable;\n  const parentRowHash = parentRow._hash as string;\n\n  for (const sliceId in parentRow.assign) {\n    if (sliceId.startsWith('_')) {\n      continue;\n    }\n\n    const sliceHash = parentRow.assign[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 = parentRow.layersTable;\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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"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@rljson/io",
|
|
3
|
-
"version": "0.0.
|
|
3
|
+
"version": "0.0.36",
|
|
4
4
|
"packageManager": "pnpm@10.9.0",
|
|
5
5
|
"description": "Low level interface for reading and writing RLJSON data",
|
|
6
6
|
"homepage": "https://github.com/rljson/io",
|
|
@@ -29,7 +29,7 @@
|
|
|
29
29
|
"updateGoldens": "cross-env UPDATE_GOLDENS=true pnpm test"
|
|
30
30
|
},
|
|
31
31
|
"devDependencies": {
|
|
32
|
-
"@types/node": "^22.15.
|
|
32
|
+
"@types/node": "^22.15.18",
|
|
33
33
|
"@typescript-eslint/eslint-plugin": "^8.32.1",
|
|
34
34
|
"@typescript-eslint/parser": "^8.32.1",
|
|
35
35
|
"@vitest/coverage-v8": "^3.1.3",
|