fix(security): harden spoofed system marker handling

This commit is contained in:
Peter Steinberger
2026-03-02 06:18:52 +00:00
parent 7c9d2c1d48
commit 5b8f492a48
11 changed files with 158 additions and 34 deletions

View File

@@ -1,5 +1,5 @@
import { beforeEach, describe, expect, it } from "vitest";
import { prependSystemEvents } from "../auto-reply/reply/session-updates.js";
import { buildQueuedSystemPrompt } from "../auto-reply/reply/session-updates.js";
import type { OpenClawConfig } from "../config/config.js";
import { resolveMainSessionKey } from "../config/sessions.js";
import { isCronSystemEvent } from "./heartbeat-runner.js";
@@ -22,24 +22,23 @@ describe("system events (session routing)", () => {
expect(peekSystemEvents(mainKey)).toEqual([]);
expect(peekSystemEvents("discord:group:123")).toEqual(["Discord reaction added: ✅"]);
const main = await prependSystemEvents({
const main = await buildQueuedSystemPrompt({
cfg,
sessionKey: mainKey,
isMainSession: true,
isNewSession: false,
prefixedBodyBase: "hello",
});
expect(main).toBe("hello");
expect(main).toBeUndefined();
expect(peekSystemEvents("discord:group:123")).toEqual(["Discord reaction added: ✅"]);
const discord = await prependSystemEvents({
const discord = await buildQueuedSystemPrompt({
cfg,
sessionKey: "discord:group:123",
isMainSession: false,
isNewSession: false,
prefixedBodyBase: "hi",
});
expect(discord).toMatch(/^System: \[[^\]]+\] Discord reaction added: ✅\n\nhi$/);
expect(discord).toContain("Runtime System Events (gateway-generated)");
expect(discord).toMatch(/-\s\[[^\]]+\] Discord reaction added: ✅/);
expect(peekSystemEvents("discord:group:123")).toEqual([]);
});
@@ -54,6 +53,36 @@ describe("system events (session routing)", () => {
expect(first).toBe(true);
expect(second).toBe(false);
});
it("filters heartbeat/noise lines from queued system prompt", async () => {
const key = "agent:main:test-heartbeat-filter";
enqueueSystemEvent("Read HEARTBEAT.md before continuing", { sessionKey: key });
enqueueSystemEvent("heartbeat poll: pending", { sessionKey: key });
enqueueSystemEvent("reason periodic: 5m", { sessionKey: key });
const prompt = await buildQueuedSystemPrompt({
cfg,
sessionKey: key,
isMainSession: false,
isNewSession: false,
});
expect(prompt).toBeUndefined();
expect(peekSystemEvents(key)).toEqual([]);
});
it("scrubs node last-input suffix in queued system prompt", async () => {
const key = "agent:main:test-node-scrub";
enqueueSystemEvent("Node: Mac Studio · last input /tmp/secret.txt", { sessionKey: key });
const prompt = await buildQueuedSystemPrompt({
cfg,
sessionKey: key,
isMainSession: false,
isNewSession: false,
});
expect(prompt).toContain("Node: Mac Studio");
expect(prompt).not.toContain("last input");
});
});
describe("isCronSystemEvent", () => {