2023-05-23 18:53:53 +00:00
|
|
|
import { Application, Router } from "./deps.ts";
|
2022-10-10 12:50:21 +00:00
|
|
|
import { SpacePrimitives } from "../common/spaces/space_primitives.ts";
|
2022-10-12 09:47:13 +00:00
|
|
|
import { AssetBundle } from "../plugos/asset_bundle/bundle.ts";
|
2023-01-13 14:41:29 +00:00
|
|
|
import { base64Decode } from "../plugos/asset_bundle/base64.ts";
|
2023-05-23 18:53:53 +00:00
|
|
|
import { ensureSettingsAndIndex } from "../common/util.ts";
|
|
|
|
import { performLocalFetch } from "../common/proxy_fetch.ts";
|
2022-10-10 12:50:21 +00:00
|
|
|
|
|
|
|
export type ServerOptions = {
|
2022-12-04 05:24:06 +00:00
|
|
|
hostname: string;
|
2022-10-10 12:50:21 +00:00
|
|
|
port: number;
|
|
|
|
pagesPath: string;
|
2023-05-23 18:53:53 +00:00
|
|
|
clientAssetBundle: AssetBundle;
|
2022-12-05 11:14:21 +00:00
|
|
|
user?: string;
|
|
|
|
pass?: string;
|
2023-05-23 18:53:53 +00:00
|
|
|
certFile?: string;
|
|
|
|
keyFile?: string;
|
|
|
|
maxFileSizeMB?: number;
|
2022-10-10 12:50:21 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
export class HttpServer {
|
|
|
|
app: Application;
|
2022-12-04 05:24:06 +00:00
|
|
|
private hostname: string;
|
2022-10-10 12:50:21 +00:00
|
|
|
private port: number;
|
2022-12-05 11:14:21 +00:00
|
|
|
user?: string;
|
2022-10-10 12:50:21 +00:00
|
|
|
settings: { [key: string]: any } = {};
|
|
|
|
abortController?: AbortController;
|
2023-05-23 18:53:53 +00:00
|
|
|
clientAssetBundle: AssetBundle;
|
2022-10-10 12:50:21 +00:00
|
|
|
|
2023-05-23 18:53:53 +00:00
|
|
|
constructor(
|
|
|
|
private spacePrimitives: SpacePrimitives,
|
|
|
|
private options: ServerOptions,
|
|
|
|
) {
|
2022-12-04 05:24:06 +00:00
|
|
|
this.hostname = options.hostname;
|
2022-10-10 12:50:21 +00:00
|
|
|
this.port = options.port;
|
2023-05-23 18:53:53 +00:00
|
|
|
this.app = new Application();
|
2023-05-24 03:42:24 +00:00
|
|
|
this.user = options.user;
|
2023-05-23 18:53:53 +00:00
|
|
|
this.clientAssetBundle = options.clientAssetBundle;
|
|
|
|
}
|
2022-10-10 12:50:21 +00:00
|
|
|
|
2023-05-23 18:53:53 +00:00
|
|
|
// Replaces some template variables in index.html in a rather ad-hoc manner, but YOLO
|
|
|
|
renderIndexHtml() {
|
|
|
|
return this.clientAssetBundle.readTextFileSync(".client/index.html")
|
|
|
|
.replaceAll(
|
|
|
|
"{{SPACE_PATH}}",
|
|
|
|
this.options.pagesPath.replaceAll("\\", "\\\\"),
|
|
|
|
).replaceAll(
|
|
|
|
"{{SYNC_ENDPOINT}}",
|
|
|
|
"/.fs",
|
|
|
|
);
|
2022-10-10 12:50:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
async start() {
|
2023-05-23 18:53:53 +00:00
|
|
|
await ensureSettingsAndIndex(this.spacePrimitives);
|
2022-12-14 19:32:26 +00:00
|
|
|
|
2022-10-10 12:50:21 +00:00
|
|
|
// Serve static files (javascript, css, html)
|
|
|
|
this.app.use(async ({ request, response }, next) => {
|
|
|
|
if (request.url.pathname === "/") {
|
2023-05-23 18:53:53 +00:00
|
|
|
// Note: we're explicitly not setting Last-Modified and If-Modified-Since header here because this page is dynamic
|
2022-10-10 12:50:21 +00:00
|
|
|
response.headers.set("Content-type", "text/html");
|
2023-05-23 18:53:53 +00:00
|
|
|
response.body = this.renderIndexHtml();
|
2022-10-10 12:50:21 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
try {
|
2023-05-23 18:53:53 +00:00
|
|
|
const assetName = request.url.pathname.slice(1);
|
2022-10-21 14:56:46 +00:00
|
|
|
if (
|
2023-05-23 18:53:53 +00:00
|
|
|
this.clientAssetBundle.has(assetName) &&
|
|
|
|
request.headers.get("If-Modified-Since") ===
|
|
|
|
utcDateString(this.clientAssetBundle.getMtime(assetName))
|
2022-10-21 14:56:46 +00:00
|
|
|
) {
|
|
|
|
response.status = 304;
|
|
|
|
return;
|
|
|
|
}
|
2022-10-10 12:50:21 +00:00
|
|
|
response.status = 200;
|
|
|
|
response.headers.set(
|
|
|
|
"Content-type",
|
2023-05-23 18:53:53 +00:00
|
|
|
this.clientAssetBundle.getMimeType(assetName),
|
2022-10-10 12:50:21 +00:00
|
|
|
);
|
2023-05-23 18:53:53 +00:00
|
|
|
const data = this.clientAssetBundle.readFileSync(
|
2022-10-12 09:47:13 +00:00
|
|
|
assetName,
|
2022-10-10 12:50:21 +00:00
|
|
|
);
|
2022-10-21 14:56:46 +00:00
|
|
|
response.headers.set("Cache-Control", "no-cache");
|
2022-10-12 09:47:13 +00:00
|
|
|
response.headers.set("Content-length", "" + data.length);
|
2023-05-23 18:53:53 +00:00
|
|
|
response.headers.set(
|
|
|
|
"Last-Modified",
|
|
|
|
utcDateString(this.clientAssetBundle.getMtime(assetName)),
|
|
|
|
);
|
2022-10-12 09:47:13 +00:00
|
|
|
|
2022-10-10 12:50:21 +00:00
|
|
|
if (request.method === "GET") {
|
2022-10-12 09:47:13 +00:00
|
|
|
response.body = data;
|
2022-10-10 12:50:21 +00:00
|
|
|
}
|
|
|
|
} catch {
|
|
|
|
await next();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2023-05-23 18:53:53 +00:00
|
|
|
// Fallback, serve index.html
|
|
|
|
this.app.use(({ request, response }, next) => {
|
|
|
|
if (
|
|
|
|
!request.url.pathname.startsWith("/.fs") &&
|
|
|
|
request.url.pathname !== "/.auth"
|
|
|
|
) {
|
|
|
|
response.headers.set("Content-type", "text/html");
|
|
|
|
response.body = this.renderIndexHtml();
|
|
|
|
} else {
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2022-10-10 12:50:21 +00:00
|
|
|
// Pages API
|
2023-05-23 18:53:53 +00:00
|
|
|
const fsRouter = this.buildFsRouter(this.spacePrimitives);
|
|
|
|
this.addPasswordAuth(this.app);
|
2022-10-10 12:50:21 +00:00
|
|
|
this.app.use(fsRouter.routes());
|
|
|
|
this.app.use(fsRouter.allowedMethods());
|
|
|
|
|
|
|
|
this.abortController = new AbortController();
|
2023-05-23 18:53:53 +00:00
|
|
|
const listenOptions: any = {
|
2022-12-14 19:32:26 +00:00
|
|
|
hostname: this.hostname,
|
|
|
|
port: this.port,
|
|
|
|
signal: this.abortController.signal,
|
2023-05-23 18:53:53 +00:00
|
|
|
};
|
|
|
|
if (this.options.keyFile) {
|
|
|
|
listenOptions.key = Deno.readTextFileSync(this.options.keyFile);
|
|
|
|
}
|
|
|
|
if (this.options.certFile) {
|
|
|
|
listenOptions.cert = Deno.readTextFileSync(this.options.certFile);
|
|
|
|
}
|
|
|
|
this.app.listen(listenOptions)
|
2022-11-25 12:08:59 +00:00
|
|
|
.catch((e: any) => {
|
|
|
|
console.log("Server listen error:", e.message);
|
|
|
|
Deno.exit(1);
|
|
|
|
});
|
2022-12-14 19:32:26 +00:00
|
|
|
const visibleHostname = this.hostname === "0.0.0.0"
|
|
|
|
? "localhost"
|
|
|
|
: this.hostname;
|
2022-10-10 12:50:21 +00:00
|
|
|
console.log(
|
2023-01-16 15:45:55 +00:00
|
|
|
`SilverBullet is now running: http://${visibleHostname}:${this.port}`,
|
2022-10-10 12:50:21 +00:00
|
|
|
);
|
2022-11-26 13:15:38 +00:00
|
|
|
}
|
|
|
|
|
2022-12-05 11:14:21 +00:00
|
|
|
private addPasswordAuth(app: Application) {
|
2022-12-22 10:21:12 +00:00
|
|
|
const excludedPaths = [
|
|
|
|
"/manifest.json",
|
|
|
|
"/favicon.png",
|
|
|
|
"/logo.png",
|
|
|
|
"/.auth",
|
|
|
|
];
|
2022-12-05 11:14:21 +00:00
|
|
|
if (this.user) {
|
2022-12-22 10:21:12 +00:00
|
|
|
const b64User = btoa(this.user);
|
|
|
|
app.use(async ({ request, response, cookies }, next) => {
|
2022-12-15 11:59:31 +00:00
|
|
|
if (!excludedPaths.includes(request.url.pathname)) {
|
2022-12-22 10:21:12 +00:00
|
|
|
const authCookie = await cookies.get("auth");
|
|
|
|
if (!authCookie || authCookie !== b64User) {
|
2023-05-23 18:53:53 +00:00
|
|
|
response.status = 401;
|
|
|
|
response.body = "Unauthorized, please authenticate";
|
2022-12-22 10:21:12 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (request.url.pathname === "/.auth") {
|
|
|
|
if (request.method === "GET") {
|
|
|
|
response.headers.set("Content-type", "text/html");
|
2023-05-23 18:53:53 +00:00
|
|
|
response.body = this.clientAssetBundle.readTextFileSync(
|
|
|
|
".client/auth.html",
|
2022-12-22 10:21:12 +00:00
|
|
|
);
|
|
|
|
return;
|
|
|
|
} else if (request.method === "POST") {
|
|
|
|
const values = await request.body({ type: "form" }).value;
|
|
|
|
const username = values.get("username"),
|
|
|
|
password = values.get("password"),
|
|
|
|
refer = values.get("refer");
|
|
|
|
if (this.user === `${username}:${password}`) {
|
|
|
|
await cookies.set("auth", b64User, {
|
|
|
|
expires: new Date(Date.now() + 1000 * 60 * 60 * 24 * 7), // in a week
|
|
|
|
sameSite: "strict",
|
|
|
|
});
|
|
|
|
response.redirect(refer || "/");
|
|
|
|
// console.log("All headers", request.headers);
|
|
|
|
} else {
|
|
|
|
response.redirect("/.auth?error=1");
|
|
|
|
}
|
|
|
|
return;
|
2022-12-15 11:59:31 +00:00
|
|
|
} else {
|
|
|
|
response.status = 401;
|
|
|
|
response.body = "Unauthorized";
|
2022-12-22 10:21:12 +00:00
|
|
|
return;
|
2022-12-15 11:59:31 +00:00
|
|
|
}
|
2022-10-17 18:35:38 +00:00
|
|
|
} else {
|
2022-12-15 11:59:31 +00:00
|
|
|
// Unauthenticated access to excluded paths
|
|
|
|
await next();
|
2022-10-17 18:35:38 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private buildFsRouter(spacePrimitives: SpacePrimitives): Router {
|
|
|
|
const fsRouter = new Router();
|
|
|
|
// File list
|
|
|
|
fsRouter.get("/", async ({ response }) => {
|
|
|
|
response.headers.set("Content-type", "application/json");
|
2023-05-23 18:53:53 +00:00
|
|
|
response.headers.set("X-Space-Path", this.options.pagesPath);
|
2023-01-13 14:41:29 +00:00
|
|
|
const files = await spacePrimitives.fetchFileList();
|
|
|
|
response.body = JSON.stringify(files);
|
2022-10-17 18:35:38 +00:00
|
|
|
});
|
|
|
|
|
2023-05-23 18:53:53 +00:00
|
|
|
// RPC
|
|
|
|
fsRouter.post("/", async ({ request, response }) => {
|
|
|
|
const body = await request.body({ type: "json" }).value;
|
|
|
|
try {
|
|
|
|
switch (body.operation) {
|
|
|
|
case "fetch": {
|
|
|
|
const result = await performLocalFetch(body.url, body.options);
|
2023-05-24 04:47:39 +00:00
|
|
|
console.log("Proxying fetch request to", body.url);
|
2023-05-23 18:53:53 +00:00
|
|
|
response.headers.set("Content-Type", "application/json");
|
|
|
|
response.body = JSON.stringify(result);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
case "shell": {
|
|
|
|
// TODO: Have a nicer way to do this
|
|
|
|
if (this.options.pagesPath.startsWith("s3://")) {
|
|
|
|
response.status = 500;
|
|
|
|
response.body = JSON.stringify({
|
|
|
|
stdout: "",
|
|
|
|
stderr: "Cannot run shell commands with S3 backend",
|
|
|
|
code: 500,
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const p = new Deno.Command(body.cmd, {
|
|
|
|
args: body.args,
|
|
|
|
cwd: this.options.pagesPath,
|
|
|
|
stdout: "piped",
|
|
|
|
stderr: "piped",
|
|
|
|
});
|
|
|
|
const output = await p.output();
|
|
|
|
const stdout = new TextDecoder().decode(output.stdout);
|
|
|
|
const stderr = new TextDecoder().decode(output.stderr);
|
|
|
|
|
|
|
|
response.headers.set("Content-Type", "application/json");
|
|
|
|
response.body = JSON.stringify({
|
|
|
|
stdout,
|
|
|
|
stderr,
|
|
|
|
code: output.code,
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
response.headers.set("Content-Type", "text/plain");
|
|
|
|
response.status = 400;
|
|
|
|
response.body = "Unknown operation";
|
|
|
|
}
|
|
|
|
} catch (e: any) {
|
|
|
|
console.log("Error", e);
|
|
|
|
response.status = 500;
|
|
|
|
response.body = e.message;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2022-10-17 18:35:38 +00:00
|
|
|
fsRouter
|
2022-10-21 14:56:46 +00:00
|
|
|
.get("\/(.+)", async ({ params, response, request }) => {
|
2022-10-17 18:35:38 +00:00
|
|
|
const name = params[0];
|
2023-05-23 18:53:53 +00:00
|
|
|
console.log("Loading file", name);
|
2022-10-17 18:35:38 +00:00
|
|
|
try {
|
|
|
|
const attachmentData = await spacePrimitives.readFile(
|
|
|
|
name,
|
|
|
|
);
|
2022-10-21 14:56:46 +00:00
|
|
|
const lastModifiedHeader = new Date(attachmentData.meta.lastModified)
|
|
|
|
.toUTCString();
|
|
|
|
if (request.headers.get("If-Modified-Since") === lastModifiedHeader) {
|
|
|
|
response.status = 304;
|
|
|
|
return;
|
|
|
|
}
|
2022-10-17 18:35:38 +00:00
|
|
|
response.status = 200;
|
|
|
|
response.headers.set(
|
|
|
|
"X-Last-Modified",
|
|
|
|
"" + attachmentData.meta.lastModified,
|
|
|
|
);
|
2022-10-21 14:56:46 +00:00
|
|
|
response.headers.set("Cache-Control", "no-cache");
|
2022-10-17 18:35:38 +00:00
|
|
|
response.headers.set("X-Permission", attachmentData.meta.perm);
|
2022-10-21 14:56:46 +00:00
|
|
|
response.headers.set(
|
|
|
|
"Last-Modified",
|
|
|
|
lastModifiedHeader,
|
|
|
|
);
|
2022-10-17 18:35:38 +00:00
|
|
|
response.headers.set("Content-Type", attachmentData.meta.contentType);
|
2023-05-23 18:53:53 +00:00
|
|
|
response.body = attachmentData.data;
|
2022-10-17 18:35:38 +00:00
|
|
|
} catch {
|
|
|
|
// console.error("Error in main router", e);
|
|
|
|
response.status = 404;
|
|
|
|
response.body = "";
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.put("\/(.+)", async ({ request, response, params }) => {
|
|
|
|
const name = params[0];
|
|
|
|
console.log("Saving file", name);
|
|
|
|
|
2023-01-13 14:41:29 +00:00
|
|
|
let body: Uint8Array;
|
|
|
|
if (
|
|
|
|
request.headers.get("X-Content-Base64")
|
|
|
|
) {
|
|
|
|
const content = await request.body({ type: "text" }).value;
|
|
|
|
body = base64Decode(content);
|
|
|
|
} else {
|
|
|
|
body = await request.body({ type: "bytes" }).value;
|
|
|
|
}
|
|
|
|
|
2022-10-17 18:35:38 +00:00
|
|
|
try {
|
|
|
|
const meta = await spacePrimitives.writeFile(
|
|
|
|
name,
|
2023-01-13 14:41:29 +00:00
|
|
|
body,
|
2022-10-17 18:35:38 +00:00
|
|
|
);
|
|
|
|
response.status = 200;
|
|
|
|
response.headers.set("Content-Type", meta.contentType);
|
|
|
|
response.headers.set("X-Last-Modified", "" + meta.lastModified);
|
|
|
|
response.headers.set("X-Content-Length", "" + meta.size);
|
|
|
|
response.headers.set("X-Permission", meta.perm);
|
|
|
|
response.body = "OK";
|
|
|
|
} catch (err) {
|
|
|
|
response.status = 500;
|
|
|
|
response.body = "Write failed";
|
|
|
|
console.error("Pipeline failed", err);
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.options("\/(.+)", async ({ response, params }) => {
|
|
|
|
const name = params[0];
|
|
|
|
try {
|
|
|
|
const meta = await spacePrimitives.getFileMeta(name);
|
|
|
|
response.status = 200;
|
|
|
|
response.headers.set("Content-Type", meta.contentType);
|
|
|
|
response.headers.set("X-Last-Modified", "" + meta.lastModified);
|
|
|
|
response.headers.set("X-Content-Length", "" + meta.size);
|
|
|
|
response.headers.set("X-Permission", meta.perm);
|
|
|
|
} catch {
|
|
|
|
response.status = 404;
|
2023-05-23 18:53:53 +00:00
|
|
|
response.body = "Not found";
|
2022-10-17 18:35:38 +00:00
|
|
|
// console.error("Options failed", err);
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.delete("\/(.+)", async ({ response, params }) => {
|
|
|
|
const name = params[0];
|
|
|
|
try {
|
|
|
|
await spacePrimitives.deleteFile(name);
|
|
|
|
response.status = 200;
|
|
|
|
response.body = "OK";
|
|
|
|
} catch (e: any) {
|
|
|
|
console.error("Error deleting attachment", e);
|
|
|
|
response.status = 200;
|
|
|
|
response.body = e.message;
|
|
|
|
}
|
|
|
|
});
|
2023-05-23 18:53:53 +00:00
|
|
|
return new Router().use("/.fs", fsRouter.routes());
|
2022-10-10 12:50:21 +00:00
|
|
|
}
|
|
|
|
|
2023-05-23 18:53:53 +00:00
|
|
|
stop() {
|
2022-10-10 12:50:21 +00:00
|
|
|
if (this.abortController) {
|
|
|
|
this.abortController.abort();
|
|
|
|
console.log("stopped server");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-05-23 18:53:53 +00:00
|
|
|
|
|
|
|
function utcDateString(mtime: number): string {
|
|
|
|
return new Date(mtime).toUTCString();
|
|
|
|
}
|