silverbullet/web/client_system.ts

212 lines
6.9 KiB
TypeScript
Raw Normal View History

2023-07-14 22:48:35 +08:00
import { PlugNamespaceHook } from "../common/hooks/plug_namespace.ts";
2023-07-14 19:44:30 +08:00
import { Manifest, SilverBulletHooks } from "../common/manifest.ts";
import buildMarkdown from "../common/markdown_parser/parser.ts";
import { CronHook } from "../plugos/hooks/cron.ts";
import { EventHook } from "../plugos/hooks/event.ts";
import { DexieKVStore } from "../plugos/lib/kv_store.dexie.ts";
import { createSandbox } from "../plugos/environments/webworker_sandbox.ts";
import assetSyscalls from "../plugos/syscalls/asset.ts";
import { eventSyscalls } from "../plugos/syscalls/event.ts";
2023-08-05 00:56:55 +08:00
import { storeSyscalls } from "../plugos/syscalls/store.ts";
2023-07-14 19:44:30 +08:00
import { SysCallMapping, System } from "../plugos/system.ts";
2023-07-14 22:56:20 +08:00
import type { Client } from "./client.ts";
2023-07-14 19:44:30 +08:00
import { CodeWidgetHook } from "./hooks/code_widget.ts";
import { CommandHook } from "./hooks/command.ts";
import { SlashCommandHook } from "./hooks/slash_command.ts";
import { clientStoreSyscalls } from "./syscalls/clientStore.ts";
import { debugSyscalls } from "./syscalls/debug.ts";
import { editorSyscalls } from "./syscalls/editor.ts";
import { sandboxFetchSyscalls } from "./syscalls/fetch.ts";
import { pageIndexSyscalls } from "./syscalls/index.ts";
import { markdownSyscalls } from "./syscalls/markdown.ts";
import { shellSyscalls } from "./syscalls/shell.ts";
import { spaceSyscalls } from "./syscalls/space.ts";
import { syncSyscalls } from "./syscalls/sync.ts";
import { systemSyscalls } from "./syscalls/system.ts";
import { yamlSyscalls } from "./syscalls/yaml.ts";
import { Space } from "./space.ts";
import {
loadMarkdownExtensions,
MDExt,
} from "../common/markdown_parser/markdown_ext.ts";
2023-08-11 00:32:41 +08:00
import { DexieMQ } from "../plugos/lib/mq.dexie.ts";
import { MQHook } from "../plugos/hooks/mq.ts";
import { mqSyscalls } from "../plugos/syscalls/mq.dexie.ts";
2023-08-26 14:31:51 +08:00
import { indexProxySyscalls } from "./syscalls/index.proxy.ts";
import { storeProxySyscalls } from "./syscalls/store.proxy.ts";
2023-08-28 23:12:15 +08:00
import { mqProxySyscalls } from "./syscalls/mq.proxy.ts";
2023-07-14 19:44:30 +08:00
export class ClientSystem {
commandHook: CommandHook;
slashCommandHook: SlashCommandHook;
2023-07-14 22:48:35 +08:00
namespaceHook: PlugNamespaceHook;
2023-07-14 19:44:30 +08:00
indexSyscalls: SysCallMapping;
codeWidgetHook: CodeWidgetHook;
plugsUpdated = false;
mdExtensions: MDExt[] = [];
2023-08-27 20:13:18 +08:00
system: System<SilverBulletHooks>;
2023-07-14 19:44:30 +08:00
constructor(
2023-08-26 14:31:51 +08:00
private client: Client,
2023-07-14 19:44:30 +08:00
private kvStore: DexieKVStore,
2023-08-11 00:32:41 +08:00
private mq: DexieMQ,
2023-08-27 20:13:18 +08:00
dbPrefix: string,
2023-07-14 19:44:30 +08:00
private eventHook: EventHook,
2023-08-26 14:31:51 +08:00
private thinClientMode: boolean,
2023-07-14 19:44:30 +08:00
) {
2023-08-27 20:13:18 +08:00
// Only set environment to "client" when running in thin client mode, otherwise we run everything locally (hybrid)
this.system = new System(thinClientMode ? "client" : undefined);
2023-07-14 19:44:30 +08:00
this.system.addHook(this.eventHook);
2023-07-14 22:48:35 +08:00
// Plug page namespace hook
this.namespaceHook = new PlugNamespaceHook();
this.system.addHook(this.namespaceHook);
2023-07-14 19:44:30 +08:00
// Cron hook
const cronHook = new CronHook(this.system);
this.system.addHook(cronHook);
2023-08-26 14:31:51 +08:00
if (thinClientMode) {
this.indexSyscalls = indexProxySyscalls(client);
} else {
this.indexSyscalls = pageIndexSyscalls(
`${dbPrefix}_page_index`,
globalThis.indexedDB,
globalThis.IDBKeyRange,
);
}
2023-07-14 19:44:30 +08:00
// Code widget hook
this.codeWidgetHook = new CodeWidgetHook();
this.system.addHook(this.codeWidgetHook);
2023-08-11 00:32:41 +08:00
// MQ hook
2023-08-28 23:12:15 +08:00
if (!this.thinClientMode) {
this.system.addHook(new MQHook(this.system, this.mq));
}
2023-08-11 00:32:41 +08:00
2023-07-14 19:44:30 +08:00
// Command hook
this.commandHook = new CommandHook();
this.commandHook.on({
commandsUpdated: (commandMap) => {
2023-08-26 14:31:51 +08:00
this.client.ui.viewDispatch({
2023-07-14 19:44:30 +08:00
type: "update-commands",
commands: commandMap,
});
},
});
this.system.addHook(this.commandHook);
// Slash command hook
2023-08-26 14:31:51 +08:00
this.slashCommandHook = new SlashCommandHook(this.client);
2023-07-14 19:44:30 +08:00
this.system.addHook(this.slashCommandHook);
this.eventHook.addLocalListener("plug:changed", async (fileName) => {
console.log("Plug updated, reloading:", fileName);
this.system.unload(fileName);
const plug = await this.system.load(
new URL(`/${fileName}`, location.href),
createSandbox,
2023-08-26 14:31:51 +08:00
this.client.settings.plugOverrides,
2023-07-14 19:44:30 +08:00
);
if ((plug.manifest! as Manifest).syntax) {
// If there are syntax extensions, rebuild the markdown parser immediately
this.updateMarkdownParser();
}
this.plugsUpdated = true;
});
2023-08-27 17:02:24 +08:00
// Debugging
// this.eventHook.addLocalListener("file:listed", (files) => {
// console.log("New file list", files);
// });
2023-08-28 23:12:15 +08:00
// this.eventHook.addLocalListener("file:changed", (file) => {
// console.log("File changed", file);
// });
2023-08-27 17:02:24 +08:00
2023-08-28 23:12:15 +08:00
// this.eventHook.addLocalListener("file:created", (file) => {
// console.log("File created", file);
// });
2023-08-27 17:02:24 +08:00
2023-08-28 23:12:15 +08:00
// this.eventHook.addLocalListener("file:deleted", (file) => {
// console.log("File deleted", file);
// });
2023-08-27 17:02:24 +08:00
2023-07-14 19:44:30 +08:00
this.registerSyscalls();
}
registerSyscalls() {
2023-08-26 14:31:51 +08:00
const storeCalls = this.thinClientMode
? storeProxySyscalls(this.client)
: storeSyscalls(this.kvStore);
2023-07-14 19:44:30 +08:00
// Slash command hook
2023-08-26 14:31:51 +08:00
this.slashCommandHook = new SlashCommandHook(this.client);
2023-07-14 19:44:30 +08:00
this.system.addHook(this.slashCommandHook);
// Syscalls available to all plugs
this.system.registerSyscalls(
[],
eventSyscalls(this.eventHook),
2023-08-26 14:31:51 +08:00
editorSyscalls(this.client),
spaceSyscalls(this.client),
systemSyscalls(this.client, this.system),
2023-07-14 19:44:30 +08:00
markdownSyscalls(buildMarkdown(this.mdExtensions)),
assetSyscalls(this.system),
yamlSyscalls(),
2023-08-28 23:12:15 +08:00
this.thinClientMode ? mqProxySyscalls(this.client) : mqSyscalls(this.mq),
2023-07-14 19:44:30 +08:00
storeCalls,
this.indexSyscalls,
debugSyscalls(),
2023-08-26 14:31:51 +08:00
syncSyscalls(this.client),
clientStoreSyscalls(this.kvStore),
2023-07-14 19:44:30 +08:00
);
// Syscalls that require some additional permissions
this.system.registerSyscalls(
["fetch"],
2023-08-26 14:31:51 +08:00
sandboxFetchSyscalls(this.client),
2023-07-14 19:44:30 +08:00
);
this.system.registerSyscalls(
["shell"],
2023-08-26 14:31:51 +08:00
shellSyscalls(this.client),
2023-07-14 19:44:30 +08:00
);
}
async reloadPlugsFromSpace(space: Space) {
console.log("Loading plugs");
await space.updatePageList();
await this.system.unloadAll();
console.log("(Re)loading plugs");
await Promise.all((await space.listPlugs()).map(async (plugName) => {
try {
await this.system.load(
new URL(plugName, location.origin),
createSandbox,
2023-08-26 14:31:51 +08:00
this.client.settings.plugOverrides,
2023-07-14 19:44:30 +08:00
);
} catch (e: any) {
console.error("Could not load plug", plugName, "error:", e.message);
}
}));
}
updateMarkdownParser() {
// Load all syntax extensions
this.mdExtensions = loadMarkdownExtensions(this.system);
// And reload the syscalls to use the new syntax extensions
this.system.registerSyscalls(
[],
markdownSyscalls(buildMarkdown(this.mdExtensions)),
);
}
localSyscall(name: string, args: any[]) {
return this.system.localSyscall("[local]", name, args);
}
}