Merge pull request #630 from adam91holt/fix/heartbeat-default-agent-only

fix: only inject heartbeat prompt for default agent
This commit is contained in:
Peter Steinberger
2026-01-10 00:34:13 +00:00
committed by GitHub
3 changed files with 75 additions and 6 deletions

View File

@@ -2,6 +2,7 @@
## Unreleased ## Unreleased
- Agents: gate heartbeat prompt to default agent sessions (including non-agent session keys). (#630) — thanks @adam91holt
- Agent: fast abort on /stop and cancel tool calls between tool boundaries. (#617) - Agent: fast abort on /stop and cancel tool calls between tool boundaries. (#617)
- Models/Auth: add OpenCode Zen (multi-model proxy) onboarding. (#623) — thanks @magimetal - Models/Auth: add OpenCode Zen (multi-model proxy) onboarding. (#623) — thanks @magimetal
- WhatsApp: refactor vCard parsing helper and improve empty contact card summaries. (#624) — thanks @steipete - WhatsApp: refactor vCard parsing helper and improve empty contact card summaries. (#624) — thanks @steipete

View File

@@ -2,10 +2,12 @@ import type { AgentMessage, AgentTool } from "@mariozechner/pi-agent-core";
import { SessionManager } from "@mariozechner/pi-coding-agent"; import { SessionManager } from "@mariozechner/pi-coding-agent";
import { Type } from "@sinclair/typebox"; import { Type } from "@sinclair/typebox";
import { describe, expect, it, vi } from "vitest"; import { describe, expect, it, vi } from "vitest";
import type { ClawdbotConfig } from "../config/config.js";
import { import {
applyGoogleTurnOrderingFix, applyGoogleTurnOrderingFix,
buildEmbeddedSandboxInfo, buildEmbeddedSandboxInfo,
createSystemPromptOverride, createSystemPromptOverride,
resolveSessionAgentIds,
splitSdkTools, splitSdkTools,
} from "./pi-embedded-runner.js"; } from "./pi-embedded-runner.js";
import type { SandboxContext } from "./sandbox.js"; import type { SandboxContext } from "./sandbox.js";
@@ -57,6 +59,38 @@ describe("buildEmbeddedSandboxInfo", () => {
}); });
}); });
describe("resolveSessionAgentIds", () => {
const cfg = {
agents: {
list: [{ id: "main" }, { id: "beta", default: true }],
},
} as ClawdbotConfig;
it("falls back to the configured default when sessionKey is missing", () => {
const { defaultAgentId, sessionAgentId } = resolveSessionAgentIds({
config: cfg,
});
expect(defaultAgentId).toBe("beta");
expect(sessionAgentId).toBe("beta");
});
it("falls back to the configured default when sessionKey is non-agent", () => {
const { sessionAgentId } = resolveSessionAgentIds({
sessionKey: "telegram:slash:123",
config: cfg,
});
expect(sessionAgentId).toBe("beta");
});
it("uses the agent id from agent session keys", () => {
const { sessionAgentId } = resolveSessionAgentIds({
sessionKey: "agent:main:main",
config: cfg,
});
expect(sessionAgentId).toBe("main");
});
});
function createStubTool(name: string): AgentTool { function createStubTool(name: string): AgentTool {
return { return {
name, name,

View File

@@ -33,9 +33,14 @@ import {
type enqueueCommand, type enqueueCommand,
enqueueCommandInLane, enqueueCommandInLane,
} from "../process/command-queue.js"; } from "../process/command-queue.js";
import {
normalizeAgentId,
parseAgentSessionKey,
} from "../routing/session-key.js";
import { normalizeMessageProvider } from "../utils/message-provider.js"; import { normalizeMessageProvider } from "../utils/message-provider.js";
import { resolveUserPath } from "../utils.js"; import { resolveUserPath } from "../utils.js";
import { resolveClawdbotAgentDir } from "./agent-paths.js"; import { resolveClawdbotAgentDir } from "./agent-paths.js";
import { resolveDefaultAgentId } from "./agent-scope.js";
import { import {
markAuthProfileFailure, markAuthProfileFailure,
markAuthProfileGood, markAuthProfileGood,
@@ -555,6 +560,19 @@ export function buildEmbeddedSandboxInfo(
}; };
} }
export function resolveSessionAgentIds(params: {
sessionKey?: string;
config?: ClawdbotConfig;
}): { defaultAgentId: string; sessionAgentId: string } {
const defaultAgentId = resolveDefaultAgentId(params.config ?? {});
const sessionKey = params.sessionKey?.trim();
const parsed = sessionKey ? parseAgentSessionKey(sessionKey) : null;
const sessionAgentId = parsed?.agentId
? normalizeAgentId(parsed.agentId)
: defaultAgentId;
return { defaultAgentId, sessionAgentId };
}
function buildEmbeddedSystemPrompt(params: { function buildEmbeddedSystemPrompt(params: {
workspaceDir: string; workspaceDir: string;
defaultThinkLevel?: ThinkLevel; defaultThinkLevel?: ThinkLevel;
@@ -882,15 +900,23 @@ export async function compactEmbeddedPiSession(params: {
params.config?.agents?.defaults?.userTimezone, params.config?.agents?.defaults?.userTimezone,
); );
const userTime = formatUserTime(new Date(), userTimezone); const userTime = formatUserTime(new Date(), userTimezone);
// Only include heartbeat prompt for the default agent
const { defaultAgentId, sessionAgentId } = resolveSessionAgentIds({
sessionKey: params.sessionKey,
config: params.config,
});
const isDefaultAgent = sessionAgentId === defaultAgentId;
const appendPrompt = buildEmbeddedSystemPrompt({ const appendPrompt = buildEmbeddedSystemPrompt({
workspaceDir: effectiveWorkspace, workspaceDir: effectiveWorkspace,
defaultThinkLevel: params.thinkLevel, defaultThinkLevel: params.thinkLevel,
extraSystemPrompt: params.extraSystemPrompt, extraSystemPrompt: params.extraSystemPrompt,
ownerNumbers: params.ownerNumbers, ownerNumbers: params.ownerNumbers,
reasoningTagHint, reasoningTagHint,
heartbeatPrompt: resolveHeartbeatPrompt( heartbeatPrompt: isDefaultAgent
params.config?.agents?.defaults?.heartbeat?.prompt, ? resolveHeartbeatPrompt(
), params.config?.agents?.defaults?.heartbeat?.prompt,
)
: undefined,
skillsPrompt, skillsPrompt,
runtimeInfo, runtimeInfo,
sandboxInfo, sandboxInfo,
@@ -1245,15 +1271,23 @@ export async function runEmbeddedPiAgent(params: {
params.config?.agents?.defaults?.userTimezone, params.config?.agents?.defaults?.userTimezone,
); );
const userTime = formatUserTime(new Date(), userTimezone); const userTime = formatUserTime(new Date(), userTimezone);
// Only include heartbeat prompt for the default agent
const { defaultAgentId, sessionAgentId } = resolveSessionAgentIds({
sessionKey: params.sessionKey,
config: params.config,
});
const isDefaultAgent = sessionAgentId === defaultAgentId;
const appendPrompt = buildEmbeddedSystemPrompt({ const appendPrompt = buildEmbeddedSystemPrompt({
workspaceDir: effectiveWorkspace, workspaceDir: effectiveWorkspace,
defaultThinkLevel: thinkLevel, defaultThinkLevel: thinkLevel,
extraSystemPrompt: params.extraSystemPrompt, extraSystemPrompt: params.extraSystemPrompt,
ownerNumbers: params.ownerNumbers, ownerNumbers: params.ownerNumbers,
reasoningTagHint, reasoningTagHint,
heartbeatPrompt: resolveHeartbeatPrompt( heartbeatPrompt: isDefaultAgent
params.config?.agents?.defaults?.heartbeat?.prompt, ? resolveHeartbeatPrompt(
), params.config?.agents?.defaults?.heartbeat?.prompt,
)
: undefined,
skillsPrompt, skillsPrompt,
runtimeInfo, runtimeInfo,
sandboxInfo, sandboxInfo,