Marks background tasks as error on session.error to release concurrency immediately, and skips/removes error tasks from queues to avoid zombie starts.
78 lines
2.2 KiB
TypeScript
78 lines
2.2 KiB
TypeScript
import { log } from "../../shared"
|
|
|
|
import { TASK_TTL_MS } from "./constants"
|
|
import { subagentSessions } from "../claude-code-session-state"
|
|
import { pruneStaleTasksAndNotifications } from "./task-poller"
|
|
|
|
import type { BackgroundTask, LaunchInput } from "./types"
|
|
import type { ConcurrencyManager } from "./concurrency"
|
|
|
|
type QueueItem = { task: BackgroundTask; input: LaunchInput }
|
|
|
|
export function pruneStaleState(args: {
|
|
tasks: Map<string, BackgroundTask>
|
|
notifications: Map<string, BackgroundTask[]>
|
|
queuesByKey: Map<string, QueueItem[]>
|
|
concurrencyManager: ConcurrencyManager
|
|
cleanupPendingByParent: (task: BackgroundTask) => void
|
|
clearNotificationsForTask: (taskId: string) => void
|
|
}): void {
|
|
const {
|
|
tasks,
|
|
notifications,
|
|
queuesByKey,
|
|
concurrencyManager,
|
|
cleanupPendingByParent,
|
|
clearNotificationsForTask,
|
|
} = args
|
|
|
|
pruneStaleTasksAndNotifications({
|
|
tasks,
|
|
notifications,
|
|
onTaskPruned: (taskId, task, errorMessage) => {
|
|
const wasPending = task.status === "pending"
|
|
const now = Date.now()
|
|
const timestamp = task.status === "pending"
|
|
? task.queuedAt?.getTime()
|
|
: task.startedAt?.getTime()
|
|
const age = timestamp ? now - timestamp : TASK_TTL_MS
|
|
|
|
log("[background-agent] Pruning stale task:", {
|
|
taskId,
|
|
status: task.status,
|
|
age: Math.round(age / 1000) + "s",
|
|
})
|
|
|
|
task.status = "error"
|
|
task.error = errorMessage
|
|
task.completedAt = new Date()
|
|
if (task.concurrencyKey) {
|
|
concurrencyManager.release(task.concurrencyKey)
|
|
task.concurrencyKey = undefined
|
|
}
|
|
|
|
cleanupPendingByParent(task)
|
|
if (wasPending) {
|
|
const key = task.model
|
|
? `${task.model.providerID}/${task.model.modelID}`
|
|
: task.agent
|
|
const queue = queuesByKey.get(key)
|
|
if (queue) {
|
|
const index = queue.findIndex((item) => item.task.id === taskId)
|
|
if (index !== -1) {
|
|
queue.splice(index, 1)
|
|
if (queue.length === 0) {
|
|
queuesByKey.delete(key)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
clearNotificationsForTask(taskId)
|
|
tasks.delete(taskId)
|
|
if (task.sessionID) {
|
|
subagentSessions.delete(task.sessionID)
|
|
}
|
|
},
|
|
})
|
|
}
|