feat(sprint4): фундамент графа — intents + роутер + переключение веток
Первый шаг графовой архитектуры из GRAPH_ARCHITECTURE.md. Заменили
«один активный промпт на всё» на «свой промпт на каждую ветку +
роутер выбирает ветку на каждой реплике».
Данные:
- Новая таблица intents (code, name, description, is_enabled,
order_index). Коды с префиксом `_` — системные (не responder).
- В agent_configs добавлен intent_id (nullable, FK SET NULL); убрана
глобальная уникальность version, вместо неё UniqueConstraint
(intent_id, version) — у каждой ветки свой счётчик версий.
- В messages добавлен intent_id (nullable, FK) — фиксируем, какую
ветку выбрал роутер для каждой реплики.
- Миграция cd0a88ef9080 в batch-режиме (SQLite не умеет ALTER для
constraints напрямую).
Сид:
- Стартовые 7 веток: new_booking, reschedule, price_question,
medical_question, general_info, escalate_human + `_router` как
системная ветка для промпта классификатора.
- Для каждой ветки — свой v1-промпт из prompts/intents/{code}.md.
- migrate_legacy_config_to_general_info: старый v1 из Спринта 3
(без intent_id) переносится на general_info с сохранением версии.
- ensure_seed_intents досиживает недостающие коды, существующие не
трогает — безопасно при добавлении новых веток.
Оркестрация и роутер:
- services/router_client.RouterClient — отдельный класс от LLMClient
(под будущую смену модели на более дешёвую). Метод classify(session,
history, text) возвращает {code, version}. Промпт классификатора
подтягивается из активного конфига ветки `_router`, fallback —
prompts/intents/_router.md. При сомнении/ошибке возвращает
general_info.
- services/chat_service.send_message теперь идёт через router.classify
→ берёт активный конфиг выбранной ветки → llm.chat. В сообщения
пишется intent_id, в треде фиксируется начальный agent_config_id.
В ответе — intent_code, intent_name, config_version, router_version.
API:
- GET /intents, GET /intents/{code}, PATCH /intents/{code} —
список веток со счётчиком версий, получение и переключение
is_enabled.
- /configs теперь требует intent_code как Query-параметр
(GET /configs, GET /configs/active) — выборка версий в рамках
ветки. POST /configs принимает intent_id.
- get_thread_detail JOIN-ит Intent — каждая реплика возвращает
intent_code + intent_name.
UI:
- settings.html переработан в 3-колоночный макет: слева список веток
с подгруппой «Системные» для `_router` (пометка «система» вместо
свитча), в центре редактор промпта/правил активной версии выбранной
ветки, справа список версий с активировать/удалить/загрузить.
Каждая ветка редактируется независимо — своя история версий,
своя активная.
- sandbox.html: у каждой реплики бейдж с intent_code, в отладке новый
блок «Решение роутера» (подсвеченный зелёным) с названием ветки,
версией её активного конфига и версией промпта роутера. Старый
«активная: v1» индикатор убран — он больше не имеет смысла (активная
у каждой ветки своя).
E2E проверено: разные реплики уходят в корректные ветки, каждая
отвечает по своему узкому промпту, промпт роутера редактируется в UI
как v2/v3 и откатывается — классификация сразу использует новую
версию.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
+35
-27
@@ -214,6 +214,17 @@
|
||||
white-space: pre-wrap;
|
||||
}
|
||||
.msg.assistant { background: var(--bot-bg); align-self: flex-start; }
|
||||
.msg-intent {
|
||||
display: inline-block;
|
||||
background: var(--chip-bg);
|
||||
color: var(--accent);
|
||||
padding: 1px 7px;
|
||||
border-radius: 10px;
|
||||
font-size: 10px;
|
||||
font-weight: 600;
|
||||
font-family: var(--mono);
|
||||
margin-right: 6px;
|
||||
}
|
||||
.msg.assistant p { margin: 0 0 8px 0; }
|
||||
.msg.assistant p:last-child { margin-bottom: 0; }
|
||||
.msg.assistant ul, .msg.assistant ol { margin: 6px 0; padding-left: 22px; }
|
||||
@@ -388,8 +399,7 @@
|
||||
<a href="/sandbox.html" class="nav-link active">Песочница</a>
|
||||
<a href="/settings.html" class="nav-link">Настройки</a>
|
||||
</nav>
|
||||
<span class="active-config" id="active-config" title="Нажмите, чтобы перейти в Настройки" onclick="location.href='/settings.html'"></span>
|
||||
<span class="status"><span class="dot" id="dot"></span><span id="status-text">проверяю…</span></span>
|
||||
<span class="status" style="margin-left:auto;"><span class="dot" id="dot"></span><span id="status-text">проверяю…</span></span>
|
||||
</header>
|
||||
|
||||
<main>
|
||||
@@ -420,6 +430,10 @@
|
||||
<aside class="col-panel">
|
||||
<div class="col-head">Отладка ответа</div>
|
||||
<div class="col-body">
|
||||
<div class="debug-section">
|
||||
<h3>Решение роутера</h3>
|
||||
<div id="debug-router"><div class="mini">— пока пусто —</div></div>
|
||||
</div>
|
||||
<div class="debug-section">
|
||||
<h3>Найденные фрагменты (по последней реплике)</h3>
|
||||
<div id="debug-chunks"><div class="mini">— пока пусто —</div></div>
|
||||
@@ -470,22 +484,6 @@ async function api(path, opts = {}) {
|
||||
return res.json();
|
||||
}
|
||||
|
||||
/* ---------- active config ---------- */
|
||||
async function refreshActiveConfig() {
|
||||
try {
|
||||
const res = await fetch("/configs/active");
|
||||
if (!res.ok) {
|
||||
$("active-config").textContent = "";
|
||||
return;
|
||||
}
|
||||
const c = await res.json();
|
||||
const label = `активная: v${c.version}${c.name ? " · " + c.name : ""}`;
|
||||
$("active-config").textContent = label;
|
||||
} catch (_) {
|
||||
$("active-config").textContent = "";
|
||||
}
|
||||
}
|
||||
|
||||
/* ---------- health ---------- */
|
||||
async function refreshHealth() {
|
||||
try {
|
||||
@@ -545,7 +543,7 @@ async function openThread(id) {
|
||||
$("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);
|
||||
if (lastAssistant) renderDebug(lastAssistant.sources, lastAssistant.assembled_prompt, lastAssistant.intent_code, lastAssistant.intent_name, null, null);
|
||||
else clearDebug();
|
||||
refreshThreads();
|
||||
} catch (e) {
|
||||
@@ -571,17 +569,18 @@ function renderMessages(messages) {
|
||||
box.innerHTML = messages.map(m => {
|
||||
const isUser = m.role === "user";
|
||||
const body = isUser ? esc(m.text) : renderMd(m.text);
|
||||
const intentBadge = m.intent_code ? `<span class="msg-intent" title="${esc(m.intent_name || m.intent_code)}">${esc(m.intent_code)}</span>` : "";
|
||||
return `
|
||||
<div class="msg ${isUser ? "user" : "assistant"}">
|
||||
<div class="msg-body">${body}</div>
|
||||
<div class="msg-meta">${esc(fmtDate(m.created_at))}</div>
|
||||
<div class="msg-meta">${intentBadge}${esc(fmtDate(m.created_at))}</div>
|
||||
</div>
|
||||
`;
|
||||
}).join("");
|
||||
box.scrollTop = box.scrollHeight;
|
||||
}
|
||||
|
||||
function appendMessage(role, text, iso) {
|
||||
function appendMessage(role, text, iso, intentCode, intentName) {
|
||||
const box = $("chat-messages");
|
||||
const empty = box.querySelector(".chat-empty");
|
||||
if (empty) empty.remove();
|
||||
@@ -589,14 +588,24 @@ function appendMessage(role, text, iso) {
|
||||
const isUser = role === "user";
|
||||
div.className = "msg " + (isUser ? "user" : "assistant");
|
||||
const body = isUser ? esc(text) : renderMd(text);
|
||||
div.innerHTML = `<div class="msg-body">${body}</div><div class="msg-meta">${esc(fmtDate(iso || new Date().toISOString()))}</div>`;
|
||||
const intentBadge = intentCode ? `<span class="msg-intent" title="${esc(intentName || intentCode)}">${esc(intentCode)}</span>` : "";
|
||||
div.innerHTML = `<div class="msg-body">${body}</div><div class="msg-meta">${intentBadge}${esc(fmtDate(iso || new Date().toISOString()))}</div>`;
|
||||
box.appendChild(div);
|
||||
box.scrollTop = box.scrollHeight;
|
||||
return div;
|
||||
}
|
||||
|
||||
/* ---------- отладка ---------- */
|
||||
function renderDebug(sources, prompt) {
|
||||
function renderDebug(sources, prompt, intentCode, intentName, configVersion, routerVersion) {
|
||||
const routerTag = routerVersion != null ? ` · роутер v${routerVersion}` : "";
|
||||
const routerLine = intentCode
|
||||
? `<div style="padding:10px 16px;background:#ecfdf5;font-size:12px;">
|
||||
<div><b>${esc(intentCode)}</b> — ${esc(intentName || '')}${configVersion ? ' · ветка v' + configVersion : ''}</div>
|
||||
${routerVersion != null ? `<div style="color:var(--muted);font-size:11px;margin-top:2px;">классифицировано роутером${routerTag.replace(' · роутер', '')}</div>` : ''}
|
||||
</div>`
|
||||
: "";
|
||||
$("debug-router").innerHTML = routerLine || '<div class="mini">— маршрутизация пока не выполнена —</div>';
|
||||
|
||||
if (sources && sources.length) {
|
||||
$("debug-chunks").innerHTML = sources.map(s => `
|
||||
<div class="chunk-card">
|
||||
@@ -617,6 +626,7 @@ function renderDebug(sources, prompt) {
|
||||
}
|
||||
|
||||
function clearDebug() {
|
||||
$("debug-router").innerHTML = '<div class="mini">— пока пусто —</div>';
|
||||
$("debug-chunks").innerHTML = '<div class="mini">— пока пусто —</div>';
|
||||
$("debug-prompt").innerHTML = '<div class="mini">— пока пусто —</div>';
|
||||
}
|
||||
@@ -655,10 +665,10 @@ async function sendMessage() {
|
||||
});
|
||||
activeThreadId = r.thread_id;
|
||||
pending.remove();
|
||||
appendMessage("assistant", r.answer);
|
||||
appendMessage("assistant", r.answer, null, r.intent_code, r.intent_name);
|
||||
$("chat-title").className = "chat-title";
|
||||
$("chat-title").textContent = r.thread_name;
|
||||
renderDebug(r.sources, r.assembled_prompt);
|
||||
renderDebug(r.sources, r.assembled_prompt, r.intent_code, r.intent_name, r.config_version, r.router_version);
|
||||
refreshThreads();
|
||||
} catch (e) {
|
||||
pending.remove();
|
||||
@@ -703,10 +713,8 @@ async function deleteThread(id, name) {
|
||||
|
||||
/* ---------- init ---------- */
|
||||
refreshHealth();
|
||||
refreshActiveConfig();
|
||||
refreshThreads();
|
||||
setInterval(refreshHealth, 15000);
|
||||
setInterval(refreshActiveConfig, 15000);
|
||||
</script>
|
||||
|
||||
</body>
|
||||
|
||||
+356
-152
@@ -20,6 +20,7 @@
|
||||
--mono: ui-monospace, SFMono-Regular, Menlo, monospace;
|
||||
}
|
||||
* { box-sizing: border-box; }
|
||||
html, body { height: 100%; }
|
||||
body {
|
||||
margin: 0;
|
||||
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, sans-serif;
|
||||
@@ -27,6 +28,8 @@
|
||||
color: var(--fg);
|
||||
font-size: 14px;
|
||||
line-height: 1.5;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
}
|
||||
header {
|
||||
background: var(--panel);
|
||||
@@ -35,9 +38,7 @@
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 16px;
|
||||
position: sticky;
|
||||
top: 0;
|
||||
z-index: 10;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
header h1 { margin: 0; font-size: 16px; font-weight: 600; }
|
||||
.nav { display: flex; gap: 4px; }
|
||||
@@ -56,32 +57,127 @@
|
||||
font-size: 13px;
|
||||
color: var(--muted);
|
||||
}
|
||||
.stats b { color: var(--fg); }
|
||||
|
||||
main {
|
||||
padding: 24px;
|
||||
max-width: 1400px;
|
||||
margin: 0 auto;
|
||||
flex: 1;
|
||||
display: grid;
|
||||
grid-template-columns: 1fr 420px;
|
||||
gap: 20px;
|
||||
grid-template-columns: 280px 1fr 380px;
|
||||
gap: 0;
|
||||
min-height: 0;
|
||||
}
|
||||
.panel {
|
||||
.col-panel {
|
||||
background: var(--panel);
|
||||
border: 1px solid var(--border);
|
||||
border-radius: 10px;
|
||||
padding: 18px 20px;
|
||||
border-right: 1px solid var(--border);
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
min-height: 0;
|
||||
}
|
||||
.panel h2 {
|
||||
margin: 0 0 14px 0;
|
||||
font-size: 15px;
|
||||
.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);
|
||||
font-weight: 600;
|
||||
}
|
||||
.panel .sub {
|
||||
.col-body {
|
||||
flex: 1;
|
||||
overflow-y: auto;
|
||||
min-height: 0;
|
||||
}
|
||||
|
||||
/* Список веток */
|
||||
.section-header {
|
||||
padding: 10px 16px 6px;
|
||||
font-size: 10px;
|
||||
text-transform: uppercase;
|
||||
letter-spacing: 0.05em;
|
||||
color: var(--muted);
|
||||
font-weight: 600;
|
||||
background: #fafbfd;
|
||||
border-bottom: 1px solid var(--border);
|
||||
}
|
||||
.system-badge {
|
||||
background: #fef3c7;
|
||||
color: #92400e;
|
||||
padding: 1px 6px;
|
||||
border-radius: 10px;
|
||||
font-size: 9px;
|
||||
font-weight: 600;
|
||||
text-transform: uppercase;
|
||||
}
|
||||
.intent-item {
|
||||
padding: 12px 16px;
|
||||
border-bottom: 1px solid var(--border);
|
||||
cursor: pointer;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 2px;
|
||||
}
|
||||
.intent-item:hover { background: #f9fafb; }
|
||||
.intent-item.active { background: var(--chip-bg); }
|
||||
.intent-item.disabled .intent-name { color: var(--muted); text-decoration: line-through; }
|
||||
.intent-top {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
}
|
||||
.intent-name { font-weight: 500; font-size: 13px; flex: 1; }
|
||||
.intent-code { font-family: var(--mono); font-size: 10px; color: var(--muted); }
|
||||
.intent-desc {
|
||||
font-size: 11px;
|
||||
color: var(--muted);
|
||||
margin-top: 2px;
|
||||
}
|
||||
.intent-version {
|
||||
font-size: 11px;
|
||||
color: var(--accent);
|
||||
font-weight: 500;
|
||||
margin-top: 2px;
|
||||
}
|
||||
.intent-version.empty { color: var(--muted); font-weight: normal; }
|
||||
|
||||
/* Редактор */
|
||||
.editor {
|
||||
padding: 20px 24px;
|
||||
max-width: 900px;
|
||||
}
|
||||
.editor-empty {
|
||||
margin: auto;
|
||||
text-align: center;
|
||||
color: var(--muted);
|
||||
padding: 60px 20px;
|
||||
font-style: italic;
|
||||
}
|
||||
.editor h2 {
|
||||
margin: 0 0 4px 0;
|
||||
font-size: 18px;
|
||||
font-weight: 600;
|
||||
}
|
||||
.editor .sub {
|
||||
font-size: 12px;
|
||||
color: var(--muted);
|
||||
margin: -10px 0 12px 0;
|
||||
margin-bottom: 18px;
|
||||
}
|
||||
.toolbar {
|
||||
display: flex;
|
||||
gap: 10px;
|
||||
margin-bottom: 14px;
|
||||
flex-wrap: wrap;
|
||||
}
|
||||
.toolbar button {
|
||||
background: none;
|
||||
border: 1px solid var(--border);
|
||||
padding: 5px 12px;
|
||||
border-radius: 6px;
|
||||
font-size: 12px;
|
||||
cursor: pointer;
|
||||
color: var(--fg);
|
||||
}
|
||||
.toolbar button:hover { background: #f9fafb; }
|
||||
.toolbar .toggle { margin-left: auto; display: inline-flex; align-items: center; gap: 6px; font-size: 12px; color: var(--muted); cursor: pointer; }
|
||||
|
||||
.field { margin-bottom: 14px; }
|
||||
.field label {
|
||||
@@ -91,8 +187,7 @@
|
||||
color: var(--muted);
|
||||
margin-bottom: 4px;
|
||||
}
|
||||
.field input[type=text],
|
||||
.field textarea {
|
||||
.field input[type=text], .field textarea {
|
||||
width: 100%;
|
||||
border: 1px solid var(--border);
|
||||
border-radius: 6px;
|
||||
@@ -102,21 +197,24 @@
|
||||
background: #fff;
|
||||
outline: none;
|
||||
}
|
||||
.field input[type=text]:focus,
|
||||
.field textarea:focus { border-color: var(--accent); }
|
||||
.field input[type=text]:focus, .field textarea:focus { border-color: var(--accent); }
|
||||
.field textarea {
|
||||
font-family: var(--mono);
|
||||
resize: vertical;
|
||||
min-height: 160px;
|
||||
min-height: 200px;
|
||||
}
|
||||
.field textarea.prompt { min-height: 260px; }
|
||||
.field textarea.rules { min-height: 160px; }
|
||||
.field textarea.prompt { min-height: 300px; }
|
||||
.field textarea.rules { min-height: 140px; }
|
||||
|
||||
.editor-actions {
|
||||
display: flex;
|
||||
gap: 10px;
|
||||
align-items: center;
|
||||
flex-wrap: wrap;
|
||||
padding-top: 6px;
|
||||
border-top: 1px solid var(--border);
|
||||
margin-top: 10px;
|
||||
padding: 14px 0 0 0;
|
||||
}
|
||||
.editor-actions button {
|
||||
background: var(--accent);
|
||||
@@ -129,11 +227,7 @@
|
||||
cursor: pointer;
|
||||
}
|
||||
.editor-actions button:hover { background: var(--accent-hover); }
|
||||
.editor-actions button.secondary {
|
||||
background: none;
|
||||
color: var(--fg);
|
||||
border: 1px solid var(--border);
|
||||
}
|
||||
.editor-actions button.secondary { background: none; color: var(--fg); border: 1px solid var(--border); }
|
||||
.editor-actions button.secondary:hover { background: #f9fafb; }
|
||||
.editor-actions label {
|
||||
display: inline-flex;
|
||||
@@ -144,13 +238,13 @@
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
/* Список версий */
|
||||
.versions { max-height: 70vh; overflow-y: auto; }
|
||||
/* Версии */
|
||||
.versions { padding: 10px; }
|
||||
.version-card {
|
||||
border: 1px solid var(--border);
|
||||
border-radius: 8px;
|
||||
padding: 12px 14px;
|
||||
margin-bottom: 10px;
|
||||
padding: 10px 12px;
|
||||
margin-bottom: 8px;
|
||||
background: #fafbfd;
|
||||
}
|
||||
.version-card.active {
|
||||
@@ -158,54 +252,34 @@
|
||||
border-color: var(--accent);
|
||||
box-shadow: 0 0 0 1px var(--accent);
|
||||
}
|
||||
.version-head {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
margin-bottom: 4px;
|
||||
}
|
||||
.version-head { display: flex; align-items: center; gap: 6px; margin-bottom: 4px; }
|
||||
.v-num {
|
||||
background: var(--chip-bg);
|
||||
color: var(--accent);
|
||||
padding: 2px 8px;
|
||||
border-radius: 10px;
|
||||
font-size: 11px;
|
||||
font-weight: 600;
|
||||
}
|
||||
.v-name {
|
||||
font-weight: 500;
|
||||
font-size: 13px;
|
||||
overflow: hidden;
|
||||
text-overflow: ellipsis;
|
||||
white-space: nowrap;
|
||||
}
|
||||
.v-name.empty { color: var(--muted); font-style: italic; font-weight: normal; }
|
||||
.v-active-badge {
|
||||
margin-left: auto;
|
||||
background: var(--ok);
|
||||
color: #fff;
|
||||
padding: 2px 8px;
|
||||
padding: 1px 7px;
|
||||
border-radius: 10px;
|
||||
font-size: 10px;
|
||||
font-weight: 600;
|
||||
}
|
||||
.v-name { font-weight: 500; font-size: 12px; flex: 1; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }
|
||||
.v-name.empty { color: var(--muted); font-style: italic; font-weight: normal; }
|
||||
.v-active-badge {
|
||||
background: var(--ok);
|
||||
color: #fff;
|
||||
padding: 1px 6px;
|
||||
border-radius: 10px;
|
||||
font-size: 9px;
|
||||
font-weight: 600;
|
||||
text-transform: uppercase;
|
||||
}
|
||||
.v-meta {
|
||||
font-size: 11px;
|
||||
color: var(--muted);
|
||||
margin-bottom: 8px;
|
||||
}
|
||||
.v-actions {
|
||||
display: flex;
|
||||
gap: 6px;
|
||||
flex-wrap: wrap;
|
||||
}
|
||||
.v-meta { font-size: 10px; color: var(--muted); margin-bottom: 6px; }
|
||||
.v-actions { display: flex; gap: 4px; flex-wrap: wrap; }
|
||||
.v-actions button {
|
||||
background: none;
|
||||
border: 1px solid var(--border);
|
||||
padding: 3px 10px;
|
||||
padding: 2px 8px;
|
||||
border-radius: 4px;
|
||||
font-size: 11px;
|
||||
font-size: 10px;
|
||||
cursor: pointer;
|
||||
color: var(--fg);
|
||||
}
|
||||
@@ -215,7 +289,7 @@
|
||||
.v-actions button.del:hover { border-color: var(--err); color: var(--err); }
|
||||
.v-actions button:disabled { opacity: 0.4; cursor: not-allowed; }
|
||||
|
||||
.mini { color: var(--muted); font-size: 12px; font-style: italic; }
|
||||
.mini { color: var(--muted); font-size: 12px; font-style: italic; padding: 14px; }
|
||||
|
||||
.toast {
|
||||
position: fixed;
|
||||
@@ -234,6 +308,34 @@
|
||||
}
|
||||
.toast.show { opacity: 1; }
|
||||
.toast.err { background: var(--err); }
|
||||
|
||||
/* Свитч включён/выключен */
|
||||
.switch {
|
||||
position: relative;
|
||||
display: inline-block;
|
||||
width: 34px;
|
||||
height: 18px;
|
||||
}
|
||||
.switch input { opacity: 0; width: 0; height: 0; }
|
||||
.slider {
|
||||
position: absolute;
|
||||
cursor: pointer;
|
||||
inset: 0;
|
||||
background: #cbd5e1;
|
||||
border-radius: 10px;
|
||||
transition: 0.2s;
|
||||
}
|
||||
.slider:before {
|
||||
content: "";
|
||||
position: absolute;
|
||||
height: 14px; width: 14px;
|
||||
left: 2px; bottom: 2px;
|
||||
background: white;
|
||||
border-radius: 50%;
|
||||
transition: 0.2s;
|
||||
}
|
||||
.switch input:checked + .slider { background: var(--ok); }
|
||||
.switch input:checked + .slider:before { transform: translateX(16px); }
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
@@ -250,37 +352,26 @@
|
||||
|
||||
<main>
|
||||
|
||||
<section class="panel">
|
||||
<h2>Редактор конфигурации агента</h2>
|
||||
<div class="sub">При сохранении всегда создаётся новая версия — существующие не меняются. Это даёт честный откат.</div>
|
||||
|
||||
<div class="field">
|
||||
<label for="f-name">Имя версии (необязательно)</label>
|
||||
<input type="text" id="f-name" placeholder="например: строгий тон, v2 после фидбэка операторов" maxlength="200">
|
||||
<aside class="col-panel">
|
||||
<div class="col-head">Ветки (intents)</div>
|
||||
<div class="col-body" id="intents-list">
|
||||
<div class="mini">загружаю…</div>
|
||||
</div>
|
||||
</aside>
|
||||
|
||||
<div class="field">
|
||||
<label for="f-prompt">Системный промпт</label>
|
||||
<textarea id="f-prompt" class="prompt" spellcheck="false"></textarea>
|
||||
</div>
|
||||
|
||||
<div class="field">
|
||||
<label for="f-rules">Правила (в дополнение к промпту; свободная markdown-форма)</label>
|
||||
<textarea id="f-rules" class="rules" spellcheck="false" placeholder="Например: - Если пациент спрашивает про цены — не называй конкретных сумм, переведи на оператора. - Если злится — сначала извинись, подтверди, что сейчас поможешь."></textarea>
|
||||
</div>
|
||||
|
||||
<div class="editor-actions">
|
||||
<button id="btn-save" onclick="saveVersion(false)">Сохранить как новую версию</button>
|
||||
<button class="secondary" onclick="loadActiveIntoEditor()">Загрузить активную в редактор</button>
|
||||
<label><input type="checkbox" id="chk-activate"> Сразу сделать активной</label>
|
||||
<section class="col-panel">
|
||||
<div class="col-head" id="editor-head">Выберите ветку слева</div>
|
||||
<div class="col-body">
|
||||
<div class="editor" id="editor">
|
||||
<div class="editor-empty">Слева — список веток. Выберите, чтобы увидеть и отредактировать её активный промпт.</div>
|
||||
</div>
|
||||
</div>
|
||||
</section>
|
||||
|
||||
<aside class="panel">
|
||||
<h2>Версии</h2>
|
||||
<div class="sub">Активная версия используется в «Песочнице» на каждый запрос.</div>
|
||||
<div class="versions" id="versions">
|
||||
<div class="mini">загружаю…</div>
|
||||
<aside class="col-panel">
|
||||
<div class="col-head">Версии <span id="versions-intent" style="color:var(--fg);text-transform:none;font-weight:normal;"></span></div>
|
||||
<div class="col-body" id="versions">
|
||||
<div class="mini">— выберите ветку —</div>
|
||||
</div>
|
||||
</aside>
|
||||
|
||||
@@ -292,7 +383,9 @@
|
||||
const $ = (id) => document.getElementById(id);
|
||||
const esc = (s) => String(s ?? "").replace(/[&<>"']/g, c => ({'&':'&','<':'<','>':'>','"':'"',"'":'''}[c]));
|
||||
|
||||
let configs = [];
|
||||
let intents = [];
|
||||
let currentIntentCode = null;
|
||||
let versions = [];
|
||||
|
||||
function toast(msg, kind = "ok") {
|
||||
const t = $("toast");
|
||||
@@ -319,52 +412,128 @@ function fmtDate(iso) {
|
||||
} catch (_) { return iso; }
|
||||
}
|
||||
|
||||
async function refreshHealth() {
|
||||
/* ---------- intents list ---------- */
|
||||
async function refreshIntents() {
|
||||
try {
|
||||
const h = await api("/health");
|
||||
const active = configs.find(c => c.is_active);
|
||||
const vTag = active ? `активная: v${active.version}${active.name ? " · " + esc(active.name) : ""}` : "нет активной";
|
||||
$("stats").innerHTML = `${vTag} · документов <b>${h.documents_count}</b> · чанков <b>${h.chunks_count}</b>`;
|
||||
const d = await api("/intents");
|
||||
intents = d.intents;
|
||||
renderIntents();
|
||||
} catch (e) {
|
||||
$("stats").textContent = "недоступен";
|
||||
$("intents-list").innerHTML = `<div class="mini" style="color:var(--err)">${esc(e.message)}</div>`;
|
||||
}
|
||||
}
|
||||
|
||||
async function refreshConfigs() {
|
||||
try {
|
||||
const d = await api("/configs");
|
||||
configs = d.configs;
|
||||
renderVersions();
|
||||
refreshHealth();
|
||||
} catch (e) {
|
||||
$("versions").innerHTML = `<div class="mini" style="color:var(--err)">${esc(e.message)}</div>`;
|
||||
}
|
||||
}
|
||||
function renderIntents() {
|
||||
const responders = intents.filter(i => !i.code.startsWith("_"));
|
||||
const system = intents.filter(i => i.code.startsWith("_"));
|
||||
|
||||
function renderVersions() {
|
||||
if (!configs.length) {
|
||||
$("versions").innerHTML = '<div class="mini">версий ещё нет</div>';
|
||||
return;
|
||||
}
|
||||
$("versions").innerHTML = configs.map(c => `
|
||||
<div class="version-card ${c.is_active ? "active" : ""}">
|
||||
<div class="version-head">
|
||||
<span class="v-num">v${c.version}</span>
|
||||
<span class="v-name ${c.name ? "" : "empty"}" title="${esc(c.name || '')}">${esc(c.name || "без имени")}</span>
|
||||
${c.is_active ? '<span class="v-active-badge">активная</span>' : ""}
|
||||
const renderItem = (i, isSystem) => `
|
||||
<div class="intent-item ${i.code === currentIntentCode ? 'active' : ''} ${i.is_enabled ? '' : 'disabled'}" onclick="selectIntent('${i.code}')">
|
||||
<div class="intent-top">
|
||||
<span class="intent-name">${esc(i.name)}</span>
|
||||
${isSystem
|
||||
? '<span class="system-badge" title="Системная ветка — не выключается">система</span>'
|
||||
: `<label class="switch" onclick="event.stopPropagation();">
|
||||
<input type="checkbox" ${i.is_enabled ? 'checked' : ''} onchange="toggleIntent('${i.code}', this.checked)">
|
||||
<span class="slider"></span>
|
||||
</label>`}
|
||||
</div>
|
||||
<div class="v-meta">${esc(fmtDate(c.created_at))} · промпт ${c.system_prompt.length} симв.${c.rules_text ? " · правил " + c.rules_text.length : ""}</div>
|
||||
<div class="v-actions">
|
||||
<button onclick="loadIntoEditor(${c.id})">Загрузить в редактор</button>
|
||||
${!c.is_active ? `<button class="primary" onclick="activate(${c.id})">Активировать</button>` : ""}
|
||||
${!c.is_active ? `<button class="del" onclick="deleteVersion(${c.id}, ${c.version})">Удалить</button>` : '<button class="del" disabled title="Активную удалить нельзя — сначала активируйте другую">Удалить</button>'}
|
||||
<div class="intent-desc">${esc(i.description)}</div>
|
||||
<div class="intent-version ${i.active_config_version ? '' : 'empty'}">
|
||||
<span class="intent-code">${esc(i.code)}</span>
|
||||
${i.active_config_version ? `· активна v${i.active_config_version}` : '· нет активной'}
|
||||
</div>
|
||||
</div>
|
||||
`).join("");
|
||||
`;
|
||||
|
||||
let html = '<div class="section-header">Ветки-ответчики</div>';
|
||||
html += responders.map(i => renderItem(i, false)).join("");
|
||||
if (system.length) {
|
||||
html += '<div class="section-header">Системные</div>';
|
||||
html += system.map(i => renderItem(i, true)).join("");
|
||||
}
|
||||
$("intents-list").innerHTML = html;
|
||||
}
|
||||
|
||||
function loadIntoEditor(id) {
|
||||
const c = configs.find(x => x.id === id);
|
||||
async function toggleIntent(code, enabled) {
|
||||
try {
|
||||
await api(`/intents/${code}`, {
|
||||
method: "PATCH",
|
||||
headers: { "Content-Type": "application/json" },
|
||||
body: JSON.stringify({ is_enabled: enabled }),
|
||||
});
|
||||
toast(`${code}: ${enabled ? 'включена' : 'выключена'}`);
|
||||
refreshIntents();
|
||||
} catch (e) {
|
||||
toast("Ошибка: " + e.message, "err");
|
||||
}
|
||||
}
|
||||
|
||||
/* ---------- select intent ---------- */
|
||||
async function selectIntent(code) {
|
||||
currentIntentCode = code;
|
||||
renderIntents();
|
||||
renderEditor();
|
||||
await refreshVersions(code);
|
||||
await loadActiveIntoEditor();
|
||||
}
|
||||
|
||||
function renderEditor() {
|
||||
const intent = intents.find(i => i.code === currentIntentCode);
|
||||
if (!intent) {
|
||||
$("editor").innerHTML = '<div class="editor-empty">Выберите ветку слева.</div>';
|
||||
$("editor-head").textContent = "Выберите ветку слева";
|
||||
return;
|
||||
}
|
||||
$("editor-head").textContent = `${intent.name} · редактор`;
|
||||
$("editor").innerHTML = `
|
||||
<h2>${esc(intent.name)}</h2>
|
||||
<div class="sub"><code>${esc(intent.code)}</code> · ${esc(intent.description)}</div>
|
||||
|
||||
<div class="field">
|
||||
<label for="f-name">Имя версии (необязательно)</label>
|
||||
<input type="text" id="f-name" placeholder="например: после фидбэка операторов 24.04" maxlength="200">
|
||||
</div>
|
||||
<div class="field">
|
||||
<label for="f-prompt">Системный промпт ветки</label>
|
||||
<textarea id="f-prompt" class="prompt" spellcheck="false"></textarea>
|
||||
</div>
|
||||
<div class="field">
|
||||
<label for="f-rules">Правила (дополнение к промпту; свободная markdown-форма)</label>
|
||||
<textarea id="f-rules" class="rules" spellcheck="false"></textarea>
|
||||
</div>
|
||||
<div class="editor-actions">
|
||||
<button onclick="saveVersion()">Сохранить как новую версию</button>
|
||||
<button class="secondary" onclick="loadActiveIntoEditor()">Перезагрузить активную</button>
|
||||
<label><input type="checkbox" id="chk-activate"> Сразу сделать активной</label>
|
||||
</div>
|
||||
`;
|
||||
}
|
||||
|
||||
async function loadActiveIntoEditor() {
|
||||
if (!currentIntentCode) return;
|
||||
const intent = intents.find(i => i.code === currentIntentCode);
|
||||
if (!intent || !intent.active_config_id) {
|
||||
// Новая ветка без активной версии — пусто.
|
||||
if ($("f-name")) {
|
||||
$("f-name").value = "";
|
||||
$("f-prompt").value = "";
|
||||
$("f-rules").value = "";
|
||||
}
|
||||
return;
|
||||
}
|
||||
try {
|
||||
const c = await api(`/configs/active?intent_code=${encodeURIComponent(currentIntentCode)}`);
|
||||
$("f-name").value = c.name ? `${c.name} (на основе v${c.version})` : `v${c.version} — копия`;
|
||||
$("f-prompt").value = c.system_prompt;
|
||||
$("f-rules").value = c.rules_text || "";
|
||||
} catch (e) {
|
||||
toast("Не удалось загрузить активную: " + e.message, "err");
|
||||
}
|
||||
}
|
||||
|
||||
function loadIntoEditor(configId) {
|
||||
const c = versions.find(x => x.id === configId);
|
||||
if (!c) return;
|
||||
$("f-name").value = c.name ? `${c.name} (на основе v${c.version})` : `v${c.version} — копия`;
|
||||
$("f-prompt").value = c.system_prompt;
|
||||
@@ -373,16 +542,47 @@ function loadIntoEditor(id) {
|
||||
window.scrollTo({ top: 0, behavior: "smooth" });
|
||||
}
|
||||
|
||||
function loadActiveIntoEditor() {
|
||||
const active = configs.find(c => c.is_active);
|
||||
if (!active) {
|
||||
toast("Активной версии нет", "err");
|
||||
return;
|
||||
/* ---------- versions ---------- */
|
||||
async function refreshVersions(code) {
|
||||
const intent = intents.find(i => i.code === code);
|
||||
$("versions-intent").textContent = intent ? ` — ${intent.name}` : "";
|
||||
try {
|
||||
const d = await api(`/configs?intent_code=${encodeURIComponent(code)}`);
|
||||
versions = d.configs;
|
||||
renderVersions();
|
||||
} catch (e) {
|
||||
$("versions").innerHTML = `<div class="mini" style="color:var(--err)">${esc(e.message)}</div>`;
|
||||
}
|
||||
loadIntoEditor(active.id);
|
||||
}
|
||||
|
||||
function renderVersions() {
|
||||
if (!versions.length) {
|
||||
$("versions").innerHTML = '<div class="mini">версий ещё нет</div>';
|
||||
return;
|
||||
}
|
||||
$("versions").innerHTML = `<div class="versions">` + versions.map(c => `
|
||||
<div class="version-card ${c.is_active ? "active" : ""}">
|
||||
<div class="version-head">
|
||||
<span class="v-num">v${c.version}</span>
|
||||
<span class="v-name ${c.name ? "" : "empty"}" title="${esc(c.name || '')}">${esc(c.name || "без имени")}</span>
|
||||
${c.is_active ? '<span class="v-active-badge">активная</span>' : ""}
|
||||
</div>
|
||||
<div class="v-meta">${esc(fmtDate(c.created_at))} · промпт ${c.system_prompt.length} симв.${c.rules_text ? " · правил " + c.rules_text.length : ""}</div>
|
||||
<div class="v-actions">
|
||||
<button onclick="loadIntoEditor(${c.id})">Загрузить</button>
|
||||
${!c.is_active ? `<button class="primary" onclick="activateVersion(${c.id})">Активировать</button>` : ""}
|
||||
${!c.is_active ? `<button class="del" onclick="deleteVersion(${c.id}, ${c.version})">Удалить</button>` : '<button class="del" disabled title="Активную удалить нельзя">Удалить</button>'}
|
||||
</div>
|
||||
</div>
|
||||
`).join("") + `</div>`;
|
||||
}
|
||||
|
||||
/* ---------- save / activate / delete ---------- */
|
||||
async function saveVersion() {
|
||||
if (!currentIntentCode) return;
|
||||
const intent = intents.find(i => i.code === currentIntentCode);
|
||||
if (!intent) return;
|
||||
|
||||
const name = $("f-name").value.trim();
|
||||
const system_prompt = $("f-prompt").value.trim();
|
||||
const rules_text = $("f-rules").value.trim();
|
||||
@@ -393,39 +593,44 @@ async function saveVersion() {
|
||||
return;
|
||||
}
|
||||
|
||||
$("btn-save").disabled = true;
|
||||
try {
|
||||
const r = await api("/configs", {
|
||||
method: "POST",
|
||||
headers: { "Content-Type": "application/json" },
|
||||
body: JSON.stringify({ name: name || null, system_prompt, rules_text, activate }),
|
||||
body: JSON.stringify({
|
||||
intent_id: intent.id,
|
||||
name: name || null,
|
||||
system_prompt,
|
||||
rules_text,
|
||||
activate,
|
||||
}),
|
||||
});
|
||||
toast(`Сохранена v${r.version}${activate ? " · активирована" : ""}`);
|
||||
toast(`v${r.version} сохранена${activate ? " · активирована" : ""}`);
|
||||
$("chk-activate").checked = false;
|
||||
await refreshConfigs();
|
||||
await refreshIntents();
|
||||
await refreshVersions(currentIntentCode);
|
||||
} catch (e) {
|
||||
toast("Ошибка: " + e.message, "err");
|
||||
} finally {
|
||||
$("btn-save").disabled = false;
|
||||
}
|
||||
}
|
||||
|
||||
async function activate(id) {
|
||||
async function activateVersion(id) {
|
||||
try {
|
||||
const r = await api(`/configs/${id}/activate`, { method: "POST" });
|
||||
toast(`Активирована v${r.version}`);
|
||||
refreshConfigs();
|
||||
await refreshIntents();
|
||||
await refreshVersions(currentIntentCode);
|
||||
} catch (e) {
|
||||
toast("Ошибка: " + e.message, "err");
|
||||
}
|
||||
}
|
||||
|
||||
async function deleteVersion(id, version) {
|
||||
if (!confirm(`Удалить версию v${version}?`)) return;
|
||||
if (!confirm(`Удалить v${version}?`)) return;
|
||||
try {
|
||||
await api(`/configs/${id}`, { method: "DELETE" });
|
||||
toast(`v${version} удалена`);
|
||||
refreshConfigs();
|
||||
await refreshVersions(currentIntentCode);
|
||||
} catch (e) {
|
||||
toast("Ошибка: " + e.message, "err");
|
||||
}
|
||||
@@ -433,9 +638,8 @@ async function deleteVersion(id, version) {
|
||||
|
||||
/* ---------- init ---------- */
|
||||
(async function init() {
|
||||
await refreshConfigs();
|
||||
// При первом заходе загружаем активную в редактор для удобства.
|
||||
loadActiveIntoEditor();
|
||||
await refreshIntents();
|
||||
if (intents.length) selectIntent(intents[0].code);
|
||||
})();
|
||||
</script>
|
||||
|
||||
|
||||
Reference in New Issue
Block a user