From 1e08ebe82449e4ea3c74290c79783e8281c5322d Mon Sep 17 00:00:00 2001
From: Ammar
Date: Sat, 17 Jan 2026 20:07:16 -0600
Subject: [PATCH 01/11] =?UTF-8?q?=F0=9F=A4=96=20fix:=20add=20compact=20ret?=
=?UTF-8?q?ry=20action=20for=20stream=20errors?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
.../Messages/ChatBarrier/RetryBarrier.tsx | 284 ++---------------
.../Messages/MessageListContext.tsx | 31 ++
.../Messages/StreamErrorMessage.tsx | 27 +-
src/browser/hooks/useCompactAndRetry.ts | 291 ++++++++++++++++++
src/common/constants/events.ts | 9 +
5 files changed, 375 insertions(+), 267 deletions(-)
create mode 100644 src/browser/components/Messages/MessageListContext.tsx
create mode 100644 src/browser/hooks/useCompactAndRetry.ts
diff --git a/src/browser/components/Messages/ChatBarrier/RetryBarrier.tsx b/src/browser/components/Messages/ChatBarrier/RetryBarrier.tsx
index 335852690b..9ede327554 100644
--- a/src/browser/components/Messages/ChatBarrier/RetryBarrier.tsx
+++ b/src/browser/components/Messages/ChatBarrier/RetryBarrier.tsx
@@ -1,19 +1,8 @@
-import React, { useEffect, useMemo, useRef, useState } from "react";
-import { useAPI } from "@/browser/contexts/API";
-import { buildSendMessageOptions } from "@/browser/hooks/useSendMessageOptions";
+import React, { useEffect, useMemo, useState } from "react";
+import { useCompactAndRetry } from "@/browser/hooks/useCompactAndRetry";
import { usePersistedState, updatePersistedState } from "@/browser/hooks/usePersistedState";
import type { RetryState } from "@/browser/hooks/useResumeManager";
import { useWorkspaceState } from "@/browser/stores/WorkspaceStore";
-import {
- getExplicitCompactionSuggestion,
- getHigherContextCompactionSuggestion,
- type CompactionSuggestion,
-} from "@/browser/utils/compaction/suggestion";
-import {
- buildCompactionEditText,
- formatCompactionCommandLine,
-} from "@/browser/utils/compaction/format";
-import { executeCompaction } from "@/browser/utils/chatCommands";
import {
isEligibleForAutoRetry,
isNonRetryableSendError,
@@ -21,15 +10,8 @@ import {
import { calculateBackoffDelay, createManualRetryState } from "@/browser/utils/messages/retryState";
import { KEYBINDS, formatKeybind } from "@/browser/utils/ui/keybinds";
import { CUSTOM_EVENTS, createCustomEvent } from "@/common/constants/events";
-import {
- getAutoRetryKey,
- getRetryStateKey,
- PREFERRED_COMPACTION_MODEL_KEY,
- VIM_ENABLED_KEY,
-} from "@/common/constants/storage";
+import { getAutoRetryKey, getRetryStateKey, VIM_ENABLED_KEY } from "@/common/constants/storage";
import { cn } from "@/common/lib/utils";
-import type { ImagePart, ProvidersConfigMap } from "@/common/orpc/types";
-import { buildContinueMessage, type DisplayedMessage } from "@/common/types/message";
import { formatSendMessageError } from "@/common/utils/errors/formatSendError";
import { formatTokens } from "@/common/utils/tokens/tokenMeterUtils";
@@ -42,18 +24,6 @@ function formatContextTokens(tokens: number): string {
return formatTokens(tokens).replace(/\.0([kM])$/, "$1");
}
-function findTriggerUserMessage(
- messages: DisplayedMessage[]
-): Extract | null {
- for (let i = messages.length - 1; i >= 0; i--) {
- const msg = messages[i];
- if (msg.type === "user") {
- return msg;
- }
- }
-
- return null;
-}
const defaultRetryState: RetryState = {
attempt: 0,
retryStartTime: Date.now(),
@@ -63,82 +33,14 @@ export const RetryBarrier: React.FC = ({ workspaceId, classNa
// Get workspace state for computing effective autoRetry
const workspaceState = useWorkspaceState(workspaceId);
- const { api } = useAPI();
- const [isRetryingWithCompaction, setIsRetryingWithCompaction] = useState(false);
- const isMountedRef = useRef(true);
- useEffect(() => {
- return () => {
- isMountedRef.current = false;
- };
- }, []);
-
- const [providersConfig, setProvidersConfig] = useState(null);
-
- const lastMessage = workspaceState
- ? workspaceState.messages[workspaceState.messages.length - 1]
- : undefined;
-
- const isContextExceeded =
- lastMessage?.type === "stream-error" && lastMessage.errorType === "context_exceeded";
-
- // Check if we're in a compaction recovery flow: the last user message was a compaction request
- // that failed. This persists the compaction UI even if the retry fails with a different error.
- const triggerUserMessage = useMemo(() => {
- if (!workspaceState) return null;
- return findTriggerUserMessage(workspaceState.messages);
- }, [workspaceState]);
-
- const isCompactionRecoveryFlow =
- lastMessage?.type === "stream-error" && !!triggerUserMessage?.compactionRequest;
-
- // Show compaction UI if either: original context_exceeded OR we're retrying a failed compaction
- const showCompactionUI = isContextExceeded || isCompactionRecoveryFlow;
-
- // This is a rare error state; we only need a snapshot of provider config to make a
- // best-effort suggestion (no subscriptions / real-time updates required).
- useEffect(() => {
- if (!api) return;
- if (!showCompactionUI) return;
- if (providersConfig) return;
-
- let active = true;
- void (async () => {
- try {
- const cfg = await api.providers.getConfig();
- if (active) {
- setProvidersConfig(cfg);
- }
- } catch {
- // Ignore failures fetching config (we just won't show a suggestion).
- }
- })();
-
- return () => {
- active = false;
- };
- }, [api, showCompactionUI, providersConfig]);
-
- // For compaction recovery, use the model from the original compaction request or fall back to workspace model
- const compactionTargetModel = useMemo(() => {
- if (!showCompactionUI) return null;
- // If retrying a failed compaction, use the model from that request
- if (triggerUserMessage?.compactionRequest?.parsed.model) {
- return triggerUserMessage.compactionRequest.parsed.model;
- }
- // Otherwise use the model from the error or workspace
- if (lastMessage?.type === "stream-error") {
- return lastMessage.model ?? workspaceState?.currentModel ?? null;
- }
- return workspaceState?.currentModel ?? null;
- }, [showCompactionUI, triggerUserMessage, lastMessage, workspaceState?.currentModel]);
-
- // Read preferences from localStorage
-
- const [preferredCompactionModel] = usePersistedState(
- PREFERRED_COMPACTION_MODEL_KEY,
- "", // Default to empty (use workspace model)
- { listener: true }
- );
+ const {
+ showCompactionUI,
+ compactionSuggestion,
+ hasCompactionRequest,
+ hasTriggerUserMessage,
+ isRetryingWithCompaction,
+ retryWithCompaction,
+ } = useCompactAndRetry({ workspaceId });
const [autoRetry, setAutoRetry] = usePersistedState(
getAutoRetryKey(workspaceId),
true, // Default to true
@@ -203,160 +105,6 @@ export const RetryBarrier: React.FC = ({ workspaceId, classNa
return () => clearInterval(interval);
}, [autoRetry, attempt, retryStartTime]);
- const compactionSuggestion = useMemo(() => {
- // Opportunistic: only attempt suggestions when we can confidently identify the model.
- if (!showCompactionUI || !compactionTargetModel) {
- return null;
- }
-
- // If we're retrying a failed compaction request, prefer a larger-context model to recover.
- if (isCompactionRecoveryFlow) {
- return getHigherContextCompactionSuggestion({
- currentModel: compactionTargetModel,
- providersConfig,
- });
- }
-
- const preferred = preferredCompactionModel.trim();
- if (preferred.length > 0) {
- const explicit = getExplicitCompactionSuggestion({
- modelId: preferred,
- providersConfig,
- });
- if (explicit) {
- return explicit;
- }
- }
-
- return getHigherContextCompactionSuggestion({
- currentModel: compactionTargetModel,
- providersConfig,
- });
- }, [
- compactionTargetModel,
- showCompactionUI,
- isCompactionRecoveryFlow,
- providersConfig,
- preferredCompactionModel,
- ]);
-
- async function handleRetryWithCompaction(): Promise {
- const insertIntoChatInput = (text: string, imageParts?: ImagePart[]): void => {
- window.dispatchEvent(
- createCustomEvent(CUSTOM_EVENTS.INSERT_TO_CHAT_INPUT, {
- text,
- mode: "replace",
- imageParts,
- })
- );
- };
-
- if (!compactionSuggestion) {
- insertIntoChatInput("/compact\n");
- return;
- }
-
- const suggestedCommandLine = formatCompactionCommandLine({
- model: compactionSuggestion.modelArg,
- });
-
- if (!api) {
- insertIntoChatInput(suggestedCommandLine + "\n");
- return;
- }
-
- if (isMountedRef.current) {
- setIsRetryingWithCompaction(true);
- }
- try {
- // Read fresh values at click-time (workspace might have switched models).
- const sendMessageOptions = buildSendMessageOptions(workspaceId);
-
- // Best-effort: fall back to the nearest user message if we can't find the exact one.
- const source = triggerUserMessage;
-
- if (!source) {
- insertIntoChatInput(suggestedCommandLine + "\n");
- return;
- }
-
- if (source.compactionRequest) {
- const maxOutputTokens = source.compactionRequest.parsed.maxOutputTokens;
- const continueMessage = source.compactionRequest.parsed.continueMessage;
-
- const result = await executeCompaction({
- api,
- workspaceId,
- sendMessageOptions,
- model: compactionSuggestion.modelId,
- maxOutputTokens,
- continueMessage,
- });
-
- if (!result.success) {
- console.error("Failed to retry compaction:", result.error);
-
- const rawCommand = formatCompactionCommandLine({
- model: compactionSuggestion.modelArg,
- maxOutputTokens,
- });
-
- const fallbackText = buildCompactionEditText({
- rawCommand,
- parsed: {
- model: compactionSuggestion.modelArg,
- maxOutputTokens,
- continueMessage,
- },
- });
-
- const shouldAppendNewline =
- !continueMessage?.text || continueMessage.text.trim().length === 0;
-
- insertIntoChatInput(
- fallbackText + (shouldAppendNewline ? "\n" : ""),
- continueMessage?.imageParts
- );
- }
-
- return;
- }
-
- const continueMessage = buildContinueMessage({
- text: source.content,
- imageParts: source.imageParts,
- reviews: source.reviews,
- model: sendMessageOptions.model,
- agentId: sendMessageOptions.agentId ?? "exec",
- });
-
- if (!continueMessage) {
- insertIntoChatInput(suggestedCommandLine + "\n");
- return;
- }
-
- const result = await executeCompaction({
- api,
- workspaceId,
- sendMessageOptions,
- model: compactionSuggestion.modelId,
- continueMessage,
- });
-
- if (!result.success) {
- console.error("Failed to start compaction:", result.error);
- insertIntoChatInput(suggestedCommandLine + "\n" + source.content, source.imageParts);
- }
- } catch (error) {
- console.error("Failed to retry with compaction", error);
- insertIntoChatInput(suggestedCommandLine + "\n");
- } finally {
- if (isMountedRef.current) {
- setIsRetryingWithCompaction(false);
- }
- }
- }
-
// Manual retry handler (user-initiated, immediate)
// Emits event to useResumeManager instead of calling resumeStream directly
// This keeps all retry logic centralized in one place
@@ -462,15 +210,19 @@ export const RetryBarrier: React.FC = ({ workspaceId, classNa
);
} else {
- const onClick = showCompactionUI ? () => void handleRetryWithCompaction() : handleManualRetry;
+ const onClick = showCompactionUI
+ ? () => {
+ retryWithCompaction().catch(() => undefined);
+ }
+ : handleManualRetry;
let label = "Retry";
if (showCompactionUI) {
if (isRetryingWithCompaction) {
label = "Starting...";
- } else if (!compactionSuggestion || !triggerUserMessage) {
+ } else if (!compactionSuggestion || !hasTriggerUserMessage) {
label = "Insert /compact";
- } else if (triggerUserMessage.compactionRequest) {
+ } else if (hasCompactionRequest) {
label = "Retry compaction";
} else {
label = "Compact & retry";
diff --git a/src/browser/components/Messages/MessageListContext.tsx b/src/browser/components/Messages/MessageListContext.tsx
new file mode 100644
index 0000000000..fad3aa3bb7
--- /dev/null
+++ b/src/browser/components/Messages/MessageListContext.tsx
@@ -0,0 +1,31 @@
+import React, { createContext, useContext } from "react";
+
+interface MessageListContextValue {
+ workspaceId: string;
+ latestMessageId: string | null;
+}
+
+const MessageListContext = createContext(null);
+
+interface MessageListProviderProps {
+ value: MessageListContextValue;
+ children: React.ReactNode;
+}
+
+export const MessageListProvider: React.FC = (props) => {
+ return (
+ {props.children}
+ );
+};
+
+export function useOptionalMessageListContext(): MessageListContextValue | null {
+ return useContext(MessageListContext);
+}
+
+export function useMessageListContext(): MessageListContextValue {
+ const context = useContext(MessageListContext);
+ if (!context) {
+ throw new Error("useMessageListContext must be used within MessageListProvider");
+ }
+ return context;
+}
diff --git a/src/browser/components/Messages/StreamErrorMessage.tsx b/src/browser/components/Messages/StreamErrorMessage.tsx
index 1709a13eae..2b5f2c0ad2 100644
--- a/src/browser/components/Messages/StreamErrorMessage.tsx
+++ b/src/browser/components/Messages/StreamErrorMessage.tsx
@@ -4,6 +4,7 @@ import { Button } from "@/browser/components/ui/button";
import { Tooltip, TooltipContent, TooltipTrigger } from "@/browser/components/ui/tooltip";
import { CUSTOM_EVENTS, createCustomEvent } from "@/common/constants/events";
import { cn } from "@/common/lib/utils";
+import { useOptionalMessageListContext } from "./MessageListContext";
import type { DisplayedMessage } from "@/common/types/message";
interface StreamErrorMessageProps {
@@ -11,8 +12,31 @@ interface StreamErrorMessageProps {
className?: string;
}
-// Note: RetryBarrier handles retry actions. This component only displays the error.
+// RetryBarrier handles auto-retry; this component exposes a compact-and-retry shortcut.
export const StreamErrorMessage: React.FC = ({ message, className }) => {
+ const messageListContext = useOptionalMessageListContext();
+ const latestMessageId = messageListContext?.latestMessageId ?? null;
+ const workspaceId = messageListContext?.workspaceId ?? null;
+ const isLatestMessage = latestMessageId === message.id;
+
+ const showCompactRetry =
+ message.errorType === "context_exceeded" && !!workspaceId && isLatestMessage;
+
+ const compactRetryAction = showCompactRetry ? (
+
+ ) : null;
const debugAction = (
@@ -68,6 +92,7 @@ export const StreamErrorMessage: React.FC = ({ message,
{message.errorType}
+ {compactRetryAction}
{showCount && (
×{message.errorCount}
diff --git a/src/browser/hooks/useCompactAndRetry.ts b/src/browser/hooks/useCompactAndRetry.ts
new file mode 100644
index 0000000000..8ad8de319a
--- /dev/null
+++ b/src/browser/hooks/useCompactAndRetry.ts
@@ -0,0 +1,291 @@
+import { useCallback, useEffect, useMemo, useRef, useState } from "react";
+import { useAPI } from "@/browser/contexts/API";
+import { buildSendMessageOptions } from "@/browser/hooks/useSendMessageOptions";
+import { usePersistedState } from "@/browser/hooks/usePersistedState";
+import { useWorkspaceState } from "@/browser/stores/WorkspaceStore";
+import {
+ buildCompactionEditText,
+ formatCompactionCommandLine,
+} from "@/browser/utils/compaction/format";
+import {
+ getExplicitCompactionSuggestion,
+ getHigherContextCompactionSuggestion,
+ type CompactionSuggestion,
+} from "@/browser/utils/compaction/suggestion";
+import { executeCompaction } from "@/browser/utils/chatCommands";
+import { CUSTOM_EVENTS, createCustomEvent, type CustomEventType } from "@/common/constants/events";
+import { PREFERRED_COMPACTION_MODEL_KEY } from "@/common/constants/storage";
+import type { ImagePart, ProvidersConfigMap } from "@/common/orpc/types";
+import { buildContinueMessage, type DisplayedMessage } from "@/common/types/message";
+
+interface CompactAndRetryState {
+ showCompactionUI: boolean;
+ compactionSuggestion: CompactionSuggestion | null;
+ isRetryingWithCompaction: boolean;
+ hasTriggerUserMessage: boolean;
+ hasCompactionRequest: boolean;
+ retryWithCompaction: () => Promise;
+}
+
+function findTriggerUserMessage(
+ messages: DisplayedMessage[]
+): Extract | null {
+ for (let i = messages.length - 1; i >= 0; i--) {
+ const msg = messages[i];
+ if (msg.type === "user") {
+ return msg;
+ }
+ }
+
+ return null;
+}
+
+export function useCompactAndRetry(props: { workspaceId: string }): CompactAndRetryState {
+ const workspaceState = useWorkspaceState(props.workspaceId);
+ const { api } = useAPI();
+ const [providersConfig, setProvidersConfig] = useState(null);
+ const [isRetryingWithCompaction, setIsRetryingWithCompaction] = useState(false);
+ const isMountedRef = useRef(true);
+
+ useEffect(() => {
+ return () => {
+ isMountedRef.current = false;
+ };
+ }, []);
+
+ const lastMessage = workspaceState
+ ? workspaceState.messages[workspaceState.messages.length - 1]
+ : undefined;
+
+ const triggerUserMessage = useMemo(() => {
+ if (!workspaceState) return null;
+ return findTriggerUserMessage(workspaceState.messages);
+ }, [workspaceState]);
+
+ const isCompactionRecoveryFlow =
+ lastMessage?.type === "stream-error" && !!triggerUserMessage?.compactionRequest;
+
+ const isContextExceeded =
+ lastMessage?.type === "stream-error" && lastMessage.errorType === "context_exceeded";
+
+ const showCompactionUI = isContextExceeded || isCompactionRecoveryFlow;
+
+ const [preferredCompactionModel] = usePersistedState(PREFERRED_COMPACTION_MODEL_KEY, "", {
+ listener: true,
+ });
+
+ useEffect(() => {
+ if (!api) return;
+ if (!showCompactionUI) return;
+ if (providersConfig) return;
+
+ let active = true;
+ const fetchProvidersConfig = async () => {
+ try {
+ const cfg = await api.providers.getConfig();
+ if (active) {
+ setProvidersConfig(cfg);
+ }
+ } catch {
+ // Ignore failures fetching config (we just won't show a suggestion).
+ }
+ };
+
+ fetchProvidersConfig().catch(() => undefined);
+
+ return () => {
+ active = false;
+ };
+ }, [api, showCompactionUI, providersConfig]);
+
+ const compactionTargetModel = useMemo(() => {
+ if (!showCompactionUI) return null;
+ if (triggerUserMessage?.compactionRequest?.parsed.model) {
+ return triggerUserMessage.compactionRequest.parsed.model;
+ }
+ if (lastMessage?.type === "stream-error") {
+ return lastMessage.model ?? workspaceState?.currentModel ?? null;
+ }
+ return workspaceState?.currentModel ?? null;
+ }, [showCompactionUI, triggerUserMessage, lastMessage, workspaceState?.currentModel]);
+
+ const compactionSuggestion = useMemo(() => {
+ if (!showCompactionUI || !compactionTargetModel) {
+ return null;
+ }
+
+ if (isCompactionRecoveryFlow) {
+ return getHigherContextCompactionSuggestion({
+ currentModel: compactionTargetModel,
+ providersConfig,
+ });
+ }
+
+ const preferred = preferredCompactionModel.trim();
+ if (preferred.length > 0) {
+ const explicit = getExplicitCompactionSuggestion({
+ modelId: preferred,
+ providersConfig,
+ });
+ if (explicit) {
+ return explicit;
+ }
+ }
+
+ return getHigherContextCompactionSuggestion({
+ currentModel: compactionTargetModel,
+ providersConfig,
+ });
+ }, [
+ compactionTargetModel,
+ showCompactionUI,
+ isCompactionRecoveryFlow,
+ providersConfig,
+ preferredCompactionModel,
+ ]);
+
+ const retryWithCompaction = useCallback(async (): Promise => {
+ const insertIntoChatInput = (text: string, imageParts?: ImagePart[]): void => {
+ window.dispatchEvent(
+ createCustomEvent(CUSTOM_EVENTS.INSERT_TO_CHAT_INPUT, {
+ text,
+ mode: "replace",
+ imageParts,
+ })
+ );
+ };
+
+ if (!compactionSuggestion) {
+ insertIntoChatInput("/compact\n");
+ return;
+ }
+
+ const suggestedCommandLine = formatCompactionCommandLine({
+ model: compactionSuggestion.modelArg,
+ });
+
+ if (!api) {
+ insertIntoChatInput(suggestedCommandLine + "\n");
+ return;
+ }
+
+ if (isMountedRef.current) {
+ setIsRetryingWithCompaction(true);
+ }
+ try {
+ const sendMessageOptions = buildSendMessageOptions(props.workspaceId);
+ const source = triggerUserMessage;
+
+ if (!source) {
+ insertIntoChatInput(suggestedCommandLine + "\n");
+ return;
+ }
+
+ if (source.compactionRequest) {
+ const maxOutputTokens = source.compactionRequest.parsed.maxOutputTokens;
+ const continueMessage = source.compactionRequest.parsed.continueMessage;
+
+ const result = await executeCompaction({
+ api,
+ workspaceId: props.workspaceId,
+ sendMessageOptions,
+ model: compactionSuggestion.modelId,
+ maxOutputTokens,
+ continueMessage,
+ });
+
+ if (!result.success) {
+ console.error("Failed to retry compaction:", result.error);
+
+ const rawCommand = formatCompactionCommandLine({
+ model: compactionSuggestion.modelArg,
+ maxOutputTokens,
+ });
+
+ const fallbackText = buildCompactionEditText({
+ rawCommand,
+ parsed: {
+ model: compactionSuggestion.modelArg,
+ maxOutputTokens,
+ continueMessage,
+ },
+ });
+
+ const shouldAppendNewline =
+ !continueMessage?.text || continueMessage.text.trim().length === 0;
+
+ insertIntoChatInput(
+ fallbackText + (shouldAppendNewline ? "\n" : ""),
+ continueMessage?.imageParts
+ );
+ }
+
+ return;
+ }
+
+ const continueMessage = buildContinueMessage({
+ text: source.content,
+ imageParts: source.imageParts,
+ reviews: source.reviews,
+ model: sendMessageOptions.model,
+ agentId: sendMessageOptions.agentId ?? "exec",
+ });
+
+ if (!continueMessage) {
+ insertIntoChatInput(suggestedCommandLine + "\n");
+ return;
+ }
+
+ const result = await executeCompaction({
+ api,
+ workspaceId: props.workspaceId,
+ sendMessageOptions,
+ model: compactionSuggestion.modelId,
+ continueMessage,
+ });
+
+ if (!result.success) {
+ console.error("Failed to start compaction:", result.error);
+ insertIntoChatInput(suggestedCommandLine + "\n" + source.content, source.imageParts);
+ }
+ } catch (error) {
+ console.error("Failed to retry with compaction", error);
+ insertIntoChatInput(suggestedCommandLine + "\n");
+ } finally {
+ if (isMountedRef.current) {
+ setIsRetryingWithCompaction(false);
+ }
+ }
+ }, [api, compactionSuggestion, props.workspaceId, triggerUserMessage]);
+
+ useEffect(() => {
+ if (!showCompactionUI) return;
+
+ const handleCompactRetryRequested = (event: Event) => {
+ const customEvent = event as CustomEventType<
+ typeof CUSTOM_EVENTS.COMPACT_AND_RETRY_REQUESTED
+ >;
+ if (customEvent.detail.workspaceId !== props.workspaceId) return;
+ if (isRetryingWithCompaction) return;
+ retryWithCompaction().catch(() => undefined);
+ };
+
+ window.addEventListener(CUSTOM_EVENTS.COMPACT_AND_RETRY_REQUESTED, handleCompactRetryRequested);
+
+ return () => {
+ window.removeEventListener(
+ CUSTOM_EVENTS.COMPACT_AND_RETRY_REQUESTED,
+ handleCompactRetryRequested
+ );
+ };
+ }, [isRetryingWithCompaction, props.workspaceId, retryWithCompaction, showCompactionUI]);
+
+ return {
+ showCompactionUI,
+ compactionSuggestion,
+ isRetryingWithCompaction,
+ hasTriggerUserMessage: !!triggerUserMessage,
+ hasCompactionRequest: !!triggerUserMessage?.compactionRequest,
+ retryWithCompaction,
+ };
+}
diff --git a/src/common/constants/events.ts b/src/common/constants/events.ts
index 1ba4b4520a..8bbe5cb5e1 100644
--- a/src/common/constants/events.ts
+++ b/src/common/constants/events.ts
@@ -52,6 +52,12 @@ export const CUSTOM_EVENTS = {
*/
RESUME_CHECK_REQUESTED: "mux:resumeCheckRequested",
+ /**
+ * Event to request a compact-and-retry flow for a workspace
+ * Detail: { workspaceId: string }
+ */
+ COMPACT_AND_RETRY_REQUESTED: "mux:compactAndRetryRequested",
+
/**
* Event to switch to a different workspace after fork
* Detail: { workspaceId: string, projectPath: string, projectName: string, workspacePath: string, branch: string }
@@ -103,6 +109,9 @@ export interface CustomEventPayloads {
workspaceId: string;
isManual?: boolean; // true when user explicitly clicks retry (bypasses eligibility checks)
};
+ [CUSTOM_EVENTS.COMPACT_AND_RETRY_REQUESTED]: {
+ workspaceId: string;
+ };
[CUSTOM_EVENTS.WORKSPACE_FORK_SWITCH]: {
workspaceId: string;
projectPath: string;
From 30ac83bb6b95211c50a4d704629d645b0015f4df Mon Sep 17 00:00:00 2001
From: Ammar
Date: Sat, 17 Jan 2026 20:49:51 -0600
Subject: [PATCH 02/11] =?UTF-8?q?=F0=9F=A4=96=20tests:=20allow=20multiple?=
=?UTF-8?q?=20compact=20retry=20buttons?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
...tExceededCompactionSuggestion.integration.test.ts | 12 ++++++------
1 file changed, 6 insertions(+), 6 deletions(-)
diff --git a/tests/ui/contextExceededCompactionSuggestion.integration.test.ts b/tests/ui/contextExceededCompactionSuggestion.integration.test.ts
index f65c026220..a45fc0bd44 100644
--- a/tests/ui/contextExceededCompactionSuggestion.integration.test.ts
+++ b/tests/ui/contextExceededCompactionSuggestion.integration.test.ts
@@ -81,8 +81,8 @@ describeIntegration("Context exceeded compaction suggestion (UI)", () => {
// And we should render an action button for one-click compaction + retry.
await waitFor(
() => {
- const button = view.queryByRole("button", { name: "Compact & retry" });
- if (!button) {
+ const buttons = view.queryAllByRole("button", { name: "Compact & retry" });
+ if (buttons.length === 0) {
throw new Error("Expected Compact & retry button");
}
},
@@ -105,7 +105,7 @@ describeIntegration("Context exceeded compaction suggestion (UI)", () => {
// Clicking the CTA should actually send a compaction request message.
// We assert on the rendered /compact command (from muxMetadata.rawCommand).
- const button = view.getByRole("button", { name: "Compact & retry" });
+ const [button] = view.getAllByRole("button", { name: "Compact & retry" });
if (view.container.textContent?.includes(expectedCompactionCommand)) {
throw new Error("Compaction command should not be present before clicking");
}
@@ -176,8 +176,8 @@ describeIntegration("Context exceeded compaction suggestion (UI)", () => {
// And we should render an action button for one-click compaction + retry.
await waitFor(
() => {
- const button = view.queryByRole("button", { name: "Compact & retry" });
- if (!button) {
+ const buttons = view.queryAllByRole("button", { name: "Compact & retry" });
+ if (buttons.length === 0) {
throw new Error("Expected Compact & retry button");
}
},
@@ -201,7 +201,7 @@ describeIntegration("Context exceeded compaction suggestion (UI)", () => {
// Clicking the CTA should actually send a compaction request message.
// We assert on the rendered /compact command (from muxMetadata.rawCommand).
- const button = view.getByRole("button", { name: "Compact & retry" });
+ const [button] = view.getAllByRole("button", { name: "Compact & retry" });
if (view.container.textContent?.includes(expectedCompactionCommand)) {
throw new Error("Compaction command should not be present before clicking");
}
From dc57b433f4fbf496107abb96805f3662a6014600 Mon Sep 17 00:00:00 2001
From: Ammar
Date: Sun, 18 Jan 2026 10:33:37 -0600
Subject: [PATCH 03/11] =?UTF-8?q?=F0=9F=A4=96=20fix:=20move=20compact=20re?=
=?UTF-8?q?try=20into=20stream=20error=20card?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
.../Messages/ChatBarrier/RetryBarrier.tsx | 87 ++---------
.../Messages/StreamErrorMessage.tsx | 138 +++++++++++++++---
src/browser/hooks/useCompactAndRetry.ts | 24 +--
src/common/constants/events.ts | 9 --
4 files changed, 126 insertions(+), 132 deletions(-)
diff --git a/src/browser/components/Messages/ChatBarrier/RetryBarrier.tsx b/src/browser/components/Messages/ChatBarrier/RetryBarrier.tsx
index 9ede327554..2813dd5c4b 100644
--- a/src/browser/components/Messages/ChatBarrier/RetryBarrier.tsx
+++ b/src/browser/components/Messages/ChatBarrier/RetryBarrier.tsx
@@ -1,5 +1,4 @@
import React, { useEffect, useMemo, useState } from "react";
-import { useCompactAndRetry } from "@/browser/hooks/useCompactAndRetry";
import { usePersistedState, updatePersistedState } from "@/browser/hooks/usePersistedState";
import type { RetryState } from "@/browser/hooks/useResumeManager";
import { useWorkspaceState } from "@/browser/stores/WorkspaceStore";
@@ -13,36 +12,23 @@ import { CUSTOM_EVENTS, createCustomEvent } from "@/common/constants/events";
import { getAutoRetryKey, getRetryStateKey, VIM_ENABLED_KEY } from "@/common/constants/storage";
import { cn } from "@/common/lib/utils";
import { formatSendMessageError } from "@/common/utils/errors/formatSendError";
-import { formatTokens } from "@/common/utils/tokens/tokenMeterUtils";
interface RetryBarrierProps {
workspaceId: string;
className?: string;
}
-function formatContextTokens(tokens: number): string {
- return formatTokens(tokens).replace(/\.0([kM])$/, "$1");
-}
-
const defaultRetryState: RetryState = {
attempt: 0,
retryStartTime: Date.now(),
};
-export const RetryBarrier: React.FC = ({ workspaceId, className }) => {
+export const RetryBarrier: React.FC = (props) => {
// Get workspace state for computing effective autoRetry
- const workspaceState = useWorkspaceState(workspaceId);
-
- const {
- showCompactionUI,
- compactionSuggestion,
- hasCompactionRequest,
- hasTriggerUserMessage,
- isRetryingWithCompaction,
- retryWithCompaction,
- } = useCompactAndRetry({ workspaceId });
+ const workspaceState = useWorkspaceState(props.workspaceId);
+
const [autoRetry, setAutoRetry] = usePersistedState(
- getAutoRetryKey(workspaceId),
+ getAutoRetryKey(props.workspaceId),
true, // Default to true
{ listener: true }
);
@@ -56,7 +42,7 @@ export const RetryBarrier: React.FC = ({ workspaceId, classNa
// Use persisted state for retry tracking (survives workspace switches)
// Read retry state (managed by useResumeManager)
const [retryState] = usePersistedState(
- getRetryStateKey(workspaceId),
+ getRetryStateKey(props.workspaceId),
defaultRetryState,
{ listener: true }
);
@@ -113,13 +99,13 @@ export const RetryBarrier: React.FC = ({ workspaceId, classNa
// Create manual retry state: immediate retry BUT preserves attempt counter
// This prevents infinite retry loops without backoff if the retry fails
- updatePersistedState(getRetryStateKey(workspaceId), createManualRetryState(attempt));
+ updatePersistedState(getRetryStateKey(props.workspaceId), createManualRetryState(attempt));
// Emit event to useResumeManager - it will handle the actual resume
// Pass isManual flag to bypass eligibility checks (user explicitly wants to retry)
window.dispatchEvent(
createCustomEvent(CUSTOM_EVENTS.RESUME_CHECK_REQUESTED, {
- workspaceId,
+ workspaceId: props.workspaceId,
isManual: true,
})
);
@@ -141,38 +127,7 @@ export const RetryBarrier: React.FC = ({ workspaceId, classNa
: formatted.message;
};
- const details = showCompactionUI ? (
-
- Context window exceeded.{" "}
- {compactionSuggestion ? (
- compactionSuggestion.kind === "preferred" ? (
- <>
- We'll compact with your configured compaction model{" "}
-
- {compactionSuggestion.displayName}
-
- {compactionSuggestion.maxInputTokens !== null ? (
- <> ({formatContextTokens(compactionSuggestion.maxInputTokens)} context)>
- ) : null}{" "}
- to unblock you. Your workspace model stays the same.
- >
- ) : (
- <>
- We'll compact with{" "}
-
- {compactionSuggestion.displayName}
-
- {compactionSuggestion.maxInputTokens !== null ? (
- <> ({formatContextTokens(compactionSuggestion.maxInputTokens)} context)>
- ) : null}{" "}
- to unblock you with a higher-context model. Your workspace model stays the same.
- >
- )
- ) : (
- <>Compact this chat to unblock you. Your workspace model stays the same.>
- )}
-
- ) : lastError ? (
+ const details = lastError ? (
Error: {getErrorMessage(lastError)}
@@ -180,7 +135,7 @@ export const RetryBarrier: React.FC = ({ workspaceId, classNa
const barrierClassName = cn(
"my-5 px-5 py-4 bg-gradient-to-br from-[rgba(255,165,0,0.1)] to-[rgba(255,140,0,0.1)] border-l-4 border-warning rounded flex flex-col gap-3",
- className
+ props.className
);
let statusIcon = "⚠️";
@@ -210,32 +165,12 @@ export const RetryBarrier: React.FC = ({ workspaceId, classNa
);
} else {
- const onClick = showCompactionUI
- ? () => {
- retryWithCompaction().catch(() => undefined);
- }
- : handleManualRetry;
-
- let label = "Retry";
- if (showCompactionUI) {
- if (isRetryingWithCompaction) {
- label = "Starting...";
- } else if (!compactionSuggestion || !hasTriggerUserMessage) {
- label = "Insert /compact";
- } else if (hasCompactionRequest) {
- label = "Retry compaction";
- } else {
- label = "Compact & retry";
- }
- }
-
actionButton = (
);
}
diff --git a/src/browser/components/Messages/StreamErrorMessage.tsx b/src/browser/components/Messages/StreamErrorMessage.tsx
index 2b5f2c0ad2..f6c8d7049d 100644
--- a/src/browser/components/Messages/StreamErrorMessage.tsx
+++ b/src/browser/components/Messages/StreamErrorMessage.tsx
@@ -2,41 +2,33 @@ import React from "react";
import { Bug } from "lucide-react";
import { Button } from "@/browser/components/ui/button";
import { Tooltip, TooltipContent, TooltipTrigger } from "@/browser/components/ui/tooltip";
+import { useCompactAndRetry } from "@/browser/hooks/useCompactAndRetry";
import { CUSTOM_EVENTS, createCustomEvent } from "@/common/constants/events";
import { cn } from "@/common/lib/utils";
-import { useOptionalMessageListContext } from "./MessageListContext";
import type { DisplayedMessage } from "@/common/types/message";
+import { formatTokens } from "@/common/utils/tokens/tokenMeterUtils";
+import { useOptionalMessageListContext } from "./MessageListContext";
interface StreamErrorMessageProps {
message: DisplayedMessage & { type: "stream-error" };
className?: string;
}
-// RetryBarrier handles auto-retry; this component exposes a compact-and-retry shortcut.
-export const StreamErrorMessage: React.FC = ({ message, className }) => {
- const messageListContext = useOptionalMessageListContext();
- const latestMessageId = messageListContext?.latestMessageId ?? null;
- const workspaceId = messageListContext?.workspaceId ?? null;
- const isLatestMessage = latestMessageId === message.id;
+interface StreamErrorMessageBaseProps extends StreamErrorMessageProps {
+ compactRetryAction?: React.ReactNode;
+ compactionDetails?: React.ReactNode;
+}
- const showCompactRetry =
- message.errorType === "context_exceeded" && !!workspaceId && isLatestMessage;
+function formatContextTokens(tokens: number): string {
+ return formatTokens(tokens).replace(/\.0([kM])$/, "$1");
+}
+
+const StreamErrorMessageBase: React.FC = (props) => {
+ const message = props.message;
+ const className = props.className;
+ const compactRetryAction = props.compactRetryAction;
+ const compactionDetails = props.compactionDetails;
- const compactRetryAction = showCompactRetry ? (
-
- ) : null;
const debugAction = (
@@ -104,6 +96,104 @@ export const StreamErrorMessage: React.FC = ({ message,
{message.error}
+ {compactionDetails}
);
};
+
+interface StreamErrorMessageWithRetryProps extends StreamErrorMessageProps {
+ workspaceId: string;
+}
+
+const StreamErrorMessageWithRetry: React.FC = (props) => {
+ const compactAndRetry = useCompactAndRetry({ workspaceId: props.workspaceId });
+ const showCompactRetry = compactAndRetry.showCompactionUI;
+
+ let compactRetryLabel = "Compact & retry";
+ if (showCompactRetry) {
+ if (compactAndRetry.isRetryingWithCompaction) {
+ compactRetryLabel = "Starting...";
+ } else if (!compactAndRetry.compactionSuggestion || !compactAndRetry.hasTriggerUserMessage) {
+ compactRetryLabel = "Insert /compact";
+ } else if (compactAndRetry.hasCompactionRequest) {
+ compactRetryLabel = "Retry compaction";
+ }
+ }
+
+ const compactRetryAction = showCompactRetry ? (
+
+ ) : null;
+
+ const compactionSuggestion = compactAndRetry.compactionSuggestion;
+ const compactionDetails = showCompactRetry ? (
+
+ Context window exceeded.{" "}
+ {compactionSuggestion ? (
+ compactionSuggestion.kind === "preferred" ? (
+ <>
+ We'll compact with your configured compaction model{" "}
+
+ {compactionSuggestion.displayName}
+
+ {compactionSuggestion.maxInputTokens !== null ? (
+ <> ({formatContextTokens(compactionSuggestion.maxInputTokens)} context)>
+ ) : null}{" "}
+ to unblock you. Your workspace model stays the same.
+ >
+ ) : (
+ <>
+ We'll compact with{" "}
+
+ {compactionSuggestion.displayName}
+
+ {compactionSuggestion.maxInputTokens !== null ? (
+ <> ({formatContextTokens(compactionSuggestion.maxInputTokens)} context)>
+ ) : null}{" "}
+ to unblock you with a higher-context model. Your workspace model stays the same.
+ >
+ )
+ ) : (
+ <>Compact this chat to unblock you. Your workspace model stays the same.>
+ )}
+
+ ) : null;
+
+ return (
+
+ );
+};
+
+// RetryBarrier handles auto-retry; compaction retry UI lives here for stream errors.
+export const StreamErrorMessage: React.FC = (props) => {
+ const messageListContext = useOptionalMessageListContext();
+ const latestMessageId = messageListContext?.latestMessageId ?? null;
+ const workspaceId = messageListContext?.workspaceId ?? null;
+ const isLatestMessage = latestMessageId === props.message.id;
+
+ if (!workspaceId || !isLatestMessage) {
+ return ;
+ }
+
+ return (
+
+ );
+};
diff --git a/src/browser/hooks/useCompactAndRetry.ts b/src/browser/hooks/useCompactAndRetry.ts
index 8ad8de319a..de0ad5e09e 100644
--- a/src/browser/hooks/useCompactAndRetry.ts
+++ b/src/browser/hooks/useCompactAndRetry.ts
@@ -13,7 +13,7 @@ import {
type CompactionSuggestion,
} from "@/browser/utils/compaction/suggestion";
import { executeCompaction } from "@/browser/utils/chatCommands";
-import { CUSTOM_EVENTS, createCustomEvent, type CustomEventType } from "@/common/constants/events";
+import { CUSTOM_EVENTS, createCustomEvent } from "@/common/constants/events";
import { PREFERRED_COMPACTION_MODEL_KEY } from "@/common/constants/storage";
import type { ImagePart, ProvidersConfigMap } from "@/common/orpc/types";
import { buildContinueMessage, type DisplayedMessage } from "@/common/types/message";
@@ -258,28 +258,6 @@ export function useCompactAndRetry(props: { workspaceId: string }): CompactAndRe
}
}, [api, compactionSuggestion, props.workspaceId, triggerUserMessage]);
- useEffect(() => {
- if (!showCompactionUI) return;
-
- const handleCompactRetryRequested = (event: Event) => {
- const customEvent = event as CustomEventType<
- typeof CUSTOM_EVENTS.COMPACT_AND_RETRY_REQUESTED
- >;
- if (customEvent.detail.workspaceId !== props.workspaceId) return;
- if (isRetryingWithCompaction) return;
- retryWithCompaction().catch(() => undefined);
- };
-
- window.addEventListener(CUSTOM_EVENTS.COMPACT_AND_RETRY_REQUESTED, handleCompactRetryRequested);
-
- return () => {
- window.removeEventListener(
- CUSTOM_EVENTS.COMPACT_AND_RETRY_REQUESTED,
- handleCompactRetryRequested
- );
- };
- }, [isRetryingWithCompaction, props.workspaceId, retryWithCompaction, showCompactionUI]);
-
return {
showCompactionUI,
compactionSuggestion,
diff --git a/src/common/constants/events.ts b/src/common/constants/events.ts
index 8bbe5cb5e1..1ba4b4520a 100644
--- a/src/common/constants/events.ts
+++ b/src/common/constants/events.ts
@@ -52,12 +52,6 @@ export const CUSTOM_EVENTS = {
*/
RESUME_CHECK_REQUESTED: "mux:resumeCheckRequested",
- /**
- * Event to request a compact-and-retry flow for a workspace
- * Detail: { workspaceId: string }
- */
- COMPACT_AND_RETRY_REQUESTED: "mux:compactAndRetryRequested",
-
/**
* Event to switch to a different workspace after fork
* Detail: { workspaceId: string, projectPath: string, projectName: string, workspacePath: string, branch: string }
@@ -109,9 +103,6 @@ export interface CustomEventPayloads {
workspaceId: string;
isManual?: boolean; // true when user explicitly clicks retry (bypasses eligibility checks)
};
- [CUSTOM_EVENTS.COMPACT_AND_RETRY_REQUESTED]: {
- workspaceId: string;
- };
[CUSTOM_EVENTS.WORKSPACE_FORK_SWITCH]: {
workspaceId: string;
projectPath: string;
From 8e1b6982d8d20957ba58a92e78c88bc40a14ee34 Mon Sep 17 00:00:00 2001
From: Ammar
Date: Sun, 18 Jan 2026 11:03:03 -0600
Subject: [PATCH 04/11] =?UTF-8?q?=F0=9F=A4=96=20fix:=20wire=20message=20li?=
=?UTF-8?q?st=20context=20in=20chat=20pane?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
src/browser/components/ChatPane.tsx | 161 +++++++++++++++-------------
1 file changed, 89 insertions(+), 72 deletions(-)
diff --git a/src/browser/components/ChatPane.tsx b/src/browser/components/ChatPane.tsx
index 086ac47693..4d6ea44c1f 100644
--- a/src/browser/components/ChatPane.tsx
+++ b/src/browser/components/ChatPane.tsx
@@ -7,6 +7,7 @@ import React, {
useDeferredValue,
useMemo,
} from "react";
+import { MessageListProvider } from "./Messages/MessageListContext";
import { cn } from "@/common/lib/utils";
import { MessageRenderer } from "./Messages/MessageRenderer";
import { InterruptedBarrier } from "./Messages/ChatBarrier/InterruptedBarrier";
@@ -170,6 +171,15 @@ export const ChatPane: React.FC = (props) => {
? transformedMessages
: deferredTransformedMessages;
+ const latestMessageId =
+ deferredMessages.length > 0
+ ? (deferredMessages[deferredMessages.length - 1]?.id ?? null)
+ : null;
+ const messageListContextValue = useMemo(
+ () => ({ workspaceId, latestMessageId }),
+ [workspaceId, latestMessageId]
+ );
+
const autoCompactionResult = useMemo(
() => checkAutoCompaction(workspaceUsage, pendingModel, use1M, autoCompactionThreshold / 100),
[workspaceUsage, pendingModel, use1M, autoCompactionThreshold]
@@ -448,6 +458,11 @@ export const ChatPane: React.FC = (props) => {
)
: false;
+ const lastMessage = workspaceState.messages[workspaceState.messages.length - 1];
+ const suppressRetryBarrier =
+ lastMessage?.type === "stream-error" && lastMessage.errorType === "context_exceeded";
+ const showRetryBarrierUI = showRetryBarrier && !suppressRetryBarrier;
+
// Handle keyboard shortcuts (using optional refs that are safe even if not initialized)
useAIViewKeybinds({
workspaceId,
@@ -557,79 +572,81 @@ export const ChatPane: React.FC = (props) => {
) : (
- <>
- {deferredMessages.map((msg, index) => {
- // Compute bash_output grouping at render-time
- const bashOutputGroup = computeBashOutputGroupInfo(deferredMessages, index);
-
- // For bash_output groups, use first message ID as expansion key
- const groupKey = bashOutputGroup
- ? deferredMessages[bashOutputGroup.firstIndex]?.id
- : undefined;
- const isGroupExpanded = groupKey ? expandedBashGroups.has(groupKey) : false;
-
- // Skip rendering middle items in a bash_output group (unless expanded)
- if (bashOutputGroup?.position === "middle" && !isGroupExpanded) {
- return null;
- }
-
- const isAtCutoff =
- editCutoffHistoryId !== undefined &&
- msg.type !== "history-hidden" &&
- msg.type !== "workspace-init" &&
- msg.historyId === editCutoffHistoryId;
-
- return (
-
-
-
+ <>
+ {deferredMessages.map((msg, index) => {
+ // Compute bash_output grouping at render-time
+ const bashOutputGroup = computeBashOutputGroupInfo(deferredMessages, index);
+
+ // For bash_output groups, use first message ID as expansion key
+ const groupKey = bashOutputGroup
+ ? deferredMessages[bashOutputGroup.firstIndex]?.id
+ : undefined;
+ const isGroupExpanded = groupKey ? expandedBashGroups.has(groupKey) : false;
+
+ // Skip rendering middle items in a bash_output group (unless expanded)
+ if (bashOutputGroup?.position === "middle" && !isGroupExpanded) {
+ return null;
+ }
+
+ const isAtCutoff =
+ editCutoffHistoryId !== undefined &&
+ msg.type !== "history-hidden" &&
+ msg.type !== "workspace-init" &&
+ msg.historyId === editCutoffHistoryId;
+
+ return (
+
+
-
- {/* Show collapsed indicator after the first item in a bash_output group */}
- {bashOutputGroup?.position === "first" && groupKey && (
- {
- setExpandedBashGroups((prev) => {
- const next = new Set(prev);
- if (next.has(groupKey)) {
- next.delete(groupKey);
- } else {
- next.add(groupKey);
- }
- return next;
- });
- }}
- />
- )}
- {isAtCutoff && }
- {shouldShowInterruptedBarrier(msg) && }
-
- );
- })}
- {/* Show RetryBarrier after the last message if needed */}
- {showRetryBarrier && }
- >
+ >
+
+
+ {/* Show collapsed indicator after the first item in a bash_output group */}
+ {bashOutputGroup?.position === "first" && groupKey && (
+ {
+ setExpandedBashGroups((prev) => {
+ const next = new Set(prev);
+ if (next.has(groupKey)) {
+ next.delete(groupKey);
+ } else {
+ next.add(groupKey);
+ }
+ return next;
+ });
+ }}
+ />
+ )}
+ {isAtCutoff && }
+ {shouldShowInterruptedBarrier(msg) && }
+
+ );
+ })}
+ {/* Show RetryBarrier after the last message if needed */}
+ {showRetryBarrierUI && }
+ >
+
)}
From 8a9221d87a457acf74dbc77e3e14e34a4813586f Mon Sep 17 00:00:00 2001
From: Ammar
Date: Sun, 18 Jan 2026 16:07:53 -0600
Subject: [PATCH 05/11] =?UTF-8?q?=F0=9F=A4=96=20fix:=20improve=20compactio?=
=?UTF-8?q?n=20CTA=20contrast?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
src/browser/components/Messages/StreamErrorMessage.tsx | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/src/browser/components/Messages/StreamErrorMessage.tsx b/src/browser/components/Messages/StreamErrorMessage.tsx
index f6c8d7049d..572f85f836 100644
--- a/src/browser/components/Messages/StreamErrorMessage.tsx
+++ b/src/browser/components/Messages/StreamErrorMessage.tsx
@@ -128,7 +128,7 @@ const StreamErrorMessageWithRetry: React.FC =
compactAndRetry.retryWithCompaction().catch(() => undefined);
}}
disabled={compactAndRetry.isRetryingWithCompaction}
- className="border-warning/40 text-warning hover:bg-warning/10 hover:text-warning h-6 px-2 text-[10px]"
+ className="border-warning/50 text-foreground bg-warning/10 hover:bg-warning/15 hover:text-foreground h-6 px-2 text-[10px]"
>
{compactRetryLabel}
@@ -137,7 +137,7 @@ const StreamErrorMessageWithRetry: React.FC =
const compactionSuggestion = compactAndRetry.compactionSuggestion;
const compactionDetails = showCompactRetry ? (
-
Context window exceeded.{" "}
+
Context window exceeded.{" "}
{compactionSuggestion ? (
compactionSuggestion.kind === "preferred" ? (
<>
From e3cf6b2c547bf27cd8abf40d90b55ba5ff4e462e Mon Sep 17 00:00:00 2001
From: Ammar
Date: Sun, 18 Jan 2026 16:41:38 -0600
Subject: [PATCH 06/11] =?UTF-8?q?=F0=9F=A4=96=20fix:=20align=20compact=20r?=
=?UTF-8?q?etry=20CTA=20and=20hide=20duplicates?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
src/browser/components/Messages/StreamErrorMessage.tsx | 4 +++-
src/browser/hooks/useCompactAndRetry.ts | 4 ++++
src/browser/utils/chatCommands.ts | 3 +++
src/browser/utils/messages/StreamingMessageAggregator.ts | 1 +
src/common/orpc/schemas/stream.ts | 1 +
src/common/types/message.ts | 2 ++
src/node/services/agentSession.ts | 7 +++++++
7 files changed, 21 insertions(+), 1 deletion(-)
diff --git a/src/browser/components/Messages/StreamErrorMessage.tsx b/src/browser/components/Messages/StreamErrorMessage.tsx
index 572f85f836..8109c17f67 100644
--- a/src/browser/components/Messages/StreamErrorMessage.tsx
+++ b/src/browser/components/Messages/StreamErrorMessage.tsx
@@ -84,7 +84,6 @@ const StreamErrorMessageBase: React.FC = (props) =>
{message.errorType}
- {compactRetryAction}
{showCount && (
×{message.errorCount}
@@ -97,6 +96,9 @@ const StreamErrorMessageBase: React.FC = (props) =>
{message.error}
{compactionDetails}
+ {compactRetryAction ? (
+ {compactRetryAction}
+ ) : null}
);
};
diff --git a/src/browser/hooks/useCompactAndRetry.ts b/src/browser/hooks/useCompactAndRetry.ts
index de0ad5e09e..36b253b107 100644
--- a/src/browser/hooks/useCompactAndRetry.ts
+++ b/src/browser/hooks/useCompactAndRetry.ts
@@ -184,6 +184,8 @@ export function useCompactAndRetry(props: { workspaceId: string }): CompactAndRe
if (source.compactionRequest) {
const maxOutputTokens = source.compactionRequest.parsed.maxOutputTokens;
const continueMessage = source.compactionRequest.parsed.continueMessage;
+ const continueMessageIsRetry =
+ source.compactionRequest.parsed.continueMessageIsRetry === true;
const result = await executeCompaction({
api,
@@ -192,6 +194,7 @@ export function useCompactAndRetry(props: { workspaceId: string }): CompactAndRe
model: compactionSuggestion.modelId,
maxOutputTokens,
continueMessage,
+ continueMessageIsRetry,
});
if (!result.success) {
@@ -242,6 +245,7 @@ export function useCompactAndRetry(props: { workspaceId: string }): CompactAndRe
sendMessageOptions,
model: compactionSuggestion.modelId,
continueMessage,
+ continueMessageIsRetry: true,
});
if (!result.success) {
diff --git a/src/browser/utils/chatCommands.ts b/src/browser/utils/chatCommands.ts
index 9173777861..0189c4a2bf 100644
--- a/src/browser/utils/chatCommands.ts
+++ b/src/browser/utils/chatCommands.ts
@@ -613,6 +613,8 @@ export interface CompactionOptions {
workspaceId: string;
maxOutputTokens?: number;
continueMessage?: ContinueMessage;
+ /** True when the continue message retries an already-sent user message. */
+ continueMessageIsRetry?: boolean;
model?: string;
sendMessageOptions: SendMessageOptions;
editMessageId?: string;
@@ -662,6 +664,7 @@ export function prepareCompactionMessage(options: CompactionOptions): {
model: effectiveModel,
maxOutputTokens: options.maxOutputTokens,
continueMessage: cm,
+ continueMessageIsRetry: options.continueMessageIsRetry,
};
const metadata: MuxFrontendMetadata = {
diff --git a/src/browser/utils/messages/StreamingMessageAggregator.ts b/src/browser/utils/messages/StreamingMessageAggregator.ts
index 32fa2a136c..6f184e756a 100644
--- a/src/browser/utils/messages/StreamingMessageAggregator.ts
+++ b/src/browser/utils/messages/StreamingMessageAggregator.ts
@@ -1735,6 +1735,7 @@ export class StreamingMessageAggregator {
model: muxMeta.parsed.model,
maxOutputTokens: muxMeta.parsed.maxOutputTokens,
continueMessage: muxMeta.parsed.continueMessage,
+ continueMessageIsRetry: muxMeta.parsed.continueMessageIsRetry,
} satisfies CompactionRequestData,
}
: undefined;
diff --git a/src/common/orpc/schemas/stream.ts b/src/common/orpc/schemas/stream.ts
index 2c85327e55..7dbe7cf3cf 100644
--- a/src/common/orpc/schemas/stream.ts
+++ b/src/common/orpc/schemas/stream.ts
@@ -409,6 +409,7 @@ export const SendMessageOptionsSchema = z.object({
providerOptions: MuxProviderOptionsSchema.optional(),
mode: AgentModeSchema.optional().catch(undefined),
muxMetadata: z.any().optional(), // Black box
+ synthetic: z.boolean().optional(),
experiments: ExperimentsSchema.optional(),
/**
* When true, workspace-specific agent definitions are disabled.
diff --git a/src/common/types/message.ts b/src/common/types/message.ts
index da8caee14d..e706880485 100644
--- a/src/common/types/message.ts
+++ b/src/common/types/message.ts
@@ -136,6 +136,8 @@ export interface CompactionRequestData {
model?: string; // Custom model override for compaction
maxOutputTokens?: number;
continueMessage?: ContinueMessage;
+ /** True when the continue message retries an already-sent user message. */
+ continueMessageIsRetry?: boolean;
}
/**
diff --git a/src/node/services/agentSession.ts b/src/node/services/agentSession.ts
index 5f224ac508..d724e3a58a 100644
--- a/src/node/services/agentSession.ts
+++ b/src/node/services/agentSession.ts
@@ -73,6 +73,7 @@ interface CompactionRequestMetadata {
type: "compaction-request";
parsed: {
continueMessage?: ContinueMessage;
+ continueMessageIsRetry?: boolean;
};
}
@@ -550,6 +551,7 @@ export class AgentSession {
timestamp: Date.now(),
toolPolicy: typedToolPolicy,
muxMetadata: typedMuxMetadata, // Pass through frontend metadata as black-box
+ synthetic: options?.synthetic ? true : undefined,
},
additionalParts
);
@@ -656,6 +658,7 @@ export class AgentSession {
"exec";
// Build options for the queued message (strip compaction-specific fields)
// agentId determines tool policy via resolveToolPolicyForAgent in aiService
+
const sanitizedOptions: Omit<
SendMessageOptions,
"muxMetadata" | "mode" | "editMessageId" | "imageParts" | "maxOutputTokens"
@@ -669,6 +672,10 @@ export class AgentSession {
disableWorkspaceAgents: options.disableWorkspaceAgents,
};
+ if (typedMuxMetadata.parsed.continueMessageIsRetry) {
+ sanitizedOptions.synthetic = true;
+ }
+
// Add image parts if present
const continueImageParts = continueMessage.imageParts;
if (continueImageParts && continueImageParts.length > 0) {
From 917ef110f32c70ef5a37fd38e26a8aedd3466f95 Mon Sep 17 00:00:00 2001
From: Ammar
Date: Sun, 18 Jan 2026 16:58:06 -0600
Subject: [PATCH 07/11] =?UTF-8?q?=F0=9F=A4=96=20fix:=20resume=20compaction?=
=?UTF-8?q?=20retries=20without=20synthetic?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
src/common/orpc/schemas/stream.ts | 1 -
src/node/services/agentSession.ts | 34 +++++++++---
src/node/services/compactionHandler.ts | 71 ++++++++++++++++++++++++--
3 files changed, 93 insertions(+), 13 deletions(-)
diff --git a/src/common/orpc/schemas/stream.ts b/src/common/orpc/schemas/stream.ts
index 7dbe7cf3cf..2c85327e55 100644
--- a/src/common/orpc/schemas/stream.ts
+++ b/src/common/orpc/schemas/stream.ts
@@ -409,7 +409,6 @@ export const SendMessageOptionsSchema = z.object({
providerOptions: MuxProviderOptionsSchema.optional(),
mode: AgentModeSchema.optional().catch(undefined),
muxMetadata: z.any().optional(), // Black box
- synthetic: z.boolean().optional(),
experiments: ExperimentsSchema.optional(),
/**
* When true, workspace-specific agent definitions are disabled.
diff --git a/src/node/services/agentSession.ts b/src/node/services/agentSession.ts
index d724e3a58a..7798e8876c 100644
--- a/src/node/services/agentSession.ts
+++ b/src/node/services/agentSession.ts
@@ -128,6 +128,7 @@ export class AgentSession {
private readonly initListeners: Array<{ event: string; handler: (...args: unknown[]) => void }> =
[];
private disposed = false;
+ private pendingCompactionRetryOptions?: SendMessageOptions;
private readonly messageQueue = new MessageQueue();
private readonly compactionHandler: CompactionHandler;
@@ -551,7 +552,6 @@ export class AgentSession {
timestamp: Date.now(),
toolPolicy: typedToolPolicy,
muxMetadata: typedMuxMetadata, // Pass through frontend metadata as black-box
- synthetic: options?.synthetic ? true : undefined,
},
additionalParts
);
@@ -672,10 +672,6 @@ export class AgentSession {
disableWorkspaceAgents: options.disableWorkspaceAgents,
};
- if (typedMuxMetadata.parsed.continueMessageIsRetry) {
- sanitizedOptions.synthetic = true;
- }
-
// Add image parts if present
const continueImageParts = continueMessage.imageParts;
if (continueImageParts && continueImageParts.length > 0) {
@@ -687,8 +683,12 @@ export class AgentSession {
sanitizedOptions.muxMetadata = metadata;
}
- this.messageQueue.add(finalText, sanitizedOptions);
- this.emitQueuedMessageChanged();
+ if (typedMuxMetadata.parsed.continueMessageIsRetry) {
+ this.pendingCompactionRetryOptions = sanitizedOptions;
+ } else {
+ this.messageQueue.add(finalText, sanitizedOptions);
+ this.emitQueuedMessageChanged();
+ }
}
if (this.disposed) {
@@ -937,6 +937,7 @@ export class AgentSession {
}
this.activeCompactionRequest = undefined;
+ this.pendingCompactionRetryOptions = undefined;
const streamError: StreamErrorMessage = {
type: "stream-error",
@@ -992,6 +993,7 @@ export class AgentSession {
forward("usage-delta", (payload) => this.emitChatEvent(payload));
forward("stream-abort", (payload) => {
this.activeCompactionRequest = undefined;
+ this.pendingCompactionRetryOptions = undefined;
this.emitChatEvent(payload);
});
forward("runtime-status", (payload) => this.emitChatEvent(payload));
@@ -999,6 +1001,11 @@ export class AgentSession {
forward("stream-end", async (payload) => {
this.activeCompactionRequest = undefined;
const handled = await this.compactionHandler.handleCompletion(payload as StreamEndEvent);
+ const retryOptions = this.pendingCompactionRetryOptions;
+ const shouldRetryAfterCompaction =
+ handled && this.compactionHandler.consumeRetryAfterCompaction();
+ this.pendingCompactionRetryOptions = undefined;
+
if (!handled) {
this.emitChatEvent(payload);
} else {
@@ -1006,6 +1013,19 @@ export class AgentSession {
// This allows the frontend to get updated postCompaction state
this.onCompactionComplete?.();
}
+
+ if (shouldRetryAfterCompaction && retryOptions) {
+ const retryResult = await this.resumeStream(retryOptions);
+ if (!retryResult.success) {
+ log.warn("Failed to resume after compaction retry", {
+ workspaceId: this.workspaceId,
+ error: retryResult.error,
+ });
+ } else {
+ return;
+ }
+ }
+
// Stream end: auto-send queued messages
this.sendQueuedMessages();
});
diff --git a/src/node/services/compactionHandler.ts b/src/node/services/compactionHandler.ts
index 57c276e114..0ecfea443a 100644
--- a/src/node/services/compactionHandler.ts
+++ b/src/node/services/compactionHandler.ts
@@ -8,7 +8,12 @@ import type { Result } from "@/common/types/result";
import { Ok, Err } from "@/common/types/result";
import type { LanguageModelV2Usage } from "@ai-sdk/provider";
-import { createMuxMessage, type MuxMessage } from "@/common/types/message";
+import {
+ createMuxMessage,
+ prepareUserMessageForSend,
+ type ContinueMessage,
+ type MuxMessage,
+} from "@/common/types/message";
import type { TelemetryService } from "@/node/services/telemetryService";
import { roundToBase2 } from "@/common/telemetry/utils";
import { log } from "@/node/services/log";
@@ -68,6 +73,9 @@ export class CompactionHandler {
private readonly telemetryService?: TelemetryService;
private readonly emitter: EventEmitter;
private readonly processedCompactionRequestIds: Set = new Set();
+
+ /** Flag indicating a retry message should be resumed after compaction. */
+ private retryAfterCompaction = false;
private readonly onCompactionComplete?: () => void;
/** Flag indicating post-compaction attachments should be generated on next turn */
@@ -110,6 +118,15 @@ export class CompactionHandler {
return this.cachedFileDiffs.map((diff) => diff.path);
}
+ /**
+ * Consume retry-after-compaction flag (clears after read).
+ */
+ consumeRetryAfterCompaction(): boolean {
+ const shouldRetry = this.retryAfterCompaction;
+ this.retryAfterCompaction = false;
+ return shouldRetry;
+ }
+
/**
* Handle compaction stream completion
*
@@ -125,12 +142,15 @@ export class CompactionHandler {
const messages = historyResult.data;
const lastUserMsg = [...messages].reverse().find((m) => m.role === "user");
- const isCompaction = lastUserMsg?.metadata?.muxMetadata?.type === "compaction-request";
+ const muxMeta = lastUserMsg?.metadata?.muxMetadata;
+ const isCompaction = muxMeta?.type === "compaction-request";
if (!isCompaction || !lastUserMsg) {
return false;
}
+ this.retryAfterCompaction = false;
+
// Dedupe: If we've already processed this compaction-request, skip
if (this.processedCompactionRequestIds.has(lastUserMsg.id)) {
return true;
@@ -175,9 +195,12 @@ export class CompactionHandler {
}
// Check if this was an idle-compaction (auto-triggered due to inactivity)
- const muxMeta = lastUserMsg.metadata?.muxMetadata;
const isIdleCompaction =
muxMeta?.type === "compaction-request" && muxMeta.source === "idle-compaction";
+ const retryContinueMessage =
+ muxMeta?.type === "compaction-request" && muxMeta.parsed.continueMessageIsRetry
+ ? muxMeta.parsed.continueMessage
+ : undefined;
// Mark as processed before performing compaction
this.processedCompactionRequestIds.add(lastUserMsg.id);
@@ -186,7 +209,8 @@ export class CompactionHandler {
summary,
event.metadata,
messages,
- isIdleCompaction
+ isIdleCompaction,
+ retryContinueMessage
);
if (!result.success) {
log.error("Compaction failed:", result.error);
@@ -238,7 +262,8 @@ export class CompactionHandler {
systemMessageTokens?: number;
},
messages: MuxMessage[],
- isIdleCompaction = false
+ isIdleCompaction = false,
+ retryContinueMessage?: ContinueMessage
): Promise> {
// CRITICAL: Delete partial.json BEFORE clearing history
// This prevents a race condition where:
@@ -304,6 +329,38 @@ export class CompactionHandler {
return Err(`Failed to append summary: ${appendResult.error}`);
}
+ let retryMessage: MuxMessage | null = null;
+ if (retryContinueMessage) {
+ const { finalText, metadata: retryMetadata } =
+ prepareUserMessageForSend(retryContinueMessage);
+ const retryImageParts = retryContinueMessage.imageParts?.map((part) => ({
+ type: "file" as const,
+ url: part.url,
+ mediaType: part.mediaType,
+ }));
+ const retryMessageMetadata = retryMetadata
+ ? { timestamp: Date.now(), muxMetadata: retryMetadata }
+ : { timestamp: Date.now() };
+
+ retryMessage = createMuxMessage(
+ `user-${Date.now()}-${Math.random().toString(36).substring(2, 11)}`,
+ "user",
+ finalText,
+ retryMessageMetadata,
+ retryImageParts
+ );
+
+ const retryAppendResult = await this.historyService.appendToHistory(
+ this.workspaceId,
+ retryMessage
+ );
+ if (!retryAppendResult.success) {
+ return Err(`Failed to append retry message: ${retryAppendResult.error}`);
+ }
+
+ this.retryAfterCompaction = true;
+ }
+
// Set flag to trigger post-compaction attachment injection on next turn
this.postCompactionAttachmentsPending = true;
@@ -319,6 +376,10 @@ export class CompactionHandler {
// Emit summary message to frontend (add type: "message" for discriminated union)
this.emitChatEvent({ ...summaryMessage, type: "message" });
+ if (retryMessage) {
+ this.emitChatEvent({ ...retryMessage, type: "message" });
+ }
+
return Ok(undefined);
}
From 3883adbcc5251e686cf259f135ceeb97cee12c5e Mon Sep 17 00:00:00 2001
From: Ammar
Date: Sun, 18 Jan 2026 18:02:54 -0600
Subject: [PATCH 08/11] fix: show compact retry command
---
src/browser/hooks/useCompactAndRetry.ts | 2 ++
src/browser/utils/compaction/format.ts | 13 +++++++++++--
.../utils/messages/StreamingMessageAggregator.ts | 7 ++-----
3 files changed, 15 insertions(+), 7 deletions(-)
diff --git a/src/browser/hooks/useCompactAndRetry.ts b/src/browser/hooks/useCompactAndRetry.ts
index 36b253b107..da78274037 100644
--- a/src/browser/hooks/useCompactAndRetry.ts
+++ b/src/browser/hooks/useCompactAndRetry.ts
@@ -195,6 +195,7 @@ export function useCompactAndRetry(props: { workspaceId: string }): CompactAndRe
maxOutputTokens,
continueMessage,
continueMessageIsRetry,
+ editMessageId: source.id,
});
if (!result.success) {
@@ -246,6 +247,7 @@ export function useCompactAndRetry(props: { workspaceId: string }): CompactAndRe
model: compactionSuggestion.modelId,
continueMessage,
continueMessageIsRetry: true,
+ editMessageId: source.id,
});
if (!result.success) {
diff --git a/src/browser/utils/compaction/format.ts b/src/browser/utils/compaction/format.ts
index a33d9d53df..075badab04 100644
--- a/src/browser/utils/compaction/format.ts
+++ b/src/browser/utils/compaction/format.ts
@@ -30,8 +30,17 @@ export function buildCompactionEditText(request: {
rawCommand: string;
parsed: CompactionRequestData;
}): string {
- const continueText = request.parsed.continueMessage?.text;
- if (typeof continueText === "string" && continueText.trim().length > 0) {
+ const continueMessage = request.parsed.continueMessage;
+ const continueText = continueMessage?.text;
+ const hasImages = (continueMessage?.imageParts?.length ?? 0) > 0;
+ const hasReviews = (continueMessage?.reviews?.length ?? 0) > 0;
+ const isDefaultResume =
+ typeof continueText === "string" &&
+ continueText.trim() === "Continue" &&
+ !hasImages &&
+ !hasReviews;
+
+ if (typeof continueText === "string" && continueText.trim().length > 0 && !isDefaultResume) {
return `${request.rawCommand}\n${continueText}`;
}
return request.rawCommand;
diff --git a/src/browser/utils/messages/StreamingMessageAggregator.ts b/src/browser/utils/messages/StreamingMessageAggregator.ts
index 6f184e756a..dd7ac28291 100644
--- a/src/browser/utils/messages/StreamingMessageAggregator.ts
+++ b/src/browser/utils/messages/StreamingMessageAggregator.ts
@@ -7,6 +7,7 @@ import type {
MuxFrontendMetadata,
} from "@/common/types/message";
import { createMuxMessage } from "@/common/types/message";
+import { buildCompactionDisplayText } from "@/browser/utils/compaction/format";
import type {
StreamStartEvent,
StreamDeltaEvent,
@@ -1740,10 +1741,6 @@ export class StreamingMessageAggregator {
}
: undefined;
- const userDisplayText =
- compactionRequest?.rawCommand ??
- (muxMeta?.type === "agent-skill" ? muxMeta.rawCommand : content);
-
// Extract reviews from muxMetadata for rich UI display (orthogonal to message type)
const reviews = muxMeta?.reviews;
@@ -1751,7 +1748,7 @@ export class StreamingMessageAggregator {
type: "user",
id: message.id,
historyId: message.id,
- content: userDisplayText,
+ content: compactionRequest ? buildCompactionDisplayText(compactionRequest) : content,
imageParts: imageParts.length > 0 ? imageParts : undefined,
historySequence,
isSynthetic: message.metadata?.synthetic === true ? true : undefined,
From ebf4711316fbddeae16ecc10de15c209899670cf Mon Sep 17 00:00:00 2001
From: Ammar
Date: Sun, 18 Jan 2026 18:09:46 -0600
Subject: [PATCH 09/11] fix: simplify compaction retry flow
---
src/browser/hooks/useCompactAndRetry.ts | 5 --
src/browser/utils/chatCommands.ts | 3 -
.../messages/StreamingMessageAggregator.ts | 1 -
src/common/types/message.ts | 2 -
src/node/services/agentSession.ts | 28 +-------
src/node/services/compactionHandler.ts | 67 +------------------
6 files changed, 5 insertions(+), 101 deletions(-)
diff --git a/src/browser/hooks/useCompactAndRetry.ts b/src/browser/hooks/useCompactAndRetry.ts
index da78274037..9a32f161f1 100644
--- a/src/browser/hooks/useCompactAndRetry.ts
+++ b/src/browser/hooks/useCompactAndRetry.ts
@@ -184,9 +184,6 @@ export function useCompactAndRetry(props: { workspaceId: string }): CompactAndRe
if (source.compactionRequest) {
const maxOutputTokens = source.compactionRequest.parsed.maxOutputTokens;
const continueMessage = source.compactionRequest.parsed.continueMessage;
- const continueMessageIsRetry =
- source.compactionRequest.parsed.continueMessageIsRetry === true;
-
const result = await executeCompaction({
api,
workspaceId: props.workspaceId,
@@ -194,7 +191,6 @@ export function useCompactAndRetry(props: { workspaceId: string }): CompactAndRe
model: compactionSuggestion.modelId,
maxOutputTokens,
continueMessage,
- continueMessageIsRetry,
editMessageId: source.id,
});
@@ -246,7 +242,6 @@ export function useCompactAndRetry(props: { workspaceId: string }): CompactAndRe
sendMessageOptions,
model: compactionSuggestion.modelId,
continueMessage,
- continueMessageIsRetry: true,
editMessageId: source.id,
});
diff --git a/src/browser/utils/chatCommands.ts b/src/browser/utils/chatCommands.ts
index 0189c4a2bf..9173777861 100644
--- a/src/browser/utils/chatCommands.ts
+++ b/src/browser/utils/chatCommands.ts
@@ -613,8 +613,6 @@ export interface CompactionOptions {
workspaceId: string;
maxOutputTokens?: number;
continueMessage?: ContinueMessage;
- /** True when the continue message retries an already-sent user message. */
- continueMessageIsRetry?: boolean;
model?: string;
sendMessageOptions: SendMessageOptions;
editMessageId?: string;
@@ -664,7 +662,6 @@ export function prepareCompactionMessage(options: CompactionOptions): {
model: effectiveModel,
maxOutputTokens: options.maxOutputTokens,
continueMessage: cm,
- continueMessageIsRetry: options.continueMessageIsRetry,
};
const metadata: MuxFrontendMetadata = {
diff --git a/src/browser/utils/messages/StreamingMessageAggregator.ts b/src/browser/utils/messages/StreamingMessageAggregator.ts
index dd7ac28291..137bbb5cde 100644
--- a/src/browser/utils/messages/StreamingMessageAggregator.ts
+++ b/src/browser/utils/messages/StreamingMessageAggregator.ts
@@ -1736,7 +1736,6 @@ export class StreamingMessageAggregator {
model: muxMeta.parsed.model,
maxOutputTokens: muxMeta.parsed.maxOutputTokens,
continueMessage: muxMeta.parsed.continueMessage,
- continueMessageIsRetry: muxMeta.parsed.continueMessageIsRetry,
} satisfies CompactionRequestData,
}
: undefined;
diff --git a/src/common/types/message.ts b/src/common/types/message.ts
index e706880485..da8caee14d 100644
--- a/src/common/types/message.ts
+++ b/src/common/types/message.ts
@@ -136,8 +136,6 @@ export interface CompactionRequestData {
model?: string; // Custom model override for compaction
maxOutputTokens?: number;
continueMessage?: ContinueMessage;
- /** True when the continue message retries an already-sent user message. */
- continueMessageIsRetry?: boolean;
}
/**
diff --git a/src/node/services/agentSession.ts b/src/node/services/agentSession.ts
index 7798e8876c..9ee01d57ed 100644
--- a/src/node/services/agentSession.ts
+++ b/src/node/services/agentSession.ts
@@ -73,7 +73,6 @@ interface CompactionRequestMetadata {
type: "compaction-request";
parsed: {
continueMessage?: ContinueMessage;
- continueMessageIsRetry?: boolean;
};
}
@@ -128,7 +127,6 @@ export class AgentSession {
private readonly initListeners: Array<{ event: string; handler: (...args: unknown[]) => void }> =
[];
private disposed = false;
- private pendingCompactionRetryOptions?: SendMessageOptions;
private readonly messageQueue = new MessageQueue();
private readonly compactionHandler: CompactionHandler;
@@ -683,12 +681,8 @@ export class AgentSession {
sanitizedOptions.muxMetadata = metadata;
}
- if (typedMuxMetadata.parsed.continueMessageIsRetry) {
- this.pendingCompactionRetryOptions = sanitizedOptions;
- } else {
- this.messageQueue.add(finalText, sanitizedOptions);
- this.emitQueuedMessageChanged();
- }
+ this.messageQueue.add(finalText, sanitizedOptions);
+ this.emitQueuedMessageChanged();
}
if (this.disposed) {
@@ -937,7 +931,6 @@ export class AgentSession {
}
this.activeCompactionRequest = undefined;
- this.pendingCompactionRetryOptions = undefined;
const streamError: StreamErrorMessage = {
type: "stream-error",
@@ -993,7 +986,6 @@ export class AgentSession {
forward("usage-delta", (payload) => this.emitChatEvent(payload));
forward("stream-abort", (payload) => {
this.activeCompactionRequest = undefined;
- this.pendingCompactionRetryOptions = undefined;
this.emitChatEvent(payload);
});
forward("runtime-status", (payload) => this.emitChatEvent(payload));
@@ -1001,10 +993,6 @@ export class AgentSession {
forward("stream-end", async (payload) => {
this.activeCompactionRequest = undefined;
const handled = await this.compactionHandler.handleCompletion(payload as StreamEndEvent);
- const retryOptions = this.pendingCompactionRetryOptions;
- const shouldRetryAfterCompaction =
- handled && this.compactionHandler.consumeRetryAfterCompaction();
- this.pendingCompactionRetryOptions = undefined;
if (!handled) {
this.emitChatEvent(payload);
@@ -1014,18 +1002,6 @@ export class AgentSession {
this.onCompactionComplete?.();
}
- if (shouldRetryAfterCompaction && retryOptions) {
- const retryResult = await this.resumeStream(retryOptions);
- if (!retryResult.success) {
- log.warn("Failed to resume after compaction retry", {
- workspaceId: this.workspaceId,
- error: retryResult.error,
- });
- } else {
- return;
- }
- }
-
// Stream end: auto-send queued messages
this.sendQueuedMessages();
});
diff --git a/src/node/services/compactionHandler.ts b/src/node/services/compactionHandler.ts
index 0ecfea443a..51021836e3 100644
--- a/src/node/services/compactionHandler.ts
+++ b/src/node/services/compactionHandler.ts
@@ -8,12 +8,7 @@ import type { Result } from "@/common/types/result";
import { Ok, Err } from "@/common/types/result";
import type { LanguageModelV2Usage } from "@ai-sdk/provider";
-import {
- createMuxMessage,
- prepareUserMessageForSend,
- type ContinueMessage,
- type MuxMessage,
-} from "@/common/types/message";
+import { createMuxMessage, type MuxMessage } from "@/common/types/message";
import type { TelemetryService } from "@/node/services/telemetryService";
import { roundToBase2 } from "@/common/telemetry/utils";
import { log } from "@/node/services/log";
@@ -74,8 +69,6 @@ export class CompactionHandler {
private readonly emitter: EventEmitter;
private readonly processedCompactionRequestIds: Set = new Set();
- /** Flag indicating a retry message should be resumed after compaction. */
- private retryAfterCompaction = false;
private readonly onCompactionComplete?: () => void;
/** Flag indicating post-compaction attachments should be generated on next turn */
@@ -118,15 +111,6 @@ export class CompactionHandler {
return this.cachedFileDiffs.map((diff) => diff.path);
}
- /**
- * Consume retry-after-compaction flag (clears after read).
- */
- consumeRetryAfterCompaction(): boolean {
- const shouldRetry = this.retryAfterCompaction;
- this.retryAfterCompaction = false;
- return shouldRetry;
- }
-
/**
* Handle compaction stream completion
*
@@ -149,8 +133,6 @@ export class CompactionHandler {
return false;
}
- this.retryAfterCompaction = false;
-
// Dedupe: If we've already processed this compaction-request, skip
if (this.processedCompactionRequestIds.has(lastUserMsg.id)) {
return true;
@@ -197,11 +179,6 @@ export class CompactionHandler {
// Check if this was an idle-compaction (auto-triggered due to inactivity)
const isIdleCompaction =
muxMeta?.type === "compaction-request" && muxMeta.source === "idle-compaction";
- const retryContinueMessage =
- muxMeta?.type === "compaction-request" && muxMeta.parsed.continueMessageIsRetry
- ? muxMeta.parsed.continueMessage
- : undefined;
-
// Mark as processed before performing compaction
this.processedCompactionRequestIds.add(lastUserMsg.id);
@@ -209,8 +186,7 @@ export class CompactionHandler {
summary,
event.metadata,
messages,
- isIdleCompaction,
- retryContinueMessage
+ isIdleCompaction
);
if (!result.success) {
log.error("Compaction failed:", result.error);
@@ -262,8 +238,7 @@ export class CompactionHandler {
systemMessageTokens?: number;
},
messages: MuxMessage[],
- isIdleCompaction = false,
- retryContinueMessage?: ContinueMessage
+ isIdleCompaction = false
): Promise> {
// CRITICAL: Delete partial.json BEFORE clearing history
// This prevents a race condition where:
@@ -329,38 +304,6 @@ export class CompactionHandler {
return Err(`Failed to append summary: ${appendResult.error}`);
}
- let retryMessage: MuxMessage | null = null;
- if (retryContinueMessage) {
- const { finalText, metadata: retryMetadata } =
- prepareUserMessageForSend(retryContinueMessage);
- const retryImageParts = retryContinueMessage.imageParts?.map((part) => ({
- type: "file" as const,
- url: part.url,
- mediaType: part.mediaType,
- }));
- const retryMessageMetadata = retryMetadata
- ? { timestamp: Date.now(), muxMetadata: retryMetadata }
- : { timestamp: Date.now() };
-
- retryMessage = createMuxMessage(
- `user-${Date.now()}-${Math.random().toString(36).substring(2, 11)}`,
- "user",
- finalText,
- retryMessageMetadata,
- retryImageParts
- );
-
- const retryAppendResult = await this.historyService.appendToHistory(
- this.workspaceId,
- retryMessage
- );
- if (!retryAppendResult.success) {
- return Err(`Failed to append retry message: ${retryAppendResult.error}`);
- }
-
- this.retryAfterCompaction = true;
- }
-
// Set flag to trigger post-compaction attachment injection on next turn
this.postCompactionAttachmentsPending = true;
@@ -376,10 +319,6 @@ export class CompactionHandler {
// Emit summary message to frontend (add type: "message" for discriminated union)
this.emitChatEvent({ ...summaryMessage, type: "message" });
- if (retryMessage) {
- this.emitChatEvent({ ...retryMessage, type: "message" });
- }
-
return Ok(undefined);
}
From d73482224fd469ecbb74b78ada15a72ba44a8057 Mon Sep 17 00:00:00 2001
From: Ammar
Date: Mon, 19 Jan 2026 09:35:33 -0600
Subject: [PATCH 10/11] fix: tidy compaction retry display
---
src/browser/hooks/useCompactAndRetry.ts | 4 +--
src/browser/utils/chatCommands.ts | 4 +--
src/browser/utils/compaction/format.ts | 45 +++++++++++++++++++------
src/common/types/message.ts | 12 +++++++
src/node/services/agentSession.ts | 9 +++++
5 files changed, 59 insertions(+), 15 deletions(-)
diff --git a/src/browser/hooks/useCompactAndRetry.ts b/src/browser/hooks/useCompactAndRetry.ts
index 9a32f161f1..154f4be76a 100644
--- a/src/browser/hooks/useCompactAndRetry.ts
+++ b/src/browser/hooks/useCompactAndRetry.ts
@@ -6,6 +6,7 @@ import { useWorkspaceState } from "@/browser/stores/WorkspaceStore";
import {
buildCompactionEditText,
formatCompactionCommandLine,
+ getCompactionContinueText,
} from "@/browser/utils/compaction/format";
import {
getExplicitCompactionSuggestion,
@@ -211,8 +212,7 @@ export function useCompactAndRetry(props: { workspaceId: string }): CompactAndRe
},
});
- const shouldAppendNewline =
- !continueMessage?.text || continueMessage.text.trim().length === 0;
+ const shouldAppendNewline = !getCompactionContinueText(continueMessage);
insertIntoChatInput(
fallbackText + (shouldAppendNewline ? "\n" : ""),
diff --git a/src/browser/utils/chatCommands.ts b/src/browser/utils/chatCommands.ts
index 9173777861..f59a99ee09 100644
--- a/src/browser/utils/chatCommands.ts
+++ b/src/browser/utils/chatCommands.ts
@@ -14,6 +14,7 @@ import {
type CompactionRequestData,
type ContinueMessage,
buildContinueMessage,
+ isDefaultContinueMessage,
} from "@/common/types/message";
import type { ReviewNoteData } from "@/common/types/review";
import type { FrontendWorkspaceMetadata } from "@/common/types/workspace";
@@ -647,8 +648,7 @@ export function prepareCompactionMessage(options: CompactionOptions): {
// misread it as a competing instruction. We still keep it in metadata so the backend resumes.
// Only treat it as the default resume when there's no other queued content (images/reviews).
const cm = options.continueMessage;
- const isDefaultResume =
- cm?.text?.trim() === "Continue" && !cm?.imageParts?.length && !cm?.reviews?.length;
+ const isDefaultResume = isDefaultContinueMessage(cm);
if (cm && !isDefaultResume) {
messageText += `\n\nThe user wants to continue with: ${cm.text}`;
diff --git a/src/browser/utils/compaction/format.ts b/src/browser/utils/compaction/format.ts
index 075badab04..f9d400f8c1 100644
--- a/src/browser/utils/compaction/format.ts
+++ b/src/browser/utils/compaction/format.ts
@@ -1,4 +1,5 @@
import type { CompactionRequestData } from "@/common/types/message";
+import { isDefaultContinueMessage } from "@/common/types/message";
/**
* Format compaction command *line* for display.
@@ -20,6 +21,22 @@ export function formatCompactionCommandLine(options: {
return cmd;
}
+/**
+ * Return the visible continue text for a compaction request.
+ * Hides the default resume sentinel ("Continue") and empty text.
+ */
+export function getCompactionContinueText(
+ continueMessage?: CompactionRequestData["continueMessage"]
+): string | null {
+ if (!continueMessage) return null;
+ if (isDefaultContinueMessage(continueMessage)) return null;
+ const continueText = continueMessage.text;
+ if (typeof continueText !== "string" || continueText.trim().length === 0) {
+ return null;
+ }
+ return continueText;
+}
+
/**
* Build the text shown in the editor when editing a /compact request.
*
@@ -30,18 +47,24 @@ export function buildCompactionEditText(request: {
rawCommand: string;
parsed: CompactionRequestData;
}): string {
- const continueMessage = request.parsed.continueMessage;
- const continueText = continueMessage?.text;
- const hasImages = (continueMessage?.imageParts?.length ?? 0) > 0;
- const hasReviews = (continueMessage?.reviews?.length ?? 0) > 0;
- const isDefaultResume =
- typeof continueText === "string" &&
- continueText.trim() === "Continue" &&
- !hasImages &&
- !hasReviews;
-
- if (typeof continueText === "string" && continueText.trim().length > 0 && !isDefaultResume) {
+ const continueText = getCompactionContinueText(request.parsed.continueMessage);
+ if (continueText) {
return `${request.rawCommand}\n${continueText}`;
}
return request.rawCommand;
}
+
+/**
+ * Build the text shown in user message bubbles for a /compact request.
+ * Uses a hard line break so the command and payload render on separate lines.
+ */
+export function buildCompactionDisplayText(request: {
+ rawCommand: string;
+ parsed: CompactionRequestData;
+}): string {
+ const continueText = getCompactionContinueText(request.parsed.continueMessage);
+ if (continueText) {
+ return `${request.rawCommand} \n${continueText}`;
+ }
+ return request.rawCommand;
+}
diff --git a/src/common/types/message.ts b/src/common/types/message.ts
index da8caee14d..6e573b9a67 100644
--- a/src/common/types/message.ts
+++ b/src/common/types/message.ts
@@ -98,6 +98,18 @@ export type PersistedContinueMessage =
mode?: "exec" | "plan";
};
+/**
+ * True when the continue message is the default resume sentinel ("Continue")
+ * with no attachments.
+ */
+export function isDefaultContinueMessage(message?: Partial): boolean {
+ if (!message) return false;
+ const text = typeof message.text === "string" ? message.text.trim() : "";
+ const hasImages = (message.imageParts?.length ?? 0) > 0;
+ const hasReviews = (message.reviews?.length ?? 0) > 0;
+ return text === "Continue" && !hasImages && !hasReviews;
+}
+
/**
* Rebuild a ContinueMessage from persisted data.
* Use this when reading from storage/history where the data may have been
diff --git a/src/node/services/agentSession.ts b/src/node/services/agentSession.ts
index 9ee01d57ed..157a2c33be 100644
--- a/src/node/services/agentSession.ts
+++ b/src/node/services/agentSession.ts
@@ -926,12 +926,17 @@ export class AgentSession {
error: string;
errorType?: string;
}): Promise {
+ const hadCompactionRequest = this.activeCompactionRequest !== undefined;
if (await this.maybeRetryCompactionOnContextExceeded(data)) {
return;
}
this.activeCompactionRequest = undefined;
+ if (hadCompactionRequest && !this.disposed) {
+ this.clearQueue();
+ }
+
const streamError: StreamErrorMessage = {
type: "stream-error",
messageId: data.messageId,
@@ -985,7 +990,11 @@ export class AgentSession {
forward("reasoning-end", (payload) => this.emitChatEvent(payload));
forward("usage-delta", (payload) => this.emitChatEvent(payload));
forward("stream-abort", (payload) => {
+ const hadCompactionRequest = this.activeCompactionRequest !== undefined;
this.activeCompactionRequest = undefined;
+ if (hadCompactionRequest && !this.disposed) {
+ this.clearQueue();
+ }
this.emitChatEvent(payload);
});
forward("runtime-status", (payload) => this.emitChatEvent(payload));
From 001855f068284e44e31ca96f9c77b3fa92167402 Mon Sep 17 00:00:00 2001
From: Ammar
Date: Mon, 19 Jan 2026 10:07:50 -0600
Subject: [PATCH 11/11] fix: dedupe compaction continue queue
---
src/node/services/agentSession.ts | 10 ++++++--
src/node/services/messageQueue.test.ts | 21 ++++++++++++++++
src/node/services/messageQueue.ts | 33 +++++++++++++++++++++++++-
3 files changed, 61 insertions(+), 3 deletions(-)
diff --git a/src/node/services/agentSession.ts b/src/node/services/agentSession.ts
index 157a2c33be..08094b627b 100644
--- a/src/node/services/agentSession.ts
+++ b/src/node/services/agentSession.ts
@@ -681,8 +681,14 @@ export class AgentSession {
sanitizedOptions.muxMetadata = metadata;
}
- this.messageQueue.add(finalText, sanitizedOptions);
- this.emitQueuedMessageChanged();
+ const dedupeKey = JSON.stringify({
+ text: finalText.trim(),
+ images: (continueImageParts ?? []).map((image) => `${image.mediaType}:${image.url}`),
+ });
+
+ if (this.messageQueue.addOnce(finalText, sanitizedOptions, dedupeKey)) {
+ this.emitQueuedMessageChanged();
+ }
}
if (this.disposed) {
diff --git a/src/node/services/messageQueue.test.ts b/src/node/services/messageQueue.test.ts
index 6487ddbc72..257dbc1dbb 100644
--- a/src/node/services/messageQueue.test.ts
+++ b/src/node/services/messageQueue.test.ts
@@ -159,6 +159,27 @@ describe("MessageQueue", () => {
});
});
+ describe("addOnce", () => {
+ it("should dedupe repeated entries by key", () => {
+ const image = { url: "", mediaType: "image/png" };
+ const addedFirst = queue.addOnce(
+ "Follow up",
+ { model: "gpt-4", imageParts: [image] },
+ "follow-up"
+ );
+ const addedSecond = queue.addOnce(
+ "Follow up",
+ { model: "gpt-4", imageParts: [image] },
+ "follow-up"
+ );
+
+ expect(addedFirst).toBe(true);
+ expect(addedSecond).toBe(false);
+ expect(queue.getMessages()).toEqual(["Follow up"]);
+ expect(queue.getImageParts()).toEqual([image]);
+ });
+ });
+
describe("multi-message batching", () => {
it("should batch multiple follow-up messages", () => {
queue.add("First message");
diff --git a/src/node/services/messageQueue.ts b/src/node/services/messageQueue.ts
index 22ff7aeb87..f1e82d1484 100644
--- a/src/node/services/messageQueue.ts
+++ b/src/node/services/messageQueue.ts
@@ -66,6 +66,7 @@ export class MessageQueue {
private firstMuxMetadata?: unknown;
private latestOptions?: SendMessageOptions;
private accumulatedImages: ImagePart[] = [];
+ private dedupeKeys: Set = new Set();
/**
* Check if the queue currently contains a compaction request.
@@ -82,12 +83,39 @@ export class MessageQueue {
* @throws Error if trying to add a compaction request when queue already has messages
*/
add(message: string, options?: SendMessageOptions & { imageParts?: ImagePart[] }): void {
+ this.addInternal(message, options);
+ }
+
+ /**
+ * Add a message to the queue once, keyed by dedupeKey.
+ * Returns true if the message was queued.
+ */
+ addOnce(
+ message: string,
+ options?: SendMessageOptions & { imageParts?: ImagePart[] },
+ dedupeKey?: string
+ ): boolean {
+ if (dedupeKey !== undefined && this.dedupeKeys.has(dedupeKey)) {
+ return false;
+ }
+
+ const didAdd = this.addInternal(message, options);
+ if (didAdd && dedupeKey !== undefined) {
+ this.dedupeKeys.add(dedupeKey);
+ }
+ return didAdd;
+ }
+
+ private addInternal(
+ message: string,
+ options?: SendMessageOptions & { imageParts?: ImagePart[] }
+ ): boolean {
const trimmedMessage = message.trim();
const hasImages = options?.imageParts && options.imageParts.length > 0;
// Reject if both text and images are empty
if (trimmedMessage.length === 0 && !hasImages) {
- return;
+ return false;
}
const incomingIsCompaction = isCompactionMetadata(options?.muxMetadata);
@@ -139,6 +167,8 @@ export class MessageQueue {
this.accumulatedImages.push(...imageParts);
}
}
+
+ return true;
}
/**
@@ -218,6 +248,7 @@ export class MessageQueue {
this.firstMuxMetadata = undefined;
this.latestOptions = undefined;
this.accumulatedImages = [];
+ this.dedupeKeys.clear();
}
/**