feat(sprint2): страница «Песочница» + навигация между страницами
Завершающий кусок Спринта 2 — UI для ведения диалогов. static/sandbox.html: - Трёхколоночная раскладка во всю высоту экрана. - Слева: список сохранённых диалогов (имя, дата последнего обновления, счётчик сообщений, превью первой реплики), кнопка «+ новый»; на каждой карточке — «переименовать» (prompt) и «удалить» (confirm). - Центр: чат в привычной стилистике (пузыри, user справа, assistant слева), Enter — отправить, Shift+Enter — перенос строки. Заголовок сверху показывает имя активного треда. - Справа: отладка ответа — найденные фрагменты со score в процентах + собранный промпт в моноширинном блоке на светлом фоне. - При отправке первой реплики тред создаётся автоматически, API возвращает thread_id и thread_name — дальше реплики уходят в тот же тред. static/index.html: в шапке добавлены ссылки «Отладка» / «Песочница», подсветка активной страницы; тот же стиль nav-ссылок продублирован в sandbox.html. routers/chat: detail сообщения ошибки теперь включает тип исключения (удобнее при диагностике), trace пишется через logger.exception. SPRINTS.md: Спринт 2 помечен как закрытый. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
+19
-1
@@ -62,6 +62,20 @@
|
||||
.dot.ok { background: var(--ok); }
|
||||
.dot.warn { background: var(--warn); }
|
||||
.dot.err { background: var(--err); }
|
||||
.nav {
|
||||
display: flex;
|
||||
gap: 4px;
|
||||
}
|
||||
.nav-link {
|
||||
text-decoration: none;
|
||||
color: var(--muted);
|
||||
padding: 6px 12px;
|
||||
border-radius: 6px;
|
||||
font-size: 13px;
|
||||
font-weight: 500;
|
||||
}
|
||||
.nav-link:hover { background: var(--chip-bg); color: var(--fg); }
|
||||
.nav-link.active { background: var(--accent); color: #fff; }
|
||||
.stats {
|
||||
margin-left: auto;
|
||||
font-size: 13px;
|
||||
@@ -324,7 +338,11 @@
|
||||
<body>
|
||||
|
||||
<header>
|
||||
<h1>Chat Agent for Patients — Debug</h1>
|
||||
<h1>Chat Agent for Patients</h1>
|
||||
<nav class="nav">
|
||||
<a href="/" class="nav-link active">Отладка</a>
|
||||
<a href="/sandbox.html" class="nav-link">Песочница</a>
|
||||
</nav>
|
||||
<span class="status"><span class="dot" id="dot"></span><span id="status-text">проверяю…</span></span>
|
||||
<span class="stats" id="stats"></span>
|
||||
</header>
|
||||
|
||||
@@ -0,0 +1,628 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="ru">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||
<title>Chat Agent for Patients — Песочница</title>
|
||||
<style>
|
||||
:root {
|
||||
--bg: #f5f6f8;
|
||||
--panel: #ffffff;
|
||||
--border: #e1e4ea;
|
||||
--muted: #6b7280;
|
||||
--fg: #111827;
|
||||
--accent: #2563eb;
|
||||
--accent-hover: #1d4ed8;
|
||||
--ok: #16a34a;
|
||||
--warn: #d97706;
|
||||
--err: #dc2626;
|
||||
--chip-bg: #eef2ff;
|
||||
--mono: ui-monospace, SFMono-Regular, Menlo, monospace;
|
||||
--user-bg: #dbeafe;
|
||||
--bot-bg: #f3f4f6;
|
||||
}
|
||||
* { box-sizing: border-box; }
|
||||
html, body { height: 100%; }
|
||||
body {
|
||||
margin: 0;
|
||||
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, sans-serif;
|
||||
background: var(--bg);
|
||||
color: var(--fg);
|
||||
font-size: 14px;
|
||||
line-height: 1.5;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
}
|
||||
header {
|
||||
background: var(--panel);
|
||||
border-bottom: 1px solid var(--border);
|
||||
padding: 14px 24px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 16px;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
header h1 { margin: 0; font-size: 16px; font-weight: 600; }
|
||||
.nav { display: flex; gap: 4px; }
|
||||
.nav-link {
|
||||
text-decoration: none;
|
||||
color: var(--muted);
|
||||
padding: 6px 12px;
|
||||
border-radius: 6px;
|
||||
font-size: 13px;
|
||||
font-weight: 500;
|
||||
}
|
||||
.nav-link:hover { background: var(--chip-bg); color: var(--fg); }
|
||||
.nav-link.active { background: var(--accent); color: #fff; }
|
||||
.status {
|
||||
margin-left: auto;
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
gap: 6px;
|
||||
padding: 4px 10px;
|
||||
border-radius: 999px;
|
||||
background: var(--chip-bg);
|
||||
font-size: 13px;
|
||||
}
|
||||
.dot { width: 8px; height: 8px; border-radius: 50%; background: var(--muted); }
|
||||
.dot.ok { background: var(--ok); }
|
||||
.dot.err { background: var(--err); }
|
||||
|
||||
main {
|
||||
flex: 1;
|
||||
display: grid;
|
||||
grid-template-columns: 280px 1fr 420px;
|
||||
gap: 0;
|
||||
min-height: 0;
|
||||
}
|
||||
.col-panel {
|
||||
background: var(--panel);
|
||||
border-right: 1px solid var(--border);
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
min-height: 0;
|
||||
}
|
||||
.col-panel:last-child { border-right: none; border-left: 1px solid var(--border); }
|
||||
.col-head {
|
||||
padding: 12px 16px;
|
||||
border-bottom: 1px solid var(--border);
|
||||
font-size: 12px;
|
||||
text-transform: uppercase;
|
||||
letter-spacing: 0.04em;
|
||||
color: var(--muted);
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
}
|
||||
.col-body {
|
||||
flex: 1;
|
||||
overflow-y: auto;
|
||||
min-height: 0;
|
||||
}
|
||||
|
||||
/* Список тредов */
|
||||
.threads-head-btn {
|
||||
margin-left: auto;
|
||||
background: var(--accent);
|
||||
color: #fff;
|
||||
border: none;
|
||||
padding: 4px 10px;
|
||||
border-radius: 6px;
|
||||
font-size: 12px;
|
||||
cursor: pointer;
|
||||
}
|
||||
.threads-head-btn:hover { background: var(--accent-hover); }
|
||||
.thread-item {
|
||||
padding: 12px 16px;
|
||||
border-bottom: 1px solid var(--border);
|
||||
cursor: pointer;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 2px;
|
||||
}
|
||||
.thread-item:hover { background: #f9fafb; }
|
||||
.thread-item.active { background: var(--chip-bg); }
|
||||
.thread-name {
|
||||
font-weight: 500;
|
||||
font-size: 13px;
|
||||
overflow: hidden;
|
||||
text-overflow: ellipsis;
|
||||
white-space: nowrap;
|
||||
}
|
||||
.thread-meta {
|
||||
font-size: 11px;
|
||||
color: var(--muted);
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
gap: 8px;
|
||||
}
|
||||
.thread-preview {
|
||||
font-size: 12px;
|
||||
color: var(--muted);
|
||||
overflow: hidden;
|
||||
text-overflow: ellipsis;
|
||||
white-space: nowrap;
|
||||
}
|
||||
.thread-actions {
|
||||
display: flex;
|
||||
gap: 6px;
|
||||
margin-top: 4px;
|
||||
opacity: 0;
|
||||
transition: opacity 0.1s;
|
||||
}
|
||||
.thread-item:hover .thread-actions,
|
||||
.thread-item.active .thread-actions { opacity: 1; }
|
||||
.thread-actions button {
|
||||
background: none;
|
||||
border: 1px solid var(--border);
|
||||
font-size: 11px;
|
||||
padding: 2px 8px;
|
||||
border-radius: 4px;
|
||||
cursor: pointer;
|
||||
color: var(--muted);
|
||||
}
|
||||
.thread-actions button:hover { background: var(--panel); color: var(--fg); }
|
||||
.thread-actions button.del:hover { border-color: var(--err); color: var(--err); }
|
||||
|
||||
/* Чат */
|
||||
.chat-head {
|
||||
padding: 12px 20px;
|
||||
border-bottom: 1px solid var(--border);
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 10px;
|
||||
background: var(--panel);
|
||||
}
|
||||
.chat-title {
|
||||
font-weight: 500;
|
||||
font-size: 14px;
|
||||
overflow: hidden;
|
||||
text-overflow: ellipsis;
|
||||
white-space: nowrap;
|
||||
}
|
||||
.chat-title.empty { color: var(--muted); font-weight: normal; }
|
||||
.chat-messages {
|
||||
flex: 1;
|
||||
overflow-y: auto;
|
||||
padding: 16px 20px;
|
||||
background: var(--bg);
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 10px;
|
||||
min-height: 0;
|
||||
}
|
||||
.msg {
|
||||
max-width: 75%;
|
||||
padding: 10px 14px;
|
||||
border-radius: 12px;
|
||||
word-wrap: break-word;
|
||||
white-space: pre-wrap;
|
||||
}
|
||||
.msg.user { background: var(--user-bg); align-self: flex-end; }
|
||||
.msg.assistant { background: var(--bot-bg); align-self: flex-start; }
|
||||
.msg-meta { font-size: 10px; color: var(--muted); margin-top: 4px; }
|
||||
.chat-empty {
|
||||
margin: auto;
|
||||
color: var(--muted);
|
||||
text-align: center;
|
||||
font-style: italic;
|
||||
padding: 40px 20px;
|
||||
}
|
||||
.chat-input {
|
||||
padding: 12px 20px;
|
||||
border-top: 1px solid var(--border);
|
||||
background: var(--panel);
|
||||
display: flex;
|
||||
gap: 10px;
|
||||
}
|
||||
.chat-input textarea {
|
||||
flex: 1;
|
||||
border: 1px solid var(--border);
|
||||
border-radius: 8px;
|
||||
padding: 10px 12px;
|
||||
font: inherit;
|
||||
font-size: 13px;
|
||||
resize: none;
|
||||
min-height: 44px;
|
||||
max-height: 160px;
|
||||
outline: none;
|
||||
}
|
||||
.chat-input textarea:focus { border-color: var(--accent); }
|
||||
.chat-input button {
|
||||
background: var(--accent);
|
||||
color: #fff;
|
||||
border: none;
|
||||
padding: 0 20px;
|
||||
border-radius: 8px;
|
||||
font-size: 13px;
|
||||
font-weight: 500;
|
||||
cursor: pointer;
|
||||
align-self: stretch;
|
||||
}
|
||||
.chat-input button:hover { background: var(--accent-hover); }
|
||||
.chat-input button:disabled { background: var(--muted); cursor: not-allowed; }
|
||||
|
||||
/* Правая панель — отладка */
|
||||
.debug-section { padding: 14px 16px; border-bottom: 1px solid var(--border); }
|
||||
.debug-section h3 {
|
||||
font-size: 12px;
|
||||
text-transform: uppercase;
|
||||
letter-spacing: 0.04em;
|
||||
color: var(--muted);
|
||||
margin: 0 0 10px 0;
|
||||
font-weight: 600;
|
||||
}
|
||||
.chunk-card {
|
||||
background: var(--panel);
|
||||
border: 1px solid var(--border);
|
||||
border-radius: 6px;
|
||||
padding: 8px 10px;
|
||||
margin-bottom: 8px;
|
||||
font-size: 12px;
|
||||
}
|
||||
.chunk-card-meta {
|
||||
font-size: 10px;
|
||||
color: var(--muted);
|
||||
display: flex;
|
||||
gap: 8px;
|
||||
flex-wrap: wrap;
|
||||
margin-bottom: 4px;
|
||||
}
|
||||
.chunk-score {
|
||||
background: var(--chip-bg);
|
||||
color: var(--accent);
|
||||
padding: 1px 6px;
|
||||
border-radius: 10px;
|
||||
font-weight: 600;
|
||||
}
|
||||
.chunk-text {
|
||||
white-space: pre-wrap;
|
||||
word-break: break-word;
|
||||
font-size: 11.5px;
|
||||
color: var(--fg);
|
||||
max-height: 100px;
|
||||
overflow-y: auto;
|
||||
}
|
||||
.prompt-box {
|
||||
background: var(--panel);
|
||||
color: var(--fg);
|
||||
border: 1px solid var(--border);
|
||||
padding: 10px 12px;
|
||||
border-radius: 6px;
|
||||
font-family: var(--mono);
|
||||
font-size: 11px;
|
||||
line-height: 1.5;
|
||||
white-space: pre-wrap;
|
||||
word-break: break-word;
|
||||
max-height: 400px;
|
||||
overflow-y: auto;
|
||||
}
|
||||
.mini { color: var(--muted); font-size: 12px; font-style: italic; }
|
||||
|
||||
.toast {
|
||||
position: fixed;
|
||||
bottom: 30px;
|
||||
left: 50%;
|
||||
transform: translateX(-50%);
|
||||
background: #111827;
|
||||
color: #fff;
|
||||
padding: 10px 16px;
|
||||
border-radius: 8px;
|
||||
font-size: 13px;
|
||||
opacity: 0;
|
||||
pointer-events: none;
|
||||
transition: opacity 0.2s;
|
||||
z-index: 100;
|
||||
}
|
||||
.toast.show { opacity: 1; }
|
||||
.toast.err { background: var(--err); }
|
||||
|
||||
.spinner {
|
||||
display: inline-block;
|
||||
width: 12px;
|
||||
height: 12px;
|
||||
border: 2px solid var(--border);
|
||||
border-top-color: var(--accent);
|
||||
border-radius: 50%;
|
||||
animation: spin 0.8s linear infinite;
|
||||
vertical-align: middle;
|
||||
}
|
||||
@keyframes spin { to { transform: rotate(360deg); } }
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
|
||||
<header>
|
||||
<h1>Chat Agent for Patients</h1>
|
||||
<nav class="nav">
|
||||
<a href="/" class="nav-link">Отладка</a>
|
||||
<a href="/sandbox.html" class="nav-link active">Песочница</a>
|
||||
</nav>
|
||||
<span class="status"><span class="dot" id="dot"></span><span id="status-text">проверяю…</span></span>
|
||||
</header>
|
||||
|
||||
<main>
|
||||
|
||||
<aside class="col-panel">
|
||||
<div class="col-head">
|
||||
Диалоги
|
||||
<button class="threads-head-btn" onclick="startNewThread()">+ новый</button>
|
||||
</div>
|
||||
<div class="col-body" id="threads-list">
|
||||
<div class="mini" style="padding:14px;">загружаю…</div>
|
||||
</div>
|
||||
</aside>
|
||||
|
||||
<section class="col-panel">
|
||||
<div class="chat-head">
|
||||
<div class="chat-title empty" id="chat-title">— выберите диалог слева или начните новый —</div>
|
||||
</div>
|
||||
<div class="chat-messages" id="chat-messages">
|
||||
<div class="chat-empty">Здесь появятся сообщения диалога.<br>Напишите что-нибудь снизу, чтобы начать.</div>
|
||||
</div>
|
||||
<form class="chat-input" id="chat-form">
|
||||
<textarea id="chat-text" placeholder="Напишите реплику пациента и нажмите Enter..." rows="1"></textarea>
|
||||
<button type="submit" id="chat-send">Отправить</button>
|
||||
</form>
|
||||
</section>
|
||||
|
||||
<aside class="col-panel">
|
||||
<div class="col-head">Отладка ответа</div>
|
||||
<div class="col-body">
|
||||
<div class="debug-section">
|
||||
<h3>Найденные фрагменты (по последней реплике)</h3>
|
||||
<div id="debug-chunks"><div class="mini">— пока пусто —</div></div>
|
||||
</div>
|
||||
<div class="debug-section">
|
||||
<h3>Собранный промпт</h3>
|
||||
<div id="debug-prompt"><div class="mini">— пока пусто —</div></div>
|
||||
</div>
|
||||
</div>
|
||||
</aside>
|
||||
|
||||
</main>
|
||||
|
||||
<div class="toast" id="toast"></div>
|
||||
|
||||
<script>
|
||||
const API = "";
|
||||
const $ = (id) => document.getElementById(id);
|
||||
const esc = (s) => String(s ?? "").replace(/[&<>"']/g, c => ({'&':'&','<':'<','>':'>','"':'"',"'":'''}[c]));
|
||||
|
||||
let activeThreadId = null;
|
||||
|
||||
function toast(msg, kind = "ok") {
|
||||
const t = $("toast");
|
||||
t.textContent = msg;
|
||||
t.className = "toast show" + (kind === "err" ? " err" : "");
|
||||
setTimeout(() => t.className = "toast", 2500);
|
||||
}
|
||||
|
||||
async function api(path, opts = {}) {
|
||||
const res = await fetch(API + path, opts);
|
||||
if (!res.ok) {
|
||||
let msg = `${res.status}`;
|
||||
try { const d = await res.json(); msg = d.detail || JSON.stringify(d); } catch (_) {}
|
||||
throw new Error(msg);
|
||||
}
|
||||
if (res.status === 204) return null;
|
||||
return res.json();
|
||||
}
|
||||
|
||||
/* ---------- health ---------- */
|
||||
async function refreshHealth() {
|
||||
try {
|
||||
const h = await api("/health");
|
||||
$("dot").className = "dot ok";
|
||||
$("status-text").textContent = `${h.chunks_count} чанков · ${h.documents_count} док.`;
|
||||
} catch (e) {
|
||||
$("dot").className = "dot err";
|
||||
$("status-text").textContent = "недоступен";
|
||||
}
|
||||
}
|
||||
|
||||
/* ---------- threads list ---------- */
|
||||
async function refreshThreads() {
|
||||
try {
|
||||
const d = await api("/threads");
|
||||
renderThreads(d.threads);
|
||||
} catch (e) {
|
||||
$("threads-list").innerHTML = `<div class="mini" style="padding:14px;color:var(--err)">${esc(e.message)}</div>`;
|
||||
}
|
||||
}
|
||||
|
||||
function renderThreads(threads) {
|
||||
if (!threads.length) {
|
||||
$("threads-list").innerHTML = '<div class="mini" style="padding:14px;">пока нет диалогов</div>';
|
||||
return;
|
||||
}
|
||||
$("threads-list").innerHTML = threads.map(t => `
|
||||
<div class="thread-item ${t.id === activeThreadId ? 'active' : ''}" onclick="openThread(${t.id})">
|
||||
<div class="thread-name" title="${esc(t.name)}">${esc(t.name)}</div>
|
||||
<div class="thread-meta">
|
||||
<span>${esc(fmtDate(t.updated_at))}</span>
|
||||
<span>${t.messages_count} сообщ.</span>
|
||||
</div>
|
||||
${t.first_message_preview ? `<div class="thread-preview">${esc(t.first_message_preview)}</div>` : ""}
|
||||
<div class="thread-actions">
|
||||
<button onclick="event.stopPropagation(); renameThread(${t.id}, ${JSON.stringify(t.name).replace(/"/g, '"')})">переименовать</button>
|
||||
<button class="del" onclick="event.stopPropagation(); deleteThread(${t.id}, ${JSON.stringify(t.name).replace(/"/g, '"')})">удалить</button>
|
||||
</div>
|
||||
</div>
|
||||
`).join("");
|
||||
}
|
||||
|
||||
function fmtDate(iso) {
|
||||
try {
|
||||
const d = new Date(iso);
|
||||
return d.toLocaleString("ru-RU", { day: "2-digit", month: "2-digit", hour: "2-digit", minute: "2-digit" });
|
||||
} catch (_) { return iso; }
|
||||
}
|
||||
|
||||
/* ---------- open / new thread ---------- */
|
||||
async function openThread(id) {
|
||||
activeThreadId = id;
|
||||
try {
|
||||
const d = await api(`/threads/${id}`);
|
||||
$("chat-title").className = "chat-title";
|
||||
$("chat-title").textContent = d.name;
|
||||
renderMessages(d.messages);
|
||||
const lastAssistant = [...d.messages].reverse().find(m => m.role === "assistant");
|
||||
if (lastAssistant) renderDebug(lastAssistant.sources, lastAssistant.assembled_prompt);
|
||||
else clearDebug();
|
||||
refreshThreads();
|
||||
} catch (e) {
|
||||
toast("Ошибка: " + e.message, "err");
|
||||
}
|
||||
}
|
||||
|
||||
function startNewThread() {
|
||||
activeThreadId = null;
|
||||
$("chat-title").className = "chat-title empty";
|
||||
$("chat-title").textContent = "— новый диалог (начните с первой реплики) —";
|
||||
$("chat-messages").innerHTML = '<div class="chat-empty">Это новый диалог.<br>Напишите реплику пациента снизу, чтобы начать.</div>';
|
||||
clearDebug();
|
||||
refreshThreads();
|
||||
}
|
||||
|
||||
function renderMessages(messages) {
|
||||
const box = $("chat-messages");
|
||||
if (!messages.length) {
|
||||
box.innerHTML = '<div class="chat-empty">Пусто. Напишите первую реплику.</div>';
|
||||
return;
|
||||
}
|
||||
box.innerHTML = messages.map(m => `
|
||||
<div class="msg ${m.role === "user" ? "user" : "assistant"}">
|
||||
${esc(m.text)}
|
||||
<div class="msg-meta">${esc(fmtDate(m.created_at))}</div>
|
||||
</div>
|
||||
`).join("");
|
||||
box.scrollTop = box.scrollHeight;
|
||||
}
|
||||
|
||||
function appendMessage(role, text, iso) {
|
||||
const box = $("chat-messages");
|
||||
const empty = box.querySelector(".chat-empty");
|
||||
if (empty) empty.remove();
|
||||
const div = document.createElement("div");
|
||||
div.className = "msg " + (role === "user" ? "user" : "assistant");
|
||||
div.innerHTML = esc(text) + `<div class="msg-meta">${esc(fmtDate(iso || new Date().toISOString()))}</div>`;
|
||||
box.appendChild(div);
|
||||
box.scrollTop = box.scrollHeight;
|
||||
return div;
|
||||
}
|
||||
|
||||
/* ---------- отладка ---------- */
|
||||
function renderDebug(sources, prompt) {
|
||||
if (sources && sources.length) {
|
||||
$("debug-chunks").innerHTML = sources.map(s => `
|
||||
<div class="chunk-card">
|
||||
<div class="chunk-card-meta">
|
||||
<span class="chunk-score">${(s.relevance_score * 100).toFixed(1)}%</span>
|
||||
<span>${esc(s.document_name || "—")}</span>
|
||||
${s.section ? `<span>${esc(s.section)}</span>` : ""}
|
||||
</div>
|
||||
<div class="chunk-text">${esc(s.chunk_text)}</div>
|
||||
</div>
|
||||
`).join("");
|
||||
} else {
|
||||
$("debug-chunks").innerHTML = '<div class="mini">источников нет</div>';
|
||||
}
|
||||
$("debug-prompt").innerHTML = prompt
|
||||
? `<div class="prompt-box">${esc(prompt)}</div>`
|
||||
: '<div class="mini">промпт пуст</div>';
|
||||
}
|
||||
|
||||
function clearDebug() {
|
||||
$("debug-chunks").innerHTML = '<div class="mini">— пока пусто —</div>';
|
||||
$("debug-prompt").innerHTML = '<div class="mini">— пока пусто —</div>';
|
||||
}
|
||||
|
||||
/* ---------- send message ---------- */
|
||||
$("chat-form").addEventListener("submit", async (e) => {
|
||||
e.preventDefault();
|
||||
sendMessage();
|
||||
});
|
||||
|
||||
$("chat-text").addEventListener("keydown", (e) => {
|
||||
if (e.key === "Enter" && !e.shiftKey) {
|
||||
e.preventDefault();
|
||||
sendMessage();
|
||||
}
|
||||
});
|
||||
|
||||
async function sendMessage() {
|
||||
const txt = $("chat-text").value.trim();
|
||||
if (!txt) return;
|
||||
$("chat-text").value = "";
|
||||
$("chat-send").disabled = true;
|
||||
$("chat-send").innerHTML = '<span class="spinner"></span>';
|
||||
|
||||
appendMessage("user", txt);
|
||||
const pending = appendMessage("assistant", "…");
|
||||
pending.style.opacity = "0.6";
|
||||
|
||||
try {
|
||||
const body = { text: txt, top_k: 5 };
|
||||
if (activeThreadId) body.thread_id = activeThreadId;
|
||||
const r = await api("/chat", {
|
||||
method: "POST",
|
||||
headers: { "Content-Type": "application/json" },
|
||||
body: JSON.stringify(body),
|
||||
});
|
||||
activeThreadId = r.thread_id;
|
||||
pending.remove();
|
||||
appendMessage("assistant", r.answer);
|
||||
$("chat-title").className = "chat-title";
|
||||
$("chat-title").textContent = r.thread_name;
|
||||
renderDebug(r.sources, r.assembled_prompt);
|
||||
refreshThreads();
|
||||
} catch (e) {
|
||||
pending.remove();
|
||||
toast("Ошибка: " + e.message, "err");
|
||||
} finally {
|
||||
$("chat-send").disabled = false;
|
||||
$("chat-send").textContent = "Отправить";
|
||||
}
|
||||
}
|
||||
|
||||
/* ---------- rename / delete ---------- */
|
||||
async function renameThread(id, currentName) {
|
||||
const newName = prompt("Новое имя диалога:", currentName);
|
||||
if (!newName || newName.trim() === currentName) return;
|
||||
try {
|
||||
await api(`/threads/${id}`, {
|
||||
method: "PATCH",
|
||||
headers: { "Content-Type": "application/json" },
|
||||
body: JSON.stringify({ name: newName.trim() }),
|
||||
});
|
||||
toast("Переименован");
|
||||
if (id === activeThreadId) {
|
||||
$("chat-title").textContent = newName.trim();
|
||||
}
|
||||
refreshThreads();
|
||||
} catch (e) {
|
||||
toast("Ошибка: " + e.message, "err");
|
||||
}
|
||||
}
|
||||
|
||||
async function deleteThread(id, name) {
|
||||
if (!confirm(`Удалить диалог «${name}» со всеми сообщениями?`)) return;
|
||||
try {
|
||||
const r = await api(`/threads/${id}`, { method: "DELETE" });
|
||||
toast(`Удалён (${r.deleted_messages} сообщ.)`);
|
||||
if (id === activeThreadId) startNewThread();
|
||||
else refreshThreads();
|
||||
} catch (e) {
|
||||
toast("Ошибка: " + e.message, "err");
|
||||
}
|
||||
}
|
||||
|
||||
/* ---------- init ---------- */
|
||||
refreshHealth();
|
||||
refreshThreads();
|
||||
setInterval(refreshHealth, 15000);
|
||||
</script>
|
||||
|
||||
</body>
|
||||
</html>
|
||||
Reference in New Issue
Block a user