mirror of
https://github.com/chatmail/core.git
synced 2026-04-17 21:46:35 +03:00
164 lines
4.9 KiB
JavaScript
164 lines
4.9 KiB
JavaScript
//@ts-check
|
|
import { execFile, spawn } from "node:child_process";
|
|
import { stat, readdir } from "node:fs/promises";
|
|
import os from "node:os";
|
|
import { join, basename } from "node:path";
|
|
import process from "node:process";
|
|
import { promisify } from "node:util";
|
|
import {
|
|
ENV_VAR_NAME,
|
|
PATH_EXECUTABLE_NAME,
|
|
SKIP_SEARCH_IN_PATH,
|
|
} from "./src/const.js";
|
|
import {
|
|
ENV_VAR_LOCATION_NOT_FOUND,
|
|
FAILED_TO_START_SERVER_EXECUTABLE,
|
|
NPM_NOT_FOUND_SUPPORTED_PLATFORM_ERROR,
|
|
NPM_NOT_FOUND_UNSUPPORTED_PLATFORM_ERROR,
|
|
} from "./src/errors.js";
|
|
|
|
// Because this is not compiled by typescript, esm needs this stuff (` with { type: "json" };`,
|
|
// nodejs still complains about it being experimental, but deno also uses it, so treefit bets taht it will become standard)
|
|
import package_json from "./package.json" with { type: "json" };
|
|
import { createRequire } from "node:module";
|
|
|
|
function findRPCServerInNodeModules() {
|
|
const arch = os.arch();
|
|
const operating_system = process.platform;
|
|
const package_name = `@deltachat/stdio-rpc-server-${operating_system}-${arch}`;
|
|
try {
|
|
const { resolve } = createRequire(import.meta.url);
|
|
return resolve(package_name);
|
|
} catch (error) {
|
|
console.debug("findRpcServerInNodeModules", error);
|
|
if (Object.keys(package_json.optionalDependencies).includes(package_name)) {
|
|
throw new Error(NPM_NOT_FOUND_SUPPORTED_PLATFORM_ERROR(package_name));
|
|
} else {
|
|
throw new Error(NPM_NOT_FOUND_UNSUPPORTED_PLATFORM_ERROR());
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @returns {Promise<string>}
|
|
*/
|
|
async function getLocationInPath() {
|
|
const exec = promisify(execFile);
|
|
|
|
if (os.platform() === "win32") {
|
|
const { stdout: executable } = await exec("where", [PATH_EXECUTABLE_NAME], {
|
|
shell: true,
|
|
});
|
|
return executable;
|
|
}
|
|
|
|
try {
|
|
const { stdout: executable } = await exec(
|
|
"command",
|
|
["-v", PATH_EXECUTABLE_NAME],
|
|
{ shell: true }
|
|
);
|
|
return executable;
|
|
} catch (error) {
|
|
if (error.code > 0) return "";
|
|
else throw error;
|
|
}
|
|
}
|
|
|
|
/** @type {import("./index").FnTypes.getRPCServerPath} */
|
|
export async function getRPCServerPath(
|
|
options = { skipSearchInPath: false, disableEnvPath: false }
|
|
) {
|
|
// @TODO: improve confusing naming of these options
|
|
const { skipSearchInPath, disableEnvPath } = options;
|
|
// 1. check if it is set as env var
|
|
if (process.env[ENV_VAR_NAME] && !disableEnvPath) {
|
|
try {
|
|
if (!(await stat(process.env[ENV_VAR_NAME])).isFile()) {
|
|
throw new Error(
|
|
`expected ${ENV_VAR_NAME} to point to the deltachat-rpc-server executable`
|
|
);
|
|
}
|
|
} catch (error) {
|
|
throw new Error(ENV_VAR_LOCATION_NOT_FOUND());
|
|
}
|
|
return process.env[ENV_VAR_NAME];
|
|
}
|
|
|
|
// 2. check if it can be found in PATH
|
|
if (!process.env[SKIP_SEARCH_IN_PATH] && !skipSearchInPath) {
|
|
const executable = await getLocationInPath();
|
|
|
|
// by just trying to execute it and then use "command -v deltachat-rpc-server" (unix) or "where deltachat-rpc-server" (windows) to get the path to the executable
|
|
if (executable.length > 1) {
|
|
// test if it is the right version
|
|
try {
|
|
// for some unknown reason it is in stderr and not in stdout
|
|
const { stderr } = await promisify(execFile)(
|
|
executable,
|
|
["--version"],
|
|
{ shell: true }
|
|
);
|
|
const version = stderr.slice(0, stderr.indexOf("\n"));
|
|
if (package_json.version !== version) {
|
|
throw new Error(
|
|
`version mismatch: (npm package: ${package_json.version}) (installed ${PATH_EXECUTABLE_NAME} version: ${version})`
|
|
);
|
|
} else {
|
|
return executable;
|
|
}
|
|
} catch (error) {
|
|
console.error(
|
|
"Found executable in PATH, but there was an error: " + error
|
|
);
|
|
console.error("So falling back to using prebuild...");
|
|
}
|
|
}
|
|
}
|
|
// 3. check for prebuilds
|
|
|
|
return findRPCServerInNodeModules();
|
|
}
|
|
|
|
import { StdioDeltaChat } from "@deltachat/jsonrpc-client";
|
|
|
|
/** @type {import("./index").FnTypes.startDeltaChat} */
|
|
export async function startDeltaChat(directory, options) {
|
|
const pathToServerBinary = await getRPCServerPath(options);
|
|
const server = spawn(pathToServerBinary, {
|
|
env: {
|
|
RUST_LOG: process.env.RUST_LOG || "info",
|
|
DC_ACCOUNTS_PATH: directory,
|
|
},
|
|
stdio: ["pipe", "pipe", options.muteStdErr ? "ignore" : "inherit"],
|
|
});
|
|
|
|
server.on("error", (err) => {
|
|
throw new Error(FAILED_TO_START_SERVER_EXECUTABLE(pathToServerBinary, err));
|
|
});
|
|
let shouldClose = false;
|
|
|
|
server.on("exit", () => {
|
|
if (shouldClose) {
|
|
return;
|
|
}
|
|
throw new Error("Server quit");
|
|
});
|
|
|
|
/** @type {import('./index').DeltaChatOverJsonRpcServer} */
|
|
//@ts-expect-error
|
|
const dc = new StdioDeltaChat(server.stdin, server.stdout, true);
|
|
|
|
dc.close = () => {
|
|
shouldClose = true;
|
|
if (!server.kill()) {
|
|
console.log("server termination failed");
|
|
}
|
|
};
|
|
|
|
//@ts-expect-error
|
|
dc.pathToServerBinary = pathToServerBinary;
|
|
|
|
return dc;
|
|
}
|