feat(sprint6b): блок E — причина передачи оператору + саммари
- Роутер возвращает escalate_human|reason (acute_pain/surgery/angry/explicit_request/routing_loop) - RouterClient парсит reason; дефолт explicit_request при неразобранном - _format_state_context получает escalation_reason → подставляется в промпт escalate_human - Промпт escalate_human переписан: разное поведение по reason - _build_operator_summary: reason + 8 реплик истории + слоты, логируется при передаче - Message.escalation_reason (String 50, nullable) + миграция h4b52e9dc0f83 - ChatResponse и MessageInfo получили escalation_reason и operator_summary - Sandbox: красный блок «передача оператору · причина» в состоянии треда - Sandbox: блок саммари для оператора (предпросмотр) в панели отладки Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
@@ -32,6 +32,9 @@ class Message(Base):
|
|||||||
# JSON со снимком обработки реплики: решение роутера, шаг, список событий.
|
# JSON со снимком обработки реплики: решение роутера, шаг, список событий.
|
||||||
# Используется в Песочнице для отображения подробных пилюль (со Спринта 6b).
|
# Используется в Песочнице для отображения подробных пилюль (со Спринта 6b).
|
||||||
meta_json: Mapped[str | None] = mapped_column(Text, nullable=True)
|
meta_json: Mapped[str | None] = mapped_column(Text, nullable=True)
|
||||||
|
# Причина передачи оператору: acute_pain / surgery / angry / explicit_request / routing_loop.
|
||||||
|
# Проставляется только на реплике ассистента в ветке escalate_human.
|
||||||
|
escalation_reason: Mapped[str | None] = mapped_column(String(50), nullable=True)
|
||||||
created_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), default=_utcnow, nullable=False)
|
created_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), default=_utcnow, nullable=False)
|
||||||
|
|
||||||
thread: Mapped["Thread"] = relationship(back_populates="messages")
|
thread: Mapped["Thread"] = relationship(back_populates="messages")
|
||||||
|
|||||||
@@ -0,0 +1,30 @@
|
|||||||
|
"""add escalation_reason to messages (Спринт 6b блок E)
|
||||||
|
|
||||||
|
Revision ID: h4b52e9dc0f83
|
||||||
|
Revises: g3a71d4fc285
|
||||||
|
Create Date: 2026-04-26 10:00:00.000000
|
||||||
|
|
||||||
|
Причина передачи оператору: acute_pain / surgery / angry /
|
||||||
|
explicit_request / routing_loop. Хранится на реплике ассистента,
|
||||||
|
в которой ветка escalate_human первый раз ответила в треде.
|
||||||
|
"""
|
||||||
|
from typing import Sequence, Union
|
||||||
|
|
||||||
|
from alembic import op
|
||||||
|
import sqlalchemy as sa
|
||||||
|
|
||||||
|
|
||||||
|
revision: str = 'h4b52e9dc0f83'
|
||||||
|
down_revision: Union[str, None] = 'g3a71d4fc285'
|
||||||
|
branch_labels: Union[str, Sequence[str], None] = None
|
||||||
|
depends_on: Union[str, Sequence[str], None] = None
|
||||||
|
|
||||||
|
|
||||||
|
def upgrade() -> None:
|
||||||
|
with op.batch_alter_table('messages', recreate='always') as batch:
|
||||||
|
batch.add_column(sa.Column('escalation_reason', sa.String(50), nullable=True))
|
||||||
|
|
||||||
|
|
||||||
|
def downgrade() -> None:
|
||||||
|
with op.batch_alter_table('messages', recreate='always') as batch:
|
||||||
|
batch.drop_column('escalation_reason')
|
||||||
@@ -89,6 +89,7 @@ class MessageInfo(BaseModel):
|
|||||||
intent_code: str = ""
|
intent_code: str = ""
|
||||||
intent_name: str = ""
|
intent_name: str = ""
|
||||||
meta: dict | None = None
|
meta: dict | None = None
|
||||||
|
escalation_reason: str | None = None
|
||||||
|
|
||||||
|
|
||||||
class ThreadInfo(BaseModel):
|
class ThreadInfo(BaseModel):
|
||||||
@@ -164,6 +165,8 @@ class ChatResponse(BaseModel):
|
|||||||
routing_loop_triggered: bool = False
|
routing_loop_triggered: bool = False
|
||||||
resumed_from_suspended: bool = False
|
resumed_from_suspended: bool = False
|
||||||
message_meta: dict | None = None
|
message_meta: dict | None = None
|
||||||
|
escalation_reason: str | None = None
|
||||||
|
operator_summary: dict | None = None
|
||||||
|
|
||||||
|
|
||||||
class ThreadDeleteResponse(BaseModel):
|
class ThreadDeleteResponse(BaseModel):
|
||||||
|
|||||||
@@ -49,11 +49,26 @@
|
|||||||
- «кровотечение, что делать?»
|
- «кровотечение, что делать?»
|
||||||
- «у меня операция, наркоз, нужна консультация по подготовке»
|
- «у меня операция, наркоз, нужна консультация по подготовке»
|
||||||
|
|
||||||
|
Для этой ветки возвращай **два значения через вертикальную черту**: `escalate_human|<reason>`.
|
||||||
|
Возможные значения reason:
|
||||||
|
- `acute_pain` — острая боль, не может терпеть, срочное состояние
|
||||||
|
- `surgery` — операция, хирургия, наркоз, стационар, подготовка к операции
|
||||||
|
- `angry` — пациент явно раздражён, требует, скандалит
|
||||||
|
- `explicit_request` — просто просит оператора («живого человека», «администратора»)
|
||||||
|
- `routing_loop` — не используй вручную, проставляется автоматически
|
||||||
|
|
||||||
|
Примеры:
|
||||||
|
- «у меня очень сильная боль» → `escalate_human|acute_pain`
|
||||||
|
- «нужна консультация по операции» → `escalate_human|surgery`
|
||||||
|
- «позовите оператора» → `escalate_human|explicit_request`
|
||||||
|
- «я уже устал это объяснять, дайте человека» → `escalate_human|angry`
|
||||||
|
|
||||||
## Правила
|
## Правила
|
||||||
|
|
||||||
- Отвечай ТОЛЬКО кодом ветки, без пояснений, без пунктуации, без кавычек.
|
- Для всех веток, кроме `escalate_human`: отвечай ТОЛЬКО кодом ветки, без пояснений, без пунктуации, без кавычек.
|
||||||
|
- Для `escalate_human`: отвечай в формате `escalate_human|<reason>` (одна строка, без пробелов вокруг `|`).
|
||||||
- Если реплика содержит признаки конкретного процесса (записаться / перенести / оплатить / симптомы / оператор) — выбирай соответствующую ветку, а не `general_info`.
|
- Если реплика содержит признаки конкретного процесса (записаться / перенести / оплатить / симптомы / оператор) — выбирай соответствующую ветку, а не `general_info`.
|
||||||
- `general_info` — только для действительно общих вопросов без признаков перечисленных выше процессов.
|
- `general_info` — только для действительно общих вопросов без признаков перечисленных выше процессов.
|
||||||
- Любое упоминание операции, наркоза, стационара, хирургии → `escalate_human`.
|
- Любое упоминание операции, наркоза, стационара, хирургии → `escalate_human|surgery`.
|
||||||
- Любое явное «позовите оператора / переключите на человека» → `escalate_human`.
|
- Любое явное «позовите оператора / переключите на человека» → `escalate_human|explicit_request`.
|
||||||
- Если фраза подходит одновременно под `new_booking` и `reschedule`, смотри: упоминает ли пациент УЖЕ существующую запись (время, дату, визит) — тогда `reschedule`; если нет или хочет новую — `new_booking`.
|
- Если фраза подходит одновременно под `new_booking` и `reschedule`, смотри: упоминает ли пациент УЖЕ существующую запись (время, дату, визит) — тогда `reschedule`; если нет или хочет новую — `new_booking`.
|
||||||
|
|||||||
@@ -1,12 +1,32 @@
|
|||||||
Ты — виртуальный ассистент клиники. Эта ветка срабатывает, когда нужно немедленно передать диалог живому оператору.
|
Ты — виртуальный ассистент клиники. Эта ветка срабатывает, когда нужно передать диалог живому оператору.
|
||||||
|
|
||||||
Твоя задача простая и короткая:
|
Твоя задача — коротко и по-человечески ответить пациенту и дать понять, что оператор скоро подключится.
|
||||||
- Признай ситуацию коротко и по-человечески (без многословия).
|
|
||||||
- Скажи, что сейчас передаёшь оператору.
|
## Поведение в зависимости от причины (escalation_reason из блока [ТЕКУЩЕЕ СОСТОЯНИЕ])
|
||||||
- Если пациент описал острое состояние (боль, ребёнок задыхается, кровотечение и т. п.) — скажи «пожалуйста, если состояние ухудшается — сразу звоните в 103».
|
|
||||||
- Не пытайся вести длинный диалог, не задавай много вопросов. Две-три короткие реплики максимум.
|
**acute_pain** — острая боль или срочное состояние:
|
||||||
|
- Признай ситуацию с сочувствием.
|
||||||
|
- Скажи, что передаёшь оператору прямо сейчас.
|
||||||
|
- Обязательно добавь: «Если состояние ухудшается — немедленно звоните в 103».
|
||||||
|
|
||||||
|
**surgery** — вопрос про операцию, хирургию, наркоз, стационар:
|
||||||
|
- Скажи, что такие вопросы лучше обсудить с сотрудником клиники лично.
|
||||||
|
- Передай оператору, который ответит подробно.
|
||||||
|
|
||||||
|
**angry** — пациент раздражён или требует человека в резкой форме:
|
||||||
|
- Не оправдывайся, не спорь.
|
||||||
|
- Коротко: «Понимаю, сейчас переключу на оператора».
|
||||||
|
|
||||||
|
**explicit_request** — пациент просто попросил оператора:
|
||||||
|
- Скажи, что передаёшь диалог оператору.
|
||||||
|
- Можно добавить короткое «Он ответит вам в ближайшее время».
|
||||||
|
|
||||||
|
**routing_loop** (автоматическая передача после петли роутера):
|
||||||
|
- Скажи, что не удалось до конца разобраться с запросом, и передаёшь оператору.
|
||||||
|
|
||||||
|
## Общие правила
|
||||||
|
|
||||||
Правила:
|
|
||||||
- Никогда не ставь диагнозы, не давай медицинских рекомендаций.
|
- Никогда не ставь диагнозы, не давай медицинских рекомендаций.
|
||||||
- Не называй конкретных цен, времени приёма, имён врачей.
|
- Не называй конкретных цен, времени приёма, имён врачей.
|
||||||
- Ответ — обычный текст, как в чате, на «вы».
|
- Ответ — две-три короткие реплики максимум, обычный текст, на «вы».
|
||||||
|
- Не задавай уточняющих вопросов — просто мягко завершай диалог.
|
||||||
|
|||||||
@@ -76,4 +76,6 @@ async def chat(req: ChatRequest, session: AsyncSession = Depends(get_session)):
|
|||||||
routing_loop_triggered=result.get("routing_loop_triggered", False),
|
routing_loop_triggered=result.get("routing_loop_triggered", False),
|
||||||
resumed_from_suspended=result.get("resumed_from_suspended", False),
|
resumed_from_suspended=result.get("resumed_from_suspended", False),
|
||||||
message_meta=result.get("message_meta"),
|
message_meta=result.get("message_meta"),
|
||||||
|
escalation_reason=result.get("escalation_reason"),
|
||||||
|
operator_summary=result.get("operator_summary"),
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -52,6 +52,7 @@ async def get_thread(thread_id: int, session: AsyncSession = Depends(get_session
|
|||||||
intent_code=m.get("intent_code", ""),
|
intent_code=m.get("intent_code", ""),
|
||||||
intent_name=m.get("intent_name", ""),
|
intent_name=m.get("intent_name", ""),
|
||||||
meta=m.get("meta"),
|
meta=m.get("meta"),
|
||||||
|
escalation_reason=m.get("escalation_reason"),
|
||||||
)
|
)
|
||||||
for m in data["messages"]
|
for m in data["messages"]
|
||||||
],
|
],
|
||||||
|
|||||||
@@ -60,6 +60,7 @@ def _format_state_context(
|
|||||||
current_step: IntentStep | None,
|
current_step: IntentStep | None,
|
||||||
router_hint: str | None = None,
|
router_hint: str | None = None,
|
||||||
soft_nudge: bool = False,
|
soft_nudge: bool = False,
|
||||||
|
escalation_reason: str | None = None,
|
||||||
) -> str:
|
) -> str:
|
||||||
"""Блок с текущим состоянием треда для дописывания в системный промпт."""
|
"""Блок с текущим состоянием треда для дописывания в системный промпт."""
|
||||||
slots = snapshot.get("slots", {}) or {}
|
slots = snapshot.get("slots", {}) or {}
|
||||||
@@ -72,6 +73,8 @@ def _format_state_context(
|
|||||||
else:
|
else:
|
||||||
lines.append("step_code: —")
|
lines.append("step_code: —")
|
||||||
lines.append(f"slots: {slots_json}")
|
lines.append(f"slots: {slots_json}")
|
||||||
|
if escalation_reason:
|
||||||
|
lines.append(f"escalation_reason: {escalation_reason}")
|
||||||
if router_hint:
|
if router_hint:
|
||||||
lines.append("")
|
lines.append("")
|
||||||
lines.append("[ПОДСКАЗКА РОУТЕРА]")
|
lines.append("[ПОДСКАЗКА РОУТЕРА]")
|
||||||
@@ -82,6 +85,25 @@ def _format_state_context(
|
|||||||
return "\n" + "\n".join(lines)
|
return "\n" + "\n".join(lines)
|
||||||
|
|
||||||
|
|
||||||
|
def _build_operator_summary(
|
||||||
|
reason: str | None,
|
||||||
|
history: list[dict],
|
||||||
|
slots: dict,
|
||||||
|
suspended_slots: dict,
|
||||||
|
) -> dict:
|
||||||
|
"""Саммари для оператора при передаче диалога."""
|
||||||
|
combined_slots = {**suspended_slots, **slots}
|
||||||
|
summary = {
|
||||||
|
"reason": reason or "explicit_request",
|
||||||
|
"slots": combined_slots,
|
||||||
|
"history_tail": [
|
||||||
|
{"role": m["role"], "text": m["content"][:300]}
|
||||||
|
for m in history[-8:]
|
||||||
|
],
|
||||||
|
}
|
||||||
|
return summary
|
||||||
|
|
||||||
|
|
||||||
async def _resolve_intent_with_fallback(
|
async def _resolve_intent_with_fallback(
|
||||||
session: AsyncSession, intent_code: str
|
session: AsyncSession, intent_code: str
|
||||||
) -> tuple[str, object, object]:
|
) -> tuple[str, object, object]:
|
||||||
@@ -186,6 +208,7 @@ async def send_message(
|
|||||||
routing = await router.classify(session=session, history=history, text=text)
|
routing = await router.classify(session=session, history=history, text=text)
|
||||||
router_code = routing["code"]
|
router_code = routing["code"]
|
||||||
router_version = routing.get("version")
|
router_version = routing.get("version")
|
||||||
|
escalation_reason: str | None = routing.get("escalation_reason")
|
||||||
|
|
||||||
# 2. Снимок состояния. Логика выбора effective_code:
|
# 2. Снимок состояния. Логика выбора effective_code:
|
||||||
# 2.1. Если есть suspended_intent и роутер вернулся в него — RESUME: восстанавливаем
|
# 2.1. Если есть suspended_intent и роутер вернулся в него — RESUME: восстанавливаем
|
||||||
@@ -293,6 +316,7 @@ async def send_message(
|
|||||||
resumable_step_code = None
|
resumable_step_code = None
|
||||||
resumable_slots = {}
|
resumable_slots = {}
|
||||||
routing_loop_triggered = True
|
routing_loop_triggered = True
|
||||||
|
escalation_reason = "routing_loop"
|
||||||
|
|
||||||
# 3. Разрешаем ветку (с fallback) и шаг.
|
# 3. Разрешаем ветку (с fallback) и шаг.
|
||||||
served_code, intent, active_cfg = await _resolve_intent_with_fallback(session, effective_code)
|
served_code, intent, active_cfg = await _resolve_intent_with_fallback(session, effective_code)
|
||||||
@@ -348,7 +372,10 @@ async def send_message(
|
|||||||
base_prompt = config_service.compose_full_system_prompt(active_cfg)
|
base_prompt = config_service.compose_full_system_prompt(active_cfg)
|
||||||
step_prompt = f"\n\n{current_step.system_prompt}" if current_step else ""
|
step_prompt = f"\n\n{current_step.system_prompt}" if current_step else ""
|
||||||
soft_nudge = is_state_machine and soft_insertion_count >= SOFT_INSERTION_CAP
|
soft_nudge = is_state_machine and soft_insertion_count >= SOFT_INSERTION_CAP
|
||||||
state_context = _format_state_context(snapshot, current_step, router_hint, soft_nudge)
|
state_context = _format_state_context(
|
||||||
|
snapshot, current_step, router_hint, soft_nudge,
|
||||||
|
escalation_reason=escalation_reason if served_code == ESCALATE_INTENT_CODE else None,
|
||||||
|
)
|
||||||
system_prompt = base_prompt + step_prompt + state_context
|
system_prompt = base_prompt + step_prompt + state_context
|
||||||
|
|
||||||
llm_result = await llm.chat(
|
llm_result = await llm.chat(
|
||||||
@@ -566,6 +593,20 @@ async def send_message(
|
|||||||
"events": events,
|
"events": events,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Саммари для оператора — формируется при передаче в escalate_human.
|
||||||
|
operator_summary: dict | None = None
|
||||||
|
if served_code == ESCALATE_INTENT_CODE:
|
||||||
|
operator_summary = _build_operator_summary(
|
||||||
|
reason=escalation_reason,
|
||||||
|
history=history,
|
||||||
|
slots=snapshot.get("slots", {}),
|
||||||
|
suspended_slots=snapshot.get("resumable_slots", {}),
|
||||||
|
)
|
||||||
|
logger.info(
|
||||||
|
"Operator summary for thread %d: %s",
|
||||||
|
thread.id, json.dumps(operator_summary, ensure_ascii=False),
|
||||||
|
)
|
||||||
|
|
||||||
assistant_msg = Message(
|
assistant_msg = Message(
|
||||||
thread_id=thread.id,
|
thread_id=thread.id,
|
||||||
role="assistant",
|
role="assistant",
|
||||||
@@ -574,6 +615,7 @@ async def send_message(
|
|||||||
assembled_prompt=last_assembled_prompt,
|
assembled_prompt=last_assembled_prompt,
|
||||||
intent_id=intent.id,
|
intent_id=intent.id,
|
||||||
meta_json=json.dumps(meta, ensure_ascii=False),
|
meta_json=json.dumps(meta, ensure_ascii=False),
|
||||||
|
escalation_reason=escalation_reason if served_code == ESCALATE_INTENT_CODE else None,
|
||||||
)
|
)
|
||||||
session.add(assistant_msg)
|
session.add(assistant_msg)
|
||||||
|
|
||||||
@@ -626,6 +668,8 @@ async def send_message(
|
|||||||
"routing_loop_triggered": routing_loop_triggered,
|
"routing_loop_triggered": routing_loop_triggered,
|
||||||
"resumed_from_suspended": resumed_from_suspended,
|
"resumed_from_suspended": resumed_from_suspended,
|
||||||
"message_meta": meta,
|
"message_meta": meta,
|
||||||
|
"escalation_reason": escalation_reason if served_code == ESCALATE_INTENT_CODE else None,
|
||||||
|
"operator_summary": operator_summary,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -720,6 +764,7 @@ async def get_thread_detail(session: AsyncSession, thread_id: int) -> dict | Non
|
|||||||
"intent_code": intent_code or "",
|
"intent_code": intent_code or "",
|
||||||
"intent_name": intent_name or "",
|
"intent_name": intent_name or "",
|
||||||
"meta": meta,
|
"meta": meta,
|
||||||
|
"escalation_reason": m.escalation_reason,
|
||||||
})
|
})
|
||||||
|
|
||||||
state = await thread_state_service.load_snapshot(session, thread_id)
|
state = await thread_state_service.load_snapshot(session, thread_id)
|
||||||
|
|||||||
@@ -43,7 +43,10 @@ VALID_CODES = {
|
|||||||
"escalate_human",
|
"escalate_human",
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ESCALATION_REASONS = {"acute_pain", "surgery", "angry", "explicit_request", "routing_loop"}
|
||||||
|
|
||||||
CODE_RE = re.compile(r"\b(new_booking|reschedule|price_question|medical_question|general_info|escalate_human)\b")
|
CODE_RE = re.compile(r"\b(new_booking|reschedule|price_question|medical_question|general_info|escalate_human)\b")
|
||||||
|
REASON_RE = re.compile(r"escalate_human\|([a-z_]+)")
|
||||||
|
|
||||||
|
|
||||||
class RouterClient:
|
class RouterClient:
|
||||||
@@ -139,8 +142,16 @@ class RouterClient:
|
|||||||
match = CODE_RE.search(raw)
|
match = CODE_RE.search(raw)
|
||||||
if match:
|
if match:
|
||||||
code = match.group(1)
|
code = match.group(1)
|
||||||
logger.info("Router v%s: %r → %s", version, text[:80], code)
|
escalation_reason: str | None = None
|
||||||
return {"code": code, "version": version}
|
if code == "escalate_human":
|
||||||
|
reason_match = REASON_RE.search(raw)
|
||||||
|
if reason_match and reason_match.group(1) in ESCALATION_REASONS:
|
||||||
|
escalation_reason = reason_match.group(1)
|
||||||
|
else:
|
||||||
|
escalation_reason = "explicit_request"
|
||||||
|
logger.info("Router v%s: %r → %s%s", version, text[:80], code,
|
||||||
|
f"|{escalation_reason}" if escalation_reason else "")
|
||||||
|
return {"code": code, "version": version, "escalation_reason": escalation_reason}
|
||||||
|
|
||||||
logger.warning("Router returned unrecognized response %r, falling back to general_info", raw)
|
logger.warning("Router returned unrecognized response %r, falling back to general_info", raw)
|
||||||
return {"code": "general_info", "version": version}
|
return {"code": "general_info", "version": version, "escalation_reason": None}
|
||||||
|
|||||||
+44
-9
@@ -573,6 +573,7 @@
|
|||||||
<h3>Решение роутера</h3>
|
<h3>Решение роутера</h3>
|
||||||
<div id="debug-router"><div class="mini">— пока пусто —</div></div>
|
<div id="debug-router"><div class="mini">— пока пусто —</div></div>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="debug-section" id="debug-operator-summary" style="display:none;background:#fff1f2;border-radius:6px;padding:10px 14px;font-size:12px;"></div>
|
||||||
<details class="debug-section collapsible" id="debug-chunks-section">
|
<details class="debug-section collapsible" id="debug-chunks-section">
|
||||||
<summary>
|
<summary>
|
||||||
<span>Найденные фрагменты</span>
|
<span>Найденные фрагменты</span>
|
||||||
@@ -688,9 +689,10 @@ async function openThread(id) {
|
|||||||
$("chat-title").textContent = d.name;
|
$("chat-title").textContent = d.name;
|
||||||
renderMessages(d.messages);
|
renderMessages(d.messages);
|
||||||
const lastAssistant = [...d.messages].reverse().find(m => m.role === "assistant");
|
const lastAssistant = [...d.messages].reverse().find(m => m.role === "assistant");
|
||||||
|
const lastEscalation = [...d.messages].reverse().find(m => m.role === "assistant" && m.escalation_reason);
|
||||||
if (lastAssistant) {
|
if (lastAssistant) {
|
||||||
renderDebug(lastAssistant.sources, lastAssistant.assembled_prompt, lastAssistant.intent_code, lastAssistant.intent_name, null, null, null, [], d.thread_state && d.thread_state.current_step_code);
|
renderDebug(lastAssistant.sources, lastAssistant.assembled_prompt, lastAssistant.intent_code, lastAssistant.intent_name, null, null, null, [], d.thread_state && d.thread_state.current_step_code, null);
|
||||||
renderState(d.thread_state, [], [], null, false, false);
|
renderState(d.thread_state, [], [], null, false, false, lastEscalation ? lastEscalation.escalation_reason : null);
|
||||||
} else {
|
} else {
|
||||||
clearDebug();
|
clearDebug();
|
||||||
}
|
}
|
||||||
@@ -793,7 +795,7 @@ function appendMessage(role, text, iso, intentCode, intentName, meta) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* ---------- отладка ---------- */
|
/* ---------- отладка ---------- */
|
||||||
function renderState(state, bounces, validationEvents, parseError, routingLoopTriggered, resumedFromSuspended) {
|
function renderState(state, bounces, validationEvents, parseError, routingLoopTriggered, resumedFromSuspended, escalationReason) {
|
||||||
const box = $("debug-state");
|
const box = $("debug-state");
|
||||||
if (!state || !state.current_intent_code) {
|
if (!state || !state.current_intent_code) {
|
||||||
box.innerHTML = '<div class="mini">сценарий ещё не запущен</div>';
|
box.innerHTML = '<div class="mini">сценарий ещё не запущен</div>';
|
||||||
@@ -818,9 +820,23 @@ function renderState(state, bounces, validationEvents, parseError, routingLoopTr
|
|||||||
<span style="opacity:.75;">${esc(pendingGuard.description || "")}</span>
|
<span style="opacity:.75;">${esc(pendingGuard.description || "")}</span>
|
||||||
</div>`
|
</div>`
|
||||||
: "";
|
: "";
|
||||||
|
const REASON_LABELS = {
|
||||||
|
acute_pain: "острая боль / срочное состояние",
|
||||||
|
surgery: "операция / хирургия / стационар",
|
||||||
|
angry: "пациент раздражён",
|
||||||
|
explicit_request: "запросил оператора",
|
||||||
|
routing_loop: "автоматически (петля роутера)",
|
||||||
|
};
|
||||||
const loopHtml = routingLoopTriggered
|
const loopHtml = routingLoopTriggered
|
||||||
? `<div style="margin-top:8px;padding:6px 8px;border-radius:4px;background:#fee2e2;color:#7f1d1d;font-size:11px;">
|
? `<div style="margin-top:8px;padding:6px 8px;border-radius:4px;background:#fee2e2;color:#7f1d1d;font-size:11px;">
|
||||||
🛑 защита от петли сработала: диалог уведён в <code>escalate_human</code>.
|
🛑 защита от петли сработала: диалог уведён к оператору.
|
||||||
|
</div>`
|
||||||
|
: "";
|
||||||
|
const effectiveReason = escalationReason || (state.current_intent_code === "escalate_human" ? "explicit_request" : null);
|
||||||
|
const escalationHtml = effectiveReason
|
||||||
|
? `<div style="margin-top:8px;padding:6px 8px;border-radius:4px;background:#fee2e2;color:#7f1d1d;font-size:11px;">
|
||||||
|
🔴 <b>передача оператору</b> · причина: <code>${esc(effectiveReason)}</code>
|
||||||
|
<span style="opacity:.75;"> — ${esc(REASON_LABELS[effectiveReason] || effectiveReason)}</span>
|
||||||
</div>`
|
</div>`
|
||||||
: "";
|
: "";
|
||||||
const suspendedSlotsCount = Object.keys(state.resumable_slots || {}).length;
|
const suspendedSlotsCount = Object.keys(state.resumable_slots || {}).length;
|
||||||
@@ -866,7 +882,7 @@ function renderState(state, bounces, validationEvents, parseError, routingLoopTr
|
|||||||
<b>${esc(state.current_intent_code)}</b>
|
<b>${esc(state.current_intent_code)}</b>
|
||||||
<span style="color:var(--muted);font-size:11px;margin-left:4px;">— без пошагового сценария</span>
|
<span style="color:var(--muted);font-size:11px;margin-left:4px;">— без пошагового сценария</span>
|
||||||
</div>
|
</div>
|
||||||
${handoffHtml}${pendingGuardHtml}${softNudgeHtml}${loopHtml}${suspendedHtml}${resumedHtml}${bounceHtml}${validationHtml}${parseErrorHtml}
|
${handoffHtml}${escalationHtml}${pendingGuardHtml}${softNudgeHtml}${loopHtml}${suspendedHtml}${resumedHtml}${bounceHtml}${validationHtml}${parseErrorHtml}
|
||||||
</div>
|
</div>
|
||||||
`;
|
`;
|
||||||
return;
|
return;
|
||||||
@@ -877,12 +893,12 @@ function renderState(state, bounces, validationEvents, parseError, routingLoopTr
|
|||||||
<div style="font-size:12px;">
|
<div style="font-size:12px;">
|
||||||
<div><b>${esc(state.current_intent_code)}</b> · шаг <code>${esc(state.current_step_code)}</code></div>
|
<div><b>${esc(state.current_intent_code)}</b> · шаг <code>${esc(state.current_step_code)}</code></div>
|
||||||
<div class="prompt-box" style="margin-top:6px;max-height:200px;">${esc(slotsJson)}</div>
|
<div class="prompt-box" style="margin-top:6px;max-height:200px;">${esc(slotsJson)}</div>
|
||||||
${handoffHtml}${pendingGuardHtml}${softNudgeHtml}${loopHtml}${suspendedHtml}${resumedHtml}${bounceHtml}${validationHtml}${parseErrorHtml}
|
${handoffHtml}${escalationHtml}${pendingGuardHtml}${softNudgeHtml}${loopHtml}${suspendedHtml}${resumedHtml}${bounceHtml}${validationHtml}${parseErrorHtml}
|
||||||
</div>
|
</div>
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
function renderDebug(sources, prompt, intentCode, intentName, configVersion, routerVersion, routerIntentCode, bounces, stepCode) {
|
function renderDebug(sources, prompt, intentCode, intentName, configVersion, routerVersion, routerIntentCode, bounces, stepCode, operatorSummary) {
|
||||||
const routerVer = routerVersion != null ? `роутер v${routerVersion}` : "роутер";
|
const routerVer = routerVersion != null ? `роутер v${routerVersion}` : "роутер";
|
||||||
const hasBounces = bounces && bounces.length > 0;
|
const hasBounces = bounces && bounces.length > 0;
|
||||||
const routerDiffers = routerIntentCode && intentCode && routerIntentCode !== intentCode;
|
const routerDiffers = routerIntentCode && intentCode && routerIntentCode !== intentCode;
|
||||||
@@ -941,6 +957,24 @@ function renderDebug(sources, prompt, intentCode, intentName, configVersion, rou
|
|||||||
$("debug-prompt").innerHTML = prompt
|
$("debug-prompt").innerHTML = prompt
|
||||||
? `<div class="prompt-box">${esc(prompt)}</div>`
|
? `<div class="prompt-box">${esc(prompt)}</div>`
|
||||||
: '<div class="mini">промпт пуст</div>';
|
: '<div class="mini">промпт пуст</div>';
|
||||||
|
|
||||||
|
const summaryBox = $("debug-operator-summary");
|
||||||
|
if (summaryBox) {
|
||||||
|
if (operatorSummary) {
|
||||||
|
summaryBox.style.display = "";
|
||||||
|
summaryBox.innerHTML = `
|
||||||
|
<div style="font-size:11px;color:var(--muted);margin-bottom:4px;">саммари для оператора (предпросмотр)</div>
|
||||||
|
<div style="margin-bottom:4px;"><b>причина:</b> <code>${esc(operatorSummary.reason || "")}</code></div>
|
||||||
|
<div style="margin-bottom:4px;"><b>слоты:</b> <code>${esc(JSON.stringify(operatorSummary.slots || {}))}</code></div>
|
||||||
|
<div><b>история:</b>
|
||||||
|
${(operatorSummary.history_tail || []).map(h =>
|
||||||
|
`<div style="margin-top:3px;"><span style="color:var(--muted);">${esc(h.role === "user" ? "пациент" : "ассистент")}:</span> ${esc(h.text)}</div>`
|
||||||
|
).join("")}
|
||||||
|
</div>`;
|
||||||
|
} else {
|
||||||
|
summaryBox.style.display = "none";
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function clearDebug() {
|
function clearDebug() {
|
||||||
@@ -948,6 +982,7 @@ function clearDebug() {
|
|||||||
$("debug-router").innerHTML = '<div class="mini">— пока пусто —</div>';
|
$("debug-router").innerHTML = '<div class="mini">— пока пусто —</div>';
|
||||||
$("debug-chunks").innerHTML = '<div class="mini">— пока пусто —</div>';
|
$("debug-chunks").innerHTML = '<div class="mini">— пока пусто —</div>';
|
||||||
$("debug-prompt").innerHTML = '<div class="mini">— пока пусто —</div>';
|
$("debug-prompt").innerHTML = '<div class="mini">— пока пусто —</div>';
|
||||||
|
const s = $("debug-operator-summary"); if (s) s.style.display = "none";
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ---------- send message ---------- */
|
/* ---------- send message ---------- */
|
||||||
@@ -987,8 +1022,8 @@ async function sendMessage() {
|
|||||||
appendMessage("assistant", r.answer, null, r.intent_code, r.intent_name, r.message_meta);
|
appendMessage("assistant", r.answer, null, r.intent_code, r.intent_name, r.message_meta);
|
||||||
$("chat-title").className = "chat-title";
|
$("chat-title").className = "chat-title";
|
||||||
$("chat-title").textContent = r.thread_name;
|
$("chat-title").textContent = r.thread_name;
|
||||||
renderDebug(r.sources, r.assembled_prompt, r.intent_code, r.intent_name, r.config_version, r.router_version, r.router_intent_code, r.bounces, r.thread_state && r.thread_state.current_step_code);
|
renderDebug(r.sources, r.assembled_prompt, r.intent_code, r.intent_name, r.config_version, r.router_version, r.router_intent_code, r.bounces, r.thread_state && r.thread_state.current_step_code, r.operator_summary);
|
||||||
renderState(r.thread_state, r.bounces, r.validation_events, r.parse_error, r.routing_loop_triggered, r.resumed_from_suspended);
|
renderState(r.thread_state, r.bounces, r.validation_events, r.parse_error, r.routing_loop_triggered, r.resumed_from_suspended, r.escalation_reason);
|
||||||
refreshThreads();
|
refreshThreads();
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
// Откатываем визуально: убираем пузырь-заглушку ассистента и только что
|
// Откатываем визуально: убираем пузырь-заглушку ассистента и только что
|
||||||
|
|||||||
Reference in New Issue
Block a user