feat(sprint6b-D): soft-insertion counter + message meta_json

- thread_state.soft_insertion_count: растёт при боковом ответе (soft_insertion=true
  в STATE_JSON без смены шага/слотов), сбрасывается при продвижении или handoff
- При soft_insertion_count >= 3 в системный промпт ветки добавляется SOFT_INSERTION_NUDGE
  — явная инструкция вернуть пациента к вопросу текущего шага
- state_machine.parse_branch_response читает флаг soft_insertion из STATE_JSON
- Новая колонка message.meta_json: {router_intent_code, served_intent_code, step_code, events}
  — хранит снимок маршрутизации каждой реплики ассистента
- «Песочница»: бейджи событий (sticky / soft_insertion / hard_handoff / resumed /
  routing_loop / validation_blocked) над каждым ответом ассистента

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
AR 15 M4
2026-04-25 20:24:22 +05:00
parent 3c71372ec8
commit 85c3ec0222
12 changed files with 257 additions and 13 deletions
+3
View File
@@ -29,6 +29,9 @@ class Message(Base):
intent_id: Mapped[int | None] = mapped_column(
ForeignKey("intents.id", ondelete="SET NULL"), nullable=True, index=True
)
# JSON со снимком обработки реплики: решение роутера, шаг, список событий.
# Используется в Песочнице для отображения подробных пилюль (со Спринта 6b).
meta_json: Mapped[str | None] = mapped_column(Text, nullable=True)
created_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), default=_utcnow, nullable=False)
thread: Mapped["Thread"] = relationship(back_populates="messages")
+2
View File
@@ -27,6 +27,8 @@ class ThreadState(Base):
current_step_code: Mapped[str | None] = mapped_column(String(50), nullable=True)
slots_json: Mapped[str] = mapped_column(Text, nullable=False, default="{}")
handoff_count: Mapped[int] = mapped_column(Integer, nullable=False, default=0)
# Счётчик «боковых вопросов» подряд — блок D Спринта 6b (v2 §4.2).
soft_insertion_count: Mapped[int] = mapped_column(Integer, nullable=False, default=0)
# Состояние прерванного сценария — блок C Спринта 6a (v2 §4.4).
suspended_intent: Mapped[str | None] = mapped_column(String(50), nullable=True)
resumable_step_code: Mapped[str | None] = mapped_column(String(50), nullable=True)