πŸ“„ File detail

components/FeedbackSurvey/useMemorySurvey.tsx

🧩 .tsxπŸ“ 213 linesπŸ’Ύ 30,418 bytesπŸ“ text
← Back to All Files

🎯 Use case

This file lives under β€œcomponents/”, which covers shared React UI pieces. On the API surface it exposes useMemorySurvey β€” mainly functions, hooks, or classes. Dependencies touch React UI and src. It composes internal code from memdir, services, tools, types, and utils (relative imports).

Generated from folder role, exports, dependency roots, and inline comments β€” not hand-reviewed for every path.

🧠 Inline summary

import { useCallback, useEffect, useMemo, useRef } from 'react'; import { isFeedbackSurveyDisabled } from 'src/services/analytics/config.js'; import { getFeatureValue_CACHED_MAY_BE_STALE } from 'src/services/analytics/growthbook.js'; import { type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, logEvent } from 'src/services/analytics/index.js'; import { isAutoMemoryEnabled } from '../.

πŸ“€ Exports (heuristic)

  • useMemorySurvey

πŸ“š External import roots

Package roots from from "…" (relative paths omitted).

  • react
  • src

πŸ–₯️ Source preview

import { useCallback, useEffect, useMemo, useRef } from 'react';
import { isFeedbackSurveyDisabled } from 'src/services/analytics/config.js';
import { getFeatureValue_CACHED_MAY_BE_STALE } from 'src/services/analytics/growthbook.js';
import { type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, logEvent } from 'src/services/analytics/index.js';
import { isAutoMemoryEnabled } from '../../memdir/paths.js';
import { isPolicyAllowed } from '../../services/policyLimits/index.js';
import { FILE_READ_TOOL_NAME } from '../../tools/FileReadTool/prompt.js';
import type { Message } from '../../types/message.js';
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';
import type { FeedbackSurveyResponse } from './utils.js';
const HIDE_THANKS_AFTER_MS = 3000;
const MEMORY_SURVEY_GATE = 'tengu_dunwich_bell';
const MEMORY_SURVEY_EVENT = 'tengu_memory_survey_event';
const SURVEY_PROBABILITY = 0.2;
const TRANSCRIPT_SHARE_TRIGGER = 'memory_survey';
const MEMORY_WORD_RE = /\bmemor(?:y|ies)\b/i;
function hasMemoryFileRead(messages: Message[]): boolean {
  for (const message of messages) {
    if (message.type !== 'assistant') {
      continue;
    }
    const content = message.message.content;
    if (!Array.isArray(content)) {
      continue;
    }
    for (const block of content) {
      if (block.type !== 'tool_use' || block.name !== FILE_READ_TOOL_NAME) {
        continue;
      }
      const input = block.input as {
        file_path?: unknown;
      };
      if (typeof input.file_path === 'string' && isAutoManagedMemoryFile(input.file_path)) {
        return true;
      }
    }
  }
  return false;
}
export function useMemorySurvey(messages: Message[], isLoading: boolean, hasActivePrompt = false, {
  enabled = true
}: {
  enabled?: boolean;
} = {}): {
  state: 'closed' | 'open' | 'thanks' | 'transcript_prompt' | 'submitting' | 'submitted';
  lastResponse: FeedbackSurveyResponse | null;
  handleSelect: (selected: FeedbackSurveyResponse) => void;
  handleTranscriptSelect: (selected: TranscriptShareResponse) => void;
} {
  // Track assistant message UUIDs that were already evaluated so we don't
  // re-roll probability on re-renders or re-scan messages for the same turn.
  const seenAssistantUuids = useRef<Set<string>>(new Set());
  // Once a memory file read is observed it stays true for the session β€”
  // skip the O(n) scan on subsequent turns.
  const memoryReadSeen = useRef(false);
  const messagesRef = useRef(messages);
  messagesRef.current = messages;
  const onOpen = useCallback((appearanceId: string) => {
    logEvent(MEMORY_SURVEY_EVENT, {
      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, {
      event_type: 'responded' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
      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') {
      return false;
    }
    if (selected_0 !== 'bad' && selected_0 !== 'good') {
      return false;
    }
    if (getGlobalConfig().transcriptShareDismissed) {
      return false;
    }
    if (!isPolicyAllowed('allow_product_feedback')) {
      return false;
    }
    return true;
  }, []);
  const onTranscriptPromptShown = useCallback((appearanceId_1: string) => {
    logEvent(MEMORY_SURVEY_EVENT, {
      event_type: 'transcript_prompt_appeared' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
      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<boolean> => {
    logEvent(MEMORY_SURVEY_EVENT, {
      event_type: `transcript_share_${selected_1}` as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
      appearance_id: appearanceId_2 as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
      trigger: TRANSCRIPT_SHARE_TRIGGER as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS
    });
    if (selected_1 === 'dont_ask_again') {
      saveGlobalConfig(current => ({
        ...current,
        transcriptShareDismissed: true
      }));
    }
    if (selected_1 === 'yes') {
      const result = await submitTranscriptShare(messagesRef.current, TRANSCRIPT_SHARE_TRIGGER, appearanceId_2);
      logEvent(MEMORY_SURVEY_EVENT, {
        event_type: (result.success ? 'transcript_share_submitted' : 'transcript_share_failed') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
        appearance_id: appearanceId_2 as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
        trigger: TRANSCRIPT_SHARE_TRIGGER as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS
      });
      return result.success;
    }
    return false;
  }, []);
  const {
    state,
    lastResponse,
    open,
    handleSelect,
    handleTranscriptSelect
  } = useSurveyState({
    hideThanksAfterMs: HIDE_THANKS_AFTER_MS,
    onOpen,
    onSelect,
    shouldShowTranscriptPrompt,
    onTranscriptPromptShown,
    onTranscriptSelect
  });
  const lastAssistant = useMemo(() => getLastAssistantMessage(messages), [messages]);
  useEffect(() => {
    if (!enabled) return;

    // /clear resets messages but REPL stays mounted β€” reset refs so a memory
    // read from the previous conversation doesn't leak into the new one.
    if (messages.length === 0) {
      memoryReadSeen.current = false;
      seenAssistantUuids.current.clear();
      return;
    }
    if (state !== 'closed' || isLoading || hasActivePrompt) {
      return;
    }

    // 3P default: survey off (no GrowthBook on Bedrock/Vertex/Foundry).
    if (!getFeatureValue_CACHED_MAY_BE_STALE(MEMORY_SURVEY_GATE, false)) {
      return;
    }
    if (!isAutoMemoryEnabled()) {
      return;
    }
    if (isFeedbackSurveyDisabled()) {
      return;
    }
    if (!isPolicyAllowed('allow_product_feedback')) {
      return;
    }
    if (isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_FEEDBACK_SURVEY)) {
      return;
    }
    if (!lastAssistant || seenAssistantUuids.current.has(lastAssistant.uuid)) {
      return;
    }
    const text = extractTextContent(lastAssistant.message.content, ' ');
    if (!MEMORY_WORD_RE.test(text)) {
      return;
    }

    // Mark as evaluated before the memory-read scan so a turn that mentions
    // "memory" but has no memory read doesn't trigger repeated O(n) scans
    // on subsequent renders with the same last assistant message.
    seenAssistantUuids.current.add(lastAssistant.uuid);
    if (!memoryReadSeen.current) {
      memoryReadSeen.current = hasMemoryFileRead(messages);
    }
    if (!memoryReadSeen.current) {
      return;
    }
    if (Math.random() < SURVEY_PROBABILITY) {
      open();
    }
  }, [enabled, state, isLoading, hasActivePrompt, lastAssistant, messages, open]);
  return {
    state,
    lastResponse,
    handleSelect,
    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":[]}