Skip to content

feat(cli): add structured memory diagnostics JSON#3785

Merged
wenshao merged 12 commits into
mainfrom
codex/memory-diagnostics
May 17, 2026
Merged

feat(cli): add structured memory diagnostics JSON#3785
wenshao merged 12 commits into
mainfrom
codex/memory-diagnostics

Conversation

@yiliang114

@yiliang114 yiliang114 commented May 2, 2026

Copy link
Copy Markdown
Collaborator

Summary

  • What changed:

    • Add collectMemoryDiagnostics() in core for structured Node/V8/process memory snapshots with session id, CLI version, resource usage, optional Linux /proc probes, and heuristic risk analysis.
    • Wire /doctor memory --json to return the structured diagnostics while preserving the latest main behavior for readable /doctor memory, --sample, and --snapshot output.
    • Emit structured memory risk results as warning messages and propagate warning through the slash-command message pipeline.
    • Add a reference design note that frames this PR as the structured diagnostics layer for [P3] Memory Diagnostics / 内存诊断 #3000.
  • Why it changed:

    • [P3] Memory Diagnostics / 内存诊断 #3000 needs a stable, low-cost diagnostic payload that can be attached to issues or used by later benchmark tooling.
    • Latest main now owns the human-readable /doctor memory path; this PR focuses on structured JSON output and warning-aware handling.
  • Reviewer focus:

    • Whether /doctor memory --json exposes the right fields for issue/debug attachment.
    • Whether risk hints are useful as warnings without turning diagnostic findings into command failures.
    • Whether Linux-only probes degrade safely on platforms where /proc is unavailable.

Validation

  • Commands run:

    cd packages/core && npx vitest run src/utils/memoryDiagnostics.test.ts
    cd packages/cli && npx vitest run src/ui/commands/doctorCommand.test.ts src/ui/hooks/slashCommandProcessor.test.ts src/ui/commands/insightCommand.test.ts src/acp-integration/session/Session.test.ts
    npm run lint
    npm run build
    npm run typecheck
    npm run bundle
    git diff --cached --check
  • Expected result:

    • /doctor memory keeps the latest main readable diagnostics behavior.
    • /doctor memory --sample and /doctor memory --snapshot keep the latest main behavior.
    • /doctor memory --json returns structured JSON with session and version metadata.
    • Risk indicators are reported through analysis.risks and render as warning messages rather than command failures.
  • Observed result:

    • Core memory diagnostics tests: 10 passed.
    • Focused CLI tests: 146 passed.
    • npm run lint: passed.
    • npm run build: passed.
    • npm run typecheck: passed.
    • npm run bundle: passed.
    • git diff --cached --check: passed.
    • The build still reports an existing vscode companion lint warning and Browserslist stale warning; these are unrelated to this PR.

Scope / Risk

  • Main risk or tradeoff:

    • The risk thresholds are heuristic hints for investigation, not definitive leak detection.
  • Not covered / not validated:

    • Heap snapshot export implementation is owned by latest main and is not changed here.
    • Continuous sampling and trend analysis are not changed here.
    • Prompt/history/tool output retention is not changed.
    • Full manual smoke testing was not run across Windows/Linux/macOS.
  • Breaking changes / migration notes:

    • None.

Linked Issues / Bugs

Progress on #3000

@yiliang114

Copy link
Copy Markdown
Collaborator Author

@copilot pls review for me

@github-actions

github-actions Bot commented May 2, 2026

Copy link
Copy Markdown
Contributor

Code Coverage Summary

Package Lines Statements Functions Branches
CLI 77.12% 77.12% 78.19% 80.54%
Core 79.2% 79.2% 81.77% 82.76%
CLI Package - Full Text Report
-------------------|---------|----------|---------|---------|-------------------
File               | % Stmts | % Branch | % Funcs | % Lines | Uncovered Line #s 
-------------------|---------|----------|---------|---------|-------------------
All files          |   77.12 |    80.54 |   78.19 |   77.12 |                   
 src               |   75.73 |    69.15 |   80.55 |   75.73 |                   
  gemini.tsx       |   68.53 |     66.4 |   76.47 |   68.53 | ...29,946-949,957 
  ...ractiveCli.ts |      80 |    68.61 |   78.57 |      80 | ...1020,1058,1161 
  ...liCommands.ts |   74.51 |     72.5 |     100 |   74.51 | ...41-265,290,391 
  ...ActiveAuth.ts |     100 |     87.5 |     100 |     100 | 66-80             
 ...cp-integration |    62.2 |    76.42 |   63.88 |    62.2 |                   
  acpAgent.ts      |   64.99 |    76.85 |   70.96 |   64.99 | ...64-966,980-988 
  authMethods.ts   |   12.19 |      100 |       0 |   12.19 | 11-31,34-38,41-50 
  errorCodes.ts    |       0 |        0 |       0 |       0 | 1-22              
  ...DirContext.ts |     100 |      100 |     100 |     100 |                   
 ...ration/service |   68.65 |    83.33 |   66.66 |   68.65 |                   
  filesystem.ts    |   68.65 |    83.33 |   66.66 |   68.65 | ...32,77-94,97-98 
 ...ration/session |   76.28 |     70.9 |      84 |   76.28 |                   
  ...ryReplayer.ts |   65.93 |    75.67 |   81.81 |   65.93 | ...40-255,268-269 
  Session.ts       |   75.46 |    69.28 |    85.1 |   75.46 | ...2468,2474-2477 
  ...entTracker.ts |   90.85 |    84.84 |      90 |   90.85 | ...35,199,251-260 
  index.ts         |       0 |        0 |       0 |       0 | 1-40              
  ...ssionUtils.ts |   84.21 |    77.77 |     100 |   84.21 | ...37-153,209-211 
  types.ts         |       0 |        0 |       0 |       0 | 1                 
 ...ssion/emitters |   96.01 |    90.75 |    92.3 |   96.01 |                   
  BaseEmitter.ts   |   76.92 |    66.66 |      80 |   76.92 | 23-24,39-40,55-56 
  ...ageEmitter.ts |     100 |    89.47 |     100 |     100 | 109,111           
  PlanEmitter.ts   |     100 |      100 |     100 |     100 |                   
  ...allEmitter.ts |   98.06 |     92.3 |     100 |   98.06 | 227-228,327,335   
  index.ts         |       0 |        0 |       0 |       0 | 1-10              
 ...ession/rewrite |   90.36 |    87.83 |   94.11 |   90.36 |                   
  LlmRewriter.ts   |      81 |       84 |     100 |      81 | ...,88-89,155-159 
  ...Middleware.ts |   95.83 |    85.71 |     100 |   95.83 | 119,127-129       
  TurnBuffer.ts    |     100 |      100 |     100 |     100 |                   
  config.ts        |     100 |      100 |     100 |     100 |                   
  index.ts         |     100 |      100 |     100 |     100 |                   
  types.ts         |       0 |        0 |       0 |       0 | 1                 
 src/auth          |    97.7 |    94.81 |   95.45 |    97.7 |                   
  allProviders.ts  |     100 |      100 |     100 |     100 |                   
  ...iderConfig.ts |    97.6 |    95.04 |     100 |    97.6 | ...61,411,433-434 
  types.ts         |       0 |        0 |       0 |       0 | 1                 
 src/auth/install  |   98.57 |    88.88 |     100 |   98.57 |                   
  ...nstallPlan.ts |   98.57 |    88.88 |     100 |   98.57 | 80,93             
 ...viders/alibaba |   96.96 |    66.66 |   66.66 |   96.96 |                   
  ...baStandard.ts |     100 |      100 |     100 |     100 |                   
  codingPlan.ts    |   93.67 |    66.66 |   66.66 |   93.67 | 83,87-89,94       
  tokenPlan.ts     |     100 |      100 |     100 |     100 |                   
 ...oviders/custom |     100 |      100 |     100 |     100 |                   
  ...omProvider.ts |     100 |      100 |     100 |     100 |                   
 ...roviders/oauth |    91.5 |    77.03 |   97.05 |    91.5 |                   
  openrouter.ts    |   84.37 |    33.33 |     100 |   84.37 | 43-48             
  ...outerOAuth.ts |    91.9 |    79.06 |   96.87 |    91.9 | ...53-655,699-701 
 ...ers/thirdParty |     100 |      100 |     100 |     100 |                   
  deepseek.ts      |     100 |      100 |     100 |     100 |                   
  idealab.ts       |     100 |      100 |     100 |     100 |                   
  minimax.ts       |     100 |      100 |     100 |     100 |                   
  modelscope.ts    |     100 |      100 |     100 |     100 |                   
  zai.ts           |     100 |      100 |     100 |     100 |                   
 src/commands      |   58.43 |    85.71 |   43.47 |   58.43 |                   
  auth.ts          |     100 |    83.33 |     100 |     100 | 11,14             
  channel.ts       |   56.66 |      100 |       0 |   56.66 | 15-19,27-34       
  extensions.tsx   |   96.55 |      100 |      50 |   96.55 | 37                
  hooks.tsx        |   66.66 |      100 |       0 |   66.66 | 20-24             
  mcp.ts           |   94.73 |      100 |      50 |   94.73 | 28                
  review.ts        |   51.85 |      100 |       0 |   51.85 | 24-35,38          
  serve.ts         |   10.97 |      100 |       0 |   10.97 | ...5,45-95,97-134 
 ...mmands/channel |   39.25 |    79.45 |      50 |   39.25 |                   
  ...l-registry.ts |    8.57 |      100 |       0 |    8.57 | 6-21,24-42        
  config-utils.ts  |      92 |      100 |   66.66 |      92 | 21-26             
  configure.ts     |    14.7 |      100 |       0 |    14.7 | 18-21,23-84       
  pairing.ts       |   26.31 |      100 |       0 |   26.31 | ...30,40-50,52-65 
  pidfile.ts       |   96.34 |    86.95 |     100 |   96.34 | 49,59,91          
  start.ts         |   30.98 |       52 |   69.23 |   30.98 | ...72-475,484-486 
  status.ts        |   17.85 |      100 |       0 |   17.85 | 15-26,32-76       
  stop.ts          |      20 |      100 |       0 |      20 | 14-48             
 ...nds/extensions |    84.5 |    88.95 |   81.81 |    84.5 |                   
  consent.ts       |   71.65 |    89.28 |   42.85 |   71.65 | ...85-141,156-162 
  disable.ts       |     100 |      100 |     100 |     100 |                   
  enable.ts        |     100 |      100 |     100 |     100 |                   
  install.ts       |    75.6 |    66.66 |   66.66 |    75.6 | ...39-142,145-153 
  link.ts          |     100 |      100 |     100 |     100 |                   
  list.ts          |     100 |      100 |     100 |     100 |                   
  new.ts           |     100 |      100 |     100 |     100 |                   
  settings.ts      |   99.15 |      100 |   83.33 |   99.15 | 151               
  uninstall.ts     |    37.5 |      100 |   33.33 |    37.5 | 23-45,57-64,67-70 
  update.ts        |   96.32 |      100 |     100 |   96.32 | 101-105           
  utils.ts         |   60.24 |    28.57 |     100 |   60.24 | ...81,83-87,89-93 
 ...les/mcp-server |       0 |        0 |       0 |       0 |                   
  example.ts       |       0 |        0 |       0 |       0 | 1-60              
 src/commands/mcp  |   92.29 |    86.08 |   88.88 |   92.29 |                   
  add.ts           |     100 |    98.03 |     100 |     100 | 293               
  list.ts          |   91.22 |    80.76 |      80 |   91.22 | ...19-121,146-147 
  reconnect.ts     |   76.72 |    71.42 |   85.71 |   76.72 | 35-48,153-175     
  remove.ts        |     100 |       80 |     100 |     100 | 21-25             
 ...ommands/review |   11.57 |      100 |       0 |   11.57 |                   
  cleanup.ts       |   17.94 |      100 |       0 |   17.94 | ...01-106,108-109 
  deterministic.ts |   13.75 |      100 |       0 |   13.75 | ...22-738,740-741 
  fetch-pr.ts      |   11.36 |      100 |       0 |   11.36 | ...80-201,203-204 
  load-rules.ts    |   11.32 |      100 |       0 |   11.32 | ...41-153,155-156 
  pr-context.ts    |    6.22 |      100 |       0 |    6.22 | ...97-312,314-315 
  presubmit.ts     |    9.35 |      100 |       0 |    9.35 | ...62-287,289-290 
 ...nds/review/lib |      30 |      100 |       0 |      30 |                   
  gh.ts            |   22.58 |      100 |       0 |   22.58 | ...49,53-54,62-69 
  git.ts           |   22.72 |      100 |       0 |   22.72 | 15-18,29-39,43-44 
  paths.ts         |   52.94 |      100 |       0 |   52.94 | ...26,37-38,42-43 
 src/config        |   92.87 |    85.31 |   88.09 |   92.87 |                   
  auth.ts          |   86.98 |    80.32 |     100 |   86.98 | ...26-227,243-244 
  config.ts        |   88.67 |     85.1 |      80 |   88.67 | ...1825,1827-1835 
  keyBindings.ts   |   96.55 |       50 |     100 |   96.55 | 193-196           
  ...idersScope.ts |      92 |       90 |     100 |      92 | 11-12             
  sandboxConfig.ts |   61.64 |    71.87 |   66.66 |   61.64 | ...54-68,73,77-89 
  settings.ts      |   85.76 |    87.25 |   89.18 |   85.76 | ...1148,1153-1156 
  ...ingsSchema.ts |     100 |      100 |     100 |     100 |                   
  ...tedFolders.ts |   96.22 |       94 |     100 |   96.22 | ...88-190,205-206 
 ...nfig/migration |   94.89 |    78.94 |   83.33 |   94.89 |                   
  index.ts         |   94.87 |    88.88 |     100 |   94.87 | 91-92             
  scheduler.ts     |   96.55 |    77.77 |     100 |   96.55 | 19-20             
  types.ts         |       0 |        0 |       0 |       0 | 1                 
 ...ation/versions |   94.74 |       96 |     100 |   94.74 |                   
  ...-v2-shared.ts |     100 |      100 |     100 |     100 |                   
  v1-to-v2.ts      |   81.75 |    90.19 |     100 |   81.75 | ...28-229,231-247 
  v2-to-v3.ts      |     100 |      100 |     100 |     100 |                   
  v3-to-v4.ts      |     100 |      100 |     100 |     100 |                   
 src/core          |     100 |      100 |     100 |     100 |                   
  auth.ts          |     100 |      100 |     100 |     100 |                   
  initializer.ts   |     100 |      100 |     100 |     100 |                   
  theme.ts         |     100 |      100 |     100 |     100 |                   
 src/dualOutput    |   63.09 |    64.51 |   55.55 |   63.09 |                   
  ...tputBridge.ts |   62.94 |    65.51 |   56.25 |   62.94 | ...22-323,331-334 
  ...utContext.tsx |     100 |      100 |     100 |     100 |                   
  index.ts         |       0 |        0 |       0 |       0 | 1-8               
 src/export        |       0 |        0 |       0 |       0 |                   
  index.ts         |       0 |        0 |       0 |       0 | 1-7               
 src/generated     |     100 |      100 |     100 |     100 |                   
  git-commit.ts    |     100 |      100 |     100 |     100 |                   
 src/i18n          |   81.47 |    75.94 |   65.71 |   81.47 |                   
  index.ts         |   63.68 |    69.56 |   53.84 |   63.68 | ...70-271,281-286 
  languages.ts     |   96.92 |    86.66 |     100 |   96.92 | 134-135,167,184   
  ...nslateKeys.ts |     100 |      100 |     100 |     100 |                   
  ...lationDict.ts |   93.33 |    66.66 |     100 |   93.33 | 15                
 src/i18n/locales  |     100 |      100 |     100 |     100 |                   
  ca.js            |     100 |      100 |     100 |     100 |                   
  de.js            |     100 |      100 |     100 |     100 |                   
  en.js            |     100 |      100 |     100 |     100 |                   
  fr.js            |     100 |      100 |     100 |     100 |                   
  ja.js            |     100 |      100 |     100 |     100 |                   
  pt.js            |     100 |      100 |     100 |     100 |                   
  ru.js            |     100 |      100 |     100 |     100 |                   
  zh-TW.js         |     100 |      100 |     100 |     100 |                   
  zh.js            |     100 |      100 |     100 |     100 |                   
 ...nonInteractive |   72.57 |    71.12 |   74.07 |   72.57 |                   
  session.ts       |   76.64 |     69.4 |   85.71 |   76.64 | ...23-824,833-843 
  types.ts         |    42.5 |      100 |   33.33 |    42.5 | ...80-581,584-585 
 ...active/control |   77.04 |    88.23 |      80 |   77.04 |                   
  ...rolContext.ts |    7.14 |        0 |       0 |    7.14 | 49-84             
  ...Dispatcher.ts |   91.66 |    91.83 |   88.88 |   91.66 | ...54-372,388,391 
  ...rolService.ts |       8 |        0 |       0 |       8 | 46-179            
 ...ol/controllers |    7.04 |       80 |   13.33 |    7.04 |                   
  ...Controller.ts |   19.32 |      100 |      60 |   19.32 | 81-118,127-210    
  ...Controller.ts |       0 |        0 |       0 |       0 | 1-56              
  ...Controller.ts |    3.96 |      100 |   11.11 |    3.96 | ...61-379,389-494 
  ...Controller.ts |   14.06 |      100 |       0 |   14.06 | ...82-117,130-133 
  ...Controller.ts |    5.21 |      100 |       0 |    5.21 | ...21-433,442-471 
 .../control/types |       0 |        0 |       0 |       0 |                   
  serviceAPIs.ts   |       0 |        0 |       0 |       0 | 1                 
 ...Interactive/io |   97.98 |    93.72 |   95.18 |   97.98 |                   
  ...putAdapter.ts |   97.89 |    92.82 |   98.07 |   97.89 | ...1303,1398-1399 
  ...putAdapter.ts |      96 |    91.66 |   85.71 |      96 | 51-52             
  ...nputReader.ts |     100 |    94.73 |     100 |     100 | 67                
  ...putAdapter.ts |   98.28 |      100 |      90 |   98.28 | 81-82,122-123     
  index.ts         |     100 |      100 |     100 |     100 |                   
 src/patches       |       0 |        0 |       0 |       0 |                   
  is-in-ci.ts      |       0 |        0 |       0 |       0 | 1-17              
 src/remoteInput   |   86.98 |       75 |   85.71 |   86.98 |                   
  ...utContext.tsx |     100 |      100 |     100 |     100 |                   
  ...putWatcher.ts |   88.12 |    76.08 |   91.66 |   88.12 | ...21-222,233-236 
  index.ts         |       0 |        0 |       0 |       0 | 1-8               
 src/serve         |      83 |    82.68 |   91.66 |      83 |                   
  auth.ts          |   85.86 |    83.87 |      80 |   85.86 | ...47-148,151-153 
  capabilities.ts  |     100 |      100 |     100 |     100 |                   
  eventBus.ts      |   87.07 |    87.93 |      85 |   87.07 | ...46-354,415-417 
  httpAcpBridge.ts |   81.14 |    79.89 |   96.66 |   81.14 | ...3309,3340-3381 
  ...oryChannel.ts |     100 |      100 |     100 |     100 |                   
  index.ts         |     100 |      100 |     100 |     100 |                   
  loopbackBinds.ts |     100 |      100 |     100 |     100 |                   
  runQwenServe.ts  |   79.01 |     87.5 |   83.33 |   79.01 | ...24-440,465-467 
  server.ts        |   84.32 |    83.96 |      85 |   84.32 | ...1175,1232-1241 
  types.ts         |     100 |      100 |     100 |     100 |                   
 src/services      |   91.67 |    91.21 |   97.56 |   91.67 |                   
  ...mandLoader.ts |     100 |    93.75 |     100 |     100 | 93                
  ...killLoader.ts |     100 |    96.15 |     100 |     100 | 47                
  ...andService.ts |    98.7 |      100 |     100 |    98.7 | 107               
  ...mandLoader.ts |   86.83 |    83.87 |     100 |   86.83 | ...30-335,340-345 
  ...omptLoader.ts |   75.84 |    80.64 |   83.33 |   75.84 | ...10-211,277-278 
  ...mandLoader.ts |     100 |      100 |     100 |     100 |                   
  ...nd-factory.ts |   91.42 |    91.66 |     100 |   91.42 | 128,137-144       
  ...ation-tool.ts |     100 |    95.45 |     100 |     100 | 125               
  ...ndMetadata.ts |   98.21 |    96.66 |     100 |   98.21 | 83,87             
  commandUtils.ts  |      96 |     90.9 |     100 |      96 | 48                
  ...and-parser.ts |   90.69 |    85.71 |     100 |   90.69 | 63-66             
  ...ionService.ts |     100 |      100 |     100 |     100 |                   
  types.ts         |     100 |      100 |     100 |     100 |                   
 ...ght/generators |    85.9 |    85.61 |   90.47 |    85.9 |                   
  DataProcessor.ts |   85.63 |     85.6 |   92.85 |   85.63 | ...1122,1126-1133 
  ...tGenerator.ts |   98.21 |    85.71 |     100 |   98.21 | 46                
  ...teRenderer.ts |   45.45 |      100 |       0 |   45.45 | 13-51             
 .../insight/types |       0 |       50 |      50 |       0 |                   
  ...sightTypes.ts |       0 |        0 |       0 |       0 |                   
  ...sightTypes.ts |       0 |        0 |       0 |       0 | 1                 
 ...mpt-processors |   97.27 |    94.04 |     100 |   97.27 |                   
  ...tProcessor.ts |     100 |      100 |     100 |     100 |                   
  ...eProcessor.ts |   94.52 |    84.21 |     100 |   94.52 | 46-47,93-94       
  ...tionParser.ts |     100 |      100 |     100 |     100 |                   
  ...lProcessor.ts |   97.41 |    95.65 |     100 |   97.41 | 95-98             
  types.ts         |     100 |      100 |     100 |     100 |                   
 src/services/tips |   97.35 |    83.07 |     100 |   97.35 |                   
  index.ts         |     100 |      100 |     100 |     100 |                   
  tipHistory.ts    |   92.45 |       70 |     100 |   92.45 | ...22,144,151,160 
  tipRegistry.ts   |     100 |    95.23 |     100 |     100 | 33                
  tipScheduler.ts  |     100 |    91.66 |     100 |     100 | 55                
 src/test-utils    |   93.75 |    83.33 |      80 |   93.75 |                   
  ...omMatchers.ts |   69.69 |       50 |      50 |   69.69 | 32-35,37-39,45-47 
  ...andContext.ts |     100 |      100 |     100 |     100 |                   
  render.tsx       |     100 |      100 |     100 |     100 |                   
 src/ui            |   66.39 |    73.22 |   57.14 |   66.39 |                   
  App.tsx          |     100 |      100 |     100 |     100 |                   
  AppContainer.tsx |   65.03 |    64.98 |   52.94 |   65.03 | ...2951,2955-2959 
  ...tionNudge.tsx |    9.58 |      100 |       0 |    9.58 | 24-94             
  ...ackDialog.tsx |   29.23 |      100 |       0 |   29.23 | 25-75             
  ...tionNudge.tsx |    7.69 |      100 |       0 |    7.69 | 25-103            
  colors.ts        |   52.72 |      100 |   23.52 |   52.72 | ...52,54-55,60-61 
  constants.ts     |     100 |      100 |     100 |     100 |                   
  keyMatchers.ts   |   95.91 |    97.05 |     100 |   95.91 | 25-26             
  ...tic-colors.ts |     100 |      100 |     100 |     100 |                   
  ...inePresets.ts |   98.17 |    88.88 |     100 |   98.17 | ...12,239,387-389 
  textConstants.ts |     100 |      100 |     100 |     100 |                   
  types.ts         |     100 |      100 |     100 |     100 |                   
 src/ui/auth       |   55.06 |    51.13 |   35.48 |   55.06 |                   
  AuthDialog.tsx   |   64.26 |    44.44 |   16.66 |   64.26 | ...59,366-388,392 
  ...nProgress.tsx |       0 |        0 |       0 |       0 | 1-64              
  ...etupSteps.tsx |    39.5 |       32 |   38.46 |    39.5 | ...69,472,478,481 
  useAuth.ts       |   76.63 |    68.29 |     100 |   76.63 | ...48,493-499,560 
  ...rSetupFlow.ts |   44.61 |    33.33 |      50 |   44.61 | ...57-378,395-438 
 src/ui/commands   |   73.46 |    81.27 |   81.85 |   73.46 |                   
  aboutCommand.ts  |     100 |      100 |     100 |     100 |                   
  agentsCommand.ts |   83.78 |      100 |      60 |   83.78 | 30-32,42-44       
  ...odeCommand.ts |     100 |      100 |     100 |     100 |                   
  arenaCommand.ts  |   62.81 |    58.73 |   65.21 |   62.81 | ...91-596,681-689 
  authCommand.ts   |     100 |      100 |     100 |     100 |                   
  branchCommand.ts |     100 |      100 |     100 |     100 |                   
  btwCommand.ts    |   95.59 |    71.42 |     100 |   95.59 | 72,154-159        
  bugCommand.ts    |   81.13 |    71.42 |     100 |   81.13 | 60-69             
  clearCommand.ts  |      92 |    76.47 |     100 |      92 | 43-44,72-73,91-92 
  ...essCommand.ts |    64.7 |       50 |      75 |    64.7 | ...48-149,163-166 
  ...extCommand.ts |   34.78 |    22.22 |   45.45 |   34.78 | ...86-521,532-533 
  copyCommand.ts   |   98.28 |    94.89 |     100 |   98.28 | ...80,280,321,327 
  deleteCommand.ts |     100 |      100 |     100 |     100 |                   
  diffCommand.ts   |   99.02 |    86.11 |     100 |   99.02 | 222,226           
  ...ryCommand.tsx |   68.09 |    77.77 |   77.77 |   68.09 | ...56-261,315-323 
  docsCommand.ts   |     100 |    88.88 |     100 |     100 | 25                
  doctorCommand.ts |   95.06 |    88.28 |     100 |   95.06 | ...92-293,320-321 
  dreamCommand.ts  |      75 |    66.66 |   66.66 |      75 | 22-27,44-47       
  editorCommand.ts |     100 |      100 |     100 |     100 |                   
  exportCommand.ts |   98.25 |    91.02 |     100 |   98.25 | ...81,198-199,364 
  ...onsCommand.ts |   48.66 |     90.9 |   63.63 |   48.66 | ...05-109,159-211 
  forgetCommand.ts |   26.82 |      100 |      50 |   26.82 | 18-51             
  goalCommand.ts   |    93.1 |    84.61 |     100 |    93.1 | ...67-170,182-185 
  helpCommand.ts   |     100 |      100 |     100 |     100 |                   
  hooksCommand.ts  |    20.4 |       40 |      40 |    20.4 | ...48-180,204-205 
  ideCommand.ts    |   60.75 |    64.28 |   41.17 |   60.75 | ...05-306,310-324 
  initCommand.ts   |   84.33 |    72.72 |     100 |   84.33 | 68,82-87,89-94    
  ...ghtCommand.ts |   74.56 |    68.42 |     100 |   74.56 | ...31-245,250-273 
  ...ageCommand.ts |   92.17 |    82.69 |     100 |   92.17 | ...43,164,173-183 
  lspCommand.ts    |     100 |    86.95 |     100 |     100 | 31,101-102        
  ...elsCommand.ts |     100 |      100 |     100 |     100 |                   
  mcpCommand.ts    |     100 |      100 |     100 |     100 |                   
  memoryCommand.ts |     100 |      100 |     100 |     100 |                   
  modelCommand.ts  |   75.09 |    78.18 |      75 |   75.09 | ...20-225,262-267 
  ...onsCommand.ts |     100 |      100 |     100 |     100 |                   
  planCommand.ts   |   78.82 |    76.92 |     100 |   78.82 | 30-35,51-56,68-73 
  quitCommand.ts   |     100 |      100 |     100 |     100 |                   
  recapCommand.ts  |   21.81 |      100 |      50 |   21.81 | 24-73             
  ...berCommand.ts |   32.43 |      100 |      50 |   32.43 | 23-57             
  renameCommand.ts |   85.71 |    86.04 |     100 |   85.71 | ...02-209,216-221 
  ...oreCommand.ts |    92.3 |    87.87 |     100 |    92.3 | ...,83-88,129-130 
  resumeCommand.ts |     100 |      100 |     100 |     100 |                   
  rewindCommand.ts |      80 |      100 |      50 |      80 | 19-21             
  ...ngsCommand.ts |     100 |      100 |     100 |     100 |                   
  ...hubCommand.ts |   81.43 |    65.21 |      80 |   81.43 | ...70-173,176-179 
  skillsCommand.ts |   15.04 |      100 |      25 |   15.04 | ...90-106,109-136 
  statsCommand.ts  |   88.19 |    84.21 |     100 |   88.19 | ...,58-61,143-146 
  ...ineCommand.ts |     100 |      100 |     100 |     100 |                   
  ...aryCommand.ts |    6.46 |      100 |      50 |    6.46 | 31-329            
  tasksCommand.ts  |   77.22 |    72.13 |     100 |   77.22 | ...46-150,172-177 
  ...tupCommand.ts |     100 |      100 |     100 |     100 |                   
  themeCommand.ts  |     100 |      100 |     100 |     100 |                   
  toolsCommand.ts  |     100 |      100 |     100 |     100 |                   
  trustCommand.ts  |     100 |      100 |     100 |     100 |                   
  types.ts         |     100 |      100 |     100 |     100 |                   
  vimCommand.ts    |   54.54 |      100 |      50 |   54.54 | 19-29             
 src/ui/components |   65.53 |    75.02 |   70.76 |   65.53 |                   
  AboutBox.tsx     |     100 |      100 |     100 |     100 |                   
  AnsiOutput.tsx   |   65.57 |      100 |      50 |   65.57 | 69-90             
  ApiKeyInput.tsx  |       0 |        0 |       0 |       0 | 1-97              
  AppHeader.tsx    |   89.39 |       75 |     100 |   89.39 | 35,37-42,44       
  ...odeDialog.tsx |     9.7 |      100 |       0 |     9.7 | 35-47,50-182      
  AsciiArt.ts      |     100 |      100 |     100 |     100 |                   
  ...Indicator.tsx |   14.63 |      100 |       0 |   14.63 | 18-56             
  ...TextInput.tsx |   77.01 |       76 |     100 |   77.01 | ...20,234-236,263 
  Composer.tsx     |    80.8 |     64.7 |     100 |    80.8 | ...85,103,154,167 
  ...entPrompt.tsx |     100 |      100 |     100 |     100 |                   
  ...ryDisplay.tsx |   75.89 |    62.06 |     100 |   75.89 | ...,88,93-108,113 
  ...geDisplay.tsx |   68.42 |    57.14 |     100 |   68.42 | 16-17,31-32,42-50 
  ...ification.tsx |   28.57 |      100 |       0 |   28.57 | 16-36             
  ...gProfiler.tsx |       0 |        0 |       0 |       0 | 1-36              
  ...ogManager.tsx |    12.2 |      100 |       0 |    12.2 | 64-490            
  ...ngsDialog.tsx |    8.44 |      100 |       0 |    8.44 | 37-195            
  ExitWarning.tsx  |     100 |      100 |     100 |     100 |                   
  ...hProgress.tsx |    87.8 |    33.33 |     100 |    87.8 | 28-31,56          
  ...ustDialog.tsx |     100 |      100 |     100 |     100 |                   
  Footer.tsx       |   79.54 |    54.54 |     100 |   79.54 | ...05-109,133-134 
  ...ngSpinner.tsx |   68.42 |       80 |      50 |   68.42 | 35-52,73,80-81    
  GoalPill.tsx     |   76.19 |    81.81 |     100 |   76.19 | 24-30,46-50       
  Header.tsx       |   98.62 |    94.28 |     100 |   98.62 | 162,164           
  Help.tsx         |   98.32 |    89.88 |     100 |   98.32 | ...24,381,447-448 
  ...emDisplay.tsx |    61.7 |       36 |     100 |    61.7 | ...42,345,348-354 
  ...ngeDialog.tsx |     100 |      100 |     100 |     100 |                   
  InputPrompt.tsx  |   82.75 |    78.96 |   83.33 |   82.75 | ...1425,1490,1540 
  ...Shortcuts.tsx |   20.87 |      100 |       0 |   20.87 | ...6,49-51,67-125 
  ...Indicator.tsx |     100 |    91.42 |     100 |     100 | 65,74             
  ...firmation.tsx |   91.42 |      100 |      50 |   91.42 | 26-31             
  MainContent.tsx  |   81.75 |       75 |     100 |   81.75 | ...70-274,282-286 
  ...elsDialog.tsx |   71.05 |    69.11 |   72.72 |   71.05 | ...77,590,601-603 
  MemoryDialog.tsx |    55.1 |    54.54 |   57.14 |    55.1 | ...56,368,381-383 
  ...geDisplay.tsx |       0 |        0 |       0 |       0 | 1-41              
  ModelDialog.tsx  |   80.12 |    63.55 |     100 |   80.12 | ...39-555,612-616 
  ...tsDisplay.tsx |     100 |    97.22 |     100 |     100 | 270               
  ...fications.tsx |   18.18 |      100 |       0 |   18.18 | 15-58             
  ...onsDialog.tsx |    2.13 |      100 |       0 |    2.13 | 62-133,148-1004   
  ...ryDisplay.tsx |     100 |      100 |     100 |     100 |                   
  ...icePrompt.tsx |   92.64 |    85.71 |     100 |   92.64 | 102-106,134-139   
  PrepareLabel.tsx |   91.66 |    77.27 |     100 |   91.66 | 73-75,77-79,110   
  ...atePrompt.tsx |    8.57 |      100 |       0 |    8.57 | 24-55,58-134      
  ...geDisplay.tsx |     100 |      100 |     100 |     100 |                   
  ...ngDisplay.tsx |   21.42 |      100 |       0 |   21.42 | 13-39             
  ...hProgress.tsx |   85.25 |    88.46 |     100 |   85.25 | 121-147           
  ...dSelector.tsx |   41.26 |    61.53 |   71.42 |   41.26 | ...74-472,476-520 
  ...ionPicker.tsx |   78.43 |    66.66 |     100 |   78.43 | ...20-422,444-466 
  ...onPreview.tsx |   92.42 |    84.37 |     100 |   92.42 | ...,70-71,143-145 
  ...ryDisplay.tsx |     100 |      100 |     100 |     100 |                   
  ...putPrompt.tsx |   72.56 |       80 |      40 |   72.56 | ...06-109,114-117 
  ...ngsDialog.tsx |   66.92 |    73.21 |     100 |   66.92 | ...12-820,826-827 
  ...ionDialog.tsx |    87.8 |      100 |   33.33 |    87.8 | 36-39,44-51       
  ...putPrompt.tsx |    15.9 |      100 |       0 |    15.9 | 20-63             
  ...Indicator.tsx |   57.14 |      100 |       0 |   57.14 | 12-15             
  ...MoreLines.tsx |      28 |      100 |       0 |      28 | 18-40             
  ...ionPicker.tsx |   17.59 |      100 |       0 |   17.59 | 55-172            
  StatsDisplay.tsx |     100 |      100 |     100 |     100 |                   
  ...ineDialog.tsx |   93.69 |    83.92 |     100 |   93.69 | ...11,273,293-295 
  ...yTodoList.tsx |   94.17 |       80 |     100 |   94.17 | 56-57,131-134     
  ...nsDisplay.tsx |   87.25 |       64 |     100 |   87.25 | ...45-147,154-156 
  ThemeDialog.tsx  |   89.95 |    46.15 |      75 |   89.95 | ...71-173,243-245 
  Tips.tsx         |   93.54 |       75 |     100 |   93.54 | 39-40             
  TodoDisplay.tsx  |     100 |      100 |     100 |     100 |                   
  ...tsDisplay.tsx |     100 |     87.5 |     100 |     100 | 31-32             
  TrustDialog.tsx  |     100 |    81.81 |     100 |     100 | 71-86             
  ...ification.tsx |   36.36 |      100 |       0 |   36.36 | 15-22             
  ...ackDialog.tsx |    7.84 |      100 |       0 |    7.84 | 24-134            
 ...nts/agent-view |   38.33 |    70.83 |   36.36 |   38.33 |                   
  ...atContent.tsx |    8.79 |      100 |       0 |    8.79 | 53-265,271-273    
  ...tChatView.tsx |   21.05 |      100 |       0 |   21.05 | 21-39             
  ...tComposer.tsx |    9.95 |      100 |       0 |    9.95 | 57-308            
  AgentFooter.tsx  |   17.07 |      100 |       0 |   17.07 | 28-66             
  AgentHeader.tsx  |   15.38 |      100 |       0 |   15.38 | 27-64             
  AgentTabBar.tsx  |    87.8 |    27.27 |     100 |    87.8 | ...,85,98-106,124 
  ...oryAdapter.ts |     100 |    91.83 |     100 |     100 | 103,109-110,138   
  index.ts         |       0 |        0 |       0 |       0 | 1-12              
 ...mponents/arena |   45.72 |    70.53 |   60.86 |   45.72 |                   
  ArenaCards.tsx   |   73.06 |    71.79 |   85.71 |   73.06 | ...83-185,321-326 
  ...ectDialog.tsx |   83.48 |    69.86 |   88.88 |   83.48 | ...88-392,409-410 
  ...artDialog.tsx |   10.15 |      100 |       0 |   10.15 | 27-161            
  ...tusDialog.tsx |    5.63 |      100 |       0 |    5.63 | 33-75,80-288      
  ...topDialog.tsx |    6.17 |      100 |       0 |    6.17 | 33-213            
 ...ackground-view |   75.63 |    84.44 |   85.29 |   75.63 |                   
  ...sksDialog.tsx |   70.92 |    80.39 |   76.19 |   70.92 | ...1118,1194-1196 
  ...TasksPill.tsx |   63.75 |    86.95 |     100 |   63.75 | 44,86-106,114-122 
  ...gentPanel.tsx |   99.53 |    93.18 |     100 |   99.53 | 123               
 ...nts/extensions |   45.28 |    33.33 |      60 |   45.28 |                   
  ...gerDialog.tsx |   44.31 |    34.14 |      75 |   44.31 | ...71-480,483-488 
  index.ts         |       0 |        0 |       0 |       0 | 1-9               
  types.ts         |     100 |      100 |     100 |     100 |                   
 ...tensions/steps |   54.88 |    94.23 |   66.66 |   54.88 |                   
  ...ctionStep.tsx |   95.12 |    92.85 |   85.71 |   95.12 | 84-86,89          
  ...etailStep.tsx |    6.18 |      100 |       0 |    6.18 | 17-128            
  ...nListStep.tsx |   88.43 |    94.73 |      80 |   88.43 | 52-53,59-72,106   
  ...electStep.tsx |   13.46 |      100 |       0 |   13.46 | 20-70             
  ...nfirmStep.tsx |   19.56 |      100 |       0 |   19.56 | 23-65             
  index.ts         |     100 |      100 |     100 |     100 |                   
 ...mponents/hooks |   68.67 |    69.07 |   69.56 |   68.67 |                   
  ...etailStep.tsx |   74.68 |    66.66 |   66.66 |   74.68 | ...71-184,188-201 
  ...etailStep.tsx |    87.4 |    73.68 |     100 |    87.4 | 41-42,99-113,119  
  ...abledStep.tsx |     100 |      100 |     100 |     100 |                   
  ...sListStep.tsx |     100 |      100 |     100 |     100 |                   
  ...entDialog.tsx |   34.51 |    47.05 |   42.85 |   34.51 | ...78,482-495,499 
  constants.ts     |     100 |      100 |     100 |     100 |                   
  index.ts         |       0 |        0 |       0 |       0 | 1-13              
  types.ts         |     100 |      100 |     100 |     100 |                   
 ...components/mcp |   20.98 |    86.36 |   83.33 |   20.98 |                   
  ...ealthPill.tsx |   68.42 |    85.71 |     100 |   68.42 | 40-46             
  ...entDialog.tsx |    3.64 |      100 |       0 |    3.64 | 41-717            
  constants.ts     |     100 |      100 |     100 |     100 |                   
  index.ts         |       0 |        0 |       0 |       0 | 1-30              
  types.ts         |     100 |      100 |     100 |     100 |                   
  utils.ts         |   95.83 |    88.88 |     100 |   95.83 | 16,20,109-110     
 ...ents/mcp/steps |   26.74 |    54.54 |   42.85 |   26.74 |                   
  ...icateStep.tsx |    5.88 |      100 |       0 |    5.88 | 40-55,58-296      
  ...electStep.tsx |   10.95 |      100 |       0 |   10.95 | 16-88             
  ...etailStep.tsx |    5.26 |      100 |       0 |    5.26 | 31-247            
  ...rListStep.tsx |   75.18 |    59.37 |     100 |   75.18 | ...53-158,169-173 
  ...etailStep.tsx |   10.41 |      100 |       0 |   10.41 | ...1,67-79,82-139 
  ToolListStep.tsx |   69.02 |       50 |     100 |   69.02 | ...22,125,134-143 
 ...nents/messages |   81.81 |     79.2 |   72.22 |   81.81 |                   
  ...ionDialog.tsx |   80.84 |     77.6 |    62.5 |   80.84 | ...98,516,534-536 
  BtwMessage.tsx   |     100 |      100 |     100 |     100 |                   
  ...upDisplay.tsx |   97.67 |    83.72 |     100 |   97.67 | 119,142,150       
  ...onMessage.tsx |   91.93 |    82.35 |     100 |   91.93 | 57-59,61,63       
  ...nMessages.tsx |   79.06 |      100 |      70 |   79.06 | ...51-264,268-280 
  DiffRenderer.tsx |   93.19 |    86.17 |     100 |   93.19 | ...09,237-238,304 
  ...tsDisplay.tsx |   97.82 |    77.27 |     100 |   97.82 | 87,89             
  ...usMessage.tsx |   58.41 |     7.69 |      50 |   58.41 | ...12-113,140-143 
  ...ssMessage.tsx |    12.5 |      100 |       0 |    12.5 | 18-59             
  ...edMessage.tsx |   16.66 |      100 |       0 |   16.66 | 22-38             
  ...sMessages.tsx |   55.67 |       40 |   28.57 |   55.67 | ...20-125,133-145 
  ...ryMessage.tsx |   14.28 |      100 |       0 |   14.28 | 23-62             
  ...onMessage.tsx |   81.02 |    69.23 |   33.33 |   81.02 | ...24-426,433-435 
  ...upMessage.tsx |      84 |    93.61 |     100 |      84 | ...56-383,405-420 
  ToolMessage.tsx  |   88.84 |    75.71 |    92.3 |   88.84 | ...44-749,776-778 
 ...ponents/shared |   85.36 |    78.48 |   95.77 |   85.36 |                   
  ...ctionList.tsx |   99.03 |    95.65 |     100 |   99.03 | 85                
  ...tonSelect.tsx |     100 |      100 |     100 |     100 |                   
  EnumSelector.tsx |     100 |    96.42 |     100 |     100 | 58                
  MaxSizedBox.tsx  |   83.01 |    86.25 |   88.88 |   83.01 | ...12-513,618-619 
  MultiSelect.tsx  |   84.31 |    74.19 |     100 |   84.31 | ...37,193-195,205 
  ...tonSelect.tsx |     100 |      100 |     100 |     100 |                   
  ...eSelector.tsx |     100 |       60 |     100 |     100 | 40-45             
  TextInput.tsx    |   77.01 |    48.78 |      80 |   77.01 | ...08-212,224-230 
  ...apsedTime.tsx |     100 |      100 |     100 |     100 |                   
  ...Indicator.tsx |     100 |      100 |     100 |     100 |                   
  text-buffer.ts   |   83.68 |    78.55 |   97.61 |   83.68 | ...2270-2272,2368 
  ...er-actions.ts |   86.71 |    67.79 |     100 |   86.71 | ...07-608,809-811 
 ...ents/subagents |   30.87 |        0 |       0 |   30.87 |                   
  constants.ts     |     100 |      100 |     100 |     100 |                   
  index.ts         |       0 |        0 |       0 |       0 | 1-11              
  reducers.tsx     |    12.1 |      100 |       0 |    12.1 | 33-190            
  types.ts         |     100 |      100 |     100 |     100 |                   
  utils.ts         |   10.95 |      100 |       0 |   10.95 | ...1,56-57,60-102 
 ...bagents/create |    9.13 |      100 |       0 |    9.13 |                   
  ...ionWizard.tsx |    7.28 |      100 |       0 |    7.28 | 34-299            
  ...rSelector.tsx |   14.75 |      100 |       0 |   14.75 | 26-85             
  ...onSummary.tsx |    4.26 |      100 |       0 |    4.26 | 27-331            
  ...tionInput.tsx |    8.63 |      100 |       0 |    8.63 | 23-177            
  ...dSelector.tsx |   33.33 |      100 |       0 |   33.33 | 20-21,26-27,36-63 
  ...nSelector.tsx |    37.5 |      100 |       0 |    37.5 | 20-21,26-27,36-58 
  ...EntryStep.tsx |   12.76 |      100 |       0 |   12.76 | 34-78             
  ToolSelector.tsx |    4.16 |      100 |       0 |    4.16 | 31-253            
 ...bagents/manage |   21.51 |    59.52 |   27.27 |   21.51 |                   
  ...ctionStep.tsx |   10.25 |      100 |       0 |   10.25 | 21-103            
  ...eleteStep.tsx |   20.93 |      100 |       0 |   20.93 | 23-62             
  ...tEditStep.tsx |   25.53 |      100 |       0 |   25.53 | ...2,37-38,51-124 
  ...ctionStep.tsx |   35.42 |    59.52 |     100 |   35.42 | ...20-432,437-439 
  ...iewerStep.tsx |   13.72 |      100 |       0 |   13.72 | 18-73             
  ...gerDialog.tsx |    6.74 |      100 |       0 |    6.74 | 35-341            
 ...mponents/views |   42.16 |    69.23 |   21.42 |   42.16 |                   
  ContextUsage.tsx |     4.7 |      100 |       0 |     4.7 | ...52-167,170-456 
  DoctorReport.tsx |     9.8 |      100 |       0 |     9.8 | 25-54,57-131      
  ...sionsList.tsx |   87.69 |    73.68 |     100 |   87.69 | 65-72             
  McpStatus.tsx    |   89.53 |    60.52 |     100 |   89.53 | ...72,175-177,262 
  SkillsList.tsx   |   27.27 |      100 |       0 |   27.27 | 18-35             
  ToolsList.tsx    |     100 |      100 |     100 |     100 |                   
 src/ui/contexts   |   77.11 |    77.66 |   80.35 |   77.11 |                   
  ...ewContext.tsx |    64.7 |    85.71 |      50 |    64.7 | ...22-225,231-241 
  AppContext.tsx   |      80 |       50 |     100 |      80 | 19-20             
  ...ewContext.tsx |   95.18 |    67.56 |      50 |   95.18 | ...94-195,222-226 
  ...deContext.tsx |     100 |      100 |     100 |     100 |                   
  ...igContext.tsx |   81.81 |       50 |     100 |   81.81 | 15-16             
  ...ssContext.tsx |   81.88 |    82.26 |     100 |   81.88 | ...1153,1159-1161 
  ...owContext.tsx |   89.28 |       80 |   66.66 |   89.28 | 34,47-48,60-62    
  ...deContext.tsx |     100 |      100 |      50 |     100 |                   
  ...onContext.tsx |   43.28 |     62.5 |    62.5 |   43.28 | ...56-259,263-266 
  ...gsContext.tsx |   83.33 |       50 |     100 |   83.33 | 17-18             
  ...usContext.tsx |     100 |      100 |     100 |     100 |                   
  ...ngContext.tsx |   71.42 |       50 |     100 |   71.42 | 17-20             
  ...utContext.tsx |   85.71 |      100 |   66.66 |   85.71 | 13-14             
  ...nsContext.tsx |   88.23 |       50 |     100 |   88.23 | 113-114           
  ...teContext.tsx |   86.66 |       50 |     100 |   86.66 | 177-178           
  ...deContext.tsx |   76.08 |    72.72 |     100 |   76.08 | 47-48,52-59,77-78 
 src/ui/editors    |   93.33 |    85.71 |   66.66 |   93.33 |                   
  ...ngsManager.ts |   93.33 |    85.71 |   66.66 |   93.33 | 49,63-64          
 src/ui/hooks      |   82.33 |    82.39 |   86.66 |   82.33 |                   
  ...dProcessor.ts |   83.12 |    82.56 |     100 |   83.12 | ...88-389,408-435 
  keyToAnsi.ts     |    3.92 |      100 |       0 |    3.92 | 19-77             
  ...dProcessor.ts |    94.8 |    70.58 |     100 |    94.8 | ...76-277,282-283 
  ...dProcessor.ts |   75.75 |    63.01 |   61.53 |   75.75 | ...84,908,927-931 
  ...amingState.ts |   12.22 |      100 |       0 |   12.22 | 54-157            
  ...agerDialog.ts |   88.23 |      100 |     100 |   88.23 | 20,24             
  ...ationFrame.ts |      32 |       60 |     100 |      32 | 42-44,51-90       
  ...odeCommand.ts |   58.82 |      100 |     100 |   58.82 | 28,33-48          
  ...enaCommand.ts |      85 |      100 |     100 |      85 | 23-24,29          
  ...aInProcess.ts |   19.81 |    66.66 |      25 |   19.81 | 57-175            
  ...Completion.ts |   92.77 |    89.09 |     100 |   92.77 | ...86-187,220-223 
  ...ifications.ts |   92.07 |    96.29 |     100 |   92.07 | 116-124           
  ...tIndicator.ts |     100 |    93.75 |     100 |     100 | 63                
  ...waySummary.ts |   96.22 |    69.69 |     100 |   96.22 | 125-127,169       
  ...ndTaskView.ts |   94.21 |    76.08 |     100 |   94.21 | 122-126,213,219   
  ...ketedPaste.ts |    23.8 |      100 |       0 |    23.8 | 19-37             
  ...nchCommand.ts |   94.36 |    74.35 |     100 |   94.36 | ...60,168-169,209 
  ...ompletion.tsx |   95.95 |    82.75 |     100 |   95.95 | ...22-223,225-226 
  ...dMigration.ts |   90.62 |       75 |     100 |   90.62 | 38-40             
  useCompletion.ts |    92.4 |     87.5 |     100 |    92.4 | 68-69,93-94,98-99 
  ...nitMessage.ts |     100 |      100 |     100 |     100 |                   
  ...extualTips.ts |   76.92 |       50 |     100 |   76.92 | 55,68,71-75,88-96 
  ...eteCommand.ts |   78.53 |    88.57 |     100 |   78.53 | ...96-104,112-113 
  ...ialogClose.ts |   15.38 |      100 |     100 |   15.38 | 83-148            
  ...oublePress.ts |   53.12 |       75 |     100 |   53.12 | 33-35,41-54       
  ...orSettings.ts |     100 |      100 |     100 |     100 |                   
  ...Completion.ts |   99.12 |     97.7 |     100 |   99.12 | 182-183           
  ...ionUpdates.ts |   93.45 |     92.3 |     100 |   93.45 | ...83-287,300-306 
  ...agerDialog.ts |   88.88 |      100 |     100 |   88.88 | 21,25             
  ...backDialog.ts |   54.47 |       50 |   33.33 |   54.47 | ...69-171,193-194 
  useFocus.ts      |     100 |      100 |     100 |     100 |                   
  ...olderTrust.ts |     100 |      100 |     100 |     100 |                   
  ...ggestions.tsx |   89.15 |     62.5 |      50 |   89.15 | ...22-124,149-150 
  ...miniStream.ts |   76.96 |       74 |   91.66 |   76.96 | ...2461,2474-2482 
  ...BranchName.ts |    90.9 |     92.3 |     100 |    90.9 | 19-20,55-58       
  ...oryManager.ts |   93.15 |    93.75 |     100 |   93.15 | 44,107-110        
  ...ooksDialog.ts |    87.5 |      100 |     100 |    87.5 | 19,23             
  ...stListener.ts |     100 |      100 |     100 |     100 |                   
  ...nAuthError.ts |   76.19 |       50 |     100 |   76.19 | 39-40,43-45       
  ...putHistory.ts |   92.59 |    85.71 |     100 |   92.59 | 63-64,72,94-96    
  ...storyStore.ts |     100 |    94.11 |     100 |     100 | 69                
  useKeypress.ts   |     100 |      100 |     100 |     100 |                   
  ...rdProtocol.ts |   36.36 |      100 |       0 |   36.36 | 24-31             
  ...unchEditor.ts |    9.67 |      100 |       0 |    9.67 | 11-32,39-90       
  ...gIndicator.ts |     100 |      100 |     100 |     100 |                   
  useLogger.ts     |   21.05 |      100 |       0 |   21.05 | 15-37             
  useMCPHealth.ts  |   63.15 |       75 |      50 |   63.15 | 42-52,64-67       
  ...elsCommand.ts |     100 |      100 |     100 |     100 |                   
  useMcpDialog.ts  |    87.5 |      100 |     100 |    87.5 | 19,23             
  ...moryDialog.ts |    87.5 |      100 |     100 |    87.5 | 19,23             
  ...oryMonitor.ts |     100 |      100 |     100 |     100 |                   
  ...ssageQueue.ts |     100 |      100 |     100 |     100 |                   
  ...delCommand.ts |     100 |       75 |     100 |     100 | 22                
  ...raseCycler.ts |   84.74 |    76.47 |     100 |   84.74 | ...49,52-53,69-71 
  ...derUpdates.ts |   86.38 |    77.19 |     100 |   86.38 | ...22,281-293,341 
  useQwenAuth.ts   |     100 |      100 |     100 |     100 |                   
  ...lScheduler.ts |    84.7 |    93.33 |     100 |    84.7 | ...71-276,372-382 
  ...oryCommand.ts |       0 |        0 |       0 |       0 | 1-7               
  ...umeCommand.ts |   97.08 |    83.33 |     100 |   97.08 | 103-104,133       
  ...ompletion.tsx |   90.59 |    83.33 |     100 |   90.59 | ...01,104,137-140 
  ...ectionList.ts |   96.98 |    95.69 |     100 |   96.98 | ...83-184,238-241 
  ...sionPicker.ts |   92.02 |    89.47 |     100 |   92.02 | ...99-501,503-505 
  ...earchInput.ts |     100 |      100 |     100 |     100 |                   
  ...ngsCommand.ts |   18.75 |      100 |       0 |   18.75 | 10-25             
  ...ellHistory.ts |   91.74 |    79.41 |     100 |   91.74 | ...74,122-123,133 
  ...oryCommand.ts |       0 |        0 |       0 |       0 | 1-73              
  ...Completion.ts |   82.67 |    85.41 |   94.73 |   82.67 | ...68-670,678-714 
  ...tateAndRef.ts |     100 |      100 |     100 |     100 |                   
  useStatusLine.ts |   97.67 |    91.66 |     100 |   97.67 | ...28-332,344-347 
  ...eateDialog.ts |   88.23 |      100 |     100 |   88.23 | 14,18             
  ...tification.ts |     100 |    85.71 |     100 |     100 | 47                
  ...alProgress.ts |   53.06 |       50 |   66.66 |   53.06 | ...53,61-68,79-85 
  ...rminalSize.ts |   76.19 |      100 |      50 |   76.19 | 21-25             
  ...emeCommand.ts |   67.01 |    29.41 |     100 |   67.01 | ...10-111,115-116 
  useTimer.ts      |   88.09 |    85.71 |     100 |   88.09 | 44-45,51-53       
  ...lMigration.ts |       0 |        0 |       0 |       0 |                   
  ...rustModify.ts |     100 |      100 |     100 |     100 |                   
  ...elcomeBack.ts |   87.36 |     90.9 |     100 |   87.36 | ...,94-96,114-115 
  vim.ts           |   83.77 |    80.31 |     100 |   83.77 | ...55,759-767,776 
 src/ui/layouts    |   89.72 |     87.5 |     100 |   89.72 |                   
  ...AppLayout.tsx |   89.88 |     87.5 |     100 |   89.88 | 51-53,93-98       
  ...AppLayout.tsx |   89.47 |     87.5 |     100 |   89.47 | 58-63             
 ...i/manageModels |   93.61 |       48 |     100 |   93.61 |                   
  manageModels.ts  |   93.61 |       48 |     100 |   93.61 | ...63-166,179,209 
 src/ui/models     |   80.24 |    79.16 |   71.42 |   80.24 |                   
  ...ableModels.ts |   80.24 |    79.16 |   71.42 |   80.24 | ...,61-71,123-125 
 ...noninteractive |     100 |      100 |    7.14 |     100 |                   
  ...eractiveUi.ts |     100 |      100 |    7.14 |     100 |                   
 src/ui/state      |   94.91 |    81.81 |     100 |   94.91 |                   
  extensions.ts    |   94.91 |    81.81 |     100 |   94.91 | 68-69,88          
 src/ui/themes     |   98.53 |    70.58 |     100 |   98.53 |                   
  ansi-light.ts    |     100 |      100 |     100 |     100 |                   
  ansi.ts          |     100 |      100 |     100 |     100 |                   
  atom-one-dark.ts |     100 |      100 |     100 |     100 |                   
  ayu-light.ts     |     100 |      100 |     100 |     100 |                   
  ayu.ts           |     100 |      100 |     100 |     100 |                   
  color-utils.ts   |     100 |      100 |     100 |     100 |                   
  default-light.ts |     100 |      100 |     100 |     100 |                   
  default.ts       |     100 |      100 |     100 |     100 |                   
  ...inal-theme.ts |   88.59 |    85.96 |     100 |   88.59 | ...57-261,266-270 
  dracula.ts       |     100 |      100 |     100 |     100 |                   
  github-dark.ts   |     100 |      100 |     100 |     100 |                   
  github-light.ts  |     100 |      100 |     100 |     100 |                   
  googlecode.ts    |     100 |      100 |     100 |     100 |                   
  no-color.ts      |     100 |      100 |     100 |     100 |                   
  qwen-dark.ts     |     100 |      100 |     100 |     100 |                   
  qwen-light.ts    |     100 |      100 |     100 |     100 |                   
  ...tic-tokens.ts |     100 |      100 |     100 |     100 |                   
  ...-of-purple.ts |     100 |      100 |     100 |     100 |                   
  theme-manager.ts |   87.98 |    82.89 |     100 |   87.98 | ...48-357,362-363 
  theme.ts         |     100 |    38.02 |     100 |     100 | ...34-449,457-461 
  xcode.ts         |     100 |      100 |     100 |     100 |                   
 src/ui/utils      |   83.91 |    82.93 |   92.56 |   83.91 |                   
  ...Colorizer.tsx |   79.53 |    83.78 |     100 |   79.53 | ...51-152,249-275 
  ...nRenderer.tsx |   68.83 |    70.14 |      50 |   68.83 | ...52-254,274-293 
  ...wnDisplay.tsx |   86.01 |    87.41 |     100 |   86.01 | ...87,704,729-754 
  ...idDiagram.tsx |   87.79 |    95.34 |     100 |   87.79 | 156-179           
  ...eRenderer.tsx |   92.08 |    80.45 |      95 |   92.08 | ...76-679,723-728 
  ...dWorkUtils.ts |     100 |      100 |     100 |     100 |                   
  ...boardUtils.ts |   59.61 |    58.82 |     100 |   59.61 | ...,86-88,107-149 
  commandUtils.ts  |    95.9 |    88.42 |     100 |    95.9 | ...62,164-165,289 
  computeStats.ts  |     100 |      100 |     100 |     100 |                   
  customBanner.ts  |   90.68 |    91.22 |     100 |   90.68 | ...13,324-327,334 
  displayUtils.ts  |   88.37 |    72.22 |     100 |   88.37 | 23,25,29,31,33    
  formatters.ts    |   95.23 |    98.27 |     100 |   95.23 | 117-120           
  gradientUtils.ts |     100 |      100 |     100 |     100 |                   
  highlight.ts     |     100 |      100 |     100 |     100 |                   
  ...oryMapping.ts |     100 |    94.28 |     100 |     100 | 29,51             
  historyUtils.ts  |   94.11 |       94 |     100 |   94.11 | 94-97             
  isNarrowWidth.ts |     100 |      100 |     100 |     100 |                   
  ...olDetector.ts |    8.23 |      100 |       0 |    8.23 | ...31-132,135-136 
  latexRenderer.ts |   94.95 |     73.8 |     100 |   94.95 | ...76-178,184-187 
  layoutUtils.ts   |     100 |      100 |     100 |     100 |                   
  ...ightLoader.ts |     100 |    89.47 |     100 |     100 | 81,110            
  ...nUtilities.ts |   69.84 |    85.71 |     100 |   69.84 | 75-91,100-101     
  ...ToolGroups.ts |   98.66 |    96.77 |     100 |   98.66 | 48-49             
  ...geRenderer.ts |   86.23 |    69.06 |   95.12 |   86.23 | ...1284,1324-1330 
  ...alRenderer.ts |   86.69 |     71.9 |     100 |   86.69 | ...1476,1513-1519 
  ...lsBySource.ts |     100 |    95.23 |     100 |     100 | 84                
  osc8.ts          |   94.71 |    87.41 |     100 |   94.71 | ...43,428,432-433 
  ...mConstants.ts |     100 |      100 |     100 |     100 |                   
  restoreGoal.ts   |   98.98 |    97.14 |     100 |   98.98 | 94                
  ...storyUtils.ts |   61.06 |    69.62 |      90 |   61.06 | ...64,412,417-439 
  ...ickerUtils.ts |     100 |      100 |     100 |     100 |                   
  ...izedOutput.ts |   94.94 |      100 |   88.88 |   94.94 | 112-117           
  ...wOptimizer.ts |     100 |    96.77 |     100 |     100 | 69                
  terminalSetup.ts |    4.37 |      100 |       0 |    4.37 | 44-393            
  textUtils.ts     |   97.35 |    94.38 |   91.66 |   97.35 | ...50-251,386-387 
  todoSnapshot.ts  |   89.11 |    93.33 |     100 |   89.11 | ...,66-78,180-181 
  updateCheck.ts   |     100 |    80.95 |     100 |     100 | 30-42             
 ...i/utils/export |   56.77 |     40.8 |   79.41 |   56.77 |                   
  collect.ts       |   55.92 |    50.58 |   86.36 |   55.92 | ...25-640,642-647 
  index.ts         |     100 |      100 |     100 |     100 |                   
  normalize.ts     |   57.47 |    20.51 |      80 |   57.47 | ...09-310,324-359 
  types.ts         |       0 |        0 |       0 |       0 | 1                 
  utils.ts         |      40 |      100 |       0 |      40 | 11-13             
 ...ort/formatters |    3.38 |      100 |       0 |    3.38 |                   
  html.ts          |    9.61 |      100 |       0 |    9.61 | ...28,34-76,82-84 
  json.ts          |      50 |      100 |       0 |      50 | 14-15             
  jsonl.ts         |     3.5 |      100 |       0 |     3.5 | 14-76             
  markdown.ts      |    0.94 |      100 |       0 |    0.94 | 13-295            
 src/utils         |   75.88 |    89.55 |   93.38 |   75.88 |                   
  acpModelUtils.ts |     100 |      100 |     100 |     100 |                   
  apiPreconnect.ts |   96.72 |    97.14 |     100 |   96.72 | 165-168           
  checks.ts        |   33.33 |      100 |       0 |   33.33 | 23-28             
  cleanup.ts       |   84.12 |    93.33 |      80 |   84.12 | 75,106-115        
  commands.ts      |     100 |      100 |     100 |     100 |                   
  commentJson.ts   |   87.17 |     90.9 |     100 |   87.17 | 64-73             
  ...Calculator.ts |     100 |      100 |     100 |     100 |                   
  deepMerge.ts     |     100 |       90 |     100 |     100 | 41-43,49          
  ...ScopeUtils.ts |   97.56 |    88.88 |     100 |   97.56 | 67                
  doctorChecks.ts  |   71.06 |       75 |     100 |   71.06 | ...95-301,325-341 
  ...putCapture.ts |   90.65 |    86.17 |     100 |   90.65 | ...72,370,372-373 
  ...arResolver.ts |   94.28 |       88 |     100 |   94.28 | 28-29,125-126     
  errors.ts        |   98.67 |    96.36 |     100 |   98.67 | 67-68             
  events.ts        |     100 |      100 |     100 |     100 |                   
  gitUtils.ts      |   91.91 |    84.61 |     100 |   91.91 | 78-81,124-127     
  ...AutoUpdate.ts |   90.76 |    93.33 |   88.88 |   90.76 | 103-114           
  ...lationInfo.ts |     100 |      100 |     100 |     100 |                   
  languageUtils.ts |   97.89 |    96.42 |     100 |   97.89 | 132-133           
  math.ts          |       0 |        0 |       0 |       0 | 1-15              
  ...iagnostics.ts |   94.57 |    83.01 |   88.88 |   94.57 | ...05,311,315-317 
  ...onfigUtils.ts |     100 |      100 |     100 |     100 |                   
  ...iveHelpers.ts |   96.79 |    93.28 |     100 |   96.79 | ...76-477,575,588 
  osc.ts           |    97.5 |      100 |   88.88 |    97.5 | 195-196           
  package.ts       |   88.88 |       80 |     100 |   88.88 | 33-34             
  processUtils.ts  |     100 |      100 |     100 |     100 |                   
  readStdin.ts     |   79.62 |       90 |      80 |   79.62 | 33-40,52-54       
  relaunch.ts      |   98.07 |    76.92 |     100 |   98.07 | 70                
  resolvePath.ts   |   66.66 |       25 |     100 |   66.66 | 12-13,16,18-19    
  sandbox.ts       |       0 |        0 |       0 |       0 | 1-1047            
  settingsUtils.ts |   82.89 |    90.75 |   89.47 |   82.89 | ...52-663,670-678 
  spawnWrapper.ts  |     100 |      100 |     100 |     100 |                   
  ...upProfiler.ts |   98.46 |    94.52 |     100 |   98.46 | 130-131,305       
  ...upWarnings.ts |     100 |      100 |     100 |     100 |                   
  stdioHelpers.ts  |     100 |       60 |     100 |     100 | 23,32             
  systemInfo.ts    |   90.32 |    89.65 |    87.5 |   90.32 | ...18-219,224-228 
  ...InfoFields.ts |   87.61 |       65 |     100 |   87.61 | ...22-123,144-145 
  ...iffPreview.ts |   94.11 |    83.33 |     100 |   94.11 | 13                
  ...entEmitter.ts |     100 |      100 |     100 |     100 |                   
  ...upWarnings.ts |   91.17 |    82.35 |     100 |   91.17 | 67-68,73-74,77-78 
  version.ts       |     100 |       50 |     100 |     100 | 11                
  windowTitle.ts   |     100 |      100 |     100 |     100 |                   
  ...WithBackup.ts |   63.15 |    81.25 |     100 |   63.15 | 93,118-157        
-------------------|---------|----------|---------|---------|-------------------
Core Package - Full Text Report
-------------------|---------|----------|---------|---------|-------------------
File               | % Stmts | % Branch | % Funcs | % Lines | Uncovered Line #s 
-------------------|---------|----------|---------|---------|-------------------
All files          |    79.2 |    82.76 |   81.77 |    79.2 |                   
 src               |     100 |      100 |     100 |     100 |                   
  index.ts         |     100 |      100 |     100 |     100 |                   
 src/__mocks__/fs  |       0 |        0 |       0 |       0 |                   
  promises.ts      |       0 |        0 |       0 |       0 | 1-48              
 src/agents        |   87.58 |    79.07 |   91.76 |   87.58 |                   
  ...transcript.ts |   92.25 |    85.71 |     100 |   92.25 | ...87,306-307,438 
  ...ent-resume.ts |    82.5 |     71.5 |   77.41 |    82.5 | ...1035-1039,1042 
  ...ound-tasks.ts |    95.4 |    86.48 |     100 |    95.4 | ...55-756,827-828 
  index.ts         |     100 |      100 |     100 |     100 |                   
 src/agents/arena  |   76.54 |    66.87 |   78.72 |   76.54 |                   
  ...gentClient.ts |   79.47 |    88.88 |   81.81 |   79.47 | ...68-183,189-204 
  ArenaManager.ts  |   75.37 |    63.37 |   78.26 |   75.37 | ...1860,1866-1867 
  arena-events.ts  |   64.44 |      100 |      50 |   64.44 | ...71-175,178-183 
  diff-summary.ts  |    87.5 |    72.34 |     100 |    87.5 | ...32-133,137-138 
  index.ts         |     100 |      100 |     100 |     100 |                   
  types.ts         |     100 |      100 |     100 |     100 |                   
 ...gents/backends |   76.29 |    86.15 |   73.04 |   76.29 |                   
  ITermBackend.ts  |   97.97 |    93.93 |     100 |   97.97 | ...78-180,255,307 
  ...essBackend.ts |   91.25 |    90.62 |   86.66 |   91.25 | ...94,249-269,328 
  TmuxBackend.ts   |    90.7 |    76.55 |   97.36 |    90.7 | ...87,697,743-747 
  detect.ts        |   31.25 |      100 |       0 |   31.25 | 34-88             
  index.ts         |     100 |      100 |     100 |     100 |                   
  iterm-it2.ts     |     100 |     92.1 |     100 |     100 | 37-38,106         
  tmux-commands.ts |    6.64 |      100 |    3.03 |    6.64 | ...93-363,386-503 
  types.ts         |     100 |      100 |     100 |     100 |                   
 ...agents/runtime |   81.14 |     76.7 |   71.42 |   81.14 |                   
  agent-context.ts |     100 |      100 |     100 |     100 |                   
  agent-core.ts    |   76.49 |    72.35 |   60.86 |   76.49 | ...1608,1635-1682 
  agent-events.ts  |     100 |      100 |     100 |     100 |                   
  ...t-headless.ts |   81.19 |    71.73 |   60.86 |   81.19 | ...98-399,402-403 
  ...nteractive.ts |   79.71 |    79.62 |      75 |   79.71 | ...54,456,458,461 
  ...statistics.ts |   98.19 |    82.35 |     100 |   98.19 | 127,151,192,225   
  agent-types.ts   |     100 |      100 |     100 |     100 |                   
  index.ts         |     100 |      100 |     100 |     100 |                   
 src/agents/tasks  |     100 |      100 |     100 |     100 |                   
  types.ts         |     100 |      100 |     100 |     100 |                   
 src/config        |   77.22 |    79.73 |   63.34 |   77.22 |                   
  config.ts        |   75.37 |    77.98 |   58.68 |   75.37 | ...3483,3494-3506 
  constants.ts     |     100 |      100 |     100 |     100 |                   
  models.ts        |     100 |      100 |     100 |     100 |                   
  storage.ts       |   95.07 |    93.44 |   89.47 |   95.07 | ...66-267,270-271 
 ...nfirmation-bus |   98.29 |    97.14 |     100 |   98.29 |                   
  message-bus.ts   |   98.14 |    97.05 |     100 |   98.14 | 42-43             
  types.ts         |     100 |      100 |     100 |     100 |                   
 src/core          |   86.28 |    82.27 |   89.86 |   86.28 |                   
  baseLlmClient.ts |   92.35 |    80.85 |   86.66 |   92.35 | ...34,342-356,495 
  client.ts        |   85.49 |    77.12 |   84.84 |   85.49 | ...1735,1774-1777 
  ...tGenerator.ts |    72.1 |    61.11 |     100 |    72.1 | ...63,365,372-375 
  ...lScheduler.ts |   81.53 |    81.77 |   93.47 |   81.53 | ...2425,2477-2481 
  geminiChat.ts    |   89.32 |     84.8 |   91.48 |   89.32 | ...1454,1521-1522 
  geminiRequest.ts |     100 |      100 |     100 |     100 |                   
  ...htProtocol.ts |    9.09 |      100 |       0 |    9.09 | 34-42,45-49,52-87 
  logger.ts        |   87.33 |    87.02 |     100 |   87.33 | ...61-565,611-625 
  ...tyDefaults.ts |     100 |      100 |     100 |     100 |                   
  ...olExecutor.ts |   92.59 |       75 |      50 |   92.59 | 41-42             
  ...on-helpers.ts |   85.71 |    70.58 |     100 |   85.71 | ...90-191,205-214 
  ...issionFlow.ts |   98.59 |    94.73 |     100 |   98.59 | 93                
  prompts.ts       |   89.16 |    86.41 |   76.92 |   89.16 | ...-965,1168-1169 
  tokenLimits.ts   |     100 |    89.47 |     100 |     100 | 51-52             
  ...okTriggers.ts |   99.31 |    90.41 |     100 |   99.31 | 124,135           
  turn.ts          |   96.42 |    88.88 |     100 |   96.42 | ...00,413-414,462 
 ...ntentGenerator |   94.92 |    82.59 |   93.87 |   94.92 |                   
  ...tGenerator.ts |   96.48 |    84.28 |   92.59 |   96.48 | ...01,919-923,963 
  converter.ts     |   94.51 |    80.72 |     100 |   94.51 | ...06-607,617,823 
  index.ts         |       0 |        0 |       0 |       0 | 1-21              
  usage.ts         |     100 |      100 |     100 |     100 |                   
 ...ntentGenerator |   91.53 |    71.64 |   93.33 |   91.53 |                   
  ...tGenerator.ts |      90 |    70.96 |   92.85 |      90 | ...80-286,304-305 
  index.ts         |     100 |       80 |     100 |     100 | 50                
 ...ntentGenerator |    92.1 |    80.38 |   90.32 |    92.1 |                   
  index.ts         |     100 |      100 |     100 |     100 |                   
  ...tGenerator.ts |   92.08 |    80.38 |   90.32 |   92.08 | ...85,895-896,924 
 ...ntentGenerator |   81.66 |    84.08 |    90.9 |   81.66 |                   
  constants.ts     |     100 |      100 |     100 |     100 |                   
  converter.ts     |   76.88 |    82.25 |    87.5 |   76.88 | ...1589,1610-1616 
  errorHandler.ts  |     100 |      100 |     100 |     100 |                   
  index.ts         |   52.38 |    44.44 |      50 |   52.38 | ...77,81-85,89-93 
  ...tGenerator.ts |    66.4 |    70.58 |   88.88 |    66.4 | ...51-157,168-169 
  pipeline.ts      |   93.67 |     84.9 |     100 |   93.67 | ...80-481,489,554 
  ...ureContext.ts |     100 |      100 |     100 |     100 |                   
  ...ingOptions.ts |       0 |        0 |       0 |       0 | 1                 
  ...CallParser.ts |   90.66 |    88.57 |     100 |   90.66 | ...15-319,349-350 
  ...kingParser.ts |     100 |    96.87 |     100 |     100 | 42                
  types.ts         |       0 |        0 |       0 |       0 | 1                 
 ...rator/provider |   96.62 |    88.82 |   95.45 |   96.62 |                   
  dashscope.ts     |   97.14 |    89.02 |   93.33 |   97.14 | ...53-254,330-331 
  deepseek.ts      |   95.55 |    90.56 |     100 |   95.55 | ...31-132,145-146 
  default.ts       |   94.62 |    86.36 |   85.71 |   94.62 | 86-87,157-159     
  index.ts         |     100 |      100 |     100 |     100 |                   
  minimax.ts       |     100 |      100 |     100 |     100 |                   
  mistral.ts       |   96.07 |    73.33 |     100 |   96.07 | 32-33             
  modelscope.ts    |     100 |      100 |     100 |     100 |                   
  openrouter.ts    |     100 |      100 |     100 |     100 |                   
  types.ts         |       0 |        0 |       0 |       0 |                   
 src/extension     |   60.56 |    79.46 |    78.4 |   60.56 |                   
  ...-converter.ts |   62.35 |    47.82 |      90 |   62.35 | ...90-791,800-832 
  ...ionManager.ts |   47.04 |    82.06 |    65.9 |   47.04 | ...1398,1408-1427 
  ...onSettings.ts |   93.46 |    93.05 |     100 |   93.46 | ...17-221,228-232 
  ...-converter.ts |   54.88 |    94.44 |      60 |   54.88 | ...35-146,158-192 
  github.ts        |   44.94 |    88.52 |      60 |   44.94 | ...53-359,398-451 
  index.ts         |     100 |      100 |     100 |     100 |                   
  marketplace.ts   |   97.29 |    93.75 |     100 |   97.29 | ...64,184-185,274 
  npm.ts           |   48.66 |    76.08 |      75 |   48.66 | ...18-420,427-431 
  override.ts      |   94.11 |    88.88 |     100 |   94.11 | 63-64,81-82       
  settings.ts      |   66.26 |      100 |      50 |   66.26 | 81-108,143-149    
  storage.ts       |     100 |      100 |     100 |     100 |                   
  ...ableSchema.ts |     100 |      100 |     100 |     100 |                   
  variables.ts     |   88.75 |    83.33 |     100 |   88.75 | ...28-231,234-237 
 src/followup      |   46.91 |     92.3 |   71.87 |   46.91 |                   
  followupState.ts |      96 |    89.74 |     100 |      96 | 159-161,218-219   
  index.ts         |     100 |      100 |     100 |     100 |                   
  overlayFs.ts     |   95.06 |       84 |     100 |   95.06 | 78,108,122,133    
  speculation.ts   |   13.22 |      100 |   16.66 |   13.22 | 88-458,518-568    
  ...onToolGate.ts |     100 |    96.29 |     100 |     100 | 93                
  ...nGenerator.ts |    38.4 |    95.12 |   33.33 |    38.4 | ...16-318,353-383 
 src/generated     |       0 |        0 |       0 |       0 |                   
  git-commit.ts    |       0 |        0 |       0 |       0 | 1-10              
 src/goals         |   88.38 |    81.14 |   94.11 |   88.38 |                   
  ...eGoalStore.ts |   81.57 |    92.85 |   81.81 |   81.57 | ...37-140,148-156 
  goalHook.ts      |   96.89 |    90.24 |     100 |   96.89 | 93-98             
  goalJudge.ts     |    83.4 |    73.13 |     100 |    83.4 | ...19-320,328-330 
  index.ts         |     100 |      100 |     100 |     100 |                   
 src/hooks         |   83.48 |    84.87 |   86.83 |   83.48 |                   
  ...okRegistry.ts |   86.48 |    77.08 |     100 |   86.48 | ...41-344,362-369 
  ...bortSignal.ts |     100 |      100 |     100 |     100 |                   
  ...terpolator.ts |   96.66 |    93.33 |     100 |   96.66 | 66-67             
  ...HookRunner.ts |   96.68 |    87.23 |     100 |   96.68 | 110-112,231-233   
  ...Aggregator.ts |    96.4 |    90.78 |     100 |    96.4 | ...91,293-294,367 
  ...entHandler.ts |   94.56 |    83.78 |   93.33 |   94.56 | ...38,795-796,806 
  hookPlanner.ts   |   84.13 |    76.59 |      90 |   84.13 | ...38,144,162-173 
  hookRegistry.ts  |   90.17 |    83.33 |     100 |   90.17 | ...33,352,356,360 
  hookRunner.ts    |   58.56 |    71.26 |   66.66 |   58.56 | ...48-749,758-759 
  hookSystem.ts    |   84.57 |      100 |   65.85 |   84.57 | ...21-622,628-629 
  ...HookRunner.ts |   75.51 |     61.9 |      80 |   75.51 | ...05-406,424-425 
  index.ts         |     100 |      100 |     100 |     100 |                   
  ...HookRunner.ts |   93.63 |    89.47 |      90 |   93.63 | ...45-353,427-428 
  ...SkillHooks.ts |   78.75 |       75 |   66.66 |   78.75 | 62-66,137-152     
  ...oksManager.ts |   96.66 |    91.66 |     100 |   96.66 | ...90,209-210,223 
  ssrfGuard.ts     |   77.22 |    85.36 |     100 |   77.22 | ...57,261-267,273 
  stopHookCap.ts   |     100 |      100 |     100 |     100 |                   
  trustedHooks.ts  |       0 |        0 |       0 |       0 | 1-124             
  types.ts         |   91.18 |    92.04 |   85.71 |   91.18 | ...40-441,501-505 
  urlValidator.ts  |     100 |      100 |     100 |     100 |                   
 src/ide           |   74.28 |    83.39 |   78.33 |   74.28 |                   
  constants.ts     |     100 |      100 |     100 |     100 |                   
  detect-ide.ts    |     100 |      100 |     100 |     100 |                   
  ide-client.ts    |    64.2 |    81.48 |   66.66 |    64.2 | ...9-970,999-1007 
  ide-installer.ts |   89.06 |    79.31 |     100 |   89.06 | ...36,143-147,160 
  ideContext.ts    |     100 |      100 |     100 |     100 |                   
  process-utils.ts |   84.84 |    71.79 |     100 |   84.84 | ...37,151,193-194 
  types.ts         |     100 |      100 |     100 |     100 |                   
 src/lsp           |   41.24 |    52.14 |   51.42 |   41.24 |                   
  ...nfigLoader.ts |   70.27 |    35.89 |   94.73 |   70.27 | ...20-422,426-432 
  ...ionFactory.ts |   42.69 |    79.16 |      50 |   42.69 | ...62-413,419-436 
  ...Normalizer.ts |   23.09 |    13.72 |   30.43 |   23.09 | ...04-905,909-924 
  ...verManager.ts |   25.31 |    62.06 |   41.66 |   25.31 | ...85-704,710-740 
  ...eLspClient.ts |   32.77 |       80 |   17.64 |   32.77 | ...84-288,294-295 
  ...LspService.ts |   48.49 |    67.16 |   65.71 |   48.49 | ...1352,1369-1379 
  constants.ts     |     100 |      100 |     100 |     100 |                   
  types.ts         |     100 |      100 |     100 |     100 |                   
 src/mcp           |   78.69 |    75.34 |   75.92 |   78.69 |                   
  constants.ts     |     100 |      100 |     100 |     100 |                   
  ...h-provider.ts |   86.95 |      100 |   33.33 |   86.95 | ...,93,97,101-102 
  ...h-provider.ts |   73.82 |    53.92 |     100 |   73.82 | ...88-895,902-904 
  ...en-storage.ts |   98.62 |    97.72 |     100 |   98.62 | 87-88             
  oauth-utils.ts   |   70.58 |    85.29 |    90.9 |   70.58 | ...70-290,315-344 
  ...n-provider.ts |   89.83 |    95.83 |   45.45 |   89.83 | ...43,147,151-152 
 .../token-storage |   79.52 |    86.66 |   86.36 |   79.52 |                   
  ...en-storage.ts |     100 |      100 |     100 |     100 |                   
  ...en-storage.ts |   82.87 |    82.35 |   92.85 |   82.87 | ...63-173,181-182 
  ...en-storage.ts |     100 |      100 |     100 |     100 |                   
  index.ts         |     100 |      100 |     100 |     100 |                   
  ...en-storage.ts |   68.14 |    82.35 |   64.28 |   68.14 | ...81-295,298-314 
  types.ts         |     100 |      100 |     100 |     100 |                   
 src/memory        |   67.43 |       76 |   65.62 |   67.43 |                   
  const.ts         |     100 |      100 |     100 |     100 |                   
  dream.ts         |   65.65 |    73.33 |      50 |   65.65 | 50,107-148        
  ...entPlanner.ts |   57.84 |    72.72 |   33.33 |   57.84 | ...35,140-147,152 
  entries.ts       |   63.77 |    79.16 |      50 |   63.77 | ...72-180,183-189 
  extract.ts       |    95.2 |    79.16 |     100 |    95.2 | 81-86,125         
  ...entPlanner.ts |   63.08 |    65.71 |   41.17 |   63.08 | ...17,222-223,332 
  ...ionPlanner.ts |       0 |        0 |       0 |       0 | 1                 
  forget.ts        |    45.8 |    61.53 |   44.44 |    45.8 | ...04,211,214-346 
  indexer.ts       |   83.87 |    45.45 |     100 |   83.87 | ...50,56-57,69-70 
  manager.ts       |   75.31 |    81.04 |    75.6 |   75.31 | ...1278,1291-1293 
  memoryAge.ts     |   90.47 |    77.77 |     100 |   90.47 | 50-51             
  paths.ts         |   55.47 |    89.47 |   85.71 |   55.47 | ...,89-90,106-114 
  prompt.ts        |   93.36 |    71.42 |     100 |   93.36 | ...58,161,228-229 
  recall.ts        |   79.56 |    69.38 |   88.88 |   79.56 | ...40-245,269-280 
  ...ceSelector.ts |   91.86 |    77.27 |     100 |   91.86 | ...07,109-110,118 
  scan.ts          |   87.91 |    68.42 |     100 |   87.91 | ...47-48,58,82-87 
  ...entPlanner.ts |    11.5 |      100 |       0 |    11.5 | ...57-192,210-298 
  status.ts        |   10.52 |      100 |       0 |   10.52 | 41-98             
  store.ts         |   94.44 |    83.33 |     100 |   94.44 | 56-57,92-93       
  types.ts         |     100 |      100 |     100 |     100 |                   
 src/mocks         |       0 |        0 |       0 |       0 |                   
  msw.ts           |       0 |        0 |       0 |       0 | 1-9               
 src/models        |   89.31 |    85.55 |    87.5 |   89.31 |                   
  constants.ts     |     100 |      100 |     100 |     100 |                   
  ...tor-config.ts |   90.24 |    91.42 |     100 |   90.24 | 142,148,151-160   
  index.ts         |     100 |      100 |     100 |     100 |                   
  ...nfigErrors.ts |   74.22 |       44 |   84.61 |   74.22 | ...,67-74,106-117 
  ...igResolver.ts |   98.63 |    92.53 |     100 |   98.63 | 161,323,329       
  modelRegistry.ts |     100 |    98.59 |     100 |     100 | 222               
  modelsConfig.ts  |   84.57 |    82.14 |   81.57 |   84.57 | ...1223,1252-1253 
  types.ts         |     100 |      100 |     100 |     100 |                   
 src/output        |     100 |      100 |     100 |     100 |                   
  ...-formatter.ts |     100 |      100 |     100 |     100 |                   
  types.ts         |     100 |      100 |     100 |     100 |                   
 src/permissions   |   71.18 |    88.73 |   48.57 |   71.18 |                   
  index.ts         |     100 |      100 |     100 |     100 |                   
  ...on-manager.ts |   81.42 |    86.66 |      80 |   81.42 | ...29-830,837-846 
  rule-parser.ts   |   95.99 |    93.18 |     100 |   95.99 | ...-864,1013-1015 
  ...-semantics.ts |   58.28 |    85.27 |    30.2 |   58.28 | ...1604-1614,1643 
  types.ts         |     100 |      100 |     100 |     100 |                   
 src/prompts       |   83.63 |      100 |    87.5 |   83.63 |                   
  mcp-prompts.ts   |   18.18 |      100 |       0 |   18.18 | 11-19             
  ...t-registry.ts |     100 |      100 |     100 |     100 |                   
 src/qwen          |   86.01 |    79.48 |   97.18 |   86.01 |                   
  ...tGenerator.ts |   98.64 |    98.18 |     100 |   98.64 | 105-106           
  qwenOAuth2.ts    |   84.99 |    74.81 |   93.33 |   84.99 | ...,985-1001,1031 
  ...kenManager.ts |   83.76 |    76.22 |     100 |   83.76 | ...62-767,788-793 
 src/services      |   85.69 |    84.06 |   90.63 |   85.69 |                   
  ...ionTrailer.ts |     100 |      100 |     100 |     100 |                   
  ...llRegistry.ts |   98.44 |    91.83 |     100 |   98.44 | 268-269           
  ...ionService.ts |    95.6 |    96.36 |     100 |    95.6 | ...32,400,402-406 
  ...ingService.ts |   84.15 |       84 |   82.85 |   84.15 | ...1241,1258-1259 
  ...ttribution.ts |   91.73 |    87.71 |      90 |   91.73 | ...80-685,826-827 
  ...utSlimming.ts |     100 |    96.77 |     100 |     100 | 133,182           
  cronScheduler.ts |   97.56 |    92.98 |     100 |   97.56 | 62-63,77,155      
  ...eryService.ts |   80.43 |    95.45 |      75 |   80.43 | ...19-134,140-141 
  ...oryService.ts |   86.25 |    74.35 |    92.3 |   86.25 | ...46-655,696-699 
  fileReadCache.ts |     100 |      100 |     100 |     100 |                   
  ...temService.ts |      90 |    84.44 |   88.88 |      90 | ...89,191,269-276 
  ...ratedFiles.ts |      96 |    88.23 |     100 |      96 | 119-120,146-147   
  gitInit.ts       |     100 |      100 |     100 |     100 |                   
  gitService.ts    |   68.75 |     92.3 |   55.55 |   68.75 | ...12-122,125-129 
  ...reeService.ts |   73.79 |       70 |   94.87 |   73.79 | ...1365,1393-1394 
  ...ionService.ts |   98.13 |     97.8 |   95.45 |   98.13 | ...32-333,380-381 
  ...orRegistry.ts |   96.54 |    91.73 |     100 |   96.54 | ...70-471,622-623 
  sessionRecap.ts  |   12.04 |      100 |       0 |   12.04 | 49-160            
  ...ionService.ts |   90.05 |    79.71 |   96.55 |   90.05 | ...1272,1276-1277 
  sessionTitle.ts  |   93.87 |    69.81 |     100 |   93.87 | ...33-236,267-268 
  ...ionService.ts |   83.01 |    78.66 |   87.75 |   83.01 | ...1482,1488-1493 
  ...UseSummary.ts |   94.73 |    87.71 |     100 |   94.73 | ...73-175,225-226 
  ...reeCleanup.ts |   14.56 |      100 |   33.33 |   14.56 | 58-185            
 ...icrocompaction |   97.69 |    89.79 |     100 |   97.69 |                   
  microcompact.ts  |   97.69 |    89.79 |     100 |   97.69 | ...68,229,233,314 
 src/skills        |    87.5 |    83.86 |   94.23 |    87.5 |                   
  index.ts         |     100 |      100 |     100 |     100 |                   
  ...activation.ts |     100 |     93.1 |     100 |     100 | 93,112            
  skill-load.ts    |   92.94 |    81.63 |     100 |   92.94 | ...06,226,238-240 
  skill-manager.ts |   83.31 |    79.66 |   90.32 |   83.31 | ...1120,1127-1131 
  skill-paths.ts   |   86.74 |    77.77 |     100 |   86.74 | ...00-101,106-107 
  symlinkScope.ts  |     100 |      100 |     100 |     100 |                   
  types.ts         |     100 |      100 |     100 |     100 |                   
 src/subagents     |   83.13 |    80.24 |   95.23 |   83.13 |                   
  ...tin-agents.ts |     100 |      100 |     100 |     100 |                   
  index.ts         |     100 |      100 |     100 |     100 |                   
  ...-selection.ts |     100 |      100 |     100 |     100 |                   
  ...nt-manager.ts |   77.21 |    72.09 |   92.85 |   77.21 | ...1180,1202-1203 
  types.ts         |     100 |      100 |     100 |     100 |                   
  validation.ts    |   92.46 |    95.18 |     100 |   92.46 | 51-56,69-74,78-83 
 src/telemetry     |   74.59 |     85.9 |   78.77 |   74.59 |                   
  config.ts        |     100 |      100 |     100 |     100 |                   
  constants.ts     |     100 |      100 |     100 |     100 |                   
  ...attributes.ts |   98.13 |       88 |     100 |   98.13 | 185-187           
  ...-exporters.ts |   46.37 |      100 |   44.44 |   46.37 | ...85,88-89,92-93 
  index.ts         |     100 |      100 |     100 |     100 |                   
  ...t.circular.ts |       0 |        0 |       0 |       0 | 1-111             
  ...-processor.ts |   93.93 |    90.21 |   94.11 |   93.93 | ...75-280,299-300 
  ...t.circular.ts |       0 |        0 |       0 |       0 | 1-128             
  loggers.ts       |    51.9 |       64 |   57.77 |    51.9 | ...1214,1231-1251 
  metrics.ts       |    74.9 |    82.95 |   74.54 |    74.9 | ...58-978,981-992 
  sanitize.ts      |      80 |    83.33 |     100 |      80 | 35-36,41-42       
  sdk.ts           |   90.45 |    83.56 |   76.92 |   90.45 | ...17-318,338-342 
  ...on-context.ts |     100 |      100 |     100 |     100 |                   
  ...on-tracing.ts |   90.69 |    87.87 |     100 |   90.69 | ...67-471,482-485 
  ...etry-utils.ts |     100 |      100 |     100 |     100 |                   
  ...l-decision.ts |     100 |      100 |     100 |     100 |                   
  ...e-id-utils.ts |     100 |      100 |     100 |     100 |                   
  tracer.ts        |   98.61 |    89.36 |     100 |   98.61 | 53,104            
  types.ts         |   79.17 |    85.83 |   83.33 |   79.17 | ...1149,1152-1181 
  uiTelemetry.ts   |   92.97 |    96.96 |   81.25 |   92.97 | ...93-194,200-207 
 ...ry/qwen-logger |   68.24 |    79.56 |   64.91 |   68.24 |                   
  event-types.ts   |       0 |        0 |       0 |       0 |                   
  qwen-logger.ts   |   68.24 |    79.34 |   64.28 |   68.24 | ...1055,1093-1094 
 src/test-utils    |   93.16 |    95.83 |   73.52 |   93.16 |                   
  config.ts        |     100 |      100 |     100 |     100 |                   
  ...st-helpers.ts |   94.11 |       90 |     100 |   94.11 | 69-70             
  index.ts         |     100 |      100 |     100 |     100 |                   
  mock-tool.ts     |   91.19 |    97.05 |   68.96 |   91.19 | ...38,202-203,216 
  ...aceContext.ts |     100 |      100 |     100 |     100 |                   
 src/tools         |   77.74 |    81.46 |   86.14 |   77.74 |                   
  ...erQuestion.ts |   88.93 |    76.74 |    90.9 |   88.93 | ...39-340,347-348 
  cron-create.ts   |   97.75 |    88.88 |   83.33 |   97.75 | 30-31             
  cron-delete.ts   |   96.82 |      100 |   83.33 |   96.82 | 26-27             
  cron-list.ts     |   96.66 |      100 |   83.33 |   96.66 | 25-26             
  diffOptions.ts   |     100 |      100 |     100 |     100 |                   
  edit.ts          |   80.52 |    85.98 |   73.33 |   80.52 | ...15-716,803-853 
  ...r-worktree.ts |   82.43 |    68.75 |    87.5 |   82.43 | ...67-170,236-237 
  exit-worktree.ts |   83.47 |       84 |    90.9 |   83.47 | ...80-281,286-299 
  exitPlanMode.ts  |   85.09 |    85.71 |     100 |   85.09 | ...60-163,177-189 
  glob.ts          |   90.63 |    88.33 |   84.61 |   90.63 | ...28,171,302,305 
  grep.ts          |   79.19 |    85.71 |   78.94 |   79.19 | ...20,560,569-576 
  ls.ts            |   96.74 |    90.27 |     100 |   96.74 | 176-181,212,216   
  lsp.ts           |   72.77 |    60.09 |   90.32 |   72.77 | ...1211,1213-1214 
  ...nt-manager.ts |   69.73 |    75.29 |   71.42 |   69.73 | ...29-732,749-786 
  mcp-client.ts    |   33.18 |    77.41 |   66.66 |   33.18 | ...1490,1494-1497 
  mcp-tool.ts      |   90.98 |    88.88 |   96.42 |   90.98 | ...95-596,646-647 
  memory-config.ts |       0 |        0 |       0 |       0 | 1-47              
  ...iable-tool.ts |     100 |    84.61 |     100 |     100 | 102,109           
  monitor.ts       |   92.36 |    83.94 |      92 |   92.36 | ...29,558-561,574 
  ...nforcement.ts |   82.44 |       90 |     100 |   82.44 | 174-185,234-247   
  read-file.ts     |   95.07 |     88.6 |      90 |   95.07 | ...99,290-293,296 
  ripGrep.ts       |   94.59 |    85.71 |   93.33 |   94.59 | ...60,463,541-542 
  ...-transport.ts |    6.34 |      100 |       0 |    6.34 | 47-145            
  send-message.ts  |   89.32 |    91.66 |   83.33 |   89.32 | 44-45,68-76       
  shell.ts         |   72.19 |    80.09 |   90.47 |   72.19 | ...3790,3839-3845 
  skill-utils.ts   |     100 |      100 |     100 |     100 |                   
  skill.ts         |   88.11 |    91.17 |   84.61 |   88.11 | ...95,399,422-444 
  ...eticOutput.ts |   95.12 |      100 |      80 |   95.12 | 87-88             
  task-stop.ts     |   93.14 |    96.15 |   85.71 |   93.14 | 39-40,54-64       
  todoWrite.ts     |   89.17 |    82.05 |   92.85 |   89.17 | ...41-546,568-569 
  tool-error.ts    |     100 |      100 |     100 |     100 |                   
  tool-names.ts    |     100 |      100 |     100 |     100 |                   
  tool-registry.ts |   74.79 |       75 |   80.48 |   74.79 | ...92-793,801-802 
  tool-search.ts   |   95.19 |    86.48 |    92.3 |   95.19 | ...47-153,208-213 
  tools.ts         |   91.98 |    90.19 |   88.88 |   91.98 | ...50-451,467-473 
  web-fetch.ts     |   88.59 |    79.48 |    92.3 |   88.59 | ...12-313,315-316 
  write-file.ts    |   82.23 |    81.17 |   83.33 |   82.23 | ...65-668,680-715 
 src/tools/agent   |   75.01 |    82.55 |   74.62 |   75.01 |                   
  agent.ts         |   75.29 |    82.86 |    75.4 |   75.29 | ...2203,2265-2272 
  fork-subagent.ts |   69.62 |    71.42 |   66.66 |   69.62 | ...04-105,140-151 
 src/utils         |   88.98 |    87.52 |   93.68 |   88.98 |                   
  LruCache.ts      |       0 |        0 |       0 |       0 | 1-41              
  ...ssageQueue.ts |     100 |      100 |     100 |     100 |                   
  ...cFileWrite.ts |   77.96 |    80.48 |     100 |   77.96 | ...35,156,173-176 
  bareMode.ts      |   27.27 |      100 |       0 |   27.27 | 9-15,18-19        
  browser.ts       |    7.69 |      100 |       0 |    7.69 | 17-56             
  bundlePaths.ts   |     100 |      100 |     100 |     100 |                   
  ...igResolver.ts |     100 |      100 |     100 |     100 |                   
  ...engthError.ts |   89.11 |    86.66 |     100 |   89.11 | ...28-129,132-133 
  cronDisplay.ts   |   42.85 |    23.07 |     100 |   42.85 | 26-31,33-45,47-54 
  cronParser.ts    |   89.74 |    85.71 |     100 |   89.74 | ...,63-64,183-186 
  debugLogger.ts   |    95.9 |    93.84 |   94.73 |    95.9 | 106-107,214-218   
  editHelper.ts    |   93.63 |    83.52 |     100 |   93.63 | ...28-429,463-464 
  editor.ts        |   97.61 |    95.71 |     100 |   97.61 | ...70-271,273-274 
  ...arResolver.ts |   94.28 |    88.88 |     100 |   94.28 | 28-29,125-126     
  ...entContext.ts |     100 |    95.45 |     100 |     100 | 83                
  errorParsing.ts  |    97.7 |    97.05 |     100 |    97.7 | 72-73             
  ...rReporting.ts |   88.46 |       90 |     100 |   88.46 | 69-74             
  errors.ts        |   70.92 |    79.59 |   53.33 |   70.92 | ...03-219,223-229 
  fetch.ts         |   70.18 |    71.42 |   71.42 |   70.18 | ...42,148,161,186 
  fileUtils.ts     |   91.41 |    86.13 |      95 |   91.41 | ...1182,1186-1192 
  forkedAgent.ts   |    78.5 |    70.73 |   85.71 |    78.5 | ...30-436,441-447 
  formatters.ts    |   81.81 |       75 |     100 |   81.81 | 15-16             
  ...eUtilities.ts |   89.21 |    86.66 |     100 |   89.21 | 16-17,49-55,65-66 
  ...rStructure.ts |   94.36 |    94.28 |     100 |   94.36 | ...17-120,330-335 
  getPty.ts        |    12.5 |      100 |       0 |    12.5 | 21-34             
  gitDiff.ts       |   92.36 |    79.53 |     100 |   92.36 | ...55-856,928-929 
  ...noreParser.ts |    92.3 |    89.36 |     100 |    92.3 | ...15-116,186-187 
  gitUtils.ts      |   56.66 |    85.71 |      75 |   56.66 | ...2,72-73,97-148 
  iconvHelper.ts   |     100 |      100 |     100 |     100 |                   
  ...rePatterns.ts |     100 |      100 |     100 |     100 |                   
  ...ionManager.ts |     100 |     90.9 |     100 |     100 | 26                
  ...lPromptIds.ts |     100 |      100 |     100 |     100 |                   
  jsonl-utils.ts   |    74.1 |    90.76 |   58.33 |    74.1 | ...23-326,336-342 
  ...-detection.ts |     100 |      100 |     100 |     100 |                   
  ...iagnostics.ts |   96.87 |    91.83 |     100 |   96.87 | 214-219,272       
  ...yDiscovery.ts |    83.9 |    79.36 |     100 |    83.9 | ...16,319,411-414 
  ...tProcessor.ts |   93.63 |       90 |     100 |   93.63 | ...96-302,384-385 
  ...Inspectors.ts |   61.53 |      100 |      50 |   61.53 | 18-23             
  modelId.ts       |   98.55 |    96.87 |     100 |   98.55 | 103               
  ...kerChecker.ts |   88.75 |    85.71 |     100 |   88.75 | 69-70,87-93       
  notebook.ts      |   94.35 |    84.78 |     100 |   94.35 | ...10,122,174-176 
  openaiLogger.ts  |   88.05 |    84.09 |     100 |   88.05 | ...44-146,169-174 
  partUtils.ts     |     100 |    98.61 |     100 |     100 | 206               
  pathReader.ts    |     100 |      100 |     100 |     100 |                   
  paths.ts         |   93.21 |    91.86 |     100 |   93.21 | ...89-390,392-394 
  pdf.ts           |   93.68 |    87.05 |     100 |   93.68 | ...96-297,321-325 
  projectPath.ts   |     100 |      100 |     100 |     100 |                   
  ...ectSummary.ts |   89.39 |    72.41 |     100 |   89.39 | ...37-142,193-196 
  ...tIdContext.ts |     100 |      100 |     100 |     100 |                   
  proxyUtils.ts    |     100 |      100 |     100 |     100 |                   
  ...rDetection.ts |   58.57 |       76 |     100 |   58.57 | ...4,88-89,95-100 
  ...noreParser.ts |   85.45 |    85.18 |     100 |   85.45 | ...59,65-66,72-73 
  rateLimit.ts     |   92.55 |    85.92 |     100 |   92.55 | ...70-272,309-310 
  readManyFiles.ts |   87.96 |    86.95 |     100 |   87.96 | ...05-207,223-234 
  retry.ts         |   89.81 |    88.05 |     100 |   89.81 | ...29,350,357-358 
  ripgrepUtils.ts  |   46.79 |    84.37 |   66.66 |   46.79 | ...45-246,258-335 
  ...sDiscovery.ts |   97.42 |    92.85 |     100 |   97.42 | ...04,182-183,202 
  ...tchOptions.ts |   81.72 |    85.04 |   95.23 |   81.72 | ...11,536,565-574 
  runtimeStatus.ts |    97.5 |    88.57 |     100 |    97.5 | 167-168           
  safeJsonParse.ts |   74.07 |    83.33 |     100 |   74.07 | 40-46             
  ...nStringify.ts |     100 |      100 |     100 |     100 |                   
  ...aConverter.ts |   90.78 |    88.23 |     100 |   90.78 | ...41-42,93,95-96 
  ...aValidator.ts |   94.57 |    80.26 |     100 |   94.57 | ...04,213-216,270 
  ...r-launcher.ts |   76.92 |     91.3 |   66.66 |   76.92 | ...34,136,157-195 
  ...orageUtils.ts |   96.89 |    85.84 |     100 |   96.89 | ...51,367,447,466 
  shell-utils.ts   |   82.93 |    89.66 |     100 |   82.93 | ...1522,1529-1533 
  ...lAstParser.ts |   95.58 |    85.79 |     100 |   95.58 | ...1059-1061,1071 
  ...nlyChecker.ts |   95.75 |    92.39 |     100 |   95.75 | ...00-301,313-314 
  sideQuery.ts     |   98.73 |    94.59 |     100 |   98.73 | 111               
  ...pEventSink.ts |     100 |       80 |     100 |     100 | 61                
  ...tGenerator.ts |     100 |      100 |     100 |     100 |                   
  ...ameContext.ts |     100 |      100 |     100 |     100 |                   
  symlink.ts       |   77.77 |       50 |     100 |   77.77 | 44,54-59          
  ...emEncoding.ts |   96.36 |    91.17 |     100 |   96.36 | 59-60,124-125     
  terminalSafe.ts  |     100 |      100 |     100 |     100 |                   
  ...Serializer.ts |   98.72 |       90 |     100 |   98.72 | 42-43,134,201-203 
  testUtils.ts     |   53.33 |      100 |   33.33 |   53.33 | ...53,59-64,70-72 
  textUtils.ts     |      60 |      100 |   66.66 |      60 | 36-55             
  thoughtUtils.ts  |     100 |    92.85 |     100 |     100 | 71                
  ...-converter.ts |   94.59 |    85.71 |     100 |   94.59 | 35-36             
  tool-utils.ts    |    93.6 |     91.3 |     100 |    93.6 | ...58-159,162-163 
  truncation.ts    |     100 |       92 |     100 |     100 | 52,71             
  windowsPath.ts   |   89.47 |    79.31 |     100 |   89.47 | ...57-58,62,90-91 
  ...aceContext.ts |   93.71 |    89.28 |   93.33 |   93.71 | ...24-225,249-251 
  xml.ts           |     100 |      100 |     100 |     100 |                   
  yaml-parser.ts   |      92 |    84.61 |     100 |      92 | 49-53,65-69       
 ...ils/filesearch |   85.77 |    81.06 |   96.42 |   85.77 |                   
  crawlCache.ts    |     100 |      100 |     100 |     100 |                   
  crawler.ts       |   82.84 |    77.49 |   94.82 |   82.84 | ...1451,1485-1486 
  fileSearch.ts    |   93.58 |    87.32 |     100 |   93.58 | ...46-247,249-250 
  ignore.ts        |     100 |      100 |     100 |     100 |                   
  result-cache.ts  |     100 |     92.3 |     100 |     100 | 46                
 ...uest-tokenizer |   56.63 |    74.52 |   74.19 |   56.63 |                   
  ...eTokenizer.ts |   41.86 |    76.47 |   69.23 |   41.86 | ...70-443,453-507 
  index.ts         |     100 |      100 |     100 |     100 |                   
  ...tTokenizer.ts |   68.39 |    69.49 |    90.9 |   68.39 | ...24-325,327-328 
  ...ageFormats.ts |      76 |      100 |   33.33 |      76 | 45-48,55-56       
  textTokenizer.ts |     100 |      100 |     100 |     100 |                   
  types.ts         |       0 |        0 |       0 |       0 | 1                 
-------------------|---------|----------|---------|---------|-------------------

For detailed HTML reports, please see the 'coverage-reports-22.x-ubuntu-latest' artifact from the main CI run.

Copilot AI commented May 2, 2026

Copy link
Copy Markdown
Contributor

@copilot pls review for me

Here's the review summary:

Overall: The PR is well-structured with good test coverage, clean interfaces, and safe degradation when Linux-only probes are unavailable.

Two bugs were fixed (commit 8e886c3):

  1. maxRSS platform conversion (memoryDiagnostics.ts): process.resourceUsage().maxRSS is in kilobytes on Linux but already in bytes on macOS/Windows. The original code always multiplied by 1024, producing 1024× incorrect values on macOS. Fixed by extracting the platform detection before building the diagnostics object and only converting on Linux.

  2. native-memory-pressure risk message (memoryDiagnostics.ts): The message said "larger than heapUsed" but the actual threshold is > 2× heapUsed. Fixed to say "more than 2× heap used (N bytes)" so the hint accurately reflects the threshold.

A new test was added to assert that maxRSS is not multiplied on a darwin platform.

Minor observations (not changed — low-impact in a diagnostics context):

  • memoryDoctorAction ignores _context so sessionId/qwenVersion are never surfaced in diagnostics; these could be passed through in a follow-up.
  • Risk thresholds (handles > 100, FDs > 500) are heuristic — the PR description already acknowledges this.
  • The reference design doc lives in docs/plans/ (consistent with the existing file there) rather than .qwen/design/.

All 3 core tests and 9 CLI tests pass; typecheck is clean.

Comment thread packages/core/src/utils/memoryDiagnostics.ts Outdated
Comment thread packages/core/src/utils/memoryDiagnostics.ts Outdated
Comment thread packages/cli/src/ui/commands/doctorCommand.ts
Comment thread packages/cli/src/ui/commands/doctorCommand.ts Outdated

@wenshao wenshao left a comment

Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

[Suggestion] formatMemoryUsage is duplicated identically in packages/core/src/utils/formatters.ts and packages/cli/src/ui/utils/formatters.ts. If someone fixes a bug or adds a tier (e.g., TB) in one copy, the other silently diverges. Consider having the CLI's formatters.ts re-export from @qwen-code/qwen-code-core instead of re-declaring an identical function.

— pai/glm-5.1 via Qwen Code /review

Comment thread packages/cli/src/acp-integration/session/Session.ts Outdated
Comment thread packages/core/src/utils/memoryDiagnostics.ts
Comment thread packages/core/src/utils/memoryDiagnostics.ts
Comment thread packages/cli/src/ui/commands/doctorCommand.ts
Comment thread packages/cli/src/acp-integration/session/Session.test.ts
Comment thread packages/cli/src/ui/commands/doctorCommand.test.ts
Comment thread packages/core/src/utils/memoryDiagnostics.ts Outdated
Comment thread packages/core/src/utils/memoryDiagnostics.ts Outdated
Comment thread packages/core/src/utils/memoryDiagnostics.ts Outdated
Comment thread packages/cli/src/ui/commands/doctorCommand.ts Outdated
Comment thread packages/cli/src/ui/commands/doctorCommand.ts
Comment thread packages/cli/src/ui/commands/doctorCommand.ts
Comment thread packages/core/src/utils/memoryDiagnostics.test.ts
@wenshao

wenshao commented May 6, 2026

Copy link
Copy Markdown
Collaborator

Code Review

Overview

Adds /doctor memory and /doctor memory --json subcommands that produce a point-in-time memory snapshot (process / V8 / resource usage / handles / FDs / Linux smaps_rollup), plus a new collectMemoryDiagnostics() core utility, heuristic risk analysis, and a small ACP serialization tweak so the parent /doctor keeps auto-submitting even though it now has subcommands.

What's good

  • Clean dependency-injection seams. MemoryDiagnosticsOptions lets every probe be swapped in tests; this is why the test file can cover Linux-only paths from any platform.
  • Graceful degradation. Optional probes (heapSpaceStatistics, openFileDescriptors, smapsRollup, _getActiveHandles/Requests) are individually try/caught with debug-only logging, so /proc-less platforms don't error out — and there are explicit tests for that.
  • Sensible heuristic refinements. The NATIVE_MEMORY_PRESSURE_MIN_BYTES = 64 MB floor and RSS_HEAP_GAP_MIN_BYTES = 256 MB floor explicitly avoid spurious risk hints during cold start; both are covered by negative tests.
  • Cross-platform maxRSS normalization. Linux returns kB, macOS/Windows bytes — converted once and tested.
  • Independent probe parallelism via Promise.all, with a test verifying that slow probes don't block faster ones.

Issues / suggestions

1. ACP UX: parent /doctor with acceptsInput: false may hide the memory subcommand from auto-submitting clients.
doctorCommand.ts sets argumentHint: '[memory]' together with acceptsInput: false. In ACP, this means selecting /doctor from a command picker sends input: null and auto-submits the parent — users can never reach /doctor memory via the picker, only by typing it manually. The comment explicitly acknowledges this trade-off, but it's worth confirming with whoever owns the ACP UX. An alternative is to keep input enabled for parents-with-subcommands so the picker reveals the subcommand path.

2. Vacuous assertion in doctorCommand.test.ts.

expect(result?.type === 'message' ? result.content : '').not.toContain('0.00 MiB');

formatMemoryUsage only ever emits KB / MB / GB — never MiB. The assertion always passes regardless of the rendering. Either change it to '0.00 MB' (matching what would actually regress) or drop it; the preceding toContain('heapUsed: 1.0 KB') already covers the small-value path.

3. Duplicate formatMemoryUsage.
Pre-existing tech debt, but this PR doubles down on it: doctorCommand.ts imports the cli copy, memoryDiagnostics.ts imports the core copy. Both files are byte-identical. Worth a follow-up to delete the cli one and re-export from core (packages/core/src/index.ts already exports formatters).

4. Heap-pressure threshold may be noisy for legitimately heavy workloads.
The analyzer flags heapUsed / heapSizeLimit ≥ 0.75. Long-running CLI sessions can sit near this without leaking. Not a blocker, but consider whether to require both a high ratio and a near-the-limit absolute value (parallel to the floors used for native pressure / RSS gap).

5. getActiveHandlesCount / getActiveRequestsCount use private Node internals.
The inline comment flags this honestly. Worth noting that Node 20+ has process.getActiveResourcesInfo() as a public API for active resource categorization — not a strict superset, but lower-risk for the count case if you ever want to drop the _getActive* dependency.

6. Minor: analysis field is double-assigned.
collectMemoryDiagnostics initializes analysis: { risks: [], recommendation: '' } and immediately overwrites it on the next statement. Safer/cleaner to compute analysis first, then build the object once.

7. Coverage gap: heap-spaces rendering.
formatHeapSpaces (top-4 + "… N more") is unit-tested only via JSON shape, not the readable summary. A small assertion that the readable output contains, e.g., old_space: would catch regressions in the formatter helper.

Risk assessment

  • Correctness: Solid. Probes are bounded, errors swallowed locally, units normalized, no observable change to existing /doctor behavior.
  • Performance: Snapshot-only; no continuous polling. Promise.all over the optional probes; on Linux the worst case is two small /proc/self/* reads. Safe to call ad hoc.
  • Security: No new network/IPC surface; only reads /proc/self/{fd,smaps_rollup} (own-process). The --json output may include smapsRollup text, which can reveal mapped file paths if pasted into a public issue — worth a one-line warning in the readable summary or in docs, but not a blocker.
  • Backward compat: None broken. New acceptsInput field on SlashCommand is optional and falls back to existing inference.

Suggested actions before merge

  • Decide on the ACP picker UX for /doctor memory (item 1).
  • Fix the '0.00 MiB' assertion (item 2).
  • Optional: file a follow-up to dedupe formatMemoryUsage (item 3).

Overall: ship-worthy as a first diagnostics layer for #3000. The test discipline is unusually good for a feature this size.

Comment thread packages/cli/src/ui/commands/doctorCommand.ts Outdated
Comment thread packages/cli/src/ui/commands/doctorCommand.ts
@yiliang114

Copy link
Copy Markdown
Collaborator Author

Updated this PR after re-reviewing the memory diagnostics feedback.

Changes since the previous draft:

  • Pass the current session id and CLI version into collectMemoryDiagnostics() so --json can be tied back to a run.
  • Render the Rss: line from smaps_rollup in readable /doctor memory output while keeping the full raw content in JSON.
  • Emit memory risk results as warning messages and wire warning handling through the slash-command message pipeline.
  • Remove the temporary analysis placeholder before running memory risk analysis and keep tests covering heap spaces, smaps, session metadata, risk rendering, and warning rendering.

Local validation:

  • cd packages/core && npx vitest run src/utils/memoryDiagnostics.test.ts (10 passed)
  • cd packages/cli && npx vitest run src/ui/commands/doctorCommand.test.ts src/acp-integration/session/Session.test.ts src/ui/hooks/slashCommandProcessor.test.ts src/ui/commands/insightCommand.test.ts (94 passed)
  • npm run lint
  • npm run build
  • npm run typecheck
  • npm run bundle
  • git diff --check

I kept the parent /doctor command with acceptsInput: false intentionally so bare /doctor still auto-submits; the memory subcommand keeps [--json] as its own argument hint.

@yiliang114 yiliang114 marked this pull request as ready for review May 16, 2026 11:28
@yiliang114 yiliang114 requested a review from wenshao May 16, 2026 11:28
@yiliang114 yiliang114 changed the title feat(cli): add memory diagnostics doctor command feat(cli): add structured memory diagnostics JSON May 16, 2026
@yiliang114

Copy link
Copy Markdown
Collaborator Author

Updated again after syncing with the latest main.

What changed in this round:

  • Resolved the merge conflict with the newer /doctor memory implementation that already exists on main.
  • Kept main behavior for readable /doctor memory, --sample, and --snapshot.
  • Scoped this PR to structured /doctor memory --json, session/version metadata, core risk analysis, and warning message plumbing.
  • Removed the earlier acceptsInput override approach because it is no longer needed after the latest main changes.
  • Updated the PR title and body to match the current scope.

Local validation after the merge:

  • cd packages/core && npx vitest run src/utils/memoryDiagnostics.test.ts (10 passed)
  • cd packages/cli && npx vitest run src/ui/commands/doctorCommand.test.ts src/ui/hooks/slashCommandProcessor.test.ts src/ui/commands/insightCommand.test.ts src/acp-integration/session/Session.test.ts (146 passed)
  • npm run lint
  • npm run build
  • npm run typecheck
  • npm run bundle
  • git diff --cached --check

GitHub now reports the PR as mergeable; CI has started on the new head and I am not waiting on the long OS matrix here.

@wenshao wenshao left a comment

Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Cross-file concerns (files not in this diff):

⚠️ [Critical] /doctor memory --json warnings exit with code 0 in nonInteractiveCli.ts:321 — The PR widens messageType to include 'warning' in nonInteractiveCliCommands.ts, but nonInteractiveCli.ts:321 only treats messageType === 'error' as failure (isError: false, exit 0). CI pipelines running health checks will miss memory warnings. The handler at :321 and :326 should also treat 'warning' as a non-zero exit.

⚠️ [Suggestion] Session.ts:2249 does not distinguish warning from info — ACP #processSlashCommandResult renders both warning and info identically through emitAgentMessage. ACP clients (Zed) cannot visually differentiate memory risk warnings.

Comment thread packages/core/src/utils/memoryDiagnostics.ts Outdated
Comment thread packages/core/src/utils/memoryDiagnostics.ts

@wenshao wenshao left a comment

Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The warning messageType was added to NonInteractiveSlashCommandResult, but the exit-code logic in nonInteractiveCli.ts still only checks messageType === 'error'. /doctor memory --json returns messageType: 'warning' when risks are detected, yet the non-interactive CLI exits with code 0. CI scripts that rely solely on exit codes will miss memory warnings — they must parse the JSON output to discover risks.

Test coverage gaps: heapSizeLimit === 0 guard in analyzeMemoryDiagnostics is not tested. Interactive-mode /doctor memory --json path is not tested.

— gpt-5.5 via Qwen Code /review

Comment thread packages/core/src/utils/memoryDiagnostics.ts
Comment thread packages/core/src/utils/memoryDiagnostics.ts

@yiliang114 yiliang114 left a comment

Copy link
Copy Markdown
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Addressed the remaining review feedback:

  • Removed the incorrect maxRSS * 1024 multiplier on Linux (Node.js >=14.10 returns bytes on all platforms)
  • Switched heapRatio to use v8HeapStats.usedHeapSize instead of memoryUsage.heapUsed for internal consistency
  • Updated test expectations accordingly

@yiliang114 yiliang114 requested a review from wenshao May 16, 2026 15:27
@yiliang114 yiliang114 force-pushed the codex/memory-diagnostics branch from a50c9c2 to 146d165 Compare May 16, 2026 16:36
@yiliang114

Copy link
Copy Markdown
Collaborator Author

Verification Report — PR #3785

/doctor memory structured diagnostics — E2E + unit test results

Environment

Date 2026-05-16 17:11:58 UTC
Branch codex/memory-diagnostics
Node v22.22.0
Platform Darwin arm64

Test Matrix

Category Pass Fail Total
Non-interactive E2E 14 0 14
Interactive (tmux) 6 0 6
Unit tests 161 0 161
Total 181 0 181

Build Pipeline

Step Result
npm run build
npm run typecheck
npm run lint ✅ (17 pre-existing in vscode-ide-companion)
npm run bundle

E2E Scenarios

Test 1: /doctor memory --json

Parsed output
heapUsed:       104.2 MB
heapTotal:      130.0 MB
rss:            246.5 MB
v8HeapLimit:    4.05 GB
activeHandles:  207
activeRequests: 3
risks:          0
recommendation: No obvious leak indicators. Check heap snapshot for retained objects.
sessionId:      5d917bbb-4280-4219-b679-88b7fbbdc65a
qwenVersion:    0.15.11
### Test 2: `/doctor memory` (readable)
Full output
Memory Diagnostics
timestamp: 2026-05-16T17:11:32.243Z
uptimeSeconds: 2.8
heapUsed: 104.3 MB
heapTotal: 130.0 MB
rss: 244.9 MB
external: 21.0 MB
arrayBuffers: 1.1 MB
v8HeapLimit: 4.05 GB
v8MallocedMemory: 1.0 MB
v8PeakMallocedMemory: 34.3 MB
detachedContexts: 0
nativeContexts: 2
maxRSS: 244.9 KB
userCPUTime: 0.58s
systemCPUTime: 0.15s
activeHandles: 207
activeRequests: 3
openFileDescriptors: unavailable
smapsRollup: unavailable
v8HeapSpaces:
  - old_space: used 46.9 MB / size 52.0 MB
  - large_object_space: used 36.8 MB / size 37.0 MB
  - new_space: used 12.6 MB / size 32.0 MB
  - trusted_space: used 4.6 MB / size 4.9 MB
  - … 7 more
risks:
  无
recommendation: No obvious leak indicators. Check heap snapshot for retained objects.
### Test 3: `/doctor memory --unknown`
Unknown argument(s): --unknown. Usage: /doctor memory [--json]

Test 4: /doctor (general checks)

Summary
pass: 14, warn: 0, fail: 0
check count: 14
  [pass] 系统/Node.js version: v22.22.0
  [pass] 系统/npm version: 10.9.4
  [pass] 系统/Platform: darwin/arm64 (24.1.0)
  [pass] Authentication/API key: configured (openai)
  [pass] Authentication/API client: client initialized
  ... and 9 more
### Test 5: interactive `/doctor memory --json`
Captured pane (excerpt)
      "memoryUsage": {
      "v8HeapStats": {
      "activeHandles": 168,
      "analysis": {
        "risks": [],
        "recommendation": "No obvious leak indicators. Check heap snapshot for retained objects."
### Test 6: interactive `/doctor memory`
Captured pane (excerpt)
        "heapUsed": 110348288,
      "activeHandles": 168,
  ● Memory Diagnostics
    heapUsed: 109.0 MB
    rss: 275.1 MB
    v8HeapLimit: 4.05 GB
    activeHandles: 168
    risks:
    recommendation: No obvious leak indicators. Check heap snapshot for retained objects.

Unit Tests

File                                          Tests  Status
packages/core/src/utils/memoryDiagnostics       10   ✅
packages/cli/src/ui/commands/doctorCommand       33   ✅
packages/cli/src/ui/hooks/slashCommandProcessor  46   ✅
packages/cli/src/acp-integration/session/Session 68   ✅
packages/cli/src/ui/commands/insightCommand       4   ✅
────────────────────────────────────────────────────────
Total                                           161   ✅

Known Issue

After rebasing onto main, the parent doctorCommand.action still contains legacy
/doctor memory --sample and --snapshot handling. This code is now unreachable
because parseSlashCommand routes /doctor memory to the subCommands entry
first. The existing --sample/--snapshot features are shadowed (not deleted).
Follow-up: either migrate these flags into the subCommand or integrate --json
into the parent action.

@wenshao

wenshao commented May 16, 2026

Copy link
Copy Markdown
Collaborator

@yiliang114 这个 PR 3 个 Test 平台全失败了(run 25968245065)。失败不是 tsc 类型错,是 i18n 严格 parity 检查没过:

 FAIL  src/i18n/mustTranslateKeys.test.ts > must-translate locale coverage
   > does not fall back to English for any built-in command description in strict-parity locale
     { code: 'zh-TW', ... }
   > does not fall back to English for any built-in command description in strict-parity locale
     { code: 'zh', id: 'zh-CN', ... }

AssertionError: expected [ 'doctor memory' ] to deeply equal []

zh / zh-TW 是 strict-parity locale,新增的 /doctor memory 子命令没补对应的描述翻译,运行时 fallback 到英文 → 测试拒绝。

修复

新增 slash 命令时同步加翻译到这两个 locale 的命令描述。看一下别的 /doctor * 子命令是怎么写的(例如 doctor 顶层命令的 zh / zh-TW 翻译),把 doctor memory 那条 description key 加上对应中文 / 繁体中文文案。

packages/cli/src/i18n/locales/zh.js + packages/cli/src/i18n/locales/zh-TW.js(这两个文件如果不存在,路径以 src/i18n/locales/ 下其他 locale 文件为准)。

本地复现命令:

cd packages/cli && npx vitest run src/i18n/mustTranslateKeys.test.ts

补完翻译后这条会从 2 failed0 failed,三平台 Test 应该都能转绿。

@wenshao wenshao left a comment

Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Code Review: /doctor memory --json in ACP tools

Summary

PR adds a dedicated /doctor memory --json path that exposes machine-readable
memory diagnostics through ACP tools. The implementation moves the core
MemoryDiagnostics type and collectMemoryDiagnostics() from CLI into packages/core,
adds /doctor memory --json as a direct handler in doctorCommand.ts (bypassing the
historical 2-message ACP stream), wires /doctor memory --json into
list_directory and read_file tool failures, and covers the change with
CLI + Core unit tests and two integration tests.

Positive findings:

  • Core module (packages/core/src/utils/memoryDiagnostics.test.ts) tests are
    comprehensive: positive path, error/missing data, and boundary conditions (539 lines)
  • Error handling with graceful degradation across all probes
  • Rate limiting and disk space checks for heap snapshots
  • Abort signal support throughout the async path

Findings

[non-blocking] 12 new t() calls reference 8 keys absent from ALL locale files

doctorCommand.ts introduces 8 new t() keys that are missing from every
locale file (en.js, zh.js, zh-TW.js, ca.js, de.js, fr.js, ja.js, pt.js, ru.js):

  • Show current process memory diagnostics
  • Unknown argument(s)
  • Usage
  • Failed to collect memory diagnostics
  • unavailable
  • available
  • none
  • Memory Diagnostics

The t() function falls back to the key itself, so English users see the raw
key text -- which happens to be English, masking the gap. But zh and zh-TW
users also see English text for these strings.

Additionally, doctorChecks.ts adds t('system') (line 170) which is also
absent from all locale files.

Recommendation: Add all 9 keys to en.js (as identity mappings) and to at
least zh.js/zh-TW.js with proper translations. The other locale files can
be left for native speakers to fill in.


[non-blocking] Dual diagnostic paths -- parent action vs subcommand action

doctorCommand.action (the parent) handles /doctor memory inline using the
CLI-local getMemoryDiagnostics() + formatMemoryDiagnostics() (structured
CLI output). The memory subcommand's own action (memoryDoctorAction) uses
the core collectMemoryDiagnostics() + formatCoreDiagnostics() (plain-text
key-value output).

Both paths are exercised by tests, but they produce different output formats
for the same logical operation:

Path Source Format
Parent action (doctorCommand.action) CLI-local getMemoryDiagnostics() Structured object with formatMemoryDiagnostics()
Subcommand action (memoryDoctorAction) Core collectMemoryDiagnostics() Plain-text key-value via formatCoreDiagnostics()

If the command dispatcher ever routes /doctor memory to the subcommand's
action directly (rather than through the parent), the output format and data
source would differ silently.

Recommendation: Consider making the parent action delegate to the
subcommand's action for the memory case, or document the routing contract
explicitly to prevent future drift.


[non-blocking] memoryDoctorAction does not handle --sample or --snapshot

The subcommand's action only recognizes --json:

const subSubCommand = args?.trim() ?? '';
const isJsonMode = subSubCommand === '--json';

If the command dispatcher routes /doctor memory --sample or
/doctor memory --snapshot to the subcommand's action, it returns an "Unknown
argument" error. The --sample and --snapshot paths are only handled by the
parent action.

Recommendation: Either add --sample/--snapshot handling to the
subcommand's action, or add an explicit guard comment explaining that these
flags are parent-action-only.


[non-blocking] acceptsInput inference bypasses subcommand hint

Session.ts's acceptsInput inference checks cmd.argumentHint and
cmd.subCommands, so doctorCommand (which has both) correctly gets
acceptsInput = true. However, the memory subcommand also has
argumentHint: '[--json]' -- the inference logic only runs on top-level
commands passed to buildAvailableCommands, so subcommand acceptsInput is
never evaluated. This is fine for the current ACP protocol but could cause
confusion if subcommands are ever exposed directly.

Recommendation: No action needed now; just noting the implicit contract.


[nit] formatCoreDiagnostics risks/recommendation indentation inconsistency

Risks are indented with - (2-space indent + dash), but the recommendation
line has no indentation prefix:

risks:
  - heap-pressure: Heap pressure detected.
recommendation: WARNING: 1 potential leak indicator(s) found.

The recommendation key appears at the same level as risks:, which is
correct for the key-value format, but the visual gap between the indented
risk items and the unindented recommendation may confuse parsers expecting
consistent indentation.


[nit] formatMemoryUsage uses mixed units

The function uses decimal MB for values >= 1 MiB:

return `${(bytes / (1024 * 1024)).toFixed(1)} MB`;

But uses binary KiB for values < 1 MiB:

return `${(bytes / 1024).toFixed(1)} KiB`;

This mixes decimal (MB) and binary (KiB) units in the same output. The CLI's
formatBytes function consistently uses binary MiB. Consider using a single
unit system (binary MiB/KiB or decimal MB/KB) throughout.

Suggested fix (in packages/core/src/utils/formatters.ts):

export const formatMemoryUsage = (bytes: number): string => {
  if (bytes < 1024) {
    return `${bytes} B`;
  }
  if (bytes < 1024 * 1024) {
    return `${(bytes / 1024).toFixed(1)} KiB`;
  }
  return `${(bytes / (1024 * 1024)).toFixed(1)} MiB`;
};

[nit] doctorChecks.ts -- normalizeConfigDir does not expand ~ on Linux

The function handles macOS's /Users/name prefix but not Linux's /home/name:

if (process.platform === 'darwin' && !configDir.startsWith('/Users/')) {
  configDir = path.join(os.homedir(), configDir);
}

On Linux, a relative configDir (e.g., '.config/qwen-code') would not be
expanded. The existing code has the same gap, but since this PR modifies the
file, it is worth noting.


[nit] checkGoogleApplicationCredentials receives raw (non-normalized) configDir

checkAuthentication normalizes configDir via normalizeConfigDir() but then
passes the original (non-normalized) configDir to checkGoogleApplicationCredentials:

const configDir = config.getConfigDir();
const normalizedDir = normalizeConfigDir(configDir);
// ...
const googleAppCredsResult = await checkGoogleApplicationCredentials(configDir);

This is fine because checkGoogleApplicationCredentials only uses it for the
config.json existence check, but it is a subtle inconsistency worth harmonizing.


[scope] Suggested improvement -- expose --sample and --snapshot through ACP

The PR description mentions exposing /doctor memory --json through ACP tools.
The --sample (memory pressure over time) and --snapshot (heap dump) features
are powerful diagnostics that would also benefit ACP tool consumers. Consider
following up with a PR that makes these available through the ACP protocol.


Test Coverage Notes

Well-covered areas:

  • Core collectMemoryDiagnostics: positive path, error/missing data, boundary
    conditions (539 lines)
  • CLI doctorCommand: interactive + non-interactive modes, abort signals,
    --json, --sample, --snapshot, error handling
  • Integration tests: headless --json, tmux readable output, stdout isolation

Gaps:

  • No standalone test for formatCoreDiagnostics (tested indirectly through
    doctorCommand.action)
  • No test for the acceptsInput inference in Session.ts with the new
    acceptsInput field
  • No test for normalizeConfigDir with Linux paths

@wenshao wenshao left a comment

Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Missing i18n keys: Several t() calls in the new memoryDoctorAction + formatters use keys not registered in locale files: 'Usage', 'Unknown argument(s)', 'Failed to collect memory diagnostics', 'Memory Diagnostics', 'unavailable', 'available'. While t() falls back to the key itself (works for English), non-English locales will see untranslated English text. Please add translations to en.js, zh.js, and zh-TW.js.

— DeepSeek/deepseek-v4-pro via Qwen Code /review

Comment thread packages/cli/src/ui/commands/doctorCommand.ts
Comment thread packages/cli/src/ui/commands/doctorCommand.ts Outdated
Comment thread packages/cli/src/ui/commands/doctorCommand.ts
Comment thread packages/core/src/utils/memoryDiagnostics.ts
Comment thread packages/core/src/utils/memoryDiagnostics.ts

@wenshao wenshao left a comment

Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Review Summary

Build & Test: All passing (tsc ✅, eslint ✅, 180/180 tests ✅, i18n parity ✅)

Verdict: Request changes — 2 Critical findings must be fixed before merge.

# Severity Issue Location
1 Critical maxRSS unit mismatch: code assumes bytes, Node returns KB (1024× under-report) memoryDiagnostics.ts:134
2 Critical warning messageType exits 0 in non-interactive CLI; CI cannot detect memory risks nonInteractiveCli.ts:326
3 Suggestion MemoryResourceUsage fields lack JSDoc unit docs (microseconds for CPU, KB for maxRSS) memoryDiagnostics.ts:56-63
4 Suggestion --json silently dropped when combined with --sample/--snapshot doctorCommand.ts:266
5 Suggestion No timeout on optionalProbe; NFS/FUSE /proc can cause indefinite hang memoryDiagnostics.ts:257
6 Suggestion Parent /doctor action: --snapshot/--sample flags parsed but unreachable outside memory subcommand doctorCommand.ts:89
7 Suggestion Arena dialog callbacks + ACP stream_messages lack warning branch; risk warnings silently swallowed ArenaStopDialog.tsx:43, Session.ts:2285
8 Suggestion countOpenFileDescriptors allocates full string array via readdir; opendir async iterator is more memory-efficient memoryDiagnostics.ts:226
9 Suggestion formatSmapsRollup can use regex instead of split+map+find doctorCommand.ts:339

Finding 1 — Critical: maxRSS unit mismatch

Empirically verified on this machine (Node v22.22.2, macOS ARM64):

process.resourceUsage().maxRSS = 40816  (kilobytes)
process.memoryUsage().rss      = 41844736  (bytes ≈ 39.9 MB)
maxRSS * 1024 / rss            = 0.998  (matches!)
maxRSS / rss                   = 0.00098  (off by 1024×)

The code comment at line 132 states "Node.js >=14.10.0 returns maxRSS in bytes on all platforms" — this contradicts the official Node.js documentation which states maxRSS returns kilobytes on Linux/macOS. The result: formatMemoryUsage(maxRSSBytes) displays 39.9 KB instead of the correct 39.9 MB.

Fix: const maxRSSBytes = resourceUsage.maxRSS * 1024; (and update the comment).

Note: Commit 146d165a00 ("correct maxRSS byte handling") removed a * 1024 multiplier, which appears to have introduced this regression.


Finding 2 — Critical: warning exits 0 in non-interactive CLI

nonInteractiveCli.ts:326:

return slashCommandResult.messageType === 'error' ? 1 : 0;

When /doctor memory detects risks, it returns messageType: 'warning'. In non-interactive mode, this maps to exit code 0 — indistinguishable from a clean run. CI pipelines that check exit codes will silently pass.

Fix: Add a warning branch:

if (slashCommandResult.messageType === 'error') return 1;
if (slashCommandResult.messageType === 'warning') return 2;
return 0;

Note: This finding was partially flagged in a prior review round as "warning exit code logic not updated." The current code still has the issue.


Finding 3 — Missing JSDoc units on MemoryResourceUsage

memoryDiagnostics.ts:56-63 — the interface fields have no unit documentation:

export interface MemoryResourceUsage {
  maxRSS: number;       // KB (per Node.js docs)
  userCPUTime: number;  // microseconds (per Node.js docs)
  systemCPUTime: number; // microseconds
}

Future consumers (tests, integrations, Bun compatibility) will silently get values wrong by orders of magnitude.


Finding 4 — --json silently dropped with --sample/--snapshot

doctorCommand.ts:266 — when shouldSampleMemory || shouldWriteHeapSnapshot, the code strips --json and delegates to the parent action, which returns plain text. No error or warning is emitted. Automated scripts expecting JSON will receive unparseable text.

Fix: Reject the combination explicitly:

if (tokens.includes('--json') && (shouldSampleMemory || shouldWriteHeapSnapshot)) {
  return { type: 'message', messageType: 'error', content: t('--json is not supported with --sample or --snapshot') };
}

Finding 5 — No timeout on optionalProbe

memoryDiagnostics.ts:257await probe() has no timeout. On NFS-mounted /proc or hung FUSE filesystems, readdir('/proc/self/fd') or readFile('/proc/self/smaps_rollup') can hang indefinitely.

Fix: Add a Promise.race with a 5-second timeout inside optionalProbe.


Finding 6 — Parent /doctor dead code

doctorCommand.ts:89-103shouldSampleMemory and shouldWriteHeapSnapshot are parsed at the parent level, but the snapshot/sample execution block is inside the if (subCommand === MEMORY_SUBCOMMAND) branch. When a user types /doctor --snapshot (without memory), subCommand is '--snapshot', which doesn't match, so the flags are silently ignored.


Finding 7 — Warning handling missing in Arena + ACP paths

  • ArenaStopDialog.tsx:43 and ArenaSelectDialog.tsx:42 type callbacks as 'info' | 'error'warning maps to MessageType.ERROR.
  • Session.ts:2285 (ACP stream_messages): only error triggers throw; warning is emitted as a plain agentMessage with no severity signal.

Finding 8 — countOpenFileDescriptors allocates full string array

memoryDiagnostics.ts:226readdir('/proc/self/fd') returns Promise<string[]>, allocating a string per fd just to read .length. On a process with fd leaks (the exact scenario this diagnostic targets), this could be thousands of strings. Using fs.opendir with an async iterator avoids the allocation.


Finding 9 — formatSmapsRollup optimization

doctorCommand.ts:339 — the function splits all lines, maps each to a trimmed string, then finds the Rss: line. A regex smapsRollup.match(/^Rss:\s*.+$/m) would be simpler and avoid intermediate allocations.


Previously flagged items now resolved: i18n missing keys ✅, formatError duplication ✅, probe failure logging ✅, getActiveHandlesCount/getActiveRequestsCount deduplication ✅.

Open items from prior review still not addressed: 7 inline comments from @wenshao remain open (see individual comment threads).

Comment thread packages/core/src/utils/memoryDiagnostics.ts
Comment thread packages/core/src/utils/memoryDiagnostics.ts
Comment thread packages/core/src/utils/memoryDiagnostics.ts
Comment thread packages/core/src/utils/memoryDiagnostics.ts
Comment thread packages/cli/src/ui/commands/doctorCommand.ts
Comment thread packages/cli/src/ui/commands/doctorCommand.ts

@wenshao wenshao left a comment

Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Review Summary

Build & Test: All passing (tsc ✅, eslint ✅, 180/180 tests ✅, i18n parity ✅)

Verdict: Request changes — 2 Critical findings must be fixed before merge.

# Severity Issue Location
1 Critical maxRSS unit mismatch: code assumes bytes, Node returns KB (1024× under-report) memoryDiagnostics.ts:134
2 Critical warning messageType exits 0 in non-interactive CLI; CI cannot detect memory risks nonInteractiveCli.ts:326
3 Suggestion MemoryResourceUsage fields lack JSDoc unit docs (microseconds for CPU, KB for maxRSS) memoryDiagnostics.ts:56-63
4 Suggestion --json silently dropped when combined with --sample/--snapshot doctorCommand.ts:266
5 Suggestion No timeout on optionalProbe; NFS/FUSE /proc can cause indefinite hang memoryDiagnostics.ts:257
6 Suggestion Parent /doctor action: --snapshot/--sample flags parsed but unreachable outside memory subcommand doctorCommand.ts:89
7 Suggestion Arena dialog callbacks + ACP stream_messages lack warning branch; risk warnings silently swallowed ArenaStopDialog.tsx:43, Session.ts:2285
8 Suggestion countOpenFileDescriptors allocates full string array via readdir; opendir async iterator is more memory-efficient memoryDiagnostics.ts:226
9 Suggestion formatSmapsRollup can use regex instead of split+map+find doctorCommand.ts:339

Finding 1 — Critical: maxRSS unit mismatch

Empirically verified on this machine (Node v22.22.2, macOS ARM64):

process.resourceUsage().maxRSS = 40816  (kilobytes)
process.memoryUsage().rss      = 41844736  (bytes ≈ 39.9 MB)
maxRSS * 1024 / rss            = 0.998  (matches!)
maxRSS / rss                   = 0.00098  (off by 1024×)

The code comment at line 132 states "Node.js >=14.10.0 returns maxRSS in bytes on all platforms" — this contradicts the official Node.js documentation which states maxRSS returns kilobytes on Linux/macOS. The result: formatMemoryUsage(maxRSSBytes) displays 39.9 KB instead of the correct 39.9 MB.

Fix: const maxRSSBytes = resourceUsage.maxRSS * 1024; (and update the comment).

Note: Commit 146d165a00 ("correct maxRSS byte handling") removed a * 1024 multiplier, which appears to have introduced this regression.


Finding 2 — Critical: warning exits 0 in non-interactive CLI

nonInteractiveCli.ts:326:

return slashCommandResult.messageType === 'error' ? 1 : 0;

When /doctor memory detects risks, it returns messageType: 'warning'. In non-interactive mode, this maps to exit code 0 — indistinguishable from a clean run. CI pipelines that check exit codes will silently pass.

Fix: Add a warning branch:

if (slashCommandResult.messageType === 'error') return 1;
if (slashCommandResult.messageType === 'warning') return 2;
return 0;

Note: This finding was partially flagged in a prior review round as "warning exit code logic not updated." The current code still has the issue.


Finding 3 — Missing JSDoc units on MemoryResourceUsage

memoryDiagnostics.ts:56-63 — the interface fields have no unit documentation:

export interface MemoryResourceUsage {
  maxRSS: number;       // KB (per Node.js docs)
  userCPUTime: number;  // microseconds (per Node.js docs)
  systemCPUTime: number; // microseconds
}

Future consumers (tests, integrations, Bun compatibility) will silently get values wrong by orders of magnitude.


Finding 4 — --json silently dropped with --sample/--snapshot

doctorCommand.ts:266 — when shouldSampleMemory || shouldWriteHeapSnapshot, the code strips --json and delegates to the parent action, which returns plain text. No error or warning is emitted. Automated scripts expecting JSON will receive unparseable text.

Fix: Reject the combination explicitly:

if (tokens.includes('--json') && (shouldSampleMemory || shouldWriteHeapSnapshot)) {
  return { type: 'message', messageType: 'error', content: t('--json is not supported with --sample or --snapshot') };
}

Finding 5 — No timeout on optionalProbe

memoryDiagnostics.ts:257await probe() has no timeout. On NFS-mounted /proc or hung FUSE filesystems, readdir('/proc/self/fd') or readFile('/proc/self/smaps_rollup') can hang indefinitely.

Fix: Add a Promise.race with a 5-second timeout inside optionalProbe.


Finding 6 — Parent /doctor dead code

doctorCommand.ts:89-103shouldSampleMemory and shouldWriteHeapSnapshot are parsed at the parent level, but the snapshot/sample execution block is inside the if (subCommand === MEMORY_SUBCOMMAND) branch. When a user types /doctor --snapshot (without memory), subCommand is '--snapshot', which doesn't match, so the flags are silently ignored.


Finding 7 — Warning handling missing in Arena + ACP paths

  • ArenaStopDialog.tsx:43 and ArenaSelectDialog.tsx:42 type callbacks as 'info' | 'error'warning maps to MessageType.ERROR.
  • Session.ts:2285 (ACP stream_messages): only error triggers throw; warning is emitted as a plain agentMessage with no severity signal.

Finding 8 — countOpenFileDescriptors allocates full string array

memoryDiagnostics.ts:226readdir('/proc/self/fd') returns Promise<string[]>, allocating a string per fd just to read .length. On a process with fd leaks (the exact scenario this diagnostic targets), this could be thousands of strings. Using fs.opendir with an async iterator avoids the allocation.


Finding 9 — formatSmapsRollup optimization

doctorCommand.ts:339 — the function splits all lines, maps each to a trimmed string, then finds the Rss: line. A regex smapsRollup.match(/^Rss:\s*.+$/m) would be simpler and avoid intermediate allocations.


Previously flagged items now resolved: i18n missing keys ✅, formatError duplication ✅, probe failure logging ✅, getActiveHandlesCount/getActiveRequestsCount deduplication ✅.

Open items from prior review still not addressed: 7 inline comments from @wenshao remain open (see individual comment threads).

Comment thread packages/core/src/utils/memoryDiagnostics.ts
Comment thread packages/core/src/utils/memoryDiagnostics.ts
Comment thread packages/core/src/utils/memoryDiagnostics.ts
Comment thread packages/core/src/utils/memoryDiagnostics.ts
Comment thread packages/cli/src/ui/commands/doctorCommand.ts
Comment thread packages/cli/src/ui/commands/doctorCommand.ts

@wenshao wenshao left a comment

Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Review Summary

Build & Test: All passing (tsc ✅, eslint ✅, 180/180 tests ✅, i18n parity ✅)

Verdict: Request changes — 2 Critical findings must be fixed before merge.

# Severity Issue Location
1 Critical maxRSS unit mismatch: code assumes bytes, Node returns KB (1024× under-report) memoryDiagnostics.ts:134
2 Critical warning messageType exits 0 in non-interactive CLI; CI cannot detect memory risks nonInteractiveCli.ts:326
3 Suggestion MemoryResourceUsage fields lack JSDoc unit docs (microseconds for CPU, KB for maxRSS) memoryDiagnostics.ts:56-63
4 Suggestion --json silently dropped when combined with --sample/--snapshot doctorCommand.ts:266
5 Suggestion No timeout on optionalProbe; NFS/FUSE /proc can cause indefinite hang memoryDiagnostics.ts:257
6 Suggestion Parent /doctor action: --snapshot/--sample flags parsed but unreachable outside memory subcommand doctorCommand.ts:89
7 Suggestion Arena dialog callbacks + ACP stream_messages lack warning branch; risk warnings silently swallowed ArenaStopDialog.tsx:43, Session.ts:2285
8 Suggestion countOpenFileDescriptors allocates full string array via readdir; opendir async iterator is more memory-efficient memoryDiagnostics.ts:226
9 Suggestion formatSmapsRollup can use regex instead of split+map+find doctorCommand.ts:339

Finding 1 — Critical: maxRSS unit mismatch

Empirically verified on this machine (Node v22.22.2, macOS ARM64):

process.resourceUsage().maxRSS = 40816  (kilobytes)
process.memoryUsage().rss      = 41844736  (bytes ≈ 39.9 MB)
maxRSS * 1024 / rss            = 0.998  (matches!)
maxRSS / rss                   = 0.00098  (off by 1024×)

The code comment at line 132 states "Node.js >=14.10.0 returns maxRSS in bytes on all platforms" — this contradicts the official Node.js documentation which states maxRSS returns kilobytes on Linux/macOS. The result: formatMemoryUsage(maxRSSBytes) displays 39.9 KB instead of the correct 39.9 MB.

Fix: const maxRSSBytes = resourceUsage.maxRSS * 1024; (and update the comment).

Note: Commit 146d165a00 ("correct maxRSS byte handling") removed a * 1024 multiplier, which appears to have introduced this regression.


Finding 2 — Critical: warning exits 0 in non-interactive CLI

nonInteractiveCli.ts:326:

return slashCommandResult.messageType === 'error' ? 1 : 0;

When /doctor memory detects risks, it returns messageType: 'warning'. In non-interactive mode, this maps to exit code 0 — indistinguishable from a clean run. CI pipelines that check exit codes will silently pass.

Fix: Add a warning branch:

if (slashCommandResult.messageType === 'error') return 1;
if (slashCommandResult.messageType === 'warning') return 2;
return 0;

Note: This finding was partially flagged in a prior review round as "warning exit code logic not updated." The current code still has the issue.


Finding 3 — Missing JSDoc units on MemoryResourceUsage

memoryDiagnostics.ts:56-63 — the interface fields have no unit documentation:

export interface MemoryResourceUsage {
  maxRSS: number;       // KB (per Node.js docs)
  userCPUTime: number;  // microseconds (per Node.js docs)
  systemCPUTime: number; // microseconds
}

Future consumers (tests, integrations, Bun compatibility) will silently get values wrong by orders of magnitude.


Finding 4 — --json silently dropped with --sample/--snapshot

doctorCommand.ts:266 — when shouldSampleMemory || shouldWriteHeapSnapshot, the code strips --json and delegates to the parent action, which returns plain text. No error or warning is emitted. Automated scripts expecting JSON will receive unparseable text.

Fix: Reject the combination explicitly:

if (tokens.includes('--json') && (shouldSampleMemory || shouldWriteHeapSnapshot)) {
  return { type: 'message', messageType: 'error', content: t('--json is not supported with --sample or --snapshot') };
}

Finding 5 — No timeout on optionalProbe

memoryDiagnostics.ts:257await probe() has no timeout. On NFS-mounted /proc or hung FUSE filesystems, readdir('/proc/self/fd') or readFile('/proc/self/smaps_rollup') can hang indefinitely.

Fix: Add a Promise.race with a 5-second timeout inside optionalProbe.


Finding 6 — Parent /doctor dead code

doctorCommand.ts:89-103shouldSampleMemory and shouldWriteHeapSnapshot are parsed at the parent level, but the snapshot/sample execution block is inside the if (subCommand === MEMORY_SUBCOMMAND) branch. When a user types /doctor --snapshot (without memory), subCommand is '--snapshot', which doesn't match, so the flags are silently ignored.


Finding 7 — Warning handling missing in Arena + ACP paths

  • ArenaStopDialog.tsx:43 and ArenaSelectDialog.tsx:42 type callbacks as 'info' | 'error'warning maps to MessageType.ERROR.
  • Session.ts:2285 (ACP stream_messages): only error triggers throw; warning is emitted as a plain agentMessage with no severity signal.

Finding 8 — countOpenFileDescriptors allocates full string array

memoryDiagnostics.ts:226readdir('/proc/self/fd') returns Promise<string[]>, allocating a string per fd just to read .length. On a process with fd leaks (the exact scenario this diagnostic targets), this could be thousands of strings. Using fs.opendir with an async iterator avoids the allocation.


Finding 9 — formatSmapsRollup optimization

doctorCommand.ts:339 — the function splits all lines, maps each to a trimmed string, then finds the Rss: line. A regex smapsRollup.match(/^Rss:\s*.+$/m) would be simpler and avoid intermediate allocations.


Previously flagged items now resolved: i18n missing keys ✅, formatError duplication ✅, probe failure logging ✅, getActiveHandlesCount/getActiveRequestsCount deduplication ✅.

Open items from prior review still not addressed: 7 inline comments from @wenshao remain open (see individual comment threads).

Comment thread packages/core/src/utils/memoryDiagnostics.ts
Comment thread packages/core/src/utils/memoryDiagnostics.ts
Comment thread packages/core/src/utils/memoryDiagnostics.ts
Comment thread packages/core/src/utils/memoryDiagnostics.ts
Comment thread packages/cli/src/ui/commands/doctorCommand.ts
Comment thread packages/cli/src/ui/commands/doctorCommand.ts
Comment thread packages/core/src/utils/memoryDiagnostics.ts
Comment thread packages/core/src/utils/memoryDiagnostics.ts
Comment thread packages/core/src/utils/memoryDiagnostics.ts
Comment thread packages/core/src/utils/memoryDiagnostics.ts
Comment thread packages/cli/src/ui/commands/doctorCommand.ts
Comment thread packages/cli/src/ui/commands/doctorCommand.ts
yiliang114 and others added 12 commits May 17, 2026 17:47
- Extract platform detection before building diagnostics so the correct
  unit conversion can be applied: multiply by 1024 on Linux (where
  process.resourceUsage().maxRSS is in KB) but leave the value unchanged
  on macOS/Windows (where it is already in bytes).
- Correct the native-memory-pressure risk message to accurately state
  that the threshold is 2× heap used, not just "larger than heapUsed".
- Add a dedicated test to assert that maxRSS is not multiplied on a
  non-Linux platform (darwin).

All 3 core and 9 CLI tests pass; typecheck clean.

Agent-Logs-Url: https://github.com/QwenLM/qwen-code/sessions/9b413337-68ed-4d5c-af99-0d42378900c3
…output

- Add 64MB absolute floor on native-memory-pressure so cold processes don't trip
  the 2x ratio check; raise active-handles threshold from 100 to 256
- Show detachedContexts, nativeContexts, maxRSS, CPU times, smapsRollup
  availability, and v8HeapSpaces summary in the readable /doctor memory output
- Validate unknown memory subcommand args with a usage hint instead of silently
  dropping them
- Wrap human-readable strings in t(...) for i18n parity with the rest of doctor
- Advertise the memory subcommand via /doctor argumentHint while keeping
  acceptsInput false so the parent still auto-submits
- Document _getActiveHandles/_getActiveRequests as undocumented Node internals
- Update tests for new thresholds, expanded output, unknown-arg path, and
  abort-during-json
- Remove incorrect * 1024 multiplier for maxRSS on Linux (Node.js >=14.10 returns bytes on all platforms)
- Use v8HeapStats.usedHeapSize for heapRatio to avoid cross-API inconsistency
- Update test expectations and rename "does not multiply" test
- Rename local formatMemoryDiagnostics to formatCoreDiagnostics to avoid
  naming conflict with the imported utility from memoryDiagnostics.js
- Update Session.test.ts to use objectContaining for _meta field added
  in recent main commits
- Align doctorCommand.test.ts assertions with current parent command
  state (argumentHint includes --sample/--snapshot from main)
…cate active count helpers

- optionalProbe/optionalSyncProbe now return null on failure so
  JSON.stringify preserves the keys instead of silently omitting them.
- Merge getActiveHandlesCount/getActiveRequestsCount into a single
  parameterized getProcessInternalCount helper.
- Update MemoryDiagnostics interface: v8HeapSpaces, openFileDescriptors,
  smapsRollup are now T | null instead of T | undefined.
@yiliang114 yiliang114 force-pushed the codex/memory-diagnostics branch from 55c4fad to b3921c2 Compare May 17, 2026 09:48

@wenshao wenshao left a comment

Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggestion: Dual memory diagnostics code paths with different thresholds

The PR introduces a new collectMemoryDiagnostics() path (core layer, heap-pressure threshold 0.75) but leaves the existing /doctor memory --sample / --snapshot path using the old getMemoryDiagnostics() (cli layer, threshold 0.85). The two paths produce different data shapes, use different risk thresholds, and the old path never returns 'warning' message type. Over time these paths will diverge — changes to the core diagnostics won't be reflected in --sample/--snapshot output.

Consider migrating --sample/--snapshot to use collectMemoryDiagnostics() as the unified data source in a follow-up PR.


— DeepSeek/deepseek-v4-pro via Qwen Code /review

if (shouldSampleMemory || shouldWriteHeapSnapshot) {
return doctorCommand.action?.(
context,
[MEMORY_SUBCOMMAND, ...tokens.filter((token) => token !== '--json')].join(

Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

[Suggestion] --json is silently dropped when combined with --sample/--snapshot

memoryDoctorAction filters out --json before delegating to the parent doctorCommand.action, which never returns JSON. The user receives text output with no error. And argumentHint lists [--json] [--sample] [--snapshot], implying they can be combined.

Suggested change
[MEMORY_SUBCOMMAND, ...tokens.filter((token) => token !== '--json')].join(
if (shouldSampleMemory || shouldWriteHeapSnapshot) {
if (tokens.includes('--json')) {
return {
type: 'message' as const,
messageType: 'error' as const,
content: `${t('--json is not compatible with --sample or --snapshot')}. ${t('Usage')}: ${MEMORY_USAGE_HINT}`,
};
}
return doctorCommand.action?.(
context,
[MEMORY_SUBCOMMAND, ...tokens].join(' '),
);
}

— DeepSeek/deepseek-v4-pro via Qwen Code /review

@wenshao wenshao merged commit eef06ce into main May 17, 2026
10 checks passed
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

None yet

Projects

None yet

Development

Successfully merging this pull request may close these issues.

3 participants