semola 0.5.2 → 0.5.3
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 +88 -13
- package/dist/cron/builder/index.cjs +166 -0
- package/dist/cron/builder/index.d.cts +28 -0
- package/dist/cron/builder/index.d.cts.map +1 -0
- package/dist/cron/builder/index.d.mts +28 -0
- package/dist/cron/builder/index.d.mts.map +1 -0
- package/dist/cron/builder/index.mjs +163 -0
- package/dist/cron/builder/index.mjs.map +1 -0
- package/dist/cron/builder/types.cjs +27 -0
- package/dist/cron/builder/types.d.cts +79 -0
- package/dist/cron/builder/types.d.cts.map +1 -0
- package/dist/cron/builder/types.d.mts +79 -0
- package/dist/cron/builder/types.d.mts.map +1 -0
- package/dist/cron/builder/types.mjs +28 -0
- package/dist/cron/builder/types.mjs.map +1 -0
- package/dist/cron/core/index.cjs +308 -0
- package/dist/cron/core/index.d.cts +39 -0
- package/dist/cron/core/index.d.cts.map +1 -0
- package/dist/cron/core/index.d.mts +39 -0
- package/dist/cron/core/index.d.mts.map +1 -0
- package/dist/cron/core/index.mjs +310 -0
- package/dist/cron/core/index.mjs.map +1 -0
- package/dist/cron/{scanner.cjs → core/scanner.cjs} +2 -2
- package/dist/cron/{scanner.mjs → core/scanner.mjs} +2 -2
- package/dist/cron/core/scanner.mjs.map +1 -0
- package/dist/cron/{types.d.cts → core/types.d.cts} +1 -1
- package/dist/cron/core/types.d.cts.map +1 -0
- package/dist/cron/{types.d.mts → core/types.d.mts} +1 -1
- package/dist/cron/core/types.d.mts.map +1 -0
- package/dist/errors/types.d.cts +1 -1
- package/dist/errors/types.d.mts +1 -1
- package/dist/lib/cache/index.d.cts +3 -3
- package/dist/lib/cache/index.d.mts +3 -3
- package/dist/lib/cron/index.cjs +12 -275
- package/dist/lib/cron/index.d.cts +4 -39
- package/dist/lib/cron/index.d.mts +4 -39
- package/dist/lib/cron/index.mjs +4 -277
- package/dist/lib/errors/index.d.cts +2 -2
- package/dist/lib/errors/index.d.cts.map +1 -1
- package/dist/lib/errors/index.d.mts +2 -2
- package/dist/lib/errors/index.d.mts.map +1 -1
- package/dist/lib/errors/index.mjs.map +1 -1
- package/dist/lib/i18n/index.cjs +6 -1
- package/dist/lib/i18n/index.d.cts.map +1 -1
- package/dist/lib/i18n/index.d.mts.map +1 -1
- package/dist/lib/i18n/index.mjs +6 -1
- package/dist/lib/i18n/index.mjs.map +1 -1
- package/dist/lib/logging/index.cjs +18 -0
- package/dist/lib/logging/index.d.cts +7 -0
- package/dist/lib/logging/index.d.mts +7 -0
- package/dist/lib/logging/index.mjs +5 -0
- package/dist/lib/orm/index.cjs +20 -0
- package/dist/lib/orm/index.d.cts +7 -0
- package/dist/lib/orm/index.d.mts +7 -0
- package/dist/lib/orm/index.mjs +6 -0
- package/dist/lib/prompts/index.d.cts +8 -8
- package/dist/lib/prompts/index.d.mts +8 -8
- package/dist/lib/pubsub/index.cjs +82 -13
- package/dist/lib/pubsub/index.d.cts +14 -5
- package/dist/lib/pubsub/index.d.cts.map +1 -1
- package/dist/lib/pubsub/index.d.mts +14 -5
- package/dist/lib/pubsub/index.d.mts.map +1 -1
- package/dist/lib/pubsub/index.mjs +82 -13
- package/dist/lib/pubsub/index.mjs.map +1 -1
- package/dist/lib/queue/index.d.cts +2 -2
- package/dist/lib/queue/index.d.mts +2 -2
- package/dist/lib/workflow/index.cjs +534 -0
- package/dist/lib/workflow/index.d.cts +7 -0
- package/dist/lib/workflow/index.d.cts.map +1 -0
- package/dist/lib/workflow/index.d.mts +7 -0
- package/dist/lib/workflow/index.d.mts.map +1 -0
- package/dist/lib/workflow/index.mjs +535 -0
- package/dist/lib/workflow/index.mjs.map +1 -0
- package/dist/logging/core/index.cjs +99 -0
- package/dist/logging/core/index.d.cts +26 -0
- package/dist/logging/core/index.d.cts.map +1 -0
- package/dist/logging/core/index.d.mts +26 -0
- package/dist/logging/core/index.d.mts.map +1 -0
- package/dist/logging/core/index.mjs +99 -0
- package/dist/logging/core/index.mjs.map +1 -0
- package/dist/logging/core/types.cjs +10 -0
- package/dist/logging/core/types.d.cts +22 -0
- package/dist/logging/core/types.d.cts.map +1 -0
- package/dist/logging/core/types.d.mts +22 -0
- package/dist/logging/core/types.d.mts.map +1 -0
- package/dist/logging/core/types.mjs +12 -0
- package/dist/logging/core/types.mjs.map +1 -0
- package/dist/logging/formatter/index.cjs +119 -0
- package/dist/logging/formatter/index.d.cts +27 -0
- package/dist/logging/formatter/index.d.cts.map +1 -0
- package/dist/logging/formatter/index.d.mts +27 -0
- package/dist/logging/formatter/index.d.mts.map +1 -0
- package/dist/logging/formatter/index.mjs +115 -0
- package/dist/logging/formatter/index.mjs.map +1 -0
- package/dist/logging/formatter/types.d.cts +5 -0
- package/dist/logging/formatter/types.d.cts.map +1 -0
- package/dist/logging/formatter/types.d.mts +5 -0
- package/dist/logging/formatter/types.d.mts.map +1 -0
- package/dist/logging/provider/index.cjs +165 -0
- package/dist/logging/provider/index.d.cts +28 -0
- package/dist/logging/provider/index.d.cts.map +1 -0
- package/dist/logging/provider/index.d.mts +28 -0
- package/dist/logging/provider/index.d.mts.map +1 -0
- package/dist/logging/provider/index.mjs +165 -0
- package/dist/logging/provider/index.mjs.map +1 -0
- package/dist/logging/provider/types.d.cts +23 -0
- package/dist/logging/provider/types.d.cts.map +1 -0
- package/dist/logging/provider/types.d.mts +23 -0
- package/dist/logging/provider/types.d.mts.map +1 -0
- package/dist/orm/column.cjs +137 -0
- package/dist/orm/column.d.cts +121 -0
- package/dist/orm/column.d.cts.map +1 -0
- package/dist/orm/column.d.mts +121 -0
- package/dist/orm/column.d.mts.map +1 -0
- package/dist/orm/column.mjs +132 -0
- package/dist/orm/column.mjs.map +1 -0
- package/dist/orm/dialect/index.cjs +14 -0
- package/dist/orm/dialect/index.mjs +16 -0
- package/dist/orm/dialect/index.mjs.map +1 -0
- package/dist/orm/dialect/mysql.cjs +31 -0
- package/dist/orm/dialect/mysql.mjs +33 -0
- package/dist/orm/dialect/mysql.mjs.map +1 -0
- package/dist/orm/dialect/postgres.cjs +23 -0
- package/dist/orm/dialect/postgres.mjs +25 -0
- package/dist/orm/dialect/postgres.mjs.map +1 -0
- package/dist/orm/dialect/sqlite.cjs +31 -0
- package/dist/orm/dialect/sqlite.mjs +33 -0
- package/dist/orm/dialect/sqlite.mjs.map +1 -0
- package/dist/orm/dialect/utils.cjs +8 -0
- package/dist/orm/dialect/utils.mjs +10 -0
- package/dist/orm/dialect/utils.mjs.map +1 -0
- package/dist/orm/internal/table-columns.cjs +31 -0
- package/dist/orm/internal/table-columns.mjs +32 -0
- package/dist/orm/internal/table-columns.mjs.map +1 -0
- package/dist/orm/internal/table-lookup.cjs +35 -0
- package/dist/orm/internal/table-lookup.mjs +35 -0
- package/dist/orm/internal/table-lookup.mjs.map +1 -0
- package/dist/orm/internal/table-relations.cjs +28 -0
- package/dist/orm/internal/table-relations.mjs +29 -0
- package/dist/orm/internal/table-relations.mjs.map +1 -0
- package/dist/orm/migration/config.cjs +7 -0
- package/dist/orm/migration/config.d.cts +7 -0
- package/dist/orm/migration/config.d.cts.map +1 -0
- package/dist/orm/migration/config.d.mts +7 -0
- package/dist/orm/migration/config.d.mts.map +1 -0
- package/dist/orm/migration/config.mjs +8 -0
- package/dist/orm/migration/config.mjs.map +1 -0
- package/dist/orm/migration/types.d.cts +20 -0
- package/dist/orm/migration/types.d.cts.map +1 -0
- package/dist/orm/migration/types.d.mts +20 -0
- package/dist/orm/migration/types.d.mts.map +1 -0
- package/dist/orm/orm.cjs +41 -0
- package/dist/orm/orm.d.cts +18 -0
- package/dist/orm/orm.d.cts.map +1 -0
- package/dist/orm/orm.d.mts +18 -0
- package/dist/orm/orm.d.mts.map +1 -0
- package/dist/orm/orm.mjs +43 -0
- package/dist/orm/orm.mjs.map +1 -0
- package/dist/orm/relation.cjs +18 -0
- package/dist/orm/relation.d.cts +8 -0
- package/dist/orm/relation.d.cts.map +1 -0
- package/dist/orm/relation.d.mts +8 -0
- package/dist/orm/relation.d.mts.map +1 -0
- package/dist/orm/relation.mjs +19 -0
- package/dist/orm/relation.mjs.map +1 -0
- package/dist/orm/runtime/builders/mutations.cjs +29 -0
- package/dist/orm/runtime/builders/mutations.mjs +28 -0
- package/dist/orm/runtime/builders/mutations.mjs.map +1 -0
- package/dist/orm/runtime/builders/select.cjs +18 -0
- package/dist/orm/runtime/builders/select.mjs +19 -0
- package/dist/orm/runtime/builders/select.mjs.map +1 -0
- package/dist/orm/runtime/client.cjs +90 -0
- package/dist/orm/runtime/client.mjs +92 -0
- package/dist/orm/runtime/client.mjs.map +1 -0
- package/dist/orm/runtime/context.cjs +49 -0
- package/dist/orm/runtime/context.mjs +51 -0
- package/dist/orm/runtime/context.mjs.map +1 -0
- package/dist/orm/runtime/dialect/index.cjs +11 -0
- package/dist/orm/runtime/dialect/index.mjs +13 -0
- package/dist/orm/runtime/dialect/index.mjs.map +1 -0
- package/dist/orm/runtime/dialect/mysql.cjs +95 -0
- package/dist/orm/runtime/dialect/mysql.mjs +97 -0
- package/dist/orm/runtime/dialect/mysql.mjs.map +1 -0
- package/dist/orm/runtime/dialect/postgres.cjs +51 -0
- package/dist/orm/runtime/dialect/postgres.mjs +53 -0
- package/dist/orm/runtime/dialect/postgres.mjs.map +1 -0
- package/dist/orm/runtime/dialect/sqlite.cjs +4 -0
- package/dist/orm/runtime/dialect/sqlite.mjs +7 -0
- package/dist/orm/runtime/dialect/sqlite.mjs.map +1 -0
- package/dist/orm/runtime/errors.cjs +19 -0
- package/dist/orm/runtime/errors.mjs +21 -0
- package/dist/orm/runtime/errors.mjs.map +1 -0
- package/dist/orm/runtime/hydrate/many.cjs +46 -0
- package/dist/orm/runtime/hydrate/many.mjs +48 -0
- package/dist/orm/runtime/hydrate/many.mjs.map +1 -0
- package/dist/orm/runtime/hydrate/one.cjs +38 -0
- package/dist/orm/runtime/hydrate/one.mjs +40 -0
- package/dist/orm/runtime/hydrate/one.mjs.map +1 -0
- package/dist/orm/runtime/hydrate.cjs +49 -0
- package/dist/orm/runtime/hydrate.mjs +51 -0
- package/dist/orm/runtime/hydrate.mjs.map +1 -0
- package/dist/orm/runtime/rows.cjs +30 -0
- package/dist/orm/runtime/rows.mjs +31 -0
- package/dist/orm/runtime/rows.mjs.map +1 -0
- package/dist/orm/runtime/utils.cjs +27 -0
- package/dist/orm/runtime/utils.mjs +27 -0
- package/dist/orm/runtime/utils.mjs.map +1 -0
- package/dist/orm/sql/parse-array.cjs +64 -0
- package/dist/orm/sql/parse-array.mjs +66 -0
- package/dist/orm/sql/parse-array.mjs.map +1 -0
- package/dist/orm/sql/plan/select.cjs +36 -0
- package/dist/orm/sql/plan/select.mjs +38 -0
- package/dist/orm/sql/plan/select.mjs.map +1 -0
- package/dist/orm/sql/plan/where/operators.cjs +95 -0
- package/dist/orm/sql/plan/where/operators.mjs +97 -0
- package/dist/orm/sql/plan/where/operators.mjs.map +1 -0
- package/dist/orm/sql/plan/where.cjs +59 -0
- package/dist/orm/sql/plan/where.mjs +61 -0
- package/dist/orm/sql/plan/where.mjs.map +1 -0
- package/dist/orm/sql/serialize/clauses.cjs +36 -0
- package/dist/orm/sql/serialize/clauses.mjs +37 -0
- package/dist/orm/sql/serialize/clauses.mjs.map +1 -0
- package/dist/orm/sql/serialize/joins.cjs +31 -0
- package/dist/orm/sql/serialize/joins.mjs +33 -0
- package/dist/orm/sql/serialize/joins.mjs.map +1 -0
- package/dist/orm/sql/serialize/values.cjs +30 -0
- package/dist/orm/sql/serialize/values.mjs +32 -0
- package/dist/orm/sql/serialize/values.mjs.map +1 -0
- package/dist/orm/sql/serialize/where/predicate.cjs +73 -0
- package/dist/orm/sql/serialize/where/predicate.mjs +75 -0
- package/dist/orm/sql/serialize/where/predicate.mjs.map +1 -0
- package/dist/orm/sql/serialize/where/tree.cjs +26 -0
- package/dist/orm/sql/serialize/where/tree.mjs +28 -0
- package/dist/orm/sql/serialize/where/tree.mjs.map +1 -0
- package/dist/orm/sql/serialize/where.cjs +10 -0
- package/dist/orm/sql/serialize/where.mjs +12 -0
- package/dist/orm/sql/serialize/where.mjs.map +1 -0
- package/dist/orm/sql/serialize.cjs +24 -0
- package/dist/orm/sql/serialize.mjs +25 -0
- package/dist/orm/sql/serialize.mjs.map +1 -0
- package/dist/orm/table.cjs +12 -0
- package/dist/orm/table.d.cts +12 -0
- package/dist/orm/table.d.cts.map +1 -0
- package/dist/orm/table.d.mts +12 -0
- package/dist/orm/table.d.mts.map +1 -0
- package/dist/orm/table.mjs +14 -0
- package/dist/orm/table.mjs.map +1 -0
- package/dist/orm/types.d.cts +183 -0
- package/dist/orm/types.d.cts.map +1 -0
- package/dist/orm/types.d.mts +183 -0
- package/dist/orm/types.d.mts.map +1 -0
- package/dist/workflow/types.d.cts +83 -0
- package/dist/workflow/types.d.cts.map +1 -0
- package/dist/workflow/types.d.mts +83 -0
- package/dist/workflow/types.d.mts.map +1 -0
- package/package.json +29 -3
- package/dist/cron/scanner.mjs.map +0 -1
- package/dist/cron/types.d.cts.map +0 -1
- package/dist/cron/types.d.mts.map +0 -1
- package/dist/lib/cron/index.d.cts.map +0 -1
- package/dist/lib/cron/index.d.mts.map +0 -1
- package/dist/lib/cron/index.mjs.map +0 -1
|
@@ -0,0 +1,66 @@
|
|
|
1
|
+
//#region src/lib/orm/sql/parse-array.ts
|
|
2
|
+
function coerceUnquotedToken(token) {
|
|
3
|
+
const trimmed = token.trim();
|
|
4
|
+
if (trimmed === "NULL") return null;
|
|
5
|
+
if (/^-?\d+(\.\d+)?$/.test(trimmed)) return Number(trimmed);
|
|
6
|
+
const lower = trimmed.toLowerCase();
|
|
7
|
+
if (lower === "true") return true;
|
|
8
|
+
if (lower === "false") return false;
|
|
9
|
+
return trimmed;
|
|
10
|
+
}
|
|
11
|
+
function parsePostgresArrayLiteral(value) {
|
|
12
|
+
const trimmed = value.trim();
|
|
13
|
+
if (!trimmed.startsWith("{") || !trimmed.endsWith("}")) return null;
|
|
14
|
+
const inner = trimmed.slice(1, -1);
|
|
15
|
+
if (inner.length === 0) return [];
|
|
16
|
+
const out = [];
|
|
17
|
+
let token = "";
|
|
18
|
+
let index = 0;
|
|
19
|
+
let inQuotes = false;
|
|
20
|
+
let tokenWasQuoted = false;
|
|
21
|
+
while (index < inner.length) {
|
|
22
|
+
const char = inner[index];
|
|
23
|
+
if (inQuotes) {
|
|
24
|
+
if (char === "\\") {
|
|
25
|
+
const next = inner[index + 1];
|
|
26
|
+
if (next) {
|
|
27
|
+
token += next;
|
|
28
|
+
index += 2;
|
|
29
|
+
continue;
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
if (char === "\"") {
|
|
33
|
+
inQuotes = false;
|
|
34
|
+
index++;
|
|
35
|
+
continue;
|
|
36
|
+
}
|
|
37
|
+
token += char;
|
|
38
|
+
index++;
|
|
39
|
+
continue;
|
|
40
|
+
}
|
|
41
|
+
if (char === "\"") {
|
|
42
|
+
inQuotes = true;
|
|
43
|
+
tokenWasQuoted = true;
|
|
44
|
+
index++;
|
|
45
|
+
continue;
|
|
46
|
+
}
|
|
47
|
+
if (char === ",") {
|
|
48
|
+
if (tokenWasQuoted) out.push(token);
|
|
49
|
+
else out.push(coerceUnquotedToken(token));
|
|
50
|
+
token = "";
|
|
51
|
+
tokenWasQuoted = false;
|
|
52
|
+
index++;
|
|
53
|
+
continue;
|
|
54
|
+
}
|
|
55
|
+
token += char;
|
|
56
|
+
index++;
|
|
57
|
+
}
|
|
58
|
+
if (inQuotes) return null;
|
|
59
|
+
if (tokenWasQuoted) out.push(token);
|
|
60
|
+
else out.push(coerceUnquotedToken(token));
|
|
61
|
+
return out;
|
|
62
|
+
}
|
|
63
|
+
//#endregion
|
|
64
|
+
export { parsePostgresArrayLiteral };
|
|
65
|
+
|
|
66
|
+
//# sourceMappingURL=parse-array.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"parse-array.mjs","names":[],"sources":["../../../src/lib/orm/sql/parse-array.ts"],"sourcesContent":["function coerceUnquotedToken(token: string) {\n const trimmed = token.trim();\n\n if (trimmed === \"NULL\") {\n return null;\n }\n\n if (/^-?\\d+(\\.\\d+)?$/.test(trimmed)) {\n return Number(trimmed);\n }\n\n const lower = trimmed.toLowerCase();\n\n if (lower === \"true\") {\n return true;\n }\n\n if (lower === \"false\") {\n return false;\n }\n\n return trimmed;\n}\n\nexport function parsePostgresArrayLiteral(value: string) {\n const trimmed = value.trim();\n\n if (!trimmed.startsWith(\"{\") || !trimmed.endsWith(\"}\")) {\n return null;\n }\n\n const inner = trimmed.slice(1, -1);\n\n if (inner.length === 0) {\n return [];\n }\n\n const out: unknown[] = [];\n let token = \"\";\n let index = 0;\n let inQuotes = false;\n let tokenWasQuoted = false;\n\n while (index < inner.length) {\n const char = inner[index];\n\n if (inQuotes) {\n if (char === \"\\\\\") {\n const next = inner[index + 1];\n\n if (next) {\n token += next;\n index += 2;\n continue;\n }\n }\n\n if (char === '\"') {\n inQuotes = false;\n index++;\n continue;\n }\n\n token += char;\n index++;\n continue;\n }\n\n if (char === '\"') {\n inQuotes = true;\n tokenWasQuoted = true;\n index++;\n continue;\n }\n\n if (char === \",\") {\n if (tokenWasQuoted) {\n out.push(token);\n } else {\n out.push(coerceUnquotedToken(token));\n }\n\n token = \"\";\n tokenWasQuoted = false;\n index++;\n continue;\n }\n\n token += char;\n index++;\n }\n\n if (inQuotes) {\n return null;\n }\n\n if (tokenWasQuoted) {\n out.push(token);\n } else {\n out.push(coerceUnquotedToken(token));\n }\n\n return out;\n}\n"],"mappings":";AAAA,SAAS,oBAAoB,OAAe;CAC1C,MAAM,UAAU,MAAM,MAAM;AAE5B,KAAI,YAAY,OACd,QAAO;AAGT,KAAI,kBAAkB,KAAK,QAAQ,CACjC,QAAO,OAAO,QAAQ;CAGxB,MAAM,QAAQ,QAAQ,aAAa;AAEnC,KAAI,UAAU,OACZ,QAAO;AAGT,KAAI,UAAU,QACZ,QAAO;AAGT,QAAO;;AAGT,SAAgB,0BAA0B,OAAe;CACvD,MAAM,UAAU,MAAM,MAAM;AAE5B,KAAI,CAAC,QAAQ,WAAW,IAAI,IAAI,CAAC,QAAQ,SAAS,IAAI,CACpD,QAAO;CAGT,MAAM,QAAQ,QAAQ,MAAM,GAAG,GAAG;AAElC,KAAI,MAAM,WAAW,EACnB,QAAO,EAAE;CAGX,MAAM,MAAiB,EAAE;CACzB,IAAI,QAAQ;CACZ,IAAI,QAAQ;CACZ,IAAI,WAAW;CACf,IAAI,iBAAiB;AAErB,QAAO,QAAQ,MAAM,QAAQ;EAC3B,MAAM,OAAO,MAAM;AAEnB,MAAI,UAAU;AACZ,OAAI,SAAS,MAAM;IACjB,MAAM,OAAO,MAAM,QAAQ;AAE3B,QAAI,MAAM;AACR,cAAS;AACT,cAAS;AACT;;;AAIJ,OAAI,SAAS,MAAK;AAChB,eAAW;AACX;AACA;;AAGF,YAAS;AACT;AACA;;AAGF,MAAI,SAAS,MAAK;AAChB,cAAW;AACX,oBAAiB;AACjB;AACA;;AAGF,MAAI,SAAS,KAAK;AAChB,OAAI,eACF,KAAI,KAAK,MAAM;OAEf,KAAI,KAAK,oBAAoB,MAAM,CAAC;AAGtC,WAAQ;AACR,oBAAiB;AACjB;AACA;;AAGF,WAAS;AACT;;AAGF,KAAI,SACF,QAAO;AAGT,KAAI,eACF,KAAI,KAAK,MAAM;KAEf,KAAI,KAAK,oBAAoB,MAAM,CAAC;AAGtC,QAAO"}
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
const require_where = require("./where.cjs");
|
|
2
|
+
//#region src/lib/orm/sql/plan/select.ts
|
|
3
|
+
function buildJoinNodes(include) {
|
|
4
|
+
if (!include) return [];
|
|
5
|
+
const joins = [];
|
|
6
|
+
for (const [relationKey, enabled] of Object.entries(include)) {
|
|
7
|
+
if (enabled !== true) continue;
|
|
8
|
+
joins.push({
|
|
9
|
+
relationKey,
|
|
10
|
+
kind: "left"
|
|
11
|
+
});
|
|
12
|
+
}
|
|
13
|
+
return joins;
|
|
14
|
+
}
|
|
15
|
+
function buildOrderByNodes(orderBy) {
|
|
16
|
+
if (!orderBy) return [];
|
|
17
|
+
const nodes = [];
|
|
18
|
+
for (const [key, direction] of Object.entries(orderBy)) nodes.push({
|
|
19
|
+
key,
|
|
20
|
+
direction: direction === "desc" ? "desc" : "asc"
|
|
21
|
+
});
|
|
22
|
+
return nodes;
|
|
23
|
+
}
|
|
24
|
+
function buildSelectPlan(input = {}) {
|
|
25
|
+
return {
|
|
26
|
+
where: require_where.buildWhereNode(input.where),
|
|
27
|
+
joins: buildJoinNodes(input.include),
|
|
28
|
+
orderBy: buildOrderByNodes(input.orderBy),
|
|
29
|
+
page: {
|
|
30
|
+
limit: input.limit,
|
|
31
|
+
offset: input.offset
|
|
32
|
+
}
|
|
33
|
+
};
|
|
34
|
+
}
|
|
35
|
+
//#endregion
|
|
36
|
+
exports.buildSelectPlan = buildSelectPlan;
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
import { buildWhereNode } from "./where.mjs";
|
|
2
|
+
//#region src/lib/orm/sql/plan/select.ts
|
|
3
|
+
function buildJoinNodes(include) {
|
|
4
|
+
if (!include) return [];
|
|
5
|
+
const joins = [];
|
|
6
|
+
for (const [relationKey, enabled] of Object.entries(include)) {
|
|
7
|
+
if (enabled !== true) continue;
|
|
8
|
+
joins.push({
|
|
9
|
+
relationKey,
|
|
10
|
+
kind: "left"
|
|
11
|
+
});
|
|
12
|
+
}
|
|
13
|
+
return joins;
|
|
14
|
+
}
|
|
15
|
+
function buildOrderByNodes(orderBy) {
|
|
16
|
+
if (!orderBy) return [];
|
|
17
|
+
const nodes = [];
|
|
18
|
+
for (const [key, direction] of Object.entries(orderBy)) nodes.push({
|
|
19
|
+
key,
|
|
20
|
+
direction: direction === "desc" ? "desc" : "asc"
|
|
21
|
+
});
|
|
22
|
+
return nodes;
|
|
23
|
+
}
|
|
24
|
+
function buildSelectPlan(input = {}) {
|
|
25
|
+
return {
|
|
26
|
+
where: buildWhereNode(input.where),
|
|
27
|
+
joins: buildJoinNodes(input.include),
|
|
28
|
+
orderBy: buildOrderByNodes(input.orderBy),
|
|
29
|
+
page: {
|
|
30
|
+
limit: input.limit,
|
|
31
|
+
offset: input.offset
|
|
32
|
+
}
|
|
33
|
+
};
|
|
34
|
+
}
|
|
35
|
+
//#endregion
|
|
36
|
+
export { buildSelectPlan };
|
|
37
|
+
|
|
38
|
+
//# sourceMappingURL=select.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"select.mjs","names":[],"sources":["../../../../src/lib/orm/sql/plan/select.ts"],"sourcesContent":["import type {\n ColDefs,\n JoinNode,\n OrderDirection,\n SelectInput,\n SelectPlan,\n} from \"../../types.js\";\nimport { buildWhereNode } from \"./where.js\";\n\nfunction buildJoinNodes<TRels>(include?: { [K in keyof TRels]?: true }) {\n if (!include) {\n return [];\n }\n\n const joins: JoinNode[] = [];\n\n for (const [relationKey, enabled] of Object.entries(include)) {\n if (enabled !== true) {\n continue;\n }\n\n joins.push({\n relationKey,\n kind: \"left\",\n });\n }\n\n return joins;\n}\n\nfunction buildOrderByNodes<T extends ColDefs>(\n orderBy?: Partial<Record<keyof T, OrderDirection>>,\n) {\n if (!orderBy) {\n return [];\n }\n\n const nodes: Array<{ key: keyof T & string; direction: OrderDirection }> = [];\n\n for (const [key, direction] of Object.entries(orderBy)) {\n nodes.push({\n key: key as keyof T & string,\n direction: direction === \"desc\" ? \"desc\" : \"asc\",\n });\n }\n\n return nodes;\n}\n\nexport function buildSelectPlan<T extends ColDefs, TRels>(\n input: SelectInput<T, TRels> = {},\n): SelectPlan<T> {\n return {\n where: buildWhereNode(input.where),\n joins: buildJoinNodes(input.include),\n orderBy: buildOrderByNodes(input.orderBy),\n page: {\n limit: input.limit,\n offset: input.offset,\n },\n };\n}\n"],"mappings":";;AASA,SAAS,eAAsB,SAAyC;AACtE,KAAI,CAAC,QACH,QAAO,EAAE;CAGX,MAAM,QAAoB,EAAE;AAE5B,MAAK,MAAM,CAAC,aAAa,YAAY,OAAO,QAAQ,QAAQ,EAAE;AAC5D,MAAI,YAAY,KACd;AAGF,QAAM,KAAK;GACT;GACA,MAAM;GACP,CAAC;;AAGJ,QAAO;;AAGT,SAAS,kBACP,SACA;AACA,KAAI,CAAC,QACH,QAAO,EAAE;CAGX,MAAM,QAAqE,EAAE;AAE7E,MAAK,MAAM,CAAC,KAAK,cAAc,OAAO,QAAQ,QAAQ,CACpD,OAAM,KAAK;EACJ;EACL,WAAW,cAAc,SAAS,SAAS;EAC5C,CAAC;AAGJ,QAAO;;AAGT,SAAgB,gBACd,QAA+B,EAAE,EAClB;AACf,QAAO;EACL,OAAO,eAAe,MAAM,MAAM;EAClC,OAAO,eAAe,MAAM,QAAQ;EACpC,SAAS,kBAAkB,MAAM,QAAQ;EACzC,MAAM;GACJ,OAAO,MAAM;GACb,QAAQ,MAAM;GACf;EACF"}
|
|
@@ -0,0 +1,95 @@
|
|
|
1
|
+
//#region src/lib/orm/sql/plan/where/operators.ts
|
|
2
|
+
function toOperatorPredicates(key, condition) {
|
|
3
|
+
const predicates = [];
|
|
4
|
+
if ("startsWith" in condition) predicates.push({
|
|
5
|
+
kind: "predicate",
|
|
6
|
+
key,
|
|
7
|
+
op: "like",
|
|
8
|
+
value: {
|
|
9
|
+
mode: "startsWith",
|
|
10
|
+
value: String(Reflect.get(condition, "startsWith"))
|
|
11
|
+
}
|
|
12
|
+
});
|
|
13
|
+
if ("endsWith" in condition) predicates.push({
|
|
14
|
+
kind: "predicate",
|
|
15
|
+
key,
|
|
16
|
+
op: "like",
|
|
17
|
+
value: {
|
|
18
|
+
mode: "endsWith",
|
|
19
|
+
value: String(Reflect.get(condition, "endsWith"))
|
|
20
|
+
}
|
|
21
|
+
});
|
|
22
|
+
if ("contains" in condition) predicates.push({
|
|
23
|
+
kind: "predicate",
|
|
24
|
+
key,
|
|
25
|
+
op: "like",
|
|
26
|
+
value: {
|
|
27
|
+
mode: "contains",
|
|
28
|
+
value: String(Reflect.get(condition, "contains"))
|
|
29
|
+
}
|
|
30
|
+
});
|
|
31
|
+
if ("gt" in condition) predicates.push({
|
|
32
|
+
kind: "predicate",
|
|
33
|
+
key,
|
|
34
|
+
op: "gt",
|
|
35
|
+
value: Reflect.get(condition, "gt")
|
|
36
|
+
});
|
|
37
|
+
if ("gte" in condition) predicates.push({
|
|
38
|
+
kind: "predicate",
|
|
39
|
+
key,
|
|
40
|
+
op: "gte",
|
|
41
|
+
value: Reflect.get(condition, "gte")
|
|
42
|
+
});
|
|
43
|
+
if ("lt" in condition) predicates.push({
|
|
44
|
+
kind: "predicate",
|
|
45
|
+
key,
|
|
46
|
+
op: "lt",
|
|
47
|
+
value: Reflect.get(condition, "lt")
|
|
48
|
+
});
|
|
49
|
+
if ("lte" in condition) predicates.push({
|
|
50
|
+
kind: "predicate",
|
|
51
|
+
key,
|
|
52
|
+
op: "lte",
|
|
53
|
+
value: Reflect.get(condition, "lte")
|
|
54
|
+
});
|
|
55
|
+
if ("in" in condition) predicates.push({
|
|
56
|
+
kind: "predicate",
|
|
57
|
+
key,
|
|
58
|
+
op: "in",
|
|
59
|
+
value: Reflect.get(condition, "in")
|
|
60
|
+
});
|
|
61
|
+
if ("notIn" in condition) predicates.push({
|
|
62
|
+
kind: "predicate",
|
|
63
|
+
key,
|
|
64
|
+
op: "not_in",
|
|
65
|
+
value: Reflect.get(condition, "notIn")
|
|
66
|
+
});
|
|
67
|
+
if ("equals" in condition) predicates.push({
|
|
68
|
+
kind: "predicate",
|
|
69
|
+
key,
|
|
70
|
+
op: "eq",
|
|
71
|
+
value: Reflect.get(condition, "equals")
|
|
72
|
+
});
|
|
73
|
+
if ("not" in condition) predicates.push({
|
|
74
|
+
kind: "predicate",
|
|
75
|
+
key,
|
|
76
|
+
op: "neq",
|
|
77
|
+
value: Reflect.get(condition, "not")
|
|
78
|
+
});
|
|
79
|
+
if ("isNull" in condition) {
|
|
80
|
+
const isNull = Reflect.get(condition, "isNull");
|
|
81
|
+
if (isNull === true) predicates.push({
|
|
82
|
+
kind: "predicate",
|
|
83
|
+
key,
|
|
84
|
+
op: "is_null"
|
|
85
|
+
});
|
|
86
|
+
if (isNull === false) predicates.push({
|
|
87
|
+
kind: "predicate",
|
|
88
|
+
key,
|
|
89
|
+
op: "is_not_null"
|
|
90
|
+
});
|
|
91
|
+
}
|
|
92
|
+
return predicates;
|
|
93
|
+
}
|
|
94
|
+
//#endregion
|
|
95
|
+
exports.toOperatorPredicates = toOperatorPredicates;
|
|
@@ -0,0 +1,97 @@
|
|
|
1
|
+
//#region src/lib/orm/sql/plan/where/operators.ts
|
|
2
|
+
function toOperatorPredicates(key, condition) {
|
|
3
|
+
const predicates = [];
|
|
4
|
+
if ("startsWith" in condition) predicates.push({
|
|
5
|
+
kind: "predicate",
|
|
6
|
+
key,
|
|
7
|
+
op: "like",
|
|
8
|
+
value: {
|
|
9
|
+
mode: "startsWith",
|
|
10
|
+
value: String(Reflect.get(condition, "startsWith"))
|
|
11
|
+
}
|
|
12
|
+
});
|
|
13
|
+
if ("endsWith" in condition) predicates.push({
|
|
14
|
+
kind: "predicate",
|
|
15
|
+
key,
|
|
16
|
+
op: "like",
|
|
17
|
+
value: {
|
|
18
|
+
mode: "endsWith",
|
|
19
|
+
value: String(Reflect.get(condition, "endsWith"))
|
|
20
|
+
}
|
|
21
|
+
});
|
|
22
|
+
if ("contains" in condition) predicates.push({
|
|
23
|
+
kind: "predicate",
|
|
24
|
+
key,
|
|
25
|
+
op: "like",
|
|
26
|
+
value: {
|
|
27
|
+
mode: "contains",
|
|
28
|
+
value: String(Reflect.get(condition, "contains"))
|
|
29
|
+
}
|
|
30
|
+
});
|
|
31
|
+
if ("gt" in condition) predicates.push({
|
|
32
|
+
kind: "predicate",
|
|
33
|
+
key,
|
|
34
|
+
op: "gt",
|
|
35
|
+
value: Reflect.get(condition, "gt")
|
|
36
|
+
});
|
|
37
|
+
if ("gte" in condition) predicates.push({
|
|
38
|
+
kind: "predicate",
|
|
39
|
+
key,
|
|
40
|
+
op: "gte",
|
|
41
|
+
value: Reflect.get(condition, "gte")
|
|
42
|
+
});
|
|
43
|
+
if ("lt" in condition) predicates.push({
|
|
44
|
+
kind: "predicate",
|
|
45
|
+
key,
|
|
46
|
+
op: "lt",
|
|
47
|
+
value: Reflect.get(condition, "lt")
|
|
48
|
+
});
|
|
49
|
+
if ("lte" in condition) predicates.push({
|
|
50
|
+
kind: "predicate",
|
|
51
|
+
key,
|
|
52
|
+
op: "lte",
|
|
53
|
+
value: Reflect.get(condition, "lte")
|
|
54
|
+
});
|
|
55
|
+
if ("in" in condition) predicates.push({
|
|
56
|
+
kind: "predicate",
|
|
57
|
+
key,
|
|
58
|
+
op: "in",
|
|
59
|
+
value: Reflect.get(condition, "in")
|
|
60
|
+
});
|
|
61
|
+
if ("notIn" in condition) predicates.push({
|
|
62
|
+
kind: "predicate",
|
|
63
|
+
key,
|
|
64
|
+
op: "not_in",
|
|
65
|
+
value: Reflect.get(condition, "notIn")
|
|
66
|
+
});
|
|
67
|
+
if ("equals" in condition) predicates.push({
|
|
68
|
+
kind: "predicate",
|
|
69
|
+
key,
|
|
70
|
+
op: "eq",
|
|
71
|
+
value: Reflect.get(condition, "equals")
|
|
72
|
+
});
|
|
73
|
+
if ("not" in condition) predicates.push({
|
|
74
|
+
kind: "predicate",
|
|
75
|
+
key,
|
|
76
|
+
op: "neq",
|
|
77
|
+
value: Reflect.get(condition, "not")
|
|
78
|
+
});
|
|
79
|
+
if ("isNull" in condition) {
|
|
80
|
+
const isNull = Reflect.get(condition, "isNull");
|
|
81
|
+
if (isNull === true) predicates.push({
|
|
82
|
+
kind: "predicate",
|
|
83
|
+
key,
|
|
84
|
+
op: "is_null"
|
|
85
|
+
});
|
|
86
|
+
if (isNull === false) predicates.push({
|
|
87
|
+
kind: "predicate",
|
|
88
|
+
key,
|
|
89
|
+
op: "is_not_null"
|
|
90
|
+
});
|
|
91
|
+
}
|
|
92
|
+
return predicates;
|
|
93
|
+
}
|
|
94
|
+
//#endregion
|
|
95
|
+
export { toOperatorPredicates };
|
|
96
|
+
|
|
97
|
+
//# sourceMappingURL=operators.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"operators.mjs","names":[],"sources":["../../../../../src/lib/orm/sql/plan/where/operators.ts"],"sourcesContent":["import type { ColDefs, WhereNode } from \"../../../types.js\";\n\nexport function toOperatorPredicates<T extends ColDefs>(\n key: keyof T & string,\n condition: object,\n) {\n const predicates: Array<WhereNode<T>> = [];\n\n if (\"startsWith\" in condition) {\n predicates.push({\n kind: \"predicate\",\n key,\n op: \"like\",\n value: {\n mode: \"startsWith\",\n value: String(Reflect.get(condition, \"startsWith\")),\n },\n });\n }\n\n if (\"endsWith\" in condition) {\n predicates.push({\n kind: \"predicate\",\n key,\n op: \"like\",\n value: {\n mode: \"endsWith\",\n value: String(Reflect.get(condition, \"endsWith\")),\n },\n });\n }\n\n if (\"contains\" in condition) {\n predicates.push({\n kind: \"predicate\",\n key,\n op: \"like\",\n value: {\n mode: \"contains\",\n value: String(Reflect.get(condition, \"contains\")),\n },\n });\n }\n\n if (\"gt\" in condition) {\n predicates.push({\n kind: \"predicate\",\n key,\n op: \"gt\",\n value: Reflect.get(condition, \"gt\"),\n });\n }\n\n if (\"gte\" in condition) {\n predicates.push({\n kind: \"predicate\",\n key,\n op: \"gte\",\n value: Reflect.get(condition, \"gte\"),\n });\n }\n\n if (\"lt\" in condition) {\n predicates.push({\n kind: \"predicate\",\n key,\n op: \"lt\",\n value: Reflect.get(condition, \"lt\"),\n });\n }\n\n if (\"lte\" in condition) {\n predicates.push({\n kind: \"predicate\",\n key,\n op: \"lte\",\n value: Reflect.get(condition, \"lte\"),\n });\n }\n\n if (\"in\" in condition) {\n predicates.push({\n kind: \"predicate\",\n key,\n op: \"in\",\n value: Reflect.get(condition, \"in\"),\n });\n }\n\n if (\"notIn\" in condition) {\n predicates.push({\n kind: \"predicate\",\n key,\n op: \"not_in\",\n value: Reflect.get(condition, \"notIn\"),\n });\n }\n\n if (\"equals\" in condition) {\n predicates.push({\n kind: \"predicate\",\n key,\n op: \"eq\",\n value: Reflect.get(condition, \"equals\"),\n });\n }\n\n if (\"not\" in condition) {\n predicates.push({\n kind: \"predicate\",\n key,\n op: \"neq\",\n value: Reflect.get(condition, \"not\"),\n });\n }\n\n if (\"isNull\" in condition) {\n const isNull = Reflect.get(condition, \"isNull\");\n\n if (isNull === true) {\n predicates.push({ kind: \"predicate\", key, op: \"is_null\" });\n }\n\n if (isNull === false) {\n predicates.push({ kind: \"predicate\", key, op: \"is_not_null\" });\n }\n }\n\n return predicates;\n}\n"],"mappings":";AAEA,SAAgB,qBACd,KACA,WACA;CACA,MAAM,aAAkC,EAAE;AAE1C,KAAI,gBAAgB,UAClB,YAAW,KAAK;EACd,MAAM;EACN;EACA,IAAI;EACJ,OAAO;GACL,MAAM;GACN,OAAO,OAAO,QAAQ,IAAI,WAAW,aAAa,CAAC;GACpD;EACF,CAAC;AAGJ,KAAI,cAAc,UAChB,YAAW,KAAK;EACd,MAAM;EACN;EACA,IAAI;EACJ,OAAO;GACL,MAAM;GACN,OAAO,OAAO,QAAQ,IAAI,WAAW,WAAW,CAAC;GAClD;EACF,CAAC;AAGJ,KAAI,cAAc,UAChB,YAAW,KAAK;EACd,MAAM;EACN;EACA,IAAI;EACJ,OAAO;GACL,MAAM;GACN,OAAO,OAAO,QAAQ,IAAI,WAAW,WAAW,CAAC;GAClD;EACF,CAAC;AAGJ,KAAI,QAAQ,UACV,YAAW,KAAK;EACd,MAAM;EACN;EACA,IAAI;EACJ,OAAO,QAAQ,IAAI,WAAW,KAAK;EACpC,CAAC;AAGJ,KAAI,SAAS,UACX,YAAW,KAAK;EACd,MAAM;EACN;EACA,IAAI;EACJ,OAAO,QAAQ,IAAI,WAAW,MAAM;EACrC,CAAC;AAGJ,KAAI,QAAQ,UACV,YAAW,KAAK;EACd,MAAM;EACN;EACA,IAAI;EACJ,OAAO,QAAQ,IAAI,WAAW,KAAK;EACpC,CAAC;AAGJ,KAAI,SAAS,UACX,YAAW,KAAK;EACd,MAAM;EACN;EACA,IAAI;EACJ,OAAO,QAAQ,IAAI,WAAW,MAAM;EACrC,CAAC;AAGJ,KAAI,QAAQ,UACV,YAAW,KAAK;EACd,MAAM;EACN;EACA,IAAI;EACJ,OAAO,QAAQ,IAAI,WAAW,KAAK;EACpC,CAAC;AAGJ,KAAI,WAAW,UACb,YAAW,KAAK;EACd,MAAM;EACN;EACA,IAAI;EACJ,OAAO,QAAQ,IAAI,WAAW,QAAQ;EACvC,CAAC;AAGJ,KAAI,YAAY,UACd,YAAW,KAAK;EACd,MAAM;EACN;EACA,IAAI;EACJ,OAAO,QAAQ,IAAI,WAAW,SAAS;EACxC,CAAC;AAGJ,KAAI,SAAS,UACX,YAAW,KAAK;EACd,MAAM;EACN;EACA,IAAI;EACJ,OAAO,QAAQ,IAAI,WAAW,MAAM;EACrC,CAAC;AAGJ,KAAI,YAAY,WAAW;EACzB,MAAM,SAAS,QAAQ,IAAI,WAAW,SAAS;AAE/C,MAAI,WAAW,KACb,YAAW,KAAK;GAAE,MAAM;GAAa;GAAK,IAAI;GAAW,CAAC;AAG5D,MAAI,WAAW,MACb,YAAW,KAAK;GAAE,MAAM;GAAa;GAAK,IAAI;GAAe,CAAC;;AAIlE,QAAO"}
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
const require_operators = require("./where/operators.cjs");
|
|
2
|
+
//#region src/lib/orm/sql/plan/where.ts
|
|
3
|
+
function combineWhereNodes(kind, nodes) {
|
|
4
|
+
if (nodes.length === 0) return;
|
|
5
|
+
if (nodes.length === 1) return nodes[0];
|
|
6
|
+
return {
|
|
7
|
+
kind,
|
|
8
|
+
nodes
|
|
9
|
+
};
|
|
10
|
+
}
|
|
11
|
+
function buildWhereNode(where) {
|
|
12
|
+
if (!where) return;
|
|
13
|
+
const nodes = [];
|
|
14
|
+
const andInputs = where.and;
|
|
15
|
+
if (andInputs) for (const entry of andInputs) {
|
|
16
|
+
const node = buildWhereNode(entry);
|
|
17
|
+
if (!node) continue;
|
|
18
|
+
nodes.push(node);
|
|
19
|
+
}
|
|
20
|
+
const orInputs = where.or;
|
|
21
|
+
if (orInputs) {
|
|
22
|
+
const orNodes = [];
|
|
23
|
+
for (const entry of orInputs) {
|
|
24
|
+
const node = buildWhereNode(entry);
|
|
25
|
+
if (!node) continue;
|
|
26
|
+
orNodes.push(node);
|
|
27
|
+
}
|
|
28
|
+
const orNode = combineWhereNodes("or", orNodes);
|
|
29
|
+
if (orNode) nodes.push(orNode);
|
|
30
|
+
}
|
|
31
|
+
for (const [key, condition] of Object.entries(where)) {
|
|
32
|
+
if (key === "and") continue;
|
|
33
|
+
if (key === "or") continue;
|
|
34
|
+
const typedKey = key;
|
|
35
|
+
if (typeof condition !== "object" || condition === null) {
|
|
36
|
+
nodes.push({
|
|
37
|
+
kind: "predicate",
|
|
38
|
+
key: typedKey,
|
|
39
|
+
op: "eq",
|
|
40
|
+
value: condition
|
|
41
|
+
});
|
|
42
|
+
continue;
|
|
43
|
+
}
|
|
44
|
+
const predicates = require_operators.toOperatorPredicates(typedKey, condition);
|
|
45
|
+
if (predicates.length === 0) {
|
|
46
|
+
nodes.push({
|
|
47
|
+
kind: "predicate",
|
|
48
|
+
key: typedKey,
|
|
49
|
+
op: "eq",
|
|
50
|
+
value: condition
|
|
51
|
+
});
|
|
52
|
+
continue;
|
|
53
|
+
}
|
|
54
|
+
for (const predicate of predicates) nodes.push(predicate);
|
|
55
|
+
}
|
|
56
|
+
return combineWhereNodes("and", nodes);
|
|
57
|
+
}
|
|
58
|
+
//#endregion
|
|
59
|
+
exports.buildWhereNode = buildWhereNode;
|
|
@@ -0,0 +1,61 @@
|
|
|
1
|
+
import { toOperatorPredicates } from "./where/operators.mjs";
|
|
2
|
+
//#region src/lib/orm/sql/plan/where.ts
|
|
3
|
+
function combineWhereNodes(kind, nodes) {
|
|
4
|
+
if (nodes.length === 0) return;
|
|
5
|
+
if (nodes.length === 1) return nodes[0];
|
|
6
|
+
return {
|
|
7
|
+
kind,
|
|
8
|
+
nodes
|
|
9
|
+
};
|
|
10
|
+
}
|
|
11
|
+
function buildWhereNode(where) {
|
|
12
|
+
if (!where) return;
|
|
13
|
+
const nodes = [];
|
|
14
|
+
const andInputs = where.and;
|
|
15
|
+
if (andInputs) for (const entry of andInputs) {
|
|
16
|
+
const node = buildWhereNode(entry);
|
|
17
|
+
if (!node) continue;
|
|
18
|
+
nodes.push(node);
|
|
19
|
+
}
|
|
20
|
+
const orInputs = where.or;
|
|
21
|
+
if (orInputs) {
|
|
22
|
+
const orNodes = [];
|
|
23
|
+
for (const entry of orInputs) {
|
|
24
|
+
const node = buildWhereNode(entry);
|
|
25
|
+
if (!node) continue;
|
|
26
|
+
orNodes.push(node);
|
|
27
|
+
}
|
|
28
|
+
const orNode = combineWhereNodes("or", orNodes);
|
|
29
|
+
if (orNode) nodes.push(orNode);
|
|
30
|
+
}
|
|
31
|
+
for (const [key, condition] of Object.entries(where)) {
|
|
32
|
+
if (key === "and") continue;
|
|
33
|
+
if (key === "or") continue;
|
|
34
|
+
const typedKey = key;
|
|
35
|
+
if (typeof condition !== "object" || condition === null) {
|
|
36
|
+
nodes.push({
|
|
37
|
+
kind: "predicate",
|
|
38
|
+
key: typedKey,
|
|
39
|
+
op: "eq",
|
|
40
|
+
value: condition
|
|
41
|
+
});
|
|
42
|
+
continue;
|
|
43
|
+
}
|
|
44
|
+
const predicates = toOperatorPredicates(typedKey, condition);
|
|
45
|
+
if (predicates.length === 0) {
|
|
46
|
+
nodes.push({
|
|
47
|
+
kind: "predicate",
|
|
48
|
+
key: typedKey,
|
|
49
|
+
op: "eq",
|
|
50
|
+
value: condition
|
|
51
|
+
});
|
|
52
|
+
continue;
|
|
53
|
+
}
|
|
54
|
+
for (const predicate of predicates) nodes.push(predicate);
|
|
55
|
+
}
|
|
56
|
+
return combineWhereNodes("and", nodes);
|
|
57
|
+
}
|
|
58
|
+
//#endregion
|
|
59
|
+
export { buildWhereNode };
|
|
60
|
+
|
|
61
|
+
//# sourceMappingURL=where.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"where.mjs","names":[],"sources":["../../../../src/lib/orm/sql/plan/where.ts"],"sourcesContent":["import type { ColDefs, WhereInput, WhereNode } from \"../../types.js\";\nimport {\n isLikePredicateValue,\n isValueOperator,\n type LikeMode,\n type LikePredicateValue,\n} from \"./where/guards.js\";\nimport { toOperatorPredicates } from \"./where/operators.js\";\n\nfunction combineWhereNodes<T extends ColDefs>(\n kind: \"and\" | \"or\",\n nodes: Array<WhereNode<T>>,\n) {\n if (nodes.length === 0) {\n return undefined;\n }\n\n if (nodes.length === 1) {\n return nodes[0];\n }\n\n return { kind, nodes };\n}\n\nexport function buildWhereNode<T extends ColDefs>(where?: WhereInput<T>) {\n if (!where) {\n return undefined;\n }\n\n const nodes: Array<WhereNode<T>> = [];\n\n const andInputs = where.and;\n\n if (andInputs) {\n for (const entry of andInputs) {\n const node = buildWhereNode(entry);\n\n if (!node) {\n continue;\n }\n\n nodes.push(node);\n }\n }\n\n const orInputs = where.or;\n\n if (orInputs) {\n const orNodes: Array<WhereNode<T>> = [];\n\n for (const entry of orInputs) {\n const node = buildWhereNode(entry);\n\n if (!node) {\n continue;\n }\n\n orNodes.push(node);\n }\n\n const orNode = combineWhereNodes(\"or\", orNodes);\n\n if (orNode) {\n nodes.push(orNode);\n }\n }\n\n for (const [key, condition] of Object.entries(where)) {\n if (key === \"and\") {\n continue;\n }\n\n if (key === \"or\") {\n continue;\n }\n\n const typedKey = key as keyof T & string;\n\n if (typeof condition !== \"object\" || condition === null) {\n nodes.push({\n kind: \"predicate\",\n key: typedKey,\n op: \"eq\",\n value: condition,\n });\n\n continue;\n }\n\n const predicates = toOperatorPredicates(typedKey, condition);\n\n if (predicates.length === 0) {\n nodes.push({\n kind: \"predicate\",\n key: typedKey,\n op: \"eq\",\n value: condition,\n });\n continue;\n }\n\n for (const predicate of predicates) {\n nodes.push(predicate);\n }\n }\n\n return combineWhereNodes(\"and\", nodes);\n}\n\nexport { isLikePredicateValue, isValueOperator };\nexport type { LikeMode, LikePredicateValue };\n"],"mappings":";;AASA,SAAS,kBACP,MACA,OACA;AACA,KAAI,MAAM,WAAW,EACnB;AAGF,KAAI,MAAM,WAAW,EACnB,QAAO,MAAM;AAGf,QAAO;EAAE;EAAM;EAAO;;AAGxB,SAAgB,eAAkC,OAAuB;AACvE,KAAI,CAAC,MACH;CAGF,MAAM,QAA6B,EAAE;CAErC,MAAM,YAAY,MAAM;AAExB,KAAI,UACF,MAAK,MAAM,SAAS,WAAW;EAC7B,MAAM,OAAO,eAAe,MAAM;AAElC,MAAI,CAAC,KACH;AAGF,QAAM,KAAK,KAAK;;CAIpB,MAAM,WAAW,MAAM;AAEvB,KAAI,UAAU;EACZ,MAAM,UAA+B,EAAE;AAEvC,OAAK,MAAM,SAAS,UAAU;GAC5B,MAAM,OAAO,eAAe,MAAM;AAElC,OAAI,CAAC,KACH;AAGF,WAAQ,KAAK,KAAK;;EAGpB,MAAM,SAAS,kBAAkB,MAAM,QAAQ;AAE/C,MAAI,OACF,OAAM,KAAK,OAAO;;AAItB,MAAK,MAAM,CAAC,KAAK,cAAc,OAAO,QAAQ,MAAM,EAAE;AACpD,MAAI,QAAQ,MACV;AAGF,MAAI,QAAQ,KACV;EAGF,MAAM,WAAW;AAEjB,MAAI,OAAO,cAAc,YAAY,cAAc,MAAM;AACvD,SAAM,KAAK;IACT,MAAM;IACN,KAAK;IACL,IAAI;IACJ,OAAO;IACR,CAAC;AAEF;;EAGF,MAAM,aAAa,qBAAqB,UAAU,UAAU;AAE5D,MAAI,WAAW,WAAW,GAAG;AAC3B,SAAM,KAAK;IACT,MAAM;IACN,KAAK;IACL,IAAI;IACJ,OAAO;IACR,CAAC;AACF;;AAGF,OAAK,MAAM,aAAa,WACtB,OAAM,KAAK,UAAU;;AAIzB,QAAO,kBAAkB,OAAO,MAAM"}
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
//#region src/lib/orm/sql/serialize/clauses.ts
|
|
2
|
+
function buildLimitClause(sql, limit, offset, dialect) {
|
|
3
|
+
if (limit != null && offset != null) return sql`LIMIT ${limit} OFFSET ${offset}`;
|
|
4
|
+
if (limit != null) return sql`LIMIT ${limit}`;
|
|
5
|
+
if (offset != null) {
|
|
6
|
+
if (dialect === "mysql") return sql`LIMIT 18446744073709551615 OFFSET ${offset}`;
|
|
7
|
+
if (dialect === "sqlite") return sql`LIMIT -1 OFFSET ${offset}`;
|
|
8
|
+
return sql`OFFSET ${offset}`;
|
|
9
|
+
}
|
|
10
|
+
return sql``;
|
|
11
|
+
}
|
|
12
|
+
function buildOrderByClause(sql, table, plan) {
|
|
13
|
+
if (plan.orderBy.length === 0) return sql``;
|
|
14
|
+
const fragments = [];
|
|
15
|
+
for (const { key, direction } of plan.orderBy) {
|
|
16
|
+
const col = table.columns[key];
|
|
17
|
+
if (!col) continue;
|
|
18
|
+
if (direction === "desc") {
|
|
19
|
+
fragments.push(sql`${sql(col.meta.sqlName)} DESC`);
|
|
20
|
+
continue;
|
|
21
|
+
}
|
|
22
|
+
fragments.push(sql`${sql(col.meta.sqlName)} ASC`);
|
|
23
|
+
}
|
|
24
|
+
const firstOrderBy = fragments[0];
|
|
25
|
+
if (!firstOrderBy) return sql``;
|
|
26
|
+
let joined = firstOrderBy;
|
|
27
|
+
for (let index = 1; index < fragments.length; index++) {
|
|
28
|
+
const fragment = fragments[index];
|
|
29
|
+
if (!fragment) continue;
|
|
30
|
+
joined = sql`${joined}, ${fragment}`;
|
|
31
|
+
}
|
|
32
|
+
return sql`ORDER BY ${joined}`;
|
|
33
|
+
}
|
|
34
|
+
//#endregion
|
|
35
|
+
exports.buildLimitClause = buildLimitClause;
|
|
36
|
+
exports.buildOrderByClause = buildOrderByClause;
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
//#region src/lib/orm/sql/serialize/clauses.ts
|
|
2
|
+
function buildLimitClause(sql, limit, offset, dialect) {
|
|
3
|
+
if (limit != null && offset != null) return sql`LIMIT ${limit} OFFSET ${offset}`;
|
|
4
|
+
if (limit != null) return sql`LIMIT ${limit}`;
|
|
5
|
+
if (offset != null) {
|
|
6
|
+
if (dialect === "mysql") return sql`LIMIT 18446744073709551615 OFFSET ${offset}`;
|
|
7
|
+
if (dialect === "sqlite") return sql`LIMIT -1 OFFSET ${offset}`;
|
|
8
|
+
return sql`OFFSET ${offset}`;
|
|
9
|
+
}
|
|
10
|
+
return sql``;
|
|
11
|
+
}
|
|
12
|
+
function buildOrderByClause(sql, table, plan) {
|
|
13
|
+
if (plan.orderBy.length === 0) return sql``;
|
|
14
|
+
const fragments = [];
|
|
15
|
+
for (const { key, direction } of plan.orderBy) {
|
|
16
|
+
const col = table.columns[key];
|
|
17
|
+
if (!col) continue;
|
|
18
|
+
if (direction === "desc") {
|
|
19
|
+
fragments.push(sql`${sql(col.meta.sqlName)} DESC`);
|
|
20
|
+
continue;
|
|
21
|
+
}
|
|
22
|
+
fragments.push(sql`${sql(col.meta.sqlName)} ASC`);
|
|
23
|
+
}
|
|
24
|
+
const firstOrderBy = fragments[0];
|
|
25
|
+
if (!firstOrderBy) return sql``;
|
|
26
|
+
let joined = firstOrderBy;
|
|
27
|
+
for (let index = 1; index < fragments.length; index++) {
|
|
28
|
+
const fragment = fragments[index];
|
|
29
|
+
if (!fragment) continue;
|
|
30
|
+
joined = sql`${joined}, ${fragment}`;
|
|
31
|
+
}
|
|
32
|
+
return sql`ORDER BY ${joined}`;
|
|
33
|
+
}
|
|
34
|
+
//#endregion
|
|
35
|
+
export { buildLimitClause, buildOrderByClause };
|
|
36
|
+
|
|
37
|
+
//# sourceMappingURL=clauses.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"clauses.mjs","names":[],"sources":["../../../../src/lib/orm/sql/serialize/clauses.ts"],"sourcesContent":["import type { SQL, TransactionSQL } from \"bun\";\nimport type { Table } from \"../../table.js\";\nimport type { ColDefs, Dialect, SelectPlan } from \"../../types.js\";\n\nexport function buildLimitClause(\n sql: SQL | TransactionSQL,\n limit?: number,\n offset?: number,\n dialect?: Dialect,\n) {\n if (limit != null && offset != null) {\n return sql`LIMIT ${limit} OFFSET ${offset}`;\n }\n\n if (limit != null) {\n return sql`LIMIT ${limit}`;\n }\n\n if (offset != null) {\n if (dialect === \"mysql\") {\n return sql`LIMIT 18446744073709551615 OFFSET ${offset}`;\n }\n\n if (dialect === \"sqlite\") {\n return sql`LIMIT -1 OFFSET ${offset}`;\n }\n\n return sql`OFFSET ${offset}`;\n }\n\n return sql``;\n}\n\nexport function buildOrderByClause<T extends ColDefs>(\n sql: SQL | TransactionSQL,\n table: Table<T>,\n plan: SelectPlan<T>,\n) {\n if (plan.orderBy.length === 0) {\n return sql``;\n }\n\n const fragments: SQL.Query<unknown>[] = [];\n\n for (const { key, direction } of plan.orderBy) {\n const col = table.columns[key];\n\n if (!col) {\n continue;\n }\n\n if (direction === \"desc\") {\n fragments.push(sql`${sql(col.meta.sqlName)} DESC`);\n continue;\n }\n\n fragments.push(sql`${sql(col.meta.sqlName)} ASC`);\n }\n\n const firstOrderBy = fragments[0];\n\n if (!firstOrderBy) {\n return sql``;\n }\n\n let joined = firstOrderBy;\n\n for (let index = 1; index < fragments.length; index++) {\n const fragment = fragments[index];\n\n if (!fragment) {\n continue;\n }\n\n joined = sql`${joined}, ${fragment}`;\n }\n\n return sql`ORDER BY ${joined}`;\n}\n"],"mappings":";AAIA,SAAgB,iBACd,KACA,OACA,QACA,SACA;AACA,KAAI,SAAS,QAAQ,UAAU,KAC7B,QAAO,GAAG,SAAS,MAAM,UAAU;AAGrC,KAAI,SAAS,KACX,QAAO,GAAG,SAAS;AAGrB,KAAI,UAAU,MAAM;AAClB,MAAI,YAAY,QACd,QAAO,GAAG,qCAAqC;AAGjD,MAAI,YAAY,SACd,QAAO,GAAG,mBAAmB;AAG/B,SAAO,GAAG,UAAU;;AAGtB,QAAO,GAAG;;AAGZ,SAAgB,mBACd,KACA,OACA,MACA;AACA,KAAI,KAAK,QAAQ,WAAW,EAC1B,QAAO,GAAG;CAGZ,MAAM,YAAkC,EAAE;AAE1C,MAAK,MAAM,EAAE,KAAK,eAAe,KAAK,SAAS;EAC7C,MAAM,MAAM,MAAM,QAAQ;AAE1B,MAAI,CAAC,IACH;AAGF,MAAI,cAAc,QAAQ;AACxB,aAAU,KAAK,GAAG,GAAG,IAAI,IAAI,KAAK,QAAQ,CAAC,OAAO;AAClD;;AAGF,YAAU,KAAK,GAAG,GAAG,IAAI,IAAI,KAAK,QAAQ,CAAC,MAAM;;CAGnD,MAAM,eAAe,UAAU;AAE/B,KAAI,CAAC,aACH,QAAO,GAAG;CAGZ,IAAI,SAAS;AAEb,MAAK,IAAI,QAAQ,GAAG,QAAQ,UAAU,QAAQ,SAAS;EACrD,MAAM,WAAW,UAAU;AAE3B,MAAI,CAAC,SACH;AAGF,WAAS,GAAG,GAAG,OAAO,IAAI;;AAG5B,QAAO,GAAG,YAAY"}
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
const require_table_lookup = require("../../internal/table-lookup.cjs");
|
|
2
|
+
const require_table_relations = require("../../internal/table-relations.cjs");
|
|
3
|
+
//#region src/lib/orm/sql/serialize/joins.ts
|
|
4
|
+
function buildJoinClauses(sql, table, relations, plan) {
|
|
5
|
+
if (plan.joins.length === 0) return sql``;
|
|
6
|
+
let joinClause = sql``;
|
|
7
|
+
const basePk = require_table_lookup.getPrimaryKeyColumn(table);
|
|
8
|
+
if (!basePk) return joinClause;
|
|
9
|
+
for (const [joinIndex, join] of plan.joins.entries()) {
|
|
10
|
+
const rel = relations[join.relationKey];
|
|
11
|
+
if (!rel) continue;
|
|
12
|
+
const target = rel.table();
|
|
13
|
+
const targetPk = require_table_lookup.getPrimaryKeyColumn(target);
|
|
14
|
+
if (!targetPk) continue;
|
|
15
|
+
const targetAlias = `${join.relationKey}_${joinIndex}`;
|
|
16
|
+
if (rel.kind === "one") {
|
|
17
|
+
joinClause = sql`${joinClause} LEFT JOIN ${sql(target.tableName)} AS ${sql(targetAlias)} ON ${sql(table.tableName)}.${sql(rel.foreignKey)} = ${sql(targetAlias)}.${sql(targetPk.col.meta.sqlName)}`;
|
|
18
|
+
continue;
|
|
19
|
+
}
|
|
20
|
+
let foreignSqlName = rel.foreignKey;
|
|
21
|
+
if (!foreignSqlName) {
|
|
22
|
+
const inferredForeignSqlName = require_table_relations.findManyForeignKeyByReference(target, basePk.col);
|
|
23
|
+
if (inferredForeignSqlName) foreignSqlName = inferredForeignSqlName;
|
|
24
|
+
}
|
|
25
|
+
if (!foreignSqlName) continue;
|
|
26
|
+
joinClause = sql`${joinClause} LEFT JOIN ${sql(target.tableName)} AS ${sql(targetAlias)} ON ${sql(targetAlias)}.${sql(foreignSqlName)} = ${sql(table.tableName)}.${sql(basePk.col.meta.sqlName)}`;
|
|
27
|
+
}
|
|
28
|
+
return joinClause;
|
|
29
|
+
}
|
|
30
|
+
//#endregion
|
|
31
|
+
exports.buildJoinClauses = buildJoinClauses;
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
import { getPrimaryKeyColumn } from "../../internal/table-lookup.mjs";
|
|
2
|
+
import { findManyForeignKeyByReference } from "../../internal/table-relations.mjs";
|
|
3
|
+
//#region src/lib/orm/sql/serialize/joins.ts
|
|
4
|
+
function buildJoinClauses(sql, table, relations, plan) {
|
|
5
|
+
if (plan.joins.length === 0) return sql``;
|
|
6
|
+
let joinClause = sql``;
|
|
7
|
+
const basePk = getPrimaryKeyColumn(table);
|
|
8
|
+
if (!basePk) return joinClause;
|
|
9
|
+
for (const [joinIndex, join] of plan.joins.entries()) {
|
|
10
|
+
const rel = relations[join.relationKey];
|
|
11
|
+
if (!rel) continue;
|
|
12
|
+
const target = rel.table();
|
|
13
|
+
const targetPk = getPrimaryKeyColumn(target);
|
|
14
|
+
if (!targetPk) continue;
|
|
15
|
+
const targetAlias = `${join.relationKey}_${joinIndex}`;
|
|
16
|
+
if (rel.kind === "one") {
|
|
17
|
+
joinClause = sql`${joinClause} LEFT JOIN ${sql(target.tableName)} AS ${sql(targetAlias)} ON ${sql(table.tableName)}.${sql(rel.foreignKey)} = ${sql(targetAlias)}.${sql(targetPk.col.meta.sqlName)}`;
|
|
18
|
+
continue;
|
|
19
|
+
}
|
|
20
|
+
let foreignSqlName = rel.foreignKey;
|
|
21
|
+
if (!foreignSqlName) {
|
|
22
|
+
const inferredForeignSqlName = findManyForeignKeyByReference(target, basePk.col);
|
|
23
|
+
if (inferredForeignSqlName) foreignSqlName = inferredForeignSqlName;
|
|
24
|
+
}
|
|
25
|
+
if (!foreignSqlName) continue;
|
|
26
|
+
joinClause = sql`${joinClause} LEFT JOIN ${sql(target.tableName)} AS ${sql(targetAlias)} ON ${sql(targetAlias)}.${sql(foreignSqlName)} = ${sql(table.tableName)}.${sql(basePk.col.meta.sqlName)}`;
|
|
27
|
+
}
|
|
28
|
+
return joinClause;
|
|
29
|
+
}
|
|
30
|
+
//#endregion
|
|
31
|
+
export { buildJoinClauses };
|
|
32
|
+
|
|
33
|
+
//# sourceMappingURL=joins.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"joins.mjs","names":[],"sources":["../../../../src/lib/orm/sql/serialize/joins.ts"],"sourcesContent":["import type { SQL, TransactionSQL } from \"bun\";\nimport {\n findManyForeignKeyByReference,\n getPrimaryKeyColumn,\n} from \"../../internal/table.js\";\nimport type { Table } from \"../../table.js\";\nimport type { ColDefs, RelationDefs, SelectPlan } from \"../../types.js\";\n\nexport function buildJoinClauses<T extends ColDefs>(\n sql: SQL | TransactionSQL,\n table: Table<T>,\n relations: RelationDefs,\n plan: SelectPlan<T>,\n) {\n if (plan.joins.length === 0) {\n return sql``;\n }\n\n let joinClause = sql``;\n const basePk = getPrimaryKeyColumn(table);\n\n if (!basePk) {\n return joinClause;\n }\n\n for (const [joinIndex, join] of plan.joins.entries()) {\n const rel = relations[join.relationKey];\n\n if (!rel) {\n continue;\n }\n\n const target = rel.table();\n const targetPk = getPrimaryKeyColumn(target);\n\n if (!targetPk) {\n continue;\n }\n\n const targetAlias = `${join.relationKey}_${joinIndex}`;\n\n if (rel.kind === \"one\") {\n joinClause = sql`${joinClause} LEFT JOIN ${sql(target.tableName)} AS ${sql(targetAlias)} ON ${sql(table.tableName)}.${sql(rel.foreignKey)} = ${sql(targetAlias)}.${sql(targetPk.col.meta.sqlName)}`;\n continue;\n }\n\n let foreignSqlName = rel.foreignKey;\n\n if (!foreignSqlName) {\n const inferredForeignSqlName = findManyForeignKeyByReference(\n target,\n basePk.col,\n );\n\n if (inferredForeignSqlName) {\n foreignSqlName = inferredForeignSqlName;\n }\n }\n\n if (!foreignSqlName) {\n continue;\n }\n\n joinClause = sql`${joinClause} LEFT JOIN ${sql(target.tableName)} AS ${sql(targetAlias)} ON ${sql(targetAlias)}.${sql(foreignSqlName)} = ${sql(table.tableName)}.${sql(basePk.col.meta.sqlName)}`;\n }\n\n return joinClause;\n}\n"],"mappings":";;;AAQA,SAAgB,iBACd,KACA,OACA,WACA,MACA;AACA,KAAI,KAAK,MAAM,WAAW,EACxB,QAAO,GAAG;CAGZ,IAAI,aAAa,GAAG;CACpB,MAAM,SAAS,oBAAoB,MAAM;AAEzC,KAAI,CAAC,OACH,QAAO;AAGT,MAAK,MAAM,CAAC,WAAW,SAAS,KAAK,MAAM,SAAS,EAAE;EACpD,MAAM,MAAM,UAAU,KAAK;AAE3B,MAAI,CAAC,IACH;EAGF,MAAM,SAAS,IAAI,OAAO;EAC1B,MAAM,WAAW,oBAAoB,OAAO;AAE5C,MAAI,CAAC,SACH;EAGF,MAAM,cAAc,GAAG,KAAK,YAAY,GAAG;AAE3C,MAAI,IAAI,SAAS,OAAO;AACtB,gBAAa,GAAG,GAAG,WAAW,aAAa,IAAI,OAAO,UAAU,CAAC,MAAM,IAAI,YAAY,CAAC,MAAM,IAAI,MAAM,UAAU,CAAC,GAAG,IAAI,IAAI,WAAW,CAAC,KAAK,IAAI,YAAY,CAAC,GAAG,IAAI,SAAS,IAAI,KAAK,QAAQ;AACjM;;EAGF,IAAI,iBAAiB,IAAI;AAEzB,MAAI,CAAC,gBAAgB;GACnB,MAAM,yBAAyB,8BAC7B,QACA,OAAO,IACR;AAED,OAAI,uBACF,kBAAiB;;AAIrB,MAAI,CAAC,eACH;AAGF,eAAa,GAAG,GAAG,WAAW,aAAa,IAAI,OAAO,UAAU,CAAC,MAAM,IAAI,YAAY,CAAC,MAAM,IAAI,YAAY,CAAC,GAAG,IAAI,eAAe,CAAC,KAAK,IAAI,MAAM,UAAU,CAAC,GAAG,IAAI,OAAO,IAAI,KAAK,QAAQ;;AAGjM,QAAO"}
|