mirror of
https://github.com/getcompanion-ai/co-mono.git
synced 2026-04-15 10:05:14 +00:00
- SettingsManager now loads .pi/settings.json from cwd (project settings) - Project settings merge with global settings (deep merge for objects) - Setters only modify global settings, project settings are read-only - Add static factories: SettingsManager.create(cwd?, agentDir?), SettingsManager.inMemory(settings?) - Add applyOverrides() for programmatic overrides - Replace 'settings' option with 'settingsManager' in CreateAgentSessionOptions - Update examples to use new pattern Incorporates PR #276 approach
155 lines
4.7 KiB
TypeScript
155 lines
4.7 KiB
TypeScript
/**
|
|
* Tests for AgentSession branching behavior.
|
|
*
|
|
* These tests verify:
|
|
* - Branching from a single message works
|
|
* - Branching in --no-session mode (in-memory only)
|
|
* - getUserMessagesForBranching returns correct entries
|
|
*/
|
|
|
|
import { existsSync, mkdirSync, rmSync } from "node:fs";
|
|
import { tmpdir } from "node:os";
|
|
import { join } from "node:path";
|
|
import { Agent, ProviderTransport } from "@mariozechner/pi-agent-core";
|
|
import { getModel } from "@mariozechner/pi-ai";
|
|
import { afterEach, beforeEach, describe, expect, it } from "vitest";
|
|
import { AgentSession } from "../src/core/agent-session.js";
|
|
import { SessionManager } from "../src/core/session-manager.js";
|
|
import { SettingsManager } from "../src/core/settings-manager.js";
|
|
import { codingTools } from "../src/core/tools/index.js";
|
|
|
|
const API_KEY = process.env.ANTHROPIC_API_KEY || process.env.ANTHROPIC_OAUTH_TOKEN;
|
|
|
|
describe.skipIf(!API_KEY)("AgentSession branching", () => {
|
|
let session: AgentSession;
|
|
let tempDir: string;
|
|
let sessionManager: SessionManager;
|
|
|
|
beforeEach(() => {
|
|
// Create temp directory for session files
|
|
tempDir = join(tmpdir(), `pi-branching-test-${Date.now()}`);
|
|
mkdirSync(tempDir, { recursive: true });
|
|
});
|
|
|
|
afterEach(async () => {
|
|
if (session) {
|
|
session.dispose();
|
|
}
|
|
if (tempDir && existsSync(tempDir)) {
|
|
rmSync(tempDir, { recursive: true });
|
|
}
|
|
});
|
|
|
|
function createSession(noSession: boolean = false) {
|
|
const model = getModel("anthropic", "claude-sonnet-4-5")!;
|
|
|
|
const transport = new ProviderTransport({
|
|
getApiKey: () => API_KEY,
|
|
});
|
|
|
|
const agent = new Agent({
|
|
transport,
|
|
initialState: {
|
|
model,
|
|
systemPrompt: "You are a helpful assistant. Be extremely concise, reply with just a few words.",
|
|
tools: codingTools,
|
|
},
|
|
});
|
|
|
|
sessionManager = noSession ? SessionManager.inMemory() : SessionManager.create(tempDir);
|
|
const settingsManager = SettingsManager.create(tempDir, tempDir);
|
|
|
|
session = new AgentSession({
|
|
agent,
|
|
sessionManager,
|
|
settingsManager,
|
|
});
|
|
|
|
// Must subscribe to enable session persistence
|
|
session.subscribe(() => {});
|
|
|
|
return session;
|
|
}
|
|
|
|
it("should allow branching from single message", async () => {
|
|
createSession();
|
|
|
|
// Send one message
|
|
await session.prompt("Say hello");
|
|
await session.agent.waitForIdle();
|
|
|
|
// Should have exactly 1 user message available for branching
|
|
const userMessages = session.getUserMessagesForBranching();
|
|
expect(userMessages.length).toBe(1);
|
|
expect(userMessages[0].text).toBe("Say hello");
|
|
|
|
// Branch from the first message
|
|
const result = await session.branch(userMessages[0].entryIndex);
|
|
expect(result.selectedText).toBe("Say hello");
|
|
expect(result.skipped).toBe(false);
|
|
|
|
// After branching, conversation should be empty (branched before the first message)
|
|
expect(session.messages.length).toBe(0);
|
|
|
|
// Session file should exist (new branch)
|
|
expect(session.sessionFile).not.toBeNull();
|
|
expect(existsSync(session.sessionFile!)).toBe(true);
|
|
});
|
|
|
|
it("should support in-memory branching in --no-session mode", async () => {
|
|
createSession(true);
|
|
|
|
// Verify sessions are disabled
|
|
expect(session.sessionFile).toBeNull();
|
|
|
|
// Send one message
|
|
await session.prompt("Say hi");
|
|
await session.agent.waitForIdle();
|
|
|
|
// Should have 1 user message
|
|
const userMessages = session.getUserMessagesForBranching();
|
|
expect(userMessages.length).toBe(1);
|
|
|
|
// Verify we have messages before branching
|
|
expect(session.messages.length).toBeGreaterThan(0);
|
|
|
|
// Branch from the first message
|
|
const result = await session.branch(userMessages[0].entryIndex);
|
|
expect(result.selectedText).toBe("Say hi");
|
|
expect(result.skipped).toBe(false);
|
|
|
|
// After branching, conversation should be empty
|
|
expect(session.messages.length).toBe(0);
|
|
|
|
// Session file should still be null (no file created)
|
|
expect(session.sessionFile).toBeNull();
|
|
});
|
|
|
|
it("should branch from middle of conversation", async () => {
|
|
createSession();
|
|
|
|
// Send multiple messages
|
|
await session.prompt("Say one");
|
|
await session.agent.waitForIdle();
|
|
|
|
await session.prompt("Say two");
|
|
await session.agent.waitForIdle();
|
|
|
|
await session.prompt("Say three");
|
|
await session.agent.waitForIdle();
|
|
|
|
// Should have 3 user messages
|
|
const userMessages = session.getUserMessagesForBranching();
|
|
expect(userMessages.length).toBe(3);
|
|
|
|
// Branch from second message (keeps first message + response)
|
|
const secondMessage = userMessages[1];
|
|
const result = await session.branch(secondMessage.entryIndex);
|
|
expect(result.selectedText).toBe("Say two");
|
|
|
|
// After branching, should have first user message + assistant response
|
|
expect(session.messages.length).toBe(2);
|
|
expect(session.messages[0].role).toBe("user");
|
|
expect(session.messages[1].role).toBe("assistant");
|
|
});
|
|
});
|