mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-10 12:14:58 +00:00
71 lines
2.0 KiB
TypeScript
71 lines
2.0 KiB
TypeScript
import { escapeRegExp } from "../utils.js";
|
|
|
|
export const HEARTBEAT_TOKEN = "HEARTBEAT_OK";
|
|
export const SILENT_REPLY_TOKEN = "NO_REPLY";
|
|
|
|
const silentExactRegexByToken = new Map<string, RegExp>();
|
|
const silentTrailingRegexByToken = new Map<string, RegExp>();
|
|
|
|
function getSilentExactRegex(token: string): RegExp {
|
|
const cached = silentExactRegexByToken.get(token);
|
|
if (cached) {
|
|
return cached;
|
|
}
|
|
const escaped = escapeRegExp(token);
|
|
const regex = new RegExp(`^\\s*${escaped}\\s*$`);
|
|
silentExactRegexByToken.set(token, regex);
|
|
return regex;
|
|
}
|
|
|
|
function getSilentTrailingRegex(token: string): RegExp {
|
|
const cached = silentTrailingRegexByToken.get(token);
|
|
if (cached) {
|
|
return cached;
|
|
}
|
|
const escaped = escapeRegExp(token);
|
|
const regex = new RegExp(`(?:^|\\s+|\\*+)${escaped}\\s*$`);
|
|
silentTrailingRegexByToken.set(token, regex);
|
|
return regex;
|
|
}
|
|
|
|
export function isSilentReplyText(
|
|
text: string | undefined,
|
|
token: string = SILENT_REPLY_TOKEN,
|
|
): boolean {
|
|
if (!text) {
|
|
return false;
|
|
}
|
|
// Match only the exact silent token with optional surrounding whitespace.
|
|
// This prevents substantive replies ending with NO_REPLY from being suppressed (#19537).
|
|
return getSilentExactRegex(token).test(text);
|
|
}
|
|
|
|
/**
|
|
* Strip a trailing silent reply token from mixed-content text.
|
|
* Returns the remaining text with the token removed (trimmed).
|
|
* If the result is empty, the entire message should be treated as silent.
|
|
*/
|
|
export function stripSilentToken(text: string, token: string = SILENT_REPLY_TOKEN): string {
|
|
return text.replace(getSilentTrailingRegex(token), "").trim();
|
|
}
|
|
|
|
export function isSilentReplyPrefixText(
|
|
text: string | undefined,
|
|
token: string = SILENT_REPLY_TOKEN,
|
|
): boolean {
|
|
if (!text) {
|
|
return false;
|
|
}
|
|
const normalized = text.trimStart().toUpperCase();
|
|
if (!normalized) {
|
|
return false;
|
|
}
|
|
if (!normalized.includes("_")) {
|
|
return false;
|
|
}
|
|
if (/[^A-Z_]/.test(normalized)) {
|
|
return false;
|
|
}
|
|
return token.toUpperCase().startsWith(normalized);
|
|
}
|