silverbullet/cmd/server.ts

159 lines
5.1 KiB
TypeScript
Raw Normal View History

2023-08-26 14:31:51 +08:00
import { Application, path } from "../server/deps.ts";
2022-10-24 19:51:26 +08:00
import { HttpServer } from "../server/http_server.ts";
import clientAssetBundle from "../dist/client_asset_bundle.json" assert {
type: "json",
};
import plugAssetBundle from "../dist/plug_asset_bundle.json" assert {
type: "json",
};
2022-10-24 19:51:26 +08:00
import { AssetBundle, AssetJson } from "../plugos/asset_bundle/bundle.ts";
import { AssetBundlePlugSpacePrimitives } from "../common/spaces/asset_bundle_space_primitives.ts";
import { DiskSpacePrimitives } from "../common/spaces/disk_space_primitives.ts";
import { SpacePrimitives } from "../common/spaces/space_primitives.ts";
import { S3SpacePrimitives } from "../server/spaces/s3_space_primitives.ts";
import { Authenticator } from "../server/auth.ts";
import { JSONKVStore } from "../plugos/lib/kv_store.json_file.ts";
2023-08-26 14:31:51 +08:00
import { ServerSystem } from "../server/server_system.ts";
2023-08-30 03:17:29 +08:00
import { sleep } from "$sb/lib/async.ts";
2023-08-26 14:31:51 +08:00
import { SilverBulletHooks } from "../common/manifest.ts";
import { System } from "../plugos/system.ts";
2023-08-30 23:25:54 +08:00
import { silverBulletDbFile } from "./constants.ts";
2022-10-24 19:51:26 +08:00
export async function serveCommand(
2023-08-26 14:31:51 +08:00
options: {
hostname?: string;
port?: number;
user?: string;
auth?: string;
cert?: string;
key?: string;
reindex?: boolean;
db?: string;
2023-09-07 18:38:20 +08:00
syncOnly?: boolean;
2023-08-26 14:31:51 +08:00
},
folder?: string,
) {
2023-08-12 02:37:13 +08:00
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;
2023-09-07 18:38:20 +08:00
const syncOnly = options.syncOnly || Deno.env.get("SB_SYNC_ONLY");
2023-08-30 23:25:54 +08:00
let dbFile = options.db || Deno.env.get("SB_DB_FILE") || silverBulletDbFile;
2023-08-26 14:31:51 +08:00
const app = new Application();
2022-10-24 19:51:26 +08:00
if (!folder) {
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);
}
}
2022-10-24 19:51:26 +08:00
console.log(
"Going to start SilverBullet binding to",
`${hostname}:${port}`,
2022-10-24 19:51:26 +08:00
);
2022-12-16 20:00:06 +08:00
if (hostname === "127.0.0.1") {
console.log(
`NOTE: SilverBullet will only be available locally (via http://localhost:${port}).
To allow outside connections, pass -L 0.0.0.0 as a flag, and put a TLS terminator on top.`,
2022-12-16 20:00:06 +08:00
);
}
let spacePrimitives: SpacePrimitives | undefined;
if (folder === "s3://") {
2023-05-29 15:53:49 +08:00
spacePrimitives = new S3SpacePrimitives({
accessKey: Deno.env.get("AWS_ACCESS_KEY_ID")!,
secretKey: Deno.env.get("AWS_SECRET_ACCESS_KEY")!,
endPoint: Deno.env.get("AWS_ENDPOINT")!,
region: Deno.env.get("AWS_REGION")!,
bucket: Deno.env.get("AWS_BUCKET")!,
});
console.log("Running in S3 mode");
2023-08-26 14:31:51 +08:00
folder = Deno.cwd();
} else {
2023-05-29 15:53:49 +08:00
// Regular disk mode
folder = path.resolve(Deno.cwd(), folder);
2023-05-29 15:53:49 +08:00
spacePrimitives = new DiskSpacePrimitives(folder);
}
2023-08-26 14:31:51 +08:00
2023-05-29 15:53:49 +08:00
spacePrimitives = new AssetBundlePlugSpacePrimitives(
spacePrimitives,
new AssetBundle(plugAssetBundle as AssetJson),
);
2022-10-24 19:51:26 +08:00
2023-08-26 14:31:51 +08:00
let system: System<SilverBulletHooks> | undefined;
2023-09-07 18:38:20 +08:00
// system = undefined in syncOnly mode (no PlugOS instance on the server)
if (!syncOnly) {
2023-08-30 03:17:29 +08:00
// Enable server-side processing
2023-08-26 14:31:51 +08:00
dbFile = path.resolve(folder, dbFile);
2023-08-30 03:17:29 +08:00
console.log(
`Running in server-processing mode, keeping state in ${dbFile}`,
);
2023-08-26 14:31:51 +08:00
const serverSystem = new ServerSystem(spacePrimitives, dbFile, app);
await serverSystem.init();
spacePrimitives = serverSystem.spacePrimitives;
system = serverSystem.system;
if (options.reindex) {
console.log("Reindexing space (requested via --reindex flag)");
2023-08-31 04:36:27 +08:00
serverSystem.system.loadedPlugs.get("index")!.invoke(
2023-08-26 14:31:51 +08:00
"reindexSpace",
[],
2023-08-31 04:36:27 +08:00
).catch(console.error);
2023-08-26 14:31:51 +08:00
}
}
const authStore = new JSONKVStore();
const authenticator = new Authenticator(authStore);
const flagUser = options.user ?? Deno.env.get("SB_USER");
if (flagUser) {
// If explicitly added via env/parameter, add on the fly
const [username, password] = flagUser.split(":");
await authenticator.register(username, password, ["admin"], "");
}
if (options.auth) {
// Load auth file
const authFile: string = options.auth;
console.log("Loading authentication credentials from", authFile);
await authStore.load(authFile);
(async () => {
// Asynchronously kick off file watcher
2023-08-26 14:31:51 +08:00
for await (const _event of Deno.watchFs(options.auth!)) {
console.log("Authentication file changed, reloading...");
await authStore.load(authFile);
}
})().catch(console.error);
}
const envAuth = Deno.env.get("SB_AUTH");
if (envAuth) {
console.log("Loading authentication from SB_AUTH");
authStore.loadString(envAuth);
}
2023-08-30 03:17:29 +08:00
const httpServer = new HttpServer(
spacePrimitives!,
app,
system,
{
hostname,
port: port,
pagesPath: folder!,
clientAssetBundle: new AssetBundle(clientAssetBundle as AssetJson),
authenticator,
keyFile: options.key,
certFile: options.cert,
},
);
await httpServer.start();
2023-08-26 14:31:51 +08:00
// Wait in an infinite loop (to keep the HTTP server running, only cancelable via Ctrl+C or other signal)
while (true) {
2023-08-26 14:31:51 +08:00
await sleep(10000);
}
2022-10-24 19:51:26 +08:00
}