feat(acp): add kimi harness support surfaces

This commit is contained in:
Peter Steinberger
2026-03-03 01:02:03 +00:00
parent f26853f14c
commit 287606e445
6 changed files with 79 additions and 10 deletions

View File

@@ -47,11 +47,12 @@ describe("acp policy", () => {
it("applies allowlist filtering for ACP agents", () => {
const cfg = {
acp: {
allowedAgents: ["Codex", "claude-code"],
allowedAgents: ["Codex", "claude-code", "kimi"],
},
} satisfies OpenClawConfig;
expect(isAcpAgentAllowedByPolicy(cfg, "codex")).toBe(true);
expect(isAcpAgentAllowedByPolicy(cfg, "claude-code")).toBe(true);
expect(isAcpAgentAllowedByPolicy(cfg, "KIMI")).toBe(true);
expect(isAcpAgentAllowedByPolicy(cfg, "gemini")).toBe(false);
expect(resolveAcpAgentPolicyError(cfg, "gemini")?.code).toBe("ACP_SESSION_INIT_FAILED");
expect(resolveAcpAgentPolicyError(cfg, "codex")).toBeNull();

View File

@@ -56,6 +56,33 @@ describe("session identifier helpers", () => {
);
});
it("adds a Kimi resume hint when agent identity is resolved", () => {
const lines = resolveAcpThreadSessionDetailLines({
sessionKey: "agent:kimi:acp:resolved-1",
meta: {
backend: "acpx",
agent: "kimi",
runtimeSessionName: "runtime-1",
identity: {
state: "resolved",
source: "status",
lastUpdatedAt: Date.now(),
acpxSessionId: "acpx-kimi-123",
agentSessionId: "kimi-inner-123",
},
mode: "persistent",
state: "idle",
lastActivityAt: Date.now(),
},
});
expect(lines).toContain("agent session id: kimi-inner-123");
expect(lines).toContain("acpx session id: acpx-kimi-123");
expect(lines).toContain(
"resume in Kimi CLI: `kimi resume kimi-inner-123` (continues this conversation).",
);
});
it("shows pending identity text for status rendering", () => {
const lines = resolveAcpSessionIdentifierLinesFromIdentity({
backend: "acpx",

View File

@@ -22,6 +22,16 @@ const ACP_AGENT_RESUME_HINT_BY_KEY = new Map<string, SessionResumeHintResolver>(
({ agentSessionId }) =>
`resume in Codex CLI: \`codex resume ${agentSessionId}\` (continues this conversation).`,
],
[
"kimi",
({ agentSessionId }) =>
`resume in Kimi CLI: \`kimi resume ${agentSessionId}\` (continues this conversation).`,
],
[
"moonshot-kimi",
({ agentSessionId }) =>
`resume in Kimi CLI: \`kimi resume ${agentSessionId}\` (continues this conversation).`,
],
]);
function normalizeText(value: unknown): string | undefined {

View File

@@ -31,7 +31,7 @@ function createAcpEnabledConfig(home: string, storePath: string): OpenClawConfig
acp: {
enabled: true,
backend: "acpx",
allowedAgents: ["codex"],
allowedAgents: ["codex", "kimi"],
dispatch: { enabled: true },
},
agents: {
@@ -62,19 +62,20 @@ function mockConfigWithAcpOverrides(
loadConfigSpy.mockReturnValue(cfg);
}
function writeAcpSessionStore(storePath: string) {
function writeAcpSessionStore(storePath: string, agent = "codex") {
const sessionKey = `agent:${agent}:acp:test`;
fs.mkdirSync(path.dirname(storePath), { recursive: true });
fs.writeFileSync(
storePath,
JSON.stringify(
{
"agent:codex:acp:test": {
[sessionKey]: {
sessionId: "acp-session-1",
updatedAt: Date.now(),
acp: {
backend: "acpx",
agent: "codex",
runtimeSessionName: "agent:codex:acp:test",
agent,
runtimeSessionName: sessionKey,
mode: "oneshot",
state: "idle",
lastActivityAt: Date.now(),
@@ -278,4 +279,30 @@ describe("agentCommand ACP runtime routing", () => {
expect(runEmbeddedPiAgentSpy).not.toHaveBeenCalled();
});
});
it("allows ACP turns for kimi when policy allowlists kimi", async () => {
await withTempHome(async (home) => {
const storePath = path.join(home, "sessions.json");
writeAcpSessionStore(storePath, "kimi");
mockConfigWithAcpOverrides(home, storePath, {
allowedAgents: ["kimi"],
});
const runTurn = vi.fn(async (_params: unknown) => {});
mockAcpManager({
runTurn: (params: unknown) => runTurn(params),
resolveSession: ({ sessionKey }) => resolveReadySession(sessionKey, "kimi"),
});
await agentCommand({ message: "ping", sessionKey: "agent:kimi:acp:test" }, runtime);
expect(runTurn).toHaveBeenCalledWith(
expect.objectContaining({
sessionKey: "agent:kimi:acp:test",
text: "ping",
}),
);
expect(runEmbeddedPiAgentSpy).not.toHaveBeenCalled();
});
});
});