Use hook-written continuation marker state to gate run completion checks and remove the noisy event-stream shutdown timeout log in run mode.
77 lines
2.3 KiB
TypeScript
77 lines
2.3 KiB
TypeScript
import type { PluginInput } from "@opencode-ai/plugin"
|
|
|
|
import type { BackgroundManager } from "../../features/background-agent"
|
|
import {
|
|
clearContinuationMarker,
|
|
} from "../../features/run-continuation-state"
|
|
import { log } from "../../shared/logger"
|
|
|
|
import { DEFAULT_SKIP_AGENTS, HOOK_NAME } from "./constants"
|
|
import type { SessionStateStore } from "./session-state"
|
|
import { handleSessionIdle } from "./idle-event"
|
|
import { handleNonIdleEvent } from "./non-idle-events"
|
|
|
|
export function createTodoContinuationHandler(args: {
|
|
ctx: PluginInput
|
|
sessionStateStore: SessionStateStore
|
|
backgroundManager?: BackgroundManager
|
|
skipAgents?: string[]
|
|
isContinuationStopped?: (sessionID: string) => boolean
|
|
}): (input: { event: { type: string; properties?: unknown } }) => Promise<void> {
|
|
const {
|
|
ctx,
|
|
sessionStateStore,
|
|
backgroundManager,
|
|
skipAgents = DEFAULT_SKIP_AGENTS,
|
|
isContinuationStopped,
|
|
} = args
|
|
|
|
return async ({ event }: { event: { type: string; properties?: unknown } }): Promise<void> => {
|
|
const props = event.properties as Record<string, unknown> | undefined
|
|
|
|
if (event.type === "session.error") {
|
|
const sessionID = props?.sessionID as string | undefined
|
|
if (!sessionID) return
|
|
|
|
const error = props?.error as { name?: string } | undefined
|
|
if (error?.name === "MessageAbortedError" || error?.name === "AbortError") {
|
|
const state = sessionStateStore.getState(sessionID)
|
|
state.abortDetectedAt = Date.now()
|
|
log(`[${HOOK_NAME}] Abort detected via session.error`, { sessionID, errorName: error.name })
|
|
}
|
|
|
|
sessionStateStore.cancelCountdown(sessionID)
|
|
log(`[${HOOK_NAME}] session.error`, { sessionID })
|
|
return
|
|
}
|
|
|
|
if (event.type === "session.idle") {
|
|
const sessionID = props?.sessionID as string | undefined
|
|
if (!sessionID) return
|
|
|
|
await handleSessionIdle({
|
|
ctx,
|
|
sessionID,
|
|
sessionStateStore,
|
|
backgroundManager,
|
|
skipAgents,
|
|
isContinuationStopped,
|
|
})
|
|
return
|
|
}
|
|
|
|
if (event.type === "session.deleted") {
|
|
const sessionInfo = props?.info as { id?: string } | undefined
|
|
if (sessionInfo?.id) {
|
|
clearContinuationMarker(ctx.directory, sessionInfo.id)
|
|
}
|
|
}
|
|
|
|
handleNonIdleEvent({
|
|
eventType: event.type,
|
|
properties: props,
|
|
sessionStateStore,
|
|
})
|
|
}
|
|
}
|