opencode-immune 1.0.17 → 1.0.19
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/plugin.js +86 -29
- package/package.json +2 -2
package/dist/plugin.js
CHANGED
|
@@ -467,12 +467,12 @@ function createTodoEnforcerChatMessage(state) {
|
|
|
467
467
|
await addManagedUltraworkSession(state, sessionID);
|
|
468
468
|
await writeUltraworkMarker(state);
|
|
469
469
|
}
|
|
470
|
-
else if (sessionID && agent && record?.kind === "root") {
|
|
471
|
-
await removeManagedUltraworkSession(state, sessionID, `session taken over by agent \"${agent}\"`);
|
|
472
|
-
}
|
|
473
470
|
else if (sessionID && agent && record?.kind === "child") {
|
|
474
471
|
await updateManagedSessionAgent(state, sessionID, agent);
|
|
475
472
|
}
|
|
473
|
+
// NOTE: Do NOT remove managed root sessions when agent changes.
|
|
474
|
+
// 0-ultrawork delegates to subagents (1-van, 7-backlog, etc.) which
|
|
475
|
+
// may appear as agent in chat.message but the session is still managed.
|
|
476
476
|
// On user message, check previous assistant turn's counters
|
|
477
477
|
// then reset for next turn
|
|
478
478
|
if (state.toolCallCount > 3 && !state.todoWriteUsed) {
|
|
@@ -791,12 +791,11 @@ function createFallbackModels(state) {
|
|
|
791
791
|
}
|
|
792
792
|
}
|
|
793
793
|
}
|
|
794
|
-
else if (getManagedSession(state, input.sessionID)?.kind === "root") {
|
|
795
|
-
await removeManagedUltraworkSession(state, input.sessionID, `session switched to agent \"${input.agent}\"`);
|
|
796
|
-
}
|
|
797
794
|
else if (getManagedSession(state, input.sessionID)?.kind === "child") {
|
|
798
795
|
await updateManagedSessionAgent(state, input.sessionID, input.agent);
|
|
799
796
|
}
|
|
797
|
+
// NOTE: Do NOT remove managed root sessions when agent changes in chat.params.
|
|
798
|
+
// Subagent calls from 0-ultrawork (1-van, 7-backlog, etc.) use the same session.
|
|
800
799
|
// Log model and agent for observability
|
|
801
800
|
const modelId = input.model && "id" in input.model
|
|
802
801
|
? input.model.id
|
|
@@ -936,28 +935,8 @@ function createMultiCycleHandler(state) {
|
|
|
936
935
|
await clearUltraworkMarker(state);
|
|
937
936
|
console.log("[opencode-immune] Multi-Cycle: ALL_CYCLES_COMPLETE detected, ultrawork marker cleared.");
|
|
938
937
|
}
|
|
939
|
-
|
|
940
|
-
|
|
941
|
-
const managedSession = getManagedSession(state, sessionID);
|
|
942
|
-
if (sessionID &&
|
|
943
|
-
messageRole === "assistant" &&
|
|
944
|
-
RATE_LIMIT_MESSAGE_PATTERN.test(messageContent)) {
|
|
945
|
-
if (managedSession && !managedSession.fallbackModel) {
|
|
946
|
-
await setSessionFallbackModel(state, sessionID, RATE_LIMIT_FALLBACK_MODEL);
|
|
947
|
-
console.log(`[opencode-immune] Rate limit message detected in chat output for session ${sessionID}. ` +
|
|
948
|
-
`Fallback model pinned to ${RATE_LIMIT_FALLBACK_MODEL.providerID}/${RATE_LIMIT_FALLBACK_MODEL.modelID}.`);
|
|
949
|
-
}
|
|
950
|
-
if (managedSession) {
|
|
951
|
-
scheduleManagedSessionRetry(state, sessionID, {
|
|
952
|
-
delayMs: 1_000,
|
|
953
|
-
reason: "rate-limit message fallback",
|
|
954
|
-
countAgainstBudget: false,
|
|
955
|
-
abortBeforePrompt: true,
|
|
956
|
-
});
|
|
957
|
-
}
|
|
958
|
-
}
|
|
959
|
-
// ── PRE_COMMIT: execute /commit ──
|
|
960
|
-
if (messageContent.includes(PRE_COMMIT_MARKER) && !state.commitPending) {
|
|
938
|
+
// ── PRE_COMMIT: execute /commit (checked before managed session guard) ──
|
|
939
|
+
if (messageContent.includes(PRE_COMMIT_MARKER) && !state.commitPending && sessionID) {
|
|
961
940
|
state.commitPending = true;
|
|
962
941
|
console.log("[opencode-immune] Multi-Cycle: PRE_COMMIT detected, executing /commit...");
|
|
963
942
|
// Small delay to let the message finish rendering
|
|
@@ -980,7 +959,7 @@ function createMultiCycleHandler(state) {
|
|
|
980
959
|
}
|
|
981
960
|
}, 2_000);
|
|
982
961
|
}
|
|
983
|
-
// ── CYCLE_COMPLETE: create new session ──
|
|
962
|
+
// ── CYCLE_COMPLETE: create new session (checked before managed session guard) ──
|
|
984
963
|
if (messageContent.includes(CYCLE_COMPLETE_MARKER)) {
|
|
985
964
|
state.cycleCount++;
|
|
986
965
|
if (state.cycleCount >= MAX_CYCLES) {
|
|
@@ -1031,6 +1010,27 @@ function createMultiCycleHandler(state) {
|
|
|
1031
1010
|
}
|
|
1032
1011
|
}, 8_000); // 8s delay: let /commit finish first
|
|
1033
1012
|
}
|
|
1013
|
+
if (!isManagedRootUltraworkSession(state, sessionID))
|
|
1014
|
+
return;
|
|
1015
|
+
const managedSession = getManagedSession(state, sessionID);
|
|
1016
|
+
// Rate-limit message detection (only for managed root sessions)
|
|
1017
|
+
if (sessionID &&
|
|
1018
|
+
messageRole === "assistant" &&
|
|
1019
|
+
RATE_LIMIT_MESSAGE_PATTERN.test(messageContent)) {
|
|
1020
|
+
if (managedSession && !managedSession.fallbackModel) {
|
|
1021
|
+
await setSessionFallbackModel(state, sessionID, RATE_LIMIT_FALLBACK_MODEL);
|
|
1022
|
+
console.log(`[opencode-immune] Rate limit message detected in chat output for session ${sessionID}. ` +
|
|
1023
|
+
`Fallback model pinned to ${RATE_LIMIT_FALLBACK_MODEL.providerID}/${RATE_LIMIT_FALLBACK_MODEL.modelID}.`);
|
|
1024
|
+
}
|
|
1025
|
+
if (managedSession) {
|
|
1026
|
+
scheduleManagedSessionRetry(state, sessionID, {
|
|
1027
|
+
delayMs: 1_000,
|
|
1028
|
+
reason: "rate-limit message fallback",
|
|
1029
|
+
countAgainstBudget: false,
|
|
1030
|
+
abortBeforePrompt: true,
|
|
1031
|
+
});
|
|
1032
|
+
}
|
|
1033
|
+
}
|
|
1034
1034
|
};
|
|
1035
1035
|
}
|
|
1036
1036
|
// ═══════════════════════════════════════════════════════════════════════════════
|
|
@@ -1044,6 +1044,7 @@ async function server(input) {
|
|
|
1044
1044
|
if (markerActive) {
|
|
1045
1045
|
const recovery = await parseTasksFile(state.input.directory);
|
|
1046
1046
|
if (recovery && recovery.phase !== "ARCHIVE: DONE") {
|
|
1047
|
+
// Active task exists with incomplete phases — resume it
|
|
1047
1048
|
state.recoveryContext = recovery;
|
|
1048
1049
|
state.autoResumeAttempted = true;
|
|
1049
1050
|
console.log(`[opencode-immune] Plugin init: ultrawork marker active, recovery context loaded: ` +
|
|
@@ -1085,6 +1086,62 @@ async function server(input) {
|
|
|
1085
1086
|
}
|
|
1086
1087
|
}, 5_000);
|
|
1087
1088
|
}
|
|
1089
|
+
else {
|
|
1090
|
+
// No active task — check if backlog has pending work to start a new cycle
|
|
1091
|
+
try {
|
|
1092
|
+
const backlogPath = (0, path_1.join)(state.input.directory, "memory-bank", "backlog.md");
|
|
1093
|
+
const backlogContent = await (0, promises_1.readFile)(backlogPath, "utf-8");
|
|
1094
|
+
const hasPendingTasks = /- \[ \]/.test(backlogContent);
|
|
1095
|
+
if (hasPendingTasks) {
|
|
1096
|
+
state.autoResumeAttempted = true;
|
|
1097
|
+
console.log(`[opencode-immune] Plugin init: no active task but backlog has pending items. ` +
|
|
1098
|
+
`Will create new session to start next cycle.`);
|
|
1099
|
+
setTimeout(async () => {
|
|
1100
|
+
try {
|
|
1101
|
+
const createResult = await state.input.client.session.create({
|
|
1102
|
+
body: {
|
|
1103
|
+
title: `AUTO-CYCLE: next backlog task`,
|
|
1104
|
+
},
|
|
1105
|
+
});
|
|
1106
|
+
const newSessionData = createResult?.data;
|
|
1107
|
+
const newSessionID = newSessionData?.id;
|
|
1108
|
+
if (!newSessionID) {
|
|
1109
|
+
console.error("[opencode-immune] Auto-cycle: Failed to create session — no session ID returned.");
|
|
1110
|
+
return;
|
|
1111
|
+
}
|
|
1112
|
+
console.log(`[opencode-immune] Auto-cycle: New session created: ${newSessionID}`);
|
|
1113
|
+
await addManagedUltraworkSession(state, newSessionID);
|
|
1114
|
+
await state.input.client.session.promptAsync({
|
|
1115
|
+
body: {
|
|
1116
|
+
agent: ULTRAWORK_AGENT,
|
|
1117
|
+
parts: [
|
|
1118
|
+
{
|
|
1119
|
+
type: "text",
|
|
1120
|
+
text: `[AUTO-CYCLE] Continue processing task backlog. Read memory-bank/tasks.md and memory-bank/backlog.md, pick the next pending task, and run the full pipeline.`,
|
|
1121
|
+
},
|
|
1122
|
+
],
|
|
1123
|
+
},
|
|
1124
|
+
path: { id: newSessionID },
|
|
1125
|
+
});
|
|
1126
|
+
console.log(`[opencode-immune] Auto-cycle prompt sent to new session ${newSessionID}`);
|
|
1127
|
+
}
|
|
1128
|
+
catch (err) {
|
|
1129
|
+
console.error("[opencode-immune] Auto-cycle: Failed to create session or send prompt:", err);
|
|
1130
|
+
}
|
|
1131
|
+
}, 5_000);
|
|
1132
|
+
}
|
|
1133
|
+
else {
|
|
1134
|
+
// No active task and no pending backlog — clear marker
|
|
1135
|
+
await clearUltraworkMarker(state);
|
|
1136
|
+
console.log(`[opencode-immune] Plugin init: no active task and no pending backlog. Marker cleared.`);
|
|
1137
|
+
}
|
|
1138
|
+
}
|
|
1139
|
+
catch {
|
|
1140
|
+
// backlog.md doesn't exist or can't be read — clear marker
|
|
1141
|
+
await clearUltraworkMarker(state);
|
|
1142
|
+
console.log(`[opencode-immune] Plugin init: no active task, backlog unreadable. Marker cleared.`);
|
|
1143
|
+
}
|
|
1144
|
+
}
|
|
1088
1145
|
}
|
|
1089
1146
|
console.log(`[opencode-immune] Plugin initialized. Directory: ${input.directory}`);
|
|
1090
1147
|
// Compose tool.execute.after handlers:
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "opencode-immune",
|
|
3
|
-
"version": "1.0.
|
|
3
|
+
"version": "1.0.19",
|
|
4
4
|
"description": "OpenCode plugin: session recovery, auto-retry, multi-cycle automation, context monitoring",
|
|
5
5
|
"exports": {
|
|
6
6
|
"./server": "./dist/plugin.js"
|
|
@@ -14,7 +14,7 @@
|
|
|
14
14
|
"prepublishOnly": "npm run build"
|
|
15
15
|
},
|
|
16
16
|
"dependencies": {
|
|
17
|
-
"@opencode-ai/plugin": "1.4.
|
|
17
|
+
"@opencode-ai/plugin": "1.4.3"
|
|
18
18
|
},
|
|
19
19
|
"devDependencies": {
|
|
20
20
|
"@types/node": "^25.5.2",
|