From 3c5048ac25a0b7154ac5cc80d4b003b4d1164005 Mon Sep 17 00:00:00 2001 From: Zef Hemel Date: Fri, 13 May 2022 14:36:26 +0200 Subject: [PATCH] Dependency builds for plugos --- packages/common/preload_modules.ts | 7 -- packages/plugos/bin/plugos-bundle.ts | 14 ++- packages/plugos/compile.ts | 88 +++++++++++++++++- packages/plugos/environments/custom_logger.ts | 3 + packages/plugos/environments/node_sandbox.ts | 6 +- packages/plugos/environments/node_worker.ts | 29 +++++- .../plugos/environments/sandbox_worker.ts | 24 +++-- packages/plugos/environments/worker.ts | 14 ++- packages/plugos/hooks/event.ts | 4 +- packages/plugos/package.json | 6 +- packages/plugos/plug.ts | 3 + packages/plugos/sandbox.ts | 22 +++++ packages/plugos/syscalls/esbuild.ts | 93 +++++++++++++++---- packages/plugos/types.ts | 3 + packages/plugs/build/test/engine.test.js | 1 + packages/plugs/build/test/engine.test.js.map | 2 +- packages/plugs/core/core.plug.yaml | 1 + packages/plugs/core/navigate.ts | 2 +- packages/plugs/global.plug.yaml | 5 + packages/plugs/package.json | 4 +- packages/plugs/plugmanager/plugmanager.ts | 2 + packages/plugs/query/engine.ts | 1 + packages/plugs/query/query.plug.yaml | 3 + packages/server/express_server.ts | 24 +++-- packages/server/server.ts | 2 - packages/web/components/panel_page.ts | 5 - packages/web/editor.tsx | 13 +++ packages/web/reducer.ts | 3 + packages/web/styles/editor.scss | 1 + 29 files changed, 309 insertions(+), 76 deletions(-) delete mode 100644 packages/common/preload_modules.ts create mode 100644 packages/plugs/global.plug.yaml diff --git a/packages/common/preload_modules.ts b/packages/common/preload_modules.ts deleted file mode 100644 index cb1c4cb3..00000000 --- a/packages/common/preload_modules.ts +++ /dev/null @@ -1,7 +0,0 @@ -// These are the node modules that will be pre-bundled with SB -// as a result they will not be included into plugos bundles and assumed to be loadable -// via require() in the sandbox -// Candidate modules for this are larger modules - -// When adding a module to this list, also manually add it to sandbox_worker.ts -export const preloadModules = ["@lezer/lr", "yaml", "handlebars"]; diff --git a/packages/plugos/bin/plugos-bundle.ts b/packages/plugos/bin/plugos-bundle.ts index ccacbeb9..24d4c6b8 100755 --- a/packages/plugos/bin/plugos-bundle.ts +++ b/packages/plugos/bin/plugos-bundle.ts @@ -1,6 +1,6 @@ #!/usr/bin/env node -import { readFile, unlink, watch, writeFile } from "fs/promises"; +import { readFile, watch, writeFile } from "fs/promises"; import path from "path"; import yargs from "yargs"; @@ -8,7 +8,7 @@ import { hideBin } from "yargs/helpers"; import { Manifest } from "../types"; import YAML from "yaml"; import { mkdirSync } from "fs"; -import { compile } from "../compile"; +import { compile, sandboxCompileModule } from "../compile"; async function bundle( manifestPath: string, @@ -24,7 +24,13 @@ async function bundle( throw new Error(`Missing 'name' in ${manifestPath}`); } - for (let [name, def] of Object.entries(manifest.functions)) { + let allModulesToExclude = excludeModules.slice(); + for (let [name, moduleSpec] of Object.entries(manifest.dependencies || {})) { + manifest.dependencies![name] = await sandboxCompileModule(moduleSpec); + allModulesToExclude.push(name); + } + + for (let [name, def] of Object.entries(manifest.functions || {})) { let jsFunctionName = "default", filePath = path.join(rootPath, def.path!); if (filePath.indexOf(":") !== -1) { @@ -35,7 +41,7 @@ async function bundle( filePath, jsFunctionName, sourceMaps, - excludeModules + allModulesToExclude ); delete def.path; } diff --git a/packages/plugos/compile.ts b/packages/plugos/compile.ts index 11c5679b..447d67eb 100644 --- a/packages/plugos/compile.ts +++ b/packages/plugos/compile.ts @@ -1,6 +1,11 @@ import esbuild from "esbuild"; -import { readFile, unlink, writeFile } from "fs/promises"; +import { mkdir, readFile, rm, symlink, unlink, writeFile } from "fs/promises"; import path from "path"; +import { tmpdir } from "os"; +import { nodeModulesDir } from "./environments/node_sandbox"; +import { promisify } from "util"; +import { execFile } from "child_process"; +const execFilePromise = promisify(execFile); export async function compile( filePath: string, @@ -22,8 +27,6 @@ export async function compile( )}";export default ${functionName};` ); } - // console.log("In:", inFile); - // console.log("Outfile:", outFile); // TODO: Figure out how to make source maps work correctly with eval() code let result = await esbuild.build({ @@ -50,6 +53,81 @@ export async function compile( if (inFile !== filePath) { await unlink(inFile); } - return `(() => { ${jsCode} - return mod;})()`; + return `(() => { ${jsCode} return mod;})()`; +} + +export async function compileModule( + cwd: string, + moduleName: string +): Promise { + let inFile = path.resolve(cwd, "_in.ts"); + await writeFile(inFile, `export * from "${moduleName}";`); + let code = await compile(inFile); + await unlink(inFile); + return code; +} + +// TODO: Reconsider this later +const exposedModules = [ + "@silverbulletmd/plugos-silverbullet-syscall", + "@plugos/plugos-syscall", +]; + +export async function sandboxCompile( + filename: string, + code: string, + functionName?: string, + installModules: string[] = [], + globalModules: string[] = [] +): Promise { + let tmpDir = `${tmpdir()}/plugos-${Math.random()}`; + await mkdir(tmpDir, { recursive: true }); + + const srcNodeModules = `${nodeModulesDir}/node_modules`; + const targetNodeModules = `${tmpDir}/node_modules`; + + await mkdir(`${targetNodeModules}/@silverbulletmd`, { recursive: true }); + await mkdir(`${targetNodeModules}/@plugos`, { recursive: true }); + for (const exposedModule of exposedModules) { + await symlink( + `${srcNodeModules}/${exposedModule}`, + `${targetNodeModules}/${exposedModule}`, + "dir" + ); + } + for (let moduleName of installModules) { + await execFilePromise("npm", ["install", moduleName], { + cwd: tmpDir, + }); + } + + await writeFile(`${tmpDir}/${filename}`, code); + + let jsCode = await compile( + `${tmpDir}/${filename}`, + functionName, + false, + globalModules + ); + await rm(tmpDir, { recursive: true }); + return jsCode; +} + +export async function sandboxCompileModule( + moduleName: string, + globalModules: string[] = [] +): Promise { + let [modulePart, path] = moduleName.split(":"); + let modulePieces = modulePart.split("@"); + let cleanModulesName = modulePieces + .slice(0, modulePieces.length - 1) + .join("@"); + return sandboxCompile( + "module.ts", + // `export * from "${cleanModulesName}${path ? path : ""}";`, + `module.exports = require("${cleanModulesName}${path ? path : ""}");`, + undefined, + [modulePart], + globalModules + ); } diff --git a/packages/plugos/environments/custom_logger.ts b/packages/plugos/environments/custom_logger.ts index 7baa9697..53b51e5f 100644 --- a/packages/plugos/environments/custom_logger.ts +++ b/packages/plugos/environments/custom_logger.ts @@ -43,6 +43,9 @@ export class ConsoleLogger { case "number": pieces.push("" + val); break; + case "undefined": + pieces.push("undefined"); + break; default: try { let s = JSON.stringify(val, null, 2); diff --git a/packages/plugos/environments/node_sandbox.ts b/packages/plugos/environments/node_sandbox.ts index 741e8ea9..e71cceb0 100644 --- a/packages/plugos/environments/node_sandbox.ts +++ b/packages/plugos/environments/node_sandbox.ts @@ -41,15 +41,11 @@ while (!fs.existsSync(nodeModulesDir + "/node_modules/vm2")) { nodeModulesDir = path.dirname(nodeModulesDir); } -export function createSandbox( - plug: Plug, - preloadedModules: string[] = [] -) { +export function createSandbox(plug: Plug) { let worker = new Worker(workerCode, { eval: true, workerData: { nodeModulesPath: path.join(nodeModulesDir, "node_modules"), - preloadedModules, }, }); return new Sandbox(plug, new NodeWorkerWrapper(worker)); diff --git a/packages/plugos/environments/node_worker.ts b/packages/plugos/environments/node_worker.ts index 6993e4d1..28a2d42f 100644 --- a/packages/plugos/environments/node_worker.ts +++ b/packages/plugos/environments/node_worker.ts @@ -2,7 +2,7 @@ import { ConsoleLogger } from "./custom_logger"; const { parentPort, - workerData: { preloadedModules, nodeModulesPath }, + workerData: { nodeModulesPath }, } = require("worker_threads"); const { VM, VMScript } = require(`${nodeModulesPath}/vm2`); @@ -26,6 +26,8 @@ let consoleLogger = new ConsoleLogger((level, message) => { }); }, false); +let loadedModules = new Map(); + let vm = new VM({ sandbox: { // Exposing some "safe" APIs @@ -39,9 +41,14 @@ let vm = new VM({ // This is only going to be called for pre-bundled modules, we won't allow // arbitrary requiring of modules require: (moduleName: string): any => { - // console.log("Loading", moduleName); - if (preloadedModules.includes(moduleName)) { - return require(`${nodeModulesPath}/${moduleName}`); + // console.log("Loading module", moduleName); + // if (preloadedModules.includes(moduleName)) { + // return require(`${nodeModulesPath}/${moduleName}`); + // } else + if (loadedModules.has(moduleName)) { + let mod = loadedModules.get(moduleName); + // console.log("And it has the value", mod); + return mod; } else { throw Error(`Cannot import arbitrary modules like ${moduleName}`); } @@ -84,6 +91,20 @@ parentPort.on("message", (data: any) => { name: data.name, }); break; + case "load-dependency": + // console.log("Asked to load dep", data.name); + try { + let r = vm.run(data.code); + // console.log("Loaded dependency", r); + loadedModules.set(data.name, r); + parentPort.postMessage({ + type: "dependency-inited", + name: data.name, + }); + } catch (e: any) { + console.error("Could not load dependency", e.message); + } + break; case "invoke": let fn = loadedFunctions.get(data.name); if (!fn) { diff --git a/packages/plugos/environments/sandbox_worker.ts b/packages/plugos/environments/sandbox_worker.ts index cca927a7..d7b28c7e 100644 --- a/packages/plugos/environments/sandbox_worker.ts +++ b/packages/plugos/environments/sandbox_worker.ts @@ -39,19 +39,12 @@ self.syscall = async (name: string, ...args: any[]) => { }); }; -const preloadedModules: { [key: string]: any } = { - "@lezer/lr": require("@lezer/lr"), - yaml: require("yaml"), - handlebars: require("handlebars/dist/handlebars"), -}; -// for (const moduleName of preloadModules) { -// preloadedModules[moduleName] = require(moduleName); -// } +let loadedModules = new Map(); // @ts-ignore self.require = (moduleName: string): any => { - // console.log("Loading", moduleName, preloadedModules[moduleName]); - return preloadedModules[moduleName]; + // console.log("Loading", moduleName, loadedModules.get(moduleName)); + return loadedModules.get(moduleName); }; // @ts-ignore @@ -75,6 +68,17 @@ self.addEventListener("message", (event: { data: WorkerMessage }) => { name: data.name, }); break; + case "load-dependency": + // console.log("Received dep", data.name); + let fn3 = new Function(`return ${data.code!}`); + let v = fn3(); + loadedModules.set(data.name!, v); + // console.log("Dep val", v); + workerPostMessage({ + type: "dependency-inited", + name: data.name, + }); + break; case "invoke": let fn = loadedFunctions.get(data.name!); if (!fn) { diff --git a/packages/plugos/environments/worker.ts b/packages/plugos/environments/worker.ts index 61e61ba9..afd48807 100644 --- a/packages/plugos/environments/worker.ts +++ b/packages/plugos/environments/worker.ts @@ -1,6 +1,12 @@ import type { LogLevel } from "./custom_logger"; -export type ControllerMessageType = "inited" | "result" | "syscall" | "log"; +export type ControllerMessageType = + | "inited" + | "dependency-inited" + | "result" + | "syscall" + | "log"; + export type ControllerMessage = { type: ControllerMessageType; id?: number; @@ -21,7 +27,11 @@ export interface WorkerLike { terminate(): void; } -export type WorkerMessageType = "load" | "invoke" | "syscall-response"; +export type WorkerMessageType = + | "load" + | "load-dependency" + | "invoke" + | "syscall-response"; export type WorkerMessage = { type: WorkerMessageType; diff --git a/packages/plugos/hooks/event.ts b/packages/plugos/hooks/event.ts index 3d1e032f..2b653097 100644 --- a/packages/plugos/hooks/event.ts +++ b/packages/plugos/hooks/event.ts @@ -67,7 +67,9 @@ export class EventHook implements Hook { validateManifest(manifest: Manifest): string[] { let errors = []; - for (const [name, functionDef] of Object.entries(manifest.functions)) { + for (const [name, functionDef] of Object.entries( + manifest.functions || {} + )) { if (functionDef.events && !Array.isArray(functionDef.events)) { errors.push("'events' key must be an array of strings"); } diff --git a/packages/plugos/package.json b/packages/plugos/package.json index 37282026..362de856 100644 --- a/packages/plugos/package.json +++ b/packages/plugos/package.json @@ -60,7 +60,8 @@ "vm2": "^3.9.9", "ws": "^8.5.0", "yaml": "^1.10.2", - "yargs": "^17.3.1" + "yargs": "^17.3.1", + "typescript": "^4.6.2" }, "devDependencies": { "@lezer/lr": "^0.15.0", @@ -82,7 +83,6 @@ "assert": "^2.0.0", "events": "^3.3.0", "parcel": "2.3.2", - "prettier": "^2.5.1", - "typescript": "^4.6.2" + "prettier": "^2.5.1" } } diff --git a/packages/plugos/plug.ts b/packages/plugos/plug.ts index 9230c749..fde12c78 100644 --- a/packages/plugos/plug.ts +++ b/packages/plugos/plug.ts @@ -27,6 +27,9 @@ export class Plug { this.manifest = manifest; // TODO: These need to be explicitly granted, not just taken this.grantedPermissions = manifest.requiredPermissions || []; + for (let [dep, code] of Object.entries(manifest.dependencies || {})) { + await this.sandbox.loadDependency(dep, code); + } } syscall(name: string, args: any[]): Promise { diff --git a/packages/plugos/sandbox.ts b/packages/plugos/sandbox.ts index 52610f34..5237dff7 100644 --- a/packages/plugos/sandbox.ts +++ b/packages/plugos/sandbox.ts @@ -18,6 +18,7 @@ export class Sandbox { protected worker: WorkerLike; protected reqId = 0; protected outstandingInits = new Map void>(); + protected outstandingDependencyInits = new Map void>(); protected outstandingInvocations = new Map< number, { resolve: (result: any) => void; reject: (e: any) => void } @@ -63,6 +64,22 @@ export class Sandbox { }); } + async loadDependency(name: string, code: string): Promise { + // console.log("Loading dependency", name); + this.worker.postMessage({ + type: "load-dependency", + name: name, + code: code, + } as WorkerMessage); + return new Promise((resolve) => { + // console.log("Loaded dependency", name); + this.outstandingDependencyInits.set(name, () => { + this.outstandingDependencyInits.delete(name); + resolve(); + }); + }); + } + async onMessage(data: ControllerMessage) { switch (data.type) { case "inited": @@ -70,6 +87,11 @@ export class Sandbox { initCb && initCb(); this.outstandingInits.delete(data.name!); break; + case "dependency-inited": + let depInitCb = this.outstandingDependencyInits.get(data.name!); + depInitCb && depInitCb(); + this.outstandingDependencyInits.delete(data.name!); + break; case "syscall": try { let result = await this.plug.syscall(data.name!, data.args!); diff --git a/packages/plugos/syscalls/esbuild.ts b/packages/plugos/syscalls/esbuild.ts index 7dff6e86..71780b53 100644 --- a/packages/plugos/syscalls/esbuild.ts +++ b/packages/plugos/syscalls/esbuild.ts @@ -10,36 +10,95 @@ const exposedModules = [ "yaml", ]; +import * as ts from "typescript"; + +type CompileError = { + message: string; + pos: number; +}; + +function checkTypeScript(scriptFile: string): void { + let program = ts.createProgram([scriptFile], { + noEmit: true, + allowJs: true, + }); + let emitResult = program.emit(); + + let allDiagnostics = ts + .getPreEmitDiagnostics(program) + .concat(emitResult.diagnostics); + + let errors: CompileError[] = []; + allDiagnostics.forEach((diagnostic) => { + if (diagnostic.file) { + let { line, character } = ts.getLineAndCharacterOfPosition( + diagnostic.file, + diagnostic.start! + ); + let message = ts.flattenDiagnosticMessageText( + diagnostic.messageText, + "\n" + ); + errors.push({ + message: ts.flattenDiagnosticMessageText(diagnostic.messageText, "\n"), + pos: diagnostic.start!, + }); + // console.log( + // `${diagnostic.file.fileName} (${line + 1},${character + 1}): ${message}` + // ); + } else { + console.log( + ts.flattenDiagnosticMessageText(diagnostic.messageText, "\n") + ); + } + }); + + let exitCode = emitResult.emitSkipped ? 1 : 0; + console.log(`Process exiting with code '${exitCode}'.`); + process.exit(exitCode); +} + export function esbuildSyscalls(): SysCallMapping { return { + "tsc.analyze": async ( + ctx, + filename: string, + code: string + ): Promise => {}, "esbuild.compile": async ( ctx, filename: string, code: string, functionName?: string ): Promise => { - let tmpDir = `${tmpdir()}/plugos-${Math.random()}`; - await mkdir(tmpDir, { recursive: true }); - - const srcNodeModules = `${nodeModulesDir}/node_modules`; - const targetNodeModules = `${tmpDir}/node_modules`; - - await mkdir(`${targetNodeModules}/@silverbulletmd`, { recursive: true }); - await mkdir(`${targetNodeModules}/@plugos`, { recursive: true }); - for (const exposedModule of exposedModules) { - await symlink( - `${srcNodeModules}/${exposedModule}`, - `${targetNodeModules}/${exposedModule}`, - "dir" - ); - } - - await writeFile(`${tmpDir}/${filename}`, code); + let tmpDir = await prepareCompileEnv(filename, code); let jsCode = await compile(`${tmpDir}/${filename}`, functionName, false, [ "yaml", + "handlebars", ]); await rm(tmpDir, { recursive: true }); return jsCode; }, }; } + +async function prepareCompileEnv(filename: string, code: string) { + let tmpDir = `${tmpdir()}/plugos-${Math.random()}`; + await mkdir(tmpDir, { recursive: true }); + + const srcNodeModules = `${nodeModulesDir}/node_modules`; + const targetNodeModules = `${tmpDir}/node_modules`; + + await mkdir(`${targetNodeModules}/@silverbulletmd`, { recursive: true }); + await mkdir(`${targetNodeModules}/@plugos`, { recursive: true }); + for (const exposedModule of exposedModules) { + await symlink( + `${srcNodeModules}/${exposedModule}`, + `${targetNodeModules}/${exposedModule}`, + "dir" + ); + } + + await writeFile(`${tmpDir}/${filename}`, code); + return tmpDir; +} diff --git a/packages/plugos/types.ts b/packages/plugos/types.ts index 4ec82e90..e498fe6e 100644 --- a/packages/plugos/types.ts +++ b/packages/plugos/types.ts @@ -3,6 +3,9 @@ import { System } from "./system"; export interface Manifest { name: string; requiredPermissions?: string[]; + dependencies?: { + [key: string]: string; + }; functions: { [key: string]: FunctionDef; }; diff --git a/packages/plugs/build/test/engine.test.js b/packages/plugs/build/test/engine.test.js index 8d467c68..14ca30fa 100644 --- a/packages/plugs/build/test/engine.test.js +++ b/packages/plugs/build/test/engine.test.js @@ -422,6 +422,7 @@ function $9072202279b76d33$export$5884dae03c64f759(parsedQuery, records) { } async function $9072202279b76d33$export$b3c659c1456e61b0(parsedQuery, data) { if (parsedQuery.render) { + console.log("Handlebars", ($parcel$interopDefault($hVExJ$handlebars))); ($parcel$interopDefault($hVExJ$handlebars)).registerHelper("json", (v)=>JSON.stringify(v) ); ($parcel$interopDefault($hVExJ$handlebars)).registerHelper("niceDate", (ts)=>$c3893eec0c49ec96$export$5dc1410f87262ed6(new Date(ts)) diff --git a/packages/plugs/build/test/engine.test.js.map b/packages/plugs/build/test/engine.test.js.map index 2fc80b21..1f60b201 100644 --- a/packages/plugs/build/test/engine.test.js.map +++ b/packages/plugs/build/test/engine.test.js.map @@ -1 +1 @@ -{"mappings":"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","sources":["packages/plugs/query/engine.test.ts","packages/plugs/query/engine.ts","packages/common/tree.ts","packages/common/parse_tree.ts","packages/plugs/query/parse-query.js","packages/plugos-silverbullet-syscall/space.ts","packages/plugos-silverbullet-syscall/syscall.ts","packages/plugs/core/dates.ts","packages/plugos-silverbullet-syscall/editor.ts"],"sourcesContent":["import { expect, test } from \"@jest/globals\";\nimport { applyQuery, parseQuery } from \"./engine\";\n\ntest(\"Test parser\", () => {\n let parsedBasicQuery = parseQuery(`page`);\n expect(parsedBasicQuery.table).toBe(\"page\");\n\n let parsedQuery1 = parseQuery(\n `task where completed = false and dueDate <= \"{{today}}\" order by dueDate desc limit 5`\n );\n expect(parsedQuery1.table).toBe(\"task\");\n expect(parsedQuery1.orderBy).toBe(\"dueDate\");\n expect(parsedQuery1.orderDesc).toBe(true);\n expect(parsedQuery1.limit).toBe(5);\n expect(parsedQuery1.filter.length).toBe(2);\n expect(parsedQuery1.filter[0]).toStrictEqual({\n op: \"=\",\n prop: \"completed\",\n value: false,\n });\n expect(parsedQuery1.filter[1]).toStrictEqual({\n op: \"<=\",\n prop: \"dueDate\",\n value: \"{{today}}\",\n });\n\n let parsedQuery2 = parseQuery(`page where name =~ /interview\\\\/.*/\"`);\n expect(parsedQuery2.table).toBe(\"page\");\n expect(parsedQuery2.filter.length).toBe(1);\n expect(parsedQuery2.filter[0]).toStrictEqual({\n op: \"=~\",\n prop: \"name\",\n value: \"interview\\\\/.*\",\n });\n\n let parsedQuery3 = parseQuery(`page where something != null`);\n expect(parsedQuery3.table).toBe(\"page\");\n expect(parsedQuery3.filter.length).toBe(1);\n expect(parsedQuery3.filter[0]).toStrictEqual({\n op: \"!=\",\n prop: \"something\",\n value: null,\n });\n\n expect(parseQuery(`page select name`).select).toStrictEqual([\"name\"]);\n expect(parseQuery(`page select name, age`).select).toStrictEqual([\n \"name\",\n \"age\",\n ]);\n\n expect(\n parseQuery(`gh-events where type in [\"PushEvent\", \"somethingElse\"]`)\n ).toStrictEqual({\n table: \"gh-events\",\n filter: [\n {\n op: \"in\",\n prop: \"type\",\n value: [\"PushEvent\", \"somethingElse\"],\n },\n ],\n });\n\n expect(parseQuery(`something render \"template/table\"`)).toStrictEqual({\n table: \"something\",\n filter: [],\n render: \"template/table\",\n });\n});\n\ntest(\"Test performing the queries\", () => {\n let data: any[] = [\n { name: \"interview/My Interview\", lastModified: 1 },\n { name: \"interview/My Interview 2\", lastModified: 2 },\n { name: \"Pete\", age: 38 },\n { name: \"Angie\", age: 28 },\n ];\n\n expect(\n applyQuery(parseQuery(`page where name =~ /interview\\\\/.*/`), data)\n ).toStrictEqual([\n { name: \"interview/My Interview\", lastModified: 1 },\n { name: \"interview/My Interview 2\", lastModified: 2 },\n ]);\n expect(\n applyQuery(\n parseQuery(`page where name =~ /interview\\\\/.*/ order by lastModified`),\n data\n )\n ).toStrictEqual([\n { name: \"interview/My Interview\", lastModified: 1 },\n { name: \"interview/My Interview 2\", lastModified: 2 },\n ]);\n expect(\n applyQuery(\n parseQuery(\n `page where name =~ /interview\\\\/.*/ order by lastModified desc`\n ),\n data\n )\n ).toStrictEqual([\n { name: \"interview/My Interview 2\", lastModified: 2 },\n { name: \"interview/My Interview\", lastModified: 1 },\n ]);\n expect(applyQuery(parseQuery(`page where age > 30`), data)).toStrictEqual([\n { name: \"Pete\", age: 38 },\n ]);\n expect(\n applyQuery(parseQuery(`page where age > 28 and age < 38`), data)\n ).toStrictEqual([]);\n expect(\n applyQuery(parseQuery(`page where age > 30 select name`), data)\n ).toStrictEqual([{ name: \"Pete\" }]);\n\n expect(\n applyQuery(parseQuery(`page where name in [\"Pete\"] select name`), data)\n ).toStrictEqual([{ name: \"Pete\" }]);\n});\n","import {\n collectNodesOfType,\n findNodeOfType,\n ParseTree,\n replaceNodesMatching,\n} from \"@silverbulletmd/common/tree\";\nimport { lezerToParseTree } from \"@silverbulletmd/common/parse_tree\";\nimport Handlebars from \"handlebars\";\nimport YAML from \"yaml\";\n\n// @ts-ignore\nimport { parser } from \"./parse-query\";\nimport { readPage } from \"@silverbulletmd/plugos-silverbullet-syscall/space\";\nimport { niceDate } from \"../core/dates\";\n\nexport type QueryProviderEvent = {\n query: ParsedQuery;\n pageName: string;\n};\n\nexport type Filter = {\n op: string;\n prop: string;\n value: any;\n};\n\nexport type ParsedQuery = {\n table: string;\n orderBy?: string;\n orderDesc?: boolean;\n limit?: number;\n filter: Filter[];\n select?: string[];\n render?: string;\n};\n\nexport function parseQuery(query: string): ParsedQuery {\n let n = lezerToParseTree(query, parser.parse(query).topNode);\n // Clean the tree a bit\n replaceNodesMatching(n, (n) => {\n if (!n.type) {\n let trimmed = n.text!.trim();\n if (!trimmed) {\n return null;\n }\n n.text = trimmed;\n }\n });\n\n // console.log(\"Parsed\", JSON.stringify(n, null, 2));\n\n let queryNode = n.children![0];\n let parsedQuery: ParsedQuery = {\n table: queryNode.children![0].children![0].text!,\n filter: [],\n };\n let orderByNode = findNodeOfType(queryNode, \"OrderClause\");\n if (orderByNode) {\n let nameNode = findNodeOfType(orderByNode, \"Name\");\n parsedQuery.orderBy = nameNode!.children![0].text!;\n let orderNode = findNodeOfType(orderByNode, \"Order\");\n parsedQuery.orderDesc = orderNode\n ? orderNode.children![0].text! === \"desc\"\n : false;\n }\n let limitNode = findNodeOfType(queryNode, \"LimitClause\");\n if (limitNode) {\n let nameNode = findNodeOfType(limitNode, \"Number\");\n parsedQuery.limit = valueNodeToVal(nameNode!);\n }\n\n let filterNodes = collectNodesOfType(queryNode, \"FilterExpr\");\n for (let filterNode of filterNodes) {\n let val: any = undefined;\n let valNode = filterNode.children![2].children![0];\n val = valueNodeToVal(valNode);\n let f: Filter = {\n prop: filterNode.children![0].children![0].text!,\n op: filterNode.children![1].text!,\n value: val,\n };\n parsedQuery.filter.push(f);\n }\n let selectNode = findNodeOfType(queryNode, \"SelectClause\");\n if (selectNode) {\n // console.log(\"Select node\", JSON.stringify(selectNode));\n parsedQuery.select = [];\n collectNodesOfType(selectNode, \"Name\").forEach((t) => {\n parsedQuery.select!.push(t.children![0].text!);\n });\n // let nameNode = findNodeOfType(selectNode, \"Number\");\n // parsedQuery.limit = +nameNode!.children![0].text!;\n }\n\n let renderNode = findNodeOfType(queryNode, \"RenderClause\");\n if (renderNode) {\n let renderNameNode = findNodeOfType(renderNode, \"String\");\n parsedQuery.render = valueNodeToVal(renderNameNode!);\n }\n\n // console.log(JSON.stringify(queryNode, null, 2));\n return parsedQuery;\n}\n\nfunction valueNodeToVal(valNode: ParseTree): any {\n switch (valNode.type) {\n case \"Number\":\n return +valNode.children![0].text!;\n case \"Bool\":\n return valNode.children![0].text! === \"true\";\n case \"Null\":\n return null;\n case \"Name\":\n return valNode.children![0].text!;\n case \"Regex\":\n let val = valNode.children![0].text!;\n return val.substring(1, val.length - 1);\n case \"String\":\n let stringVal = valNode.children![0].text!;\n return stringVal.substring(1, stringVal.length - 1);\n case \"List\":\n return collectNodesOfType(valNode, \"Value\").map((t) =>\n valueNodeToVal(t.children![0])\n );\n }\n}\n\nexport function applyQuery(parsedQuery: ParsedQuery, records: T[]): T[] {\n let resultRecords: any[] = [];\n if (parsedQuery.filter.length === 0) {\n resultRecords = records.slice();\n } else {\n recordLoop: for (let record of records) {\n const recordAny: any = record;\n for (let { op, prop, value } of parsedQuery.filter) {\n switch (op) {\n case \"=\":\n if (!(recordAny[prop] == value)) {\n continue recordLoop;\n }\n break;\n case \"!=\":\n if (!(recordAny[prop] != value)) {\n continue recordLoop;\n }\n break;\n case \"<\":\n if (!(recordAny[prop] < value)) {\n continue recordLoop;\n }\n break;\n case \"<=\":\n if (!(recordAny[prop] <= value)) {\n continue recordLoop;\n }\n break;\n case \">\":\n if (!(recordAny[prop] > value)) {\n continue recordLoop;\n }\n break;\n case \">=\":\n if (!(recordAny[prop] >= value)) {\n continue recordLoop;\n }\n break;\n case \"=~\":\n // TODO: Cache regexps somehow\n if (!new RegExp(value).exec(recordAny[prop])) {\n continue recordLoop;\n }\n break;\n case \"!=~\":\n if (new RegExp(value).exec(recordAny[prop])) {\n continue recordLoop;\n }\n break;\n case \"in\":\n if (!value.includes(recordAny[prop])) {\n continue recordLoop;\n }\n break;\n }\n }\n resultRecords.push(recordAny);\n }\n }\n // Now the sorting\n if (parsedQuery.orderBy) {\n resultRecords = resultRecords.sort((a: any, b: any) => {\n const orderBy = parsedQuery.orderBy!;\n const orderDesc = parsedQuery.orderDesc!;\n if (a[orderBy] === b[orderBy]) {\n return 0;\n }\n\n if (a[orderBy] < b[orderBy]) {\n return orderDesc ? 1 : -1;\n } else {\n return orderDesc ? -1 : 1;\n }\n });\n }\n if (parsedQuery.limit) {\n resultRecords = resultRecords.slice(0, parsedQuery.limit);\n }\n if (parsedQuery.select) {\n resultRecords = resultRecords.map((rec) => {\n let newRec: any = {};\n for (let k of parsedQuery.select!) {\n newRec[k] = rec[k];\n }\n return newRec;\n });\n }\n return resultRecords;\n}\n\nexport async function renderQuery(\n parsedQuery: ParsedQuery,\n data: any[]\n): Promise {\n if (parsedQuery.render) {\n Handlebars.registerHelper(\"json\", (v) => JSON.stringify(v));\n Handlebars.registerHelper(\"niceDate\", (ts) => niceDate(new Date(ts)));\n Handlebars.registerHelper(\"yaml\", (v, prefix) => {\n if (typeof prefix === \"string\") {\n let yaml = YAML.stringify(v)\n .split(\"\\n\")\n .join(\"\\n\" + prefix)\n .trim();\n if (Array.isArray(v)) {\n return \"\\n\" + prefix + yaml;\n } else {\n return yaml;\n }\n } else {\n return YAML.stringify(v).trim();\n }\n });\n let { text: templateText } = await readPage(parsedQuery.render);\n let template = Handlebars.compile(templateText, { noEscape: true });\n return template(data);\n }\n\n return \"ERROR\";\n}\n","export type ParseTree = {\n type?: string; // undefined === text node\n from?: number;\n to?: number;\n text?: string;\n children?: ParseTree[];\n // Only present after running addParentPointers\n parent?: ParseTree;\n};\n\nexport function addParentPointers(tree: ParseTree) {\n if (!tree.children) {\n return;\n }\n for (let child of tree.children) {\n if (child.parent) {\n // Already added parent pointers before\n return;\n }\n child.parent = tree;\n addParentPointers(child);\n }\n}\n\nexport function removeParentPointers(tree: ParseTree) {\n delete tree.parent;\n if (!tree.children) {\n return;\n }\n for (let child of tree.children) {\n removeParentPointers(child);\n }\n}\n\nexport function findParentMatching(\n tree: ParseTree,\n matchFn: (tree: ParseTree) => boolean\n): ParseTree | null {\n let node = tree.parent;\n while (node) {\n if (matchFn(node)) {\n return node;\n }\n node = node.parent;\n }\n return null;\n}\n\nexport function collectNodesOfType(\n tree: ParseTree,\n nodeType: string\n): ParseTree[] {\n return collectNodesMatching(tree, (n) => n.type === nodeType);\n}\n\nexport function collectNodesMatching(\n tree: ParseTree,\n matchFn: (tree: ParseTree) => boolean\n): ParseTree[] {\n if (matchFn(tree)) {\n return [tree];\n }\n let results: ParseTree[] = [];\n if (tree.children) {\n for (let child of tree.children) {\n results = [...results, ...collectNodesMatching(child, matchFn)];\n }\n }\n return results;\n}\n\n// return value: returning undefined = not matched, continue, null = delete, new node = replace\nexport function replaceNodesMatching(\n tree: ParseTree,\n substituteFn: (tree: ParseTree) => ParseTree | null | undefined\n) {\n if (tree.children) {\n let children = tree.children.slice();\n for (let child of children) {\n let subst = substituteFn(child);\n if (subst !== undefined) {\n let pos = tree.children.indexOf(child);\n if (subst) {\n tree.children.splice(pos, 1, subst);\n } else {\n // null = delete\n tree.children.splice(pos, 1);\n }\n } else {\n replaceNodesMatching(child, substituteFn);\n }\n }\n }\n}\n\nexport function findNodeMatching(\n tree: ParseTree,\n matchFn: (tree: ParseTree) => boolean\n): ParseTree | null {\n return collectNodesMatching(tree, matchFn)[0];\n}\n\nexport function findNodeOfType(\n tree: ParseTree,\n nodeType: string\n): ParseTree | null {\n return collectNodesMatching(tree, (n) => n.type === nodeType)[0];\n}\n\n// Finds non-text node at position\nexport function nodeAtPos(tree: ParseTree, pos: number): ParseTree | null {\n if (pos < tree.from! || pos > tree.to!) {\n return null;\n }\n if (!tree.children) {\n return tree;\n }\n for (let child of tree.children) {\n let n = nodeAtPos(child, pos);\n if (n && n.text !== undefined) {\n // Got a text node, let's return its parent\n return tree;\n } else if (n) {\n // Got it\n return n;\n }\n }\n return null;\n}\n\n// Turn ParseTree back into text\nexport function renderToText(tree: ParseTree): string {\n let pieces: string[] = [];\n if (tree.text !== undefined) {\n return tree.text;\n }\n for (let child of tree.children!) {\n pieces.push(renderToText(child));\n }\n return pieces.join(\"\");\n}\n","import type { SyntaxNode } from \"@lezer/common\";\nimport type { Language } from \"@codemirror/language\";\nimport { ParseTree } from \"./tree\";\n\nexport function lezerToParseTree(\n text: string,\n n: SyntaxNode,\n offset = 0\n): ParseTree {\n let children: ParseTree[] = [];\n let nodeText: string | undefined;\n let child = n.firstChild;\n while (child) {\n children.push(lezerToParseTree(text, child));\n child = child.nextSibling;\n }\n\n if (children.length === 0) {\n children = [\n {\n from: n.from + offset,\n to: n.to + offset,\n text: text.substring(n.from, n.to),\n },\n ];\n } else {\n let newChildren: ParseTree[] = [];\n let index = n.from;\n for (let child of children) {\n let s = text.substring(index, child.from);\n if (s) {\n newChildren.push({\n from: index + offset,\n to: child.from! + offset,\n text: s,\n });\n }\n newChildren.push(child);\n index = child.to!;\n }\n let s = text.substring(index, n.to);\n if (s) {\n newChildren.push({ from: index + offset, to: n.to + offset, text: s });\n }\n children = newChildren;\n }\n\n let result: ParseTree = {\n type: n.name,\n from: n.from + offset,\n to: n.to + offset,\n };\n if (children.length > 0) {\n result.children = children;\n }\n if (nodeText) {\n result.text = nodeText;\n }\n return result;\n}\n\nexport function parse(language: Language, text: string): ParseTree {\n let tree = lezerToParseTree(text, language.parser.parse(text).topNode);\n // replaceNodesMatching(tree, (n): MarkdownTree | undefined | null => {\n // if (n.type === \"FencedCode\") {\n // let infoN = findNodeMatching(n, (n) => n.type === \"CodeInfo\");\n // let language = infoN!.children![0].text;\n // let textN = findNodeMatching(n, (n) => n.type === \"CodeText\");\n // let text = textN!.children![0].text!;\n //\n // console.log(language, text);\n // switch (language) {\n // case \"yaml\":\n // let parsed = StreamLanguage.define(yaml).parser.parse(text);\n // let subTree = treeToAST(text, parsed.topNode, n.from);\n // // console.log(JSON.stringify(subTree, null, 2));\n // subTree.type = \"yaml\";\n // return subTree;\n // }\n // }\n // return;\n // });\n return tree;\n}\n","// This file was generated by lezer-generator. You probably shouldn't edit it.\nimport {LRParser} from \"@lezer/lr\"\nexport const parser = LRParser.deserialize({\n version: 13,\n states: \"&fOVQPOOOmQQO'#C^QOQPOOOtQPO'#C`OyQQO'#CkO!OQPO'#CmO!TQPO'#CnO!YQPO'#CoOOQO'#Cp'#CpO!_QQO,58xO!fQQO'#CcO#TQQO'#CaOOQO'#Ca'#CaOOQO,58z,58zO#lQPO,59VOOQO,59X,59XO#qQQO'#D`OOQO,59Y,59YOOQO,59Z,59ZOOQO-E6n-E6nO$YQQO,58}OtQPO,58|O$qQQO1G.qO%]QPO'#CrO%bQQO,59zOOQO'#Cg'#CgOOQO'#Ci'#CiO$YQQO'#CjOOQO'#Cd'#CdOOQO1G.i1G.iOOQO1G.h1G.hOOQO'#Cl'#ClOOQO7+$]7+$]OOQO,59^,59^OOQO-E6p-E6pO%yQPO'#C|O&RQPO,59UO$YQQO'#CqO&WQPO,59hOOQO1G.p1G.pOOQO,59],59]OOQO-E6o-E6o\",\n stateData: \"&`~OiOS~ORPO~OjRO|SO!QTO!RUO!TVO~OgQX~P[ORYO~O}^O~OX_O~OR`O~OYbO~OgQa~P[OkdOsdOtdOudOvdOwdOxdOydOzdO~O{eOgTXjTX|TX!QTX!RTX!TTX~ORfO~OqgOg!SXj!SX|!SX!Q!SX!R!SX!T!SX~OXlOYlO[lOliOmiOnjOokO~O!OoO!PoOg_ij_i|_i!Q_i!R_i!T_i~ORqO~OqgOg!Saj!Sa|!Sa!Q!Sa!R!Sa!T!Sa~OquOrpX~OrwO~OquOrpa~O\",\n goto: \"#d!TPP!UP!X!]!`!c!iPP!rP!r!r!X!w!X!X!X!z#Q#WPPPPPPPPP#^PPPPPPPPPPPPPPPPP#aRQOTWPXR]RR[RQZRRneQmdQskRxuVldkuRpfQXPRcXQvsRyvQh`RrhRtkRaU\",\n nodeNames: \"⚠ Program Query Name WhereClause LogicalExpr AndExpr FilterExpr Value Number String Bool Regex Null List OrderClause Order LimitClause SelectClause RenderClause\",\n maxTerm: 51,\n skippedNodes: [0],\n repeatNodeCount: 3,\n tokenData: \"Ap~R}X^$Opq$Oqr$srs%W|}%r}!O%w!P!Q&Y!Q!['P!^!_'X!_!`'f!`!a's!c!}%w!}#O(Q#P#Q(V#R#S%w#T#U([#U#V*q#V#W%w#W#X+m#X#Y%w#Y#Z-i#Z#]%w#]#^/y#^#`%w#`#a0u#a#b%w#b#c3Y#c#d5U#d#f%w#f#g7i#g#h:e#h#i=a#i#k%w#k#l?]#l#o%w#y#z$O$f$g$O#BY#BZ$O$IS$I_$O$Ip$Iq%W$Iq$Ir%W$I|$JO$O$JT$JU$O$KV$KW$O&FU&FV$O~$TYi~X^$Opq$O#y#z$O$f$g$O#BY#BZ$O$IS$I_$O$I|$JO$O$JT$JU$O$KV$KW$O&FU&FV$O~$vP!_!`$y~%OPu~#r#s%R~%WOy~~%ZUOr%Wrs%ms$Ip%W$Ip$Iq%m$Iq$Ir%m$Ir~%W~%rOY~~%wOq~P%|SRP}!O%w!c!}%w#R#S%w#T#o%w~&_V[~OY&YZ]&Y^!P&Y!P!Q&t!Q#O&Y#O#P&y#P~&Y~&yO[~~&|PO~&Y~'UPX~!Q!['P~'^Pk~!_!`'a~'fOs~~'kPt~#r#s'n~'sOx~~'xPw~!_!`'{~(QOv~~(VOo~~([Or~R(aWRP}!O%w!c!}%w#R#S%w#T#b%w#b#c(y#c#g%w#g#h)u#h#o%wR)OURP}!O%w!c!}%w#R#S%w#T#W%w#W#X)b#X#o%wR)iS{QRP}!O%w!c!}%w#R#S%w#T#o%wR)zURP}!O%w!c!}%w#R#S%w#T#V%w#V#W*^#W#o%wR*eS!PQRP}!O%w!c!}%w#R#S%w#T#o%wR*vURP}!O%w!c!}%w#R#S%w#T#m%w#m#n+Y#n#o%wR+aS}QRP}!O%w!c!}%w#R#S%w#T#o%wR+rURP}!O%w!c!}%w#R#S%w#T#X%w#X#Y,U#Y#o%wR,ZURP}!O%w!c!}%w#R#S%w#T#g%w#g#h,m#h#o%wR,rURP}!O%w!c!}%w#R#S%w#T#V%w#V#W-U#W#o%wR-]S!OQRP}!O%w!c!}%w#R#S%w#T#o%wR-nTRP}!O%w!c!}%w#R#S%w#T#U-}#U#o%wR.SURP}!O%w!c!}%w#R#S%w#T#`%w#`#a.f#a#o%wR.kURP}!O%w!c!}%w#R#S%w#T#g%w#g#h.}#h#o%wR/SURP}!O%w!c!}%w#R#S%w#T#X%w#X#Y/f#Y#o%wR/mSmQRP}!O%w!c!}%w#R#S%w#T#o%wR0OURP}!O%w!c!}%w#R#S%w#T#b%w#b#c0b#c#o%wR0iSzQRP}!O%w!c!}%w#R#S%w#T#o%wR0zURP}!O%w!c!}%w#R#S%w#T#]%w#]#^1^#^#o%wR1cURP}!O%w!c!}%w#R#S%w#T#a%w#a#b1u#b#o%wR1zURP}!O%w!c!}%w#R#S%w#T#]%w#]#^2^#^#o%wR2cURP}!O%w!c!}%w#R#S%w#T#h%w#h#i2u#i#o%wR2|S!QQRP}!O%w!c!}%w#R#S%w#T#o%wR3_URP}!O%w!c!}%w#R#S%w#T#i%w#i#j3q#j#o%wR3vURP}!O%w!c!}%w#R#S%w#T#`%w#`#a4Y#a#o%wR4_URP}!O%w!c!}%w#R#S%w#T#`%w#`#a4q#a#o%wR4xSnQRP}!O%w!c!}%w#R#S%w#T#o%wR5ZURP}!O%w!c!}%w#R#S%w#T#f%w#f#g5m#g#o%wR5rURP}!O%w!c!}%w#R#S%w#T#W%w#W#X6U#X#o%wR6ZURP}!O%w!c!}%w#R#S%w#T#X%w#X#Y6m#Y#o%wR6rURP}!O%w!c!}%w#R#S%w#T#f%w#f#g7U#g#o%wR7]S|QRP}!O%w!c!}%w#R#S%w#T#o%wR7nURP}!O%w!c!}%w#R#S%w#T#X%w#X#Y8Q#Y#o%wR8VURP}!O%w!c!}%w#R#S%w#T#b%w#b#c8i#c#o%wR8nURP}!O%w!c!}%w#R#S%w#T#W%w#W#X9Q#X#o%wR9VURP}!O%w!c!}%w#R#S%w#T#X%w#X#Y9i#Y#o%wR9nURP}!O%w!c!}%w#R#S%w#T#f%w#f#g:Q#g#o%wR:XS!TQRP}!O%w!c!}%w#R#S%w#T#o%wR:jURP}!O%w!c!}%w#R#S%w#T#X%w#X#Y:|#Y#o%wR;RURP}!O%w!c!}%w#R#S%w#T#`%w#`#a;e#a#o%wR;jURP}!O%w!c!}%w#R#S%w#T#X%w#X#Y;|#Y#o%wRa#j#o%wR>fURP}!O%w!c!}%w#R#S%w#T#X%w#X#Y>x#Y#o%wR?PSlQRP}!O%w!c!}%w#R#S%w#T#o%wR?bURP}!O%w!c!}%w#R#S%w#T#[%w#[#]?t#]#o%wR?yURP}!O%w!c!}%w#R#S%w#T#X%w#X#Y@]#Y#o%wR@bURP}!O%w!c!}%w#R#S%w#T#f%w#f#g@t#g#o%wR@yURP}!O%w!c!}%w#R#S%w#T#X%w#X#YA]#Y#o%wRAdSjQRP}!O%w!c!}%w#R#S%w#T#o%w\",\n tokenizers: [0, 1],\n topRules: {\"Program\":[0,1]},\n tokenPrec: 0\n})\n","import { syscall } from \"./syscall\";\nimport { PageMeta } from \"../common/types\";\n\nexport async function listPages(unfiltered = false): Promise {\n return syscall(\"space.listPages\", unfiltered);\n}\n\nexport async function readPage(\n name: string\n): Promise<{ text: string; meta: PageMeta }> {\n return syscall(\"space.readPage\", name);\n}\n\nexport async function writePage(name: string, text: string): Promise {\n return syscall(\"space.writePage\", name, text);\n}\n\nexport async function deletePage(name: string): Promise {\n return syscall(\"space.deletePage\", name);\n}\n","declare global {\n function syscall(name: string, ...args: any[]): Promise;\n}\n\nexport const syscall = self.syscall;\n","import { insertAtCursor } from \"@silverbulletmd/plugos-silverbullet-syscall/editor\";\n\nconst dateMatchRegex = /(\\d{4}\\-\\d{2}\\-\\d{2})/g;\n\nexport function niceDate(d: Date): string {\n return d.toISOString().split(\"T\")[0];\n}\n\nexport async function insertToday() {\n await insertAtCursor(niceDate(new Date()));\n}\n\nexport async function insertTomorrow() {\n let d = new Date();\n d.setDate(d.getDate() + 1);\n await insertAtCursor(niceDate(d));\n}\n","import { syscall } from \"./syscall\";\nimport { FilterOption } from \"../common/types\";\n\nexport function getCurrentPage(): Promise {\n return syscall(\"editor.getCurrentPage\");\n}\n\nexport function setPage(newName: string): Promise {\n return syscall(\"editor.setPage\", newName);\n}\n\nexport function getText(): Promise {\n return syscall(\"editor.getText\");\n}\n\nexport function getCursor(): Promise {\n return syscall(\"editor.getCursor\");\n}\n\nexport function save(): Promise {\n return syscall(\"editor.save\");\n}\n\nexport function navigate(name: string, pos?: number): Promise {\n return syscall(\"editor.navigate\", name, pos);\n}\n\nexport function reloadPage(): Promise {\n return syscall(\"editor.reloadPage\");\n}\n\nexport function openUrl(url: string): Promise {\n return syscall(\"editor.openUrl\", url);\n}\n\nexport function flashNotification(message: string): Promise {\n return syscall(\"editor.flashNotification\", message);\n}\n\nexport function filterBox(\n label: string,\n options: FilterOption[],\n helpText: string = \"\",\n placeHolder: string = \"\"\n): Promise {\n return syscall(\"editor.filterBox\", label, options, helpText, placeHolder);\n}\n\nexport function showRhs(\n html: string,\n script?: string,\n flex = 1\n): Promise {\n return syscall(\"editor.showRhs\", html, script, flex);\n}\n\nexport function hideRhs(): Promise {\n return syscall(\"editor.hideRhs\");\n}\n\nexport function showLhs(\n html: string,\n script?: string,\n flex = 1\n): Promise {\n return syscall(\"editor.showLhs\", html, script, flex);\n}\n\nexport function hideLhs(): Promise {\n return syscall(\"editor.hideLhs\");\n}\n\nexport function showBhs(\n html: string,\n script?: string,\n flex = 1\n): Promise {\n return syscall(\"editor.showBhs\", html, script, flex);\n}\n\nexport function hideBhs(): Promise {\n return syscall(\"editor.hideBhs\");\n}\n\nexport function insertAtPos(text: string, pos: number): Promise {\n return syscall(\"editor.insertAtPos\", text, pos);\n}\n\nexport function replaceRange(\n from: number,\n to: number,\n text: string\n): Promise {\n return syscall(\"editor.replaceRange\", from, to, text);\n}\n\nexport function moveCursor(pos: number): Promise {\n return syscall(\"editor.moveCursor\", pos);\n}\n\nexport function insertAtCursor(text: string): Promise {\n return syscall(\"editor.insertAtCursor\", text);\n}\n\nexport function matchBefore(\n re: string\n): Promise<{ from: number; to: number; text: string } | null> {\n return syscall(\"editor.matchBefore\", re);\n}\n\nexport function dispatch(change: any): Promise {\n return syscall(\"editor.dispatch\", change);\n}\n\nexport function prompt(\n message: string,\n defaultValue = \"\"\n): Promise {\n return syscall(\"editor.prompt\", message, defaultValue);\n}\n"],"names":[],"version":3,"file":"engine.test.js.map","sourceRoot":"../../../../"} \ No newline at end of file +{"mappings":"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","sources":["packages/plugs/query/engine.test.ts","packages/plugs/query/engine.ts","packages/common/tree.ts","packages/common/parse_tree.ts","packages/plugs/query/parse-query.js","packages/plugos-silverbullet-syscall/space.ts","packages/plugos-silverbullet-syscall/syscall.ts","packages/plugs/core/dates.ts","packages/plugos-silverbullet-syscall/editor.ts"],"sourcesContent":["import { expect, test } from \"@jest/globals\";\nimport { applyQuery, parseQuery } from \"./engine\";\n\ntest(\"Test parser\", () => {\n let parsedBasicQuery = parseQuery(`page`);\n expect(parsedBasicQuery.table).toBe(\"page\");\n\n let parsedQuery1 = parseQuery(\n `task where completed = false and dueDate <= \"{{today}}\" order by dueDate desc limit 5`\n );\n expect(parsedQuery1.table).toBe(\"task\");\n expect(parsedQuery1.orderBy).toBe(\"dueDate\");\n expect(parsedQuery1.orderDesc).toBe(true);\n expect(parsedQuery1.limit).toBe(5);\n expect(parsedQuery1.filter.length).toBe(2);\n expect(parsedQuery1.filter[0]).toStrictEqual({\n op: \"=\",\n prop: \"completed\",\n value: false,\n });\n expect(parsedQuery1.filter[1]).toStrictEqual({\n op: \"<=\",\n prop: \"dueDate\",\n value: \"{{today}}\",\n });\n\n let parsedQuery2 = parseQuery(`page where name =~ /interview\\\\/.*/\"`);\n expect(parsedQuery2.table).toBe(\"page\");\n expect(parsedQuery2.filter.length).toBe(1);\n expect(parsedQuery2.filter[0]).toStrictEqual({\n op: \"=~\",\n prop: \"name\",\n value: \"interview\\\\/.*\",\n });\n\n let parsedQuery3 = parseQuery(`page where something != null`);\n expect(parsedQuery3.table).toBe(\"page\");\n expect(parsedQuery3.filter.length).toBe(1);\n expect(parsedQuery3.filter[0]).toStrictEqual({\n op: \"!=\",\n prop: \"something\",\n value: null,\n });\n\n expect(parseQuery(`page select name`).select).toStrictEqual([\"name\"]);\n expect(parseQuery(`page select name, age`).select).toStrictEqual([\n \"name\",\n \"age\",\n ]);\n\n expect(\n parseQuery(`gh-events where type in [\"PushEvent\", \"somethingElse\"]`)\n ).toStrictEqual({\n table: \"gh-events\",\n filter: [\n {\n op: \"in\",\n prop: \"type\",\n value: [\"PushEvent\", \"somethingElse\"],\n },\n ],\n });\n\n expect(parseQuery(`something render \"template/table\"`)).toStrictEqual({\n table: \"something\",\n filter: [],\n render: \"template/table\",\n });\n});\n\ntest(\"Test performing the queries\", () => {\n let data: any[] = [\n { name: \"interview/My Interview\", lastModified: 1 },\n { name: \"interview/My Interview 2\", lastModified: 2 },\n { name: \"Pete\", age: 38 },\n { name: \"Angie\", age: 28 },\n ];\n\n expect(\n applyQuery(parseQuery(`page where name =~ /interview\\\\/.*/`), data)\n ).toStrictEqual([\n { name: \"interview/My Interview\", lastModified: 1 },\n { name: \"interview/My Interview 2\", lastModified: 2 },\n ]);\n expect(\n applyQuery(\n parseQuery(`page where name =~ /interview\\\\/.*/ order by lastModified`),\n data\n )\n ).toStrictEqual([\n { name: \"interview/My Interview\", lastModified: 1 },\n { name: \"interview/My Interview 2\", lastModified: 2 },\n ]);\n expect(\n applyQuery(\n parseQuery(\n `page where name =~ /interview\\\\/.*/ order by lastModified desc`\n ),\n data\n )\n ).toStrictEqual([\n { name: \"interview/My Interview 2\", lastModified: 2 },\n { name: \"interview/My Interview\", lastModified: 1 },\n ]);\n expect(applyQuery(parseQuery(`page where age > 30`), data)).toStrictEqual([\n { name: \"Pete\", age: 38 },\n ]);\n expect(\n applyQuery(parseQuery(`page where age > 28 and age < 38`), data)\n ).toStrictEqual([]);\n expect(\n applyQuery(parseQuery(`page where age > 30 select name`), data)\n ).toStrictEqual([{ name: \"Pete\" }]);\n\n expect(\n applyQuery(parseQuery(`page where name in [\"Pete\"] select name`), data)\n ).toStrictEqual([{ name: \"Pete\" }]);\n});\n","import {\n collectNodesOfType,\n findNodeOfType,\n ParseTree,\n replaceNodesMatching,\n} from \"@silverbulletmd/common/tree\";\nimport { lezerToParseTree } from \"@silverbulletmd/common/parse_tree\";\nimport Handlebars from \"handlebars\";\nimport YAML from \"yaml\";\n\n// @ts-ignore\nimport { parser } from \"./parse-query\";\nimport { readPage } from \"@silverbulletmd/plugos-silverbullet-syscall/space\";\nimport { niceDate } from \"../core/dates\";\n\nexport type QueryProviderEvent = {\n query: ParsedQuery;\n pageName: string;\n};\n\nexport type Filter = {\n op: string;\n prop: string;\n value: any;\n};\n\nexport type ParsedQuery = {\n table: string;\n orderBy?: string;\n orderDesc?: boolean;\n limit?: number;\n filter: Filter[];\n select?: string[];\n render?: string;\n};\n\nexport function parseQuery(query: string): ParsedQuery {\n let n = lezerToParseTree(query, parser.parse(query).topNode);\n // Clean the tree a bit\n replaceNodesMatching(n, (n) => {\n if (!n.type) {\n let trimmed = n.text!.trim();\n if (!trimmed) {\n return null;\n }\n n.text = trimmed;\n }\n });\n\n // console.log(\"Parsed\", JSON.stringify(n, null, 2));\n\n let queryNode = n.children![0];\n let parsedQuery: ParsedQuery = {\n table: queryNode.children![0].children![0].text!,\n filter: [],\n };\n let orderByNode = findNodeOfType(queryNode, \"OrderClause\");\n if (orderByNode) {\n let nameNode = findNodeOfType(orderByNode, \"Name\");\n parsedQuery.orderBy = nameNode!.children![0].text!;\n let orderNode = findNodeOfType(orderByNode, \"Order\");\n parsedQuery.orderDesc = orderNode\n ? orderNode.children![0].text! === \"desc\"\n : false;\n }\n let limitNode = findNodeOfType(queryNode, \"LimitClause\");\n if (limitNode) {\n let nameNode = findNodeOfType(limitNode, \"Number\");\n parsedQuery.limit = valueNodeToVal(nameNode!);\n }\n\n let filterNodes = collectNodesOfType(queryNode, \"FilterExpr\");\n for (let filterNode of filterNodes) {\n let val: any = undefined;\n let valNode = filterNode.children![2].children![0];\n val = valueNodeToVal(valNode);\n let f: Filter = {\n prop: filterNode.children![0].children![0].text!,\n op: filterNode.children![1].text!,\n value: val,\n };\n parsedQuery.filter.push(f);\n }\n let selectNode = findNodeOfType(queryNode, \"SelectClause\");\n if (selectNode) {\n // console.log(\"Select node\", JSON.stringify(selectNode));\n parsedQuery.select = [];\n collectNodesOfType(selectNode, \"Name\").forEach((t) => {\n parsedQuery.select!.push(t.children![0].text!);\n });\n // let nameNode = findNodeOfType(selectNode, \"Number\");\n // parsedQuery.limit = +nameNode!.children![0].text!;\n }\n\n let renderNode = findNodeOfType(queryNode, \"RenderClause\");\n if (renderNode) {\n let renderNameNode = findNodeOfType(renderNode, \"String\");\n parsedQuery.render = valueNodeToVal(renderNameNode!);\n }\n\n // console.log(JSON.stringify(queryNode, null, 2));\n return parsedQuery;\n}\n\nfunction valueNodeToVal(valNode: ParseTree): any {\n switch (valNode.type) {\n case \"Number\":\n return +valNode.children![0].text!;\n case \"Bool\":\n return valNode.children![0].text! === \"true\";\n case \"Null\":\n return null;\n case \"Name\":\n return valNode.children![0].text!;\n case \"Regex\":\n let val = valNode.children![0].text!;\n return val.substring(1, val.length - 1);\n case \"String\":\n let stringVal = valNode.children![0].text!;\n return stringVal.substring(1, stringVal.length - 1);\n case \"List\":\n return collectNodesOfType(valNode, \"Value\").map((t) =>\n valueNodeToVal(t.children![0])\n );\n }\n}\n\nexport function applyQuery(parsedQuery: ParsedQuery, records: T[]): T[] {\n let resultRecords: any[] = [];\n if (parsedQuery.filter.length === 0) {\n resultRecords = records.slice();\n } else {\n recordLoop: for (let record of records) {\n const recordAny: any = record;\n for (let { op, prop, value } of parsedQuery.filter) {\n switch (op) {\n case \"=\":\n if (!(recordAny[prop] == value)) {\n continue recordLoop;\n }\n break;\n case \"!=\":\n if (!(recordAny[prop] != value)) {\n continue recordLoop;\n }\n break;\n case \"<\":\n if (!(recordAny[prop] < value)) {\n continue recordLoop;\n }\n break;\n case \"<=\":\n if (!(recordAny[prop] <= value)) {\n continue recordLoop;\n }\n break;\n case \">\":\n if (!(recordAny[prop] > value)) {\n continue recordLoop;\n }\n break;\n case \">=\":\n if (!(recordAny[prop] >= value)) {\n continue recordLoop;\n }\n break;\n case \"=~\":\n // TODO: Cache regexps somehow\n if (!new RegExp(value).exec(recordAny[prop])) {\n continue recordLoop;\n }\n break;\n case \"!=~\":\n if (new RegExp(value).exec(recordAny[prop])) {\n continue recordLoop;\n }\n break;\n case \"in\":\n if (!value.includes(recordAny[prop])) {\n continue recordLoop;\n }\n break;\n }\n }\n resultRecords.push(recordAny);\n }\n }\n // Now the sorting\n if (parsedQuery.orderBy) {\n resultRecords = resultRecords.sort((a: any, b: any) => {\n const orderBy = parsedQuery.orderBy!;\n const orderDesc = parsedQuery.orderDesc!;\n if (a[orderBy] === b[orderBy]) {\n return 0;\n }\n\n if (a[orderBy] < b[orderBy]) {\n return orderDesc ? 1 : -1;\n } else {\n return orderDesc ? -1 : 1;\n }\n });\n }\n if (parsedQuery.limit) {\n resultRecords = resultRecords.slice(0, parsedQuery.limit);\n }\n if (parsedQuery.select) {\n resultRecords = resultRecords.map((rec) => {\n let newRec: any = {};\n for (let k of parsedQuery.select!) {\n newRec[k] = rec[k];\n }\n return newRec;\n });\n }\n return resultRecords;\n}\n\nexport async function renderQuery(\n parsedQuery: ParsedQuery,\n data: any[]\n): Promise {\n if (parsedQuery.render) {\n console.log(\"Handlebars\", Handlebars);\n Handlebars.registerHelper(\"json\", (v) => JSON.stringify(v));\n Handlebars.registerHelper(\"niceDate\", (ts) => niceDate(new Date(ts)));\n Handlebars.registerHelper(\"yaml\", (v, prefix) => {\n if (typeof prefix === \"string\") {\n let yaml = YAML.stringify(v)\n .split(\"\\n\")\n .join(\"\\n\" + prefix)\n .trim();\n if (Array.isArray(v)) {\n return \"\\n\" + prefix + yaml;\n } else {\n return yaml;\n }\n } else {\n return YAML.stringify(v).trim();\n }\n });\n let { text: templateText } = await readPage(parsedQuery.render);\n let template = Handlebars.compile(templateText, { noEscape: true });\n return template(data);\n }\n\n return \"ERROR\";\n}\n","export type ParseTree = {\n type?: string; // undefined === text node\n from?: number;\n to?: number;\n text?: string;\n children?: ParseTree[];\n // Only present after running addParentPointers\n parent?: ParseTree;\n};\n\nexport function addParentPointers(tree: ParseTree) {\n if (!tree.children) {\n return;\n }\n for (let child of tree.children) {\n if (child.parent) {\n // Already added parent pointers before\n return;\n }\n child.parent = tree;\n addParentPointers(child);\n }\n}\n\nexport function removeParentPointers(tree: ParseTree) {\n delete tree.parent;\n if (!tree.children) {\n return;\n }\n for (let child of tree.children) {\n removeParentPointers(child);\n }\n}\n\nexport function findParentMatching(\n tree: ParseTree,\n matchFn: (tree: ParseTree) => boolean\n): ParseTree | null {\n let node = tree.parent;\n while (node) {\n if (matchFn(node)) {\n return node;\n }\n node = node.parent;\n }\n return null;\n}\n\nexport function collectNodesOfType(\n tree: ParseTree,\n nodeType: string\n): ParseTree[] {\n return collectNodesMatching(tree, (n) => n.type === nodeType);\n}\n\nexport function collectNodesMatching(\n tree: ParseTree,\n matchFn: (tree: ParseTree) => boolean\n): ParseTree[] {\n if (matchFn(tree)) {\n return [tree];\n }\n let results: ParseTree[] = [];\n if (tree.children) {\n for (let child of tree.children) {\n results = [...results, ...collectNodesMatching(child, matchFn)];\n }\n }\n return results;\n}\n\n// return value: returning undefined = not matched, continue, null = delete, new node = replace\nexport function replaceNodesMatching(\n tree: ParseTree,\n substituteFn: (tree: ParseTree) => ParseTree | null | undefined\n) {\n if (tree.children) {\n let children = tree.children.slice();\n for (let child of children) {\n let subst = substituteFn(child);\n if (subst !== undefined) {\n let pos = tree.children.indexOf(child);\n if (subst) {\n tree.children.splice(pos, 1, subst);\n } else {\n // null = delete\n tree.children.splice(pos, 1);\n }\n } else {\n replaceNodesMatching(child, substituteFn);\n }\n }\n }\n}\n\nexport function findNodeMatching(\n tree: ParseTree,\n matchFn: (tree: ParseTree) => boolean\n): ParseTree | null {\n return collectNodesMatching(tree, matchFn)[0];\n}\n\nexport function findNodeOfType(\n tree: ParseTree,\n nodeType: string\n): ParseTree | null {\n return collectNodesMatching(tree, (n) => n.type === nodeType)[0];\n}\n\n// Finds non-text node at position\nexport function nodeAtPos(tree: ParseTree, pos: number): ParseTree | null {\n if (pos < tree.from! || pos > tree.to!) {\n return null;\n }\n if (!tree.children) {\n return tree;\n }\n for (let child of tree.children) {\n let n = nodeAtPos(child, pos);\n if (n && n.text !== undefined) {\n // Got a text node, let's return its parent\n return tree;\n } else if (n) {\n // Got it\n return n;\n }\n }\n return null;\n}\n\n// Turn ParseTree back into text\nexport function renderToText(tree: ParseTree): string {\n let pieces: string[] = [];\n if (tree.text !== undefined) {\n return tree.text;\n }\n for (let child of tree.children!) {\n pieces.push(renderToText(child));\n }\n return pieces.join(\"\");\n}\n","import type { SyntaxNode } from \"@lezer/common\";\nimport type { Language } from \"@codemirror/language\";\nimport { ParseTree } from \"./tree\";\n\nexport function lezerToParseTree(\n text: string,\n n: SyntaxNode,\n offset = 0\n): ParseTree {\n let children: ParseTree[] = [];\n let nodeText: string | undefined;\n let child = n.firstChild;\n while (child) {\n children.push(lezerToParseTree(text, child));\n child = child.nextSibling;\n }\n\n if (children.length === 0) {\n children = [\n {\n from: n.from + offset,\n to: n.to + offset,\n text: text.substring(n.from, n.to),\n },\n ];\n } else {\n let newChildren: ParseTree[] = [];\n let index = n.from;\n for (let child of children) {\n let s = text.substring(index, child.from);\n if (s) {\n newChildren.push({\n from: index + offset,\n to: child.from! + offset,\n text: s,\n });\n }\n newChildren.push(child);\n index = child.to!;\n }\n let s = text.substring(index, n.to);\n if (s) {\n newChildren.push({ from: index + offset, to: n.to + offset, text: s });\n }\n children = newChildren;\n }\n\n let result: ParseTree = {\n type: n.name,\n from: n.from + offset,\n to: n.to + offset,\n };\n if (children.length > 0) {\n result.children = children;\n }\n if (nodeText) {\n result.text = nodeText;\n }\n return result;\n}\n\nexport function parse(language: Language, text: string): ParseTree {\n let tree = lezerToParseTree(text, language.parser.parse(text).topNode);\n // replaceNodesMatching(tree, (n): MarkdownTree | undefined | null => {\n // if (n.type === \"FencedCode\") {\n // let infoN = findNodeMatching(n, (n) => n.type === \"CodeInfo\");\n // let language = infoN!.children![0].text;\n // let textN = findNodeMatching(n, (n) => n.type === \"CodeText\");\n // let text = textN!.children![0].text!;\n //\n // console.log(language, text);\n // switch (language) {\n // case \"yaml\":\n // let parsed = StreamLanguage.define(yaml).parser.parse(text);\n // let subTree = treeToAST(text, parsed.topNode, n.from);\n // // console.log(JSON.stringify(subTree, null, 2));\n // subTree.type = \"yaml\";\n // return subTree;\n // }\n // }\n // return;\n // });\n return tree;\n}\n","// This file was generated by lezer-generator. You probably shouldn't edit it.\nimport {LRParser} from \"@lezer/lr\"\nexport const parser = LRParser.deserialize({\n version: 13,\n states: \"&fOVQPOOOmQQO'#C^QOQPOOOtQPO'#C`OyQQO'#CkO!OQPO'#CmO!TQPO'#CnO!YQPO'#CoOOQO'#Cp'#CpO!_QQO,58xO!fQQO'#CcO#TQQO'#CaOOQO'#Ca'#CaOOQO,58z,58zO#lQPO,59VOOQO,59X,59XO#qQQO'#D`OOQO,59Y,59YOOQO,59Z,59ZOOQO-E6n-E6nO$YQQO,58}OtQPO,58|O$qQQO1G.qO%]QPO'#CrO%bQQO,59zOOQO'#Cg'#CgOOQO'#Ci'#CiO$YQQO'#CjOOQO'#Cd'#CdOOQO1G.i1G.iOOQO1G.h1G.hOOQO'#Cl'#ClOOQO7+$]7+$]OOQO,59^,59^OOQO-E6p-E6pO%yQPO'#C|O&RQPO,59UO$YQQO'#CqO&WQPO,59hOOQO1G.p1G.pOOQO,59],59]OOQO-E6o-E6o\",\n stateData: \"&`~OiOS~ORPO~OjRO|SO!QTO!RUO!TVO~OgQX~P[ORYO~O}^O~OX_O~OR`O~OYbO~OgQa~P[OkdOsdOtdOudOvdOwdOxdOydOzdO~O{eOgTXjTX|TX!QTX!RTX!TTX~ORfO~OqgOg!SXj!SX|!SX!Q!SX!R!SX!T!SX~OXlOYlO[lOliOmiOnjOokO~O!OoO!PoOg_ij_i|_i!Q_i!R_i!T_i~ORqO~OqgOg!Saj!Sa|!Sa!Q!Sa!R!Sa!T!Sa~OquOrpX~OrwO~OquOrpa~O\",\n goto: \"#d!TPP!UP!X!]!`!c!iPP!rP!r!r!X!w!X!X!X!z#Q#WPPPPPPPPP#^PPPPPPPPPPPPPPPPP#aRQOTWPXR]RR[RQZRRneQmdQskRxuVldkuRpfQXPRcXQvsRyvQh`RrhRtkRaU\",\n nodeNames: \"⚠ Program Query Name WhereClause LogicalExpr AndExpr FilterExpr Value Number String Bool Regex Null List OrderClause Order LimitClause SelectClause RenderClause\",\n maxTerm: 51,\n skippedNodes: [0],\n repeatNodeCount: 3,\n tokenData: \"Ap~R}X^$Opq$Oqr$srs%W|}%r}!O%w!P!Q&Y!Q!['P!^!_'X!_!`'f!`!a's!c!}%w!}#O(Q#P#Q(V#R#S%w#T#U([#U#V*q#V#W%w#W#X+m#X#Y%w#Y#Z-i#Z#]%w#]#^/y#^#`%w#`#a0u#a#b%w#b#c3Y#c#d5U#d#f%w#f#g7i#g#h:e#h#i=a#i#k%w#k#l?]#l#o%w#y#z$O$f$g$O#BY#BZ$O$IS$I_$O$Ip$Iq%W$Iq$Ir%W$I|$JO$O$JT$JU$O$KV$KW$O&FU&FV$O~$TYi~X^$Opq$O#y#z$O$f$g$O#BY#BZ$O$IS$I_$O$I|$JO$O$JT$JU$O$KV$KW$O&FU&FV$O~$vP!_!`$y~%OPu~#r#s%R~%WOy~~%ZUOr%Wrs%ms$Ip%W$Ip$Iq%m$Iq$Ir%m$Ir~%W~%rOY~~%wOq~P%|SRP}!O%w!c!}%w#R#S%w#T#o%w~&_V[~OY&YZ]&Y^!P&Y!P!Q&t!Q#O&Y#O#P&y#P~&Y~&yO[~~&|PO~&Y~'UPX~!Q!['P~'^Pk~!_!`'a~'fOs~~'kPt~#r#s'n~'sOx~~'xPw~!_!`'{~(QOv~~(VOo~~([Or~R(aWRP}!O%w!c!}%w#R#S%w#T#b%w#b#c(y#c#g%w#g#h)u#h#o%wR)OURP}!O%w!c!}%w#R#S%w#T#W%w#W#X)b#X#o%wR)iS{QRP}!O%w!c!}%w#R#S%w#T#o%wR)zURP}!O%w!c!}%w#R#S%w#T#V%w#V#W*^#W#o%wR*eS!PQRP}!O%w!c!}%w#R#S%w#T#o%wR*vURP}!O%w!c!}%w#R#S%w#T#m%w#m#n+Y#n#o%wR+aS}QRP}!O%w!c!}%w#R#S%w#T#o%wR+rURP}!O%w!c!}%w#R#S%w#T#X%w#X#Y,U#Y#o%wR,ZURP}!O%w!c!}%w#R#S%w#T#g%w#g#h,m#h#o%wR,rURP}!O%w!c!}%w#R#S%w#T#V%w#V#W-U#W#o%wR-]S!OQRP}!O%w!c!}%w#R#S%w#T#o%wR-nTRP}!O%w!c!}%w#R#S%w#T#U-}#U#o%wR.SURP}!O%w!c!}%w#R#S%w#T#`%w#`#a.f#a#o%wR.kURP}!O%w!c!}%w#R#S%w#T#g%w#g#h.}#h#o%wR/SURP}!O%w!c!}%w#R#S%w#T#X%w#X#Y/f#Y#o%wR/mSmQRP}!O%w!c!}%w#R#S%w#T#o%wR0OURP}!O%w!c!}%w#R#S%w#T#b%w#b#c0b#c#o%wR0iSzQRP}!O%w!c!}%w#R#S%w#T#o%wR0zURP}!O%w!c!}%w#R#S%w#T#]%w#]#^1^#^#o%wR1cURP}!O%w!c!}%w#R#S%w#T#a%w#a#b1u#b#o%wR1zURP}!O%w!c!}%w#R#S%w#T#]%w#]#^2^#^#o%wR2cURP}!O%w!c!}%w#R#S%w#T#h%w#h#i2u#i#o%wR2|S!QQRP}!O%w!c!}%w#R#S%w#T#o%wR3_URP}!O%w!c!}%w#R#S%w#T#i%w#i#j3q#j#o%wR3vURP}!O%w!c!}%w#R#S%w#T#`%w#`#a4Y#a#o%wR4_URP}!O%w!c!}%w#R#S%w#T#`%w#`#a4q#a#o%wR4xSnQRP}!O%w!c!}%w#R#S%w#T#o%wR5ZURP}!O%w!c!}%w#R#S%w#T#f%w#f#g5m#g#o%wR5rURP}!O%w!c!}%w#R#S%w#T#W%w#W#X6U#X#o%wR6ZURP}!O%w!c!}%w#R#S%w#T#X%w#X#Y6m#Y#o%wR6rURP}!O%w!c!}%w#R#S%w#T#f%w#f#g7U#g#o%wR7]S|QRP}!O%w!c!}%w#R#S%w#T#o%wR7nURP}!O%w!c!}%w#R#S%w#T#X%w#X#Y8Q#Y#o%wR8VURP}!O%w!c!}%w#R#S%w#T#b%w#b#c8i#c#o%wR8nURP}!O%w!c!}%w#R#S%w#T#W%w#W#X9Q#X#o%wR9VURP}!O%w!c!}%w#R#S%w#T#X%w#X#Y9i#Y#o%wR9nURP}!O%w!c!}%w#R#S%w#T#f%w#f#g:Q#g#o%wR:XS!TQRP}!O%w!c!}%w#R#S%w#T#o%wR:jURP}!O%w!c!}%w#R#S%w#T#X%w#X#Y:|#Y#o%wR;RURP}!O%w!c!}%w#R#S%w#T#`%w#`#a;e#a#o%wR;jURP}!O%w!c!}%w#R#S%w#T#X%w#X#Y;|#Y#o%wRa#j#o%wR>fURP}!O%w!c!}%w#R#S%w#T#X%w#X#Y>x#Y#o%wR?PSlQRP}!O%w!c!}%w#R#S%w#T#o%wR?bURP}!O%w!c!}%w#R#S%w#T#[%w#[#]?t#]#o%wR?yURP}!O%w!c!}%w#R#S%w#T#X%w#X#Y@]#Y#o%wR@bURP}!O%w!c!}%w#R#S%w#T#f%w#f#g@t#g#o%wR@yURP}!O%w!c!}%w#R#S%w#T#X%w#X#YA]#Y#o%wRAdSjQRP}!O%w!c!}%w#R#S%w#T#o%w\",\n tokenizers: [0, 1],\n topRules: {\"Program\":[0,1]},\n tokenPrec: 0\n})\n","import { syscall } from \"./syscall\";\nimport { PageMeta } from \"../common/types\";\n\nexport async function listPages(unfiltered = false): Promise {\n return syscall(\"space.listPages\", unfiltered);\n}\n\nexport async function readPage(\n name: string\n): Promise<{ text: string; meta: PageMeta }> {\n return syscall(\"space.readPage\", name);\n}\n\nexport async function writePage(name: string, text: string): Promise {\n return syscall(\"space.writePage\", name, text);\n}\n\nexport async function deletePage(name: string): Promise {\n return syscall(\"space.deletePage\", name);\n}\n","declare global {\n function syscall(name: string, ...args: any[]): Promise;\n}\n\nexport const syscall = self.syscall;\n","import { insertAtCursor } from \"@silverbulletmd/plugos-silverbullet-syscall/editor\";\n\nconst dateMatchRegex = /(\\d{4}\\-\\d{2}\\-\\d{2})/g;\n\nexport function niceDate(d: Date): string {\n return d.toISOString().split(\"T\")[0];\n}\n\nexport async function insertToday() {\n await insertAtCursor(niceDate(new Date()));\n}\n\nexport async function insertTomorrow() {\n let d = new Date();\n d.setDate(d.getDate() + 1);\n await insertAtCursor(niceDate(d));\n}\n","import { syscall } from \"./syscall\";\nimport { FilterOption } from \"../common/types\";\n\nexport function getCurrentPage(): Promise {\n return syscall(\"editor.getCurrentPage\");\n}\n\nexport function setPage(newName: string): Promise {\n return syscall(\"editor.setPage\", newName);\n}\n\nexport function getText(): Promise {\n return syscall(\"editor.getText\");\n}\n\nexport function getCursor(): Promise {\n return syscall(\"editor.getCursor\");\n}\n\nexport function save(): Promise {\n return syscall(\"editor.save\");\n}\n\nexport function navigate(name: string, pos?: number): Promise {\n return syscall(\"editor.navigate\", name, pos);\n}\n\nexport function reloadPage(): Promise {\n return syscall(\"editor.reloadPage\");\n}\n\nexport function openUrl(url: string): Promise {\n return syscall(\"editor.openUrl\", url);\n}\n\nexport function flashNotification(message: string): Promise {\n return syscall(\"editor.flashNotification\", message);\n}\n\nexport function filterBox(\n label: string,\n options: FilterOption[],\n helpText: string = \"\",\n placeHolder: string = \"\"\n): Promise {\n return syscall(\"editor.filterBox\", label, options, helpText, placeHolder);\n}\n\nexport function showRhs(\n html: string,\n script?: string,\n flex = 1\n): Promise {\n return syscall(\"editor.showRhs\", html, script, flex);\n}\n\nexport function hideRhs(): Promise {\n return syscall(\"editor.hideRhs\");\n}\n\nexport function showLhs(\n html: string,\n script?: string,\n flex = 1\n): Promise {\n return syscall(\"editor.showLhs\", html, script, flex);\n}\n\nexport function hideLhs(): Promise {\n return syscall(\"editor.hideLhs\");\n}\n\nexport function showBhs(\n html: string,\n script?: string,\n flex = 1\n): Promise {\n return syscall(\"editor.showBhs\", html, script, flex);\n}\n\nexport function hideBhs(): Promise {\n return syscall(\"editor.hideBhs\");\n}\n\nexport function insertAtPos(text: string, pos: number): Promise {\n return syscall(\"editor.insertAtPos\", text, pos);\n}\n\nexport function replaceRange(\n from: number,\n to: number,\n text: string\n): Promise {\n return syscall(\"editor.replaceRange\", from, to, text);\n}\n\nexport function moveCursor(pos: number): Promise {\n return syscall(\"editor.moveCursor\", pos);\n}\n\nexport function insertAtCursor(text: string): Promise {\n return syscall(\"editor.insertAtCursor\", text);\n}\n\nexport function matchBefore(\n re: string\n): Promise<{ from: number; to: number; text: string } | null> {\n return syscall(\"editor.matchBefore\", re);\n}\n\nexport function dispatch(change: any): Promise {\n return syscall(\"editor.dispatch\", change);\n}\n\nexport function prompt(\n message: string,\n defaultValue = \"\"\n): Promise {\n return syscall(\"editor.prompt\", message, defaultValue);\n}\n"],"names":[],"version":3,"file":"engine.test.js.map","sourceRoot":"../../../../"} \ No newline at end of file diff --git a/packages/plugs/core/core.plug.yaml b/packages/plugs/core/core.plug.yaml index 91c1887b..eb57442e 100644 --- a/packages/plugs/core/core.plug.yaml +++ b/packages/plugs/core/core.plug.yaml @@ -19,6 +19,7 @@ syntax: styles: color: "#0330cb" textDecoration: underline + cursor: pointer functions: clearPageIndex: path: "./page.ts:clearPageIndex" diff --git a/packages/plugs/core/navigate.ts b/packages/plugs/core/navigate.ts index 3876888c..a35dcece 100644 --- a/packages/plugs/core/navigate.ts +++ b/packages/plugs/core/navigate.ts @@ -14,7 +14,7 @@ async function actionClickOrActionEnter(mdTree: ParseTree | null) { if (!mdTree) { return; } - console.log("Attempting to navigate based on syntax node", mdTree); + // console.log("Attempting to navigate based on syntax node", mdTree); switch (mdTree.type) { case "WikiLinkPage": let pageLink = mdTree.children![0].text!; diff --git a/packages/plugs/global.plug.yaml b/packages/plugs/global.plug.yaml new file mode 100644 index 00000000..0c6ac0b0 --- /dev/null +++ b/packages/plugs/global.plug.yaml @@ -0,0 +1,5 @@ +name: global +dependencies: + yaml: "yaml@2" + handlebars: "handlebars@4.7.7:/dist/handlebars" + "@lezer/lr": "@lezer/lr@0.15.4" diff --git a/packages/plugs/package.json b/packages/plugs/package.json index fb54b12b..27e2ec35 100644 --- a/packages/plugs/package.json +++ b/packages/plugs/package.json @@ -8,8 +8,8 @@ "license": "MIT", "scripts": { "generate": "lezer-generator query/query.grammar -o query/parse-query.js", - "watch": "plugos-bundle -w --dist dist --exclude @lezer/lr yaml handlebars -- */*.plug.yaml", - "build": "plugos-bundle --dist dist --exclude @lezer/lr yaml handlebars -- */*.plug.yaml", + "watch": "plugos-bundle --dist ../common/dist global.plug.yaml && plugos-bundle -w --dist dist --exclude @lezer/lr yaml handlebars -- */*.plug.yaml", + "build": "plugos-bundle --dist ../common/dist global.plug.yaml && plugos-bundle --dist dist --exclude @lezer/lr yaml handlebars -- */*.plug.yaml", "test": "jest build/test" }, "files": [ diff --git a/packages/plugs/plugmanager/plugmanager.ts b/packages/plugs/plugmanager/plugmanager.ts index acd5f471..00fcd9da 100644 --- a/packages/plugs/plugmanager/plugmanager.ts +++ b/packages/plugs/plugmanager/plugmanager.ts @@ -36,6 +36,7 @@ export async function compileCommand() { ); console.log("Wrote this plug", manifest); await hideBhs(); + await reloadPlugs(); } catch (e: any) { await showBhs(e.message); @@ -136,6 +137,7 @@ export async function updatePlugs() { return; } let plugYaml = codeTextNode.children![0].text; + console.log("YAML", YAML); let plugList = YAML.parse(plugYaml!); console.log("Plug YAML", plugList); let allPlugNames: string[] = []; diff --git a/packages/plugs/query/engine.ts b/packages/plugs/query/engine.ts index 2024cf7b..7c0adbec 100644 --- a/packages/plugs/query/engine.ts +++ b/packages/plugs/query/engine.ts @@ -221,6 +221,7 @@ export async function renderQuery( data: any[] ): Promise { if (parsedQuery.render) { + console.log("Handlebars", Handlebars); Handlebars.registerHelper("json", (v) => JSON.stringify(v)); Handlebars.registerHelper("niceDate", (ts) => niceDate(new Date(ts))); Handlebars.registerHelper("yaml", (v, prefix) => { diff --git a/packages/plugs/query/query.plug.yaml b/packages/plugs/query/query.plug.yaml index c9cf669f..6d5f55fd 100644 --- a/packages/plugs/query/query.plug.yaml +++ b/packages/plugs/query/query.plug.yaml @@ -1,4 +1,7 @@ name: query +# dependencies: +# yaml: "yaml@2" +# "@lezer/lr": "@lezer/lr@0.15.4" functions: updateMaterializedQueriesOnPage: path: ./materialized_queries.ts:updateMaterializedQueriesOnPage diff --git a/packages/server/express_server.ts b/packages/server/express_server.ts index 83ae9ed3..bfbb3302 100644 --- a/packages/server/express_server.ts +++ b/packages/server/express_server.ts @@ -27,9 +27,12 @@ import { systemSyscalls } from "./syscalls/system"; import { plugPrefix } from "@silverbulletmd/common/spaces/constants"; import { Authenticator } from "./auth"; -import { nextTick } from "process"; import sandboxSyscalls from "@plugos/plugos/syscalls/sandbox"; +import globalModules from "../common/dist/global.plug.json"; + +import { safeRun } from "./util"; + const safeFilename = /^[a-zA-Z0-9_\-\.]+$/; export type ServerOptions = { @@ -37,7 +40,6 @@ export type ServerOptions = { pagesPath: string; distDir: string; builtinPlugDir: string; - preloadedModules: string[]; token?: string; }; export class ExpressServer { @@ -50,7 +52,6 @@ export class ExpressServer { private port: number; private server?: Server; builtinPlugDir: string; - preloadedModules: string[]; token?: string; constructor(options: ServerOptions) { @@ -59,7 +60,6 @@ export class ExpressServer { this.builtinPlugDir = options.builtinPlugDir; this.distDir = options.distDir; this.system = new System("server"); - this.preloadedModules = options.preloadedModules; this.token = options.token; // Setup system @@ -96,6 +96,18 @@ export class ExpressServer { ); this.system.addHook(new EndpointHook(this.app, "/_")); + this.system.on({ + plugLoaded: (plug) => { + safeRun(async () => { + for (let [modName, code] of Object.entries( + globalModules.dependencies + )) { + await plug.sandbox.loadDependency(modName, code); + } + }); + }, + }); + this.eventHook.addLocalListener( "get-plug:builtin", async (plugName: string): Promise => { @@ -165,9 +177,7 @@ export class ExpressServer { console.log("Reloading plugs"); for (let pageInfo of allPlugs) { let { text } = await this.space.readPage(pageInfo.name); - await this.system.load(JSON.parse(text), (p) => - createSandbox(p, this.preloadedModules) - ); + await this.system.load(JSON.parse(text), createSandbox); } this.rebuildMdExtensions(); } diff --git a/packages/server/server.ts b/packages/server/server.ts index 52e9a716..ccfcd72a 100755 --- a/packages/server/server.ts +++ b/packages/server/server.ts @@ -1,6 +1,5 @@ #!/usr/bin/env -S node --enable-source-maps import { nodeModulesDir } from "@plugos/plugos/environments/node_sandbox"; -import { preloadModules } from "@silverbulletmd/common/preload_modules"; import { realpathSync } from "fs"; import yargs from "yargs"; import { hideBin } from "yargs/helpers"; @@ -39,7 +38,6 @@ console.log("Builtin plug dist dir", plugDistDir); const expressServer = new ExpressServer({ port: port, pagesPath: pagesPath, - preloadedModules: preloadModules, distDir: webappDistDir, builtinPlugDir: plugDistDir, token: args.token, diff --git a/packages/web/components/panel_page.ts b/packages/web/components/panel_page.ts index 2fe1f1e9..6da3b3cb 100644 --- a/packages/web/components/panel_page.ts +++ b/packages/web/components/panel_page.ts @@ -1,8 +1,3 @@ -declare global { - function syscall(name: string, ...args: any[]): Promise; - // function require(moduleName: string): any; -} - window.addEventListener("message", (message) => { const data = message.data; switch (data.type) { diff --git a/packages/web/editor.tsx b/packages/web/editor.tsx index cec57bb0..16263aeb 100644 --- a/packages/web/editor.tsx +++ b/packages/web/editor.tsx @@ -59,6 +59,7 @@ import { FilterList } from "./components/filter"; import { FilterOption } from "@silverbulletmd/common/types"; import { syntaxTree } from "@codemirror/language"; import sandboxSyscalls from "@plugos/plugos/syscalls/sandbox"; +import globalModules from "../common/dist/global.plug.json"; class PageState { constructor( @@ -131,6 +132,18 @@ export class Editor { clientStoreSyscalls(), sandboxSyscalls(this.system) ); + + this.system.on({ + plugLoaded: (plug) => { + safeRun(async () => { + for (let [modName, code] of Object.entries( + globalModules.dependencies + )) { + await plug.sandbox.loadDependency(modName, code); + } + }); + }, + }); } get currentPage(): string | undefined { diff --git a/packages/web/reducer.ts b/packages/web/reducer.ts index 692dbd31..8b46a0fa 100644 --- a/packages/web/reducer.ts +++ b/packages/web/reducer.ts @@ -1,5 +1,8 @@ import { Action, AppViewState } from "./types"; +let m = new Map(); +m.size; + export default function reducer( state: AppViewState, action: Action diff --git a/packages/web/styles/editor.scss b/packages/web/styles/editor.scss index 496d9067..d344ede8 100644 --- a/packages/web/styles/editor.scss +++ b/packages/web/styles/editor.scss @@ -171,6 +171,7 @@ cursor: pointer; } .wiki-link { + cursor: pointer; color: #a8abbd; }