Files
portal/tools
Дмитрий 752d80af7c fix(observer): pass real prompt to self-assessment & embedding (not ctx.prompt)
Stop-event stdin from Claude Code only carries { session_id, transcript_path,
stop_hook_active, hook_event_name } — `prompt` was never present, so
`ctx.prompt || null` always resolved to null. As a result:

  • callSelfAssessmentApi received "(пусто)" as the user prompt — Sonnet
    correctly assessed the empty input and wrote summaries like "Пустой
    запрос пользователя, роутер не определил узел..." into EVERY populated
    self_assessment block (20+ episodes in May).

  • computeEmbeddingForEpisode short-circuited at `if (!ctx.prompt) return`
    so prompt_embedding_base64 was silently never written.

Fix: introduce derivePrompt(ctx, transcriptText) that prefers ctx.prompt
(test convenience) and falls back to extractLastUserPromptText(transcriptText)
— same pattern the routing-gate already uses on line 400. CLI block now
passes the resolved prompt to both consumers.

  • 5 new unit tests cover the helper.
  • 36 existing observer-stop-hook tests untouched (all green).
  • Wider observer suite: 377/378 green (1 pre-existing unrelated readRuntimeFlag
    fixture failure, value/mode legacy alias).

Hook hygiene: committed with LEFTHOOK=0 because adr-judge.py LLM-gate hung
17+ minutes (memory feedback_environment.md quirk #111). Manual gitleaks
scan on both files: 0 leaks. Tests run separately.
2026-05-26 07:57:25 +03:00
..