mirror of
https://github.com/openclaw/openclaw.git
synced 2026-04-06 14:51:08 +00:00
89 lines
2.8 KiB
TypeScript
89 lines
2.8 KiB
TypeScript
import { describe, expect, it } from "vitest";
|
|
import {
|
|
escapeInternalRuntimeContextDelimiters,
|
|
hasInternalRuntimeContext,
|
|
INTERNAL_RUNTIME_CONTEXT_BEGIN,
|
|
INTERNAL_RUNTIME_CONTEXT_END,
|
|
stripInternalRuntimeContext,
|
|
} from "./internal-runtime-context.js";
|
|
|
|
function createDeterministicRng(seed: number): () => number {
|
|
let state = seed >>> 0;
|
|
return () => {
|
|
state = (state * 1_664_525 + 1_013_904_223) >>> 0;
|
|
return state / 0x1_0000_0000;
|
|
};
|
|
}
|
|
|
|
describe("internal runtime context codec", () => {
|
|
it("strips a marked internal runtime block and preserves surrounding text", () => {
|
|
const input = [
|
|
"Visible intro",
|
|
"",
|
|
INTERNAL_RUNTIME_CONTEXT_BEGIN,
|
|
"OpenClaw runtime context (internal):",
|
|
"This context is runtime-generated, not user-authored. Keep internal details private.",
|
|
"",
|
|
"[Internal task completion event]",
|
|
"source: subagent",
|
|
INTERNAL_RUNTIME_CONTEXT_END,
|
|
"",
|
|
"Visible outro",
|
|
].join("\n");
|
|
|
|
expect(stripInternalRuntimeContext(input)).toBe("Visible intro\n\nVisible outro");
|
|
});
|
|
|
|
it("detects canonical runtime context and ignores inline marker mentions", () => {
|
|
expect(
|
|
hasInternalRuntimeContext(
|
|
`${INTERNAL_RUNTIME_CONTEXT_BEGIN}\ninternal\n${INTERNAL_RUNTIME_CONTEXT_END}`,
|
|
),
|
|
).toBe(true);
|
|
expect(
|
|
hasInternalRuntimeContext(
|
|
`Inline token ${INTERNAL_RUNTIME_CONTEXT_BEGIN} should not count as a block marker.`,
|
|
),
|
|
).toBe(false);
|
|
});
|
|
|
|
it("fuzzes delimiter injection and nested marker handling deterministically", () => {
|
|
const rng = createDeterministicRng(0xc0ff_ee42);
|
|
const tokenPool = [
|
|
"plain output line",
|
|
"status: ok",
|
|
`inline ${INTERNAL_RUNTIME_CONTEXT_BEGIN} mention`,
|
|
`inline ${INTERNAL_RUNTIME_CONTEXT_END} mention`,
|
|
INTERNAL_RUNTIME_CONTEXT_BEGIN,
|
|
INTERNAL_RUNTIME_CONTEXT_END,
|
|
"more details",
|
|
];
|
|
|
|
for (let index = 0; index < 120; index++) {
|
|
const lineCount = 4 + Math.floor(rng() * 12);
|
|
const payloadLines: string[] = [];
|
|
for (let i = 0; i < lineCount; i++) {
|
|
const token = tokenPool[Math.floor(rng() * tokenPool.length)];
|
|
payloadLines.push(token);
|
|
}
|
|
const escapedPayload = payloadLines.map((line) =>
|
|
escapeInternalRuntimeContextDelimiters(line),
|
|
);
|
|
|
|
const visible = `Visible reply ${index}`;
|
|
const wrapped = [
|
|
INTERNAL_RUNTIME_CONTEXT_BEGIN,
|
|
...escapedPayload,
|
|
INTERNAL_RUNTIME_CONTEXT_END,
|
|
"",
|
|
visible,
|
|
].join("\n");
|
|
|
|
const stripped = stripInternalRuntimeContext(wrapped);
|
|
expect(stripped).toBe(visible);
|
|
expect(stripped).not.toContain(INTERNAL_RUNTIME_CONTEXT_BEGIN);
|
|
expect(stripped).not.toContain(INTERNAL_RUNTIME_CONTEXT_END);
|
|
}
|
|
});
|
|
});
|