2024-02-29 22:23:05 +08:00
|
|
|
import { FunctionMap, KV, Query, QueryExpression } from "../types.ts";
|
2024-02-03 02:19:07 +08:00
|
|
|
import { evalQueryExpression } from "$sb/lib/query_expression.ts";
|
2022-10-14 21:11:33 +08:00
|
|
|
|
2023-10-03 20:16:33 +08:00
|
|
|
/**
|
|
|
|
* Looks for an attribute assignment in the expression, and returns the expression assigned to the attribute or throws an error when not found
|
|
|
|
* Side effect: effectively removes the attribute assignment from the expression (by replacing it with true = true)
|
|
|
|
*/
|
|
|
|
export function liftAttributeFilter(
|
|
|
|
expression: QueryExpression | undefined,
|
|
|
|
attributeName: string,
|
|
|
|
): QueryExpression {
|
|
|
|
if (!expression) {
|
|
|
|
throw new Error(`Cannot find attribute assignment for ${attributeName}`);
|
|
|
|
}
|
|
|
|
switch (expression[0]) {
|
|
|
|
case "=": {
|
|
|
|
if (expression[1][0] === "attr" && expression[1][1] === attributeName) {
|
|
|
|
const val = expression[2];
|
|
|
|
// Remove the filter by changing it to true = true
|
|
|
|
expression[1] = ["boolean", true];
|
|
|
|
expression[2] = ["boolean", true];
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case "and":
|
|
|
|
case "or": {
|
|
|
|
const newOp1 = liftAttributeFilter(expression[1], attributeName);
|
|
|
|
if (newOp1) {
|
|
|
|
return newOp1;
|
|
|
|
}
|
|
|
|
const newOp2 = liftAttributeFilter(expression[2], attributeName);
|
|
|
|
if (newOp2) {
|
|
|
|
return newOp2;
|
|
|
|
}
|
|
|
|
throw new Error(`Cannot find attribute assignment for ${attributeName}`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
throw new Error(`Cannot find attribute assignment for ${attributeName}`);
|
|
|
|
}
|
|
|
|
|
2024-02-03 02:19:07 +08:00
|
|
|
export async function applyQuery<T>(
|
|
|
|
query: Query,
|
|
|
|
allItems: T[],
|
|
|
|
variables: Record<string, any>,
|
|
|
|
functionMap: FunctionMap,
|
|
|
|
): Promise<T[]> {
|
2023-10-03 20:16:33 +08:00
|
|
|
// Filter
|
|
|
|
if (query.filter) {
|
2024-02-03 02:19:07 +08:00
|
|
|
const filteredItems: T[] = [];
|
|
|
|
for (const item of allItems) {
|
|
|
|
if (
|
|
|
|
await evalQueryExpression(
|
|
|
|
query.filter,
|
|
|
|
item,
|
|
|
|
variables,
|
|
|
|
functionMap,
|
|
|
|
)
|
|
|
|
) {
|
|
|
|
filteredItems.push(item);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
allItems = filteredItems;
|
2023-10-03 20:16:33 +08:00
|
|
|
}
|
|
|
|
// Add dummy keys, then remove them
|
2024-02-03 02:19:07 +08:00
|
|
|
return (await applyQueryNoFilterKV(
|
2023-10-03 20:16:33 +08:00
|
|
|
query,
|
|
|
|
allItems.map((v) => ({ key: [], value: v })),
|
2024-02-03 02:19:07 +08:00
|
|
|
variables,
|
|
|
|
functionMap,
|
|
|
|
)).map((v) => v.value);
|
2023-10-03 20:16:33 +08:00
|
|
|
}
|
|
|
|
|
2024-02-03 02:19:07 +08:00
|
|
|
export async function applyQueryNoFilterKV(
|
2023-10-03 20:16:33 +08:00
|
|
|
query: Query,
|
|
|
|
allItems: KV[],
|
2024-02-03 02:19:07 +08:00
|
|
|
variables: Record<string, any>,
|
|
|
|
functionMap: FunctionMap,
|
|
|
|
): Promise<KV[]> {
|
2023-10-03 20:16:33 +08:00
|
|
|
// Order by
|
|
|
|
if (query.orderBy) {
|
2023-10-03 22:54:03 +08:00
|
|
|
allItems = allItems.sort((a, b) => {
|
2023-10-03 20:16:33 +08:00
|
|
|
const aVal = a.value;
|
|
|
|
const bVal = b.value;
|
|
|
|
for (const { expr, desc } of query.orderBy!) {
|
2024-02-03 02:19:07 +08:00
|
|
|
const evalA = evalQueryExpression(
|
|
|
|
expr,
|
|
|
|
aVal,
|
|
|
|
variables,
|
|
|
|
functionMap,
|
|
|
|
);
|
|
|
|
if (evalA instanceof Promise) {
|
|
|
|
throw new Error("Cannot order by a promise");
|
|
|
|
}
|
|
|
|
const evalB = evalQueryExpression(
|
|
|
|
expr,
|
|
|
|
bVal,
|
|
|
|
variables,
|
|
|
|
functionMap,
|
|
|
|
);
|
|
|
|
if (evalB instanceof Promise) {
|
|
|
|
throw new Error("Cannot order by a promise");
|
|
|
|
}
|
2023-10-03 20:16:33 +08:00
|
|
|
if (
|
|
|
|
evalA < evalB || evalA === undefined
|
|
|
|
) {
|
|
|
|
return desc ? 1 : -1;
|
2023-04-19 17:01:27 +08:00
|
|
|
}
|
2023-10-03 20:16:33 +08:00
|
|
|
if (
|
|
|
|
evalA > evalB || evalB === undefined
|
|
|
|
) {
|
|
|
|
return desc ? -1 : 1;
|
2023-04-19 17:01:27 +08:00
|
|
|
}
|
2022-10-14 21:11:33 +08:00
|
|
|
}
|
2023-10-03 20:16:33 +08:00
|
|
|
// Consider them equal. This helps with comparing arrays (like tags)
|
2023-04-19 17:01:27 +08:00
|
|
|
return 0;
|
2022-10-14 21:11:33 +08:00
|
|
|
});
|
|
|
|
}
|
2023-04-19 17:01:27 +08:00
|
|
|
|
2023-10-03 20:16:33 +08:00
|
|
|
if (query.select) {
|
|
|
|
for (let i = 0; i < allItems.length; i++) {
|
|
|
|
const rec = allItems[i].value;
|
2022-10-16 01:02:56 +08:00
|
|
|
const newRec: any = {};
|
2023-10-03 20:16:33 +08:00
|
|
|
for (const { name, expr } of query.select) {
|
|
|
|
newRec[name] = expr
|
2024-02-03 02:19:07 +08:00
|
|
|
? await evalQueryExpression(expr, rec, variables, functionMap)
|
2023-10-03 20:16:33 +08:00
|
|
|
: rec[name];
|
2022-10-14 21:11:33 +08:00
|
|
|
}
|
2023-10-03 20:16:33 +08:00
|
|
|
allItems[i].value = newRec;
|
|
|
|
}
|
|
|
|
}
|
2023-10-03 21:24:07 +08:00
|
|
|
if (query.distinct) {
|
|
|
|
// Remove duplicates
|
|
|
|
const valueSet = new Set<string>();
|
|
|
|
const uniqueItems: KV[] = [];
|
|
|
|
for (const item of allItems) {
|
|
|
|
const value = JSON.stringify(item.value);
|
|
|
|
if (!valueSet.has(value)) {
|
|
|
|
valueSet.add(value);
|
|
|
|
uniqueItems.push(item);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
allItems = uniqueItems;
|
|
|
|
}
|
|
|
|
|
2023-10-03 20:16:33 +08:00
|
|
|
if (query.limit) {
|
2024-02-03 02:19:07 +08:00
|
|
|
const limit = await evalQueryExpression(
|
|
|
|
query.limit,
|
|
|
|
{},
|
|
|
|
variables,
|
|
|
|
functionMap,
|
|
|
|
);
|
2023-10-03 20:16:33 +08:00
|
|
|
if (allItems.length > limit) {
|
|
|
|
allItems = allItems.slice(0, limit);
|
|
|
|
}
|
2022-10-14 21:11:33 +08:00
|
|
|
}
|
2023-10-03 20:16:33 +08:00
|
|
|
return allItems;
|
2022-10-14 21:11:33 +08:00
|
|
|
}
|