import { addParentPointers, collectNodesMatching, ParseTree, renderToText, } from "$sb/lib/tree.ts"; export const queryRegex = /()(.+?)()/gs; export const directiveStartRegex = //s; export const directiveEndRegex = //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(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; for (const { op, prop, value } of parsedQuery.filter) { 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) => { const newRec: any = {}; for (const k of parsedQuery.select!) { newRec[k] = rec[k]; } return newRec; }); } return resultRecords; } export function removeQueries(pt: ParseTree) { addParentPointers(pt); collectNodesMatching(pt, (t) => { if (t.type !== "CommentBlock") { return false; } const text = t.children![0].text!; const match = directiveStartRegex.exec(text); if (!match) { return false; } const directiveType = match[1]; const parentChildren = t.parent!.children!; const index = parentChildren.indexOf(t); const nodesToReplace: ParseTree[] = []; for (let i = index + 1; i < parentChildren.length; i++) { const n = parentChildren[i]; if (n.type === "CommentBlock") { const text = n.children![0].text!; const match = directiveEndRegex.exec(text); if (match && match[1] === directiveType) { break; } } nodesToReplace.push(n); } const renderedText = nodesToReplace.map(renderToText).join(""); parentChildren.splice(index + 1, nodesToReplace.length, { text: new Array(renderedText.length + 1).join(" "), }); return true; }); }