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.
Files changed (2) hide show
  1. package/dist/plugin.js +86 -29
  2. 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
- if (!isManagedRootUltraworkSession(state, sessionID))
940
- return;
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.17",
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.2"
17
+ "@opencode-ai/plugin": "1.4.3"
18
18
  },
19
19
  "devDependencies": {
20
20
  "@types/node": "^25.5.2",