mirror of
https://github.com/getcompanion-ai/co-mono.git
synced 2026-04-15 17:00:59 +00:00
Major changes: - Replace monolithic SessionEvent with reason discriminator with individual event types: session_start, session_before_switch, session_switch, session_before_new, session_new, session_before_branch, session_branch, session_before_compact, session_compact, session_shutdown - Each event has dedicated result type (SessionBeforeSwitchResult, etc.) - HookHandler type now allows bare return statements (void in return type) - HookAPI.on() has proper overloads for each event with correct typing Additional fixes: - AgentSession now always subscribes to agent in constructor (was only subscribing when external subscribe() called, breaking internal handlers) - Standardize on undefined over null throughout codebase - HookUIContext methods return undefined instead of null - SessionManager methods return undefined instead of null - Simplify hook exports to 'export type * from types.js' - Add detailed JSDoc for skipConversationRestore vs cancel - Fix createBranchedSession to rebuild index in persist mode - newSession() now returns the session file path Updated all example hooks, tests, and emission sites to use new event types.
49 lines
1.5 KiB
TypeScript
49 lines
1.5 KiB
TypeScript
/**
|
|
* Auto-Commit on Exit Hook
|
|
*
|
|
* Automatically commits changes when the agent exits.
|
|
* Uses the last assistant message to generate a commit message.
|
|
*/
|
|
|
|
import type { HookAPI } from "@mariozechner/pi-coding-agent/hooks";
|
|
|
|
export default function (pi: HookAPI) {
|
|
pi.on("session_shutdown", async (_event, ctx) => {
|
|
// Check for uncommitted changes
|
|
const { stdout: status, code } = await pi.exec("git", ["status", "--porcelain"]);
|
|
|
|
if (code !== 0 || status.trim().length === 0) {
|
|
// Not a git repo or no changes
|
|
return;
|
|
}
|
|
|
|
// Find the last assistant message for commit context
|
|
const entries = ctx.sessionManager.getEntries();
|
|
let lastAssistantText = "";
|
|
for (let i = entries.length - 1; i >= 0; i--) {
|
|
const entry = entries[i];
|
|
if (entry.type === "message" && entry.message.role === "assistant") {
|
|
const content = entry.message.content;
|
|
if (Array.isArray(content)) {
|
|
lastAssistantText = content
|
|
.filter((c): c is { type: "text"; text: string } => c.type === "text")
|
|
.map((c) => c.text)
|
|
.join("\n");
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Generate a simple commit message
|
|
const firstLine = lastAssistantText.split("\n")[0] || "Work in progress";
|
|
const commitMessage = `[pi] ${firstLine.slice(0, 50)}${firstLine.length > 50 ? "..." : ""}`;
|
|
|
|
// Stage and commit
|
|
await pi.exec("git", ["add", "-A"]);
|
|
const { code: commitCode } = await pi.exec("git", ["commit", "-m", commitMessage]);
|
|
|
|
if (commitCode === 0 && ctx.hasUI) {
|
|
ctx.ui.notify(`Auto-committed: ${commitMessage}`, "info");
|
|
}
|
|
});
|
|
}
|