Files
openclaw/extensions/clickclack/src/inbound.ts
Peter Steinberger 1507a9701b refactor: centralize inbound supplemental context
* refactor: centralize inbound supplemental context

* refactor: trim supplemental finalizer typing

* docs: clarify supplemental context projection

* refactor: move inbound finalization into core

* refactor: simplify channel inbound facts

* refactor: fold supplemental media into inbound finalizer

* refactor: migrate channel inbound callers to builder

* docs: mark inbound finalizer compat types deprecated

* refactor: wire runtime turn context builder

* refactor: replace channel turn runtime API

* fix: respect discord quote visibility

* fix: avoid deprecated line dispatch helper

* refactor: deprecate channel message SDK seams

* docs: trim channel outbound SDK page

* test: migrate irc inbound assertion

* refactor: deprecate outbound SDK facades

* refactor: deprecate channel helper SDK facades

* refactor: deprecate channel streaming SDK facade

* refactor: move direct dm helpers into inbound SDK

* chore: mark legacy test-utils SDK alias deprecated

* refactor: remove unused allow-from read helper

* refactor: route remaining channel dispatch through core

* refactor: enforce modern extension SDK imports

* test: give slow image root tests more time

* ci: support node fallback on windows

* fix: add transcripts tool display metadata

* refactor: trim legacy channel test seams

* fix: preserve channel compat after rebase

* fix: keep deprecated channel inbound aliases

* fix: preserve discord thread context visibility

* fix: clean final rebase conflicts

* fix: preserve channel message dispatch aliases

* fix: sync channel refactor after rebase

* fix: sync channel refactor after latest main

* fix: dedupe memory-core subagent mock

* test: align clickclack inbound dispatch assertions

* fix: sync plugin sdk api hash after rebase

* fix: sync channel refactor after latest main

* fix: sync plugin sdk api hash after rebase

* fix: sync plugin sdk api hash after latest main

* test: remove stale inbound context awaits
2026-05-27 09:26:06 +01:00

211 lines
6.4 KiB
TypeScript

import type { OpenClawConfig } from "openclaw/plugin-sdk/config-contracts";
import { resolveClickClackInboundAccess, type ClickClackInboundAccess } from "./access.js";
import { sendClickClackText } from "./outbound.js";
import { getClickClackRuntime } from "./runtime.js";
import { buildClickClackTarget } from "./target.js";
import type { ClickClackMessage, CoreConfig, ResolvedClickClackAccount } from "./types.js";
const CHANNEL_ID = "clickclack" as const;
function resolveAccountAgentRoute(params: {
cfg: OpenClawConfig;
account: ResolvedClickClackAccount;
target: string;
isDirect: boolean;
}) {
const runtime = getClickClackRuntime();
const route = runtime.channel.routing.resolveAgentRoute({
cfg: params.cfg,
channel: CHANNEL_ID,
accountId: params.account.accountId,
peer: {
kind: params.isDirect ? "direct" : "channel",
id: params.target,
},
});
const agentId = params.account.agentId ?? route.agentId;
if (agentId === route.agentId) {
return route;
}
return {
...route,
agentId,
sessionKey: runtime.channel.routing.buildAgentSessionKey({
agentId,
channel: CHANNEL_ID,
accountId: params.account.accountId,
peer: {
kind: params.isDirect ? "direct" : "channel",
id: params.target,
},
}),
};
}
async function dispatchModelReply(params: {
account: ResolvedClickClackAccount;
cfg: OpenClawConfig;
message: ClickClackMessage;
route: { agentId: string };
target: string;
}) {
const runtime = getClickClackRuntime();
const result = await runtime.llm.complete({
agentId: params.route.agentId,
model: params.account.model,
maxTokens: 96,
purpose: "clickclack bot reply",
systemPrompt: params.account.systemPrompt,
messages: [
{
role: "user",
content: params.message.body,
},
],
});
const text = result.text.trim();
if (!text) {
return;
}
await sendClickClackText({
cfg: params.cfg as CoreConfig,
accountId: params.account.accountId,
to: params.target,
text,
threadId: params.message.parent_message_id ? params.message.thread_root_id : undefined,
replyToId: params.message.id,
});
}
export async function handleClickClackInbound(params: {
account: ResolvedClickClackAccount;
config: CoreConfig;
message: ClickClackMessage;
access?: ClickClackInboundAccess;
}) {
const runtime = getClickClackRuntime();
const message = params.message;
const access =
params.access ??
(await resolveClickClackInboundAccess({
account: params.account,
config: params.config,
message,
}));
if (!access.shouldDispatch) {
return;
}
const isDirect = Boolean(message.direct_conversation_id);
const target = buildClickClackTarget(
isDirect
? { chatType: "direct", kind: "dm", id: message.author_id }
: { chatType: "group", kind: "channel", id: message.channel_id ?? "" },
);
const route = resolveAccountAgentRoute({
cfg: params.config as OpenClawConfig,
account: params.account,
target,
isDirect,
});
if (params.account.replyMode === "model") {
await dispatchModelReply({
account: params.account,
cfg: params.config as OpenClawConfig,
message,
route,
target,
});
return;
}
const senderName = message.author?.display_name || message.author_id;
const previousTimestamp = runtime.channel.session.readSessionUpdatedAt({
storePath: runtime.channel.session.resolveStorePath(params.config.session?.store, {
agentId: route.agentId,
}),
sessionKey: route.sessionKey,
});
const body = runtime.channel.reply.formatAgentEnvelope({
channel: "ClickClack",
from: senderName,
timestamp: new Date(message.created_at),
previousTimestamp,
envelope: runtime.channel.reply.resolveEnvelopeFormatOptions(params.config as OpenClawConfig),
body: message.body,
});
const storePath = runtime.channel.session.resolveStorePath(params.config.session?.store, {
agentId: route.agentId,
});
const ctxPayload = runtime.channel.reply.finalizeInboundContext({
Body: body,
BodyForAgent: message.body,
RawBody: message.body,
CommandBody: message.body,
From: target,
To: target,
SessionKey: route.sessionKey,
AccountId: route.accountId ?? params.account.accountId,
ChatType: isDirect ? "direct" : "group",
WasMentioned: isDirect ? undefined : true,
ConversationLabel: isDirect ? senderName : message.channel_id,
GroupChannel: message.channel_id,
NativeChannelId: message.channel_id || message.direct_conversation_id,
MessageThreadId: message.parent_message_id ? message.thread_root_id : undefined,
ThreadParentId: message.parent_message_id ? message.thread_root_id : undefined,
SenderName: senderName,
SenderId: message.author_id,
Provider: CHANNEL_ID,
Surface: CHANNEL_ID,
MessageSid: message.id,
MessageSidFull: message.id,
ReplyToId: message.id,
Timestamp: message.created_at,
OriginatingChannel: CHANNEL_ID,
OriginatingTo: target,
CommandAuthorized: access.commandAuthorized,
});
await runtime.channel.inbound.dispatchReply({
cfg: params.config as OpenClawConfig,
channel: CHANNEL_ID,
accountId: params.account.accountId,
agentId: route.agentId,
routeSessionKey: route.sessionKey,
storePath,
ctxPayload,
recordInboundSession: runtime.channel.session.recordInboundSession,
dispatchReplyWithBufferedBlockDispatcher:
runtime.channel.reply.dispatchReplyWithBufferedBlockDispatcher,
delivery: {
deliver: async (payload) => {
const text =
payload && typeof payload === "object" && "text" in payload
? ((payload as { text?: string }).text ?? "")
: "";
if (!text.trim()) {
return;
}
await sendClickClackText({
cfg: params.config,
accountId: params.account.accountId,
to: target,
text,
threadId: message.parent_message_id ? message.thread_root_id : undefined,
replyToId: message.id,
});
},
onError: (error) => {
throw error instanceof Error
? error
: new Error(`clickclack dispatch failed: ${String(error)}`);
},
},
replyPipeline: {},
record: {
onRecordError: (error) => {
throw error instanceof Error
? error
: new Error(`clickclack session record failed: ${String(error)}`);
},
},
});
}