mirror of
https://github.com/moltbot/moltbot.git
synced 2026-03-09 15:35:17 +00:00
461 lines
13 KiB
TypeScript
461 lines
13 KiB
TypeScript
import { promises as fs } from "node:fs";
|
|
import path from "node:path";
|
|
import type { OpenClawPluginApi } from "openclaw/plugin-sdk/device-pair";
|
|
import { listDevicePairing } from "openclaw/plugin-sdk/device-pair";
|
|
|
|
const NOTIFY_STATE_FILE = "device-pair-notify.json";
|
|
const NOTIFY_POLL_INTERVAL_MS = 10_000;
|
|
const NOTIFY_MAX_SEEN_AGE_MS = 24 * 60 * 60 * 1000;
|
|
|
|
type NotifySubscription = {
|
|
to: string;
|
|
accountId?: string;
|
|
messageThreadId?: number;
|
|
mode: "persistent" | "once";
|
|
addedAtMs: number;
|
|
};
|
|
|
|
type NotifyStateFile = {
|
|
subscribers: NotifySubscription[];
|
|
notifiedRequestIds: Record<string, number>;
|
|
};
|
|
|
|
export type PendingPairingRequest = {
|
|
requestId: string;
|
|
deviceId: string;
|
|
displayName?: string;
|
|
platform?: string;
|
|
remoteIp?: string;
|
|
ts?: number;
|
|
};
|
|
|
|
export function formatPendingRequests(pending: PendingPairingRequest[]): string {
|
|
if (pending.length === 0) {
|
|
return "No pending device pairing requests.";
|
|
}
|
|
const lines: string[] = ["Pending device pairing requests:"];
|
|
for (const req of pending) {
|
|
const label = req.displayName?.trim() || req.deviceId;
|
|
const platform = req.platform?.trim();
|
|
const ip = req.remoteIp?.trim();
|
|
const parts = [
|
|
`- ${req.requestId}`,
|
|
label ? `name=${label}` : null,
|
|
platform ? `platform=${platform}` : null,
|
|
ip ? `ip=${ip}` : null,
|
|
].filter(Boolean);
|
|
lines.push(parts.join(" · "));
|
|
}
|
|
return lines.join("\n");
|
|
}
|
|
|
|
function resolveNotifyStatePath(stateDir: string): string {
|
|
return path.join(stateDir, NOTIFY_STATE_FILE);
|
|
}
|
|
|
|
function normalizeNotifyState(raw: unknown): NotifyStateFile {
|
|
const root = typeof raw === "object" && raw !== null ? (raw as Record<string, unknown>) : {};
|
|
const subscribersRaw = Array.isArray(root.subscribers) ? root.subscribers : [];
|
|
const notifiedRaw =
|
|
typeof root.notifiedRequestIds === "object" && root.notifiedRequestIds !== null
|
|
? (root.notifiedRequestIds as Record<string, unknown>)
|
|
: {};
|
|
|
|
const subscribers: NotifySubscription[] = [];
|
|
for (const item of subscribersRaw) {
|
|
if (typeof item !== "object" || item === null) {
|
|
continue;
|
|
}
|
|
const record = item as Record<string, unknown>;
|
|
const to = typeof record.to === "string" ? record.to.trim() : "";
|
|
if (!to) {
|
|
continue;
|
|
}
|
|
const accountId =
|
|
typeof record.accountId === "string" && record.accountId.trim()
|
|
? record.accountId.trim()
|
|
: undefined;
|
|
const messageThreadId =
|
|
typeof record.messageThreadId === "number" && Number.isFinite(record.messageThreadId)
|
|
? Math.trunc(record.messageThreadId)
|
|
: undefined;
|
|
const mode = record.mode === "once" ? "once" : "persistent";
|
|
const addedAtMs =
|
|
typeof record.addedAtMs === "number" && Number.isFinite(record.addedAtMs)
|
|
? Math.trunc(record.addedAtMs)
|
|
: Date.now();
|
|
subscribers.push({
|
|
to,
|
|
accountId,
|
|
messageThreadId,
|
|
mode,
|
|
addedAtMs,
|
|
});
|
|
}
|
|
|
|
const notifiedRequestIds: Record<string, number> = {};
|
|
for (const [requestId, ts] of Object.entries(notifiedRaw)) {
|
|
if (!requestId.trim()) {
|
|
continue;
|
|
}
|
|
if (typeof ts !== "number" || !Number.isFinite(ts) || ts <= 0) {
|
|
continue;
|
|
}
|
|
notifiedRequestIds[requestId] = Math.trunc(ts);
|
|
}
|
|
|
|
return { subscribers, notifiedRequestIds };
|
|
}
|
|
|
|
async function readNotifyState(filePath: string): Promise<NotifyStateFile> {
|
|
try {
|
|
const content = await fs.readFile(filePath, "utf8");
|
|
return normalizeNotifyState(JSON.parse(content));
|
|
} catch {
|
|
return { subscribers: [], notifiedRequestIds: {} };
|
|
}
|
|
}
|
|
|
|
async function writeNotifyState(filePath: string, state: NotifyStateFile): Promise<void> {
|
|
await fs.mkdir(path.dirname(filePath), { recursive: true });
|
|
const content = JSON.stringify(state, null, 2);
|
|
await fs.writeFile(filePath, `${content}\n`, "utf8");
|
|
}
|
|
|
|
function notifySubscriberKey(subscriber: {
|
|
to: string;
|
|
accountId?: string;
|
|
messageThreadId?: number;
|
|
}): string {
|
|
return [subscriber.to, subscriber.accountId ?? "", subscriber.messageThreadId ?? ""].join("|");
|
|
}
|
|
|
|
type NotifyTarget = {
|
|
to: string;
|
|
accountId?: string;
|
|
messageThreadId?: number;
|
|
};
|
|
|
|
function resolveNotifyTarget(ctx: {
|
|
senderId?: string;
|
|
from?: string;
|
|
to?: string;
|
|
accountId?: string;
|
|
messageThreadId?: number;
|
|
}): NotifyTarget | null {
|
|
const to = ctx.senderId?.trim() || ctx.from?.trim() || ctx.to?.trim() || "";
|
|
if (!to) {
|
|
return null;
|
|
}
|
|
return {
|
|
to,
|
|
...(ctx.accountId ? { accountId: ctx.accountId } : {}),
|
|
...(ctx.messageThreadId != null ? { messageThreadId: ctx.messageThreadId } : {}),
|
|
};
|
|
}
|
|
|
|
function upsertNotifySubscriber(
|
|
subscribers: NotifySubscription[],
|
|
target: NotifyTarget,
|
|
mode: NotifySubscription["mode"],
|
|
): boolean {
|
|
const key = notifySubscriberKey(target);
|
|
const index = subscribers.findIndex((entry) => notifySubscriberKey(entry) === key);
|
|
const next: NotifySubscription = {
|
|
...target,
|
|
mode,
|
|
addedAtMs: Date.now(),
|
|
};
|
|
if (index === -1) {
|
|
subscribers.push(next);
|
|
return true;
|
|
}
|
|
const existing = subscribers[index];
|
|
if (existing?.mode === mode) {
|
|
return false;
|
|
}
|
|
subscribers[index] = next;
|
|
return true;
|
|
}
|
|
|
|
function buildPairingRequestNotificationText(request: PendingPairingRequest): string {
|
|
const label = request.displayName?.trim() || request.deviceId;
|
|
const platform = request.platform?.trim();
|
|
const ip = request.remoteIp?.trim();
|
|
const lines = [
|
|
"📲 New device pairing request",
|
|
`ID: ${request.requestId}`,
|
|
`Name: ${label}`,
|
|
...(platform ? [`Platform: ${platform}`] : []),
|
|
...(ip ? [`IP: ${ip}`] : []),
|
|
"",
|
|
`Approve: /pair approve ${request.requestId}`,
|
|
"List pending: /pair pending",
|
|
];
|
|
return lines.join("\n");
|
|
}
|
|
|
|
function requestTimestampMs(request: PendingPairingRequest): number | null {
|
|
if (typeof request.ts !== "number" || !Number.isFinite(request.ts)) {
|
|
return null;
|
|
}
|
|
const ts = Math.trunc(request.ts);
|
|
return ts > 0 ? ts : null;
|
|
}
|
|
|
|
function shouldNotifySubscriberForRequest(
|
|
subscriber: NotifySubscription,
|
|
request: PendingPairingRequest,
|
|
): boolean {
|
|
if (subscriber.mode !== "once") {
|
|
return true;
|
|
}
|
|
const ts = requestTimestampMs(request);
|
|
// One-shot subscriptions should only notify for new requests created after arming.
|
|
if (ts == null) {
|
|
return false;
|
|
}
|
|
return ts >= subscriber.addedAtMs;
|
|
}
|
|
|
|
async function notifySubscriber(params: {
|
|
api: OpenClawPluginApi;
|
|
subscriber: NotifySubscription;
|
|
text: string;
|
|
}): Promise<boolean> {
|
|
const send = params.api.runtime?.channel?.telegram?.sendMessageTelegram;
|
|
if (!send) {
|
|
params.api.logger.warn("device-pair: telegram runtime unavailable for pairing notifications");
|
|
return false;
|
|
}
|
|
|
|
try {
|
|
await send(params.subscriber.to, params.text, {
|
|
...(params.subscriber.accountId ? { accountId: params.subscriber.accountId } : {}),
|
|
...(params.subscriber.messageThreadId != null
|
|
? { messageThreadId: params.subscriber.messageThreadId }
|
|
: {}),
|
|
});
|
|
return true;
|
|
} catch (err) {
|
|
params.api.logger.warn(
|
|
`device-pair: failed to send pairing notification to ${params.subscriber.to}: ${String(
|
|
(err as Error)?.message ?? err,
|
|
)}`,
|
|
);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
async function notifyPendingPairingRequests(params: {
|
|
api: OpenClawPluginApi;
|
|
statePath: string;
|
|
}): Promise<void> {
|
|
const state = await readNotifyState(params.statePath);
|
|
const pairing = await listDevicePairing();
|
|
const pending = pairing.pending as PendingPairingRequest[];
|
|
const now = Date.now();
|
|
const pendingIds = new Set(pending.map((entry) => entry.requestId));
|
|
let changed = false;
|
|
|
|
for (const [requestId, ts] of Object.entries(state.notifiedRequestIds)) {
|
|
if (!pendingIds.has(requestId) || now - ts > NOTIFY_MAX_SEEN_AGE_MS) {
|
|
delete state.notifiedRequestIds[requestId];
|
|
changed = true;
|
|
}
|
|
}
|
|
|
|
if (state.subscribers.length > 0) {
|
|
const oneShotDelivered = new Set<string>();
|
|
for (const request of pending) {
|
|
if (state.notifiedRequestIds[request.requestId]) {
|
|
continue;
|
|
}
|
|
|
|
const text = buildPairingRequestNotificationText(request);
|
|
let delivered = false;
|
|
for (const subscriber of state.subscribers) {
|
|
if (!shouldNotifySubscriberForRequest(subscriber, request)) {
|
|
continue;
|
|
}
|
|
const sent = await notifySubscriber({
|
|
api: params.api,
|
|
subscriber,
|
|
text,
|
|
});
|
|
delivered = delivered || sent;
|
|
if (sent && subscriber.mode === "once") {
|
|
oneShotDelivered.add(notifySubscriberKey(subscriber));
|
|
}
|
|
}
|
|
|
|
if (delivered) {
|
|
state.notifiedRequestIds[request.requestId] = now;
|
|
changed = true;
|
|
}
|
|
}
|
|
if (oneShotDelivered.size > 0) {
|
|
const initialCount = state.subscribers.length;
|
|
state.subscribers = state.subscribers.filter(
|
|
(subscriber) => !oneShotDelivered.has(notifySubscriberKey(subscriber)),
|
|
);
|
|
if (state.subscribers.length !== initialCount) {
|
|
changed = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (changed) {
|
|
await writeNotifyState(params.statePath, state);
|
|
}
|
|
}
|
|
|
|
export async function armPairNotifyOnce(params: {
|
|
api: OpenClawPluginApi;
|
|
ctx: {
|
|
channel: string;
|
|
senderId?: string;
|
|
from?: string;
|
|
to?: string;
|
|
accountId?: string;
|
|
messageThreadId?: number;
|
|
};
|
|
}): Promise<boolean> {
|
|
if (params.ctx.channel !== "telegram") {
|
|
return false;
|
|
}
|
|
const target = resolveNotifyTarget(params.ctx);
|
|
if (!target) {
|
|
return false;
|
|
}
|
|
|
|
const stateDir = params.api.runtime.state.resolveStateDir();
|
|
const statePath = resolveNotifyStatePath(stateDir);
|
|
const state = await readNotifyState(statePath);
|
|
let changed = false;
|
|
|
|
if (upsertNotifySubscriber(state.subscribers, target, "once")) {
|
|
changed = true;
|
|
}
|
|
|
|
if (changed) {
|
|
await writeNotifyState(statePath, state);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
export async function handleNotifyCommand(params: {
|
|
api: OpenClawPluginApi;
|
|
ctx: {
|
|
channel: string;
|
|
senderId?: string;
|
|
from?: string;
|
|
to?: string;
|
|
accountId?: string;
|
|
messageThreadId?: number;
|
|
};
|
|
action: string;
|
|
}): Promise<{ text: string }> {
|
|
if (params.ctx.channel !== "telegram") {
|
|
return { text: "Pairing notifications are currently supported only on Telegram." };
|
|
}
|
|
|
|
const target = resolveNotifyTarget(params.ctx);
|
|
if (!target) {
|
|
return { text: "Could not resolve Telegram target for this chat." };
|
|
}
|
|
|
|
const stateDir = params.api.runtime.state.resolveStateDir();
|
|
const statePath = resolveNotifyStatePath(stateDir);
|
|
const state = await readNotifyState(statePath);
|
|
const targetKey = notifySubscriberKey(target);
|
|
const current = state.subscribers.find((entry) => notifySubscriberKey(entry) === targetKey);
|
|
|
|
if (params.action === "on" || params.action === "enable") {
|
|
if (upsertNotifySubscriber(state.subscribers, target, "persistent")) {
|
|
await writeNotifyState(statePath, state);
|
|
}
|
|
return {
|
|
text:
|
|
"✅ Pair request notifications enabled for this Telegram chat.\n" +
|
|
"I will ping here when a new device pairing request arrives.",
|
|
};
|
|
}
|
|
|
|
if (params.action === "off" || params.action === "disable") {
|
|
const currentIndex = state.subscribers.findIndex(
|
|
(entry) => notifySubscriberKey(entry) === targetKey,
|
|
);
|
|
if (currentIndex !== -1) {
|
|
state.subscribers.splice(currentIndex, 1);
|
|
await writeNotifyState(statePath, state);
|
|
}
|
|
return { text: "✅ Pair request notifications disabled for this Telegram chat." };
|
|
}
|
|
|
|
if (params.action === "once" || params.action === "arm") {
|
|
await armPairNotifyOnce({
|
|
api: params.api,
|
|
ctx: params.ctx,
|
|
});
|
|
return {
|
|
text:
|
|
"✅ One-shot pairing notification armed for this Telegram chat.\n" +
|
|
"I will notify on the next new pairing request, then auto-disable.",
|
|
};
|
|
}
|
|
|
|
if (params.action === "status" || params.action === "") {
|
|
const pending = await listDevicePairing();
|
|
const enabled = Boolean(current);
|
|
const mode = current?.mode ?? "off";
|
|
return {
|
|
text: [
|
|
`Pair request notifications: ${enabled ? "enabled" : "disabled"} for this chat.`,
|
|
`Mode: ${mode}`,
|
|
`Subscribers: ${state.subscribers.length}`,
|
|
`Pending requests: ${pending.pending.length}`,
|
|
"",
|
|
"Use /pair notify on|off|once",
|
|
].join("\n"),
|
|
};
|
|
}
|
|
|
|
return { text: "Usage: /pair notify on|off|once|status" };
|
|
}
|
|
|
|
export function registerPairingNotifierService(api: OpenClawPluginApi): void {
|
|
let notifyInterval: ReturnType<typeof setInterval> | null = null;
|
|
|
|
api.registerService({
|
|
id: "device-pair-notifier",
|
|
start: async (ctx) => {
|
|
const statePath = resolveNotifyStatePath(ctx.stateDir);
|
|
const tick = async () => {
|
|
await notifyPendingPairingRequests({ api, statePath });
|
|
};
|
|
|
|
await tick().catch((err) => {
|
|
api.logger.warn(
|
|
`device-pair: initial notify poll failed: ${String((err as Error)?.message ?? err)}`,
|
|
);
|
|
});
|
|
|
|
notifyInterval = setInterval(() => {
|
|
tick().catch((err) => {
|
|
api.logger.warn(
|
|
`device-pair: notify poll failed: ${String((err as Error)?.message ?? err)}`,
|
|
);
|
|
});
|
|
}, NOTIFY_POLL_INTERVAL_MS);
|
|
notifyInterval.unref?.();
|
|
},
|
|
stop: async () => {
|
|
if (notifyInterval) {
|
|
clearInterval(notifyInterval);
|
|
notifyInterval = null;
|
|
}
|
|
},
|
|
});
|
|
}
|