silverbullet/plug-api/lib/query.ts

169 lines
4.7 KiB
TypeScript
Raw Normal View History

2022-10-14 21:11:33 +08:00
import {
addParentPointers,
collectNodesMatching,
ParseTree,
renderToText,
} from "$sb/lib/tree.ts";
2022-10-14 21:11:33 +08:00
export const queryRegex =
/(<!--\s*#query\s+(.+?)-->)(.+?)(<!--\s*\/query\s*-->)/gs;
export const directiveStartRegex = /<!--\s*#([\w\-]+)\s+(.+?)-->/s;
export const directiveEndRegex = /<!--\s*\/([\w\-]+)\s*-->/s;
export type QueryFilter = {
op: string;
prop: string;
value: any;
};
export type ParsedQuery = {
table: string;
orderBy?: string;
orderDesc?: boolean;
limit?: number;
filter: QueryFilter[];
select?: string[];
render?: string;
};
export function applyQuery<T>(parsedQuery: ParsedQuery, records: T[]): T[] {
let resultRecords: any[] = [];
if (parsedQuery.filter.length === 0) {
resultRecords = records.slice();
} else {
recordLoop:
for (const record of records) {
const recordAny: any = record;
2022-10-16 01:02:56 +08:00
for (const { op, prop, value } of parsedQuery.filter) {
2022-10-14 21:11:33 +08:00
switch (op) {
case "=": {
const recordPropVal = recordAny[prop];
if (Array.isArray(recordPropVal) && !Array.isArray(value)) {
// Record property is an array, and value is a scalar: find the value in the array
if (!recordPropVal.includes(value)) {
continue recordLoop;
}
} else if (Array.isArray(recordPropVal) && Array.isArray(value)) {
// Record property is an array, and value is an array: find the value in the array
if (!recordPropVal.some((v) => value.includes(v))) {
continue recordLoop;
}
} else if (!(recordPropVal == value)) {
// Both are scalars: exact value
continue recordLoop;
}
break;
}
case "!=":
if (!(recordAny[prop] != value)) {
continue recordLoop;
}
break;
case "<":
if (!(recordAny[prop] < value)) {
continue recordLoop;
}
break;
case "<=":
if (!(recordAny[prop] <= value)) {
continue recordLoop;
}
break;
case ">":
if (!(recordAny[prop] > value)) {
continue recordLoop;
}
break;
case ">=":
if (!(recordAny[prop] >= value)) {
continue recordLoop;
}
break;
case "=~":
// TODO: Cache regexps somehow
if (!new RegExp(value).exec(recordAny[prop])) {
continue recordLoop;
}
break;
case "!=~":
if (new RegExp(value).exec(recordAny[prop])) {
continue recordLoop;
}
break;
case "in":
if (!value.includes(recordAny[prop])) {
continue recordLoop;
}
break;
}
}
resultRecords.push(recordAny);
}
}
// Now the sorting
if (parsedQuery.orderBy) {
resultRecords = resultRecords.sort((a: any, b: any) => {
const orderBy = parsedQuery.orderBy!;
const orderDesc = parsedQuery.orderDesc!;
if (a[orderBy] === b[orderBy]) {
return 0;
}
if (a[orderBy] < b[orderBy]) {
return orderDesc ? 1 : -1;
} else {
return orderDesc ? -1 : 1;
}
});
}
if (parsedQuery.limit) {
resultRecords = resultRecords.slice(0, parsedQuery.limit);
}
if (parsedQuery.select) {
resultRecords = resultRecords.map((rec) => {
2022-10-16 01:02:56 +08:00
const newRec: any = {};
for (const k of parsedQuery.select!) {
2022-10-14 21:11:33 +08:00
newRec[k] = rec[k];
}
return newRec;
});
}
return resultRecords;
}
export function removeQueries(pt: ParseTree) {
addParentPointers(pt);
collectNodesMatching(pt, (t) => {
if (t.type !== "CommentBlock") {
return false;
}
2022-10-16 01:02:56 +08:00
const text = t.children![0].text!;
const match = directiveStartRegex.exec(text);
2022-10-14 21:11:33 +08:00
if (!match) {
return false;
}
2022-10-16 01:02:56 +08:00
const directiveType = match[1];
const parentChildren = t.parent!.children!;
const index = parentChildren.indexOf(t);
const nodesToReplace: ParseTree[] = [];
2022-10-14 21:11:33 +08:00
for (let i = index + 1; i < parentChildren.length; i++) {
2022-10-16 01:02:56 +08:00
const n = parentChildren[i];
2022-10-14 21:11:33 +08:00
if (n.type === "CommentBlock") {
2022-10-16 01:02:56 +08:00
const text = n.children![0].text!;
const match = directiveEndRegex.exec(text);
2022-10-14 21:11:33 +08:00
if (match && match[1] === directiveType) {
break;
}
}
nodesToReplace.push(n);
}
2022-10-16 01:02:56 +08:00
const renderedText = nodesToReplace.map(renderToText).join("");
2022-10-14 21:11:33 +08:00
parentChildren.splice(index + 1, nodesToReplace.length, {
text: new Array(renderedText.length + 1).join(" "),
});
return true;
});
}