85c3ec0222
- 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>
38 lines
1.7 KiB
Python
38 lines
1.7 KiB
Python
from datetime import datetime, timezone
|
|
from typing import TYPE_CHECKING
|
|
|
|
from sqlalchemy import DateTime, ForeignKey, Integer, String, Text
|
|
from sqlalchemy.orm import Mapped, mapped_column, relationship
|
|
|
|
from db.base import Base
|
|
|
|
if TYPE_CHECKING:
|
|
from db.models.thread import Thread
|
|
|
|
|
|
def _utcnow() -> datetime:
|
|
return datetime.now(timezone.utc)
|
|
|
|
|
|
class Message(Base):
|
|
__tablename__ = "messages"
|
|
|
|
id: Mapped[int] = mapped_column(Integer, primary_key=True, autoincrement=True)
|
|
thread_id: Mapped[int] = mapped_column(
|
|
ForeignKey("threads.id", ondelete="CASCADE"), nullable=False, index=True
|
|
)
|
|
role: Mapped[str] = mapped_column(String(20), nullable=False) # "user" | "assistant"
|
|
text: Mapped[str] = mapped_column(Text, nullable=False)
|
|
sources_json: Mapped[str | None] = mapped_column(Text, nullable=True)
|
|
assembled_prompt: Mapped[str | None] = mapped_column(Text, nullable=True)
|
|
# Ветка, которую выбрал роутер для этой реплики (проставляется со Спринта 4).
|
|
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")
|