mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-09 06:40:42 +00:00
* fix(core): align auto-reply threading behavior * fix(core): propagate reply threading through outbound and gateway * fix(whatsapp): use cached metadata for native quoted replies * feat(whatsapp): add configurable native reply quoting
174 lines
5.9 KiB
TypeScript
174 lines
5.9 KiB
TypeScript
import { getChannelPlugin } from "../../channels/plugins/index.js";
|
|
import type { ChannelThreadingAdapter } from "../../channels/plugins/types.core.js";
|
|
import { normalizeAnyChannelId } from "../../channels/registry.js";
|
|
import type { ReplyToMode } from "../../config/types.js";
|
|
import type { OpenClawConfig } from "../../config/types.openclaw.js";
|
|
import { normalizeOptionalLowercaseString } from "../../shared/string-coerce.js";
|
|
import type { OriginatingChannelType } from "../templating.js";
|
|
import type { ReplyPayload, ReplyThreadingPolicy } from "../types.js";
|
|
import { isSingleUseReplyToMode } from "./reply-reference.js";
|
|
|
|
type ReplyToModeChannelConfig = {
|
|
replyToMode?: ReplyToMode;
|
|
replyToModeByChatType?: Partial<Record<"direct" | "group" | "channel", ReplyToMode>>;
|
|
dm?: {
|
|
replyToMode?: ReplyToMode;
|
|
};
|
|
};
|
|
|
|
function normalizeReplyToModeChatType(
|
|
chatType?: string | null,
|
|
): "direct" | "group" | "channel" | undefined {
|
|
return chatType === "direct" || chatType === "group" || chatType === "channel"
|
|
? chatType
|
|
: undefined;
|
|
}
|
|
|
|
export function resolveConfiguredReplyToMode(
|
|
cfg: OpenClawConfig,
|
|
channel?: OriginatingChannelType,
|
|
chatType?: string | null,
|
|
): ReplyToMode {
|
|
const provider = normalizeAnyChannelId(channel) ?? normalizeOptionalLowercaseString(channel);
|
|
if (!provider) {
|
|
return "all";
|
|
}
|
|
const channelConfig = (cfg.channels as Record<string, ReplyToModeChannelConfig> | undefined)?.[
|
|
provider
|
|
];
|
|
const normalizedChatType = normalizeReplyToModeChatType(chatType);
|
|
if (normalizedChatType) {
|
|
const scopedMode = channelConfig?.replyToModeByChatType?.[normalizedChatType];
|
|
if (scopedMode !== undefined) {
|
|
return scopedMode;
|
|
}
|
|
}
|
|
if (normalizedChatType === "direct") {
|
|
const legacyDirectMode = channelConfig?.dm?.replyToMode;
|
|
if (legacyDirectMode !== undefined) {
|
|
return legacyDirectMode;
|
|
}
|
|
}
|
|
return channelConfig?.replyToMode ?? "all";
|
|
}
|
|
|
|
export function resolveReplyToModeWithThreading(
|
|
cfg: OpenClawConfig,
|
|
threading: ChannelThreadingAdapter | undefined,
|
|
params: {
|
|
channel?: OriginatingChannelType;
|
|
accountId?: string | null;
|
|
chatType?: string | null;
|
|
} = {},
|
|
): ReplyToMode {
|
|
const resolved = threading?.resolveReplyToMode?.({
|
|
cfg,
|
|
accountId: params.accountId,
|
|
chatType: params.chatType,
|
|
});
|
|
return resolved ?? resolveConfiguredReplyToMode(cfg, params.channel, params.chatType);
|
|
}
|
|
|
|
export function resolveReplyToMode(
|
|
cfg: OpenClawConfig,
|
|
channel?: OriginatingChannelType,
|
|
accountId?: string | null,
|
|
chatType?: string | null,
|
|
): ReplyToMode {
|
|
const normalizedAccountId = normalizeOptionalLowercaseString(accountId);
|
|
if (!normalizedAccountId) {
|
|
return resolveConfiguredReplyToMode(cfg, channel, chatType);
|
|
}
|
|
const provider = normalizeAnyChannelId(channel) ?? normalizeOptionalLowercaseString(channel);
|
|
const threading = provider ? getChannelPlugin(provider)?.threading : undefined;
|
|
return resolveReplyToModeWithThreading(cfg, threading, {
|
|
channel,
|
|
accountId: normalizedAccountId,
|
|
chatType,
|
|
});
|
|
}
|
|
|
|
export function createReplyToModeFilter(
|
|
mode: ReplyToMode,
|
|
opts: { allowExplicitReplyTagsWhenOff?: boolean } = {},
|
|
) {
|
|
let hasThreaded = false;
|
|
return (payload: ReplyPayload): ReplyPayload => {
|
|
if (!payload.replyToId) {
|
|
return payload;
|
|
}
|
|
if (mode === "off") {
|
|
const isExplicit = Boolean(payload.replyToTag) || Boolean(payload.replyToCurrent);
|
|
// Compaction notices must never be threaded when replyToMode=off — even
|
|
// if they carry explicit reply tags (replyToCurrent). Honouring the
|
|
// explicit tag here would make status notices appear in-thread while
|
|
// normal assistant replies stay off-thread, contradicting the off-mode
|
|
// expectation. Strip replyToId unconditionally for compaction payloads.
|
|
if (opts.allowExplicitReplyTagsWhenOff && isExplicit && !payload.isCompactionNotice) {
|
|
return payload;
|
|
}
|
|
return { ...payload, replyToId: undefined };
|
|
}
|
|
if (mode === "all") {
|
|
return payload;
|
|
}
|
|
if (isSingleUseReplyToMode(mode) && hasThreaded) {
|
|
// Compaction notices are transient status messages that should always
|
|
// appear in-thread, even after the first assistant block has already
|
|
// consumed the "first" slot. Let them keep their replyToId.
|
|
if (payload.isCompactionNotice) {
|
|
return payload;
|
|
}
|
|
return { ...payload, replyToId: undefined };
|
|
}
|
|
// Compaction notices are transient status messages — they should be
|
|
// threaded (so they appear in-context), but they must not consume the
|
|
// "first" slot of the replyToMode=first|batched filter. Skip advancing
|
|
// hasThreaded so the real assistant reply still gets replyToId.
|
|
if (isSingleUseReplyToMode(mode) && !payload.isCompactionNotice) {
|
|
hasThreaded = true;
|
|
}
|
|
return payload;
|
|
};
|
|
}
|
|
|
|
export function resolveImplicitCurrentMessageReplyAllowance(
|
|
mode: ReplyToMode | undefined,
|
|
policy?: ReplyThreadingPolicy,
|
|
): boolean {
|
|
const implicitCurrentMessage = policy?.implicitCurrentMessage ?? "default";
|
|
if (implicitCurrentMessage === "allow") {
|
|
return true;
|
|
}
|
|
if (implicitCurrentMessage === "deny") {
|
|
return false;
|
|
}
|
|
return mode !== "batched";
|
|
}
|
|
|
|
export function resolveBatchedReplyThreadingPolicy(
|
|
mode: ReplyToMode,
|
|
isBatched: boolean,
|
|
): ReplyThreadingPolicy | undefined {
|
|
if (mode !== "batched") {
|
|
return undefined;
|
|
}
|
|
return {
|
|
implicitCurrentMessage: isBatched ? "allow" : "deny",
|
|
};
|
|
}
|
|
|
|
export function createReplyToModeFilterForChannel(
|
|
mode: ReplyToMode,
|
|
channel?: OriginatingChannelType,
|
|
) {
|
|
const normalized = normalizeOptionalLowercaseString(channel);
|
|
const isWebchat = normalized === "webchat";
|
|
// Default: allow explicit reply tags/directives even when replyToMode is "off".
|
|
// Unknown channels fail closed; internal webchat stays allowed.
|
|
const allowExplicitReplyTagsWhenOff = normalized ? true : isWebchat;
|
|
return createReplyToModeFilter(mode, {
|
|
allowExplicitReplyTagsWhenOff,
|
|
});
|
|
}
|