Skip to content

Commit c822824

Browse files
committed
fix(qqbot): validate cron payload base64
1 parent fe97f1f commit c822824

3 files changed

Lines changed: 17 additions & 1 deletion

File tree

CHANGELOG.md

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -42,6 +42,7 @@ Docs: https://docs.openclaw.ai
4242
- QA channel: skip malformed inline inbound attachment base64 instead of staging silently corrupted media for agent turns.
4343
- Microsoft Teams: reject malformed inline HTML image base64 padding instead of decoding corrupted `data:` image attachments.
4444
- Voice-call realtime: ignore malformed provider media-frame base64 before forwarding audio into bridge and transcription paths.
45+
- QQBot: reject malformed stored cron payload base64 before JSON decoding structured reminder data.
4546
- Models config/auth: stop inferring provider env-var markers from broad `^[A-Z_][A-Z0-9_]*$` strings, and resolve config-backed provider `apiKey` values only through structured env SecretRefs (`secrets.providers[id]` / `secrets.defaults`), so unrelated env vars cannot accidentally become provider credentials. Thanks @sallyom.
4647
- Media fetch: skip allocating and buffering the response body for bodyless media responses (HEAD probes and 204-style empty bodies), avoiding wasted heap on streams that carry no payload. Thanks @shakkernerd.
4748
- CLI/onboarding: forward provider-specific auth flags (e.g. `--openai-api-key`) through the onboarding wizard so they reach provider auth methods via `ctx.opts`, letting `--openai-api-key "$OPENAI_API_KEY"` skip the redundant "use existing env var?" prompt in non-interactive harnesses. (#81669) Thanks @sjf.

extensions/qqbot/src/engine/utils/payload.test.ts

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -59,6 +59,9 @@ describe("engine/utils/payload", () => {
5959
it("reports cron decode errors without throwing", () => {
6060
expect(decodeCronPayload("plain")).toEqual({ isCronPayload: false });
6161
expect(decodeCronPayload("QQBOT_CRON:").error).toBe("Cron payload body is empty");
62+
expect(decodeCronPayload("QQBOT_CRON:AAA@@@").error).toBe(
63+
"Failed to decode cron payload: Cron payload body is not valid base64",
64+
);
6265

6366
const wrongType = Buffer.from('{"type":"media"}', "utf-8").toString("base64");
6467
expect(decodeCronPayload(`QQBOT_CRON:${wrongType}`).error).toBe(

extensions/qqbot/src/engine/utils/payload.ts

Lines changed: 13 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -44,6 +44,18 @@ function formatErr(e: unknown): string {
4444
return e instanceof Error ? e.message : String(e);
4545
}
4646

47+
function normalizeBase64ForCompare(value: string): string {
48+
return value.replace(/=+$/u, "").replace(/-/gu, "+").replace(/_/gu, "/");
49+
}
50+
51+
function decodeStrictBase64Utf8(value: string): string {
52+
const buffer = Buffer.from(value, "base64");
53+
if (normalizeBase64ForCompare(buffer.toString("base64")) !== normalizeBase64ForCompare(value)) {
54+
throw new Error("Cron payload body is not valid base64");
55+
}
56+
return buffer.toString("utf-8");
57+
}
58+
4759
/** Parse model output that may start with the QQ Bot structured payload prefix. */
4860
export function parseQQBotPayload(text: string): ParseResult {
4961
const trimmedText = text.trim();
@@ -114,7 +126,7 @@ export function decodeCronPayload(message: string): {
114126
}
115127

116128
try {
117-
const jsonString = Buffer.from(base64Content, "base64").toString("utf-8");
129+
const jsonString = decodeStrictBase64Utf8(base64Content);
118130
const payload = JSON.parse(jsonString) as CronReminderPayload;
119131

120132
if (payload.type !== "cron_reminder") {

0 commit comments

Comments
 (0)