feat(node): add TypeScript sibling project for npm-based install
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>
This commit is contained in:
104
node/src/autostart/linux.ts
Normal file
104
node/src/autostart/linux.ts
Normal file
@@ -0,0 +1,104 @@
|
||||
import { existsSync, mkdirSync, writeFileSync, unlinkSync, rmSync, readFileSync } from "node:fs";
|
||||
import { join, dirname } from "node:path";
|
||||
import { homedir } from "node:os";
|
||||
import { run, cliEntry, type AutostartManager, type AutostartInstallOpts } from "./index.js";
|
||||
import { userConfigPath } from "../config.js";
|
||||
|
||||
const UNIT_NAME = "claude-mailbox.service";
|
||||
|
||||
function unitPath(): string {
|
||||
const xdg = process.env["XDG_CONFIG_HOME"] || join(homedir(), ".config");
|
||||
return join(xdg, "systemd", "user", UNIT_NAME);
|
||||
}
|
||||
|
||||
function ensureConfigSeeded(opts: AutostartInstallOpts): string {
|
||||
const path = userConfigPath();
|
||||
if (!existsSync(path)) {
|
||||
mkdirSync(dirname(path), { recursive: true });
|
||||
const seed: Record<string, unknown> = {};
|
||||
if (opts.port !== undefined) seed.port = opts.port;
|
||||
if (opts.bind !== undefined) seed.bind = opts.bind;
|
||||
if (opts.dbPath !== undefined) seed.dbPath = opts.dbPath;
|
||||
writeFileSync(path, JSON.stringify(seed, null, 2) + "\n", "utf8");
|
||||
}
|
||||
return path;
|
||||
}
|
||||
|
||||
function shellQuote(s: string): string {
|
||||
return `'${s.replace(/'/g, `'\\''`)}'`;
|
||||
}
|
||||
|
||||
function buildUnit(node: string, script: string, configPath: string): string {
|
||||
const exec = `${shellQuote(node)} ${shellQuote(script)} serve --config ${shellQuote(configPath)}`;
|
||||
return `[Unit]
|
||||
Description=ClaudeMailbox MCP mail daemon
|
||||
After=network.target
|
||||
|
||||
[Service]
|
||||
Type=simple
|
||||
ExecStart=${exec}
|
||||
Restart=on-failure
|
||||
RestartSec=2
|
||||
|
||||
[Install]
|
||||
WantedBy=default.target
|
||||
`;
|
||||
}
|
||||
|
||||
function systemctl(args: string[]): { status: number; stdout: string; stderr: string } {
|
||||
return run("systemctl", ["--user", ...args]);
|
||||
}
|
||||
|
||||
export function linuxManager(): AutostartManager {
|
||||
return {
|
||||
mode: "default",
|
||||
async install(opts) {
|
||||
const configPath = ensureConfigSeeded(opts);
|
||||
const { node, script } = cliEntry();
|
||||
const path = unitPath();
|
||||
mkdirSync(dirname(path), { recursive: true });
|
||||
writeFileSync(path, buildUnit(node, script, configPath), "utf8");
|
||||
const reload = systemctl(["daemon-reload"]);
|
||||
if (reload.status !== 0) {
|
||||
throw new Error(`systemctl daemon-reload failed: ${reload.stderr || reload.stdout}`);
|
||||
}
|
||||
const enable = systemctl(["enable", "--now", UNIT_NAME]);
|
||||
if (enable.status !== 0) {
|
||||
throw new Error(`systemctl enable --now failed: ${enable.stderr || enable.stdout}`);
|
||||
}
|
||||
},
|
||||
async uninstall(purge) {
|
||||
systemctl(["disable", "--now", UNIT_NAME]);
|
||||
const path = unitPath();
|
||||
if (existsSync(path)) unlinkSync(path);
|
||||
systemctl(["daemon-reload"]);
|
||||
if (purge) {
|
||||
const cfg = userConfigPath();
|
||||
if (existsSync(cfg)) {
|
||||
try {
|
||||
const parsed = JSON.parse(readFileSync(cfg, "utf8")) as { dbPath?: string };
|
||||
if (parsed.dbPath && existsSync(parsed.dbPath)) {
|
||||
rmSync(parsed.dbPath, { force: true });
|
||||
}
|
||||
} catch {
|
||||
// ignore
|
||||
}
|
||||
unlinkSync(cfg);
|
||||
}
|
||||
}
|
||||
},
|
||||
async start() {
|
||||
const r = systemctl(["start", UNIT_NAME]);
|
||||
if (r.status !== 0) throw new Error(`systemctl start failed: ${r.stderr || r.stdout}`);
|
||||
},
|
||||
async stop() {
|
||||
systemctl(["stop", UNIT_NAME]);
|
||||
},
|
||||
async status() {
|
||||
if (!existsSync(unitPath())) return "NotInstalled";
|
||||
const r = systemctl(["is-active", UNIT_NAME]);
|
||||
if (r.stdout.trim() === "active") return "Running";
|
||||
return "Stopped";
|
||||
},
|
||||
};
|
||||
}
|
||||
Reference in New Issue
Block a user