Files
openclaw/src/agents/pi-embedded-runner.limithistoryturns.test.ts
2026-02-22 11:17:47 +00:00

128 lines
4.1 KiB
TypeScript

import type { AgentMessage } from "@mariozechner/pi-agent-core";
import { describe, expect, it } from "vitest";
import { limitHistoryTurns } from "./pi-embedded-runner.js";
describe("limitHistoryTurns", () => {
const mockUsage = {
input: 1,
output: 1,
cacheRead: 0,
cacheWrite: 0,
totalTokens: 2,
cost: {
input: 0,
output: 0,
cacheRead: 0,
cacheWrite: 0,
total: 0,
},
} as const;
const userMessage = (text: string): AgentMessage =>
({
role: "user",
content: [{ type: "text", text }],
timestamp: Date.now(),
}) as AgentMessage;
const assistantTextMessage = (text: string): AgentMessage =>
({
role: "assistant",
content: [{ type: "text", text }],
stopReason: "stop",
api: "openai-responses",
provider: "openai",
model: "mock-1",
usage: mockUsage,
timestamp: Date.now(),
}) as AgentMessage;
const assistantToolCallMessage = (id: string): AgentMessage =>
({
role: "assistant",
content: [{ type: "toolCall", id, name: "exec", arguments: {} }],
stopReason: "stop",
api: "openai-responses",
provider: "openai",
model: "mock-1",
usage: mockUsage,
timestamp: Date.now(),
}) as AgentMessage;
const firstText = (message: AgentMessage): string | undefined => {
if (!("content" in message)) {
return undefined;
}
const content = message.content;
if (typeof content === "string") {
return content;
}
const first = content[0];
return first?.type === "text" ? first.text : undefined;
};
const makeMessages = (roles: ("user" | "assistant")[]): AgentMessage[] =>
roles.map((role, i) =>
role === "user" ? userMessage(`message ${i}`) : assistantTextMessage(`message ${i}`),
);
it("returns all messages when limit is undefined", () => {
const messages = makeMessages(["user", "assistant", "user", "assistant"]);
expect(limitHistoryTurns(messages, undefined)).toBe(messages);
});
it("returns all messages when limit is 0", () => {
const messages = makeMessages(["user", "assistant", "user", "assistant"]);
expect(limitHistoryTurns(messages, 0)).toBe(messages);
});
it("returns all messages when limit is negative", () => {
const messages = makeMessages(["user", "assistant", "user", "assistant"]);
expect(limitHistoryTurns(messages, -1)).toBe(messages);
});
it("returns empty array when messages is empty", () => {
expect(limitHistoryTurns([], 5)).toEqual([]);
});
it("keeps all messages when fewer user turns than limit", () => {
const messages = makeMessages(["user", "assistant", "user", "assistant"]);
expect(limitHistoryTurns(messages, 10)).toBe(messages);
});
it("limits to last N user turns", () => {
const messages = makeMessages(["user", "assistant", "user", "assistant", "user", "assistant"]);
const limited = limitHistoryTurns(messages, 2);
expect(limited.length).toBe(4);
expect(firstText(limited[0])).toBe("message 2");
});
it("handles single user turn limit", () => {
const messages = makeMessages(["user", "assistant", "user", "assistant", "user", "assistant"]);
const limited = limitHistoryTurns(messages, 1);
expect(limited.length).toBe(2);
expect(firstText(limited[0])).toBe("message 4");
expect(firstText(limited[1])).toBe("message 5");
});
it("handles messages with multiple assistant responses per user turn", () => {
const messages = makeMessages(["user", "assistant", "assistant", "user", "assistant"]);
const limited = limitHistoryTurns(messages, 1);
expect(limited.length).toBe(2);
expect(limited[0].role).toBe("user");
expect(limited[1].role).toBe("assistant");
});
it("preserves message content integrity", () => {
const messages: AgentMessage[] = [
userMessage("first"),
assistantToolCallMessage("1"),
userMessage("second"),
assistantTextMessage("response"),
];
const limited = limitHistoryTurns(messages, 1);
expect(firstText(limited[0])).toBe("second");
expect(firstText(limited[1])).toBe("response");
});
});