Sidecar lifecycle:
- Move startSideCar() out of register() into an api.on("gateway_start", ...)
handler. register() runs in every CLI subprocess that loads plugins
(e.g. `openclaw completion`, `openclaw doctor`); eagerly spawning a
long-lived process there hung `openclaw update`'s post-update steps.
- Spawn the sidecar with detached: true, stdio routed to a log file fd,
and call .unref() so the host's event loop is never held by the child.
Even if a future caller invokes startSideCar in a non-gateway context,
it can no longer block that host from exiting.
- Sidecar logs now go to ~/.openclaw/logs/dirigent-sidecar.log instead of
being piped through the host logger.
Plugin SDK convention update:
- Wrap default export with definePluginEntry({ id, name, description, register })
per the current openclaw plugin authoring contract.
- Switch all imports from the deprecated root barrel "openclaw/plugin-sdk"
to focused subpaths "openclaw/plugin-sdk/core" and
"openclaw/plugin-sdk/plugin-entry".
- Modernize openclaw.plugin.json: drop entry/version, add description,
declare contracts.tools[] for the 6 tools, set activation.onStartup: true
so gateway_start fires for this plugin at boot.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
113 lines
3.6 KiB
TypeScript
113 lines
3.6 KiB
TypeScript
import type { OpenClawPluginApi } from "openclaw/plugin-sdk/core";
|
|
import type { ChannelStore } from "../core/channel-store.js";
|
|
|
|
type Deps = {
|
|
api: OpenClawPluginApi;
|
|
channelStore: ChannelStore;
|
|
moderatorBotUserId: string | undefined;
|
|
scheduleIdentifier: string;
|
|
moderatorServiceUrl: string | undefined;
|
|
moderatorServiceToken: string | undefined;
|
|
debugMode: boolean;
|
|
onNewMessage: (event: {
|
|
channelId: string;
|
|
messageId: string;
|
|
senderId: string;
|
|
guildId?: string;
|
|
}) => Promise<void>;
|
|
};
|
|
|
|
function sendJson(res: import("node:http").ServerResponse, status: number, payload: unknown): void {
|
|
res.writeHead(status, { "Content-Type": "application/json; charset=utf-8" });
|
|
res.end(JSON.stringify(payload));
|
|
}
|
|
|
|
function readBody(req: import("node:http").IncomingMessage): Promise<Record<string, unknown>> {
|
|
return new Promise((resolve, reject) => {
|
|
let body = "";
|
|
req.on("data", (chunk: Buffer) => {
|
|
body += chunk.toString();
|
|
if (body.length > 1_000_000) {
|
|
req.destroy();
|
|
reject(new Error("body too large"));
|
|
}
|
|
});
|
|
req.on("end", () => {
|
|
try {
|
|
resolve(body ? (JSON.parse(body) as Record<string, unknown>) : {});
|
|
} catch {
|
|
reject(new Error("invalid_json"));
|
|
}
|
|
});
|
|
req.on("error", reject);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Register Dirigent plugin HTTP routes that the moderator service calls back into.
|
|
*
|
|
* Routes:
|
|
* POST /dirigent/api/moderator/message — inbound message notification from moderator service
|
|
* GET /dirigent/api/moderator/status — health/status check
|
|
*/
|
|
export function registerDirigentApi(deps: Deps): void {
|
|
const { api, moderatorServiceUrl, onNewMessage } = deps;
|
|
|
|
// ── POST /dirigent/api/moderator/message ─────────────────────────────────────
|
|
// Called by the moderator service on every Discord MESSAGE_CREATE event.
|
|
api.registerHttpRoute({
|
|
path: "/dirigent/api/moderator/message",
|
|
auth: "plugin",
|
|
match: "exact",
|
|
handler: async (req, res) => {
|
|
if (req.method !== "POST") {
|
|
res.writeHead(405);
|
|
res.end();
|
|
return;
|
|
}
|
|
|
|
let body: Record<string, unknown>;
|
|
try {
|
|
body = await readBody(req);
|
|
} catch (err) {
|
|
return sendJson(res, 400, { ok: false, error: String(err) });
|
|
}
|
|
|
|
const channelId = typeof body.channelId === "string" ? body.channelId : undefined;
|
|
const messageId = typeof body.messageId === "string" ? body.messageId : undefined;
|
|
const senderId = typeof body.senderId === "string" ? body.senderId : undefined;
|
|
const guildId = typeof body.guildId === "string" ? body.guildId : undefined;
|
|
|
|
if (!channelId || !senderId) {
|
|
return sendJson(res, 400, { ok: false, error: "channelId and senderId required" });
|
|
}
|
|
|
|
try {
|
|
await onNewMessage({
|
|
channelId,
|
|
messageId: messageId ?? "",
|
|
senderId,
|
|
guildId,
|
|
});
|
|
return sendJson(res, 200, { ok: true });
|
|
} catch (err) {
|
|
api.logger.warn(`dirigent: moderator/message handler error: ${String(err)}`);
|
|
return sendJson(res, 500, { ok: false, error: String(err) });
|
|
}
|
|
},
|
|
});
|
|
|
|
// ── GET /dirigent/api/moderator/status ───────────────────────────────────────
|
|
api.registerHttpRoute({
|
|
path: "/dirigent/api/moderator/status",
|
|
auth: "plugin",
|
|
match: "exact",
|
|
handler: (_req, res) => {
|
|
return sendJson(res, 200, {
|
|
ok: true,
|
|
moderatorServiceUrl: moderatorServiceUrl ?? null,
|
|
});
|
|
},
|
|
});
|
|
}
|