From 523b8c0a4a46abcac54c7c82b034bbcb99940f70 Mon Sep 17 00:00:00 2001 From: YoVinchen Date: Thu, 9 Apr 2026 14:01:41 +0800 Subject: [PATCH] Strip dead OTel event noise from telemetry compatibility paths The open build no longer exports OpenTelemetry events, but several user-prompt, tool, hook, API, and survey paths were still constructing and calling a no-op logOTelEvent helper. This removes those dead calls, drops the now-unused helper module, and deletes an unreferenced GrowthBook experiment event artifact so the remaining compatibility layer is less distracting during future audits. Constraint: Keep the local logEvent and tracing compatibility surfaces untouched where they still structure control flow Constraint: Avoid touching unrelated bridge and session changes already present in the worktree Rejected: Remove sessionTracing compatibility entirely | Call surface is still broad and intertwined with non-telemetry control flow Rejected: Leave no-op OTel event calls in place | They add audit noise without preserving behavior Confidence: high Scope-risk: narrow Reversibility: clean Directive: Continue treating remaining telemetry-named helpers as removable only when their call sites are proven behavior-free Tested: bun test src/services/analytics/index.test.ts src/components/FeedbackSurvey/submitTranscriptShare.test.ts Tested: bun run ./scripts/build.ts Not-tested: bun x tsc --noEmit (repository has pre-existing unrelated type errors) --- .../FeedbackSurvey/useFeedbackSurvey.tsx | 19 +- .../FeedbackSurvey/useMemorySurvey.tsx | 19 +- .../FeedbackSurvey/usePostCompactSurvey.tsx | 14 +- src/hooks/toolPermission/permissionLogging.ts | 8 +- src/services/api/logging.ts | 23 -- src/services/tools/toolExecution.ts | 58 +---- .../v1/growthbook_experiment_event.ts | 223 ------------------ src/utils/hooks.ts | 32 --- .../processUserInput/processSlashCommand.tsx | 9 +- .../processUserInput/processTextPrompt.ts | 20 -- src/utils/telemetry/events.ts | 14 -- 11 files changed, 11 insertions(+), 428 deletions(-) delete mode 100644 src/types/generated/events_mono/growthbook/v1/growthbook_experiment_event.ts delete mode 100644 src/utils/telemetry/events.ts diff --git a/src/components/FeedbackSurvey/useFeedbackSurvey.tsx b/src/components/FeedbackSurvey/useFeedbackSurvey.tsx index 20bab3d..fda055b 100644 --- a/src/components/FeedbackSurvey/useFeedbackSurvey.tsx +++ b/src/components/FeedbackSurvey/useFeedbackSurvey.tsx @@ -9,7 +9,6 @@ import { isEnvTruthy } from '../../utils/envUtils.js'; import { getLastAssistantMessage } from '../../utils/messages.js'; import { getMainLoopModel } from '../../utils/model/model.js'; import { getInitialSettings } from '../../utils/settings/settings.js'; -import { logOTelEvent } from '../../utils/telemetry/events.js'; import { submitTranscriptShare, type TranscriptShareTrigger } from './submitTranscriptShare.js'; import type { TranscriptShareResponse } from './TranscriptSharePrompt.js'; import { useSurveyState } from './useSurveyState.js'; @@ -99,11 +98,6 @@ export function useFeedbackSurvey(messages: Message[], isLoading: boolean, submi last_assistant_message_id: lastAssistantMessageIdRef.current as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, survey_type: surveyType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS }); - void logOTelEvent('feedback_survey', { - event_type: 'appeared', - appearance_id: appearanceId, - survey_type: surveyType - }); }, [updateLastShownTime, surveyType]); const onSelect = useCallback((appearanceId_0: string, selected: FeedbackSurveyResponse) => { updateLastShownTime(Date.now(), submitCountRef.current); @@ -114,12 +108,6 @@ export function useFeedbackSurvey(messages: Message[], isLoading: boolean, submi last_assistant_message_id: lastAssistantMessageIdRef.current as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, survey_type: surveyType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS }); - void logOTelEvent('feedback_survey', { - event_type: 'responded', - appearance_id: appearanceId_0, - response: selected, - survey_type: surveyType - }); }, [updateLastShownTime, surveyType]); const shouldShowTranscriptPrompt = useCallback((selected_0: FeedbackSurveyResponse) => { // Only bad and good ratings trigger the transcript ask @@ -150,11 +138,6 @@ export function useFeedbackSurvey(messages: Message[], isLoading: boolean, submi survey_type: surveyType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, trigger: trigger as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS }); - void logOTelEvent('feedback_survey', { - event_type: 'transcript_prompt_appeared', - appearance_id: appearanceId_1, - survey_type: surveyType - }); }, [surveyType]); const onTranscriptSelect = useCallback(async (appearanceId_2: string, selected_1: TranscriptShareResponse, surveyResponse_0: FeedbackSurveyResponse | null): Promise => { const trigger_0: TranscriptShareTrigger = surveyResponse_0 === 'good' ? 'good_feedback_survey' : 'bad_feedback_survey'; @@ -293,4 +276,4 @@ export function useFeedbackSurvey(messages: Message[], isLoading: boolean, submi handleTranscriptSelect }; } -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["useCallback","useEffect","useMemo","useRef","useState","useDynamicConfig","isFeedbackSurveyDisabled","AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS","logEvent","isPolicyAllowed","Message","getGlobalConfig","saveGlobalConfig","isEnvTruthy","getLastAssistantMessage","getMainLoopModel","getInitialSettings","logOTelEvent","submitTranscriptShare","TranscriptShareTrigger","TranscriptShareResponse","useSurveyState","FeedbackSurveyResponse","FeedbackSurveyType","FeedbackSurveyConfig","minTimeBeforeFeedbackMs","minTimeBetweenFeedbackMs","minTimeBetweenGlobalFeedbackMs","minUserTurnsBeforeFeedback","minUserTurnsBetweenFeedback","hideThanksAfterMs","onForModels","probability","TranscriptAskConfig","DEFAULT_FEEDBACK_SURVEY_CONFIG","DEFAULT_TRANSCRIPT_ASK_CONFIG","useFeedbackSurvey","messages","isLoading","submitCount","surveyType","hasActivePrompt","state","lastResponse","handleSelect","selected","handleTranscriptSelect","lastAssistantMessageIdRef","current","message","id","feedbackSurvey","setFeedbackSurvey","timeLastShown","submitCountAtLastAppearance","config","badTranscriptAskConfig","goodTranscriptAskConfig","settingsRate","feedbackSurveyRate","sessionStartTime","Date","now","submitCountAtSessionStart","submitCountRef","messagesRef","probabilityPassedRef","lastEligibleSubmitCountRef","updateLastShownTime","timestamp","submitCountValue","prev","feedbackSurveyState","lastShownTime","onOpen","appearanceId","event_type","appearance_id","last_assistant_message_id","survey_type","onSelect","response","shouldShowTranscriptPrompt","transcriptShareDismissed","Math","random","onTranscriptPromptShown","surveyResponse","trigger","onTranscriptSelect","Promise","result","success","open","currentModel","isModelAllowed","length","includes","shouldOpen","process","env","CLAUDE_FORCE_DISPLAY_SURVEY","CLAUDE_CODE_DISABLE_FEEDBACK_SURVEY","timeSinceLastShown","timeSinceSessionStart","globalFeedbackState","timeSinceGlobalLastShown"],"sources":["useFeedbackSurvey.tsx"],"sourcesContent":["import { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport { useDynamicConfig } from 'src/hooks/useDynamicConfig.js'\nimport { isFeedbackSurveyDisabled } from 'src/services/analytics/config.js'\nimport {\n  type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n  logEvent,\n} from 'src/services/analytics/index.js'\nimport { isPolicyAllowed } from '../../services/policyLimits/index.js'\nimport type { Message } from '../../types/message.js'\nimport { getGlobalConfig, saveGlobalConfig } from '../../utils/config.js'\nimport { isEnvTruthy } from '../../utils/envUtils.js'\nimport { getLastAssistantMessage } from '../../utils/messages.js'\nimport { getMainLoopModel } from '../../utils/model/model.js'\nimport { getInitialSettings } from '../../utils/settings/settings.js'\nimport { logOTelEvent } from '../../utils/telemetry/events.js'\nimport {\n  submitTranscriptShare,\n  type TranscriptShareTrigger,\n} from './submitTranscriptShare.js'\nimport type { TranscriptShareResponse } from './TranscriptSharePrompt.js'\nimport { useSurveyState } from './useSurveyState.js'\nimport type { FeedbackSurveyResponse, FeedbackSurveyType } from './utils.js'\n\ntype FeedbackSurveyConfig = {\n  minTimeBeforeFeedbackMs: number\n  minTimeBetweenFeedbackMs: number\n  minTimeBetweenGlobalFeedbackMs: number\n  minUserTurnsBeforeFeedback: number\n  minUserTurnsBetweenFeedback: number\n  hideThanksAfterMs: number\n  onForModels: string[]\n  probability: number\n}\n\ntype TranscriptAskConfig = {\n  probability: number\n}\n\nconst DEFAULT_FEEDBACK_SURVEY_CONFIG: FeedbackSurveyConfig = {\n  minTimeBeforeFeedbackMs: 600000,\n  minTimeBetweenFeedbackMs: 3600000,\n  minTimeBetweenGlobalFeedbackMs: 100000000,\n  minUserTurnsBeforeFeedback: 5,\n  minUserTurnsBetweenFeedback: 10,\n  hideThanksAfterMs: 3000,\n  onForModels: ['*'],\n  probability: 0.005,\n}\n\nconst DEFAULT_TRANSCRIPT_ASK_CONFIG: TranscriptAskConfig = {\n  probability: 0,\n}\n\nexport function useFeedbackSurvey(\n  messages: Message[],\n  isLoading: boolean,\n  submitCount: number,\n  surveyType: FeedbackSurveyType = 'session',\n  hasActivePrompt: boolean = false,\n): {\n  state:\n    | 'closed'\n    | 'open'\n    | 'thanks'\n    | 'transcript_prompt'\n    | 'submitting'\n    | 'submitted'\n  lastResponse: FeedbackSurveyResponse | null\n  handleSelect: (selected: FeedbackSurveyResponse) => boolean\n  handleTranscriptSelect: (selected: TranscriptShareResponse) => void\n} {\n  const lastAssistantMessageIdRef = useRef('unknown')\n  lastAssistantMessageIdRef.current =\n    getLastAssistantMessage(messages)?.message?.id || 'unknown'\n  const [feedbackSurvey, setFeedbackSurvey] = useState<{\n    timeLastShown: number | null\n    submitCountAtLastAppearance: number | null\n  }>(() => ({ timeLastShown: null, submitCountAtLastAppearance: null }))\n  const config = useDynamicConfig<FeedbackSurveyConfig>(\n    'tengu_feedback_survey_config',\n    DEFAULT_FEEDBACK_SURVEY_CONFIG,\n  )\n  const badTranscriptAskConfig = useDynamicConfig<TranscriptAskConfig>(\n    'tengu_bad_survey_transcript_ask_config',\n    DEFAULT_TRANSCRIPT_ASK_CONFIG,\n  )\n  const goodTranscriptAskConfig = useDynamicConfig<TranscriptAskConfig>(\n    'tengu_good_survey_transcript_ask_config',\n    DEFAULT_TRANSCRIPT_ASK_CONFIG,\n  )\n  const settingsRate = getInitialSettings().feedbackSurveyRate\n  const sessionStartTime = useRef(Date.now())\n  const submitCountAtSessionStart = useRef(submitCount)\n  const submitCountRef = useRef(submitCount)\n  submitCountRef.current = submitCount\n  const messagesRef = useRef(messages)\n  messagesRef.current = messages\n  // Probability gate: roll once when eligibility conditions are met, not on every\n  // useMemo re-evaluation. Without this, each dependency change (submitCount,\n  // isLoading toggle, etc.) re-rolls Math.random(), making the survey almost\n  // certain to appear after enough renders.\n  const probabilityPassedRef = useRef(false)\n  const lastEligibleSubmitCountRef = useRef<number | null>(null)\n\n  const updateLastShownTime = useCallback(\n    (timestamp: number, submitCountValue: number) => {\n      setFeedbackSurvey(prev => {\n        if (\n          prev.timeLastShown === timestamp &&\n          prev.submitCountAtLastAppearance === submitCountValue\n        ) {\n          return prev\n        }\n        return {\n          timeLastShown: timestamp,\n          submitCountAtLastAppearance: submitCountValue,\n        }\n      })\n      // Persist cross-session pacing state (previously done by onChangeAppState observer)\n      if (getGlobalConfig().feedbackSurveyState?.lastShownTime !== timestamp) {\n        saveGlobalConfig(current => ({\n          ...current,\n          feedbackSurveyState: {\n            lastShownTime: timestamp,\n          },\n        }))\n      }\n    },\n    [],\n  )\n\n  const onOpen = useCallback(\n    (appearanceId: string) => {\n      updateLastShownTime(Date.now(), submitCountRef.current)\n      logEvent('tengu_feedback_survey_event', {\n        event_type:\n          'appeared' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        appearance_id:\n          appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        last_assistant_message_id:\n          lastAssistantMessageIdRef.current as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        survey_type:\n          surveyType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      })\n      void logOTelEvent('feedback_survey', {\n        event_type: 'appeared',\n        appearance_id: appearanceId,\n        survey_type: surveyType,\n      })\n    },\n    [updateLastShownTime, surveyType],\n  )\n\n  const onSelect = useCallback(\n    (appearanceId: string, selected: FeedbackSurveyResponse) => {\n      updateLastShownTime(Date.now(), submitCountRef.current)\n      logEvent('tengu_feedback_survey_event', {\n        event_type:\n          'responded' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        appearance_id:\n          appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        response:\n          selected as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        last_assistant_message_id:\n          lastAssistantMessageIdRef.current as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        survey_type:\n          surveyType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      })\n      void logOTelEvent('feedback_survey', {\n        event_type: 'responded',\n        appearance_id: appearanceId,\n        response: selected,\n        survey_type: surveyType,\n      })\n    },\n    [updateLastShownTime, surveyType],\n  )\n\n  const shouldShowTranscriptPrompt = useCallback(\n    (selected: FeedbackSurveyResponse) => {\n      // Only bad and good ratings trigger the transcript ask\n      if (selected !== 'bad' && selected !== 'good') {\n        return false\n      }\n\n      // Don't show if user previously chose \"Don't ask again\"\n      if (getGlobalConfig().transcriptShareDismissed) {\n        return false\n      }\n\n      // Don't show if product feedback is blocked by org policy (ZDR)\n      if (!isPolicyAllowed('allow_product_feedback')) {\n        return false\n      }\n\n      // Probability gate from GrowthBook config (separate per rating)\n      const probability =\n        selected === 'bad'\n          ? badTranscriptAskConfig.probability\n          : goodTranscriptAskConfig.probability\n      return Math.random() <= probability\n    },\n    [badTranscriptAskConfig.probability, goodTranscriptAskConfig.probability],\n  )\n\n  const onTranscriptPromptShown = useCallback(\n    (appearanceId: string, surveyResponse: FeedbackSurveyResponse) => {\n      const trigger: TranscriptShareTrigger =\n        surveyResponse === 'good'\n          ? 'good_feedback_survey'\n          : 'bad_feedback_survey'\n      logEvent('tengu_feedback_survey_event', {\n        event_type:\n          'transcript_prompt_appeared' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        appearance_id:\n          appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        last_assistant_message_id:\n          lastAssistantMessageIdRef.current as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        survey_type:\n          surveyType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        trigger:\n          trigger as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      })\n      void logOTelEvent('feedback_survey', {\n        event_type: 'transcript_prompt_appeared',\n        appearance_id: appearanceId,\n        survey_type: surveyType,\n      })\n    },\n    [surveyType],\n  )\n\n  const onTranscriptSelect = useCallback(\n    async (\n      appearanceId: string,\n      selected: TranscriptShareResponse,\n      surveyResponse: FeedbackSurveyResponse | null,\n    ): Promise<boolean> => {\n      const trigger: TranscriptShareTrigger =\n        surveyResponse === 'good'\n          ? 'good_feedback_survey'\n          : 'bad_feedback_survey'\n\n      logEvent('tengu_feedback_survey_event', {\n        event_type:\n          `transcript_share_${selected}` as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        appearance_id:\n          appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        last_assistant_message_id:\n          lastAssistantMessageIdRef.current as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        survey_type:\n          surveyType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        trigger:\n          trigger as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      })\n\n      if (selected === 'dont_ask_again') {\n        saveGlobalConfig(current => ({\n          ...current,\n          transcriptShareDismissed: true,\n        }))\n      }\n\n      if (selected === 'yes') {\n        const result = await submitTranscriptShare(\n          messagesRef.current,\n          trigger,\n          appearanceId,\n        )\n        logEvent('tengu_feedback_survey_event', {\n          event_type: (result.success\n            ? 'transcript_share_submitted'\n            : 'transcript_share_failed') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n          appearance_id:\n            appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n          trigger:\n            trigger as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        })\n        return result.success\n      }\n\n      return false\n    },\n    [surveyType],\n  )\n\n  const { state, lastResponse, open, handleSelect, handleTranscriptSelect } =\n    useSurveyState({\n      hideThanksAfterMs: config.hideThanksAfterMs,\n      onOpen,\n      onSelect,\n      shouldShowTranscriptPrompt,\n      onTranscriptPromptShown,\n      onTranscriptSelect,\n    })\n\n  const currentModel = getMainLoopModel()\n  const isModelAllowed = useMemo(() => {\n    if (config.onForModels.length === 0) {\n      return false\n    }\n    if (config.onForModels.includes('*')) {\n      return true\n    }\n    return config.onForModels.includes(currentModel)\n  }, [config.onForModels, currentModel])\n\n  const shouldOpen = useMemo(() => {\n    if (state !== 'closed') {\n      return false\n    }\n\n    if (isLoading) {\n      return false\n    }\n\n    // Don't show survey when permission or ask question prompts are visible\n    if (hasActivePrompt) {\n      return false\n    }\n\n    // Force display for testing\n    if (\n      process.env.CLAUDE_FORCE_DISPLAY_SURVEY &&\n      !feedbackSurvey.timeLastShown\n    ) {\n      return true\n    }\n\n    if (!isModelAllowed) {\n      return false\n    }\n\n    if (isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_FEEDBACK_SURVEY)) {\n      return false\n    }\n\n    if (isFeedbackSurveyDisabled()) {\n      return false\n    }\n\n    // Check if product feedback is allowed by org policy\n    if (!isPolicyAllowed('allow_product_feedback')) {\n      return false\n    }\n\n    // Check session-local pacing\n    if (feedbackSurvey.timeLastShown) {\n      // Check time elapsed since last appearance in this session\n      const timeSinceLastShown = Date.now() - feedbackSurvey.timeLastShown\n      if (timeSinceLastShown < config.minTimeBetweenFeedbackMs) {\n        return false\n      }\n      // Check user turn requirement for subsequent appearances\n      if (\n        feedbackSurvey.submitCountAtLastAppearance !== null &&\n        submitCount <\n          feedbackSurvey.submitCountAtLastAppearance +\n            config.minUserTurnsBetweenFeedback\n      ) {\n        return false\n      }\n    } else {\n      // First appearance in this session\n      const timeSinceSessionStart = Date.now() - sessionStartTime.current\n      if (timeSinceSessionStart < config.minTimeBeforeFeedbackMs) {\n        return false\n      }\n      if (\n        submitCount <\n        submitCountAtSessionStart.current + config.minUserTurnsBeforeFeedback\n      ) {\n        return false\n      }\n    }\n\n    // Probability check: roll once per eligibility window to avoid re-rolling\n    // on every useMemo re-evaluation (which would make triggering near-certain).\n    if (lastEligibleSubmitCountRef.current !== submitCount) {\n      lastEligibleSubmitCountRef.current = submitCount\n      probabilityPassedRef.current =\n        Math.random() <= (settingsRate ?? config.probability)\n    }\n    if (!probabilityPassedRef.current) {\n      return false\n    }\n\n    // Check global pacing (across all sessions)\n    // Leave this till last because it reads from the filesystem which is expensive.\n    const globalFeedbackState = getGlobalConfig().feedbackSurveyState\n    if (globalFeedbackState?.lastShownTime) {\n      const timeSinceGlobalLastShown =\n        Date.now() - globalFeedbackState.lastShownTime\n      if (timeSinceGlobalLastShown < config.minTimeBetweenGlobalFeedbackMs) {\n        return false\n      }\n    }\n\n    return true\n  }, [\n    state,\n    isLoading,\n    hasActivePrompt,\n    isModelAllowed,\n    feedbackSurvey.timeLastShown,\n    feedbackSurvey.submitCountAtLastAppearance,\n    submitCount,\n    config.minTimeBetweenFeedbackMs,\n    config.minTimeBetweenGlobalFeedbackMs,\n    config.minUserTurnsBetweenFeedback,\n    config.minTimeBeforeFeedbackMs,\n    config.minUserTurnsBeforeFeedback,\n    config.probability,\n    settingsRate,\n  ])\n\n  useEffect(() => {\n    if (shouldOpen) {\n      open()\n    }\n  }, [shouldOpen, open])\n\n  return { state, lastResponse, handleSelect, handleTranscriptSelect }\n}\n"],"mappings":"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","ignoreList":[]} \ No newline at end of file +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["useCallback","useEffect","useMemo","useRef","useState","useDynamicConfig","isFeedbackSurveyDisabled","AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS","logEvent","isPolicyAllowed","Message","getGlobalConfig","saveGlobalConfig","isEnvTruthy","getLastAssistantMessage","getMainLoopModel","getInitialSettings","logOTelEvent","submitTranscriptShare","TranscriptShareTrigger","TranscriptShareResponse","useSurveyState","FeedbackSurveyResponse","FeedbackSurveyType","FeedbackSurveyConfig","minTimeBeforeFeedbackMs","minTimeBetweenFeedbackMs","minTimeBetweenGlobalFeedbackMs","minUserTurnsBeforeFeedback","minUserTurnsBetweenFeedback","hideThanksAfterMs","onForModels","probability","TranscriptAskConfig","DEFAULT_FEEDBACK_SURVEY_CONFIG","DEFAULT_TRANSCRIPT_ASK_CONFIG","useFeedbackSurvey","messages","isLoading","submitCount","surveyType","hasActivePrompt","state","lastResponse","handleSelect","selected","handleTranscriptSelect","lastAssistantMessageIdRef","current","message","id","feedbackSurvey","setFeedbackSurvey","timeLastShown","submitCountAtLastAppearance","config","badTranscriptAskConfig","goodTranscriptAskConfig","settingsRate","feedbackSurveyRate","sessionStartTime","Date","now","submitCountAtSessionStart","submitCountRef","messagesRef","probabilityPassedRef","lastEligibleSubmitCountRef","updateLastShownTime","timestamp","submitCountValue","prev","feedbackSurveyState","lastShownTime","onOpen","appearanceId","event_type","appearance_id","last_assistant_message_id","survey_type","onSelect","response","shouldShowTranscriptPrompt","transcriptShareDismissed","Math","random","onTranscriptPromptShown","surveyResponse","trigger","onTranscriptSelect","Promise","result","success","open","currentModel","isModelAllowed","length","includes","shouldOpen","process","env","CLAUDE_FORCE_DISPLAY_SURVEY","CLAUDE_CODE_DISABLE_FEEDBACK_SURVEY","timeSinceLastShown","timeSinceSessionStart","globalFeedbackState","timeSinceGlobalLastShown"],"sources":["useFeedbackSurvey.tsx"],"sourcesContent":["import { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport { useDynamicConfig } from 'src/hooks/useDynamicConfig.js'\nimport { isFeedbackSurveyDisabled } from 'src/services/analytics/config.js'\nimport {\n  type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n  logEvent,\n} from 'src/services/analytics/index.js'\nimport { isPolicyAllowed } from '../../services/policyLimits/index.js'\nimport type { Message } from '../../types/message.js'\nimport { getGlobalConfig, saveGlobalConfig } from '../../utils/config.js'\nimport { isEnvTruthy } from '../../utils/envUtils.js'\nimport { getLastAssistantMessage } from '../../utils/messages.js'\nimport { getMainLoopModel } from '../../utils/model/model.js'\nimport { getInitialSettings } from '../../utils/settings/settings.js'\nimport { logOTelEvent } from '../../utils/telemetry/events.js'\nimport {\n  submitTranscriptShare,\n  type TranscriptShareTrigger,\n} from './submitTranscriptShare.js'\nimport type { TranscriptShareResponse } from './TranscriptSharePrompt.js'\nimport { useSurveyState } from './useSurveyState.js'\nimport type { FeedbackSurveyResponse, FeedbackSurveyType } from './utils.js'\n\ntype FeedbackSurveyConfig = {\n  minTimeBeforeFeedbackMs: number\n  minTimeBetweenFeedbackMs: number\n  minTimeBetweenGlobalFeedbackMs: number\n  minUserTurnsBeforeFeedback: number\n  minUserTurnsBetweenFeedback: number\n  hideThanksAfterMs: number\n  onForModels: string[]\n  probability: number\n}\n\ntype TranscriptAskConfig = {\n  probability: number\n}\n\nconst DEFAULT_FEEDBACK_SURVEY_CONFIG: FeedbackSurveyConfig = {\n  minTimeBeforeFeedbackMs: 600000,\n  minTimeBetweenFeedbackMs: 3600000,\n  minTimeBetweenGlobalFeedbackMs: 100000000,\n  minUserTurnsBeforeFeedback: 5,\n  minUserTurnsBetweenFeedback: 10,\n  hideThanksAfterMs: 3000,\n  onForModels: ['*'],\n  probability: 0.005,\n}\n\nconst DEFAULT_TRANSCRIPT_ASK_CONFIG: TranscriptAskConfig = {\n  probability: 0,\n}\n\nexport function useFeedbackSurvey(\n  messages: Message[],\n  isLoading: boolean,\n  submitCount: number,\n  surveyType: FeedbackSurveyType = 'session',\n  hasActivePrompt: boolean = false,\n): {\n  state:\n    | 'closed'\n    | 'open'\n    | 'thanks'\n    | 'transcript_prompt'\n    | 'submitting'\n    | 'submitted'\n  lastResponse: FeedbackSurveyResponse | null\n  handleSelect: (selected: FeedbackSurveyResponse) => boolean\n  handleTranscriptSelect: (selected: TranscriptShareResponse) => void\n} {\n  const lastAssistantMessageIdRef = useRef('unknown')\n  lastAssistantMessageIdRef.current =\n    getLastAssistantMessage(messages)?.message?.id || 'unknown'\n  const [feedbackSurvey, setFeedbackSurvey] = useState<{\n    timeLastShown: number | null\n    submitCountAtLastAppearance: number | null\n  }>(() => ({ timeLastShown: null, submitCountAtLastAppearance: null }))\n  const config = useDynamicConfig<FeedbackSurveyConfig>(\n    'tengu_feedback_survey_config',\n    DEFAULT_FEEDBACK_SURVEY_CONFIG,\n  )\n  const badTranscriptAskConfig = useDynamicConfig<TranscriptAskConfig>(\n    'tengu_bad_survey_transcript_ask_config',\n    DEFAULT_TRANSCRIPT_ASK_CONFIG,\n  )\n  const goodTranscriptAskConfig = useDynamicConfig<TranscriptAskConfig>(\n    'tengu_good_survey_transcript_ask_config',\n    DEFAULT_TRANSCRIPT_ASK_CONFIG,\n  )\n  const settingsRate = getInitialSettings().feedbackSurveyRate\n  const sessionStartTime = useRef(Date.now())\n  const submitCountAtSessionStart = useRef(submitCount)\n  const submitCountRef = useRef(submitCount)\n  submitCountRef.current = submitCount\n  const messagesRef = useRef(messages)\n  messagesRef.current = messages\n  // Probability gate: roll once when eligibility conditions are met, not on every\n  // useMemo re-evaluation. Without this, each dependency change (submitCount,\n  // isLoading toggle, etc.) re-rolls Math.random(), making the survey almost\n  // certain to appear after enough renders.\n  const probabilityPassedRef = useRef(false)\n  const lastEligibleSubmitCountRef = useRef<number | null>(null)\n\n  const updateLastShownTime = useCallback(\n    (timestamp: number, submitCountValue: number) => {\n      setFeedbackSurvey(prev => {\n        if (\n          prev.timeLastShown === timestamp &&\n          prev.submitCountAtLastAppearance === submitCountValue\n        ) {\n          return prev\n        }\n        return {\n          timeLastShown: timestamp,\n          submitCountAtLastAppearance: submitCountValue,\n        }\n      })\n      // Persist cross-session pacing state (previously done by onChangeAppState observer)\n      if (getGlobalConfig().feedbackSurveyState?.lastShownTime !== timestamp) {\n        saveGlobalConfig(current => ({\n          ...current,\n          feedbackSurveyState: {\n            lastShownTime: timestamp,\n          },\n        }))\n      }\n    },\n    [],\n  )\n\n  const onOpen = useCallback(\n    (appearanceId: string) => {\n      updateLastShownTime(Date.now(), submitCountRef.current)\n      logEvent('tengu_feedback_survey_event', {\n        event_type:\n          'appeared' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        appearance_id:\n          appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        last_assistant_message_id:\n          lastAssistantMessageIdRef.current as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        survey_type:\n          surveyType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      })\n      void logOTelEvent('feedback_survey', {\n        event_type: 'appeared',\n        appearance_id: appearanceId,\n        survey_type: surveyType,\n      })\n    },\n    [updateLastShownTime, surveyType],\n  )\n\n  const onSelect = useCallback(\n    (appearanceId: string, selected: FeedbackSurveyResponse) => {\n      updateLastShownTime(Date.now(), submitCountRef.current)\n      logEvent('tengu_feedback_survey_event', {\n        event_type:\n          'responded' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        appearance_id:\n          appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        response:\n          selected as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        last_assistant_message_id:\n          lastAssistantMessageIdRef.current as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        survey_type:\n          surveyType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      })\n      void logOTelEvent('feedback_survey', {\n        event_type: 'responded',\n        appearance_id: appearanceId,\n        response: selected,\n        survey_type: surveyType,\n      })\n    },\n    [updateLastShownTime, surveyType],\n  )\n\n  const shouldShowTranscriptPrompt = useCallback(\n    (selected: FeedbackSurveyResponse) => {\n      // Only bad and good ratings trigger the transcript ask\n      if (selected !== 'bad' && selected !== 'good') {\n        return false\n      }\n\n      // Don't show if user previously chose \"Don't ask again\"\n      if (getGlobalConfig().transcriptShareDismissed) {\n        return false\n      }\n\n      // Don't show if product feedback is blocked by org policy (ZDR)\n      if (!isPolicyAllowed('allow_product_feedback')) {\n        return false\n      }\n\n      // Probability gate from GrowthBook config (separate per rating)\n      const probability =\n        selected === 'bad'\n          ? badTranscriptAskConfig.probability\n          : goodTranscriptAskConfig.probability\n      return Math.random() <= probability\n    },\n    [badTranscriptAskConfig.probability, goodTranscriptAskConfig.probability],\n  )\n\n  const onTranscriptPromptShown = useCallback(\n    (appearanceId: string, surveyResponse: FeedbackSurveyResponse) => {\n      const trigger: TranscriptShareTrigger =\n        surveyResponse === 'good'\n          ? 'good_feedback_survey'\n          : 'bad_feedback_survey'\n      logEvent('tengu_feedback_survey_event', {\n        event_type:\n          'transcript_prompt_appeared' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        appearance_id:\n          appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        last_assistant_message_id:\n          lastAssistantMessageIdRef.current as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        survey_type:\n          surveyType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        trigger:\n          trigger as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      })\n      void logOTelEvent('feedback_survey', {\n        event_type: 'transcript_prompt_appeared',\n        appearance_id: appearanceId,\n        survey_type: surveyType,\n      })\n    },\n    [surveyType],\n  )\n\n  const onTranscriptSelect = useCallback(\n    async (\n      appearanceId: string,\n      selected: TranscriptShareResponse,\n      surveyResponse: FeedbackSurveyResponse | null,\n    ): Promise<boolean> => {\n      const trigger: TranscriptShareTrigger =\n        surveyResponse === 'good'\n          ? 'good_feedback_survey'\n          : 'bad_feedback_survey'\n\n      logEvent('tengu_feedback_survey_event', {\n        event_type:\n          `transcript_share_${selected}` as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        appearance_id:\n          appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        last_assistant_message_id:\n          lastAssistantMessageIdRef.current as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        survey_type:\n          surveyType as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        trigger:\n          trigger as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      })\n\n      if (selected === 'dont_ask_again') {\n        saveGlobalConfig(current => ({\n          ...current,\n          transcriptShareDismissed: true,\n        }))\n      }\n\n      if (selected === 'yes') {\n        const result = await submitTranscriptShare(\n          messagesRef.current,\n          trigger,\n          appearanceId,\n        )\n        logEvent('tengu_feedback_survey_event', {\n          event_type: (result.success\n            ? 'transcript_share_submitted'\n            : 'transcript_share_failed') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n          appearance_id:\n            appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n          trigger:\n            trigger as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        })\n        return result.success\n      }\n\n      return false\n    },\n    [surveyType],\n  )\n\n  const { state, lastResponse, open, handleSelect, handleTranscriptSelect } =\n    useSurveyState({\n      hideThanksAfterMs: config.hideThanksAfterMs,\n      onOpen,\n      onSelect,\n      shouldShowTranscriptPrompt,\n      onTranscriptPromptShown,\n      onTranscriptSelect,\n    })\n\n  const currentModel = getMainLoopModel()\n  const isModelAllowed = useMemo(() => {\n    if (config.onForModels.length === 0) {\n      return false\n    }\n    if (config.onForModels.includes('*')) {\n      return true\n    }\n    return config.onForModels.includes(currentModel)\n  }, [config.onForModels, currentModel])\n\n  const shouldOpen = useMemo(() => {\n    if (state !== 'closed') {\n      return false\n    }\n\n    if (isLoading) {\n      return false\n    }\n\n    // Don't show survey when permission or ask question prompts are visible\n    if (hasActivePrompt) {\n      return false\n    }\n\n    // Force display for testing\n    if (\n      process.env.CLAUDE_FORCE_DISPLAY_SURVEY &&\n      !feedbackSurvey.timeLastShown\n    ) {\n      return true\n    }\n\n    if (!isModelAllowed) {\n      return false\n    }\n\n    if (isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_FEEDBACK_SURVEY)) {\n      return false\n    }\n\n    if (isFeedbackSurveyDisabled()) {\n      return false\n    }\n\n    // Check if product feedback is allowed by org policy\n    if (!isPolicyAllowed('allow_product_feedback')) {\n      return false\n    }\n\n    // Check session-local pacing\n    if (feedbackSurvey.timeLastShown) {\n      // Check time elapsed since last appearance in this session\n      const timeSinceLastShown = Date.now() - feedbackSurvey.timeLastShown\n      if (timeSinceLastShown < config.minTimeBetweenFeedbackMs) {\n        return false\n      }\n      // Check user turn requirement for subsequent appearances\n      if (\n        feedbackSurvey.submitCountAtLastAppearance !== null &&\n        submitCount <\n          feedbackSurvey.submitCountAtLastAppearance +\n            config.minUserTurnsBetweenFeedback\n      ) {\n        return false\n      }\n    } else {\n      // First appearance in this session\n      const timeSinceSessionStart = Date.now() - sessionStartTime.current\n      if (timeSinceSessionStart < config.minTimeBeforeFeedbackMs) {\n        return false\n      }\n      if (\n        submitCount <\n        submitCountAtSessionStart.current + config.minUserTurnsBeforeFeedback\n      ) {\n        return false\n      }\n    }\n\n    // Probability check: roll once per eligibility window to avoid re-rolling\n    // on every useMemo re-evaluation (which would make triggering near-certain).\n    if (lastEligibleSubmitCountRef.current !== submitCount) {\n      lastEligibleSubmitCountRef.current = submitCount\n      probabilityPassedRef.current =\n        Math.random() <= (settingsRate ?? config.probability)\n    }\n    if (!probabilityPassedRef.current) {\n      return false\n    }\n\n    // Check global pacing (across all sessions)\n    // Leave this till last because it reads from the filesystem which is expensive.\n    const globalFeedbackState = getGlobalConfig().feedbackSurveyState\n    if (globalFeedbackState?.lastShownTime) {\n      const timeSinceGlobalLastShown =\n        Date.now() - globalFeedbackState.lastShownTime\n      if (timeSinceGlobalLastShown < config.minTimeBetweenGlobalFeedbackMs) {\n        return false\n      }\n    }\n\n    return true\n  }, [\n    state,\n    isLoading,\n    hasActivePrompt,\n    isModelAllowed,\n    feedbackSurvey.timeLastShown,\n    feedbackSurvey.submitCountAtLastAppearance,\n    submitCount,\n    config.minTimeBetweenFeedbackMs,\n    config.minTimeBetweenGlobalFeedbackMs,\n    config.minUserTurnsBetweenFeedback,\n    config.minTimeBeforeFeedbackMs,\n    config.minUserTurnsBeforeFeedback,\n    config.probability,\n    settingsRate,\n  ])\n\n  useEffect(() => {\n    if (shouldOpen) {\n      open()\n    }\n  }, [shouldOpen, open])\n\n  return { state, lastResponse, handleSelect, handleTranscriptSelect }\n}\n"],"mappings":"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","ignoreList":[]} diff --git a/src/components/FeedbackSurvey/useMemorySurvey.tsx b/src/components/FeedbackSurvey/useMemorySurvey.tsx index e7d9b18..0440a1b 100644 --- a/src/components/FeedbackSurvey/useMemorySurvey.tsx +++ b/src/components/FeedbackSurvey/useMemorySurvey.tsx @@ -10,7 +10,6 @@ import { getGlobalConfig, saveGlobalConfig } from '../../utils/config.js'; import { isEnvTruthy } from '../../utils/envUtils.js'; import { isAutoManagedMemoryFile } from '../../utils/memoryFileDetection.js'; import { extractTextContent, getLastAssistantMessage } from '../../utils/messages.js'; -import { logOTelEvent } from '../../utils/telemetry/events.js'; import { submitTranscriptShare } from './submitTranscriptShare.js'; import type { TranscriptShareResponse } from './TranscriptSharePrompt.js'; import { useSurveyState } from './useSurveyState.js'; @@ -67,11 +66,6 @@ export function useMemorySurvey(messages: Message[], isLoading: boolean, hasActi event_type: 'appeared' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, appearance_id: appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS }); - void logOTelEvent('feedback_survey', { - event_type: 'appeared', - appearance_id: appearanceId, - survey_type: 'memory' - }); }, []); const onSelect = useCallback((appearanceId_0: string, selected: FeedbackSurveyResponse) => { logEvent(MEMORY_SURVEY_EVENT, { @@ -79,12 +73,6 @@ export function useMemorySurvey(messages: Message[], isLoading: boolean, hasActi appearance_id: appearanceId_0 as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, response: selected as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS }); - void logOTelEvent('feedback_survey', { - event_type: 'responded', - appearance_id: appearanceId_0, - response: selected, - survey_type: 'memory' - }); }, []); const shouldShowTranscriptPrompt = useCallback((selected_0: FeedbackSurveyResponse) => { if ("external" !== 'ant') { @@ -107,11 +95,6 @@ export function useMemorySurvey(messages: Message[], isLoading: boolean, hasActi appearance_id: appearanceId_1 as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, trigger: TRANSCRIPT_SHARE_TRIGGER as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS }); - void logOTelEvent('feedback_survey', { - event_type: 'transcript_prompt_appeared', - appearance_id: appearanceId_1, - survey_type: 'memory' - }); }, []); const onTranscriptSelect = useCallback(async (appearanceId_2: string, selected_1: TranscriptShareResponse): Promise => { logEvent(MEMORY_SURVEY_EVENT, { @@ -210,4 +193,4 @@ export function useMemorySurvey(messages: Message[], isLoading: boolean, hasActi handleTranscriptSelect }; } -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["useCallback","useEffect","useMemo","useRef","isFeedbackSurveyDisabled","getFeatureValue_CACHED_MAY_BE_STALE","AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS","logEvent","isAutoMemoryEnabled","isPolicyAllowed","FILE_READ_TOOL_NAME","Message","getGlobalConfig","saveGlobalConfig","isEnvTruthy","isAutoManagedMemoryFile","extractTextContent","getLastAssistantMessage","logOTelEvent","submitTranscriptShare","TranscriptShareResponse","useSurveyState","FeedbackSurveyResponse","HIDE_THANKS_AFTER_MS","MEMORY_SURVEY_GATE","MEMORY_SURVEY_EVENT","SURVEY_PROBABILITY","TRANSCRIPT_SHARE_TRIGGER","MEMORY_WORD_RE","hasMemoryFileRead","messages","message","type","content","Array","isArray","block","name","input","file_path","useMemorySurvey","isLoading","hasActivePrompt","enabled","state","lastResponse","handleSelect","selected","handleTranscriptSelect","seenAssistantUuids","Set","memoryReadSeen","messagesRef","current","onOpen","appearanceId","event_type","appearance_id","survey_type","onSelect","response","shouldShowTranscriptPrompt","transcriptShareDismissed","onTranscriptPromptShown","trigger","onTranscriptSelect","Promise","result","success","open","hideThanksAfterMs","lastAssistant","length","clear","process","env","CLAUDE_CODE_DISABLE_FEEDBACK_SURVEY","has","uuid","text","test","add","Math","random"],"sources":["useMemorySurvey.tsx"],"sourcesContent":["import { useCallback, useEffect, useMemo, useRef } from 'react'\nimport { isFeedbackSurveyDisabled } from 'src/services/analytics/config.js'\nimport { getFeatureValue_CACHED_MAY_BE_STALE } from 'src/services/analytics/growthbook.js'\nimport {\n  type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n  logEvent,\n} from 'src/services/analytics/index.js'\nimport { isAutoMemoryEnabled } from '../../memdir/paths.js'\nimport { isPolicyAllowed } from '../../services/policyLimits/index.js'\nimport { FILE_READ_TOOL_NAME } from '../../tools/FileReadTool/prompt.js'\nimport type { Message } from '../../types/message.js'\nimport { getGlobalConfig, saveGlobalConfig } from '../../utils/config.js'\nimport { isEnvTruthy } from '../../utils/envUtils.js'\nimport { isAutoManagedMemoryFile } from '../../utils/memoryFileDetection.js'\nimport {\n  extractTextContent,\n  getLastAssistantMessage,\n} from '../../utils/messages.js'\nimport { logOTelEvent } from '../../utils/telemetry/events.js'\nimport { submitTranscriptShare } from './submitTranscriptShare.js'\nimport type { TranscriptShareResponse } from './TranscriptSharePrompt.js'\nimport { useSurveyState } from './useSurveyState.js'\nimport type { FeedbackSurveyResponse } from './utils.js'\n\nconst HIDE_THANKS_AFTER_MS = 3000\nconst MEMORY_SURVEY_GATE = 'tengu_dunwich_bell'\nconst MEMORY_SURVEY_EVENT = 'tengu_memory_survey_event'\nconst SURVEY_PROBABILITY = 0.2\nconst TRANSCRIPT_SHARE_TRIGGER = 'memory_survey'\n\nconst MEMORY_WORD_RE = /\\bmemor(?:y|ies)\\b/i\n\nfunction hasMemoryFileRead(messages: Message[]): boolean {\n  for (const message of messages) {\n    if (message.type !== 'assistant') {\n      continue\n    }\n    const content = message.message.content\n    if (!Array.isArray(content)) {\n      continue\n    }\n    for (const block of content) {\n      if (block.type !== 'tool_use' || block.name !== FILE_READ_TOOL_NAME) {\n        continue\n      }\n      const input = block.input as { file_path?: unknown }\n      if (\n        typeof input.file_path === 'string' &&\n        isAutoManagedMemoryFile(input.file_path)\n      ) {\n        return true\n      }\n    }\n  }\n  return false\n}\n\nexport function useMemorySurvey(\n  messages: Message[],\n  isLoading: boolean,\n  hasActivePrompt = false,\n  { enabled = true }: { enabled?: boolean } = {},\n): {\n  state:\n    | 'closed'\n    | 'open'\n    | 'thanks'\n    | 'transcript_prompt'\n    | 'submitting'\n    | 'submitted'\n  lastResponse: FeedbackSurveyResponse | null\n  handleSelect: (selected: FeedbackSurveyResponse) => void\n  handleTranscriptSelect: (selected: TranscriptShareResponse) => void\n} {\n  // Track assistant message UUIDs that were already evaluated so we don't\n  // re-roll probability on re-renders or re-scan messages for the same turn.\n  const seenAssistantUuids = useRef<Set<string>>(new Set())\n  // Once a memory file read is observed it stays true for the session —\n  // skip the O(n) scan on subsequent turns.\n  const memoryReadSeen = useRef(false)\n  const messagesRef = useRef(messages)\n  messagesRef.current = messages\n\n  const onOpen = useCallback((appearanceId: string) => {\n    logEvent(MEMORY_SURVEY_EVENT, {\n      event_type:\n        'appeared' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      appearance_id:\n        appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n    })\n    void logOTelEvent('feedback_survey', {\n      event_type: 'appeared',\n      appearance_id: appearanceId,\n      survey_type: 'memory',\n    })\n  }, [])\n\n  const onSelect = useCallback(\n    (appearanceId: string, selected: FeedbackSurveyResponse) => {\n      logEvent(MEMORY_SURVEY_EVENT, {\n        event_type:\n          'responded' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        appearance_id:\n          appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        response:\n          selected as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      })\n      void logOTelEvent('feedback_survey', {\n        event_type: 'responded',\n        appearance_id: appearanceId,\n        response: selected,\n        survey_type: 'memory',\n      })\n    },\n    [],\n  )\n\n  const shouldShowTranscriptPrompt = useCallback(\n    (selected: FeedbackSurveyResponse) => {\n      if (\"external\" !== 'ant') {\n        return false\n      }\n      if (selected !== 'bad' && selected !== 'good') {\n        return false\n      }\n      if (getGlobalConfig().transcriptShareDismissed) {\n        return false\n      }\n      if (!isPolicyAllowed('allow_product_feedback')) {\n        return false\n      }\n      return true\n    },\n    [],\n  )\n\n  const onTranscriptPromptShown = useCallback((appearanceId: string) => {\n    logEvent(MEMORY_SURVEY_EVENT, {\n      event_type:\n        'transcript_prompt_appeared' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      appearance_id:\n        appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      trigger:\n        TRANSCRIPT_SHARE_TRIGGER as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n    })\n    void logOTelEvent('feedback_survey', {\n      event_type: 'transcript_prompt_appeared',\n      appearance_id: appearanceId,\n      survey_type: 'memory',\n    })\n  }, [])\n\n  const onTranscriptSelect = useCallback(\n    async (\n      appearanceId: string,\n      selected: TranscriptShareResponse,\n    ): Promise<boolean> => {\n      logEvent(MEMORY_SURVEY_EVENT, {\n        event_type:\n          `transcript_share_${selected}` as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        appearance_id:\n          appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        trigger:\n          TRANSCRIPT_SHARE_TRIGGER as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      })\n\n      if (selected === 'dont_ask_again') {\n        saveGlobalConfig(current => ({\n          ...current,\n          transcriptShareDismissed: true,\n        }))\n      }\n\n      if (selected === 'yes') {\n        const result = await submitTranscriptShare(\n          messagesRef.current,\n          TRANSCRIPT_SHARE_TRIGGER,\n          appearanceId,\n        )\n        logEvent(MEMORY_SURVEY_EVENT, {\n          event_type: (result.success\n            ? 'transcript_share_submitted'\n            : 'transcript_share_failed') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n          appearance_id:\n            appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n          trigger:\n            TRANSCRIPT_SHARE_TRIGGER as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        })\n        return result.success\n      }\n\n      return false\n    },\n    [],\n  )\n\n  const { state, lastResponse, open, handleSelect, handleTranscriptSelect } =\n    useSurveyState({\n      hideThanksAfterMs: HIDE_THANKS_AFTER_MS,\n      onOpen,\n      onSelect,\n      shouldShowTranscriptPrompt,\n      onTranscriptPromptShown,\n      onTranscriptSelect,\n    })\n\n  const lastAssistant = useMemo(\n    () => getLastAssistantMessage(messages),\n    [messages],\n  )\n\n  useEffect(() => {\n    if (!enabled) return\n\n    // /clear resets messages but REPL stays mounted — reset refs so a memory\n    // read from the previous conversation doesn't leak into the new one.\n    if (messages.length === 0) {\n      memoryReadSeen.current = false\n      seenAssistantUuids.current.clear()\n      return\n    }\n\n    if (state !== 'closed' || isLoading || hasActivePrompt) {\n      return\n    }\n\n    // 3P default: survey off (no GrowthBook on Bedrock/Vertex/Foundry).\n    if (!getFeatureValue_CACHED_MAY_BE_STALE(MEMORY_SURVEY_GATE, false)) {\n      return\n    }\n\n    if (!isAutoMemoryEnabled()) {\n      return\n    }\n\n    if (isFeedbackSurveyDisabled()) {\n      return\n    }\n\n    if (!isPolicyAllowed('allow_product_feedback')) {\n      return\n    }\n\n    if (isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_FEEDBACK_SURVEY)) {\n      return\n    }\n\n    if (!lastAssistant || seenAssistantUuids.current.has(lastAssistant.uuid)) {\n      return\n    }\n\n    const text = extractTextContent(lastAssistant.message.content, ' ')\n    if (!MEMORY_WORD_RE.test(text)) {\n      return\n    }\n\n    // Mark as evaluated before the memory-read scan so a turn that mentions\n    // \"memory\" but has no memory read doesn't trigger repeated O(n) scans\n    // on subsequent renders with the same last assistant message.\n    seenAssistantUuids.current.add(lastAssistant.uuid)\n\n    if (!memoryReadSeen.current) {\n      memoryReadSeen.current = hasMemoryFileRead(messages)\n    }\n    if (!memoryReadSeen.current) {\n      return\n    }\n\n    if (Math.random() < SURVEY_PROBABILITY) {\n      open()\n    }\n  }, [\n    enabled,\n    state,\n    isLoading,\n    hasActivePrompt,\n    lastAssistant,\n    messages,\n    open,\n  ])\n\n  return { state, lastResponse, handleSelect, handleTranscriptSelect }\n}\n"],"mappings":"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","ignoreList":[]} \ No newline at end of file +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["useCallback","useEffect","useMemo","useRef","isFeedbackSurveyDisabled","getFeatureValue_CACHED_MAY_BE_STALE","AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS","logEvent","isAutoMemoryEnabled","isPolicyAllowed","FILE_READ_TOOL_NAME","Message","getGlobalConfig","saveGlobalConfig","isEnvTruthy","isAutoManagedMemoryFile","extractTextContent","getLastAssistantMessage","logOTelEvent","submitTranscriptShare","TranscriptShareResponse","useSurveyState","FeedbackSurveyResponse","HIDE_THANKS_AFTER_MS","MEMORY_SURVEY_GATE","MEMORY_SURVEY_EVENT","SURVEY_PROBABILITY","TRANSCRIPT_SHARE_TRIGGER","MEMORY_WORD_RE","hasMemoryFileRead","messages","message","type","content","Array","isArray","block","name","input","file_path","useMemorySurvey","isLoading","hasActivePrompt","enabled","state","lastResponse","handleSelect","selected","handleTranscriptSelect","seenAssistantUuids","Set","memoryReadSeen","messagesRef","current","onOpen","appearanceId","event_type","appearance_id","survey_type","onSelect","response","shouldShowTranscriptPrompt","transcriptShareDismissed","onTranscriptPromptShown","trigger","onTranscriptSelect","Promise","result","success","open","hideThanksAfterMs","lastAssistant","length","clear","process","env","CLAUDE_CODE_DISABLE_FEEDBACK_SURVEY","has","uuid","text","test","add","Math","random"],"sources":["useMemorySurvey.tsx"],"sourcesContent":["import { useCallback, useEffect, useMemo, useRef } from 'react'\nimport { isFeedbackSurveyDisabled } from 'src/services/analytics/config.js'\nimport { getFeatureValue_CACHED_MAY_BE_STALE } from 'src/services/analytics/growthbook.js'\nimport {\n  type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n  logEvent,\n} from 'src/services/analytics/index.js'\nimport { isAutoMemoryEnabled } from '../../memdir/paths.js'\nimport { isPolicyAllowed } from '../../services/policyLimits/index.js'\nimport { FILE_READ_TOOL_NAME } from '../../tools/FileReadTool/prompt.js'\nimport type { Message } from '../../types/message.js'\nimport { getGlobalConfig, saveGlobalConfig } from '../../utils/config.js'\nimport { isEnvTruthy } from '../../utils/envUtils.js'\nimport { isAutoManagedMemoryFile } from '../../utils/memoryFileDetection.js'\nimport {\n  extractTextContent,\n  getLastAssistantMessage,\n} from '../../utils/messages.js'\nimport { logOTelEvent } from '../../utils/telemetry/events.js'\nimport { submitTranscriptShare } from './submitTranscriptShare.js'\nimport type { TranscriptShareResponse } from './TranscriptSharePrompt.js'\nimport { useSurveyState } from './useSurveyState.js'\nimport type { FeedbackSurveyResponse } from './utils.js'\n\nconst HIDE_THANKS_AFTER_MS = 3000\nconst MEMORY_SURVEY_GATE = 'tengu_dunwich_bell'\nconst MEMORY_SURVEY_EVENT = 'tengu_memory_survey_event'\nconst SURVEY_PROBABILITY = 0.2\nconst TRANSCRIPT_SHARE_TRIGGER = 'memory_survey'\n\nconst MEMORY_WORD_RE = /\\bmemor(?:y|ies)\\b/i\n\nfunction hasMemoryFileRead(messages: Message[]): boolean {\n  for (const message of messages) {\n    if (message.type !== 'assistant') {\n      continue\n    }\n    const content = message.message.content\n    if (!Array.isArray(content)) {\n      continue\n    }\n    for (const block of content) {\n      if (block.type !== 'tool_use' || block.name !== FILE_READ_TOOL_NAME) {\n        continue\n      }\n      const input = block.input as { file_path?: unknown }\n      if (\n        typeof input.file_path === 'string' &&\n        isAutoManagedMemoryFile(input.file_path)\n      ) {\n        return true\n      }\n    }\n  }\n  return false\n}\n\nexport function useMemorySurvey(\n  messages: Message[],\n  isLoading: boolean,\n  hasActivePrompt = false,\n  { enabled = true }: { enabled?: boolean } = {},\n): {\n  state:\n    | 'closed'\n    | 'open'\n    | 'thanks'\n    | 'transcript_prompt'\n    | 'submitting'\n    | 'submitted'\n  lastResponse: FeedbackSurveyResponse | null\n  handleSelect: (selected: FeedbackSurveyResponse) => void\n  handleTranscriptSelect: (selected: TranscriptShareResponse) => void\n} {\n  // Track assistant message UUIDs that were already evaluated so we don't\n  // re-roll probability on re-renders or re-scan messages for the same turn.\n  const seenAssistantUuids = useRef<Set<string>>(new Set())\n  // Once a memory file read is observed it stays true for the session —\n  // skip the O(n) scan on subsequent turns.\n  const memoryReadSeen = useRef(false)\n  const messagesRef = useRef(messages)\n  messagesRef.current = messages\n\n  const onOpen = useCallback((appearanceId: string) => {\n    logEvent(MEMORY_SURVEY_EVENT, {\n      event_type:\n        'appeared' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      appearance_id:\n        appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n    })\n    void logOTelEvent('feedback_survey', {\n      event_type: 'appeared',\n      appearance_id: appearanceId,\n      survey_type: 'memory',\n    })\n  }, [])\n\n  const onSelect = useCallback(\n    (appearanceId: string, selected: FeedbackSurveyResponse) => {\n      logEvent(MEMORY_SURVEY_EVENT, {\n        event_type:\n          'responded' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        appearance_id:\n          appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        response:\n          selected as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      })\n      void logOTelEvent('feedback_survey', {\n        event_type: 'responded',\n        appearance_id: appearanceId,\n        response: selected,\n        survey_type: 'memory',\n      })\n    },\n    [],\n  )\n\n  const shouldShowTranscriptPrompt = useCallback(\n    (selected: FeedbackSurveyResponse) => {\n      if (\"external\" !== 'ant') {\n        return false\n      }\n      if (selected !== 'bad' && selected !== 'good') {\n        return false\n      }\n      if (getGlobalConfig().transcriptShareDismissed) {\n        return false\n      }\n      if (!isPolicyAllowed('allow_product_feedback')) {\n        return false\n      }\n      return true\n    },\n    [],\n  )\n\n  const onTranscriptPromptShown = useCallback((appearanceId: string) => {\n    logEvent(MEMORY_SURVEY_EVENT, {\n      event_type:\n        'transcript_prompt_appeared' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      appearance_id:\n        appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      trigger:\n        TRANSCRIPT_SHARE_TRIGGER as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n    })\n    void logOTelEvent('feedback_survey', {\n      event_type: 'transcript_prompt_appeared',\n      appearance_id: appearanceId,\n      survey_type: 'memory',\n    })\n  }, [])\n\n  const onTranscriptSelect = useCallback(\n    async (\n      appearanceId: string,\n      selected: TranscriptShareResponse,\n    ): Promise<boolean> => {\n      logEvent(MEMORY_SURVEY_EVENT, {\n        event_type:\n          `transcript_share_${selected}` as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        appearance_id:\n          appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        trigger:\n          TRANSCRIPT_SHARE_TRIGGER as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      })\n\n      if (selected === 'dont_ask_again') {\n        saveGlobalConfig(current => ({\n          ...current,\n          transcriptShareDismissed: true,\n        }))\n      }\n\n      if (selected === 'yes') {\n        const result = await submitTranscriptShare(\n          messagesRef.current,\n          TRANSCRIPT_SHARE_TRIGGER,\n          appearanceId,\n        )\n        logEvent(MEMORY_SURVEY_EVENT, {\n          event_type: (result.success\n            ? 'transcript_share_submitted'\n            : 'transcript_share_failed') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n          appearance_id:\n            appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n          trigger:\n            TRANSCRIPT_SHARE_TRIGGER as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        })\n        return result.success\n      }\n\n      return false\n    },\n    [],\n  )\n\n  const { state, lastResponse, open, handleSelect, handleTranscriptSelect } =\n    useSurveyState({\n      hideThanksAfterMs: HIDE_THANKS_AFTER_MS,\n      onOpen,\n      onSelect,\n      shouldShowTranscriptPrompt,\n      onTranscriptPromptShown,\n      onTranscriptSelect,\n    })\n\n  const lastAssistant = useMemo(\n    () => getLastAssistantMessage(messages),\n    [messages],\n  )\n\n  useEffect(() => {\n    if (!enabled) return\n\n    // /clear resets messages but REPL stays mounted — reset refs so a memory\n    // read from the previous conversation doesn't leak into the new one.\n    if (messages.length === 0) {\n      memoryReadSeen.current = false\n      seenAssistantUuids.current.clear()\n      return\n    }\n\n    if (state !== 'closed' || isLoading || hasActivePrompt) {\n      return\n    }\n\n    // 3P default: survey off (no GrowthBook on Bedrock/Vertex/Foundry).\n    if (!getFeatureValue_CACHED_MAY_BE_STALE(MEMORY_SURVEY_GATE, false)) {\n      return\n    }\n\n    if (!isAutoMemoryEnabled()) {\n      return\n    }\n\n    if (isFeedbackSurveyDisabled()) {\n      return\n    }\n\n    if (!isPolicyAllowed('allow_product_feedback')) {\n      return\n    }\n\n    if (isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_FEEDBACK_SURVEY)) {\n      return\n    }\n\n    if (!lastAssistant || seenAssistantUuids.current.has(lastAssistant.uuid)) {\n      return\n    }\n\n    const text = extractTextContent(lastAssistant.message.content, ' ')\n    if (!MEMORY_WORD_RE.test(text)) {\n      return\n    }\n\n    // Mark as evaluated before the memory-read scan so a turn that mentions\n    // \"memory\" but has no memory read doesn't trigger repeated O(n) scans\n    // on subsequent renders with the same last assistant message.\n    seenAssistantUuids.current.add(lastAssistant.uuid)\n\n    if (!memoryReadSeen.current) {\n      memoryReadSeen.current = hasMemoryFileRead(messages)\n    }\n    if (!memoryReadSeen.current) {\n      return\n    }\n\n    if (Math.random() < SURVEY_PROBABILITY) {\n      open()\n    }\n  }, [\n    enabled,\n    state,\n    isLoading,\n    hasActivePrompt,\n    lastAssistant,\n    messages,\n    open,\n  ])\n\n  return { state, lastResponse, handleSelect, handleTranscriptSelect }\n}\n"],"mappings":"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","ignoreList":[]} diff --git a/src/components/FeedbackSurvey/usePostCompactSurvey.tsx b/src/components/FeedbackSurvey/usePostCompactSurvey.tsx index b33281a..ed8a10d 100644 --- a/src/components/FeedbackSurvey/usePostCompactSurvey.tsx +++ b/src/components/FeedbackSurvey/usePostCompactSurvey.tsx @@ -7,7 +7,6 @@ import { shouldUseSessionMemoryCompaction } from '../../services/compact/session import type { Message } from '../../types/message.js'; import { isEnvTruthy } from '../../utils/envUtils.js'; import { isCompactBoundaryMessage } from '../../utils/messages.js'; -import { logOTelEvent } from '../../utils/telemetry/events.js'; import { useSurveyState } from './useSurveyState.js'; import type { FeedbackSurveyResponse } from './utils.js'; const HIDE_THANKS_AFTER_MS = 3000; @@ -183,12 +182,6 @@ function _temp2(appearanceId_0, selected) { response: selected as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, session_memory_compaction_enabled: smCompactionEnabled_0 as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS }); - logOTelEvent("feedback_survey", { - event_type: "responded", - appearance_id: appearanceId_0, - response: selected, - survey_type: "post_compact" - }); } function _temp(appearanceId) { const smCompactionEnabled = shouldUseSessionMemoryCompaction(); @@ -197,10 +190,5 @@ function _temp(appearanceId) { appearance_id: appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, session_memory_compaction_enabled: smCompactionEnabled as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS }); - logOTelEvent("feedback_survey", { - event_type: "appeared", - appearance_id: appearanceId, - survey_type: "post_compact" - }); } -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["useCallback","useEffect","useMemo","useRef","useState","isFeedbackSurveyDisabled","checkStatsigFeatureGate_CACHED_MAY_BE_STALE","AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS","logEvent","shouldUseSessionMemoryCompaction","Message","isEnvTruthy","isCompactBoundaryMessage","logOTelEvent","useSurveyState","FeedbackSurveyResponse","HIDE_THANKS_AFTER_MS","POST_COMPACT_SURVEY_GATE","SURVEY_PROBABILITY","hasMessageAfterBoundary","messages","boundaryUuid","boundaryIndex","findIndex","msg","uuid","i","length","type","usePostCompactSurvey","isLoading","t0","t1","$","_c","hasActivePrompt","undefined","t2","enabled","t3","gateEnabled","setGateEnabled","t4","Symbol","for","Set","seenCompactBoundaries","pendingCompactBoundaryUuid","onOpen","_temp","onSelect","_temp2","t5","hideThanksAfterMs","state","lastResponse","open","handleSelect","t6","t7","t8","filter","_temp3","map","_temp4","currentCompactBoundaries","t10","t9","process","env","CLAUDE_CODE_DISABLE_FEEDBACK_SURVEY","current","Math","random","newBoundaries","Array","from","has","t11","msg_0","appearanceId_0","selected","smCompactionEnabled_0","event_type","appearance_id","appearanceId","response","session_memory_compaction_enabled","smCompactionEnabled","survey_type"],"sources":["usePostCompactSurvey.tsx"],"sourcesContent":["import { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport { isFeedbackSurveyDisabled } from 'src/services/analytics/config.js'\nimport { checkStatsigFeatureGate_CACHED_MAY_BE_STALE } from 'src/services/analytics/growthbook.js'\nimport {\n  type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n  logEvent,\n} from 'src/services/analytics/index.js'\nimport { shouldUseSessionMemoryCompaction } from '../../services/compact/sessionMemoryCompact.js'\nimport type { Message } from '../../types/message.js'\nimport { isEnvTruthy } from '../../utils/envUtils.js'\nimport { isCompactBoundaryMessage } from '../../utils/messages.js'\nimport { logOTelEvent } from '../../utils/telemetry/events.js'\nimport { useSurveyState } from './useSurveyState.js'\nimport type { FeedbackSurveyResponse } from './utils.js'\n\nconst HIDE_THANKS_AFTER_MS = 3000\nconst POST_COMPACT_SURVEY_GATE = 'tengu_post_compact_survey'\nconst SURVEY_PROBABILITY = 0.2 // Show survey 20% of the time after compaction\n\nfunction hasMessageAfterBoundary(\n  messages: Message[],\n  boundaryUuid: string,\n): boolean {\n  const boundaryIndex = messages.findIndex(msg => msg.uuid === boundaryUuid)\n  if (boundaryIndex === -1) {\n    return false\n  }\n\n  // Check if there's a user or assistant message after the boundary\n  for (let i = boundaryIndex + 1; i < messages.length; i++) {\n    const msg = messages[i]\n    if (msg && (msg.type === 'user' || msg.type === 'assistant')) {\n      return true\n    }\n  }\n  return false\n}\n\nexport function usePostCompactSurvey(\n  messages: Message[],\n  isLoading: boolean,\n  hasActivePrompt = false,\n  { enabled = true }: { enabled?: boolean } = {},\n): {\n  state:\n    | 'closed'\n    | 'open'\n    | 'thanks'\n    | 'transcript_prompt'\n    | 'submitting'\n    | 'submitted'\n  lastResponse: FeedbackSurveyResponse | null\n  handleSelect: (selected: FeedbackSurveyResponse) => void\n} {\n  const [gateEnabled, setGateEnabled] = useState<boolean | null>(null)\n  const seenCompactBoundaries = useRef<Set<string>>(new Set())\n  // Track the compact boundary we're waiting on (to show survey after next message)\n  const pendingCompactBoundaryUuid = useRef<string | null>(null)\n\n  const onOpen = useCallback((appearanceId: string) => {\n    const smCompactionEnabled = shouldUseSessionMemoryCompaction()\n    logEvent('tengu_post_compact_survey_event', {\n      event_type:\n        'appeared' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      appearance_id:\n        appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      session_memory_compaction_enabled:\n        smCompactionEnabled as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n    })\n    void logOTelEvent('feedback_survey', {\n      event_type: 'appeared',\n      appearance_id: appearanceId,\n      survey_type: 'post_compact',\n    })\n  }, [])\n\n  const onSelect = useCallback(\n    (appearanceId: string, selected: FeedbackSurveyResponse) => {\n      const smCompactionEnabled = shouldUseSessionMemoryCompaction()\n      logEvent('tengu_post_compact_survey_event', {\n        event_type:\n          'responded' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        appearance_id:\n          appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        response:\n          selected as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        session_memory_compaction_enabled:\n          smCompactionEnabled as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      })\n      void logOTelEvent('feedback_survey', {\n        event_type: 'responded',\n        appearance_id: appearanceId,\n        response: selected,\n        survey_type: 'post_compact',\n      })\n    },\n    [],\n  )\n\n  const { state, lastResponse, open, handleSelect } = useSurveyState({\n    hideThanksAfterMs: HIDE_THANKS_AFTER_MS,\n    onOpen,\n    onSelect,\n  })\n\n  // Check the feature gate on mount\n  useEffect(() => {\n    if (!enabled) return\n    setGateEnabled(\n      checkStatsigFeatureGate_CACHED_MAY_BE_STALE(POST_COMPACT_SURVEY_GATE),\n    )\n  }, [enabled])\n\n  // Find compact boundary messages\n  const currentCompactBoundaries = useMemo(\n    () =>\n      new Set(\n        messages\n          .filter(msg => isCompactBoundaryMessage(msg))\n          .map(msg => msg.uuid),\n      ),\n    [messages],\n  )\n\n  // Detect new compact boundaries and defer showing survey until next message\n  useEffect(() => {\n    if (!enabled) return\n\n    // Don't process if already showing\n    if (state !== 'closed' || isLoading) {\n      return\n    }\n\n    // Don't show survey when permission or ask question prompts are visible\n    if (hasActivePrompt) {\n      return\n    }\n\n    // Check if the gate is enabled\n    if (gateEnabled !== true) {\n      return\n    }\n\n    if (isFeedbackSurveyDisabled()) {\n      return\n    }\n\n    // Check if survey is explicitly disabled\n    if (isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_FEEDBACK_SURVEY)) {\n      return\n    }\n\n    // First, check if we have a pending compact and a new message has arrived\n    if (pendingCompactBoundaryUuid.current !== null) {\n      if (\n        hasMessageAfterBoundary(messages, pendingCompactBoundaryUuid.current)\n      ) {\n        // A new message arrived after the compact - decide whether to show survey\n        pendingCompactBoundaryUuid.current = null\n\n        // Only show survey 20% of the time\n        if (Math.random() < SURVEY_PROBABILITY) {\n          open()\n        }\n        return\n      }\n    }\n\n    // Find new compact boundaries that we haven't seen yet\n    const newBoundaries = Array.from(currentCompactBoundaries).filter(\n      uuid => !seenCompactBoundaries.current.has(uuid),\n    )\n\n    if (newBoundaries.length > 0) {\n      // Mark these boundaries as seen\n      seenCompactBoundaries.current = new Set(currentCompactBoundaries)\n\n      // Don't show survey immediately - wait for next message\n      // Store the most recent new boundary UUID\n      pendingCompactBoundaryUuid.current =\n        newBoundaries[newBoundaries.length - 1]!\n    }\n  }, [\n    enabled,\n    currentCompactBoundaries,\n    state,\n    isLoading,\n    hasActivePrompt,\n    gateEnabled,\n    messages,\n    open,\n  ])\n\n  return { state, lastResponse, handleSelect }\n}\n"],"mappings":"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","ignoreList":[]} \ No newline at end of file +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["useCallback","useEffect","useMemo","useRef","useState","isFeedbackSurveyDisabled","checkStatsigFeatureGate_CACHED_MAY_BE_STALE","AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS","logEvent","shouldUseSessionMemoryCompaction","Message","isEnvTruthy","isCompactBoundaryMessage","logOTelEvent","useSurveyState","FeedbackSurveyResponse","HIDE_THANKS_AFTER_MS","POST_COMPACT_SURVEY_GATE","SURVEY_PROBABILITY","hasMessageAfterBoundary","messages","boundaryUuid","boundaryIndex","findIndex","msg","uuid","i","length","type","usePostCompactSurvey","isLoading","t0","t1","$","_c","hasActivePrompt","undefined","t2","enabled","t3","gateEnabled","setGateEnabled","t4","Symbol","for","Set","seenCompactBoundaries","pendingCompactBoundaryUuid","onOpen","_temp","onSelect","_temp2","t5","hideThanksAfterMs","state","lastResponse","open","handleSelect","t6","t7","t8","filter","_temp3","map","_temp4","currentCompactBoundaries","t10","t9","process","env","CLAUDE_CODE_DISABLE_FEEDBACK_SURVEY","current","Math","random","newBoundaries","Array","from","has","t11","msg_0","appearanceId_0","selected","smCompactionEnabled_0","event_type","appearance_id","appearanceId","response","session_memory_compaction_enabled","smCompactionEnabled","survey_type"],"sources":["usePostCompactSurvey.tsx"],"sourcesContent":["import { useCallback, useEffect, useMemo, useRef, useState } from 'react'\nimport { isFeedbackSurveyDisabled } from 'src/services/analytics/config.js'\nimport { checkStatsigFeatureGate_CACHED_MAY_BE_STALE } from 'src/services/analytics/growthbook.js'\nimport {\n  type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n  logEvent,\n} from 'src/services/analytics/index.js'\nimport { shouldUseSessionMemoryCompaction } from '../../services/compact/sessionMemoryCompact.js'\nimport type { Message } from '../../types/message.js'\nimport { isEnvTruthy } from '../../utils/envUtils.js'\nimport { isCompactBoundaryMessage } from '../../utils/messages.js'\nimport { logOTelEvent } from '../../utils/telemetry/events.js'\nimport { useSurveyState } from './useSurveyState.js'\nimport type { FeedbackSurveyResponse } from './utils.js'\n\nconst HIDE_THANKS_AFTER_MS = 3000\nconst POST_COMPACT_SURVEY_GATE = 'tengu_post_compact_survey'\nconst SURVEY_PROBABILITY = 0.2 // Show survey 20% of the time after compaction\n\nfunction hasMessageAfterBoundary(\n  messages: Message[],\n  boundaryUuid: string,\n): boolean {\n  const boundaryIndex = messages.findIndex(msg => msg.uuid === boundaryUuid)\n  if (boundaryIndex === -1) {\n    return false\n  }\n\n  // Check if there's a user or assistant message after the boundary\n  for (let i = boundaryIndex + 1; i < messages.length; i++) {\n    const msg = messages[i]\n    if (msg && (msg.type === 'user' || msg.type === 'assistant')) {\n      return true\n    }\n  }\n  return false\n}\n\nexport function usePostCompactSurvey(\n  messages: Message[],\n  isLoading: boolean,\n  hasActivePrompt = false,\n  { enabled = true }: { enabled?: boolean } = {},\n): {\n  state:\n    | 'closed'\n    | 'open'\n    | 'thanks'\n    | 'transcript_prompt'\n    | 'submitting'\n    | 'submitted'\n  lastResponse: FeedbackSurveyResponse | null\n  handleSelect: (selected: FeedbackSurveyResponse) => void\n} {\n  const [gateEnabled, setGateEnabled] = useState<boolean | null>(null)\n  const seenCompactBoundaries = useRef<Set<string>>(new Set())\n  // Track the compact boundary we're waiting on (to show survey after next message)\n  const pendingCompactBoundaryUuid = useRef<string | null>(null)\n\n  const onOpen = useCallback((appearanceId: string) => {\n    const smCompactionEnabled = shouldUseSessionMemoryCompaction()\n    logEvent('tengu_post_compact_survey_event', {\n      event_type:\n        'appeared' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      appearance_id:\n        appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      session_memory_compaction_enabled:\n        smCompactionEnabled as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n    })\n    void logOTelEvent('feedback_survey', {\n      event_type: 'appeared',\n      appearance_id: appearanceId,\n      survey_type: 'post_compact',\n    })\n  }, [])\n\n  const onSelect = useCallback(\n    (appearanceId: string, selected: FeedbackSurveyResponse) => {\n      const smCompactionEnabled = shouldUseSessionMemoryCompaction()\n      logEvent('tengu_post_compact_survey_event', {\n        event_type:\n          'responded' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        appearance_id:\n          appearanceId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        response:\n          selected as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        session_memory_compaction_enabled:\n          smCompactionEnabled as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      })\n      void logOTelEvent('feedback_survey', {\n        event_type: 'responded',\n        appearance_id: appearanceId,\n        response: selected,\n        survey_type: 'post_compact',\n      })\n    },\n    [],\n  )\n\n  const { state, lastResponse, open, handleSelect } = useSurveyState({\n    hideThanksAfterMs: HIDE_THANKS_AFTER_MS,\n    onOpen,\n    onSelect,\n  })\n\n  // Check the feature gate on mount\n  useEffect(() => {\n    if (!enabled) return\n    setGateEnabled(\n      checkStatsigFeatureGate_CACHED_MAY_BE_STALE(POST_COMPACT_SURVEY_GATE),\n    )\n  }, [enabled])\n\n  // Find compact boundary messages\n  const currentCompactBoundaries = useMemo(\n    () =>\n      new Set(\n        messages\n          .filter(msg => isCompactBoundaryMessage(msg))\n          .map(msg => msg.uuid),\n      ),\n    [messages],\n  )\n\n  // Detect new compact boundaries and defer showing survey until next message\n  useEffect(() => {\n    if (!enabled) return\n\n    // Don't process if already showing\n    if (state !== 'closed' || isLoading) {\n      return\n    }\n\n    // Don't show survey when permission or ask question prompts are visible\n    if (hasActivePrompt) {\n      return\n    }\n\n    // Check if the gate is enabled\n    if (gateEnabled !== true) {\n      return\n    }\n\n    if (isFeedbackSurveyDisabled()) {\n      return\n    }\n\n    // Check if survey is explicitly disabled\n    if (isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_FEEDBACK_SURVEY)) {\n      return\n    }\n\n    // First, check if we have a pending compact and a new message has arrived\n    if (pendingCompactBoundaryUuid.current !== null) {\n      if (\n        hasMessageAfterBoundary(messages, pendingCompactBoundaryUuid.current)\n      ) {\n        // A new message arrived after the compact - decide whether to show survey\n        pendingCompactBoundaryUuid.current = null\n\n        // Only show survey 20% of the time\n        if (Math.random() < SURVEY_PROBABILITY) {\n          open()\n        }\n        return\n      }\n    }\n\n    // Find new compact boundaries that we haven't seen yet\n    const newBoundaries = Array.from(currentCompactBoundaries).filter(\n      uuid => !seenCompactBoundaries.current.has(uuid),\n    )\n\n    if (newBoundaries.length > 0) {\n      // Mark these boundaries as seen\n      seenCompactBoundaries.current = new Set(currentCompactBoundaries)\n\n      // Don't show survey immediately - wait for next message\n      // Store the most recent new boundary UUID\n      pendingCompactBoundaryUuid.current =\n        newBoundaries[newBoundaries.length - 1]!\n    }\n  }, [\n    enabled,\n    currentCompactBoundaries,\n    state,\n    isLoading,\n    hasActivePrompt,\n    gateEnabled,\n    messages,\n    open,\n  ])\n\n  return { state, lastResponse, handleSelect }\n}\n"],"mappings":"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","ignoreList":[]} diff --git a/src/hooks/toolPermission/permissionLogging.ts b/src/hooks/toolPermission/permissionLogging.ts index fa0dc5a..cfdbf98 100644 --- a/src/hooks/toolPermission/permissionLogging.ts +++ b/src/hooks/toolPermission/permissionLogging.ts @@ -1,6 +1,6 @@ // Centralized analytics/telemetry logging for tool permission decisions. // All permission approve/reject events flow through logPermissionDecision(), -// which fans out to Statsig analytics, OTel telemetry, and code-edit metrics. +// which fans out to analytics compatibility calls and code-edit metrics. import { feature } from 'bun:bundle' import { type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, @@ -11,7 +11,6 @@ import { getCodeEditToolDecisionCounter } from '../../bootstrap/state.js' import type { Tool as ToolType, ToolUseContext } from '../../Tool.js' import { getLanguageName } from '../../utils/cliHighlight.js' import { SandboxManager } from '../../utils/sandbox/sandbox-adapter.js' -import { logOTelEvent } from '../../utils/telemetry/events.js' import type { PermissionApprovalSource, PermissionRejectionSource, @@ -227,11 +226,6 @@ function logPermissionDecision( timestamp: Date.now(), }) - void logOTelEvent('tool_decision', { - decision, - source: sourceString, - tool_name: sanitizeToolNameForAnalytics(tool.name), - }) } export { isCodeEditingTool, buildCodeEditToolAttributes, logPermissionDecision } diff --git a/src/services/api/logging.ts b/src/services/api/logging.ts index 2bcf651..efe52d6 100644 --- a/src/services/api/logging.ts +++ b/src/services/api/logging.ts @@ -22,7 +22,6 @@ import { logError } from 'src/utils/log.js' import { getAPIProviderForStatsig } from 'src/utils/model/providers.js' import type { PermissionMode } from 'src/utils/permissions/PermissionMode.js' import { jsonStringify } from 'src/utils/slowOperations.js' -import { logOTelEvent } from 'src/utils/telemetry/events.js' import { endLLMRequestSpan, isBetaTracingEnabled, @@ -364,16 +363,6 @@ export function logAPIError({ ...getAnthropicEnvMetadata(), }) - // Log API error event for OTLP - void logOTelEvent('api_error', { - model: model, - error: errStr, - status_code: String(status), - duration_ms: String(durationMs), - attempt: String(attempt), - speed: fastMode ? 'fast' : 'normal', - }) - // Pass the span to correctly match responses to requests when beta tracing is enabled endLLMRequestSpan(llmSpan, { success: false, @@ -714,18 +703,6 @@ export function logAPISuccessAndDuration({ previousRequestId, betas, }) - // Log API request event for OTLP - void logOTelEvent('api_request', { - model, - input_tokens: String(usage.input_tokens), - output_tokens: String(usage.output_tokens), - cache_read_tokens: String(usage.cache_read_input_tokens), - cache_creation_tokens: String(usage.cache_creation_input_tokens), - cost_usd: String(costUSD), - duration_ms: String(durationMs), - speed: fastMode ? 'fast' : 'normal', - }) - // Extract model output, thinking output, and tool call flag when beta tracing is enabled let modelOutput: string | undefined let thinkingOutput: string | undefined diff --git a/src/services/tools/toolExecution.ts b/src/services/tools/toolExecution.ts index 40ef38e..c8ac40f 100644 --- a/src/services/tools/toolExecution.ts +++ b/src/services/tools/toolExecution.ts @@ -87,7 +87,6 @@ import { } from '../../utils/sessionActivity.js' import { jsonStringify } from '../../utils/slowOperations.js' import { Stream } from '../../utils/stream.js' -import { logOTelEvent } from '../../utils/telemetry/events.js' import { addToolContentEvent, endToolBlockedOnUserSpan, @@ -945,28 +944,18 @@ async function checkPermissionsAndCallTool( ) } - // Emit tool_decision OTel event and code-edit counter if the interactive - // permission path didn't already log it (headless mode bypasses permission - // logging, so we need to emit both the generic event and the code-edit - // counter here) + // Increment the code-edit counter here when the interactive permission path + // did not already log a decision (headless mode bypasses permission logging). if ( permissionDecision.behavior !== 'ask' && !toolUseContext.toolDecisions?.has(toolUseID) ) { - const decision = - permissionDecision.behavior === 'allow' ? 'accept' : 'reject' - const source = decisionReasonToOTelSource( - permissionDecision.decisionReason, - permissionDecision.behavior, - ) - void logOTelEvent('tool_decision', { - decision, - source, - tool_name: sanitizeToolNameForAnalytics(tool.name), - }) - // Increment code-edit tool decision counter for headless mode if (isCodeEditingTool(tool.name)) { + const source = decisionReasonToOTelSource( + permissionDecision.decisionReason, + permissionDecision.behavior, + ) void buildCodeEditToolAttributes( tool, processedInput, @@ -1373,27 +1362,10 @@ async function checkPermissionsAndCallTool( } } - // Log tool result event for OTLP with tool parameters and decision context const mcpServerScope = isMcpTool(tool) ? getMcpServerScopeFromToolName(tool.name) : null - void logOTelEvent('tool_result', { - tool_name: sanitizeToolNameForAnalytics(tool.name), - success: 'true', - duration_ms: String(durationMs), - ...(Object.keys(toolParameters).length > 0 && { - tool_parameters: jsonStringify(toolParameters), - }), - ...(telemetryToolInput && { tool_input: telemetryToolInput }), - tool_result_size_bytes: String(toolResultSizeBytes), - ...(decisionInfo && { - decision_source: decisionInfo.source, - decision_type: decisionInfo.decision, - }), - ...(mcpServerScope && { mcp_server_scope: mcpServerScope }), - }) - // Run PostToolUse hooks let toolOutput = result.data const hookResults = [] @@ -1666,27 +1638,9 @@ async function checkPermissionsAndCallTool( mcpServerBaseUrl, ), }) - // Log tool result error event for OTLP with tool parameters and decision context const mcpServerScope = isMcpTool(tool) ? getMcpServerScopeFromToolName(tool.name) : null - - void logOTelEvent('tool_result', { - tool_name: sanitizeToolNameForAnalytics(tool.name), - use_id: toolUseID, - success: 'false', - duration_ms: String(durationMs), - error: errorMessage(error), - ...(Object.keys(toolParameters).length > 0 && { - tool_parameters: jsonStringify(toolParameters), - }), - ...(telemetryToolInput && { tool_input: telemetryToolInput }), - ...(decisionInfo && { - decision_source: decisionInfo.source, - decision_type: decisionInfo.decision, - }), - ...(mcpServerScope && { mcp_server_scope: mcpServerScope }), - }) } const content = formatError(error) diff --git a/src/types/generated/events_mono/growthbook/v1/growthbook_experiment_event.ts b/src/types/generated/events_mono/growthbook/v1/growthbook_experiment_event.ts deleted file mode 100644 index fbda511..0000000 --- a/src/types/generated/events_mono/growthbook/v1/growthbook_experiment_event.ts +++ /dev/null @@ -1,223 +0,0 @@ -// Code generated by protoc-gen-ts_proto. DO NOT EDIT. -// versions: -// protoc-gen-ts_proto v2.6.1 -// protoc unknown -// source: events_mono/growthbook/v1/growthbook_experiment_event.proto - -/* eslint-disable */ -import { Timestamp } from '../../../google/protobuf/timestamp.js' -import { PublicApiAuth } from '../../common/v1/auth.js' - -/** - * GrowthBook experiment assignment event - * This event tracks when a user is exposed to an experiment variant - * See: https://docs.growthbook.io/guide/bigquery - */ -export interface GrowthbookExperimentEvent { - /** Unique event identifier (for deduplication) */ - event_id?: string | undefined - /** When user was exposed to experiment (maps to GrowthBook's timestamp column) */ - timestamp?: Date | undefined - /** Experiment tracking key (maps to GrowthBook's experiment_id column) */ - experiment_id?: string | undefined - /** Variation index: 0=control, 1+=variants (maps to GrowthBook's variation_id column) */ - variation_id?: number | undefined - /** Environment where assignment occurred */ - environment?: string | undefined - /** User attributes at time of assignment */ - user_attributes?: string | undefined - /** Experiment metadata */ - experiment_metadata?: string | undefined - /** Device identifier for the client */ - device_id?: string | undefined - /** Authentication context automatically injected by the API */ - auth?: PublicApiAuth | undefined - /** Session identifier for tracking user sessions */ - session_id?: string | undefined - /** Anonymous identifier for unauthenticated users */ - anonymous_id?: string | undefined - /** Event metadata variables (automatically populated by internal-tools-common event_logging library) */ - event_metadata_vars?: string | undefined -} - -function createBaseGrowthbookExperimentEvent(): GrowthbookExperimentEvent { - return { - event_id: '', - timestamp: undefined, - experiment_id: '', - variation_id: 0, - environment: '', - user_attributes: '', - experiment_metadata: '', - device_id: '', - auth: undefined, - session_id: '', - anonymous_id: '', - event_metadata_vars: '', - } -} - -export const GrowthbookExperimentEvent: MessageFns = - { - fromJSON(object: any): GrowthbookExperimentEvent { - return { - event_id: isSet(object.event_id) - ? globalThis.String(object.event_id) - : '', - timestamp: isSet(object.timestamp) - ? fromJsonTimestamp(object.timestamp) - : undefined, - experiment_id: isSet(object.experiment_id) - ? globalThis.String(object.experiment_id) - : '', - variation_id: isSet(object.variation_id) - ? globalThis.Number(object.variation_id) - : 0, - environment: isSet(object.environment) - ? globalThis.String(object.environment) - : '', - user_attributes: isSet(object.user_attributes) - ? globalThis.String(object.user_attributes) - : '', - experiment_metadata: isSet(object.experiment_metadata) - ? globalThis.String(object.experiment_metadata) - : '', - device_id: isSet(object.device_id) - ? globalThis.String(object.device_id) - : '', - auth: isSet(object.auth) - ? PublicApiAuth.fromJSON(object.auth) - : undefined, - session_id: isSet(object.session_id) - ? globalThis.String(object.session_id) - : '', - anonymous_id: isSet(object.anonymous_id) - ? globalThis.String(object.anonymous_id) - : '', - event_metadata_vars: isSet(object.event_metadata_vars) - ? globalThis.String(object.event_metadata_vars) - : '', - } - }, - - toJSON(message: GrowthbookExperimentEvent): unknown { - const obj: any = {} - if (message.event_id !== undefined) { - obj.event_id = message.event_id - } - if (message.timestamp !== undefined) { - obj.timestamp = message.timestamp.toISOString() - } - if (message.experiment_id !== undefined) { - obj.experiment_id = message.experiment_id - } - if (message.variation_id !== undefined) { - obj.variation_id = Math.round(message.variation_id) - } - if (message.environment !== undefined) { - obj.environment = message.environment - } - if (message.user_attributes !== undefined) { - obj.user_attributes = message.user_attributes - } - if (message.experiment_metadata !== undefined) { - obj.experiment_metadata = message.experiment_metadata - } - if (message.device_id !== undefined) { - obj.device_id = message.device_id - } - if (message.auth !== undefined) { - obj.auth = PublicApiAuth.toJSON(message.auth) - } - if (message.session_id !== undefined) { - obj.session_id = message.session_id - } - if (message.anonymous_id !== undefined) { - obj.anonymous_id = message.anonymous_id - } - if (message.event_metadata_vars !== undefined) { - obj.event_metadata_vars = message.event_metadata_vars - } - return obj - }, - - create, I>>( - base?: I, - ): GrowthbookExperimentEvent { - return GrowthbookExperimentEvent.fromPartial(base ?? ({} as any)) - }, - fromPartial, I>>( - object: I, - ): GrowthbookExperimentEvent { - const message = createBaseGrowthbookExperimentEvent() - message.event_id = object.event_id ?? '' - message.timestamp = object.timestamp ?? undefined - message.experiment_id = object.experiment_id ?? '' - message.variation_id = object.variation_id ?? 0 - message.environment = object.environment ?? '' - message.user_attributes = object.user_attributes ?? '' - message.experiment_metadata = object.experiment_metadata ?? '' - message.device_id = object.device_id ?? '' - message.auth = - object.auth !== undefined && object.auth !== null - ? PublicApiAuth.fromPartial(object.auth) - : undefined - message.session_id = object.session_id ?? '' - message.anonymous_id = object.anonymous_id ?? '' - message.event_metadata_vars = object.event_metadata_vars ?? '' - return message - }, - } - -type Builtin = - | Date - | Function - | Uint8Array - | string - | number - | boolean - | undefined - -type DeepPartial = T extends Builtin - ? T - : T extends globalThis.Array - ? globalThis.Array> - : T extends ReadonlyArray - ? ReadonlyArray> - : T extends {} - ? { [K in keyof T]?: DeepPartial } - : Partial - -type KeysOfUnion = T extends T ? keyof T : never -type Exact = P extends Builtin - ? P - : P & { [K in keyof P]: Exact } & { - [K in Exclude>]: never - } - -function fromTimestamp(t: Timestamp): Date { - let millis = (t.seconds || 0) * 1_000 - millis += (t.nanos || 0) / 1_000_000 - return new globalThis.Date(millis) -} - -function fromJsonTimestamp(o: any): Date { - if (o instanceof globalThis.Date) { - return o - } else if (typeof o === 'string') { - return new globalThis.Date(o) - } else { - return fromTimestamp(Timestamp.fromJSON(o)) - } -} - -function isSet(value: any): boolean { - return value !== null && value !== undefined -} - -interface MessageFns { - fromJSON(object: any): T - toJSON(message: T): unknown - create, I>>(base?: I): T - fromPartial, I>>(object: I): T -} diff --git a/src/utils/hooks.ts b/src/utils/hooks.ts index b92537e..c6799eb 100644 --- a/src/utils/hooks.ts +++ b/src/utils/hooks.ts @@ -55,7 +55,6 @@ import { logEvent, type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, } from 'src/services/analytics/index.js' -import { logOTelEvent } from './telemetry/events.js' import { ALLOWED_OFFICIAL_MARKETPLACE_NAMES } from './plugins/schemas.js' import { startHookSpan, @@ -2071,18 +2070,6 @@ async function* executeHooks({ ? jsonStringify(getHookDefinitionsForTelemetry(matchingHooks)) : '[]' - // Log hook execution start to OTEL (only for beta tracing) - if (isBetaTracingEnabled()) { - void logOTelEvent('hook_execution_start', { - hook_event: hookEvent, - hook_name: hookName, - num_hooks: String(matchingHooks.length), - managed_only: String(shouldAllowManagedHooksOnly()), - hook_definitions: hookDefinitionsJson, - hook_source: shouldAllowManagedHooksOnly() ? 'policySettings' : 'merged', - }) - } - // Start hook span for beta tracing const hookSpan = startHookSpan( hookEvent, @@ -2943,25 +2930,6 @@ async function* executeHooks({ totalDurationMs, }) - // Log hook execution completion to OTEL (only for beta tracing) - if (isBetaTracingEnabled()) { - const hookDefinitionsComplete = - getHookDefinitionsForTelemetry(matchingHooks) - - void logOTelEvent('hook_execution_complete', { - hook_event: hookEvent, - hook_name: hookName, - num_hooks: String(matchingHooks.length), - num_success: String(outcomes.success), - num_blocking: String(outcomes.blocking), - num_non_blocking_error: String(outcomes.non_blocking_error), - num_cancelled: String(outcomes.cancelled), - managed_only: String(shouldAllowManagedHooksOnly()), - hook_definitions: jsonStringify(hookDefinitionsComplete), - hook_source: shouldAllowManagedHooksOnly() ? 'policySettings' : 'merged', - }) - } - // End hook span for beta tracing endHookSpan(hookSpan, { numSuccess: outcomes.success, diff --git a/src/utils/processUserInput/processSlashCommand.tsx b/src/utils/processUserInput/processSlashCommand.tsx index a0bf451..8529532 100644 --- a/src/utils/processUserInput/processSlashCommand.tsx +++ b/src/utils/processUserInput/processSlashCommand.tsx @@ -40,7 +40,6 @@ import { isRestrictedToPluginOnly, isSourceAdminTrusted } from '../settings/plug import { parseSlashCommand } from '../slashCommandParsing.js'; import { sleep } from '../sleep.js'; import { recordSkillUsage } from '../suggestions/skillUsageTracking.js'; -import { logOTelEvent, redactIfDisabled } from '../telemetry/events.js'; import { buildPluginCommandTelemetryFields } from '../telemetry/pluginTelemetry.js'; import { getAssistantMessageContentLength } from '../tokens.js'; import { createAgentId } from '../uuid.js'; @@ -362,12 +361,6 @@ export async function processSlashCommand(inputString: string, precedingInputBlo const promptId = randomUUID(); setPromptId(promptId); logEvent('tengu_input_prompt', {}); - // Log user prompt event for OTLP - void logOTelEvent('user_prompt', { - prompt_length: String(inputString.length), - prompt: redactIfDisabled(inputString), - 'prompt.id': promptId - }); return { messages: [createUserMessage({ content: prepareUserContent({ @@ -919,4 +912,4 @@ async function getMessagesForPromptSlashCommand(command: CommandBase & PromptCom command }; } -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["feature","ContentBlockParam","TextBlockParam","randomUUID","setPromptId","builtInCommandNames","Command","CommandBase","findCommand","getCommand","getCommandName","hasCommand","PromptCommand","NO_CONTENT_MESSAGE","SetToolJSXFn","ToolUseContext","AssistantMessage","AttachmentMessage","Message","NormalizedUserMessage","ProgressMessage","UserMessage","addInvokedSkill","getSessionId","COMMAND_MESSAGE_TAG","COMMAND_NAME_TAG","CanUseToolFn","AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS","AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED","logEvent","getDumpPromptsPath","buildPostCompactMessages","resetMicrocompactState","Progress","AgentProgress","runAgent","renderToolUseProgressMessage","CommandResultDisplay","createAbortController","getAgentContext","createAttachmentMessage","getAttachmentMessages","logForDebugging","isEnvTruthy","AbortError","MalformedCommandError","getDisplayPath","extractResultText","prepareForkedCommandContext","getFsImplementation","isFullscreenEnvEnabled","toArray","registerSkillHooks","logError","enqueuePendingNotification","createCommandInputMessage","createSyntheticUserCaveatMessage","createSystemMessage","createUserInterruptionMessage","createUserMessage","formatCommandInputTags","isCompactBoundaryMessage","isSystemLocalCommandMessage","normalizeMessages","prepareUserContent","ModelAlias","parseToolListFromCLI","hasPermissionsToUseTool","isOfficialMarketplaceName","parsePluginIdentifier","isRestrictedToPluginOnly","isSourceAdminTrusted","parseSlashCommand","sleep","recordSkillUsage","logOTelEvent","redactIfDisabled","buildPluginCommandTelemetryFields","getAssistantMessageContentLength","createAgentId","getWorkload","ProcessUserInputBaseResult","ProcessUserInputContext","SlashCommandResult","command","MCP_SETTLE_POLL_MS","MCP_SETTLE_TIMEOUT_MS","executeForkedSlashCommand","args","context","precedingInputBlocks","setToolJSX","canUseTool","Promise","agentId","pluginMarketplace","pluginInfo","repository","marketplace","undefined","command_name","name","invocation_trigger","_PROTO_plugin_name","pluginManifest","_PROTO_marketplace_name","skillContent","modifiedGetAppState","baseAgent","promptMessages","agentDefinition","effort","agentType","getAppState","kairosEnabled","bgAbortController","commandName","spawnTimeWorkload","enqueueResult","value","mode","priority","isMeta","skipSlashCommands","workload","deadline","Date","now","s","mcp","clients","some","c","type","freshTools","options","refreshTools","tools","agentMessages","message","toolUseContext","abortController","isAsync","querySource","model","availableTools","override","push","resultText","catch","err","Error","String","messages","shouldQuery","progressMessages","parentToolUseID","toolUseCounter","createProgressMessage","data","prompt","toolUseID","timestamp","toISOString","uuid","updateProgress","jsx","verbose","shouldHidePromptInput","shouldContinueAnimation","showSpinner","normalizedNew","contentLength","setResponseLength","len","normalizedMsg","content","inputString","trim","looksLikeCommand","test","processSlashCommand","imageContentBlocks","attachmentMessages","isAlreadyProcessing","parsed","errorMessage","parsedArgs","isMcp","sanitizedCommandName","has","commands","isFilePath","stat","input","unknownMessage","promptId","prompt_length","length","newMessages","messageShouldQuery","allowedTools","returnedCommand","nextInput","submitNextInput","getMessagesForSlashCommand","eventData","Record","isOfficial","plugin_repository","plugin_name","version","plugin_version","Object","assign","skill_name","skill_source","source","loadedFrom","skill_loaded_from","kind","skill_kind","startsWith","looksLikeFilePath","isCompactResult","every","_isAlreadyProcessing","userInvocable","resolve","doneWasCalled","onDone","result","display","metaMessages","map","skipTranscript","endsWith","formatCommandInput","load","then","mod","call","isNonInteractiveSession","isLocalJSXCommand","isImmediate","immediate","e","clearLocalJSX","displayArgs","isSensitive","userMessage","syntheticCaveatMessage","slashCommandMessages","displayText","compactionResultWithSlashMessages","compactionResult","messagesToKeep","getMessagesForPromptSlashCommand","toolUse","formatSkillLoadingMetadata","skillName","_progressMessage","join","formatSlashCommandLoadingMetadata","filter","Boolean","formatCommandLoadingMetadata","progressMessage","processPromptSlashCommand","process","env","CLAUDE_CODE_COORDINATOR_MODE","metadata","parts","description","whenToUse","skillAllowedTools","summaryContent","text","getPromptForCommand","hooksAllowedForThisSkill","hooks","sessionId","setAppState","skillRoot","skillPath","b","additionalAllowedTools","mainMessageContent","block","skipSkillDiscovery"],"sources":["processSlashCommand.tsx"],"sourcesContent":["import { feature } from 'bun:bundle'\nimport type {\n  ContentBlockParam,\n  TextBlockParam,\n} from '@anthropic-ai/sdk/resources'\nimport { randomUUID } from 'crypto'\nimport { setPromptId } from 'src/bootstrap/state.js'\nimport {\n  builtInCommandNames,\n  type Command,\n  type CommandBase,\n  findCommand,\n  getCommand,\n  getCommandName,\n  hasCommand,\n  type PromptCommand,\n} from 'src/commands.js'\nimport { NO_CONTENT_MESSAGE } from 'src/constants/messages.js'\nimport type { SetToolJSXFn, ToolUseContext } from 'src/Tool.js'\nimport type {\n  AssistantMessage,\n  AttachmentMessage,\n  Message,\n  NormalizedUserMessage,\n  ProgressMessage,\n  UserMessage,\n} from 'src/types/message.js'\nimport { addInvokedSkill, getSessionId } from '../../bootstrap/state.js'\nimport { COMMAND_MESSAGE_TAG, COMMAND_NAME_TAG } from '../../constants/xml.js'\nimport type { CanUseToolFn } from '../../hooks/useCanUseTool.js'\nimport {\n  type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n  type AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED,\n  logEvent,\n} from '../../services/analytics/index.js'\nimport { getDumpPromptsPath } from '../../services/api/dumpPrompts.js'\nimport { buildPostCompactMessages } from '../../services/compact/compact.js'\nimport { resetMicrocompactState } from '../../services/compact/microCompact.js'\nimport type { Progress as AgentProgress } from '../../tools/AgentTool/AgentTool.js'\nimport { runAgent } from '../../tools/AgentTool/runAgent.js'\nimport { renderToolUseProgressMessage } from '../../tools/AgentTool/UI.js'\nimport type { CommandResultDisplay } from '../../types/command.js'\nimport { createAbortController } from '../abortController.js'\nimport { getAgentContext } from '../agentContext.js'\nimport {\n  createAttachmentMessage,\n  getAttachmentMessages,\n} from '../attachments.js'\nimport { logForDebugging } from '../debug.js'\nimport { isEnvTruthy } from '../envUtils.js'\nimport { AbortError, MalformedCommandError } from '../errors.js'\nimport { getDisplayPath } from '../file.js'\nimport {\n  extractResultText,\n  prepareForkedCommandContext,\n} from '../forkedAgent.js'\nimport { getFsImplementation } from '../fsOperations.js'\nimport { isFullscreenEnvEnabled } from '../fullscreen.js'\nimport { toArray } from '../generators.js'\nimport { registerSkillHooks } from '../hooks/registerSkillHooks.js'\nimport { logError } from '../log.js'\nimport { enqueuePendingNotification } from '../messageQueueManager.js'\nimport {\n  createCommandInputMessage,\n  createSyntheticUserCaveatMessage,\n  createSystemMessage,\n  createUserInterruptionMessage,\n  createUserMessage,\n  formatCommandInputTags,\n  isCompactBoundaryMessage,\n  isSystemLocalCommandMessage,\n  normalizeMessages,\n  prepareUserContent,\n} from '../messages.js'\nimport type { ModelAlias } from '../model/aliases.js'\nimport { parseToolListFromCLI } from '../permissions/permissionSetup.js'\nimport { hasPermissionsToUseTool } from '../permissions/permissions.js'\nimport {\n  isOfficialMarketplaceName,\n  parsePluginIdentifier,\n} from '../plugins/pluginIdentifier.js'\nimport {\n  isRestrictedToPluginOnly,\n  isSourceAdminTrusted,\n} from '../settings/pluginOnlyPolicy.js'\nimport { parseSlashCommand } from '../slashCommandParsing.js'\nimport { sleep } from '../sleep.js'\nimport { recordSkillUsage } from '../suggestions/skillUsageTracking.js'\nimport { logOTelEvent, redactIfDisabled } from '../telemetry/events.js'\nimport { buildPluginCommandTelemetryFields } from '../telemetry/pluginTelemetry.js'\nimport { getAssistantMessageContentLength } from '../tokens.js'\nimport { createAgentId } from '../uuid.js'\nimport { getWorkload } from '../workloadContext.js'\nimport type {\n  ProcessUserInputBaseResult,\n  ProcessUserInputContext,\n} from './processUserInput.js'\n\ntype SlashCommandResult = ProcessUserInputBaseResult & {\n  command: Command\n}\n\n// Poll interval and deadline for MCP settle before launching a background\n// forked subagent. MCP servers typically connect within 1-3s of startup;\n// 10s headroom covers slow SSE handshakes.\nconst MCP_SETTLE_POLL_MS = 200\nconst MCP_SETTLE_TIMEOUT_MS = 10_000\n\n/**\n * Executes a slash command with context: fork in a sub-agent.\n */\nasync function executeForkedSlashCommand(\n  command: CommandBase & PromptCommand,\n  args: string,\n  context: ProcessUserInputContext,\n  precedingInputBlocks: ContentBlockParam[],\n  setToolJSX: SetToolJSXFn,\n  canUseTool: CanUseToolFn,\n): Promise<SlashCommandResult> {\n  const agentId = createAgentId()\n\n  const pluginMarketplace = command.pluginInfo\n    ? parsePluginIdentifier(command.pluginInfo.repository).marketplace\n    : undefined\n  logEvent('tengu_slash_command_forked', {\n    command_name:\n      command.name as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n    invocation_trigger:\n      'user-slash' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n    ...(command.pluginInfo && {\n      _PROTO_plugin_name: command.pluginInfo.pluginManifest\n        .name as AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED,\n      ...(pluginMarketplace && {\n        _PROTO_marketplace_name:\n          pluginMarketplace as AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED,\n      }),\n      ...buildPluginCommandTelemetryFields(command.pluginInfo),\n    }),\n  })\n\n  const { skillContent, modifiedGetAppState, baseAgent, promptMessages } =\n    await prepareForkedCommandContext(command, args, context)\n\n  // Merge skill's effort into the agent definition so runAgent applies it\n  const agentDefinition =\n    command.effort !== undefined\n      ? { ...baseAgent, effort: command.effort }\n      : baseAgent\n\n  logForDebugging(\n    `Executing forked slash command /${command.name} with agent ${agentDefinition.agentType}`,\n  )\n\n  // Assistant mode: fire-and-forget. Launch subagent in background, return\n  // immediately, re-enqueue the result as an isMeta prompt when done.\n  // Without this, N scheduled tasks on startup = N serial (subagent + main\n  // agent turn) cycles blocking user input. With this, N subagents run in\n  // parallel and results trickle into the queue as they finish.\n  //\n  // Gated on kairosEnabled (not CLAUDE_CODE_BRIEF) because the closed loop\n  // depends on assistant-mode invariants: scheduled_tasks.json exists,\n  // the main agent knows to pipe results through SendUserMessage, and\n  // isMeta prompts are hidden. Outside assistant mode, context:fork commands\n  // are user-invoked skills (/commit etc.) that should run synchronously\n  // with the progress UI.\n  if (feature('KAIROS') && (await context.getAppState()).kairosEnabled) {\n    // Standalone abortController — background subagents survive main-thread\n    // ESC (same policy as AgentTool's async path). They're cron-driven; if\n    // killed mid-run they just re-fire on the next schedule.\n    const bgAbortController = createAbortController()\n    const commandName = getCommandName(command)\n\n    // Workload: handlePromptSubmit wraps the entire turn in runWithWorkload\n    // (AsyncLocalStorage). ALS context is captured when this `void` fires\n    // and survives every await inside — isolated from the parent's\n    // continuation. The detached closure's runAgent calls see the cron tag\n    // automatically. We still capture the value here ONLY for the\n    // re-enqueued result prompt below: that second turn runs in a fresh\n    // handlePromptSubmit → fresh runWithWorkload boundary (which always\n    // establishes a new context, even for `undefined`) → so it needs its\n    // own QueuedCommand.workload tag to preserve attribution.\n    const spawnTimeWorkload = getWorkload()\n\n    // Re-enter the queue as a hidden prompt. isMeta: hides from queue\n    // preview + placeholder + transcript. skipSlashCommands: prevents\n    // re-parsing if the result text happens to start with '/'. When\n    // drained, this triggers a main-agent turn that sees the result and\n    // decides whether to SendUserMessage. Propagate workload so that\n    // second turn is also tagged.\n    const enqueueResult = (value: string): void =>\n      enqueuePendingNotification({\n        value,\n        mode: 'prompt',\n        priority: 'later',\n        isMeta: true,\n        skipSlashCommands: true,\n        workload: spawnTimeWorkload,\n      })\n\n    void (async () => {\n      // Wait for MCP servers to settle. Scheduled tasks fire at startup and\n      // all N drain within ~1ms (since we return immediately), capturing\n      // context.options.tools before MCP connects. The sync path\n      // accidentally avoided this — tasks serialized, so task N's drain\n      // happened after task N-1's 30s run, by which time MCP was up.\n      // Poll until no 'pending' clients remain, then refresh.\n      const deadline = Date.now() + MCP_SETTLE_TIMEOUT_MS\n      while (Date.now() < deadline) {\n        const s = context.getAppState()\n        if (!s.mcp.clients.some(c => c.type === 'pending')) break\n        await sleep(MCP_SETTLE_POLL_MS)\n      }\n      const freshTools =\n        context.options.refreshTools?.() ?? context.options.tools\n\n      const agentMessages: Message[] = []\n      for await (const message of runAgent({\n        agentDefinition,\n        promptMessages,\n        toolUseContext: {\n          ...context,\n          getAppState: modifiedGetAppState,\n          abortController: bgAbortController,\n        },\n        canUseTool,\n        isAsync: true,\n        querySource: 'agent:custom',\n        model: command.model as ModelAlias | undefined,\n        availableTools: freshTools,\n        override: { agentId },\n      })) {\n        agentMessages.push(message)\n      }\n      const resultText = extractResultText(agentMessages, 'Command completed')\n      logForDebugging(\n        `Background forked command /${commandName} completed (agent ${agentId})`,\n      )\n      enqueueResult(\n        `<scheduled-task-result command=\"/${commandName}\">\\n${resultText}\\n</scheduled-task-result>`,\n      )\n    })().catch(err => {\n      logError(err)\n      enqueueResult(\n        `<scheduled-task-result command=\"/${commandName}\" status=\"failed\">\\n${err instanceof Error ? err.message : String(err)}\\n</scheduled-task-result>`,\n      )\n    })\n\n    // Nothing to render, nothing to query — the background runner re-enters\n    // the queue on its own schedule.\n    return { messages: [], shouldQuery: false, command }\n  }\n\n  // Collect messages from the forked agent\n  const agentMessages: Message[] = []\n\n  // Build progress messages for the agent progress UI\n  const progressMessages: ProgressMessage<AgentProgress>[] = []\n  const parentToolUseID = `forked-command-${command.name}`\n  let toolUseCounter = 0\n\n  // Helper to create a progress message from an agent message\n  const createProgressMessage = (\n    message: AssistantMessage | NormalizedUserMessage,\n  ): ProgressMessage<AgentProgress> => {\n    toolUseCounter++\n    return {\n      type: 'progress',\n      data: {\n        message,\n        type: 'agent_progress',\n        prompt: skillContent,\n        agentId,\n      },\n      parentToolUseID,\n      toolUseID: `${parentToolUseID}-${toolUseCounter}`,\n      timestamp: new Date().toISOString(),\n      uuid: randomUUID(),\n    }\n  }\n\n  // Helper to update progress display using agent progress UI\n  const updateProgress = (): void => {\n    setToolJSX({\n      jsx: renderToolUseProgressMessage(progressMessages, {\n        tools: context.options.tools,\n        verbose: false,\n      }),\n      shouldHidePromptInput: false,\n      shouldContinueAnimation: true,\n      showSpinner: true,\n    })\n  }\n\n  // Show initial \"Initializing…\" state\n  updateProgress()\n\n  // Run the sub-agent\n  try {\n    for await (const message of runAgent({\n      agentDefinition,\n      promptMessages,\n      toolUseContext: {\n        ...context,\n        getAppState: modifiedGetAppState,\n      },\n      canUseTool,\n      isAsync: false,\n      querySource: 'agent:custom',\n      model: command.model as ModelAlias | undefined,\n      availableTools: context.options.tools,\n    })) {\n      agentMessages.push(message)\n      const normalizedNew = normalizeMessages([message])\n\n      // Add progress message for assistant messages (which contain tool uses)\n      if (message.type === 'assistant') {\n        // Increment token count in spinner for assistant messages\n        const contentLength = getAssistantMessageContentLength(message)\n        if (contentLength > 0) {\n          context.setResponseLength(len => len + contentLength)\n        }\n\n        const normalizedMsg = normalizedNew[0]\n        if (normalizedMsg && normalizedMsg.type === 'assistant') {\n          progressMessages.push(createProgressMessage(message))\n          updateProgress()\n        }\n      }\n\n      // Add progress message for user messages (which contain tool results)\n      if (message.type === 'user') {\n        const normalizedMsg = normalizedNew[0]\n        if (normalizedMsg && normalizedMsg.type === 'user') {\n          progressMessages.push(createProgressMessage(normalizedMsg))\n          updateProgress()\n        }\n      }\n    }\n  } finally {\n    // Clear the progress display\n    setToolJSX(null)\n  }\n\n  let resultText = extractResultText(agentMessages, 'Command completed')\n\n  logForDebugging(\n    `Forked slash command /${command.name} completed with agent ${agentId}`,\n  )\n\n  // Prepend debug log for ant users so it appears inside the command output\n  if (\"external\" === 'ant') {\n    resultText = `[ANT-ONLY] API calls: ${getDisplayPath(getDumpPromptsPath(agentId))}\\n${resultText}`\n  }\n\n  // Return the result as a user message (simulates the agent's output)\n  const messages: UserMessage[] = [\n    createUserMessage({\n      content: prepareUserContent({\n        inputString: `/${getCommandName(command)} ${args}`.trim(),\n        precedingInputBlocks,\n      }),\n    }),\n    createUserMessage({\n      content: `<local-command-stdout>\\n${resultText}\\n</local-command-stdout>`,\n    }),\n  ]\n\n  return {\n    messages,\n    shouldQuery: false,\n    command,\n    resultText,\n  }\n}\n\n/**\n * Determines if a string looks like a valid command name.\n * Valid command names only contain letters, numbers, colons, hyphens, and underscores.\n *\n * @param commandName - The potential command name to check\n * @returns true if it looks like a command name, false if it contains non-command characters\n */\nexport function looksLikeCommand(commandName: string): boolean {\n  // Command names should only contain [a-zA-Z0-9:_-]\n  // If it contains other characters, it's probably a file path or other input\n  return !/[^a-zA-Z0-9:\\-_]/.test(commandName)\n}\n\nexport async function processSlashCommand(\n  inputString: string,\n  precedingInputBlocks: ContentBlockParam[],\n  imageContentBlocks: ContentBlockParam[],\n  attachmentMessages: AttachmentMessage[],\n  context: ProcessUserInputContext,\n  setToolJSX: SetToolJSXFn,\n  uuid?: string,\n  isAlreadyProcessing?: boolean,\n  canUseTool?: CanUseToolFn,\n): Promise<ProcessUserInputBaseResult> {\n  const parsed = parseSlashCommand(inputString)\n  if (!parsed) {\n    logEvent('tengu_input_slash_missing', {})\n    const errorMessage = 'Commands are in the form `/command [args]`'\n    return {\n      messages: [\n        createSyntheticUserCaveatMessage(),\n        ...attachmentMessages,\n        createUserMessage({\n          content: prepareUserContent({\n            inputString: errorMessage,\n            precedingInputBlocks,\n          }),\n        }),\n      ],\n      shouldQuery: false,\n      resultText: errorMessage,\n    }\n  }\n\n  const { commandName, args: parsedArgs, isMcp } = parsed\n\n  const sanitizedCommandName = isMcp\n    ? 'mcp'\n    : !builtInCommandNames().has(commandName)\n      ? 'custom'\n      : commandName\n\n  // Check if it's a real command before processing\n  if (!hasCommand(commandName, context.options.commands)) {\n    // Check if this looks like a command name vs a file path or other input\n    // Also check if it's an actual file path that exists\n    let isFilePath = false\n    try {\n      await getFsImplementation().stat(`/${commandName}`)\n      isFilePath = true\n    } catch {\n      // Not a file path — treat as command name\n    }\n    if (looksLikeCommand(commandName) && !isFilePath) {\n      logEvent('tengu_input_slash_invalid', {\n        input:\n          commandName as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      })\n\n      const unknownMessage = `Unknown skill: ${commandName}`\n      return {\n        messages: [\n          createSyntheticUserCaveatMessage(),\n          ...attachmentMessages,\n          createUserMessage({\n            content: prepareUserContent({\n              inputString: unknownMessage,\n              precedingInputBlocks,\n            }),\n          }),\n          // gh-32591: preserve args so the user can copy/resubmit without\n          // retyping. System warning is UI-only (filtered before API).\n          ...(parsedArgs\n            ? [\n                createSystemMessage(\n                  `Args from unknown skill: ${parsedArgs}`,\n                  'warning',\n                ),\n              ]\n            : []),\n        ],\n        shouldQuery: false,\n        resultText: unknownMessage,\n      }\n    }\n\n    const promptId = randomUUID()\n    setPromptId(promptId)\n    logEvent('tengu_input_prompt', {})\n    // Log user prompt event for OTLP\n    void logOTelEvent('user_prompt', {\n      prompt_length: String(inputString.length),\n      prompt: redactIfDisabled(inputString),\n      'prompt.id': promptId,\n    })\n    return {\n      messages: [\n        createUserMessage({\n          content: prepareUserContent({ inputString, precedingInputBlocks }),\n          uuid: uuid,\n        }),\n        ...attachmentMessages,\n      ],\n      shouldQuery: true,\n    }\n  }\n\n  // Track slash command usage for feature discovery\n\n  const {\n    messages: newMessages,\n    shouldQuery: messageShouldQuery,\n    allowedTools,\n    model,\n    effort,\n    command: returnedCommand,\n    resultText,\n    nextInput,\n    submitNextInput,\n  } = await getMessagesForSlashCommand(\n    commandName,\n    parsedArgs,\n    setToolJSX,\n    context,\n    precedingInputBlocks,\n    imageContentBlocks,\n    isAlreadyProcessing,\n    canUseTool,\n    uuid,\n  )\n\n  // Local slash commands that skip messages\n  if (newMessages.length === 0) {\n    const eventData: Record<string, boolean | number | undefined> = {\n      input:\n        sanitizedCommandName as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n    }\n\n    // Add plugin metadata if this is a plugin command\n    if (returnedCommand.type === 'prompt' && returnedCommand.pluginInfo) {\n      const { pluginManifest, repository } = returnedCommand.pluginInfo\n      const { marketplace } = parsePluginIdentifier(repository)\n      const isOfficial = isOfficialMarketplaceName(marketplace)\n      // _PROTO_* routes to PII-tagged plugin_name/marketplace_name BQ columns\n      // (unredacted, all users); plugin_name/plugin_repository stay in\n      // additional_metadata as redacted variants for general-access dashboards.\n      eventData._PROTO_plugin_name =\n        pluginManifest.name as AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED\n      if (marketplace) {\n        eventData._PROTO_marketplace_name =\n          marketplace as AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED\n      }\n      eventData.plugin_repository = (\n        isOfficial ? repository : 'third-party'\n      ) as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS\n      eventData.plugin_name = (\n        isOfficial ? pluginManifest.name : 'third-party'\n      ) as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS\n      if (isOfficial && pluginManifest.version) {\n        eventData.plugin_version =\n          pluginManifest.version as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS\n      }\n      Object.assign(\n        eventData,\n        buildPluginCommandTelemetryFields(returnedCommand.pluginInfo),\n      )\n    }\n\n    logEvent('tengu_input_command', {\n      ...eventData,\n      invocation_trigger:\n        'user-slash' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      ...(\"external\" === 'ant' && {\n        skill_name:\n          commandName as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        ...(returnedCommand.type === 'prompt' && {\n          skill_source:\n            returnedCommand.source as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        }),\n        ...(returnedCommand.loadedFrom && {\n          skill_loaded_from:\n            returnedCommand.loadedFrom as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        }),\n        ...(returnedCommand.kind && {\n          skill_kind:\n            returnedCommand.kind as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        }),\n      }),\n    })\n    return {\n      messages: [],\n      shouldQuery: false,\n\n      model,\n      nextInput,\n      submitNextInput,\n    }\n  }\n\n  // For invalid commands, preserve both the user message and error\n  if (\n    newMessages.length === 2 &&\n    newMessages[1]!.type === 'user' &&\n    typeof newMessages[1]!.message.content === 'string' &&\n    newMessages[1]!.message.content.startsWith('Unknown command:')\n  ) {\n    // Don't log as invalid if it looks like a common file path\n    const looksLikeFilePath =\n      inputString.startsWith('/var') ||\n      inputString.startsWith('/tmp') ||\n      inputString.startsWith('/private')\n\n    if (!looksLikeFilePath) {\n      logEvent('tengu_input_slash_invalid', {\n        input:\n          commandName as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      })\n    }\n\n    return {\n      messages: [createSyntheticUserCaveatMessage(), ...newMessages],\n      shouldQuery: messageShouldQuery,\n      allowedTools,\n\n      model,\n    }\n  }\n\n  // A valid command\n  const eventData: Record<string, boolean | number | undefined> = {\n    input:\n      sanitizedCommandName as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n  }\n\n  // Add plugin metadata if this is a plugin command\n  if (returnedCommand.type === 'prompt' && returnedCommand.pluginInfo) {\n    const { pluginManifest, repository } = returnedCommand.pluginInfo\n    const { marketplace } = parsePluginIdentifier(repository)\n    const isOfficial = isOfficialMarketplaceName(marketplace)\n    eventData._PROTO_plugin_name =\n      pluginManifest.name as AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED\n    if (marketplace) {\n      eventData._PROTO_marketplace_name =\n        marketplace as AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED\n    }\n    eventData.plugin_repository = (\n      isOfficial ? repository : 'third-party'\n    ) as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS\n    eventData.plugin_name = (\n      isOfficial ? pluginManifest.name : 'third-party'\n    ) as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS\n    if (isOfficial && pluginManifest.version) {\n      eventData.plugin_version =\n        pluginManifest.version as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS\n    }\n    Object.assign(\n      eventData,\n      buildPluginCommandTelemetryFields(returnedCommand.pluginInfo),\n    )\n  }\n\n  logEvent('tengu_input_command', {\n    ...eventData,\n    invocation_trigger:\n      'user-slash' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n    ...(\"external\" === 'ant' && {\n      skill_name:\n        commandName as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      ...(returnedCommand.type === 'prompt' && {\n        skill_source:\n          returnedCommand.source as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      }),\n      ...(returnedCommand.loadedFrom && {\n        skill_loaded_from:\n          returnedCommand.loadedFrom as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      }),\n      ...(returnedCommand.kind && {\n        skill_kind:\n          returnedCommand.kind as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      }),\n    }),\n  })\n\n  // Check if this is a compact result which handle their own synthetic caveat message ordering\n  const isCompactResult =\n    newMessages.length > 0 &&\n    newMessages[0] &&\n    isCompactBoundaryMessage(newMessages[0])\n\n  return {\n    messages:\n      messageShouldQuery ||\n      newMessages.every(isSystemLocalCommandMessage) ||\n      isCompactResult\n        ? newMessages\n        : [createSyntheticUserCaveatMessage(), ...newMessages],\n    shouldQuery: messageShouldQuery,\n    allowedTools,\n    model,\n    effort,\n    resultText,\n    nextInput,\n    submitNextInput,\n  }\n}\n\nasync function getMessagesForSlashCommand(\n  commandName: string,\n  args: string,\n  setToolJSX: SetToolJSXFn,\n  context: ProcessUserInputContext,\n  precedingInputBlocks: ContentBlockParam[],\n  imageContentBlocks: ContentBlockParam[],\n  _isAlreadyProcessing?: boolean,\n  canUseTool?: CanUseToolFn,\n  uuid?: string,\n): Promise<SlashCommandResult> {\n  const command = getCommand(commandName, context.options.commands)\n\n  // Track skill usage for ranking (only for prompt commands that are user-invocable)\n  if (command.type === 'prompt' && command.userInvocable !== false) {\n    recordSkillUsage(commandName)\n  }\n\n  // Check if the command is user-invocable\n  // Skills with userInvocable === false can only be invoked by the model via SkillTool\n  if (command.userInvocable === false) {\n    return {\n      messages: [\n        createUserMessage({\n          content: prepareUserContent({\n            inputString: `/${commandName}`,\n            precedingInputBlocks,\n          }),\n        }),\n        createUserMessage({\n          content: `This skill can only be invoked by Claude, not directly by users. Ask Claude to use the \"${commandName}\" skill for you.`,\n        }),\n      ],\n      shouldQuery: false,\n      command,\n    }\n  }\n\n  try {\n    switch (command.type) {\n      case 'local-jsx': {\n        return new Promise<SlashCommandResult>(resolve => {\n          let doneWasCalled = false\n          const onDone = (\n            result?: string,\n            options?: {\n              display?: CommandResultDisplay\n              shouldQuery?: boolean\n              metaMessages?: string[]\n              nextInput?: string\n              submitNextInput?: boolean\n            },\n          ) => {\n            doneWasCalled = true\n            // If display is 'skip', don't add any messages to the conversation\n            if (options?.display === 'skip') {\n              void resolve({\n                messages: [],\n                shouldQuery: false,\n                command,\n                nextInput: options?.nextInput,\n                submitNextInput: options?.submitNextInput,\n              })\n              return\n            }\n\n            // Meta messages are model-visible but hidden from the user\n            const metaMessages = (options?.metaMessages ?? []).map(\n              (content: string) => createUserMessage({ content, isMeta: true }),\n            )\n\n            // In fullscreen the command just showed as a centered modal\n            // pane — the transient notification is enough feedback. The\n            // \"❯ /config\" + \"⎿ dismissed\" transcript entries are\n            // type:system subtype:local_command (user-visible but NOT sent\n            // to the model), so skipping them doesn't affect model context.\n            // Outside fullscreen keep them so scrollback shows what ran.\n            // Only skip \"<Name> dismissed\" modal-close notifications —\n            // commands that early-exit before showing a modal (/ultraplan\n            // usage, /rename, /proactive) use display:system for actual\n            // output that must reach the transcript.\n            const skipTranscript =\n              isFullscreenEnvEnabled() &&\n              typeof result === 'string' &&\n              result.endsWith(' dismissed')\n\n            void resolve({\n              messages:\n                options?.display === 'system'\n                  ? skipTranscript\n                    ? metaMessages\n                    : [\n                        createCommandInputMessage(\n                          formatCommandInput(command, args),\n                        ),\n                        createCommandInputMessage(\n                          `<local-command-stdout>${result}</local-command-stdout>`,\n                        ),\n                        ...metaMessages,\n                      ]\n                  : [\n                      createUserMessage({\n                        content: prepareUserContent({\n                          inputString: formatCommandInput(command, args),\n                          precedingInputBlocks,\n                        }),\n                      }),\n                      result\n                        ? createUserMessage({\n                            content: `<local-command-stdout>${result}</local-command-stdout>`,\n                          })\n                        : createUserMessage({\n                            content: `<local-command-stdout>${NO_CONTENT_MESSAGE}</local-command-stdout>`,\n                          }),\n                      ...metaMessages,\n                    ],\n              shouldQuery: options?.shouldQuery ?? false,\n              command,\n              nextInput: options?.nextInput,\n              submitNextInput: options?.submitNextInput,\n            })\n          }\n\n          void command\n            .load()\n            .then(mod => mod.call(onDone, { ...context, canUseTool }, args))\n            .then(jsx => {\n              if (jsx == null) return\n              if (context.options.isNonInteractiveSession) {\n                void resolve({\n                  messages: [],\n                  shouldQuery: false,\n                  command,\n                })\n                return\n              }\n              // Guard: if onDone fired during mod.call() (early-exit path\n              // that calls onDone then returns JSX), skip setToolJSX. This\n              // chain is fire-and-forget — the outer Promise resolves when\n              // onDone is called, so executeUserInput may have already run\n              // its setToolJSX({clearLocalJSX: true}) before we get here.\n              // Setting isLocalJSXCommand after clear leaves it stuck true,\n              // blocking useQueueProcessor and TextInput focus.\n              if (doneWasCalled) return\n              setToolJSX({\n                jsx,\n                shouldHidePromptInput: true,\n                showSpinner: false,\n                isLocalJSXCommand: true,\n                isImmediate: command.immediate === true,\n              })\n            })\n            .catch(e => {\n              // If load()/call() throws and onDone never fired, the outer\n              // Promise hangs forever, leaving queryGuard stuck in\n              // 'dispatching' and deadlocking the queue processor.\n              logError(e)\n              if (doneWasCalled) return\n              doneWasCalled = true\n              setToolJSX({\n                jsx: null,\n                shouldHidePromptInput: false,\n                clearLocalJSX: true,\n              })\n              void resolve({ messages: [], shouldQuery: false, command })\n            })\n        })\n      }\n      case 'local': {\n        const displayArgs = command.isSensitive && args.trim() ? '***' : args\n        const userMessage = createUserMessage({\n          content: prepareUserContent({\n            inputString: formatCommandInput(command, displayArgs),\n            precedingInputBlocks,\n          }),\n        })\n\n        try {\n          const syntheticCaveatMessage = createSyntheticUserCaveatMessage()\n          const mod = await command.load()\n          const result = await mod.call(args, context)\n\n          if (result.type === 'skip') {\n            return {\n              messages: [],\n              shouldQuery: false,\n              command,\n            }\n          }\n\n          // Use discriminated union to handle different result types\n          if (result.type === 'compact') {\n            // Append slash command messages to messagesToKeep so that\n            // attachments and hookResults come after user messages\n            const slashCommandMessages = [\n              syntheticCaveatMessage,\n              userMessage,\n              ...(result.displayText\n                ? [\n                    createUserMessage({\n                      content: `<local-command-stdout>${result.displayText}</local-command-stdout>`,\n                      // --resume looks at latest timestamp message to determine which message to resume from\n                      // This is a perf optimization to avoid having to recaculcate the leaf node every time\n                      // Since we're creating a bunch of synthetic messages for compact, it's important to set\n                      // the timestamp of the last message to be slightly after the current time\n                      // This is mostly important for sdk / -p mode\n                      timestamp: new Date(Date.now() + 100).toISOString(),\n                    }),\n                  ]\n                : []),\n            ]\n            const compactionResultWithSlashMessages = {\n              ...result.compactionResult,\n              messagesToKeep: [\n                ...(result.compactionResult.messagesToKeep ?? []),\n                ...slashCommandMessages,\n              ],\n            }\n            // Reset microcompact state since full compact replaces all\n            // messages — old tool IDs are no longer relevant. Budget state\n            // (on toolUseContext) needs no reset: stale entries are inert\n            // (UUIDs never repeat, so they're never looked up).\n            resetMicrocompactState()\n            return {\n              messages: buildPostCompactMessages(\n                compactionResultWithSlashMessages,\n              ),\n              shouldQuery: false,\n              command,\n            }\n          }\n\n          // Text result — use system message so it doesn't render as a user bubble\n          return {\n            messages: [\n              userMessage,\n              createCommandInputMessage(\n                `<local-command-stdout>${result.value}</local-command-stdout>`,\n              ),\n            ],\n            shouldQuery: false,\n            command,\n            resultText: result.value,\n          }\n        } catch (e) {\n          logError(e)\n          return {\n            messages: [\n              userMessage,\n              createCommandInputMessage(\n                `<local-command-stderr>${String(e)}</local-command-stderr>`,\n              ),\n            ],\n            shouldQuery: false,\n            command,\n          }\n        }\n      }\n      case 'prompt': {\n        try {\n          // Check if command should run as forked sub-agent\n          if (command.context === 'fork') {\n            return await executeForkedSlashCommand(\n              command,\n              args,\n              context,\n              precedingInputBlocks,\n              setToolJSX,\n              canUseTool ?? hasPermissionsToUseTool,\n            )\n          }\n\n          return await getMessagesForPromptSlashCommand(\n            command,\n            args,\n            context,\n            precedingInputBlocks,\n            imageContentBlocks,\n            uuid,\n          )\n        } catch (e) {\n          // Handle abort errors specially to show proper \"Interrupted\" message\n          if (e instanceof AbortError) {\n            return {\n              messages: [\n                createUserMessage({\n                  content: prepareUserContent({\n                    inputString: formatCommandInput(command, args),\n                    precedingInputBlocks,\n                  }),\n                }),\n                createUserInterruptionMessage({ toolUse: false }),\n              ],\n              shouldQuery: false,\n              command,\n            }\n          }\n          return {\n            messages: [\n              createUserMessage({\n                content: prepareUserContent({\n                  inputString: formatCommandInput(command, args),\n                  precedingInputBlocks,\n                }),\n              }),\n              createUserMessage({\n                content: `<local-command-stderr>${String(e)}</local-command-stderr>`,\n              }),\n            ],\n            shouldQuery: false,\n            command,\n          }\n        }\n      }\n    }\n  } catch (e) {\n    if (e instanceof MalformedCommandError) {\n      return {\n        messages: [\n          createUserMessage({\n            content: prepareUserContent({\n              inputString: e.message,\n              precedingInputBlocks,\n            }),\n          }),\n        ],\n        shouldQuery: false,\n        command,\n      }\n    }\n    throw e\n  }\n}\n\nfunction formatCommandInput(command: CommandBase, args: string): string {\n  return formatCommandInputTags(getCommandName(command), args)\n}\n\n/**\n * Formats the metadata for a skill loading message.\n * Used by the Skill tool and for subagent skill preloading.\n */\nexport function formatSkillLoadingMetadata(\n  skillName: string,\n  _progressMessage: string = 'loading',\n): string {\n  // Use skill name only - UserCommandMessage renders as \"Skill(name)\"\n  return [\n    `<${COMMAND_MESSAGE_TAG}>${skillName}</${COMMAND_MESSAGE_TAG}>`,\n    `<${COMMAND_NAME_TAG}>${skillName}</${COMMAND_NAME_TAG}>`,\n    `<skill-format>true</skill-format>`,\n  ].join('\\n')\n}\n\n/**\n * Formats the metadata for a slash command loading message.\n */\nfunction formatSlashCommandLoadingMetadata(\n  commandName: string,\n  args?: string,\n): string {\n  return [\n    `<${COMMAND_MESSAGE_TAG}>${commandName}</${COMMAND_MESSAGE_TAG}>`,\n    `<${COMMAND_NAME_TAG}>/${commandName}</${COMMAND_NAME_TAG}>`,\n    args ? `<command-args>${args}</command-args>` : null,\n  ]\n    .filter(Boolean)\n    .join('\\n')\n}\n\n/**\n * Formats the loading metadata for a command (skill or slash command).\n * User-invocable skills use slash command format (/name), while model-only\n * skills use the skill format (\"The X skill is running\").\n */\nfunction formatCommandLoadingMetadata(\n  command: CommandBase & PromptCommand,\n  args?: string,\n): string {\n  // Use command.name (the qualified name including plugin prefix, e.g.\n  // \"product-management:feature-spec\") instead of userFacingName() which may\n  // strip the plugin prefix via displayName fallback.\n  // User-invocable skills should show as /command-name like regular slash commands\n  if (command.userInvocable !== false) {\n    return formatSlashCommandLoadingMetadata(command.name, args)\n  }\n  // Model-only skills (userInvocable: false) show as \"The X skill is running\"\n  if (\n    command.loadedFrom === 'skills' ||\n    command.loadedFrom === 'plugin' ||\n    command.loadedFrom === 'mcp'\n  ) {\n    return formatSkillLoadingMetadata(command.name, command.progressMessage)\n  }\n  return formatSlashCommandLoadingMetadata(command.name, args)\n}\n\nexport async function processPromptSlashCommand(\n  commandName: string,\n  args: string,\n  commands: Command[],\n  context: ToolUseContext,\n  imageContentBlocks: ContentBlockParam[] = [],\n): Promise<SlashCommandResult> {\n  const command = findCommand(commandName, commands)\n  if (!command) {\n    throw new MalformedCommandError(`Unknown command: ${commandName}`)\n  }\n  if (command.type !== 'prompt') {\n    throw new Error(\n      `Unexpected ${command.type} command. Expected 'prompt' command. Use /${commandName} directly in the main conversation.`,\n    )\n  }\n  return getMessagesForPromptSlashCommand(\n    command,\n    args,\n    context,\n    [],\n    imageContentBlocks,\n  )\n}\n\nasync function getMessagesForPromptSlashCommand(\n  command: CommandBase & PromptCommand,\n  args: string,\n  context: ToolUseContext,\n  precedingInputBlocks: ContentBlockParam[] = [],\n  imageContentBlocks: ContentBlockParam[] = [],\n  uuid?: string,\n): Promise<SlashCommandResult> {\n  // In coordinator mode (main thread only), skip loading the full skill content\n  // and permissions. The coordinator only has Agent + TaskStop tools, so the\n  // skill content and allowedTools are useless. Instead, send a brief summary\n  // telling the coordinator how to delegate this skill to a worker.\n  //\n  // Workers run in-process and inherit CLAUDE_CODE_COORDINATOR_MODE from the\n  // parent env, so we also check !context.agentId: agentId is only set for\n  // subagents, letting workers fall through to getPromptForCommand and receive\n  // the real skill content when they invoke the Skill tool.\n  if (\n    feature('COORDINATOR_MODE') &&\n    isEnvTruthy(process.env.CLAUDE_CODE_COORDINATOR_MODE) &&\n    !context.agentId\n  ) {\n    const metadata = formatCommandLoadingMetadata(command, args)\n    const parts: string[] = [\n      `Skill \"/${command.name}\" is available for workers.`,\n    ]\n    if (command.description) {\n      parts.push(`Description: ${command.description}`)\n    }\n    if (command.whenToUse) {\n      parts.push(`When to use: ${command.whenToUse}`)\n    }\n    const skillAllowedTools = command.allowedTools ?? []\n    if (skillAllowedTools.length > 0) {\n      parts.push(\n        `This skill grants workers additional tool permissions: ${skillAllowedTools.join(', ')}`,\n      )\n    }\n    parts.push(\n      `\\nInstruct a worker to use this skill by including \"Use the /${command.name} skill\" in your Agent prompt. The worker has access to the Skill tool and will receive the skill's content and permissions when it invokes it.`,\n    )\n    const summaryContent: ContentBlockParam[] = [\n      { type: 'text', text: parts.join('\\n') },\n    ]\n    return {\n      messages: [\n        createUserMessage({ content: metadata, uuid }),\n        createUserMessage({ content: summaryContent, isMeta: true }),\n      ],\n      shouldQuery: true,\n      model: command.model,\n      effort: command.effort,\n      command,\n    }\n  }\n\n  const result = await command.getPromptForCommand(args, context)\n\n  // Register skill hooks if defined. Under [\"hooks\"]-only (skills not locked),\n  // user skills still load and reach this point — block hook REGISTRATION here\n  // where source is known. Mirrors the agent frontmatter gate in runAgent.ts.\n  const hooksAllowedForThisSkill =\n    !isRestrictedToPluginOnly('hooks') || isSourceAdminTrusted(command.source)\n  if (command.hooks && hooksAllowedForThisSkill) {\n    const sessionId = getSessionId()\n    registerSkillHooks(\n      context.setAppState,\n      sessionId,\n      command.hooks,\n      command.name,\n      command.type === 'prompt' ? command.skillRoot : undefined,\n    )\n  }\n\n  // Record skill invocation for compaction preservation, scoped by agent context.\n  // Skills are tagged with their agentId so only skills belonging to the current\n  // agent are restored during compaction (preventing cross-agent leaks).\n  const skillPath = command.source\n    ? `${command.source}:${command.name}`\n    : command.name\n  const skillContent = result\n    .filter((b): b is TextBlockParam => b.type === 'text')\n    .map(b => b.text)\n    .join('\\n\\n')\n  addInvokedSkill(\n    command.name,\n    skillPath,\n    skillContent,\n    getAgentContext()?.agentId ?? null,\n  )\n\n  const metadata = formatCommandLoadingMetadata(command, args)\n\n  const additionalAllowedTools = parseToolListFromCLI(\n    command.allowedTools ?? [],\n  )\n\n  // Create content for the main message, including any pasted images\n  const mainMessageContent: ContentBlockParam[] =\n    imageContentBlocks.length > 0 || precedingInputBlocks.length > 0\n      ? [...imageContentBlocks, ...precedingInputBlocks, ...result]\n      : result\n\n  // Extract attachments from command arguments (@-mentions, MCP resources,\n  // agent mentions in SKILL.md). skipSkillDiscovery prevents the SKILL.md\n  // content itself from triggering discovery — it's meta-content, not user\n  // intent, and a large SKILL.md (e.g. 110KB) would fire chunked AKI queries\n  // adding seconds of latency to every skill invocation.\n  const attachmentMessages = await toArray(\n    getAttachmentMessages(\n      result\n        .filter((block): block is TextBlockParam => block.type === 'text')\n        .map(block => block.text)\n        .join(' '),\n      context,\n      null,\n      [], // queuedCommands - handled by query.ts for mid-turn attachments\n      context.messages,\n      'repl_main_thread',\n      { skipSkillDiscovery: true },\n    ),\n  )\n\n  const messages = [\n    createUserMessage({\n      content: metadata,\n      uuid,\n    }),\n    createUserMessage({\n      content: mainMessageContent,\n      isMeta: true,\n    }),\n    ...attachmentMessages,\n    createAttachmentMessage({\n      type: 'command_permissions',\n      allowedTools: additionalAllowedTools,\n      model: command.model,\n    }),\n  ]\n\n  return {\n    messages,\n    shouldQuery: true,\n    allowedTools: additionalAllowedTools,\n    model: command.model,\n    effort: command.effort,\n    command,\n  }\n}\n"],"mappings":"AAAA,SAASA,OAAO,QAAQ,YAAY;AACpC,cACEC,iBAAiB,EACjBC,cAAc,QACT,6BAA6B;AACpC,SAASC,UAAU,QAAQ,QAAQ;AACnC,SAASC,WAAW,QAAQ,wBAAwB;AACpD,SACEC,mBAAmB,EACnB,KAAKC,OAAO,EACZ,KAAKC,WAAW,EAChBC,WAAW,EACXC,UAAU,EACVC,cAAc,EACdC,UAAU,EACV,KAAKC,aAAa,QACb,iBAAiB;AACxB,SAASC,kBAAkB,QAAQ,2BAA2B;AAC9D,cAAcC,YAAY,EAAEC,cAAc,QAAQ,aAAa;AAC/D,cACEC,gBAAgB,EAChBC,iBAAiB,EACjBC,OAAO,EACPC,qBAAqB,EACrBC,eAAe,EACfC,WAAW,QACN,sBAAsB;AAC7B,SAASC,eAAe,EAAEC,YAAY,QAAQ,0BAA0B;AACxE,SAASC,mBAAmB,EAAEC,gBAAgB,QAAQ,wBAAwB;AAC9E,cAAcC,YAAY,QAAQ,8BAA8B;AAChE,SACE,KAAKC,0DAA0D,EAC/D,KAAKC,+CAA+C,EACpDC,QAAQ,QACH,mCAAmC;AAC1C,SAASC,kBAAkB,QAAQ,mCAAmC;AACtE,SAASC,wBAAwB,QAAQ,mCAAmC;AAC5E,SAASC,sBAAsB,QAAQ,wCAAwC;AAC/E,cAAcC,QAAQ,IAAIC,aAAa,QAAQ,oCAAoC;AACnF,SAASC,QAAQ,QAAQ,mCAAmC;AAC5D,SAASC,4BAA4B,QAAQ,6BAA6B;AAC1E,cAAcC,oBAAoB,QAAQ,wBAAwB;AAClE,SAASC,qBAAqB,QAAQ,uBAAuB;AAC7D,SAASC,eAAe,QAAQ,oBAAoB;AACpD,SACEC,uBAAuB,EACvBC,qBAAqB,QAChB,mBAAmB;AAC1B,SAASC,eAAe,QAAQ,aAAa;AAC7C,SAASC,WAAW,QAAQ,gBAAgB;AAC5C,SAASC,UAAU,EAAEC,qBAAqB,QAAQ,cAAc;AAChE,SAASC,cAAc,QAAQ,YAAY;AAC3C,SACEC,iBAAiB,EACjBC,2BAA2B,QACtB,mBAAmB;AAC1B,SAASC,mBAAmB,QAAQ,oBAAoB;AACxD,SAASC,sBAAsB,QAAQ,kBAAkB;AACzD,SAASC,OAAO,QAAQ,kBAAkB;AAC1C,SAASC,kBAAkB,QAAQ,gCAAgC;AACnE,SAASC,QAAQ,QAAQ,WAAW;AACpC,SAASC,0BAA0B,QAAQ,2BAA2B;AACtE,SACEC,yBAAyB,EACzBC,gCAAgC,EAChCC,mBAAmB,EACnBC,6BAA6B,EAC7BC,iBAAiB,EACjBC,sBAAsB,EACtBC,wBAAwB,EACxBC,2BAA2B,EAC3BC,iBAAiB,EACjBC,kBAAkB,QACb,gBAAgB;AACvB,cAAcC,UAAU,QAAQ,qBAAqB;AACrD,SAASC,oBAAoB,QAAQ,mCAAmC;AACxE,SAASC,uBAAuB,QAAQ,+BAA+B;AACvE,SACEC,yBAAyB,EACzBC,qBAAqB,QAChB,gCAAgC;AACvC,SACEC,wBAAwB,EACxBC,oBAAoB,QACf,iCAAiC;AACxC,SAASC,iBAAiB,QAAQ,2BAA2B;AAC7D,SAASC,KAAK,QAAQ,aAAa;AACnC,SAASC,gBAAgB,QAAQ,sCAAsC;AACvE,SAASC,YAAY,EAAEC,gBAAgB,QAAQ,wBAAwB;AACvE,SAASC,iCAAiC,QAAQ,iCAAiC;AACnF,SAASC,gCAAgC,QAAQ,cAAc;AAC/D,SAASC,aAAa,QAAQ,YAAY;AAC1C,SAASC,WAAW,QAAQ,uBAAuB;AACnD,cACEC,0BAA0B,EAC1BC,uBAAuB,QAClB,uBAAuB;AAE9B,KAAKC,kBAAkB,GAAGF,0BAA0B,GAAG;EACrDG,OAAO,EAAE9E,OAAO;AAClB,CAAC;;AAED;AACA;AACA;AACA,MAAM+E,kBAAkB,GAAG,GAAG;AAC9B,MAAMC,qBAAqB,GAAG,MAAM;;AAEpC;AACA;AACA;AACA,eAAeC,yBAAyBA,CACtCH,OAAO,EAAE7E,WAAW,GAAGK,aAAa,EACpC4E,IAAI,EAAE,MAAM,EACZC,OAAO,EAAEP,uBAAuB,EAChCQ,oBAAoB,EAAEzF,iBAAiB,EAAE,EACzC0F,UAAU,EAAE7E,YAAY,EACxB8E,UAAU,EAAElE,YAAY,CACzB,EAAEmE,OAAO,CAACV,kBAAkB,CAAC,CAAC;EAC7B,MAAMW,OAAO,GAAGf,aAAa,CAAC,CAAC;EAE/B,MAAMgB,iBAAiB,GAAGX,OAAO,CAACY,UAAU,GACxC3B,qBAAqB,CAACe,OAAO,CAACY,UAAU,CAACC,UAAU,CAAC,CAACC,WAAW,GAChEC,SAAS;EACbtE,QAAQ,CAAC,4BAA4B,EAAE;IACrCuE,YAAY,EACVhB,OAAO,CAACiB,IAAI,IAAI1E,0DAA0D;IAC5E2E,kBAAkB,EAChB,YAAY,IAAI3E,0DAA0D;IAC5E,IAAIyD,OAAO,CAACY,UAAU,IAAI;MACxBO,kBAAkB,EAAEnB,OAAO,CAACY,UAAU,CAACQ,cAAc,CAClDH,IAAI,IAAIzE,+CAA+C;MAC1D,IAAImE,iBAAiB,IAAI;QACvBU,uBAAuB,EACrBV,iBAAiB,IAAInE;MACzB,CAAC,CAAC;MACF,GAAGiD,iCAAiC,CAACO,OAAO,CAACY,UAAU;IACzD,CAAC;EACH,CAAC,CAAC;EAEF,MAAM;IAAEU,YAAY;IAAEC,mBAAmB;IAAEC,SAAS;IAAEC;EAAe,CAAC,GACpE,MAAM7D,2BAA2B,CAACoC,OAAO,EAAEI,IAAI,EAAEC,OAAO,CAAC;;EAE3D;EACA,MAAMqB,eAAe,GACnB1B,OAAO,CAAC2B,MAAM,KAAKZ,SAAS,GACxB;IAAE,GAAGS,SAAS;IAAEG,MAAM,EAAE3B,OAAO,CAAC2B;EAAO,CAAC,GACxCH,SAAS;EAEflE,eAAe,CACb,mCAAmC0C,OAAO,CAACiB,IAAI,eAAeS,eAAe,CAACE,SAAS,EACzF,CAAC;;EAED;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,IAAIhH,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,MAAMyF,OAAO,CAACwB,WAAW,CAAC,CAAC,EAAEC,aAAa,EAAE;IACpE;IACA;IACA;IACA,MAAMC,iBAAiB,GAAG7E,qBAAqB,CAAC,CAAC;IACjD,MAAM8E,WAAW,GAAG1G,cAAc,CAAC0E,OAAO,CAAC;;IAE3C;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA,MAAMiC,iBAAiB,GAAGrC,WAAW,CAAC,CAAC;;IAEvC;IACA;IACA;IACA;IACA;IACA;IACA,MAAMsC,aAAa,GAAGA,CAACC,KAAK,EAAE,MAAM,CAAC,EAAE,IAAI,IACzCjE,0BAA0B,CAAC;MACzBiE,KAAK;MACLC,IAAI,EAAE,QAAQ;MACdC,QAAQ,EAAE,OAAO;MACjBC,MAAM,EAAE,IAAI;MACZC,iBAAiB,EAAE,IAAI;MACvBC,QAAQ,EAAEP;IACZ,CAAC,CAAC;IAEJ,KAAK,CAAC,YAAY;MAChB;MACA;MACA;MACA;MACA;MACA;MACA,MAAMQ,QAAQ,GAAGC,IAAI,CAACC,GAAG,CAAC,CAAC,GAAGzC,qBAAqB;MACnD,OAAOwC,IAAI,CAACC,GAAG,CAAC,CAAC,GAAGF,QAAQ,EAAE;QAC5B,MAAMG,CAAC,GAAGvC,OAAO,CAACwB,WAAW,CAAC,CAAC;QAC/B,IAAI,CAACe,CAAC,CAACC,GAAG,CAACC,OAAO,CAACC,IAAI,CAACC,CAAC,IAAIA,CAAC,CAACC,IAAI,KAAK,SAAS,CAAC,EAAE;QACpD,MAAM5D,KAAK,CAACY,kBAAkB,CAAC;MACjC;MACA,MAAMiD,UAAU,GACd7C,OAAO,CAAC8C,OAAO,CAACC,YAAY,GAAG,CAAC,IAAI/C,OAAO,CAAC8C,OAAO,CAACE,KAAK;MAE3D,MAAMC,aAAa,EAAExH,OAAO,EAAE,GAAG,EAAE;MACnC,WAAW,MAAMyH,OAAO,IAAIxG,QAAQ,CAAC;QACnC2E,eAAe;QACfD,cAAc;QACd+B,cAAc,EAAE;UACd,GAAGnD,OAAO;UACVwB,WAAW,EAAEN,mBAAmB;UAChCkC,eAAe,EAAE1B;QACnB,CAAC;QACDvB,UAAU;QACVkD,OAAO,EAAE,IAAI;QACbC,WAAW,EAAE,cAAc;QAC3BC,KAAK,EAAE5D,OAAO,CAAC4D,KAAK,IAAI/E,UAAU,GAAG,SAAS;QAC9CgF,cAAc,EAAEX,UAAU;QAC1BY,QAAQ,EAAE;UAAEpD;QAAQ;MACtB,CAAC,CAAC,EAAE;QACF4C,aAAa,CAACS,IAAI,CAACR,OAAO,CAAC;MAC7B;MACA,MAAMS,UAAU,GAAGrG,iBAAiB,CAAC2F,aAAa,EAAE,mBAAmB,CAAC;MACxEhG,eAAe,CACb,8BAA8B0E,WAAW,qBAAqBtB,OAAO,GACvE,CAAC;MACDwB,aAAa,CACX,oCAAoCF,WAAW,OAAOgC,UAAU,4BAClE,CAAC;IACH,CAAC,EAAE,CAAC,CAACC,KAAK,CAACC,GAAG,IAAI;MAChBjG,QAAQ,CAACiG,GAAG,CAAC;MACbhC,aAAa,CACX,oCAAoCF,WAAW,uBAAuBkC,GAAG,YAAYC,KAAK,GAAGD,GAAG,CAACX,OAAO,GAAGa,MAAM,CAACF,GAAG,CAAC,4BACxH,CAAC;IACH,CAAC,CAAC;;IAEF;IACA;IACA,OAAO;MAAEG,QAAQ,EAAE,EAAE;MAAEC,WAAW,EAAE,KAAK;MAAEtE;IAAQ,CAAC;EACtD;;EAEA;EACA,MAAMsD,aAAa,EAAExH,OAAO,EAAE,GAAG,EAAE;;EAEnC;EACA,MAAMyI,gBAAgB,EAAEvI,eAAe,CAACc,aAAa,CAAC,EAAE,GAAG,EAAE;EAC7D,MAAM0H,eAAe,GAAG,kBAAkBxE,OAAO,CAACiB,IAAI,EAAE;EACxD,IAAIwD,cAAc,GAAG,CAAC;;EAEtB;EACA,MAAMC,qBAAqB,GAAGA,CAC5BnB,OAAO,EAAE3H,gBAAgB,GAAGG,qBAAqB,CAClD,EAAEC,eAAe,CAACc,aAAa,CAAC,IAAI;IACnC2H,cAAc,EAAE;IAChB,OAAO;MACLxB,IAAI,EAAE,UAAU;MAChB0B,IAAI,EAAE;QACJpB,OAAO;QACPN,IAAI,EAAE,gBAAgB;QACtB2B,MAAM,EAAEtD,YAAY;QACpBZ;MACF,CAAC;MACD8D,eAAe;MACfK,SAAS,EAAE,GAAGL,eAAe,IAAIC,cAAc,EAAE;MACjDK,SAAS,EAAE,IAAIpC,IAAI,CAAC,CAAC,CAACqC,WAAW,CAAC,CAAC;MACnCC,IAAI,EAAEjK,UAAU,CAAC;IACnB,CAAC;EACH,CAAC;;EAED;EACA,MAAMkK,cAAc,GAAGA,CAAA,CAAE,EAAE,IAAI,IAAI;IACjC1E,UAAU,CAAC;MACT2E,GAAG,EAAElI,4BAA4B,CAACuH,gBAAgB,EAAE;QAClDlB,KAAK,EAAEhD,OAAO,CAAC8C,OAAO,CAACE,KAAK;QAC5B8B,OAAO,EAAE;MACX,CAAC,CAAC;MACFC,qBAAqB,EAAE,KAAK;MAC5BC,uBAAuB,EAAE,IAAI;MAC7BC,WAAW,EAAE;IACf,CAAC,CAAC;EACJ,CAAC;;EAED;EACAL,cAAc,CAAC,CAAC;;EAEhB;EACA,IAAI;IACF,WAAW,MAAM1B,OAAO,IAAIxG,QAAQ,CAAC;MACnC2E,eAAe;MACfD,cAAc;MACd+B,cAAc,EAAE;QACd,GAAGnD,OAAO;QACVwB,WAAW,EAAEN;MACf,CAAC;MACDf,UAAU;MACVkD,OAAO,EAAE,KAAK;MACdC,WAAW,EAAE,cAAc;MAC3BC,KAAK,EAAE5D,OAAO,CAAC4D,KAAK,IAAI/E,UAAU,GAAG,SAAS;MAC9CgF,cAAc,EAAExD,OAAO,CAAC8C,OAAO,CAACE;IAClC,CAAC,CAAC,EAAE;MACFC,aAAa,CAACS,IAAI,CAACR,OAAO,CAAC;MAC3B,MAAMgC,aAAa,GAAG5G,iBAAiB,CAAC,CAAC4E,OAAO,CAAC,CAAC;;MAElD;MACA,IAAIA,OAAO,CAACN,IAAI,KAAK,WAAW,EAAE;QAChC;QACA,MAAMuC,aAAa,GAAG9F,gCAAgC,CAAC6D,OAAO,CAAC;QAC/D,IAAIiC,aAAa,GAAG,CAAC,EAAE;UACrBnF,OAAO,CAACoF,iBAAiB,CAACC,GAAG,IAAIA,GAAG,GAAGF,aAAa,CAAC;QACvD;QAEA,MAAMG,aAAa,GAAGJ,aAAa,CAAC,CAAC,CAAC;QACtC,IAAII,aAAa,IAAIA,aAAa,CAAC1C,IAAI,KAAK,WAAW,EAAE;UACvDsB,gBAAgB,CAACR,IAAI,CAACW,qBAAqB,CAACnB,OAAO,CAAC,CAAC;UACrD0B,cAAc,CAAC,CAAC;QAClB;MACF;;MAEA;MACA,IAAI1B,OAAO,CAACN,IAAI,KAAK,MAAM,EAAE;QAC3B,MAAM0C,aAAa,GAAGJ,aAAa,CAAC,CAAC,CAAC;QACtC,IAAII,aAAa,IAAIA,aAAa,CAAC1C,IAAI,KAAK,MAAM,EAAE;UAClDsB,gBAAgB,CAACR,IAAI,CAACW,qBAAqB,CAACiB,aAAa,CAAC,CAAC;UAC3DV,cAAc,CAAC,CAAC;QAClB;MACF;IACF;EACF,CAAC,SAAS;IACR;IACA1E,UAAU,CAAC,IAAI,CAAC;EAClB;EAEA,IAAIyD,UAAU,GAAGrG,iBAAiB,CAAC2F,aAAa,EAAE,mBAAmB,CAAC;EAEtEhG,eAAe,CACb,yBAAyB0C,OAAO,CAACiB,IAAI,yBAAyBP,OAAO,EACvE,CAAC;;EAED;EACA,IAAI,UAAU,KAAK,KAAK,EAAE;IACxBsD,UAAU,GAAG,yBAAyBtG,cAAc,CAAChB,kBAAkB,CAACgE,OAAO,CAAC,CAAC,KAAKsD,UAAU,EAAE;EACpG;;EAEA;EACA,MAAMK,QAAQ,EAAEpI,WAAW,EAAE,GAAG,CAC9BsC,iBAAiB,CAAC;IAChBqH,OAAO,EAAEhH,kBAAkB,CAAC;MAC1BiH,WAAW,EAAE,IAAIvK,cAAc,CAAC0E,OAAO,CAAC,IAAII,IAAI,EAAE,CAAC0F,IAAI,CAAC,CAAC;MACzDxF;IACF,CAAC;EACH,CAAC,CAAC,EACF/B,iBAAiB,CAAC;IAChBqH,OAAO,EAAE,2BAA2B5B,UAAU;EAChD,CAAC,CAAC,CACH;EAED,OAAO;IACLK,QAAQ;IACRC,WAAW,EAAE,KAAK;IAClBtE,OAAO;IACPgE;EACF,CAAC;AACH;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO,SAAS+B,gBAAgBA,CAAC/D,WAAW,EAAE,MAAM,CAAC,EAAE,OAAO,CAAC;EAC7D;EACA;EACA,OAAO,CAAC,kBAAkB,CAACgE,IAAI,CAAChE,WAAW,CAAC;AAC9C;AAEA,OAAO,eAAeiE,mBAAmBA,CACvCJ,WAAW,EAAE,MAAM,EACnBvF,oBAAoB,EAAEzF,iBAAiB,EAAE,EACzCqL,kBAAkB,EAAErL,iBAAiB,EAAE,EACvCsL,kBAAkB,EAAEtK,iBAAiB,EAAE,EACvCwE,OAAO,EAAEP,uBAAuB,EAChCS,UAAU,EAAE7E,YAAY,EACxBsJ,IAAa,CAAR,EAAE,MAAM,EACboB,mBAA6B,CAAT,EAAE,OAAO,EAC7B5F,UAAyB,CAAd,EAAElE,YAAY,CAC1B,EAAEmE,OAAO,CAACZ,0BAA0B,CAAC,CAAC;EACrC,MAAMwG,MAAM,GAAGjH,iBAAiB,CAACyG,WAAW,CAAC;EAC7C,IAAI,CAACQ,MAAM,EAAE;IACX5J,QAAQ,CAAC,2BAA2B,EAAE,CAAC,CAAC,CAAC;IACzC,MAAM6J,YAAY,GAAG,4CAA4C;IACjE,OAAO;MACLjC,QAAQ,EAAE,CACRjG,gCAAgC,CAAC,CAAC,EAClC,GAAG+H,kBAAkB,EACrB5H,iBAAiB,CAAC;QAChBqH,OAAO,EAAEhH,kBAAkB,CAAC;UAC1BiH,WAAW,EAAES,YAAY;UACzBhG;QACF,CAAC;MACH,CAAC,CAAC,CACH;MACDgE,WAAW,EAAE,KAAK;MAClBN,UAAU,EAAEsC;IACd,CAAC;EACH;EAEA,MAAM;IAAEtE,WAAW;IAAE5B,IAAI,EAAEmG,UAAU;IAAEC;EAAM,CAAC,GAAGH,MAAM;EAEvD,MAAMI,oBAAoB,GAAGD,KAAK,GAC9B,KAAK,GACL,CAACvL,mBAAmB,CAAC,CAAC,CAACyL,GAAG,CAAC1E,WAAW,CAAC,GACrC,QAAQ,GACRA,WAAW;;EAEjB;EACA,IAAI,CAACzG,UAAU,CAACyG,WAAW,EAAE3B,OAAO,CAAC8C,OAAO,CAACwD,QAAQ,CAAC,EAAE;IACtD;IACA;IACA,IAAIC,UAAU,GAAG,KAAK;IACtB,IAAI;MACF,MAAM/I,mBAAmB,CAAC,CAAC,CAACgJ,IAAI,CAAC,IAAI7E,WAAW,EAAE,CAAC;MACnD4E,UAAU,GAAG,IAAI;IACnB,CAAC,CAAC,MAAM;MACN;IAAA;IAEF,IAAIb,gBAAgB,CAAC/D,WAAW,CAAC,IAAI,CAAC4E,UAAU,EAAE;MAChDnK,QAAQ,CAAC,2BAA2B,EAAE;QACpCqK,KAAK,EACH9E,WAAW,IAAIzF;MACnB,CAAC,CAAC;MAEF,MAAMwK,cAAc,GAAG,kBAAkB/E,WAAW,EAAE;MACtD,OAAO;QACLqC,QAAQ,EAAE,CACRjG,gCAAgC,CAAC,CAAC,EAClC,GAAG+H,kBAAkB,EACrB5H,iBAAiB,CAAC;UAChBqH,OAAO,EAAEhH,kBAAkB,CAAC;YAC1BiH,WAAW,EAAEkB,cAAc;YAC3BzG;UACF,CAAC;QACH,CAAC,CAAC;QACF;QACA;QACA,IAAIiG,UAAU,GACV,CACElI,mBAAmB,CACjB,4BAA4BkI,UAAU,EAAE,EACxC,SACF,CAAC,CACF,GACD,EAAE,CAAC,CACR;QACDjC,WAAW,EAAE,KAAK;QAClBN,UAAU,EAAE+C;MACd,CAAC;IACH;IAEA,MAAMC,QAAQ,GAAGjM,UAAU,CAAC,CAAC;IAC7BC,WAAW,CAACgM,QAAQ,CAAC;IACrBvK,QAAQ,CAAC,oBAAoB,EAAE,CAAC,CAAC,CAAC;IAClC;IACA,KAAK8C,YAAY,CAAC,aAAa,EAAE;MAC/B0H,aAAa,EAAE7C,MAAM,CAACyB,WAAW,CAACqB,MAAM,CAAC;MACzCtC,MAAM,EAAEpF,gBAAgB,CAACqG,WAAW,CAAC;MACrC,WAAW,EAAEmB;IACf,CAAC,CAAC;IACF,OAAO;MACL3C,QAAQ,EAAE,CACR9F,iBAAiB,CAAC;QAChBqH,OAAO,EAAEhH,kBAAkB,CAAC;UAAEiH,WAAW;UAAEvF;QAAqB,CAAC,CAAC;QAClE0E,IAAI,EAAEA;MACR,CAAC,CAAC,EACF,GAAGmB,kBAAkB,CACtB;MACD7B,WAAW,EAAE;IACf,CAAC;EACH;;EAEA;;EAEA,MAAM;IACJD,QAAQ,EAAE8C,WAAW;IACrB7C,WAAW,EAAE8C,kBAAkB;IAC/BC,YAAY;IACZzD,KAAK;IACLjC,MAAM;IACN3B,OAAO,EAAEsH,eAAe;IACxBtD,UAAU;IACVuD,SAAS;IACTC;EACF,CAAC,GAAG,MAAMC,0BAA0B,CAClCzF,WAAW,EACXuE,UAAU,EACVhG,UAAU,EACVF,OAAO,EACPC,oBAAoB,EACpB4F,kBAAkB,EAClBE,mBAAmB,EACnB5F,UAAU,EACVwE,IACF,CAAC;;EAED;EACA,IAAImC,WAAW,CAACD,MAAM,KAAK,CAAC,EAAE;IAC5B,MAAMQ,SAAS,EAAEC,MAAM,CAAC,MAAM,EAAE,OAAO,GAAG,MAAM,GAAG,SAAS,CAAC,GAAG;MAC9Db,KAAK,EACHL,oBAAoB,IAAIlK;IAC5B,CAAC;;IAED;IACA,IAAI+K,eAAe,CAACrE,IAAI,KAAK,QAAQ,IAAIqE,eAAe,CAAC1G,UAAU,EAAE;MACnE,MAAM;QAAEQ,cAAc;QAAEP;MAAW,CAAC,GAAGyG,eAAe,CAAC1G,UAAU;MACjE,MAAM;QAAEE;MAAY,CAAC,GAAG7B,qBAAqB,CAAC4B,UAAU,CAAC;MACzD,MAAM+G,UAAU,GAAG5I,yBAAyB,CAAC8B,WAAW,CAAC;MACzD;MACA;MACA;MACA4G,SAAS,CAACvG,kBAAkB,GAC1BC,cAAc,CAACH,IAAI,IAAIzE,+CAA+C;MACxE,IAAIsE,WAAW,EAAE;QACf4G,SAAS,CAACrG,uBAAuB,GAC/BP,WAAW,IAAItE,+CAA+C;MAClE;MACAkL,SAAS,CAACG,iBAAiB,GAAG,CAC5BD,UAAU,GAAG/G,UAAU,GAAG,aAAa,KACpCtE,0DAA0D;MAC/DmL,SAAS,CAACI,WAAW,GAAG,CACtBF,UAAU,GAAGxG,cAAc,CAACH,IAAI,GAAG,aAAa,KAC7C1E,0DAA0D;MAC/D,IAAIqL,UAAU,IAAIxG,cAAc,CAAC2G,OAAO,EAAE;QACxCL,SAAS,CAACM,cAAc,GACtB5G,cAAc,CAAC2G,OAAO,IAAIxL,0DAA0D;MACxF;MACA0L,MAAM,CAACC,MAAM,CACXR,SAAS,EACTjI,iCAAiC,CAAC6H,eAAe,CAAC1G,UAAU,CAC9D,CAAC;IACH;IAEAnE,QAAQ,CAAC,qBAAqB,EAAE;MAC9B,GAAGiL,SAAS;MACZxG,kBAAkB,EAChB,YAAY,IAAI3E,0DAA0D;MAC5E,IAAI,UAAU,KAAK,KAAK,IAAI;QAC1B4L,UAAU,EACRnG,WAAW,IAAIzF,0DAA0D;QAC3E,IAAI+K,eAAe,CAACrE,IAAI,KAAK,QAAQ,IAAI;UACvCmF,YAAY,EACVd,eAAe,CAACe,MAAM,IAAI9L;QAC9B,CAAC,CAAC;QACF,IAAI+K,eAAe,CAACgB,UAAU,IAAI;UAChCC,iBAAiB,EACfjB,eAAe,CAACgB,UAAU,IAAI/L;QAClC,CAAC,CAAC;QACF,IAAI+K,eAAe,CAACkB,IAAI,IAAI;UAC1BC,UAAU,EACRnB,eAAe,CAACkB,IAAI,IAAIjM;QAC5B,CAAC;MACH,CAAC;IACH,CAAC,CAAC;IACF,OAAO;MACL8H,QAAQ,EAAE,EAAE;MACZC,WAAW,EAAE,KAAK;MAElBV,KAAK;MACL2D,SAAS;MACTC;IACF,CAAC;EACH;;EAEA;EACA,IACEL,WAAW,CAACD,MAAM,KAAK,CAAC,IACxBC,WAAW,CAAC,CAAC,CAAC,CAAC,CAAClE,IAAI,KAAK,MAAM,IAC/B,OAAOkE,WAAW,CAAC,CAAC,CAAC,CAAC,CAAC5D,OAAO,CAACqC,OAAO,KAAK,QAAQ,IACnDuB,WAAW,CAAC,CAAC,CAAC,CAAC,CAAC5D,OAAO,CAACqC,OAAO,CAAC8C,UAAU,CAAC,kBAAkB,CAAC,EAC9D;IACA;IACA,MAAMC,iBAAiB,GACrB9C,WAAW,CAAC6C,UAAU,CAAC,MAAM,CAAC,IAC9B7C,WAAW,CAAC6C,UAAU,CAAC,MAAM,CAAC,IAC9B7C,WAAW,CAAC6C,UAAU,CAAC,UAAU,CAAC;IAEpC,IAAI,CAACC,iBAAiB,EAAE;MACtBlM,QAAQ,CAAC,2BAA2B,EAAE;QACpCqK,KAAK,EACH9E,WAAW,IAAIzF;MACnB,CAAC,CAAC;IACJ;IAEA,OAAO;MACL8H,QAAQ,EAAE,CAACjG,gCAAgC,CAAC,CAAC,EAAE,GAAG+I,WAAW,CAAC;MAC9D7C,WAAW,EAAE8C,kBAAkB;MAC/BC,YAAY;MAEZzD;IACF,CAAC;EACH;;EAEA;EACA,MAAM8D,SAAS,EAAEC,MAAM,CAAC,MAAM,EAAE,OAAO,GAAG,MAAM,GAAG,SAAS,CAAC,GAAG;IAC9Db,KAAK,EACHL,oBAAoB,IAAIlK;EAC5B,CAAC;;EAED;EACA,IAAI+K,eAAe,CAACrE,IAAI,KAAK,QAAQ,IAAIqE,eAAe,CAAC1G,UAAU,EAAE;IACnE,MAAM;MAAEQ,cAAc;MAAEP;IAAW,CAAC,GAAGyG,eAAe,CAAC1G,UAAU;IACjE,MAAM;MAAEE;IAAY,CAAC,GAAG7B,qBAAqB,CAAC4B,UAAU,CAAC;IACzD,MAAM+G,UAAU,GAAG5I,yBAAyB,CAAC8B,WAAW,CAAC;IACzD4G,SAAS,CAACvG,kBAAkB,GAC1BC,cAAc,CAACH,IAAI,IAAIzE,+CAA+C;IACxE,IAAIsE,WAAW,EAAE;MACf4G,SAAS,CAACrG,uBAAuB,GAC/BP,WAAW,IAAItE,+CAA+C;IAClE;IACAkL,SAAS,CAACG,iBAAiB,GAAG,CAC5BD,UAAU,GAAG/G,UAAU,GAAG,aAAa,KACpCtE,0DAA0D;IAC/DmL,SAAS,CAACI,WAAW,GAAG,CACtBF,UAAU,GAAGxG,cAAc,CAACH,IAAI,GAAG,aAAa,KAC7C1E,0DAA0D;IAC/D,IAAIqL,UAAU,IAAIxG,cAAc,CAAC2G,OAAO,EAAE;MACxCL,SAAS,CAACM,cAAc,GACtB5G,cAAc,CAAC2G,OAAO,IAAIxL,0DAA0D;IACxF;IACA0L,MAAM,CAACC,MAAM,CACXR,SAAS,EACTjI,iCAAiC,CAAC6H,eAAe,CAAC1G,UAAU,CAC9D,CAAC;EACH;EAEAnE,QAAQ,CAAC,qBAAqB,EAAE;IAC9B,GAAGiL,SAAS;IACZxG,kBAAkB,EAChB,YAAY,IAAI3E,0DAA0D;IAC5E,IAAI,UAAU,KAAK,KAAK,IAAI;MAC1B4L,UAAU,EACRnG,WAAW,IAAIzF,0DAA0D;MAC3E,IAAI+K,eAAe,CAACrE,IAAI,KAAK,QAAQ,IAAI;QACvCmF,YAAY,EACVd,eAAe,CAACe,MAAM,IAAI9L;MAC9B,CAAC,CAAC;MACF,IAAI+K,eAAe,CAACgB,UAAU,IAAI;QAChCC,iBAAiB,EACfjB,eAAe,CAACgB,UAAU,IAAI/L;MAClC,CAAC,CAAC;MACF,IAAI+K,eAAe,CAACkB,IAAI,IAAI;QAC1BC,UAAU,EACRnB,eAAe,CAACkB,IAAI,IAAIjM;MAC5B,CAAC;IACH,CAAC;EACH,CAAC,CAAC;;EAEF;EACA,MAAMqM,eAAe,GACnBzB,WAAW,CAACD,MAAM,GAAG,CAAC,IACtBC,WAAW,CAAC,CAAC,CAAC,IACd1I,wBAAwB,CAAC0I,WAAW,CAAC,CAAC,CAAC,CAAC;EAE1C,OAAO;IACL9C,QAAQ,EACN+C,kBAAkB,IAClBD,WAAW,CAAC0B,KAAK,CAACnK,2BAA2B,CAAC,IAC9CkK,eAAe,GACXzB,WAAW,GACX,CAAC/I,gCAAgC,CAAC,CAAC,EAAE,GAAG+I,WAAW,CAAC;IAC1D7C,WAAW,EAAE8C,kBAAkB;IAC/BC,YAAY;IACZzD,KAAK;IACLjC,MAAM;IACNqC,UAAU;IACVuD,SAAS;IACTC;EACF,CAAC;AACH;AAEA,eAAeC,0BAA0BA,CACvCzF,WAAW,EAAE,MAAM,EACnB5B,IAAI,EAAE,MAAM,EACZG,UAAU,EAAE7E,YAAY,EACxB2E,OAAO,EAAEP,uBAAuB,EAChCQ,oBAAoB,EAAEzF,iBAAiB,EAAE,EACzCqL,kBAAkB,EAAErL,iBAAiB,EAAE,EACvCiO,oBAA8B,CAAT,EAAE,OAAO,EAC9BtI,UAAyB,CAAd,EAAElE,YAAY,EACzB0I,IAAa,CAAR,EAAE,MAAM,CACd,EAAEvE,OAAO,CAACV,kBAAkB,CAAC,CAAC;EAC7B,MAAMC,OAAO,GAAG3E,UAAU,CAAC2G,WAAW,EAAE3B,OAAO,CAAC8C,OAAO,CAACwD,QAAQ,CAAC;;EAEjE;EACA,IAAI3G,OAAO,CAACiD,IAAI,KAAK,QAAQ,IAAIjD,OAAO,CAAC+I,aAAa,KAAK,KAAK,EAAE;IAChEzJ,gBAAgB,CAAC0C,WAAW,CAAC;EAC/B;;EAEA;EACA;EACA,IAAIhC,OAAO,CAAC+I,aAAa,KAAK,KAAK,EAAE;IACnC,OAAO;MACL1E,QAAQ,EAAE,CACR9F,iBAAiB,CAAC;QAChBqH,OAAO,EAAEhH,kBAAkB,CAAC;UAC1BiH,WAAW,EAAE,IAAI7D,WAAW,EAAE;UAC9B1B;QACF,CAAC;MACH,CAAC,CAAC,EACF/B,iBAAiB,CAAC;QAChBqH,OAAO,EAAE,2FAA2F5D,WAAW;MACjH,CAAC,CAAC,CACH;MACDsC,WAAW,EAAE,KAAK;MAClBtE;IACF,CAAC;EACH;EAEA,IAAI;IACF,QAAQA,OAAO,CAACiD,IAAI;MAClB,KAAK,WAAW;QAAE;UAChB,OAAO,IAAIxC,OAAO,CAACV,kBAAkB,CAAC,CAACiJ,OAAO,IAAI;YAChD,IAAIC,aAAa,GAAG,KAAK;YACzB,MAAMC,MAAM,GAAGA,CACbC,MAAe,CAAR,EAAE,MAAM,EACfhG,OAMC,CANO,EAAE;cACRiG,OAAO,CAAC,EAAEnM,oBAAoB;cAC9BqH,WAAW,CAAC,EAAE,OAAO;cACrB+E,YAAY,CAAC,EAAE,MAAM,EAAE;cACvB9B,SAAS,CAAC,EAAE,MAAM;cAClBC,eAAe,CAAC,EAAE,OAAO;YAC3B,CAAC,KACE;cACHyB,aAAa,GAAG,IAAI;cACpB;cACA,IAAI9F,OAAO,EAAEiG,OAAO,KAAK,MAAM,EAAE;gBAC/B,KAAKJ,OAAO,CAAC;kBACX3E,QAAQ,EAAE,EAAE;kBACZC,WAAW,EAAE,KAAK;kBAClBtE,OAAO;kBACPuH,SAAS,EAAEpE,OAAO,EAAEoE,SAAS;kBAC7BC,eAAe,EAAErE,OAAO,EAAEqE;gBAC5B,CAAC,CAAC;gBACF;cACF;;cAEA;cACA,MAAM6B,YAAY,GAAG,CAAClG,OAAO,EAAEkG,YAAY,IAAI,EAAE,EAAEC,GAAG,CACpD,CAAC1D,OAAO,EAAE,MAAM,KAAKrH,iBAAiB,CAAC;gBAAEqH,OAAO;gBAAEtD,MAAM,EAAE;cAAK,CAAC,CAClE,CAAC;;cAED;cACA;cACA;cACA;cACA;cACA;cACA;cACA;cACA;cACA;cACA,MAAMiH,cAAc,GAClBzL,sBAAsB,CAAC,CAAC,IACxB,OAAOqL,MAAM,KAAK,QAAQ,IAC1BA,MAAM,CAACK,QAAQ,CAAC,YAAY,CAAC;cAE/B,KAAKR,OAAO,CAAC;gBACX3E,QAAQ,EACNlB,OAAO,EAAEiG,OAAO,KAAK,QAAQ,GACzBG,cAAc,GACZF,YAAY,GACZ,CACElL,yBAAyB,CACvBsL,kBAAkB,CAACzJ,OAAO,EAAEI,IAAI,CAClC,CAAC,EACDjC,yBAAyB,CACvB,yBAAyBgL,MAAM,yBACjC,CAAC,EACD,GAAGE,YAAY,CAChB,GACH,CACE9K,iBAAiB,CAAC;kBAChBqH,OAAO,EAAEhH,kBAAkB,CAAC;oBAC1BiH,WAAW,EAAE4D,kBAAkB,CAACzJ,OAAO,EAAEI,IAAI,CAAC;oBAC9CE;kBACF,CAAC;gBACH,CAAC,CAAC,EACF6I,MAAM,GACF5K,iBAAiB,CAAC;kBAChBqH,OAAO,EAAE,yBAAyBuD,MAAM;gBAC1C,CAAC,CAAC,GACF5K,iBAAiB,CAAC;kBAChBqH,OAAO,EAAE,yBAAyBnK,kBAAkB;gBACtD,CAAC,CAAC,EACN,GAAG4N,YAAY,CAChB;gBACP/E,WAAW,EAAEnB,OAAO,EAAEmB,WAAW,IAAI,KAAK;gBAC1CtE,OAAO;gBACPuH,SAAS,EAAEpE,OAAO,EAAEoE,SAAS;gBAC7BC,eAAe,EAAErE,OAAO,EAAEqE;cAC5B,CAAC,CAAC;YACJ,CAAC;YAED,KAAKxH,OAAO,CACT0J,IAAI,CAAC,CAAC,CACNC,IAAI,CAACC,GAAG,IAAIA,GAAG,CAACC,IAAI,CAACX,MAAM,EAAE;cAAE,GAAG7I,OAAO;cAAEG;YAAW,CAAC,EAAEJ,IAAI,CAAC,CAAC,CAC/DuJ,IAAI,CAACzE,GAAG,IAAI;cACX,IAAIA,GAAG,IAAI,IAAI,EAAE;cACjB,IAAI7E,OAAO,CAAC8C,OAAO,CAAC2G,uBAAuB,EAAE;gBAC3C,KAAKd,OAAO,CAAC;kBACX3E,QAAQ,EAAE,EAAE;kBACZC,WAAW,EAAE,KAAK;kBAClBtE;gBACF,CAAC,CAAC;gBACF;cACF;cACA;cACA;cACA;cACA;cACA;cACA;cACA;cACA,IAAIiJ,aAAa,EAAE;cACnB1I,UAAU,CAAC;gBACT2E,GAAG;gBACHE,qBAAqB,EAAE,IAAI;gBAC3BE,WAAW,EAAE,KAAK;gBAClByE,iBAAiB,EAAE,IAAI;gBACvBC,WAAW,EAAEhK,OAAO,CAACiK,SAAS,KAAK;cACrC,CAAC,CAAC;YACJ,CAAC,CAAC,CACDhG,KAAK,CAACiG,CAAC,IAAI;cACV;cACA;cACA;cACAjM,QAAQ,CAACiM,CAAC,CAAC;cACX,IAAIjB,aAAa,EAAE;cACnBA,aAAa,GAAG,IAAI;cACpB1I,UAAU,CAAC;gBACT2E,GAAG,EAAE,IAAI;gBACTE,qBAAqB,EAAE,KAAK;gBAC5B+E,aAAa,EAAE;cACjB,CAAC,CAAC;cACF,KAAKnB,OAAO,CAAC;gBAAE3E,QAAQ,EAAE,EAAE;gBAAEC,WAAW,EAAE,KAAK;gBAAEtE;cAAQ,CAAC,CAAC;YAC7D,CAAC,CAAC;UACN,CAAC,CAAC;QACJ;MACA,KAAK,OAAO;QAAE;UACZ,MAAMoK,WAAW,GAAGpK,OAAO,CAACqK,WAAW,IAAIjK,IAAI,CAAC0F,IAAI,CAAC,CAAC,GAAG,KAAK,GAAG1F,IAAI;UACrE,MAAMkK,WAAW,GAAG/L,iBAAiB,CAAC;YACpCqH,OAAO,EAAEhH,kBAAkB,CAAC;cAC1BiH,WAAW,EAAE4D,kBAAkB,CAACzJ,OAAO,EAAEoK,WAAW,CAAC;cACrD9J;YACF,CAAC;UACH,CAAC,CAAC;UAEF,IAAI;YACF,MAAMiK,sBAAsB,GAAGnM,gCAAgC,CAAC,CAAC;YACjE,MAAMwL,GAAG,GAAG,MAAM5J,OAAO,CAAC0J,IAAI,CAAC,CAAC;YAChC,MAAMP,MAAM,GAAG,MAAMS,GAAG,CAACC,IAAI,CAACzJ,IAAI,EAAEC,OAAO,CAAC;YAE5C,IAAI8I,MAAM,CAAClG,IAAI,KAAK,MAAM,EAAE;cAC1B,OAAO;gBACLoB,QAAQ,EAAE,EAAE;gBACZC,WAAW,EAAE,KAAK;gBAClBtE;cACF,CAAC;YACH;;YAEA;YACA,IAAImJ,MAAM,CAAClG,IAAI,KAAK,SAAS,EAAE;cAC7B;cACA;cACA,MAAMuH,oBAAoB,GAAG,CAC3BD,sBAAsB,EACtBD,WAAW,EACX,IAAInB,MAAM,CAACsB,WAAW,GAClB,CACElM,iBAAiB,CAAC;gBAChBqH,OAAO,EAAE,yBAAyBuD,MAAM,CAACsB,WAAW,yBAAyB;gBAC7E;gBACA;gBACA;gBACA;gBACA;gBACA3F,SAAS,EAAE,IAAIpC,IAAI,CAACA,IAAI,CAACC,GAAG,CAAC,CAAC,GAAG,GAAG,CAAC,CAACoC,WAAW,CAAC;cACpD,CAAC,CAAC,CACH,GACD,EAAE,CAAC,CACR;cACD,MAAM2F,iCAAiC,GAAG;gBACxC,GAAGvB,MAAM,CAACwB,gBAAgB;gBAC1BC,cAAc,EAAE,CACd,IAAIzB,MAAM,CAACwB,gBAAgB,CAACC,cAAc,IAAI,EAAE,CAAC,EACjD,GAAGJ,oBAAoB;cAE3B,CAAC;cACD;cACA;cACA;cACA;cACA5N,sBAAsB,CAAC,CAAC;cACxB,OAAO;gBACLyH,QAAQ,EAAE1H,wBAAwB,CAChC+N,iCACF,CAAC;gBACDpG,WAAW,EAAE,KAAK;gBAClBtE;cACF,CAAC;YACH;;YAEA;YACA,OAAO;cACLqE,QAAQ,EAAE,CACRiG,WAAW,EACXnM,yBAAyB,CACvB,yBAAyBgL,MAAM,CAAChH,KAAK,yBACvC,CAAC,CACF;cACDmC,WAAW,EAAE,KAAK;cAClBtE,OAAO;cACPgE,UAAU,EAAEmF,MAAM,CAAChH;YACrB,CAAC;UACH,CAAC,CAAC,OAAO+H,CAAC,EAAE;YACVjM,QAAQ,CAACiM,CAAC,CAAC;YACX,OAAO;cACL7F,QAAQ,EAAE,CACRiG,WAAW,EACXnM,yBAAyB,CACvB,yBAAyBiG,MAAM,CAAC8F,CAAC,CAAC,yBACpC,CAAC,CACF;cACD5F,WAAW,EAAE,KAAK;cAClBtE;YACF,CAAC;UACH;QACF;MACA,KAAK,QAAQ;QAAE;UACb,IAAI;YACF;YACA,IAAIA,OAAO,CAACK,OAAO,KAAK,MAAM,EAAE;cAC9B,OAAO,MAAMF,yBAAyB,CACpCH,OAAO,EACPI,IAAI,EACJC,OAAO,EACPC,oBAAoB,EACpBC,UAAU,EACVC,UAAU,IAAIzB,uBAChB,CAAC;YACH;YAEA,OAAO,MAAM8L,gCAAgC,CAC3C7K,OAAO,EACPI,IAAI,EACJC,OAAO,EACPC,oBAAoB,EACpB4F,kBAAkB,EAClBlB,IACF,CAAC;UACH,CAAC,CAAC,OAAOkF,CAAC,EAAE;YACV;YACA,IAAIA,CAAC,YAAY1M,UAAU,EAAE;cAC3B,OAAO;gBACL6G,QAAQ,EAAE,CACR9F,iBAAiB,CAAC;kBAChBqH,OAAO,EAAEhH,kBAAkB,CAAC;oBAC1BiH,WAAW,EAAE4D,kBAAkB,CAACzJ,OAAO,EAAEI,IAAI,CAAC;oBAC9CE;kBACF,CAAC;gBACH,CAAC,CAAC,EACFhC,6BAA6B,CAAC;kBAAEwM,OAAO,EAAE;gBAAM,CAAC,CAAC,CAClD;gBACDxG,WAAW,EAAE,KAAK;gBAClBtE;cACF,CAAC;YACH;YACA,OAAO;cACLqE,QAAQ,EAAE,CACR9F,iBAAiB,CAAC;gBAChBqH,OAAO,EAAEhH,kBAAkB,CAAC;kBAC1BiH,WAAW,EAAE4D,kBAAkB,CAACzJ,OAAO,EAAEI,IAAI,CAAC;kBAC9CE;gBACF,CAAC;cACH,CAAC,CAAC,EACF/B,iBAAiB,CAAC;gBAChBqH,OAAO,EAAE,yBAAyBxB,MAAM,CAAC8F,CAAC,CAAC;cAC7C,CAAC,CAAC,CACH;cACD5F,WAAW,EAAE,KAAK;cAClBtE;YACF,CAAC;UACH;QACF;IACF;EACF,CAAC,CAAC,OAAOkK,CAAC,EAAE;IACV,IAAIA,CAAC,YAAYzM,qBAAqB,EAAE;MACtC,OAAO;QACL4G,QAAQ,EAAE,CACR9F,iBAAiB,CAAC;UAChBqH,OAAO,EAAEhH,kBAAkB,CAAC;YAC1BiH,WAAW,EAAEqE,CAAC,CAAC3G,OAAO;YACtBjD;UACF,CAAC;QACH,CAAC,CAAC,CACH;QACDgE,WAAW,EAAE,KAAK;QAClBtE;MACF,CAAC;IACH;IACA,MAAMkK,CAAC;EACT;AACF;AAEA,SAAST,kBAAkBA,CAACzJ,OAAO,EAAE7E,WAAW,EAAEiF,IAAI,EAAE,MAAM,CAAC,EAAE,MAAM,CAAC;EACtE,OAAO5B,sBAAsB,CAAClD,cAAc,CAAC0E,OAAO,CAAC,EAAEI,IAAI,CAAC;AAC9D;;AAEA;AACA;AACA;AACA;AACA,OAAO,SAAS2K,0BAA0BA,CACxCC,SAAS,EAAE,MAAM,EACjBC,gBAAgB,EAAE,MAAM,GAAG,SAAS,CACrC,EAAE,MAAM,CAAC;EACR;EACA,OAAO,CACL,IAAI7O,mBAAmB,IAAI4O,SAAS,KAAK5O,mBAAmB,GAAG,EAC/D,IAAIC,gBAAgB,IAAI2O,SAAS,KAAK3O,gBAAgB,GAAG,EACzD,mCAAmC,CACpC,CAAC6O,IAAI,CAAC,IAAI,CAAC;AACd;;AAEA;AACA;AACA;AACA,SAASC,iCAAiCA,CACxCnJ,WAAW,EAAE,MAAM,EACnB5B,IAAa,CAAR,EAAE,MAAM,CACd,EAAE,MAAM,CAAC;EACR,OAAO,CACL,IAAIhE,mBAAmB,IAAI4F,WAAW,KAAK5F,mBAAmB,GAAG,EACjE,IAAIC,gBAAgB,KAAK2F,WAAW,KAAK3F,gBAAgB,GAAG,EAC5D+D,IAAI,GAAG,iBAAiBA,IAAI,iBAAiB,GAAG,IAAI,CACrD,CACEgL,MAAM,CAACC,OAAO,CAAC,CACfH,IAAI,CAAC,IAAI,CAAC;AACf;;AAEA;AACA;AACA;AACA;AACA;AACA,SAASI,4BAA4BA,CACnCtL,OAAO,EAAE7E,WAAW,GAAGK,aAAa,EACpC4E,IAAa,CAAR,EAAE,MAAM,CACd,EAAE,MAAM,CAAC;EACR;EACA;EACA;EACA;EACA,IAAIJ,OAAO,CAAC+I,aAAa,KAAK,KAAK,EAAE;IACnC,OAAOoC,iCAAiC,CAACnL,OAAO,CAACiB,IAAI,EAAEb,IAAI,CAAC;EAC9D;EACA;EACA,IACEJ,OAAO,CAACsI,UAAU,KAAK,QAAQ,IAC/BtI,OAAO,CAACsI,UAAU,KAAK,QAAQ,IAC/BtI,OAAO,CAACsI,UAAU,KAAK,KAAK,EAC5B;IACA,OAAOyC,0BAA0B,CAAC/K,OAAO,CAACiB,IAAI,EAAEjB,OAAO,CAACuL,eAAe,CAAC;EAC1E;EACA,OAAOJ,iCAAiC,CAACnL,OAAO,CAACiB,IAAI,EAAEb,IAAI,CAAC;AAC9D;AAEA,OAAO,eAAeoL,yBAAyBA,CAC7CxJ,WAAW,EAAE,MAAM,EACnB5B,IAAI,EAAE,MAAM,EACZuG,QAAQ,EAAEzL,OAAO,EAAE,EACnBmF,OAAO,EAAE1E,cAAc,EACvBuK,kBAAkB,EAAErL,iBAAiB,EAAE,GAAG,EAAE,CAC7C,EAAE4F,OAAO,CAACV,kBAAkB,CAAC,CAAC;EAC7B,MAAMC,OAAO,GAAG5E,WAAW,CAAC4G,WAAW,EAAE2E,QAAQ,CAAC;EAClD,IAAI,CAAC3G,OAAO,EAAE;IACZ,MAAM,IAAIvC,qBAAqB,CAAC,oBAAoBuE,WAAW,EAAE,CAAC;EACpE;EACA,IAAIhC,OAAO,CAACiD,IAAI,KAAK,QAAQ,EAAE;IAC7B,MAAM,IAAIkB,KAAK,CACb,cAAcnE,OAAO,CAACiD,IAAI,6CAA6CjB,WAAW,qCACpF,CAAC;EACH;EACA,OAAO6I,gCAAgC,CACrC7K,OAAO,EACPI,IAAI,EACJC,OAAO,EACP,EAAE,EACF6F,kBACF,CAAC;AACH;AAEA,eAAe2E,gCAAgCA,CAC7C7K,OAAO,EAAE7E,WAAW,GAAGK,aAAa,EACpC4E,IAAI,EAAE,MAAM,EACZC,OAAO,EAAE1E,cAAc,EACvB2E,oBAAoB,EAAEzF,iBAAiB,EAAE,GAAG,EAAE,EAC9CqL,kBAAkB,EAAErL,iBAAiB,EAAE,GAAG,EAAE,EAC5CmK,IAAa,CAAR,EAAE,MAAM,CACd,EAAEvE,OAAO,CAACV,kBAAkB,CAAC,CAAC;EAC7B;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA,IACEnF,OAAO,CAAC,kBAAkB,CAAC,IAC3B2C,WAAW,CAACkO,OAAO,CAACC,GAAG,CAACC,4BAA4B,CAAC,IACrD,CAACtL,OAAO,CAACK,OAAO,EAChB;IACA,MAAMkL,QAAQ,GAAGN,4BAA4B,CAACtL,OAAO,EAAEI,IAAI,CAAC;IAC5D,MAAMyL,KAAK,EAAE,MAAM,EAAE,GAAG,CACtB,WAAW7L,OAAO,CAACiB,IAAI,6BAA6B,CACrD;IACD,IAAIjB,OAAO,CAAC8L,WAAW,EAAE;MACvBD,KAAK,CAAC9H,IAAI,CAAC,gBAAgB/D,OAAO,CAAC8L,WAAW,EAAE,CAAC;IACnD;IACA,IAAI9L,OAAO,CAAC+L,SAAS,EAAE;MACrBF,KAAK,CAAC9H,IAAI,CAAC,gBAAgB/D,OAAO,CAAC+L,SAAS,EAAE,CAAC;IACjD;IACA,MAAMC,iBAAiB,GAAGhM,OAAO,CAACqH,YAAY,IAAI,EAAE;IACpD,IAAI2E,iBAAiB,CAAC9E,MAAM,GAAG,CAAC,EAAE;MAChC2E,KAAK,CAAC9H,IAAI,CACR,0DAA0DiI,iBAAiB,CAACd,IAAI,CAAC,IAAI,CAAC,EACxF,CAAC;IACH;IACAW,KAAK,CAAC9H,IAAI,CACR,gEAAgE/D,OAAO,CAACiB,IAAI,gJAC9E,CAAC;IACD,MAAMgL,cAAc,EAAEpR,iBAAiB,EAAE,GAAG,CAC1C;MAAEoI,IAAI,EAAE,MAAM;MAAEiJ,IAAI,EAAEL,KAAK,CAACX,IAAI,CAAC,IAAI;IAAE,CAAC,CACzC;IACD,OAAO;MACL7G,QAAQ,EAAE,CACR9F,iBAAiB,CAAC;QAAEqH,OAAO,EAAEgG,QAAQ;QAAE5G;MAAK,CAAC,CAAC,EAC9CzG,iBAAiB,CAAC;QAAEqH,OAAO,EAAEqG,cAAc;QAAE3J,MAAM,EAAE;MAAK,CAAC,CAAC,CAC7D;MACDgC,WAAW,EAAE,IAAI;MACjBV,KAAK,EAAE5D,OAAO,CAAC4D,KAAK;MACpBjC,MAAM,EAAE3B,OAAO,CAAC2B,MAAM;MACtB3B;IACF,CAAC;EACH;EAEA,MAAMmJ,MAAM,GAAG,MAAMnJ,OAAO,CAACmM,mBAAmB,CAAC/L,IAAI,EAAEC,OAAO,CAAC;;EAE/D;EACA;EACA;EACA,MAAM+L,wBAAwB,GAC5B,CAAClN,wBAAwB,CAAC,OAAO,CAAC,IAAIC,oBAAoB,CAACa,OAAO,CAACqI,MAAM,CAAC;EAC5E,IAAIrI,OAAO,CAACqM,KAAK,IAAID,wBAAwB,EAAE;IAC7C,MAAME,SAAS,GAAGnQ,YAAY,CAAC,CAAC;IAChC6B,kBAAkB,CAChBqC,OAAO,CAACkM,WAAW,EACnBD,SAAS,EACTtM,OAAO,CAACqM,KAAK,EACbrM,OAAO,CAACiB,IAAI,EACZjB,OAAO,CAACiD,IAAI,KAAK,QAAQ,GAAGjD,OAAO,CAACwM,SAAS,GAAGzL,SAClD,CAAC;EACH;;EAEA;EACA;EACA;EACA,MAAM0L,SAAS,GAAGzM,OAAO,CAACqI,MAAM,GAC5B,GAAGrI,OAAO,CAACqI,MAAM,IAAIrI,OAAO,CAACiB,IAAI,EAAE,GACnCjB,OAAO,CAACiB,IAAI;EAChB,MAAMK,YAAY,GAAG6H,MAAM,CACxBiC,MAAM,CAAC,CAACsB,CAAC,CAAC,EAAEA,CAAC,IAAI5R,cAAc,IAAI4R,CAAC,CAACzJ,IAAI,KAAK,MAAM,CAAC,CACrDqG,GAAG,CAACoD,CAAC,IAAIA,CAAC,CAACR,IAAI,CAAC,CAChBhB,IAAI,CAAC,MAAM,CAAC;EACfhP,eAAe,CACb8D,OAAO,CAACiB,IAAI,EACZwL,SAAS,EACTnL,YAAY,EACZnE,eAAe,CAAC,CAAC,EAAEuD,OAAO,IAAI,IAChC,CAAC;EAED,MAAMkL,QAAQ,GAAGN,4BAA4B,CAACtL,OAAO,EAAEI,IAAI,CAAC;EAE5D,MAAMuM,sBAAsB,GAAG7N,oBAAoB,CACjDkB,OAAO,CAACqH,YAAY,IAAI,EAC1B,CAAC;;EAED;EACA,MAAMuF,kBAAkB,EAAE/R,iBAAiB,EAAE,GAC3CqL,kBAAkB,CAACgB,MAAM,GAAG,CAAC,IAAI5G,oBAAoB,CAAC4G,MAAM,GAAG,CAAC,GAC5D,CAAC,GAAGhB,kBAAkB,EAAE,GAAG5F,oBAAoB,EAAE,GAAG6I,MAAM,CAAC,GAC3DA,MAAM;;EAEZ;EACA;EACA;EACA;EACA;EACA,MAAMhD,kBAAkB,GAAG,MAAMpI,OAAO,CACtCV,qBAAqB,CACnB8L,MAAM,CACHiC,MAAM,CAAC,CAACyB,KAAK,CAAC,EAAEA,KAAK,IAAI/R,cAAc,IAAI+R,KAAK,CAAC5J,IAAI,KAAK,MAAM,CAAC,CACjEqG,GAAG,CAACuD,KAAK,IAAIA,KAAK,CAACX,IAAI,CAAC,CACxBhB,IAAI,CAAC,GAAG,CAAC,EACZ7K,OAAO,EACP,IAAI,EACJ,EAAE;EAAE;EACJA,OAAO,CAACgE,QAAQ,EAChB,kBAAkB,EAClB;IAAEyI,kBAAkB,EAAE;EAAK,CAC7B,CACF,CAAC;EAED,MAAMzI,QAAQ,GAAG,CACf9F,iBAAiB,CAAC;IAChBqH,OAAO,EAAEgG,QAAQ;IACjB5G;EACF,CAAC,CAAC,EACFzG,iBAAiB,CAAC;IAChBqH,OAAO,EAAEgH,kBAAkB;IAC3BtK,MAAM,EAAE;EACV,CAAC,CAAC,EACF,GAAG6D,kBAAkB,EACrB/I,uBAAuB,CAAC;IACtB6F,IAAI,EAAE,qBAAqB;IAC3BoE,YAAY,EAAEsF,sBAAsB;IACpC/I,KAAK,EAAE5D,OAAO,CAAC4D;EACjB,CAAC,CAAC,CACH;EAED,OAAO;IACLS,QAAQ;IACRC,WAAW,EAAE,IAAI;IACjB+C,YAAY,EAAEsF,sBAAsB;IACpC/I,KAAK,EAAE5D,OAAO,CAAC4D,KAAK;IACpBjC,MAAM,EAAE3B,OAAO,CAAC2B,MAAM;IACtB3B;EACF,CAAC;AACH","ignoreList":[]} \ No newline at end of file +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["feature","ContentBlockParam","TextBlockParam","randomUUID","setPromptId","builtInCommandNames","Command","CommandBase","findCommand","getCommand","getCommandName","hasCommand","PromptCommand","NO_CONTENT_MESSAGE","SetToolJSXFn","ToolUseContext","AssistantMessage","AttachmentMessage","Message","NormalizedUserMessage","ProgressMessage","UserMessage","addInvokedSkill","getSessionId","COMMAND_MESSAGE_TAG","COMMAND_NAME_TAG","CanUseToolFn","AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS","AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED","logEvent","getDumpPromptsPath","buildPostCompactMessages","resetMicrocompactState","Progress","AgentProgress","runAgent","renderToolUseProgressMessage","CommandResultDisplay","createAbortController","getAgentContext","createAttachmentMessage","getAttachmentMessages","logForDebugging","isEnvTruthy","AbortError","MalformedCommandError","getDisplayPath","extractResultText","prepareForkedCommandContext","getFsImplementation","isFullscreenEnvEnabled","toArray","registerSkillHooks","logError","enqueuePendingNotification","createCommandInputMessage","createSyntheticUserCaveatMessage","createSystemMessage","createUserInterruptionMessage","createUserMessage","formatCommandInputTags","isCompactBoundaryMessage","isSystemLocalCommandMessage","normalizeMessages","prepareUserContent","ModelAlias","parseToolListFromCLI","hasPermissionsToUseTool","isOfficialMarketplaceName","parsePluginIdentifier","isRestrictedToPluginOnly","isSourceAdminTrusted","parseSlashCommand","sleep","recordSkillUsage","logOTelEvent","redactIfDisabled","buildPluginCommandTelemetryFields","getAssistantMessageContentLength","createAgentId","getWorkload","ProcessUserInputBaseResult","ProcessUserInputContext","SlashCommandResult","command","MCP_SETTLE_POLL_MS","MCP_SETTLE_TIMEOUT_MS","executeForkedSlashCommand","args","context","precedingInputBlocks","setToolJSX","canUseTool","Promise","agentId","pluginMarketplace","pluginInfo","repository","marketplace","undefined","command_name","name","invocation_trigger","_PROTO_plugin_name","pluginManifest","_PROTO_marketplace_name","skillContent","modifiedGetAppState","baseAgent","promptMessages","agentDefinition","effort","agentType","getAppState","kairosEnabled","bgAbortController","commandName","spawnTimeWorkload","enqueueResult","value","mode","priority","isMeta","skipSlashCommands","workload","deadline","Date","now","s","mcp","clients","some","c","type","freshTools","options","refreshTools","tools","agentMessages","message","toolUseContext","abortController","isAsync","querySource","model","availableTools","override","push","resultText","catch","err","Error","String","messages","shouldQuery","progressMessages","parentToolUseID","toolUseCounter","createProgressMessage","data","prompt","toolUseID","timestamp","toISOString","uuid","updateProgress","jsx","verbose","shouldHidePromptInput","shouldContinueAnimation","showSpinner","normalizedNew","contentLength","setResponseLength","len","normalizedMsg","content","inputString","trim","looksLikeCommand","test","processSlashCommand","imageContentBlocks","attachmentMessages","isAlreadyProcessing","parsed","errorMessage","parsedArgs","isMcp","sanitizedCommandName","has","commands","isFilePath","stat","input","unknownMessage","promptId","prompt_length","length","newMessages","messageShouldQuery","allowedTools","returnedCommand","nextInput","submitNextInput","getMessagesForSlashCommand","eventData","Record","isOfficial","plugin_repository","plugin_name","version","plugin_version","Object","assign","skill_name","skill_source","source","loadedFrom","skill_loaded_from","kind","skill_kind","startsWith","looksLikeFilePath","isCompactResult","every","_isAlreadyProcessing","userInvocable","resolve","doneWasCalled","onDone","result","display","metaMessages","map","skipTranscript","endsWith","formatCommandInput","load","then","mod","call","isNonInteractiveSession","isLocalJSXCommand","isImmediate","immediate","e","clearLocalJSX","displayArgs","isSensitive","userMessage","syntheticCaveatMessage","slashCommandMessages","displayText","compactionResultWithSlashMessages","compactionResult","messagesToKeep","getMessagesForPromptSlashCommand","toolUse","formatSkillLoadingMetadata","skillName","_progressMessage","join","formatSlashCommandLoadingMetadata","filter","Boolean","formatCommandLoadingMetadata","progressMessage","processPromptSlashCommand","process","env","CLAUDE_CODE_COORDINATOR_MODE","metadata","parts","description","whenToUse","skillAllowedTools","summaryContent","text","getPromptForCommand","hooksAllowedForThisSkill","hooks","sessionId","setAppState","skillRoot","skillPath","b","additionalAllowedTools","mainMessageContent","block","skipSkillDiscovery"],"sources":["processSlashCommand.tsx"],"sourcesContent":["import { feature } from 'bun:bundle'\nimport type {\n  ContentBlockParam,\n  TextBlockParam,\n} from '@anthropic-ai/sdk/resources'\nimport { randomUUID } from 'crypto'\nimport { setPromptId } from 'src/bootstrap/state.js'\nimport {\n  builtInCommandNames,\n  type Command,\n  type CommandBase,\n  findCommand,\n  getCommand,\n  getCommandName,\n  hasCommand,\n  type PromptCommand,\n} from 'src/commands.js'\nimport { NO_CONTENT_MESSAGE } from 'src/constants/messages.js'\nimport type { SetToolJSXFn, ToolUseContext } from 'src/Tool.js'\nimport type {\n  AssistantMessage,\n  AttachmentMessage,\n  Message,\n  NormalizedUserMessage,\n  ProgressMessage,\n  UserMessage,\n} from 'src/types/message.js'\nimport { addInvokedSkill, getSessionId } from '../../bootstrap/state.js'\nimport { COMMAND_MESSAGE_TAG, COMMAND_NAME_TAG } from '../../constants/xml.js'\nimport type { CanUseToolFn } from '../../hooks/useCanUseTool.js'\nimport {\n  type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n  type AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED,\n  logEvent,\n} from '../../services/analytics/index.js'\nimport { getDumpPromptsPath } from '../../services/api/dumpPrompts.js'\nimport { buildPostCompactMessages } from '../../services/compact/compact.js'\nimport { resetMicrocompactState } from '../../services/compact/microCompact.js'\nimport type { Progress as AgentProgress } from '../../tools/AgentTool/AgentTool.js'\nimport { runAgent } from '../../tools/AgentTool/runAgent.js'\nimport { renderToolUseProgressMessage } from '../../tools/AgentTool/UI.js'\nimport type { CommandResultDisplay } from '../../types/command.js'\nimport { createAbortController } from '../abortController.js'\nimport { getAgentContext } from '../agentContext.js'\nimport {\n  createAttachmentMessage,\n  getAttachmentMessages,\n} from '../attachments.js'\nimport { logForDebugging } from '../debug.js'\nimport { isEnvTruthy } from '../envUtils.js'\nimport { AbortError, MalformedCommandError } from '../errors.js'\nimport { getDisplayPath } from '../file.js'\nimport {\n  extractResultText,\n  prepareForkedCommandContext,\n} from '../forkedAgent.js'\nimport { getFsImplementation } from '../fsOperations.js'\nimport { isFullscreenEnvEnabled } from '../fullscreen.js'\nimport { toArray } from '../generators.js'\nimport { registerSkillHooks } from '../hooks/registerSkillHooks.js'\nimport { logError } from '../log.js'\nimport { enqueuePendingNotification } from '../messageQueueManager.js'\nimport {\n  createCommandInputMessage,\n  createSyntheticUserCaveatMessage,\n  createSystemMessage,\n  createUserInterruptionMessage,\n  createUserMessage,\n  formatCommandInputTags,\n  isCompactBoundaryMessage,\n  isSystemLocalCommandMessage,\n  normalizeMessages,\n  prepareUserContent,\n} from '../messages.js'\nimport type { ModelAlias } from '../model/aliases.js'\nimport { parseToolListFromCLI } from '../permissions/permissionSetup.js'\nimport { hasPermissionsToUseTool } from '../permissions/permissions.js'\nimport {\n  isOfficialMarketplaceName,\n  parsePluginIdentifier,\n} from '../plugins/pluginIdentifier.js'\nimport {\n  isRestrictedToPluginOnly,\n  isSourceAdminTrusted,\n} from '../settings/pluginOnlyPolicy.js'\nimport { parseSlashCommand } from '../slashCommandParsing.js'\nimport { sleep } from '../sleep.js'\nimport { recordSkillUsage } from '../suggestions/skillUsageTracking.js'\nimport { logOTelEvent, redactIfDisabled } from '../telemetry/events.js'\nimport { buildPluginCommandTelemetryFields } from '../telemetry/pluginTelemetry.js'\nimport { getAssistantMessageContentLength } from '../tokens.js'\nimport { createAgentId } from '../uuid.js'\nimport { getWorkload } from '../workloadContext.js'\nimport type {\n  ProcessUserInputBaseResult,\n  ProcessUserInputContext,\n} from './processUserInput.js'\n\ntype SlashCommandResult = ProcessUserInputBaseResult & {\n  command: Command\n}\n\n// Poll interval and deadline for MCP settle before launching a background\n// forked subagent. MCP servers typically connect within 1-3s of startup;\n// 10s headroom covers slow SSE handshakes.\nconst MCP_SETTLE_POLL_MS = 200\nconst MCP_SETTLE_TIMEOUT_MS = 10_000\n\n/**\n * Executes a slash command with context: fork in a sub-agent.\n */\nasync function executeForkedSlashCommand(\n  command: CommandBase & PromptCommand,\n  args: string,\n  context: ProcessUserInputContext,\n  precedingInputBlocks: ContentBlockParam[],\n  setToolJSX: SetToolJSXFn,\n  canUseTool: CanUseToolFn,\n): Promise<SlashCommandResult> {\n  const agentId = createAgentId()\n\n  const pluginMarketplace = command.pluginInfo\n    ? parsePluginIdentifier(command.pluginInfo.repository).marketplace\n    : undefined\n  logEvent('tengu_slash_command_forked', {\n    command_name:\n      command.name as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n    invocation_trigger:\n      'user-slash' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n    ...(command.pluginInfo && {\n      _PROTO_plugin_name: command.pluginInfo.pluginManifest\n        .name as AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED,\n      ...(pluginMarketplace && {\n        _PROTO_marketplace_name:\n          pluginMarketplace as AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED,\n      }),\n      ...buildPluginCommandTelemetryFields(command.pluginInfo),\n    }),\n  })\n\n  const { skillContent, modifiedGetAppState, baseAgent, promptMessages } =\n    await prepareForkedCommandContext(command, args, context)\n\n  // Merge skill's effort into the agent definition so runAgent applies it\n  const agentDefinition =\n    command.effort !== undefined\n      ? { ...baseAgent, effort: command.effort }\n      : baseAgent\n\n  logForDebugging(\n    `Executing forked slash command /${command.name} with agent ${agentDefinition.agentType}`,\n  )\n\n  // Assistant mode: fire-and-forget. Launch subagent in background, return\n  // immediately, re-enqueue the result as an isMeta prompt when done.\n  // Without this, N scheduled tasks on startup = N serial (subagent + main\n  // agent turn) cycles blocking user input. With this, N subagents run in\n  // parallel and results trickle into the queue as they finish.\n  //\n  // Gated on kairosEnabled (not CLAUDE_CODE_BRIEF) because the closed loop\n  // depends on assistant-mode invariants: scheduled_tasks.json exists,\n  // the main agent knows to pipe results through SendUserMessage, and\n  // isMeta prompts are hidden. Outside assistant mode, context:fork commands\n  // are user-invoked skills (/commit etc.) that should run synchronously\n  // with the progress UI.\n  if (feature('KAIROS') && (await context.getAppState()).kairosEnabled) {\n    // Standalone abortController — background subagents survive main-thread\n    // ESC (same policy as AgentTool's async path). They're cron-driven; if\n    // killed mid-run they just re-fire on the next schedule.\n    const bgAbortController = createAbortController()\n    const commandName = getCommandName(command)\n\n    // Workload: handlePromptSubmit wraps the entire turn in runWithWorkload\n    // (AsyncLocalStorage). ALS context is captured when this `void` fires\n    // and survives every await inside — isolated from the parent's\n    // continuation. The detached closure's runAgent calls see the cron tag\n    // automatically. We still capture the value here ONLY for the\n    // re-enqueued result prompt below: that second turn runs in a fresh\n    // handlePromptSubmit → fresh runWithWorkload boundary (which always\n    // establishes a new context, even for `undefined`) → so it needs its\n    // own QueuedCommand.workload tag to preserve attribution.\n    const spawnTimeWorkload = getWorkload()\n\n    // Re-enter the queue as a hidden prompt. isMeta: hides from queue\n    // preview + placeholder + transcript. skipSlashCommands: prevents\n    // re-parsing if the result text happens to start with '/'. When\n    // drained, this triggers a main-agent turn that sees the result and\n    // decides whether to SendUserMessage. Propagate workload so that\n    // second turn is also tagged.\n    const enqueueResult = (value: string): void =>\n      enqueuePendingNotification({\n        value,\n        mode: 'prompt',\n        priority: 'later',\n        isMeta: true,\n        skipSlashCommands: true,\n        workload: spawnTimeWorkload,\n      })\n\n    void (async () => {\n      // Wait for MCP servers to settle. Scheduled tasks fire at startup and\n      // all N drain within ~1ms (since we return immediately), capturing\n      // context.options.tools before MCP connects. The sync path\n      // accidentally avoided this — tasks serialized, so task N's drain\n      // happened after task N-1's 30s run, by which time MCP was up.\n      // Poll until no 'pending' clients remain, then refresh.\n      const deadline = Date.now() + MCP_SETTLE_TIMEOUT_MS\n      while (Date.now() < deadline) {\n        const s = context.getAppState()\n        if (!s.mcp.clients.some(c => c.type === 'pending')) break\n        await sleep(MCP_SETTLE_POLL_MS)\n      }\n      const freshTools =\n        context.options.refreshTools?.() ?? context.options.tools\n\n      const agentMessages: Message[] = []\n      for await (const message of runAgent({\n        agentDefinition,\n        promptMessages,\n        toolUseContext: {\n          ...context,\n          getAppState: modifiedGetAppState,\n          abortController: bgAbortController,\n        },\n        canUseTool,\n        isAsync: true,\n        querySource: 'agent:custom',\n        model: command.model as ModelAlias | undefined,\n        availableTools: freshTools,\n        override: { agentId },\n      })) {\n        agentMessages.push(message)\n      }\n      const resultText = extractResultText(agentMessages, 'Command completed')\n      logForDebugging(\n        `Background forked command /${commandName} completed (agent ${agentId})`,\n      )\n      enqueueResult(\n        `<scheduled-task-result command=\"/${commandName}\">\\n${resultText}\\n</scheduled-task-result>`,\n      )\n    })().catch(err => {\n      logError(err)\n      enqueueResult(\n        `<scheduled-task-result command=\"/${commandName}\" status=\"failed\">\\n${err instanceof Error ? err.message : String(err)}\\n</scheduled-task-result>`,\n      )\n    })\n\n    // Nothing to render, nothing to query — the background runner re-enters\n    // the queue on its own schedule.\n    return { messages: [], shouldQuery: false, command }\n  }\n\n  // Collect messages from the forked agent\n  const agentMessages: Message[] = []\n\n  // Build progress messages for the agent progress UI\n  const progressMessages: ProgressMessage<AgentProgress>[] = []\n  const parentToolUseID = `forked-command-${command.name}`\n  let toolUseCounter = 0\n\n  // Helper to create a progress message from an agent message\n  const createProgressMessage = (\n    message: AssistantMessage | NormalizedUserMessage,\n  ): ProgressMessage<AgentProgress> => {\n    toolUseCounter++\n    return {\n      type: 'progress',\n      data: {\n        message,\n        type: 'agent_progress',\n        prompt: skillContent,\n        agentId,\n      },\n      parentToolUseID,\n      toolUseID: `${parentToolUseID}-${toolUseCounter}`,\n      timestamp: new Date().toISOString(),\n      uuid: randomUUID(),\n    }\n  }\n\n  // Helper to update progress display using agent progress UI\n  const updateProgress = (): void => {\n    setToolJSX({\n      jsx: renderToolUseProgressMessage(progressMessages, {\n        tools: context.options.tools,\n        verbose: false,\n      }),\n      shouldHidePromptInput: false,\n      shouldContinueAnimation: true,\n      showSpinner: true,\n    })\n  }\n\n  // Show initial \"Initializing…\" state\n  updateProgress()\n\n  // Run the sub-agent\n  try {\n    for await (const message of runAgent({\n      agentDefinition,\n      promptMessages,\n      toolUseContext: {\n        ...context,\n        getAppState: modifiedGetAppState,\n      },\n      canUseTool,\n      isAsync: false,\n      querySource: 'agent:custom',\n      model: command.model as ModelAlias | undefined,\n      availableTools: context.options.tools,\n    })) {\n      agentMessages.push(message)\n      const normalizedNew = normalizeMessages([message])\n\n      // Add progress message for assistant messages (which contain tool uses)\n      if (message.type === 'assistant') {\n        // Increment token count in spinner for assistant messages\n        const contentLength = getAssistantMessageContentLength(message)\n        if (contentLength > 0) {\n          context.setResponseLength(len => len + contentLength)\n        }\n\n        const normalizedMsg = normalizedNew[0]\n        if (normalizedMsg && normalizedMsg.type === 'assistant') {\n          progressMessages.push(createProgressMessage(message))\n          updateProgress()\n        }\n      }\n\n      // Add progress message for user messages (which contain tool results)\n      if (message.type === 'user') {\n        const normalizedMsg = normalizedNew[0]\n        if (normalizedMsg && normalizedMsg.type === 'user') {\n          progressMessages.push(createProgressMessage(normalizedMsg))\n          updateProgress()\n        }\n      }\n    }\n  } finally {\n    // Clear the progress display\n    setToolJSX(null)\n  }\n\n  let resultText = extractResultText(agentMessages, 'Command completed')\n\n  logForDebugging(\n    `Forked slash command /${command.name} completed with agent ${agentId}`,\n  )\n\n  // Prepend debug log for ant users so it appears inside the command output\n  if (\"external\" === 'ant') {\n    resultText = `[ANT-ONLY] API calls: ${getDisplayPath(getDumpPromptsPath(agentId))}\\n${resultText}`\n  }\n\n  // Return the result as a user message (simulates the agent's output)\n  const messages: UserMessage[] = [\n    createUserMessage({\n      content: prepareUserContent({\n        inputString: `/${getCommandName(command)} ${args}`.trim(),\n        precedingInputBlocks,\n      }),\n    }),\n    createUserMessage({\n      content: `<local-command-stdout>\\n${resultText}\\n</local-command-stdout>`,\n    }),\n  ]\n\n  return {\n    messages,\n    shouldQuery: false,\n    command,\n    resultText,\n  }\n}\n\n/**\n * Determines if a string looks like a valid command name.\n * Valid command names only contain letters, numbers, colons, hyphens, and underscores.\n *\n * @param commandName - The potential command name to check\n * @returns true if it looks like a command name, false if it contains non-command characters\n */\nexport function looksLikeCommand(commandName: string): boolean {\n  // Command names should only contain [a-zA-Z0-9:_-]\n  // If it contains other characters, it's probably a file path or other input\n  return !/[^a-zA-Z0-9:\\-_]/.test(commandName)\n}\n\nexport async function processSlashCommand(\n  inputString: string,\n  precedingInputBlocks: ContentBlockParam[],\n  imageContentBlocks: ContentBlockParam[],\n  attachmentMessages: AttachmentMessage[],\n  context: ProcessUserInputContext,\n  setToolJSX: SetToolJSXFn,\n  uuid?: string,\n  isAlreadyProcessing?: boolean,\n  canUseTool?: CanUseToolFn,\n): Promise<ProcessUserInputBaseResult> {\n  const parsed = parseSlashCommand(inputString)\n  if (!parsed) {\n    logEvent('tengu_input_slash_missing', {})\n    const errorMessage = 'Commands are in the form `/command [args]`'\n    return {\n      messages: [\n        createSyntheticUserCaveatMessage(),\n        ...attachmentMessages,\n        createUserMessage({\n          content: prepareUserContent({\n            inputString: errorMessage,\n            precedingInputBlocks,\n          }),\n        }),\n      ],\n      shouldQuery: false,\n      resultText: errorMessage,\n    }\n  }\n\n  const { commandName, args: parsedArgs, isMcp } = parsed\n\n  const sanitizedCommandName = isMcp\n    ? 'mcp'\n    : !builtInCommandNames().has(commandName)\n      ? 'custom'\n      : commandName\n\n  // Check if it's a real command before processing\n  if (!hasCommand(commandName, context.options.commands)) {\n    // Check if this looks like a command name vs a file path or other input\n    // Also check if it's an actual file path that exists\n    let isFilePath = false\n    try {\n      await getFsImplementation().stat(`/${commandName}`)\n      isFilePath = true\n    } catch {\n      // Not a file path — treat as command name\n    }\n    if (looksLikeCommand(commandName) && !isFilePath) {\n      logEvent('tengu_input_slash_invalid', {\n        input:\n          commandName as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      })\n\n      const unknownMessage = `Unknown skill: ${commandName}`\n      return {\n        messages: [\n          createSyntheticUserCaveatMessage(),\n          ...attachmentMessages,\n          createUserMessage({\n            content: prepareUserContent({\n              inputString: unknownMessage,\n              precedingInputBlocks,\n            }),\n          }),\n          // gh-32591: preserve args so the user can copy/resubmit without\n          // retyping. System warning is UI-only (filtered before API).\n          ...(parsedArgs\n            ? [\n                createSystemMessage(\n                  `Args from unknown skill: ${parsedArgs}`,\n                  'warning',\n                ),\n              ]\n            : []),\n        ],\n        shouldQuery: false,\n        resultText: unknownMessage,\n      }\n    }\n\n    const promptId = randomUUID()\n    setPromptId(promptId)\n    logEvent('tengu_input_prompt', {})\n    // Log user prompt event for OTLP\n    void logOTelEvent('user_prompt', {\n      prompt_length: String(inputString.length),\n      prompt: redactIfDisabled(inputString),\n      'prompt.id': promptId,\n    })\n    return {\n      messages: [\n        createUserMessage({\n          content: prepareUserContent({ inputString, precedingInputBlocks }),\n          uuid: uuid,\n        }),\n        ...attachmentMessages,\n      ],\n      shouldQuery: true,\n    }\n  }\n\n  // Track slash command usage for feature discovery\n\n  const {\n    messages: newMessages,\n    shouldQuery: messageShouldQuery,\n    allowedTools,\n    model,\n    effort,\n    command: returnedCommand,\n    resultText,\n    nextInput,\n    submitNextInput,\n  } = await getMessagesForSlashCommand(\n    commandName,\n    parsedArgs,\n    setToolJSX,\n    context,\n    precedingInputBlocks,\n    imageContentBlocks,\n    isAlreadyProcessing,\n    canUseTool,\n    uuid,\n  )\n\n  // Local slash commands that skip messages\n  if (newMessages.length === 0) {\n    const eventData: Record<string, boolean | number | undefined> = {\n      input:\n        sanitizedCommandName as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n    }\n\n    // Add plugin metadata if this is a plugin command\n    if (returnedCommand.type === 'prompt' && returnedCommand.pluginInfo) {\n      const { pluginManifest, repository } = returnedCommand.pluginInfo\n      const { marketplace } = parsePluginIdentifier(repository)\n      const isOfficial = isOfficialMarketplaceName(marketplace)\n      // _PROTO_* routes to PII-tagged plugin_name/marketplace_name BQ columns\n      // (unredacted, all users); plugin_name/plugin_repository stay in\n      // additional_metadata as redacted variants for general-access dashboards.\n      eventData._PROTO_plugin_name =\n        pluginManifest.name as AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED\n      if (marketplace) {\n        eventData._PROTO_marketplace_name =\n          marketplace as AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED\n      }\n      eventData.plugin_repository = (\n        isOfficial ? repository : 'third-party'\n      ) as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS\n      eventData.plugin_name = (\n        isOfficial ? pluginManifest.name : 'third-party'\n      ) as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS\n      if (isOfficial && pluginManifest.version) {\n        eventData.plugin_version =\n          pluginManifest.version as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS\n      }\n      Object.assign(\n        eventData,\n        buildPluginCommandTelemetryFields(returnedCommand.pluginInfo),\n      )\n    }\n\n    logEvent('tengu_input_command', {\n      ...eventData,\n      invocation_trigger:\n        'user-slash' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      ...(\"external\" === 'ant' && {\n        skill_name:\n          commandName as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        ...(returnedCommand.type === 'prompt' && {\n          skill_source:\n            returnedCommand.source as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        }),\n        ...(returnedCommand.loadedFrom && {\n          skill_loaded_from:\n            returnedCommand.loadedFrom as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        }),\n        ...(returnedCommand.kind && {\n          skill_kind:\n            returnedCommand.kind as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n        }),\n      }),\n    })\n    return {\n      messages: [],\n      shouldQuery: false,\n\n      model,\n      nextInput,\n      submitNextInput,\n    }\n  }\n\n  // For invalid commands, preserve both the user message and error\n  if (\n    newMessages.length === 2 &&\n    newMessages[1]!.type === 'user' &&\n    typeof newMessages[1]!.message.content === 'string' &&\n    newMessages[1]!.message.content.startsWith('Unknown command:')\n  ) {\n    // Don't log as invalid if it looks like a common file path\n    const looksLikeFilePath =\n      inputString.startsWith('/var') ||\n      inputString.startsWith('/tmp') ||\n      inputString.startsWith('/private')\n\n    if (!looksLikeFilePath) {\n      logEvent('tengu_input_slash_invalid', {\n        input:\n          commandName as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      })\n    }\n\n    return {\n      messages: [createSyntheticUserCaveatMessage(), ...newMessages],\n      shouldQuery: messageShouldQuery,\n      allowedTools,\n\n      model,\n    }\n  }\n\n  // A valid command\n  const eventData: Record<string, boolean | number | undefined> = {\n    input:\n      sanitizedCommandName as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n  }\n\n  // Add plugin metadata if this is a plugin command\n  if (returnedCommand.type === 'prompt' && returnedCommand.pluginInfo) {\n    const { pluginManifest, repository } = returnedCommand.pluginInfo\n    const { marketplace } = parsePluginIdentifier(repository)\n    const isOfficial = isOfficialMarketplaceName(marketplace)\n    eventData._PROTO_plugin_name =\n      pluginManifest.name as AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED\n    if (marketplace) {\n      eventData._PROTO_marketplace_name =\n        marketplace as AnalyticsMetadata_I_VERIFIED_THIS_IS_PII_TAGGED\n    }\n    eventData.plugin_repository = (\n      isOfficial ? repository : 'third-party'\n    ) as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS\n    eventData.plugin_name = (\n      isOfficial ? pluginManifest.name : 'third-party'\n    ) as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS\n    if (isOfficial && pluginManifest.version) {\n      eventData.plugin_version =\n        pluginManifest.version as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS\n    }\n    Object.assign(\n      eventData,\n      buildPluginCommandTelemetryFields(returnedCommand.pluginInfo),\n    )\n  }\n\n  logEvent('tengu_input_command', {\n    ...eventData,\n    invocation_trigger:\n      'user-slash' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n    ...(\"external\" === 'ant' && {\n      skill_name:\n        commandName as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      ...(returnedCommand.type === 'prompt' && {\n        skill_source:\n          returnedCommand.source as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      }),\n      ...(returnedCommand.loadedFrom && {\n        skill_loaded_from:\n          returnedCommand.loadedFrom as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      }),\n      ...(returnedCommand.kind && {\n        skill_kind:\n          returnedCommand.kind as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,\n      }),\n    }),\n  })\n\n  // Check if this is a compact result which handle their own synthetic caveat message ordering\n  const isCompactResult =\n    newMessages.length > 0 &&\n    newMessages[0] &&\n    isCompactBoundaryMessage(newMessages[0])\n\n  return {\n    messages:\n      messageShouldQuery ||\n      newMessages.every(isSystemLocalCommandMessage) ||\n      isCompactResult\n        ? newMessages\n        : [createSyntheticUserCaveatMessage(), ...newMessages],\n    shouldQuery: messageShouldQuery,\n    allowedTools,\n    model,\n    effort,\n    resultText,\n    nextInput,\n    submitNextInput,\n  }\n}\n\nasync function getMessagesForSlashCommand(\n  commandName: string,\n  args: string,\n  setToolJSX: SetToolJSXFn,\n  context: ProcessUserInputContext,\n  precedingInputBlocks: ContentBlockParam[],\n  imageContentBlocks: ContentBlockParam[],\n  _isAlreadyProcessing?: boolean,\n  canUseTool?: CanUseToolFn,\n  uuid?: string,\n): Promise<SlashCommandResult> {\n  const command = getCommand(commandName, context.options.commands)\n\n  // Track skill usage for ranking (only for prompt commands that are user-invocable)\n  if (command.type === 'prompt' && command.userInvocable !== false) {\n    recordSkillUsage(commandName)\n  }\n\n  // Check if the command is user-invocable\n  // Skills with userInvocable === false can only be invoked by the model via SkillTool\n  if (command.userInvocable === false) {\n    return {\n      messages: [\n        createUserMessage({\n          content: prepareUserContent({\n            inputString: `/${commandName}`,\n            precedingInputBlocks,\n          }),\n        }),\n        createUserMessage({\n          content: `This skill can only be invoked by Claude, not directly by users. Ask Claude to use the \"${commandName}\" skill for you.`,\n        }),\n      ],\n      shouldQuery: false,\n      command,\n    }\n  }\n\n  try {\n    switch (command.type) {\n      case 'local-jsx': {\n        return new Promise<SlashCommandResult>(resolve => {\n          let doneWasCalled = false\n          const onDone = (\n            result?: string,\n            options?: {\n              display?: CommandResultDisplay\n              shouldQuery?: boolean\n              metaMessages?: string[]\n              nextInput?: string\n              submitNextInput?: boolean\n            },\n          ) => {\n            doneWasCalled = true\n            // If display is 'skip', don't add any messages to the conversation\n            if (options?.display === 'skip') {\n              void resolve({\n                messages: [],\n                shouldQuery: false,\n                command,\n                nextInput: options?.nextInput,\n                submitNextInput: options?.submitNextInput,\n              })\n              return\n            }\n\n            // Meta messages are model-visible but hidden from the user\n            const metaMessages = (options?.metaMessages ?? []).map(\n              (content: string) => createUserMessage({ content, isMeta: true }),\n            )\n\n            // In fullscreen the command just showed as a centered modal\n            // pane — the transient notification is enough feedback. The\n            // \"❯ /config\" + \"⎿ dismissed\" transcript entries are\n            // type:system subtype:local_command (user-visible but NOT sent\n            // to the model), so skipping them doesn't affect model context.\n            // Outside fullscreen keep them so scrollback shows what ran.\n            // Only skip \"<Name> dismissed\" modal-close notifications —\n            // commands that early-exit before showing a modal (/ultraplan\n            // usage, /rename, /proactive) use display:system for actual\n            // output that must reach the transcript.\n            const skipTranscript =\n              isFullscreenEnvEnabled() &&\n              typeof result === 'string' &&\n              result.endsWith(' dismissed')\n\n            void resolve({\n              messages:\n                options?.display === 'system'\n                  ? skipTranscript\n                    ? metaMessages\n                    : [\n                        createCommandInputMessage(\n                          formatCommandInput(command, args),\n                        ),\n                        createCommandInputMessage(\n                          `<local-command-stdout>${result}</local-command-stdout>`,\n                        ),\n                        ...metaMessages,\n                      ]\n                  : [\n                      createUserMessage({\n                        content: prepareUserContent({\n                          inputString: formatCommandInput(command, args),\n                          precedingInputBlocks,\n                        }),\n                      }),\n                      result\n                        ? createUserMessage({\n                            content: `<local-command-stdout>${result}</local-command-stdout>`,\n                          })\n                        : createUserMessage({\n                            content: `<local-command-stdout>${NO_CONTENT_MESSAGE}</local-command-stdout>`,\n                          }),\n                      ...metaMessages,\n                    ],\n              shouldQuery: options?.shouldQuery ?? false,\n              command,\n              nextInput: options?.nextInput,\n              submitNextInput: options?.submitNextInput,\n            })\n          }\n\n          void command\n            .load()\n            .then(mod => mod.call(onDone, { ...context, canUseTool }, args))\n            .then(jsx => {\n              if (jsx == null) return\n              if (context.options.isNonInteractiveSession) {\n                void resolve({\n                  messages: [],\n                  shouldQuery: false,\n                  command,\n                })\n                return\n              }\n              // Guard: if onDone fired during mod.call() (early-exit path\n              // that calls onDone then returns JSX), skip setToolJSX. This\n              // chain is fire-and-forget — the outer Promise resolves when\n              // onDone is called, so executeUserInput may have already run\n              // its setToolJSX({clearLocalJSX: true}) before we get here.\n              // Setting isLocalJSXCommand after clear leaves it stuck true,\n              // blocking useQueueProcessor and TextInput focus.\n              if (doneWasCalled) return\n              setToolJSX({\n                jsx,\n                shouldHidePromptInput: true,\n                showSpinner: false,\n                isLocalJSXCommand: true,\n                isImmediate: command.immediate === true,\n              })\n            })\n            .catch(e => {\n              // If load()/call() throws and onDone never fired, the outer\n              // Promise hangs forever, leaving queryGuard stuck in\n              // 'dispatching' and deadlocking the queue processor.\n              logError(e)\n              if (doneWasCalled) return\n              doneWasCalled = true\n              setToolJSX({\n                jsx: null,\n                shouldHidePromptInput: false,\n                clearLocalJSX: true,\n              })\n              void resolve({ messages: [], shouldQuery: false, command })\n            })\n        })\n      }\n      case 'local': {\n        const displayArgs = command.isSensitive && args.trim() ? '***' : args\n        const userMessage = createUserMessage({\n          content: prepareUserContent({\n            inputString: formatCommandInput(command, displayArgs),\n            precedingInputBlocks,\n          }),\n        })\n\n        try {\n          const syntheticCaveatMessage = createSyntheticUserCaveatMessage()\n          const mod = await command.load()\n          const result = await mod.call(args, context)\n\n          if (result.type === 'skip') {\n            return {\n              messages: [],\n              shouldQuery: false,\n              command,\n            }\n          }\n\n          // Use discriminated union to handle different result types\n          if (result.type === 'compact') {\n            // Append slash command messages to messagesToKeep so that\n            // attachments and hookResults come after user messages\n            const slashCommandMessages = [\n              syntheticCaveatMessage,\n              userMessage,\n              ...(result.displayText\n                ? [\n                    createUserMessage({\n                      content: `<local-command-stdout>${result.displayText}</local-command-stdout>`,\n                      // --resume looks at latest timestamp message to determine which message to resume from\n                      // This is a perf optimization to avoid having to recaculcate the leaf node every time\n                      // Since we're creating a bunch of synthetic messages for compact, it's important to set\n                      // the timestamp of the last message to be slightly after the current time\n                      // This is mostly important for sdk / -p mode\n                      timestamp: new Date(Date.now() + 100).toISOString(),\n                    }),\n                  ]\n                : []),\n            ]\n            const compactionResultWithSlashMessages = {\n              ...result.compactionResult,\n              messagesToKeep: [\n                ...(result.compactionResult.messagesToKeep ?? []),\n                ...slashCommandMessages,\n              ],\n            }\n            // Reset microcompact state since full compact replaces all\n            // messages — old tool IDs are no longer relevant. Budget state\n            // (on toolUseContext) needs no reset: stale entries are inert\n            // (UUIDs never repeat, so they're never looked up).\n            resetMicrocompactState()\n            return {\n              messages: buildPostCompactMessages(\n                compactionResultWithSlashMessages,\n              ),\n              shouldQuery: false,\n              command,\n            }\n          }\n\n          // Text result — use system message so it doesn't render as a user bubble\n          return {\n            messages: [\n              userMessage,\n              createCommandInputMessage(\n                `<local-command-stdout>${result.value}</local-command-stdout>`,\n              ),\n            ],\n            shouldQuery: false,\n            command,\n            resultText: result.value,\n          }\n        } catch (e) {\n          logError(e)\n          return {\n            messages: [\n              userMessage,\n              createCommandInputMessage(\n                `<local-command-stderr>${String(e)}</local-command-stderr>`,\n              ),\n            ],\n            shouldQuery: false,\n            command,\n          }\n        }\n      }\n      case 'prompt': {\n        try {\n          // Check if command should run as forked sub-agent\n          if (command.context === 'fork') {\n            return await executeForkedSlashCommand(\n              command,\n              args,\n              context,\n              precedingInputBlocks,\n              setToolJSX,\n              canUseTool ?? hasPermissionsToUseTool,\n            )\n          }\n\n          return await getMessagesForPromptSlashCommand(\n            command,\n            args,\n            context,\n            precedingInputBlocks,\n            imageContentBlocks,\n            uuid,\n          )\n        } catch (e) {\n          // Handle abort errors specially to show proper \"Interrupted\" message\n          if (e instanceof AbortError) {\n            return {\n              messages: [\n                createUserMessage({\n                  content: prepareUserContent({\n                    inputString: formatCommandInput(command, args),\n                    precedingInputBlocks,\n                  }),\n                }),\n                createUserInterruptionMessage({ toolUse: false }),\n              ],\n              shouldQuery: false,\n              command,\n            }\n          }\n          return {\n            messages: [\n              createUserMessage({\n                content: prepareUserContent({\n                  inputString: formatCommandInput(command, args),\n                  precedingInputBlocks,\n                }),\n              }),\n              createUserMessage({\n                content: `<local-command-stderr>${String(e)}</local-command-stderr>`,\n              }),\n            ],\n            shouldQuery: false,\n            command,\n          }\n        }\n      }\n    }\n  } catch (e) {\n    if (e instanceof MalformedCommandError) {\n      return {\n        messages: [\n          createUserMessage({\n            content: prepareUserContent({\n              inputString: e.message,\n              precedingInputBlocks,\n            }),\n          }),\n        ],\n        shouldQuery: false,\n        command,\n      }\n    }\n    throw e\n  }\n}\n\nfunction formatCommandInput(command: CommandBase, args: string): string {\n  return formatCommandInputTags(getCommandName(command), args)\n}\n\n/**\n * Formats the metadata for a skill loading message.\n * Used by the Skill tool and for subagent skill preloading.\n */\nexport function formatSkillLoadingMetadata(\n  skillName: string,\n  _progressMessage: string = 'loading',\n): string {\n  // Use skill name only - UserCommandMessage renders as \"Skill(name)\"\n  return [\n    `<${COMMAND_MESSAGE_TAG}>${skillName}</${COMMAND_MESSAGE_TAG}>`,\n    `<${COMMAND_NAME_TAG}>${skillName}</${COMMAND_NAME_TAG}>`,\n    `<skill-format>true</skill-format>`,\n  ].join('\\n')\n}\n\n/**\n * Formats the metadata for a slash command loading message.\n */\nfunction formatSlashCommandLoadingMetadata(\n  commandName: string,\n  args?: string,\n): string {\n  return [\n    `<${COMMAND_MESSAGE_TAG}>${commandName}</${COMMAND_MESSAGE_TAG}>`,\n    `<${COMMAND_NAME_TAG}>/${commandName}</${COMMAND_NAME_TAG}>`,\n    args ? `<command-args>${args}</command-args>` : null,\n  ]\n    .filter(Boolean)\n    .join('\\n')\n}\n\n/**\n * Formats the loading metadata for a command (skill or slash command).\n * User-invocable skills use slash command format (/name), while model-only\n * skills use the skill format (\"The X skill is running\").\n */\nfunction formatCommandLoadingMetadata(\n  command: CommandBase & PromptCommand,\n  args?: string,\n): string {\n  // Use command.name (the qualified name including plugin prefix, e.g.\n  // \"product-management:feature-spec\") instead of userFacingName() which may\n  // strip the plugin prefix via displayName fallback.\n  // User-invocable skills should show as /command-name like regular slash commands\n  if (command.userInvocable !== false) {\n    return formatSlashCommandLoadingMetadata(command.name, args)\n  }\n  // Model-only skills (userInvocable: false) show as \"The X skill is running\"\n  if (\n    command.loadedFrom === 'skills' ||\n    command.loadedFrom === 'plugin' ||\n    command.loadedFrom === 'mcp'\n  ) {\n    return formatSkillLoadingMetadata(command.name, command.progressMessage)\n  }\n  return formatSlashCommandLoadingMetadata(command.name, args)\n}\n\nexport async function processPromptSlashCommand(\n  commandName: string,\n  args: string,\n  commands: Command[],\n  context: ToolUseContext,\n  imageContentBlocks: ContentBlockParam[] = [],\n): Promise<SlashCommandResult> {\n  const command = findCommand(commandName, commands)\n  if (!command) {\n    throw new MalformedCommandError(`Unknown command: ${commandName}`)\n  }\n  if (command.type !== 'prompt') {\n    throw new Error(\n      `Unexpected ${command.type} command. Expected 'prompt' command. Use /${commandName} directly in the main conversation.`,\n    )\n  }\n  return getMessagesForPromptSlashCommand(\n    command,\n    args,\n    context,\n    [],\n    imageContentBlocks,\n  )\n}\n\nasync function getMessagesForPromptSlashCommand(\n  command: CommandBase & PromptCommand,\n  args: string,\n  context: ToolUseContext,\n  precedingInputBlocks: ContentBlockParam[] = [],\n  imageContentBlocks: ContentBlockParam[] = [],\n  uuid?: string,\n): Promise<SlashCommandResult> {\n  // In coordinator mode (main thread only), skip loading the full skill content\n  // and permissions. The coordinator only has Agent + TaskStop tools, so the\n  // skill content and allowedTools are useless. Instead, send a brief summary\n  // telling the coordinator how to delegate this skill to a worker.\n  //\n  // Workers run in-process and inherit CLAUDE_CODE_COORDINATOR_MODE from the\n  // parent env, so we also check !context.agentId: agentId is only set for\n  // subagents, letting workers fall through to getPromptForCommand and receive\n  // the real skill content when they invoke the Skill tool.\n  if (\n    feature('COORDINATOR_MODE') &&\n    isEnvTruthy(process.env.CLAUDE_CODE_COORDINATOR_MODE) &&\n    !context.agentId\n  ) {\n    const metadata = formatCommandLoadingMetadata(command, args)\n    const parts: string[] = [\n      `Skill \"/${command.name}\" is available for workers.`,\n    ]\n    if (command.description) {\n      parts.push(`Description: ${command.description}`)\n    }\n    if (command.whenToUse) {\n      parts.push(`When to use: ${command.whenToUse}`)\n    }\n    const skillAllowedTools = command.allowedTools ?? []\n    if (skillAllowedTools.length > 0) {\n      parts.push(\n        `This skill grants workers additional tool permissions: ${skillAllowedTools.join(', ')}`,\n      )\n    }\n    parts.push(\n      `\\nInstruct a worker to use this skill by including \"Use the /${command.name} skill\" in your Agent prompt. The worker has access to the Skill tool and will receive the skill's content and permissions when it invokes it.`,\n    )\n    const summaryContent: ContentBlockParam[] = [\n      { type: 'text', text: parts.join('\\n') },\n    ]\n    return {\n      messages: [\n        createUserMessage({ content: metadata, uuid }),\n        createUserMessage({ content: summaryContent, isMeta: true }),\n      ],\n      shouldQuery: true,\n      model: command.model,\n      effort: command.effort,\n      command,\n    }\n  }\n\n  const result = await command.getPromptForCommand(args, context)\n\n  // Register skill hooks if defined. Under [\"hooks\"]-only (skills not locked),\n  // user skills still load and reach this point — block hook REGISTRATION here\n  // where source is known. Mirrors the agent frontmatter gate in runAgent.ts.\n  const hooksAllowedForThisSkill =\n    !isRestrictedToPluginOnly('hooks') || isSourceAdminTrusted(command.source)\n  if (command.hooks && hooksAllowedForThisSkill) {\n    const sessionId = getSessionId()\n    registerSkillHooks(\n      context.setAppState,\n      sessionId,\n      command.hooks,\n      command.name,\n      command.type === 'prompt' ? command.skillRoot : undefined,\n    )\n  }\n\n  // Record skill invocation for compaction preservation, scoped by agent context.\n  // Skills are tagged with their agentId so only skills belonging to the current\n  // agent are restored during compaction (preventing cross-agent leaks).\n  const skillPath = command.source\n    ? `${command.source}:${command.name}`\n    : command.name\n  const skillContent = result\n    .filter((b): b is TextBlockParam => b.type === 'text')\n    .map(b => b.text)\n    .join('\\n\\n')\n  addInvokedSkill(\n    command.name,\n    skillPath,\n    skillContent,\n    getAgentContext()?.agentId ?? null,\n  )\n\n  const metadata = formatCommandLoadingMetadata(command, args)\n\n  const additionalAllowedTools = parseToolListFromCLI(\n    command.allowedTools ?? [],\n  )\n\n  // Create content for the main message, including any pasted images\n  const mainMessageContent: ContentBlockParam[] =\n    imageContentBlocks.length > 0 || precedingInputBlocks.length > 0\n      ? [...imageContentBlocks, ...precedingInputBlocks, ...result]\n      : result\n\n  // Extract attachments from command arguments (@-mentions, MCP resources,\n  // agent mentions in SKILL.md). skipSkillDiscovery prevents the SKILL.md\n  // content itself from triggering discovery — it's meta-content, not user\n  // intent, and a large SKILL.md (e.g. 110KB) would fire chunked AKI queries\n  // adding seconds of latency to every skill invocation.\n  const attachmentMessages = await toArray(\n    getAttachmentMessages(\n      result\n        .filter((block): block is TextBlockParam => block.type === 'text')\n        .map(block => block.text)\n        .join(' '),\n      context,\n      null,\n      [], // queuedCommands - handled by query.ts for mid-turn attachments\n      context.messages,\n      'repl_main_thread',\n      { skipSkillDiscovery: true },\n    ),\n  )\n\n  const messages = [\n    createUserMessage({\n      content: metadata,\n      uuid,\n    }),\n    createUserMessage({\n      content: mainMessageContent,\n      isMeta: true,\n    }),\n    ...attachmentMessages,\n    createAttachmentMessage({\n      type: 'command_permissions',\n      allowedTools: additionalAllowedTools,\n      model: command.model,\n    }),\n  ]\n\n  return {\n    messages,\n    shouldQuery: true,\n    allowedTools: additionalAllowedTools,\n    model: command.model,\n    effort: command.effort,\n    command,\n  }\n}\n"],"mappings":"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","ignoreList":[]} diff --git a/src/utils/processUserInput/processTextPrompt.ts b/src/utils/processUserInput/processTextPrompt.ts index cc91c5b..b6898fc 100644 --- a/src/utils/processUserInput/processTextPrompt.ts +++ b/src/utils/processUserInput/processTextPrompt.ts @@ -9,7 +9,6 @@ import type { import { logEvent } from '../../services/analytics/index.js' import type { PermissionMode } from '../../types/permissions.js' import { createUserMessage } from '../messages.js' -import { logOTelEvent, redactIfDisabled } from '../telemetry/events.js' import { startInteractionSpan } from '../telemetry/sessionTracing.js' import { matchesKeepGoingKeyword, @@ -37,25 +36,6 @@ export function processTextPrompt( : input.find(block => block.type === 'text')?.text || '' startInteractionSpan(userPromptText) - // Emit user_prompt OTEL event for both string (CLI) and array (SDK/VS Code) - // input shapes. Previously gated on `typeof input === 'string'`, so VS Code - // sessions never emitted user_prompt (anthropics/claude-code#33301). - // For array input, use the LAST text block: createUserContent pushes the - // user's message last (after any /attachment context blocks), - // so .findLast gets the actual prompt. userPromptText (first block) is kept - // unchanged for startInteractionSpan to preserve existing span attributes. - const otelPromptText = - typeof input === 'string' - ? input - : input.findLast(block => block.type === 'text')?.text || '' - if (otelPromptText) { - void logOTelEvent('user_prompt', { - prompt_length: String(otelPromptText.length), - prompt: redactIfDisabled(otelPromptText), - 'prompt.id': promptId, - }) - } - const isNegative = matchesNegativeKeyword(userPromptText) const isKeepGoing = matchesKeepGoingKeyword(userPromptText) logEvent('tengu_input_prompt', { diff --git a/src/utils/telemetry/events.ts b/src/utils/telemetry/events.ts deleted file mode 100644 index 936b36f..0000000 --- a/src/utils/telemetry/events.ts +++ /dev/null @@ -1,14 +0,0 @@ -/** - * OpenTelemetry event egress is disabled in this build. - */ - -export function redactIfDisabled(_content: string): string { - return '' -} - -export async function logOTelEvent( - _eventName: string, - _metadata: { [key: string]: string | undefined } = {}, -): Promise { - return -}