Now all communication happens over sockets

pull/3/head
Zef Hemel 2022-03-11 11:49:42 +01:00
parent 5c5e232034
commit da4bf4a9ab
20 changed files with 467 additions and 430 deletions

View File

@ -1,78 +1,16 @@
import { Manifest } from "./types"; import { Manifest } from "./types";
import { WebworkerSandbox } from "./worker_sandbox";
interface SysCallMapping { interface SysCallMapping {
// TODO: Better typing // TODO: Better typing
[key: string]: any; [key: string]: any;
} }
export class FunctionWorker { export interface Sandbox {
private worker: Worker; isLoaded(name: string): boolean;
private inited: Promise<any>; load(name: string, code: string): Promise<void>;
private initCallback: any; invoke(name: string, args: any[]): Promise<any>;
private invokeResolve?: (result?: any) => void; stop(): void;
private invokeReject?: (reason?: any) => void;
private plug: Plug<any>;
constructor(plug: Plug<any>, name: string, code: string) {
// let worker = window.Worker;
this.worker = new Worker(new URL("function_worker.ts", import.meta.url), {
type: "module",
});
this.worker.onmessage = this.onmessage.bind(this);
this.worker.postMessage({
type: "boot",
name: name,
code: code,
});
this.inited = new Promise((resolve) => {
this.initCallback = resolve;
});
this.plug = plug;
}
async onmessage(evt: MessageEvent) {
let data = evt.data;
if (!data) return;
switch (data.type) {
case "inited":
this.initCallback();
break;
case "syscall":
let result = await this.plug.system.syscall(data.name, data.args);
this.worker.postMessage({
type: "syscall-response",
id: data.id,
data: result,
});
break;
case "result":
this.invokeResolve!(data.result);
break;
case "error":
this.invokeReject!(data.reason);
break;
default:
console.error("Unknown message type", data);
}
}
async invoke(args: Array<any>): Promise<any> {
await this.inited;
this.worker.postMessage({
type: "invoke",
args: args,
});
return new Promise((resolve, reject) => {
this.invokeResolve = resolve;
this.invokeReject = reject;
});
}
stop() {
this.worker.terminate();
}
} }
export interface PlugLoader<HookT> { export interface PlugLoader<HookT> {
@ -81,12 +19,13 @@ export interface PlugLoader<HookT> {
export class Plug<HookT> { export class Plug<HookT> {
system: System<HookT>; system: System<HookT>;
private runningFunctions: Map<string, FunctionWorker>; // private runningFunctions: Map<string, FunctionWorker>;
functionWorker: WebworkerSandbox;
public manifest?: Manifest<HookT>; public manifest?: Manifest<HookT>;
constructor(system: System<HookT>, name: string) { constructor(system: System<HookT>, name: string) {
this.system = system; this.system = system;
this.runningFunctions = new Map<string, FunctionWorker>(); this.functionWorker = new WebworkerSandbox(this);
} }
async load(manifest: Manifest<HookT>) { async load(manifest: Manifest<HookT>) {
@ -95,16 +34,13 @@ export class Plug<HookT> {
} }
async invoke(name: string, args: Array<any>): Promise<any> { async invoke(name: string, args: Array<any>): Promise<any> {
let worker = this.runningFunctions.get(name); if (!this.functionWorker.isLoaded(name)) {
if (!worker) { await this.functionWorker.load(
worker = new FunctionWorker(
this,
name, name,
this.manifest!.functions[name].code! this.manifest!.functions[name].code!
); );
this.runningFunctions.set(name, worker);
} }
return await worker.invoke(args); return await this.functionWorker.invoke(name, args);
} }
async dispatchEvent(name: string, data?: any): Promise<any[]> { async dispatchEvent(name: string, data?: any): Promise<any[]> {
@ -122,13 +58,7 @@ export class Plug<HookT> {
} }
async stop() { async stop() {
for (const [functionname, worker] of Object.entries( this.functionWorker.stop();
this.runningFunctions
)) {
console.log(`Stopping ${functionname}`);
worker.stop();
}
this.runningFunctions = new Map<string, FunctionWorker>();
} }
} }
@ -141,7 +71,7 @@ export class System<HookT> {
this.registeredSyscalls = {}; this.registeredSyscalls = {};
} }
registerSyscalls(...registrationObjects: Array<SysCallMapping>) { registerSyscalls(...registrationObjects: SysCallMapping[]) {
for (const registrationObject of registrationObjects) { for (const registrationObject of registrationObjects) {
for (let p in registrationObject) { for (let p in registrationObject) {
this.registeredSyscalls[p] = registrationObject[p]; this.registeredSyscalls[p] = registrationObject[p];

7
plugbox/src/sandbox.html Normal file
View File

@ -0,0 +1,7 @@
<html>
<body>
<script type="module">
import "./function_worker";
</script>
</body>
</html>

View File

@ -1,11 +1,13 @@
declare global { declare global {
function syscall(id: string, name: string, args: any[]): Promise<any>; function syscall(id: number, name: string, args: any[]): Promise<any>;
} }
import { ControllerMessage, WorkerMessage, WorkerMessageType } from "./types";
import { safeRun } from "./util"; import { safeRun } from "./util";
let func: Function | null = null;
let pendingRequests = new Map<string, (result: unknown) => void>();
self.syscall = async (id: string, name: string, args: any[]) => { let loadedFunctions = new Map<string, Function>();
let pendingRequests = new Map<number, (result: unknown) => void>();
self.syscall = async (id: number, name: string, args: any[]) => {
return await new Promise((resolve, reject) => { return await new Promise((resolve, reject) => {
pendingRequests.set(id, resolve); pendingRequests.set(id, resolve);
self.postMessage({ self.postMessage({
@ -38,51 +40,56 @@ function wrapScript(code: string): string {
return fn["default"].apply(null, arguments);`; return fn["default"].apply(null, arguments);`;
} }
self.addEventListener("message", (event) => { self.addEventListener("message", (event: { data: WorkerMessage }) => {
safeRun(async () => { safeRun(async () => {
let messageEvent = event; let messageEvent = event;
let data = messageEvent.data; let data = messageEvent.data;
switch (data.type) { switch (data.type) {
case "boot": case "load":
console.log("Booting", data.name); console.log("Booting", data.name);
func = new Function(wrapScript(data.code)); loadedFunctions.set(data.name!, new Function(wrapScript(data.code!)));
self.postMessage({ self.postMessage({
type: "inited", type: "inited",
}); name: data.name,
} as ControllerMessage);
break; break;
case "invoke": case "invoke":
if (!func) { let fn = loadedFunctions.get(data.name!);
throw new Error("No function loaded"); if (!fn) {
throw new Error(`Function not loaded: ${data.name}`);
} }
try { try {
let result = await Promise.resolve(func(...(data.args || []))); let result = await Promise.resolve(fn(...(data.args || [])));
self.postMessage({ self.postMessage({
type: "result", type: "result",
id: data.id,
result: result, result: result,
}); } as ControllerMessage);
} catch (e: any) { } catch (e: any) {
self.postMessage({ self.postMessage({
type: "error", type: "error",
id: data.id,
reason: e.message, reason: e.message,
}); } as ControllerMessage);
throw e; throw e;
} }
break; break;
case "syscall-response": case "syscall-response":
let id = data.id; let syscallId = data.id!;
const lookup = pendingRequests.get(id); const lookup = pendingRequests.get(syscallId);
if (!lookup) { if (!lookup) {
console.log( console.log(
"Current outstanding requests", "Current outstanding requests",
pendingRequests, pendingRequests,
"looking up", "looking up",
id syscallId
); );
throw Error("Invalid request id"); throw Error("Invalid request id");
} }
pendingRequests.delete(id); pendingRequests.delete(syscallId);
lookup(data.data); lookup(data.data);
break;
} }
}); });
}); });

View File

@ -2,6 +2,28 @@ export type EventHook = {
events: { [key: string]: string[] }; events: { [key: string]: string[] };
}; };
export type WorkerMessageType = "load" | "invoke" | "syscall-response";
export type WorkerMessage = {
type: WorkerMessageType;
id?: number;
name?: string;
code?: string;
args?: any[];
data?: any;
};
export type ControllerMessageType = "inited" | "result" | "error" | "syscall";
export type ControllerMessage = {
type: ControllerMessageType;
id?: number;
name?: string;
reason?: string;
args?: any[];
result: any;
};
export interface Manifest<HookT> { export interface Manifest<HookT> {
hooks: HookT & EventHook; hooks: HookT & EventHook;
functions: { functions: {

View File

@ -0,0 +1,86 @@
import { ControllerMessage, WorkerMessage } from "./types";
import { Plug, Sandbox } from "./runtime";
export class WebworkerSandbox implements Sandbox {
private worker: Worker;
private reqId = 0;
private outstandingInits = new Map<string, () => void>();
private outstandingInvocations = new Map<
number,
{ resolve: (result: any) => void; reject: (e: any) => void }
>();
private loadedFunctions = new Set<string>();
constructor(readonly plug: Plug<any>) {
this.worker = new Worker(new URL("sandbox_worker.ts", import.meta.url), {
type: "module",
});
this.worker.onmessage = this.onmessage.bind(this);
}
isLoaded(name: string) {
return this.loadedFunctions.has(name);
}
async load(name: string, code: string): Promise<void> {
this.worker.postMessage({
type: "load",
name: name,
code: code,
} as WorkerMessage);
return new Promise((resolve) => {
this.loadedFunctions.add(name);
this.outstandingInits.set(name, resolve);
});
}
async onmessage(evt: { data: ControllerMessage }) {
let data = evt.data;
if (!data) return;
switch (data.type) {
case "inited":
let initCb = this.outstandingInits.get(data.name!);
initCb && initCb();
this.outstandingInits.delete(data.name!);
break;
case "syscall":
let result = await this.plug.system.syscall(data.name!, data.args!);
this.worker.postMessage({
type: "syscall-response",
id: data.id,
data: result,
} as WorkerMessage);
break;
case "result":
let resultCb = this.outstandingInvocations.get(data.id!);
resultCb && resultCb.resolve(data.result);
break;
case "error":
let errCb = this.outstandingInvocations.get(data.result.id!);
errCb && errCb.reject(data.reason);
break;
default:
console.error("Unknown message type", data);
}
}
async invoke(name: string, args: any[]): Promise<any> {
this.reqId++;
this.worker.postMessage({
type: "invoke",
id: this.reqId,
name,
args,
});
return new Promise((resolve, reject) => {
this.outstandingInvocations.set(this.reqId, { resolve, reject });
});
}
stop() {
this.worker.terminate();
}
}

View File

@ -1,9 +1,9 @@
declare global { declare global {
function syscall(id: string, name: string, args: any[]): Promise<any>; function syscall(id: number, name: string, args: any[]): Promise<any>;
} }
export async function syscall(name: string, ...args: any[]): Promise<any> { export async function syscall(name: string, ...args: any[]): Promise<any> {
let reqId = "" + Math.floor(Math.random() * 1000000); let reqId = Math.floor(Math.random() * 1000000);
// console.log("Syscall", name, reqId); // console.log("Syscall", name, reqId);
return await self.syscall(reqId, name, args); return await self.syscall(reqId, name, args);
// return new Promise((resolve, reject) => { // return new Promise((resolve, reject) => {

View File

@ -25,8 +25,6 @@ export async function deletePage() {
await syscall("editor.navigate", "start"); await syscall("editor.navigate", "start");
console.log("Deleting page from space"); console.log("Deleting page from space");
await syscall("space.deletePage", pageName); await syscall("space.deletePage", pageName);
console.log("Reloading page list");
await syscall("space.reloadPageList");
} }
export async function renamePage() { export async function renamePage() {
@ -50,8 +48,6 @@ export async function renamePage() {
await syscall("space.writePage", newName, text); await syscall("space.writePage", newName, text);
console.log("Deleting page from space"); console.log("Deleting page from space");
await syscall("space.deletePage", oldName); await syscall("space.deletePage", oldName);
console.log("Reloading page list");
await syscall("space.reloadPageList");
console.log("Navigating to new page"); console.log("Navigating to new page");
await syscall("editor.navigate", newName); await syscall("editor.navigate", newName);
@ -63,6 +59,7 @@ export async function renamePage() {
for (let pageToUpdate of pageToUpdateSet) { for (let pageToUpdate of pageToUpdateSet) {
console.log("Now going to update links in", pageToUpdate); console.log("Now going to update links in", pageToUpdate);
let { text } = await syscall("space.readPage", pageToUpdate); let { text } = await syscall("space.readPage", pageToUpdate);
console.log("Received text", text);
if (!text) { if (!text) {
// Page likely does not exist, but at least we can skip it // Page likely does not exist, but at least we can skip it
continue; continue;

4
plugs/yarn.lock Normal file
View File

@ -0,0 +1,4 @@
# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY.
# yarn lockfile v1

View File

@ -8,118 +8,28 @@ import { Cursor, cursorEffect } from "../../webapp/src/cursorEffect";
import { Socket } from "socket.io"; import { Socket } from "socket.io";
import { DiskStorage } from "./disk_storage"; import { DiskStorage } from "./disk_storage";
import { PageMeta } from "./server"; import { PageMeta } from "./server";
import { Client, Page } from "./types"; import { ClientPageState, Page } from "./types";
import { safeRun } from "./util"; import { safeRun } from "./util";
export class RealtimeStorage extends DiskStorage { export class SocketAPI {
openPages = new Map<string, Page>(); openPages = new Map<string, Page>();
connectedSockets: Set<Socket> = new Set();
private disconnectClient(client: Client, page: Page) { pageStore: DiskStorage;
page.clients.delete(client);
if (page.clients.size === 0) {
console.log("No more clients for", page.name, "flushing");
this.flushPageToDisk(page.name, page);
this.openPages.delete(page.name);
} else {
page.cursors.delete(client.socket.id);
this.broadcastCursors(page);
}
}
private broadcastCursors(page: Page) {
page.clients.forEach((client) => {
client.socket.emit("cursors", Object.fromEntries(page.cursors.entries()));
});
}
private flushPageToDisk(name: string, page: Page) {
super
.writePage(name, page.text.sliceString(0))
.then((meta) => {
console.log(`Wrote page ${name} to disk`);
page.meta = meta;
})
.catch((e) => {
console.log(`Could not write ${name} to disk:`, e);
});
}
// Override
async readPage(pageName: string): Promise<{ text: string; meta: PageMeta }> {
let page = this.openPages.get(pageName);
if (page) {
console.log("Serving page from memory", pageName);
return {
text: page.text.sliceString(0),
meta: page.meta,
};
} else {
return super.readPage(pageName);
}
}
async writePage(pageName: string, text: string): Promise<PageMeta> {
let page = this.openPages.get(pageName);
if (page) {
for (let client of page.clients) {
client.socket.emit("reload", pageName);
}
this.openPages.delete(pageName);
}
return super.writePage(pageName, text);
}
disconnectPageSocket(socket: Socket, pageName: string) {
let page = this.openPages.get(pageName);
if (page) {
for (let client of page.clients) {
if (client.socket === socket) {
this.disconnectClient(client, page);
}
}
}
}
constructor(rootPath: string, io: Server) { constructor(rootPath: string, io: Server) {
super(rootPath); this.pageStore = new DiskStorage(rootPath);
this.fileWatcher(rootPath);
// setInterval(() => {
// console.log("Currently open pages:", this.openPages.keys());
// }, 10000);
// Disk watcher
fs.watch(
rootPath,
{
recursive: true,
persistent: false,
},
(eventType, filename) => {
safeRun(async () => {
if (path.extname(filename) !== ".md") {
return;
}
let localPath = path.join(rootPath, filename);
let pageName = filename.substring(0, filename.length - 3);
let s = await stat(localPath);
// console.log("Edit in", pageName);
const openPage = this.openPages.get(pageName);
if (openPage) {
if (openPage.meta.lastModified < s.mtime.getTime()) {
console.log("Page changed on disk outside of editor, reloading");
this.openPages.delete(pageName);
for (let client of openPage.clients) {
client.socket.emit("reload", pageName);
}
}
}
});
}
);
io.on("connection", (socket) => { io.on("connection", (socket) => {
console.log("Connected", socket.id); console.log("Connected", socket.id);
let clientOpenPages = new Set<string>(); this.connectedSockets.add(socket);
const socketOpenPages = new Set<string>();
socket.on("disconnect", () => {
console.log("Disconnected", socket.id);
socketOpenPages.forEach(disconnectPageSocket);
this.connectedSockets.delete(socket);
});
function onCall(eventName: string, cb: (...args: any[]) => Promise<any>) { function onCall(eventName: string, cb: (...args: any[]) => Promise<any>) {
socket.on(eventName, (reqId: number, ...args) => { socket.on(eventName, (reqId: number, ...args) => {
@ -129,11 +39,23 @@ export class RealtimeStorage extends DiskStorage {
}); });
} }
const _this = this;
function disconnectPageSocket(pageName: string) {
let page = _this.openPages.get(pageName);
if (page) {
for (let client of page.clientStates) {
if (client.socket === socket) {
_this.disconnectClient(client, page);
}
}
}
}
onCall("openPage", async (pageName: string) => { onCall("openPage", async (pageName: string) => {
let page = this.openPages.get(pageName); let page = this.openPages.get(pageName);
if (!page) { if (!page) {
try { try {
let { text, meta } = await super.readPage(pageName); let { text, meta } = await this.pageStore.readPage(pageName);
page = new Page(pageName, text, meta); page = new Page(pageName, text, meta);
} catch (e) { } catch (e) {
console.log("Creating new page", pageName); console.log("Creating new page", pageName);
@ -141,8 +63,8 @@ export class RealtimeStorage extends DiskStorage {
} }
this.openPages.set(pageName, page); this.openPages.set(pageName, page);
} }
page.clients.add(new Client(socket, page.version)); page.clientStates.add(new ClientPageState(socket, page.version));
clientOpenPages.add(pageName); socketOpenPages.add(pageName);
console.log("Opened page", pageName); console.log("Opened page", pageName);
this.broadcastCursors(page); this.broadcastCursors(page);
return page.toJSON(); return page.toJSON();
@ -150,8 +72,8 @@ export class RealtimeStorage extends DiskStorage {
socket.on("closePage", (pageName: string) => { socket.on("closePage", (pageName: string) => {
console.log("Closing page", pageName); console.log("Closing page", pageName);
clientOpenPages.delete(pageName); socketOpenPages.delete(pageName);
this.disconnectPageSocket(socket, pageName); disconnectPageSocket(pageName);
}); });
onCall( onCall(
@ -169,13 +91,13 @@ export class RealtimeStorage extends DiskStorage {
pageName, pageName,
this.openPages.keys() this.openPages.keys()
); );
return; return false;
} }
if (version !== page.version) { if (version !== page.version) {
console.error("Invalid version", version, page.version); console.error("Invalid version", version, page.version);
return false; return false;
} else { } else {
console.log("Applying", updates.length, "updates"); console.log("Applying", updates.length, "updates to", pageName);
let transformedUpdates = []; let transformedUpdates = [];
let textChanged = false; let textChanged = false;
for (let update of updates) { for (let update of updates) {
@ -225,7 +147,7 @@ export class RealtimeStorage extends DiskStorage {
} }
// TODO: Optimize this // TODO: Optimize this
let oldestVersion = Infinity; let oldestVersion = Infinity;
page.clients.forEach((client) => { page.clientStates.forEach((client) => {
oldestVersion = Math.min(client.version, oldestVersion); oldestVersion = Math.min(client.version, oldestVersion);
if (client.socket === socket) { if (client.socket === socket) {
client.version = version; client.version = version;
@ -242,12 +164,139 @@ export class RealtimeStorage extends DiskStorage {
} }
); );
socket.on("disconnect", () => { onCall(
console.log("Disconnected", socket.id); "readPage",
clientOpenPages.forEach((pageName) => { async (pageName: string): Promise<{ text: string; meta: PageMeta }> => {
this.disconnectPageSocket(socket, pageName); let page = this.openPages.get(pageName);
if (page) {
console.log("Serving page from memory", pageName);
return {
text: page.text.sliceString(0),
meta: page.meta,
};
} else {
return this.pageStore.readPage(pageName);
}
}
);
onCall("writePage", async (pageName: string, text: string) => {
let page = this.openPages.get(pageName);
if (page) {
for (let client of page.clientStates) {
client.socket.emit("reloadPage", pageName);
}
this.openPages.delete(pageName);
}
return this.pageStore.writePage(pageName, text);
}); });
onCall("deletePage", async (pageName: string) => {
this.openPages.delete(pageName);
socketOpenPages.delete(pageName);
// Cascading of this to all connected clients will be handled by file watcher
return this.pageStore.deletePage(pageName);
});
onCall("listPages", async (): Promise<PageMeta[]> => {
return this.pageStore.listPages();
});
onCall("getPageMeta", async (pageName: string): Promise<PageMeta> => {
let page = this.openPages.get(pageName);
if (page) {
return page.meta;
}
return this.pageStore.getPageMeta(pageName);
}); });
}); });
} }
private disconnectClient(client: ClientPageState, page: Page) {
page.clientStates.delete(client);
if (page.clientStates.size === 0) {
console.log("No more clients for", page.name, "flushing");
this.flushPageToDisk(page.name, page);
this.openPages.delete(page.name);
} else {
page.cursors.delete(client.socket.id);
this.broadcastCursors(page);
}
}
private broadcastCursors(page: Page) {
page.clientStates.forEach((client) => {
client.socket.emit(
"cursorSnapshot",
page.name,
Object.fromEntries(page.cursors.entries())
);
});
}
private flushPageToDisk(name: string, page: Page) {
safeRun(async () => {
let meta = await this.pageStore.writePage(name, page.text.sliceString(0));
console.log(`Wrote page ${name} to disk`);
page.meta = meta;
});
}
private fileWatcher(rootPath: string) {
fs.watch(
rootPath,
{
recursive: true,
persistent: false,
},
(eventType, filename) => {
safeRun(async () => {
if (path.extname(filename) !== ".md") {
return;
}
let localPath = path.join(rootPath, filename);
let pageName = filename.substring(0, filename.length - 3);
// console.log("Edit in", pageName, eventType);
let modifiedTime = 0;
try {
let s = await stat(localPath);
modifiedTime = s.mtime.getTime();
} catch (e) {
// File was deleted
console.log("Deleted", pageName);
for (let socket of this.connectedSockets) {
socket.emit("pageDeleted", pageName);
}
return;
}
const openPage = this.openPages.get(pageName);
if (openPage) {
if (openPage.meta.lastModified < modifiedTime) {
console.log("Page changed on disk outside of editor, reloading");
this.openPages.delete(pageName);
const meta = {
name: pageName,
lastModified: modifiedTime,
} as PageMeta;
for (let client of openPage.clientStates) {
client.socket.emit("pageChanged", meta);
}
}
}
if (eventType === "rename") {
// This most likely means a new file was created, let's push new file listings to all connected sockets
console.log(
"New file created, broadcasting to all connected sockets"
);
for (let socket of this.connectedSockets) {
socket.emit("pageCreated", {
name: pageName,
lastModified: modifiedTime,
} as PageMeta);
}
}
});
}
);
}
} }

View File

@ -1,12 +1,8 @@
import bodyParser from "body-parser";
import cors from "cors";
import express from "express"; import express from "express";
import { readFile } from "fs/promises"; import { readFile } from "fs/promises";
import http from "http"; import http from "http";
import { Server } from "socket.io"; import { Server } from "socket.io";
import stream from "stream"; import { SocketAPI } from "./api";
import { promisify } from "util";
import { RealtimeStorage } from "./realtime_storage";
const app = express(); const app = express();
const server = http.createServer(app); const server = http.createServer(app);
@ -18,7 +14,6 @@ const io = new Server(server, {
}); });
const port = 3000; const port = 3000;
const pipeline = promisify(stream.pipeline);
export const pagesPath = "../pages"; export const pagesPath = "../pages";
const distDir = `${__dirname}/../../webapp/dist`; const distDir = `${__dirname}/../../webapp/dist`;
@ -29,80 +24,7 @@ export type PageMeta = {
}; };
app.use("/", express.static(distDir)); app.use("/", express.static(distDir));
let filesystem = new SocketAPI(pagesPath, io);
let fsRouter = express.Router();
// let diskFS = new DiskFS(pagesPath);
let filesystem = new RealtimeStorage(pagesPath, io);
// Page list
fsRouter.route("/").get(async (req, res) => {
res.json(await filesystem.listPages());
});
fsRouter
.route(/\/(.+)/)
.get(async (req, res) => {
let reqPath = req.params[0];
console.log("Getting", reqPath);
try {
let { text, meta } = await filesystem.readPage(reqPath);
res.status(200);
res.header("Last-Modified", "" + meta.lastModified);
res.header("Content-Type", "text/markdown");
res.send(text);
} catch (e) {
res.status(200);
res.send("");
}
})
.put(bodyParser.text({ type: "*/*" }), async (req, res) => {
let reqPath = req.params[0];
try {
let meta = await filesystem.writePage(reqPath, req.body);
res.status(200);
res.header("Last-Modified", "" + meta.lastModified);
res.send("OK");
} catch (err) {
res.status(500);
res.send("Write failed");
console.error("Pipeline failed", err);
}
})
.options(async (req, res) => {
let reqPath = req.params[0];
try {
const meta = await filesystem.getPageMeta(reqPath);
res.status(200);
res.header("Last-Modified", "" + meta.lastModified);
res.header("Content-Type", "text/markdown");
res.send("");
} catch (e) {
res.status(200);
res.send("");
}
})
.delete(async (req, res) => {
let reqPath = req.params[0];
try {
await filesystem.deletePage(reqPath);
res.status(200);
res.send("OK");
} catch (e) {
console.error("Error deleting file", reqPath, e);
res.status(500);
res.send("OK");
}
});
app.use(
"/fs",
cors({
methods: "GET,HEAD,PUT,OPTIONS,POST,DELETE",
preflightContinue: true,
}),
fsRouter
);
// Fallback, serve index.html // Fallback, serve index.html
let cachedIndex: string | undefined = undefined; let cachedIndex: string | undefined = undefined;
@ -113,7 +35,6 @@ app.get("/*", async (req, res) => {
res.status(200).header("Content-Type", "text/html").send(cachedIndex); res.status(200).header("Content-Type", "text/html").send(cachedIndex);
}); });
//sup
server.listen(port, () => { server.listen(port, () => {
console.log(`Server istening on port ${port}`); console.log(`Server istening on port ${port}`);
}); });

View File

@ -4,7 +4,7 @@ import { Socket } from "socket.io";
import { Cursor } from "../../webapp/src/cursorEffect"; import { Cursor } from "../../webapp/src/cursorEffect";
import { PageMeta } from "./server"; import { PageMeta } from "./server";
export class Client { export class ClientPageState {
constructor(public socket: Socket, public version: number) {} constructor(public socket: Socket, public version: number) {}
} }
@ -12,7 +12,7 @@ export class Page {
versionOffset = 0; versionOffset = 0;
updates: Update[] = []; updates: Update[] = [];
cursors = new Map<string, Cursor>(); cursors = new Map<string, Cursor>();
clients = new Set<Client>(); clientStates = new Set<ClientPageState>();
pending: ((value: any) => void)[] = []; pending: ((value: any) => void)[] = [];

View File

@ -1,12 +1,12 @@
import { Editor } from "./editor"; import { Editor } from "./editor";
import { HttpRemoteSpace } from "./space"; import { RealtimeSpace } from "./space";
import { safeRun } from "./util"; import { safeRun } from "./util";
import { io } from "socket.io-client"; import { io } from "socket.io-client";
let socket = io(`http://${location.hostname}:3000`); let socket = io(`http://${location.hostname}:3000`);
let editor = new Editor( let editor = new Editor(
new HttpRemoteSpace(`http://${location.hostname}:3000/fs`, socket), new RealtimeSpace(socket),
document.getElementById("root")! document.getElementById("root")!
); );

View File

@ -10,8 +10,8 @@ import {
receiveUpdates, receiveUpdates,
sendableUpdates, sendableUpdates,
} from "@codemirror/collab"; } from "@codemirror/collab";
import { RangeSetBuilder, Range } from "@codemirror/rangeset"; import { RangeSetBuilder } from "@codemirror/rangeset";
import { EditorState, StateEffect, StateField, Text } from "@codemirror/state"; import { Text } from "@codemirror/state";
import { import {
Decoration, Decoration,
DecorationSet, DecorationSet,
@ -21,7 +21,7 @@ import {
WidgetType, WidgetType,
} from "@codemirror/view"; } from "@codemirror/view";
import { Cursor, cursorEffect } from "./cursorEffect"; import { Cursor, cursorEffect } from "./cursorEffect";
import { HttpRemoteSpace } from "./space"; import { RealtimeSpace, SpaceEventHandlers } from "./space";
const throttleInterval = 250; const throttleInterval = 250;
@ -85,7 +85,7 @@ export function collabExtension(
pageName: string, pageName: string,
clientID: string, clientID: string,
doc: Document, doc: Document,
space: HttpRemoteSpace, space: RealtimeSpace,
reloadCallback: () => void reloadCallback: () => void
) { ) {
let plugin = ViewPlugin.fromClass( let plugin = ViewPlugin.fromClass(
@ -95,7 +95,15 @@ export function collabExtension(
private failedPushes = 0; private failedPushes = 0;
decorations: DecorationSet; decorations: DecorationSet;
private cursorPositions: Map<string, Cursor> = doc.cursors; private cursorPositions: Map<string, Cursor> = doc.cursors;
throttledPush: () => void;
throttledPush = throttle(() => this.push(), throttleInterval);
eventHandlers: Partial<SpaceEventHandlers> = {
cursorSnapshot: (pageName, cursors) => {
console.log("Received new cursor snapshot", cursors);
this.cursorPositions = new Map(Object.entries(cursors));
},
};
buildDecorations(view: EditorView) { buildDecorations(view: EditorView) {
let builder = new RangeSetBuilder<Decoration>(); let builder = new RangeSetBuilder<Decoration>();
@ -128,18 +136,7 @@ export function collabExtension(
this.pull(); this.pull();
} }
this.decorations = this.buildDecorations(view); this.decorations = this.buildDecorations(view);
this.throttledPush = throttle(() => this.push(), throttleInterval); space.on(this.eventHandlers);
console.log("Created collabo plug");
space.addEventListener("cursors", this.updateCursors);
}
updateCursors(cursorEvent: any) {
this.cursorPositions = new Map();
console.log("Received new cursor snapshot", cursorEvent.detail, this);
for (let userId in cursorEvent.detail) {
this.cursorPositions.set(userId, cursorEvent.detail[userId]);
}
} }
update(update: ViewUpdate) { update(update: ViewUpdate) {
@ -190,7 +187,7 @@ export function collabExtension(
let success = await space.pushUpdates(pageName, version, updates); let success = await space.pushUpdates(pageName, version, updates);
this.pushing = false; this.pushing = false;
if (!success) { if (!success && !this.done) {
this.failedPushes++; this.failedPushes++;
if (this.failedPushes > 10) { if (this.failedPushes > 10) {
// Not sure if 10 is a good number, but YOLO // Not sure if 10 is a good number, but YOLO
@ -198,14 +195,16 @@ export function collabExtension(
reloadCallback(); reloadCallback();
return this.destroy(); return this.destroy();
} }
console.log("Push failed temporarily, but will try again"); console.log(
`Push for page ${pageName} failed temporarily, but will try again`
);
} else { } else {
this.failedPushes = 0; this.failedPushes = 0;
} }
// Regardless of whether the push failed or new updates came in // Regardless of whether the push failed or new updates came in
// while it was running, try again if there's updates remaining // while it was running, try again if there's updates remaining
if (sendableUpdates(this.view.state).length) { if (!this.done && sendableUpdates(this.view.state).length) {
// setTimeout(() => this.push(), 100); // setTimeout(() => this.push(), 100);
this.throttledPush(); this.throttledPush();
} }
@ -236,7 +235,7 @@ export function collabExtension(
destroy() { destroy() {
this.done = true; this.done = true;
space.removeEventListener("cursors", this.updateCursors); space.off(this.eventHandlers);
} }
}, },
{ {
@ -252,7 +251,6 @@ export function collabExtension(
return tr.effects.filter((e) => e.is(cursorEffect)); return tr.effects.filter((e) => e.is(cursorEffect));
}, },
}), }),
// cursorField,
plugin, plugin,
]; ];
} }

View File

@ -100,7 +100,7 @@ export function FilterList({
ref={searchBoxRef} ref={searchBoxRef}
onChange={filter} onChange={filter}
onKeyDown={(e: React.KeyboardEvent) => { onKeyDown={(e: React.KeyboardEvent) => {
console.log("Key up", e.key); // console.log("Key up", e.key);
if (onKeyPress) { if (onKeyPress) {
onKeyPress(e.key, text); onKeyPress(e.key, text);
} }

View File

@ -7,7 +7,7 @@ export function PageNavigator({
onNavigate, onNavigate,
currentPage, currentPage,
}: { }: {
allPages: PageMeta[]; allPages: Set<PageMeta>;
onNavigate: (page: string | undefined) => void; onNavigate: (page: string | undefined) => void;
currentPage?: string; currentPage?: string;
}) { }) {
@ -17,10 +17,10 @@ export function PageNavigator({
continue; continue;
} }
// Order by last modified date in descending order // Order by last modified date in descending order
let orderId = -pageMeta.lastModified.getTime(); let orderId = -pageMeta.lastModified;
// Unless it was opened and is still in memory // Unless it was opened and is still in memory
if (pageMeta.lastOpened) { if (pageMeta.lastOpened) {
orderId = -pageMeta.lastOpened.getTime(); orderId = -pageMeta.lastOpened;
} }
options.push({ options.push({
...pageMeta, ...pageMeta,

View File

@ -45,7 +45,7 @@ import { slashCommandRegexp } from "./types";
import reducer from "./reducer"; import reducer from "./reducer";
import { smartQuoteKeymap } from "./smart_quotes"; import { smartQuoteKeymap } from "./smart_quotes";
import { HttpRemoteSpace } from "./space"; import { RealtimeSpace } from "./space";
import customMarkdownStyle from "./style"; import customMarkdownStyle from "./style";
import dbSyscalls from "./syscalls/db.localstorage"; import dbSyscalls from "./syscalls/db.localstorage";
import editorSyscalls from "./syscalls/editor.browser"; import editorSyscalls from "./syscalls/editor.browser";
@ -84,7 +84,7 @@ export class Editor implements AppEventDispatcher {
viewState: AppViewState; viewState: AppViewState;
viewDispatch: React.Dispatch<Action>; viewDispatch: React.Dispatch<Action>;
openPages: Map<string, PageState>; openPages: Map<string, PageState>;
space: HttpRemoteSpace; space: RealtimeSpace;
editorCommands: Map<string, AppCommand>; editorCommands: Map<string, AppCommand>;
plugs: Plug<NuggetHook>[]; plugs: Plug<NuggetHook>[];
indexer: Indexer; indexer: Indexer;
@ -92,7 +92,7 @@ export class Editor implements AppEventDispatcher {
pageNavigator: IPageNavigator; pageNavigator: IPageNavigator;
indexCurrentPageDebounced: () => any; indexCurrentPageDebounced: () => any;
constructor(space: HttpRemoteSpace, parent: Element) { constructor(space: RealtimeSpace, parent: Element) {
this.editorCommands = new Map(); this.editorCommands = new Map();
this.openPages = new Map(); this.openPages = new Map();
this.plugs = []; this.plugs = [];
@ -114,7 +114,7 @@ export class Editor implements AppEventDispatcher {
} }
async init() { async init() {
await this.loadPageList(); // await this.loadPageList();
await this.loadPlugs(); await this.loadPlugs();
this.focus(); this.focus();
@ -127,8 +127,10 @@ export class Editor implements AppEventDispatcher {
if (this.currentPage) { if (this.currentPage) {
let pageState = this.openPages.get(this.currentPage)!; let pageState = this.openPages.get(this.currentPage)!;
if (pageState) {
pageState.selection = this.editorView!.state.selection; pageState.selection = this.editorView!.state.selection;
pageState.scrollTop = this.editorView!.scrollDOM.scrollTop; pageState.scrollTop = this.editorView!.scrollDOM.scrollTop;
}
this.space.closePage(this.currentPage); this.space.closePage(this.currentPage);
} }
@ -136,19 +138,25 @@ export class Editor implements AppEventDispatcher {
await this.loadPage(pageName); await this.loadPage(pageName);
}); });
this.space.addEventListener("connect", () => { this.space.on({
connect: () => {
if (this.currentPage) { if (this.currentPage) {
console.log("Connected to socket, fetch fresh?"); console.log("Connected to socket, fetch fresh?");
this.reloadPage(); this.reloadPage();
} }
}); },
pageChanged: (meta) => {
this.space.addEventListener("reload", (e) => { if (this.currentPage === meta.name) {
let pageName = (e as CustomEvent).detail; console.log("page changed on disk, reloading");
if (this.currentPage === pageName) {
console.log("Was told to reload the page");
this.reloadPage(); this.reloadPage();
} }
},
pageListUpdated: (pages) => {
this.viewDispatch({
type: "pages-listed",
pages: pages,
});
},
}); });
if (this.pageNavigator.getCurrentPage() === "") { if (this.pageNavigator.getCurrentPage() === "") {
@ -411,14 +419,6 @@ export class Editor implements AppEventDispatcher {
} }
} }
async loadPageList() {
let pagesMeta = await this.space.listPages();
this.viewDispatch({
type: "pages-listed",
pages: pagesMeta,
});
}
focus() { focus() {
this.editorView!.focus(); this.editorView!.focus();
} }

View File

@ -9,10 +9,12 @@ export default function reducer(
case "page-loaded": case "page-loaded":
return { return {
...state, ...state,
allPages: state.allPages.map((pageMeta) => allPages: new Set(
[...state.allPages].map((pageMeta) =>
pageMeta.name === action.name pageMeta.name === action.name
? { ...pageMeta, lastOpened: new Date() } ? { ...pageMeta, lastOpened: Date.now() }
: pageMeta : pageMeta
)
), ),
currentPage: action.name, currentPage: action.name,
}; };

View File

@ -14,27 +14,81 @@ export interface Space {
getPageMeta(name: string): Promise<PageMeta>; getPageMeta(name: string): Promise<PageMeta>;
} }
export class HttpRemoteSpace extends EventTarget implements Space { export type SpaceEventHandlers = {
url: string; connect: () => void;
cursorSnapshot: (
pageName: string,
cursors: { [key: string]: Cursor }
) => void;
pageCreated: (meta: PageMeta) => void;
pageChanged: (meta: PageMeta) => void;
pageDeleted: (name: string) => void;
pageListUpdated: (pages: Set<PageMeta>) => void;
};
abstract class EventEmitter<HandlerT> {
private handlers: Partial<HandlerT>[] = [];
on(handlers: Partial<HandlerT>) {
this.handlers.push(handlers);
}
off(handlers: Partial<HandlerT>) {
this.handlers = this.handlers.filter((h) => h !== handlers);
}
emit(eventName: keyof HandlerT, ...args: any[]) {
for (let handler of this.handlers) {
let fn: any = handler[eventName];
if (fn) {
fn(...args);
}
}
}
}
export class RealtimeSpace
extends EventEmitter<SpaceEventHandlers>
implements Space
{
socket: Socket; socket: Socket;
reqId = 0; reqId = 0;
allPages = new Set<PageMeta>();
constructor(url: string, socket: Socket) { constructor(socket: Socket) {
super(); super();
this.url = url;
this.socket = socket; this.socket = socket;
socket.on("connect", () => { [
console.log("connected to socket"); "connect",
this.dispatchEvent(new Event("connect")); "cursorSnapshot",
"pageCreated",
"pageChanged",
"pageDeleted",
].forEach((eventName) => {
socket.on(eventName, (...args) => {
this.emit(eventName as keyof SpaceEventHandlers, ...args);
}); });
socket.on("reload", (pageName: string) => {
this.dispatchEvent(new CustomEvent("reload", { detail: pageName }));
}); });
this.wsCall("listPages").then((pages) => {
socket.on("cursors", (cursors) => { this.allPages = new Set(pages);
this.dispatchEvent(new CustomEvent("cursors", { detail: cursors })); this.emit("pageListUpdated", this.allPages);
});
this.on({
pageCreated: (meta) => {
this.allPages.add(meta);
console.log("New page created", meta);
this.emit("pageListUpdated", this.allPages);
},
pageDeleted: (name) => {
console.log("Page delete", name);
this.allPages.forEach((meta) => {
if (name === meta.name) {
this.allPages.delete(meta);
}
});
this.emit("pageListUpdated", this.allPages);
},
}); });
} }
@ -76,14 +130,7 @@ export class HttpRemoteSpace extends EventTarget implements Space {
} }
async listPages(): Promise<PageMeta[]> { async listPages(): Promise<PageMeta[]> {
let req = await fetch(this.url, { return Array.from(this.allPages);
method: "GET",
});
return (await req.json()).map((meta: any) => ({
name: meta.name,
lastModified: new Date(meta.lastModified),
}));
} }
async openPage(name: string): Promise<Document> { async openPage(name: string): Promise<Document> {
@ -101,47 +148,18 @@ export class HttpRemoteSpace extends EventTarget implements Space {
} }
async readPage(name: string): Promise<{ text: string; meta: PageMeta }> { async readPage(name: string): Promise<{ text: string; meta: PageMeta }> {
let req = await fetch(`${this.url}/${name}`, { return this.wsCall("readPage", name);
method: "GET",
});
return {
text: await req.text(),
meta: {
lastModified: new Date(+req.headers.get("Last-Modified")!),
name: name,
},
};
} }
async writePage(name: string, text: string): Promise<PageMeta> { async writePage(name: string, text: string): Promise<PageMeta> {
let req = await fetch(`${this.url}/${name}`, { return this.wsCall("writePage", name, text);
method: "PUT",
body: text,
});
// 201 (Created) means a new page was created
return {
lastModified: new Date(+req.headers.get("Last-Modified")!),
name: name,
created: req.status === 201,
};
} }
async deletePage(name: string): Promise<void> { async deletePage(name: string): Promise<void> {
let req = await fetch(`${this.url}/${name}`, { return this.wsCall("deletePage", name);
method: "DELETE",
});
if (req.status !== 200) {
throw Error(`Failed to delete page: ${req.statusText}`);
}
} }
async getPageMeta(name: string): Promise<PageMeta> { async getPageMeta(name: string): Promise<PageMeta> {
let req = await fetch(`${this.url}/${name}`, { return this.wsCall("deletePage", name);
method: "OPTIONS",
});
return {
name: name,
lastModified: new Date(+req.headers.get("Last-Modified")!),
};
} }
} }

View File

@ -3,10 +3,7 @@ import { PageMeta } from "../types";
export default (editor: Editor) => ({ export default (editor: Editor) => ({
"space.listPages": (): PageMeta[] => { "space.listPages": (): PageMeta[] => {
return editor.viewState.allPages; return [...editor.viewState.allPages];
},
"space.reloadPageList": async () => {
await editor.loadPageList();
}, },
"space.reindex": async () => { "space.reindex": async () => {
await editor.indexer.reindexSpace(editor.space, editor); await editor.indexer.reindexSpace(editor.space, editor);

View File

@ -10,10 +10,9 @@ export type Manifest = plugbox.Manifest<NuggetHook>;
export type PageMeta = { export type PageMeta = {
name: string; name: string;
lastModified: Date; lastModified: number;
version?: number; version?: number;
created?: boolean; lastOpened?: number;
lastOpened?: Date;
}; };
export type AppCommand = { export type AppCommand = {
@ -40,20 +39,20 @@ export type AppViewState = {
currentPage?: string; currentPage?: string;
showPageNavigator: boolean; showPageNavigator: boolean;
showCommandPalette: boolean; showCommandPalette: boolean;
allPages: PageMeta[]; allPages: Set<PageMeta>;
commands: Map<string, AppCommand>; commands: Map<string, AppCommand>;
}; };
export const initialViewState: AppViewState = { export const initialViewState: AppViewState = {
showPageNavigator: false, showPageNavigator: false,
showCommandPalette: false, showCommandPalette: false,
allPages: [], allPages: new Set(),
commands: new Map(), commands: new Map(),
}; };
export type Action = export type Action =
| { type: "page-loaded"; name: string } | { type: "page-loaded"; name: string }
| { type: "pages-listed"; pages: PageMeta[] } | { type: "pages-listed"; pages: Set<PageMeta> }
| { type: "start-navigate" } | { type: "start-navigate" }
| { type: "stop-navigate" } | { type: "stop-navigate" }
| { type: "update-commands"; commands: Map<string, AppCommand> } | { type: "update-commands"; commands: Map<string, AppCommand> }