@peers-app/peers-sdk 0.1.4
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/README.md +1 -0
- package/dist/context/data-context.d.ts +31 -0
- package/dist/context/data-context.js +56 -0
- package/dist/context/index.d.ts +3 -0
- package/dist/context/index.js +19 -0
- package/dist/context/user-context-singleton.d.ts +11 -0
- package/dist/context/user-context-singleton.js +121 -0
- package/dist/context/user-context.d.ts +55 -0
- package/dist/context/user-context.js +205 -0
- package/dist/data/assistants.d.ts +68 -0
- package/dist/data/assistants.js +64 -0
- package/dist/data/change-tracking.d.ts +219 -0
- package/dist/data/change-tracking.js +119 -0
- package/dist/data/channels.d.ts +29 -0
- package/dist/data/channels.js +25 -0
- package/dist/data/data-locks.d.ts +37 -0
- package/dist/data/data-locks.js +180 -0
- package/dist/data/data-locks.test.d.ts +1 -0
- package/dist/data/data-locks.test.js +456 -0
- package/dist/data/device-sync-info.d.ts +19 -0
- package/dist/data/device-sync-info.js +24 -0
- package/dist/data/devices.d.ts +51 -0
- package/dist/data/devices.js +36 -0
- package/dist/data/embeddings.d.ts +47 -0
- package/dist/data/embeddings.js +36 -0
- package/dist/data/files/file-read-stream.d.ts +27 -0
- package/dist/data/files/file-read-stream.js +195 -0
- package/dist/data/files/file-write-stream.d.ts +20 -0
- package/dist/data/files/file-write-stream.js +113 -0
- package/dist/data/files/file.types.d.ts +47 -0
- package/dist/data/files/file.types.js +55 -0
- package/dist/data/files/files.d.ts +28 -0
- package/dist/data/files/files.js +127 -0
- package/dist/data/files/files.test.d.ts +1 -0
- package/dist/data/files/files.test.js +728 -0
- package/dist/data/files/index.d.ts +4 -0
- package/dist/data/files/index.js +23 -0
- package/dist/data/group-member-roles.d.ts +9 -0
- package/dist/data/group-member-roles.js +25 -0
- package/dist/data/group-members.d.ts +39 -0
- package/dist/data/group-members.js +68 -0
- package/dist/data/group-members.test.d.ts +1 -0
- package/dist/data/group-members.test.js +287 -0
- package/dist/data/group-permissions.d.ts +8 -0
- package/dist/data/group-permissions.js +73 -0
- package/dist/data/group-share.d.ts +50 -0
- package/dist/data/group-share.js +196 -0
- package/dist/data/groups.d.ts +50 -0
- package/dist/data/groups.js +73 -0
- package/dist/data/groups.test.d.ts +1 -0
- package/dist/data/groups.test.js +153 -0
- package/dist/data/index.d.ts +31 -0
- package/dist/data/index.js +47 -0
- package/dist/data/knowledge/knowledge-frames.d.ts +34 -0
- package/dist/data/knowledge/knowledge-frames.js +34 -0
- package/dist/data/knowledge/knowledge-links.d.ts +30 -0
- package/dist/data/knowledge/knowledge-links.js +25 -0
- package/dist/data/knowledge/knowledge-values.d.ts +35 -0
- package/dist/data/knowledge/knowledge-values.js +35 -0
- package/dist/data/knowledge/peer-types.d.ts +112 -0
- package/dist/data/knowledge/peer-types.js +27 -0
- package/dist/data/knowledge/predicates.d.ts +34 -0
- package/dist/data/knowledge/predicates.js +27 -0
- package/dist/data/messages.d.ts +57 -0
- package/dist/data/messages.js +97 -0
- package/dist/data/orm/client-proxy.data-source.d.ts +27 -0
- package/dist/data/orm/client-proxy.data-source.js +65 -0
- package/dist/data/orm/cursor.d.ts +25 -0
- package/dist/data/orm/cursor.js +47 -0
- package/dist/data/orm/cursor.test.d.ts +1 -0
- package/dist/data/orm/cursor.test.js +315 -0
- package/dist/data/orm/data-query.d.ts +96 -0
- package/dist/data/orm/data-query.js +208 -0
- package/dist/data/orm/data-query.mongo.d.ts +17 -0
- package/dist/data/orm/data-query.mongo.js +267 -0
- package/dist/data/orm/data-query.mongo.test.d.ts +1 -0
- package/dist/data/orm/data-query.mongo.test.js +398 -0
- package/dist/data/orm/data-query.sqlite.d.ts +14 -0
- package/dist/data/orm/data-query.sqlite.js +297 -0
- package/dist/data/orm/data-query.sqlite.test.d.ts +1 -0
- package/dist/data/orm/data-query.sqlite.test.js +377 -0
- package/dist/data/orm/data-query.test.d.ts +1 -0
- package/dist/data/orm/data-query.test.js +553 -0
- package/dist/data/orm/decorators.d.ts +6 -0
- package/dist/data/orm/decorators.js +21 -0
- package/dist/data/orm/dependency-injection.test.d.ts +1 -0
- package/dist/data/orm/dependency-injection.test.js +171 -0
- package/dist/data/orm/doc.d.ts +26 -0
- package/dist/data/orm/doc.js +124 -0
- package/dist/data/orm/event-registry.d.ts +24 -0
- package/dist/data/orm/event-registry.js +40 -0
- package/dist/data/orm/event-registry.test.d.ts +1 -0
- package/dist/data/orm/event-registry.test.js +44 -0
- package/dist/data/orm/factory.d.ts +8 -0
- package/dist/data/orm/factory.js +147 -0
- package/dist/data/orm/index.d.ts +16 -0
- package/dist/data/orm/index.js +32 -0
- package/dist/data/orm/multi-cursors.d.ts +11 -0
- package/dist/data/orm/multi-cursors.js +146 -0
- package/dist/data/orm/multi-cursors.test.d.ts +1 -0
- package/dist/data/orm/multi-cursors.test.js +455 -0
- package/dist/data/orm/sql-db.d.ts +6 -0
- package/dist/data/orm/sql-db.js +2 -0
- package/dist/data/orm/sql.data-source.d.ts +38 -0
- package/dist/data/orm/sql.data-source.js +379 -0
- package/dist/data/orm/sql.data-source.test.d.ts +1 -0
- package/dist/data/orm/sql.data-source.test.js +406 -0
- package/dist/data/orm/subscribable.data-source.d.ts +25 -0
- package/dist/data/orm/subscribable.data-source.js +72 -0
- package/dist/data/orm/table-container-events.test.d.ts +1 -0
- package/dist/data/orm/table-container-events.test.js +93 -0
- package/dist/data/orm/table-container.d.ts +39 -0
- package/dist/data/orm/table-container.js +96 -0
- package/dist/data/orm/table-definitions.system.d.ts +9 -0
- package/dist/data/orm/table-definitions.system.js +29 -0
- package/dist/data/orm/table-definitions.type.d.ts +19 -0
- package/dist/data/orm/table-definitions.type.js +2 -0
- package/dist/data/orm/table-dependencies.d.ts +32 -0
- package/dist/data/orm/table-dependencies.js +2 -0
- package/dist/data/orm/table.d.ts +42 -0
- package/dist/data/orm/table.event-source.test.d.ts +1 -0
- package/dist/data/orm/table.event-source.test.js +341 -0
- package/dist/data/orm/table.js +244 -0
- package/dist/data/orm/types.d.ts +20 -0
- package/dist/data/orm/types.js +115 -0
- package/dist/data/orm/types.test.d.ts +1 -0
- package/dist/data/orm/types.test.js +71 -0
- package/dist/data/package-permissions.d.ts +7 -0
- package/dist/data/package-permissions.js +18 -0
- package/dist/data/packages.d.ts +92 -0
- package/dist/data/packages.js +90 -0
- package/dist/data/peer-events/peer-event-handlers.d.ts +21 -0
- package/dist/data/peer-events/peer-event-handlers.js +28 -0
- package/dist/data/peer-events/peer-event-types.d.ts +119 -0
- package/dist/data/peer-events/peer-event-types.js +29 -0
- package/dist/data/peer-events/peer-events.d.ts +41 -0
- package/dist/data/peer-events/peer-events.js +102 -0
- package/dist/data/persistent-vars.d.ts +87 -0
- package/dist/data/persistent-vars.js +230 -0
- package/dist/data/tool-tests.d.ts +37 -0
- package/dist/data/tool-tests.js +27 -0
- package/dist/data/tools.d.ts +358 -0
- package/dist/data/tools.js +48 -0
- package/dist/data/user-permissions.d.ts +15 -0
- package/dist/data/user-permissions.js +39 -0
- package/dist/data/user-permissions.test.d.ts +1 -0
- package/dist/data/user-permissions.test.js +252 -0
- package/dist/data/users.d.ts +38 -0
- package/dist/data/users.js +73 -0
- package/dist/data/workflow-logs.d.ts +106 -0
- package/dist/data/workflow-logs.js +67 -0
- package/dist/data/workflow-runs.d.ts +103 -0
- package/dist/data/workflow-runs.js +313 -0
- package/dist/data/workflows.d.ts +16 -0
- package/dist/data/workflows.js +21 -0
- package/dist/device/connection.d.ts +41 -0
- package/dist/device/connection.js +249 -0
- package/dist/device/connection.test.d.ts +1 -0
- package/dist/device/connection.test.js +292 -0
- package/dist/device/device-election.d.ts +36 -0
- package/dist/device/device-election.js +137 -0
- package/dist/device/device.d.ts +22 -0
- package/dist/device/device.js +110 -0
- package/dist/device/device.test.d.ts +1 -0
- package/dist/device/device.test.js +203 -0
- package/dist/device/get-trust-level.d.ts +3 -0
- package/dist/device/get-trust-level.js +87 -0
- package/dist/device/socket.type.d.ts +20 -0
- package/dist/device/socket.type.js +15 -0
- package/dist/device/streamed-socket.d.ts +27 -0
- package/dist/device/streamed-socket.js +154 -0
- package/dist/device/streamed-socket.test.d.ts +1 -0
- package/dist/device/streamed-socket.test.js +44 -0
- package/dist/events.d.ts +35 -0
- package/dist/events.js +128 -0
- package/dist/index.d.ts +33 -0
- package/dist/index.js +50 -0
- package/dist/keys.d.ts +51 -0
- package/dist/keys.js +234 -0
- package/dist/keys.test.d.ts +1 -0
- package/dist/keys.test.js +215 -0
- package/dist/mentions.d.ts +9 -0
- package/dist/mentions.js +46 -0
- package/dist/observable.d.ts +19 -0
- package/dist/observable.js +112 -0
- package/dist/observable.test.d.ts +1 -0
- package/dist/observable.test.js +183 -0
- package/dist/package-loader/get-require.d.ts +10 -0
- package/dist/package-loader/get-require.js +31 -0
- package/dist/package-loader/index.d.ts +1 -0
- package/dist/package-loader/index.js +17 -0
- package/dist/package-loader/package-loader.d.ts +16 -0
- package/dist/package-loader/package-loader.js +102 -0
- package/dist/peers-ui/peers-ui.d.ts +15 -0
- package/dist/peers-ui/peers-ui.js +23 -0
- package/dist/peers-ui/peers-ui.types.d.ts +35 -0
- package/dist/peers-ui/peers-ui.types.js +3 -0
- package/dist/rpc-types.d.ts +45 -0
- package/dist/rpc-types.js +47 -0
- package/dist/serial-json.d.ts +5 -0
- package/dist/serial-json.js +186 -0
- package/dist/serial-json.test.d.ts +1 -0
- package/dist/serial-json.test.js +86 -0
- package/dist/system-ids.d.ts +6 -0
- package/dist/system-ids.js +10 -0
- package/dist/tools/index.d.ts +1 -0
- package/dist/tools/index.js +17 -0
- package/dist/tools/tools-factory.d.ts +5 -0
- package/dist/tools/tools-factory.js +34 -0
- package/dist/types/app-nav.d.ts +18 -0
- package/dist/types/app-nav.js +10 -0
- package/dist/types/assistant-runner-args.d.ts +9 -0
- package/dist/types/assistant-runner-args.js +2 -0
- package/dist/types/field-type.d.ts +37 -0
- package/dist/types/field-type.js +26 -0
- package/dist/types/peer-device.d.ts +40 -0
- package/dist/types/peer-device.js +14 -0
- package/dist/types/peers-package.d.ts +23 -0
- package/dist/types/peers-package.js +2 -0
- package/dist/types/workflow-logger.d.ts +2 -0
- package/dist/types/workflow-logger.js +2 -0
- package/dist/types/workflow-run-context.d.ts +12 -0
- package/dist/types/workflow-run-context.js +2 -0
- package/dist/types/workflow.d.ts +72 -0
- package/dist/types/workflow.js +24 -0
- package/dist/types/zod-types.d.ts +7 -0
- package/dist/types/zod-types.js +12 -0
- package/dist/users.query.d.ts +13 -0
- package/dist/users.query.js +134 -0
- package/dist/utils.d.ts +39 -0
- package/dist/utils.js +240 -0
- package/dist/utils.test.d.ts +1 -0
- package/dist/utils.test.js +140 -0
- package/package.json +50 -0
|
@@ -0,0 +1,297 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.dataFilterToSqlWhere = dataFilterToSqlWhere;
|
|
4
|
+
exports.dataQueryToSqliteQuery = dataQueryToSqliteQuery;
|
|
5
|
+
exports.dataFilterToSqliteTextSearch = dataFilterToSqliteTextSearch;
|
|
6
|
+
exports.substrDirectiveToSqlite = substrDirectiveToSqlite;
|
|
7
|
+
const lodash_1 = require("lodash");
|
|
8
|
+
function dataFilterToSqlWhere(filter) {
|
|
9
|
+
if (isScalarValue(filter)) {
|
|
10
|
+
return dataFieldScalarToSqlValue(filter);
|
|
11
|
+
}
|
|
12
|
+
if ((0, lodash_1.isArray)(filter)) {
|
|
13
|
+
return handleOrArray(filter);
|
|
14
|
+
}
|
|
15
|
+
return handleFilterObject(filter);
|
|
16
|
+
}
|
|
17
|
+
function handleOrArray(filters) {
|
|
18
|
+
const conditions = filters.map(f => dataFilterToSqlWhere(f));
|
|
19
|
+
const orStatement = conditions.join(' OR ').trim();
|
|
20
|
+
return `(${orStatement || '1=1'})`;
|
|
21
|
+
}
|
|
22
|
+
function handleFilterObject(filter) {
|
|
23
|
+
const fieldNames = Object.keys(filter);
|
|
24
|
+
if (!fieldNames.length) {
|
|
25
|
+
return '(1=1)';
|
|
26
|
+
}
|
|
27
|
+
const conditions = fieldNames.map(name => handleFilterField(name, filter[name]));
|
|
28
|
+
return `(${conditions.join(' AND ')})`;
|
|
29
|
+
}
|
|
30
|
+
function handleFilterField(name, value) {
|
|
31
|
+
if (name === '$or') {
|
|
32
|
+
return handleOrOperator(value);
|
|
33
|
+
}
|
|
34
|
+
if (name === '$and') {
|
|
35
|
+
return handleAndOperator(value);
|
|
36
|
+
}
|
|
37
|
+
if (name === '$substr') {
|
|
38
|
+
return substrDirectiveToSqlite(value);
|
|
39
|
+
}
|
|
40
|
+
if (name === '$expr') {
|
|
41
|
+
return handleExprOperator(value);
|
|
42
|
+
}
|
|
43
|
+
return handleFieldCondition(name, value);
|
|
44
|
+
}
|
|
45
|
+
function handleFieldCondition(fieldName, value) {
|
|
46
|
+
const escapedName = escapeFieldName(fieldName);
|
|
47
|
+
if (value === undefined || value === null) {
|
|
48
|
+
return `"${escapedName}" IS NULL`;
|
|
49
|
+
}
|
|
50
|
+
if ((0, lodash_1.isArray)(value)) {
|
|
51
|
+
throw new Error(`Invalid filter: Arrays are not allowed as direct field values. Use $in instead: { ${fieldName}: { $in: [${value.slice(0, 2).map(v => typeof v === 'string' ? `'${v}'` : v).join(', ')}${value.length > 2 ? ', ...' : ''}] } }`);
|
|
52
|
+
}
|
|
53
|
+
if (typeof value === 'object') {
|
|
54
|
+
return handleObjectValue(escapedName, value);
|
|
55
|
+
}
|
|
56
|
+
return `"${escapedName}" = ${dataFieldScalarToSqlValue(value)}`;
|
|
57
|
+
}
|
|
58
|
+
function handleObjectValue(fieldName, value) {
|
|
59
|
+
const keys = Object.keys(value);
|
|
60
|
+
if (keys.length !== 1 || !keys[0].startsWith('$')) {
|
|
61
|
+
throw new Error(`Invalid object value for field ${fieldName}`);
|
|
62
|
+
}
|
|
63
|
+
const operator = keys[0];
|
|
64
|
+
const operatorValue = value[operator];
|
|
65
|
+
return handleFieldOperator(fieldName, operator, operatorValue);
|
|
66
|
+
}
|
|
67
|
+
function handleFieldOperator(fieldName, operator, value) {
|
|
68
|
+
switch (operator) {
|
|
69
|
+
case '$in':
|
|
70
|
+
return handleInOperator(fieldName, value);
|
|
71
|
+
case '$nin':
|
|
72
|
+
return handleNinOperator(fieldName, value);
|
|
73
|
+
case '$matchWords':
|
|
74
|
+
return handleMatchWordsOperator(fieldName, value);
|
|
75
|
+
case '$exists':
|
|
76
|
+
return handleExistsOperator(fieldName, value);
|
|
77
|
+
case '$ne':
|
|
78
|
+
return handleNeOperator(fieldName, value);
|
|
79
|
+
case '$eq':
|
|
80
|
+
case '$gt':
|
|
81
|
+
case '$gte':
|
|
82
|
+
case '$lt':
|
|
83
|
+
case '$lte':
|
|
84
|
+
return handleComparisonOperator(fieldName, operator, value);
|
|
85
|
+
default:
|
|
86
|
+
throw new Error(`Unknown operator: ${operator}`);
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
function handleOrOperator(value) {
|
|
90
|
+
const conditions = value.map(f => dataFilterToSqlWhere(f));
|
|
91
|
+
const orStatement = conditions.join(' OR ').trim();
|
|
92
|
+
return `(${orStatement || '1=1'})`;
|
|
93
|
+
}
|
|
94
|
+
function handleAndOperator(value) {
|
|
95
|
+
if (value.length === 0) {
|
|
96
|
+
return '(1=1)';
|
|
97
|
+
}
|
|
98
|
+
const conditions = value.map(f => dataFilterToSqlWhere(f));
|
|
99
|
+
return `(${conditions.join(' AND ')})`;
|
|
100
|
+
}
|
|
101
|
+
function handleExprOperator(value) {
|
|
102
|
+
let expr = '';
|
|
103
|
+
let lhsSet = false;
|
|
104
|
+
for (const exprPart of Object.values(value)) {
|
|
105
|
+
if (!lhsSet) {
|
|
106
|
+
expr += dataFilterToSqlWhere(exprPart);
|
|
107
|
+
lhsSet = true;
|
|
108
|
+
}
|
|
109
|
+
else {
|
|
110
|
+
let tmp = dataFilterToSqlWhere({ $$$$: exprPart });
|
|
111
|
+
tmp = tmp.substring(7, tmp.length - 1);
|
|
112
|
+
expr += tmp;
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
return expr;
|
|
116
|
+
}
|
|
117
|
+
function handleInOperator(fieldName, value) {
|
|
118
|
+
if (!(0, lodash_1.isArray)(value)) {
|
|
119
|
+
throw new Error(`$in value must be an array`);
|
|
120
|
+
}
|
|
121
|
+
const inValues = value.map(v => dataFilterToSqlWhere(v)).join(', ');
|
|
122
|
+
return `"${fieldName}" IN (${inValues})`;
|
|
123
|
+
}
|
|
124
|
+
function handleNinOperator(fieldName, value) {
|
|
125
|
+
if (!(0, lodash_1.isArray)(value)) {
|
|
126
|
+
throw new Error(`$nin value must be an array`);
|
|
127
|
+
}
|
|
128
|
+
const ninValues = value.map(v => dataFilterToSqlWhere(v)).join(', ');
|
|
129
|
+
return `"${fieldName}" NOT IN (${ninValues})`;
|
|
130
|
+
}
|
|
131
|
+
function handleMatchWordsOperator(fieldName, value) {
|
|
132
|
+
if (typeof value !== 'string') {
|
|
133
|
+
throw new Error(`$matchWords value must be a string`);
|
|
134
|
+
}
|
|
135
|
+
return value.split(' ').map((word) => {
|
|
136
|
+
const escapedWord = word.replace(/'/g, "''");
|
|
137
|
+
return `"${fieldName}" LIKE '%${escapedWord}%'`;
|
|
138
|
+
}).join(' AND ');
|
|
139
|
+
}
|
|
140
|
+
function handleExistsOperator(fieldName, value) {
|
|
141
|
+
if (value) {
|
|
142
|
+
return `"${fieldName}" IS NOT NULL`;
|
|
143
|
+
}
|
|
144
|
+
else {
|
|
145
|
+
return `"${fieldName}" IS NULL`;
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
function handleNeOperator(fieldName, value) {
|
|
149
|
+
const strValue = dataFilterToSqlWhere(value);
|
|
150
|
+
return `("${fieldName}" <> ${strValue} OR "${fieldName}" IS NULL)`;
|
|
151
|
+
}
|
|
152
|
+
function handleComparisonOperator(fieldName, operator, value) {
|
|
153
|
+
const sqlOperator = {
|
|
154
|
+
'$eq': '=',
|
|
155
|
+
'$ne': '<>',
|
|
156
|
+
'$gt': '>',
|
|
157
|
+
'$gte': '>=',
|
|
158
|
+
'$lt': '<',
|
|
159
|
+
'$lte': '<='
|
|
160
|
+
}[operator];
|
|
161
|
+
if (!sqlOperator) {
|
|
162
|
+
throw new Error(`Unknown operator: ${operator}`);
|
|
163
|
+
}
|
|
164
|
+
const strValue = dataFilterToSqlWhere(value);
|
|
165
|
+
return `"${fieldName}" ${sqlOperator} ${strValue}`;
|
|
166
|
+
}
|
|
167
|
+
function escapeFieldName(name) {
|
|
168
|
+
return name.replace(/"/g, `""`);
|
|
169
|
+
}
|
|
170
|
+
function isScalarValue(value) {
|
|
171
|
+
return (typeof value !== 'object' ||
|
|
172
|
+
(0, lodash_1.isDate)(value) ||
|
|
173
|
+
value === null);
|
|
174
|
+
}
|
|
175
|
+
function dataFieldScalarToSqlValue(value) {
|
|
176
|
+
if (value === undefined || value === null) {
|
|
177
|
+
return 'NULL';
|
|
178
|
+
}
|
|
179
|
+
else if (typeof value === 'boolean') {
|
|
180
|
+
return value ? '1' : '0';
|
|
181
|
+
}
|
|
182
|
+
else if (typeof value === 'number') {
|
|
183
|
+
return String(value);
|
|
184
|
+
}
|
|
185
|
+
else if (typeof value === 'string') {
|
|
186
|
+
if (value.startsWith('$')) {
|
|
187
|
+
const escapedValue = value.substring(1).replace(/"/g, `""`);
|
|
188
|
+
return `"${escapedValue}"`;
|
|
189
|
+
}
|
|
190
|
+
else {
|
|
191
|
+
let processedValue = value;
|
|
192
|
+
if (value.startsWith('\\$')) {
|
|
193
|
+
processedValue = value.substring(1);
|
|
194
|
+
}
|
|
195
|
+
const escapedValue = processedValue.replace(/'/g, "''");
|
|
196
|
+
return `'${escapedValue}'`;
|
|
197
|
+
}
|
|
198
|
+
}
|
|
199
|
+
else if (value instanceof Date) {
|
|
200
|
+
return `'${value.toISOString()}'`;
|
|
201
|
+
}
|
|
202
|
+
else {
|
|
203
|
+
throw new Error(`Unknown data field type: ${value}`);
|
|
204
|
+
}
|
|
205
|
+
}
|
|
206
|
+
/**
|
|
207
|
+
* Converts a DataQuery to a complete SQLite query string
|
|
208
|
+
*/
|
|
209
|
+
function dataQueryToSqliteQuery(dataQuery) {
|
|
210
|
+
let orderBy = "1";
|
|
211
|
+
const sortBy = dataQuery.sortBy;
|
|
212
|
+
if (sortBy?.length) {
|
|
213
|
+
const sortBys = sortBy.map(s => String(s));
|
|
214
|
+
orderBy = sortBys.map(s => {
|
|
215
|
+
if (s.startsWith('-')) {
|
|
216
|
+
return `"${s.substring(1)}" DESC`;
|
|
217
|
+
}
|
|
218
|
+
else {
|
|
219
|
+
return `"${s}" ASC`;
|
|
220
|
+
}
|
|
221
|
+
}).join(', ');
|
|
222
|
+
}
|
|
223
|
+
let where = dataFilterToSqlWhere(dataQuery.filter || {}) || '(1=1)';
|
|
224
|
+
if (dataQuery.textSearch?.trim()) {
|
|
225
|
+
const textSearch = dataFilterToSqliteTextSearch(dataQuery);
|
|
226
|
+
if (textSearch) {
|
|
227
|
+
if (where === '(1=1)') {
|
|
228
|
+
where = textSearch;
|
|
229
|
+
}
|
|
230
|
+
else {
|
|
231
|
+
where = `(${where}) AND (${textSearch})`;
|
|
232
|
+
}
|
|
233
|
+
}
|
|
234
|
+
}
|
|
235
|
+
if ((!where.trim()) || where === '()') {
|
|
236
|
+
console.warn(`dataQueryToSqliteQuery: where clause is empty, using '1=1'`);
|
|
237
|
+
where = '(1=1)';
|
|
238
|
+
}
|
|
239
|
+
let sql = `
|
|
240
|
+
SELECT
|
|
241
|
+
*
|
|
242
|
+
FROM "${dataQuery.table.tableName}"
|
|
243
|
+
WHERE ${where}
|
|
244
|
+
ORDER BY ${orderBy}
|
|
245
|
+
`;
|
|
246
|
+
if (dataQuery.page || dataQuery.pageSize) {
|
|
247
|
+
const page = Math.round(dataQuery.page || 1);
|
|
248
|
+
const pageSize = Math.round(dataQuery.pageSize || 100);
|
|
249
|
+
sql += `
|
|
250
|
+
LIMIT ${pageSize}
|
|
251
|
+
OFFSET ${pageSize * (page - 1)}
|
|
252
|
+
`.substring(3);
|
|
253
|
+
}
|
|
254
|
+
sql = sql
|
|
255
|
+
.trim()
|
|
256
|
+
.replace(/ /g, '');
|
|
257
|
+
return sql;
|
|
258
|
+
}
|
|
259
|
+
/**
|
|
260
|
+
* Converts DataQuery text search to SQLite LIKE clauses
|
|
261
|
+
*/
|
|
262
|
+
function dataFilterToSqliteTextSearch(query) {
|
|
263
|
+
const textSearch = query.textSearch;
|
|
264
|
+
if (!textSearch?.trim()) {
|
|
265
|
+
return '';
|
|
266
|
+
}
|
|
267
|
+
const fieldNames = query.table.metaData.fields.map(f => f.name);
|
|
268
|
+
// escape single quotes
|
|
269
|
+
const escapedTextSearch = textSearch.replace(/'/g, "''");
|
|
270
|
+
if (query.textSearchExact) {
|
|
271
|
+
const strFilter = fieldNames.map(name => {
|
|
272
|
+
return `"${name}" LIKE '%${escapedTextSearch}%'`;
|
|
273
|
+
}).join(' OR ');
|
|
274
|
+
return `(${strFilter})`;
|
|
275
|
+
}
|
|
276
|
+
else {
|
|
277
|
+
const words = escapedTextSearch.replace(/\s/mg, ' ').split(' ').filter(s => s);
|
|
278
|
+
const strFilters = [];
|
|
279
|
+
for (const word of words) {
|
|
280
|
+
const strFilter = fieldNames.map(name => {
|
|
281
|
+
return `"${name}" LIKE '%${word}%'`;
|
|
282
|
+
}).join(' OR ');
|
|
283
|
+
strFilters.push(`(${strFilter})`);
|
|
284
|
+
}
|
|
285
|
+
return `(${strFilters.join(' AND ')})`;
|
|
286
|
+
}
|
|
287
|
+
}
|
|
288
|
+
/**
|
|
289
|
+
* Converts $substr directive to SQLite substr function
|
|
290
|
+
*/
|
|
291
|
+
function substrDirectiveToSqlite(substrArgs) {
|
|
292
|
+
const args = substrArgs;
|
|
293
|
+
let arg0 = dataFilterToSqlWhere(args[0]);
|
|
294
|
+
let arg1 = dataFilterToSqlWhere(args[1]);
|
|
295
|
+
let arg2 = dataFilterToSqlWhere(args[2]);
|
|
296
|
+
return `substr(${arg0}, ${arg1}, ${arg2})`;
|
|
297
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,377 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const data_query_sqlite_1 = require("./data-query.sqlite");
|
|
4
|
+
describe('dataFilterToSqlWhere - Restructured Version', () => {
|
|
5
|
+
describe('Main orchestrator function', () => {
|
|
6
|
+
it('should handle scalar values', () => {
|
|
7
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(null)).toBe('NULL');
|
|
8
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(undefined)).toBe('NULL');
|
|
9
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(true)).toBe('1');
|
|
10
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(false)).toBe('0');
|
|
11
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(42)).toBe('42');
|
|
12
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)('test')).toBe("'test'");
|
|
13
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(new Date('2023-01-01T00:00:00.000Z'))).toBe("'2023-01-01T00:00:00.000Z'");
|
|
14
|
+
});
|
|
15
|
+
it('should handle empty filter object', () => {
|
|
16
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)({})).toBe('(1=1)');
|
|
17
|
+
});
|
|
18
|
+
it('should delegate to array handler for arrays', () => {
|
|
19
|
+
const filter = [{ name: 'John' }, { age: 30 }];
|
|
20
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`(("name" = 'John') OR ("age" = 30))`);
|
|
21
|
+
});
|
|
22
|
+
it('should delegate to object handler for filter objects', () => {
|
|
23
|
+
const filter = { name: 'John', age: 30 };
|
|
24
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("name" = 'John' AND "age" = 30)`);
|
|
25
|
+
});
|
|
26
|
+
});
|
|
27
|
+
describe('handleOrArray function', () => {
|
|
28
|
+
it('should handle empty arrays', () => {
|
|
29
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)([])).toBe('(1=1)');
|
|
30
|
+
});
|
|
31
|
+
it('should handle single condition arrays', () => {
|
|
32
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)([{ name: 'John' }])).toBe(`(("name" = 'John'))`);
|
|
33
|
+
});
|
|
34
|
+
it('should handle multiple conditions with OR', () => {
|
|
35
|
+
const filter = [{ name: 'John' }, { age: 30 }, { city: 'NYC' }];
|
|
36
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`(("name" = 'John') OR ("age" = 30) OR ("city" = 'NYC'))`);
|
|
37
|
+
});
|
|
38
|
+
it('should handle complex nested conditions', () => {
|
|
39
|
+
const filter = [
|
|
40
|
+
{ name: 'John', age: 30 },
|
|
41
|
+
{ city: { $in: ['NYC', 'LA'] } }
|
|
42
|
+
];
|
|
43
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`(("name" = 'John' AND "age" = 30) OR ("city" IN ('NYC', 'LA')))`);
|
|
44
|
+
});
|
|
45
|
+
});
|
|
46
|
+
describe('handleFilterObject function', () => {
|
|
47
|
+
it('should handle multiple field conditions with AND', () => {
|
|
48
|
+
const filter = { name: 'John', age: 30, active: true };
|
|
49
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("name" = 'John' AND "age" = 30 AND "active" = 1)`);
|
|
50
|
+
});
|
|
51
|
+
it('should handle single field condition', () => {
|
|
52
|
+
const filter = { name: 'John' };
|
|
53
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("name" = 'John')`);
|
|
54
|
+
});
|
|
55
|
+
});
|
|
56
|
+
describe('Special operators', () => {
|
|
57
|
+
describe('$or operator', () => {
|
|
58
|
+
it('should handle $or with multiple conditions', () => {
|
|
59
|
+
const filter = { $or: [{ name: 'John' }, { age: 30 }] };
|
|
60
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`((("name" = 'John') OR ("age" = 30)))`);
|
|
61
|
+
});
|
|
62
|
+
it('should handle empty $or array', () => {
|
|
63
|
+
const filter = { $or: [] };
|
|
64
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`((1=1))`);
|
|
65
|
+
});
|
|
66
|
+
it('should handle $or with empty objects', () => {
|
|
67
|
+
const filter = { $or: [{}] };
|
|
68
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`(((1=1)))`);
|
|
69
|
+
});
|
|
70
|
+
});
|
|
71
|
+
describe('$and operator', () => {
|
|
72
|
+
it('should handle $and with multiple conditions', () => {
|
|
73
|
+
const filter = { $and: [{ name: 'John' }, { age: 30 }] };
|
|
74
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`((("name" = 'John') AND ("age" = 30)))`);
|
|
75
|
+
});
|
|
76
|
+
it('should handle empty $and array', () => {
|
|
77
|
+
const filter = { $and: [] };
|
|
78
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`((1=1))`);
|
|
79
|
+
});
|
|
80
|
+
it('should handle $and with empty objects', () => {
|
|
81
|
+
const filter = { $and: [{}] };
|
|
82
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`(((1=1)))`);
|
|
83
|
+
});
|
|
84
|
+
});
|
|
85
|
+
describe('$expr operator', () => {
|
|
86
|
+
it('should handle simple expressions', () => {
|
|
87
|
+
const filter = { $expr: [{ $substr: ['$id', 1, 1] }, { $eq: 'a' }] };
|
|
88
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`((substr("id", 1, 1)) = 'a')`);
|
|
89
|
+
});
|
|
90
|
+
it('should handle expressions with column references', () => {
|
|
91
|
+
const filter = { $expr: [{ $substr: ['$col1', 2, 4] }, { $gte: '$col2' }] };
|
|
92
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`((substr("col1", 2, 4)) >= "col2")`);
|
|
93
|
+
});
|
|
94
|
+
});
|
|
95
|
+
describe('$substr operator', () => {
|
|
96
|
+
it('should handle substr with column reference', () => {
|
|
97
|
+
const filter = { $substr: ['$col1', 2, 4] };
|
|
98
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`(substr("col1", 2, 4))`);
|
|
99
|
+
});
|
|
100
|
+
it('should handle substr with string literal', () => {
|
|
101
|
+
const filter = { $substr: ['test string', 1, 4] };
|
|
102
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`(substr('test string', 1, 4))`);
|
|
103
|
+
});
|
|
104
|
+
it('should handle substr with escaped quotes', () => {
|
|
105
|
+
const filter = { $substr: ["test's string", 1, 4] };
|
|
106
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`(substr('test''s string', 1, 4))`);
|
|
107
|
+
});
|
|
108
|
+
});
|
|
109
|
+
});
|
|
110
|
+
describe('Field operators', () => {
|
|
111
|
+
describe('$in operator', () => {
|
|
112
|
+
it('should handle string arrays', () => {
|
|
113
|
+
const filter = { name: { $in: ['John', 'Jane', 'Bob'] } };
|
|
114
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("name" IN ('John', 'Jane', 'Bob'))`);
|
|
115
|
+
});
|
|
116
|
+
it('should handle number arrays', () => {
|
|
117
|
+
const filter = { age: { $in: [20, 30, 40] } };
|
|
118
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("age" IN (20, 30, 40))`);
|
|
119
|
+
});
|
|
120
|
+
it('should handle mixed type arrays', () => {
|
|
121
|
+
const filter = { value: { $in: ['test', 42, null, true] } };
|
|
122
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("value" IN ('test', 42, NULL, 1))`);
|
|
123
|
+
});
|
|
124
|
+
it('should handle empty arrays', () => {
|
|
125
|
+
const filter = { name: { $in: [] } };
|
|
126
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("name" IN ())`);
|
|
127
|
+
});
|
|
128
|
+
it('should throw error for non-array values', () => {
|
|
129
|
+
const filter = { name: { $in: 'John' } };
|
|
130
|
+
expect(() => (0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toThrow('$in value must be an array');
|
|
131
|
+
});
|
|
132
|
+
});
|
|
133
|
+
describe('$nin operator', () => {
|
|
134
|
+
it('should handle string arrays', () => {
|
|
135
|
+
const filter = { name: { $nin: ['John', 'Jane'] } };
|
|
136
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("name" NOT IN ('John', 'Jane'))`);
|
|
137
|
+
});
|
|
138
|
+
it('should handle number arrays', () => {
|
|
139
|
+
const filter = { age: { $nin: [20, 30] } };
|
|
140
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("age" NOT IN (20, 30))`);
|
|
141
|
+
});
|
|
142
|
+
it('should throw error for non-array values', () => {
|
|
143
|
+
const filter = { name: { $nin: 'John' } };
|
|
144
|
+
expect(() => (0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toThrow('$nin value must be an array');
|
|
145
|
+
});
|
|
146
|
+
});
|
|
147
|
+
describe('$matchWords operator', () => {
|
|
148
|
+
it('should handle single word', () => {
|
|
149
|
+
const filter = { description: { $matchWords: 'test' } };
|
|
150
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("description" LIKE '%test%')`);
|
|
151
|
+
});
|
|
152
|
+
it('should handle multiple words', () => {
|
|
153
|
+
const filter = { description: { $matchWords: 'hello world' } };
|
|
154
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("description" LIKE '%hello%' AND "description" LIKE '%world%')`);
|
|
155
|
+
});
|
|
156
|
+
it('should handle words with quotes', () => {
|
|
157
|
+
const filter = { description: { $matchWords: "John's world" } };
|
|
158
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("description" LIKE '%John''s%' AND "description" LIKE '%world%')`);
|
|
159
|
+
});
|
|
160
|
+
it('should handle multiple quotes', () => {
|
|
161
|
+
const filter = { description: { $matchWords: "It's John's" } };
|
|
162
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("description" LIKE '%It''s%' AND "description" LIKE '%John''s%')`);
|
|
163
|
+
});
|
|
164
|
+
it('should throw error for non-string values', () => {
|
|
165
|
+
const filter = { description: { $matchWords: 123 } };
|
|
166
|
+
expect(() => (0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toThrow('$matchWords value must be a string');
|
|
167
|
+
});
|
|
168
|
+
});
|
|
169
|
+
describe('$exists operator', () => {
|
|
170
|
+
it('should handle $exists: true', () => {
|
|
171
|
+
const filter = { name: { $exists: true } };
|
|
172
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("name" IS NOT NULL)`);
|
|
173
|
+
});
|
|
174
|
+
it('should handle $exists: false', () => {
|
|
175
|
+
const filter = { name: { $exists: false } };
|
|
176
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("name" IS NULL)`);
|
|
177
|
+
});
|
|
178
|
+
it('should handle truthy values', () => {
|
|
179
|
+
const filter = { name: { $exists: 1 } };
|
|
180
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("name" IS NOT NULL)`);
|
|
181
|
+
});
|
|
182
|
+
it('should handle falsy values', () => {
|
|
183
|
+
const filter = { name: { $exists: 0 } };
|
|
184
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("name" IS NULL)`);
|
|
185
|
+
});
|
|
186
|
+
});
|
|
187
|
+
describe('$ne operator', () => {
|
|
188
|
+
it('should handle string values', () => {
|
|
189
|
+
const filter = { name: { $ne: 'John' } };
|
|
190
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`(("name" <> 'John' OR "name" IS NULL))`);
|
|
191
|
+
});
|
|
192
|
+
it('should handle number values', () => {
|
|
193
|
+
const filter = { age: { $ne: 30 } };
|
|
194
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`(("age" <> 30 OR "age" IS NULL))`);
|
|
195
|
+
});
|
|
196
|
+
it('should handle null values', () => {
|
|
197
|
+
const filter = { name: { $ne: null } };
|
|
198
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`(("name" <> NULL OR "name" IS NULL))`);
|
|
199
|
+
});
|
|
200
|
+
it('should escape quotes in strings', () => {
|
|
201
|
+
const filter = { name: { $ne: "O'Hara" } };
|
|
202
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`(("name" <> 'O''Hara' OR "name" IS NULL))`);
|
|
203
|
+
});
|
|
204
|
+
});
|
|
205
|
+
describe('Comparison operators', () => {
|
|
206
|
+
it('should handle $eq operator', () => {
|
|
207
|
+
const filter = { age: { $eq: 30 } };
|
|
208
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("age" = 30)`);
|
|
209
|
+
});
|
|
210
|
+
it('should handle $gt operator', () => {
|
|
211
|
+
const filter = { age: { $gt: 30 } };
|
|
212
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("age" > 30)`);
|
|
213
|
+
});
|
|
214
|
+
it('should handle $gte operator', () => {
|
|
215
|
+
const filter = { age: { $gte: 30 } };
|
|
216
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("age" >= 30)`);
|
|
217
|
+
});
|
|
218
|
+
it('should handle $lt operator', () => {
|
|
219
|
+
const filter = { age: { $lt: 30 } };
|
|
220
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("age" < 30)`);
|
|
221
|
+
});
|
|
222
|
+
it('should handle $lte operator', () => {
|
|
223
|
+
const filter = { age: { $lte: 30 } };
|
|
224
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("age" <= 30)`);
|
|
225
|
+
});
|
|
226
|
+
it('should handle string values with comparison operators', () => {
|
|
227
|
+
const filter = { name: { $gt: 'John' } };
|
|
228
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("name" > 'John')`);
|
|
229
|
+
});
|
|
230
|
+
it('should handle date values', () => {
|
|
231
|
+
const date = new Date('2023-01-01T00:00:00.000Z');
|
|
232
|
+
const filter = { created: { $gte: date } };
|
|
233
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("created" >= '2023-01-01T00:00:00.000Z')`);
|
|
234
|
+
});
|
|
235
|
+
});
|
|
236
|
+
});
|
|
237
|
+
describe('Field conditions', () => {
|
|
238
|
+
it('should handle null field values', () => {
|
|
239
|
+
const filter = { name: null };
|
|
240
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("name" IS NULL)`);
|
|
241
|
+
});
|
|
242
|
+
it('should handle undefined field values', () => {
|
|
243
|
+
const filter = { name: undefined };
|
|
244
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("name" IS NULL)`);
|
|
245
|
+
});
|
|
246
|
+
it('should handle simple field values', () => {
|
|
247
|
+
const filter = { name: 'John', age: 30, active: true };
|
|
248
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("name" = 'John' AND "age" = 30 AND "active" = 1)`);
|
|
249
|
+
});
|
|
250
|
+
it('should throw error for direct array usage', () => {
|
|
251
|
+
const filter = { names: ['John', 'Jane'] };
|
|
252
|
+
expect(() => (0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toThrow(`Invalid filter: Arrays are not allowed as direct field values. Use $in instead: { names: { $in: ['John', 'Jane'] } }`);
|
|
253
|
+
});
|
|
254
|
+
it('should throw error for direct array usage with more items', () => {
|
|
255
|
+
const filter = { names: ['John', 'Jane', 'Bob', 'Alice'] };
|
|
256
|
+
expect(() => (0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toThrow(`Invalid filter: Arrays are not allowed as direct field values. Use $in instead: { names: { $in: ['John', 'Jane', ...] } }`);
|
|
257
|
+
});
|
|
258
|
+
});
|
|
259
|
+
describe('Field name escaping', () => {
|
|
260
|
+
it('should escape field names with double quotes', () => {
|
|
261
|
+
const filter = { 'field"name': 'test' };
|
|
262
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("field""name" = 'test')`);
|
|
263
|
+
});
|
|
264
|
+
it('should escape field names with multiple double quotes', () => {
|
|
265
|
+
const filter = { 'field"with"quotes': 'test' };
|
|
266
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("field""with""quotes" = 'test')`);
|
|
267
|
+
});
|
|
268
|
+
it('should handle normal field names', () => {
|
|
269
|
+
const filter = { fieldName: 'test' };
|
|
270
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("fieldName" = 'test')`);
|
|
271
|
+
});
|
|
272
|
+
});
|
|
273
|
+
describe('String value handling', () => {
|
|
274
|
+
it('should escape single quotes in string values', () => {
|
|
275
|
+
const filter = { name: "John's" };
|
|
276
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("name" = 'John''s')`);
|
|
277
|
+
});
|
|
278
|
+
it('should escape multiple single quotes', () => {
|
|
279
|
+
const filter = { name: "It's John's" };
|
|
280
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("name" = 'It''s John''s')`);
|
|
281
|
+
});
|
|
282
|
+
it('should handle column references with $ prefix', () => {
|
|
283
|
+
const filter = { name: '$other_column' };
|
|
284
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("name" = "other_column")`);
|
|
285
|
+
});
|
|
286
|
+
it('should handle escaped $ prefix', () => {
|
|
287
|
+
const filter = { name: '\\$literal' };
|
|
288
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("name" = '$literal')`);
|
|
289
|
+
});
|
|
290
|
+
it('should handle backslash not followed by $', () => {
|
|
291
|
+
const filter = { name: '\\literal' };
|
|
292
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("name" = '\\literal')`);
|
|
293
|
+
});
|
|
294
|
+
it('should escape double quotes in column references', () => {
|
|
295
|
+
const filter = { name: '$col"with"quotes' };
|
|
296
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("name" = "col""with""quotes")`);
|
|
297
|
+
});
|
|
298
|
+
});
|
|
299
|
+
describe('Error cases', () => {
|
|
300
|
+
it('should throw error for unknown operators', () => {
|
|
301
|
+
const filter = { name: { $unknown: 'test' } };
|
|
302
|
+
expect(() => (0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toThrow('Unknown operator: $unknown');
|
|
303
|
+
});
|
|
304
|
+
it('should throw error for invalid object values', () => {
|
|
305
|
+
const filter = { name: { invalidKey: 'test' } };
|
|
306
|
+
expect(() => (0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toThrow('Invalid object value for field name');
|
|
307
|
+
});
|
|
308
|
+
it('should throw error for multiple operators in object', () => {
|
|
309
|
+
const filter = { name: { $eq: 'test', $ne: 'other' } };
|
|
310
|
+
expect(() => (0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toThrow('Invalid object value for field name');
|
|
311
|
+
});
|
|
312
|
+
it('should throw error for unknown scalar types', () => {
|
|
313
|
+
const filter = { name: Symbol('test') };
|
|
314
|
+
expect(() => (0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toThrow('Cannot convert a Symbol value to a string');
|
|
315
|
+
});
|
|
316
|
+
});
|
|
317
|
+
describe('Complex nested scenarios', () => {
|
|
318
|
+
it('should handle deeply nested conditions', () => {
|
|
319
|
+
const filter = {
|
|
320
|
+
$and: [
|
|
321
|
+
{ name: 'John' },
|
|
322
|
+
{
|
|
323
|
+
$or: [
|
|
324
|
+
{ age: { $gte: 30 } },
|
|
325
|
+
{ city: { $in: ['NYC', 'LA'] } }
|
|
326
|
+
]
|
|
327
|
+
}
|
|
328
|
+
]
|
|
329
|
+
};
|
|
330
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`((("name" = 'John') AND (((\"age\" >= 30) OR (\"city\" IN ('NYC', 'LA'))))))`);
|
|
331
|
+
});
|
|
332
|
+
it('should handle mixed operators and conditions', () => {
|
|
333
|
+
const filter = {
|
|
334
|
+
name: { $ne: 'John' },
|
|
335
|
+
$and: [
|
|
336
|
+
{ age: { $gte: 25 } },
|
|
337
|
+
{ age: { $lte: 65 } }
|
|
338
|
+
],
|
|
339
|
+
$or: [
|
|
340
|
+
{ city: 'NYC' },
|
|
341
|
+
{ state: 'CA' }
|
|
342
|
+
]
|
|
343
|
+
};
|
|
344
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`(("name" <> 'John' OR "name" IS NULL) AND (("age" >= 25) AND ("age" <= 65)) AND (("city" = 'NYC') OR ("state" = 'CA')))`);
|
|
345
|
+
});
|
|
346
|
+
it('should handle arrays with complex conditions', () => {
|
|
347
|
+
const filter = [
|
|
348
|
+
{ name: 'John', age: { $gte: 30 } },
|
|
349
|
+
{ city: { $in: ['NYC', 'LA'] }, active: true }
|
|
350
|
+
];
|
|
351
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`(("name" = 'John' AND "age" >= 30) OR ("city" IN ('NYC', 'LA') AND "active" = 1))`);
|
|
352
|
+
});
|
|
353
|
+
});
|
|
354
|
+
describe('Edge cases', () => {
|
|
355
|
+
it('should handle empty strings', () => {
|
|
356
|
+
const filter = { name: '' };
|
|
357
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("name" = '')`);
|
|
358
|
+
});
|
|
359
|
+
it('should handle zero values', () => {
|
|
360
|
+
const filter = { count: 0 };
|
|
361
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("count" = 0)`);
|
|
362
|
+
});
|
|
363
|
+
it('should handle boolean false', () => {
|
|
364
|
+
const filter = { active: false };
|
|
365
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("active" = 0)`);
|
|
366
|
+
});
|
|
367
|
+
it('should handle very long strings', () => {
|
|
368
|
+
const longString = 'a'.repeat(1000);
|
|
369
|
+
const filter = { description: longString };
|
|
370
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("description" = '${longString}')`);
|
|
371
|
+
});
|
|
372
|
+
it('should handle special characters in field names', () => {
|
|
373
|
+
const filter = { 'field-with-dashes': 'test', 'field_with_underscores': 'test2' };
|
|
374
|
+
expect((0, data_query_sqlite_1.dataFilterToSqlWhere)(filter)).toBe(`("field-with-dashes" = 'test' AND "field_with_underscores" = 'test2')`);
|
|
375
|
+
});
|
|
376
|
+
});
|
|
377
|
+
});
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|