Compare commits
3 Commits
dev
...
fix/issue-
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
d46946c85f | ||
|
|
3b588283b1 | ||
|
|
816e46a967 |
@@ -224,6 +224,12 @@ function stubNotifyParentSession(manager: BackgroundManager): void {
|
|||||||
;(manager as unknown as { notifyParentSession: () => Promise<void> }).notifyParentSession = async () => {}
|
;(manager as unknown as { notifyParentSession: () => Promise<void> }).notifyParentSession = async () => {}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function flushBackgroundNotifications(): Promise<void> {
|
||||||
|
for (let i = 0; i < 6; i++) {
|
||||||
|
await Promise.resolve()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function createToastRemoveTaskTracker(): { removeTaskCalls: string[]; resetToastManager: () => void } {
|
function createToastRemoveTaskTracker(): { removeTaskCalls: string[]; resetToastManager: () => void } {
|
||||||
_resetTaskToastManagerForTesting()
|
_resetTaskToastManagerForTesting()
|
||||||
const toastManager = initTaskToastManager({
|
const toastManager = initTaskToastManager({
|
||||||
@@ -1306,11 +1312,20 @@ describe("BackgroundManager.tryCompleteTask", () => {
|
|||||||
expect(abortedSessionIDs).toEqual(["session-1"])
|
expect(abortedSessionIDs).toEqual(["session-1"])
|
||||||
})
|
})
|
||||||
|
|
||||||
test("should clean pendingByParent even when notifyParentSession throws", async () => {
|
test("should clean pendingByParent even when promptAsync notification fails", async () => {
|
||||||
// given
|
// given
|
||||||
;(manager as unknown as { notifyParentSession: () => Promise<void> }).notifyParentSession = async () => {
|
const client = {
|
||||||
|
session: {
|
||||||
|
prompt: async () => ({}),
|
||||||
|
promptAsync: async () => {
|
||||||
throw new Error("notify failed")
|
throw new Error("notify failed")
|
||||||
|
},
|
||||||
|
abort: async () => ({}),
|
||||||
|
messages: async () => ({ data: [] }),
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
manager.shutdown()
|
||||||
|
manager = new BackgroundManager({ client, directory: tmpdir() } as unknown as PluginInput)
|
||||||
|
|
||||||
const task: BackgroundTask = {
|
const task: BackgroundTask = {
|
||||||
id: "task-pending-cleanup",
|
id: "task-pending-cleanup",
|
||||||
@@ -1424,7 +1439,7 @@ describe("BackgroundManager.tryCompleteTask", () => {
|
|||||||
// then
|
// then
|
||||||
expect(rejectedCount).toBe(0)
|
expect(rejectedCount).toBe(0)
|
||||||
expect(promptBodies.length).toBe(2)
|
expect(promptBodies.length).toBe(2)
|
||||||
expect(promptBodies.some((b) => b.noReply === false)).toBe(true)
|
expect(promptBodies.filter((body) => body.noReply === false)).toHaveLength(1)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -1932,7 +1947,6 @@ describe("BackgroundManager - Non-blocking Queue Integration", () => {
|
|||||||
test("should cancel running task and release concurrency", async () => {
|
test("should cancel running task and release concurrency", async () => {
|
||||||
// given
|
// given
|
||||||
const manager = createBackgroundManager()
|
const manager = createBackgroundManager()
|
||||||
stubNotifyParentSession(manager)
|
|
||||||
|
|
||||||
const concurrencyManager = getConcurrencyManager(manager)
|
const concurrencyManager = getConcurrencyManager(manager)
|
||||||
const concurrencyKey = "test-provider/test-model"
|
const concurrencyKey = "test-provider/test-model"
|
||||||
@@ -2890,7 +2904,7 @@ describe("BackgroundManager.shutdown session abort", () => {
|
|||||||
})
|
})
|
||||||
|
|
||||||
describe("BackgroundManager.handleEvent - session.deleted cascade", () => {
|
describe("BackgroundManager.handleEvent - session.deleted cascade", () => {
|
||||||
test("should cancel descendant tasks when parent session is deleted", () => {
|
test("should cancel descendant tasks and keep them until delayed cleanup", async () => {
|
||||||
// given
|
// given
|
||||||
const manager = createBackgroundManager()
|
const manager = createBackgroundManager()
|
||||||
const parentSessionID = "session-parent"
|
const parentSessionID = "session-parent"
|
||||||
@@ -2937,21 +2951,26 @@ describe("BackgroundManager.handleEvent - session.deleted cascade", () => {
|
|||||||
properties: { info: { id: parentSessionID } },
|
properties: { info: { id: parentSessionID } },
|
||||||
})
|
})
|
||||||
|
|
||||||
|
await flushBackgroundNotifications()
|
||||||
|
|
||||||
// then
|
// then
|
||||||
expect(taskMap.has(childTask.id)).toBe(false)
|
expect(taskMap.has(childTask.id)).toBe(true)
|
||||||
expect(taskMap.has(siblingTask.id)).toBe(false)
|
expect(taskMap.has(siblingTask.id)).toBe(true)
|
||||||
expect(taskMap.has(grandchildTask.id)).toBe(false)
|
expect(taskMap.has(grandchildTask.id)).toBe(true)
|
||||||
expect(taskMap.has(unrelatedTask.id)).toBe(true)
|
expect(taskMap.has(unrelatedTask.id)).toBe(true)
|
||||||
expect(childTask.status).toBe("cancelled")
|
expect(childTask.status).toBe("cancelled")
|
||||||
expect(siblingTask.status).toBe("cancelled")
|
expect(siblingTask.status).toBe("cancelled")
|
||||||
expect(grandchildTask.status).toBe("cancelled")
|
expect(grandchildTask.status).toBe("cancelled")
|
||||||
expect(pendingByParent.get(parentSessionID)).toBeUndefined()
|
expect(pendingByParent.get(parentSessionID)).toBeUndefined()
|
||||||
expect(pendingByParent.get("session-child")).toBeUndefined()
|
expect(pendingByParent.get("session-child")).toBeUndefined()
|
||||||
|
expect(getCompletionTimers(manager).has(childTask.id)).toBe(true)
|
||||||
|
expect(getCompletionTimers(manager).has(siblingTask.id)).toBe(true)
|
||||||
|
expect(getCompletionTimers(manager).has(grandchildTask.id)).toBe(true)
|
||||||
|
|
||||||
manager.shutdown()
|
manager.shutdown()
|
||||||
})
|
})
|
||||||
|
|
||||||
test("should remove tasks from toast manager when session is deleted", () => {
|
test("should remove cancelled tasks from toast manager while preserving delayed cleanup", async () => {
|
||||||
//#given
|
//#given
|
||||||
const { removeTaskCalls, resetToastManager } = createToastRemoveTaskTracker()
|
const { removeTaskCalls, resetToastManager } = createToastRemoveTaskTracker()
|
||||||
const manager = createBackgroundManager()
|
const manager = createBackgroundManager()
|
||||||
@@ -2980,9 +2999,13 @@ describe("BackgroundManager.handleEvent - session.deleted cascade", () => {
|
|||||||
properties: { info: { id: parentSessionID } },
|
properties: { info: { id: parentSessionID } },
|
||||||
})
|
})
|
||||||
|
|
||||||
|
await flushBackgroundNotifications()
|
||||||
|
|
||||||
//#then
|
//#then
|
||||||
expect(removeTaskCalls).toContain(childTask.id)
|
expect(removeTaskCalls).toContain(childTask.id)
|
||||||
expect(removeTaskCalls).toContain(grandchildTask.id)
|
expect(removeTaskCalls).toContain(grandchildTask.id)
|
||||||
|
expect(getCompletionTimers(manager).has(childTask.id)).toBe(true)
|
||||||
|
expect(getCompletionTimers(manager).has(grandchildTask.id)).toBe(true)
|
||||||
|
|
||||||
manager.shutdown()
|
manager.shutdown()
|
||||||
resetToastManager()
|
resetToastManager()
|
||||||
@@ -3045,7 +3068,7 @@ describe("BackgroundManager.handleEvent - session.error", () => {
|
|||||||
return task
|
return task
|
||||||
}
|
}
|
||||||
|
|
||||||
test("sets task to error, releases concurrency, and cleans up", async () => {
|
test("sets task to error, releases concurrency, and keeps it until delayed cleanup", async () => {
|
||||||
//#given
|
//#given
|
||||||
const manager = createBackgroundManager()
|
const manager = createBackgroundManager()
|
||||||
const concurrencyManager = getConcurrencyManager(manager)
|
const concurrencyManager = getConcurrencyManager(manager)
|
||||||
@@ -3078,18 +3101,21 @@ describe("BackgroundManager.handleEvent - session.error", () => {
|
|||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
|
await flushBackgroundNotifications()
|
||||||
|
|
||||||
//#then
|
//#then
|
||||||
expect(task.status).toBe("error")
|
expect(task.status).toBe("error")
|
||||||
expect(task.error).toBe("Model not found: kimi-for-coding/k2p5.")
|
expect(task.error).toBe("Model not found: kimi-for-coding/k2p5.")
|
||||||
expect(task.completedAt).toBeInstanceOf(Date)
|
expect(task.completedAt).toBeInstanceOf(Date)
|
||||||
expect(concurrencyManager.getCount(concurrencyKey)).toBe(0)
|
expect(concurrencyManager.getCount(concurrencyKey)).toBe(0)
|
||||||
expect(getTaskMap(manager).has(task.id)).toBe(false)
|
expect(getTaskMap(manager).has(task.id)).toBe(true)
|
||||||
expect(getPendingByParent(manager).get(task.parentSessionID)).toBeUndefined()
|
expect(getPendingByParent(manager).get(task.parentSessionID)).toBeUndefined()
|
||||||
|
expect(getCompletionTimers(manager).has(task.id)).toBe(true)
|
||||||
|
|
||||||
manager.shutdown()
|
manager.shutdown()
|
||||||
})
|
})
|
||||||
|
|
||||||
test("removes errored task from toast manager", () => {
|
test("should remove errored task from toast manager while preserving delayed cleanup", async () => {
|
||||||
//#given
|
//#given
|
||||||
const { removeTaskCalls, resetToastManager } = createToastRemoveTaskTracker()
|
const { removeTaskCalls, resetToastManager } = createToastRemoveTaskTracker()
|
||||||
const manager = createBackgroundManager()
|
const manager = createBackgroundManager()
|
||||||
@@ -3111,8 +3137,11 @@ describe("BackgroundManager.handleEvent - session.error", () => {
|
|||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
|
await flushBackgroundNotifications()
|
||||||
|
|
||||||
//#then
|
//#then
|
||||||
expect(removeTaskCalls).toContain(task.id)
|
expect(removeTaskCalls).toContain(task.id)
|
||||||
|
expect(getCompletionTimers(manager).has(task.id)).toBe(true)
|
||||||
|
|
||||||
manager.shutdown()
|
manager.shutdown()
|
||||||
resetToastManager()
|
resetToastManager()
|
||||||
@@ -3393,7 +3422,7 @@ describe("BackgroundManager.pruneStaleTasksAndNotifications - removes pruned tas
|
|||||||
manager.shutdown()
|
manager.shutdown()
|
||||||
})
|
})
|
||||||
|
|
||||||
test("removes stale task from toast manager", () => {
|
test("removes stale task from toast manager", async () => {
|
||||||
//#given
|
//#given
|
||||||
const { removeTaskCalls, resetToastManager } = createToastRemoveTaskTracker()
|
const { removeTaskCalls, resetToastManager } = createToastRemoveTaskTracker()
|
||||||
const manager = createBackgroundManager()
|
const manager = createBackgroundManager()
|
||||||
@@ -3408,6 +3437,7 @@ describe("BackgroundManager.pruneStaleTasksAndNotifications - removes pruned tas
|
|||||||
|
|
||||||
//#when
|
//#when
|
||||||
pruneStaleTasksAndNotificationsForTest(manager)
|
pruneStaleTasksAndNotificationsForTest(manager)
|
||||||
|
await flushBackgroundNotifications()
|
||||||
|
|
||||||
//#then
|
//#then
|
||||||
expect(removeTaskCalls).toContain(staleTask.id)
|
expect(removeTaskCalls).toContain(staleTask.id)
|
||||||
@@ -3415,6 +3445,53 @@ describe("BackgroundManager.pruneStaleTasksAndNotifications - removes pruned tas
|
|||||||
manager.shutdown()
|
manager.shutdown()
|
||||||
resetToastManager()
|
resetToastManager()
|
||||||
})
|
})
|
||||||
|
|
||||||
|
test("keeps stale task until notification cleanup after notifying parent", async () => {
|
||||||
|
//#given
|
||||||
|
const notifications: string[] = []
|
||||||
|
const { removeTaskCalls, resetToastManager } = createToastRemoveTaskTracker()
|
||||||
|
const client = {
|
||||||
|
session: {
|
||||||
|
prompt: async () => ({}),
|
||||||
|
promptAsync: async (args: { path: { id: string }; body: Record<string, unknown> & { noReply?: boolean; parts?: unknown[] } }) => {
|
||||||
|
const firstPart = args.body.parts?.[0]
|
||||||
|
if (firstPart && typeof firstPart === "object" && "text" in firstPart && typeof firstPart.text === "string") {
|
||||||
|
notifications.push(firstPart.text)
|
||||||
|
}
|
||||||
|
return {}
|
||||||
|
},
|
||||||
|
abort: async () => ({}),
|
||||||
|
messages: async () => ({ data: [] }),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
const manager = new BackgroundManager({ client, directory: tmpdir() } as unknown as PluginInput)
|
||||||
|
const staleTask = createMockTask({
|
||||||
|
id: "task-stale-notify-cleanup",
|
||||||
|
sessionID: "session-stale-notify-cleanup",
|
||||||
|
parentSessionID: "parent-stale-notify-cleanup",
|
||||||
|
status: "running",
|
||||||
|
startedAt: new Date(Date.now() - 31 * 60 * 1000),
|
||||||
|
})
|
||||||
|
getTaskMap(manager).set(staleTask.id, staleTask)
|
||||||
|
getPendingByParent(manager).set(staleTask.parentSessionID, new Set([staleTask.id]))
|
||||||
|
|
||||||
|
//#when
|
||||||
|
pruneStaleTasksAndNotificationsForTest(manager)
|
||||||
|
await flushBackgroundNotifications()
|
||||||
|
|
||||||
|
//#then
|
||||||
|
const retainedTask = getTaskMap(manager).get(staleTask.id)
|
||||||
|
expect(retainedTask?.status).toBe("error")
|
||||||
|
expect(getTaskMap(manager).has(staleTask.id)).toBe(true)
|
||||||
|
expect(notifications).toHaveLength(1)
|
||||||
|
expect(notifications[0]).toContain("[ALL BACKGROUND TASKS COMPLETE]")
|
||||||
|
expect(notifications[0]).toContain(staleTask.description)
|
||||||
|
expect(getCompletionTimers(manager).has(staleTask.id)).toBe(true)
|
||||||
|
expect(removeTaskCalls).toContain(staleTask.id)
|
||||||
|
|
||||||
|
manager.shutdown()
|
||||||
|
resetToastManager()
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
describe("BackgroundManager.completionTimers - Memory Leak Fix", () => {
|
describe("BackgroundManager.completionTimers - Memory Leak Fix", () => {
|
||||||
@@ -3518,7 +3595,7 @@ describe("BackgroundManager.completionTimers - Memory Leak Fix", () => {
|
|||||||
expect(completionTimers.size).toBe(0)
|
expect(completionTimers.size).toBe(0)
|
||||||
})
|
})
|
||||||
|
|
||||||
test("should cancel timer when task is deleted via session.deleted", () => {
|
test("should preserve cleanup timer when terminal task session is deleted", () => {
|
||||||
// given
|
// given
|
||||||
const manager = createBackgroundManager()
|
const manager = createBackgroundManager()
|
||||||
const task: BackgroundTask = {
|
const task: BackgroundTask = {
|
||||||
@@ -3547,7 +3624,7 @@ describe("BackgroundManager.completionTimers - Memory Leak Fix", () => {
|
|||||||
})
|
})
|
||||||
|
|
||||||
// then
|
// then
|
||||||
expect(completionTimers.has(task.id)).toBe(false)
|
expect(completionTimers.has(task.id)).toBe(true)
|
||||||
|
|
||||||
manager.shutdown()
|
manager.shutdown()
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -390,7 +390,6 @@ export class BackgroundManager {
|
|||||||
}).catch(() => {})
|
}).catch(() => {})
|
||||||
|
|
||||||
this.markForNotification(existingTask)
|
this.markForNotification(existingTask)
|
||||||
this.cleanupPendingByParent(existingTask)
|
|
||||||
this.enqueueNotificationForParent(existingTask.parentSessionID, () => this.notifyParentSession(existingTask)).catch(err => {
|
this.enqueueNotificationForParent(existingTask.parentSessionID, () => this.notifyParentSession(existingTask)).catch(err => {
|
||||||
log("[background-agent] Failed to notify on error:", err)
|
log("[background-agent] Failed to notify on error:", err)
|
||||||
})
|
})
|
||||||
@@ -661,7 +660,6 @@ export class BackgroundManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
this.markForNotification(existingTask)
|
this.markForNotification(existingTask)
|
||||||
this.cleanupPendingByParent(existingTask)
|
|
||||||
this.enqueueNotificationForParent(existingTask.parentSessionID, () => this.notifyParentSession(existingTask)).catch(err => {
|
this.enqueueNotificationForParent(existingTask.parentSessionID, () => this.notifyParentSession(existingTask)).catch(err => {
|
||||||
log("[background-agent] Failed to notify on resume error:", err)
|
log("[background-agent] Failed to notify on resume error:", err)
|
||||||
})
|
})
|
||||||
@@ -804,16 +802,14 @@ export class BackgroundManager {
|
|||||||
this.idleDeferralTimers.delete(task.id)
|
this.idleDeferralTimers.delete(task.id)
|
||||||
}
|
}
|
||||||
|
|
||||||
this.cleanupPendingByParent(task)
|
|
||||||
this.tasks.delete(task.id)
|
|
||||||
this.clearNotificationsForTask(task.id)
|
|
||||||
const toastManager = getTaskToastManager()
|
|
||||||
if (toastManager) {
|
|
||||||
toastManager.removeTask(task.id)
|
|
||||||
}
|
|
||||||
if (task.sessionID) {
|
if (task.sessionID) {
|
||||||
subagentSessions.delete(task.sessionID)
|
SessionCategoryRegistry.remove(task.sessionID)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this.markForNotification(task)
|
||||||
|
this.enqueueNotificationForParent(task.parentSessionID, () => this.notifyParentSession(task)).catch(err => {
|
||||||
|
log("[background-agent] Error in notifyParentSession for errored task:", { taskId: task.id, error: err })
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
if (event.type === "session.deleted") {
|
if (event.type === "session.deleted") {
|
||||||
@@ -834,47 +830,30 @@ export class BackgroundManager {
|
|||||||
|
|
||||||
if (tasksToCancel.size === 0) return
|
if (tasksToCancel.size === 0) return
|
||||||
|
|
||||||
|
const deletedSessionIDs = new Set<string>([sessionID])
|
||||||
|
for (const task of tasksToCancel.values()) {
|
||||||
|
if (task.sessionID) {
|
||||||
|
deletedSessionIDs.add(task.sessionID)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
for (const task of tasksToCancel.values()) {
|
for (const task of tasksToCancel.values()) {
|
||||||
if (task.status === "running" || task.status === "pending") {
|
if (task.status === "running" || task.status === "pending") {
|
||||||
void this.cancelTask(task.id, {
|
void this.cancelTask(task.id, {
|
||||||
source: "session.deleted",
|
source: "session.deleted",
|
||||||
reason: "Session deleted",
|
reason: "Session deleted",
|
||||||
skipNotification: true,
|
}).then(() => {
|
||||||
|
if (deletedSessionIDs.has(task.parentSessionID)) {
|
||||||
|
this.pendingNotifications.delete(task.parentSessionID)
|
||||||
|
}
|
||||||
}).catch(err => {
|
}).catch(err => {
|
||||||
|
if (deletedSessionIDs.has(task.parentSessionID)) {
|
||||||
|
this.pendingNotifications.delete(task.parentSessionID)
|
||||||
|
}
|
||||||
log("[background-agent] Failed to cancel task on session.deleted:", { taskId: task.id, error: err })
|
log("[background-agent] Failed to cancel task on session.deleted:", { taskId: task.id, error: err })
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
const existingTimer = this.completionTimers.get(task.id)
|
|
||||||
if (existingTimer) {
|
|
||||||
clearTimeout(existingTimer)
|
|
||||||
this.completionTimers.delete(task.id)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const idleTimer = this.idleDeferralTimers.get(task.id)
|
|
||||||
if (idleTimer) {
|
|
||||||
clearTimeout(idleTimer)
|
|
||||||
this.idleDeferralTimers.delete(task.id)
|
|
||||||
}
|
|
||||||
|
|
||||||
this.cleanupPendingByParent(task)
|
|
||||||
this.tasks.delete(task.id)
|
|
||||||
this.clearNotificationsForTask(task.id)
|
|
||||||
const toastManager = getTaskToastManager()
|
|
||||||
if (toastManager) {
|
|
||||||
toastManager.removeTask(task.id)
|
|
||||||
}
|
|
||||||
if (task.sessionID) {
|
|
||||||
subagentSessions.delete(task.sessionID)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const task of tasksToCancel.values()) {
|
|
||||||
if (task.parentSessionID) {
|
|
||||||
this.pendingNotifications.delete(task.parentSessionID)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
SessionCategoryRegistry.remove(sessionID)
|
SessionCategoryRegistry.remove(sessionID)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1094,8 +1073,6 @@ export class BackgroundManager {
|
|||||||
this.idleDeferralTimers.delete(task.id)
|
this.idleDeferralTimers.delete(task.id)
|
||||||
}
|
}
|
||||||
|
|
||||||
this.cleanupPendingByParent(task)
|
|
||||||
|
|
||||||
if (abortSession && task.sessionID) {
|
if (abortSession && task.sessionID) {
|
||||||
this.client.session.abort({
|
this.client.session.abort({
|
||||||
path: { id: task.sessionID },
|
path: { id: task.sessionID },
|
||||||
@@ -1202,9 +1179,6 @@ export class BackgroundManager {
|
|||||||
|
|
||||||
this.markForNotification(task)
|
this.markForNotification(task)
|
||||||
|
|
||||||
// Ensure pending tracking is cleaned up even if notification fails
|
|
||||||
this.cleanupPendingByParent(task)
|
|
||||||
|
|
||||||
const idleTimer = this.idleDeferralTimers.get(task.id)
|
const idleTimer = this.idleDeferralTimers.get(task.id)
|
||||||
if (idleTimer) {
|
if (idleTimer) {
|
||||||
clearTimeout(idleTimer)
|
clearTimeout(idleTimer)
|
||||||
@@ -1260,7 +1234,10 @@ export class BackgroundManager {
|
|||||||
this.pendingByParent.delete(task.parentSessionID)
|
this.pendingByParent.delete(task.parentSessionID)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
allComplete = true
|
remainingCount = Array.from(this.tasks.values())
|
||||||
|
.filter(t => t.parentSessionID === task.parentSessionID && t.id !== task.id && (t.status === "running" || t.status === "pending"))
|
||||||
|
.length
|
||||||
|
allComplete = remainingCount === 0
|
||||||
}
|
}
|
||||||
|
|
||||||
const completedTasks = allComplete
|
const completedTasks = allComplete
|
||||||
@@ -1268,7 +1245,13 @@ export class BackgroundManager {
|
|||||||
.filter(t => t.parentSessionID === task.parentSessionID && t.status !== "running" && t.status !== "pending")
|
.filter(t => t.parentSessionID === task.parentSessionID && t.status !== "running" && t.status !== "pending")
|
||||||
: []
|
: []
|
||||||
|
|
||||||
const statusText = task.status === "completed" ? "COMPLETED" : task.status === "interrupt" ? "INTERRUPTED" : "CANCELLED"
|
const statusText = task.status === "completed"
|
||||||
|
? "COMPLETED"
|
||||||
|
: task.status === "interrupt"
|
||||||
|
? "INTERRUPTED"
|
||||||
|
: task.status === "error"
|
||||||
|
? "ERROR"
|
||||||
|
: "CANCELLED"
|
||||||
const errorInfo = task.error ? `\n**Error:** ${task.error}` : ""
|
const errorInfo = task.error ? `\n**Error:** ${task.error}` : ""
|
||||||
|
|
||||||
let notification: string
|
let notification: string
|
||||||
@@ -1399,8 +1382,13 @@ Use \`background_output(task_id="${task.id}")\` to retrieve this result when rea
|
|||||||
}
|
}
|
||||||
const timer = setTimeout(() => {
|
const timer = setTimeout(() => {
|
||||||
this.completionTimers.delete(taskId)
|
this.completionTimers.delete(taskId)
|
||||||
if (this.tasks.has(taskId)) {
|
const taskToRemove = this.tasks.get(taskId)
|
||||||
|
if (taskToRemove) {
|
||||||
this.clearNotificationsForTask(taskId)
|
this.clearNotificationsForTask(taskId)
|
||||||
|
if (taskToRemove.sessionID) {
|
||||||
|
subagentSessions.delete(taskToRemove.sessionID)
|
||||||
|
SessionCategoryRegistry.remove(taskToRemove.sessionID)
|
||||||
|
}
|
||||||
this.tasks.delete(taskId)
|
this.tasks.delete(taskId)
|
||||||
log("[background-agent] Removed completed task from memory:", taskId)
|
log("[background-agent] Removed completed task from memory:", taskId)
|
||||||
}
|
}
|
||||||
@@ -1435,11 +1423,21 @@ Use \`background_output(task_id="${task.id}")\` to retrieve this result when rea
|
|||||||
task.status = "error"
|
task.status = "error"
|
||||||
task.error = errorMessage
|
task.error = errorMessage
|
||||||
task.completedAt = new Date()
|
task.completedAt = new Date()
|
||||||
|
this.taskHistory.record(task.parentSessionID, { id: task.id, sessionID: task.sessionID, agent: task.agent, description: task.description, status: "error", category: task.category, startedAt: task.startedAt, completedAt: task.completedAt })
|
||||||
if (task.concurrencyKey) {
|
if (task.concurrencyKey) {
|
||||||
this.concurrencyManager.release(task.concurrencyKey)
|
this.concurrencyManager.release(task.concurrencyKey)
|
||||||
task.concurrencyKey = undefined
|
task.concurrencyKey = undefined
|
||||||
}
|
}
|
||||||
this.cleanupPendingByParent(task)
|
const existingTimer = this.completionTimers.get(taskId)
|
||||||
|
if (existingTimer) {
|
||||||
|
clearTimeout(existingTimer)
|
||||||
|
this.completionTimers.delete(taskId)
|
||||||
|
}
|
||||||
|
const idleTimer = this.idleDeferralTimers.get(taskId)
|
||||||
|
if (idleTimer) {
|
||||||
|
clearTimeout(idleTimer)
|
||||||
|
this.idleDeferralTimers.delete(taskId)
|
||||||
|
}
|
||||||
if (wasPending) {
|
if (wasPending) {
|
||||||
const key = task.model
|
const key = task.model
|
||||||
? `${task.model.providerID}/${task.model.modelID}`
|
? `${task.model.providerID}/${task.model.modelID}`
|
||||||
@@ -1455,16 +1453,10 @@ Use \`background_output(task_id="${task.id}")\` to retrieve this result when rea
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
this.clearNotificationsForTask(taskId)
|
this.markForNotification(task)
|
||||||
const toastManager = getTaskToastManager()
|
this.enqueueNotificationForParent(task.parentSessionID, () => this.notifyParentSession(task)).catch(err => {
|
||||||
if (toastManager) {
|
log("[background-agent] Error in notifyParentSession for stale-pruned task:", { taskId: task.id, error: err })
|
||||||
toastManager.removeTask(taskId)
|
})
|
||||||
}
|
|
||||||
this.tasks.delete(taskId)
|
|
||||||
if (task.sessionID) {
|
|
||||||
subagentSessions.delete(task.sessionID)
|
|
||||||
SessionCategoryRegistry.remove(task.sessionID)
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -422,4 +422,38 @@ describe("pruneStaleTasksAndNotifications", () => {
|
|||||||
//#then
|
//#then
|
||||||
expect(pruned).toContain("old-task")
|
expect(pruned).toContain("old-task")
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it("should skip terminal tasks even when they exceeded TTL", () => {
|
||||||
|
//#given
|
||||||
|
const tasks = new Map<string, BackgroundTask>()
|
||||||
|
const oldStartedAt = new Date(Date.now() - 31 * 60 * 1000)
|
||||||
|
const terminalStatuses: BackgroundTask["status"][] = ["completed", "error", "cancelled", "interrupt"]
|
||||||
|
|
||||||
|
for (const status of terminalStatuses) {
|
||||||
|
tasks.set(status, {
|
||||||
|
id: status,
|
||||||
|
parentSessionID: "parent",
|
||||||
|
parentMessageID: "msg",
|
||||||
|
description: status,
|
||||||
|
prompt: status,
|
||||||
|
agent: "explore",
|
||||||
|
status,
|
||||||
|
startedAt: oldStartedAt,
|
||||||
|
completedAt: new Date(),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const pruned: string[] = []
|
||||||
|
|
||||||
|
//#when
|
||||||
|
pruneStaleTasksAndNotifications({
|
||||||
|
tasks,
|
||||||
|
notifications: new Map<string, BackgroundTask[]>(),
|
||||||
|
onTaskPruned: (taskId) => pruned.push(taskId),
|
||||||
|
})
|
||||||
|
|
||||||
|
//#then
|
||||||
|
expect(pruned).toEqual([])
|
||||||
|
expect(Array.from(tasks.keys())).toEqual(terminalStatuses)
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -12,6 +12,13 @@ import {
|
|||||||
TASK_TTL_MS,
|
TASK_TTL_MS,
|
||||||
} from "./constants"
|
} from "./constants"
|
||||||
|
|
||||||
|
const TERMINAL_TASK_STATUSES = new Set<BackgroundTask["status"]>([
|
||||||
|
"completed",
|
||||||
|
"error",
|
||||||
|
"cancelled",
|
||||||
|
"interrupt",
|
||||||
|
])
|
||||||
|
|
||||||
export function pruneStaleTasksAndNotifications(args: {
|
export function pruneStaleTasksAndNotifications(args: {
|
||||||
tasks: Map<string, BackgroundTask>
|
tasks: Map<string, BackgroundTask>
|
||||||
notifications: Map<string, BackgroundTask[]>
|
notifications: Map<string, BackgroundTask[]>
|
||||||
@@ -21,6 +28,8 @@ export function pruneStaleTasksAndNotifications(args: {
|
|||||||
const now = Date.now()
|
const now = Date.now()
|
||||||
|
|
||||||
for (const [taskId, task] of tasks.entries()) {
|
for (const [taskId, task] of tasks.entries()) {
|
||||||
|
if (TERMINAL_TASK_STATUSES.has(task.status)) continue
|
||||||
|
|
||||||
const timestamp = task.status === "pending"
|
const timestamp = task.status === "pending"
|
||||||
? task.queuedAt?.getTime()
|
? task.queuedAt?.getTime()
|
||||||
: task.startedAt?.getTime()
|
: task.startedAt?.getTime()
|
||||||
|
|||||||
Reference in New Issue
Block a user