mirror of
https://github.com/openclaw/openclaw.git
synced 2026-06-04 07:04:06 +00:00
* feat: add core session goals * feat: polish session goals in tui * fix: resolve goal tool session stores * fix: keep get goal read-only * fix: migrate legacy goal session slots * fix: persist goal token accounting * fix: validate goal session rows * refactor: remove unshipped goal legacy handling * fix: handle goal commands in local tui * fix: satisfy goal tool display checks * fix: reset goal budget on overdue resume * feat: surface session goals across control surfaces * test: update gateway protocol test import * test: align goal fixture types with protocol * fix: scope selected global transcript usage fallback * fix: scope selected global web subscriptions * fix: preserve selected global agent during chat dispatch * fix: scope chat inject to selected global agents
82 lines
2.5 KiB
TypeScript
82 lines
2.5 KiB
TypeScript
import { afterEach, describe, expect, it, vi } from "vitest";
|
|
import { emitSessionTranscriptUpdate, onSessionTranscriptUpdate } from "./transcript-events.js";
|
|
|
|
const cleanup: Array<() => void> = [];
|
|
|
|
afterEach(() => {
|
|
while (cleanup.length > 0) {
|
|
cleanup.pop()?.();
|
|
}
|
|
});
|
|
|
|
describe("transcript events", () => {
|
|
it("emits trimmed session file updates", () => {
|
|
const listener = vi.fn();
|
|
cleanup.push(onSessionTranscriptUpdate(listener));
|
|
|
|
emitSessionTranscriptUpdate(" /tmp/session.jsonl ");
|
|
|
|
expect(listener).toHaveBeenCalledTimes(1);
|
|
expect(listener).toHaveBeenCalledWith({ sessionFile: "/tmp/session.jsonl" });
|
|
});
|
|
|
|
it("includes optional session metadata when provided", () => {
|
|
const listener = vi.fn();
|
|
cleanup.push(onSessionTranscriptUpdate(listener));
|
|
|
|
emitSessionTranscriptUpdate({
|
|
sessionFile: " /tmp/session.jsonl ",
|
|
sessionKey: " agent:main:main ",
|
|
agentId: " main ",
|
|
message: { role: "assistant", content: "hi" },
|
|
messageId: " msg-1 ",
|
|
messageSeq: 2,
|
|
});
|
|
|
|
expect(listener).toHaveBeenCalledWith({
|
|
sessionFile: "/tmp/session.jsonl",
|
|
sessionKey: "agent:main:main",
|
|
agentId: "main",
|
|
message: { role: "assistant", content: "hi" },
|
|
messageId: "msg-1",
|
|
messageSeq: 2,
|
|
});
|
|
});
|
|
|
|
it("drops invalid message sequence values", () => {
|
|
const listener = vi.fn();
|
|
cleanup.push(onSessionTranscriptUpdate(listener));
|
|
|
|
emitSessionTranscriptUpdate({
|
|
sessionFile: "/tmp/session.jsonl",
|
|
messageSeq: 0,
|
|
});
|
|
emitSessionTranscriptUpdate({
|
|
sessionFile: "/tmp/session.jsonl",
|
|
messageSeq: 1.5,
|
|
});
|
|
emitSessionTranscriptUpdate({
|
|
sessionFile: "/tmp/session.jsonl",
|
|
messageSeq: Number.POSITIVE_INFINITY,
|
|
});
|
|
|
|
expect(listener).toHaveBeenCalledTimes(3);
|
|
expect(listener).toHaveBeenNthCalledWith(1, { sessionFile: "/tmp/session.jsonl" });
|
|
expect(listener).toHaveBeenNthCalledWith(2, { sessionFile: "/tmp/session.jsonl" });
|
|
expect(listener).toHaveBeenNthCalledWith(3, { sessionFile: "/tmp/session.jsonl" });
|
|
});
|
|
|
|
it("continues notifying other listeners when one throws", () => {
|
|
const first = vi.fn(() => {
|
|
throw new Error("boom");
|
|
});
|
|
const second = vi.fn();
|
|
cleanup.push(onSessionTranscriptUpdate(first));
|
|
cleanup.push(onSessionTranscriptUpdate(second));
|
|
|
|
expect(emitSessionTranscriptUpdate("/tmp/session.jsonl")).toBeUndefined();
|
|
expect(first).toHaveBeenCalledTimes(1);
|
|
expect(second).toHaveBeenCalledTimes(1);
|
|
});
|
|
});
|