mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-06 19:00:45 +00:00
Simplify plugin installation and runtime loading around package-manager-owned dependencies, with Jiti reserved for local/TS fallback paths. Also scans npm plugin install roots so hoisted transitive dependencies are covered by dependency denylist and node_modules symlink checks.
202 lines
6.5 KiB
TypeScript
202 lines
6.5 KiB
TypeScript
import { readFile, writeFile } from "node:fs/promises";
|
|
import path from "node:path";
|
|
import type { MacosGuest } from "./guest-transports.ts";
|
|
import { run, say, shellQuote, warn } from "./host-command.ts";
|
|
|
|
export type DiscordSmokePhase = "fresh" | "upgrade";
|
|
|
|
export interface MacosDiscordConfig {
|
|
channelId: string;
|
|
guildId: string;
|
|
token: string;
|
|
}
|
|
|
|
export class MacosDiscordSmoke {
|
|
constructor(
|
|
private input: {
|
|
config: MacosDiscordConfig;
|
|
guest: MacosGuest;
|
|
guestNode: string;
|
|
guestOpenClaw: string;
|
|
guestOpenClawEntry: string;
|
|
runDir: string;
|
|
vmName: string;
|
|
},
|
|
) {}
|
|
|
|
configure(): void {
|
|
const guilds = JSON.stringify({
|
|
[this.input.config.guildId]: {
|
|
channels: {
|
|
[this.input.config.channelId]: {
|
|
enabled: true,
|
|
requireMention: false,
|
|
},
|
|
},
|
|
},
|
|
});
|
|
this.input.guest.sh(`set -eu
|
|
${this.input.guestNode} ${this.input.guestOpenClawEntry} config set channels.discord.token ${shellQuote(this.input.config.token)}
|
|
${this.input.guestNode} ${this.input.guestOpenClawEntry} config set channels.discord.enabled true
|
|
${this.input.guestNode} ${this.input.guestOpenClawEntry} config set channels.discord.groupPolicy allowlist
|
|
${this.input.guestNode} ${this.input.guestOpenClawEntry} config set channels.discord.guilds ${shellQuote(guilds)} --strict-json
|
|
${this.input.guestNode} ${this.input.guestOpenClawEntry} doctor --fix --yes --non-interactive
|
|
${this.input.guestNode} ${this.input.guestOpenClawEntry} gateway restart
|
|
${this.input.guestNode} ${this.input.guestOpenClawEntry} channels status --probe --json`);
|
|
}
|
|
|
|
async runRoundtrip(phase: DiscordSmokePhase): Promise<void> {
|
|
const nonce = `${Date.now()}-${Math.floor(Math.random() * 100000)}`;
|
|
const outboundNonce = `${phase}-out-${nonce}`;
|
|
const inboundNonce = `${phase}-in-${nonce}`;
|
|
const outboundLog = path.join(this.input.runDir, `${phase}.discord-send.json`);
|
|
const sentIdFile = path.join(this.input.runDir, `${phase}.discord-sent-message-id`);
|
|
const hostIdFile = path.join(this.input.runDir, `${phase}.discord-host-message-id`);
|
|
const outbound = this.input.guest.exec([
|
|
this.input.guestOpenClaw,
|
|
"message",
|
|
"send",
|
|
"--channel",
|
|
"discord",
|
|
"--target",
|
|
`channel:${this.input.config.channelId}`,
|
|
"--message",
|
|
`parallels-macos-smoke-outbound-${outboundNonce}`,
|
|
"--silent",
|
|
"--json",
|
|
]);
|
|
await writeFile(outboundLog, `${outbound}\n`, "utf8");
|
|
const sentId = this.discordMessageId(outbound);
|
|
await writeFile(sentIdFile, `${sentId}\n`, "utf8");
|
|
await this.waitForHostVisibility(outboundNonce, sentId);
|
|
const hostId = await this.postDiscordMessage(`parallels-macos-smoke-inbound-${inboundNonce}`);
|
|
await writeFile(hostIdFile, `${hostId}\n`, "utf8");
|
|
this.waitForGuestReadback(inboundNonce);
|
|
}
|
|
|
|
async cleanupMessages(): Promise<void> {
|
|
for (const name of [
|
|
"fresh.discord-sent-message-id",
|
|
"fresh.discord-host-message-id",
|
|
"upgrade.discord-sent-message-id",
|
|
"upgrade.discord-host-message-id",
|
|
]) {
|
|
const filePath = path.join(this.input.runDir, name);
|
|
const id = await readFile(filePath, "utf8").catch(() => "");
|
|
if (id.trim()) {
|
|
await this.discordApi(
|
|
"DELETE",
|
|
`/channels/${this.input.config.channelId}/messages/${id.trim()}`,
|
|
).catch(() => "");
|
|
}
|
|
}
|
|
}
|
|
|
|
stopVmAfterSuccessfulSmoke(freshDiscord: string, upgradeDiscord: string): void {
|
|
if (freshDiscord !== "pass" && upgradeDiscord !== "pass") {
|
|
return;
|
|
}
|
|
say(`Stop ${this.input.vmName} after successful Discord smoke`);
|
|
const result = run("prlctl", ["stop", this.input.vmName], {
|
|
check: false,
|
|
quiet: true,
|
|
timeoutMs: 120_000,
|
|
});
|
|
if (result.status !== 0) {
|
|
warn(
|
|
`failed to stop ${this.input.vmName} after successful Discord smoke (rc=${result.status})`,
|
|
);
|
|
}
|
|
}
|
|
|
|
private discordMessageId(payloadText: string): string {
|
|
const payload = JSON.parse(payloadText) as {
|
|
payload?: { messageId?: string; result?: { messageId?: string } };
|
|
};
|
|
const id = payload.payload?.messageId || payload.payload?.result?.messageId;
|
|
if (!id) {
|
|
throw new Error("messageId missing from send output");
|
|
}
|
|
return id;
|
|
}
|
|
|
|
private async discordApi(method: string, apiPath: string, payload?: unknown): Promise<string> {
|
|
const args = [
|
|
"-fsS",
|
|
"-X",
|
|
method,
|
|
"-H",
|
|
`Authorization: Bot ${this.input.config.token}`,
|
|
...(payload == null
|
|
? []
|
|
: ["-H", "Content-Type: application/json", "--data", JSON.stringify(payload)]),
|
|
`https://discord.com/api/v10${apiPath}`,
|
|
];
|
|
return run("curl", args, { quiet: true }).stdout;
|
|
}
|
|
|
|
private async waitForHostVisibility(nonce: string, messageId: string): Promise<void> {
|
|
const deadline = Date.now() + 180_000;
|
|
while (Date.now() < deadline) {
|
|
const direct = await this.discordApi(
|
|
"GET",
|
|
`/channels/${this.input.config.channelId}/messages/${messageId}`,
|
|
).catch(() => "");
|
|
if (direct.includes(nonce)) {
|
|
return;
|
|
}
|
|
const recent = await this.discordApi(
|
|
"GET",
|
|
`/channels/${this.input.config.channelId}/messages?limit=20`,
|
|
).catch(() => "");
|
|
if (recent.includes(nonce)) {
|
|
return;
|
|
}
|
|
run("sleep", ["2"], { quiet: true });
|
|
}
|
|
throw new Error("Discord host visibility timed out");
|
|
}
|
|
|
|
private async postDiscordMessage(content: string): Promise<string> {
|
|
const response = await this.discordApi(
|
|
"POST",
|
|
`/channels/${this.input.config.channelId}/messages`,
|
|
{
|
|
content,
|
|
flags: 4096,
|
|
},
|
|
);
|
|
const id = (JSON.parse(response) as { id?: string }).id;
|
|
if (!id) {
|
|
throw new Error("host Discord post missing message id");
|
|
}
|
|
return id;
|
|
}
|
|
|
|
private waitForGuestReadback(nonce: string): void {
|
|
const deadline = Date.now() + 180_000;
|
|
while (Date.now() < deadline) {
|
|
const result = this.input.guest.run(
|
|
[
|
|
this.input.guestOpenClaw,
|
|
"message",
|
|
"read",
|
|
"--channel",
|
|
"discord",
|
|
"--target",
|
|
`channel:${this.input.config.channelId}`,
|
|
"--limit",
|
|
"20",
|
|
"--json",
|
|
],
|
|
{ check: false },
|
|
);
|
|
if (result.status === 0 && result.stdout.includes(nonce)) {
|
|
return;
|
|
}
|
|
run("sleep", ["3"], { quiet: true });
|
|
}
|
|
throw new Error("Discord guest readback timed out");
|
|
}
|
|
}
|