silverbullet/cmd/server.ts

168 lines
4.8 KiB
TypeScript

import { HttpServer } from "../server/http_server.ts";
import clientAssetBundle from "../dist/client_asset_bundle.json" with {
type: "json",
};
import plugAssetBundle from "../dist/plug_asset_bundle.json" with {
type: "json",
};
import { AssetBundle, type AssetJson } from "../lib/asset_bundle/bundle.ts";
import { determineDatabaseBackend } from "../server/db_backend.ts";
import { runPlug } from "../cmd/plug_run.ts";
import { PrefixedKvPrimitives } from "$lib/data/prefixed_kv_primitives.ts";
import { sleep } from "$lib/async.ts";
export type AuthOptions = {
authToken?: string;
user: string;
pass: string;
lockoutTime: number;
lockoutLimit: number;
};
export async function serveCommand(
options: {
hostname?: string;
port?: number;
user?: string;
auth?: string;
cert?: string;
key?: string;
reindex?: boolean;
syncOnly?: boolean;
},
folder?: string,
) {
const hostname = options.hostname || Deno.env.get("SB_HOSTNAME") ||
"127.0.0.1";
const port = options.port ||
(Deno.env.get("SB_PORT") && +Deno.env.get("SB_PORT")!) || 3000;
const syncOnly = options.syncOnly || !!Deno.env.get("SB_SYNC_ONLY");
const readOnly = !!Deno.env.get("SB_READ_ONLY");
if (syncOnly) {
console.log("Running in sync-only mode (no backend processing)");
}
if (!folder) {
// Didn't get a folder as an argument, check if we got it as an environment variable
folder = Deno.env.get("SB_FOLDER");
if (!folder) {
console.error(
"No folder specified. Please pass a folder as an argument or set SB_FOLDER environment variable.",
);
Deno.exit(1);
}
}
const baseKvPrimitives = await determineDatabaseBackend(folder);
console.log(
"Going to start SilverBullet binding to",
`${hostname}:${port}`,
);
if (hostname === "127.0.0.1") {
console.info(
`SilverBullet will only be available locally, to allow outside connections, pass -L0.0.0.0 as a flag, and put a TLS terminator on top.`,
);
}
const userAuth = options.user ?? Deno.env.get("SB_USER");
let userCredentials: AuthOptions | undefined;
if (userAuth) {
const [user, pass] = userAuth.split(":");
userCredentials = {
user,
pass,
// 10 failed login attempts in 1 minute
lockoutLimit: 10,
lockoutTime: 60,
};
// Override lockout settings if they are set in the environment
if (Deno.env.get("SB_LOCKOUT_LIMIT")) {
userCredentials.lockoutLimit = Number(Deno.env.get("SB_LOCKOUT_LIMIT"));
}
if (Deno.env.get("SB_LOCKOUT_TIME")) {
userCredentials.lockoutTime = Number(Deno.env.get("SB_LOCKOUT_TIME"));
}
if (Deno.env.get("SB_AUTH_TOKEN")) {
userCredentials.authToken = Deno.env.get("SB_AUTH_TOKEN");
}
console.log(
`User authentication enabled for user "${user}" with lockout limit ${userCredentials.lockoutLimit} and lockout time ${userCredentials.lockoutTime}s`,
);
}
const backendConfig = Deno.env.get("SB_SHELL_BACKEND") || "local";
const enableSpaceScript = Deno.env.get("SB_SPACE_SCRIPT") !== "off";
const plugAssets = new AssetBundle(plugAssetBundle as AssetJson);
if (readOnly) {
console.log("Performing initial space indexing...");
await runPlug(
folder,
"index.reindexSpace",
[],
plugAssets,
new PrefixedKvPrimitives(baseKvPrimitives, ["*"]),
);
console.log(
"Now indexing again to make sure any additional space script indexers are run...",
);
await runPlug(
folder,
"index.reindexSpace",
[true], // noClear
plugAssets,
new PrefixedKvPrimitives(baseKvPrimitives, ["*"]),
);
}
const clientAssets = new AssetBundle(clientAssetBundle as AssetJson);
const manifestName = Deno.env.get("SB_NAME");
const manifestDescription = Deno.env.get("SB_DESCRIPTION");
if (manifestName || manifestDescription) {
const manifestData = JSON.parse(
clientAssets.readTextFileSync(".client/manifest.json"),
);
if (manifestName) {
manifestData.name = manifestData.short_name = manifestName;
}
if (manifestDescription) {
manifestData.description = manifestDescription;
}
clientAssets.writeTextFileSync(
".client/manifest.json",
"application/json",
JSON.stringify(manifestData),
);
}
const httpServer = new HttpServer({
hostname,
port,
clientAssetBundle: clientAssets,
plugAssetBundle: plugAssets,
baseKvPrimitives,
keyFile: options.key,
certFile: options.cert,
auth: userCredentials,
syncOnly,
readOnly,
shellBackend: backendConfig,
enableSpaceScript,
pagesPath: folder,
});
await httpServer.start();
// Wait in an infinite loop (to keep the HTTP server running, only cancelable via Ctrl+C or other signal)
while (true) {
await sleep(10000);
}
}