Introduces @kuns/claude-mailbox under node/, a wire-compatible TypeScript port of the .NET daemon that distributes via the public Gitea npm registry. The .NET project stays in src/ClaudeMailbox/ untouched; users pick whichever flavor they prefer. - node/ project: fastify + @modelcontextprotocol/sdk StreamableHTTPServerTransport + better-sqlite3, schema and wire surface match the C# version (port 47822, X-Mailbox header, MCP tool names, snake_case SQLite columns) - Cross-platform autostart: Scheduled Task (Win, no admin) / Windows Service (Win, --service) / launchd (mac) / systemd --user (linux) - 9/9 vitest tests pass; end-to-end /health + send/check round-trip verified - CI split: existing ci.yml/release.yml renamed to *-dotnet.yml with path filters, new ci-node.yml and release-node.yml publish to Gitea npm registry - install.ps1 / install.sh bootstrap one-liners at repo root; homebrew/ contains a tap formula template - README install section reordered: npm path primary, dotnet publish secondary Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
60 lines
1.6 KiB
TypeScript
60 lines
1.6 KiB
TypeScript
import { spawnSync, type SpawnSyncOptionsWithStringEncoding } from "node:child_process";
|
|
import { fileURLToPath } from "node:url";
|
|
import { dirname, resolve } from "node:path";
|
|
|
|
export interface AutostartInstallOpts {
|
|
port?: number;
|
|
bind?: string;
|
|
dbPath?: string;
|
|
}
|
|
|
|
export interface AutostartManager {
|
|
readonly mode: "default" | "service";
|
|
install(opts: AutostartInstallOpts): Promise<void>;
|
|
uninstall(purge: boolean): Promise<void>;
|
|
start(): Promise<void>;
|
|
stop(): Promise<void>;
|
|
status(): Promise<"Running" | "Stopped" | "NotInstalled">;
|
|
}
|
|
|
|
export interface RunResult {
|
|
status: number;
|
|
stdout: string;
|
|
stderr: string;
|
|
}
|
|
|
|
export function run(file: string, args: string[]): RunResult {
|
|
const opts: SpawnSyncOptionsWithStringEncoding = {
|
|
encoding: "utf8",
|
|
stdio: ["ignore", "pipe", "pipe"],
|
|
shell: false,
|
|
};
|
|
const r = spawnSync(file, args, opts);
|
|
return {
|
|
status: r.status ?? -1,
|
|
stdout: typeof r.stdout === "string" ? r.stdout : "",
|
|
stderr: typeof r.stderr === "string" ? r.stderr : "",
|
|
};
|
|
}
|
|
|
|
export function cliEntry(): { node: string; script: string } {
|
|
const here = dirname(fileURLToPath(import.meta.url));
|
|
return {
|
|
node: process.execPath,
|
|
script: resolve(here, "..", "cli.js"),
|
|
};
|
|
}
|
|
|
|
export async function autostartManager(mode: "default" | "service" = "default"): Promise<AutostartManager> {
|
|
if (process.platform === "win32") {
|
|
const mod = await import("./windows.js");
|
|
return mod.windowsManager(mode);
|
|
}
|
|
if (process.platform === "darwin") {
|
|
const mod = await import("./darwin.js");
|
|
return mod.darwinManager();
|
|
}
|
|
const mod = await import("./linux.js");
|
|
return mod.linuxManager();
|
|
}
|