From 99288c195d157d56468c34ec416043d49ebde67a Mon Sep 17 00:00:00 2001 From: dctouch Date: Thu, 9 Apr 2026 16:32:19 +0300 Subject: [PATCH] =?UTF-8?q?=D0=90=D0=94=D0=A0=D0=95=D0=A1=D0=9D=D0=AB?= =?UTF-8?q?=D0=99=20=D0=A0=D0=95=D0=96=D0=98=D0=9C=20-=20=D0=B0=D0=B2?= =?UTF-8?q?=D1=82=D0=BE=D1=80=D0=B0=D0=BD=20=D0=B8=D1=81=D1=82=D0=BE=D1=80?= =?UTF-8?q?=D0=B8=D1=8F=20-=20=D0=B1=D0=B0=D0=B7=D0=BE=D0=B2=D0=B0=D1=8F?= =?UTF-8?q?=20=D0=B2=D0=B5=D1=80=D1=81=D0=B8=D1=8F=20+=20=D0=B4=D0=BE?= =?UTF-8?q?=D0=BF=20=D0=BA=D0=BB=D1=8F=20+=20=D0=BA=D0=BE=D0=BD=D1=84?= =?UTF-8?q?=D0=B8=D0=B3=20=D0=B4=D0=B8=D0=B7=D0=B0=D0=BF=D0=B9=D0=BD=D0=B0?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- designconfig.ts | 18 + docs/TECH/assistant_canon.md | 51 + docs/TECH/capabilities_registry.json | 193 +++ docs/TECH/history_colibration.md | 567 +++++++++ docs/TECH/manual_case_decision_schema.json | 39 + llm_normalizer/backend/dist/config.js | 10 +- .../backend/dist/routes/autoRuns.js | 799 +++++++++++- llm_normalizer/backend/dist/server.js | 2 + .../backend/dist/services/answerComposer.js | 76 +- .../backend/dist/services/assistantCanon.js | 41 + .../dist/services/assistantDataLayer.js | 182 +-- .../dist/services/assistantRuntimeGuards.js | 42 +- .../backend/dist/services/assistantService.js | 18 +- .../dist/services/capabilitiesRegistry.js | 182 +++ .../backend/dist/services/lifecycleRuntime.js | 146 +-- .../backend/dist/services/promptBuilder.js | 8 +- .../backend/dist/services/routeHintAdapter.js | 2 +- llm_normalizer/backend/src/config.ts | 19 + llm_normalizer/backend/src/routes/autoRuns.ts | 941 +++++++++++++- llm_normalizer/backend/src/server.ts | 14 +- .../backend/src/services/answerComposer.ts | 76 +- .../backend/src/services/assistantCanon.ts | 38 + .../src/services/assistantDataLayer.ts | 182 +-- .../src/services/assistantRuntimeGuards.ts | 42 +- .../backend/src/services/assistantService.ts | 18 +- .../src/services/capabilitiesRegistry.ts | 205 ++++ .../backend/src/services/lifecycleRuntime.ts | 146 +-- .../backend/src/services/promptBuilder.ts | 8 +- .../backend/src/services/routeHintAdapter.ts | 2 +- .../frontend/dist/assets/index-BDtb8kxy.js | 13 + .../frontend/dist/assets/index-BMWPMdQA.css | 1 - .../frontend/dist/assets/index-D6Y_lHrc.js | 12 - .../frontend/dist/assets/index-iFxz5cXp.css | 1 + llm_normalizer/frontend/dist/index.html | 4 +- llm_normalizer/frontend/src/App.tsx | 87 +- llm_normalizer/frontend/src/api/client.ts | 96 ++ .../src/components/AutoRunsHistoryPanel.tsx | 1080 +++++++++++++---- .../frontend/src/components/PanelFrame.tsx | 22 +- llm_normalizer/frontend/src/state/types.ts | 149 +++ llm_normalizer/frontend/src/styles.css | 495 ++++++-- 40 files changed, 5174 insertions(+), 853 deletions(-) create mode 100644 designconfig.ts create mode 100644 docs/TECH/assistant_canon.md create mode 100644 docs/TECH/capabilities_registry.json create mode 100644 docs/TECH/history_colibration.md create mode 100644 docs/TECH/manual_case_decision_schema.json create mode 100644 llm_normalizer/backend/dist/services/assistantCanon.js create mode 100644 llm_normalizer/backend/dist/services/capabilitiesRegistry.js create mode 100644 llm_normalizer/backend/src/services/assistantCanon.ts create mode 100644 llm_normalizer/backend/src/services/capabilitiesRegistry.ts create mode 100644 llm_normalizer/frontend/dist/assets/index-BDtb8kxy.js delete mode 100644 llm_normalizer/frontend/dist/assets/index-BMWPMdQA.css delete mode 100644 llm_normalizer/frontend/dist/assets/index-D6Y_lHrc.js create mode 100644 llm_normalizer/frontend/dist/assets/index-iFxz5cXp.css diff --git a/designconfig.ts b/designconfig.ts new file mode 100644 index 0000000..aee0db1 --- /dev/null +++ b/designconfig.ts @@ -0,0 +1,18 @@ +export const designConfig = { + colors: { + backgroundRgb: "18, 18, 18", + mainSurfaceRgb: "25, 25, 25", + horizontalSurfaceRgb: "30, 30, 30", + focusSurfaceRgb: "35, 35, 35", + activeRgb: "167, 59, 255", + activeTextRgb: "240, 240, 240", + textMainRgb: "240, 240, 240", + textMutedRgb: "166, 166, 166", + dangerRgb: "126, 126, 126", + scrollbarTrackRgb: "20, 20, 20", + scrollbarThumbRgb: "30, 30, 30", + scrollbarThumbHoverRgb: "30, 50, 30" + } +} as const; + +export type DesignConfig = typeof designConfig; diff --git a/docs/TECH/assistant_canon.md b/docs/TECH/assistant_canon.md new file mode 100644 index 0000000..783ad2b --- /dev/null +++ b/docs/TECH/assistant_canon.md @@ -0,0 +1,51 @@ +# Assistant Behavior Canon + +Schema version: `assistant_canon_v1` +Updated at: `2026-04-09` + +## Mission +Assist users with 1C data analysis in read-only mode: accurate, honest, and useful. + +## Core Rules +1. Never fabricate capabilities. +2. Never claim operational/admin actions in 1C. +3. Never expose internal technical pipeline details in user-facing replies. +4. Always separate: + - what is confirmed, + - what is inferred, + - what is unavailable. +5. For unsupported questions, provide a soft boundary and nearest useful supported action. + +## Covered Case Behavior +1. Answer directly and concretely. +2. Keep response concise and business-oriented. +3. Mention period/entity assumptions only when they impact correctness. + +## Partial Coverage Behavior +1. Explicitly state covered vs uncovered parts. +2. Ask only minimal clarifications required for correctness. +3. Propose the next best executable query. + +## Out-of-Scope Behavior +1. Do not output raw errors or internal route/classifier terms. +2. Use plain language boundary: + - "I cannot perform this action directly." +3. Offer safe alternatives: + - how to inspect in 1C, + - what data can be checked right now. + +## High-Risk Behavior +1. No destructive guidance. +2. No unsafe legal/financial certainty without data confirmation. +3. Prefer "confirmed data only" framing for factual claims. + +## Capability Disclosure Behavior +1. Use 3-level disclosure: + - L1: capability groups, + - L2: operations in selected group, + - L3: exact actionable query. +2. Never dump full internal route catalog by default. + +## Tone +1. Professional, calm, non-technical language. +2. Respectful boundary statements without refusal-only dead ends. diff --git a/docs/TECH/capabilities_registry.json b/docs/TECH/capabilities_registry.json new file mode 100644 index 0000000..c651fdf --- /dev/null +++ b/docs/TECH/capabilities_registry.json @@ -0,0 +1,193 @@ +{ + "schema_version": "capabilities_registry_v1", + "updated_at": "2026-04-09T00:00:00.000Z", + "assistant_mode": "read_only", + "groups": [ + { + "group_code": "vat", + "group_title": "НДС", + "description": "Расчеты и аналитика по НДС на основании данных 1С.", + "risk_level": "high", + "maturity_status": "partial", + "supported_operations": [ + "vat_period_snapshot", + "vat_payable_forecast", + "vat_turnover_breakdown" + ], + "unsupported_operations": [ + "submit_tax_declaration", + "legal_tax_advice_as_final" + ], + "required_entities": [ + "period", + "organization" + ], + "optional_entities": [ + "counterparty", + "account_scope" + ], + "typical_queries": [ + "Сколько НДС к уплате за период?", + "Покажи срез НДС на дату.", + "Почему НДС к уплате ноль?" + ], + "related_routes": [ + "address_vat_payable_forecast_v1" + ], + "safe_alternatives": [ + "Показать движения по счетам 68/19 за период", + "Показать сверочный срез по документам НДС" + ], + "one_c_hints": [ + "РегистрБухгалтерии.Хозрасчетный", + "Книга покупок/продаж" + ] + }, + { + "group_code": "counterparties", + "group_title": "Контрагенты", + "description": "Срезы активности, платежей и документов по контрагентам.", + "risk_level": "medium", + "maturity_status": "production_ready", + "supported_operations": [ + "list_documents_by_counterparty", + "bank_operations_by_counterparty", + "list_contracts_by_counterparty" + ], + "unsupported_operations": [ + "edit_counterparty_card", + "merge_counterparties" + ], + "required_entities": [ + "counterparty_scope_or_contract" + ], + "optional_entities": [ + "period", + "organization" + ], + "typical_queries": [ + "Покажи документы по контрагенту.", + "Какие операции по банку были с контрагентом?", + "Какие договоры есть у контрагента?" + ], + "related_routes": [ + "address_documents_by_counterparty_v1", + "address_bank_operations_by_counterparty_v1" + ], + "safe_alternatives": [ + "Ограничить период и организацию", + "Уточнить ИНН/наименование контрагента" + ], + "one_c_hints": [ + "Справочник.Контрагенты", + "Документы расчетов" + ] + }, + { + "group_code": "settlements", + "group_title": "Задолженности и расчеты", + "description": "Аналитика закрытия расчетов, сальдо и признаков незакрытых цепочек.", + "risk_level": "medium", + "maturity_status": "production_ready", + "supported_operations": [ + "settlement_closure_state", + "advance_offset_state", + "open_items_snapshot" + ], + "unsupported_operations": [ + "force_close_settlements", + "writeoff_execution" + ], + "required_entities": [ + "period", + "account_scope" + ], + "optional_entities": [ + "counterparty", + "contract" + ], + "typical_queries": [ + "Закрылись ли расчеты по счету 60/62?", + "Есть ли незакрытые авансы?", + "Покажи незакрытые договоры." + ], + "related_routes": [ + "prove_settlement_closure_state" + ], + "safe_alternatives": [ + "Уточнить контрагента", + "Уточнить договор/объект расчетов" + ], + "one_c_hints": [ + "Счета 60, 62, 76", + "Регистры взаиморасчетов" + ] + }, + { + "group_code": "cash_and_balances", + "group_title": "Деньги и остатки", + "description": "Остатки и динамика по денежным счетам и кассе.", + "risk_level": "medium", + "maturity_status": "partial", + "supported_operations": [ + "balance_snapshot", + "turnover_by_period" + ], + "unsupported_operations": [ + "payment_execution", + "bank_statement_import" + ], + "required_entities": [ + "period" + ], + "optional_entities": [ + "account_scope", + "organization" + ], + "typical_queries": [ + "Какой остаток по счету 51 на дату?", + "Покажи движение денег за месяц." + ], + "related_routes": [ + "address_balance_snapshot_v1" + ], + "safe_alternatives": [ + "Уточнить счет и период", + "Показать обороты вместо итоговой суммы" + ], + "one_c_hints": [ + "Счет 50, 51, 52, 55" + ] + }, + { + "group_code": "capability_boundaries", + "group_title": "Ограничения", + "description": "Операции, которые ассистент не выполняет в этом рантайме.", + "risk_level": "high", + "maturity_status": "production_ready", + "supported_operations": [ + "explain_boundary", + "suggest_safe_next_step" + ], + "unsupported_operations": [ + "configure_1c", + "admin_server_actions", + "create_or_post_documents", + "destructive_database_actions" + ], + "required_entities": [], + "optional_entities": [], + "typical_queries": [ + "Можешь настроить 1С?", + "Можешь удалить базу?", + "Можешь подготовить и провести документ?" + ], + "related_routes": [], + "safe_alternatives": [ + "Дать безопасный диагностический план для 1С/ИТ-админа", + "Подсказать точный запрос к данным в read-only" + ], + "one_c_hints": [] + } + ] +} diff --git a/docs/TECH/history_colibration.md b/docs/TECH/history_colibration.md new file mode 100644 index 0000000..6b5f55e --- /dev/null +++ b/docs/TECH/history_colibration.md @@ -0,0 +1,567 @@ +Да, тут уже напрашивается не просто “ещё одно поле в автопрогонах”, а **нормальная управляющая схема**. +То есть у вас должно быть не только “модель ответила / оценка 5-балльная”, а три опоры: + +1. **эталон идеального поведения ассистента**; +2. **ручная разметка результата прогона с управленческим смыслом**; +3. **канонический файл возможностей ассистента по отработанным маршрутам 1С**. + +И тогда автопрогоны перестают быть просто логами, а становятся контуром развития системы. + +Ниже я собрал это так, чтобы можно было почти целиком отдать в Codex. + +--- + +# Как я бы это сформулировал концептуально + +## 1. Нужен отдельный блок: «Эталон поведения ассистента» + +Это не просто описание “каким хотелось бы видеть ответ”. +Это должен быть **формальный канон**, который понимают: + +* сам ассистент; +* система автопрогонов; +* пост-анализ; +* Codex, который потом дорабатывает маршруты и поведение. + +То есть это не prose-блок “идеальная работа”, а именно **Assistant Canon / Behavior Canon**. + +### Что в нём должно быть + +#### А. Что такое хороший ответ + +Хороший ответ ассистента: + +* отвечает по существу, если кейс реально покрыт; +* не врёт, если кейс не покрыт; +* не выдаёт технические внутренности вместо нормальной коммуникации; +* не ломается на смежных вопросах; +* умеет мягко ограничить себя; +* умеет предложить близкий поддерживаемый сценарий; +* умеет подсказать, где это обычно смотреть в 1С; +* не вываливает полный список возможностей без запроса; +* раскрывает возможности по группам и по мере уточнения. + +#### Б. Что такое плохой ответ + +Плохой ответ ассистента: + +* выдумывает функцию, которой нет; +* делает вид, что может точно ответить, когда не может; +* отвечает внутренним техническим языком; +* сухо отказывает без пользы; +* валит пользователя в огромный список умений; +* не понимает, когда вопрос надо передать в “не покрыто, но рядом”; +* не различает безопасный общий ответ и рискованный прикладной совет. + +#### В. Какой идеал поведения на границе покрытия + +Вот это вообще ключевой блок. + +Ассистент должен: + +* честно понимать границу своих возможностей; +* не маскировать отсутствие маршрута; +* не ломаться; +* не отвечать “не поддерживается” в лоб; +* не уходить в системные сообщения; +* давать человекочитаемое ограничение; +* предлагать ближайший полезный путь. + +Это и есть ваш **эталон**. + +--- + +## 2. Нужна ручная разметка не только по качеству, но и по судьбе вопроса + +Вот это очень сильная мысль. +Пятибалльная оценка сама по себе почти бесполезна, потому что она **не говорит, что делать дальше**. + +Нужна ещё одна сущность: + +## **Decision Markup / Route Decision Markup** + +То есть по каждому прогону вы размечаете не только “норм / не норм”, а **каково управленческое решение по классу вопроса**. + +### Я бы добавил в UI не просто кнопку, а выпадающий классификатор + +Например: + +* `covered_ok` — кейс нормальный, покрывается, поведение ок; +* `covered_but_bad_answer` — кейс должен покрываться, но ответ плохой; +* `good_question_to_implement` — хороший вопрос, его надо брать в отработку; +* `out_of_scope_but_answer_softly` — вопрос не планируется покрывать, но нужно мягко и полезно отвечать; +* `unsafe_question_limit_strictly` — вопрос рискованный, на него нужно отвечать осторожно и ограниченно; +* `bad_test_case` — сам тестовый вопрос мусорный / нерелевантный; +* `needs_routing_extension` — нужен новый маршрут или расширение маршрутизации; +* `needs_capability_registry_update` — кейс выявил дыру в файле возможностей; +* `needs_dialog_policy_fix` — маршрут, возможно, не нужен, но политика ответа плохая. + +Вот это уже даст системе смысл. + +### Если хочется совсем по-простому + +Можно ввести более короткий список: + +* **Отрабатывается** +* **Должно отрабатываться** +* **Не будет отрабатываться** +* **Отвечать мягким ограничением** +* **Высокорисковый вопрос** +* **Плохой тест-кейс** + +Но я бы всё-таки оставил более инженерный набор, а в UI уже сделал человекочитаемые названия. + +--- + +## 3. Нужен канонический файл возможностей ассистента + +Да, это обязательно. И это должен быть не просто текстовый файл “что умеем”. +Это должен быть **Capabilities Registry / Supported Routes Registry**. + +И он должен быть источником истины для трёх вещей: + +* ответа ассистента; +* классификации покрываемости; +* автопрогонов и анализа. + +### Что там должно быть + +Для каждого маршрута / домена: + +* код маршрута; +* человекочитаемая группа; +* краткое описание; +* что реально умеется; +* что не умеется; +* обязательные параметры; +* типовые формулировки вопросов; +* похожие смежные сценарии; +* безопасные альтернативы; +* подсказка, где это обычно смотреть в 1С; +* уровень риска; +* статус зрелости: + + * production-ready + * partial + * planned + * deprecated + +### Пример групп + +Не надо сразу показывать всё пользователю. +Надо хранить глубоко, а наружу отдавать по группам. + +Например: + +* НДС +* Контрагенты +* Задолженности +* Деньги и остатки +* Платежи и движения +* Аналитика по периодам +* Справочные бухгалтерские вопросы + +А дальше уже внутри группы: + +* что умеется конкретно. + +То есть если юзер спрашивает “что ты можешь по НДС?”, ассистент отвечает не списком из 80 пунктов, а компактной группой возможностей по НДС. + +--- + +## 4. Надо отдельно зафиксировать правило раскрытия возможностей + +Это тоже очень важный продуктовый момент. + +### Ассистент не должен + +* автоматически вываливать весь список поддерживаемого; +* отвечать каталогом без запроса; +* перегружать пользователя техническими деталями маршрутов. + +### Ассистент должен + +* раскрывать возможности **по группам**; +* сначала давать верхнеуровневую сегментацию; +* при уточнении — углубляться; +* говорить в продуктовой, а не внутренне-технической логике. + +То есть: +“Могу помочь с НДС, остатками и движением денег, контрагентами и задолженностями, а также с частью аналитики по периодам. Если хочешь, могу уточнить отдельно по любому из этих блоков.” + +А уже потом: +“По НДС могу показать суммы, динамику по периодам, сверку по организации, сравнительные разрезы...” + +--- + +## 5. Нужен отдельный тип разметки: «вопрос хороший, но ещё не покрыт» + +Это, по сути, мост между автопрогоном и roadmap. + +То есть если в прогоне всплыл вопрос: + +* он адекватный; +* он реально нужен; +* пользователь его точно задаст; +* сейчас он не покрыт, + +то это не просто “ответ плохой”. +Это **кандидат на новый маршрут / на расширение текущего покрытия**. + +Поэтому в ручной разметке должен быть отдельный флаг: + +* `candidate_for_implementation` + или +* `planned_route_gap` + +Именно его потом должен видеть Codex и дальше использовать как список задач на развитие. + +--- + +## 6. Надо разделить две разные проблемы + +Сейчас у тебя в одном описании смешаны две вещи, а их лучше развести. + +### Проблема 1. Функциональное покрытие + +Что система реально умеет по данным и маршрутам. + +### Проблема 2. Поведенческая зрелость + +Как система ведёт себя, когда вопрос: + +* вне покрытия; +* частично в покрытии; +* опасный; +* слишком общий; +* смежный; +* абстрактный. + +То есть даже если маршрут не реализован, поведение всё равно может быть: + +* хорошим; +* плохим; +* опасным; +* слишком техническим; +* бесполезным. + +И автопрогоны должны это различать. + +--- + +# Ниже — готовая мини-ТЗшка для Codex + +## Мини-ТЗ: эталон поведения ассистента, ручная разметка прогонов и канонический файл возможностей + +### Цель + +Доработать систему автопрогонов бухгалтерского ассистента так, чтобы она оценивала не только качество конкретного ответа, но и соответствие эталонному поведению ассистента, а также позволяла вручную размечать судьбу вопроса: покрывается, должен быть отработан, не будет отрабатываться, должен обрабатываться мягким ограничением, требует нового маршрута или требует доработки политики ответа. + +--- + +## 1. Ввести отдельную сущность: Assistant Behavior Canon + +Нужен канонический блок, описывающий эталонную работу ассистента. + +### Требования + +Создать отдельную структуру/файл, который будет использоваться: + +* в логике ответа ассистента; +* в пост-анализе автопрогонов; +* в интерфейсе разметки прогонов; +* в дальнейшей работе Codex по развитию маршрутов и поведения. + +### В Assistant Behavior Canon зафиксировать: + +#### 1.1. Поведение на покрытых кейсах + +* отвечать уверенно и по существу; +* не уходить в лишние оговорки; +* не использовать технические внутренние формулировки. + +#### 1.2. Поведение на частично покрытых кейсах + +* явно разделять, что ассистент может сделать, а что нет; +* не маскировать ограничения; +* предлагать полезное продолжение. + +#### 1.3. Поведение на непокрытых, но близких кейсах + +* не выдумывать поддержку функциональности; +* мягко и по-человечески объяснять ограничение; +* предлагать ближайший поддерживаемый сценарий; +* при уместности подсказывать, где это обычно посмотреть в 1С. + +#### 1.4. Поведение на высокорисковых вопросах + +* не выдавать неподтверждённые рекомендации как надёжный ответ; +* не делать вид, что прикладная логика существует, если её нет; +* сохранять полезность без ложной уверенности. + +#### 1.5. Поведение при вопросе “что ты умеешь” + +* не вываливать весь список возможностей сразу; +* раскрывать возможности по крупным группам; +* углубляться только после уточнения; +* использовать человекочитаемые продуктовые группы, а не внутренние названия маршрутов. + +--- + +## 2. Ввести канонический файл возможностей ассистента + +Нужен отдельный реестр отработанных возможностей ассистента по маршрутам 1С. + +### Назначение + +Этот файл является источником истины для: + +* определения покрытия вопроса; +* ответа ассистента на вопросы о своих возможностях; +* similarity-логики; +* автопрогонов и пост-анализа; +* Codex при дальнейшем развитии маршрутов. + +### Для каждого маршрута / домена хранить: + +* `route_code` +* `group_code` +* `group_title` +* `title` +* `description` +* `supported_operations` +* `unsupported_operations` +* `required_entities` +* `optional_entities` +* `typical_queries` +* `related_routes` +* `safe_alternatives` +* `one_c_hints` +* `risk_level` +* `maturity_status` (`production_ready`, `partial`, `planned`, `deprecated`) + +### Требования к пользовательскому раскрытию возможностей + +Ассистент должен уметь: + +* сначала показывать верхнеуровневые группы; +* по запросу раскрывать детали внутри выбранной группы; +* не использовать длинные технические перечни без необходимости. + +--- + +## 3. Доработать интерфейс автопрогонов: ручная управленческая разметка + +Существующую 5-балльную оценку оставить, но дополнить отдельной выпадающей ручной классификацией результата прогона. + +### Добавить новое поле: + +`manual_case_decision` + +### Возможные значения: + +* `covered_ok` +* `covered_but_bad_answer` +* `candidate_for_implementation` +* `needs_routing_extension` +* `out_of_scope_but_answer_softly` +* `unsafe_question_limit_strictly` +* `needs_dialog_policy_fix` +* `needs_capability_registry_update` +* `bad_test_case` + +### Смысл значений + +* `covered_ok` — кейс уже покрыт, поведение нормальное; +* `covered_but_bad_answer` — кейс покрывается, но ответ/диалог плохой; +* `candidate_for_implementation` — хороший пользовательский кейс, которого пока нет, его стоит брать в разработку; +* `needs_routing_extension` — нужен новый маршрут или расширение существующего; +* `out_of_scope_but_answer_softly` — кейс не планируется покрывать, но нужен качественный мягкий ответ без техничности; +* `unsafe_question_limit_strictly` — кейс относится к рискованным, и ассистент должен ограничивать себя особенно строго; +* `needs_dialog_policy_fix` — проблема не в маршруте, а в стиле/логике ответа; +* `needs_capability_registry_update` — реестр возможностей неактуален или недостаточно формализован; +* `bad_test_case` — вопрос мусорный, нерелевантный или бесполезный для развития системы. + +### Дополнительно + +Для каждой ручной метки предусмотреть: + +* короткий комментарий; +* автора разметки; +* timestamp; +* возможность использовать эту разметку в пост-анализе и отборе задач для Codex. + +--- + +## 4. Доработать логику пост-анализа прогонов + +После прогона система должна уметь отделять: + +* ошибки покрытия; +* ошибки маршрутизации; +* ошибки политики ответа; +* хорошие, но ещё не покрытые кейсы; +* мусорные тест-кейсы; +* высокорисковые кейсы; +* кейсы на обновление файла возможностей. + +### На выходе пост-анализа нужны агрегаты: + +* список кейсов на доработку маршрутов; +* список кейсов на доработку policy; +* список кейсов на обновление capabilities registry; +* список кейсов, которые сознательно не будут покрываться, но требуют мягкого ограничения; +* список кейсов, пригодных для новых regression suites. + +--- + +## 5. Встроить связь между ручной разметкой и дальнейшей работой Codex + +Codex должен видеть не только сам диалог и оценку, но и управленческое решение по нему. + +### Требование + +При выгрузке данных для дальнейшего анализа и доработок обязательно передавать: + +* question / dialog trace; +* current route / current coverage decision; +* 5-балльную оценку; +* `manual_case_decision`; +* комментарий аналитика; +* ссылку на ближайший домен из capabilities registry; +* признак: нужно ли брать кейс в маршрутную отработку. + +### Ожидаемое поведение + +Если кейс помечен как: + +* `candidate_for_implementation` или `needs_routing_extension` — Codex рассматривает его как материал для новой/расширенной маршрутной логики; +* `out_of_scope_but_answer_softly` — Codex улучшает не маршруты, а policy-слой ответа; +* `needs_capability_registry_update` — Codex актуализирует реестр возможностей; +* `unsafe_question_limit_strictly` — Codex усиливает безопасное поведение и ограничения; +* `covered_but_bad_answer` — Codex чинит существующий покрываемый сценарий, а не создаёт новый. + +--- + +## 6. Добавить в эталон обязательное правило минимизации технических ответов + +Это отдельное критичное требование. + +### Ассистент не должен + +* отвечать внутренними техническими терминами; +* ссылаться на отсутствие маршрута, домена, интента, пайплайна, классификатора; +* создавать ощущение поломки системы. + +### Ассистент должен + +* говорить естественно; +* объяснять ограничения человеческим языком; +* сохранять полезность даже при отказе; +* ориентировать пользователя в доступных соседних возможностях. + +--- + +## 7. Добавить в эталон обязательное правило сегментированного раскрытия возможностей + +Если пользователь спрашивает: + +* “что ты умеешь?” +* “что можешь по НДС?” +* “что можешь по остаткам?” +* “что умеешь по деньгам / поставщикам / задолженностям?” + +ассистент должен отвечать через иерархию: + +### Уровень 1 + +Крупные продуктовые группы: + +* НДС +* Контрагенты +* Задолженности +* Деньги и остатки +* Движение и платежи +* Аналитика по периодам +* Справочные бухгалтерские вопросы + +### Уровень 2 + +Уточнение по выбранной группе: + +* что внутри группы реально доступно; +* какие ограничения есть; +* что можно сделать следующим шагом. + +### Уровень 3 + +Точечный ответ по конкретному запросу. + +--- + +## 8. Критерии приёмки + +1. В системе появился отдельный канонический блок эталонного поведения ассистента. +2. Появился отдельный файл/реестр возможностей по маршрутам 1С. +3. В UI автопрогонов добавлена ручная управленческая разметка результата. +4. Ручная разметка сохраняется в логах и участвует в пост-анализе. +5. Система умеет отделять “не покрыто, но стоит реализовать” от “не покрыто и не планируется, но нужно мягко отвечать”. +6. Ассистент перестаёт отвечать внутренним техническим языком на границе покрытия. +7. Ассистент умеет раскрывать свои возможности по группам, а не полным списком. +8. Codex получает достаточно данных, чтобы понимать, что чинить: маршрут, policy, capabilities registry или сам тест-кейс. + +--- + +# Что я бы ещё добавил от себя + +Я бы прямо выделил в ТЗ отдельный артефакт: + +## `assistant_canon.md` + +В нём: + +* идеал поведения; +* анти-паттерны; +* примеры хороших ответов; +* примеры плохих ответов; +* правила раскрытия возможностей; +* правила мягкого ограничения; +* правила поведения на рискованных вопросах. + +И отдельно: + +## `capabilities_registry.json` + +или `capabilities_registry.yaml` + +И ещё: + +## `manual_case_decision_schema.json` + +Чтобы UI и пост-анализ работали по одному словарю значений. + +--- + +# Если совсем коротко, в чём суть + +Тебе сейчас нужен не просто “ещё один контрол в модалке”, а вот такая конструкция: + +**Эталон ассистента** +→ задаёт идеальное поведение + +**Файл возможностей** +→ задаёт фактическое покрытие + +**Ручная разметка кейса** +→ задаёт управленческое решение, что с этим вопросом делать дальше + +**Codex** +→ уже понимает, нужно ли: + +* чинить ответ, +* расширять маршрут, +* обновлять capabilities, +* улучшать мягкий отказ, +* или вообще выкинуть тест-кейс. + +Если хочешь, я следующим сообщением могу собрать это ещё в более прикладной форме: **короткое ТЗ на 30–40 строк для прямой отправки в Codex**, без пояснений и лирики. diff --git a/docs/TECH/manual_case_decision_schema.json b/docs/TECH/manual_case_decision_schema.json new file mode 100644 index 0000000..f515aa9 --- /dev/null +++ b/docs/TECH/manual_case_decision_schema.json @@ -0,0 +1,39 @@ +{ + "schema_version": "manual_case_decision_schema_v1", + "updated_at": "2026-04-09T00:00:00.000Z", + "title": "Manual Case Decision Schema", + "description": "Management decision for assistant auto-run case annotation.", + "enum": [ + "covered_ok", + "covered_but_bad_answer", + "candidate_for_implementation", + "needs_routing_extension", + "out_of_scope_but_answer_softly", + "unsafe_question_limit_strictly", + "needs_dialog_policy_fix", + "needs_capability_registry_update", + "bad_test_case" + ], + "labels": { + "covered_ok": "Покрыто и ок", + "covered_but_bad_answer": "Покрыто, но ответ плохой", + "candidate_for_implementation": "Кандидат на внедрение", + "needs_routing_extension": "Нужно расширение маршрутизации", + "out_of_scope_but_answer_softly": "Вне скоупа, но нужен мягкий ответ", + "unsafe_question_limit_strictly": "Высокий риск, строгие ограничения", + "needs_dialog_policy_fix": "Нужен фикс диалоговой политики", + "needs_capability_registry_update": "Нужно обновить реестр возможностей", + "bad_test_case": "Плохой тест-кейс" + }, + "queue_mapping": { + "covered_ok": "none", + "covered_but_bad_answer": "policy_fix", + "candidate_for_implementation": "routing_extension", + "needs_routing_extension": "routing_extension", + "out_of_scope_but_answer_softly": "soft_boundary", + "unsafe_question_limit_strictly": "safety_policy", + "needs_dialog_policy_fix": "policy_fix", + "needs_capability_registry_update": "capability_registry", + "bad_test_case": "testset_hygiene" + } +} diff --git a/llm_normalizer/backend/dist/config.js b/llm_normalizer/backend/dist/config.js index 33223ac..df6d2c7 100644 --- a/llm_normalizer/backend/dist/config.js +++ b/llm_normalizer/backend/dist/config.js @@ -3,7 +3,8 @@ var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); -exports.ARCH_EXPORT_2020_DIR = exports.SCHEMAS_DIR = exports.EVAL_DATASETS_DIR = exports.REPORTS_DIR = exports.PROMPTS_DIR = exports.ASSISTANT_SESSIONS_DIR = exports.EVAL_CASES_DIR = exports.PRESETS_DIR = exports.TRACES_DIR = exports.DATA_DIR = exports.VAT_PAYABLE_19_PREFIXES = exports.VAT_PAYABLE_68_PREFIXES = exports.ASSISTANT_MCP_LIVE_LIMIT = exports.ASSISTANT_MCP_TIMEOUT_MS = exports.ASSISTANT_MCP_CHANNEL = exports.ASSISTANT_MCP_PROXY_URL = exports.FEATURE_ASSISTANT_LIVING_CHAT_ROUTER_V1 = exports.FEATURE_ASSISTANT_ADDRESS_QUERY_LIVE_V1 = exports.FEATURE_ASSISTANT_ADDRESS_QUERY_LLM_PREDECOMPOSE_V1 = exports.FEATURE_ASSISTANT_ADDRESS_QUERY_V1 = exports.FEATURE_ASSISTANT_MCP_RUNTIME_V1 = exports.FEATURE_ASSISTANT_GRAPH_RUNTIME_V1 = exports.FEATURE_ASSISTANT_LIFECYCLE_ANSWER_V1 = exports.FEATURE_ASSISTANT_LIFECYCLE_RUNTIME_V1 = exports.FEATURE_ASSISTANT_STAGE2_EVAL_V1 = exports.FEATURE_ASSISTANT_PROBLEM_UNIT_CONTINUITY_V1 = exports.FEATURE_ASSISTANT_PROBLEM_CENTRIC_ANSWER_V1 = exports.FEATURE_ASSISTANT_PROBLEM_UNITS_V1 = exports.FEATURE_ASSISTANT_ACCOUNTANT_EVAL_V1 = exports.FEATURE_ASSISTANT_ANSWER_POLICY_V11 = exports.FEATURE_ASSISTANT_ANTI_GENERIC_RANKING_GUARD_V1 = exports.FEATURE_ASSISTANT_MIN_EVIDENCE_GATE_V1 = exports.FEATURE_ASSISTANT_BROAD_GUARD_V1 = exports.FEATURE_ASSISTANT_EVIDENCE_ENRICHMENT_V1 = exports.FEATURE_ASSISTANT_STATE_FOLLOWUP_BINDING_V1 = exports.FEATURE_ASSISTANT_CONTRACTS_V11 = exports.FEATURE_ASSISTANT_INVESTIGATION_STATE_V1 = exports.DEFAULT_PROMPT_VERSION = exports.DEFAULT_MAX_OUTPUT_TOKENS = exports.DEFAULT_TEMPERATURE = exports.DEFAULT_MODEL = exports.DEFAULT_OPENAI_BASE_URL = exports.TIMEZONE = exports.PORT = exports.MODULE_ROOT = exports.BACKEND_ROOT = void 0; +exports.ARCH_EXPORT_2020_DIR = exports.SCHEMAS_DIR = exports.EVAL_DATASETS_DIR = exports.REPORTS_DIR = exports.PROMPTS_DIR = exports.AUTORUN_GENERATOR_HISTORY_FILE = exports.AUTORUN_GENERATOR_DIR = exports.AUTORUN_ANNOTATIONS_FILE = exports.AUTORUN_ANNOTATIONS_DIR = exports.ASSISTANT_SESSIONS_DIR = exports.EVAL_CASES_DIR = exports.PRESETS_DIR = exports.TRACES_DIR = exports.DATA_DIR = exports.VAT_PAYABLE_19_PREFIXES = exports.VAT_PAYABLE_68_PREFIXES = exports.ASSISTANT_MCP_LIVE_LIMIT = exports.ASSISTANT_MCP_TIMEOUT_MS = exports.ASSISTANT_MCP_CHANNEL = exports.ASSISTANT_MCP_PROXY_URL = exports.FEATURE_ASSISTANT_LIVING_CHAT_ROUTER_V1 = exports.FEATURE_ASSISTANT_ADDRESS_QUERY_LIVE_V1 = exports.FEATURE_ASSISTANT_ADDRESS_QUERY_LLM_PREDECOMPOSE_V1 = exports.FEATURE_ASSISTANT_ADDRESS_QUERY_V1 = exports.FEATURE_ASSISTANT_MCP_RUNTIME_V1 = exports.FEATURE_ASSISTANT_GRAPH_RUNTIME_V1 = exports.FEATURE_ASSISTANT_LIFECYCLE_ANSWER_V1 = exports.FEATURE_ASSISTANT_LIFECYCLE_RUNTIME_V1 = exports.FEATURE_ASSISTANT_STAGE2_EVAL_V1 = exports.FEATURE_ASSISTANT_PROBLEM_UNIT_CONTINUITY_V1 = exports.FEATURE_ASSISTANT_PROBLEM_CENTRIC_ANSWER_V1 = exports.FEATURE_ASSISTANT_PROBLEM_UNITS_V1 = exports.FEATURE_ASSISTANT_ACCOUNTANT_EVAL_V1 = exports.FEATURE_ASSISTANT_ANSWER_POLICY_V11 = exports.FEATURE_ASSISTANT_ANTI_GENERIC_RANKING_GUARD_V1 = exports.FEATURE_ASSISTANT_MIN_EVIDENCE_GATE_V1 = exports.FEATURE_ASSISTANT_BROAD_GUARD_V1 = exports.FEATURE_ASSISTANT_EVIDENCE_ENRICHMENT_V1 = exports.FEATURE_ASSISTANT_STATE_FOLLOWUP_BINDING_V1 = exports.FEATURE_ASSISTANT_CONTRACTS_V11 = exports.FEATURE_ASSISTANT_INVESTIGATION_STATE_V1 = exports.DEFAULT_PROMPT_VERSION = exports.DEFAULT_MAX_OUTPUT_TOKENS = exports.DEFAULT_TEMPERATURE = exports.DEFAULT_MODEL = exports.DEFAULT_OPENAI_BASE_URL = exports.TIMEZONE = exports.PORT = exports.MODULE_ROOT = exports.BACKEND_ROOT = void 0; +exports.MANUAL_CASE_DECISION_SCHEMA_FILE = exports.ASSISTANT_CAPABILITIES_REGISTRY_FILE = exports.ASSISTANT_CANON_FILE = void 0; const path_1 = __importDefault(require("path")); exports.BACKEND_ROOT = path_1.default.resolve(__dirname, ".."); exports.MODULE_ROOT = path_1.default.resolve(exports.BACKEND_ROOT, ".."); @@ -71,8 +72,15 @@ exports.TRACES_DIR = path_1.default.resolve(exports.DATA_DIR, "traces"); exports.PRESETS_DIR = path_1.default.resolve(exports.DATA_DIR, "presets"); exports.EVAL_CASES_DIR = path_1.default.resolve(exports.DATA_DIR, "eval_cases"); exports.ASSISTANT_SESSIONS_DIR = path_1.default.resolve(exports.DATA_DIR, "assistant_sessions"); +exports.AUTORUN_ANNOTATIONS_DIR = path_1.default.resolve(exports.DATA_DIR, "autorun_annotations"); +exports.AUTORUN_ANNOTATIONS_FILE = path_1.default.resolve(exports.AUTORUN_ANNOTATIONS_DIR, "annotations.json"); +exports.AUTORUN_GENERATOR_DIR = path_1.default.resolve(exports.DATA_DIR, "autorun_generators"); +exports.AUTORUN_GENERATOR_HISTORY_FILE = path_1.default.resolve(exports.AUTORUN_GENERATOR_DIR, "history.json"); exports.PROMPTS_DIR = path_1.default.resolve(exports.MODULE_ROOT, "prompts"); exports.REPORTS_DIR = path_1.default.resolve(exports.MODULE_ROOT, "reports"); exports.EVAL_DATASETS_DIR = path_1.default.resolve(exports.MODULE_ROOT, "eval_cases"); exports.SCHEMAS_DIR = path_1.default.resolve(exports.BACKEND_ROOT, "src", "schemas"); exports.ARCH_EXPORT_2020_DIR = path_1.default.resolve(exports.MODULE_ROOT, "..", "docs", "ARCH", "2020экспорт"); +exports.ASSISTANT_CANON_FILE = path_1.default.resolve(exports.MODULE_ROOT, "..", "docs", "TECH", "assistant_canon.md"); +exports.ASSISTANT_CAPABILITIES_REGISTRY_FILE = path_1.default.resolve(exports.MODULE_ROOT, "..", "docs", "TECH", "capabilities_registry.json"); +exports.MANUAL_CASE_DECISION_SCHEMA_FILE = path_1.default.resolve(exports.MODULE_ROOT, "..", "docs", "TECH", "manual_case_decision_schema.json"); diff --git a/llm_normalizer/backend/dist/routes/autoRuns.js b/llm_normalizer/backend/dist/routes/autoRuns.js index a155a48..eab10d1 100644 --- a/llm_normalizer/backend/dist/routes/autoRuns.js +++ b/llm_normalizer/backend/dist/routes/autoRuns.js @@ -9,6 +9,29 @@ const path_1 = __importDefault(require("path")); const express_1 = require("express"); const config_1 = require("../config"); const http_1 = require("../utils/http"); +const capabilitiesRegistry_1 = require("../services/capabilitiesRegistry"); +const MANUAL_CASE_DECISIONS = [ + "covered_ok", + "covered_but_bad_answer", + "candidate_for_implementation", + "needs_routing_extension", + "out_of_scope_but_answer_softly", + "unsafe_question_limit_strictly", + "needs_dialog_policy_fix", + "needs_capability_registry_update", + "bad_test_case" +]; +const DECISION_QUEUE_MAP = { + covered_ok: "none", + covered_but_bad_answer: "policy_fix", + candidate_for_implementation: "routing_extension", + needs_routing_extension: "routing_extension", + out_of_scope_but_answer_softly: "soft_boundary", + unsafe_question_limit_strictly: "safety_policy", + needs_dialog_policy_fix: "policy_fix", + needs_capability_registry_update: "capability_registry", + bad_test_case: "testset_hygiene" +}; function toRecord(value) { if (!value || typeof value !== "object" || Array.isArray(value)) { return null; @@ -67,6 +90,304 @@ function clampInt(value, min, max, fallback) { return max; return rounded; } +function parseManualCaseDecision(value, fallback = "needs_dialog_policy_fix") { + const normalized = toStringSafe(value); + if (!normalized) + return fallback; + return (MANUAL_CASE_DECISIONS.includes(normalized) ? normalized : fallback); +} +function parseAnnotationAuthor(value) { + const author = toStringSafe(value); + if (!author) + return null; + return author.slice(0, 80); +} +function readManualDecisionSchema() { + const fallback = { + schema_version: "manual_case_decision_schema_v1_fallback", + enum: MANUAL_CASE_DECISIONS, + labels: { + covered_ok: "Покрыто и ок", + covered_but_bad_answer: "Покрыто, но ответ плохой", + candidate_for_implementation: "Кандидат на внедрение", + needs_routing_extension: "Нужно расширение маршрутизации", + out_of_scope_but_answer_softly: "Вне скоупа, но нужен мягкий ответ", + unsafe_question_limit_strictly: "Высокий риск, строгие ограничения", + needs_dialog_policy_fix: "Нужен фикс диалоговой политики", + needs_capability_registry_update: "Нужно обновить реестр возможностей", + bad_test_case: "Плохой тест-кейс" + }, + queue_mapping: DECISION_QUEUE_MAP + }; + if (!fs_1.default.existsSync(config_1.MANUAL_CASE_DECISION_SCHEMA_FILE)) { + return fallback; + } + try { + const parsed = JSON.parse(fs_1.default.readFileSync(config_1.MANUAL_CASE_DECISION_SCHEMA_FILE, "utf-8")); + const record = toRecord(parsed); + return record ?? fallback; + } + catch { + return fallback; + } +} +function readAutoGenHistory() { + if (!fs_1.default.existsSync(config_1.AUTORUN_GENERATOR_HISTORY_FILE)) + return []; + try { + const parsed = JSON.parse(fs_1.default.readFileSync(config_1.AUTORUN_GENERATOR_HISTORY_FILE, "utf-8")); + if (!Array.isArray(parsed)) + return []; + return parsed + .map((item) => toRecord(item)) + .filter((item) => item !== null) + .map((item) => ({ + generation_id: toStringSafe(item.generation_id) ?? "", + created_at: toStringSafe(item.created_at) ?? new Date().toISOString(), + mode: toStringSafe(item.mode) ?? "codex_creative", + count: clampInt(toNumberSafe(item.count), 1, 300, 20), + domain: toStringSafe(item.domain), + questions: toArray(item.questions) + .map((q) => toStringSafe(q)) + .filter((q) => q !== null) + .slice(0, 500), + generated_by: toStringSafe(item.generated_by), + saved_case_set_file: toStringSafe(item.saved_case_set_file), + context: toRecord(item.context) + ? { + llm_provider: toStringSafe(toRecord(item.context)?.llm_provider), + model: toStringSafe(toRecord(item.context)?.model), + assistant_prompt_version: toStringSafe(toRecord(item.context)?.assistant_prompt_version), + decomposition_prompt_version: toStringSafe(toRecord(item.context)?.decomposition_prompt_version), + prompt_fingerprint: toStringSafe(toRecord(item.context)?.prompt_fingerprint) + } + : null + })) + .filter((item) => item.generation_id.length > 0) + .sort((a, b) => Date.parse(b.created_at) - Date.parse(a.created_at)); + } + catch { + return []; + } +} +function writeAutoGenHistory(records) { + const dir = path_1.default.dirname(config_1.AUTORUN_GENERATOR_HISTORY_FILE); + if (!fs_1.default.existsSync(dir)) { + fs_1.default.mkdirSync(dir, { recursive: true }); + } + fs_1.default.writeFileSync(config_1.AUTORUN_GENERATOR_HISTORY_FILE, JSON.stringify(records, null, 2), "utf-8"); +} +function readEvalDatasetCases(filePath) { + try { + const parsed = JSON.parse(fs_1.default.readFileSync(filePath, "utf-8")); + if (Array.isArray(parsed)) { + return parsed.map((item) => toRecord(item)).filter((item) => item !== null); + } + const record = toRecord(parsed); + if (!record) + return []; + const cases = toArray(record.cases).map((item) => toRecord(item)).filter((item) => item !== null); + return cases; + } + catch { + return []; + } +} +function collectCanonicalQuestions(limit = 300) { + if (!fs_1.default.existsSync(config_1.EVAL_DATASETS_DIR)) { + return []; + } + const entries = fs_1.default.readdirSync(config_1.EVAL_DATASETS_DIR, { withFileTypes: true }); + const questions = []; + for (const entry of entries) { + if (!entry.isFile() || !entry.name.endsWith(".json")) + continue; + const fullPath = path_1.default.resolve(config_1.EVAL_DATASETS_DIR, entry.name); + const cases = readEvalDatasetCases(fullPath); + for (const testCase of cases) { + const rawQuestion = toStringSafe(testCase.raw_question) ?? toStringSafe(testCase.user_message) ?? toStringSafe(testCase.query); + if (rawQuestion) { + questions.push(rawQuestion); + } + } + } + return Array.from(new Set(questions)).slice(0, limit); +} +function normalizeDomainHint(value) { + const domain = toStringSafe(value); + if (!domain) + return null; + return domain.toLowerCase(); +} +function fallbackDomainTemplates(domain) { + if (domain?.includes("vat") || domain?.includes("ндс")) { + return [ + "Сколько НДС к уплате на дату по организации?", + "Покажи прогноз НДС за период по организации.", + "Почему по НДС сейчас ноль и из чего сложился расчет?" + ]; + } + if (domain?.includes("counter") || domain?.includes("контраг")) { + return [ + "Покажи топ контрагентов по сумме платежей за период.", + "Какой самый крупный договор у выбранной организации?", + "Какие документы были по контрагенту за весь период?" + ]; + } + if (domain?.includes("settlement") || domain?.includes("задолж") || domain?.includes("расчет")) { + return [ + "Какие незакрытые расчеты висят на конец периода?", + "Есть ли незакрытые авансы по поставщикам?", + "Покажи цепочки закрытия по счетам 60/62." + ]; + } + return [ + "С какой организацией сейчас можно работать в активном контуре?", + "Покажи ключевые операции за выбранный период.", + "Какие вопросы по этому домену ассистент поддерживает прямо сейчас?" + ]; +} +function mutateIntoQwenStyle(base, index) { + const wrappers = ["йо ", "слушай ", "подскажи плиз ", "короче ", "мож ", "а ну-ка "]; + const tails = ["", " без воды", " по факту", " и коротко", " прям сейчас", " за весь период"]; + const typoMap = [ + [/\bкомпания\b/gi, "компиния"], + [/\bсейчас\b/gi, "щас"], + [/\bпожалуйста\b/gi, "плиз"], + [/\bкакая\b/gi, "кака"], + [/\bчто\b/gi, "че"] + ]; + const prefix = wrappers[index % wrappers.length]; + const tail = tails[index % tails.length]; + let text = `${prefix}${base}${tail}`.trim(); + if (index % 2 === 0) { + const [pattern, replacement] = typoMap[index % typoMap.length]; + text = text.replace(pattern, replacement); + } + return text; +} +function generateQwenSeedQuestions(count, domain) { + const seed = collectCanonicalQuestions(450); + const source = seed.length > 0 ? seed : fallbackDomainTemplates(domain); + const filtered = domain + ? source.filter((item) => item.toLowerCase().includes(domain) || fallbackDomainTemplates(domain).includes(item)) + : source; + const bag = filtered.length > 0 ? filtered : source; + const out = []; + for (let index = 0; index < count; index += 1) { + const base = bag[index % bag.length]; + out.push(mutateIntoQwenStyle(base, index)); + } + return Array.from(new Set(out)).slice(0, count); +} +function generateCodexCreativeQuestions(count, domain) { + const domainTemplates = fallbackDomainTemplates(domain); + const patterns = [ + "Дай бизнес-срез по состоянию на дату: {q}", + "Нужен аккуратный ответ как бухгалтеру: {q}", + "Если данных не хватает, скажи что уточнить, но сначала попробуй: {q}", + "Сформулируй результат без технички и с шагом дальше: {q}", + "Проверь в read-only и скажи что видно: {q}" + ]; + const out = []; + for (let index = 0; index < count; index += 1) { + const base = domainTemplates[index % domainTemplates.length]; + const pattern = patterns[index % patterns.length]; + out.push(pattern.replace("{q}", base)); + } + return Array.from(new Set(out)).slice(0, count); +} +function generateAutogenId() { + return `gen-${Date.now().toString(36)}-${Math.random().toString(36).slice(2, 9)}`; +} +function readAnnotations() { + if (!fs_1.default.existsSync(config_1.AUTORUN_ANNOTATIONS_FILE)) { + return []; + } + try { + const raw = fs_1.default.readFileSync(config_1.AUTORUN_ANNOTATIONS_FILE, "utf-8"); + const parsed = JSON.parse(raw); + if (!Array.isArray(parsed)) { + return []; + } + return parsed + .map((item) => toRecord(item)) + .filter((item) => item !== null) + .map((item) => { + const context = toRecord(item.context); + return { + annotation_id: toStringSafe(item.annotation_id) ?? "", + run_id: toStringSafe(item.run_id) ?? "", + case_id: toStringSafe(item.case_id) ?? "", + session_id: toStringSafe(item.session_id) ?? "", + message_index: clampInt(toNumberSafe(item.message_index), 0, 100_000, 0), + rating: clampInt(toNumberSafe(item.rating), 1, 5, 1), + comment: toStringSafe(item.comment) ?? "", + manual_case_decision: parseManualCaseDecision(item.manual_case_decision), + annotation_author: parseAnnotationAuthor(item.annotation_author), + created_at: toStringSafe(item.created_at) ?? new Date().toISOString(), + updated_at: toStringSafe(item.updated_at) ?? new Date().toISOString(), + context: { + message_id: toStringSafe(context?.message_id), + trace_id: toStringSafe(context?.trace_id), + reply_type: toStringSafe(context?.reply_type), + eval_target: toStringSafe(context?.eval_target) ?? "unknown", + prompt_version: toStringSafe(context?.prompt_version), + domain: toStringSafe(context?.domain), + query_class: toStringSafe(context?.query_class) + } + }; + }) + .filter((item) => item.annotation_id && item.run_id && item.case_id); + } + catch { + return []; + } +} +function writeAnnotations(items) { + fs_1.default.writeFileSync(config_1.AUTORUN_ANNOTATIONS_FILE, JSON.stringify(items, null, 2), "utf-8"); +} +function annotationKey(runId, caseId, messageIndex) { + return `${runId}::${caseId}::${messageIndex}`; +} +function buildAnnotationStatsMap(runId, annotations) { + const scoped = annotations.filter((item) => item.run_id === runId); + const buckets = new Map(); + for (const item of scoped) { + const bucket = buckets.get(item.case_id) ?? { count: 0, ratings: [], latestMs: null }; + bucket.count += 1; + bucket.ratings.push(item.rating); + const ms = Date.parse(item.updated_at); + if (Number.isFinite(ms) && (bucket.latestMs === null || ms > bucket.latestMs)) { + bucket.latestMs = ms; + } + buckets.set(item.case_id, bucket); + } + const result = new Map(); + for (const [caseId, bucket] of buckets.entries()) { + const avg = bucket.ratings.length > 0 ? Number((bucket.ratings.reduce((a, b) => a + b, 0) / bucket.ratings.length).toFixed(2)) : null; + result.set(caseId, { + count: bucket.count, + latest_at: bucket.latestMs === null ? null : new Date(bucket.latestMs).toISOString(), + avg_rating: avg + }); + } + return result; +} +function buildAnnotationsByMessageIndex(runId, caseId, annotations) { + const map = new Map(); + for (const item of annotations) { + if (item.run_id !== runId || item.case_id !== caseId) + continue; + const current = map.get(item.message_index); + const currentMs = current ? Date.parse(current.updated_at) : null; + const nextMs = Date.parse(item.updated_at); + if (!current || (!Number.isNaN(nextMs) && (currentMs === null || nextMs >= currentMs))) { + map.set(item.message_index, item); + } + } + return map; +} function resolveRunTarget(input) { const explicit = toStringSafe(input.report.eval_target); if (explicit === "assistant_stage1" || explicit === "assistant_stage2" || explicit === "assistant_p0" || explicit === "normalizer") { @@ -221,7 +542,7 @@ function getResultCases(report) { .map((item) => toRecord(item)) .filter((item) => item !== null); } -function buildCaseSummaries(report, runId, checkDialogAvailability) { +function buildCaseSummaries(report, runId, checkDialogAvailability, annotationStatsByCase) { const results = getResultCases(report); return results.map((item, index) => { const caseId = toStringSafe(item.case_id) ?? `case-${index + 1}`; @@ -235,6 +556,7 @@ function buildCaseSummaries(report, runId, checkDialogAvailability) { const dialogAvailable = checkDialogAvailability ? fs_1.default.existsSync(path_1.default.resolve(config_1.ASSISTANT_SESSIONS_DIR, `${sessionId}.json`)) : false; + const annotationStats = annotationStatsByCase?.get(caseId); return { case_id: caseId, domain: toStringSafe(item.domain), @@ -245,6 +567,9 @@ function buildCaseSummaries(report, runId, checkDialogAvailability) { reply_type: toStringSafe(item.reply_type), session_id: sessionId, dialog_available: dialogAvailable, + commented_count: annotationStats?.count ?? 0, + latest_annotation_at: annotationStats?.latest_at ?? null, + avg_rating: annotationStats?.avg_rating ?? null, checks, metric_subscores: metricSubscores }; @@ -521,6 +846,7 @@ function loadSessionDialog(runId, caseId) { .map((item) => toRecord(item)) .filter((item) => item !== null); const messages = conversation.map((item) => ({ + message_id: toStringSafe(item.message_id), role: toStringSafe(item.role) ?? "unknown", text: toStringSafe(item.text) ?? "", created_at: toStringSafe(item.created_at), @@ -588,6 +914,7 @@ function buildFallbackDialog(run, caseId) { session_id: sessionId, messages: [ { + message_id: null, role: "user", text: userText, created_at: null, @@ -595,6 +922,7 @@ function buildFallbackDialog(run, caseId) { reply_type: null }, { + message_id: null, role: "assistant", text: assistantSummaryParts.join("\n"), created_at: null, @@ -606,6 +934,175 @@ function buildFallbackDialog(run, caseId) { assistant_mode: null }; } +function withMessageAnnotations(runId, caseId, messages, annotations) { + const byIndex = buildAnnotationsByMessageIndex(runId, caseId, annotations); + return messages.map((message, index) => { + const annotation = byIndex.get(index) ?? null; + return { + ...message, + message_index: index, + commented: annotation !== null, + annotation + }; + }); +} +function generateAnnotationId() { + return `ann-${Date.now().toString(36)}-${Math.random().toString(36).slice(2, 9)}`; +} +function parseComment(value) { + const text = toStringSafe(value) ?? ""; + return text.trim(); +} +function parseDecisionFilter(value) { + const normalized = toStringSafe(value); + if (!normalized || normalized === "all") + return "all"; + return parseManualCaseDecision(normalized); +} +function parseAutoGenMode(value) { + const normalized = toStringSafe(value)?.toLowerCase() ?? ""; + if (normalized === "qwen_seed" || normalized === "codex_creative") { + return normalized; + } + return "codex_creative"; +} +function parseAutogenCount(value) { + return clampInt(toNumberSafe(value), 1, 200, 24); +} +function parseAutogenDomain(value) { + const domain = normalizeDomainHint(value); + if (!domain) + return null; + return domain.slice(0, 80); +} +function hasAnyRunFilterQuery(query) { + return Boolean(toStringSafe(query.from) ?? + toStringSafe(query.to) ?? + toStringSafe(query.target) ?? + toStringSafe(query.mode) ?? + toStringSafe(query.use_mock) ?? + toStringSafe(query.prompt_contains)); +} +function buildAutogenCaseSetFileName(mode, generationId) { + const now = new Date(); + const stamp = [ + now.getUTCFullYear(), + String(now.getUTCMonth() + 1).padStart(2, "0"), + String(now.getUTCDate()).padStart(2, "0"), + String(now.getUTCHours()).padStart(2, "0"), + String(now.getUTCMinutes()).padStart(2, "0"), + String(now.getUTCSeconds()).padStart(2, "0") + ].join(""); + return `assistant_autogen_${mode}_${stamp}_${generationId}.json`; +} +function buildAutogenCaseSetPayload(input) { + const cases = input.questions.map((question, index) => ({ + case_id: `AUTO-${String(index + 1).padStart(3, "0")}`, + scenario_tag: `${input.mode}_${input.domain ?? "general"}`, + question_type: "direct", + broadness_level: "medium", + turns: [{ user_message: question }], + expected_hints: { + expected_reply_type: null, + expected_degraded_to: null + } + })); + return { + suite_id: `assistant_autogen_${input.generationId}`, + suite_version: "0.1.0", + schema_version: "assistant_autogen_suite_v0_1", + generated_at: new Date().toISOString(), + generation_id: input.generationId, + mode: input.mode, + domain: input.domain, + scenario_count: cases.length, + case_ids: cases.map((item) => item.case_id), + cases + }; +} +function collectPostAnalysis(annotations, runMap, limitPerQueue) { + const byDecision = {}; + const byQueue = {}; + const byDomain = new Map(); + const queues = { + routing_extension: [], + policy_fix: [], + capability_registry: [], + soft_boundary: [], + safety_policy: [], + testset_hygiene: [], + covered_ok: [] + }; + const registry = (0, capabilitiesRegistry_1.loadCapabilitiesRegistry)(); + for (const item of annotations) { + byDecision[item.manual_case_decision] = (byDecision[item.manual_case_decision] ?? 0) + 1; + const queueKey = DECISION_QUEUE_MAP[item.manual_case_decision]; + byQueue[queueKey] = (byQueue[queueKey] ?? 0) + 1; + const run = runMap.get(item.run_id) ?? null; + const caseSummary = run + ? buildCaseSummaries(run.report, run.run_id, false).find((candidate) => candidate.case_id === item.case_id) ?? null + : null; + const nearestGroup = (0, capabilitiesRegistry_1.resolveNearestCapabilityGroup)({ + domain: caseSummary?.domain ?? item.context.domain, + queryClass: caseSummary?.query_class ?? item.context.query_class + }) ?? + registry.groups[0] ?? + null; + const domainKey = caseSummary?.domain ?? item.context.domain ?? "unknown"; + byDomain.set(domainKey, (byDomain.get(domainKey) ?? 0) + 1); + const view = { + annotation_id: item.annotation_id, + run_id: item.run_id, + case_id: item.case_id, + message_index: item.message_index, + rating: item.rating, + comment: item.comment, + manual_case_decision: item.manual_case_decision, + annotation_author: item.annotation_author, + updated_at: item.updated_at, + domain: caseSummary?.domain ?? item.context.domain ?? null, + query_class: caseSummary?.query_class ?? item.context.query_class ?? null, + trace_id: item.context.trace_id ?? caseSummary?.trace_id ?? null, + reply_type: item.context.reply_type ?? caseSummary?.reply_type ?? null, + nearest_capability_group: nearestGroup + ? { + group_code: nearestGroup.group_code, + group_title: nearestGroup.group_title, + maturity_status: nearestGroup.maturity_status + } + : null + }; + if (queueKey === "none") { + if (queues.covered_ok.length < limitPerQueue) + queues.covered_ok.push(view); + continue; + } + if (!queues[queueKey]) { + queues[queueKey] = []; + } + if (queues[queueKey].length < limitPerQueue) { + queues[queueKey].push(view); + } + } + const domainSummary = Array.from(byDomain.entries()) + .map(([domain, total]) => ({ domain, total })) + .sort((a, b) => b.total - a.total); + return { + stats: { + annotations_total: annotations.length, + by_decision: byDecision, + by_queue: byQueue, + domains_total: domainSummary.length + }, + domain_summary: domainSummary, + queues, + recommended_regression_candidates: [ + ...queues.routing_extension.slice(0, 20), + ...queues.policy_fix.slice(0, 20), + ...queues.safety_policy.slice(0, 20) + ].slice(0, 60) + }; +} function buildAutoRunsRouter() { const router = (0, express_1.Router)(); router.get("/api/autoruns/history", (req, res) => { @@ -648,13 +1145,18 @@ function buildAutoRunsRouter() { if (!run) { throw new http_1.ApiError("AUTORUN_NOT_FOUND", `Run not found: ${runId}`, 404); } - const cases = buildCaseSummaries(run.report, run.run_id, true); + const annotations = readAnnotations(); + const annotationStatsByCase = buildAnnotationStatsMap(runId, annotations); + const cases = buildCaseSummaries(run.report, run.run_id, true, annotationStatsByCase); const coverage = buildCoverageFromCases(cases); (0, http_1.ok)(res, { ok: true, run: buildRunSummary(run), coverage, cases, + annotations_summary: { + total: annotations.filter((item) => item.run_id === runId).length + }, report: run.report }); } @@ -675,11 +1177,302 @@ function buildAutoRunsRouter() { } const sessionDialog = loadSessionDialog(runId, caseId); const dialog = sessionDialog ?? buildFallbackDialog(run, caseId); + const annotations = readAnnotations(); + const messages = withMessageAnnotations(runId, caseId, dialog.messages, annotations); (0, http_1.ok)(res, { ok: true, run_id: runId, case_id: caseId, - ...dialog + ...dialog, + messages, + annotations: annotations + .filter((item) => item.run_id === runId && item.case_id === caseId) + .sort((a, b) => Date.parse(b.updated_at) - Date.parse(a.updated_at)) + }); + } + catch (error) { + next(error); + } + }); + router.get("/api/autoruns/annotations", (req, res, next) => { + try { + const runIdFilter = toStringSafe(req.query.run_id); + const caseIdFilter = toStringSafe(req.query.case_id); + const minRatingRaw = toNumberSafe(req.query.min_rating); + const minRating = minRatingRaw === null ? null : clampInt(minRatingRaw, 1, 5, 1); + const decisionFilter = parseDecisionFilter(req.query.manual_case_decision); + const limit = clampInt(toNumberSafe(req.query.limit), 1, 2000, 400); + const scanLimit = clampInt(toNumberSafe(req.query.scan_limit), 50, 5000, 2500); + const annotations = readAnnotations() + .filter((item) => (runIdFilter ? item.run_id === runIdFilter : true)) + .filter((item) => (caseIdFilter ? item.case_id === caseIdFilter : true)) + .filter((item) => (minRating === null ? true : item.rating >= minRating)) + .filter((item) => (decisionFilter === "all" ? true : item.manual_case_decision === decisionFilter)) + .sort((a, b) => Date.parse(b.updated_at) - Date.parse(a.updated_at)) + .slice(0, limit); + const runIndex = indexRuns(scanLimit); + const runMap = new Map(runIndex.map((item) => [item.run_id, item])); + const items = annotations.map((item) => { + const run = runMap.get(item.run_id) ?? null; + const runSummary = run ? buildRunSummary(run) : null; + const cases = run ? buildCaseSummaries(run.report, run.run_id, false) : []; + const caseSummary = cases.find((candidate) => candidate.case_id === item.case_id) ?? null; + return { + ...item, + run: runSummary, + case_summary: caseSummary, + technical_context: { + report_path: run?.report_path ?? null, + trace_id: item.context.trace_id, + reply_type: item.context.reply_type, + domain: item.context.domain, + query_class: item.context.query_class, + checks: caseSummary?.checks ?? null, + metric_subscores: caseSummary?.metric_subscores ?? null + } + }; + }); + const avgRating = items.length > 0 ? Number((items.reduce((acc, item) => acc + item.rating, 0) / items.length).toFixed(2)) : null; + const byDecision = items.reduce((acc, item) => { + acc[item.manual_case_decision] = (acc[item.manual_case_decision] ?? 0) + 1; + return acc; + }, {}); + (0, http_1.ok)(res, { + ok: true, + generated_at: new Date().toISOString(), + filters_applied: { + run_id: runIdFilter ?? null, + case_id: caseIdFilter ?? null, + min_rating: minRating, + manual_case_decision: decisionFilter, + limit + }, + stats: { + total: items.length, + avg_rating: avgRating, + by_decision: byDecision + }, + available_manual_case_decisions: MANUAL_CASE_DECISIONS, + manual_case_decision_schema: readManualDecisionSchema(), + items + }); + } + catch (error) { + next(error); + } + }); + router.post("/api/autoruns/annotations", (req, res, next) => { + try { + const body = toRecord(req.body); + if (!body) { + throw new http_1.ApiError("INVALID_ANNOTATION_PAYLOAD", "JSON body is required", 400); + } + const runId = toStringSafe(body.run_id); + const caseId = toStringSafe(body.case_id); + const messageIndexRaw = toNumberSafe(body.message_index); + const ratingRaw = toNumberSafe(body.rating); + const comment = parseComment(body.comment); + const manualCaseDecision = parseManualCaseDecision(body.manual_case_decision); + const annotationAuthor = parseAnnotationAuthor(body.annotation_author); + if (!runId || !caseId) { + throw new http_1.ApiError("INVALID_ANNOTATION_PAYLOAD", "run_id and case_id are required", 400); + } + if (messageIndexRaw === null) { + throw new http_1.ApiError("INVALID_ANNOTATION_PAYLOAD", "message_index is required", 400); + } + const messageIndex = clampInt(messageIndexRaw, 0, 100_000, 0); + if (ratingRaw === null) { + throw new http_1.ApiError("INVALID_ANNOTATION_PAYLOAD", "rating is required", 400); + } + const rating = clampInt(ratingRaw, 1, 5, 1); + if (comment.length === 0) { + throw new http_1.ApiError("INVALID_ANNOTATION_PAYLOAD", "comment is required", 400); + } + const run = findRunById(runId); + if (!run) { + throw new http_1.ApiError("AUTORUN_NOT_FOUND", `Run not found: ${runId}`, 404); + } + const cases = buildCaseSummaries(run.report, run.run_id, false); + const caseSummary = cases.find((item) => item.case_id === caseId) ?? null; + if (!caseSummary) { + throw new http_1.ApiError("AUTORUN_CASE_NOT_FOUND", `Case not found: ${caseId} in run ${runId}`, 404); + } + const sessionDialog = loadSessionDialog(runId, caseId); + const dialog = sessionDialog ?? buildFallbackDialog(run, caseId); + if (messageIndex >= dialog.messages.length) { + throw new http_1.ApiError("AUTORUN_MESSAGE_NOT_FOUND", `Message index ${messageIndex} out of range`, 400); + } + const targetMessage = dialog.messages[messageIndex]; + const targetRole = toStringSafe(targetMessage.role) ?? "unknown"; + if (targetRole !== "assistant") { + throw new http_1.ApiError("AUTORUN_MESSAGE_NOT_ASSISTANT", "Only assistant answers can be annotated", 400); + } + const nowIso = new Date().toISOString(); + const annotations = readAnnotations(); + const key = annotationKey(runId, caseId, messageIndex); + const existingIndex = annotations.findIndex((item) => annotationKey(item.run_id, item.case_id, item.message_index) === key); + const existing = existingIndex >= 0 ? annotations[existingIndex] : null; + const annotation = { + annotation_id: existing?.annotation_id ?? generateAnnotationId(), + run_id: runId, + case_id: caseId, + session_id: caseSummary.session_id, + message_index: messageIndex, + rating, + comment, + manual_case_decision: manualCaseDecision, + annotation_author: annotationAuthor, + created_at: existing?.created_at ?? nowIso, + updated_at: nowIso, + context: { + message_id: toStringSafe(targetMessage.message_id), + trace_id: toStringSafe(targetMessage.trace_id) ?? caseSummary.trace_id, + reply_type: toStringSafe(targetMessage.reply_type) ?? caseSummary.reply_type, + eval_target: run.eval_target, + prompt_version: toStringSafe(run.report.prompt_version), + domain: caseSummary.domain, + query_class: caseSummary.query_class + } + }; + if (existingIndex >= 0) { + annotations[existingIndex] = annotation; + } + else { + annotations.push(annotation); + } + writeAnnotations(annotations); + const annotationStatsByCase = buildAnnotationStatsMap(runId, annotations); + const caseStats = annotationStatsByCase.get(caseId) ?? null; + (0, http_1.ok)(res, { + ok: true, + annotation, + case_annotation_stats: caseStats + }); + } + catch (error) { + next(error); + } + }); + router.get("/api/autoruns/manual-decision-schema", (_req, res) => { + (0, http_1.ok)(res, { + ok: true, + schema: readManualDecisionSchema(), + enum: MANUAL_CASE_DECISIONS + }); + }); + router.get("/api/autoruns/post-analysis", (req, res, next) => { + try { + const query = req.query; + const runIdFilter = toStringSafe(query.run_id); + const limitPerQueue = clampInt(toNumberSafe(query.limit_per_queue), 5, 250, 40); + const annotationLimit = clampInt(toNumberSafe(query.annotation_limit), 20, 5000, 1500); + const scanLimit = clampInt(toNumberSafe(query.scan_limit), 50, 5000, 2500); + const runFilters = parseFilters(query); + const applyRunFilters = hasAnyRunFilterQuery(query); + const runIndex = indexRuns(Math.max(scanLimit, runFilters.scan_limit)); + const filteredRuns = applyRunFilters ? runIndex.filter((run) => matchesFilters(run, runFilters)) : runIndex; + const runMap = new Map(filteredRuns.map((run) => [run.run_id, run])); + const scopedAnnotations = readAnnotations() + .filter((item) => (runIdFilter ? item.run_id === runIdFilter : true)) + .filter((item) => (runMap.size > 0 ? runMap.has(item.run_id) : true)) + .sort((a, b) => Date.parse(b.updated_at) - Date.parse(a.updated_at)) + .slice(0, annotationLimit); + const analysis = collectPostAnalysis(scopedAnnotations, runMap, limitPerQueue); + (0, http_1.ok)(res, { + ok: true, + generated_at: new Date().toISOString(), + filters_applied: { + run_id: runIdFilter ?? null, + run_filters_applied: applyRunFilters, + limit_per_queue: limitPerQueue, + annotation_limit: annotationLimit, + scan_limit: scanLimit + }, + runs_considered: filteredRuns.slice(0, 500).map((item) => buildRunSummary(item)), + manual_case_decision_schema: readManualDecisionSchema(), + post_analysis: analysis + }); + } + catch (error) { + next(error); + } + }); + router.get("/api/autoruns/autogen/history", (req, res, next) => { + try { + const limit = clampInt(toNumberSafe(req.query.limit), 1, 500, 120); + const rawMode = toStringSafe(req.query.mode); + const includeAllModes = !rawMode || !["qwen_seed", "codex_creative"].includes(rawMode); + const modeFilter = rawMode ?? "codex_creative"; + const items = readAutoGenHistory() + .filter((item) => (includeAllModes ? true : item.mode === modeFilter)) + .slice(0, limit); + (0, http_1.ok)(res, { + ok: true, + generated_at: new Date().toISOString(), + items + }); + } + catch (error) { + next(error); + } + }); + router.post("/api/autoruns/autogen/generate", (req, res, next) => { + try { + const body = toRecord(req.body); + if (!body) { + throw new http_1.ApiError("INVALID_AUTOGEN_PAYLOAD", "JSON body is required", 400); + } + const mode = parseAutoGenMode(body.mode); + const count = parseAutogenCount(body.count); + const domain = parseAutogenDomain(body.domain); + const persistCaseSet = toBooleanSafe(body.persist_to_eval_cases) ?? true; + const generatedBy = parseAnnotationAuthor(body.generated_by); + const context = toRecord(body.context); + const questions = mode === "qwen_seed" + ? generateQwenSeedQuestions(count, domain) + : generateCodexCreativeQuestions(count, domain); + const generationId = generateAutogenId(); + let savedCaseSetFile = null; + if (persistCaseSet) { + if (!fs_1.default.existsSync(config_1.EVAL_CASES_DIR)) { + fs_1.default.mkdirSync(config_1.EVAL_CASES_DIR, { recursive: true }); + } + const fileName = buildAutogenCaseSetFileName(mode, generationId); + const filePath = path_1.default.resolve(config_1.EVAL_CASES_DIR, fileName); + const payload = buildAutogenCaseSetPayload({ + generationId, + mode, + domain, + questions + }); + fs_1.default.writeFileSync(filePath, JSON.stringify(payload, null, 2), "utf-8"); + savedCaseSetFile = fileName; + } + const record = { + generation_id: generationId, + created_at: new Date().toISOString(), + mode, + count: questions.length, + domain, + questions, + generated_by: generatedBy, + saved_case_set_file: savedCaseSetFile, + context: context + ? { + llm_provider: toStringSafe(context.llm_provider), + model: toStringSafe(context.model), + assistant_prompt_version: toStringSafe(context.assistant_prompt_version), + decomposition_prompt_version: toStringSafe(context.decomposition_prompt_version), + prompt_fingerprint: toStringSafe(context.prompt_fingerprint) + } + : null + }; + const history = readAutoGenHistory(); + history.unshift(record); + writeAutoGenHistory(history.slice(0, 500)); + (0, http_1.ok)(res, { + ok: true, + generation: record }); } catch (error) { diff --git a/llm_normalizer/backend/dist/server.js b/llm_normalizer/backend/dist/server.js index 1eaa096..3fa8348 100644 --- a/llm_normalizer/backend/dist/server.js +++ b/llm_normalizer/backend/dist/server.js @@ -31,6 +31,8 @@ function createApp() { (0, files_1.ensureDir)(config_1.EVAL_CASES_DIR); (0, files_1.ensureDir)(config_1.REPORTS_DIR); (0, files_1.ensureDir)(config_1.ASSISTANT_SESSIONS_DIR); + (0, files_1.ensureDir)(config_1.AUTORUN_ANNOTATIONS_DIR); + (0, files_1.ensureDir)(config_1.AUTORUN_GENERATOR_DIR); const app = (0, express_1.default)(); app.use((0, cors_1.default)()); app.use(express_1.default.json({ type: ["application/json", "application/*+json"], limit: "2mb" })); diff --git a/llm_normalizer/backend/dist/services/answerComposer.js b/llm_normalizer/backend/dist/services/answerComposer.js index ed7c681..63c40a5 100644 --- a/llm_normalizer/backend/dist/services/answerComposer.js +++ b/llm_normalizer/backend/dist/services/answerComposer.js @@ -1158,29 +1158,29 @@ function buildProblemCentricActions(input) { actions.push("Проверьте зачет аванса или взаимозачет и связку платежа с закрытием расчета."); } if (unitTypes.has("broken_chain_segment")) { - actions.push("Проверьте СЃРІСЏР·РєСѓ выписка -> документ -> РїСЂРѕРІРѕРґРєР° РїРѕ проблемным участкам цепочки."); + actions.push("Проверьте связку выписка -> документ -> проводка по проблемным участкам цепочки."); } if (unitTypes.has("unresolved_settlement_cluster")) { - actions.push("Сверьте хвосты РїРѕ расчетам: закрылся ли документ оплаты корректным закрывающим документом."); + actions.push("Сверьте хвосты по расчетам: закрылся ли документ оплаты корректным закрывающим документом."); } if (unitTypes.has("period_risk_cluster")) { - actions.push("Оцените влияние дефекта РЅР° закрытие периода Рё корректность регламентных операций."); + actions.push("Оцените влияние дефекта на закрытие периода и корректность регламентных операций."); } if (unitTypes.has("cross_branch_inconsistency_cluster")) { - actions.push("Сверьте противоречия между документами, проводками Рё регистрами РїРѕ НДС/межконтурным СЃРІСЏР·СЏРј."); + actions.push("Сверьте противоречия между документами, проводками и регистрами по НДС/межконтурным связям."); } if (unitTypes.has("lifecycle_anomaly_node")) { - actions.push("Проверьте lifecycle объекта: ожидаемый этап РЅРµ должен оставаться РІ partially_linked состоянии."); + actions.push("Проверьте lifecycle объекта: ожидаемый этап не должен оставаться в partially_linked состоянии."); } for (const unit of input.units) { if (unit.lifecycle_defect_type === "stale_active_state") { - actions.push("Проверьте, почему объект завис: ожидаемый переход РЅРµ должен оставаться РІ активной стадии."); + actions.push("Проверьте, почему объект завис: ожидаемый переход не должен оставаться в активной стадии."); } if (unit.lifecycle_defect_type === "misclosed_state") { - actions.push("Проверьте закрывающий документ Рё РїСЂРѕРІРѕРґРєРё: закрытие может быть формальным, РЅРѕ некорректным РїРѕ пути."); + actions.push("Проверьте закрывающий документ и проводки: закрытие может быть формальным, но некорректным по пути."); } if (unit.lifecycle_defect_type === "cross_branch_state_conflict") { - actions.push("Сверьте бухгалтерскую Рё смежную ветки (например, НДС/расчеты): обнаружен межконтурный конфликт состояния."); + actions.push("Сверьте бухгалтерскую и смежную ветки (например, НДС/расчеты): обнаружен межконтурный конфликт состояния."); } } if (input.missingAnchors.period && input.mode !== "clarification_required") { @@ -1188,20 +1188,20 @@ function buildProblemCentricActions(input) { } if (input.mode === "clarification_required") { if (input.missingAnchors.period) { - actions.push("Уточните период проверки, чтобы зафиксировать границы проблемного контура."); + actions.push("Уточните период проверки, чтобы зафиксировать границы проблемного контура."); } if (input.missingAnchors.account) { - actions.push("Уточните счет или РіСЂСѓРїРїСѓ счетов для предметной локализации дефекта."); + actions.push("Уточните счет или группу счетов для предметной локализации дефекта."); } if (input.missingAnchors.documentOrObject) { - actions.push("Укажите конкретный документ или объект трассировки для проверки механизма отклонения."); + actions.push("Укажите конкретный документ или объект трассировки для проверки механизма отклонения."); } if (input.missingAnchors.counterparty) { - actions.push("Укажите контрагента/РґРѕРіРѕРІРѕСЂ, чтобы проверить хвосты Рё разрывы РЅР° конкретной СЃРІСЏР·РєРµ."); + actions.push("Укажите контрагента/договор, чтобы проверить хвосты и разрывы на конкретной связке."); } } if (input.coverageReport.requirements_uncovered.length > 0) { - actions.push(`Закройте непокрытые требования: ${input.coverageReport.requirements_uncovered.join(", ")}.`); + actions.push(`Закройте непокрытые требования: ${input.coverageReport.requirements_uncovered.join(", ")}.`); } return uniqueStrings(actions, 6); } @@ -1215,25 +1215,25 @@ function buildProblemCentricClarifications(input) { questions.push("Уточните период (например, июль 2020), в котором нужно проверить проблемный кластер."); } if (input.missingAnchors.account) { - questions.push("Уточните счет или СЃРІСЏР·РєСѓ счетов (например, 51/60), РіРґРµ РІС‹ ожидаете дефект."); + questions.push("Уточните счет или связку счетов (например, 51/60), где вы ожидаете дефект."); } if (input.missingAnchors.documentOrObject) { - questions.push("Укажите документ/объект, РѕС‚ которого РЅСѓР¶РЅРѕ строить проверку цепочки."); + questions.push("Укажите документ/объект, от которого нужно строить проверку цепочки."); } if (input.missingAnchors.counterparty) { - questions.push("Укажите контрагента или РґРѕРіРѕРІРѕСЂ, РїРѕ которому проверить незакрытую экспозицию."); + questions.push("Укажите контрагента или договор, по которому проверить незакрытую экспозицию."); } if (unitTypes.has("broken_chain_segment")) { - questions.push("Уточните участок цепочки: выписка, платежный документ или РїСЂРѕРІРѕРґРєР°."); + questions.push("Уточните участок цепочки: выписка, платежный документ или проводка."); } if (unitTypes.has("period_risk_cluster")) { - questions.push("Уточните, какой этап закрытия периода критичен: начисление, закрытие счетов или НДС-блок."); + questions.push("Уточните, какой этап закрытия периода критичен: начисление, закрытие счетов или НДС-блок."); } if (unitTypes.has("unresolved_settlement_cluster")) { - questions.push("Уточните, интересуют хвосты поставщиков, покупателей или РѕР±Р° направления."); + questions.push("Уточните, интересуют хвосты поставщиков, покупателей или оба направления."); } if (input.coverageReport.clarification_needed_for.length > 0) { - questions.push(`Закройте уточнения для требований: ${input.coverageReport.clarification_needed_for.join(", ")}.`); + questions.push(`Закройте уточнения для требований: ${input.coverageReport.clarification_needed_for.join(", ")}.`); } return uniqueStrings(questions, 6); } @@ -1402,10 +1402,10 @@ function detectMissingAnchors(userMessage, retrievalResults = [], options) { hasPeriodAnchorInRetrieval(retrievalResults) || Boolean(options?.normalizationPeriodExplicit) || hasPeriodAnchorInCompanyAnchors(options?.companyAnchors); - const hasAccount = /(?:\bсчет\b|\baccount\b|\bschet\b|\b(?:0[1-9]|[1-9]\d)(?:\.\d{2})?\b|\b(?:60|62)\.\d{2}\s*\/\s*(?:60|62)\.\d{2}\b)/i.test(lower) || hasAccountAnchorInRetrieval(retrievalResults); - const hasDocumentOrObject = /(?:документ|invoice|guid|object|obj|#\d+|\b№\s*[a-zа-я0-9-]+\b|\bid\b|\bref\b|dokument|doc)/i.test(lower); - const hasCounterparty = /(?:контрагент|supplier|buyer|customer|kontragent|postavsh|pokupatel|договор|contract)/i.test(lower); - const hasAnomalyType = /(?:аномал|risk|отклон|разрыв|mismatch|duplicate|tail|цепочк|anomali|hvost)/i.test(lower); + const hasAccount = /(?:\bсчет\b|\baccount\b|\bschet\b|\b(?:0[1-9]|[1-9]\d)(?:\.\d{2})?\b|\b(?:60|62)\.\d{2}\s*\/\s*(?:60|62)\.\d{2}\b)/i.test(lower) || hasAccountAnchorInRetrieval(retrievalResults); + const hasDocumentOrObject = /(?:документ|invoice|guid|object|obj|#\d+|\b№\s*[a-zа-я0-9-]+\b|\bid\b|\bref\b|dokument|doc)/i.test(lower); + const hasCounterparty = /(?:контрагент|supplier|buyer|customer|kontragent|postavsh|pokupatel|договор|contract)/i.test(lower); + const hasAnomalyType = /(?:аномал|risk|отклон|разрыв|mismatch|duplicate|tail|цепочк|anomali|hvost)/i.test(lower); return { period: !hasPeriod, account: !hasAccount, @@ -1424,50 +1424,50 @@ function buildClarificationQuestions(input) { questions.push("Уточните период проверки (например, июль 2020)."); } if (input.missingAnchors.account) { - questions.push("Уточните счет или РіСЂСѓРїРїСѓ счетов (например, 19, 60, 62)."); + questions.push("Уточните счет или группу счетов (например, 19, 60, 62)."); } if (input.missingAnchors.documentOrObject) { - questions.push("Укажите документ/GUID/конкретный объект для трассировки."); + questions.push("Укажите документ/GUID/конкретный объект для трассировки."); } if (input.missingAnchors.counterparty) { - questions.push("Укажите контрагента или РіСЂСѓРїРїСѓ контрагентов."); + questions.push("Укажите контрагента или группу контрагентов."); } if (input.policySignals.broad_query_detected && input.missingAnchors.anomalyType) { - questions.push("Уточните тип отклонения: разрыв цепочки, неверный документ или аномальный СЂРёСЃРє."); + questions.push("Уточните тип отклонения: разрыв цепочки, неверный документ или аномальный риск."); } if (input.coverageReport.clarification_needed_for.length > 0) { - questions.push(`Закройте уточнения для требований: ${input.coverageReport.clarification_needed_for.join(", ")}.`); + questions.push(`Закройте уточнения для требований: ${input.coverageReport.clarification_needed_for.join(", ")}.`); } return uniqueStrings(questions, 6); } function buildRecommendedActions(input) { const actions = []; if (input.mode === "focused_grounded") { - actions.push("Проверьте 1-2 ключевые записи РІ учетной базе Рё зафиксируйте итог РІ рабочем файле проверки."); + actions.push("Проверьте 1-2 ключевые записи в учетной базе и зафиксируйте итог в рабочем файле проверки."); } if (input.mode === "broad_partial") { - actions.push("Сузьте запрос РґРѕ периода + счета или периода + документа Рё повторите проверку."); + actions.push("Сузьте запрос до периода + счета или периода + документа и повторите проверку."); } if (input.mode === "clarification_required") { - actions.push("Дайте недостающие СЏРєРѕСЂСЏ (период/счет/объект), иначе сильный factual вывод невозможен."); + actions.push("Дайте недостающие якоря (период/счет/объект), иначе сильный factual вывод невозможен."); } if (input.coverageReport.requirements_uncovered.length > 0) { - actions.push(`Закройте непокрытые требования: ${input.coverageReport.requirements_uncovered.join(", ")}.`); + actions.push(`Закройте непокрытые требования: ${input.coverageReport.requirements_uncovered.join(", ")}.`); } if (input.coverageReport.requirements_partially_covered.length > 0) { - actions.push(`Доуточните частично покрытые требования: ${input.coverageReport.requirements_partially_covered.join(", ")}.`); + actions.push(`Доуточните частично покрытые требования: ${input.coverageReport.requirements_partially_covered.join(", ")}.`); } if (input.policySignals.broad_query_detected && input.policySignals.narrowing_strength !== "strong") { - actions.push("Добавьте более СѓР·РєРёР№ контекст: тип отклонения, РіСЂСѓРїРїСѓ документов Рё бизнес-участок."); + actions.push("Добавьте более узкий контекст: тип отклонения, группу документов и бизнес-участок."); } if (input.limitationReasonCodes.includes("snapshot_only")) { - actions.push("Сверьте критичные выводы СЃ live source-of-record РІ 1C."); + actions.push("Сверьте критичные выводы с live source-of-record в 1C."); } if (input.limitationReasonCodes.includes("weak_source_mapping")) { - actions.push("Проверьте source mapping для связей document/register РїРѕ указанным ref."); + actions.push("Проверьте source mapping для связей document/register по указанным ref."); } if (input.sourceRefs.length > 0) { - actions.push(`Начните проверку СЃ ${input.sourceRefs.length} подтвержденных записей Рё сверьте РёС… СЃ первичными документами.`); + actions.push(`Начните проверку с ${input.sourceRefs.length} подтвержденных записей и сверьте их с первичными документами.`); } return uniqueStrings(actions, 6); } diff --git a/llm_normalizer/backend/dist/services/assistantCanon.js b/llm_normalizer/backend/dist/services/assistantCanon.js new file mode 100644 index 0000000..e983b99 --- /dev/null +++ b/llm_normalizer/backend/dist/services/assistantCanon.js @@ -0,0 +1,41 @@ +"use strict"; +var __importDefault = (this && this.__importDefault) || function (mod) { + return (mod && mod.__esModule) ? mod : { "default": mod }; +}; +Object.defineProperty(exports, "__esModule", { value: true }); +exports.loadAssistantCanonExcerpt = loadAssistantCanonExcerpt; +const fs_1 = __importDefault(require("fs")); +const config_1 = require("../config"); +const FALLBACK_CANON = [ + "Не выдумывай возможности.", + "Не обещай настройку 1С и админ-действия.", + "Не показывай внутренние технические термины пользователю.", + "Говори по-человечески и предлагай ближайший полезный поддерживаемый шаг." +].join(" "); +let cache = null; +function stripMarkdown(input) { + return input + .replace(/^#{1,6}\s+/gm, "") + .replace(/[`*_>\-\[\]\(\)]/g, " ") + .replace(/\s+/g, " ") + .trim(); +} +function loadAssistantCanonExcerpt(maxChars = 900) { + try { + const mtimeMs = fs_1.default.existsSync(config_1.ASSISTANT_CANON_FILE) ? fs_1.default.statSync(config_1.ASSISTANT_CANON_FILE).mtimeMs : -1; + if (cache && cache.mtimeMs === mtimeMs) { + return cache.excerpt; + } + if (!fs_1.default.existsSync(config_1.ASSISTANT_CANON_FILE)) { + return FALLBACK_CANON; + } + const raw = fs_1.default.readFileSync(config_1.ASSISTANT_CANON_FILE, "utf-8"); + const normalized = stripMarkdown(raw); + const excerpt = normalized.length > maxChars ? `${normalized.slice(0, maxChars).trim()}...` : normalized; + cache = { mtimeMs, excerpt: excerpt || FALLBACK_CANON }; + return cache.excerpt; + } + catch { + return cache?.excerpt ?? FALLBACK_CANON; + } +} diff --git a/llm_normalizer/backend/dist/services/assistantDataLayer.js b/llm_normalizer/backend/dist/services/assistantDataLayer.js index 0e3f99f..0e1701f 100644 --- a/llm_normalizer/backend/dist/services/assistantDataLayer.js +++ b/llm_normalizer/backend/dist/services/assistantDataLayer.js @@ -536,8 +536,8 @@ const P0_DOMAIN_CARDS = [ /закрыт[а-яё]*\s+период/i, /close\s+operation/i, /allocation/i, - /закр/i, - /перио/i, + /закр/i, + /перио/i, /\u0437\u0430\u043a\u0440\u044b\u0442(?:\u0438|\u0438\u0435|\u044b|)\s*(?:\u043c\u0435\u0441\u044f\u0446|\u0441\u0447\u0435\u0442)/i, /\u0440\u0435\u0433\u043b\u0430\u043c\u0435\u043d\u0442/i, /\u0437\u0430\u0442\u0440\u0430\u0442/i, @@ -559,14 +559,14 @@ function parseDateCandidate(value) { } function extractDate(record) { const attrs = record.attributes ?? {}; - const directKeys = ["Period", "Date", "Дата", "Период", "ДатаСобытия"]; + const directKeys = ["Period", "Date", "Дата", "Период", "ДатаСобытия"]; for (const key of directKeys) { if (attrs[key] !== undefined && attrs[key] !== null) { return String(attrs[key]); } } for (const [key, value] of Object.entries(attrs)) { - if (/period|date|дата|период/i.test(key) && typeof value === "string" && value.trim()) { + if (/period|date|дата|период/i.test(key) && typeof value === "string" && value.trim()) { return value; } } @@ -593,7 +593,7 @@ function countNavigationLinks(record) { return count; } function findCounterpartyLinks(record) { - return record.links.filter((link) => link.target_entity === "Counterparty" || /supplier|buyer|counterparty/i.test(link.relation) || /постав|РїРѕРєСѓРї/i.test(link.source_field)); + return record.links.filter((link) => link.target_entity === "Counterparty" || /supplier|buyer|counterparty/i.test(link.relation) || /постав|покуп/i.test(link.source_field)); } function extractGuids(text) { const matches = text.match(/[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}/gi) ?? []; @@ -1305,14 +1305,14 @@ function inferPeriodScope(fragmentText) { granularity: "year" }; } - if (/квартал|quarter/i.test(fragmentText)) { + if (/квартал|quarter/i.test(fragmentText)) { return { from: null, to: null, granularity: "quarter" }; } - if (/месяц|month|период/i.test(fragmentText)) { + if (/месяц|month|период/i.test(fragmentText)) { return { from: null, to: null, @@ -1378,7 +1378,7 @@ function buildSemanticRetrievalProfile(fragmentText) { const excludedInterpretations = []; const rankingBasis = ["closure_risk", "repeatability", "financial_impact"]; const explanationFocus = ["why_selected", "where_chain_breaks", "what_business_risk"]; - if (/банк|выписк|расчетн|платеж|банк|выписк|расчетн|платеж|bank|payment|statement|platezh|vypisk/i.test(lower)) { + if (/банк|выписк|расчетн|платеж|банк|выписк|расчетн|платеж|bank|payment|statement|platezh|vypisk/i.test(lower)) { pushMany(domainScope, ["bank", "settlements"]); pushMany(documentTypes, ["bank_statement", "payment_order", "settlement_document"]); pushMany(entityTypes, ["counterparty", "contract", "document", "posting"]); @@ -1390,51 +1390,51 @@ function buildSemanticRetrievalProfile(fragmentText) { const hasDeferredExpenseAccountScope = accountScope.some((item) => item === "97"); const hasMonthCloseCostsAccountScope = accountScope.some((item) => CLOSE_COST_ACCOUNTS.includes(item)); const hasExplicitMonthCloseLexicalMarker = /(?:закрыти[ея]\s+месяц|закрыт[а-яё]*\s+период|закрытие\s+счетов|регламентн|косвенн|затрат|распределени|рбп|амортиз|финансовых\s+результат|month\s*close|period\s*close|close\s+period|close\s+operation)/i.test(lower) || - (/закр/i.test(lower) && /перио/i.test(lower)); - if (/постав|постав|supplier|vendor/i.test(lower) || hasSettlementAccountScope) { + (/закр/i.test(lower) && /перио/i.test(lower)); + if (/постав|постав|supplier|vendor/i.test(lower) || hasSettlementAccountScope) { pushMany(domainScope, ["suppliers", "settlements"]); pushMany(documentTypes, ["supplier_receipt", "settlement_document"]); pushMany(entityTypes, ["counterparty", "contract", "document", "posting"]); pushMany(relationPatterns, ["payment_to_settlement", "contract_to_documents"]); } - if (/покупат|покупат|customer|buyer/i.test(lower) || hasSettlementAccountScope) { + if (/покупат|покупат|customer|buyer/i.test(lower) || hasSettlementAccountScope) { pushMany(domainScope, ["customers", "settlements"]); pushMany(documentTypes, ["sales_document", "settlement_document"]); pushMany(entityTypes, ["counterparty", "contract", "document", "posting"]); pushMany(relationPatterns, ["payment_to_settlement", "contract_to_documents"]); } - if (/РЅРґСЃ|ндс|vat|РєРЅРёРіР° РїРѕРєСѓРїРѕРє|РєРЅРёРіР° продаж|счет.?фактур|книг[аи]\s+покуп|книг[аи]\s+продаж|сч[её]т(?:а|у|ом|е)?[-\s]?фактур(?:а|ы|е|у|ой)?|вычет|налогов(?:ый|ого)?\s+эффект/i.test(lower) || + if (/ндс|vat|книга\s+покупок|книга\s+продаж|счет.?фактур|книг[аи]\s+покуп|книг[аи]\s+продаж|сч[её]т(?:а|у|ом|е)?[-\s]?фактур(?:а|ы|е|у|ой)?|вычет|налогов(?:ый|ого)?\s+эффект/i.test(lower) || hasVatAccountScope) { pushMany(domainScope, ["vat", "taxes"]); pushMany(documentTypes, ["invoice", "vat_document"]); pushMany(entityTypes, ["document", "tax_entry", "posting"]); pushMany(relationPatterns, ["invoice_to_vat", "document_to_posting"]); } - if (/РѕСЃ|РѕСЃРЅРѕРІРЅ(ые|ых)\s+сред|(?:^|[^a-zа-яё])ос(?:$|[^a-zа-яё])|основн(ые|ых|ым)?\s+средств|fixed asset|amort|амортиз|амортиз/i.test(lower) || + if (/ос|основн(ые|ых)\s+сред|(?:^|[^a-zа-яё])ос(?:$|[^a-zа-яё])|основн(ые|ых|ым)?\s+средств|fixed asset|amort|амортиз|амортиз/i.test(lower) || hasFixedAssetAccountScope) { pushMany(domainScope, ["fixed_assets"]); pushMany(documentTypes, ["fixed_asset_card", "fixed_asset_acceptance", "depreciation_document"]); pushMany(entityTypes, ["fixed_asset", "document", "posting"]); pushMany(relationPatterns, ["asset_card_to_depreciation", "document_to_posting"]); } - if (/СЂР±Рї|расходы будущих периодов|рбп|расходы\s+будущих\s+периодов|deferred|writeoff/i.test(lower) || + if (/рбп|расходы будущих периодов|рбп|расходы\s+будущих\s+периодов|deferred|writeoff/i.test(lower) || hasDeferredExpenseAccountScope) { pushMany(domainScope, ["deferred_expense", "period_close"]); pushMany(documentTypes, ["deferred_expense_document", "period_close_document"]); pushMany(entityTypes, ["document", "posting"]); pushMany(relationPatterns, ["deferred_expense_to_writeoff", "document_to_posting"]); } - if (/цепоч|разрыв|СЃРІСЏР·|документ.*РїСЂРѕРІРѕРґ|РіРґРµ рвет|Р¶РёРІСѓС‚ отдельно|цепоч|разрыв|связ|документ.*провод|chain|break/i.test(lower)) { + if (/цепоч|разрыв|связ|документ.*провод|где рвет|живут отдельно|цепоч|разрыв|связ|документ.*провод|chain|break/i.test(lower)) { pushMany(relationPatterns, ["document_to_posting", "contract_to_documents"]); pushMany(explanationFocus, ["what_conflicts_with_what", "why_not_closed"]); } - if (/аномал|СЂРёСЃРє|С…РІРѕСЃС‚|РїРѕРґРѕР·СЂ|искаж|аномал|риск|хвост|подозр|искаж|suspic|risk/i.test(lower)) { + if (/аномал|риск|хвост|подозр|искаж|аномал|риск|хвост|подозр|искаж|suspic|risk/i.test(lower)) { pushMany(anomalyPatterns, ["missing_link", "broken_lifecycle", "amount_independent_risk"]); } if (WRONG_DOCUMENT_MARKERS.test(lower)) { pushMany(anomalyPatterns, ["wrong_document_type", "posting_mismatch", "broken_lifecycle"]); } - if (/Р¶РёРІСѓС‚ отдельно|РЅРµ СЃРІСЏР·|без СЃРІСЏР·Рё|живут\s+отдельно|не\s+связ|без\s+связи|missing link/i.test(lower)) { + if (/живут отдельно|не связ|без связи|живут\s+отдельно|не\s+связ|без\s+связи|missing link/i.test(lower)) { pushMany(anomalyPatterns, ["missing_link", "cross_domain_inconsistency"]); } if (REPEATED_ANOMALY_MARKERS.test(lower)) { @@ -1446,10 +1446,10 @@ function buildSemanticRetrievalProfile(fragmentText) { pushMany(anomalyPatterns, ["closure_risk", "broken_lifecycle"]); pushMany(documentTypes, ["period_close_document"]); } - if (/РЅРµ РІ платеже|не\s+в\s+платеже|not payment/i.test(lower)) { + if (/не\s+в\s+платеже|not payment/i.test(lower)) { pushMany(excludedInterpretations, ["simple_payment_delay"]); } - if (/РЅРµ РїРѕ СЃСѓРјРј|РЅРµ СЃСѓРјРјР°|не\s+по\s+сумм|не\s+сумм|not by amount/i.test(lower)) { + if (/не\s+по\s+сумм|не\s+сумм|не\s+сумма|not by amount/i.test(lower)) { pushMany(excludedInterpretations, ["amount_only_anomaly"]); pushMany(rankingBasis, ["amount_independent_risk"]); } @@ -1735,16 +1735,16 @@ function inferAccountsFromRecord(record, corpus) { accounts.push(token.split(".")[0]); } for (const key of Object.keys(record.attributes ?? {})) { - if (/счетбанк|расчетн.*счет|bank account|банковскийсчет/i.test(key)) { + if (/счетбанк|расчетн.*счет|bank account|банковскийсчет/i.test(key)) { accounts.push("51"); } - if (/счетучетарасчетовсконтрагентом/i.test(key)) { + if (/счетучетарасчетовсконтрагентом/i.test(key)) { accounts.push("60"); } - if (/счетучетандс/i.test(key)) { + if (/счетучетандс/i.test(key)) { accounts.push("19"); } - if (/субконтодт/i.test(key)) { + if (/субконтодт/i.test(key)) { accounts.push("60"); } } @@ -1752,28 +1752,28 @@ function inferAccountsFromRecord(record, corpus) { } function inferDocumentTypesFromRecord(record, corpus) { const items = []; - if (/банковскиевыписки|выписк|расчетногосчета|spisaniesraschetnogoscheta|bank/i.test(corpus)) { + if (/банковскиевыписки|выписк|расчетногосчета|spisaniesraschetnogoscheta|bank/i.test(corpus)) { pushMany(items, ["bank_statement", "payment_order"]); } - if (/поступлениетоваровуслуг|поступлен/i.test(corpus)) { + if (/поступлениетоваровуслуг|поступлен/i.test(corpus)) { items.push("supplier_receipt"); } - if (/реализациятоваровуслуг|реализац/i.test(corpus)) { + if (/реализациятоваровуслуг|реализац/i.test(corpus)) { items.push("sales_document"); } - if (/РЅРґСЃ|счетфактур|РєРЅРёРіРёРїРѕРєСѓРїРѕРє|книгипродаж|vat|invoice/i.test(corpus)) { + if (/ндс|счетфактур|книгипокупок|книгипродаж|vat|invoice/i.test(corpus)) { pushMany(items, ["invoice", "vat_document"]); } - if (/корректировк|ручн|manual/i.test(corpus)) { + if (/корректировк|ручн|manual/i.test(corpus)) { items.push("manual_operation"); } - if (/закрытие|регламент/i.test(corpus)) { + if (/закрытие|регламент/i.test(corpus)) { items.push("period_close_document"); } - if (/РѕСЃРЅРѕРІРЅ|амортиз|fixed_asset/i.test(corpus)) { + if (/основн|амортиз|fixed_asset/i.test(corpus)) { pushMany(items, ["fixed_asset_card", "depreciation_document"]); } - if (/расходыбудущихпериодов|deferred|97/.test(corpus)) { + if (/расходыбудущихпериодов|deferred|97/.test(corpus)) { items.push("deferred_expense_document"); } if (record.source_entity.startsWith("Document") || record.source_entity.startsWith("DocumentJournal")) { @@ -1801,10 +1801,10 @@ function inferDomainsFromRecord(corpus, documentTypes, record) { if (documentTypes.some((item) => item === "deferred_expense_document")) { pushMany(domains, ["deferred_expense", "period_close"]); } - if (/закрытие|регламент|period close/i.test(corpus)) { + if (/закрытие|регламент|period close/i.test(corpus)) { domains.push("period_close"); } - const hasSettlementLexicalAnchor = /(?:settlement|payment|bank|statement|supplier|customer|buyer|vendor|60\b|62\b|51\b|оплат|банк|выписк|расчет|постав|РїРѕРєСѓРї)/i.test(corpus); + const hasSettlementLexicalAnchor = /(?:settlement|payment|bank|statement|supplier|customer|buyer|vendor|60\b|62\b|51\b|оплат|банк|выписк|расчет|постав|покуп)/i.test(corpus); const hasSettlementDocAnchor = documentTypes.some((item) => item === "bank_statement" || item === "payment_order" || item === "supplier_receipt" || item === "sales_document"); const hasSettlementDomainAnchor = domains.includes("bank") || domains.includes("suppliers") || domains.includes("customers") || domains.includes("supplier_payments"); if (findCounterpartyLinks(record).length > 0 && (hasSettlementLexicalAnchor || hasSettlementDocAnchor || hasSettlementDomainAnchor)) { @@ -1824,13 +1824,13 @@ function inferEntityTypes(record) { entities.push("counterparty"); } const corpus = collectTextFromRecord(record); - if (/РґРѕРіРѕРІРѕСЂ|contract/i.test(corpus)) { + if (/договор|contract/i.test(corpus)) { entities.push("contract"); } - if (/РѕСЃРЅРѕРІРЅ|fixed_asset|инвентар/i.test(corpus)) { + if (/основн|fixed_asset|инвентар/i.test(corpus)) { entities.push("fixed_asset"); } - if (/РЅРґСЃ|РєРЅРёРіРёРїРѕРєСѓРїРѕРє|книгипродаж|vat|invoice/i.test(corpus)) { + if (/ндс|книгипокупок|книгипродаж|vat|invoice/i.test(corpus)) { entities.push("tax_entry"); } return uniqueStrings(entities); @@ -1842,25 +1842,25 @@ function inferRelationPatterns(record, corpus) { if (hasDocLinks) { patterns.push("document_to_posting"); } - if (hasCounterparty && hasDocLinks && /платеж|bank|settlement|расчет/i.test(corpus)) { + if (hasCounterparty && hasDocLinks && /платеж|bank|settlement|расчет/i.test(corpus)) { patterns.push("payment_to_settlement"); } - if (/банковскиевыписки|выписк|statement/i.test(corpus) && hasDocLinks) { + if (/банковскиевыписки|выписк|statement/i.test(corpus) && hasDocLinks) { patterns.push("statement_to_document"); } - if (/РѕСЃРЅРѕРІРЅ|fixed_asset|амортиз/i.test(corpus)) { + if (/основн|fixed_asset|амортиз/i.test(corpus)) { patterns.push("asset_card_to_depreciation"); } - if (/расходыбудущихпериодов|97|deferred/i.test(corpus)) { + if (/расходыбудущихпериодов|97|deferred/i.test(corpus)) { patterns.push("deferred_expense_to_writeoff"); } - if (/РЅРґСЃ|счетфактур|РєРЅРёРіРёРїРѕРєСѓРїРѕРє|книгипродаж|vat|invoice/i.test(corpus)) { + if (/ндс|счетфактур|книгипокупок|книгипродаж|vat|invoice/i.test(corpus)) { patterns.push("invoice_to_vat"); } - if (/РґРѕРіРѕРІРѕСЂ|contract/i.test(corpus) && hasDocLinks) { + if (/договор|contract/i.test(corpus) && hasDocLinks) { patterns.push("contract_to_documents"); } - if (/склад|товар|материал|receipt/i.test(corpus)) { + if (/склад|товар|материал|receipt/i.test(corpus)) { patterns.push("receipt_to_stock_movement"); } return uniqueStrings(patterns); @@ -1899,7 +1899,7 @@ function inferAnomalyPatterns(record, corpus, relationPatterns, lifecycleMarkers if (relationPatterns.includes("document_to_posting") && !record.attributes.Recorder) { anomalies.push("posting_mismatch"); } - if (/ручн|manual|корректировк/.test(corpus)) { + if (/ручн|manual|корректировк/.test(corpus)) { anomalies.push("manual_intervention_suspicion"); } if (lifecycleMarkers.includes("period_boundary") && (unknownLinks > 0 || zeroGuidValues > 0)) { @@ -1911,7 +1911,7 @@ function inferAnomalyPatterns(record, corpus, relationPatterns, lifecycleMarkers if (!hasCounterparty && !hasDocLinks && zeroGuidValues > 0) { anomalies.push("silent_orphan"); } - const hasAmountSignal = Object.keys(record.attributes ?? {}).some((key) => /sum|СЃСѓРјРј|итого|amount/i.test(key)); + const hasAmountSignal = Object.keys(record.attributes ?? {}).some((key) => /sum|сумм|итого|amount/i.test(key)); if (!hasAmountSignal && anomalies.length > 0) { anomalies.push("amount_independent_risk"); } @@ -1954,15 +1954,15 @@ function evaluateExcludedInterpretations(profile, signals, record) { const structural = ["missing_link", "broken_lifecycle", "posting_mismatch", "wrong_document_type", "closure_risk"]; const hasStructural = signals.anomaly_patterns.some((item) => structural.includes(item)); if (!hasStructural) { - reasons.push("Исключено как simple_payment_delay без структурного дефекта."); + reasons.push("Исключено как simple_payment_delay без структурного дефекта."); } } if (interpretationSet.has("amount_only_anomaly")) { - const hasAmountSignal = Object.keys(record.attributes ?? {}).some((key) => /sum|СЃСѓРјРј|итого|amount/i.test(key)); + const hasAmountSignal = Object.keys(record.attributes ?? {}).some((key) => /sum|сумм|итого|amount/i.test(key)); const structural = ["missing_link", "broken_lifecycle", "posting_mismatch", "cross_domain_inconsistency", "silent_orphan"]; const hasStructural = signals.anomaly_patterns.some((item) => structural.includes(item)); if (hasAmountSignal && !hasStructural) { - reasons.push("Исключено как amount-only аномалия без структурных признаков."); + reasons.push("Исключено как amount-only аномалия без структурных признаков."); } } return { @@ -2016,22 +2016,22 @@ function evaluateRecordAgainstProfile(record, profile) { const graphTraversal = evaluateGraphTraversalForRecord(profile, signals); const matchReasons = []; if (accountMatch && profile.account_scope.length > 0) { - matchReasons.push("Совпал account_scope."); + matchReasons.push("Совпал account_scope."); } if (domainMatch && profile.domain_scope.length > 0) { - matchReasons.push("Совпал domain_scope."); + matchReasons.push("Совпал domain_scope."); } if (documentMatch && profile.document_types.length > 0) { - matchReasons.push("Совпал document_types."); + matchReasons.push("Совпал document_types."); } if (relationMatch && profile.relation_patterns.length > 0) { - matchReasons.push("Совпали relation_patterns."); + matchReasons.push("Совпали relation_patterns."); } if (anomalyMatch && profile.anomaly_patterns.length > 0) { - matchReasons.push("Совпали anomaly_patterns."); + matchReasons.push("Совпали anomaly_patterns."); } if (lifecycleMatch && profile.lifecycle_stage_filters.length > 0) { - matchReasons.push("Совпал lifecycle_stage_filters."); + matchReasons.push("Совпал lifecycle_stage_filters."); } if (graphTraversal.domain_match) { matchReasons.push("Graph traversal domain matched."); @@ -2206,7 +2206,7 @@ class AssistantDataLayer { business_interpretation: [], confidence: "low", limitations: ["Snapshot data files could not be loaded."], - errors: ["Слой данных недоступен: РЅРµ удалось загрузить snapshot-файлы."] + errors: ["Слой данных недоступен: не удалось загрузить snapshot-файлы."] }; } let result = null; @@ -3037,8 +3037,8 @@ class AssistantDataLayer { ? "medium" : "low", business_interpretation: group.risk_factors.size > 0 - ? "Есть признаки разрыва расчетной цепочки: часть связей/этапов lifecycle подтверждена неполно." - : "Есть связанная операционная цепочка, РЅРѕ явные СЂРёСЃРє-паттерны выражены слабо.", + ? "Есть признаки разрыва расчетной цепочки: часть связей/этапов lifecycle подтверждена неполно." + : "Есть связанная операционная цепочка, но явные риск-паттерны выражены слабо.", relation_types: Array.from(group.relations.entries()) .sort((left, right) => right[1] - left[1]) .map((item) => item[0]), @@ -3086,24 +3086,24 @@ class AssistantDataLayer { evidence: [], why_included: [], selection_reason: [ - "РџРѕРёСЃРє строился РїРѕ semantic retrieval profile, РЅРѕ подходящие контрагенты РЅРµ найдены.", - "Фильтрация использовала пересечение account/domain/document/relation/anomaly ограничений.", + "Поиск строился по semantic retrieval profile, но подходящие контрагенты не найдены.", + "Фильтрация использовала пересечение account/domain/document/relation/anomaly ограничений.", domainCard ? `P0 domain purity enforced for ${domainCard.id}.` : "P0 domain purity was not enforced.", guidFilter.length > 0 - ? "GUID-фильтрация включена." - : `GUID отсутствовал, выполнено semantic narrowing (${filtered.length}/${sourceRecords.length}).`, + ? "GUID-фильтрация включена." + : `GUID отсутствовал, выполнено semantic narrowing (${filtered.length}/${sourceRecords.length}).`, `Graph planner mode=${graphTraversalRuntime.planner_mode}, eligible=${graphTraversalRuntime.graph_eligible}, applied=${graphTraversalRuntime.traversal_applied}.`, `Graph ranking signals: ${graphTraversalRuntime.ranking_shift_signals.join(",") || "none"}.` ], risk_factors: semanticProfile.anomaly_patterns, business_interpretation: [ - "РџРѕ текущему профилю запроса устойчивых разрывов цепочки РЅРµ обнаружено.", - "Для точечного drilldown добавьте GUID или уточните период/контрагента." + "По текущему профилю запроса устойчивых разрывов цепочки не обнаружено.", + "Для точечного drilldown добавьте GUID или уточните период/контрагента." ], confidence: "medium", limitations: [ - guidFilter.length > 0 ? "РџРѕРёСЃРє ограничен переданными GUID." : "РџРѕРёСЃРє выполнен РїРѕ semantic narrowing без GUID.", - "Источник данных — snapshot 2020 (read-only), Р° РЅРµ live состояние базы 1РЎ.", + guidFilter.length > 0 ? "Поиск ограничен переданными GUID." : "Поиск выполнен по semantic narrowing без GUID.", + "Источник данных — snapshot 2020 (read-only), а не live состояние базы 1С.", domainCard ? "Domain purity guardrail может исключить cross-domain записи на этапе source selection." : "Domain purity guardrail не применялся." ], errors: [] @@ -3144,33 +3144,33 @@ class AssistantDataLayer { }, evidence: evidence.slice(0, 12), why_included: [ - `Семантическое сужение выполнено РїРѕ профилю ${semanticProfile.query_subject}.`, + `Семантическое сужение выполнено по профилю ${semanticProfile.query_subject}.`, domainCard ? `P0 domain purity enforced for ${domainCard.id}.` : "P0 domain purity was not enforced.", semanticProfile.account_scope.length > 0 - ? `Учитывались счета: ${semanticProfile.account_scope.join(", ")}.` - : "Счета РЅРµ были заданы СЏРІРЅРѕ, использованы domain/document/relation ограничения.", - `После narrowing осталось ${filtered.length} РёР· ${sourceRecords.length} записей.`, + ? `Учитывались счета: ${semanticProfile.account_scope.join(", ")}.` + : "Счета не были заданы явно, использованы domain/document/relation ограничения.", + `После narrowing осталось ${filtered.length} из ${sourceRecords.length} записей.`, `Graph traversal mode=${graphTraversalRuntime.planner_mode}, matched=${graphTraversalRuntime.matched_candidates}/${graphTraversalRuntime.evaluated_candidates}.` ], selection_reason: [ - "Отбор основан РЅР° пересечении account_scope + domain_scope + document_types + relation_patterns + anomaly_patterns.", - "GUID-mode отключен: full scan без ограничителей РЅРµ использовался.", - `Ранжирование выполнено РїРѕ basis: ${semanticProfile.ranking_basis.join(", ")}.`, + "Отбор основан на пересечении account_scope + domain_scope + document_types + relation_patterns + anomaly_patterns.", + "GUID-mode отключен: full scan без ограничителей не использовался.", + `Ранжирование выполнено по basis: ${semanticProfile.ranking_basis.join(", ")}.`, domainCard ? `Domain gate source scope: ${sourceScope.join(", ")}.` : "Domain gate source scope not applied.", `Graph signal counts: ${JSON.stringify(graphTraversalRuntime.signal_counts)}.`, `Graph ranking signals: ${graphTraversalRuntime.ranking_shift_signals.join(",") || "none"}.` ], risk_factors: aggregatedRiskFactors.length > 0 ? aggregatedRiskFactors - : ["Высокая плотность операций РїРѕ контрагенту может указывать РЅР° незакрытые цепочки."], + : ["Высокая плотность операций по контрагенту может указывать на незакрытые цепочки."], business_interpretation: [ - "Результат отражает РЅРµ просто объем операций, Р° структурные признаки разрыва цепочки Рё lifecycle-конфликта.", - "Контрагенты РІ топе приоритетны для проверки РЅР° неверный тип закрывающего документа Рё незавершенные СЃРІСЏР·Рё." + "Результат отражает не просто объем операций, а структурные признаки разрыва цепочки и lifecycle-конфликта.", + "Контрагенты в топе приоритетны для проверки на неверный тип закрывающего документа и незавершенные связи." ], confidence: "high", limitations: [ - guidFilter.length > 0 ? "Выборка ограничена GUID РёР· запроса." : "Выборка ограничена semantic retrieval profile.", - "Источник данных — snapshot 2020 (read-only), РЅРµ live контур 1РЎ.", + guidFilter.length > 0 ? "Выборка ограничена GUID из запроса." : "Выборка ограничена semantic retrieval profile.", + "Источник данных — snapshot 2020 (read-only), не live контур 1С.", domainCard ? "Domain purity guardrail может исключить cross-domain элементы на этапе source selection." : "Domain purity guardrail не применялся." ], errors: [] @@ -3455,12 +3455,12 @@ class AssistantDataLayer { })), why_included: items.length > 0 ? [ - "Показаны сущности СЃ максимальным количеством записей.", + "Показаны сущности с максимальным количеством записей.", domainCard ? `P0 domain purity enforced for ${domainCard.id}.` : "P0 domain purity was not enforced." ] : [], selection_reason: [ - "Ранжирование выполнено РїРѕ records_count РїРѕ убыванию.", + "Ранжирование выполнено по records_count по убыванию.", domainCard ? `Domain gate source scope: ${sourceScope.join(", ")}.` : "Domain gate source scope not applied." ], risk_factors: uniqueStrings(["entity_volume_spike", ...semanticProfile.anomaly_patterns]), @@ -3469,7 +3469,7 @@ class AssistantDataLayer { ], confidence: "medium", limitations: [ - "Ранжирование РїРѕ объему РЅРµ всегда эквивалентно бизнес-СЂРёСЃРєСѓ.", + "Ранжирование по объему не всегда эквивалентно бизнес-риску.", domainCard ? "Domain purity guardrail может исключить cross-domain записи на batch-слое." : "Domain purity guardrail не применялся." ], errors: [] @@ -3604,7 +3604,7 @@ class AssistantDataLayer { domainCard ? `Domain gate source scope: ${sourceScope.join(", ")}.` : "Domain gate source scope not applied." ], risk_factors: semanticProfile.anomaly_patterns, - business_interpretation: ["Слой отражает базовый factual-срез документов для оперативной сверки."], + business_interpretation: ["Слой отражает базовый factual-срез документов для оперативной сверки."], confidence: "high", limitations: [ "Это read-only snapshot, а не онлайн-состояние 1С.", @@ -3634,11 +3634,11 @@ class AssistantDataLayer { }, evidence: [], why_included: [], - selection_reason: ["Для drilldown требуется GUID или достаточные business anchors (номер/дата/СЃСѓРјРјР°/счет)."], + selection_reason: ["Для drilldown требуется GUID или достаточные business anchors (номер/дата/сумма/счет)."], risk_factors: [], - business_interpretation: ["Без GUID или business anchors точечный drilldown невозможен."], + business_interpretation: ["Без GUID или business anchors точечный drilldown невозможен."], confidence: "low", - limitations: ["Добавьте GUID или СЏРєРѕСЂСЏ: номер документа, дату, СЃСѓРјРјСѓ, счет."], + limitations: ["Добавьте GUID или якоря: номер документа, дату, сумму, счет."], errors: [] }; } @@ -3686,14 +3686,14 @@ class AssistantDataLayer { }, evidence: matches.slice(0, 10), why_included: matches.length > 0 - ? ["Включены source-of-record записи, совпавшие РїРѕ business anchors (номер/дата/СЃСѓРјРјР°/счет)."] + ? ["Включены source-of-record записи, совпавшие по business anchors (номер/дата/сумма/счет)."] : [], selection_reason: [ - "GUID отсутствует, использован business-anchor trace РїРѕ атрибутам документа Рё расчетов." + "GUID отсутствует, использован business-anchor trace по атрибутам документа и расчетов." ], risk_factors: [], business_interpretation: [ - "Drilldown опирается РЅР° business anchors, поэтому вывод требует первичной проверки РІ source-of-record." + "Drilldown опирается на business anchors, поэтому вывод требует первичной проверки в source-of-record." ], confidence: matches.length > 0 ? "medium" : "low", limitations: [ @@ -3722,12 +3722,12 @@ class AssistantDataLayer { matched_records: matches.length }, evidence: matches.slice(0, 10), - why_included: matches.length > 0 ? ["Включены записи, содержащие GUID РёР· запроса."] : [], - selection_reason: ["РџРѕРёСЃРє РїРѕ source_id, linked target_id Рё строковым атрибутам."], + why_included: matches.length > 0 ? ["Включены записи, содержащие GUID из запроса."] : [], + selection_reason: ["Поиск по source_id, linked target_id и строковым атрибутам."], risk_factors: [], - business_interpretation: ["Результат показывает source-of-record объекты РїРѕ переданным идентификаторам."], + business_interpretation: ["Результат показывает source-of-record объекты по переданным идентификаторам."], confidence: matches.length > 0 ? "high" : "medium", - limitations: ["РџРѕРёСЃРє ограничен локальным snapshot-пакетом."], + limitations: ["Поиск ограничен локальным snapshot-пакетом."], errors: [] }; } diff --git a/llm_normalizer/backend/dist/services/assistantRuntimeGuards.js b/llm_normalizer/backend/dist/services/assistantRuntimeGuards.js index e935948..0c398aa 100644 --- a/llm_normalizer/backend/dist/services/assistantRuntimeGuards.js +++ b/llm_normalizer/backend/dist/services/assistantRuntimeGuards.js @@ -378,7 +378,7 @@ function parseDateLike(raw) { if (dayMonthYear) { return normalizeDateIso({ year: parseYear(dayMonthYear[3]), month: dayMonthYear[2], day: dayMonthYear[1] }); } - const rusMonthYear = value.match(/\b(январь|февраль|март|апрель|май|РёСЋРЅСЊ|июль|август|сентябрь|октябрь|РЅРѕСЏР±СЂСЊ|декабрь)\s+(20\d{2})\b/i); + const rusMonthYear = value.match(/\b(январь|февраль|март|апрель|май|июнь|июль|август|сентябрь|октябрь|ноябрь|декабрь)\s+(20\d{2})\b/i); if (rusMonthYear) { const month = RUS_MONTH_TO_NUMBER[String(rusMonthYear[1] ?? "").toLowerCase()]; if (!month) @@ -467,7 +467,7 @@ function normalizedAnchorFromFragments(normalized) { source: `normalized_time_scope:${type || "unknown"}` }; } - if (/(?:июл|july|РёСЋР»)/i.test(value)) { + if (/(?:июл|july|июл)/i.test(value)) { return { value: `${JULY_YEAR}-${JULY_MONTH}`, source: `normalized_time_scope:${type || "unknown"}` @@ -491,7 +491,7 @@ function resolveJulyAnchor(rawText) { const explicitYear = lower.match(/\b(20\d{2})\b/)?.[1] ?? null; const dayByNamedJuly = lower.match(/(?:^|\D)(0?[1-9]|[12]\d|3[01])\s+(?:июл(?:я|ь)?|july)(?:\D|$)/i); const dayByNumeric = lower.match(/\b(0?[1-9]|[12]\d|3[01])[./-](0?7)(?:[./-](\d{2}|\d{4}))?\b/); - const monthByNamed = /(?:июл|july|РёСЋР»)/i.test(lower); + const monthByNamed = /(?:июл|july|июл)/i.test(lower); const monthByNumeric = /\b20\d{2}[-/.]0?7\b/.test(lower); if (!dayByNamedJuly && !dayByNumeric && !monthByNamed && !monthByNumeric) { return { @@ -662,7 +662,7 @@ function applyTemporalHintToExecutionPlan(executionPlan, temporal) { return item; } const text = String(item.fragment_text ?? "").trim(); - if (/2020-07|июл|РёСЋР»|july/i.test(text)) { + if (/2020-07|июл|июл|july/i.test(text)) { return item; } return { @@ -681,7 +681,7 @@ function resolveDomainPolarityGuard(input) { prefixes.has("62") || prefixes.has("51") || prefixes.has("76") || - /(?:расч[её]т|оплат|аванс|долг|settlement|payment|tail|хвост|незакры|зач[её]т|расч|оплат|аванс|долг|С…РІРѕСЃС‚)/i.test(lower); + /(?:расч[её]т|оплат|аванс|долг|settlement|payment|tail|хвост|незакры|зач[её]т|расч|оплат|аванс|долг|хвост)/i.test(lower); if (!settlementSignal) { return { applied: false, @@ -700,12 +700,12 @@ function resolveDomainPolarityGuard(input) { reason_codes: [] }; } - const supplierScore = (/(?:поставщ|supplier|vendor|кредитор|обязательств|payable|поставщ|кредитор|обязательств)/i.test(lower) ? 2 : 0) + + const supplierScore = (/(?:поставщ|supplier|vendor|кредитор|обязательств|payable|поставщ|кредитор|обязательств)/i.test(lower) ? 2 : 0) + (prefixes.has("60") ? 2 : 0) + - (/(?:сч[её]т\s*60|по\s*60|счет\s*60|РїРѕ\s*60)/i.test(lower) ? 1 : 0); - const customerScore = (/(?:покупат|customer|buyer|дебитор|receivable|покупат|дебитор)/i.test(lower) ? 2 : 0) + + (/(?:сч[её]т\s*60|по\s*60|счет\s*60)/i.test(lower) ? 1 : 0); + const customerScore = (/(?:покупат|customer|buyer|дебитор|receivable|покупат|дебитор)/i.test(lower) ? 2 : 0) + (prefixes.has("62") ? 2 : 0) + - (/(?:сч[её]т\s*62|по\s*62|счет\s*62|РїРѕ\s*62)/i.test(lower) ? 1 : 0); + (/(?:сч[её]т\s*62|по\s*62|счет\s*62)/i.test(lower) ? 1 : 0); let polarity = "mixed_or_unresolved"; if (supplierScore > 0 || customerScore > 0) { if (supplierScore >= customerScore + 2) { @@ -758,10 +758,10 @@ function applyPolarityHintToExecutionPlan(executionPlan, polarity) { return item; } const text = String(item.fragment_text ?? "").trim(); - if (polarity.polarity === "supplier_payable" && /(поставщ|supplier|сч[её]т\s*60|по\s*60|поставщ|счет\s*60|РїРѕ\s*60)/i.test(text)) { + if (polarity.polarity === "supplier_payable" && /(поставщ|supplier|сч[её]т\s*60|по\s*60|поставщ|счет\s*60)/i.test(text)) { return item; } - if (polarity.polarity === "customer_receivable" && /(покупат|customer|сч[её]т\s*62|по\s*62|покупат|счет\s*62|РїРѕ\s*62)/i.test(text)) { + if (polarity.polarity === "customer_receivable" && /(покупат|customer|сч[её]т\s*62|по\s*62|покупат|счет\s*62)/i.test(text)) { return item; } return { @@ -771,10 +771,10 @@ function applyPolarityHintToExecutionPlan(executionPlan, polarity) { }); } function containsReceivableSignal(value) { - return /(?:customer_settlement|stale_receivable|receivable_closed|receivable|дебитор)/i.test(value); + return /(?:customer_settlement|stale_receivable|receivable_closed|receivable|дебитор)/i.test(value); } function containsPayableSignal(value) { - return /(?:bank_settlement|payable|обязательств|supplier|поставщ|счет\s*60|\b60(?:\.\d{2})?\b)/i.test(value); + return /(?:bank_settlement|payable|обязательств|supplier|поставщ|счет\s*60|\b60(?:\.\d{2})?\b)/i.test(value); } function problemUnitCorpus(unit) { return [ @@ -1319,15 +1319,15 @@ function applyEligibilityToGroundingCheck(groundingCheck, eligibility) { ? "no_grounded_answer" : "partial"; const reasonMap = { - admissible_evidence_count_zero: "Недостаточно допустимого evidence для обоснованного ответа.", - critical_domain_or_account_contradiction: "Есть критическое противоречие РїРѕ domain/account scope.", - temporal_guard_failed_out_of_snapshot_window: "Temporal anchor вышел Р·Р° РѕРєРЅРѕ company snapshot (июль 2020).", - temporal_guard_ambiguous_limited: "Temporal anchor РЅРµ разрешен надежно РІ пределах company snapshot.", + admissible_evidence_count_zero: "Недостаточно допустимого evidence для обоснованного ответа.", + critical_domain_or_account_contradiction: "Есть критическое противоречие по domain/account scope.", + temporal_guard_failed_out_of_snapshot_window: "Temporal anchor вышел за окно company snapshot (июль 2020).", + temporal_guard_ambiguous_limited: "Temporal anchor не разрешен надежно в пределах company snapshot.", business_scope_generic_unresolved: "Business scope остался generic и не подтвержден как company-specific для доказательного ответа.", - polarity_guard_limited_unresolved_polarity: "РќРµ удалось надежно определить supplier/customer polarity.", - polarity_guard_blocked_conflict: "Обнаружен конфликт supplier/customer polarity РІ retrieval-контуре.", - claim_anchor_coverage_insufficient: "Недостаточно покрытия required anchors для claim-bound grounding.", - targeted_evidence_hit_rate_zero: "Targeted evidence acquisition РЅРµ дал допустимых попаданий РїРѕ claim target path." + polarity_guard_limited_unresolved_polarity: "Не удалось надежно определить supplier/customer polarity.", + polarity_guard_blocked_conflict: "Обнаружен конфликт supplier/customer polarity в retrieval-контуре.", + claim_anchor_coverage_insufficient: "Недостаточно покрытия required anchors для claim-bound grounding.", + targeted_evidence_hit_rate_zero: "Targeted evidence acquisition не дал допустимых попаданий по claim target path." }; const reasons = [ ...(Array.isArray(groundingCheck.reasons) ? groundingCheck.reasons : []), diff --git a/llm_normalizer/backend/dist/services/assistantService.js b/llm_normalizer/backend/dist/services/assistantService.js index b59eee9..bc862b4 100644 --- a/llm_normalizer/backend/dist/services/assistantService.js +++ b/llm_normalizer/backend/dist/services/assistantService.js @@ -65,6 +65,8 @@ const addressFilterExtractor_1 = __importStar(require("./addressFilterExtractor" const predecomposeContract_1 = __importStar(require("./address_runtime/predecomposeContract")); const openaiResponsesClient_1 = __importStar(require("./openaiResponsesClient")); const addressMcpClient_1 = __importStar(require("./addressMcpClient")); +const capabilitiesRegistry_1 = __importStar(require("./capabilitiesRegistry")); +const assistantCanon_1 = __importStar(require("./assistantCanon")); const iconv_lite_1 = __importDefault(require("iconv-lite")); const DATA_SCOPE_CACHE_TTL_MS = 60_000; const dataScopeProbeCache = new Map(); @@ -3903,17 +3905,7 @@ function buildLivingChatPrompt(userMessage, conversationWindow) { return `${contextBlock}Сообщение пользователя:\n${userMessage}`; } function buildAssistantCapabilityContractReply() { - return [ - "Я ассистент по анализу данных 1С в режиме чтения.", - "Что умею сейчас:", - "1. Находить документы, операции, договоры и остатки по контрагенту/договору/периоду.", - "2. Делать агрегаты по базе: активность, роли контрагентов, top-срезы по суммам и операциям.", - "3. Кратко объяснять результат и подсказывать следующий точный запрос.", - "Что не умею:", - "1. Не настраиваю 1С и не меняю конфигурацию.", - "2. Не создаю и не провожу документы в базе.", - "3. Не выполняю админские действия на сервере." - ].join("\n"); + return (0, capabilitiesRegistry_1.buildCapabilityContractReplyFromRegistry)(); } function normalizeScopeLabel(value) { const repaired = repairAddressMojibake(String(value ?? "")); @@ -5016,6 +5008,7 @@ class AssistantService { else { const conversationWindow = buildLivingChatContextWindow(session.items); const userPrompt = buildLivingChatPrompt(userMessage, conversationWindow); + const canonExcerpt = (0, assistantCanon_1.loadAssistantCanonExcerpt)(520); const chatResponse = await this.chatClient.chat({ llmProvider: payload.llmProvider, apiKey: String(payload.apiKey ?? process.env.OPENAI_API_KEY ?? ""), @@ -5029,7 +5022,8 @@ class AssistantService { "Работай честно: не заявляй действия, которые недоступны в этом рантайме.", "Разрешено: анализ и объяснение данных, формулировка запросов, подсказки по следующему шагу.", "Запрещено: обещать настройку 1С, админ-действия, создание/проведение документов или любые изменения в базе.", - "Если пользователь спрашивает про возможности, отвечай только по этому контракту." + "Если пользователь спрашивает про возможности, отвечай только по этому контракту.", + `Канон поведения: ${canonExcerpt}` ].join(" "), developerPrompt: "Формат: коротко и по сути, без JSON и без служебных блоков. Пиши человеко-понятно.", userMessage: userPrompt, diff --git a/llm_normalizer/backend/dist/services/capabilitiesRegistry.js b/llm_normalizer/backend/dist/services/capabilitiesRegistry.js new file mode 100644 index 0000000..5db7352 --- /dev/null +++ b/llm_normalizer/backend/dist/services/capabilitiesRegistry.js @@ -0,0 +1,182 @@ +"use strict"; +var __importDefault = (this && this.__importDefault) || function (mod) { + return (mod && mod.__esModule) ? mod : { "default": mod }; +}; +Object.defineProperty(exports, "__esModule", { value: true }); +exports.loadCapabilitiesRegistry = loadCapabilitiesRegistry; +exports.buildCapabilityContractReplyFromRegistry = buildCapabilityContractReplyFromRegistry; +exports.resolveNearestCapabilityGroup = resolveNearestCapabilityGroup; +const fs_1 = __importDefault(require("fs")); +const config_1 = require("../config"); +const FALLBACK_REGISTRY = { + schema_version: "capabilities_registry_fallback_v1", + updated_at: "2026-04-09T00:00:00.000Z", + assistant_mode: "read_only", + groups: [ + { + group_code: "vat", + group_title: "НДС", + description: "Срезы и расчеты НДС на базе данных 1С.", + risk_level: "high", + maturity_status: "partial", + supported_operations: ["vat_period_snapshot", "vat_payable_forecast"], + unsupported_operations: ["submit_tax_declaration"], + required_entities: ["period", "organization"], + optional_entities: ["counterparty"], + typical_queries: ["Сколько НДС к уплате за период?"], + related_routes: [], + safe_alternatives: ["Показать движения по 68/19 за период"], + one_c_hints: ["РегистрБухгалтерии.Хозрасчетный"] + }, + { + group_code: "counterparties", + group_title: "Контрагенты", + description: "Документы, операции, договоры и срезы по контрагентам.", + risk_level: "medium", + maturity_status: "production_ready", + supported_operations: ["list_documents_by_counterparty", "list_contracts_by_counterparty"], + unsupported_operations: ["edit_counterparty_card"], + required_entities: ["counterparty_scope_or_contract"], + optional_entities: ["period", "organization"], + typical_queries: ["Покажи документы по контрагенту"], + related_routes: [], + safe_alternatives: ["Уточнить ИНН/наименование контрагента"], + one_c_hints: ["Справочник.Контрагенты"] + }, + { + group_code: "boundaries", + group_title: "Ограничения", + description: "Операции, которые ассистент не выполняет.", + risk_level: "high", + maturity_status: "production_ready", + supported_operations: ["explain_boundary", "suggest_safe_next_step"], + unsupported_operations: ["configure_1c", "admin_server_actions", "create_or_post_documents"], + required_entities: [], + optional_entities: [], + typical_queries: ["Можешь настроить 1С?"], + related_routes: [], + safe_alternatives: ["Сформировать план диагностики для 1С/ИТ-админа"], + one_c_hints: [] + } + ] +}; +let cache = null; +function toRecord(value) { + if (!value || typeof value !== "object" || Array.isArray(value)) + return null; + return value; +} +function toStringSafe(value) { + if (typeof value !== "string") + return null; + const trimmed = value.trim(); + return trimmed.length > 0 ? trimmed : null; +} +function toArray(value) { + return Array.isArray(value) ? value : []; +} +function readRegistryFromFile() { + if (!fs_1.default.existsSync(config_1.ASSISTANT_CAPABILITIES_REGISTRY_FILE)) + return null; + try { + const raw = fs_1.default.readFileSync(config_1.ASSISTANT_CAPABILITIES_REGISTRY_FILE, "utf-8"); + const parsed = JSON.parse(raw); + const root = toRecord(parsed); + if (!root) + return null; + const groups = toArray(root.groups) + .map((item) => toRecord(item)) + .filter((item) => item !== null) + .map((item) => ({ + group_code: toStringSafe(item.group_code) ?? "unknown_group", + group_title: toStringSafe(item.group_title) ?? "Группа", + description: toStringSafe(item.description) ?? "", + risk_level: toStringSafe(item.risk_level) ?? "medium", + maturity_status: toStringSafe(item.maturity_status) ?? + "partial", + supported_operations: toArray(item.supported_operations) + .map((v) => toStringSafe(v)) + .filter((v) => v !== null), + unsupported_operations: toArray(item.unsupported_operations) + .map((v) => toStringSafe(v)) + .filter((v) => v !== null), + required_entities: toArray(item.required_entities) + .map((v) => toStringSafe(v)) + .filter((v) => v !== null), + optional_entities: toArray(item.optional_entities) + .map((v) => toStringSafe(v)) + .filter((v) => v !== null), + typical_queries: toArray(item.typical_queries) + .map((v) => toStringSafe(v)) + .filter((v) => v !== null), + related_routes: toArray(item.related_routes) + .map((v) => toStringSafe(v)) + .filter((v) => v !== null), + safe_alternatives: toArray(item.safe_alternatives) + .map((v) => toStringSafe(v)) + .filter((v) => v !== null), + one_c_hints: toArray(item.one_c_hints) + .map((v) => toStringSafe(v)) + .filter((v) => v !== null) + })); + if (groups.length === 0) + return null; + return { + schema_version: toStringSafe(root.schema_version) ?? "capabilities_registry_v1", + updated_at: toStringSafe(root.updated_at) ?? new Date().toISOString(), + assistant_mode: toStringSafe(root.assistant_mode) ?? "read_only", + groups + }; + } + catch { + return null; + } +} +function loadCapabilitiesRegistry() { + try { + const mtimeMs = fs_1.default.existsSync(config_1.ASSISTANT_CAPABILITIES_REGISTRY_FILE) + ? fs_1.default.statSync(config_1.ASSISTANT_CAPABILITIES_REGISTRY_FILE).mtimeMs + : -1; + if (cache && cache.mtimeMs === mtimeMs) { + return cache.value; + } + const value = readRegistryFromFile() ?? FALLBACK_REGISTRY; + cache = { mtimeMs, value }; + return value; + } + catch { + return cache?.value ?? FALLBACK_REGISTRY; + } +} +function buildCapabilityContractReplyFromRegistry() { + const registry = loadCapabilitiesRegistry(); + const topGroups = registry.groups.slice(0, 6); + const groupLines = topGroups.map((group, index) => { + const ops = group.supported_operations.slice(0, 3).join(", "); + return `${index + 1}. ${group.group_title}: ${group.description}${ops ? ` (например: ${ops})` : ""}.`; + }); + return [ + "Я ассистент по анализу данных 1С в режиме чтения.", + "Что умею по группам:", + ...groupLines, + "Если хотите, раскрою любую группу точечно и дам готовую формулировку запроса.", + "Что не делаю: не настраиваю 1С, не меняю конфигурацию, не создаю и не провожу документы, не выполняю админ-действия на сервере." + ].join("\n"); +} +function resolveNearestCapabilityGroup(input) { + const registry = loadCapabilitiesRegistry(); + const haystack = `${String(input.domain ?? "")} ${String(input.queryClass ?? "")}`.toLowerCase(); + if (!haystack.trim()) + return null; + const scoring = registry.groups.map((group) => { + let score = 0; + const bucket = `${group.group_code} ${group.group_title} ${group.description} ${group.supported_operations.join(" ")}`.toLowerCase(); + for (const token of haystack.split(/[\s._/-]+/g).filter(Boolean)) { + if (bucket.includes(token)) + score += 1; + } + return { group, score }; + }); + scoring.sort((a, b) => b.score - a.score); + return scoring[0] && scoring[0].score > 0 ? scoring[0].group : null; +} diff --git a/llm_normalizer/backend/dist/services/lifecycleRuntime.js b/llm_normalizer/backend/dist/services/lifecycleRuntime.js index 360f80d..828883c 100644 --- a/llm_normalizer/backend/dist/services/lifecycleRuntime.js +++ b/llm_normalizer/backend/dist/services/lifecycleRuntime.js @@ -127,53 +127,53 @@ const LIFECYCLE_DOMAIN_MODELS = { states: [ { state_code: "initiated_payment", - state_label: "Платеж инициирован", + state_label: "Платеж инициирован", state_class: "initial", entry_conditions: ["payment_order_created"], exit_conditions: ["bank_recorded"], is_terminal: false, is_problematic: false, - business_meaning: "Есть инициирование платежа." + business_meaning: "Есть инициирование платежа." }, { state_code: "bank_recorded", - state_label: "Платеж отражен банком", + state_label: "Платеж отражен банком", state_class: "active", entry_conditions: ["bank_statement_recorded"], exit_conditions: ["settlement_linked"], is_terminal: false, is_problematic: false, - business_meaning: "Движение денег зафиксировано, ожидается расчетное закрытие." + business_meaning: "Движение денег зафиксировано, ожидается расчетное закрытие." }, { state_code: "settlement_closed", - state_label: "Расчет закрыт", + state_label: "Расчет закрыт", state_class: "terminal", entry_conditions: ["payment_to_settlement_linked"], exit_conditions: [], is_terminal: true, is_problematic: false, - business_meaning: "Платеж доведен РґРѕ расчетного результата." + business_meaning: "Платеж доведен до расчетного результата." }, { state_code: "stale_unlinked_payment", - state_label: "Платеж завис без закрытия", + state_label: "Платеж завис без закрытия", state_class: "problematic", entry_conditions: ["bank_recorded", "missing_link"], exit_conditions: ["settlement_closed"], is_terminal: false, is_problematic: true, - business_meaning: "Платеж отражен, РЅРѕ ожидаемая СЃРІСЏР·СЊ РїРѕ расчету РЅРµ завершена." + business_meaning: "Платеж отражен, но ожидаемая связь по расчету не завершена." }, { state_code: "misclosed_payment", - state_label: "Платеж закрыт некорректно", + state_label: "Платеж закрыт некорректно", state_class: "problematic", entry_conditions: ["wrong_document_type_or_posting_mismatch"], exit_conditions: ["settlement_closed"], is_terminal: false, is_problematic: true, - business_meaning: "Формальное закрытие есть, РЅРѕ путь закрытия неверный." + business_meaning: "Формальное закрытие есть, но путь закрытия неверный." } ], transitions: [ @@ -184,7 +184,7 @@ const LIFECYCLE_DOMAIN_MODELS = { required_evidence: ["bank_statement_recorded"], optional_evidence: ["payment_order"], forbidden_conditions: [], - business_meaning: "Платеж должен появиться РІРѕ выписке." + business_meaning: "Платеж должен появиться во выписке." }, { from_state: "bank_recorded", @@ -193,7 +193,7 @@ const LIFECYCLE_DOMAIN_MODELS = { required_evidence: ["payment_to_settlement_link"], optional_evidence: ["document_to_posting"], forbidden_conditions: ["wrong_document_type"], - business_meaning: "После выписки должен закрываться расчет." + business_meaning: "После выписки должен закрываться расчет." } ], defects: [] @@ -205,43 +205,43 @@ const LIFECYCLE_DOMAIN_MODELS = { states: [ { state_code: "invoice_issued", - state_label: "Реализация отражена", + state_label: "Реализация отражена", state_class: "initial", entry_conditions: ["realization_document_exists"], exit_conditions: ["payment_recorded"], is_terminal: false, is_problematic: false, - business_meaning: "Возникла дебиторская позиция." + business_meaning: "Возникла дебиторская позиция." }, { state_code: "payment_recorded", - state_label: "Оплата отражена", + state_label: "Оплата отражена", state_class: "active", entry_conditions: ["payment_document_exists"], exit_conditions: ["receivable_closed"], is_terminal: false, is_problematic: false, - business_meaning: "Оплата есть, ожидается корректное закрытие." + business_meaning: "Оплата есть, ожидается корректное закрытие." }, { state_code: "receivable_closed", - state_label: "Дебиторка закрыта", + state_label: "Дебиторка закрыта", state_class: "terminal", entry_conditions: ["closing_document_linked"], exit_conditions: [], is_terminal: true, is_problematic: false, - business_meaning: "Дебиторская позиция закрыта корректно." + business_meaning: "Дебиторская позиция закрыта корректно." }, { state_code: "stale_receivable", - state_label: "Дебиторка зависла", + state_label: "Дебиторка зависла", state_class: "problematic", entry_conditions: ["unresolved_settlement"], exit_conditions: ["receivable_closed"], is_terminal: false, is_problematic: true, - business_meaning: "Позиция остается незавершенной дольше ожидаемого." + business_meaning: "Позиция остается незавершенной дольше ожидаемого." } ], transitions: [ @@ -252,7 +252,7 @@ const LIFECYCLE_DOMAIN_MODELS = { required_evidence: ["payment_document_exists"], optional_evidence: [], forbidden_conditions: [], - business_meaning: "После реализации ожидается оплата/зачет." + business_meaning: "После реализации ожидается оплата/зачет." }, { from_state: "payment_recorded", @@ -261,7 +261,7 @@ const LIFECYCLE_DOMAIN_MODELS = { required_evidence: ["closing_document_linked"], optional_evidence: ["register_movement_exists"], forbidden_conditions: ["cross_branch_inconsistency"], - business_meaning: "Оплата должна завершаться корректным закрытием расчета." + business_meaning: "Оплата должна завершаться корректным закрытием расчета." } ], defects: [] @@ -273,43 +273,43 @@ const LIFECYCLE_DOMAIN_MODELS = { states: [ { state_code: "recognized", - state_label: "РБП признан", + state_label: "РБП признан", state_class: "initial", entry_conditions: ["deferred_expense_created"], exit_conditions: ["writeoff_started"], is_terminal: false, is_problematic: false, - business_meaning: "РБП поставлен РЅР° учет." + business_meaning: "РБП поставлен на учет." }, { state_code: "partially_written_off", - state_label: "Частичное списание", + state_label: "Частичное списание", state_class: "active", entry_conditions: ["partial_writeoff_exists"], exit_conditions: ["fully_written_off"], is_terminal: false, is_problematic: false, - business_meaning: "Списание идет РїРѕ графику." + business_meaning: "Списание идет по графику." }, { state_code: "fully_written_off", - state_label: "РБП полностью списан", + state_label: "РБП полностью списан", state_class: "terminal", entry_conditions: ["full_writeoff_exists"], exit_conditions: [], is_terminal: true, is_problematic: false, - business_meaning: "РБП завершил lifecycle." + business_meaning: "РБП завершил lifecycle." }, { state_code: "overdue_writeoff", - state_label: "Просроченное списание", + state_label: "Просроченное списание", state_class: "problematic", entry_conditions: ["period_boundary", "missing_link"], exit_conditions: ["fully_written_off"], is_terminal: false, is_problematic: true, - business_meaning: "РБП живет дольше допустимого РѕРєРЅР°." + business_meaning: "РБП живет дольше допустимого окна." } ], transitions: [], @@ -322,53 +322,53 @@ const LIFECYCLE_DOMAIN_MODELS = { states: [ { state_code: "capitalized", - state_label: "Капвложения отражены", + state_label: "Капвложения отражены", state_class: "initial", entry_conditions: ["capitalization_document_exists"], exit_conditions: ["accepted_for_accounting"], is_terminal: false, is_problematic: false, - business_meaning: "Объект зафиксирован как вложение." + business_meaning: "Объект зафиксирован как вложение." }, { state_code: "accepted_for_accounting", - state_label: "РџСЂРёРЅСЏС‚ Рє учету", + state_label: "Принят к учету", state_class: "active", entry_conditions: ["acceptance_document_exists"], exit_conditions: ["depreciation_active"], is_terminal: false, is_problematic: false, - business_meaning: "Объект переведен РІ РѕСЃРЅРѕРІРЅРѕР№ контур учета." + business_meaning: "Объект переведен в основной контур учета." }, { state_code: "depreciation_active", - state_label: "Амортизация активна", + state_label: "Амортизация активна", state_class: "active", entry_conditions: ["depreciation_register_movement"], exit_conditions: ["disposed"], is_terminal: false, is_problematic: false, - business_meaning: "Жизненный цикл РћРЎ идет штатно." + business_meaning: "Жизненный цикл ОС идет штатно." }, { state_code: "contradictory_asset_state", - state_label: "Противоречивый статус РћРЎ", + state_label: "Противоречивый статус ОС", state_class: "problematic", entry_conditions: ["posting_mismatch_or_wrong_path"], exit_conditions: ["depreciation_active"], is_terminal: false, is_problematic: true, - business_meaning: "Статус РћРЎ формально есть, РЅРѕ смыслово противоречив." + business_meaning: "Статус ОС формально есть, но смыслово противоречив." }, { state_code: "disposed", - state_label: "Выбыл", + state_label: "Выбыл", state_class: "terminal", entry_conditions: ["disposal_document_exists"], exit_conditions: [], is_terminal: true, is_problematic: false, - business_meaning: "Жизненный цикл РћРЎ завершен." + business_meaning: "Жизненный цикл ОС завершен." } ], transitions: [], @@ -381,43 +381,43 @@ const LIFECYCLE_DOMAIN_MODELS = { states: [ { state_code: "vat_registered", - state_label: "НДС отражен документно", + state_label: "НДС отражен документно", state_class: "initial", entry_conditions: ["invoice_registered"], exit_conditions: ["vat_reflected"], is_terminal: false, is_problematic: false, - business_meaning: "Сформирован первичный документный слой НДС." + business_meaning: "Сформирован первичный документный слой НДС." }, { state_code: "vat_reflected", - state_label: "НДС отражен РІ учете", + state_label: "НДС отражен в учете", state_class: "active", entry_conditions: ["vat_register_movement"], exit_conditions: ["vat_deducted"], is_terminal: false, is_problematic: false, - business_meaning: "НДС РїСЂРѕС…РѕРґРёС‚ штатную стадию отражения." + business_meaning: "НДС проходит штатную стадию отражения." }, { state_code: "vat_deducted", - state_label: "НДС РїСЂРёРЅСЏС‚ Рє вычету", + state_label: "НДС принят к вычету", state_class: "terminal", entry_conditions: ["deduction_confirmed"], exit_conditions: [], is_terminal: true, is_problematic: false, - business_meaning: "НДС-цепочка завершена корректно." + business_meaning: "НДС-цепочка завершена корректно." }, { state_code: "vat_conflict", - state_label: "Конфликт НДС-цепочки", + state_label: "Конфликт НДС-цепочки", state_class: "problematic", entry_conditions: ["cross_branch_inconsistency"], exit_conditions: ["vat_reflected"], is_terminal: false, is_problematic: true, - business_meaning: "Бухгалтерская Рё налоговая ветки расходятся." + business_meaning: "Бухгалтерская и налоговая ветки расходятся." } ], transitions: [], @@ -430,53 +430,53 @@ const LIFECYCLE_DOMAIN_MODELS = { states: [ { state_code: "preclose_checks", - state_label: "Предзакрытие", + state_label: "Предзакрытие", state_class: "active", entry_conditions: ["period_scope_detected"], exit_conditions: ["close_ready"], is_terminal: false, is_problematic: false, - business_meaning: "Идет проверка готовности периода." + business_meaning: "Идет проверка готовности периода." }, { state_code: "close_ready", - state_label: "Готов Рє закрытию", + state_label: "Готов к закрытию", state_class: "active", entry_conditions: ["no_blockers_detected"], exit_conditions: ["close_completed"], is_terminal: false, is_problematic: false, - business_meaning: "Период может быть закрыт." + business_meaning: "Период может быть закрыт." }, { state_code: "close_completed", - state_label: "Закрытие завершено", + state_label: "Закрытие завершено", state_class: "terminal", entry_conditions: ["close_operation_done"], exit_conditions: [], is_terminal: true, is_problematic: false, - business_meaning: "Период закрыт." + business_meaning: "Период закрыт." }, { state_code: "close_blocked", - state_label: "Закрытие заблокировано", + state_label: "Закрытие заблокировано", state_class: "problematic", entry_conditions: ["period_close_risk_or_stale_state"], exit_conditions: ["close_ready"], is_terminal: false, is_problematic: true, - business_meaning: "Есть lifecycle-дефекты, влияющие РЅР° закрытие." + business_meaning: "Есть lifecycle-дефекты, влияющие на закрытие." }, { state_code: "close_contradicted", - state_label: "Закрыт формально, РЅРѕ СЃ противоречием", + state_label: "Закрыт формально, но с противоречием", state_class: "problematic", entry_conditions: ["misclosed_or_cross_branch_conflict"], exit_conditions: ["close_completed"], is_terminal: false, is_problematic: true, - business_meaning: "Формальное закрытие РЅРµ согласовано СЃ фактическими ветками." + business_meaning: "Формальное закрытие не согласовано с фактическими ветками." } ], transitions: [], @@ -488,7 +488,7 @@ const SHARED_DEFECTS = [ defect_code: "missing_expected_transition", defect_class: "path", severity_hint: "medium", - business_meaning: "Ожидаемый переход РЅРµ произошел.", + business_meaning: "Ожидаемый переход не произошел.", evidence_requirements: ["expected_state", "missing_transition_signal"], period_impact_potential: "indirect" }, @@ -496,7 +496,7 @@ const SHARED_DEFECTS = [ defect_code: "invalid_transition", defect_class: "path", severity_hint: "high", - business_meaning: "Переход произошел РїРѕ некорректному пути.", + business_meaning: "Переход произошел по некорректному пути.", evidence_requirements: ["invalid_transition_signal"], period_impact_potential: "indirect" }, @@ -504,7 +504,7 @@ const SHARED_DEFECTS = [ defect_code: "stale_active_state", defect_class: "timing", severity_hint: "high", - business_meaning: "Объект завис РІ активном состоянии.", + business_meaning: "Объект завис в активном состоянии.", evidence_requirements: ["stale_marker", "missing_transition_signal"], period_impact_potential: "direct" }, @@ -512,7 +512,7 @@ const SHARED_DEFECTS = [ defect_code: "contradictory_state", defect_class: "consistency", severity_hint: "high", - business_meaning: "Статусы объекта противоречат РґСЂСѓРі РґСЂСѓРіСѓ.", + business_meaning: "Статусы объекта противоречат друг другу.", evidence_requirements: ["contradiction_signal"], period_impact_potential: "direct" }, @@ -520,7 +520,7 @@ const SHARED_DEFECTS = [ defect_code: "premature_terminal_state", defect_class: "closure", severity_hint: "medium", - business_meaning: "Терминальное состояние наступило преждевременно.", + business_meaning: "Терминальное состояние наступило преждевременно.", evidence_requirements: ["terminal_state", "missing_required_previous_state"], period_impact_potential: "indirect" }, @@ -528,7 +528,7 @@ const SHARED_DEFECTS = [ defect_code: "misclosed_state", defect_class: "closure", severity_hint: "high", - business_meaning: "Контур формально закрыт, РЅРѕ закрыт неверно.", + business_meaning: "Контур формально закрыт, но закрыт неверно.", evidence_requirements: ["wrong_closure_path"], period_impact_potential: "direct" }, @@ -536,7 +536,7 @@ const SHARED_DEFECTS = [ defect_code: "orphan_intermediate_state", defect_class: "path", severity_hint: "medium", - business_meaning: "Промежуточная стадия осталась без корректного продолжения.", + business_meaning: "Промежуточная стадия осталась без корректного продолжения.", evidence_requirements: ["intermediate_state_without_next"], period_impact_potential: "indirect" }, @@ -544,7 +544,7 @@ const SHARED_DEFECTS = [ defect_code: "cross_branch_state_conflict", defect_class: "consistency", severity_hint: "high", - business_meaning: "Состояния соседних веток учета противоречат РґСЂСѓРі РґСЂСѓРіСѓ.", + business_meaning: "Состояния соседних веток учета противоречат друг другу.", evidence_requirements: ["cross_branch_conflict_signal"], period_impact_potential: "direct" } @@ -845,23 +845,23 @@ function staleDurationHint(domain, defect, input) { return "unknown_snapshot_window"; } function lifecycleInterpretation(input) { - const base = `Текущая стадия: ${input.currentState}; ожидаемая стадия: ${input.expectedState}.`; + const base = `Текущая стадия: ${input.currentState}; ожидаемая стадия: ${input.expectedState}.`; if (input.defect === "stale_active_state") { - return `${base} Объект завис РІРѕ времени Рё РЅРµ дошел РґРѕ ожидаемого перехода.`; + return `${base} Объект завис во времени и не дошел до ожидаемого перехода.`; } if (input.defect === "misclosed_state") { - return `${base} Контур закрыт формально, РЅРѕ путь закрытия противоречит бухгалтерской логике.`; + return `${base} Контур закрыт формально, но путь закрытия противоречит бухгалтерской логике.`; } if (input.defect === "cross_branch_state_conflict") { - return `${base} Между ветками домена ${input.domain} обнаружено противоречие состояний.`; + return `${base} Между ветками домена ${input.domain} обнаружено противоречие состояний.`; } if (input.defect === "missing_expected_transition") { - return `${base} РќРµ зафиксирован ожидаемый переход (${input.missingTransition ?? "unknown_transition"}).`; + return `${base} Не зафиксирован ожидаемый переход (${input.missingTransition ?? "unknown_transition"}).`; } if (input.defect === "invalid_transition") { - return `${base} Зафиксирован некорректный переход (${input.invalidTransition ?? "invalid_transition"}).`; + return `${base} Зафиксирован некорректный переход (${input.invalidTransition ?? "invalid_transition"}).`; } - return `${base} Lifecycle-разрешение РЅРµ выявило критичный дефект, РЅРѕ состояние требует наблюдения.`; + return `${base} Lifecycle-разрешение не выявило критичный дефект, но состояние требует наблюдения.`; } function resolveLifecycle(input) { const lifecycle_domain = inferLifecycleDomain(input); diff --git a/llm_normalizer/backend/dist/services/promptBuilder.js b/llm_normalizer/backend/dist/services/promptBuilder.js index 529235f..3848e75 100644 --- a/llm_normalizer/backend/dist/services/promptBuilder.js +++ b/llm_normalizer/backend/dist/services/promptBuilder.js @@ -67,9 +67,9 @@ const BUILTIN_PROMPT_PRESETS = { }, normalizer_v1_1_2_1: { id: "default-normalizer-v1_1_2_1", - name: "Стандартный пресет NDC v1.1.2.1", + name: "Стандартный пресет NDC v1.1.2.1", promptVersion: "normalizer_v1_1_2_1", - schemaNotes: "v1.1.2.1: stable prompt baseline v1.1.2 + accounting-review phrasing anchors for 30-case validation pack. Схема normalized_query_v1 без изменений.", + schemaNotes: "v1.1.2.1: stable prompt baseline v1.1.2 + accounting-review phrasing anchors for 30-case validation pack. Схема normalized_query_v1 без изменений.", files: { system: path_1.default.join("system", "default.txt"), developer: path_1.default.join("developer", "normalizer_v1_1_2_1.txt"), @@ -79,9 +79,9 @@ const BUILTIN_PROMPT_PRESETS = { }, normalizer_v2: { id: "default-normalizer-v2", - name: "Стандартный пресет NDC v2", + name: "Стандартный пресет NDC v2", promptVersion: "normalizer_v2", - schemaNotes: "v2: decomposition-first pre-router. LLM returns fragments + scope + flags; deterministic routing happens in code. Схема normalized_query_v2.", + schemaNotes: "v2: decomposition-first pre-router. LLM returns fragments + scope + flags; deterministic routing happens in code. Схема normalized_query_v2.", files: { system: path_1.default.join("system", "default.txt"), developer: path_1.default.join("developer", "normalizer_v2.txt"), diff --git a/llm_normalizer/backend/dist/services/routeHintAdapter.js b/llm_normalizer/backend/dist/services/routeHintAdapter.js index e8c6f83..3d04d68 100644 --- a/llm_normalizer/backend/dist/services/routeHintAdapter.js +++ b/llm_normalizer/backend/dist/services/routeHintAdapter.js @@ -38,7 +38,7 @@ const PERIOD_IMPACT_PATTERN = /(?:period\s*close|month\s*close|month-end|residua const CAUSAL_PATTERN = /(?:\bwhy\b|\bbecause\b|\breason\b|explain\s+mechanism|почему|объясни|механизм|причин)/i; const AMBIGUITY_PATTERN = /(?:\bmaybe\b|\bperhaps\b|not\s+sure|i\s+only\s+know|part\s+may\s+be\s+missing|возможно|может\s+быть|не\s+уверен|не\s+знаю|часть\s+цепочки\s+не\s+подтвержд)/i; const TRANSLIT_PROBLEM_PATTERN = /(?:raschet|oplata|zakryt|nds|vychet|zatrat|ostatok|cepoch|perehod|pochemu|prichin|period)/i; -const DOMAIN_LEXICAL_ANCHOR_PATTERN = /(?:\b(?:settlement|payment|bank|supplier|customer|vat|nds|invoice|register|book|period\s*close|month\s*close|close\s*operation|allocation|residual|cost|expenses?)\b|оплат|расчет|РЅРґСЃ|СЃС‡[её]С‚.?фактур|РєРЅРёРі[аи]|затрат|закрыт|остатк)/i; +const DOMAIN_LEXICAL_ANCHOR_PATTERN = /(?:\b(?:settlement|payment|bank|supplier|customer|vat|nds|invoice|register|book|period\s*close|month\s*close|close\s*operation|allocation|residual|cost|expenses?)\b|оплат|расчет|ндс|сч[её]С‚.?фактур|книг[аи]|затрат|закрыт|остатк)/i; exports.ROUTE_DISCIPLINE_RULE_TABLE = [ { query_class: "exact_object_trace", diff --git a/llm_normalizer/backend/src/config.ts b/llm_normalizer/backend/src/config.ts index 952b59a..3dc0a74 100644 --- a/llm_normalizer/backend/src/config.ts +++ b/llm_normalizer/backend/src/config.ts @@ -134,9 +134,28 @@ export const TRACES_DIR = path.resolve(DATA_DIR, "traces"); export const PRESETS_DIR = path.resolve(DATA_DIR, "presets"); export const EVAL_CASES_DIR = path.resolve(DATA_DIR, "eval_cases"); export const ASSISTANT_SESSIONS_DIR = path.resolve(DATA_DIR, "assistant_sessions"); +export const AUTORUN_ANNOTATIONS_DIR = path.resolve(DATA_DIR, "autorun_annotations"); +export const AUTORUN_ANNOTATIONS_FILE = path.resolve(AUTORUN_ANNOTATIONS_DIR, "annotations.json"); +export const AUTORUN_GENERATOR_DIR = path.resolve(DATA_DIR, "autorun_generators"); +export const AUTORUN_GENERATOR_HISTORY_FILE = path.resolve(AUTORUN_GENERATOR_DIR, "history.json"); export const PROMPTS_DIR = path.resolve(MODULE_ROOT, "prompts"); export const REPORTS_DIR = path.resolve(MODULE_ROOT, "reports"); export const EVAL_DATASETS_DIR = path.resolve(MODULE_ROOT, "eval_cases"); export const SCHEMAS_DIR = path.resolve(BACKEND_ROOT, "src", "schemas"); export const ARCH_EXPORT_2020_DIR = path.resolve(MODULE_ROOT, "..", "docs", "ARCH", "2020экспорт"); +export const ASSISTANT_CANON_FILE = path.resolve(MODULE_ROOT, "..", "docs", "TECH", "assistant_canon.md"); +export const ASSISTANT_CAPABILITIES_REGISTRY_FILE = path.resolve( + MODULE_ROOT, + "..", + "docs", + "TECH", + "capabilities_registry.json" +); +export const MANUAL_CASE_DECISION_SCHEMA_FILE = path.resolve( + MODULE_ROOT, + "..", + "docs", + "TECH", + "manual_case_decision_schema.json" +); diff --git a/llm_normalizer/backend/src/routes/autoRuns.ts b/llm_normalizer/backend/src/routes/autoRuns.ts index 771c34b..deda1aa 100644 --- a/llm_normalizer/backend/src/routes/autoRuns.ts +++ b/llm_normalizer/backend/src/routes/autoRuns.ts @@ -1,11 +1,55 @@ import fs from "fs"; import path from "path"; import { Router } from "express"; -import { ASSISTANT_SESSIONS_DIR, EVAL_CASES_DIR, REPORTS_DIR } from "../config"; +import { + ASSISTANT_SESSIONS_DIR, + AUTORUN_ANNOTATIONS_FILE, + AUTORUN_GENERATOR_HISTORY_FILE, + EVAL_CASES_DIR, + EVAL_DATASETS_DIR, + MANUAL_CASE_DECISION_SCHEMA_FILE, + REPORTS_DIR +} from "../config"; import { ApiError, ok } from "../utils/http"; +import { loadCapabilitiesRegistry, resolveNearestCapabilityGroup } from "../services/capabilitiesRegistry"; type AutoRunTarget = "normalizer" | "assistant_stage1" | "assistant_stage2" | "assistant_p0" | "unknown"; type AutoRunTrend = "up" | "down" | "flat"; +type AutoGenMode = "qwen_seed" | "codex_creative"; +type ManualCaseDecision = + | "covered_ok" + | "covered_but_bad_answer" + | "candidate_for_implementation" + | "needs_routing_extension" + | "out_of_scope_but_answer_softly" + | "unsafe_question_limit_strictly" + | "needs_dialog_policy_fix" + | "needs_capability_registry_update" + | "bad_test_case"; + +const MANUAL_CASE_DECISIONS: ManualCaseDecision[] = [ + "covered_ok", + "covered_but_bad_answer", + "candidate_for_implementation", + "needs_routing_extension", + "out_of_scope_but_answer_softly", + "unsafe_question_limit_strictly", + "needs_dialog_policy_fix", + "needs_capability_registry_update", + "bad_test_case" +]; + +const DECISION_QUEUE_MAP: Record = { + covered_ok: "none", + covered_but_bad_answer: "policy_fix", + candidate_for_implementation: "routing_extension", + needs_routing_extension: "routing_extension", + out_of_scope_but_answer_softly: "soft_boundary", + unsafe_question_limit_strictly: "safety_policy", + needs_dialog_policy_fix: "policy_fix", + needs_capability_registry_update: "capability_registry", + bad_test_case: "testset_hygiene" +}; interface IndexedRun { run_id: string; @@ -71,6 +115,9 @@ interface CaseSummary { reply_type: string | null; session_id: string; dialog_available: boolean; + commented_count: number; + latest_annotation_at: string | null; + avg_rating: number | null; checks: Record | null; metric_subscores: Record | null; } @@ -87,6 +134,53 @@ interface HistoryStats { domain_coverage: DomainCoverage[]; } +interface AutoRunAnnotationRecord { + annotation_id: string; + run_id: string; + case_id: string; + session_id: string; + message_index: number; + rating: number; + comment: string; + manual_case_decision: ManualCaseDecision; + annotation_author: string | null; + created_at: string; + updated_at: string; + context: { + message_id: string | null; + trace_id: string | null; + reply_type: string | null; + eval_target: AutoRunTarget | "unknown"; + prompt_version: string | null; + domain: string | null; + query_class: string | null; + }; +} + +interface AnnotationStatsByCase { + count: number; + latest_at: string | null; + avg_rating: number | null; +} + +interface AutoGenHistoryRecord { + generation_id: string; + created_at: string; + mode: AutoGenMode; + count: number; + domain: string | null; + questions: string[]; + generated_by: string | null; + saved_case_set_file: string | null; + context: { + llm_provider: string | null; + model: string | null; + assistant_prompt_version: string | null; + decomposition_prompt_version: string | null; + prompt_fingerprint: string | null; + } | null; +} + function toRecord(value: unknown): Record | null { if (!value || typeof value !== "object" || Array.isArray(value)) { return null; @@ -148,6 +242,311 @@ function clampInt(value: number | null, min: number, max: number, fallback: numb return rounded; } +function parseManualCaseDecision(value: unknown, fallback: ManualCaseDecision = "needs_dialog_policy_fix"): ManualCaseDecision { + const normalized = toStringSafe(value); + if (!normalized) return fallback; + return (MANUAL_CASE_DECISIONS.includes(normalized as ManualCaseDecision) ? normalized : fallback) as ManualCaseDecision; +} + +function parseAnnotationAuthor(value: unknown): string | null { + const author = toStringSafe(value); + if (!author) return null; + return author.slice(0, 80); +} + +function readManualDecisionSchema(): Record { + const fallback: Record = { + schema_version: "manual_case_decision_schema_v1_fallback", + enum: MANUAL_CASE_DECISIONS, + labels: { + covered_ok: "Покрыто и ок", + covered_but_bad_answer: "Покрыто, но ответ плохой", + candidate_for_implementation: "Кандидат на внедрение", + needs_routing_extension: "Нужно расширение маршрутизации", + out_of_scope_but_answer_softly: "Вне скоупа, но нужен мягкий ответ", + unsafe_question_limit_strictly: "Высокий риск, строгие ограничения", + needs_dialog_policy_fix: "Нужен фикс диалоговой политики", + needs_capability_registry_update: "Нужно обновить реестр возможностей", + bad_test_case: "Плохой тест-кейс" + }, + queue_mapping: DECISION_QUEUE_MAP + }; + if (!fs.existsSync(MANUAL_CASE_DECISION_SCHEMA_FILE)) { + return fallback; + } + try { + const parsed = JSON.parse(fs.readFileSync(MANUAL_CASE_DECISION_SCHEMA_FILE, "utf-8")) as unknown; + const record = toRecord(parsed); + return record ?? fallback; + } catch { + return fallback; + } +} + +function readAutoGenHistory(): AutoGenHistoryRecord[] { + if (!fs.existsSync(AUTORUN_GENERATOR_HISTORY_FILE)) return []; + try { + const parsed = JSON.parse(fs.readFileSync(AUTORUN_GENERATOR_HISTORY_FILE, "utf-8")) as unknown; + if (!Array.isArray(parsed)) return []; + return parsed + .map((item) => toRecord(item)) + .filter((item): item is Record => item !== null) + .map((item) => ({ + generation_id: toStringSafe(item.generation_id) ?? "", + created_at: toStringSafe(item.created_at) ?? new Date().toISOString(), + mode: (toStringSafe(item.mode) as AutoGenMode | null) ?? "codex_creative", + count: clampInt(toNumberSafe(item.count), 1, 300, 20), + domain: toStringSafe(item.domain), + questions: toArray(item.questions) + .map((q) => toStringSafe(q)) + .filter((q): q is string => q !== null) + .slice(0, 500), + generated_by: toStringSafe(item.generated_by), + saved_case_set_file: toStringSafe(item.saved_case_set_file), + context: toRecord(item.context) + ? { + llm_provider: toStringSafe(toRecord(item.context)?.llm_provider), + model: toStringSafe(toRecord(item.context)?.model), + assistant_prompt_version: toStringSafe(toRecord(item.context)?.assistant_prompt_version), + decomposition_prompt_version: toStringSafe(toRecord(item.context)?.decomposition_prompt_version), + prompt_fingerprint: toStringSafe(toRecord(item.context)?.prompt_fingerprint) + } + : null + })) + .filter((item) => item.generation_id.length > 0) + .sort((a, b) => Date.parse(b.created_at) - Date.parse(a.created_at)); + } catch { + return []; + } +} + +function writeAutoGenHistory(records: AutoGenHistoryRecord[]): void { + const dir = path.dirname(AUTORUN_GENERATOR_HISTORY_FILE); + if (!fs.existsSync(dir)) { + fs.mkdirSync(dir, { recursive: true }); + } + fs.writeFileSync(AUTORUN_GENERATOR_HISTORY_FILE, JSON.stringify(records, null, 2), "utf-8"); +} + +function readEvalDatasetCases(filePath: string): Array> { + try { + const parsed = JSON.parse(fs.readFileSync(filePath, "utf-8")) as unknown; + if (Array.isArray(parsed)) { + return parsed.map((item) => toRecord(item)).filter((item): item is Record => item !== null); + } + const record = toRecord(parsed); + if (!record) return []; + const cases = toArray(record.cases).map((item) => toRecord(item)).filter((item): item is Record => item !== null); + return cases; + } catch { + return []; + } +} + +function collectCanonicalQuestions(limit = 300): string[] { + if (!fs.existsSync(EVAL_DATASETS_DIR)) { + return []; + } + const entries = fs.readdirSync(EVAL_DATASETS_DIR, { withFileTypes: true }); + const questions: string[] = []; + for (const entry of entries) { + if (!entry.isFile() || !entry.name.endsWith(".json")) continue; + const fullPath = path.resolve(EVAL_DATASETS_DIR, entry.name); + const cases = readEvalDatasetCases(fullPath); + for (const testCase of cases) { + const rawQuestion = toStringSafe(testCase.raw_question) ?? toStringSafe(testCase.user_message) ?? toStringSafe(testCase.query); + if (rawQuestion) { + questions.push(rawQuestion); + } + } + } + return Array.from(new Set(questions)).slice(0, limit); +} + +function normalizeDomainHint(value: unknown): string | null { + const domain = toStringSafe(value); + if (!domain) return null; + return domain.toLowerCase(); +} + +function fallbackDomainTemplates(domain: string | null): string[] { + if (domain?.includes("vat") || domain?.includes("ндс")) { + return [ + "Сколько НДС к уплате на дату по организации?", + "Покажи прогноз НДС за период по организации.", + "Почему по НДС сейчас ноль и из чего сложился расчет?" + ]; + } + if (domain?.includes("counter") || domain?.includes("контраг")) { + return [ + "Покажи топ контрагентов по сумме платежей за период.", + "Какой самый крупный договор у выбранной организации?", + "Какие документы были по контрагенту за весь период?" + ]; + } + if (domain?.includes("settlement") || domain?.includes("задолж") || domain?.includes("расчет")) { + return [ + "Какие незакрытые расчеты висят на конец периода?", + "Есть ли незакрытые авансы по поставщикам?", + "Покажи цепочки закрытия по счетам 60/62." + ]; + } + return [ + "С какой организацией сейчас можно работать в активном контуре?", + "Покажи ключевые операции за выбранный период.", + "Какие вопросы по этому домену ассистент поддерживает прямо сейчас?" + ]; +} + +function mutateIntoQwenStyle(base: string, index: number): string { + const wrappers = ["йо ", "слушай ", "подскажи плиз ", "короче ", "мож ", "а ну-ка "]; + const tails = ["", " без воды", " по факту", " и коротко", " прям сейчас", " за весь период"]; + const typoMap: Array<[RegExp, string]> = [ + [/\bкомпания\b/gi, "компиния"], + [/\bсейчас\b/gi, "щас"], + [/\bпожалуйста\b/gi, "плиз"], + [/\bкакая\b/gi, "кака"], + [/\bчто\b/gi, "че"] + ]; + const prefix = wrappers[index % wrappers.length]; + const tail = tails[index % tails.length]; + let text = `${prefix}${base}${tail}`.trim(); + if (index % 2 === 0) { + const [pattern, replacement] = typoMap[index % typoMap.length]; + text = text.replace(pattern, replacement); + } + return text; +} + +function generateQwenSeedQuestions(count: number, domain: string | null): string[] { + const seed = collectCanonicalQuestions(450); + const source = seed.length > 0 ? seed : fallbackDomainTemplates(domain); + const filtered = domain + ? source.filter((item) => item.toLowerCase().includes(domain) || fallbackDomainTemplates(domain).includes(item)) + : source; + const bag = filtered.length > 0 ? filtered : source; + const out: string[] = []; + for (let index = 0; index < count; index += 1) { + const base = bag[index % bag.length]; + out.push(mutateIntoQwenStyle(base, index)); + } + return Array.from(new Set(out)).slice(0, count); +} + +function generateCodexCreativeQuestions(count: number, domain: string | null): string[] { + const domainTemplates = fallbackDomainTemplates(domain); + const patterns = [ + "Дай бизнес-срез по состоянию на дату: {q}", + "Нужен аккуратный ответ как бухгалтеру: {q}", + "Если данных не хватает, скажи что уточнить, но сначала попробуй: {q}", + "Сформулируй результат без технички и с шагом дальше: {q}", + "Проверь в read-only и скажи что видно: {q}" + ]; + const out: string[] = []; + for (let index = 0; index < count; index += 1) { + const base = domainTemplates[index % domainTemplates.length]; + const pattern = patterns[index % patterns.length]; + out.push(pattern.replace("{q}", base)); + } + return Array.from(new Set(out)).slice(0, count); +} + +function generateAutogenId(): string { + return `gen-${Date.now().toString(36)}-${Math.random().toString(36).slice(2, 9)}`; +} + +function readAnnotations(): AutoRunAnnotationRecord[] { + if (!fs.existsSync(AUTORUN_ANNOTATIONS_FILE)) { + return []; + } + try { + const raw = fs.readFileSync(AUTORUN_ANNOTATIONS_FILE, "utf-8"); + const parsed = JSON.parse(raw) as unknown; + if (!Array.isArray(parsed)) { + return []; + } + return parsed + .map((item) => toRecord(item)) + .filter((item): item is Record => item !== null) + .map((item) => { + const context = toRecord(item.context); + return { + annotation_id: toStringSafe(item.annotation_id) ?? "", + run_id: toStringSafe(item.run_id) ?? "", + case_id: toStringSafe(item.case_id) ?? "", + session_id: toStringSafe(item.session_id) ?? "", + message_index: clampInt(toNumberSafe(item.message_index), 0, 100_000, 0), + rating: clampInt(toNumberSafe(item.rating), 1, 5, 1), + comment: toStringSafe(item.comment) ?? "", + manual_case_decision: parseManualCaseDecision(item.manual_case_decision), + annotation_author: parseAnnotationAuthor(item.annotation_author), + created_at: toStringSafe(item.created_at) ?? new Date().toISOString(), + updated_at: toStringSafe(item.updated_at) ?? new Date().toISOString(), + context: { + message_id: toStringSafe(context?.message_id), + trace_id: toStringSafe(context?.trace_id), + reply_type: toStringSafe(context?.reply_type), + eval_target: (toStringSafe(context?.eval_target) as AutoRunTarget | null) ?? "unknown", + prompt_version: toStringSafe(context?.prompt_version), + domain: toStringSafe(context?.domain), + query_class: toStringSafe(context?.query_class) + } + } satisfies AutoRunAnnotationRecord; + }) + .filter((item) => item.annotation_id && item.run_id && item.case_id); + } catch { + return []; + } +} + +function writeAnnotations(items: AutoRunAnnotationRecord[]): void { + fs.writeFileSync(AUTORUN_ANNOTATIONS_FILE, JSON.stringify(items, null, 2), "utf-8"); +} + +function annotationKey(runId: string, caseId: string, messageIndex: number): string { + return `${runId}::${caseId}::${messageIndex}`; +} + +function buildAnnotationStatsMap(runId: string, annotations: AutoRunAnnotationRecord[]): Map { + const scoped = annotations.filter((item) => item.run_id === runId); + const buckets = new Map(); + for (const item of scoped) { + const bucket = buckets.get(item.case_id) ?? { count: 0, ratings: [], latestMs: null }; + bucket.count += 1; + bucket.ratings.push(item.rating); + const ms = Date.parse(item.updated_at); + if (Number.isFinite(ms) && (bucket.latestMs === null || ms > bucket.latestMs)) { + bucket.latestMs = ms; + } + buckets.set(item.case_id, bucket); + } + + const result = new Map(); + for (const [caseId, bucket] of buckets.entries()) { + const avg = bucket.ratings.length > 0 ? Number((bucket.ratings.reduce((a, b) => a + b, 0) / bucket.ratings.length).toFixed(2)) : null; + result.set(caseId, { + count: bucket.count, + latest_at: bucket.latestMs === null ? null : new Date(bucket.latestMs).toISOString(), + avg_rating: avg + }); + } + return result; +} + +function buildAnnotationsByMessageIndex(runId: string, caseId: string, annotations: AutoRunAnnotationRecord[]): Map { + const map = new Map(); + for (const item of annotations) { + if (item.run_id !== runId || item.case_id !== caseId) continue; + const current = map.get(item.message_index); + const currentMs = current ? Date.parse(current.updated_at) : null; + const nextMs = Date.parse(item.updated_at); + if (!current || (!Number.isNaN(nextMs) && (currentMs === null || nextMs >= currentMs))) { + map.set(item.message_index, item); + } + } + return map; +} + function resolveRunTarget(input: { report: Record; runId: string; reportPath: string }): AutoRunTarget { const explicit = toStringSafe(input.report.eval_target); if (explicit === "assistant_stage1" || explicit === "assistant_stage2" || explicit === "assistant_p0" || explicit === "normalizer") { @@ -313,7 +712,12 @@ function getResultCases(report: Record): Array => item !== null); } -function buildCaseSummaries(report: Record, runId: string, checkDialogAvailability: boolean): CaseSummary[] { +function buildCaseSummaries( + report: Record, + runId: string, + checkDialogAvailability: boolean, + annotationStatsByCase?: Map +): CaseSummary[] { const results = getResultCases(report); return results.map((item, index) => { const caseId = toStringSafe(item.case_id) ?? `case-${index + 1}`; @@ -328,6 +732,7 @@ function buildCaseSummaries(report: Record, runId: string, chec const dialogAvailable = checkDialogAvailability ? fs.existsSync(path.resolve(ASSISTANT_SESSIONS_DIR, `${sessionId}.json`)) : false; + const annotationStats = annotationStatsByCase?.get(caseId); return { case_id: caseId, @@ -339,6 +744,9 @@ function buildCaseSummaries(report: Record, runId: string, chec reply_type: toStringSafe(item.reply_type), session_id: sessionId, dialog_available: dialogAvailable, + commented_count: annotationStats?.count ?? 0, + latest_annotation_at: annotationStats?.latest_at ?? null, + avg_rating: annotationStats?.avg_rating ?? null, checks, metric_subscores: metricSubscores }; @@ -632,6 +1040,7 @@ function loadSessionDialog(runId: string, caseId: string): { .map((item) => toRecord(item)) .filter((item): item is Record => item !== null); const messages = conversation.map((item) => ({ + message_id: toStringSafe(item.message_id), role: toStringSafe(item.role) ?? "unknown", text: toStringSafe(item.text) ?? "", created_at: toStringSafe(item.created_at), @@ -708,6 +1117,7 @@ function buildFallbackDialog(run: IndexedRun, caseId: string): { session_id: sessionId, messages: [ { + message_id: null, role: "user", text: userText, created_at: null, @@ -715,6 +1125,7 @@ function buildFallbackDialog(run: IndexedRun, caseId: string): { reply_type: null }, { + message_id: null, role: "assistant", text: assistantSummaryParts.join("\n"), created_at: null, @@ -727,6 +1138,209 @@ function buildFallbackDialog(run: IndexedRun, caseId: string): { }; } +function withMessageAnnotations( + runId: string, + caseId: string, + messages: Array>, + annotations: AutoRunAnnotationRecord[] +): Array> { + const byIndex = buildAnnotationsByMessageIndex(runId, caseId, annotations); + return messages.map((message, index) => { + const annotation = byIndex.get(index) ?? null; + return { + ...message, + message_index: index, + commented: annotation !== null, + annotation + }; + }); +} + +function generateAnnotationId(): string { + return `ann-${Date.now().toString(36)}-${Math.random().toString(36).slice(2, 9)}`; +} + +function parseComment(value: unknown): string { + const text = toStringSafe(value) ?? ""; + return text.trim(); +} + +function parseDecisionFilter(value: unknown): ManualCaseDecision | "all" { + const normalized = toStringSafe(value); + if (!normalized || normalized === "all") return "all"; + return parseManualCaseDecision(normalized); +} + +function parseAutoGenMode(value: unknown): AutoGenMode { + const normalized = toStringSafe(value)?.toLowerCase() ?? ""; + if (normalized === "qwen_seed" || normalized === "codex_creative") { + return normalized; + } + return "codex_creative"; +} + +function parseAutogenCount(value: unknown): number { + return clampInt(toNumberSafe(value), 1, 200, 24); +} + +function parseAutogenDomain(value: unknown): string | null { + const domain = normalizeDomainHint(value); + if (!domain) return null; + return domain.slice(0, 80); +} + +function hasAnyRunFilterQuery(query: Record): boolean { + return Boolean( + toStringSafe(query.from) ?? + toStringSafe(query.to) ?? + toStringSafe(query.target) ?? + toStringSafe(query.mode) ?? + toStringSafe(query.use_mock) ?? + toStringSafe(query.prompt_contains) + ); +} + +function buildAutogenCaseSetFileName(mode: AutoGenMode, generationId: string): string { + const now = new Date(); + const stamp = [ + now.getUTCFullYear(), + String(now.getUTCMonth() + 1).padStart(2, "0"), + String(now.getUTCDate()).padStart(2, "0"), + String(now.getUTCHours()).padStart(2, "0"), + String(now.getUTCMinutes()).padStart(2, "0"), + String(now.getUTCSeconds()).padStart(2, "0") + ].join(""); + return `assistant_autogen_${mode}_${stamp}_${generationId}.json`; +} + +function buildAutogenCaseSetPayload(input: { + generationId: string; + mode: AutoGenMode; + domain: string | null; + questions: string[]; +}): Record { + const cases = input.questions.map((question, index) => ({ + case_id: `AUTO-${String(index + 1).padStart(3, "0")}`, + scenario_tag: `${input.mode}_${input.domain ?? "general"}`, + question_type: "direct", + broadness_level: "medium", + turns: [{ user_message: question }], + expected_hints: { + expected_reply_type: null, + expected_degraded_to: null + } + })); + return { + suite_id: `assistant_autogen_${input.generationId}`, + suite_version: "0.1.0", + schema_version: "assistant_autogen_suite_v0_1", + generated_at: new Date().toISOString(), + generation_id: input.generationId, + mode: input.mode, + domain: input.domain, + scenario_count: cases.length, + case_ids: cases.map((item) => item.case_id), + cases + }; +} + +function collectPostAnalysis( + annotations: AutoRunAnnotationRecord[], + runMap: Map, + limitPerQueue: number +): Record { + const byDecision: Record = {}; + const byQueue: Record = {}; + const byDomain = new Map(); + + const queues: Record>> = { + routing_extension: [], + policy_fix: [], + capability_registry: [], + soft_boundary: [], + safety_policy: [], + testset_hygiene: [], + covered_ok: [] + }; + + const registry = loadCapabilitiesRegistry(); + + for (const item of annotations) { + byDecision[item.manual_case_decision] = (byDecision[item.manual_case_decision] ?? 0) + 1; + const queueKey = DECISION_QUEUE_MAP[item.manual_case_decision]; + byQueue[queueKey] = (byQueue[queueKey] ?? 0) + 1; + + const run = runMap.get(item.run_id) ?? null; + const caseSummary = run + ? buildCaseSummaries(run.report, run.run_id, false).find((candidate) => candidate.case_id === item.case_id) ?? null + : null; + const nearestGroup = + resolveNearestCapabilityGroup({ + domain: caseSummary?.domain ?? item.context.domain, + queryClass: caseSummary?.query_class ?? item.context.query_class + }) ?? + registry.groups[0] ?? + null; + + const domainKey = caseSummary?.domain ?? item.context.domain ?? "unknown"; + byDomain.set(domainKey, (byDomain.get(domainKey) ?? 0) + 1); + + const view = { + annotation_id: item.annotation_id, + run_id: item.run_id, + case_id: item.case_id, + message_index: item.message_index, + rating: item.rating, + comment: item.comment, + manual_case_decision: item.manual_case_decision, + annotation_author: item.annotation_author, + updated_at: item.updated_at, + domain: caseSummary?.domain ?? item.context.domain ?? null, + query_class: caseSummary?.query_class ?? item.context.query_class ?? null, + trace_id: item.context.trace_id ?? caseSummary?.trace_id ?? null, + reply_type: item.context.reply_type ?? caseSummary?.reply_type ?? null, + nearest_capability_group: nearestGroup + ? { + group_code: nearestGroup.group_code, + group_title: nearestGroup.group_title, + maturity_status: nearestGroup.maturity_status + } + : null + }; + + if (queueKey === "none") { + if (queues.covered_ok.length < limitPerQueue) queues.covered_ok.push(view); + continue; + } + if (!queues[queueKey]) { + queues[queueKey] = []; + } + if (queues[queueKey].length < limitPerQueue) { + queues[queueKey].push(view); + } + } + + const domainSummary = Array.from(byDomain.entries()) + .map(([domain, total]) => ({ domain, total })) + .sort((a, b) => b.total - a.total); + + return { + stats: { + annotations_total: annotations.length, + by_decision: byDecision, + by_queue: byQueue, + domains_total: domainSummary.length + }, + domain_summary: domainSummary, + queues, + recommended_regression_candidates: [ + ...queues.routing_extension.slice(0, 20), + ...queues.policy_fix.slice(0, 20), + ...queues.safety_policy.slice(0, 20) + ].slice(0, 60) + }; +} + export function buildAutoRunsRouter(): Router { const router = Router(); @@ -777,7 +1391,9 @@ export function buildAutoRunsRouter(): Router { if (!run) { throw new ApiError("AUTORUN_NOT_FOUND", `Run not found: ${runId}`, 404); } - const cases = buildCaseSummaries(run.report, run.run_id, true); + const annotations = readAnnotations(); + const annotationStatsByCase = buildAnnotationStatsMap(runId, annotations); + const cases = buildCaseSummaries(run.report, run.run_id, true, annotationStatsByCase); const coverage = buildCoverageFromCases(cases); ok(res, { @@ -785,6 +1401,9 @@ export function buildAutoRunsRouter(): Router { run: buildRunSummary(run), coverage, cases, + annotations_summary: { + total: annotations.filter((item) => item.run_id === runId).length + }, report: run.report }); } catch (error) { @@ -806,11 +1425,325 @@ export function buildAutoRunsRouter(): Router { const sessionDialog = loadSessionDialog(runId, caseId); const dialog = sessionDialog ?? buildFallbackDialog(run, caseId); + const annotations = readAnnotations(); + const messages = withMessageAnnotations(runId, caseId, dialog.messages, annotations); ok(res, { ok: true, run_id: runId, case_id: caseId, - ...dialog + ...dialog, + messages, + annotations: annotations + .filter((item) => item.run_id === runId && item.case_id === caseId) + .sort((a, b) => Date.parse(b.updated_at) - Date.parse(a.updated_at)) + }); + } catch (error) { + next(error); + } + }); + + router.get("/api/autoruns/annotations", (req, res, next) => { + try { + const runIdFilter = toStringSafe((req.query as Record).run_id); + const caseIdFilter = toStringSafe((req.query as Record).case_id); + const minRatingRaw = toNumberSafe((req.query as Record).min_rating); + const minRating = minRatingRaw === null ? null : clampInt(minRatingRaw, 1, 5, 1); + const decisionFilter = parseDecisionFilter((req.query as Record).manual_case_decision); + const limit = clampInt(toNumberSafe((req.query as Record).limit), 1, 2000, 400); + const scanLimit = clampInt(toNumberSafe((req.query as Record).scan_limit), 50, 5000, 2500); + + const annotations = readAnnotations() + .filter((item) => (runIdFilter ? item.run_id === runIdFilter : true)) + .filter((item) => (caseIdFilter ? item.case_id === caseIdFilter : true)) + .filter((item) => (minRating === null ? true : item.rating >= minRating)) + .filter((item) => (decisionFilter === "all" ? true : item.manual_case_decision === decisionFilter)) + .sort((a, b) => Date.parse(b.updated_at) - Date.parse(a.updated_at)) + .slice(0, limit); + + const runIndex = indexRuns(scanLimit); + const runMap = new Map(runIndex.map((item) => [item.run_id, item])); + const items = annotations.map((item) => { + const run = runMap.get(item.run_id) ?? null; + const runSummary = run ? buildRunSummary(run) : null; + const cases = run ? buildCaseSummaries(run.report, run.run_id, false) : []; + const caseSummary = cases.find((candidate) => candidate.case_id === item.case_id) ?? null; + return { + ...item, + run: runSummary, + case_summary: caseSummary, + technical_context: { + report_path: run?.report_path ?? null, + trace_id: item.context.trace_id, + reply_type: item.context.reply_type, + domain: item.context.domain, + query_class: item.context.query_class, + checks: caseSummary?.checks ?? null, + metric_subscores: caseSummary?.metric_subscores ?? null + } + }; + }); + + const avgRating = + items.length > 0 ? Number((items.reduce((acc, item) => acc + item.rating, 0) / items.length).toFixed(2)) : null; + const byDecision = items.reduce>((acc, item) => { + acc[item.manual_case_decision] = (acc[item.manual_case_decision] ?? 0) + 1; + return acc; + }, {}); + + ok(res, { + ok: true, + generated_at: new Date().toISOString(), + filters_applied: { + run_id: runIdFilter ?? null, + case_id: caseIdFilter ?? null, + min_rating: minRating, + manual_case_decision: decisionFilter, + limit + }, + stats: { + total: items.length, + avg_rating: avgRating, + by_decision: byDecision + }, + available_manual_case_decisions: MANUAL_CASE_DECISIONS, + manual_case_decision_schema: readManualDecisionSchema(), + items + }); + } catch (error) { + next(error); + } + }); + + router.post("/api/autoruns/annotations", (req, res, next) => { + try { + const body = toRecord(req.body); + if (!body) { + throw new ApiError("INVALID_ANNOTATION_PAYLOAD", "JSON body is required", 400); + } + const runId = toStringSafe(body.run_id); + const caseId = toStringSafe(body.case_id); + const messageIndexRaw = toNumberSafe(body.message_index); + const ratingRaw = toNumberSafe(body.rating); + const comment = parseComment(body.comment); + const manualCaseDecision = parseManualCaseDecision(body.manual_case_decision); + const annotationAuthor = parseAnnotationAuthor(body.annotation_author); + + if (!runId || !caseId) { + throw new ApiError("INVALID_ANNOTATION_PAYLOAD", "run_id and case_id are required", 400); + } + if (messageIndexRaw === null) { + throw new ApiError("INVALID_ANNOTATION_PAYLOAD", "message_index is required", 400); + } + const messageIndex = clampInt(messageIndexRaw, 0, 100_000, 0); + if (ratingRaw === null) { + throw new ApiError("INVALID_ANNOTATION_PAYLOAD", "rating is required", 400); + } + const rating = clampInt(ratingRaw, 1, 5, 1); + if (comment.length === 0) { + throw new ApiError("INVALID_ANNOTATION_PAYLOAD", "comment is required", 400); + } + + const run = findRunById(runId); + if (!run) { + throw new ApiError("AUTORUN_NOT_FOUND", `Run not found: ${runId}`, 404); + } + const cases = buildCaseSummaries(run.report, run.run_id, false); + const caseSummary = cases.find((item) => item.case_id === caseId) ?? null; + if (!caseSummary) { + throw new ApiError("AUTORUN_CASE_NOT_FOUND", `Case not found: ${caseId} in run ${runId}`, 404); + } + + const sessionDialog = loadSessionDialog(runId, caseId); + const dialog = sessionDialog ?? buildFallbackDialog(run, caseId); + if (messageIndex >= dialog.messages.length) { + throw new ApiError("AUTORUN_MESSAGE_NOT_FOUND", `Message index ${messageIndex} out of range`, 400); + } + const targetMessage = dialog.messages[messageIndex]; + const targetRole = toStringSafe(targetMessage.role) ?? "unknown"; + if (targetRole !== "assistant") { + throw new ApiError("AUTORUN_MESSAGE_NOT_ASSISTANT", "Only assistant answers can be annotated", 400); + } + + const nowIso = new Date().toISOString(); + const annotations = readAnnotations(); + const key = annotationKey(runId, caseId, messageIndex); + const existingIndex = annotations.findIndex((item) => annotationKey(item.run_id, item.case_id, item.message_index) === key); + const existing = existingIndex >= 0 ? annotations[existingIndex] : null; + + const annotation: AutoRunAnnotationRecord = { + annotation_id: existing?.annotation_id ?? generateAnnotationId(), + run_id: runId, + case_id: caseId, + session_id: caseSummary.session_id, + message_index: messageIndex, + rating, + comment, + manual_case_decision: manualCaseDecision, + annotation_author: annotationAuthor, + created_at: existing?.created_at ?? nowIso, + updated_at: nowIso, + context: { + message_id: toStringSafe(targetMessage.message_id), + trace_id: toStringSafe(targetMessage.trace_id) ?? caseSummary.trace_id, + reply_type: toStringSafe(targetMessage.reply_type) ?? caseSummary.reply_type, + eval_target: run.eval_target, + prompt_version: toStringSafe(run.report.prompt_version), + domain: caseSummary.domain, + query_class: caseSummary.query_class + } + }; + + if (existingIndex >= 0) { + annotations[existingIndex] = annotation; + } else { + annotations.push(annotation); + } + writeAnnotations(annotations); + + const annotationStatsByCase = buildAnnotationStatsMap(runId, annotations); + const caseStats = annotationStatsByCase.get(caseId) ?? null; + + ok(res, { + ok: true, + annotation, + case_annotation_stats: caseStats + }); + } catch (error) { + next(error); + } + }); + + router.get("/api/autoruns/manual-decision-schema", (_req, res) => { + ok(res, { + ok: true, + schema: readManualDecisionSchema(), + enum: MANUAL_CASE_DECISIONS + }); + }); + + router.get("/api/autoruns/post-analysis", (req, res, next) => { + try { + const query = req.query as Record; + const runIdFilter = toStringSafe(query.run_id); + const limitPerQueue = clampInt(toNumberSafe(query.limit_per_queue), 5, 250, 40); + const annotationLimit = clampInt(toNumberSafe(query.annotation_limit), 20, 5000, 1500); + const scanLimit = clampInt(toNumberSafe(query.scan_limit), 50, 5000, 2500); + + const runFilters = parseFilters(query); + const applyRunFilters = hasAnyRunFilterQuery(query); + const runIndex = indexRuns(Math.max(scanLimit, runFilters.scan_limit)); + const filteredRuns = applyRunFilters ? runIndex.filter((run) => matchesFilters(run, runFilters)) : runIndex; + const runMap = new Map(filteredRuns.map((run) => [run.run_id, run])); + + const scopedAnnotations = readAnnotations() + .filter((item) => (runIdFilter ? item.run_id === runIdFilter : true)) + .filter((item) => (runMap.size > 0 ? runMap.has(item.run_id) : true)) + .sort((a, b) => Date.parse(b.updated_at) - Date.parse(a.updated_at)) + .slice(0, annotationLimit); + + const analysis = collectPostAnalysis(scopedAnnotations, runMap, limitPerQueue); + + ok(res, { + ok: true, + generated_at: new Date().toISOString(), + filters_applied: { + run_id: runIdFilter ?? null, + run_filters_applied: applyRunFilters, + limit_per_queue: limitPerQueue, + annotation_limit: annotationLimit, + scan_limit: scanLimit + }, + runs_considered: filteredRuns.slice(0, 500).map((item) => buildRunSummary(item)), + manual_case_decision_schema: readManualDecisionSchema(), + post_analysis: analysis + }); + } catch (error) { + next(error); + } + }); + + router.get("/api/autoruns/autogen/history", (req, res, next) => { + try { + const limit = clampInt(toNumberSafe((req.query as Record).limit), 1, 500, 120); + const rawMode = toStringSafe((req.query as Record).mode); + const includeAllModes = !rawMode || !["qwen_seed", "codex_creative"].includes(rawMode); + const modeFilter = (rawMode as AutoGenMode | null) ?? "codex_creative"; + const items = readAutoGenHistory() + .filter((item) => (includeAllModes ? true : item.mode === modeFilter)) + .slice(0, limit); + + ok(res, { + ok: true, + generated_at: new Date().toISOString(), + items + }); + } catch (error) { + next(error); + } + }); + + router.post("/api/autoruns/autogen/generate", (req, res, next) => { + try { + const body = toRecord(req.body); + if (!body) { + throw new ApiError("INVALID_AUTOGEN_PAYLOAD", "JSON body is required", 400); + } + const mode = parseAutoGenMode(body.mode); + const count = parseAutogenCount(body.count); + const domain = parseAutogenDomain(body.domain); + const persistCaseSet = toBooleanSafe(body.persist_to_eval_cases) ?? true; + const generatedBy = parseAnnotationAuthor(body.generated_by); + const context = toRecord(body.context); + + const questions = + mode === "qwen_seed" + ? generateQwenSeedQuestions(count, domain) + : generateCodexCreativeQuestions(count, domain); + const generationId = generateAutogenId(); + + let savedCaseSetFile: string | null = null; + if (persistCaseSet) { + if (!fs.existsSync(EVAL_CASES_DIR)) { + fs.mkdirSync(EVAL_CASES_DIR, { recursive: true }); + } + const fileName = buildAutogenCaseSetFileName(mode, generationId); + const filePath = path.resolve(EVAL_CASES_DIR, fileName); + const payload = buildAutogenCaseSetPayload({ + generationId, + mode, + domain, + questions + }); + fs.writeFileSync(filePath, JSON.stringify(payload, null, 2), "utf-8"); + savedCaseSetFile = fileName; + } + + const record: AutoGenHistoryRecord = { + generation_id: generationId, + created_at: new Date().toISOString(), + mode, + count: questions.length, + domain, + questions, + generated_by: generatedBy, + saved_case_set_file: savedCaseSetFile, + context: context + ? { + llm_provider: toStringSafe(context.llm_provider), + model: toStringSafe(context.model), + assistant_prompt_version: toStringSafe(context.assistant_prompt_version), + decomposition_prompt_version: toStringSafe(context.decomposition_prompt_version), + prompt_fingerprint: toStringSafe(context.prompt_fingerprint) + } + : null + }; + const history = readAutoGenHistory(); + history.unshift(record); + writeAutoGenHistory(history.slice(0, 500)); + + ok(res, { + ok: true, + generation: record }); } catch (error) { next(error); diff --git a/llm_normalizer/backend/src/server.ts b/llm_normalizer/backend/src/server.ts index 70c2e63..50bab81 100644 --- a/llm_normalizer/backend/src/server.ts +++ b/llm_normalizer/backend/src/server.ts @@ -1,7 +1,17 @@ import "dotenv/config"; import cors from "cors"; import express from "express"; -import { PORT, PRESETS_DIR, TRACES_DIR, EVAL_CASES_DIR, REPORTS_DIR, TIMEZONE, ASSISTANT_SESSIONS_DIR } from "./config"; +import { + PORT, + PRESETS_DIR, + TRACES_DIR, + EVAL_CASES_DIR, + REPORTS_DIR, + TIMEZONE, + ASSISTANT_SESSIONS_DIR, + AUTORUN_ANNOTATIONS_DIR, + AUTORUN_GENERATOR_DIR +} from "./config"; import { buildAccountingAgentRouter } from "./routes/accountingAgent"; import { buildAssistantRouter } from "./routes/assistant"; import { buildAutoRunsRouter } from "./routes/autoRuns"; @@ -27,6 +37,8 @@ export function createApp(): express.Express { ensureDir(EVAL_CASES_DIR); ensureDir(REPORTS_DIR); ensureDir(ASSISTANT_SESSIONS_DIR); + ensureDir(AUTORUN_ANNOTATIONS_DIR); + ensureDir(AUTORUN_GENERATOR_DIR); const app = express(); app.use(cors()); diff --git a/llm_normalizer/backend/src/services/answerComposer.ts b/llm_normalizer/backend/src/services/answerComposer.ts index 17387e6..49d095f 100644 --- a/llm_normalizer/backend/src/services/answerComposer.ts +++ b/llm_normalizer/backend/src/services/answerComposer.ts @@ -1368,29 +1368,29 @@ function buildProblemCentricActions(input: { } if (unitTypes.has("broken_chain_segment")) { - actions.push("Проверьте СЃРІСЏР·РєСѓ выписка -> документ -> РїСЂРѕРІРѕРґРєР° РїРѕ проблемным участкам цепочки."); + actions.push("Проверьте связку выписка -> документ -> проводка по проблемным участкам цепочки."); } if (unitTypes.has("unresolved_settlement_cluster")) { - actions.push("Сверьте хвосты РїРѕ расчетам: закрылся ли документ оплаты корректным закрывающим документом."); + actions.push("Сверьте хвосты по расчетам: закрылся ли документ оплаты корректным закрывающим документом."); } if (unitTypes.has("period_risk_cluster")) { - actions.push("Оцените влияние дефекта РЅР° закрытие периода Рё корректность регламентных операций."); + actions.push("Оцените влияние дефекта на закрытие периода и корректность регламентных операций."); } if (unitTypes.has("cross_branch_inconsistency_cluster")) { - actions.push("Сверьте противоречия между документами, проводками Рё регистрами РїРѕ НДС/межконтурным СЃРІСЏР·СЏРј."); + actions.push("Сверьте противоречия между документами, проводками и регистрами по НДС/межконтурным связям."); } if (unitTypes.has("lifecycle_anomaly_node")) { - actions.push("Проверьте lifecycle объекта: ожидаемый этап РЅРµ должен оставаться РІ partially_linked состоянии."); + actions.push("Проверьте lifecycle объекта: ожидаемый этап не должен оставаться в partially_linked состоянии."); } for (const unit of input.units) { if (unit.lifecycle_defect_type === "stale_active_state") { - actions.push("Проверьте, почему объект завис: ожидаемый переход РЅРµ должен оставаться РІ активной стадии."); + actions.push("Проверьте, почему объект завис: ожидаемый переход не должен оставаться в активной стадии."); } if (unit.lifecycle_defect_type === "misclosed_state") { - actions.push("Проверьте закрывающий документ Рё РїСЂРѕРІРѕРґРєРё: закрытие может быть формальным, РЅРѕ некорректным РїРѕ пути."); + actions.push("Проверьте закрывающий документ и проводки: закрытие может быть формальным, но некорректным по пути."); } if (unit.lifecycle_defect_type === "cross_branch_state_conflict") { - actions.push("Сверьте бухгалтерскую Рё смежную ветки (например, НДС/расчеты): обнаружен межконтурный конфликт состояния."); + actions.push("Сверьте бухгалтерскую и смежную ветки (например, НДС/расчеты): обнаружен межконтурный конфликт состояния."); } } @@ -1400,21 +1400,21 @@ function buildProblemCentricActions(input: { if (input.mode === "clarification_required") { if (input.missingAnchors.period) { - actions.push("Уточните период проверки, чтобы зафиксировать границы проблемного контура."); + actions.push("Уточните период проверки, чтобы зафиксировать границы проблемного контура."); } if (input.missingAnchors.account) { - actions.push("Уточните счет или РіСЂСѓРїРїСѓ счетов для предметной локализации дефекта."); + actions.push("Уточните счет или группу счетов для предметной локализации дефекта."); } if (input.missingAnchors.documentOrObject) { - actions.push("Укажите конкретный документ или объект трассировки для проверки механизма отклонения."); + actions.push("Укажите конкретный документ или объект трассировки для проверки механизма отклонения."); } if (input.missingAnchors.counterparty) { - actions.push("Укажите контрагента/РґРѕРіРѕРІРѕСЂ, чтобы проверить хвосты Рё разрывы РЅР° конкретной СЃРІСЏР·РєРµ."); + actions.push("Укажите контрагента/договор, чтобы проверить хвосты и разрывы на конкретной связке."); } } if (input.coverageReport.requirements_uncovered.length > 0) { - actions.push(`Закройте непокрытые требования: ${input.coverageReport.requirements_uncovered.join(", ")}.`); + actions.push(`Закройте непокрытые требования: ${input.coverageReport.requirements_uncovered.join(", ")}.`); } return uniqueStrings(actions, 6); @@ -1437,25 +1437,25 @@ function buildProblemCentricClarifications(input: { questions.push("Уточните период (например, июль 2020), в котором нужно проверить проблемный кластер."); } if (input.missingAnchors.account) { - questions.push("Уточните счет или СЃРІСЏР·РєСѓ счетов (например, 51/60), РіРґРµ РІС‹ ожидаете дефект."); + questions.push("Уточните счет или связку счетов (например, 51/60), где вы ожидаете дефект."); } if (input.missingAnchors.documentOrObject) { - questions.push("Укажите документ/объект, РѕС‚ которого РЅСѓР¶РЅРѕ строить проверку цепочки."); + questions.push("Укажите документ/объект, от которого нужно строить проверку цепочки."); } if (input.missingAnchors.counterparty) { - questions.push("Укажите контрагента или РґРѕРіРѕРІРѕСЂ, РїРѕ которому проверить незакрытую экспозицию."); + questions.push("Укажите контрагента или договор, по которому проверить незакрытую экспозицию."); } if (unitTypes.has("broken_chain_segment")) { - questions.push("Уточните участок цепочки: выписка, платежный документ или РїСЂРѕРІРѕРґРєР°."); + questions.push("Уточните участок цепочки: выписка, платежный документ или проводка."); } if (unitTypes.has("period_risk_cluster")) { - questions.push("Уточните, какой этап закрытия периода критичен: начисление, закрытие счетов или НДС-блок."); + questions.push("Уточните, какой этап закрытия периода критичен: начисление, закрытие счетов или НДС-блок."); } if (unitTypes.has("unresolved_settlement_cluster")) { - questions.push("Уточните, интересуют хвосты поставщиков, покупателей или РѕР±Р° направления."); + questions.push("Уточните, интересуют хвосты поставщиков, покупателей или оба направления."); } if (input.coverageReport.clarification_needed_for.length > 0) { - questions.push(`Закройте уточнения для требований: ${input.coverageReport.clarification_needed_for.join(", ")}.`); + questions.push(`Закройте уточнения для требований: ${input.coverageReport.clarification_needed_for.join(", ")}.`); } return uniqueStrings(questions, 6); @@ -1641,13 +1641,13 @@ function detectMissingAnchors( Boolean(options?.normalizationPeriodExplicit) || hasPeriodAnchorInCompanyAnchors(options?.companyAnchors); const hasAccount = - /(?:\bсчет\b|\baccount\b|\bschet\b|\b(?:0[1-9]|[1-9]\d)(?:\.\d{2})?\b|\b(?:60|62)\.\d{2}\s*\/\s*(?:60|62)\.\d{2}\b)/i.test( + /(?:\bсчет\b|\baccount\b|\bschet\b|\b(?:0[1-9]|[1-9]\d)(?:\.\d{2})?\b|\b(?:60|62)\.\d{2}\s*\/\s*(?:60|62)\.\d{2}\b)/i.test( lower ) || hasAccountAnchorInRetrieval(retrievalResults); const hasDocumentOrObject = - /(?:документ|invoice|guid|object|obj|#\d+|\b№\s*[a-zа-я0-9-]+\b|\bid\b|\bref\b|dokument|doc)/i.test(lower); - const hasCounterparty = /(?:контрагент|supplier|buyer|customer|kontragent|postavsh|pokupatel|договор|contract)/i.test(lower); - const hasAnomalyType = /(?:аномал|risk|отклон|разрыв|mismatch|duplicate|tail|цепочк|anomali|hvost)/i.test(lower); + /(?:документ|invoice|guid|object|obj|#\d+|\b№\s*[a-zа-я0-9-]+\b|\bid\b|\bref\b|dokument|doc)/i.test(lower); + const hasCounterparty = /(?:контрагент|supplier|buyer|customer|kontragent|postavsh|pokupatel|договор|contract)/i.test(lower); + const hasAnomalyType = /(?:аномал|risk|отклон|разрыв|mismatch|duplicate|tail|цепочк|anomali|hvost)/i.test(lower); return { period: !hasPeriod, @@ -1674,19 +1674,19 @@ function buildClarificationQuestions(input: { questions.push("Уточните период проверки (например, июль 2020)."); } if (input.missingAnchors.account) { - questions.push("Уточните счет или РіСЂСѓРїРїСѓ счетов (например, 19, 60, 62)."); + questions.push("Уточните счет или группу счетов (например, 19, 60, 62)."); } if (input.missingAnchors.documentOrObject) { - questions.push("Укажите документ/GUID/конкретный объект для трассировки."); + questions.push("Укажите документ/GUID/конкретный объект для трассировки."); } if (input.missingAnchors.counterparty) { - questions.push("Укажите контрагента или РіСЂСѓРїРїСѓ контрагентов."); + questions.push("Укажите контрагента или группу контрагентов."); } if (input.policySignals.broad_query_detected && input.missingAnchors.anomalyType) { - questions.push("Уточните тип отклонения: разрыв цепочки, неверный документ или аномальный СЂРёСЃРє."); + questions.push("Уточните тип отклонения: разрыв цепочки, неверный документ или аномальный риск."); } if (input.coverageReport.clarification_needed_for.length > 0) { - questions.push(`Закройте уточнения для требований: ${input.coverageReport.clarification_needed_for.join(", ")}.`); + questions.push(`Закройте уточнения для требований: ${input.coverageReport.clarification_needed_for.join(", ")}.`); } return uniqueStrings(questions, 6); @@ -1701,31 +1701,31 @@ function buildRecommendedActions(input: { }): string[] { const actions: string[] = []; if (input.mode === "focused_grounded") { - actions.push("Проверьте 1-2 ключевые записи РІ учетной базе Рё зафиксируйте итог РІ рабочем файле проверки."); + actions.push("Проверьте 1-2 ключевые записи в учетной базе и зафиксируйте итог в рабочем файле проверки."); } if (input.mode === "broad_partial") { - actions.push("Сузьте запрос РґРѕ периода + счета или периода + документа Рё повторите проверку."); + actions.push("Сузьте запрос до периода + счета или периода + документа и повторите проверку."); } if (input.mode === "clarification_required") { - actions.push("Дайте недостающие СЏРєРѕСЂСЏ (период/счет/объект), иначе сильный factual вывод невозможен."); + actions.push("Дайте недостающие якоря (период/счет/объект), иначе сильный factual вывод невозможен."); } if (input.coverageReport.requirements_uncovered.length > 0) { - actions.push(`Закройте непокрытые требования: ${input.coverageReport.requirements_uncovered.join(", ")}.`); + actions.push(`Закройте непокрытые требования: ${input.coverageReport.requirements_uncovered.join(", ")}.`); } if (input.coverageReport.requirements_partially_covered.length > 0) { - actions.push(`Доуточните частично покрытые требования: ${input.coverageReport.requirements_partially_covered.join(", ")}.`); + actions.push(`Доуточните частично покрытые требования: ${input.coverageReport.requirements_partially_covered.join(", ")}.`); } if (input.policySignals.broad_query_detected && input.policySignals.narrowing_strength !== "strong") { - actions.push("Добавьте более СѓР·РєРёР№ контекст: тип отклонения, РіСЂСѓРїРїСѓ документов Рё бизнес-участок."); + actions.push("Добавьте более узкий контекст: тип отклонения, группу документов и бизнес-участок."); } if (input.limitationReasonCodes.includes("snapshot_only")) { - actions.push("Сверьте критичные выводы СЃ live source-of-record РІ 1C."); + actions.push("Сверьте критичные выводы с live source-of-record в 1C."); } if (input.limitationReasonCodes.includes("weak_source_mapping")) { - actions.push("Проверьте source mapping для связей document/register РїРѕ указанным ref."); + actions.push("Проверьте source mapping для связей document/register по указанным ref."); } if (input.sourceRefs.length > 0) { - actions.push(`Начните проверку СЃ ${input.sourceRefs.length} подтвержденных записей Рё сверьте РёС… СЃ первичными документами.`); + actions.push(`Начните проверку с ${input.sourceRefs.length} подтвержденных записей и сверьте их с первичными документами.`); } return uniqueStrings(actions, 6); diff --git a/llm_normalizer/backend/src/services/assistantCanon.ts b/llm_normalizer/backend/src/services/assistantCanon.ts new file mode 100644 index 0000000..46b1a00 --- /dev/null +++ b/llm_normalizer/backend/src/services/assistantCanon.ts @@ -0,0 +1,38 @@ +import fs from "fs"; +import { ASSISTANT_CANON_FILE } from "../config"; + +const FALLBACK_CANON = [ + "Не выдумывай возможности.", + "Не обещай настройку 1С и админ-действия.", + "Не показывай внутренние технические термины пользователю.", + "Говори по-человечески и предлагай ближайший полезный поддерживаемый шаг." +].join(" "); + +let cache: { mtimeMs: number; excerpt: string } | null = null; + +function stripMarkdown(input: string): string { + return input + .replace(/^#{1,6}\s+/gm, "") + .replace(/[`*_>\-\[\]\(\)]/g, " ") + .replace(/\s+/g, " ") + .trim(); +} + +export function loadAssistantCanonExcerpt(maxChars = 900): string { + try { + const mtimeMs = fs.existsSync(ASSISTANT_CANON_FILE) ? fs.statSync(ASSISTANT_CANON_FILE).mtimeMs : -1; + if (cache && cache.mtimeMs === mtimeMs) { + return cache.excerpt; + } + if (!fs.existsSync(ASSISTANT_CANON_FILE)) { + return FALLBACK_CANON; + } + const raw = fs.readFileSync(ASSISTANT_CANON_FILE, "utf-8"); + const normalized = stripMarkdown(raw); + const excerpt = normalized.length > maxChars ? `${normalized.slice(0, maxChars).trim()}...` : normalized; + cache = { mtimeMs, excerpt: excerpt || FALLBACK_CANON }; + return cache.excerpt; + } catch { + return cache?.excerpt ?? FALLBACK_CANON; + } +} diff --git a/llm_normalizer/backend/src/services/assistantDataLayer.ts b/llm_normalizer/backend/src/services/assistantDataLayer.ts index 3ccbcf5..d729e39 100644 --- a/llm_normalizer/backend/src/services/assistantDataLayer.ts +++ b/llm_normalizer/backend/src/services/assistantDataLayer.ts @@ -731,8 +731,8 @@ const P0_DOMAIN_CARDS: P0DomainCard[] = [ /закрыт[а-яё]*\s+период/i, /close\s+operation/i, /allocation/i, - /закр/i, - /перио/i, + /закр/i, + /перио/i, /\u0437\u0430\u043a\u0440\u044b\u0442(?:\u0438|\u0438\u0435|\u044b|)\s*(?:\u043c\u0435\u0441\u044f\u0446|\u0441\u0447\u0435\u0442)/i, /\u0440\u0435\u0433\u043b\u0430\u043c\u0435\u043d\u0442/i, /\u0437\u0430\u0442\u0440\u0430\u0442/i, @@ -756,14 +756,14 @@ function parseDateCandidate(value: unknown): number | null { function extractDate(record: SnapshotRecord): string | null { const attrs = record.attributes ?? {}; - const directKeys = ["Period", "Date", "Дата", "Период", "ДатаСобытия"]; + const directKeys = ["Period", "Date", "Дата", "Период", "ДатаСобытия"]; for (const key of directKeys) { if (attrs[key] !== undefined && attrs[key] !== null) { return String(attrs[key]); } } for (const [key, value] of Object.entries(attrs)) { - if (/period|date|дата|период/i.test(key) && typeof value === "string" && value.trim()) { + if (/period|date|дата|период/i.test(key) && typeof value === "string" && value.trim()) { return value; } } @@ -795,7 +795,7 @@ function countNavigationLinks(record: SnapshotRecord): number { function findCounterpartyLinks(record: SnapshotRecord): SnapshotLink[] { return record.links.filter( (link) => - link.target_entity === "Counterparty" || /supplier|buyer|counterparty/i.test(link.relation) || /постав|РїРѕРєСѓРї/i.test(link.source_field) + link.target_entity === "Counterparty" || /supplier|buyer|counterparty/i.test(link.relation) || /постав|покуп/i.test(link.source_field) ); } @@ -1723,14 +1723,14 @@ function inferPeriodScope(fragmentText: string): SemanticPeriodScope { granularity: "year" }; } - if (/квартал|quarter/i.test(fragmentText)) { + if (/квартал|quarter/i.test(fragmentText)) { return { from: null, to: null, granularity: "quarter" }; } - if (/месяц|month|период/i.test(fragmentText)) { + if (/месяц|month|период/i.test(fragmentText)) { return { from: null, to: null, @@ -1808,7 +1808,7 @@ function buildSemanticRetrievalProfile(fragmentText: string): SemanticRetrievalP const rankingBasis: string[] = ["closure_risk", "repeatability", "financial_impact"]; const explanationFocus: string[] = ["why_selected", "where_chain_breaks", "what_business_risk"]; - if (/банк|выписк|расчетн|платеж|банк|выписк|расчетн|платеж|bank|payment|statement|platezh|vypisk/i.test(lower)) { + if (/банк|выписк|расчетн|платеж|банк|выписк|расчетн|платеж|bank|payment|statement|platezh|vypisk/i.test(lower)) { pushMany(domainScope, ["bank", "settlements"]); pushMany(documentTypes, ["bank_statement", "payment_order", "settlement_document"]); pushMany(entityTypes, ["counterparty", "contract", "document", "posting"]); @@ -1823,22 +1823,22 @@ function buildSemanticRetrievalProfile(fragmentText: string): SemanticRetrievalP /(?:закрыти[ея]\s+месяц|закрыт[а-яё]*\s+период|закрытие\s+счетов|регламентн|косвенн|затрат|распределени|рбп|амортиз|финансовых\s+результат|month\s*close|period\s*close|close\s+period|close\s+operation)/i.test( lower ) || - (/закр/i.test(lower) && /перио/i.test(lower)); + (/закр/i.test(lower) && /перио/i.test(lower)); - if (/постав|постав|supplier|vendor/i.test(lower) || hasSettlementAccountScope) { + if (/постав|постав|supplier|vendor/i.test(lower) || hasSettlementAccountScope) { pushMany(domainScope, ["suppliers", "settlements"]); pushMany(documentTypes, ["supplier_receipt", "settlement_document"]); pushMany(entityTypes, ["counterparty", "contract", "document", "posting"]); pushMany(relationPatterns, ["payment_to_settlement", "contract_to_documents"]); } - if (/покупат|покупат|customer|buyer/i.test(lower) || hasSettlementAccountScope) { + if (/покупат|покупат|customer|buyer/i.test(lower) || hasSettlementAccountScope) { pushMany(domainScope, ["customers", "settlements"]); pushMany(documentTypes, ["sales_document", "settlement_document"]); pushMany(entityTypes, ["counterparty", "contract", "document", "posting"]); pushMany(relationPatterns, ["payment_to_settlement", "contract_to_documents"]); } if ( - /РЅРґСЃ|ндс|vat|РєРЅРёРіР° РїРѕРєСѓРїРѕРє|РєРЅРёРіР° продаж|счет.?фактур|книг[аи]\s+покуп|книг[аи]\s+продаж|сч[её]т(?:а|у|ом|е)?[-\s]?фактур(?:а|ы|е|у|ой)?|вычет|налогов(?:ый|ого)?\s+эффект/i.test( + /ндс|vat|книга\s+покупок|книга\s+продаж|счет.?фактур|книг[аи]\s+покуп|книг[аи]\s+продаж|сч[её]т(?:а|у|ом|е)?[-\s]?фактур(?:а|ы|е|у|ой)?|вычет|налогов(?:ый|ого)?\s+эффект/i.test( lower ) || hasVatAccountScope @@ -1849,7 +1849,7 @@ function buildSemanticRetrievalProfile(fragmentText: string): SemanticRetrievalP pushMany(relationPatterns, ["invoice_to_vat", "document_to_posting"]); } if ( - /РѕСЃ|РѕСЃРЅРѕРІРЅ(ые|ых)\s+сред|(?:^|[^a-zа-яё])ос(?:$|[^a-zа-яё])|основн(ые|ых|ым)?\s+средств|fixed asset|amort|амортиз|амортиз/i.test( + /ос|основн(ые|ых)\s+сред|(?:^|[^a-zа-яё])ос(?:$|[^a-zа-яё])|основн(ые|ых|ым)?\s+средств|fixed asset|amort|амортиз|амортиз/i.test( lower ) || hasFixedAssetAccountScope @@ -1860,7 +1860,7 @@ function buildSemanticRetrievalProfile(fragmentText: string): SemanticRetrievalP pushMany(relationPatterns, ["asset_card_to_depreciation", "document_to_posting"]); } if ( - /СЂР±Рї|расходы будущих периодов|рбп|расходы\s+будущих\s+периодов|deferred|writeoff/i.test(lower) || + /рбп|расходы будущих периодов|рбп|расходы\s+будущих\s+периодов|deferred|writeoff/i.test(lower) || hasDeferredExpenseAccountScope ) { pushMany(domainScope, ["deferred_expense", "period_close"]); @@ -1868,17 +1868,17 @@ function buildSemanticRetrievalProfile(fragmentText: string): SemanticRetrievalP pushMany(entityTypes, ["document", "posting"]); pushMany(relationPatterns, ["deferred_expense_to_writeoff", "document_to_posting"]); } - if (/цепоч|разрыв|СЃРІСЏР·|документ.*РїСЂРѕРІРѕРґ|РіРґРµ рвет|Р¶РёРІСѓС‚ отдельно|цепоч|разрыв|связ|документ.*провод|chain|break/i.test(lower)) { + if (/цепоч|разрыв|связ|документ.*провод|где рвет|живут отдельно|цепоч|разрыв|связ|документ.*провод|chain|break/i.test(lower)) { pushMany(relationPatterns, ["document_to_posting", "contract_to_documents"]); pushMany(explanationFocus, ["what_conflicts_with_what", "why_not_closed"]); } - if (/аномал|СЂРёСЃРє|С…РІРѕСЃС‚|РїРѕРґРѕР·СЂ|искаж|аномал|риск|хвост|подозр|искаж|suspic|risk/i.test(lower)) { + if (/аномал|риск|хвост|подозр|искаж|аномал|риск|хвост|подозр|искаж|suspic|risk/i.test(lower)) { pushMany(anomalyPatterns, ["missing_link", "broken_lifecycle", "amount_independent_risk"]); } if (WRONG_DOCUMENT_MARKERS.test(lower)) { pushMany(anomalyPatterns, ["wrong_document_type", "posting_mismatch", "broken_lifecycle"]); } - if (/Р¶РёРІСѓС‚ отдельно|РЅРµ СЃРІСЏР·|без СЃРІСЏР·Рё|живут\s+отдельно|не\s+связ|без\s+связи|missing link/i.test(lower)) { + if (/живут отдельно|не связ|без связи|живут\s+отдельно|не\s+связ|без\s+связи|missing link/i.test(lower)) { pushMany(anomalyPatterns, ["missing_link", "cross_domain_inconsistency"]); } if (REPEATED_ANOMALY_MARKERS.test(lower)) { @@ -1890,10 +1890,10 @@ function buildSemanticRetrievalProfile(fragmentText: string): SemanticRetrievalP pushMany(anomalyPatterns, ["closure_risk", "broken_lifecycle"]); pushMany(documentTypes, ["period_close_document"]); } - if (/РЅРµ РІ платеже|не\s+в\s+платеже|not payment/i.test(lower)) { + if (/не\s+в\s+платеже|not payment/i.test(lower)) { pushMany(excludedInterpretations, ["simple_payment_delay"]); } - if (/РЅРµ РїРѕ СЃСѓРјРј|РЅРµ СЃСѓРјРјР°|не\s+по\s+сумм|не\s+сумм|not by amount/i.test(lower)) { + if (/не\s+по\s+сумм|не\s+сумм|не\s+сумма|not by amount/i.test(lower)) { pushMany(excludedInterpretations, ["amount_only_anomaly"]); pushMany(rankingBasis, ["amount_independent_risk"]); } @@ -2288,16 +2288,16 @@ function inferAccountsFromRecord(record: SnapshotRecord, corpus: string): string accounts.push(token.split(".")[0]); } for (const key of Object.keys(record.attributes ?? {})) { - if (/счетбанк|расчетн.*счет|bank account|банковскийсчет/i.test(key)) { + if (/счетбанк|расчетн.*счет|bank account|банковскийсчет/i.test(key)) { accounts.push("51"); } - if (/счетучетарасчетовсконтрагентом/i.test(key)) { + if (/счетучетарасчетовсконтрагентом/i.test(key)) { accounts.push("60"); } - if (/счетучетандс/i.test(key)) { + if (/счетучетандс/i.test(key)) { accounts.push("19"); } - if (/субконтодт/i.test(key)) { + if (/субконтодт/i.test(key)) { accounts.push("60"); } } @@ -2306,28 +2306,28 @@ function inferAccountsFromRecord(record: SnapshotRecord, corpus: string): string function inferDocumentTypesFromRecord(record: SnapshotRecord, corpus: string): string[] { const items: string[] = []; - if (/банковскиевыписки|выписк|расчетногосчета|spisaniesraschetnogoscheta|bank/i.test(corpus)) { + if (/банковскиевыписки|выписк|расчетногосчета|spisaniesraschetnogoscheta|bank/i.test(corpus)) { pushMany(items, ["bank_statement", "payment_order"]); } - if (/поступлениетоваровуслуг|поступлен/i.test(corpus)) { + if (/поступлениетоваровуслуг|поступлен/i.test(corpus)) { items.push("supplier_receipt"); } - if (/реализациятоваровуслуг|реализац/i.test(corpus)) { + if (/реализациятоваровуслуг|реализац/i.test(corpus)) { items.push("sales_document"); } - if (/РЅРґСЃ|счетфактур|РєРЅРёРіРёРїРѕРєСѓРїРѕРє|книгипродаж|vat|invoice/i.test(corpus)) { + if (/ндс|счетфактур|книгипокупок|книгипродаж|vat|invoice/i.test(corpus)) { pushMany(items, ["invoice", "vat_document"]); } - if (/корректировк|ручн|manual/i.test(corpus)) { + if (/корректировк|ручн|manual/i.test(corpus)) { items.push("manual_operation"); } - if (/закрытие|регламент/i.test(corpus)) { + if (/закрытие|регламент/i.test(corpus)) { items.push("period_close_document"); } - if (/РѕСЃРЅРѕРІРЅ|амортиз|fixed_asset/i.test(corpus)) { + if (/основн|амортиз|fixed_asset/i.test(corpus)) { pushMany(items, ["fixed_asset_card", "depreciation_document"]); } - if (/расходыбудущихпериодов|deferred|97/.test(corpus)) { + if (/расходыбудущихпериодов|deferred|97/.test(corpus)) { items.push("deferred_expense_document"); } if (record.source_entity.startsWith("Document") || record.source_entity.startsWith("DocumentJournal")) { @@ -2356,11 +2356,11 @@ function inferDomainsFromRecord(corpus: string, documentTypes: string[], record: if (documentTypes.some((item) => item === "deferred_expense_document")) { pushMany(domains, ["deferred_expense", "period_close"]); } - if (/закрытие|регламент|period close/i.test(corpus)) { + if (/закрытие|регламент|period close/i.test(corpus)) { domains.push("period_close"); } const hasSettlementLexicalAnchor = - /(?:settlement|payment|bank|statement|supplier|customer|buyer|vendor|60\b|62\b|51\b|оплат|банк|выписк|расчет|постав|РїРѕРєСѓРї)/i.test( + /(?:settlement|payment|bank|statement|supplier|customer|buyer|vendor|60\b|62\b|51\b|оплат|банк|выписк|расчет|постав|покуп)/i.test( corpus ); const hasSettlementDocAnchor = documentTypes.some( @@ -2386,13 +2386,13 @@ function inferEntityTypes(record: SnapshotRecord): string[] { entities.push("counterparty"); } const corpus = collectTextFromRecord(record); - if (/РґРѕРіРѕРІРѕСЂ|contract/i.test(corpus)) { + if (/договор|contract/i.test(corpus)) { entities.push("contract"); } - if (/РѕСЃРЅРѕРІРЅ|fixed_asset|инвентар/i.test(corpus)) { + if (/основн|fixed_asset|инвентар/i.test(corpus)) { entities.push("fixed_asset"); } - if (/РЅРґСЃ|РєРЅРёРіРёРїРѕРєСѓРїРѕРє|книгипродаж|vat|invoice/i.test(corpus)) { + if (/ндс|книгипокупок|книгипродаж|vat|invoice/i.test(corpus)) { entities.push("tax_entry"); } return uniqueStrings(entities); @@ -2406,25 +2406,25 @@ function inferRelationPatterns(record: SnapshotRecord, corpus: string): string[] if (hasDocLinks) { patterns.push("document_to_posting"); } - if (hasCounterparty && hasDocLinks && /платеж|bank|settlement|расчет/i.test(corpus)) { + if (hasCounterparty && hasDocLinks && /платеж|bank|settlement|расчет/i.test(corpus)) { patterns.push("payment_to_settlement"); } - if (/банковскиевыписки|выписк|statement/i.test(corpus) && hasDocLinks) { + if (/банковскиевыписки|выписк|statement/i.test(corpus) && hasDocLinks) { patterns.push("statement_to_document"); } - if (/РѕСЃРЅРѕРІРЅ|fixed_asset|амортиз/i.test(corpus)) { + if (/основн|fixed_asset|амортиз/i.test(corpus)) { patterns.push("asset_card_to_depreciation"); } - if (/расходыбудущихпериодов|97|deferred/i.test(corpus)) { + if (/расходыбудущихпериодов|97|deferred/i.test(corpus)) { patterns.push("deferred_expense_to_writeoff"); } - if (/РЅРґСЃ|счетфактур|РєРЅРёРіРёРїРѕРєСѓРїРѕРє|книгипродаж|vat|invoice/i.test(corpus)) { + if (/ндс|счетфактур|книгипокупок|книгипродаж|vat|invoice/i.test(corpus)) { patterns.push("invoice_to_vat"); } - if (/РґРѕРіРѕРІРѕСЂ|contract/i.test(corpus) && hasDocLinks) { + if (/договор|contract/i.test(corpus) && hasDocLinks) { patterns.push("contract_to_documents"); } - if (/склад|товар|материал|receipt/i.test(corpus)) { + if (/склад|товар|материал|receipt/i.test(corpus)) { patterns.push("receipt_to_stock_movement"); } return uniqueStrings(patterns); @@ -2466,7 +2466,7 @@ function inferAnomalyPatterns(record: SnapshotRecord, corpus: string, relationPa if (relationPatterns.includes("document_to_posting") && !record.attributes.Recorder) { anomalies.push("posting_mismatch"); } - if (/ручн|manual|корректировк/.test(corpus)) { + if (/ручн|manual|корректировк/.test(corpus)) { anomalies.push("manual_intervention_suspicion"); } if (lifecycleMarkers.includes("period_boundary") && (unknownLinks > 0 || zeroGuidValues > 0)) { @@ -2478,7 +2478,7 @@ function inferAnomalyPatterns(record: SnapshotRecord, corpus: string, relationPa if (!hasCounterparty && !hasDocLinks && zeroGuidValues > 0) { anomalies.push("silent_orphan"); } - const hasAmountSignal = Object.keys(record.attributes ?? {}).some((key) => /sum|СЃСѓРјРј|итого|amount/i.test(key)); + const hasAmountSignal = Object.keys(record.attributes ?? {}).some((key) => /sum|сумм|итого|amount/i.test(key)); if (!hasAmountSignal && anomalies.length > 0) { anomalies.push("amount_independent_risk"); } @@ -2530,15 +2530,15 @@ function evaluateExcludedInterpretations( const structural = ["missing_link", "broken_lifecycle", "posting_mismatch", "wrong_document_type", "closure_risk"]; const hasStructural = signals.anomaly_patterns.some((item) => structural.includes(item)); if (!hasStructural) { - reasons.push("Исключено как simple_payment_delay без структурного дефекта."); + reasons.push("Исключено как simple_payment_delay без структурного дефекта."); } } if (interpretationSet.has("amount_only_anomaly")) { - const hasAmountSignal = Object.keys(record.attributes ?? {}).some((key) => /sum|СЃСѓРјРј|итого|amount/i.test(key)); + const hasAmountSignal = Object.keys(record.attributes ?? {}).some((key) => /sum|сумм|итого|amount/i.test(key)); const structural = ["missing_link", "broken_lifecycle", "posting_mismatch", "cross_domain_inconsistency", "silent_orphan"]; const hasStructural = signals.anomaly_patterns.some((item) => structural.includes(item)); if (hasAmountSignal && !hasStructural) { - reasons.push("Исключено как amount-only аномалия без структурных признаков."); + reasons.push("Исключено как amount-only аномалия без структурных признаков."); } } @@ -2611,22 +2611,22 @@ function evaluateRecordAgainstProfile(record: SnapshotRecord, profile: SemanticR const matchReasons: string[] = []; if (accountMatch && profile.account_scope.length > 0) { - matchReasons.push("Совпал account_scope."); + matchReasons.push("Совпал account_scope."); } if (domainMatch && profile.domain_scope.length > 0) { - matchReasons.push("Совпал domain_scope."); + matchReasons.push("Совпал domain_scope."); } if (documentMatch && profile.document_types.length > 0) { - matchReasons.push("Совпал document_types."); + matchReasons.push("Совпал document_types."); } if (relationMatch && profile.relation_patterns.length > 0) { - matchReasons.push("Совпали relation_patterns."); + matchReasons.push("Совпали relation_patterns."); } if (anomalyMatch && profile.anomaly_patterns.length > 0) { - matchReasons.push("Совпали anomaly_patterns."); + matchReasons.push("Совпали anomaly_patterns."); } if (lifecycleMatch && profile.lifecycle_stage_filters.length > 0) { - matchReasons.push("Совпал lifecycle_stage_filters."); + matchReasons.push("Совпал lifecycle_stage_filters."); } if (graphTraversal.domain_match) { matchReasons.push("Graph traversal domain matched."); @@ -2818,7 +2818,7 @@ export class AssistantDataLayer { business_interpretation: [], confidence: "low", limitations: ["Snapshot data files could not be loaded."], - errors: ["Слой данных недоступен: РЅРµ удалось загрузить snapshot-файлы."] + errors: ["Слой данных недоступен: не удалось загрузить snapshot-файлы."] }; } let result: RawRetrievalResult | null = null; @@ -3718,8 +3718,8 @@ export class AssistantDataLayer { : "low", business_interpretation: group.risk_factors.size > 0 - ? "Есть признаки разрыва расчетной цепочки: часть связей/этапов lifecycle подтверждена неполно." - : "Есть связанная операционная цепочка, РЅРѕ явные СЂРёСЃРє-паттерны выражены слабо.", + ? "Есть признаки разрыва расчетной цепочки: часть связей/этапов lifecycle подтверждена неполно." + : "Есть связанная операционная цепочка, но явные риск-паттерны выражены слабо.", relation_types: Array.from(group.relations.entries()) .sort((left, right) => right[1] - left[1]) .map((item) => item[0]), @@ -3768,24 +3768,24 @@ export class AssistantDataLayer { evidence: [], why_included: [], selection_reason: [ - "РџРѕРёСЃРє строился РїРѕ semantic retrieval profile, РЅРѕ подходящие контрагенты РЅРµ найдены.", - "Фильтрация использовала пересечение account/domain/document/relation/anomaly ограничений.", + "Поиск строился по semantic retrieval profile, но подходящие контрагенты не найдены.", + "Фильтрация использовала пересечение account/domain/document/relation/anomaly ограничений.", domainCard ? `P0 domain purity enforced for ${domainCard.id}.` : "P0 domain purity was not enforced.", guidFilter.length > 0 - ? "GUID-фильтрация включена." - : `GUID отсутствовал, выполнено semantic narrowing (${filtered.length}/${sourceRecords.length}).`, + ? "GUID-фильтрация включена." + : `GUID отсутствовал, выполнено semantic narrowing (${filtered.length}/${sourceRecords.length}).`, `Graph planner mode=${graphTraversalRuntime.planner_mode}, eligible=${graphTraversalRuntime.graph_eligible}, applied=${graphTraversalRuntime.traversal_applied}.`, `Graph ranking signals: ${graphTraversalRuntime.ranking_shift_signals.join(",") || "none"}.` ], risk_factors: semanticProfile.anomaly_patterns, business_interpretation: [ - "РџРѕ текущему профилю запроса устойчивых разрывов цепочки РЅРµ обнаружено.", - "Для точечного drilldown добавьте GUID или уточните период/контрагента." + "По текущему профилю запроса устойчивых разрывов цепочки не обнаружено.", + "Для точечного drilldown добавьте GUID или уточните период/контрагента." ], confidence: "medium", limitations: [ - guidFilter.length > 0 ? "РџРѕРёСЃРє ограничен переданными GUID." : "РџРѕРёСЃРє выполнен РїРѕ semantic narrowing без GUID.", - "Источник данных — snapshot 2020 (read-only), Р° РЅРµ live состояние базы 1РЎ.", + guidFilter.length > 0 ? "Поиск ограничен переданными GUID." : "Поиск выполнен по semantic narrowing без GUID.", + "Источник данных — snapshot 2020 (read-only), а не live состояние базы 1С.", domainCard ? "Domain purity guardrail может исключить cross-domain записи на этапе source selection." : "Domain purity guardrail не применялся." ], errors: [] @@ -3833,18 +3833,18 @@ export class AssistantDataLayer { }, evidence: evidence.slice(0, 12), why_included: [ - `Семантическое сужение выполнено РїРѕ профилю ${semanticProfile.query_subject}.`, + `Семантическое сужение выполнено по профилю ${semanticProfile.query_subject}.`, domainCard ? `P0 domain purity enforced for ${domainCard.id}.` : "P0 domain purity was not enforced.", semanticProfile.account_scope.length > 0 - ? `Учитывались счета: ${semanticProfile.account_scope.join(", ")}.` - : "Счета РЅРµ были заданы СЏРІРЅРѕ, использованы domain/document/relation ограничения.", - `После narrowing осталось ${filtered.length} РёР· ${sourceRecords.length} записей.`, + ? `Учитывались счета: ${semanticProfile.account_scope.join(", ")}.` + : "Счета не были заданы явно, использованы domain/document/relation ограничения.", + `После narrowing осталось ${filtered.length} из ${sourceRecords.length} записей.`, `Graph traversal mode=${graphTraversalRuntime.planner_mode}, matched=${graphTraversalRuntime.matched_candidates}/${graphTraversalRuntime.evaluated_candidates}.` ], selection_reason: [ - "Отбор основан РЅР° пересечении account_scope + domain_scope + document_types + relation_patterns + anomaly_patterns.", - "GUID-mode отключен: full scan без ограничителей РЅРµ использовался.", - `Ранжирование выполнено РїРѕ basis: ${semanticProfile.ranking_basis.join(", ")}.`, + "Отбор основан на пересечении account_scope + domain_scope + document_types + relation_patterns + anomaly_patterns.", + "GUID-mode отключен: full scan без ограничителей не использовался.", + `Ранжирование выполнено по basis: ${semanticProfile.ranking_basis.join(", ")}.`, domainCard ? `Domain gate source scope: ${sourceScope.join(", ")}.` : "Domain gate source scope not applied.", `Graph signal counts: ${JSON.stringify(graphTraversalRuntime.signal_counts)}.`, `Graph ranking signals: ${graphTraversalRuntime.ranking_shift_signals.join(",") || "none"}.` @@ -3852,15 +3852,15 @@ export class AssistantDataLayer { risk_factors: aggregatedRiskFactors.length > 0 ? aggregatedRiskFactors - : ["Высокая плотность операций РїРѕ контрагенту может указывать РЅР° незакрытые цепочки."], + : ["Высокая плотность операций по контрагенту может указывать на незакрытые цепочки."], business_interpretation: [ - "Результат отражает РЅРµ просто объем операций, Р° структурные признаки разрыва цепочки Рё lifecycle-конфликта.", - "Контрагенты РІ топе приоритетны для проверки РЅР° неверный тип закрывающего документа Рё незавершенные СЃРІСЏР·Рё." + "Результат отражает не просто объем операций, а структурные признаки разрыва цепочки и lifecycle-конфликта.", + "Контрагенты в топе приоритетны для проверки на неверный тип закрывающего документа и незавершенные связи." ], confidence: "high", limitations: [ - guidFilter.length > 0 ? "Выборка ограничена GUID РёР· запроса." : "Выборка ограничена semantic retrieval profile.", - "Источник данных — snapshot 2020 (read-only), РЅРµ live контур 1РЎ.", + guidFilter.length > 0 ? "Выборка ограничена GUID из запроса." : "Выборка ограничена semantic retrieval profile.", + "Источник данных — snapshot 2020 (read-only), не live контур 1С.", domainCard ? "Domain purity guardrail может исключить cross-domain элементы на этапе source selection." : "Domain purity guardrail не применялся." ], errors: [] @@ -4177,12 +4177,12 @@ export class AssistantDataLayer { })), why_included: items.length > 0 ? [ - "Показаны сущности СЃ максимальным количеством записей.", + "Показаны сущности с максимальным количеством записей.", domainCard ? `P0 domain purity enforced for ${domainCard.id}.` : "P0 domain purity was not enforced." ] : [], selection_reason: [ - "Ранжирование выполнено РїРѕ records_count РїРѕ убыванию.", + "Ранжирование выполнено по records_count по убыванию.", domainCard ? `Domain gate source scope: ${sourceScope.join(", ")}.` : "Domain gate source scope not applied." ], risk_factors: uniqueStrings(["entity_volume_spike", ...semanticProfile.anomaly_patterns]), @@ -4191,7 +4191,7 @@ export class AssistantDataLayer { ], confidence: "medium", limitations: [ - "Ранжирование РїРѕ объему РЅРµ всегда эквивалентно бизнес-СЂРёСЃРєСѓ.", + "Ранжирование по объему не всегда эквивалентно бизнес-риску.", domainCard ? "Domain purity guardrail может исключить cross-domain записи на batch-слое." : "Domain purity guardrail не применялся." ], errors: [] @@ -4354,7 +4354,7 @@ export class AssistantDataLayer { domainCard ? `Domain gate source scope: ${sourceScope.join(", ")}.` : "Domain gate source scope not applied." ], risk_factors: semanticProfile.anomaly_patterns, - business_interpretation: ["Слой отражает базовый factual-срез документов для оперативной сверки."], + business_interpretation: ["Слой отражает базовый factual-срез документов для оперативной сверки."], confidence: "high", limitations: [ "Это read-only snapshot, а не онлайн-состояние 1С.", @@ -4385,11 +4385,11 @@ export class AssistantDataLayer { }, evidence: [], why_included: [], - selection_reason: ["Для drilldown требуется GUID или достаточные business anchors (номер/дата/СЃСѓРјРјР°/счет)."], + selection_reason: ["Для drilldown требуется GUID или достаточные business anchors (номер/дата/сумма/счет)."], risk_factors: [], - business_interpretation: ["Без GUID или business anchors точечный drilldown невозможен."], + business_interpretation: ["Без GUID или business anchors точечный drilldown невозможен."], confidence: "low", - limitations: ["Добавьте GUID или СЏРєРѕСЂСЏ: номер документа, дату, СЃСѓРјРјСѓ, счет."], + limitations: ["Добавьте GUID или якоря: номер документа, дату, сумму, счет."], errors: [] }; } @@ -4440,14 +4440,14 @@ export class AssistantDataLayer { evidence: matches.slice(0, 10), why_included: matches.length > 0 - ? ["Включены source-of-record записи, совпавшие РїРѕ business anchors (номер/дата/СЃСѓРјРјР°/счет)."] + ? ["Включены source-of-record записи, совпавшие по business anchors (номер/дата/сумма/счет)."] : [], selection_reason: [ - "GUID отсутствует, использован business-anchor trace РїРѕ атрибутам документа Рё расчетов." + "GUID отсутствует, использован business-anchor trace по атрибутам документа и расчетов." ], risk_factors: [], business_interpretation: [ - "Drilldown опирается РЅР° business anchors, поэтому вывод требует первичной проверки РІ source-of-record." + "Drilldown опирается на business anchors, поэтому вывод требует первичной проверки в source-of-record." ], confidence: matches.length > 0 ? "medium" : "low", limitations: [ @@ -4478,12 +4478,12 @@ export class AssistantDataLayer { matched_records: matches.length }, evidence: matches.slice(0, 10), - why_included: matches.length > 0 ? ["Включены записи, содержащие GUID РёР· запроса."] : [], - selection_reason: ["РџРѕРёСЃРє РїРѕ source_id, linked target_id Рё строковым атрибутам."], + why_included: matches.length > 0 ? ["Включены записи, содержащие GUID из запроса."] : [], + selection_reason: ["Поиск по source_id, linked target_id и строковым атрибутам."], risk_factors: [], - business_interpretation: ["Результат показывает source-of-record объекты РїРѕ переданным идентификаторам."], + business_interpretation: ["Результат показывает source-of-record объекты по переданным идентификаторам."], confidence: matches.length > 0 ? "high" : "medium", - limitations: ["РџРѕРёСЃРє ограничен локальным snapshot-пакетом."], + limitations: ["Поиск ограничен локальным snapshot-пакетом."], errors: [] }; } diff --git a/llm_normalizer/backend/src/services/assistantRuntimeGuards.ts b/llm_normalizer/backend/src/services/assistantRuntimeGuards.ts index fcbe5bd..a97eeec 100644 --- a/llm_normalizer/backend/src/services/assistantRuntimeGuards.ts +++ b/llm_normalizer/backend/src/services/assistantRuntimeGuards.ts @@ -431,7 +431,7 @@ function parseDateLike(raw: string): string | null { return normalizeDateIso({ year: parseYear(dayMonthYear[3]), month: dayMonthYear[2], day: dayMonthYear[1] }); } const rusMonthYear = value.match( - /\b(январь|февраль|март|апрель|май|РёСЋРЅСЊ|июль|август|сентябрь|октябрь|РЅРѕСЏР±СЂСЊ|декабрь)\s+(20\d{2})\b/i + /\b(январь|февраль|март|апрель|май|июнь|июль|август|сентябрь|октябрь|ноябрь|декабрь)\s+(20\d{2})\b/i ); if (rusMonthYear) { const month = RUS_MONTH_TO_NUMBER[String(rusMonthYear[1] ?? "").toLowerCase()]; @@ -530,7 +530,7 @@ function normalizedAnchorFromFragments(normalized: NormalizedPayload | null | un source: `normalized_time_scope:${type || "unknown"}` }; } - if (/(?:июл|july|РёСЋР»)/i.test(value)) { + if (/(?:июл|july|июл)/i.test(value)) { return { value: `${JULY_YEAR}-${JULY_MONTH}`, source: `normalized_time_scope:${type || "unknown"}` @@ -564,7 +564,7 @@ function resolveJulyAnchor(rawText: string): TemporalAnchorResolution { const explicitYear = lower.match(/\b(20\d{2})\b/)?.[1] ?? null; const dayByNamedJuly = lower.match(/(?:^|\D)(0?[1-9]|[12]\d|3[01])\s+(?:июл(?:я|ь)?|july)(?:\D|$)/i); const dayByNumeric = lower.match(/\b(0?[1-9]|[12]\d|3[01])[./-](0?7)(?:[./-](\d{2}|\d{4}))?\b/); - const monthByNamed = /(?:июл|july|РёСЋР»)/i.test(lower); + const monthByNamed = /(?:июл|july|июл)/i.test(lower); const monthByNumeric = /\b20\d{2}[-/.]0?7\b/.test(lower); if (!dayByNamedJuly && !dayByNumeric && !monthByNamed && !monthByNumeric) { return { @@ -771,7 +771,7 @@ export function applyTemporalHintToExecutionPlan< return item; } const text = String(item.fragment_text ?? "").trim(); - if (/2020-07|июл|РёСЋР»|july/i.test(text)) { + if (/2020-07|июл|июл|july/i.test(text)) { return item; } return { @@ -819,7 +819,7 @@ export function resolveDomainPolarityGuard(input: { prefixes.has("62") || prefixes.has("51") || prefixes.has("76") || - /(?:расч[её]т|оплат|аванс|долг|settlement|payment|tail|хвост|незакры|зач[её]т|расч|оплат|аванс|долг|С…РІРѕСЃС‚)/i.test(lower); + /(?:расч[её]т|оплат|аванс|долг|settlement|payment|tail|хвост|незакры|зач[её]т|расч|оплат|аванс|долг|хвост)/i.test(lower); if (!settlementSignal) { return { applied: false, @@ -839,13 +839,13 @@ export function resolveDomainPolarityGuard(input: { }; } const supplierScore = - (/(?:поставщ|supplier|vendor|кредитор|обязательств|payable|поставщ|кредитор|обязательств)/i.test(lower) ? 2 : 0) + + (/(?:поставщ|supplier|vendor|кредитор|обязательств|payable|поставщ|кредитор|обязательств)/i.test(lower) ? 2 : 0) + (prefixes.has("60") ? 2 : 0) + - (/(?:сч[её]т\s*60|по\s*60|счет\s*60|РїРѕ\s*60)/i.test(lower) ? 1 : 0); + (/(?:сч[её]т\s*60|по\s*60|счет\s*60)/i.test(lower) ? 1 : 0); const customerScore = - (/(?:покупат|customer|buyer|дебитор|receivable|покупат|дебитор)/i.test(lower) ? 2 : 0) + + (/(?:покупат|customer|buyer|дебитор|receivable|покупат|дебитор)/i.test(lower) ? 2 : 0) + (prefixes.has("62") ? 2 : 0) + - (/(?:сч[её]т\s*62|по\s*62|счет\s*62|РїРѕ\s*62)/i.test(lower) ? 1 : 0); + (/(?:сч[её]т\s*62|по\s*62|счет\s*62)/i.test(lower) ? 1 : 0); let polarity: DomainPolarity = "mixed_or_unresolved"; if (supplierScore > 0 || customerScore > 0) { @@ -903,10 +903,10 @@ export function applyPolarityHintToExecutionPlan< return item; } const text = String(item.fragment_text ?? "").trim(); - if (polarity.polarity === "supplier_payable" && /(поставщ|supplier|сч[её]т\s*60|по\s*60|поставщ|счет\s*60|РїРѕ\s*60)/i.test(text)) { + if (polarity.polarity === "supplier_payable" && /(поставщ|supplier|сч[её]т\s*60|по\s*60|поставщ|счет\s*60)/i.test(text)) { return item; } - if (polarity.polarity === "customer_receivable" && /(покупат|customer|сч[её]т\s*62|по\s*62|покупат|счет\s*62|РїРѕ\s*62)/i.test(text)) { + if (polarity.polarity === "customer_receivable" && /(покупат|customer|сч[её]т\s*62|по\s*62|покупат|счет\s*62)/i.test(text)) { return item; } return { @@ -917,11 +917,11 @@ export function applyPolarityHintToExecutionPlan< } function containsReceivableSignal(value: string): boolean { - return /(?:customer_settlement|stale_receivable|receivable_closed|receivable|дебитор)/i.test(value); + return /(?:customer_settlement|stale_receivable|receivable_closed|receivable|дебитор)/i.test(value); } function containsPayableSignal(value: string): boolean { - return /(?:bank_settlement|payable|обязательств|supplier|поставщ|счет\s*60|\b60(?:\.\d{2})?\b)/i.test(value); + return /(?:bank_settlement|payable|обязательств|supplier|поставщ|счет\s*60|\b60(?:\.\d{2})?\b)/i.test(value); } function problemUnitCorpus(unit: ProblemUnit): string { @@ -1590,15 +1590,15 @@ export function applyEligibilityToGroundingCheck = { - admissible_evidence_count_zero: "Недостаточно допустимого evidence для обоснованного ответа.", - critical_domain_or_account_contradiction: "Есть критическое противоречие РїРѕ domain/account scope.", - temporal_guard_failed_out_of_snapshot_window: "Temporal anchor вышел Р·Р° РѕРєРЅРѕ company snapshot (июль 2020).", - temporal_guard_ambiguous_limited: "Temporal anchor РЅРµ разрешен надежно РІ пределах company snapshot.", + admissible_evidence_count_zero: "Недостаточно допустимого evidence для обоснованного ответа.", + critical_domain_or_account_contradiction: "Есть критическое противоречие по domain/account scope.", + temporal_guard_failed_out_of_snapshot_window: "Temporal anchor вышел за окно company snapshot (июль 2020).", + temporal_guard_ambiguous_limited: "Temporal anchor не разрешен надежно в пределах company snapshot.", business_scope_generic_unresolved: "Business scope остался generic и не подтвержден как company-specific для доказательного ответа.", - polarity_guard_limited_unresolved_polarity: "РќРµ удалось надежно определить supplier/customer polarity.", - polarity_guard_blocked_conflict: "Обнаружен конфликт supplier/customer polarity РІ retrieval-контуре.", - claim_anchor_coverage_insufficient: "Недостаточно покрытия required anchors для claim-bound grounding.", - targeted_evidence_hit_rate_zero: "Targeted evidence acquisition РЅРµ дал допустимых попаданий РїРѕ claim target path." + polarity_guard_limited_unresolved_polarity: "Не удалось надежно определить supplier/customer polarity.", + polarity_guard_blocked_conflict: "Обнаружен конфликт supplier/customer polarity в retrieval-контуре.", + claim_anchor_coverage_insufficient: "Недостаточно покрытия required anchors для claim-bound grounding.", + targeted_evidence_hit_rate_zero: "Targeted evidence acquisition не дал допустимых попаданий по claim target path." }; const reasons = [ ...(Array.isArray(groundingCheck.reasons) ? groundingCheck.reasons : []), diff --git a/llm_normalizer/backend/src/services/assistantService.ts b/llm_normalizer/backend/src/services/assistantService.ts index 5a9ccb1..df5b1d6 100644 --- a/llm_normalizer/backend/src/services/assistantService.ts +++ b/llm_normalizer/backend/src/services/assistantService.ts @@ -19,6 +19,8 @@ import * as addressFilterExtractor_1 from "./addressFilterExtractor"; import * as predecomposeContract_1 from "./address_runtime/predecomposeContract"; import * as openaiResponsesClient_1 from "./openaiResponsesClient"; import * as addressMcpClient_1 from "./addressMcpClient"; +import * as capabilitiesRegistry_1 from "./capabilitiesRegistry"; +import * as assistantCanon_1 from "./assistantCanon"; import iconv from "iconv-lite"; const DATA_SCOPE_CACHE_TTL_MS = 60_000; const dataScopeProbeCache = new Map(); @@ -3859,17 +3861,7 @@ function buildLivingChatPrompt(userMessage, conversationWindow) { return `${contextBlock}Сообщение пользователя:\n${userMessage}`; } function buildAssistantCapabilityContractReply() { - return [ - "Я ассистент по анализу данных 1С в режиме чтения.", - "Что умею сейчас:", - "1. Находить документы, операции, договоры и остатки по контрагенту/договору/периоду.", - "2. Делать агрегаты по базе: активность, роли контрагентов, top-срезы по суммам и операциям.", - "3. Кратко объяснять результат и подсказывать следующий точный запрос.", - "Что не умею:", - "1. Не настраиваю 1С и не меняю конфигурацию.", - "2. Не создаю и не провожу документы в базе.", - "3. Не выполняю админские действия на сервере." - ].join("\n"); + return (0, capabilitiesRegistry_1.buildCapabilityContractReplyFromRegistry)(); } function normalizeScopeLabel(value) { const repaired = repairAddressMojibake(String(value ?? "")); @@ -4971,6 +4963,7 @@ export class AssistantService { else { const conversationWindow = buildLivingChatContextWindow(session.items); const userPrompt = buildLivingChatPrompt(userMessage, conversationWindow); + const canonExcerpt = (0, assistantCanon_1.loadAssistantCanonExcerpt)(520); const chatResponse = await this.chatClient.chat({ llmProvider: payload.llmProvider, apiKey: String(payload.apiKey ?? process.env.OPENAI_API_KEY ?? ""), @@ -4984,7 +4977,8 @@ export class AssistantService { "Работай честно: не заявляй действия, которые недоступны в этом рантайме.", "Разрешено: анализ и объяснение данных, формулировка запросов, подсказки по следующему шагу.", "Запрещено: обещать настройку 1С, админ-действия, создание/проведение документов или любые изменения в базе.", - "Если пользователь спрашивает про возможности, отвечай только по этому контракту." + "Если пользователь спрашивает про возможности, отвечай только по этому контракту.", + `Канон поведения: ${canonExcerpt}` ].join(" "), developerPrompt: "Формат: коротко и по сути, без JSON и без служебных блоков. Пиши человеко-понятно.", userMessage: userPrompt, diff --git a/llm_normalizer/backend/src/services/capabilitiesRegistry.ts b/llm_normalizer/backend/src/services/capabilitiesRegistry.ts new file mode 100644 index 0000000..42da52b --- /dev/null +++ b/llm_normalizer/backend/src/services/capabilitiesRegistry.ts @@ -0,0 +1,205 @@ +import fs from "fs"; +import { ASSISTANT_CAPABILITIES_REGISTRY_FILE } from "../config"; + +export type CapabilityMaturity = "production_ready" | "partial" | "planned" | "deprecated"; + +export interface CapabilityGroup { + group_code: string; + group_title: string; + description: string; + risk_level: "low" | "medium" | "high"; + maturity_status: CapabilityMaturity; + supported_operations: string[]; + unsupported_operations: string[]; + required_entities: string[]; + optional_entities: string[]; + typical_queries: string[]; + related_routes: string[]; + safe_alternatives: string[]; + one_c_hints: string[]; +} + +export interface CapabilityRegistry { + schema_version: string; + updated_at: string; + assistant_mode: "read_only" | "mixed"; + groups: CapabilityGroup[]; +} + +const FALLBACK_REGISTRY: CapabilityRegistry = { + schema_version: "capabilities_registry_fallback_v1", + updated_at: "2026-04-09T00:00:00.000Z", + assistant_mode: "read_only", + groups: [ + { + group_code: "vat", + group_title: "НДС", + description: "Срезы и расчеты НДС на базе данных 1С.", + risk_level: "high", + maturity_status: "partial", + supported_operations: ["vat_period_snapshot", "vat_payable_forecast"], + unsupported_operations: ["submit_tax_declaration"], + required_entities: ["period", "organization"], + optional_entities: ["counterparty"], + typical_queries: ["Сколько НДС к уплате за период?"], + related_routes: [], + safe_alternatives: ["Показать движения по 68/19 за период"], + one_c_hints: ["РегистрБухгалтерии.Хозрасчетный"] + }, + { + group_code: "counterparties", + group_title: "Контрагенты", + description: "Документы, операции, договоры и срезы по контрагентам.", + risk_level: "medium", + maturity_status: "production_ready", + supported_operations: ["list_documents_by_counterparty", "list_contracts_by_counterparty"], + unsupported_operations: ["edit_counterparty_card"], + required_entities: ["counterparty_scope_or_contract"], + optional_entities: ["period", "organization"], + typical_queries: ["Покажи документы по контрагенту"], + related_routes: [], + safe_alternatives: ["Уточнить ИНН/наименование контрагента"], + one_c_hints: ["Справочник.Контрагенты"] + }, + { + group_code: "boundaries", + group_title: "Ограничения", + description: "Операции, которые ассистент не выполняет.", + risk_level: "high", + maturity_status: "production_ready", + supported_operations: ["explain_boundary", "suggest_safe_next_step"], + unsupported_operations: ["configure_1c", "admin_server_actions", "create_or_post_documents"], + required_entities: [], + optional_entities: [], + typical_queries: ["Можешь настроить 1С?"], + related_routes: [], + safe_alternatives: ["Сформировать план диагностики для 1С/ИТ-админа"], + one_c_hints: [] + } + ] +}; + +let cache: { mtimeMs: number; value: CapabilityRegistry } | null = null; + +function toRecord(value: unknown): Record | null { + if (!value || typeof value !== "object" || Array.isArray(value)) return null; + return value as Record; +} + +function toStringSafe(value: unknown): string | null { + if (typeof value !== "string") return null; + const trimmed = value.trim(); + return trimmed.length > 0 ? trimmed : null; +} + +function toArray(value: unknown): unknown[] { + return Array.isArray(value) ? value : []; +} + +function readRegistryFromFile(): CapabilityRegistry | null { + if (!fs.existsSync(ASSISTANT_CAPABILITIES_REGISTRY_FILE)) return null; + try { + const raw = fs.readFileSync(ASSISTANT_CAPABILITIES_REGISTRY_FILE, "utf-8"); + const parsed = JSON.parse(raw) as unknown; + const root = toRecord(parsed); + if (!root) return null; + const groups = toArray(root.groups) + .map((item) => toRecord(item)) + .filter((item): item is Record => item !== null) + .map((item) => ({ + group_code: toStringSafe(item.group_code) ?? "unknown_group", + group_title: toStringSafe(item.group_title) ?? "Группа", + description: toStringSafe(item.description) ?? "", + risk_level: (toStringSafe(item.risk_level) as "low" | "medium" | "high" | null) ?? "medium", + maturity_status: + (toStringSafe(item.maturity_status) as CapabilityMaturity | null) ?? + ("partial" as CapabilityMaturity), + supported_operations: toArray(item.supported_operations) + .map((v) => toStringSafe(v)) + .filter((v): v is string => v !== null), + unsupported_operations: toArray(item.unsupported_operations) + .map((v) => toStringSafe(v)) + .filter((v): v is string => v !== null), + required_entities: toArray(item.required_entities) + .map((v) => toStringSafe(v)) + .filter((v): v is string => v !== null), + optional_entities: toArray(item.optional_entities) + .map((v) => toStringSafe(v)) + .filter((v): v is string => v !== null), + typical_queries: toArray(item.typical_queries) + .map((v) => toStringSafe(v)) + .filter((v): v is string => v !== null), + related_routes: toArray(item.related_routes) + .map((v) => toStringSafe(v)) + .filter((v): v is string => v !== null), + safe_alternatives: toArray(item.safe_alternatives) + .map((v) => toStringSafe(v)) + .filter((v): v is string => v !== null), + one_c_hints: toArray(item.one_c_hints) + .map((v) => toStringSafe(v)) + .filter((v): v is string => v !== null) + })); + + if (groups.length === 0) return null; + + return { + schema_version: toStringSafe(root.schema_version) ?? "capabilities_registry_v1", + updated_at: toStringSafe(root.updated_at) ?? new Date().toISOString(), + assistant_mode: (toStringSafe(root.assistant_mode) as "read_only" | "mixed" | null) ?? "read_only", + groups + }; + } catch { + return null; + } +} + +export function loadCapabilitiesRegistry(): CapabilityRegistry { + try { + const mtimeMs = fs.existsSync(ASSISTANT_CAPABILITIES_REGISTRY_FILE) + ? fs.statSync(ASSISTANT_CAPABILITIES_REGISTRY_FILE).mtimeMs + : -1; + if (cache && cache.mtimeMs === mtimeMs) { + return cache.value; + } + const value = readRegistryFromFile() ?? FALLBACK_REGISTRY; + cache = { mtimeMs, value }; + return value; + } catch { + return cache?.value ?? FALLBACK_REGISTRY; + } +} + +export function buildCapabilityContractReplyFromRegistry(): string { + const registry = loadCapabilitiesRegistry(); + const topGroups = registry.groups.slice(0, 6); + const groupLines = topGroups.map((group, index) => { + const ops = group.supported_operations.slice(0, 3).join(", "); + return `${index + 1}. ${group.group_title}: ${group.description}${ops ? ` (например: ${ops})` : ""}.`; + }); + + return [ + "Я ассистент по анализу данных 1С в режиме чтения.", + "Что умею по группам:", + ...groupLines, + "Если хотите, раскрою любую группу точечно и дам готовую формулировку запроса.", + "Что не делаю: не настраиваю 1С, не меняю конфигурацию, не создаю и не провожу документы, не выполняю админ-действия на сервере." + ].join("\n"); +} + +export function resolveNearestCapabilityGroup(input: { domain?: string | null; queryClass?: string | null }): CapabilityGroup | null { + const registry = loadCapabilitiesRegistry(); + const haystack = `${String(input.domain ?? "")} ${String(input.queryClass ?? "")}`.toLowerCase(); + if (!haystack.trim()) return null; + + const scoring: Array<{ group: CapabilityGroup; score: number }> = registry.groups.map((group) => { + let score = 0; + const bucket = `${group.group_code} ${group.group_title} ${group.description} ${group.supported_operations.join(" ")}`.toLowerCase(); + for (const token of haystack.split(/[\s._/-]+/g).filter(Boolean)) { + if (bucket.includes(token)) score += 1; + } + return { group, score }; + }); + + scoring.sort((a, b) => b.score - a.score); + return scoring[0] && scoring[0].score > 0 ? scoring[0].group : null; +} diff --git a/llm_normalizer/backend/src/services/lifecycleRuntime.ts b/llm_normalizer/backend/src/services/lifecycleRuntime.ts index f02a518..66f10f8 100644 --- a/llm_normalizer/backend/src/services/lifecycleRuntime.ts +++ b/llm_normalizer/backend/src/services/lifecycleRuntime.ts @@ -150,53 +150,53 @@ const LIFECYCLE_DOMAIN_MODELS: Record = { states: [ { state_code: "initiated_payment", - state_label: "Платеж инициирован", + state_label: "Платеж инициирован", state_class: "initial", entry_conditions: ["payment_order_created"], exit_conditions: ["bank_recorded"], is_terminal: false, is_problematic: false, - business_meaning: "Есть инициирование платежа." + business_meaning: "Есть инициирование платежа." }, { state_code: "bank_recorded", - state_label: "Платеж отражен банком", + state_label: "Платеж отражен банком", state_class: "active", entry_conditions: ["bank_statement_recorded"], exit_conditions: ["settlement_linked"], is_terminal: false, is_problematic: false, - business_meaning: "Движение денег зафиксировано, ожидается расчетное закрытие." + business_meaning: "Движение денег зафиксировано, ожидается расчетное закрытие." }, { state_code: "settlement_closed", - state_label: "Расчет закрыт", + state_label: "Расчет закрыт", state_class: "terminal", entry_conditions: ["payment_to_settlement_linked"], exit_conditions: [], is_terminal: true, is_problematic: false, - business_meaning: "Платеж доведен РґРѕ расчетного результата." + business_meaning: "Платеж доведен до расчетного результата." }, { state_code: "stale_unlinked_payment", - state_label: "Платеж завис без закрытия", + state_label: "Платеж завис без закрытия", state_class: "problematic", entry_conditions: ["bank_recorded", "missing_link"], exit_conditions: ["settlement_closed"], is_terminal: false, is_problematic: true, - business_meaning: "Платеж отражен, РЅРѕ ожидаемая СЃРІСЏР·СЊ РїРѕ расчету РЅРµ завершена." + business_meaning: "Платеж отражен, но ожидаемая связь по расчету не завершена." }, { state_code: "misclosed_payment", - state_label: "Платеж закрыт некорректно", + state_label: "Платеж закрыт некорректно", state_class: "problematic", entry_conditions: ["wrong_document_type_or_posting_mismatch"], exit_conditions: ["settlement_closed"], is_terminal: false, is_problematic: true, - business_meaning: "Формальное закрытие есть, РЅРѕ путь закрытия неверный." + business_meaning: "Формальное закрытие есть, но путь закрытия неверный." } ], transitions: [ @@ -207,7 +207,7 @@ const LIFECYCLE_DOMAIN_MODELS: Record = { required_evidence: ["bank_statement_recorded"], optional_evidence: ["payment_order"], forbidden_conditions: [], - business_meaning: "Платеж должен появиться РІРѕ выписке." + business_meaning: "Платеж должен появиться во выписке." }, { from_state: "bank_recorded", @@ -216,7 +216,7 @@ const LIFECYCLE_DOMAIN_MODELS: Record = { required_evidence: ["payment_to_settlement_link"], optional_evidence: ["document_to_posting"], forbidden_conditions: ["wrong_document_type"], - business_meaning: "После выписки должен закрываться расчет." + business_meaning: "После выписки должен закрываться расчет." } ], defects: [] @@ -228,43 +228,43 @@ const LIFECYCLE_DOMAIN_MODELS: Record = { states: [ { state_code: "invoice_issued", - state_label: "Реализация отражена", + state_label: "Реализация отражена", state_class: "initial", entry_conditions: ["realization_document_exists"], exit_conditions: ["payment_recorded"], is_terminal: false, is_problematic: false, - business_meaning: "Возникла дебиторская позиция." + business_meaning: "Возникла дебиторская позиция." }, { state_code: "payment_recorded", - state_label: "Оплата отражена", + state_label: "Оплата отражена", state_class: "active", entry_conditions: ["payment_document_exists"], exit_conditions: ["receivable_closed"], is_terminal: false, is_problematic: false, - business_meaning: "Оплата есть, ожидается корректное закрытие." + business_meaning: "Оплата есть, ожидается корректное закрытие." }, { state_code: "receivable_closed", - state_label: "Дебиторка закрыта", + state_label: "Дебиторка закрыта", state_class: "terminal", entry_conditions: ["closing_document_linked"], exit_conditions: [], is_terminal: true, is_problematic: false, - business_meaning: "Дебиторская позиция закрыта корректно." + business_meaning: "Дебиторская позиция закрыта корректно." }, { state_code: "stale_receivable", - state_label: "Дебиторка зависла", + state_label: "Дебиторка зависла", state_class: "problematic", entry_conditions: ["unresolved_settlement"], exit_conditions: ["receivable_closed"], is_terminal: false, is_problematic: true, - business_meaning: "Позиция остается незавершенной дольше ожидаемого." + business_meaning: "Позиция остается незавершенной дольше ожидаемого." } ], transitions: [ @@ -275,7 +275,7 @@ const LIFECYCLE_DOMAIN_MODELS: Record = { required_evidence: ["payment_document_exists"], optional_evidence: [], forbidden_conditions: [], - business_meaning: "После реализации ожидается оплата/зачет." + business_meaning: "После реализации ожидается оплата/зачет." }, { from_state: "payment_recorded", @@ -284,7 +284,7 @@ const LIFECYCLE_DOMAIN_MODELS: Record = { required_evidence: ["closing_document_linked"], optional_evidence: ["register_movement_exists"], forbidden_conditions: ["cross_branch_inconsistency"], - business_meaning: "Оплата должна завершаться корректным закрытием расчета." + business_meaning: "Оплата должна завершаться корректным закрытием расчета." } ], defects: [] @@ -296,43 +296,43 @@ const LIFECYCLE_DOMAIN_MODELS: Record = { states: [ { state_code: "recognized", - state_label: "РБП признан", + state_label: "РБП признан", state_class: "initial", entry_conditions: ["deferred_expense_created"], exit_conditions: ["writeoff_started"], is_terminal: false, is_problematic: false, - business_meaning: "РБП поставлен РЅР° учет." + business_meaning: "РБП поставлен на учет." }, { state_code: "partially_written_off", - state_label: "Частичное списание", + state_label: "Частичное списание", state_class: "active", entry_conditions: ["partial_writeoff_exists"], exit_conditions: ["fully_written_off"], is_terminal: false, is_problematic: false, - business_meaning: "Списание идет РїРѕ графику." + business_meaning: "Списание идет по графику." }, { state_code: "fully_written_off", - state_label: "РБП полностью списан", + state_label: "РБП полностью списан", state_class: "terminal", entry_conditions: ["full_writeoff_exists"], exit_conditions: [], is_terminal: true, is_problematic: false, - business_meaning: "РБП завершил lifecycle." + business_meaning: "РБП завершил lifecycle." }, { state_code: "overdue_writeoff", - state_label: "Просроченное списание", + state_label: "Просроченное списание", state_class: "problematic", entry_conditions: ["period_boundary", "missing_link"], exit_conditions: ["fully_written_off"], is_terminal: false, is_problematic: true, - business_meaning: "РБП живет дольше допустимого РѕРєРЅР°." + business_meaning: "РБП живет дольше допустимого окна." } ], transitions: [], @@ -345,53 +345,53 @@ const LIFECYCLE_DOMAIN_MODELS: Record = { states: [ { state_code: "capitalized", - state_label: "Капвложения отражены", + state_label: "Капвложения отражены", state_class: "initial", entry_conditions: ["capitalization_document_exists"], exit_conditions: ["accepted_for_accounting"], is_terminal: false, is_problematic: false, - business_meaning: "Объект зафиксирован как вложение." + business_meaning: "Объект зафиксирован как вложение." }, { state_code: "accepted_for_accounting", - state_label: "РџСЂРёРЅСЏС‚ Рє учету", + state_label: "Принят к учету", state_class: "active", entry_conditions: ["acceptance_document_exists"], exit_conditions: ["depreciation_active"], is_terminal: false, is_problematic: false, - business_meaning: "Объект переведен РІ РѕСЃРЅРѕРІРЅРѕР№ контур учета." + business_meaning: "Объект переведен в основной контур учета." }, { state_code: "depreciation_active", - state_label: "Амортизация активна", + state_label: "Амортизация активна", state_class: "active", entry_conditions: ["depreciation_register_movement"], exit_conditions: ["disposed"], is_terminal: false, is_problematic: false, - business_meaning: "Жизненный цикл РћРЎ идет штатно." + business_meaning: "Жизненный цикл ОС идет штатно." }, { state_code: "contradictory_asset_state", - state_label: "Противоречивый статус РћРЎ", + state_label: "Противоречивый статус ОС", state_class: "problematic", entry_conditions: ["posting_mismatch_or_wrong_path"], exit_conditions: ["depreciation_active"], is_terminal: false, is_problematic: true, - business_meaning: "Статус РћРЎ формально есть, РЅРѕ смыслово противоречив." + business_meaning: "Статус ОС формально есть, но смыслово противоречив." }, { state_code: "disposed", - state_label: "Выбыл", + state_label: "Выбыл", state_class: "terminal", entry_conditions: ["disposal_document_exists"], exit_conditions: [], is_terminal: true, is_problematic: false, - business_meaning: "Жизненный цикл РћРЎ завершен." + business_meaning: "Жизненный цикл ОС завершен." } ], transitions: [], @@ -404,43 +404,43 @@ const LIFECYCLE_DOMAIN_MODELS: Record = { states: [ { state_code: "vat_registered", - state_label: "НДС отражен документно", + state_label: "НДС отражен документно", state_class: "initial", entry_conditions: ["invoice_registered"], exit_conditions: ["vat_reflected"], is_terminal: false, is_problematic: false, - business_meaning: "Сформирован первичный документный слой НДС." + business_meaning: "Сформирован первичный документный слой НДС." }, { state_code: "vat_reflected", - state_label: "НДС отражен РІ учете", + state_label: "НДС отражен в учете", state_class: "active", entry_conditions: ["vat_register_movement"], exit_conditions: ["vat_deducted"], is_terminal: false, is_problematic: false, - business_meaning: "НДС РїСЂРѕС…РѕРґРёС‚ штатную стадию отражения." + business_meaning: "НДС проходит штатную стадию отражения." }, { state_code: "vat_deducted", - state_label: "НДС РїСЂРёРЅСЏС‚ Рє вычету", + state_label: "НДС принят к вычету", state_class: "terminal", entry_conditions: ["deduction_confirmed"], exit_conditions: [], is_terminal: true, is_problematic: false, - business_meaning: "НДС-цепочка завершена корректно." + business_meaning: "НДС-цепочка завершена корректно." }, { state_code: "vat_conflict", - state_label: "Конфликт НДС-цепочки", + state_label: "Конфликт НДС-цепочки", state_class: "problematic", entry_conditions: ["cross_branch_inconsistency"], exit_conditions: ["vat_reflected"], is_terminal: false, is_problematic: true, - business_meaning: "Бухгалтерская Рё налоговая ветки расходятся." + business_meaning: "Бухгалтерская и налоговая ветки расходятся." } ], transitions: [], @@ -453,53 +453,53 @@ const LIFECYCLE_DOMAIN_MODELS: Record = { states: [ { state_code: "preclose_checks", - state_label: "Предзакрытие", + state_label: "Предзакрытие", state_class: "active", entry_conditions: ["period_scope_detected"], exit_conditions: ["close_ready"], is_terminal: false, is_problematic: false, - business_meaning: "Идет проверка готовности периода." + business_meaning: "Идет проверка готовности периода." }, { state_code: "close_ready", - state_label: "Готов Рє закрытию", + state_label: "Готов к закрытию", state_class: "active", entry_conditions: ["no_blockers_detected"], exit_conditions: ["close_completed"], is_terminal: false, is_problematic: false, - business_meaning: "Период может быть закрыт." + business_meaning: "Период может быть закрыт." }, { state_code: "close_completed", - state_label: "Закрытие завершено", + state_label: "Закрытие завершено", state_class: "terminal", entry_conditions: ["close_operation_done"], exit_conditions: [], is_terminal: true, is_problematic: false, - business_meaning: "Период закрыт." + business_meaning: "Период закрыт." }, { state_code: "close_blocked", - state_label: "Закрытие заблокировано", + state_label: "Закрытие заблокировано", state_class: "problematic", entry_conditions: ["period_close_risk_or_stale_state"], exit_conditions: ["close_ready"], is_terminal: false, is_problematic: true, - business_meaning: "Есть lifecycle-дефекты, влияющие РЅР° закрытие." + business_meaning: "Есть lifecycle-дефекты, влияющие на закрытие." }, { state_code: "close_contradicted", - state_label: "Закрыт формально, РЅРѕ СЃ противоречием", + state_label: "Закрыт формально, но с противоречием", state_class: "problematic", entry_conditions: ["misclosed_or_cross_branch_conflict"], exit_conditions: ["close_completed"], is_terminal: false, is_problematic: true, - business_meaning: "Формальное закрытие РЅРµ согласовано СЃ фактическими ветками." + business_meaning: "Формальное закрытие не согласовано с фактическими ветками." } ], transitions: [], @@ -512,7 +512,7 @@ const SHARED_DEFECTS: LifecycleDefectDefinition[] = [ defect_code: "missing_expected_transition", defect_class: "path", severity_hint: "medium", - business_meaning: "Ожидаемый переход РЅРµ произошел.", + business_meaning: "Ожидаемый переход не произошел.", evidence_requirements: ["expected_state", "missing_transition_signal"], period_impact_potential: "indirect" }, @@ -520,7 +520,7 @@ const SHARED_DEFECTS: LifecycleDefectDefinition[] = [ defect_code: "invalid_transition", defect_class: "path", severity_hint: "high", - business_meaning: "Переход произошел РїРѕ некорректному пути.", + business_meaning: "Переход произошел по некорректному пути.", evidence_requirements: ["invalid_transition_signal"], period_impact_potential: "indirect" }, @@ -528,7 +528,7 @@ const SHARED_DEFECTS: LifecycleDefectDefinition[] = [ defect_code: "stale_active_state", defect_class: "timing", severity_hint: "high", - business_meaning: "Объект завис РІ активном состоянии.", + business_meaning: "Объект завис в активном состоянии.", evidence_requirements: ["stale_marker", "missing_transition_signal"], period_impact_potential: "direct" }, @@ -536,7 +536,7 @@ const SHARED_DEFECTS: LifecycleDefectDefinition[] = [ defect_code: "contradictory_state", defect_class: "consistency", severity_hint: "high", - business_meaning: "Статусы объекта противоречат РґСЂСѓРі РґСЂСѓРіСѓ.", + business_meaning: "Статусы объекта противоречат друг другу.", evidence_requirements: ["contradiction_signal"], period_impact_potential: "direct" }, @@ -544,7 +544,7 @@ const SHARED_DEFECTS: LifecycleDefectDefinition[] = [ defect_code: "premature_terminal_state", defect_class: "closure", severity_hint: "medium", - business_meaning: "Терминальное состояние наступило преждевременно.", + business_meaning: "Терминальное состояние наступило преждевременно.", evidence_requirements: ["terminal_state", "missing_required_previous_state"], period_impact_potential: "indirect" }, @@ -552,7 +552,7 @@ const SHARED_DEFECTS: LifecycleDefectDefinition[] = [ defect_code: "misclosed_state", defect_class: "closure", severity_hint: "high", - business_meaning: "Контур формально закрыт, РЅРѕ закрыт неверно.", + business_meaning: "Контур формально закрыт, но закрыт неверно.", evidence_requirements: ["wrong_closure_path"], period_impact_potential: "direct" }, @@ -560,7 +560,7 @@ const SHARED_DEFECTS: LifecycleDefectDefinition[] = [ defect_code: "orphan_intermediate_state", defect_class: "path", severity_hint: "medium", - business_meaning: "Промежуточная стадия осталась без корректного продолжения.", + business_meaning: "Промежуточная стадия осталась без корректного продолжения.", evidence_requirements: ["intermediate_state_without_next"], period_impact_potential: "indirect" }, @@ -568,7 +568,7 @@ const SHARED_DEFECTS: LifecycleDefectDefinition[] = [ defect_code: "cross_branch_state_conflict", defect_class: "consistency", severity_hint: "high", - business_meaning: "Состояния соседних веток учета противоречат РґСЂСѓРі РґСЂСѓРіСѓ.", + business_meaning: "Состояния соседних веток учета противоречат друг другу.", evidence_requirements: ["cross_branch_conflict_signal"], period_impact_potential: "direct" } @@ -905,23 +905,23 @@ function lifecycleInterpretation(input: { missingTransition: string | null; invalidTransition: string | null; }): string { - const base = `Текущая стадия: ${input.currentState}; ожидаемая стадия: ${input.expectedState}.`; + const base = `Текущая стадия: ${input.currentState}; ожидаемая стадия: ${input.expectedState}.`; if (input.defect === "stale_active_state") { - return `${base} Объект завис РІРѕ времени Рё РЅРµ дошел РґРѕ ожидаемого перехода.`; + return `${base} Объект завис во времени и не дошел до ожидаемого перехода.`; } if (input.defect === "misclosed_state") { - return `${base} Контур закрыт формально, РЅРѕ путь закрытия противоречит бухгалтерской логике.`; + return `${base} Контур закрыт формально, но путь закрытия противоречит бухгалтерской логике.`; } if (input.defect === "cross_branch_state_conflict") { - return `${base} Между ветками домена ${input.domain} обнаружено противоречие состояний.`; + return `${base} Между ветками домена ${input.domain} обнаружено противоречие состояний.`; } if (input.defect === "missing_expected_transition") { - return `${base} РќРµ зафиксирован ожидаемый переход (${input.missingTransition ?? "unknown_transition"}).`; + return `${base} Не зафиксирован ожидаемый переход (${input.missingTransition ?? "unknown_transition"}).`; } if (input.defect === "invalid_transition") { - return `${base} Зафиксирован некорректный переход (${input.invalidTransition ?? "invalid_transition"}).`; + return `${base} Зафиксирован некорректный переход (${input.invalidTransition ?? "invalid_transition"}).`; } - return `${base} Lifecycle-разрешение РЅРµ выявило критичный дефект, РЅРѕ состояние требует наблюдения.`; + return `${base} Lifecycle-разрешение не выявило критичный дефект, но состояние требует наблюдения.`; } export function resolveLifecycle(input: LifecycleResolverInput): LifecycleResolution { diff --git a/llm_normalizer/backend/src/services/promptBuilder.ts b/llm_normalizer/backend/src/services/promptBuilder.ts index c7a90e1..28a9bd4 100644 --- a/llm_normalizer/backend/src/services/promptBuilder.ts +++ b/llm_normalizer/backend/src/services/promptBuilder.ts @@ -78,10 +78,10 @@ const BUILTIN_PROMPT_PRESETS: Record{for(const F of k)if(F.type==="childList")for(const H of F.addedNodes)H.tagName==="LINK"&&H.rel==="modulepreload"&&D(H)}).observe(document,{childList:!0,subtree:!0});function p(k){const F={};return k.integrity&&(F.integrity=k.integrity),k.referrerPolicy&&(F.referrerPolicy=k.referrerPolicy),k.crossOrigin==="use-credentials"?F.credentials="include":k.crossOrigin==="anonymous"?F.credentials="omit":F.credentials="same-origin",F}function D(k){if(k.ep)return;k.ep=!0;const F=p(k);fetch(k.href,F)}})();function rc(a){return a&&a.__esModule&&Object.prototype.hasOwnProperty.call(a,"default")?a.default:a}var bo={exports:{}},rl={},ei={exports:{}},se={};var Fu;function Xd(){if(Fu)return se;Fu=1;var a=Symbol.for("react.element"),h=Symbol.for("react.portal"),p=Symbol.for("react.fragment"),D=Symbol.for("react.strict_mode"),k=Symbol.for("react.profiler"),F=Symbol.for("react.provider"),H=Symbol.for("react.context"),Z=Symbol.for("react.forward_ref"),N=Symbol.for("react.suspense"),te=Symbol.for("react.memo"),ce=Symbol.for("react.lazy"),z=Symbol.iterator;function L(m){return m===null||typeof m!="object"?null:(m=z&&m[z]||m["@@iterator"],typeof m=="function"?m:null)}var Le={isMounted:function(){return!1},enqueueForceUpdate:function(){},enqueueReplaceState:function(){},enqueueSetState:function(){}},me=Object.assign,re={};function G(m,_,Y){this.props=m,this.context=_,this.refs=re,this.updater=Y||Le}G.prototype.isReactComponent={},G.prototype.setState=function(m,_){if(typeof m!="object"&&typeof m!="function"&&m!=null)throw Error("setState(...): takes an object of state variables to update or a function which returns an object of state variables.");this.updater.enqueueSetState(this,m,_,"setState")},G.prototype.forceUpdate=function(m){this.updater.enqueueForceUpdate(this,m,"forceUpdate")};function De(){}De.prototype=G.prototype;function ye(m,_,Y){this.props=m,this.context=_,this.refs=re,this.updater=Y||Le}var We=ye.prototype=new De;We.constructor=ye,me(We,G.prototype),We.isPureReactComponent=!0;var xe=Array.isArray,ze=Object.prototype.hasOwnProperty,X={current:null},$e={key:!0,ref:!0,__self:!0,__source:!0};function Ne(m,_,Y){var J,le={},ee=null,de=null;if(_!=null)for(J in _.ref!==void 0&&(de=_.ref),_.key!==void 0&&(ee=""+_.key),_)ze.call(_,J)&&!$e.hasOwnProperty(J)&&(le[J]=_[J]);var ae=arguments.length-2;if(ae===1)le.children=Y;else if(1>>1,_=R[m];if(0>>1;mk(le,T))ee<_&&0>k(de,le)?(R[m]=de,R[ee]=T,m=ee):(R[m]=le,R[J]=T,m=J);else if(ee<_&&0>k(de,T))R[m]=de,R[ee]=T,m=ee;else break e}}return V}function k(R,V){var T=R.sortIndex-V.sortIndex;return T!==0?T:R.id-V.id}if(typeof performance=="object"&&typeof performance.now=="function"){var F=performance;a.unstable_now=function(){return F.now()}}else{var H=Date,Z=H.now();a.unstable_now=function(){return H.now()-Z}}var N=[],te=[],ce=1,z=null,L=3,Le=!1,me=!1,re=!1,G=typeof setTimeout=="function"?setTimeout:null,De=typeof clearTimeout=="function"?clearTimeout:null,ye=typeof setImmediate<"u"?setImmediate:null;typeof navigator<"u"&&navigator.scheduling!==void 0&&navigator.scheduling.isInputPending!==void 0&&navigator.scheduling.isInputPending.bind(navigator.scheduling);function We(R){for(var V=p(te);V!==null;){if(V.callback===null)D(te);else if(V.startTime<=R)D(te),V.sortIndex=V.expirationTime,h(N,V);else break;V=p(te)}}function xe(R){if(re=!1,We(R),!me)if(p(N)!==null)me=!0,b(ze);else{var V=p(te);V!==null&&pe(xe,V.startTime-R)}}function ze(R,V){me=!1,re&&(re=!1,De(Ne),Ne=-1),Le=!0;var T=L;try{for(We(V),z=p(N);z!==null&&(!(z.expirationTime>V)||R&&!gt());){var m=z.callback;if(typeof m=="function"){z.callback=null,L=z.priorityLevel;var _=m(z.expirationTime<=V);V=a.unstable_now(),typeof _=="function"?z.callback=_:z===p(N)&&D(N),We(V)}else D(N);z=p(N)}if(z!==null)var Y=!0;else{var J=p(te);J!==null&&pe(xe,J.startTime-V),Y=!1}return Y}finally{z=null,L=T,Le=!1}}var X=!1,$e=null,Ne=-1,ut=5,be=-1;function gt(){return!(a.unstable_now()-beR||125m?(R.sortIndex=T,h(te,R),p(N)===null&&R===p(te)&&(re?(De(Ne),Ne=-1):re=!0,pe(xe,T-m))):(R.sortIndex=_,h(N,R),me||Le||(me=!0,b(ze))),R},a.unstable_shouldYield=gt,a.unstable_wrapCallback=function(R){var V=L;return function(){var T=L;L=V;try{return R.apply(this,arguments)}finally{L=T}}}})(ri)),ri}var Vu;function nf(){return Vu||(Vu=1,ni.exports=tf()),ni.exports}var Qu;function rf(){if(Qu)return vt;Qu=1;var a=oi(),h=nf();function p(e){for(var t="https://reactjs.org/docs/error-decoder.html?invariant="+e,n=1;n"u"||typeof window.document>"u"||typeof window.document.createElement>"u"),N=Object.prototype.hasOwnProperty,te=/^[:A-Z_a-z\u00C0-\u00D6\u00D8-\u00F6\u00F8-\u02FF\u0370-\u037D\u037F-\u1FFF\u200C-\u200D\u2070-\u218F\u2C00-\u2FEF\u3001-\uD7FF\uF900-\uFDCF\uFDF0-\uFFFD][:A-Z_a-z\u00C0-\u00D6\u00D8-\u00F6\u00F8-\u02FF\u0370-\u037D\u037F-\u1FFF\u200C-\u200D\u2070-\u218F\u2C00-\u2FEF\u3001-\uD7FF\uF900-\uFDCF\uFDF0-\uFFFD\-.0-9\u00B7\u0300-\u036F\u203F-\u2040]*$/,ce={},z={};function L(e){return N.call(z,e)?!0:N.call(ce,e)?!1:te.test(e)?z[e]=!0:(ce[e]=!0,!1)}function Le(e,t,n,r){if(n!==null&&n.type===0)return!1;switch(typeof t){case"function":case"symbol":return!0;case"boolean":return r?!1:n!==null?!n.acceptsBooleans:(e=e.toLowerCase().slice(0,5),e!=="data-"&&e!=="aria-");default:return!1}}function me(e,t,n,r){if(t===null||typeof t>"u"||Le(e,t,n,r))return!0;if(r)return!1;if(n!==null)switch(n.type){case 3:return!t;case 4:return t===!1;case 5:return isNaN(t);case 6:return isNaN(t)||1>t}return!1}function re(e,t,n,r,l,s,i){this.acceptsBooleans=t===2||t===3||t===4,this.attributeName=r,this.attributeNamespace=l,this.mustUseProperty=n,this.propertyName=e,this.type=t,this.sanitizeURL=s,this.removeEmptyString=i}var G={};"children dangerouslySetInnerHTML defaultValue defaultChecked innerHTML suppressContentEditableWarning suppressHydrationWarning style".split(" ").forEach(function(e){G[e]=new re(e,0,!1,e,null,!1,!1)}),[["acceptCharset","accept-charset"],["className","class"],["htmlFor","for"],["httpEquiv","http-equiv"]].forEach(function(e){var t=e[0];G[t]=new re(t,1,!1,e[1],null,!1,!1)}),["contentEditable","draggable","spellCheck","value"].forEach(function(e){G[e]=new re(e,2,!1,e.toLowerCase(),null,!1,!1)}),["autoReverse","externalResourcesRequired","focusable","preserveAlpha"].forEach(function(e){G[e]=new re(e,2,!1,e,null,!1,!1)}),"allowFullScreen async autoFocus autoPlay controls default defer disabled disablePictureInPicture disableRemotePlayback formNoValidate hidden loop noModule noValidate open playsInline readOnly required reversed scoped seamless itemScope".split(" ").forEach(function(e){G[e]=new re(e,3,!1,e.toLowerCase(),null,!1,!1)}),["checked","multiple","muted","selected"].forEach(function(e){G[e]=new re(e,3,!0,e,null,!1,!1)}),["capture","download"].forEach(function(e){G[e]=new re(e,4,!1,e,null,!1,!1)}),["cols","rows","size","span"].forEach(function(e){G[e]=new re(e,6,!1,e,null,!1,!1)}),["rowSpan","start"].forEach(function(e){G[e]=new re(e,5,!1,e.toLowerCase(),null,!1,!1)});var De=/[\-:]([a-z])/g;function ye(e){return e[1].toUpperCase()}"accent-height alignment-baseline arabic-form baseline-shift cap-height clip-path clip-rule color-interpolation color-interpolation-filters color-profile color-rendering dominant-baseline enable-background fill-opacity fill-rule flood-color flood-opacity font-family font-size font-size-adjust font-stretch font-style font-variant font-weight glyph-name glyph-orientation-horizontal glyph-orientation-vertical horiz-adv-x horiz-origin-x image-rendering letter-spacing lighting-color marker-end marker-mid marker-start overline-position overline-thickness paint-order panose-1 pointer-events rendering-intent shape-rendering stop-color stop-opacity strikethrough-position strikethrough-thickness stroke-dasharray stroke-dashoffset stroke-linecap stroke-linejoin stroke-miterlimit stroke-opacity stroke-width text-anchor text-decoration text-rendering underline-position underline-thickness unicode-bidi unicode-range units-per-em v-alphabetic v-hanging v-ideographic v-mathematical vector-effect vert-adv-y vert-origin-x vert-origin-y word-spacing writing-mode xmlns:xlink x-height".split(" ").forEach(function(e){var t=e.replace(De,ye);G[t]=new re(t,1,!1,e,null,!1,!1)}),"xlink:actuate xlink:arcrole xlink:role xlink:show xlink:title xlink:type".split(" ").forEach(function(e){var t=e.replace(De,ye);G[t]=new re(t,1,!1,e,"http://www.w3.org/1999/xlink",!1,!1)}),["xml:base","xml:lang","xml:space"].forEach(function(e){var t=e.replace(De,ye);G[t]=new re(t,1,!1,e,"http://www.w3.org/XML/1998/namespace",!1,!1)}),["tabIndex","crossOrigin"].forEach(function(e){G[e]=new re(e,1,!1,e.toLowerCase(),null,!1,!1)}),G.xlinkHref=new re("xlinkHref",1,!1,"xlink:href","http://www.w3.org/1999/xlink",!0,!1),["src","href","action","formAction"].forEach(function(e){G[e]=new re(e,1,!1,e.toLowerCase(),null,!0,!0)});function We(e,t,n,r){var l=G.hasOwnProperty(t)?G[t]:null;(l!==null?l.type!==0:r||!(2u||l[i]!==s[u]){var d=` +`+l[i].replace(" at new "," at ");return e.displayName&&d.includes("")&&(d=d.replace("",e.displayName)),d}while(1<=i&&0<=u);break}}}finally{Y=!1,Error.prepareStackTrace=n}return(e=e?e.displayName||e.name:"")?_(e):""}function le(e){switch(e.tag){case 5:return _(e.type);case 16:return _("Lazy");case 13:return _("Suspense");case 19:return _("SuspenseList");case 0:case 2:case 15:return e=J(e.type,!1),e;case 11:return e=J(e.type.render,!1),e;case 1:return e=J(e.type,!0),e;default:return""}}function ee(e){if(e==null)return null;if(typeof e=="function")return e.displayName||e.name||null;if(typeof e=="string")return e;switch(e){case $e:return"Fragment";case X:return"Portal";case ut:return"Profiler";case Ne:return"StrictMode";case He:return"Suspense";case Oe:return"SuspenseList"}if(typeof e=="object")switch(e.$$typeof){case gt:return(e.displayName||"Context")+".Consumer";case be:return(e._context.displayName||"Context")+".Provider";case he:var t=e.render;return e=e.displayName,e||(e=t.displayName||t.name||"",e=e!==""?"ForwardRef("+e+")":"ForwardRef"),e;case Ye:return t=e.displayName||null,t!==null?t:ee(e.type)||"Memo";case b:t=e._payload,e=e._init;try{return ee(e(t))}catch{}}return null}function de(e){var t=e.type;switch(e.tag){case 24:return"Cache";case 9:return(t.displayName||"Context")+".Consumer";case 10:return(t._context.displayName||"Context")+".Provider";case 18:return"DehydratedFragment";case 11:return e=t.render,e=e.displayName||e.name||"",t.displayName||(e!==""?"ForwardRef("+e+")":"ForwardRef");case 7:return"Fragment";case 5:return t;case 4:return"Portal";case 3:return"Root";case 6:return"Text";case 16:return ee(t);case 8:return t===Ne?"StrictMode":"Mode";case 22:return"Offscreen";case 12:return"Profiler";case 21:return"Scope";case 13:return"Suspense";case 19:return"SuspenseList";case 25:return"TracingMarker";case 1:case 0:case 17:case 2:case 14:case 15:if(typeof t=="function")return t.displayName||t.name||null;if(typeof t=="string")return t}return null}function ae(e){switch(typeof e){case"boolean":case"number":case"string":case"undefined":return e;case"object":return e;default:return""}}function oe(e){var t=e.type;return(e=e.nodeName)&&e.toLowerCase()==="input"&&(t==="checkbox"||t==="radio")}function ke(e){var t=oe(e)?"checked":"value",n=Object.getOwnPropertyDescriptor(e.constructor.prototype,t),r=""+e[t];if(!e.hasOwnProperty(t)&&typeof n<"u"&&typeof n.get=="function"&&typeof n.set=="function"){var l=n.get,s=n.set;return Object.defineProperty(e,t,{configurable:!0,get:function(){return l.call(this)},set:function(i){r=""+i,s.call(this,i)}}),Object.defineProperty(e,t,{enumerable:n.enumerable}),{getValue:function(){return r},setValue:function(i){r=""+i},stopTracking:function(){e._valueTracker=null,delete e[t]}}}}function Ot(e){e._valueTracker||(e._valueTracker=ke(e))}function an(e){if(!e)return!1;var t=e._valueTracker;if(!t)return!0;var n=t.getValue(),r="";return e&&(r=oe(e)?e.checked?"true":"false":e.value),e=r,e!==n?(t.setValue(e),!0):!1}function Gt(e){if(e=e||(typeof document<"u"?document:void 0),typeof e>"u")return null;try{return e.activeElement||e.body}catch{return e.body}}function kt(e,t){var n=t.checked;return T({},t,{defaultChecked:void 0,defaultValue:void 0,value:void 0,checked:n??e._wrapperState.initialChecked})}function Dn(e,t){var n=t.defaultValue==null?"":t.defaultValue,r=t.checked!=null?t.checked:t.defaultChecked;n=ae(t.value!=null?t.value:n),e._wrapperState={initialChecked:r,initialValue:n,controlled:t.type==="checkbox"||t.type==="radio"?t.checked!=null:t.value!=null}}function un(e,t){t=t.checked,t!=null&&We(e,"checked",t,!1)}function ct(e,t){un(e,t);var n=ae(t.value),r=t.type;if(n!=null)r==="number"?(n===0&&e.value===""||e.value!=n)&&(e.value=""+n):e.value!==""+n&&(e.value=""+n);else if(r==="submit"||r==="reset"){e.removeAttribute("value");return}t.hasOwnProperty("value")?et(e,t.type,n):t.hasOwnProperty("defaultValue")&&et(e,t.type,ae(t.defaultValue)),t.checked==null&&t.defaultChecked!=null&&(e.defaultChecked=!!t.defaultChecked)}function ie(e,t,n){if(t.hasOwnProperty("value")||t.hasOwnProperty("defaultValue")){var r=t.type;if(!(r!=="submit"&&r!=="reset"||t.value!==void 0&&t.value!==null))return;t=""+e._wrapperState.initialValue,n||t===e.value||(e.value=t),e.defaultValue=t}n=e.name,n!==""&&(e.name=""),e.defaultChecked=!!e._wrapperState.initialChecked,n!==""&&(e.name=n)}function et(e,t,n){(t!=="number"||Gt(e.ownerDocument)!==e)&&(n==null?e.defaultValue=""+e._wrapperState.initialValue:e.defaultValue!==""+n&&(e.defaultValue=""+n))}var Mt=Array.isArray;function yt(e,t,n,r){if(e=e.options,t){t={};for(var l=0;l"+t.valueOf().toString()+"",t=Ct.firstChild;e.firstChild;)e.removeChild(e.firstChild);for(;t.firstChild;)e.appendChild(t.firstChild)}});function Je(e,t){if(t){var n=e.firstChild;if(n&&n===e.lastChild&&n.nodeType===3){n.nodeValue=t;return}}e.textContent=t}var K={animationIterationCount:!0,aspectRatio:!0,borderImageOutset:!0,borderImageSlice:!0,borderImageWidth:!0,boxFlex:!0,boxFlexGroup:!0,boxOrdinalGroup:!0,columnCount:!0,columns:!0,flex:!0,flexGrow:!0,flexPositive:!0,flexShrink:!0,flexNegative:!0,flexOrder:!0,gridArea:!0,gridRow:!0,gridRowEnd:!0,gridRowSpan:!0,gridRowStart:!0,gridColumn:!0,gridColumnEnd:!0,gridColumnSpan:!0,gridColumnStart:!0,fontWeight:!0,lineClamp:!0,lineHeight:!0,opacity:!0,order:!0,orphans:!0,tabSize:!0,widows:!0,zIndex:!0,zoom:!0,fillOpacity:!0,floodOpacity:!0,stopOpacity:!0,strokeDasharray:!0,strokeDashoffset:!0,strokeMiterlimit:!0,strokeOpacity:!0,strokeWidth:!0},fn=["Webkit","ms","Moz","O"];Object.keys(K).forEach(function(e){fn.forEach(function(t){t=t+e.charAt(0).toUpperCase()+e.substring(1),K[t]=K[e]})});function tt(e,t,n){return t==null||typeof t=="boolean"||t===""?"":n||typeof t!="number"||t===0||K.hasOwnProperty(e)&&K[e]?(""+t).trim():t+"px"}function Yt(e,t){e=e.style;for(var n in t)if(t.hasOwnProperty(n)){var r=n.indexOf("--")===0,l=tt(n,t[n],r);n==="float"&&(n="cssFloat"),r?e.setProperty(n,l):e[n]=l}}var pn=T({menuitem:!0},{area:!0,base:!0,br:!0,col:!0,embed:!0,hr:!0,img:!0,input:!0,keygen:!0,link:!0,meta:!0,param:!0,source:!0,track:!0,wbr:!0});function Dt(e,t){if(t){if(pn[e]&&(t.children!=null||t.dangerouslySetInnerHTML!=null))throw Error(p(137,e));if(t.dangerouslySetInnerHTML!=null){if(t.children!=null)throw Error(p(60));if(typeof t.dangerouslySetInnerHTML!="object"||!("__html"in t.dangerouslySetInnerHTML))throw Error(p(61))}if(t.style!=null&&typeof t.style!="object")throw Error(p(62))}}function mn(e,t){if(e.indexOf("-")===-1)return typeof t.is=="string";switch(e){case"annotation-xml":case"color-profile":case"font-face":case"font-face-src":case"font-face-uri":case"font-face-format":case"font-face-name":case"missing-glyph":return!1;default:return!0}}var hn=null;function c(e){return e=e.target||e.srcElement||window,e.correspondingUseElement&&(e=e.correspondingUseElement),e.nodeType===3?e.parentNode:e}var E=null,ne=null,Ee=null;function Jt(e){if(e=Hr(e)){if(typeof E!="function")throw Error(p(280));var t=e.stateNode;t&&(t=Rl(t),E(e.stateNode,e.type,t))}}function Xn(e){ne?Ee?Ee.push(e):Ee=[e]:ne=e}function Zn(){if(ne){var e=ne,t=Ee;if(Ee=ne=null,Jt(e),t)for(e=0;e>>=0,e===0?32:31-(pc(e)/mc|0)|0}var dl=64,fl=4194304;function kr(e){switch(e&-e){case 1:return 1;case 2:return 2;case 4:return 4;case 8:return 8;case 16:return 16;case 32:return 32;case 64:case 128:case 256:case 512:case 1024:case 2048:case 4096:case 8192:case 16384:case 32768:case 65536:case 131072:case 262144:case 524288:case 1048576:case 2097152:return e&4194240;case 4194304:case 8388608:case 16777216:case 33554432:case 67108864:return e&130023424;case 134217728:return 134217728;case 268435456:return 268435456;case 536870912:return 536870912;case 1073741824:return 1073741824;default:return e}}function pl(e,t){var n=e.pendingLanes;if(n===0)return 0;var r=0,l=e.suspendedLanes,s=e.pingedLanes,i=n&268435455;if(i!==0){var u=i&~l;u!==0?r=kr(u):(s&=i,s!==0&&(r=kr(s)))}else i=n&~l,i!==0?r=kr(i):s!==0&&(r=kr(s));if(r===0)return 0;if(t!==0&&t!==r&&(t&l)===0&&(l=r&-r,s=t&-t,l>=s||l===16&&(s&4194240)!==0))return t;if((r&4)!==0&&(r|=n&16),t=e.entangledLanes,t!==0)for(e=e.entanglements,t&=r;0n;n++)t.push(e);return t}function Cr(e,t,n){e.pendingLanes|=t,t!==536870912&&(e.suspendedLanes=0,e.pingedLanes=0),e=e.eventTimes,t=31-It(t),e[t]=n}function yc(e,t){var n=e.pendingLanes&~t;e.pendingLanes=t,e.suspendedLanes=0,e.pingedLanes=0,e.expiredLanes&=t,e.mutableReadLanes&=t,e.entangledLanes&=t,t=e.entanglements;var r=e.eventTimes;for(e=e.expirationTimes;0=Or),Li=" ",zi=!1;function Oi(e,t){switch(e){case"keyup":return Kc.indexOf(t.keyCode)!==-1;case"keydown":return t.keyCode!==229;case"keypress":case"mousedown":case"focusout":return!0;default:return!1}}function Mi(e){return e=e.detail,typeof e=="object"&&"data"in e?e.data:null}var tr=!1;function Gc(e,t){switch(e){case"compositionend":return Mi(t);case"keypress":return t.which!==32?null:(zi=!0,Li);case"textInput":return e=t.data,e===Li&&zi?null:e;default:return null}}function Yc(e,t){if(tr)return e==="compositionend"||!zs&&Oi(e,t)?(e=Ci(),yl=Ns=_n=null,tr=!1,e):null;switch(e){case"paste":return null;case"keypress":if(!(t.ctrlKey||t.altKey||t.metaKey)||t.ctrlKey&&t.altKey){if(t.char&&1=t)return{node:n,offset:t-e};e=r}e:{for(;n;){if(n.nextSibling){n=n.nextSibling;break e}n=n.parentNode}n=void 0}n=Bi(n)}}function Vi(e,t){return e&&t?e===t?!0:e&&e.nodeType===3?!1:t&&t.nodeType===3?Vi(e,t.parentNode):"contains"in e?e.contains(t):e.compareDocumentPosition?!!(e.compareDocumentPosition(t)&16):!1:!1}function Qi(){for(var e=window,t=Gt();t instanceof e.HTMLIFrameElement;){try{var n=typeof t.contentWindow.location.href=="string"}catch{n=!1}if(n)e=t.contentWindow;else break;t=Gt(e.document)}return t}function Ds(e){var t=e&&e.nodeName&&e.nodeName.toLowerCase();return t&&(t==="input"&&(e.type==="text"||e.type==="search"||e.type==="tel"||e.type==="url"||e.type==="password")||t==="textarea"||e.contentEditable==="true")}function ld(e){var t=Qi(),n=e.focusedElem,r=e.selectionRange;if(t!==n&&n&&n.ownerDocument&&Vi(n.ownerDocument.documentElement,n)){if(r!==null&&Ds(n)){if(t=r.start,e=r.end,e===void 0&&(e=t),"selectionStart"in n)n.selectionStart=t,n.selectionEnd=Math.min(e,n.value.length);else if(e=(t=n.ownerDocument||document)&&t.defaultView||window,e.getSelection){e=e.getSelection();var l=n.textContent.length,s=Math.min(r.start,l);r=r.end===void 0?s:Math.min(r.end,l),!e.extend&&s>r&&(l=r,r=s,s=l),l=Hi(n,s);var i=Hi(n,r);l&&i&&(e.rangeCount!==1||e.anchorNode!==l.node||e.anchorOffset!==l.offset||e.focusNode!==i.node||e.focusOffset!==i.offset)&&(t=t.createRange(),t.setStart(l.node,l.offset),e.removeAllRanges(),s>r?(e.addRange(t),e.extend(i.node,i.offset)):(t.setEnd(i.node,i.offset),e.addRange(t)))}}for(t=[],e=n;e=e.parentNode;)e.nodeType===1&&t.push({element:e,left:e.scrollLeft,top:e.scrollTop});for(typeof n.focus=="function"&&n.focus(),n=0;n=document.documentMode,nr=null,Is=null,Ar=null,As=!1;function Wi(e,t,n){var r=n.window===n?n.document:n.nodeType===9?n:n.ownerDocument;As||nr==null||nr!==Gt(r)||(r=nr,"selectionStart"in r&&Ds(r)?r={start:r.selectionStart,end:r.selectionEnd}:(r=(r.ownerDocument&&r.ownerDocument.defaultView||window).getSelection(),r={anchorNode:r.anchorNode,anchorOffset:r.anchorOffset,focusNode:r.focusNode,focusOffset:r.focusOffset}),Ar&&Ir(Ar,r)||(Ar=r,r=Nl(Is,"onSelect"),0ir||(e.current=Ys[ir],Ys[ir]=null,ir--)}function _e(e,t){ir++,Ys[ir]=e.current,e.current=t}var kn={},nt=jn(kn),dt=jn(!1),Un=kn;function ar(e,t){var n=e.type.contextTypes;if(!n)return kn;var r=e.stateNode;if(r&&r.__reactInternalMemoizedUnmaskedChildContext===t)return r.__reactInternalMemoizedMaskedChildContext;var l={},s;for(s in n)l[s]=t[s];return r&&(e=e.stateNode,e.__reactInternalMemoizedUnmaskedChildContext=t,e.__reactInternalMemoizedMaskedChildContext=l),l}function ft(e){return e=e.childContextTypes,e!=null}function Tl(){je(dt),je(nt)}function oa(e,t,n){if(nt.current!==kn)throw Error(p(168));_e(nt,t),_e(dt,n)}function ia(e,t,n){var r=e.stateNode;if(t=t.childContextTypes,typeof r.getChildContext!="function")return n;r=r.getChildContext();for(var l in r)if(!(l in t))throw Error(p(108,de(e)||"Unknown",l));return T({},n,r)}function Ll(e){return e=(e=e.stateNode)&&e.__reactInternalMemoizedMergedChildContext||kn,Un=nt.current,_e(nt,e),_e(dt,dt.current),!0}function aa(e,t,n){var r=e.stateNode;if(!r)throw Error(p(169));n?(e=ia(e,t,Un),r.__reactInternalMemoizedMergedChildContext=e,je(dt),je(nt),_e(nt,e)):je(dt),_e(dt,n)}var Zt=null,zl=!1,Js=!1;function ua(e){Zt===null?Zt=[e]:Zt.push(e)}function vd(e){zl=!0,ua(e)}function Cn(){if(!Js&&Zt!==null){Js=!0;var e=0,t=ve;try{var n=Zt;for(ve=1;e>=i,l-=i,bt=1<<32-It(t)+l|n<q?(Ge=W,W=null):Ge=W.sibling;var fe=S(v,W,g[q],C);if(fe===null){W===null&&(W=Ge);break}e&&W&&fe.alternate===null&&t(v,W),f=s(fe,f,q),Q===null?$=fe:Q.sibling=fe,Q=fe,W=Ge}if(q===g.length)return n(v,W),Ce&&Bn(v,q),$;if(W===null){for(;qq?(Ge=W,W=null):Ge=W.sibling;var Mn=S(v,W,fe.value,C);if(Mn===null){W===null&&(W=Ge);break}e&&W&&Mn.alternate===null&&t(v,W),f=s(Mn,f,q),Q===null?$=Mn:Q.sibling=Mn,Q=Mn,W=Ge}if(fe.done)return n(v,W),Ce&&Bn(v,q),$;if(W===null){for(;!fe.done;q++,fe=g.next())fe=j(v,fe.value,C),fe!==null&&(f=s(fe,f,q),Q===null?$=fe:Q.sibling=fe,Q=fe);return Ce&&Bn(v,q),$}for(W=r(v,W);!fe.done;q++,fe=g.next())fe=M(W,v,q,fe.value,C),fe!==null&&(e&&fe.alternate!==null&&W.delete(fe.key===null?q:fe.key),f=s(fe,f,q),Q===null?$=fe:Q.sibling=fe,Q=fe);return e&&W.forEach(function(Jd){return t(v,Jd)}),Ce&&Bn(v,q),$}function Fe(v,f,g,C){if(typeof g=="object"&&g!==null&&g.type===$e&&g.key===null&&(g=g.props.children),typeof g=="object"&&g!==null){switch(g.$$typeof){case ze:e:{for(var $=g.key,Q=f;Q!==null;){if(Q.key===$){if($=g.type,$===$e){if(Q.tag===7){n(v,Q.sibling),f=l(Q,g.props.children),f.return=v,v=f;break e}}else if(Q.elementType===$||typeof $=="object"&&$!==null&&$.$$typeof===b&&ha($)===Q.type){n(v,Q.sibling),f=l(Q,g.props),f.ref=Vr(v,Q,g),f.return=v,v=f;break e}n(v,Q);break}else t(v,Q);Q=Q.sibling}g.type===$e?(f=Yn(g.props.children,v.mode,C,g.key),f.return=v,v=f):(C=os(g.type,g.key,g.props,null,v.mode,C),C.ref=Vr(v,f,g),C.return=v,v=C)}return i(v);case X:e:{for(Q=g.key;f!==null;){if(f.key===Q)if(f.tag===4&&f.stateNode.containerInfo===g.containerInfo&&f.stateNode.implementation===g.implementation){n(v,f.sibling),f=l(f,g.children||[]),f.return=v,v=f;break e}else{n(v,f);break}else t(v,f);f=f.sibling}f=Go(g,v.mode,C),f.return=v,v=f}return i(v);case b:return Q=g._init,Fe(v,f,Q(g._payload),C)}if(Mt(g))return A(v,f,g,C);if(V(g))return U(v,f,g,C);Il(v,g)}return typeof g=="string"&&g!==""||typeof g=="number"?(g=""+g,f!==null&&f.tag===6?(n(v,f.sibling),f=l(f,g),f.return=v,v=f):(n(v,f),f=qo(g,v.mode,C),f.return=v,v=f),i(v)):n(v,f)}return Fe}var fr=va(!0),ga=va(!1),Al=jn(null),Fl=null,pr=null,no=null;function ro(){no=pr=Fl=null}function lo(e){var t=Al.current;je(Al),e._currentValue=t}function so(e,t,n){for(;e!==null;){var r=e.alternate;if((e.childLanes&t)!==t?(e.childLanes|=t,r!==null&&(r.childLanes|=t)):r!==null&&(r.childLanes&t)!==t&&(r.childLanes|=t),e===n)break;e=e.return}}function mr(e,t){Fl=e,no=pr=null,e=e.dependencies,e!==null&&e.firstContext!==null&&((e.lanes&t)!==0&&(pt=!0),e.firstContext=null)}function Rt(e){var t=e._currentValue;if(no!==e)if(e={context:e,memoizedValue:t,next:null},pr===null){if(Fl===null)throw Error(p(308));pr=e,Fl.dependencies={lanes:0,firstContext:e}}else pr=pr.next=e;return t}var Hn=null;function oo(e){Hn===null?Hn=[e]:Hn.push(e)}function ya(e,t,n,r){var l=t.interleaved;return l===null?(n.next=n,oo(t)):(n.next=l.next,l.next=n),t.interleaved=n,tn(e,r)}function tn(e,t){e.lanes|=t;var n=e.alternate;for(n!==null&&(n.lanes|=t),n=e,e=e.return;e!==null;)e.childLanes|=t,n=e.alternate,n!==null&&(n.childLanes|=t),n=e,e=e.return;return n.tag===3?n.stateNode:null}var Nn=!1;function io(e){e.updateQueue={baseState:e.memoizedState,firstBaseUpdate:null,lastBaseUpdate:null,shared:{pending:null,interleaved:null,lanes:0},effects:null}}function xa(e,t){e=e.updateQueue,t.updateQueue===e&&(t.updateQueue={baseState:e.baseState,firstBaseUpdate:e.firstBaseUpdate,lastBaseUpdate:e.lastBaseUpdate,shared:e.shared,effects:e.effects})}function nn(e,t){return{eventTime:e,lane:t,tag:0,payload:null,callback:null,next:null}}function En(e,t,n){var r=e.updateQueue;if(r===null)return null;if(r=r.shared,(ue&2)!==0){var l=r.pending;return l===null?t.next=t:(t.next=l.next,l.next=t),r.pending=t,tn(e,n)}return l=r.interleaved,l===null?(t.next=t,oo(r)):(t.next=l.next,l.next=t),r.interleaved=t,tn(e,n)}function Ul(e,t,n){if(t=t.updateQueue,t!==null&&(t=t.shared,(n&4194240)!==0)){var r=t.lanes;r&=e.pendingLanes,n|=r,t.lanes=n,Ss(e,n)}}function _a(e,t){var n=e.updateQueue,r=e.alternate;if(r!==null&&(r=r.updateQueue,n===r)){var l=null,s=null;if(n=n.firstBaseUpdate,n!==null){do{var i={eventTime:n.eventTime,lane:n.lane,tag:n.tag,payload:n.payload,callback:n.callback,next:null};s===null?l=s=i:s=s.next=i,n=n.next}while(n!==null);s===null?l=s=t:s=s.next=t}else l=s=t;n={baseState:r.baseState,firstBaseUpdate:l,lastBaseUpdate:s,shared:r.shared,effects:r.effects},e.updateQueue=n;return}e=n.lastBaseUpdate,e===null?n.firstBaseUpdate=t:e.next=t,n.lastBaseUpdate=t}function $l(e,t,n,r){var l=e.updateQueue;Nn=!1;var s=l.firstBaseUpdate,i=l.lastBaseUpdate,u=l.shared.pending;if(u!==null){l.shared.pending=null;var d=u,y=d.next;d.next=null,i===null?s=y:i.next=y,i=d;var w=e.alternate;w!==null&&(w=w.updateQueue,u=w.lastBaseUpdate,u!==i&&(u===null?w.firstBaseUpdate=y:u.next=y,w.lastBaseUpdate=d))}if(s!==null){var j=l.baseState;i=0,w=y=d=null,u=s;do{var S=u.lane,M=u.eventTime;if((r&S)===S){w!==null&&(w=w.next={eventTime:M,lane:0,tag:u.tag,payload:u.payload,callback:u.callback,next:null});e:{var A=e,U=u;switch(S=t,M=n,U.tag){case 1:if(A=U.payload,typeof A=="function"){j=A.call(M,j,S);break e}j=A;break e;case 3:A.flags=A.flags&-65537|128;case 0:if(A=U.payload,S=typeof A=="function"?A.call(M,j,S):A,S==null)break e;j=T({},j,S);break e;case 2:Nn=!0}}u.callback!==null&&u.lane!==0&&(e.flags|=64,S=l.effects,S===null?l.effects=[u]:S.push(u))}else M={eventTime:M,lane:S,tag:u.tag,payload:u.payload,callback:u.callback,next:null},w===null?(y=w=M,d=j):w=w.next=M,i|=S;if(u=u.next,u===null){if(u=l.shared.pending,u===null)break;S=u,u=S.next,S.next=null,l.lastBaseUpdate=S,l.shared.pending=null}}while(!0);if(w===null&&(d=j),l.baseState=d,l.firstBaseUpdate=y,l.lastBaseUpdate=w,t=l.shared.interleaved,t!==null){l=t;do i|=l.lane,l=l.next;while(l!==t)}else s===null&&(l.shared.lanes=0);Wn|=i,e.lanes=i,e.memoizedState=j}}function Sa(e,t,n){if(e=t.effects,t.effects=null,e!==null)for(t=0;tn?n:4,e(!0);var r=po.transition;po.transition={};try{e(!1),t()}finally{ve=n,po.transition=r}}function $a(){return Tt().memoizedState}function _d(e,t,n){var r=Ln(e);if(n={lane:r,action:n,hasEagerState:!1,eagerState:null,next:null},Ba(e))Ha(t,n);else if(n=ya(e,t,n,r),n!==null){var l=it();Ht(n,e,r,l),Va(n,t,r)}}function Sd(e,t,n){var r=Ln(e),l={lane:r,action:n,hasEagerState:!1,eagerState:null,next:null};if(Ba(e))Ha(t,l);else{var s=e.alternate;if(e.lanes===0&&(s===null||s.lanes===0)&&(s=t.lastRenderedReducer,s!==null))try{var i=t.lastRenderedState,u=s(i,n);if(l.hasEagerState=!0,l.eagerState=u,At(u,i)){var d=t.interleaved;d===null?(l.next=l,oo(t)):(l.next=d.next,d.next=l),t.interleaved=l;return}}catch{}n=ya(e,t,l,r),n!==null&&(l=it(),Ht(n,e,r,l),Va(n,t,r))}}function Ba(e){var t=e.alternate;return e===Re||t!==null&&t===Re}function Ha(e,t){qr=Vl=!0;var n=e.pending;n===null?t.next=t:(t.next=n.next,n.next=t),e.pending=t}function Va(e,t,n){if((n&4194240)!==0){var r=t.lanes;r&=e.pendingLanes,n|=r,t.lanes=n,Ss(e,n)}}var Kl={readContext:Rt,useCallback:rt,useContext:rt,useEffect:rt,useImperativeHandle:rt,useInsertionEffect:rt,useLayoutEffect:rt,useMemo:rt,useReducer:rt,useRef:rt,useState:rt,useDebugValue:rt,useDeferredValue:rt,useTransition:rt,useMutableSource:rt,useSyncExternalStore:rt,useId:rt,unstable_isNewReconciler:!1},wd={readContext:Rt,useCallback:function(e,t){return Kt().memoizedState=[e,t===void 0?null:t],e},useContext:Rt,useEffect:za,useImperativeHandle:function(e,t,n){return n=n!=null?n.concat([e]):null,Ql(4194308,4,Da.bind(null,t,e),n)},useLayoutEffect:function(e,t){return Ql(4194308,4,e,t)},useInsertionEffect:function(e,t){return Ql(4,2,e,t)},useMemo:function(e,t){var n=Kt();return t=t===void 0?null:t,e=e(),n.memoizedState=[e,t],e},useReducer:function(e,t,n){var r=Kt();return t=n!==void 0?n(t):t,r.memoizedState=r.baseState=t,e={pending:null,interleaved:null,lanes:0,dispatch:null,lastRenderedReducer:e,lastRenderedState:t},r.queue=e,e=e.dispatch=_d.bind(null,Re,e),[r.memoizedState,e]},useRef:function(e){var t=Kt();return e={current:e},t.memoizedState=e},useState:Ta,useDebugValue:_o,useDeferredValue:function(e){return Kt().memoizedState=e},useTransition:function(){var e=Ta(!1),t=e[0];return e=xd.bind(null,e[1]),Kt().memoizedState=e,[t,e]},useMutableSource:function(){},useSyncExternalStore:function(e,t,n){var r=Re,l=Kt();if(Ce){if(n===void 0)throw Error(p(407));n=n()}else{if(n=t(),qe===null)throw Error(p(349));(Qn&30)!==0||Ca(r,t,n)}l.memoizedState=n;var s={value:n,getSnapshot:t};return l.queue=s,za(Ea.bind(null,r,s,e),[e]),r.flags|=2048,Jr(9,Na.bind(null,r,s,n,t),void 0,null),n},useId:function(){var e=Kt(),t=qe.identifierPrefix;if(Ce){var n=en,r=bt;n=(r&~(1<<32-It(r)-1)).toString(32)+n,t=":"+t+"R"+n,n=Gr++,0<\/script>",e=e.removeChild(e.firstChild)):typeof r.is=="string"?e=i.createElement(n,{is:r.is}):(e=i.createElement(n),n==="select"&&(i=e,r.multiple?i.multiple=!0:r.size&&(i.size=r.size))):e=i.createElementNS(e,n),e[Qt]=t,e[Br]=r,uu(e,t,!1,!1),t.stateNode=e;e:{switch(i=mn(n,r),n){case"dialog":we("cancel",e),we("close",e),l=r;break;case"iframe":case"object":case"embed":we("load",e),l=r;break;case"video":case"audio":for(l=0;lxr&&(t.flags|=128,r=!0,Xr(s,!1),t.lanes=4194304)}else{if(!r)if(e=Bl(i),e!==null){if(t.flags|=128,r=!0,n=e.updateQueue,n!==null&&(t.updateQueue=n,t.flags|=4),Xr(s,!0),s.tail===null&&s.tailMode==="hidden"&&!i.alternate&&!Ce)return lt(t),null}else 2*Ae()-s.renderingStartTime>xr&&n!==1073741824&&(t.flags|=128,r=!0,Xr(s,!1),t.lanes=4194304);s.isBackwards?(i.sibling=t.child,t.child=i):(n=s.last,n!==null?n.sibling=i:t.child=i,s.last=i)}return s.tail!==null?(t=s.tail,s.rendering=t,s.tail=t.sibling,s.renderingStartTime=Ae(),t.sibling=null,n=Pe.current,_e(Pe,r?n&1|2:n&1),t):(lt(t),null);case 22:case 23:return Qo(),r=t.memoizedState!==null,e!==null&&e.memoizedState!==null!==r&&(t.flags|=8192),r&&(t.mode&1)!==0?(jt&1073741824)!==0&&(lt(t),t.subtreeFlags&6&&(t.flags|=8192)):lt(t),null;case 24:return null;case 25:return null}throw Error(p(156,t.tag))}function Td(e,t){switch(Zs(t),t.tag){case 1:return ft(t.type)&&Tl(),e=t.flags,e&65536?(t.flags=e&-65537|128,t):null;case 3:return hr(),je(dt),je(nt),fo(),e=t.flags,(e&65536)!==0&&(e&128)===0?(t.flags=e&-65537|128,t):null;case 5:return uo(t),null;case 13:if(je(Pe),e=t.memoizedState,e!==null&&e.dehydrated!==null){if(t.alternate===null)throw Error(p(340));dr()}return e=t.flags,e&65536?(t.flags=e&-65537|128,t):null;case 19:return je(Pe),null;case 4:return hr(),null;case 10:return lo(t.type._context),null;case 22:case 23:return Qo(),null;case 24:return null;default:return null}}var Jl=!1,st=!1,Ld=typeof WeakSet=="function"?WeakSet:Set,I=null;function gr(e,t){var n=e.ref;if(n!==null)if(typeof n=="function")try{n(null)}catch(r){Me(e,t,r)}else n.current=null}function zo(e,t,n){try{n()}catch(r){Me(e,t,r)}}var fu=!1;function zd(e,t){if(Vs=vl,e=Qi(),Ds(e)){if("selectionStart"in e)var n={start:e.selectionStart,end:e.selectionEnd};else e:{n=(n=e.ownerDocument)&&n.defaultView||window;var r=n.getSelection&&n.getSelection();if(r&&r.rangeCount!==0){n=r.anchorNode;var l=r.anchorOffset,s=r.focusNode;r=r.focusOffset;try{n.nodeType,s.nodeType}catch{n=null;break e}var i=0,u=-1,d=-1,y=0,w=0,j=e,S=null;t:for(;;){for(var M;j!==n||l!==0&&j.nodeType!==3||(u=i+l),j!==s||r!==0&&j.nodeType!==3||(d=i+r),j.nodeType===3&&(i+=j.nodeValue.length),(M=j.firstChild)!==null;)S=j,j=M;for(;;){if(j===e)break t;if(S===n&&++y===l&&(u=i),S===s&&++w===r&&(d=i),(M=j.nextSibling)!==null)break;j=S,S=j.parentNode}j=M}n=u===-1||d===-1?null:{start:u,end:d}}else n=null}n=n||{start:0,end:0}}else n=null;for(Qs={focusedElem:e,selectionRange:n},vl=!1,I=t;I!==null;)if(t=I,e=t.child,(t.subtreeFlags&1028)!==0&&e!==null)e.return=t,I=e;else for(;I!==null;){t=I;try{var A=t.alternate;if((t.flags&1024)!==0)switch(t.tag){case 0:case 11:case 15:break;case 1:if(A!==null){var U=A.memoizedProps,Fe=A.memoizedState,v=t.stateNode,f=v.getSnapshotBeforeUpdate(t.elementType===t.type?U:Ut(t.type,U),Fe);v.__reactInternalSnapshotBeforeUpdate=f}break;case 3:var g=t.stateNode.containerInfo;g.nodeType===1?g.textContent="":g.nodeType===9&&g.documentElement&&g.removeChild(g.documentElement);break;case 5:case 6:case 4:case 17:break;default:throw Error(p(163))}}catch(C){Me(t,t.return,C)}if(e=t.sibling,e!==null){e.return=t.return,I=e;break}I=t.return}return A=fu,fu=!1,A}function Zr(e,t,n){var r=t.updateQueue;if(r=r!==null?r.lastEffect:null,r!==null){var l=r=r.next;do{if((l.tag&e)===e){var s=l.destroy;l.destroy=void 0,s!==void 0&&zo(t,n,s)}l=l.next}while(l!==r)}}function Xl(e,t){if(t=t.updateQueue,t=t!==null?t.lastEffect:null,t!==null){var n=t=t.next;do{if((n.tag&e)===e){var r=n.create;n.destroy=r()}n=n.next}while(n!==t)}}function Oo(e){var t=e.ref;if(t!==null){var n=e.stateNode;e.tag,e=n,typeof t=="function"?t(e):t.current=e}}function pu(e){var t=e.alternate;t!==null&&(e.alternate=null,pu(t)),e.child=null,e.deletions=null,e.sibling=null,e.tag===5&&(t=e.stateNode,t!==null&&(delete t[Qt],delete t[Br],delete t[Gs],delete t[md],delete t[hd])),e.stateNode=null,e.return=null,e.dependencies=null,e.memoizedProps=null,e.memoizedState=null,e.pendingProps=null,e.stateNode=null,e.updateQueue=null}function mu(e){return e.tag===5||e.tag===3||e.tag===4}function hu(e){e:for(;;){for(;e.sibling===null;){if(e.return===null||mu(e.return))return null;e=e.return}for(e.sibling.return=e.return,e=e.sibling;e.tag!==5&&e.tag!==6&&e.tag!==18;){if(e.flags&2||e.child===null||e.tag===4)continue e;e.child.return=e,e=e.child}if(!(e.flags&2))return e.stateNode}}function Mo(e,t,n){var r=e.tag;if(r===5||r===6)e=e.stateNode,t?n.nodeType===8?n.parentNode.insertBefore(e,t):n.insertBefore(e,t):(n.nodeType===8?(t=n.parentNode,t.insertBefore(e,n)):(t=n,t.appendChild(e)),n=n._reactRootContainer,n!=null||t.onclick!==null||(t.onclick=Pl));else if(r!==4&&(e=e.child,e!==null))for(Mo(e,t,n),e=e.sibling;e!==null;)Mo(e,t,n),e=e.sibling}function Do(e,t,n){var r=e.tag;if(r===5||r===6)e=e.stateNode,t?n.insertBefore(e,t):n.appendChild(e);else if(r!==4&&(e=e.child,e!==null))for(Do(e,t,n),e=e.sibling;e!==null;)Do(e,t,n),e=e.sibling}var Xe=null,$t=!1;function Pn(e,t,n){for(n=n.child;n!==null;)vu(e,t,n),n=n.sibling}function vu(e,t,n){if(Vt&&typeof Vt.onCommitFiberUnmount=="function")try{Vt.onCommitFiberUnmount(cl,n)}catch{}switch(n.tag){case 5:st||gr(n,t);case 6:var r=Xe,l=$t;Xe=null,Pn(e,t,n),Xe=r,$t=l,Xe!==null&&($t?(e=Xe,n=n.stateNode,e.nodeType===8?e.parentNode.removeChild(n):e.removeChild(n)):Xe.removeChild(n.stateNode));break;case 18:Xe!==null&&($t?(e=Xe,n=n.stateNode,e.nodeType===8?qs(e.parentNode,n):e.nodeType===1&&qs(e,n),Tr(e)):qs(Xe,n.stateNode));break;case 4:r=Xe,l=$t,Xe=n.stateNode.containerInfo,$t=!0,Pn(e,t,n),Xe=r,$t=l;break;case 0:case 11:case 14:case 15:if(!st&&(r=n.updateQueue,r!==null&&(r=r.lastEffect,r!==null))){l=r=r.next;do{var s=l,i=s.destroy;s=s.tag,i!==void 0&&((s&2)!==0||(s&4)!==0)&&zo(n,t,i),l=l.next}while(l!==r)}Pn(e,t,n);break;case 1:if(!st&&(gr(n,t),r=n.stateNode,typeof r.componentWillUnmount=="function"))try{r.props=n.memoizedProps,r.state=n.memoizedState,r.componentWillUnmount()}catch(u){Me(n,t,u)}Pn(e,t,n);break;case 21:Pn(e,t,n);break;case 22:n.mode&1?(st=(r=st)||n.memoizedState!==null,Pn(e,t,n),st=r):Pn(e,t,n);break;default:Pn(e,t,n)}}function gu(e){var t=e.updateQueue;if(t!==null){e.updateQueue=null;var n=e.stateNode;n===null&&(n=e.stateNode=new Ld),t.forEach(function(r){var l=Bd.bind(null,e,r);n.has(r)||(n.add(r),r.then(l,l))})}}function Bt(e,t){var n=t.deletions;if(n!==null)for(var r=0;rl&&(l=i),r&=~s}if(r=l,r=Ae()-r,r=(120>r?120:480>r?480:1080>r?1080:1920>r?1920:3e3>r?3e3:4320>r?4320:1960*Md(r/1960))-r,10e?16:e,Tn===null)var r=!1;else{if(e=Tn,Tn=null,ns=0,(ue&6)!==0)throw Error(p(331));var l=ue;for(ue|=4,I=e.current;I!==null;){var s=I,i=s.child;if((I.flags&16)!==0){var u=s.deletions;if(u!==null){for(var d=0;dAe()-Fo?qn(e,0):Ao|=n),ht(e,t)}function Tu(e,t){t===0&&((e.mode&1)===0?t=1:(t=fl,fl<<=1,(fl&130023424)===0&&(fl=4194304)));var n=it();e=tn(e,t),e!==null&&(Cr(e,t,n),ht(e,n))}function $d(e){var t=e.memoizedState,n=0;t!==null&&(n=t.retryLane),Tu(e,n)}function Bd(e,t){var n=0;switch(e.tag){case 13:var r=e.stateNode,l=e.memoizedState;l!==null&&(n=l.retryLane);break;case 19:r=e.stateNode;break;default:throw Error(p(314))}r!==null&&r.delete(t),Tu(e,n)}var Lu;Lu=function(e,t,n){if(e!==null)if(e.memoizedProps!==t.pendingProps||dt.current)pt=!0;else{if((e.lanes&n)===0&&(t.flags&128)===0)return pt=!1,Pd(e,t,n);pt=(e.flags&131072)!==0}else pt=!1,Ce&&(t.flags&1048576)!==0&&ca(t,Ml,t.index);switch(t.lanes=0,t.tag){case 2:var r=t.type;Yl(e,t),e=t.pendingProps;var l=ar(t,nt.current);mr(t,n),l=ho(null,t,r,e,l,n);var s=vo();return t.flags|=1,typeof l=="object"&&l!==null&&typeof l.render=="function"&&l.$$typeof===void 0?(t.tag=1,t.memoizedState=null,t.updateQueue=null,ft(r)?(s=!0,Ll(t)):s=!1,t.memoizedState=l.state!==null&&l.state!==void 0?l.state:null,io(t),l.updater=ql,t.stateNode=l,l._reactInternals=t,wo(t,r,e,n),t=No(null,t,r,!0,s,n)):(t.tag=0,Ce&&s&&Xs(t),ot(null,t,l,n),t=t.child),t;case 16:r=t.elementType;e:{switch(Yl(e,t),e=t.pendingProps,l=r._init,r=l(r._payload),t.type=r,l=t.tag=Vd(r),e=Ut(r,e),l){case 0:t=Co(null,t,r,e,n);break e;case 1:t=ru(null,t,r,e,n);break e;case 11:t=Za(null,t,r,e,n);break e;case 14:t=ba(null,t,r,Ut(r.type,e),n);break e}throw Error(p(306,r,""))}return t;case 0:return r=t.type,l=t.pendingProps,l=t.elementType===r?l:Ut(r,l),Co(e,t,r,l,n);case 1:return r=t.type,l=t.pendingProps,l=t.elementType===r?l:Ut(r,l),ru(e,t,r,l,n);case 3:e:{if(lu(t),e===null)throw Error(p(387));r=t.pendingProps,s=t.memoizedState,l=s.element,xa(e,t),$l(t,r,null,n);var i=t.memoizedState;if(r=i.element,s.isDehydrated)if(s={element:r,isDehydrated:!1,cache:i.cache,pendingSuspenseBoundaries:i.pendingSuspenseBoundaries,transitions:i.transitions},t.updateQueue.baseState=s,t.memoizedState=s,t.flags&256){l=vr(Error(p(423)),t),t=su(e,t,r,n,l);break e}else if(r!==l){l=vr(Error(p(424)),t),t=su(e,t,r,n,l);break e}else for(wt=wn(t.stateNode.containerInfo.firstChild),St=t,Ce=!0,Ft=null,n=ga(t,null,r,n),t.child=n;n;)n.flags=n.flags&-3|4096,n=n.sibling;else{if(dr(),r===l){t=rn(e,t,n);break e}ot(e,t,r,n)}t=t.child}return t;case 5:return wa(t),e===null&&eo(t),r=t.type,l=t.pendingProps,s=e!==null?e.memoizedProps:null,i=l.children,Ws(r,l)?i=null:s!==null&&Ws(r,s)&&(t.flags|=32),nu(e,t),ot(e,t,i,n),t.child;case 6:return e===null&&eo(t),null;case 13:return ou(e,t,n);case 4:return ao(t,t.stateNode.containerInfo),r=t.pendingProps,e===null?t.child=fr(t,null,r,n):ot(e,t,r,n),t.child;case 11:return r=t.type,l=t.pendingProps,l=t.elementType===r?l:Ut(r,l),Za(e,t,r,l,n);case 7:return ot(e,t,t.pendingProps,n),t.child;case 8:return ot(e,t,t.pendingProps.children,n),t.child;case 12:return ot(e,t,t.pendingProps.children,n),t.child;case 10:e:{if(r=t.type._context,l=t.pendingProps,s=t.memoizedProps,i=l.value,_e(Al,r._currentValue),r._currentValue=i,s!==null)if(At(s.value,i)){if(s.children===l.children&&!dt.current){t=rn(e,t,n);break e}}else for(s=t.child,s!==null&&(s.return=t);s!==null;){var u=s.dependencies;if(u!==null){i=s.child;for(var d=u.firstContext;d!==null;){if(d.context===r){if(s.tag===1){d=nn(-1,n&-n),d.tag=2;var y=s.updateQueue;if(y!==null){y=y.shared;var w=y.pending;w===null?d.next=d:(d.next=w.next,w.next=d),y.pending=d}}s.lanes|=n,d=s.alternate,d!==null&&(d.lanes|=n),so(s.return,n,t),u.lanes|=n;break}d=d.next}}else if(s.tag===10)i=s.type===t.type?null:s.child;else if(s.tag===18){if(i=s.return,i===null)throw Error(p(341));i.lanes|=n,u=i.alternate,u!==null&&(u.lanes|=n),so(i,n,t),i=s.sibling}else i=s.child;if(i!==null)i.return=s;else for(i=s;i!==null;){if(i===t){i=null;break}if(s=i.sibling,s!==null){s.return=i.return,i=s;break}i=i.return}s=i}ot(e,t,l.children,n),t=t.child}return t;case 9:return l=t.type,r=t.pendingProps.children,mr(t,n),l=Rt(l),r=r(l),t.flags|=1,ot(e,t,r,n),t.child;case 14:return r=t.type,l=Ut(r,t.pendingProps),l=Ut(r.type,l),ba(e,t,r,l,n);case 15:return eu(e,t,t.type,t.pendingProps,n);case 17:return r=t.type,l=t.pendingProps,l=t.elementType===r?l:Ut(r,l),Yl(e,t),t.tag=1,ft(r)?(e=!0,Ll(t)):e=!1,mr(t,n),Wa(t,r,l),wo(t,r,l,n),No(null,t,r,!0,e,n);case 19:return au(e,t,n);case 22:return tu(e,t,n)}throw Error(p(156,t.tag))};function zu(e,t){return di(e,t)}function Hd(e,t,n,r){this.tag=e,this.key=n,this.sibling=this.child=this.return=this.stateNode=this.type=this.elementType=null,this.index=0,this.ref=null,this.pendingProps=t,this.dependencies=this.memoizedState=this.updateQueue=this.memoizedProps=null,this.mode=r,this.subtreeFlags=this.flags=0,this.deletions=null,this.childLanes=this.lanes=0,this.alternate=null}function zt(e,t,n,r){return new Hd(e,t,n,r)}function Ko(e){return e=e.prototype,!(!e||!e.isReactComponent)}function Vd(e){if(typeof e=="function")return Ko(e)?1:0;if(e!=null){if(e=e.$$typeof,e===he)return 11;if(e===Ye)return 14}return 2}function On(e,t){var n=e.alternate;return n===null?(n=zt(e.tag,t,e.key,e.mode),n.elementType=e.elementType,n.type=e.type,n.stateNode=e.stateNode,n.alternate=e,e.alternate=n):(n.pendingProps=t,n.type=e.type,n.flags=0,n.subtreeFlags=0,n.deletions=null),n.flags=e.flags&14680064,n.childLanes=e.childLanes,n.lanes=e.lanes,n.child=e.child,n.memoizedProps=e.memoizedProps,n.memoizedState=e.memoizedState,n.updateQueue=e.updateQueue,t=e.dependencies,n.dependencies=t===null?null:{lanes:t.lanes,firstContext:t.firstContext},n.sibling=e.sibling,n.index=e.index,n.ref=e.ref,n}function os(e,t,n,r,l,s){var i=2;if(r=e,typeof e=="function")Ko(e)&&(i=1);else if(typeof e=="string")i=5;else e:switch(e){case $e:return Yn(n.children,l,s,t);case Ne:i=8,l|=8;break;case ut:return e=zt(12,n,t,l|2),e.elementType=ut,e.lanes=s,e;case He:return e=zt(13,n,t,l),e.elementType=He,e.lanes=s,e;case Oe:return e=zt(19,n,t,l),e.elementType=Oe,e.lanes=s,e;case pe:return is(n,l,s,t);default:if(typeof e=="object"&&e!==null)switch(e.$$typeof){case be:i=10;break e;case gt:i=9;break e;case he:i=11;break e;case Ye:i=14;break e;case b:i=16,r=null;break e}throw Error(p(130,e==null?e:typeof e,""))}return t=zt(i,n,t,l),t.elementType=e,t.type=r,t.lanes=s,t}function Yn(e,t,n,r){return e=zt(7,e,r,t),e.lanes=n,e}function is(e,t,n,r){return e=zt(22,e,r,t),e.elementType=pe,e.lanes=n,e.stateNode={isHidden:!1},e}function qo(e,t,n){return e=zt(6,e,null,t),e.lanes=n,e}function Go(e,t,n){return t=zt(4,e.children!==null?e.children:[],e.key,t),t.lanes=n,t.stateNode={containerInfo:e.containerInfo,pendingChildren:null,implementation:e.implementation},t}function Qd(e,t,n,r,l){this.tag=t,this.containerInfo=e,this.finishedWork=this.pingCache=this.current=this.pendingChildren=null,this.timeoutHandle=-1,this.callbackNode=this.pendingContext=this.context=null,this.callbackPriority=0,this.eventTimes=_s(0),this.expirationTimes=_s(-1),this.entangledLanes=this.finishedLanes=this.mutableReadLanes=this.expiredLanes=this.pingedLanes=this.suspendedLanes=this.pendingLanes=0,this.entanglements=_s(0),this.identifierPrefix=r,this.onRecoverableError=l,this.mutableSourceEagerHydrationData=null}function Yo(e,t,n,r,l,s,i,u,d){return e=new Qd(e,t,n,u,d),t===1?(t=1,s===!0&&(t|=8)):t=0,s=zt(3,null,null,t),e.current=s,s.stateNode=e,s.memoizedState={element:r,isDehydrated:n,cache:null,transitions:null,pendingSuspenseBoundaries:null},io(s),e}function Wd(e,t,n){var r=3"u"||typeof __REACT_DEVTOOLS_GLOBAL_HOOK__.checkDCE!="function"))try{__REACT_DEVTOOLS_GLOBAL_HOOK__.checkDCE(a)}catch(h){console.error(h)}}return a(),ti.exports=rf(),ti.exports}var Ku;function sf(){if(Ku)return ms;Ku=1;var a=lf();return ms.createRoot=a.createRoot,ms.hydrateRoot=a.hydrateRoot,ms}var of=sf();const af=rc(of),uf="/api";async function Te(a,h){const p=await fetch(`${uf}${a}`,{...h,headers:{"Content-Type":"application/json",...h?.headers??{}}}),D=await p.json();if(!p.ok){const k=D.error?.message??"Ошибка запроса";throw new Error(k)}return D}const Ue={async listModels(a){return Te("/llm/models",{method:"POST",body:JSON.stringify({llmProvider:a.llmProvider,apiKey:a.apiKey,model:a.model,baseUrl:a.baseUrl})})},async testConnection(a){return Te("/llm/test-connection",{method:"POST",body:JSON.stringify({llmProvider:a.llmProvider,apiKey:a.apiKey,model:a.model,baseUrl:a.baseUrl})})},async normalize(a){return Te("/normalize",{method:"POST",body:JSON.stringify({llmProvider:a.connection.llmProvider,apiKey:a.connection.apiKey,model:a.connection.model,baseUrl:a.connection.baseUrl,temperature:a.connection.temperature,maxOutputTokens:a.connection.maxOutputTokens,promptVersion:a.promptVersion,systemPrompt:a.prompts.systemPrompt,developerPrompt:a.prompts.developerPrompt,domainPrompt:a.prompts.domainPrompt,fewShotExamples:a.prompts.fewShotExamples,userQuestion:a.query.userQuestion,context:{period_hint:a.query.periodHint??"",business_context:a.query.businessContext??"",expected_route:a.query.expectedRoute??""},saveAsTestCase:!!a.saveAsTestCase,useMock:!!a.useMock})})},async loadHistory(){return Te("/history")},async loadTrace(a){return Te(`/history/${a}`)},async loadPresets(){return Te("/presets")},async savePreset(a){return Te("/presets/save",{method:"POST",body:JSON.stringify(a)})},async runEval(a){return Te("/eval/run",{method:"POST",body:JSON.stringify({normalizeConfig:{llmProvider:a.connection.llmProvider,apiKey:a.connection.apiKey,model:a.connection.model,baseUrl:a.connection.baseUrl,temperature:a.connection.temperature,maxOutputTokens:a.connection.maxOutputTokens,promptVersion:a.promptVersion,systemPrompt:a.prompts.systemPrompt,developerPrompt:a.prompts.developerPrompt,domainPrompt:a.prompts.domainPrompt,fewShotExamples:a.prompts.fewShotExamples},caseIds:a.caseIds,useMock:!!a.useMock,mode:a.mode??"standard",caseSetFile:a.caseSetFile,rawQuestions:a.rawQuestions})})},async startRun(){return Te("/accounting-agent/v1/runs/start",{method:"POST",body:JSON.stringify({initiator:"ndc_operator",source:"gui"})})},async finishRun(a){return Te("/accounting-agent/v1/runs/finish",{method:"POST",body:JSON.stringify({runId:a,status:"DONE",source:"gui",reason:"Остановлено оператором из GUI"})})},async listRuns(){return Te("/accounting-agent/v1/runs")},async listResults(){return Te("/accounting-agent/v1/results")},async runTrace(a){return Te(`/accounting-agent/v1/trace/run/${a}`)},async sendAssistantMessage(a){return Te("/assistant/message",{method:"POST",body:JSON.stringify({session_id:a.sessionId??"",mode:"assistant",message:a.userMessage,user_message:a.userMessage,llmProvider:a.connection.llmProvider,apiKey:a.connection.apiKey,model:a.connection.model,baseUrl:a.connection.baseUrl,temperature:a.connection.temperature,maxOutputTokens:a.connection.maxOutputTokens,promptVersion:a.promptVersion??"address_query_runtime_v1",systemPrompt:a.prompts.systemPrompt,developerPrompt:a.prompts.developerPrompt,domainPrompt:a.prompts.domainPrompt,fewShotExamples:a.prompts.fewShotExamples,context:{period_hint:a.context?.periodHint??"",business_context:a.context?.businessContext??""},useMock:!!a.useMock})})},async loadAssistantSession(a){return Te(`/assistant/session/${a}`)},async loadAutoRunsHistory(a){const h=new URLSearchParams;a?.from&&h.set("from",a.from),a?.to&&h.set("to",a.to),a?.target&&h.set("target",a.target),a?.mode&&h.set("mode",a.mode),a?.use_mock&&h.set("use_mock",a.use_mock),a?.prompt_contains&&h.set("prompt_contains",a.prompt_contains),typeof a?.limit=="number"&&h.set("limit",String(a.limit)),typeof a?.scan_limit=="number"&&h.set("scan_limit",String(a.scan_limit));const p=h.toString();return Te(`/autoruns/history${p?`?${p}`:""}`)},async loadAutoRunDetail(a){return Te(`/autoruns/history/${encodeURIComponent(a)}`)},async loadAutoRunCaseDialog(a,h){return Te(`/autoruns/history/${encodeURIComponent(a)}/case/${encodeURIComponent(h)}/dialog`)},async loadAutoRunAnnotations(a){const h=new URLSearchParams;a?.run_id&&h.set("run_id",a.run_id),a?.case_id&&h.set("case_id",a.case_id),typeof a?.min_rating=="number"&&h.set("min_rating",String(a.min_rating)),a?.manual_case_decision&&h.set("manual_case_decision",a.manual_case_decision),typeof a?.limit=="number"&&h.set("limit",String(a.limit));const p=h.toString();return Te(`/autoruns/annotations${p?`?${p}`:""}`)},async saveAutoRunAnnotation(a){return Te("/autoruns/annotations",{method:"POST",body:JSON.stringify(a)})},async loadAutoRunPostAnalysis(a){const h=new URLSearchParams;a?.run_id&&h.set("run_id",a.run_id),typeof a?.limit_per_queue=="number"&&h.set("limit_per_queue",String(a.limit_per_queue)),typeof a?.annotation_limit=="number"&&h.set("annotation_limit",String(a.annotation_limit)),typeof a?.scan_limit=="number"&&h.set("scan_limit",String(a.scan_limit)),a?.from&&h.set("from",a.from),a?.to&&h.set("to",a.to),a?.target&&h.set("target",a.target),a?.mode&&h.set("mode",a.mode),a?.use_mock&&h.set("use_mock",a.use_mock),a?.prompt_contains&&h.set("prompt_contains",a.prompt_contains);const p=h.toString();return Te(`/autoruns/post-analysis${p?`?${p}`:""}`)},async loadAutoRunAutogenHistory(a){const h=new URLSearchParams;a?.mode&&h.set("mode",a.mode),typeof a?.limit=="number"&&h.set("limit",String(a.limit));const p=h.toString();return Te(`/autoruns/autogen/history${p?`?${p}`:""}`)},async generateAutoRunQuestions(a){return Te("/autoruns/autogen/generate",{method:"POST",body:JSON.stringify(a)})}};function at({value:a}){return o.jsx("pre",{className:"json-view",children:JSON.stringify(a??{},null,2)})}function on({title:a,subtitle:h,actions:p,className:D,hideHeader:k,children:F}){return o.jsxs("section",{className:D?`panel-frame ${D}`:"panel-frame",children:[k?null:o.jsxs("header",{className:"panel-header",children:[o.jsxs("div",{children:[o.jsx("h2",{children:a}),h?o.jsx("p",{children:h}):null]}),p?o.jsx("div",{className:"panel-actions",children:p}):null]}),o.jsx("div",{className:"panel-body",children:F})]})}const qu={fromLocal:"",toLocal:"",target:"all",mode:"all",useMock:"any",promptContains:"",limit:120},hs="needs_dialog_policy_fix",cf={mode:"codex_creative",count:24,domain:"",persistToEvalCases:!0,generatedBy:"manual_reviewer"};function df(a){const h=a.getFullYear(),p=String(a.getMonth()+1).padStart(2,"0"),D=String(a.getDate()).padStart(2,"0"),k=String(a.getHours()).padStart(2,"0"),F=String(a.getMinutes()).padStart(2,"0");return`${h}-${p}-${D}T${k}:${F}`}function Gu(){const a=new Date;return a.setDate(a.getDate()-14),df(a)}function vs(a){if(!a.trim())return;const h=Date.parse(a);if(Number.isFinite(h))return new Date(h).toISOString()}function Sr(a){if(!a)return"нет данных";const h=Date.parse(a);return Number.isFinite(h)?new Date(h).toLocaleString("ru-RU"):a}function ff(a,h){return h<=0?0:Math.max(0,Math.min(100,Number((a/h*100).toFixed(1))))}function ll(a){return typeof a!="number"?"нет данных":`${a.toFixed(1)}%`}function pf(a){return a==="assistant_stage1"?"assistant/s1":a==="assistant_stage2"?"assistant/s2":a==="assistant_p0"?"assistant/p0":a}function Yu(a){return a==="up"?"Рост":a==="down"?"Регресс":"Без изменений"}function mf(a,h){return a.find(p=>p.case_id===h)??null}function Ju(a){const h=Math.max(1,Math.min(5,Math.round(a)));return`${"●".repeat(h)}${"○".repeat(5-h)}`}function Xu(a){return a.length===0?o.jsx("p",{className:"muted",children:"Покрытие доменов пока не сформировано."}):o.jsx("div",{className:"autoruns-coverage-list",children:a.map(h=>{const p=ff(h.closed_cases,h.total_cases);return o.jsxs("div",{className:"autoruns-coverage-item",children:[o.jsxs("div",{className:"autoruns-coverage-head",children:[o.jsx("strong",{children:h.domain}),o.jsxs("span",{children:[h.closed_cases,"/",h.total_cases," (",p,"%)"]})]}),o.jsx("div",{className:"autoruns-coverage-bar",children:o.jsx("div",{style:{width:`${p}%`}})})]},h.domain)})})}function hf({connection:a,prompts:h,assistantPromptVersion:p,decompositionPromptVersion:D,showAssistantMode:k,showDecompositionMode:F,showProgressMode:H,onLog:Z}){const[N,te]=P.useState({...qu,fromLocal:Gu()}),[ce,z]=P.useState("settings"),[L,Le]=P.useState(null),[me,re]=P.useState(null),[G,De]=P.useState(null),[ye,We]=P.useState([]),[xe,ze]=P.useState("all"),[X,$e]=P.useState(null),[Ne,ut]=P.useState([]),[be,gt]=P.useState(""),[he,He]=P.useState(""),[Oe,Ye]=P.useState(""),[b,pe]=P.useState(cf),[R,V]=P.useState([]),[T,m]=P.useState(null),[_,Y]=P.useState(!1),[J,le]=P.useState(!1),[ee,de]=P.useState(!1),[ae,oe]=P.useState(!1),[ke,Ot]=P.useState(!1),[an,Gt]=P.useState(!1),[kt,Dn]=P.useState(!1),[un,ct]=P.useState(""),[ie,et]=P.useState({open:!1,messageIndex:-1,rating:3,comment:"",manualCaseDecision:hs,annotationAuthor:"manual_reviewer",saving:!1,error:""}),Mt=P.useRef(!1),yt=L?.items.find(c=>c.run_id===he)??me?.run??null,xt=me?mf(me.cases,Oe):null,ge=ye.find(c=>c.annotation_id===be)??null,cn=G?.messages.find(c=>c.message_index===ie.messageIndex)??null,dn=P.useMemo(()=>{if(ye.length===0)return null;const c=ye.reduce((E,ne)=>E+ne.rating,0)/ye.length;return Number(c.toFixed(2))},[ye]),Jn=P.useMemo(()=>{const c=[...L?.items??[]];return he&&!c.some(E=>E.run_id===he)&&me?.run&&c.unshift(me.run),c},[L?.items,me?.run,he]),Se=P.useCallback(c=>{Z?.(`[autoruns] ${c}`)},[Z]),Ct=P.useCallback(async()=>{Dn(!0);try{const c=await Ue.loadAutoRunAnnotations({limit:800,manual_case_decision:xe});We(c.items),$e(c.manual_case_decision_schema??null),ut(c.available_manual_case_decisions??[]),gt(E=>c.items.length===0?"":c.items.some(ne=>ne.annotation_id===E)?E:c.items[0].annotation_id)}catch(c){Se(`Annotations load error: ${c instanceof Error?c.message:String(c)}`)}finally{Dn(!1)}},[xe,Se]),Nt=P.useCallback(async()=>{de(!0);try{const c=await Ue.loadAutoRunAutogenHistory({limit:180});V(c.items)}catch(c){Se(`Autogen history load error: ${c instanceof Error?c.message:String(c)}`)}finally{de(!1)}},[Se]),Je=P.useCallback(async()=>{le(!0);try{const c=await Ue.loadAutoRunPostAnalysis({run_id:he||void 0,limit_per_queue:30,annotation_limit:1500,from:vs(N.fromLocal),to:vs(N.toLocal),target:N.target,mode:N.mode,use_mock:N.useMock,prompt_contains:N.promptContains.trim()||void 0});m(c)}catch(c){Se(`Post-analysis load error: ${c instanceof Error?c.message:String(c)}`),m(null)}finally{le(!1)}},[N.fromLocal,N.mode,N.promptContains,N.target,N.toLocal,N.useMock,Se,he]),K=P.useCallback(async()=>{Y(!0),ct("");try{const c=[h.systemPrompt,h.developerPrompt,h.domainPrompt,h.schemaNotes,h.fewShotExamples].join(` +`).slice(0,900),E=await Ue.generateAutoRunQuestions({mode:b.mode,count:b.count,domain:b.domain.trim()||void 0,persist_to_eval_cases:b.persistToEvalCases,generated_by:b.generatedBy.trim()||void 0,context:{llm_provider:a.llmProvider,model:a.model,assistant_prompt_version:p,decomposition_prompt_version:D,prompt_fingerprint:c}});Se(`Generated ${E.generation.count} questions (${E.generation.mode}) id=${E.generation.generation_id}`+(E.generation.saved_case_set_file?` saved=${E.generation.saved_case_set_file}`:"")),await Nt()}catch(c){const E=c instanceof Error?c.message:String(c);ct(`Автогенерация: ${E}`),Se(`Autogen generate error: ${E}`)}finally{Y(!1)}},[p,b.count,b.domain,b.generatedBy,b.mode,b.persistToEvalCases,a.llmProvider,a.model,D,Nt,Se,h.developerPrompt,h.domainPrompt,h.fewShotExamples,h.schemaNotes,h.systemPrompt]),fn=P.useCallback(async(c,E)=>{Gt(!0);try{const ne=await Ue.loadAutoRunCaseDialog(c,E);De(ne)}catch(ne){const Ee=ne instanceof Error?ne.message:String(ne);ct(`Диалог кейса: ${Ee}`),De(null),Se(`Dialog load error for ${c}/${E}: ${Ee}`)}finally{Gt(!1)}},[Se]),tt=P.useCallback(async(c,E)=>{Ot(!0);try{const ne=await Ue.loadAutoRunDetail(c);re(ne);const Ee=(E&&ne.cases.some(Jt=>Jt.case_id===E)?E:"")||ne.cases[0]?.case_id||"";He(c),Ye(Ee),Ee?await fn(c,Ee):De(null)}catch(ne){const Ee=ne instanceof Error?ne.message:String(ne);ct(`Детализация прогона: ${Ee}`),re(null),De(null),Se(`Run detail load error for ${c}: ${Ee}`)}finally{Ot(!1)}},[fn,Se]),Yt=P.useCallback(async c=>{oe(!0),ct("");try{const E=await Ue.loadAutoRunsHistory({from:vs(N.fromLocal),to:vs(N.toLocal),target:N.target,mode:N.mode,use_mock:N.useMock,prompt_contains:N.promptContains.trim()||void 0,limit:N.limit});if(Le(E),E.items.length===0){He(""),Ye(""),re(null),De(null);return}const ne=c?.keepSelection??!0,Ee=c?.preferredRunId??"",Jt=c?.preferredCaseId??"",Xn=ne&&Ee&&E.items.some(Zn=>Zn.run_id===Ee)?Ee:E.items[0].run_id;await tt(Xn,ne?Jt:void 0),Je()}catch(E){const ne=E instanceof Error?E.message:String(E);ct(`История прогонов: ${ne}`),Se(`History load error: ${ne}`)}finally{oe(!1)}},[N.fromLocal,N.limit,N.mode,N.promptContains,N.target,N.toLocal,N.useMock,Je,tt,Se]),pn=P.useCallback(c=>{c.role==="assistant"&&et({open:!0,messageIndex:c.message_index,rating:c.annotation?.rating??3,comment:c.annotation?.comment??"",manualCaseDecision:c.annotation?.manual_case_decision??hs,annotationAuthor:c.annotation?.annotation_author??b.generatedBy,saving:!1,error:""})},[b.generatedBy]),Dt=P.useCallback(()=>{et(c=>c.saving?c:{open:!1,messageIndex:-1,rating:3,comment:"",manualCaseDecision:hs,annotationAuthor:b.generatedBy,saving:!1,error:""})},[b.generatedBy]),mn=P.useCallback(async()=>{if(!(!he||!Oe||ie.messageIndex<0)){if(!ie.comment.trim()){et(c=>({...c,error:"Добавьте комментарий."}));return}et(c=>({...c,saving:!0,error:""}));try{await Ue.saveAutoRunAnnotation({run_id:he,case_id:Oe,message_index:ie.messageIndex,rating:ie.rating,comment:ie.comment.trim(),manual_case_decision:ie.manualCaseDecision,annotation_author:ie.annotationAuthor.trim()||void 0}),await Promise.all([tt(he,Oe),Ct(),Je()]),Dt()}catch(c){et(E=>({...E,saving:!1,error:c instanceof Error?c.message:String(c)}))}}},[Dt,ie.annotationAuthor,ie.comment,ie.manualCaseDecision,ie.messageIndex,ie.rating,Ct,Je,tt,Oe,he]),hn=P.useCallback(async c=>{gt(c.annotation_id),z("settings"),await tt(c.run_id,c.case_id),L?.items.some(E=>E.run_id===c.run_id)||ct("Комментарий относится к прогону вне текущего фильтра. Детали загружены напрямую.")},[L?.items,tt]);return P.useEffect(()=>{Mt.current||(Mt.current=!0,Yt({keepSelection:!1}),Nt(),Je())},[Nt,Yt,Je]),P.useEffect(()=>{Mt.current&&Ct()},[xe,Ct]),o.jsxs(on,{className:"autoruns-frame",title:"",hideHeader:!0,children:[o.jsxs("div",{className:"autoruns-columns",children:[o.jsxs("section",{className:"autoruns-col",children:[o.jsx("h3",{children:"Левая панель"}),o.jsxs("div",{className:"tab-row",children:[o.jsx("button",{type:"button",className:ce==="settings"?"tab active":"tab",onClick:()=>z("settings"),children:"Настройки"}),o.jsx("button",{type:"button",className:ce==="comments"?"tab active":"tab",onClick:()=>z("comments"),children:"Комментарии"})]}),ce==="settings"?o.jsxs(o.Fragment,{children:[o.jsx("h4",{children:"Настройки выборки"}),o.jsxs("div",{className:"autoruns-form-grid",children:[o.jsxs("label",{children:["Дата с",o.jsx("input",{type:"datetime-local",value:N.fromLocal,onChange:c=>te(E=>({...E,fromLocal:c.target.value}))})]}),o.jsxs("label",{children:["Дата по",o.jsx("input",{type:"datetime-local",value:N.toLocal,onChange:c=>te(E=>({...E,toLocal:c.target.value}))})]}),o.jsxs("label",{children:["Целевой контур",o.jsxs("select",{value:N.target,onChange:c=>te(E=>({...E,target:c.target.value})),children:[o.jsx("option",{value:"all",children:"все"}),(L?.available.targets??[]).map(c=>o.jsx("option",{value:c,children:c},c))]})]}),o.jsxs("label",{children:["Режим",o.jsxs("select",{value:N.mode,onChange:c=>te(E=>({...E,mode:c.target.value})),children:[o.jsx("option",{value:"all",children:"все"}),(L?.available.modes??[]).map(c=>o.jsx("option",{value:c,children:c},c))]})]}),o.jsxs("label",{children:["Использовать mock",o.jsxs("select",{value:N.useMock,onChange:c=>te(E=>({...E,useMock:c.target.value})),children:[o.jsx("option",{value:"any",children:"любой"}),o.jsx("option",{value:"true",children:"да"}),o.jsx("option",{value:"false",children:"нет"})]})]}),o.jsxs("label",{children:["Лимит",o.jsx("input",{type:"number",min:1,max:500,value:N.limit,onChange:c=>te(E=>({...E,limit:Number(c.target.value||120)}))})]}),o.jsxs("label",{className:"full-width",children:["Версия промпта содержит",o.jsx("input",{value:N.promptContains,onChange:c=>te(E=>({...E,promptContains:c.target.value})),placeholder:"normalizer_v2_0_2 / address_query_runtime_v1",list:"autoruns-prompt-versions"})]})]}),o.jsx("datalist",{id:"autoruns-prompt-versions",children:(L?.available.prompt_versions??[]).map(c=>o.jsx("option",{value:c},c))}),o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",disabled:ae,onClick:()=>{Yt({keepSelection:!1})},children:ae?"Обновляю...":"Применить"}),o.jsx("button",{type:"button",className:"tab",onClick:()=>{te({...qu,fromLocal:Gu()}),ct("")},children:"Сбросить фильтры"})]}),o.jsx("h4",{children:"Контур генерации"}),o.jsxs("div",{className:"autoruns-meta-list",children:[o.jsxs("div",{children:[o.jsx("span",{children:"Провайдер:"}),o.jsx("strong",{children:a.llmProvider})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Модель:"}),o.jsx("strong",{children:a.model||"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Промпт ассистента:"}),o.jsx("strong",{children:p})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Промпт декомпозиции:"}),o.jsx("strong",{children:D})]})]}),o.jsx("h4",{children:"Автогенерация вопросов"}),o.jsxs("div",{className:"autoruns-form-grid",children:[o.jsxs("label",{children:["Режим генерации",o.jsxs("select",{value:b.mode,onChange:c=>pe(E=>({...E,mode:c.target.value})),children:[o.jsx("option",{value:"codex_creative",children:"codex_creative"}),o.jsx("option",{value:"qwen_seed",children:"qwen_seed"})]})]}),o.jsxs("label",{children:["Кол-во",o.jsx("input",{type:"number",min:1,max:200,value:b.count,onChange:c=>pe(E=>({...E,count:Math.max(1,Math.min(200,Number(c.target.value||24)))}))})]}),o.jsxs("label",{children:["Домен (опц.)",o.jsx("input",{value:b.domain,onChange:c=>pe(E=>({...E,domain:c.target.value})),placeholder:"vat / settlements / counterparties"})]}),o.jsxs("label",{children:["Кто генерирует",o.jsx("input",{value:b.generatedBy,onChange:c=>pe(E=>({...E,generatedBy:c.target.value})),placeholder:"manual_reviewer"})]}),o.jsxs("label",{className:"checkbox-row",children:[o.jsx("input",{type:"checkbox",checked:b.persistToEvalCases,onChange:c=>pe(E=>({...E,persistToEvalCases:c.target.checked}))}),"Сохранять кейс-сет в `eval_cases`"]})]}),o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",disabled:_,onClick:()=>{K()},children:_?"Генерирую...":"Сгенерировать пачку"}),o.jsx("button",{type:"button",className:"tab",disabled:ee,onClick:()=>{Nt()},children:ee?"Обновляю...":"Обновить историю"})]}),o.jsxs("div",{className:"autoruns-autogen-list",children:[ee?o.jsx("p",{className:"muted",children:"Загружаю историю автогенераций..."}):null,!ee&&R.length===0?o.jsx("p",{className:"muted",children:"История автогенераций пока пустая."}):null,R.slice(0,30).map(c=>o.jsxs("article",{className:"autoruns-autogen-item",children:[o.jsxs("header",{children:[o.jsx("strong",{children:Sr(c.created_at)}),o.jsx("span",{children:c.mode})]}),o.jsxs("div",{className:"autoruns-run-meta",children:["id=",c.generation_id," | count=",c.count]}),o.jsxs("div",{className:"autoruns-run-meta",children:["домен=",c.domain??"общий",c.generated_by?` | автор=${c.generated_by}`:""]}),c.saved_case_set_file?o.jsxs("div",{className:"autoruns-run-meta",children:["кейс-сет=",c.saved_case_set_file]}):null,(c.questions??[]).length>0?o.jsx("p",{children:c.questions[0]}):null]},c.generation_id))]}),o.jsxs("details",{className:"autoruns-prompt-details",children:[o.jsx("summary",{children:"Копия активного промпта (только чтение)"}),o.jsxs("label",{children:["Системный",o.jsx("textarea",{readOnly:!0,value:h.systemPrompt})]}),o.jsxs("label",{children:["Разработчика",o.jsx("textarea",{readOnly:!0,value:h.developerPrompt})]}),o.jsxs("label",{children:["Доменный",o.jsx("textarea",{readOnly:!0,value:h.domainPrompt})]}),o.jsxs("label",{children:["Заметки по схеме",o.jsx("textarea",{readOnly:!0,value:h.schemaNotes})]}),o.jsxs("label",{children:["Примеры few-shot",o.jsx("textarea",{readOnly:!0,value:h.fewShotExamples})]})]})]}):o.jsxs(o.Fragment,{children:[o.jsx("h4",{children:"Размеченные ответы"}),o.jsx("div",{className:"autoruns-form-grid",children:o.jsxs("label",{children:["Фильтр решений",o.jsxs("select",{value:xe,onChange:c=>ze(c.target.value),children:[o.jsx("option",{value:"all",children:"все"}),(Ne.length>0?Ne:X?.enum??[]).map(c=>o.jsx("option",{value:c,children:String(X?.labels?.[c]??c)},c))]})]})}),o.jsxs("div",{className:"autoruns-stats-grid",children:[o.jsxs("div",{children:[o.jsx("span",{children:"Комментариев"}),o.jsx("strong",{children:ye.length})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Средний рейтинг"}),o.jsx("strong",{children:dn===null?"нет данных":`${dn.toFixed(2)} / 5`})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Последний"}),o.jsx("strong",{children:ye.length>0?Sr(ye[0].updated_at):"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Статус"}),o.jsx("strong",{children:kt?"обновляю":"готово"})]})]}),o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",disabled:kt,onClick:()=>{Ct()},children:kt?"Обновляю...":"Обновить список"}),o.jsx("button",{type:"button",className:"tab",disabled:J,onClick:()=>{Je()},children:J?"Идет пост-анализ...":"Обновить пост-анализ"})]}),o.jsxs("div",{className:"autoruns-comments-list",children:[kt?o.jsx("p",{className:"muted",children:"Загружаю комментарии..."}):null,!kt&&ye.length===0?o.jsx("p",{className:"muted",children:"Пока нет откомментированных ответов."}):null,ye.map(c=>o.jsxs("button",{type:"button",className:be===c.annotation_id?"autoruns-comment-item selected":"autoruns-comment-item",onClick:()=>{hn(c)},children:[o.jsxs("div",{className:"autoruns-comment-head",children:[o.jsx("strong",{children:Ju(c.rating)}),o.jsx("span",{children:Sr(c.updated_at)})]}),o.jsx("div",{className:"autoruns-run-meta",children:c.run_id}),o.jsxs("div",{className:"autoruns-run-meta",children:["case=",c.case_id," | msg=",c.message_index]}),o.jsxs("div",{className:"autoruns-run-meta",children:["decision=",c.manual_case_decision,c.annotation_author?` | author=${c.annotation_author}`:""]}),o.jsx("p",{children:c.comment})]},c.annotation_id))]}),ge?o.jsxs(o.Fragment,{children:[o.jsx("h4",{children:"Тех-контекст брака"}),o.jsxs("div",{className:"autoruns-meta-list",children:[o.jsxs("div",{children:[o.jsx("span",{children:"trace:"}),o.jsx("strong",{children:ge.technical_context.trace_id??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"reply_type:"}),o.jsx("strong",{children:ge.technical_context.reply_type??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"domain:"}),o.jsx("strong",{children:ge.technical_context.domain??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"query_class:"}),o.jsx("strong",{children:ge.technical_context.query_class??"нет данных"})]})]}),o.jsx("h4",{children:"JSON разбор"}),o.jsx(at,{value:{annotation_id:ge.annotation_id,run_id:ge.run_id,case_id:ge.case_id,message_index:ge.message_index,rating:ge.rating,comment:ge.comment,manual_case_decision:ge.manual_case_decision,annotation_author:ge.annotation_author,context:ge.context,technical_context:ge.technical_context,case_summary:ge.case_summary?{case_id:ge.case_summary.case_id,domain:ge.case_summary.domain,query_class:ge.case_summary.query_class,checks:ge.case_summary.checks,metric_subscores:ge.case_summary.metric_subscores}:null}})]}):null]}),un?o.jsx("p",{className:"error-text",children:un}):null]}),o.jsxs("section",{className:"autoruns-col",children:[o.jsx("h3",{children:"Выдача прогонов"}),o.jsxs("div",{className:"autoruns-stats-grid",children:[o.jsxs("div",{children:[o.jsx("span",{children:"Всего"}),o.jsx("strong",{children:L?.stats.runs_total??0})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Средний score"}),o.jsx("strong",{children:ll(L?.stats.avg_score_index??null)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Тренд"}),o.jsx("strong",{children:L?Yu(L.stats.trend):"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Блокеры"}),o.jsx("strong",{children:L?.stats.blocking_runs??0})]})]}),o.jsxs("div",{className:"autoruns-run-list",children:[(L?.items??[]).map(c=>o.jsxs("button",{type:"button",className:he===c.run_id?"autoruns-run-item selected":"autoruns-run-item",onClick:()=>{tt(c.run_id)},children:[o.jsxs("div",{className:"autoruns-run-head",children:[o.jsx("strong",{children:Sr(c.run_timestamp)}),o.jsx("span",{children:pf(c.eval_target)})]}),o.jsx("div",{className:"autoruns-run-meta",children:c.run_id}),o.jsxs("div",{className:"autoruns-run-meta",children:["режим=",c.mode??"нет данных"," | mock=",String(c.use_mock)]}),c.llm_provider||c.model?o.jsxs("div",{className:"autoruns-run-meta",children:["llm=",c.llm_provider??"нет данных"," | модель=",c.model??"нет данных"]}):null,o.jsxs("div",{className:"autoruns-run-meta",children:["промпт=",c.prompt_version??"нет данных"]}),o.jsxs("div",{className:"autoruns-run-foot",children:[o.jsxs("span",{children:["оценка: ",ll(c.score_index)]}),o.jsxs("span",{children:["закрыто/открыто: ",c.closed_cases,"/",c.open_cases]})]}),o.jsxs("div",{className:"autoruns-run-foot",children:[o.jsxs("span",{children:["блокеры: ",c.blocking_failures]}),o.jsxs("span",{children:["качество: ",c.quality_failures]})]})]},c.run_id)),(L?.items.length??0)===0?o.jsx("p",{className:"muted",children:"За выбранный диапазон прогонов нет."}):null]})]}),o.jsxs("section",{className:"autoruns-col",children:[o.jsx("h3",{children:"Диалог прогона"}),o.jsxs("div",{className:"autoruns-dialog-toolbar",children:[o.jsxs("label",{children:["Прогон",o.jsx("select",{value:he,onChange:c=>{const E=c.target.value;tt(E)},children:Jn.map(c=>o.jsxs("option",{value:c.run_id,children:[Sr(c.run_timestamp)," | ",c.run_id]},c.run_id))})]}),o.jsxs("label",{children:["Кейс",o.jsx("select",{value:Oe,onChange:c=>{const E=c.target.value;Ye(E),he&&E&&fn(he,E)},children:(me?.cases??[]).map(c=>o.jsxs("option",{value:c.case_id,children:[c.case_id," | ",c.status]},c.case_id))})]})]}),o.jsx("div",{className:"autoruns-case-list",children:(me?.cases??[]).map(c=>o.jsxs("button",{type:"button",className:Oe===c.case_id?"autoruns-case-item selected":"autoruns-case-item",onClick:()=>{Ye(c.case_id),he&&fn(he,c.case_id)},children:[o.jsx("span",{children:c.case_id}),o.jsxs("span",{children:[c.status,c.commented_count>0?` | 💬${c.commented_count}`:""]})]},c.case_id))}),o.jsxs("div",{className:"autoruns-dialog-view",children:[an||ke?o.jsx("p",{className:"muted",children:"Загружаю диалог..."}):null,!an&&!ke&&(G?.messages.length??0)===0?o.jsx("p",{className:"muted",children:"Диалог для этого кейса не найден."}):null,(G?.messages??[]).map((c,E)=>{const ne=c.role==="assistant"?"assistant":"user";return o.jsxs("article",{className:`autoruns-msg ${ne}`,children:[o.jsxs("header",{children:[o.jsx("strong",{children:ne==="assistant"?"Система":"Модель/вопрос"}),o.jsxs("div",{className:"autoruns-msg-head-actions",children:[o.jsx("span",{children:c.created_at?Sr(c.created_at):"нет данных"}),ne==="assistant"?o.jsx("button",{type:"button",className:c.commented?"autoruns-comment-icon commented":"autoruns-comment-icon",onClick:()=>pn(c),title:"Комментировать ответ системы",children:"💬"}):null]})]}),o.jsx("p",{children:c.text}),ne==="assistant"&&c.annotation?o.jsxs("div",{className:"autoruns-msg-annotation",children:[o.jsx("strong",{children:Ju(c.annotation.rating)}),o.jsx("span",{children:c.annotation.comment}),o.jsxs("span",{className:"muted",children:[c.annotation.manual_case_decision,c.annotation.annotation_author?` | ${c.annotation.annotation_author}`:""]})]}):null,(c.trace_id||c.reply_type)&&o.jsxs("footer",{children:[c.trace_id?o.jsxs("span",{children:["trace=",c.trace_id]}):null,c.reply_type?o.jsxs("span",{children:["reply_type=",c.reply_type]}):null]})]},c.message_id??`${ne}-${E}`)})]})]}),k?o.jsxs("section",{className:"autoruns-col",children:[o.jsx("h3",{children:"Режим ассистента"}),o.jsxs("div",{className:"autoruns-meta-list",children:[o.jsxs("div",{children:[o.jsx("span",{children:"источник:"}),o.jsx("strong",{children:G?.source??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"сессия:"}),o.jsx("strong",{children:G?.session_id??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"контур прогона:"}),o.jsx("strong",{children:yt?.eval_target??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"оценка прогона:"}),o.jsx("strong",{children:ll(yt?.score_index??null)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"комментарии:"}),o.jsx("strong",{children:me?.annotations_summary?.total??0})]})]}),o.jsx("h4",{children:"Пакет режима ассистента"}),o.jsx(at,{value:G?.assistant_mode??{note:"assistant_mode недоступен"}}),o.jsx("h4",{style:{marginTop:12},children:"Проверки кейса"}),o.jsx(at,{value:xt?.checks??{note:"checks недоступен"}}),o.jsx("h4",{style:{marginTop:12},children:"Сабскор метрик"}),o.jsx(at,{value:xt?.metric_subscores??{note:"metric_subscores недоступен"}})]}):null,F?o.jsxs("section",{className:"autoruns-col",children:[o.jsx("h3",{children:"Режим декомпозиции"}),o.jsxs("div",{className:"autoruns-meta-list",children:[o.jsxs("div",{children:[o.jsx("span",{children:"кейс:"}),o.jsx("strong",{children:xt?.case_id??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"домен:"}),o.jsx("strong",{children:xt?.domain??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"класс запроса:"}),o.jsx("strong",{children:xt?.query_class??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"trace:"}),o.jsx("strong",{children:xt?.trace_id??"нет данных"})]})]}),o.jsx("h4",{children:"Шаги декомпозиции"}),(G?.decomposition.length??0)>0?o.jsx("ol",{className:"autoruns-decomposition-list",children:(G?.decomposition??[]).map((c,E)=>o.jsx("li",{children:c},`${E}-${c.slice(0,24)}`))}):o.jsx("p",{className:"muted",children:"В логах кейса нет явной декомпозиции."})]}):null,H?o.jsxs("section",{className:"autoruns-col",children:[o.jsx("h3",{children:"Прогресс / регресс"}),o.jsxs("div",{className:"autoruns-stats-grid",children:[o.jsxs("div",{children:[o.jsx("span",{children:"Последний score"}),o.jsx("strong",{children:ll(L?.stats.latest_score_index??null)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Предыдущий"}),o.jsx("strong",{children:ll(L?.stats.previous_score_index??null)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Тренд"}),o.jsx("strong",{children:L?Yu(L.stats.trend):"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Пробелы качества"}),o.jsx("strong",{children:L?.stats.quality_gap_runs??0})]})]}),o.jsx("h4",{children:"Покрытие доменов (история)"}),Xu(L?.stats.domain_coverage??[]),o.jsx("h4",{style:{marginTop:14},children:"Покрытие доменов (выбранный прогон)"}),Xu(me?.coverage.domain_coverage??[]),o.jsx("h4",{style:{marginTop:14},children:"Очереди фиксов пост-анализа"}),J?o.jsx("p",{className:"muted",children:"Собираю пост-анализ..."}):null,J?null:o.jsx("div",{className:"autoruns-stats-grid",children:Object.entries(T?.post_analysis.stats.by_queue??{}).map(([c,E])=>o.jsxs("div",{children:[o.jsx("span",{children:c}),o.jsx("strong",{children:E})]},c))}),o.jsxs("div",{className:"autoruns-autogen-list",children:[(T?.post_analysis.recommended_regression_candidates??[]).slice(0,12).map(c=>o.jsxs("article",{className:"autoruns-autogen-item",children:[o.jsxs("header",{children:[o.jsx("strong",{children:c.manual_case_decision}),o.jsxs("span",{children:[c.rating,"/5"]})]}),o.jsxs("div",{className:"autoruns-run-meta",children:[c.domain??"неизвестно"," / ",c.query_class??"неизвестно"]}),o.jsx("p",{children:c.comment})]},c.annotation_id)),!J&&(T?.post_analysis.recommended_regression_candidates.length??0)===0?o.jsx("p",{className:"muted",children:"Рекомендованных кандидатов пока нет."}):null]})]}):null]}),ie.open?o.jsx("div",{className:"autoruns-comment-modal-backdrop",onClick:c=>{c.target===c.currentTarget&&Dt()},children:o.jsxs("div",{className:"autoruns-comment-modal",children:[o.jsx("h3",{children:"Комментарий к ответу системы"}),o.jsx("p",{className:"muted",children:"Оцените ответ по 5-балльной шкале и добавьте комментарий по браку."}),cn?o.jsxs("details",{className:"autoruns-prompt-details",open:!0,children:[o.jsx("summary",{children:"Ответ системы"}),o.jsx("p",{className:"autoruns-comment-quote",children:cn.text})]}):null,o.jsx("div",{className:"autoruns-rating-row",role:"group","aria-label":"Рейтинг ответа",children:[1,2,3,4,5].map(c=>o.jsx("button",{type:"button",className:ie.rating>=c?"autoruns-rating-dot active":"autoruns-rating-dot",onClick:()=>et(E=>({...E,rating:c})),disabled:ie.saving,"aria-label":`Оценка ${c}`,children:ie.rating>=c?"●":"○"},c))}),o.jsxs("div",{className:"autoruns-form-grid",children:[o.jsxs("label",{children:["Решение по кейсу",o.jsx("select",{value:ie.manualCaseDecision,onChange:c=>et(E=>({...E,manualCaseDecision:c.target.value})),disabled:ie.saving,children:(Ne.length>0?Ne:X?.enum??[hs]).map(c=>o.jsx("option",{value:c,children:String(X?.labels?.[c]??c)},c))})]}),o.jsxs("label",{children:["Автор комментария",o.jsx("input",{value:ie.annotationAuthor,onChange:c=>et(E=>({...E,annotationAuthor:c.target.value})),placeholder:"manual_reviewer",disabled:ie.saving})]})]}),o.jsxs("label",{children:["Комментарий",o.jsx("textarea",{value:ie.comment,onChange:c=>et(E=>({...E,comment:c.target.value})),placeholder:"Почему ответ бракованный, что именно пошло не так, какие технические детали проверить.",rows:4,disabled:ie.saving})]}),ie.error?o.jsx("p",{className:"error-text",children:ie.error}):null,o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",onClick:()=>{mn()},disabled:ie.saving,children:ie.saving?"Сохраняю...":"Готово"}),o.jsx("button",{type:"button",className:"tab",onClick:Dt,disabled:ie.saving,children:"Отмена"})]})]})}):null]})}const vf=/(?:^|\n)\s*#{0,6}\s*(?:debug_payload_json|technical_breakdown_json|route_summary_json|debug_payload|technical_breakdown)\b/i,gf=[/\b(?:debug_payload_json|technical_breakdown_json)\b/i,/\b(?:route_summary|semantic_profile|domain_scope|relation_patterns|account_scope)\b/i,/\b(?:coverage_report|retrieval_status|problem_unit_state|candidate_evidence)\b/i,/\b(?:graph_domain_scope|graph_runtime|selection_reason|why_included)\b/i];function yf(a){try{return JSON.stringify(a,null,2)}catch{return String(a)}}function xf(a){const h=String(a??""),p=h.match(vf);return(p?h.slice(0,p.index):h).replace(/###\s*(?:debug_payload_json|technical_breakdown_json|route_summary_json)[\s\S]*?(?:```[\s\S]*?```|$)/gi,"").replace(/(?:^|\n)\s*#{0,6}\s*(?:debug_payload_json|technical_breakdown_json|route_summary_json)\b[\s\S]*$/gi,"").split(/\r?\n/g).map(H=>H.trimEnd()).filter(H=>H.trim().length>0).filter(H=>!gf.some(Z=>Z.test(H))).join(` +`).trim()}function _f(a,h,p="default"){const D=p==="technical",k=[];k.push("# Assistant conversation export"),k.push(`session_id: ${a||"n/a"}`),k.push(`export_mode: ${p}`),k.push(`exported_at: ${new Date().toISOString()}`),k.push("");for(let F=0;F{re.current&&(re.current.scrollTop=re.current.scrollHeight)},[h,Le]),P.useEffect(()=>()=>{G.current!==null&&window.clearTimeout(G.current)},[]);async function ze(X){if(h.length===0)return;const $e=_f(a,h,X),Ne=await jf($e);xe(X==="technical"?"тех":"чат"),ye(Ne?"success":"error"),G.current!==null&&window.clearTimeout(G.current),G.current=window.setTimeout(()=>{ye("idle")},2200)}return o.jsxs(on,{title:"Режим ассистента",subtitle:"Диалоговый слой поверх normalizer, маршрутизации и factual retrieval.",actions:o.jsxs("div",{className:"assistant-panel-actions",children:[o.jsx("button",{type:"button",className:"assistant-copy-btn",onClick:()=>{ze("default")},disabled:h.length===0,title:"Экспорт только user-facing чата",children:"Скопировать чат"}),o.jsx("button",{type:"button",className:"assistant-copy-btn",onClick:()=>{ze("technical")},disabled:h.length===0,title:"Технический экспорт с debug payload",children:"Скопировать техчат"}),De==="success"?o.jsxs("span",{className:"assistant-copy-feedback success",children:["Скопировано (",We,")"]}):null,De==="error"?o.jsx("span",{className:"assistant-copy-feedback error",children:"Ошибка копирования"}):null,o.jsx("span",{className:"status-chip",children:a?`session: ${a}`:"новая сессия"})]}),children:[o.jsxs("div",{ref:re,className:"assistant-chat-list",children:[h.length===0?o.jsx("div",{className:"assistant-empty muted",children:"Диалог пуст. Отправьте первый вопрос, чтобы запустить контур ассистента."}):null,h.map(X=>o.jsxs("article",{className:`assistant-msg ${X.role}`,children:[o.jsxs("header",{className:"assistant-msg-head",children:[o.jsx("strong",{children:Sf(X.role)}),o.jsx("span",{children:wf(X.created_at)})]}),o.jsx("div",{className:"assistant-msg-body",children:X.text}),X.role==="assistant"&&X.debug?o.jsxs("details",{className:"assistant-debug",children:[o.jsx("summary",{children:"Показать технический разбор"}),o.jsx(at,{value:X.debug})]}):null]},X.message_id))]}),o.jsxs("div",{className:"assistant-compose",children:[o.jsxs("div",{className:"grid-two",children:[o.jsxs("label",{children:["Подсказка по периоду",o.jsx("input",{value:k,onChange:X=>F(X.target.value)})]}),o.jsxs("label",{children:["Бизнес-контекст",o.jsx("input",{value:H,onChange:X=>Z(X.target.value)})]})]}),o.jsxs("label",{className:"full-width",children:["Сообщение",o.jsx("textarea",{value:p,onChange:X=>D(X.target.value),rows:4,placeholder:"Введите вопрос к данным компании..."})]}),o.jsxs("div",{className:"button-row",children:[o.jsxs("label",{className:"checkbox-row",children:[o.jsx("input",{type:"checkbox",checked:N,onChange:X=>te(X.target.checked)}),"Mock-режим"]}),o.jsx("button",{type:"button",onClick:()=>ce(),disabled:L||!p.trim(),children:L?"Выполняю...":"Отправить"}),o.jsx("button",{type:"button",onClick:()=>z(),disabled:L&&h.length===0,children:"Сбросить сессию"})]}),Le?o.jsx("p",{className:"diff-summary",children:Le}):null,me?o.jsx("p",{className:"error-text",children:me}):null]})]})}function Zu({value:a,modelOptions:h,modelsBusy:p,onChange:D,onReloadModels:k,onTestConnection:F,onSaveLocalConfig:H,lastStatus:Z,busy:N}){const te=a.llmProvider==="local",ce=h.includes(a.model);return o.jsxs(on,{title:"LLM Connection",subtitle:"Switch between OpenAI cloud and local OpenAI-compatible server.",actions:o.jsx("span",{className:"status-chip",children:Z||"Status: not checked"}),children:[o.jsxs("div",{className:"grid-two",children:[o.jsxs("label",{children:["Provider",o.jsxs("select",{value:a.llmProvider,onChange:z=>{const L=z.target.value==="local"?"local":"openai";D({...a,llmProvider:L,baseUrl:L==="local"?"http://127.0.0.1:1234/v1":"https://api.openai.com/v1"})},children:[o.jsx("option",{value:"openai",children:"OpenAI (token)"}),o.jsx("option",{value:"local",children:"Local (LM Studio / OpenAI-compatible)"})]})]}),o.jsxs("label",{children:["Model",o.jsxs("select",{value:ce?a.model:"__manual__",onChange:z=>{const L=z.target.value;L!=="__manual__"&&D({...a,model:L})},children:[o.jsx("option",{value:"__manual__",children:"Manual input"}),h.map(z=>o.jsx("option",{value:z,children:z},z))]})]}),o.jsxs("label",{children:["Model ID (manual)",o.jsx("input",{value:a.model,onChange:z=>D({...a,model:z.target.value}),placeholder:"qwen2.5-14b-instruct or lmstudio loaded model id"})]}),te?null:o.jsxs("label",{className:"full-width",children:["OpenAI API Key",o.jsx("input",{type:"password",value:a.apiKey,onChange:z=>D({...a,apiKey:z.target.value}),placeholder:"sk-..."})]}),o.jsxs("label",{className:te?"full-width":void 0,children:[te?"Local server base URL":"Base URL",o.jsx("input",{value:a.baseUrl,onChange:z=>D({...a,baseUrl:z.target.value}),placeholder:te?"http://127.0.0.1:1234/v1":"https://api.openai.com/v1"})]}),o.jsxs("label",{children:["Temperature",o.jsx("input",{type:"number",step:"0.1",value:a.temperature,onChange:z=>D({...a,temperature:Number(z.target.value)})})]}),o.jsxs("label",{children:["Max output tokens",o.jsx("input",{type:"number",value:a.maxOutputTokens,onChange:z=>D({...a,maxOutputTokens:Number(z.target.value)})})]})]}),o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",onClick:()=>H(),children:"Save local config"}),o.jsx("button",{type:"button",onClick:()=>k(),disabled:N||p,children:p?"Loading models...":"Load model list"}),o.jsx("button",{type:"button",onClick:()=>F(),disabled:N,children:N?"Checking...":"Test connection"})]})]})}function Cf({items:a,onRefresh:h,onOpenTrace:p}){return o.jsx(on,{title:"История нормализаций",subtitle:"Короткий вопрос, confidence, route hint и статус валидации.",actions:o.jsx("button",{type:"button",onClick:()=>h(),children:"Обновить"}),children:o.jsxs("div",{className:"history-list",children:[a.length===0?o.jsx("p",{className:"muted",children:"История пока пустая."}):null,a.map(D=>o.jsxs("button",{type:"button",className:"history-item",onClick:()=>p(D.trace_id),children:[o.jsxs("div",{className:"history-row",children:[o.jsx("strong",{children:D.route_hint??"route: n/a"}),o.jsx("span",{children:D.validation_passed?"schema: ok":"schema: fail"})]}),o.jsx("p",{children:D.question_short}),o.jsxs("div",{className:"history-row",children:[o.jsx("span",{children:D.model}),o.jsx("span",{children:new Date(D.timestamp).toLocaleString("ru-RU")})]})]},D.trace_id))]})})}function sn(a){return a==null||a===""?"—":String(a)}function Nf({result:a}){return o.jsx(on,{title:"Runtime метрики",subtitle:"trace_id, токены, latency и статус валидации.",children:o.jsxs("div",{className:"metrics-grid",children:[o.jsxs("div",{children:[o.jsx("span",{children:"trace_id"}),o.jsx("strong",{children:sn(a?.trace_id)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"request_started_at"}),o.jsx("strong",{children:sn(a?new Date(Date.now()-a.latency_ms).toISOString():null)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"request_finished_at"}),o.jsx("strong",{children:sn(a?new Date().toISOString():null)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"latency_ms"}),o.jsx("strong",{children:sn(a?.latency_ms)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"input_tokens"}),o.jsx("strong",{children:sn(a?.usage?.input_tokens)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"output_tokens"}),o.jsx("strong",{children:sn(a?.usage?.output_tokens)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"total_tokens"}),o.jsx("strong",{children:sn(a?.usage?.total_tokens)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"validation_status"}),o.jsx("strong",{children:a?.validation?.passed?"passed":"failed"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"prompt_version"}),o.jsx("strong",{children:sn(a?.prompt_version)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"schema_version"}),o.jsx("strong",{children:sn(a?.schema_version)})]})]})})}const Ef={normalized:"Normalized JSON",fragments:"Fragment View",scope:"Scope View",flags:"Flags View",route:"Route Simulation",raw:"Raw model output",validation:"Validation",logs:"Logs"};function Pf(a){return a&&typeof a=="object"?a:null}function Rf({tab:a,onTabChange:h,result:p,appLogs:D}){const k=["normalized","fragments","scope","flags","route","raw","validation","logs"],F=Pf(p?.normalized),H=String(F?.schema_version??""),Z=H==="normalized_query_v2"||H==="normalized_query_v2_0_1"||H==="normalized_query_v2_0_2",N=Z?{fragments:F?.fragments??[],discarded_fragments:F?.discarded_fragments??[]}:{note:"Fragment View доступен для normalized_query_v2."},te=Z?{message_in_scope:F?.message_in_scope??null,scope_confidence:F?.scope_confidence??null,contains_multiple_tasks:F?.contains_multiple_tasks??null,global_notes:F?.global_notes??null}:{note:"Scope View доступен для normalized_query_v2."},ce=Z?Array.isArray(F?.fragments)?(F?.fragments).map(z=>({fragment_id:z.fragment_id??null,domain_relevance:z.domain_relevance??null,candidate_labels:z.candidate_labels??[],execution_readiness:z.execution_readiness??null,clarification_reason:z.clarification_reason??null,soft_assumption_used:z.soft_assumption_used??[],route_status:z.route_status??null,no_route_reason:z.no_route_reason??null,flags:z.flags??{}})):[]:{note:"Flags View доступен для normalized_query_v2."};return o.jsxs(on,{title:"Выходные данные",subtitle:"Structured output и диагностические вкладки.",children:[o.jsx("div",{className:"tab-row",children:k.map(z=>o.jsx("button",{type:"button",className:a===z?"tab active":"tab",onClick:()=>h(z),children:Ef[z]},z))}),a==="normalized"?o.jsx(at,{value:p?.normalized??{note:"Нет данных."}}):null,a==="fragments"?o.jsx(at,{value:N}):null,a==="scope"?o.jsx(at,{value:te}):null,a==="flags"?o.jsx(at,{value:ce}):null,a==="route"?o.jsx(at,{value:p?.route_hint_summary??{note:"Нет данных."}}):null,a==="raw"?o.jsx(at,{value:p?.raw_model_output??{note:"Нет данных."}}):null,a==="validation"?o.jsx(at,{value:p?.validation??{note:"Нет данных."}}):null,a==="logs"?o.jsx(at,{value:D}):null]})}function bu({value:a,onChange:h,presets:p,selectedPresetId:D,onSelectPreset:k,onLoadPreset:F,onSavePreset:H,onResetDefaults:Z,onDiffPrevious:N,presetName:te,onPresetNameChange:ce,diffSummary:z}){return o.jsxs(on,{title:"Prompt Manager",subtitle:"Системный, developer и domain уровни управляются отдельно.",children:[o.jsxs("div",{className:"grid-two",children:[o.jsxs("label",{children:["Системный prompt",o.jsx("textarea",{value:a.systemPrompt,onChange:L=>h({...a,systemPrompt:L.target.value}),rows:6})]}),o.jsxs("label",{children:["Developer / Instruction prompt",o.jsx("textarea",{value:a.developerPrompt,onChange:L=>h({...a,developerPrompt:L.target.value}),rows:6})]}),o.jsxs("label",{children:["Domain prompt",o.jsx("textarea",{value:a.domainPrompt,onChange:L=>h({...a,domainPrompt:L.target.value}),rows:6})]}),o.jsxs("label",{children:["Schema notes",o.jsx("textarea",{value:a.schemaNotes,onChange:L=>h({...a,schemaNotes:L.target.value}),rows:6})]}),o.jsxs("label",{className:"full-width",children:["Few-shot examples",o.jsx("textarea",{value:a.fewShotExamples,onChange:L=>h({...a,fewShotExamples:L.target.value}),rows:8})]})]}),o.jsxs("div",{className:"button-row",children:[o.jsxs("select",{value:D,onChange:L=>k(L.target.value),children:[o.jsx("option",{value:"",children:"Выберите preset..."}),p.map(L=>o.jsx("option",{value:L.id,children:L.name},L.id))]}),o.jsx("button",{type:"button",onClick:()=>F(),children:"Загрузить preset"}),o.jsx("input",{value:te,onChange:L=>ce(L.target.value),placeholder:"Имя для сохранения"}),o.jsx("button",{type:"button",onClick:()=>H(),children:"Сохранить preset"}),o.jsx("button",{type:"button",onClick:()=>N(),children:"Diff с предыдущим"}),o.jsx("button",{type:"button",onClick:()=>Z(),children:"Сбросить к default"})]}),z?o.jsx("p",{className:"diff-summary",children:z}):null]})}function Tf({value:a,onChange:h,onApplyBatchFormat:p,onNormalize:D,busy:k,useMock:F,onUseMockChange:H,errorMessage:Z}){return o.jsxs(on,{title:"Запрос пользователя",subtitle:"NDC semantic front-end: нормализуем, но не отвечаем за бухгалтерскую суть.",children:[o.jsxs("div",{className:"grid-two",children:[o.jsxs("label",{className:"full-width",children:["Raw user question",o.jsx("textarea",{value:a.userQuestion,onChange:N=>h({...a,userQuestion:N.target.value}),rows:6,placeholder:"Например: По каким покупателям у нас на конец июня висят отгрузки без оплаты..."})]}),o.jsxs("label",{className:"full-width",children:["Batch queries (`;` separator)",o.jsx("textarea",{value:a.batchQuestionsRaw,onChange:N=>h({...a,batchQuestionsRaw:N.target.value}),onBlur:()=>p(),rows:8,placeholder:"Вопрос 1; Вопрос 2; Вопрос 3"})]}),o.jsxs("label",{children:["Optional period context",o.jsx("input",{value:a.periodHint,onChange:N=>h({...a,periodHint:N.target.value})})]}),o.jsxs("label",{children:["Optional business context",o.jsx("input",{value:a.businessContext,onChange:N=>h({...a,businessContext:N.target.value})})]}),o.jsxs("label",{children:["Optional expected route (eval)",o.jsx("input",{value:a.expectedRoute,onChange:N=>h({...a,expectedRoute:N.target.value})})]})]}),o.jsxs("div",{className:"button-row",children:[o.jsxs("label",{className:"checkbox-row",children:[o.jsx("input",{type:"checkbox",checked:F,onChange:N=>H(N.target.checked)}),"Mock-режим (без вызова OpenAI)"]}),o.jsx("button",{type:"button",onClick:()=>p(),disabled:k||!a.batchQuestionsRaw.trim(),children:"Применить `;` в переносы"}),o.jsx("button",{type:"button",onClick:()=>D(!1),disabled:k||!a.userQuestion.trim(),children:k?"Нормализуем...":"Normalize"}),o.jsx("button",{type:"button",onClick:()=>D(!0),disabled:k||!a.userQuestion.trim(),children:k?"Сохраняем...":"Normalize + Save as test case"})]}),Z?o.jsx("p",{className:"error-text",children:Z}):null]})}function Lf({runs:a,selectedRunId:h,onSelectRun:p,onStartRun:D,onFinishRun:k,onRefreshRuns:F,onRunEval:H,onCopyEvalReport:Z,evalBusy:N,traceItems:te,evalReport:ce}){return o.jsxs(on,{title:"NDC Run Monitor",subtitle:"Важно: кнопка Запустить run создает только run-сущность. Кнопка eval запускает batch-проверку normalizer v2.0.2.",children:[o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",onClick:()=>D(),children:"Запустить run"}),o.jsx("button",{type:"button",onClick:()=>k(),disabled:!h,children:"Завершить выбранный run"}),o.jsx("button",{type:"button",onClick:()=>F(),children:"Обновить runs"}),o.jsx("button",{type:"button",onClick:()=>H(),disabled:N,children:N?"Идет eval v2.0.2...":"Запустить eval v2.0.2"})]}),o.jsxs("div",{className:"runtime-grid",children:[o.jsxs("div",{className:"runtime-runs",children:[a.map(z=>o.jsxs("button",{type:"button",className:h===z.runId?"history-item selected":"history-item",onClick:()=>p(z.runId),children:[o.jsxs("div",{className:"history-row",children:[o.jsx("strong",{children:z.status}),o.jsx("span",{children:z.runId})]}),o.jsxs("div",{className:"history-row",children:[o.jsx("span",{children:z.sessionId}),o.jsx("span",{children:new Date(z.updatedAt).toLocaleString("ru-RU")})]})]},z.runId)),a.length===0?o.jsx("p",{className:"muted",children:"Нет активных запусков."}):null]}),o.jsxs("div",{children:[o.jsx("h3",{children:"Trace выбранного run"}),o.jsx(at,{value:te}),o.jsxs("div",{className:"eval-report-wrap",children:[o.jsx("h3",{style:{marginTop:12},children:"Отчет eval"}),o.jsx(at,{value:ce??{note:"Eval пока не запускался"}}),o.jsx("button",{type:"button",className:"copy-cube-button",title:"Скопировать отчет eval",onClick:()=>Z(),children:"⧉"})]})]})]})]})}const zf={llmProvider:"openai",apiKey:"",model:"gpt-4o-mini",baseUrl:"https://api.openai.com/v1",temperature:0,maxOutputTokens:700},ec={systemPrompt:"Ты semantic-normalizer для бухгалтерского ассистента NDC. Возвращай только JSON по схеме normalized_query_v2_0_2.",developerPrompt:"Сначала делай decomposition сообщения на task fragments, затем определяй domain scope и route-critical flags. Для каждого fragment заполняй execution_readiness + route_status + no_route_reason. Если fragment routable, не оставляй его в no_route.",domainPrompt:"Контур: данные текущего предприятия в 1С/NDC. In-scope: документы, проводки, взаиморасчеты, остатки, периодное закрытие, аномалии и контрольные проверки. Out-of-scope: общая теория, законы и оффтоп.",schemaNotes:"schema_version: normalized_query_v2_0_2. Строгий JSON без дополнительных полей.",fewShotExamples:"Q: Проверь по поставщикам хвосты и разложи цепочку документов/оплат. => fragment in_scope, flags: multi_entity + chain_explanation. Q: Как вообще по ФСБУ? => out_of_scope/generic_accounting."},Of={userQuestion:"",batchQuestionsRaw:"",periodHint:"",businessContext:"",expectedRoute:""},Mf={colors:{backgroundRgb:"18, 18, 18",mainSurfaceRgb:"23, 23, 23",horizontalSurfaceRgb:"40, 40, 40",focusSurfaceRgb:"30, 30, 30",activeRgb:"167, 59, 255",activeTextRgb:"18, 18, 18",textMainRgb:"240, 240, 240",textMutedRgb:"166, 166, 166",dangerRgb:"205, 126, 126",scrollbarTrackRgb:"20, 20, 20",scrollbarThumbRgb:"30, 30, 30",scrollbarThumbHoverRgb:"30, 50, 30"}},tc="ndc_normalizer_session_config_v1",li=["Анализ запроса","Получение данных","Подготовка ответа"],Df="assistant",si="normalizer_v2_0_2",nc="address_query_runtime_v1";function If(a){return`[${new Date().toLocaleTimeString("ru-RU")}] ${a}`}function Af(a,h){if(!h)return"Previous preset is not selected.";const D=["systemPrompt","developerPrompt","domainPrompt","schemaNotes","fewShotExamples"].filter(k=>a[k]!==h[k]).map(k=>`${k}: ${Math.abs(a[k].length-h[k].length)} chars delta`);return D.length===0?"No changes against previous preset.":`Changed fields: ${D.length}. ${D.join(" | ")}`}function Ff(){const[a,h]=P.useState(zf),[p,D]=P.useState(ec),[k,F]=P.useState(Of),[H,Z]=P.useState(null),[N,te]=P.useState([]),[ce,z]=P.useState([]),[L,Le]=P.useState("normalized"),[me,re]=P.useState(!1),[G,De]=P.useState(!1),[ye,We]=P.useState([]),[xe,ze]=P.useState(""),[X,$e]=P.useState([]),[Ne,ut]=P.useState(""),[be,gt]=P.useState("NDC custom preset"),[he,He]=P.useState(null),[Oe,Ye]=P.useState(""),[b,pe]=P.useState(!1),[R,V]=P.useState([]),[T,m]=P.useState(""),[_,Y]=P.useState([]),[J,le]=P.useState(!1),[ee,de]=P.useState(null),[ae,oe]=P.useState(""),[ke,Ot]=P.useState(Df),[an,Gt]=P.useState(!0),[kt,Dn]=P.useState(!0),[un,ct]=P.useState(!0),[ie,et]=P.useState(""),[Mt,yt]=P.useState([]),[xt,ge]=P.useState(""),[cn,dn]=P.useState(!1),[Jn,Se]=P.useState(""),[Ct,Nt]=P.useState(""),Je=P.useRef(!1);P.useEffect(()=>{const x=document.documentElement,{colors:O}=Mf;x.style.setProperty("--rgb-background",O.backgroundRgb),x.style.setProperty("--rgb-surface-main",O.mainSurfaceRgb),x.style.setProperty("--rgb-surface-horizontal",O.horizontalSurfaceRgb),x.style.setProperty("--rgb-surface-focus",O.focusSurfaceRgb),x.style.setProperty("--rgb-active",O.activeRgb),x.style.setProperty("--rgb-active-text",O.activeTextRgb),x.style.setProperty("--rgb-text-main",O.textMainRgb),x.style.setProperty("--rgb-text-muted",O.textMutedRgb),x.style.setProperty("--rgb-danger",O.dangerRgb),x.style.setProperty("--rgb-scrollbar-track",O.scrollbarTrackRgb),x.style.setProperty("--rgb-scrollbar-thumb",O.scrollbarThumbRgb),x.style.setProperty("--rgb-scrollbar-thumb-hover",O.scrollbarThumbHoverRgb)},[]);const K=x=>{z(O=>[If(x),...O].slice(0,300))};function fn(){let x=0;Se(li[0]);const O=window.setInterval(()=>{x=Math.min(x+1,li.length-1),Se(li[x])},650);return()=>window.clearInterval(O)}P.useEffect(()=>{const x=localStorage.getItem(tc);if(x)try{const O=JSON.parse(x);h(B=>({...B,llmProvider:O.llmProvider==="local"?"local":"openai",model:O.model??B.model,baseUrl:O.baseUrl??B.baseUrl,temperature:O.temperature??B.temperature,maxOutputTokens:O.maxOutputTokens??B.maxOutputTokens}))}catch{}tt(),Yt(),pn()},[]);async function tt(){try{const x=await Ue.loadHistory();te(x.items??[])}catch(x){K(`History load error: ${x instanceof Error?x.message:String(x)}`)}}async function Yt(){try{const O=(await Ue.loadPresets()).presets??[];if($e(O),Je.current)return;const B=O.find(Ie=>Ie.prompt_version===si)??O.find(Ie=>Ie.id==="default-normalizer-v2_0_2");if(!B){Je.current=!0,K(`Preset autoload skipped: ${si} not found.`);return}ut(B.id),He(p),D({systemPrompt:B.systemPrompt,developerPrompt:B.developerPrompt,domainPrompt:B.domainPrompt,schemaNotes:B.schemaNotes??"",fewShotExamples:B.fewShotExamples??""}),Je.current=!0,K(`Preset autoloaded: ${B.name} (${B.prompt_version}).`)}catch(x){K(`Presets load error: ${x instanceof Error?x.message:String(x)}`)}}async function pn(){try{const x=await Ue.listRuns();V(x.items??[])}catch(x){K(`Runs load error: ${x instanceof Error?x.message:String(x)}`)}}function Dt(){localStorage.setItem(tc,JSON.stringify({model:a.model,llmProvider:a.llmProvider,baseUrl:a.baseUrl,temperature:a.temperature,maxOutputTokens:a.maxOutputTokens})),K("Local config saved (without API key).")}async function mn(){re(!0),oe("");try{const x=await Ue.testConnection(a);x.provider==="local"?x.model_found===!0?(ze(`LOCAL OK - ${x.model}`),K(`Local model is available: ${x.model} (catalog size=${x.models_count??"n/a"}).`)):x.model_found===!1?(ze(`LOCAL OK, model not loaded - ${x.model}`),K(`Local server is reachable, but model '${x.model}' is not in loaded catalog. Use 'Load model list' and select one of loaded models.`)):(ze(`LOCAL OK (model list unavailable) - ${x.model}`),K("Local server is reachable, but model catalog could not be verified.")):(ze(`OPENAI OK - ${x.model}`),K(`OpenAI connection ok: ${x.model}`))}catch(x){const O=x instanceof Error?x.message:String(x);ze("Connection error"),oe(`Test connection: ${O}`),K(`Test connection error: ${O}`)}finally{re(!1)}}async function hn(){De(!0);try{const O=(await Ue.listModels(a)).models??[];We(O),O.length>0&&h(B=>B.model&&O.includes(B.model)?B:{...B,model:O[0]}),K(`Model catalog loaded (${a.llmProvider}): ${O.length} items.`)}catch(x){const O=x instanceof Error?x.message:String(x);K(`Load model list error: ${O}`)}finally{De(!1)}}P.useEffect(()=>{We([])},[a.llmProvider,a.baseUrl]);async function c(x){re(!0),oe("");try{const O=await Ue.normalize({connection:a,prompts:p,promptVersion:"normalizer_v2_0_2",query:{userQuestion:k.userQuestion,periodHint:k.periodHint,businessContext:k.businessContext,expectedRoute:k.expectedRoute},saveAsTestCase:x,useMock:b});Z(O),Le("normalized"),K(`Normalize done: trace=${O.trace_id}, validation=${O.validation.passed?"passed":"failed"}`),tt()}catch(O){const B=O instanceof Error?O.message:String(O);oe(`Normalize: ${B}`),K(`Normalize error: ${B}`)}finally{re(!1)}}function E(){const x=X.find(O=>O.id===Ne);if(!x){K("Preset is not selected.");return}He(p),D({systemPrompt:x.systemPrompt,developerPrompt:x.developerPrompt,domainPrompt:x.domainPrompt,schemaNotes:x.schemaNotes??"",fewShotExamples:x.fewShotExamples??""}),K(`Preset loaded: ${x.name}`)}async function ne(){try{await Ue.savePreset({name:be||"NDC preset",prompt_version:"normalizer_v2_0_2",systemPrompt:p.systemPrompt,developerPrompt:p.developerPrompt,domainPrompt:p.domainPrompt,schemaNotes:p.schemaNotes,fewShotExamples:p.fewShotExamples}),K("Preset saved."),await Yt()}catch(x){K(`Preset save error: ${x instanceof Error?x.message:String(x)}`)}}function Ee(){D(ec),K("Prompt panel reset to defaults.")}function Jt(){const x=Af(p,he);Ye(x),K(x)}function Xn(){const x=k.batchQuestionsRaw.split(";").map(O=>O.trim()).filter(Boolean).join(` + +`);x&&(F(O=>({...O,batchQuestionsRaw:x})),K("Batch field formatted: `;` converted to blank-line separators."))}async function Zn(x){try{const B=(await Ue.loadTrace(x)).trace,Ie=B.parsed_normalized_json??null;Z({trace_id:String(B.trace_id??x),ok:!!B.validation_result?.passed,normalized:Ie,route_hint_summary:B.route_hint_summary??(Ie?{route_hint:Ie.route_hint??null,confidence:Ie.confidence?.route_hint??null}:null),raw_model_output:B.raw_model_response??{},validation:B.validation_result??{passed:!1,errors:["validation not found"]},usage:B.usage??{input_tokens:0,output_tokens:0,total_tokens:0},latency_ms:Number(B.latency_ms??0),prompt_version:String(B.prompt_version??"unknown"),schema_version:String(B.schema_version??"unknown")}),Le("raw"),oe(""),K(`Trace opened: ${x}`)}catch(O){const B=O instanceof Error?O.message:String(O);oe(`Trace: ${B}`),K(`Trace open error ${x}: ${B}`)}}async function sl(){try{const x=await Ue.startRun();m(x.run.runId),K(`Run started: ${x.run.runId}`),K("Tip: start run does not execute normalize by itself. Use 'Run eval v2.0.2' button."),await pn()}catch(x){K(`Run start error: ${x instanceof Error?x.message:String(x)}`)}}async function ol(){if(T)try{await Ue.finishRun(T),K(`Run finished: ${T}`),await pn()}catch(x){K(`Run finish error: ${x instanceof Error?x.message:String(x)}`)}}async function wr(){le(!0),oe("");try{K("Starting eval in v2 contour.");const x=k.batchQuestionsRaw.trim()||k.userQuestion.trim();if(!x)throw new Error("Fill batch field or Raw user question first.");const O=await Ue.runEval({connection:a,prompts:p,promptVersion:"normalizer_v2_0_2",mode:"single-pass-strict",rawQuestions:x,useMock:b});de(O.report),K("Eval v2.0.2 run finished.");const B=O.report;if(B.run_id&&K(`Eval run id: ${B.run_id}`),B.metrics){const Ie=B.metrics;K(`Eval metrics v2.0.2: schema=${Ie.schema_validation_pass_rate??"n/a"}%, route_accuracy=${Ie.route_resolution_accuracy??"n/a"}%, no_route_precision=${Ie.no_route_precision??"n/a"}%, state_consistency=${Ie.execution_state_consistency_rate??"n/a"}%`)}await tt()}catch(x){const O=x instanceof Error?x.message:String(x);O.includes("Legacy eval runner supports normalized_query_v1 only")?(de({status:"plan_only",prompt_version:"normalizer_v2",reason:"backend eval runner is still legacy-v1 only",plan_file:"reports/v2_pilot_eval_plan.md",next_steps:["run cheap mock sanity for schema/fragment/scope","run small real batch (10-15 messages, temperature=0)","run challenge-30 replay with v2 metrics"]}),K("Backend is legacy-only for eval right now. Showing v2 pilot plan.")):(oe(`Eval: ${O}`),K(`Eval run error: ${O}`))}finally{le(!1)}}async function il(){try{const x=JSON.stringify(ee??{},null,2);await navigator.clipboard.writeText(x),K("Eval report copied to clipboard.")}catch(x){K(`Eval report copy error: ${x instanceof Error?x.message:String(x)}`)}}function In(){et(""),yt([]),ge(""),Se(""),Nt(""),K("Assistant session reset.")}async function jr(){const x=xt.trim();if(!x)return;dn(!0),Nt(""),ge(""),yt(B=>[...B,{message_id:`local-${Date.now()}`,session_id:ie||"pending",role:"user",text:x,reply_type:null,created_at:new Date().toISOString(),trace_id:null,debug:null}]);const O=fn();try{const B=await Ue.sendAssistantMessage({connection:a,prompts:p,userMessage:x,sessionId:ie||void 0,promptVersion:nc,context:{periodHint:k.periodHint,businessContext:k.businessContext},useMock:b});et(B.session_id),yt(B.conversation),Se("Ответ готов"),K(`Assistant reply received: trace=${B.debug.trace_id}`)}catch(B){const Ie=B instanceof Error?B.message:String(B);Nt(Ie),Se("Ошибка ассистента"),K(`Assistant error: ${Ie}`)}finally{O(),dn(!1)}}return P.useEffect(()=>{if(!T){Y([]);return}Ue.runTrace(T).then(x=>Y(x.items)).catch(x=>K(`Run trace error: ${x instanceof Error?x.message:String(x)}`))},[T]),o.jsxs("main",{className:`app-root ${ke==="autoruns"?"app-root-autoruns":""}`,children:[o.jsxs("header",{className:"app-topbar",children:[o.jsxs("div",{className:"mode-switch-row",children:[o.jsx("button",{type:"button",className:ke==="assistant"?"tab active":"tab",onClick:()=>Ot("assistant"),children:"Ассистент"}),o.jsx("button",{type:"button",className:ke==="decomposition"?"tab active":"tab",onClick:()=>Ot("decomposition"),children:"Декомпозиция"}),o.jsx("button",{type:"button",className:ke==="autoruns"?"tab active":"tab",onClick:()=>Ot("autoruns"),children:"История автопрогонов"})]}),ke==="autoruns"?o.jsxs("div",{className:"mode-switch-row mode-switch-row-right",children:[o.jsx("button",{type:"button",className:an?"tab active":"tab",onClick:()=>Gt(x=>!x),children:"Режим ассистента"}),o.jsx("button",{type:"button",className:kt?"tab active":"tab",onClick:()=>Dn(x=>!x),children:"Режим декомпозиции"}),o.jsx("button",{type:"button",className:un?"tab active":"tab",onClick:()=>ct(x=>!x),children:"Прогресс/регресс"})]}):null]}),ke==="assistant"?o.jsxs("div",{className:"layout-grid",children:[o.jsx(Zu,{value:a,modelOptions:ye,modelsBusy:G,onChange:h,onReloadModels:hn,onSaveLocalConfig:Dt,onTestConnection:mn,lastStatus:xe,busy:me||cn}),o.jsx(bu,{value:p,onChange:D,presets:X,selectedPresetId:Ne,onSelectPreset:ut,onLoadPreset:E,onSavePreset:ne,onResetDefaults:Ee,onDiffPrevious:Jt,presetName:be,onPresetNameChange:gt,diffSummary:Oe}),o.jsx(kf,{sessionId:ie,conversation:Mt,inputValue:xt,onInputChange:ge,periodHint:k.periodHint,onPeriodHintChange:x=>F(O=>({...O,periodHint:x})),businessContext:k.businessContext,onBusinessContextChange:x=>F(O=>({...O,businessContext:x})),useMock:b,onUseMockChange:pe,onSend:jr,onClear:In,busy:cn,statusText:Jn,errorMessage:Ct})]}):ke==="decomposition"?o.jsxs("div",{className:"layout-grid",children:[o.jsx(Zu,{value:a,modelOptions:ye,modelsBusy:G,onChange:h,onReloadModels:hn,onSaveLocalConfig:Dt,onTestConnection:mn,lastStatus:xe,busy:me}),o.jsx(bu,{value:p,onChange:D,presets:X,selectedPresetId:Ne,onSelectPreset:ut,onLoadPreset:E,onSavePreset:ne,onResetDefaults:Ee,onDiffPrevious:Jt,presetName:be,onPresetNameChange:gt,diffSummary:Oe}),o.jsx(Tf,{value:k,onChange:F,onApplyBatchFormat:Xn,onNormalize:c,busy:me,useMock:b,onUseMockChange:pe,errorMessage:ae}),o.jsx(Rf,{tab:L,onTabChange:Le,result:H,appLogs:ce}),o.jsx(Nf,{result:H}),o.jsx(Cf,{items:N,onRefresh:tt,onOpenTrace:Zn}),o.jsx(Lf,{runs:R,selectedRunId:T,onSelectRun:m,onStartRun:sl,onFinishRun:ol,onRefreshRuns:pn,onRunEval:wr,onCopyEvalReport:il,evalBusy:J,traceItems:_,evalReport:ee})]}):o.jsx("div",{className:"layout-grid layout-grid-autoruns",children:o.jsx(hf,{connection:a,prompts:p,assistantPromptVersion:nc,decompositionPromptVersion:si,showAssistantMode:an,showDecompositionMode:kt,showProgressMode:un,onLog:K})})]})}af.createRoot(document.getElementById("root")).render(o.jsx(ef.StrictMode,{children:o.jsx(Ff,{})})); diff --git a/llm_normalizer/frontend/dist/assets/index-BMWPMdQA.css b/llm_normalizer/frontend/dist/assets/index-BMWPMdQA.css deleted file mode 100644 index ffda417..0000000 --- a/llm_normalizer/frontend/dist/assets/index-BMWPMdQA.css +++ /dev/null @@ -1 +0,0 @@ -@import"https://fonts.googleapis.com/css2?family=Manrope:wght@400;600;700;800&family=Space+Grotesk:wght@500;700&display=swap";:root{--bg-main: #060a08;--bg-soft: #0e1511;--bg-panel: rgba(15, 24, 19, .85);--bg-panel-accent: rgba(38, 62, 45, .45);--line: rgba(143, 255, 173, .3);--line-strong: rgba(143, 255, 173, .72);--text-main: #f3ffee;--text-muted: #9bb8a5;--lime-main: #8fffad;--lime-press: #59db83;--danger: #ff7e7e;--radius-lg: 20px;--radius-md: 14px;--shadow: 0 16px 44px rgba(0, 0, 0, .35)}*{box-sizing:border-box}html,body,#root{margin:0;min-height:100%;font-family:Manrope,Segoe UI,sans-serif;background:radial-gradient(circle at 15% -10%,#1f3b2f 0%,transparent 40%),radial-gradient(circle at 90% 10%,#1d2f24 0%,transparent 35%),linear-gradient(165deg,var(--bg-main) 0%,#070d0a 100%);color:var(--text-main)}.app-root{max-width:1720px;margin:0 auto;padding:28px 24px 42px}.hero{margin-bottom:20px;padding:20px 22px;border:1px solid var(--line);border-radius:var(--radius-lg);background:linear-gradient(145deg,#111d16f0,#0a100deb);box-shadow:var(--shadow);animation:rise .5s ease-out}.hero h1{font-family:Space Grotesk,Manrope,sans-serif;margin:0 0 8px;font-size:clamp(1.5rem,3vw,2.3rem);letter-spacing:.04em}.hero p{margin:0;color:var(--text-muted)}.layout-grid{display:grid;grid-template-columns:repeat(12,1fr);gap:16px}.mode-switch-row{display:flex;gap:8px;margin:0 0 14px}.panel-frame{grid-column:span 12;border:1px solid var(--line);border-radius:var(--radius-lg);background:linear-gradient(165deg,var(--bg-panel),rgba(10,17,13,.88));overflow:hidden;box-shadow:var(--shadow);animation:rise .4s ease-out}.panel-header{display:flex;align-items:flex-start;justify-content:space-between;gap:14px;padding:14px 18px 10px;border-bottom:1px solid var(--line);background:linear-gradient(90deg,var(--bg-panel-accent),transparent 70%)}.panel-header h2{margin:0;font-size:1.02rem;letter-spacing:.02em}.panel-header p{margin:6px 0 0;font-size:.85rem;color:var(--text-muted)}.panel-body{padding:14px 18px 18px}.status-chip{border:1px solid var(--line);border-radius:999px;padding:4px 10px;color:var(--lime-main);font-size:.78rem}.assistant-panel-actions{display:flex;align-items:center;justify-content:flex-end;flex-wrap:wrap;gap:8px}.assistant-copy-btn{background:transparent;border-color:var(--line);color:var(--text-main);box-shadow:none;transform:none}.assistant-copy-btn:hover{background:#8fffad24;filter:none;box-shadow:none;transform:none}.assistant-copy-feedback{font-size:.76rem;color:var(--text-muted)}.assistant-copy-feedback.success{color:var(--lime-main)}.assistant-copy-feedback.error{color:var(--danger)}input,select,textarea,button{font-family:Manrope,sans-serif}label{display:flex;flex-direction:column;gap:6px;color:var(--text-muted);font-size:.84rem}input,select,textarea{border:1px solid rgba(170,255,194,.26);border-radius:var(--radius-md);background:#070d0aeb;color:var(--text-main);padding:10px 12px;outline:none;transition:border-color .18s ease,box-shadow .18s ease}input:focus,select:focus,textarea:focus{border-color:var(--line-strong);box-shadow:0 0 0 3px #8fffad33}textarea{resize:vertical;min-height:86px}button{border:1px solid rgba(176,255,199,.35);border-radius:999px;background:linear-gradient(180deg,#8fffad,#62e286);color:#08100a;font-weight:700;font-size:.83rem;letter-spacing:.02em;cursor:pointer;padding:9px 14px;transition:transform .16s ease,filter .2s ease,box-shadow .18s ease}button:hover{filter:brightness(1.04);transform:translateY(-1px);box-shadow:0 8px 20px #7fffaa52}button:disabled{opacity:.5;cursor:not-allowed;transform:none}.button-row{display:flex;flex-wrap:wrap;align-items:center;gap:10px;margin-top:12px}.checkbox-row{flex-direction:row;align-items:center;gap:8px;color:var(--text-main)}.grid-two{display:grid;grid-template-columns:repeat(2,minmax(0,1fr));gap:12px}.full-width{grid-column:1 / -1}.tab-row{display:flex;flex-wrap:wrap;gap:8px;margin-bottom:12px}.tab{background:transparent;color:var(--text-main);border:1px solid var(--line)}.tab.active{border-color:var(--line-strong);background:linear-gradient(180deg,#8fffad40,#8fffad1a)}.assistant-chat-list{max-height:420px;overflow:auto;display:grid;gap:10px;padding:4px;border:1px solid rgba(156,255,189,.2);border-radius:var(--radius-md);background:#060a08a6}.assistant-empty{padding:18px;text-align:center}.assistant-msg{border:1px solid rgba(157,255,190,.2);border-radius:12px;background:#0b120ebd;padding:10px}.assistant-msg.user{border-color:#6ec6ff61;background:#0d1418c2}.assistant-msg.assistant{border-color:#9dffbe40}.assistant-msg-head{display:flex;justify-content:space-between;gap:8px;margin-bottom:6px;font-size:.78rem;color:var(--text-muted)}.assistant-msg-body{white-space:pre-wrap;line-height:1.45;font-size:.9rem}.assistant-trace{margin-top:6px;color:var(--text-muted);font-size:.75rem}.assistant-debug{margin-top:8px}.assistant-debug summary{cursor:pointer;color:var(--lime-main);font-size:.8rem}.assistant-compose{margin-top:12px;display:grid;gap:10px}.json-view{margin:0;width:100%;min-height:180px;max-height:420px;overflow:auto;background:#050806;border:1px solid rgba(156,255,189,.2);border-radius:var(--radius-md);padding:12px;color:#d6ffdf;font-family:JetBrains Mono,Consolas,monospace;font-size:.78rem;line-height:1.45}.metrics-grid{display:grid;grid-template-columns:repeat(5,minmax(0,1fr));gap:10px}.metrics-grid div{background:#0b120eb3;border:1px solid rgba(157,255,190,.24);border-radius:12px;padding:10px;display:flex;flex-direction:column;gap:4px}.metrics-grid span{color:var(--text-muted);font-size:.75rem}.metrics-grid strong{font-size:.84rem;color:var(--lime-main)}.history-list{display:grid;gap:8px;max-height:340px;overflow:auto}.history-item{width:100%;text-align:left;border-radius:12px;border:1px solid rgba(157,255,190,.24);background:#0b120ec7;color:var(--text-main);padding:10px}.history-item p{margin:8px 0;color:var(--text-muted);font-size:.82rem}.history-item.selected{border-color:var(--line-strong)}.history-row{display:flex;justify-content:space-between;gap:8px;font-size:.76rem;color:var(--text-muted)}.runtime-grid{display:grid;grid-template-columns:1.2fr 1fr;gap:12px}.runtime-runs{max-height:360px;overflow:auto;display:grid;gap:8px}.eval-report-wrap{position:relative}.copy-cube-button{position:absolute;right:10px;bottom:10px;width:34px;height:34px;border-radius:10px;padding:0;min-width:34px;display:grid;place-items:center;font-size:.92rem;line-height:1}.muted{color:var(--text-muted)}.diff-summary{margin-top:10px;font-size:.82rem;color:var(--lime-main)}.error-text{margin-top:10px;color:var(--danger);font-size:.84rem}.autoruns-columns{display:grid;gap:12px}.autoruns-col{border:1px solid rgba(157,255,190,.2);border-radius:14px;background:#080d0ab8;padding:12px;min-height:220px}.autoruns-col h3{margin:0 0 10px;font-size:.95rem}.autoruns-col h4{margin:12px 0 8px;font-size:.82rem;color:var(--text-muted)}.autoruns-form-grid{display:grid;grid-template-columns:repeat(2,minmax(0,1fr));gap:8px}.autoruns-meta-list{display:grid;gap:8px}.autoruns-meta-list>div{display:flex;justify-content:space-between;gap:8px;border:1px solid rgba(157,255,190,.14);border-radius:10px;background:#0a120da6;padding:8px 9px;font-size:.79rem}.autoruns-meta-list span{color:var(--text-muted)}.autoruns-prompt-details summary{cursor:pointer;color:var(--lime-main);font-size:.8rem;margin-bottom:8px}.autoruns-prompt-details textarea{min-height:68px}.autoruns-stats-grid{display:grid;grid-template-columns:repeat(2,minmax(0,1fr));gap:8px;margin-bottom:10px}.autoruns-stats-grid>div{border:1px solid rgba(157,255,190,.2);border-radius:10px;background:#0a120db3;padding:8px;display:grid;gap:3px}.autoruns-stats-grid span{color:var(--text-muted);font-size:.74rem}.autoruns-stats-grid strong{color:var(--lime-main);font-size:.84rem}.autoruns-run-list{display:grid;gap:8px;max-height:760px;overflow:auto;padding-right:2px}.autoruns-run-item{width:100%;text-align:left;border-radius:12px;border:1px solid rgba(157,255,190,.23);background:#0b120ebf;color:var(--text-main);padding:10px;display:grid;gap:5px}.autoruns-run-item.selected{border-color:var(--line-strong)}.autoruns-run-head,.autoruns-run-foot{display:flex;justify-content:space-between;gap:8px;font-size:.76rem}.autoruns-run-meta{color:var(--text-muted);font-size:.75rem;word-break:break-word}.autoruns-dialog-toolbar{display:grid;grid-template-columns:repeat(2,minmax(0,1fr));gap:8px}.autoruns-case-list{margin-top:8px;display:grid;gap:6px;max-height:170px;overflow:auto}.autoruns-case-item{width:100%;text-align:left;border-radius:10px;border:1px solid rgba(157,255,190,.22);background:#090e0bb8;color:var(--text-main);padding:7px 8px;display:flex;justify-content:space-between;gap:6px;font-size:.76rem}.autoruns-case-item.selected{border-color:var(--line-strong)}.autoruns-dialog-view{margin-top:10px;border:1px solid rgba(157,255,190,.2);border-radius:12px;background:#050806b3;padding:10px;max-height:570px;overflow:auto;display:grid;gap:8px}.autoruns-msg{border:1px solid rgba(157,255,190,.22);border-radius:12px;background:#0b120ecc;padding:8px 10px;display:grid;gap:6px}.autoruns-msg header,.autoruns-msg footer{display:flex;justify-content:space-between;gap:8px;font-size:.74rem;color:var(--text-muted)}.autoruns-msg p{margin:0;white-space:pre-wrap;line-height:1.35;font-size:.84rem}.autoruns-msg.assistant{margin-right:12%}.autoruns-msg.user{margin-left:12%;border-color:#5fb3ff59;background:#0a1218bf}.autoruns-decomposition-list{margin:0;padding-left:18px;display:grid;gap:7px;font-size:.8rem}.autoruns-coverage-list{display:grid;gap:8px}.autoruns-coverage-item{border:1px solid rgba(157,255,190,.2);border-radius:10px;background:#0b120ead;padding:8px}.autoruns-coverage-head{display:flex;justify-content:space-between;gap:8px;font-size:.76rem;margin-bottom:5px}.autoruns-coverage-head span{color:var(--text-muted)}.autoruns-coverage-bar{height:7px;border-radius:999px;background:#9dffbe24;overflow:hidden}.autoruns-coverage-bar>div{height:100%;border-radius:999px;background:linear-gradient(90deg,#6ee0ff,#8fffad)}@media(max-width:1200px){.metrics-grid{grid-template-columns:repeat(3,minmax(0,1fr))}.autoruns-columns{grid-template-columns:1fr!important}}@media(max-width:920px){.grid-two,.runtime-grid{grid-template-columns:1fr}.metrics-grid{grid-template-columns:repeat(2,minmax(0,1fr))}.autoruns-form-grid,.autoruns-dialog-toolbar,.autoruns-stats-grid{grid-template-columns:1fr}}@media(max-width:640px){.app-root{padding:18px 12px 24px}.metrics-grid{grid-template-columns:1fr}}@keyframes rise{0%{opacity:0;transform:translateY(8px)}to{opacity:1;transform:translateY(0)}} diff --git a/llm_normalizer/frontend/dist/assets/index-D6Y_lHrc.js b/llm_normalizer/frontend/dist/assets/index-D6Y_lHrc.js deleted file mode 100644 index 4769f75..0000000 --- a/llm_normalizer/frontend/dist/assets/index-D6Y_lHrc.js +++ /dev/null @@ -1,12 +0,0 @@ -(function(){const g=document.createElement("link").relList;if(g&&g.supports&&g.supports("modulepreload"))return;for(const k of document.querySelectorAll('link[rel="modulepreload"]'))O(k);new MutationObserver(k=>{for(const E of k)if(E.type==="childList")for(const M of E.addedNodes)M.tagName==="LINK"&&M.rel==="modulepreload"&&O(M)}).observe(document,{childList:!0,subtree:!0});function p(k){const E={};return k.integrity&&(E.integrity=k.integrity),k.referrerPolicy&&(E.referrerPolicy=k.referrerPolicy),k.crossOrigin==="use-credentials"?E.credentials="include":k.crossOrigin==="anonymous"?E.credentials="omit":E.credentials="same-origin",E}function O(k){if(k.ep)return;k.ep=!0;const E=p(k);fetch(k.href,E)}})();function ec(u){return u&&u.__esModule&&Object.prototype.hasOwnProperty.call(u,"default")?u.default:u}var qs={exports:{}},qr={},Gs={exports:{}},ee={};var Ma;function qd(){if(Ma)return ee;Ma=1;var u=Symbol.for("react.element"),g=Symbol.for("react.portal"),p=Symbol.for("react.fragment"),O=Symbol.for("react.strict_mode"),k=Symbol.for("react.profiler"),E=Symbol.for("react.provider"),M=Symbol.for("react.context"),$=Symbol.for("react.forward_ref"),H=Symbol.for("react.suspense"),te=Symbol.for("react.memo"),oe=Symbol.for("react.lazy"),N=Symbol.iterator;function F(m){return m===null||typeof m!="object"?null:(m=N&&m[N]||m["@@iterator"],typeof m=="function"?m:null)}var ce={isMounted:function(){return!1},enqueueForceUpdate:function(){},enqueueReplaceState:function(){},enqueueSetState:function(){}},he=Object.assign,b={};function J(m,c,C){this.props=m,this.context=c,this.refs=b,this.updater=C||ce}J.prototype.isReactComponent={},J.prototype.setState=function(m,c){if(typeof m!="object"&&typeof m!="function"&&m!=null)throw Error("setState(...): takes an object of state variables to update or a function which returns an object of state variables.");this.updater.enqueueSetState(this,m,c,"setState")},J.prototype.forceUpdate=function(m){this.updater.enqueueForceUpdate(this,m,"forceUpdate")};function De(){}De.prototype=J.prototype;function Pe(m,c,C){this.props=m,this.context=c,this.refs=b,this.updater=C||ce}var Te=Pe.prototype=new De;Te.constructor=Pe,he(Te,J.prototype),Te.isPureReactComponent=!0;var ve=Array.isArray,xe=Object.prototype.hasOwnProperty,G={current:null},ke={key:!0,ref:!0,__self:!0,__source:!0};function ye(m,c,C){var I,Y={},Z=null,ue=null;if(c!=null)for(I in c.ref!==void 0&&(ue=c.ref),c.key!==void 0&&(Z=""+c.key),c)xe.call(c,I)&&!ke.hasOwnProperty(I)&&(Y[I]=c[I]);var re=arguments.length-2;if(re===1)Y.children=C;else if(1>>1,c=P[m];if(0>>1;mk(Y,T))Zk(ue,Y)?(P[m]=ue,P[Z]=T,m=Z):(P[m]=Y,P[I]=T,m=I);else if(Zk(ue,T))P[m]=ue,P[Z]=T,m=Z;else break e}}return B}function k(P,B){var T=P.sortIndex-B.sortIndex;return T!==0?T:P.id-B.id}if(typeof performance=="object"&&typeof performance.now=="function"){var E=performance;u.unstable_now=function(){return E.now()}}else{var M=Date,$=M.now();u.unstable_now=function(){return M.now()-$}}var H=[],te=[],oe=1,N=null,F=3,ce=!1,he=!1,b=!1,J=typeof setTimeout=="function"?setTimeout:null,De=typeof clearTimeout=="function"?clearTimeout:null,Pe=typeof setImmediate<"u"?setImmediate:null;typeof navigator<"u"&&navigator.scheduling!==void 0&&navigator.scheduling.isInputPending!==void 0&&navigator.scheduling.isInputPending.bind(navigator.scheduling);function Te(P){for(var B=p(te);B!==null;){if(B.callback===null)O(te);else if(B.startTime<=P)O(te),B.sortIndex=B.expirationTime,g(H,B);else break;B=p(te)}}function ve(P){if(b=!1,Te(P),!he)if(p(H)!==null)he=!0,de(xe);else{var B=p(te);B!==null&&se(ve,B.startTime-P)}}function xe(P,B){he=!1,b&&(b=!1,De(ye),ye=-1),ce=!0;var T=F;try{for(Te(B),N=p(H);N!==null&&(!(N.expirationTime>B)||P&&!qe());){var m=N.callback;if(typeof m=="function"){N.callback=null,F=N.priorityLevel;var c=m(N.expirationTime<=B);B=u.unstable_now(),typeof c=="function"?N.callback=c:N===p(H)&&O(H),Te(B)}else O(H);N=p(H)}if(N!==null)var C=!0;else{var I=p(te);I!==null&&se(ve,I.startTime-B),C=!1}return C}finally{N=null,F=T,ce=!1}}var G=!1,ke=null,ye=-1,Ae=5,Xe=-1;function qe(){return!(u.unstable_now()-XeP||125m?(P.sortIndex=T,g(te,P),p(H)===null&&P===p(te)&&(b?(De(ye),ye=-1):b=!0,se(ve,T-m))):(P.sortIndex=c,g(H,P),he||ce||(he=!0,de(xe))),P},u.unstable_shouldYield=qe,u.unstable_wrapCallback=function(P){var B=F;return function(){var T=F;F=B;try{return P.apply(this,arguments)}finally{F=T}}}})(bs)),bs}var $a;function ef(){return $a||($a=1,Zs.exports=bd()),Zs.exports}var Va;function tf(){if(Va)return ot;Va=1;var u=ri(),g=ef();function p(e){for(var t="https://reactjs.org/docs/error-decoder.html?invariant="+e,n=1;n"u"||typeof window.document>"u"||typeof window.document.createElement>"u"),H=Object.prototype.hasOwnProperty,te=/^[:A-Z_a-z\u00C0-\u00D6\u00D8-\u00F6\u00F8-\u02FF\u0370-\u037D\u037F-\u1FFF\u200C-\u200D\u2070-\u218F\u2C00-\u2FEF\u3001-\uD7FF\uF900-\uFDCF\uFDF0-\uFFFD][:A-Z_a-z\u00C0-\u00D6\u00D8-\u00F6\u00F8-\u02FF\u0370-\u037D\u037F-\u1FFF\u200C-\u200D\u2070-\u218F\u2C00-\u2FEF\u3001-\uD7FF\uF900-\uFDCF\uFDF0-\uFFFD\-.0-9\u00B7\u0300-\u036F\u203F-\u2040]*$/,oe={},N={};function F(e){return H.call(N,e)?!0:H.call(oe,e)?!1:te.test(e)?N[e]=!0:(oe[e]=!0,!1)}function ce(e,t,n,r){if(n!==null&&n.type===0)return!1;switch(typeof t){case"function":case"symbol":return!0;case"boolean":return r?!1:n!==null?!n.acceptsBooleans:(e=e.toLowerCase().slice(0,5),e!=="data-"&&e!=="aria-");default:return!1}}function he(e,t,n,r){if(t===null||typeof t>"u"||ce(e,t,n,r))return!0;if(r)return!1;if(n!==null)switch(n.type){case 3:return!t;case 4:return t===!1;case 5:return isNaN(t);case 6:return isNaN(t)||1>t}return!1}function b(e,t,n,r,l,o,i){this.acceptsBooleans=t===2||t===3||t===4,this.attributeName=r,this.attributeNamespace=l,this.mustUseProperty=n,this.propertyName=e,this.type=t,this.sanitizeURL=o,this.removeEmptyString=i}var J={};"children dangerouslySetInnerHTML defaultValue defaultChecked innerHTML suppressContentEditableWarning suppressHydrationWarning style".split(" ").forEach(function(e){J[e]=new b(e,0,!1,e,null,!1,!1)}),[["acceptCharset","accept-charset"],["className","class"],["htmlFor","for"],["httpEquiv","http-equiv"]].forEach(function(e){var t=e[0];J[t]=new b(t,1,!1,e[1],null,!1,!1)}),["contentEditable","draggable","spellCheck","value"].forEach(function(e){J[e]=new b(e,2,!1,e.toLowerCase(),null,!1,!1)}),["autoReverse","externalResourcesRequired","focusable","preserveAlpha"].forEach(function(e){J[e]=new b(e,2,!1,e,null,!1,!1)}),"allowFullScreen async autoFocus autoPlay controls default defer disabled disablePictureInPicture disableRemotePlayback formNoValidate hidden loop noModule noValidate open playsInline readOnly required reversed scoped seamless itemScope".split(" ").forEach(function(e){J[e]=new b(e,3,!1,e.toLowerCase(),null,!1,!1)}),["checked","multiple","muted","selected"].forEach(function(e){J[e]=new b(e,3,!0,e,null,!1,!1)}),["capture","download"].forEach(function(e){J[e]=new b(e,4,!1,e,null,!1,!1)}),["cols","rows","size","span"].forEach(function(e){J[e]=new b(e,6,!1,e,null,!1,!1)}),["rowSpan","start"].forEach(function(e){J[e]=new b(e,5,!1,e.toLowerCase(),null,!1,!1)});var De=/[\-:]([a-z])/g;function Pe(e){return e[1].toUpperCase()}"accent-height alignment-baseline arabic-form baseline-shift cap-height clip-path clip-rule color-interpolation color-interpolation-filters color-profile color-rendering dominant-baseline enable-background fill-opacity fill-rule flood-color flood-opacity font-family font-size font-size-adjust font-stretch font-style font-variant font-weight glyph-name glyph-orientation-horizontal glyph-orientation-vertical horiz-adv-x horiz-origin-x image-rendering letter-spacing lighting-color marker-end marker-mid marker-start overline-position overline-thickness paint-order panose-1 pointer-events rendering-intent shape-rendering stop-color stop-opacity strikethrough-position strikethrough-thickness stroke-dasharray stroke-dashoffset stroke-linecap stroke-linejoin stroke-miterlimit stroke-opacity stroke-width text-anchor text-decoration text-rendering underline-position underline-thickness unicode-bidi unicode-range units-per-em v-alphabetic v-hanging v-ideographic v-mathematical vector-effect vert-adv-y vert-origin-x vert-origin-y word-spacing writing-mode xmlns:xlink x-height".split(" ").forEach(function(e){var t=e.replace(De,Pe);J[t]=new b(t,1,!1,e,null,!1,!1)}),"xlink:actuate xlink:arcrole xlink:role xlink:show xlink:title xlink:type".split(" ").forEach(function(e){var t=e.replace(De,Pe);J[t]=new b(t,1,!1,e,"http://www.w3.org/1999/xlink",!1,!1)}),["xml:base","xml:lang","xml:space"].forEach(function(e){var t=e.replace(De,Pe);J[t]=new b(t,1,!1,e,"http://www.w3.org/XML/1998/namespace",!1,!1)}),["tabIndex","crossOrigin"].forEach(function(e){J[e]=new b(e,1,!1,e.toLowerCase(),null,!1,!1)}),J.xlinkHref=new b("xlinkHref",1,!1,"xlink:href","http://www.w3.org/1999/xlink",!0,!1),["src","href","action","formAction"].forEach(function(e){J[e]=new b(e,1,!1,e.toLowerCase(),null,!0,!0)});function Te(e,t,n,r){var l=J.hasOwnProperty(t)?J[t]:null;(l!==null?l.type!==0:r||!(2a||l[i]!==o[a]){var d=` -`+l[i].replace(" at new "," at ");return e.displayName&&d.includes("")&&(d=d.replace("",e.displayName)),d}while(1<=i&&0<=a);break}}}finally{C=!1,Error.prepareStackTrace=n}return(e=e?e.displayName||e.name:"")?c(e):""}function Y(e){switch(e.tag){case 5:return c(e.type);case 16:return c("Lazy");case 13:return c("Suspense");case 19:return c("SuspenseList");case 0:case 2:case 15:return e=I(e.type,!1),e;case 11:return e=I(e.type.render,!1),e;case 1:return e=I(e.type,!0),e;default:return""}}function Z(e){if(e==null)return null;if(typeof e=="function")return e.displayName||e.name||null;if(typeof e=="string")return e;switch(e){case ke:return"Fragment";case G:return"Portal";case Ae:return"Profiler";case ye:return"StrictMode";case je:return"Suspense";case He:return"SuspenseList"}if(typeof e=="object")switch(e.$$typeof){case qe:return(e.displayName||"Context")+".Consumer";case Xe:return(e._context.displayName||"Context")+".Provider";case be:var t=e.render;return e=e.displayName,e||(e=t.displayName||t.name||"",e=e!==""?"ForwardRef("+e+")":"ForwardRef"),e;case Ge:return t=e.displayName||null,t!==null?t:Z(e.type)||"Memo";case de:t=e._payload,e=e._init;try{return Z(e(t))}catch{}}return null}function ue(e){var t=e.type;switch(e.tag){case 24:return"Cache";case 9:return(t.displayName||"Context")+".Consumer";case 10:return(t._context.displayName||"Context")+".Provider";case 18:return"DehydratedFragment";case 11:return e=t.render,e=e.displayName||e.name||"",t.displayName||(e!==""?"ForwardRef("+e+")":"ForwardRef");case 7:return"Fragment";case 5:return t;case 4:return"Portal";case 3:return"Root";case 6:return"Text";case 16:return Z(t);case 8:return t===ye?"StrictMode":"Mode";case 22:return"Offscreen";case 12:return"Profiler";case 21:return"Scope";case 13:return"Suspense";case 19:return"SuspenseList";case 25:return"TracingMarker";case 1:case 0:case 17:case 2:case 14:case 15:if(typeof t=="function")return t.displayName||t.name||null;if(typeof t=="string")return t}return null}function re(e){switch(typeof e){case"boolean":case"number":case"string":case"undefined":return e;case"object":return e;default:return""}}function ne(e){var t=e.type;return(e=e.nodeName)&&e.toLowerCase()==="input"&&(t==="checkbox"||t==="radio")}function Re(e){var t=ne(e)?"checked":"value",n=Object.getOwnPropertyDescriptor(e.constructor.prototype,t),r=""+e[t];if(!e.hasOwnProperty(t)&&typeof n<"u"&&typeof n.get=="function"&&typeof n.set=="function"){var l=n.get,o=n.set;return Object.defineProperty(e,t,{configurable:!0,get:function(){return l.call(this)},set:function(i){r=""+i,o.call(this,i)}}),Object.defineProperty(e,t,{enumerable:n.enumerable}),{getValue:function(){return r},setValue:function(i){r=""+i},stopTracking:function(){e._valueTracker=null,delete e[t]}}}}function $t(e){e._valueTracker||(e._valueTracker=Re(e))}function Rn(e){if(!e)return!1;var t=e._valueTracker;if(!t)return!0;var n=t.getValue(),r="";return e&&(r=ne(e)?e.checked?"true":"false":e.value),e=r,e!==n?(t.setValue(e),!0):!1}function fn(e){if(e=e||(typeof document<"u"?document:void 0),typeof e>"u")return null;try{return e.activeElement||e.body}catch{return e.body}}function or(e,t){var n=t.checked;return T({},t,{defaultChecked:void 0,defaultValue:void 0,value:void 0,checked:n??e._wrapperState.initialChecked})}function zn(e,t){var n=t.defaultValue==null?"":t.defaultValue,r=t.checked!=null?t.checked:t.defaultChecked;n=re(t.value!=null?t.value:n),e._wrapperState={initialChecked:r,initialValue:n,controlled:t.type==="checkbox"||t.type==="radio"?t.checked!=null:t.value!=null}}function sr(e,t){t=t.checked,t!=null&&Te(e,"checked",t,!1)}function pn(e,t){sr(e,t);var n=re(t.value),r=t.type;if(n!=null)r==="number"?(n===0&&e.value===""||e.value!=n)&&(e.value=""+n):e.value!==""+n&&(e.value=""+n);else if(r==="submit"||r==="reset"){e.removeAttribute("value");return}t.hasOwnProperty("value")?Ln(e,t.type,n):t.hasOwnProperty("defaultValue")&&Ln(e,t.type,re(t.defaultValue)),t.checked==null&&t.defaultChecked!=null&&(e.defaultChecked=!!t.defaultChecked)}function ir(e,t,n){if(t.hasOwnProperty("value")||t.hasOwnProperty("defaultValue")){var r=t.type;if(!(r!=="submit"&&r!=="reset"||t.value!==void 0&&t.value!==null))return;t=""+e._wrapperState.initialValue,n||t===e.value||(e.value=t),e.defaultValue=t}n=e.name,n!==""&&(e.name=""),e.defaultChecked=!!e._wrapperState.initialChecked,n!==""&&(e.name=n)}function Ln(e,t,n){(t!=="number"||fn(e.ownerDocument)!==e)&&(n==null?e.defaultValue=""+e._wrapperState.initialValue:e.defaultValue!==""+n&&(e.defaultValue=""+n))}var mn=Array.isArray;function dt(e,t,n,r){if(e=e.options,t){t={};for(var l=0;l"+t.valueOf().toString()+"",t=hn.firstChild;e.firstChild;)e.removeChild(e.firstChild);for(;t.firstChild;)e.appendChild(t.firstChild)}});function Ht(e,t){if(t){var n=e.firstChild;if(n&&n===e.lastChild&&n.nodeType===3){n.nodeValue=t;return}}e.textContent=t}var Bt={animationIterationCount:!0,aspectRatio:!0,borderImageOutset:!0,borderImageSlice:!0,borderImageWidth:!0,boxFlex:!0,boxFlexGroup:!0,boxOrdinalGroup:!0,columnCount:!0,columns:!0,flex:!0,flexGrow:!0,flexPositive:!0,flexShrink:!0,flexNegative:!0,flexOrder:!0,gridArea:!0,gridRow:!0,gridRowEnd:!0,gridRowSpan:!0,gridRowStart:!0,gridColumn:!0,gridColumnEnd:!0,gridColumnSpan:!0,gridColumnStart:!0,fontWeight:!0,lineClamp:!0,lineHeight:!0,opacity:!0,order:!0,orphans:!0,tabSize:!0,widows:!0,zIndex:!0,zoom:!0,fillOpacity:!0,floodOpacity:!0,stopOpacity:!0,strokeDasharray:!0,strokeDashoffset:!0,strokeMiterlimit:!0,strokeOpacity:!0,strokeWidth:!0},Zr=["Webkit","ms","Moz","O"];Object.keys(Bt).forEach(function(e){Zr.forEach(function(t){t=t+e.charAt(0).toUpperCase()+e.substring(1),Bt[t]=Bt[e]})});function br(e,t,n){return t==null||typeof t=="boolean"||t===""?"":n||typeof t!="number"||t===0||Bt.hasOwnProperty(e)&&Bt[e]?(""+t).trim():t+"px"}function ar(e,t){e=e.style;for(var n in t)if(t.hasOwnProperty(n)){var r=n.indexOf("--")===0,l=br(n,t[n],r);n==="float"&&(n="cssFloat"),r?e.setProperty(n,l):e[n]=l}}var el=T({menuitem:!0},{area:!0,base:!0,br:!0,col:!0,embed:!0,hr:!0,img:!0,input:!0,keygen:!0,link:!0,meta:!0,param:!0,source:!0,track:!0,wbr:!0});function Mn(e,t){if(t){if(el[e]&&(t.children!=null||t.dangerouslySetInnerHTML!=null))throw Error(p(137,e));if(t.dangerouslySetInnerHTML!=null){if(t.children!=null)throw Error(p(60));if(typeof t.dangerouslySetInnerHTML!="object"||!("__html"in t.dangerouslySetInnerHTML))throw Error(p(61))}if(t.style!=null&&typeof t.style!="object")throw Error(p(62))}}function In(e,t){if(e.indexOf("-")===-1)return typeof t.is=="string";switch(e){case"annotation-xml":case"color-profile":case"font-face":case"font-face-src":case"font-face-uri":case"font-face-format":case"font-face-name":case"missing-glyph":return!1;default:return!0}}var cr=null;function dr(e){return e=e.target||e.srcElement||window,e.correspondingUseElement&&(e=e.correspondingUseElement),e.nodeType===3?e.parentNode:e}var fr=null,Qt=null,Wt=null;function tl(e){if(e=Dr(e)){if(typeof fr!="function")throw Error(p(280));var t=e.stateNode;t&&(t=El(t),fr(e.stateNode,e.type,t))}}function nl(e){Qt?Wt?Wt.push(e):Wt=[e]:Qt=e}function rl(){if(Qt){var e=Qt,t=Wt;if(Wt=Qt=null,tl(e),t)for(e=0;e>>=0,e===0?32:31-(dc(e)/fc|0)|0}var ul=64,al=4194304;function vr(e){switch(e&-e){case 1:return 1;case 2:return 2;case 4:return 4;case 8:return 8;case 16:return 16;case 32:return 32;case 64:case 128:case 256:case 512:case 1024:case 2048:case 4096:case 8192:case 16384:case 32768:case 65536:case 131072:case 262144:case 524288:case 1048576:case 2097152:return e&4194240;case 4194304:case 8388608:case 16777216:case 33554432:case 67108864:return e&130023424;case 134217728:return 134217728;case 268435456:return 268435456;case 536870912:return 536870912;case 1073741824:return 1073741824;default:return e}}function cl(e,t){var n=e.pendingLanes;if(n===0)return 0;var r=0,l=e.suspendedLanes,o=e.pingedLanes,i=n&268435455;if(i!==0){var a=i&~l;a!==0?r=vr(a):(o&=i,o!==0&&(r=vr(o)))}else i=n&~l,i!==0?r=vr(i):o!==0&&(r=vr(o));if(r===0)return 0;if(t!==0&&t!==r&&(t&l)===0&&(l=r&-r,o=t&-t,l>=o||l===16&&(o&4194240)!==0))return t;if((r&4)!==0&&(r|=n&16),t=e.entangledLanes,t!==0)for(e=e.entanglements,t&=r;0n;n++)t.push(e);return t}function yr(e,t,n){e.pendingLanes|=t,t!==536870912&&(e.suspendedLanes=0,e.pingedLanes=0),e=e.eventTimes,t=31-gt(t),e[t]=n}function vc(e,t){var n=e.pendingLanes&~t;e.pendingLanes=t,e.suspendedLanes=0,e.pingedLanes=0,e.expiredLanes&=t,e.mutableReadLanes&=t,e.entangledLanes&=t,t=e.entanglements;var r=e.eventTimes;for(e=e.expirationTimes;0=Cr),Pi=" ",Ti=!1;function Ri(e,t){switch(e){case"keyup":return Qc.indexOf(t.keyCode)!==-1;case"keydown":return t.keyCode!==229;case"keypress":case"mousedown":case"focusout":return!0;default:return!1}}function zi(e){return e=e.detail,typeof e=="object"&&"data"in e?e.data:null}var An=!1;function Kc(e,t){switch(e){case"compositionend":return zi(t);case"keypress":return t.which!==32?null:(Ti=!0,Pi);case"textInput":return e=t.data,e===Pi&&Ti?null:e;default:return null}}function Yc(e,t){if(An)return e==="compositionend"||!To&&Ri(e,t)?(e=wi(),hl=ko=Gt=null,An=!1,e):null;switch(e){case"paste":return null;case"keypress":if(!(t.ctrlKey||t.altKey||t.metaKey)||t.ctrlKey&&t.altKey){if(t.char&&1=t)return{node:n,offset:t-e};e=r}e:{for(;n;){if(n.nextSibling){n=n.nextSibling;break e}n=n.parentNode}n=void 0}n=Ui(n)}}function $i(e,t){return e&&t?e===t?!0:e&&e.nodeType===3?!1:t&&t.nodeType===3?$i(e,t.parentNode):"contains"in e?e.contains(t):e.compareDocumentPosition?!!(e.compareDocumentPosition(t)&16):!1:!1}function Vi(){for(var e=window,t=fn();t instanceof e.HTMLIFrameElement;){try{var n=typeof t.contentWindow.location.href=="string"}catch{n=!1}if(n)e=t.contentWindow;else break;t=fn(e.document)}return t}function Lo(e){var t=e&&e.nodeName&&e.nodeName.toLowerCase();return t&&(t==="input"&&(e.type==="text"||e.type==="search"||e.type==="tel"||e.type==="url"||e.type==="password")||t==="textarea"||e.contentEditable==="true")}function nd(e){var t=Vi(),n=e.focusedElem,r=e.selectionRange;if(t!==n&&n&&n.ownerDocument&&$i(n.ownerDocument.documentElement,n)){if(r!==null&&Lo(n)){if(t=r.start,e=r.end,e===void 0&&(e=t),"selectionStart"in n)n.selectionStart=t,n.selectionEnd=Math.min(e,n.value.length);else if(e=(t=n.ownerDocument||document)&&t.defaultView||window,e.getSelection){e=e.getSelection();var l=n.textContent.length,o=Math.min(r.start,l);r=r.end===void 0?o:Math.min(r.end,l),!e.extend&&o>r&&(l=r,r=o,o=l),l=Ai(n,o);var i=Ai(n,r);l&&i&&(e.rangeCount!==1||e.anchorNode!==l.node||e.anchorOffset!==l.offset||e.focusNode!==i.node||e.focusOffset!==i.offset)&&(t=t.createRange(),t.setStart(l.node,l.offset),e.removeAllRanges(),o>r?(e.addRange(t),e.extend(i.node,i.offset)):(t.setEnd(i.node,i.offset),e.addRange(t)))}}for(t=[],e=n;e=e.parentNode;)e.nodeType===1&&t.push({element:e,left:e.scrollLeft,top:e.scrollTop});for(typeof n.focus=="function"&&n.focus(),n=0;n=document.documentMode,$n=null,Oo=null,Tr=null,Do=!1;function Hi(e,t,n){var r=n.window===n?n.document:n.nodeType===9?n:n.ownerDocument;Do||$n==null||$n!==fn(r)||(r=$n,"selectionStart"in r&&Lo(r)?r={start:r.selectionStart,end:r.selectionEnd}:(r=(r.ownerDocument&&r.ownerDocument.defaultView||window).getSelection(),r={anchorNode:r.anchorNode,anchorOffset:r.anchorOffset,focusNode:r.focusNode,focusOffset:r.focusOffset}),Tr&&Pr(Tr,r)||(Tr=r,r=kl(Oo,"onSelect"),0Wn||(e.current=Ko[Wn],Ko[Wn]=null,Wn--)}function fe(e,t){Wn++,Ko[Wn]=e.current,e.current=t}var en={},Be=bt(en),et=bt(!1),xn=en;function Kn(e,t){var n=e.type.contextTypes;if(!n)return en;var r=e.stateNode;if(r&&r.__reactInternalMemoizedUnmaskedChildContext===t)return r.__reactInternalMemoizedMaskedChildContext;var l={},o;for(o in n)l[o]=t[o];return r&&(e=e.stateNode,e.__reactInternalMemoizedUnmaskedChildContext=t,e.__reactInternalMemoizedMaskedChildContext=l),l}function tt(e){return e=e.childContextTypes,e!=null}function Nl(){me(et),me(Be)}function ru(e,t,n){if(Be.current!==en)throw Error(p(168));fe(Be,t),fe(et,n)}function lu(e,t,n){var r=e.stateNode;if(t=t.childContextTypes,typeof r.getChildContext!="function")return n;r=r.getChildContext();for(var l in r)if(!(l in t))throw Error(p(108,ue(e)||"Unknown",l));return T({},n,r)}function Pl(e){return e=(e=e.stateNode)&&e.__reactInternalMemoizedMergedChildContext||en,xn=Be.current,fe(Be,e),fe(et,et.current),!0}function ou(e,t,n){var r=e.stateNode;if(!r)throw Error(p(169));n?(e=lu(e,t,xn),r.__reactInternalMemoizedMergedChildContext=e,me(et),me(Be),fe(Be,e)):me(et),fe(et,n)}var zt=null,Tl=!1,Yo=!1;function su(e){zt===null?zt=[e]:zt.push(e)}function md(e){Tl=!0,su(e)}function tn(){if(!Yo&&zt!==null){Yo=!0;var e=0,t=ae;try{var n=zt;for(ae=1;e>=i,l-=i,Lt=1<<32-gt(t)+l|n<X?(Fe=K,K=null):Fe=K.sibling;var ie=S(h,K,v[X],j);if(ie===null){K===null&&(K=Fe);break}e&&K&&ie.alternate===null&&t(h,K),f=o(ie,f,X),W===null?U=ie:W.sibling=ie,W=ie,K=Fe}if(X===v.length)return n(h,K),ge&&_n(h,X),U;if(K===null){for(;XX?(Fe=K,K=null):Fe=K.sibling;var dn=S(h,K,ie.value,j);if(dn===null){K===null&&(K=Fe);break}e&&K&&dn.alternate===null&&t(h,K),f=o(dn,f,X),W===null?U=dn:W.sibling=dn,W=dn,K=Fe}if(ie.done)return n(h,K),ge&&_n(h,X),U;if(K===null){for(;!ie.done;X++,ie=v.next())ie=w(h,ie.value,j),ie!==null&&(f=o(ie,f,X),W===null?U=ie:W.sibling=ie,W=ie);return ge&&_n(h,X),U}for(K=r(h,K);!ie.done;X++,ie=v.next())ie=R(K,h,X,ie.value,j),ie!==null&&(e&&ie.alternate!==null&&K.delete(ie.key===null?X:ie.key),f=o(ie,f,X),W===null?U=ie:W.sibling=ie,W=ie);return e&&K.forEach(function(Xd){return t(h,Xd)}),ge&&_n(h,X),U}function Ee(h,f,v,j){if(typeof v=="object"&&v!==null&&v.type===ke&&v.key===null&&(v=v.props.children),typeof v=="object"&&v!==null){switch(v.$$typeof){case xe:e:{for(var U=v.key,W=f;W!==null;){if(W.key===U){if(U=v.type,U===ke){if(W.tag===7){n(h,W.sibling),f=l(W,v.props.children),f.return=h,h=f;break e}}else if(W.elementType===U||typeof U=="object"&&U!==null&&U.$$typeof===de&&fu(U)===W.type){n(h,W.sibling),f=l(W,v.props),f.ref=Mr(h,W,v),f.return=h,h=f;break e}n(h,W);break}else t(h,W);W=W.sibling}v.type===ke?(f=Tn(v.props.children,h.mode,j,v.key),f.return=h,h=f):(j=ro(v.type,v.key,v.props,null,h.mode,j),j.ref=Mr(h,f,v),j.return=h,h=j)}return i(h);case G:e:{for(W=v.key;f!==null;){if(f.key===W)if(f.tag===4&&f.stateNode.containerInfo===v.containerInfo&&f.stateNode.implementation===v.implementation){n(h,f.sibling),f=l(f,v.children||[]),f.return=h,h=f;break e}else{n(h,f);break}else t(h,f);f=f.sibling}f=Qs(v,h.mode,j),f.return=h,h=f}return i(h);case de:return W=v._init,Ee(h,f,W(v._payload),j)}if(mn(v))return L(h,f,v,j);if(B(v))return D(h,f,v,j);Ol(h,v)}return typeof v=="string"&&v!==""||typeof v=="number"?(v=""+v,f!==null&&f.tag===6?(n(h,f.sibling),f=l(f,v),f.return=h,h=f):(n(h,f),f=Bs(v,h.mode,j),f.return=h,h=f),i(h)):n(h,f)}return Ee}var Gn=pu(!0),mu=pu(!1),Dl=bt(null),Ml=null,Jn=null,bo=null;function es(){bo=Jn=Ml=null}function ts(e){var t=Dl.current;me(Dl),e._currentValue=t}function ns(e,t,n){for(;e!==null;){var r=e.alternate;if((e.childLanes&t)!==t?(e.childLanes|=t,r!==null&&(r.childLanes|=t)):r!==null&&(r.childLanes&t)!==t&&(r.childLanes|=t),e===n)break;e=e.return}}function Zn(e,t){Ml=e,bo=Jn=null,e=e.dependencies,e!==null&&e.firstContext!==null&&((e.lanes&t)!==0&&(nt=!0),e.firstContext=null)}function mt(e){var t=e._currentValue;if(bo!==e)if(e={context:e,memoizedValue:t,next:null},Jn===null){if(Ml===null)throw Error(p(308));Jn=e,Ml.dependencies={lanes:0,firstContext:e}}else Jn=Jn.next=e;return t}var wn=null;function rs(e){wn===null?wn=[e]:wn.push(e)}function hu(e,t,n,r){var l=t.interleaved;return l===null?(n.next=n,rs(t)):(n.next=l.next,l.next=n),t.interleaved=n,Dt(e,r)}function Dt(e,t){e.lanes|=t;var n=e.alternate;for(n!==null&&(n.lanes|=t),n=e,e=e.return;e!==null;)e.childLanes|=t,n=e.alternate,n!==null&&(n.childLanes|=t),n=e,e=e.return;return n.tag===3?n.stateNode:null}var nn=!1;function ls(e){e.updateQueue={baseState:e.memoizedState,firstBaseUpdate:null,lastBaseUpdate:null,shared:{pending:null,interleaved:null,lanes:0},effects:null}}function vu(e,t){e=e.updateQueue,t.updateQueue===e&&(t.updateQueue={baseState:e.baseState,firstBaseUpdate:e.firstBaseUpdate,lastBaseUpdate:e.lastBaseUpdate,shared:e.shared,effects:e.effects})}function Mt(e,t){return{eventTime:e,lane:t,tag:0,payload:null,callback:null,next:null}}function rn(e,t,n){var r=e.updateQueue;if(r===null)return null;if(r=r.shared,(le&2)!==0){var l=r.pending;return l===null?t.next=t:(t.next=l.next,l.next=t),r.pending=t,Dt(e,n)}return l=r.interleaved,l===null?(t.next=t,rs(r)):(t.next=l.next,l.next=t),r.interleaved=t,Dt(e,n)}function Il(e,t,n){if(t=t.updateQueue,t!==null&&(t=t.shared,(n&4194240)!==0)){var r=t.lanes;r&=e.pendingLanes,n|=r,t.lanes=n,go(e,n)}}function yu(e,t){var n=e.updateQueue,r=e.alternate;if(r!==null&&(r=r.updateQueue,n===r)){var l=null,o=null;if(n=n.firstBaseUpdate,n!==null){do{var i={eventTime:n.eventTime,lane:n.lane,tag:n.tag,payload:n.payload,callback:n.callback,next:null};o===null?l=o=i:o=o.next=i,n=n.next}while(n!==null);o===null?l=o=t:o=o.next=t}else l=o=t;n={baseState:r.baseState,firstBaseUpdate:l,lastBaseUpdate:o,shared:r.shared,effects:r.effects},e.updateQueue=n;return}e=n.lastBaseUpdate,e===null?n.firstBaseUpdate=t:e.next=t,n.lastBaseUpdate=t}function Fl(e,t,n,r){var l=e.updateQueue;nn=!1;var o=l.firstBaseUpdate,i=l.lastBaseUpdate,a=l.shared.pending;if(a!==null){l.shared.pending=null;var d=a,y=d.next;d.next=null,i===null?o=y:i.next=y,i=d;var _=e.alternate;_!==null&&(_=_.updateQueue,a=_.lastBaseUpdate,a!==i&&(a===null?_.firstBaseUpdate=y:a.next=y,_.lastBaseUpdate=d))}if(o!==null){var w=l.baseState;i=0,_=y=d=null,a=o;do{var S=a.lane,R=a.eventTime;if((r&S)===S){_!==null&&(_=_.next={eventTime:R,lane:0,tag:a.tag,payload:a.payload,callback:a.callback,next:null});e:{var L=e,D=a;switch(S=t,R=n,D.tag){case 1:if(L=D.payload,typeof L=="function"){w=L.call(R,w,S);break e}w=L;break e;case 3:L.flags=L.flags&-65537|128;case 0:if(L=D.payload,S=typeof L=="function"?L.call(R,w,S):L,S==null)break e;w=T({},w,S);break e;case 2:nn=!0}}a.callback!==null&&a.lane!==0&&(e.flags|=64,S=l.effects,S===null?l.effects=[a]:S.push(a))}else R={eventTime:R,lane:S,tag:a.tag,payload:a.payload,callback:a.callback,next:null},_===null?(y=_=R,d=w):_=_.next=R,i|=S;if(a=a.next,a===null){if(a=l.shared.pending,a===null)break;S=a,a=S.next,S.next=null,l.lastBaseUpdate=S,l.shared.pending=null}}while(!0);if(_===null&&(d=w),l.baseState=d,l.firstBaseUpdate=y,l.lastBaseUpdate=_,t=l.shared.interleaved,t!==null){l=t;do i|=l.lane,l=l.next;while(l!==t)}else o===null&&(l.shared.lanes=0);Cn|=i,e.lanes=i,e.memoizedState=w}}function gu(e,t,n){if(e=t.effects,t.effects=null,e!==null)for(t=0;tn?n:4,e(!0);var r=as.transition;as.transition={};try{e(!1),t()}finally{ae=n,as.transition=r}}function Fu(){return ht().memoizedState}function gd(e,t,n){var r=un(e);if(n={lane:r,action:n,hasEagerState:!1,eagerState:null,next:null},Uu(e))Au(t,n);else if(n=hu(e,t,n,r),n!==null){var l=Ze();jt(n,e,r,l),$u(n,t,r)}}function xd(e,t,n){var r=un(e),l={lane:r,action:n,hasEagerState:!1,eagerState:null,next:null};if(Uu(e))Au(t,l);else{var o=e.alternate;if(e.lanes===0&&(o===null||o.lanes===0)&&(o=t.lastRenderedReducer,o!==null))try{var i=t.lastRenderedState,a=o(i,n);if(l.hasEagerState=!0,l.eagerState=a,xt(a,i)){var d=t.interleaved;d===null?(l.next=l,rs(t)):(l.next=d.next,d.next=l),t.interleaved=l;return}}catch{}n=hu(e,t,l,r),n!==null&&(l=Ze(),jt(n,e,r,l),$u(n,t,r))}}function Uu(e){var t=e.alternate;return e===_e||t!==null&&t===_e}function Au(e,t){Ar=$l=!0;var n=e.pending;n===null?t.next=t:(t.next=n.next,n.next=t),e.pending=t}function $u(e,t,n){if((n&4194240)!==0){var r=t.lanes;r&=e.pendingLanes,n|=r,t.lanes=n,go(e,n)}}var Bl={readContext:mt,useCallback:Qe,useContext:Qe,useEffect:Qe,useImperativeHandle:Qe,useInsertionEffect:Qe,useLayoutEffect:Qe,useMemo:Qe,useReducer:Qe,useRef:Qe,useState:Qe,useDebugValue:Qe,useDeferredValue:Qe,useTransition:Qe,useMutableSource:Qe,useSyncExternalStore:Qe,useId:Qe,unstable_isNewReconciler:!1},Sd={readContext:mt,useCallback:function(e,t){return Pt().memoizedState=[e,t===void 0?null:t],e},useContext:mt,useEffect:Tu,useImperativeHandle:function(e,t,n){return n=n!=null?n.concat([e]):null,Vl(4194308,4,Lu.bind(null,t,e),n)},useLayoutEffect:function(e,t){return Vl(4194308,4,e,t)},useInsertionEffect:function(e,t){return Vl(4,2,e,t)},useMemo:function(e,t){var n=Pt();return t=t===void 0?null:t,e=e(),n.memoizedState=[e,t],e},useReducer:function(e,t,n){var r=Pt();return t=n!==void 0?n(t):t,r.memoizedState=r.baseState=t,e={pending:null,interleaved:null,lanes:0,dispatch:null,lastRenderedReducer:e,lastRenderedState:t},r.queue=e,e=e.dispatch=gd.bind(null,_e,e),[r.memoizedState,e]},useRef:function(e){var t=Pt();return e={current:e},t.memoizedState=e},useState:Nu,useDebugValue:vs,useDeferredValue:function(e){return Pt().memoizedState=e},useTransition:function(){var e=Nu(!1),t=e[0];return e=yd.bind(null,e[1]),Pt().memoizedState=e,[t,e]},useMutableSource:function(){},useSyncExternalStore:function(e,t,n){var r=_e,l=Pt();if(ge){if(n===void 0)throw Error(p(407));n=n()}else{if(n=t(),Ie===null)throw Error(p(349));(jn&30)!==0||wu(r,t,n)}l.memoizedState=n;var o={value:n,getSnapshot:t};return l.queue=o,Tu(ju.bind(null,r,o,e),[e]),r.flags|=2048,Hr(9,ku.bind(null,r,o,n,t),void 0,null),n},useId:function(){var e=Pt(),t=Ie.identifierPrefix;if(ge){var n=Ot,r=Lt;n=(r&~(1<<32-gt(r)-1)).toString(32)+n,t=":"+t+"R"+n,n=$r++,0<\/script>",e=e.removeChild(e.firstChild)):typeof r.is=="string"?e=i.createElement(n,{is:r.is}):(e=i.createElement(n),n==="select"&&(i=e,r.multiple?i.multiple=!0:r.size&&(i.size=r.size))):e=i.createElementNS(e,n),e[Et]=t,e[Or]=r,sa(e,t,!1,!1),t.stateNode=e;e:{switch(i=In(n,r),n){case"dialog":pe("cancel",e),pe("close",e),l=r;break;case"iframe":case"object":case"embed":pe("load",e),l=r;break;case"video":case"audio":for(l=0;lrr&&(t.flags|=128,r=!0,Br(o,!1),t.lanes=4194304)}else{if(!r)if(e=Ul(i),e!==null){if(t.flags|=128,r=!0,n=e.updateQueue,n!==null&&(t.updateQueue=n,t.flags|=4),Br(o,!0),o.tail===null&&o.tailMode==="hidden"&&!i.alternate&&!ge)return We(t),null}else 2*Ce()-o.renderingStartTime>rr&&n!==1073741824&&(t.flags|=128,r=!0,Br(o,!1),t.lanes=4194304);o.isBackwards?(i.sibling=t.child,t.child=i):(n=o.last,n!==null?n.sibling=i:t.child=i,o.last=i)}return o.tail!==null?(t=o.tail,o.rendering=t,o.tail=t.sibling,o.renderingStartTime=Ce(),t.sibling=null,n=Se.current,fe(Se,r?n&1|2:n&1),t):(We(t),null);case 22:case 23:return $s(),r=t.memoizedState!==null,e!==null&&e.memoizedState!==null!==r&&(t.flags|=8192),r&&(t.mode&1)!==0?(ct&1073741824)!==0&&(We(t),t.subtreeFlags&6&&(t.flags|=8192)):We(t),null;case 24:return null;case 25:return null}throw Error(p(156,t.tag))}function Pd(e,t){switch(qo(t),t.tag){case 1:return tt(t.type)&&Nl(),e=t.flags,e&65536?(t.flags=e&-65537|128,t):null;case 3:return bn(),me(et),me(Be),us(),e=t.flags,(e&65536)!==0&&(e&128)===0?(t.flags=e&-65537|128,t):null;case 5:return ss(t),null;case 13:if(me(Se),e=t.memoizedState,e!==null&&e.dehydrated!==null){if(t.alternate===null)throw Error(p(340));qn()}return e=t.flags,e&65536?(t.flags=e&-65537|128,t):null;case 19:return me(Se),null;case 4:return bn(),null;case 10:return ts(t.type._context),null;case 22:case 23:return $s(),null;case 24:return null;default:return null}}var Yl=!1,Ke=!1,Td=typeof WeakSet=="function"?WeakSet:Set,z=null;function tr(e,t){var n=e.ref;if(n!==null)if(typeof n=="function")try{n(null)}catch(r){we(e,t,r)}else n.current=null}function Ps(e,t,n){try{n()}catch(r){we(e,t,r)}}var aa=!1;function Rd(e,t){if($o=pl,e=Vi(),Lo(e)){if("selectionStart"in e)var n={start:e.selectionStart,end:e.selectionEnd};else e:{n=(n=e.ownerDocument)&&n.defaultView||window;var r=n.getSelection&&n.getSelection();if(r&&r.rangeCount!==0){n=r.anchorNode;var l=r.anchorOffset,o=r.focusNode;r=r.focusOffset;try{n.nodeType,o.nodeType}catch{n=null;break e}var i=0,a=-1,d=-1,y=0,_=0,w=e,S=null;t:for(;;){for(var R;w!==n||l!==0&&w.nodeType!==3||(a=i+l),w!==o||r!==0&&w.nodeType!==3||(d=i+r),w.nodeType===3&&(i+=w.nodeValue.length),(R=w.firstChild)!==null;)S=w,w=R;for(;;){if(w===e)break t;if(S===n&&++y===l&&(a=i),S===o&&++_===r&&(d=i),(R=w.nextSibling)!==null)break;w=S,S=w.parentNode}w=R}n=a===-1||d===-1?null:{start:a,end:d}}else n=null}n=n||{start:0,end:0}}else n=null;for(Vo={focusedElem:e,selectionRange:n},pl=!1,z=t;z!==null;)if(t=z,e=t.child,(t.subtreeFlags&1028)!==0&&e!==null)e.return=t,z=e;else for(;z!==null;){t=z;try{var L=t.alternate;if((t.flags&1024)!==0)switch(t.tag){case 0:case 11:case 15:break;case 1:if(L!==null){var D=L.memoizedProps,Ee=L.memoizedState,h=t.stateNode,f=h.getSnapshotBeforeUpdate(t.elementType===t.type?D:_t(t.type,D),Ee);h.__reactInternalSnapshotBeforeUpdate=f}break;case 3:var v=t.stateNode.containerInfo;v.nodeType===1?v.textContent="":v.nodeType===9&&v.documentElement&&v.removeChild(v.documentElement);break;case 5:case 6:case 4:case 17:break;default:throw Error(p(163))}}catch(j){we(t,t.return,j)}if(e=t.sibling,e!==null){e.return=t.return,z=e;break}z=t.return}return L=aa,aa=!1,L}function Qr(e,t,n){var r=t.updateQueue;if(r=r!==null?r.lastEffect:null,r!==null){var l=r=r.next;do{if((l.tag&e)===e){var o=l.destroy;l.destroy=void 0,o!==void 0&&Ps(t,n,o)}l=l.next}while(l!==r)}}function Xl(e,t){if(t=t.updateQueue,t=t!==null?t.lastEffect:null,t!==null){var n=t=t.next;do{if((n.tag&e)===e){var r=n.create;n.destroy=r()}n=n.next}while(n!==t)}}function Ts(e){var t=e.ref;if(t!==null){var n=e.stateNode;e.tag,e=n,typeof t=="function"?t(e):t.current=e}}function ca(e){var t=e.alternate;t!==null&&(e.alternate=null,ca(t)),e.child=null,e.deletions=null,e.sibling=null,e.tag===5&&(t=e.stateNode,t!==null&&(delete t[Et],delete t[Or],delete t[Wo],delete t[fd],delete t[pd])),e.stateNode=null,e.return=null,e.dependencies=null,e.memoizedProps=null,e.memoizedState=null,e.pendingProps=null,e.stateNode=null,e.updateQueue=null}function da(e){return e.tag===5||e.tag===3||e.tag===4}function fa(e){e:for(;;){for(;e.sibling===null;){if(e.return===null||da(e.return))return null;e=e.return}for(e.sibling.return=e.return,e=e.sibling;e.tag!==5&&e.tag!==6&&e.tag!==18;){if(e.flags&2||e.child===null||e.tag===4)continue e;e.child.return=e,e=e.child}if(!(e.flags&2))return e.stateNode}}function Rs(e,t,n){var r=e.tag;if(r===5||r===6)e=e.stateNode,t?n.nodeType===8?n.parentNode.insertBefore(e,t):n.insertBefore(e,t):(n.nodeType===8?(t=n.parentNode,t.insertBefore(e,n)):(t=n,t.appendChild(e)),n=n._reactRootContainer,n!=null||t.onclick!==null||(t.onclick=Cl));else if(r!==4&&(e=e.child,e!==null))for(Rs(e,t,n),e=e.sibling;e!==null;)Rs(e,t,n),e=e.sibling}function zs(e,t,n){var r=e.tag;if(r===5||r===6)e=e.stateNode,t?n.insertBefore(e,t):n.appendChild(e);else if(r!==4&&(e=e.child,e!==null))for(zs(e,t,n),e=e.sibling;e!==null;)zs(e,t,n),e=e.sibling}var $e=null,wt=!1;function ln(e,t,n){for(n=n.child;n!==null;)pa(e,t,n),n=n.sibling}function pa(e,t,n){if(Ct&&typeof Ct.onCommitFiberUnmount=="function")try{Ct.onCommitFiberUnmount(il,n)}catch{}switch(n.tag){case 5:Ke||tr(n,t);case 6:var r=$e,l=wt;$e=null,ln(e,t,n),$e=r,wt=l,$e!==null&&(wt?(e=$e,n=n.stateNode,e.nodeType===8?e.parentNode.removeChild(n):e.removeChild(n)):$e.removeChild(n.stateNode));break;case 18:$e!==null&&(wt?(e=$e,n=n.stateNode,e.nodeType===8?Qo(e.parentNode,n):e.nodeType===1&&Qo(e,n),wr(e)):Qo($e,n.stateNode));break;case 4:r=$e,l=wt,$e=n.stateNode.containerInfo,wt=!0,ln(e,t,n),$e=r,wt=l;break;case 0:case 11:case 14:case 15:if(!Ke&&(r=n.updateQueue,r!==null&&(r=r.lastEffect,r!==null))){l=r=r.next;do{var o=l,i=o.destroy;o=o.tag,i!==void 0&&((o&2)!==0||(o&4)!==0)&&Ps(n,t,i),l=l.next}while(l!==r)}ln(e,t,n);break;case 1:if(!Ke&&(tr(n,t),r=n.stateNode,typeof r.componentWillUnmount=="function"))try{r.props=n.memoizedProps,r.state=n.memoizedState,r.componentWillUnmount()}catch(a){we(n,t,a)}ln(e,t,n);break;case 21:ln(e,t,n);break;case 22:n.mode&1?(Ke=(r=Ke)||n.memoizedState!==null,ln(e,t,n),Ke=r):ln(e,t,n);break;default:ln(e,t,n)}}function ma(e){var t=e.updateQueue;if(t!==null){e.updateQueue=null;var n=e.stateNode;n===null&&(n=e.stateNode=new Td),t.forEach(function(r){var l=Ad.bind(null,e,r);n.has(r)||(n.add(r),r.then(l,l))})}}function kt(e,t){var n=t.deletions;if(n!==null)for(var r=0;rl&&(l=i),r&=~o}if(r=l,r=Ce()-r,r=(120>r?120:480>r?480:1080>r?1080:1920>r?1920:3e3>r?3e3:4320>r?4320:1960*Ld(r/1960))-r,10e?16:e,sn===null)var r=!1;else{if(e=sn,sn=null,bl=0,(le&6)!==0)throw Error(p(331));var l=le;for(le|=4,z=e.current;z!==null;){var o=z,i=o.child;if((z.flags&16)!==0){var a=o.deletions;if(a!==null){for(var d=0;dCe()-Ds?Nn(e,0):Os|=n),lt(e,t)}function Na(e,t){t===0&&((e.mode&1)===0?t=1:(t=al,al<<=1,(al&130023424)===0&&(al=4194304)));var n=Ze();e=Dt(e,t),e!==null&&(yr(e,t,n),lt(e,n))}function Ud(e){var t=e.memoizedState,n=0;t!==null&&(n=t.retryLane),Na(e,n)}function Ad(e,t){var n=0;switch(e.tag){case 13:var r=e.stateNode,l=e.memoizedState;l!==null&&(n=l.retryLane);break;case 19:r=e.stateNode;break;default:throw Error(p(314))}r!==null&&r.delete(t),Na(e,n)}var Pa;Pa=function(e,t,n){if(e!==null)if(e.memoizedProps!==t.pendingProps||et.current)nt=!0;else{if((e.lanes&n)===0&&(t.flags&128)===0)return nt=!1,Ed(e,t,n);nt=(e.flags&131072)!==0}else nt=!1,ge&&(t.flags&1048576)!==0&&iu(t,zl,t.index);switch(t.lanes=0,t.tag){case 2:var r=t.type;Kl(e,t),e=t.pendingProps;var l=Kn(t,Be.current);Zn(t,n),l=ds(null,t,r,e,l,n);var o=fs();return t.flags|=1,typeof l=="object"&&l!==null&&typeof l.render=="function"&&l.$$typeof===void 0?(t.tag=1,t.memoizedState=null,t.updateQueue=null,tt(r)?(o=!0,Pl(t)):o=!1,t.memoizedState=l.state!==null&&l.state!==void 0?l.state:null,ls(t),l.updater=Ql,t.stateNode=l,l._reactInternals=t,gs(t,r,e,n),t=ws(null,t,r,!0,o,n)):(t.tag=0,ge&&o&&Xo(t),Je(null,t,l,n),t=t.child),t;case 16:r=t.elementType;e:{switch(Kl(e,t),e=t.pendingProps,l=r._init,r=l(r._payload),t.type=r,l=t.tag=Vd(r),e=_t(r,e),l){case 0:t=_s(null,t,r,e,n);break e;case 1:t=ea(null,t,r,e,n);break e;case 11:t=qu(null,t,r,e,n);break e;case 14:t=Gu(null,t,r,_t(r.type,e),n);break e}throw Error(p(306,r,""))}return t;case 0:return r=t.type,l=t.pendingProps,l=t.elementType===r?l:_t(r,l),_s(e,t,r,l,n);case 1:return r=t.type,l=t.pendingProps,l=t.elementType===r?l:_t(r,l),ea(e,t,r,l,n);case 3:e:{if(ta(t),e===null)throw Error(p(387));r=t.pendingProps,o=t.memoizedState,l=o.element,vu(e,t),Fl(t,r,null,n);var i=t.memoizedState;if(r=i.element,o.isDehydrated)if(o={element:r,isDehydrated:!1,cache:i.cache,pendingSuspenseBoundaries:i.pendingSuspenseBoundaries,transitions:i.transitions},t.updateQueue.baseState=o,t.memoizedState=o,t.flags&256){l=er(Error(p(423)),t),t=na(e,t,r,n,l);break e}else if(r!==l){l=er(Error(p(424)),t),t=na(e,t,r,n,l);break e}else for(at=Zt(t.stateNode.containerInfo.firstChild),ut=t,ge=!0,St=null,n=mu(t,null,r,n),t.child=n;n;)n.flags=n.flags&-3|4096,n=n.sibling;else{if(qn(),r===l){t=It(e,t,n);break e}Je(e,t,r,n)}t=t.child}return t;case 5:return xu(t),e===null&&Jo(t),r=t.type,l=t.pendingProps,o=e!==null?e.memoizedProps:null,i=l.children,Ho(r,l)?i=null:o!==null&&Ho(r,o)&&(t.flags|=32),bu(e,t),Je(e,t,i,n),t.child;case 6:return e===null&&Jo(t),null;case 13:return ra(e,t,n);case 4:return os(t,t.stateNode.containerInfo),r=t.pendingProps,e===null?t.child=Gn(t,null,r,n):Je(e,t,r,n),t.child;case 11:return r=t.type,l=t.pendingProps,l=t.elementType===r?l:_t(r,l),qu(e,t,r,l,n);case 7:return Je(e,t,t.pendingProps,n),t.child;case 8:return Je(e,t,t.pendingProps.children,n),t.child;case 12:return Je(e,t,t.pendingProps.children,n),t.child;case 10:e:{if(r=t.type._context,l=t.pendingProps,o=t.memoizedProps,i=l.value,fe(Dl,r._currentValue),r._currentValue=i,o!==null)if(xt(o.value,i)){if(o.children===l.children&&!et.current){t=It(e,t,n);break e}}else for(o=t.child,o!==null&&(o.return=t);o!==null;){var a=o.dependencies;if(a!==null){i=o.child;for(var d=a.firstContext;d!==null;){if(d.context===r){if(o.tag===1){d=Mt(-1,n&-n),d.tag=2;var y=o.updateQueue;if(y!==null){y=y.shared;var _=y.pending;_===null?d.next=d:(d.next=_.next,_.next=d),y.pending=d}}o.lanes|=n,d=o.alternate,d!==null&&(d.lanes|=n),ns(o.return,n,t),a.lanes|=n;break}d=d.next}}else if(o.tag===10)i=o.type===t.type?null:o.child;else if(o.tag===18){if(i=o.return,i===null)throw Error(p(341));i.lanes|=n,a=i.alternate,a!==null&&(a.lanes|=n),ns(i,n,t),i=o.sibling}else i=o.child;if(i!==null)i.return=o;else for(i=o;i!==null;){if(i===t){i=null;break}if(o=i.sibling,o!==null){o.return=i.return,i=o;break}i=i.return}o=i}Je(e,t,l.children,n),t=t.child}return t;case 9:return l=t.type,r=t.pendingProps.children,Zn(t,n),l=mt(l),r=r(l),t.flags|=1,Je(e,t,r,n),t.child;case 14:return r=t.type,l=_t(r,t.pendingProps),l=_t(r.type,l),Gu(e,t,r,l,n);case 15:return Ju(e,t,t.type,t.pendingProps,n);case 17:return r=t.type,l=t.pendingProps,l=t.elementType===r?l:_t(r,l),Kl(e,t),t.tag=1,tt(r)?(e=!0,Pl(t)):e=!1,Zn(t,n),Hu(t,r,l),gs(t,r,l,n),ws(null,t,r,!0,e,n);case 19:return oa(e,t,n);case 22:return Zu(e,t,n)}throw Error(p(156,t.tag))};function Ta(e,t){return ui(e,t)}function $d(e,t,n,r){this.tag=e,this.key=n,this.sibling=this.child=this.return=this.stateNode=this.type=this.elementType=null,this.index=0,this.ref=null,this.pendingProps=t,this.dependencies=this.memoizedState=this.updateQueue=this.memoizedProps=null,this.mode=r,this.subtreeFlags=this.flags=0,this.deletions=null,this.childLanes=this.lanes=0,this.alternate=null}function yt(e,t,n,r){return new $d(e,t,n,r)}function Hs(e){return e=e.prototype,!(!e||!e.isReactComponent)}function Vd(e){if(typeof e=="function")return Hs(e)?1:0;if(e!=null){if(e=e.$$typeof,e===be)return 11;if(e===Ge)return 14}return 2}function cn(e,t){var n=e.alternate;return n===null?(n=yt(e.tag,t,e.key,e.mode),n.elementType=e.elementType,n.type=e.type,n.stateNode=e.stateNode,n.alternate=e,e.alternate=n):(n.pendingProps=t,n.type=e.type,n.flags=0,n.subtreeFlags=0,n.deletions=null),n.flags=e.flags&14680064,n.childLanes=e.childLanes,n.lanes=e.lanes,n.child=e.child,n.memoizedProps=e.memoizedProps,n.memoizedState=e.memoizedState,n.updateQueue=e.updateQueue,t=e.dependencies,n.dependencies=t===null?null:{lanes:t.lanes,firstContext:t.firstContext},n.sibling=e.sibling,n.index=e.index,n.ref=e.ref,n}function ro(e,t,n,r,l,o){var i=2;if(r=e,typeof e=="function")Hs(e)&&(i=1);else if(typeof e=="string")i=5;else e:switch(e){case ke:return Tn(n.children,l,o,t);case ye:i=8,l|=8;break;case Ae:return e=yt(12,n,t,l|2),e.elementType=Ae,e.lanes=o,e;case je:return e=yt(13,n,t,l),e.elementType=je,e.lanes=o,e;case He:return e=yt(19,n,t,l),e.elementType=He,e.lanes=o,e;case se:return lo(n,l,o,t);default:if(typeof e=="object"&&e!==null)switch(e.$$typeof){case Xe:i=10;break e;case qe:i=9;break e;case be:i=11;break e;case Ge:i=14;break e;case de:i=16,r=null;break e}throw Error(p(130,e==null?e:typeof e,""))}return t=yt(i,n,t,l),t.elementType=e,t.type=r,t.lanes=o,t}function Tn(e,t,n,r){return e=yt(7,e,r,t),e.lanes=n,e}function lo(e,t,n,r){return e=yt(22,e,r,t),e.elementType=se,e.lanes=n,e.stateNode={isHidden:!1},e}function Bs(e,t,n){return e=yt(6,e,null,t),e.lanes=n,e}function Qs(e,t,n){return t=yt(4,e.children!==null?e.children:[],e.key,t),t.lanes=n,t.stateNode={containerInfo:e.containerInfo,pendingChildren:null,implementation:e.implementation},t}function Hd(e,t,n,r,l){this.tag=t,this.containerInfo=e,this.finishedWork=this.pingCache=this.current=this.pendingChildren=null,this.timeoutHandle=-1,this.callbackNode=this.pendingContext=this.context=null,this.callbackPriority=0,this.eventTimes=yo(0),this.expirationTimes=yo(-1),this.entangledLanes=this.finishedLanes=this.mutableReadLanes=this.expiredLanes=this.pingedLanes=this.suspendedLanes=this.pendingLanes=0,this.entanglements=yo(0),this.identifierPrefix=r,this.onRecoverableError=l,this.mutableSourceEagerHydrationData=null}function Ws(e,t,n,r,l,o,i,a,d){return e=new Hd(e,t,n,a,d),t===1?(t=1,o===!0&&(t|=8)):t=0,o=yt(3,null,null,t),e.current=o,o.stateNode=e,o.memoizedState={element:r,isDehydrated:n,cache:null,transitions:null,pendingSuspenseBoundaries:null},ls(o),e}function Bd(e,t,n){var r=3"u"||typeof __REACT_DEVTOOLS_GLOBAL_HOOK__.checkDCE!="function"))try{__REACT_DEVTOOLS_GLOBAL_HOOK__.checkDCE(u)}catch(g){console.error(g)}}return u(),Js.exports=tf(),Js.exports}var Ba;function rf(){if(Ba)return fo;Ba=1;var u=nf();return fo.createRoot=u.createRoot,fo.hydrateRoot=u.hydrateRoot,fo}var lf=rf();const of=ec(lf),sf="/api";async function Ue(u,g){const p=await fetch(`${sf}${u}`,{...g,headers:{"Content-Type":"application/json",...g?.headers??{}}}),O=await p.json();if(!p.ok){const k=O.error?.message??"Ошибка запроса";throw new Error(k)}return O}const Ye={async listModels(u){return Ue("/llm/models",{method:"POST",body:JSON.stringify({llmProvider:u.llmProvider,apiKey:u.apiKey,model:u.model,baseUrl:u.baseUrl})})},async testConnection(u){return Ue("/llm/test-connection",{method:"POST",body:JSON.stringify({llmProvider:u.llmProvider,apiKey:u.apiKey,model:u.model,baseUrl:u.baseUrl})})},async normalize(u){return Ue("/normalize",{method:"POST",body:JSON.stringify({llmProvider:u.connection.llmProvider,apiKey:u.connection.apiKey,model:u.connection.model,baseUrl:u.connection.baseUrl,temperature:u.connection.temperature,maxOutputTokens:u.connection.maxOutputTokens,promptVersion:u.promptVersion,systemPrompt:u.prompts.systemPrompt,developerPrompt:u.prompts.developerPrompt,domainPrompt:u.prompts.domainPrompt,fewShotExamples:u.prompts.fewShotExamples,userQuestion:u.query.userQuestion,context:{period_hint:u.query.periodHint??"",business_context:u.query.businessContext??"",expected_route:u.query.expectedRoute??""},saveAsTestCase:!!u.saveAsTestCase,useMock:!!u.useMock})})},async loadHistory(){return Ue("/history")},async loadTrace(u){return Ue(`/history/${u}`)},async loadPresets(){return Ue("/presets")},async savePreset(u){return Ue("/presets/save",{method:"POST",body:JSON.stringify(u)})},async runEval(u){return Ue("/eval/run",{method:"POST",body:JSON.stringify({normalizeConfig:{llmProvider:u.connection.llmProvider,apiKey:u.connection.apiKey,model:u.connection.model,baseUrl:u.connection.baseUrl,temperature:u.connection.temperature,maxOutputTokens:u.connection.maxOutputTokens,promptVersion:u.promptVersion,systemPrompt:u.prompts.systemPrompt,developerPrompt:u.prompts.developerPrompt,domainPrompt:u.prompts.domainPrompt,fewShotExamples:u.prompts.fewShotExamples},caseIds:u.caseIds,useMock:!!u.useMock,mode:u.mode??"standard",caseSetFile:u.caseSetFile,rawQuestions:u.rawQuestions})})},async startRun(){return Ue("/accounting-agent/v1/runs/start",{method:"POST",body:JSON.stringify({initiator:"ndc_operator",source:"gui"})})},async finishRun(u){return Ue("/accounting-agent/v1/runs/finish",{method:"POST",body:JSON.stringify({runId:u,status:"DONE",source:"gui",reason:"Остановлено оператором из GUI"})})},async listRuns(){return Ue("/accounting-agent/v1/runs")},async listResults(){return Ue("/accounting-agent/v1/results")},async runTrace(u){return Ue(`/accounting-agent/v1/trace/run/${u}`)},async sendAssistantMessage(u){return Ue("/assistant/message",{method:"POST",body:JSON.stringify({session_id:u.sessionId??"",mode:"assistant",message:u.userMessage,user_message:u.userMessage,llmProvider:u.connection.llmProvider,apiKey:u.connection.apiKey,model:u.connection.model,baseUrl:u.connection.baseUrl,temperature:u.connection.temperature,maxOutputTokens:u.connection.maxOutputTokens,promptVersion:u.promptVersion??"address_query_runtime_v1",systemPrompt:u.prompts.systemPrompt,developerPrompt:u.prompts.developerPrompt,domainPrompt:u.prompts.domainPrompt,fewShotExamples:u.prompts.fewShotExamples,context:{period_hint:u.context?.periodHint??"",business_context:u.context?.businessContext??""},useMock:!!u.useMock})})},async loadAssistantSession(u){return Ue(`/assistant/session/${u}`)},async loadAutoRunsHistory(u){const g=new URLSearchParams;u?.from&&g.set("from",u.from),u?.to&&g.set("to",u.to),u?.target&&g.set("target",u.target),u?.mode&&g.set("mode",u.mode),u?.use_mock&&g.set("use_mock",u.use_mock),u?.prompt_contains&&g.set("prompt_contains",u.prompt_contains),typeof u?.limit=="number"&&g.set("limit",String(u.limit)),typeof u?.scan_limit=="number"&&g.set("scan_limit",String(u.scan_limit));const p=g.toString();return Ue(`/autoruns/history${p?`?${p}`:""}`)},async loadAutoRunDetail(u){return Ue(`/autoruns/history/${encodeURIComponent(u)}`)},async loadAutoRunCaseDialog(u,g){return Ue(`/autoruns/history/${encodeURIComponent(u)}/case/${encodeURIComponent(g)}/dialog`)}};function st({value:u}){return s.jsx("pre",{className:"json-view",children:JSON.stringify(u??{},null,2)})}function At({title:u,subtitle:g,actions:p,children:O}){return s.jsxs("section",{className:"panel-frame",children:[s.jsxs("header",{className:"panel-header",children:[s.jsxs("div",{children:[s.jsx("h2",{children:u}),g?s.jsx("p",{children:g}):null]}),p?s.jsx("div",{className:"panel-actions",children:p}):null]}),s.jsx("div",{className:"panel-body",children:O})]})}const Qa={fromLocal:"",toLocal:"",target:"all",mode:"all",useMock:"any",promptContains:"",limit:120};function uf(u){const g=u.getFullYear(),p=String(u.getMonth()+1).padStart(2,"0"),O=String(u.getDate()).padStart(2,"0"),k=String(u.getHours()).padStart(2,"0"),E=String(u.getMinutes()).padStart(2,"0");return`${g}-${p}-${O}T${k}:${E}`}function Wa(){const u=new Date;return u.setDate(u.getDate()-14),uf(u)}function Ka(u){if(!u.trim())return;const g=Date.parse(u);if(Number.isFinite(g))return new Date(g).toISOString()}function ei(u){const g=Date.parse(u);return Number.isFinite(g)?new Date(g).toLocaleString("ru-RU"):u}function af(u,g){return g<=0?0:Math.max(0,Math.min(100,Number((u/g*100).toFixed(1))))}function Gr(u){return typeof u!="number"?"n/a":`${u.toFixed(1)}%`}function cf(u){return u==="assistant_stage1"?"assistant/s1":u==="assistant_stage2"?"assistant/s2":u==="assistant_p0"?"assistant/p0":u}function Ya(u){return u==="up"?"Рост":u==="down"?"Регресс":"Без изменений"}function df(u,g){return u.find(p=>p.case_id===g)??null}function Xa(u){return u.length===0?s.jsx("p",{className:"muted",children:"Покрытие доменов пока не сформировано."}):s.jsx("div",{className:"autoruns-coverage-list",children:u.map(g=>{const p=af(g.closed_cases,g.total_cases);return s.jsxs("div",{className:"autoruns-coverage-item",children:[s.jsxs("div",{className:"autoruns-coverage-head",children:[s.jsx("strong",{children:g.domain}),s.jsxs("span",{children:[g.closed_cases,"/",g.total_cases," (",p,"%)"]})]}),s.jsx("div",{className:"autoruns-coverage-bar",children:s.jsx("div",{style:{width:`${p}%`}})})]},g.domain)})})}function ff({connection:u,prompts:g,assistantPromptVersion:p,decompositionPromptVersion:O,onLog:k}){const[E,M]=A.useState({...Qa,fromLocal:Wa()}),[$,H]=A.useState(null),[te,oe]=A.useState(null),[N,F]=A.useState(null),[ce,he]=A.useState(""),[b,J]=A.useState(""),[De,Pe]=A.useState(!1),[Te,ve]=A.useState(!1),[xe,G]=A.useState(!1),[ke,ye]=A.useState(""),[Ae,Xe]=A.useState(!0),[qe,be]=A.useState(!0),[je,He]=A.useState(!0),Ge=$?.items.find(c=>c.run_id===ce)??null,de=te?df(te.cases,b):null,se=A.useCallback(c=>{k?.(`[autoruns] ${c}`)},[k]),P=A.useCallback(async(c,C)=>{G(!0);try{const I=await Ye.loadAutoRunCaseDialog(c,C);F(I)}catch(I){const Y=I instanceof Error?I.message:String(I);ye(`Диалог кейса: ${Y}`),se(`Dialog load error for ${c}/${C}: ${Y}`),F(null)}finally{G(!1)}},[se]),B=A.useCallback(async(c,C)=>{ve(!0);try{const I=await Ye.loadAutoRunDetail(c);oe(I);const Y=(C&&I.cases.some(Z=>Z.case_id===C)?C:"")||I.cases[0]?.case_id||"";J(Y),Y?await P(c,Y):F(null)}catch(I){const Y=I instanceof Error?I.message:String(I);ye(`Детализация прогона: ${Y}`),se(`Run detail load error for ${c}: ${Y}`),oe(null),F(null)}finally{ve(!1)}},[P,se]),T=A.useCallback(async c=>{Pe(!0),ye("");try{const C=await Ye.loadAutoRunsHistory({from:Ka(E.fromLocal),to:Ka(E.toLocal),target:E.target,mode:E.mode,use_mock:E.useMock,prompt_contains:E.promptContains.trim()||void 0,limit:E.limit});if(H(C),!(C.items.length>0)){he(""),J(""),oe(null),F(null);return}const Y=c?.keepSelection??!0,Z=c?.preferredRunId??"",ue=c?.preferredCaseId??"",re=Y&&Z&&C.items.some(ne=>ne.run_id===Z)?Z:C.items[0].run_id;he(re),await B(re,Y?ue:void 0)}catch(C){const I=C instanceof Error?C.message:String(C);ye(`История прогонов: ${I}`),se(`History load error: ${I}`)}finally{Pe(!1)}},[E.fromLocal,E.limit,E.mode,E.promptContains,E.target,E.toLocal,E.useMock,B,se]);A.useEffect(()=>{T({keepSelection:!1})},[T]);const m=A.useMemo(()=>{const c=["minmax(290px, 340px)","minmax(300px, 360px)","minmax(420px, 1fr)"];return Ae&&c.push("minmax(280px, 320px)"),qe&&c.push("minmax(280px, 320px)"),je&&c.push("minmax(280px, 320px)"),c.join(" ")},[Ae,qe,je]);return s.jsx(At,{title:"История автопрогонов",subtitle:"Центральный экран диагностики: фильтры, список прогонов, диалог по кейсу, режимы ассистента/декомпозиции и тренд качества.",actions:s.jsxs("div",{className:"assistant-panel-actions",children:[s.jsx("button",{type:"button",className:Ae?"tab active":"tab",onClick:()=>Xe(c=>!c),children:"Режим ассистента"}),s.jsx("button",{type:"button",className:qe?"tab active":"tab",onClick:()=>be(c=>!c),children:"Режим декомпозиции"}),s.jsx("button",{type:"button",className:je?"tab active":"tab",onClick:()=>He(c=>!c),children:"Прогресс/регресс"})]}),children:s.jsxs("div",{className:"autoruns-columns",style:{gridTemplateColumns:m},children:[s.jsxs("section",{className:"autoruns-col",children:[s.jsx("h3",{children:"Настройки выборки"}),s.jsxs("div",{className:"autoruns-form-grid",children:[s.jsxs("label",{children:["Дата с",s.jsx("input",{type:"datetime-local",value:E.fromLocal,onChange:c=>M(C=>({...C,fromLocal:c.target.value}))})]}),s.jsxs("label",{children:["Дата по",s.jsx("input",{type:"datetime-local",value:E.toLocal,onChange:c=>M(C=>({...C,toLocal:c.target.value}))})]}),s.jsxs("label",{children:["Целевой контур",s.jsxs("select",{value:E.target,onChange:c=>M(C=>({...C,target:c.target.value})),children:[s.jsx("option",{value:"all",children:"all"}),($?.available.targets??[]).map(c=>s.jsx("option",{value:c,children:c},c))]})]}),s.jsxs("label",{children:["Режим",s.jsxs("select",{value:E.mode,onChange:c=>M(C=>({...C,mode:c.target.value})),children:[s.jsx("option",{value:"all",children:"all"}),($?.available.modes??[]).map(c=>s.jsx("option",{value:c,children:c},c))]})]}),s.jsxs("label",{children:["use_mock",s.jsxs("select",{value:E.useMock,onChange:c=>M(C=>({...C,useMock:c.target.value})),children:[s.jsx("option",{value:"any",children:"any"}),s.jsx("option",{value:"true",children:"true"}),s.jsx("option",{value:"false",children:"false"})]})]}),s.jsxs("label",{children:["Лимит",s.jsx("input",{type:"number",min:1,max:500,value:E.limit,onChange:c=>M(C=>({...C,limit:Number(c.target.value||120)}))})]}),s.jsxs("label",{className:"full-width",children:["Версия промпта содержит",s.jsx("input",{value:E.promptContains,onChange:c=>M(C=>({...C,promptContains:c.target.value})),placeholder:"normalizer_v2_0_2 / address_query_runtime_v1",list:"autoruns-prompt-versions"})]})]}),s.jsx("datalist",{id:"autoruns-prompt-versions",children:($?.available.prompt_versions??[]).map(c=>s.jsx("option",{value:c},c))}),s.jsxs("div",{className:"button-row",children:[s.jsx("button",{type:"button",disabled:De,onClick:()=>{T({keepSelection:!1})},children:De?"Обновляю...":"Применить"}),s.jsx("button",{type:"button",className:"tab",onClick:()=>{M({...Qa,fromLocal:Wa()}),ye("")},children:"Сбросить фильтры"})]}),s.jsx("h4",{children:"Контур генерации"}),s.jsxs("div",{className:"autoruns-meta-list",children:[s.jsxs("div",{children:[s.jsx("span",{children:"Провайдер:"}),s.jsx("strong",{children:u.llmProvider})]}),s.jsxs("div",{children:[s.jsx("span",{children:"Модель:"}),s.jsx("strong",{children:u.model||"n/a"})]}),s.jsxs("div",{children:[s.jsx("span",{children:"Prompt assistant:"}),s.jsx("strong",{children:p})]}),s.jsxs("div",{children:[s.jsx("span",{children:"Prompt decomposition:"}),s.jsx("strong",{children:O})]})]}),s.jsxs("details",{className:"autoruns-prompt-details",children:[s.jsx("summary",{children:"Дублирование главного промпта (read-only)"}),s.jsxs("label",{children:["System",s.jsx("textarea",{readOnly:!0,value:g.systemPrompt})]}),s.jsxs("label",{children:["Developer",s.jsx("textarea",{readOnly:!0,value:g.developerPrompt})]}),s.jsxs("label",{children:["Domain",s.jsx("textarea",{readOnly:!0,value:g.domainPrompt})]}),s.jsxs("label",{children:["Schema notes",s.jsx("textarea",{readOnly:!0,value:g.schemaNotes})]}),s.jsxs("label",{children:["Few-shot",s.jsx("textarea",{readOnly:!0,value:g.fewShotExamples})]})]}),ke?s.jsx("p",{className:"error-text",children:ke}):null]}),s.jsxs("section",{className:"autoruns-col",children:[s.jsx("h3",{children:"Выдача прогонов"}),s.jsxs("div",{className:"autoruns-stats-grid",children:[s.jsxs("div",{children:[s.jsx("span",{children:"Всего"}),s.jsx("strong",{children:$?.stats.runs_total??0})]}),s.jsxs("div",{children:[s.jsx("span",{children:"Средний score"}),s.jsx("strong",{children:Gr($?.stats.avg_score_index??null)})]}),s.jsxs("div",{children:[s.jsx("span",{children:"Тренд"}),s.jsx("strong",{children:$?Ya($.stats.trend):"n/a"})]}),s.jsxs("div",{children:[s.jsx("span",{children:"Блокеры"}),s.jsx("strong",{children:$?.stats.blocking_runs??0})]})]}),s.jsxs("div",{className:"autoruns-run-list",children:[($?.items??[]).map(c=>s.jsxs("button",{type:"button",className:ce===c.run_id?"autoruns-run-item selected":"autoruns-run-item",onClick:()=>{he(c.run_id),B(c.run_id)},children:[s.jsxs("div",{className:"autoruns-run-head",children:[s.jsx("strong",{children:ei(c.run_timestamp)}),s.jsx("span",{children:cf(c.eval_target)})]}),s.jsx("div",{className:"autoruns-run-meta",children:c.run_id}),s.jsxs("div",{className:"autoruns-run-meta",children:["mode=",c.mode??"n/a"," | mock=",String(c.use_mock)]}),c.llm_provider||c.model?s.jsxs("div",{className:"autoruns-run-meta",children:["llm=",c.llm_provider??"n/a"," | model=",c.model??"n/a"]}):null,s.jsxs("div",{className:"autoruns-run-meta",children:["prompt=",c.prompt_version??"n/a"]}),s.jsxs("div",{className:"autoruns-run-foot",children:[s.jsxs("span",{children:["score: ",Gr(c.score_index)]}),s.jsxs("span",{children:["closed/open: ",c.closed_cases,"/",c.open_cases]})]}),s.jsxs("div",{className:"autoruns-run-foot",children:[s.jsxs("span",{children:["blocking: ",c.blocking_failures]}),s.jsxs("span",{children:["quality: ",c.quality_failures]})]})]},c.run_id)),($?.items.length??0)===0?s.jsx("p",{className:"muted",children:"За выбранный диапазон прогонов нет."}):null]})]}),s.jsxs("section",{className:"autoruns-col",children:[s.jsx("h3",{children:"Диалог прогона"}),s.jsxs("div",{className:"autoruns-dialog-toolbar",children:[s.jsxs("label",{children:["Прогон",s.jsx("select",{value:ce,onChange:c=>{const C=c.target.value;he(C),B(C)},children:($?.items??[]).map(c=>s.jsxs("option",{value:c.run_id,children:[ei(c.run_timestamp)," | ",c.run_id]},c.run_id))})]}),s.jsxs("label",{children:["Кейc",s.jsx("select",{value:b,onChange:c=>{const C=c.target.value;J(C),ce&&C&&P(ce,C)},children:(te?.cases??[]).map(c=>s.jsxs("option",{value:c.case_id,children:[c.case_id," | ",c.status]},c.case_id))})]})]}),s.jsx("div",{className:"autoruns-case-list",children:(te?.cases??[]).map(c=>s.jsxs("button",{type:"button",className:b===c.case_id?"autoruns-case-item selected":"autoruns-case-item",onClick:()=>{J(c.case_id),ce&&P(ce,c.case_id)},children:[s.jsx("span",{children:c.case_id}),s.jsx("span",{children:c.status})]},c.case_id))}),s.jsxs("div",{className:"autoruns-dialog-view",children:[xe||Te?s.jsx("p",{className:"muted",children:"Загружаю диалог..."}):null,!xe&&!Te&&(N?.messages.length??0)===0?s.jsx("p",{className:"muted",children:"Диалог для этого кейса не найден."}):null,(N?.messages??[]).map((c,C)=>{const I=c.role==="assistant"?"assistant":"user";return s.jsxs("article",{className:`autoruns-msg ${I}`,children:[s.jsxs("header",{children:[s.jsx("strong",{children:I==="assistant"?"Система":"Модель/вопрос"}),s.jsx("span",{children:c.created_at?ei(c.created_at):"n/a"})]}),s.jsx("p",{children:c.text}),(c.trace_id||c.reply_type)&&s.jsxs("footer",{children:[c.trace_id?s.jsxs("span",{children:["trace=",c.trace_id]}):null,c.reply_type?s.jsxs("span",{children:["reply_type=",c.reply_type]}):null]})]},`${I}-${C}`)})]})]}),Ae?s.jsxs("section",{className:"autoruns-col",children:[s.jsx("h3",{children:"Режим ассистента"}),s.jsxs("div",{className:"autoruns-meta-list",children:[s.jsxs("div",{children:[s.jsx("span",{children:"source:"}),s.jsx("strong",{children:N?.source??"n/a"})]}),s.jsxs("div",{children:[s.jsx("span",{children:"session:"}),s.jsx("strong",{children:N?.session_id??"n/a"})]}),s.jsxs("div",{children:[s.jsx("span",{children:"run target:"}),s.jsx("strong",{children:Ge?.eval_target??"n/a"})]}),s.jsxs("div",{children:[s.jsx("span",{children:"run score:"}),s.jsx("strong",{children:Gr(Ge?.score_index??null)})]})]}),s.jsx("h4",{children:"Assistant mode payload"}),s.jsx(st,{value:N?.assistant_mode??{note:"assistant_mode unavailable"}}),s.jsx("h4",{style:{marginTop:12},children:"Case checks"}),s.jsx(st,{value:de?.checks??{note:"checks unavailable"}}),s.jsx("h4",{style:{marginTop:12},children:"Metric subscores"}),s.jsx(st,{value:de?.metric_subscores??{note:"metric_subscores unavailable"}})]}):null,qe?s.jsxs("section",{className:"autoruns-col",children:[s.jsx("h3",{children:"Режим декомпозиции"}),s.jsxs("div",{className:"autoruns-meta-list",children:[s.jsxs("div",{children:[s.jsx("span",{children:"Case:"}),s.jsx("strong",{children:de?.case_id??"n/a"})]}),s.jsxs("div",{children:[s.jsx("span",{children:"Domain:"}),s.jsx("strong",{children:de?.domain??"n/a"})]}),s.jsxs("div",{children:[s.jsx("span",{children:"Query class:"}),s.jsx("strong",{children:de?.query_class??"n/a"})]}),s.jsxs("div",{children:[s.jsx("span",{children:"Trace:"}),s.jsx("strong",{children:de?.trace_id??"n/a"})]})]}),s.jsx("h4",{children:"Шаги декомпозиции"}),(N?.decomposition.length??0)>0?s.jsx("ol",{className:"autoruns-decomposition-list",children:(N?.decomposition??[]).map((c,C)=>s.jsx("li",{children:c},`${C}-${c.slice(0,24)}`))}):s.jsx("p",{className:"muted",children:"В логах кейса нет явной декомпозиции."})]}):null,je?s.jsxs("section",{className:"autoruns-col",children:[s.jsx("h3",{children:"Прогресс / регресс"}),s.jsxs("div",{className:"autoruns-stats-grid",children:[s.jsxs("div",{children:[s.jsx("span",{children:"Latest score"}),s.jsx("strong",{children:Gr($?.stats.latest_score_index??null)})]}),s.jsxs("div",{children:[s.jsx("span",{children:"Previous"}),s.jsx("strong",{children:Gr($?.stats.previous_score_index??null)})]}),s.jsxs("div",{children:[s.jsx("span",{children:"Trend"}),s.jsx("strong",{children:$?Ya($.stats.trend):"n/a"})]}),s.jsxs("div",{children:[s.jsx("span",{children:"Quality gaps"}),s.jsx("strong",{children:$?.stats.quality_gap_runs??0})]})]}),s.jsx("h4",{children:"Покрытие доменов (история)"}),Xa($?.stats.domain_coverage??[]),s.jsx("h4",{style:{marginTop:14},children:"Покрытие доменов (выбранный прогон)"}),Xa(te?.coverage.domain_coverage??[])]}):null]})})}const pf=/(?:^|\n)\s*#{0,6}\s*(?:debug_payload_json|technical_breakdown_json|route_summary_json|debug_payload|technical_breakdown)\b/i,mf=[/\b(?:debug_payload_json|technical_breakdown_json)\b/i,/\b(?:route_summary|semantic_profile|domain_scope|relation_patterns|account_scope)\b/i,/\b(?:coverage_report|retrieval_status|problem_unit_state|candidate_evidence)\b/i,/\b(?:graph_domain_scope|graph_runtime|selection_reason|why_included)\b/i];function hf(u){try{return JSON.stringify(u,null,2)}catch{return String(u)}}function vf(u){const g=String(u??""),p=g.match(pf);return(p?g.slice(0,p.index):g).replace(/###\s*(?:debug_payload_json|technical_breakdown_json|route_summary_json)[\s\S]*?(?:```[\s\S]*?```|$)/gi,"").replace(/(?:^|\n)\s*#{0,6}\s*(?:debug_payload_json|technical_breakdown_json|route_summary_json)\b[\s\S]*$/gi,"").split(/\r?\n/g).map(M=>M.trimEnd()).filter(M=>M.trim().length>0).filter(M=>!mf.some($=>$.test(M))).join(` -`).trim()}function yf(u,g,p="default"){const O=p==="technical",k=[];k.push("# Assistant conversation export"),k.push(`session_id: ${u||"n/a"}`),k.push(`export_mode: ${p}`),k.push(`exported_at: ${new Date().toISOString()}`),k.push("");for(let E=0;E{b.current&&(b.current.scrollTop=b.current.scrollHeight)},[g,ce]),A.useEffect(()=>()=>{J.current!==null&&window.clearTimeout(J.current)},[]);async function xe(G){if(g.length===0)return;const ke=yf(u,g,G),ye=await Sf(ke);ve(G==="technical"?"тех":"чат"),Pe(ye?"success":"error"),J.current!==null&&window.clearTimeout(J.current),J.current=window.setTimeout(()=>{Pe("idle")},2200)}return s.jsxs(At,{title:"Режим ассистента",subtitle:"Диалоговый слой поверх normalizer, маршрутизации и factual retrieval.",actions:s.jsxs("div",{className:"assistant-panel-actions",children:[s.jsx("button",{type:"button",className:"assistant-copy-btn",onClick:()=>{xe("default")},disabled:g.length===0,title:"Экспорт только user-facing чата",children:"Скопировать чат"}),s.jsx("button",{type:"button",className:"assistant-copy-btn",onClick:()=>{xe("technical")},disabled:g.length===0,title:"Технический экспорт с debug payload",children:"Скопировать техчат"}),De==="success"?s.jsxs("span",{className:"assistant-copy-feedback success",children:["Скопировано (",Te,")"]}):null,De==="error"?s.jsx("span",{className:"assistant-copy-feedback error",children:"Ошибка копирования"}):null,s.jsx("span",{className:"status-chip",children:u?`session: ${u}`:"новая сессия"})]}),children:[s.jsxs("div",{ref:b,className:"assistant-chat-list",children:[g.length===0?s.jsx("div",{className:"assistant-empty muted",children:"Диалог пуст. Отправьте первый вопрос, чтобы запустить контур ассистента."}):null,g.map(G=>s.jsxs("article",{className:`assistant-msg ${G.role}`,children:[s.jsxs("header",{className:"assistant-msg-head",children:[s.jsx("strong",{children:gf(G.role)}),s.jsx("span",{children:xf(G.created_at)})]}),s.jsx("div",{className:"assistant-msg-body",children:G.text}),G.role==="assistant"&&G.debug?s.jsxs("details",{className:"assistant-debug",children:[s.jsx("summary",{children:"Показать технический разбор"}),s.jsx(st,{value:G.debug})]}):null]},G.message_id))]}),s.jsxs("div",{className:"assistant-compose",children:[s.jsxs("div",{className:"grid-two",children:[s.jsxs("label",{children:["Подсказка по периоду",s.jsx("input",{value:k,onChange:G=>E(G.target.value)})]}),s.jsxs("label",{children:["Бизнес-контекст",s.jsx("input",{value:M,onChange:G=>$(G.target.value)})]})]}),s.jsxs("label",{className:"full-width",children:["Сообщение",s.jsx("textarea",{value:p,onChange:G=>O(G.target.value),rows:4,placeholder:"Введите вопрос к данным компании..."})]}),s.jsxs("div",{className:"button-row",children:[s.jsxs("label",{className:"checkbox-row",children:[s.jsx("input",{type:"checkbox",checked:H,onChange:G=>te(G.target.checked)}),"Mock-режим"]}),s.jsx("button",{type:"button",onClick:()=>oe(),disabled:F||!p.trim(),children:F?"Выполняю...":"Отправить"}),s.jsx("button",{type:"button",onClick:()=>N(),disabled:F&&g.length===0,children:"Сбросить сессию"})]}),ce?s.jsx("p",{className:"diff-summary",children:ce}):null,he?s.jsx("p",{className:"error-text",children:he}):null]})]})}function qa({value:u,modelOptions:g,modelsBusy:p,onChange:O,onReloadModels:k,onTestConnection:E,onSaveLocalConfig:M,lastStatus:$,busy:H}){const te=u.llmProvider==="local",oe=g.includes(u.model);return s.jsxs(At,{title:"LLM Connection",subtitle:"Switch between OpenAI cloud and local OpenAI-compatible server.",actions:s.jsx("span",{className:"status-chip",children:$||"Status: not checked"}),children:[s.jsxs("div",{className:"grid-two",children:[s.jsxs("label",{children:["Provider",s.jsxs("select",{value:u.llmProvider,onChange:N=>{const F=N.target.value==="local"?"local":"openai";O({...u,llmProvider:F,baseUrl:F==="local"?"http://127.0.0.1:1234/v1":"https://api.openai.com/v1"})},children:[s.jsx("option",{value:"openai",children:"OpenAI (token)"}),s.jsx("option",{value:"local",children:"Local (LM Studio / OpenAI-compatible)"})]})]}),s.jsxs("label",{children:["Model",s.jsxs("select",{value:oe?u.model:"__manual__",onChange:N=>{const F=N.target.value;F!=="__manual__"&&O({...u,model:F})},children:[s.jsx("option",{value:"__manual__",children:"Manual input"}),g.map(N=>s.jsx("option",{value:N,children:N},N))]})]}),s.jsxs("label",{children:["Model ID (manual)",s.jsx("input",{value:u.model,onChange:N=>O({...u,model:N.target.value}),placeholder:"qwen2.5-14b-instruct or lmstudio loaded model id"})]}),te?null:s.jsxs("label",{className:"full-width",children:["OpenAI API Key",s.jsx("input",{type:"password",value:u.apiKey,onChange:N=>O({...u,apiKey:N.target.value}),placeholder:"sk-..."})]}),s.jsxs("label",{className:te?"full-width":void 0,children:[te?"Local server base URL":"Base URL",s.jsx("input",{value:u.baseUrl,onChange:N=>O({...u,baseUrl:N.target.value}),placeholder:te?"http://127.0.0.1:1234/v1":"https://api.openai.com/v1"})]}),s.jsxs("label",{children:["Temperature",s.jsx("input",{type:"number",step:"0.1",value:u.temperature,onChange:N=>O({...u,temperature:Number(N.target.value)})})]}),s.jsxs("label",{children:["Max output tokens",s.jsx("input",{type:"number",value:u.maxOutputTokens,onChange:N=>O({...u,maxOutputTokens:Number(N.target.value)})})]})]}),s.jsxs("div",{className:"button-row",children:[s.jsx("button",{type:"button",onClick:()=>M(),children:"Save local config"}),s.jsx("button",{type:"button",onClick:()=>k(),disabled:H||p,children:p?"Loading models...":"Load model list"}),s.jsx("button",{type:"button",onClick:()=>E(),disabled:H,children:H?"Checking...":"Test connection"})]})]})}function wf({items:u,onRefresh:g,onOpenTrace:p}){return s.jsx(At,{title:"История нормализаций",subtitle:"Короткий вопрос, confidence, route hint и статус валидации.",actions:s.jsx("button",{type:"button",onClick:()=>g(),children:"Обновить"}),children:s.jsxs("div",{className:"history-list",children:[u.length===0?s.jsx("p",{className:"muted",children:"История пока пустая."}):null,u.map(O=>s.jsxs("button",{type:"button",className:"history-item",onClick:()=>p(O.trace_id),children:[s.jsxs("div",{className:"history-row",children:[s.jsx("strong",{children:O.route_hint??"route: n/a"}),s.jsx("span",{children:O.validation_passed?"schema: ok":"schema: fail"})]}),s.jsx("p",{children:O.question_short}),s.jsxs("div",{className:"history-row",children:[s.jsx("span",{children:O.model}),s.jsx("span",{children:new Date(O.timestamp).toLocaleString("ru-RU")})]})]},O.trace_id))]})})}function Ut(u){return u==null||u===""?"—":String(u)}function kf({result:u}){return s.jsx(At,{title:"Runtime метрики",subtitle:"trace_id, токены, latency и статус валидации.",children:s.jsxs("div",{className:"metrics-grid",children:[s.jsxs("div",{children:[s.jsx("span",{children:"trace_id"}),s.jsx("strong",{children:Ut(u?.trace_id)})]}),s.jsxs("div",{children:[s.jsx("span",{children:"request_started_at"}),s.jsx("strong",{children:Ut(u?new Date(Date.now()-u.latency_ms).toISOString():null)})]}),s.jsxs("div",{children:[s.jsx("span",{children:"request_finished_at"}),s.jsx("strong",{children:Ut(u?new Date().toISOString():null)})]}),s.jsxs("div",{children:[s.jsx("span",{children:"latency_ms"}),s.jsx("strong",{children:Ut(u?.latency_ms)})]}),s.jsxs("div",{children:[s.jsx("span",{children:"input_tokens"}),s.jsx("strong",{children:Ut(u?.usage?.input_tokens)})]}),s.jsxs("div",{children:[s.jsx("span",{children:"output_tokens"}),s.jsx("strong",{children:Ut(u?.usage?.output_tokens)})]}),s.jsxs("div",{children:[s.jsx("span",{children:"total_tokens"}),s.jsx("strong",{children:Ut(u?.usage?.total_tokens)})]}),s.jsxs("div",{children:[s.jsx("span",{children:"validation_status"}),s.jsx("strong",{children:u?.validation?.passed?"passed":"failed"})]}),s.jsxs("div",{children:[s.jsx("span",{children:"prompt_version"}),s.jsx("strong",{children:Ut(u?.prompt_version)})]}),s.jsxs("div",{children:[s.jsx("span",{children:"schema_version"}),s.jsx("strong",{children:Ut(u?.schema_version)})]})]})})}const jf={normalized:"Normalized JSON",fragments:"Fragment View",scope:"Scope View",flags:"Flags View",route:"Route Simulation",raw:"Raw model output",validation:"Validation",logs:"Logs"};function Cf(u){return u&&typeof u=="object"?u:null}function Ef({tab:u,onTabChange:g,result:p,appLogs:O}){const k=["normalized","fragments","scope","flags","route","raw","validation","logs"],E=Cf(p?.normalized),M=String(E?.schema_version??""),$=M==="normalized_query_v2"||M==="normalized_query_v2_0_1"||M==="normalized_query_v2_0_2",H=$?{fragments:E?.fragments??[],discarded_fragments:E?.discarded_fragments??[]}:{note:"Fragment View доступен для normalized_query_v2."},te=$?{message_in_scope:E?.message_in_scope??null,scope_confidence:E?.scope_confidence??null,contains_multiple_tasks:E?.contains_multiple_tasks??null,global_notes:E?.global_notes??null}:{note:"Scope View доступен для normalized_query_v2."},oe=$?Array.isArray(E?.fragments)?(E?.fragments).map(N=>({fragment_id:N.fragment_id??null,domain_relevance:N.domain_relevance??null,candidate_labels:N.candidate_labels??[],execution_readiness:N.execution_readiness??null,clarification_reason:N.clarification_reason??null,soft_assumption_used:N.soft_assumption_used??[],route_status:N.route_status??null,no_route_reason:N.no_route_reason??null,flags:N.flags??{}})):[]:{note:"Flags View доступен для normalized_query_v2."};return s.jsxs(At,{title:"Выходные данные",subtitle:"Structured output и диагностические вкладки.",children:[s.jsx("div",{className:"tab-row",children:k.map(N=>s.jsx("button",{type:"button",className:u===N?"tab active":"tab",onClick:()=>g(N),children:jf[N]},N))}),u==="normalized"?s.jsx(st,{value:p?.normalized??{note:"Нет данных."}}):null,u==="fragments"?s.jsx(st,{value:H}):null,u==="scope"?s.jsx(st,{value:te}):null,u==="flags"?s.jsx(st,{value:oe}):null,u==="route"?s.jsx(st,{value:p?.route_hint_summary??{note:"Нет данных."}}):null,u==="raw"?s.jsx(st,{value:p?.raw_model_output??{note:"Нет данных."}}):null,u==="validation"?s.jsx(st,{value:p?.validation??{note:"Нет данных."}}):null,u==="logs"?s.jsx(st,{value:O}):null]})}function Ga({value:u,onChange:g,presets:p,selectedPresetId:O,onSelectPreset:k,onLoadPreset:E,onSavePreset:M,onResetDefaults:$,onDiffPrevious:H,presetName:te,onPresetNameChange:oe,diffSummary:N}){return s.jsxs(At,{title:"Prompt Manager",subtitle:"Системный, developer и domain уровни управляются отдельно.",children:[s.jsxs("div",{className:"grid-two",children:[s.jsxs("label",{children:["Системный prompt",s.jsx("textarea",{value:u.systemPrompt,onChange:F=>g({...u,systemPrompt:F.target.value}),rows:6})]}),s.jsxs("label",{children:["Developer / Instruction prompt",s.jsx("textarea",{value:u.developerPrompt,onChange:F=>g({...u,developerPrompt:F.target.value}),rows:6})]}),s.jsxs("label",{children:["Domain prompt",s.jsx("textarea",{value:u.domainPrompt,onChange:F=>g({...u,domainPrompt:F.target.value}),rows:6})]}),s.jsxs("label",{children:["Schema notes",s.jsx("textarea",{value:u.schemaNotes,onChange:F=>g({...u,schemaNotes:F.target.value}),rows:6})]}),s.jsxs("label",{className:"full-width",children:["Few-shot examples",s.jsx("textarea",{value:u.fewShotExamples,onChange:F=>g({...u,fewShotExamples:F.target.value}),rows:8})]})]}),s.jsxs("div",{className:"button-row",children:[s.jsxs("select",{value:O,onChange:F=>k(F.target.value),children:[s.jsx("option",{value:"",children:"Выберите preset..."}),p.map(F=>s.jsx("option",{value:F.id,children:F.name},F.id))]}),s.jsx("button",{type:"button",onClick:()=>E(),children:"Загрузить preset"}),s.jsx("input",{value:te,onChange:F=>oe(F.target.value),placeholder:"Имя для сохранения"}),s.jsx("button",{type:"button",onClick:()=>M(),children:"Сохранить preset"}),s.jsx("button",{type:"button",onClick:()=>H(),children:"Diff с предыдущим"}),s.jsx("button",{type:"button",onClick:()=>$(),children:"Сбросить к default"})]}),N?s.jsx("p",{className:"diff-summary",children:N}):null]})}function Nf({value:u,onChange:g,onApplyBatchFormat:p,onNormalize:O,busy:k,useMock:E,onUseMockChange:M,errorMessage:$}){return s.jsxs(At,{title:"Запрос пользователя",subtitle:"NDC semantic front-end: нормализуем, но не отвечаем за бухгалтерскую суть.",children:[s.jsxs("div",{className:"grid-two",children:[s.jsxs("label",{className:"full-width",children:["Raw user question",s.jsx("textarea",{value:u.userQuestion,onChange:H=>g({...u,userQuestion:H.target.value}),rows:6,placeholder:"Например: По каким покупателям у нас на конец июня висят отгрузки без оплаты..."})]}),s.jsxs("label",{className:"full-width",children:["Batch queries (`;` separator)",s.jsx("textarea",{value:u.batchQuestionsRaw,onChange:H=>g({...u,batchQuestionsRaw:H.target.value}),onBlur:()=>p(),rows:8,placeholder:"Вопрос 1; Вопрос 2; Вопрос 3"})]}),s.jsxs("label",{children:["Optional period context",s.jsx("input",{value:u.periodHint,onChange:H=>g({...u,periodHint:H.target.value})})]}),s.jsxs("label",{children:["Optional business context",s.jsx("input",{value:u.businessContext,onChange:H=>g({...u,businessContext:H.target.value})})]}),s.jsxs("label",{children:["Optional expected route (eval)",s.jsx("input",{value:u.expectedRoute,onChange:H=>g({...u,expectedRoute:H.target.value})})]})]}),s.jsxs("div",{className:"button-row",children:[s.jsxs("label",{className:"checkbox-row",children:[s.jsx("input",{type:"checkbox",checked:E,onChange:H=>M(H.target.checked)}),"Mock-режим (без вызова OpenAI)"]}),s.jsx("button",{type:"button",onClick:()=>p(),disabled:k||!u.batchQuestionsRaw.trim(),children:"Применить `;` в переносы"}),s.jsx("button",{type:"button",onClick:()=>O(!1),disabled:k||!u.userQuestion.trim(),children:k?"Нормализуем...":"Normalize"}),s.jsx("button",{type:"button",onClick:()=>O(!0),disabled:k||!u.userQuestion.trim(),children:k?"Сохраняем...":"Normalize + Save as test case"})]}),$?s.jsx("p",{className:"error-text",children:$}):null]})}function Pf({runs:u,selectedRunId:g,onSelectRun:p,onStartRun:O,onFinishRun:k,onRefreshRuns:E,onRunEval:M,onCopyEvalReport:$,evalBusy:H,traceItems:te,evalReport:oe}){return s.jsxs(At,{title:"NDC Run Monitor",subtitle:"Важно: кнопка Запустить run создает только run-сущность. Кнопка eval запускает batch-проверку normalizer v2.0.2.",children:[s.jsxs("div",{className:"button-row",children:[s.jsx("button",{type:"button",onClick:()=>O(),children:"Запустить run"}),s.jsx("button",{type:"button",onClick:()=>k(),disabled:!g,children:"Завершить выбранный run"}),s.jsx("button",{type:"button",onClick:()=>E(),children:"Обновить runs"}),s.jsx("button",{type:"button",onClick:()=>M(),disabled:H,children:H?"Идет eval v2.0.2...":"Запустить eval v2.0.2"})]}),s.jsxs("div",{className:"runtime-grid",children:[s.jsxs("div",{className:"runtime-runs",children:[u.map(N=>s.jsxs("button",{type:"button",className:g===N.runId?"history-item selected":"history-item",onClick:()=>p(N.runId),children:[s.jsxs("div",{className:"history-row",children:[s.jsx("strong",{children:N.status}),s.jsx("span",{children:N.runId})]}),s.jsxs("div",{className:"history-row",children:[s.jsx("span",{children:N.sessionId}),s.jsx("span",{children:new Date(N.updatedAt).toLocaleString("ru-RU")})]})]},N.runId)),u.length===0?s.jsx("p",{className:"muted",children:"Нет активных запусков."}):null]}),s.jsxs("div",{children:[s.jsx("h3",{children:"Trace выбранного run"}),s.jsx(st,{value:te}),s.jsxs("div",{className:"eval-report-wrap",children:[s.jsx("h3",{style:{marginTop:12},children:"Отчет eval"}),s.jsx(st,{value:oe??{note:"Eval пока не запускался"}}),s.jsx("button",{type:"button",className:"copy-cube-button",title:"Скопировать отчет eval",onClick:()=>$(),children:"⧉"})]})]})]})]})}const Tf={llmProvider:"openai",apiKey:"",model:"gpt-4o-mini",baseUrl:"https://api.openai.com/v1",temperature:0,maxOutputTokens:700},Ja={systemPrompt:"Ты semantic-normalizer для бухгалтерского ассистента NDC. Возвращай только JSON по схеме normalized_query_v2_0_2.",developerPrompt:"Сначала делай decomposition сообщения на task fragments, затем определяй domain scope и route-critical flags. Для каждого fragment заполняй execution_readiness + route_status + no_route_reason. Если fragment routable, не оставляй его в no_route.",domainPrompt:"Контур: данные текущего предприятия в 1С/NDC. In-scope: документы, проводки, взаиморасчеты, остатки, периодное закрытие, аномалии и контрольные проверки. Out-of-scope: общая теория, законы и оффтоп.",schemaNotes:"schema_version: normalized_query_v2_0_2. Строгий JSON без дополнительных полей.",fewShotExamples:"Q: Проверь по поставщикам хвосты и разложи цепочку документов/оплат. => fragment in_scope, flags: multi_entity + chain_explanation. Q: Как вообще по ФСБУ? => out_of_scope/generic_accounting."},Rf={userQuestion:"",batchQuestionsRaw:"",periodHint:"",businessContext:"",expectedRoute:""},Za="ndc_normalizer_session_config_v1",ti=["Analyzing request","Fetching data","Composing answer"],zf="assistant",ni="normalizer_v2_0_2",ba="address_query_runtime_v1";function Lf(u){return`[${new Date().toLocaleTimeString("ru-RU")}] ${u}`}function Of(u,g){if(!g)return"Previous preset is not selected.";const O=["systemPrompt","developerPrompt","domainPrompt","schemaNotes","fewShotExamples"].filter(k=>u[k]!==g[k]).map(k=>`${k}: ${Math.abs(u[k].length-g[k].length)} chars delta`);return O.length===0?"No changes against previous preset.":`Changed fields: ${O.length}. ${O.join(" | ")}`}function Df(){const[u,g]=A.useState(Tf),[p,O]=A.useState(Ja),[k,E]=A.useState(Rf),[M,$]=A.useState(null),[H,te]=A.useState([]),[oe,N]=A.useState([]),[F,ce]=A.useState("normalized"),[he,b]=A.useState(!1),[J,De]=A.useState(!1),[Pe,Te]=A.useState([]),[ve,xe]=A.useState(""),[G,ke]=A.useState([]),[ye,Ae]=A.useState(""),[Xe,qe]=A.useState("NDC custom preset"),[be,je]=A.useState(null),[He,Ge]=A.useState(""),[de,se]=A.useState(!1),[P,B]=A.useState([]),[T,m]=A.useState(""),[c,C]=A.useState([]),[I,Y]=A.useState(!1),[Z,ue]=A.useState(null),[re,ne]=A.useState(""),[Re,$t]=A.useState(zf),[Rn,fn]=A.useState(""),[or,zn]=A.useState([]),[sr,pn]=A.useState(""),[ir,Ln]=A.useState(!1),[mn,dt]=A.useState(""),[ur,On]=A.useState(""),Dn=A.useRef(!1),q=x=>{N(V=>[Lf(x),...V].slice(0,300))};function Jr(){let x=0;dt(ti[0]);const V=window.setInterval(()=>{x=Math.min(x+1,ti.length-1),dt(ti[x])},650);return()=>window.clearInterval(V)}A.useEffect(()=>{const x=localStorage.getItem(Za);if(x)try{const V=JSON.parse(x);g(Q=>({...Q,llmProvider:V.llmProvider==="local"?"local":"openai",model:V.model??Q.model,baseUrl:V.baseUrl??Q.baseUrl,temperature:V.temperature??Q.temperature,maxOutputTokens:V.maxOutputTokens??Q.maxOutputTokens}))}catch{}Vt(),hn(),vn()},[]);async function Vt(){try{const x=await Ye.loadHistory();te(x.items??[])}catch(x){q(`History load error: ${x instanceof Error?x.message:String(x)}`)}}async function hn(){try{const V=(await Ye.loadPresets()).presets??[];if(ke(V),Dn.current)return;const Q=V.find(ze=>ze.prompt_version===ni)??V.find(ze=>ze.id==="default-normalizer-v2_0_2");if(!Q){Dn.current=!0,q(`Preset autoload skipped: ${ni} not found.`);return}Ae(Q.id),je(p),O({systemPrompt:Q.systemPrompt,developerPrompt:Q.developerPrompt,domainPrompt:Q.domainPrompt,schemaNotes:Q.schemaNotes??"",fewShotExamples:Q.fewShotExamples??""}),Dn.current=!0,q(`Preset autoloaded: ${Q.name} (${Q.prompt_version}).`)}catch(x){q(`Presets load error: ${x instanceof Error?x.message:String(x)}`)}}async function vn(){try{const x=await Ye.listRuns();B(x.items??[])}catch(x){q(`Runs load error: ${x instanceof Error?x.message:String(x)}`)}}function Ht(){localStorage.setItem(Za,JSON.stringify({model:u.model,llmProvider:u.llmProvider,baseUrl:u.baseUrl,temperature:u.temperature,maxOutputTokens:u.maxOutputTokens})),q("Local config saved (without API key).")}async function Bt(){b(!0),ne("");try{const x=await Ye.testConnection(u);x.provider==="local"?x.model_found===!0?(xe(`LOCAL OK - ${x.model}`),q(`Local model is available: ${x.model} (catalog size=${x.models_count??"n/a"}).`)):x.model_found===!1?(xe(`LOCAL OK, model not loaded - ${x.model}`),q(`Local server is reachable, but model '${x.model}' is not in loaded catalog. Use 'Load model list' and select one of loaded models.`)):(xe(`LOCAL OK (model list unavailable) - ${x.model}`),q("Local server is reachable, but model catalog could not be verified.")):(xe(`OPENAI OK - ${x.model}`),q(`OpenAI connection ok: ${x.model}`))}catch(x){const V=x instanceof Error?x.message:String(x);xe("Connection error"),ne(`Test connection: ${V}`),q(`Test connection error: ${V}`)}finally{b(!1)}}async function Zr(){De(!0);try{const V=(await Ye.listModels(u)).models??[];Te(V),V.length>0&&g(Q=>Q.model&&V.includes(Q.model)?Q:{...Q,model:V[0]}),q(`Model catalog loaded (${u.llmProvider}): ${V.length} items.`)}catch(x){const V=x instanceof Error?x.message:String(x);q(`Load model list error: ${V}`)}finally{De(!1)}}A.useEffect(()=>{Te([])},[u.llmProvider,u.baseUrl]);async function br(x){b(!0),ne("");try{const V=await Ye.normalize({connection:u,prompts:p,promptVersion:"normalizer_v2_0_2",query:{userQuestion:k.userQuestion,periodHint:k.periodHint,businessContext:k.businessContext,expectedRoute:k.expectedRoute},saveAsTestCase:x,useMock:de});$(V),ce("normalized"),q(`Normalize done: trace=${V.trace_id}, validation=${V.validation.passed?"passed":"failed"}`),Vt()}catch(V){const Q=V instanceof Error?V.message:String(V);ne(`Normalize: ${Q}`),q(`Normalize error: ${Q}`)}finally{b(!1)}}function ar(){const x=G.find(V=>V.id===ye);if(!x){q("Preset is not selected.");return}je(p),O({systemPrompt:x.systemPrompt,developerPrompt:x.developerPrompt,domainPrompt:x.domainPrompt,schemaNotes:x.schemaNotes??"",fewShotExamples:x.fewShotExamples??""}),q(`Preset loaded: ${x.name}`)}async function el(){try{await Ye.savePreset({name:Xe||"NDC preset",prompt_version:"normalizer_v2_0_2",systemPrompt:p.systemPrompt,developerPrompt:p.developerPrompt,domainPrompt:p.domainPrompt,schemaNotes:p.schemaNotes,fewShotExamples:p.fewShotExamples}),q("Preset saved."),await hn()}catch(x){q(`Preset save error: ${x instanceof Error?x.message:String(x)}`)}}function Mn(){O(Ja),q("Prompt panel reset to defaults.")}function In(){const x=Of(p,be);Ge(x),q(x)}function cr(){const x=k.batchQuestionsRaw.split(";").map(V=>V.trim()).filter(Boolean).join(` - -`);x&&(E(V=>({...V,batchQuestionsRaw:x})),q("Batch field formatted: `;` converted to blank-line separators."))}async function dr(x){try{const Q=(await Ye.loadTrace(x)).trace,ze=Q.parsed_normalized_json??null;$({trace_id:String(Q.trace_id??x),ok:!!Q.validation_result?.passed,normalized:ze,route_hint_summary:Q.route_hint_summary??(ze?{route_hint:ze.route_hint??null,confidence:ze.confidence?.route_hint??null}:null),raw_model_output:Q.raw_model_response??{},validation:Q.validation_result??{passed:!1,errors:["validation not found"]},usage:Q.usage??{input_tokens:0,output_tokens:0,total_tokens:0},latency_ms:Number(Q.latency_ms??0),prompt_version:String(Q.prompt_version??"unknown"),schema_version:String(Q.schema_version??"unknown")}),ce("raw"),ne(""),q(`Trace opened: ${x}`)}catch(V){const Q=V instanceof Error?V.message:String(V);ne(`Trace: ${Q}`),q(`Trace open error ${x}: ${Q}`)}}async function fr(){try{const x=await Ye.startRun();m(x.run.runId),q(`Run started: ${x.run.runId}`),q("Tip: start run does not execute normalize by itself. Use 'Run eval v2.0.2' button."),await vn()}catch(x){q(`Run start error: ${x instanceof Error?x.message:String(x)}`)}}async function Qt(){if(T)try{await Ye.finishRun(T),q(`Run finished: ${T}`),await vn()}catch(x){q(`Run finish error: ${x instanceof Error?x.message:String(x)}`)}}async function Wt(){Y(!0),ne("");try{q("Starting eval in v2 contour.");const x=k.batchQuestionsRaw.trim()||k.userQuestion.trim();if(!x)throw new Error("Fill batch field or Raw user question first.");const V=await Ye.runEval({connection:u,prompts:p,promptVersion:"normalizer_v2_0_2",mode:"single-pass-strict",rawQuestions:x,useMock:de});ue(V.report),q("Eval v2.0.2 run finished.");const Q=V.report;if(Q.run_id&&q(`Eval run id: ${Q.run_id}`),Q.metrics){const ze=Q.metrics;q(`Eval metrics v2.0.2: schema=${ze.schema_validation_pass_rate??"n/a"}%, route_accuracy=${ze.route_resolution_accuracy??"n/a"}%, no_route_precision=${ze.no_route_precision??"n/a"}%, state_consistency=${ze.execution_state_consistency_rate??"n/a"}%`)}await Vt()}catch(x){const V=x instanceof Error?x.message:String(x);V.includes("Legacy eval runner supports normalized_query_v1 only")?(ue({status:"plan_only",prompt_version:"normalizer_v2",reason:"backend eval runner is still legacy-v1 only",plan_file:"reports/v2_pilot_eval_plan.md",next_steps:["run cheap mock sanity for schema/fragment/scope","run small real batch (10-15 messages, temperature=0)","run challenge-30 replay with v2 metrics"]}),q("Backend is legacy-only for eval right now. Showing v2 pilot plan.")):(ne(`Eval: ${V}`),q(`Eval run error: ${V}`))}finally{Y(!1)}}async function tl(){try{const x=JSON.stringify(Z??{},null,2);await navigator.clipboard.writeText(x),q("Eval report copied to clipboard.")}catch(x){q(`Eval report copy error: ${x instanceof Error?x.message:String(x)}`)}}function nl(){fn(""),zn([]),pn(""),dt(""),On(""),q("Assistant session reset.")}async function rl(){const x=sr.trim();if(!x)return;Ln(!0),On(""),pn(""),zn(Q=>[...Q,{message_id:`local-${Date.now()}`,session_id:Rn||"pending",role:"user",text:x,reply_type:null,created_at:new Date().toISOString(),trace_id:null,debug:null}]);const V=Jr();try{const Q=await Ye.sendAssistantMessage({connection:u,prompts:p,userMessage:x,sessionId:Rn||void 0,promptVersion:ba,context:{periodHint:k.periodHint,businessContext:k.businessContext},useMock:de});fn(Q.session_id),zn(Q.conversation),dt("Reply is ready"),q(`Assistant reply received: trace=${Q.debug.trace_id}`)}catch(Q){const ze=Q instanceof Error?Q.message:String(Q);On(ze),dt("Assistant error"),q(`Assistant error: ${ze}`)}finally{V(),Ln(!1)}}return A.useEffect(()=>{if(!T){C([]);return}Ye.runTrace(T).then(x=>C(x.items)).catch(x=>q(`Run trace error: ${x instanceof Error?x.message:String(x)}`))},[T]),s.jsxs("main",{className:"app-root",children:[s.jsxs("div",{className:"hero",children:[s.jsx("h1",{children:"NDC AI First Layer"}),s.jsx("p",{children:"Three modes in one UI: assistant, decomposition diagnostics, and auto-run history with regression visibility."})]}),s.jsxs("div",{className:"mode-switch-row",children:[s.jsx("button",{type:"button",className:Re==="assistant"?"tab active":"tab",onClick:()=>$t("assistant"),children:"Assistant"}),s.jsx("button",{type:"button",className:Re==="decomposition"?"tab active":"tab",onClick:()=>$t("decomposition"),children:"Decomposition"}),s.jsx("button",{type:"button",className:Re==="autoruns"?"tab active":"tab",onClick:()=>$t("autoruns"),children:"AutoRun History"})]}),Re==="assistant"?s.jsxs("div",{className:"layout-grid",children:[s.jsx(qa,{value:u,modelOptions:Pe,modelsBusy:J,onChange:g,onReloadModels:Zr,onSaveLocalConfig:Ht,onTestConnection:Bt,lastStatus:ve,busy:he||ir}),s.jsx(Ga,{value:p,onChange:O,presets:G,selectedPresetId:ye,onSelectPreset:Ae,onLoadPreset:ar,onSavePreset:el,onResetDefaults:Mn,onDiffPrevious:In,presetName:Xe,onPresetNameChange:qe,diffSummary:He}),s.jsx(_f,{sessionId:Rn,conversation:or,inputValue:sr,onInputChange:pn,periodHint:k.periodHint,onPeriodHintChange:x=>E(V=>({...V,periodHint:x})),businessContext:k.businessContext,onBusinessContextChange:x=>E(V=>({...V,businessContext:x})),useMock:de,onUseMockChange:se,onSend:rl,onClear:nl,busy:ir,statusText:mn,errorMessage:ur})]}):Re==="decomposition"?s.jsxs("div",{className:"layout-grid",children:[s.jsx(qa,{value:u,modelOptions:Pe,modelsBusy:J,onChange:g,onReloadModels:Zr,onSaveLocalConfig:Ht,onTestConnection:Bt,lastStatus:ve,busy:he}),s.jsx(Ga,{value:p,onChange:O,presets:G,selectedPresetId:ye,onSelectPreset:Ae,onLoadPreset:ar,onSavePreset:el,onResetDefaults:Mn,onDiffPrevious:In,presetName:Xe,onPresetNameChange:qe,diffSummary:He}),s.jsx(Nf,{value:k,onChange:E,onApplyBatchFormat:cr,onNormalize:br,busy:he,useMock:de,onUseMockChange:se,errorMessage:re}),s.jsx(Ef,{tab:F,onTabChange:ce,result:M,appLogs:oe}),s.jsx(kf,{result:M}),s.jsx(wf,{items:H,onRefresh:Vt,onOpenTrace:dr}),s.jsx(Pf,{runs:P,selectedRunId:T,onSelectRun:m,onStartRun:fr,onFinishRun:Qt,onRefreshRuns:vn,onRunEval:Wt,onCopyEvalReport:tl,evalBusy:I,traceItems:c,evalReport:Z})]}):s.jsx("div",{className:"layout-grid",children:s.jsx(ff,{connection:u,prompts:p,assistantPromptVersion:ba,decompositionPromptVersion:ni,onLog:q})})]})}of.createRoot(document.getElementById("root")).render(s.jsx(Zd.StrictMode,{children:s.jsx(Df,{})})); diff --git a/llm_normalizer/frontend/dist/assets/index-iFxz5cXp.css b/llm_normalizer/frontend/dist/assets/index-iFxz5cXp.css new file mode 100644 index 0000000..3921e16 --- /dev/null +++ b/llm_normalizer/frontend/dist/assets/index-iFxz5cXp.css @@ -0,0 +1 @@ +@import"https://fonts.googleapis.com/css2?family=Manrope:wght@400;600;700;800&family=Space+Grotesk:wght@500;700&display=swap";:root{--rgb-background: 16, 16, 19;--rgb-surface-main: 26, 26, 31;--rgb-surface-horizontal: 32, 32, 38;--rgb-surface-focus: 40, 40, 47;--rgb-active: 228, 142, 92;--rgb-active-text: 18, 18, 18;--rgb-text-main: 240, 240, 240;--rgb-text-muted: 166, 166, 170;--rgb-danger: 255, 126, 126;--rgb-scrollbar-track: 31, 31, 36;--rgb-scrollbar-thumb: 74, 74, 82;--rgb-scrollbar-thumb-hover: 90, 90, 100;--bg-main: rgb(var(--rgb-background));--bg-soft: rgb(var(--rgb-surface-main));--bg-panel: rgb(var(--rgb-surface-main));--bg-panel-accent: rgb(var(--rgb-surface-horizontal));--surface-horizontal: rgb(var(--rgb-surface-horizontal));--surface-focus: rgb(var(--rgb-surface-focus));--line: transparent;--line-strong: rgba(var(--rgb-active), .48);--text-main: rgb(var(--rgb-text-main));--text-muted: rgb(var(--rgb-text-muted));--lime-main: rgb(var(--rgb-text-main));--lime-press: rgb(var(--rgb-text-main));--danger: rgb(var(--rgb-danger));--radius-lg: 20px;--radius-md: 14px;--shadow: none;--autoruns-col-width: 360px}*{box-sizing:border-box;scrollbar-width:thin;scrollbar-color:rgb(var(--rgb-scrollbar-thumb)) rgb(var(--rgb-scrollbar-track))}*::-webkit-scrollbar{width:10px;height:10px}*::-webkit-scrollbar-track{background:rgb(var(--rgb-scrollbar-track))}*::-webkit-scrollbar-thumb{background:rgb(var(--rgb-scrollbar-thumb));border-radius:999px;border:2px solid rgb(var(--rgb-scrollbar-track))}*::-webkit-scrollbar-thumb:hover{background:rgb(var(--rgb-scrollbar-thumb-hover))}html,body,#root{margin:0;min-height:100dvh;font-family:Manrope,Segoe UI,sans-serif;background:var(--bg-main);color:var(--text-main)}.app-root{max-width:1720px;margin:0 auto;padding:12px 16px 16px}.app-root.app-root-autoruns{max-width:none;width:100%;min-height:100dvh;max-height:100dvh;display:flex;flex-direction:column;overflow:hidden}.app-topbar{display:flex;align-items:center;justify-content:space-between;gap:12px;margin:0 0 12px;padding:0;min-height:38px}.layout-grid{display:grid;grid-template-columns:repeat(12,1fr);gap:16px}.layout-grid.layout-grid-autoruns{min-height:0;flex:1 1 auto;grid-template-columns:minmax(0,1fr)}.mode-switch-row{display:flex;gap:8px;margin:0;padding:0}.mode-switch-row.mode-switch-row-right{margin-left:auto;justify-content:flex-end}.panel-frame{grid-column:span 12;border:none;border-radius:var(--radius-lg);background:var(--bg-panel);overflow:hidden;box-shadow:none;animation:rise .4s ease-out;display:flex;flex-direction:column;min-height:0}.panel-header{display:flex;align-items:flex-start;justify-content:space-between;gap:14px;padding:14px 18px 10px;border-bottom:none;background:var(--bg-panel-accent)}.panel-header h2{margin:0;font-size:1.02rem;letter-spacing:.02em}.panel-header p{margin:6px 0 0;font-size:.85rem;color:var(--text-muted)}.panel-body{padding:10px 12px 12px;min-height:0}.app-root-autoruns .autoruns-frame{height:100%}.app-root-autoruns .autoruns-frame .panel-body{flex:1 1 auto;overflow:hidden;display:flex;flex-direction:column;gap:10px;padding:10px 12px 12px;background:rgb(var(--rgb-background))}.status-chip{border:none;border-radius:999px;padding:4px 10px;color:var(--lime-main);font-size:.78rem;background:rgb(var(--rgb-surface-focus))}.assistant-panel-actions{display:flex;align-items:center;justify-content:flex-end;flex-wrap:wrap;gap:8px}.assistant-copy-btn{background:transparent;border-color:transparent;color:var(--text-main);box-shadow:none;transform:none}.assistant-copy-btn:hover{background:rgb(var(--rgb-surface-focus));filter:none;box-shadow:none;transform:none}.assistant-copy-feedback{font-size:.76rem;color:var(--text-muted)}.assistant-copy-feedback.success{color:var(--lime-main)}.assistant-copy-feedback.error{color:var(--danger)}input,select,textarea,button{font-family:Manrope,sans-serif}label{display:flex;flex-direction:column;gap:6px;color:var(--text-muted);font-size:.84rem}input,select,textarea{border:none;border-radius:var(--radius-md);background:rgb(var(--rgb-surface-horizontal));color:var(--text-main);padding:10px 12px;outline:none;transition:background-color .18s ease}input:focus,select:focus,textarea:focus{border-color:transparent;box-shadow:none;outline:none;background:rgb(var(--rgb-surface-focus))}textarea{resize:vertical;min-height:86px}button{border:none;border-radius:999px;background:rgb(var(--rgb-surface-horizontal));color:rgb(var(--rgb-text-main));font-weight:700;font-size:.83rem;letter-spacing:.02em;cursor:pointer;padding:9px 14px;transition:background .2s ease,color .2s ease;outline:none;box-shadow:none}button:hover{border-color:transparent;background:rgb(var(--rgb-surface-focus))}button:disabled{opacity:.52;cursor:not-allowed}.button-row{display:flex;flex-wrap:wrap;align-items:center;gap:10px;margin-top:12px}.checkbox-row{flex-direction:row;align-items:center;gap:8px;color:var(--text-main)}.grid-two{display:grid;grid-template-columns:repeat(2,minmax(0,1fr));gap:12px}.full-width{grid-column:1 / -1}.tab-row{display:flex;flex-wrap:wrap;gap:8px;margin-bottom:12px}.tab{background:rgb(var(--rgb-surface-main));color:var(--text-main);border:none}.tab.active{border-color:transparent;background:rgb(var(--rgb-active));color:rgb(var(--rgb-active-text))}.assistant-chat-list{max-height:420px;overflow:auto;display:grid;gap:10px;padding:4px;border:none;border-radius:var(--radius-md);background:rgb(var(--rgb-surface-horizontal))}.assistant-empty{padding:18px;text-align:center}.assistant-msg{border:none;border-radius:12px;background:rgb(var(--rgb-surface-main));padding:10px}.assistant-msg.user{border-color:transparent;background:rgb(var(--rgb-surface-focus))}.assistant-msg.assistant{border-color:transparent}.assistant-msg-head{display:flex;justify-content:space-between;gap:8px;margin-bottom:6px;font-size:.78rem;color:var(--text-muted)}.assistant-msg-body{white-space:pre-wrap;line-height:1.45;font-size:.9rem}.assistant-trace{margin-top:6px;color:var(--text-muted);font-size:.75rem}.assistant-debug{margin-top:8px}.assistant-debug summary{cursor:pointer;color:var(--lime-main);font-size:.8rem}.assistant-compose{margin-top:12px;display:grid;gap:10px}.json-view{margin:0;width:100%;min-height:180px;max-height:420px;overflow:auto;background:rgb(var(--rgb-surface-horizontal));border:none;border-radius:var(--radius-md);padding:12px;color:rgb(var(--rgb-text-main));font-family:JetBrains Mono,Consolas,monospace;font-size:.78rem;line-height:1.45}.metrics-grid{display:grid;grid-template-columns:repeat(5,minmax(0,1fr));gap:10px}.metrics-grid div{background:rgba(var(--rgb-surface-main),.8);border:none;border-radius:12px;padding:10px;display:flex;flex-direction:column;gap:4px}.metrics-grid span{color:var(--text-muted);font-size:.75rem}.metrics-grid strong{font-size:.84rem;color:var(--lime-main)}.history-list{display:grid;gap:8px;max-height:340px;overflow:auto}.history-item{width:100%;text-align:left;border-radius:12px;border:none;background:rgb(var(--rgb-surface-main));color:var(--text-main);padding:10px}.history-item p{margin:8px 0;color:var(--text-muted);font-size:.82rem}.history-item.selected{border-color:var(--line-strong)}.history-row{display:flex;justify-content:space-between;gap:8px;font-size:.76rem;color:var(--text-muted)}.runtime-grid{display:grid;grid-template-columns:1.2fr 1fr;gap:12px}.runtime-runs{max-height:360px;overflow:auto;display:grid;gap:8px}.eval-report-wrap{position:relative}.copy-cube-button{position:absolute;right:10px;bottom:10px;width:34px;height:34px;border-radius:10px;padding:0;min-width:34px;display:grid;place-items:center;font-size:.92rem;line-height:1}.muted{color:var(--text-muted)}.diff-summary{margin-top:10px;font-size:.82rem;color:var(--lime-main)}.error-text{margin-top:10px;color:var(--danger);font-size:.84rem}.autoruns-columns{display:flex;gap:12px;width:100%;min-height:0;flex:1 1 auto;overflow-x:auto;overflow-y:hidden;padding-bottom:4px}.autoruns-col{flex:0 0 var(--autoruns-col-width);width:var(--autoruns-col-width);height:100%;min-height:0;overflow:auto;border:none;border-radius:14px;background:rgb(var(--rgb-surface-main));padding:12px;scrollbar-gutter:stable}.autoruns-col:nth-child(3){flex-basis:440px;width:440px}.autoruns-col h3{margin:0 0 10px;font-size:.95rem}.autoruns-col h4{margin:12px 0 8px;font-size:.82rem;color:var(--text-muted)}.autoruns-form-grid{display:grid;grid-template-columns:repeat(2,minmax(0,1fr));gap:8px}.autoruns-meta-list{display:grid;gap:8px}.autoruns-meta-list>div{display:flex;justify-content:space-between;gap:8px;border:none;border-radius:10px;background:rgb(var(--rgb-surface-horizontal));padding:8px 9px;font-size:.79rem}.autoruns-meta-list span{color:var(--text-muted)}.autoruns-prompt-details summary{cursor:pointer;color:var(--text-main);font-size:.8rem;margin-bottom:8px}.autoruns-prompt-details textarea{min-height:68px}.autoruns-stats-grid{display:grid;grid-template-columns:repeat(2,minmax(0,1fr));gap:8px;margin-bottom:10px}.autoruns-stats-grid>div{border:none;border-radius:10px;background:rgb(var(--rgb-surface-horizontal));padding:8px;display:grid;gap:3px}.autoruns-stats-grid span{color:var(--text-muted);font-size:.74rem}.autoruns-stats-grid strong{color:var(--lime-main);font-size:.84rem}.autoruns-run-list{display:grid;gap:8px;max-height:none;min-height:0;flex:1 1 auto;overflow:auto;padding-right:2px}.autoruns-run-item{width:100%;text-align:left;border-radius:12px;border:none;background:rgb(var(--rgb-surface-horizontal));color:var(--text-main);padding:10px;display:grid;gap:5px}.autoruns-run-item.selected{border-color:var(--line-strong)}.autoruns-run-head,.autoruns-run-foot{display:flex;justify-content:space-between;gap:8px;font-size:.76rem}.autoruns-run-meta{color:var(--text-muted);font-size:.75rem;word-break:break-word}.autoruns-dialog-toolbar{display:grid;grid-template-columns:repeat(2,minmax(0,1fr));gap:8px}.autoruns-case-list{margin-top:8px;display:grid;gap:6px;max-height:180px;overflow:auto}.autoruns-case-item{width:100%;text-align:left;border-radius:10px;border:none;background:rgb(var(--rgb-surface-horizontal));color:var(--text-main);padding:7px 8px;display:flex;justify-content:space-between;gap:6px;font-size:.76rem}.autoruns-case-item.selected{border-color:var(--line-strong)}.autoruns-dialog-view{margin-top:10px;border:none;border-radius:12px;background:rgb(var(--rgb-surface-horizontal));padding:10px;max-height:none;min-height:0;flex:1 1 auto;overflow:auto;display:grid;gap:8px}.autoruns-msg{border:none;border-radius:12px;background:rgb(var(--rgb-surface-focus));padding:8px 10px;display:grid;gap:6px}.autoruns-msg header,.autoruns-msg footer{display:flex;justify-content:space-between;gap:8px;font-size:.74rem;color:var(--text-muted)}.autoruns-msg-head-actions{display:flex;align-items:center;gap:8px}.autoruns-msg p{margin:0;white-space:pre-wrap;line-height:1.35;font-size:.84rem}.autoruns-comment-icon{border:none;background:rgb(var(--rgb-surface-horizontal));color:var(--text-main);border-radius:999px;min-width:28px;min-height:28px;padding:0 8px;line-height:1;box-shadow:none;transform:none}.autoruns-comment-icon:hover{border-color:var(--line-strong);box-shadow:none;transform:none}.autoruns-comment-icon.commented{border-color:var(--line-strong);color:var(--lime-main);background:rgb(var(--rgb-active));color:rgb(var(--rgb-active-text))}.autoruns-msg-annotation{display:grid;gap:4px;border:none;border-radius:10px;background:rgb(var(--rgb-surface-horizontal));padding:7px 8px;font-size:.78rem}.autoruns-comments-list{display:grid;gap:8px;max-height:none;min-height:0;flex:1 1 auto;overflow:auto;padding-right:2px}.autoruns-autogen-list{display:grid;gap:8px;max-height:none;min-height:0;overflow:auto;padding-right:2px}.autoruns-autogen-item{border:none;border-radius:10px;background:rgb(var(--rgb-surface-horizontal));padding:8px;display:grid;gap:5px}.autoruns-autogen-item header{display:flex;justify-content:space-between;gap:8px;font-size:.76rem}.autoruns-autogen-item p{margin:0;color:var(--text-muted);white-space:pre-wrap;font-size:.8rem}.autoruns-comment-item{width:100%;text-align:left;border-radius:12px;border:none;background:rgb(var(--rgb-surface-horizontal));color:var(--text-main);padding:9px;display:grid;gap:6px}.autoruns-comment-item p{margin:0;white-space:pre-wrap;color:var(--text-muted);font-size:.79rem}.autoruns-comment-item.selected{border-color:var(--line-strong)}.autoruns-comment-head{display:flex;justify-content:space-between;gap:8px;font-size:.75rem}.autoruns-msg.assistant{margin-right:12%}.autoruns-msg.user{margin-left:12%;border-color:transparent;background:rgb(var(--rgb-surface-focus))}.autoruns-decomposition-list{margin:0;padding-left:18px;display:grid;gap:7px;font-size:.8rem}.autoruns-comment-modal-backdrop{position:fixed;inset:0;background:rgba(var(--rgb-background),.74);display:grid;place-items:center;z-index:1800;padding:12px}.autoruns-comment-modal{width:min(660px,100%);border:none;border-radius:16px;background:rgb(var(--rgb-surface-horizontal));box-shadow:var(--shadow);padding:14px;display:grid;gap:10px}.autoruns-comment-modal h3{margin:0;font-size:.95rem}.autoruns-comment-quote{margin:0;border:none;border-radius:10px;background:rgb(var(--rgb-surface-focus));padding:8px;white-space:pre-wrap;max-height:150px;overflow:auto;font-size:.82rem}.autoruns-rating-row{display:flex;gap:8px}.autoruns-rating-dot{width:34px;height:34px;border-radius:999px;padding:0;border:none;background:rgb(var(--rgb-surface-focus));color:var(--text-muted);font-size:.95rem;box-shadow:none;transform:none}.autoruns-rating-dot:hover{border-color:var(--line-strong);box-shadow:none;transform:none}.autoruns-rating-dot.active{border-color:var(--line-strong);color:rgb(var(--rgb-active-text));background:rgb(var(--rgb-active))}.autoruns-coverage-list{display:grid;gap:8px}.autoruns-coverage-item{border:none;border-radius:10px;background:rgb(var(--rgb-surface-horizontal));padding:8px}.autoruns-coverage-head{display:flex;justify-content:space-between;gap:8px;font-size:.76rem;margin-bottom:5px}.autoruns-coverage-head span{color:var(--text-muted)}.autoruns-coverage-bar{height:7px;border-radius:999px;background:rgb(var(--rgb-surface-focus));overflow:hidden}.autoruns-coverage-bar>div{height:100%;border-radius:999px;background:rgb(var(--rgb-active))}@media(max-width:1200px){:root{--autoruns-col-width: 340px}.metrics-grid{grid-template-columns:repeat(3,minmax(0,1fr))}}@media(max-width:920px){:root{--autoruns-col-width: 320px}.grid-two,.runtime-grid{grid-template-columns:1fr}.metrics-grid{grid-template-columns:repeat(2,minmax(0,1fr))}.autoruns-form-grid,.autoruns-dialog-toolbar,.autoruns-stats-grid{grid-template-columns:1fr}}@media(max-width:640px){:root{--autoruns-col-width: 300px}.app-root{padding:18px 12px 24px}.metrics-grid{grid-template-columns:1fr}}@keyframes rise{0%{opacity:0;transform:translateY(8px)}to{opacity:1;transform:translateY(0)}} diff --git a/llm_normalizer/frontend/dist/index.html b/llm_normalizer/frontend/dist/index.html index 25f7393..08a87f5 100644 --- a/llm_normalizer/frontend/dist/index.html +++ b/llm_normalizer/frontend/dist/index.html @@ -4,8 +4,8 @@ NDC AI Normalizer Playground - - + +
diff --git a/llm_normalizer/frontend/src/App.tsx b/llm_normalizer/frontend/src/App.tsx index 68eb533..32e3848 100644 --- a/llm_normalizer/frontend/src/App.tsx +++ b/llm_normalizer/frontend/src/App.tsx @@ -10,6 +10,7 @@ import { PromptPanel } from "./components/PromptPanel"; import { QueryPanel } from "./components/QueryPanel"; import { RuntimePanel } from "./components/RuntimePanel"; import { DEFAULT_CONNECTION, DEFAULT_PROMPTS, DEFAULT_QUERY } from "./state/defaults"; +import { designConfig } from "../../../designconfig"; import type { AssistantConversationItem, ConnectionState, @@ -23,7 +24,7 @@ import type { } from "./state/types"; const SESSION_CONFIG_KEY = "ndc_normalizer_session_config_v1"; -const ASSISTANT_STAGES = ["Analyzing request", "Fetching data", "Composing answer"]; +const ASSISTANT_STAGES = ["Анализ запроса", "Получение данных", "Подготовка ответа"]; const DEFAULT_UI_MODE: UiMode = "assistant"; const AUTOLOAD_PROMPT_VERSION = "normalizer_v2_0_2"; const ASSISTANT_PROMPT_VERSION = "address_query_runtime_v1"; @@ -79,6 +80,9 @@ export default function App() { const [lastError, setLastError] = useState(""); const [uiMode, setUiMode] = useState(DEFAULT_UI_MODE); + const [showAutorunsAssistantMode, setShowAutorunsAssistantMode] = useState(true); + const [showAutorunsDecompositionMode, setShowAutorunsDecompositionMode] = useState(true); + const [showAutorunsProgressMode, setShowAutorunsProgressMode] = useState(true); const [assistantSessionId, setAssistantSessionId] = useState(""); const [assistantConversation, setAssistantConversation] = useState([]); const [assistantInput, setAssistantInput] = useState(""); @@ -87,6 +91,23 @@ export default function App() { const [assistantError, setAssistantError] = useState(""); const presetAutoloadDoneRef = useRef(false); + useEffect(() => { + const root = document.documentElement; + const { colors } = designConfig; + root.style.setProperty("--rgb-background", colors.backgroundRgb); + root.style.setProperty("--rgb-surface-main", colors.mainSurfaceRgb); + root.style.setProperty("--rgb-surface-horizontal", colors.horizontalSurfaceRgb); + root.style.setProperty("--rgb-surface-focus", colors.focusSurfaceRgb); + root.style.setProperty("--rgb-active", colors.activeRgb); + root.style.setProperty("--rgb-active-text", colors.activeTextRgb); + root.style.setProperty("--rgb-text-main", colors.textMainRgb); + root.style.setProperty("--rgb-text-muted", colors.textMutedRgb); + root.style.setProperty("--rgb-danger", colors.dangerRgb); + root.style.setProperty("--rgb-scrollbar-track", colors.scrollbarTrackRgb); + root.style.setProperty("--rgb-scrollbar-thumb", colors.scrollbarThumbRgb); + root.style.setProperty("--rgb-scrollbar-thumb-hover", colors.scrollbarThumbHoverRgb); + }, []); + const log = (message: string) => { setAppLogs((prev) => [withTs(message), ...prev].slice(0, 300)); }; @@ -509,12 +530,12 @@ export default function App() { }); setAssistantSessionId(response.session_id); setAssistantConversation(response.conversation); - setAssistantStatus("Reply is ready"); + setAssistantStatus("Ответ готов"); log(`Assistant reply received: trace=${response.debug.trace_id}`); } catch (error) { const message = error instanceof Error ? error.message : String(error); setAssistantError(message); - setAssistantStatus("Assistant error"); + setAssistantStatus("Ошибка ассистента"); log(`Assistant error: ${message}`); } finally { stopTicker(); @@ -534,23 +555,46 @@ export default function App() { }, [selectedRunId]); return ( -
-
-

NDC AI First Layer

-

Three modes in one UI: assistant, decomposition diagnostics, and auto-run history with regression visibility.

-
+
+
+
+ + + +
-
- - - -
+ {uiMode === "autoruns" ? ( +
+ + + +
+ ) : null} +
{uiMode === "assistant" ? (
@@ -660,12 +704,15 @@ export default function App() { />
) : ( -
+
diff --git a/llm_normalizer/frontend/src/api/client.ts b/llm_normalizer/frontend/src/api/client.ts index 327734a..35d30d7 100644 --- a/llm_normalizer/frontend/src/api/client.ts +++ b/llm_normalizer/frontend/src/api/client.ts @@ -1,11 +1,17 @@ import type { + AutoGenHistoryResponse, + AutoGenMode, + AutoRunAnnotationsResponse, + AutoRunAnnotationRecord, AutoRunDetailResponse, AutoRunDialogResponse, AutoRunHistoryResponse, + AutoRunPostAnalysisResponse, AssistantMessageResultState, AssistantConversationItem, ConnectionState, HistoryItem, + ManualCaseDecision, NormalizeResultState, PromptState, RuntimeRun @@ -281,5 +287,95 @@ export const apiClient = { async loadAutoRunCaseDialog(runId: string, caseId: string): Promise { return request(`/autoruns/history/${encodeURIComponent(runId)}/case/${encodeURIComponent(caseId)}/dialog`); + }, + + async loadAutoRunAnnotations(input?: { + run_id?: string; + case_id?: string; + min_rating?: number; + manual_case_decision?: ManualCaseDecision | "all"; + limit?: number; + }): Promise { + const params = new URLSearchParams(); + if (input?.run_id) params.set("run_id", input.run_id); + if (input?.case_id) params.set("case_id", input.case_id); + if (typeof input?.min_rating === "number") params.set("min_rating", String(input.min_rating)); + if (input?.manual_case_decision) params.set("manual_case_decision", input.manual_case_decision); + if (typeof input?.limit === "number") params.set("limit", String(input.limit)); + const query = params.toString(); + return request(`/autoruns/annotations${query ? `?${query}` : ""}`); + }, + + async saveAutoRunAnnotation(input: { + run_id: string; + case_id: string; + message_index: number; + rating: number; + comment: string; + manual_case_decision: ManualCaseDecision; + annotation_author?: string; + }): Promise<{ ok: boolean; annotation: AutoRunAnnotationRecord; case_annotation_stats: { count: number; latest_at: string | null; avg_rating: number | null } | null }> { + return request("/autoruns/annotations", { + method: "POST", + body: JSON.stringify(input) + }); + }, + + async loadAutoRunPostAnalysis(input?: { + run_id?: string; + limit_per_queue?: number; + annotation_limit?: number; + scan_limit?: number; + from?: string; + to?: string; + target?: string; + mode?: string; + use_mock?: "any" | "true" | "false"; + prompt_contains?: string; + }): Promise { + const params = new URLSearchParams(); + if (input?.run_id) params.set("run_id", input.run_id); + if (typeof input?.limit_per_queue === "number") params.set("limit_per_queue", String(input.limit_per_queue)); + if (typeof input?.annotation_limit === "number") params.set("annotation_limit", String(input.annotation_limit)); + if (typeof input?.scan_limit === "number") params.set("scan_limit", String(input.scan_limit)); + if (input?.from) params.set("from", input.from); + if (input?.to) params.set("to", input.to); + if (input?.target) params.set("target", input.target); + if (input?.mode) params.set("mode", input.mode); + if (input?.use_mock) params.set("use_mock", input.use_mock); + if (input?.prompt_contains) params.set("prompt_contains", input.prompt_contains); + const query = params.toString(); + return request(`/autoruns/post-analysis${query ? `?${query}` : ""}`); + }, + + async loadAutoRunAutogenHistory(input?: { + mode?: AutoGenMode; + limit?: number; + }): Promise { + const params = new URLSearchParams(); + if (input?.mode) params.set("mode", input.mode); + if (typeof input?.limit === "number") params.set("limit", String(input.limit)); + const query = params.toString(); + return request(`/autoruns/autogen/history${query ? `?${query}` : ""}`); + }, + + async generateAutoRunQuestions(input: { + mode: AutoGenMode; + count: number; + domain?: string; + persist_to_eval_cases?: boolean; + generated_by?: string; + context?: { + llm_provider?: string; + model?: string; + assistant_prompt_version?: string; + decomposition_prompt_version?: string; + prompt_fingerprint?: string; + }; + }): Promise<{ ok: boolean; generation: { generation_id: string; created_at: string; mode: AutoGenMode; count: number; domain: string | null; questions: string[]; generated_by: string | null; saved_case_set_file: string | null; context: Record | null } }> { + return request("/autoruns/autogen/generate", { + method: "POST", + body: JSON.stringify(input) + }); } }; diff --git a/llm_normalizer/frontend/src/components/AutoRunsHistoryPanel.tsx b/llm_normalizer/frontend/src/components/AutoRunsHistoryPanel.tsx index d241153..d98c6ba 100644 --- a/llm_normalizer/frontend/src/components/AutoRunsHistoryPanel.tsx +++ b/llm_normalizer/frontend/src/components/AutoRunsHistoryPanel.tsx @@ -1,13 +1,20 @@ -import { useCallback, useEffect, useMemo, useState } from "react"; +import { useCallback, useEffect, useMemo, useRef, useState } from "react"; import { apiClient } from "../api/client"; import type { + AutoGenHistoryRecord, + AutoGenMode, + AutoRunAnnotationListItem, + AutoRunAnnotationRecord, AutoRunCaseSummary, AutoRunDetailResponse, + AutoRunDialogMessage, AutoRunDialogResponse, AutoRunDomainCoverage, AutoRunHistoryResponse, + AutoRunPostAnalysisResponse, AutoRunSummary, ConnectionState, + ManualCaseDecision, PromptState } from "../state/types"; import { JsonView } from "./JsonView"; @@ -18,10 +25,14 @@ interface AutoRunsHistoryPanelProps { prompts: PromptState; assistantPromptVersion: string; decompositionPromptVersion: string; + showAssistantMode: boolean; + showDecompositionMode: boolean; + showProgressMode: boolean; onLog?: (message: string) => void; } type UseMockFilter = "any" | "true" | "false"; +type LeftTabMode = "settings" | "comments"; interface AutoRunsFilters { fromLocal: string; @@ -33,6 +44,25 @@ interface AutoRunsFilters { limit: number; } +interface CommentModalState { + open: boolean; + messageIndex: number; + rating: number; + comment: string; + manualCaseDecision: ManualCaseDecision; + annotationAuthor: string; + saving: boolean; + error: string; +} + +interface AutoGenSettingsState { + mode: AutoGenMode; + count: number; + domain: string; + persistToEvalCases: boolean; + generatedBy: string; +} + const DEFAULT_FILTERS: AutoRunsFilters = { fromLocal: "", toLocal: "", @@ -43,6 +73,16 @@ const DEFAULT_FILTERS: AutoRunsFilters = { limit: 120 }; +const DEFAULT_MANUAL_DECISION: ManualCaseDecision = "needs_dialog_policy_fix"; + +const DEFAULT_AUTOGEN_SETTINGS: AutoGenSettingsState = { + mode: "codex_creative", + count: 24, + domain: "", + persistToEvalCases: true, + generatedBy: "manual_reviewer" +}; + function dateToInputValue(date: Date): string { const year = date.getFullYear(); const month = String(date.getMonth() + 1).padStart(2, "0"); @@ -65,7 +105,8 @@ function localInputToIso(value: string): string | undefined { return new Date(parsed).toISOString(); } -function formatDateTime(iso: string): string { +function formatDateTime(iso: string | null): string { + if (!iso) return "нет данных"; const parsed = Date.parse(iso); if (!Number.isFinite(parsed)) return iso; return new Date(parsed).toLocaleString("ru-RU"); @@ -77,7 +118,7 @@ function toPercent(closed: number, total: number): number { } function formatScore(value: number | null): string { - if (typeof value !== "number") return "n/a"; + if (typeof value !== "number") return "нет данных"; return `${value.toFixed(1)}%`; } @@ -98,6 +139,11 @@ function getSelectedCase(cases: AutoRunCaseSummary[], caseId: string): AutoRunCa return cases.find((item) => item.case_id === caseId) ?? null; } +function renderRatingDots(rating: number): string { + const safe = Math.max(1, Math.min(5, Math.round(rating))); + return `${"●".repeat(safe)}${"○".repeat(5 - safe)}`; +} + function renderCoverageRows(items: AutoRunDomainCoverage[]) { if (items.length === 0) { return

Покрытие доменов пока не сформировано.

; @@ -130,28 +176,69 @@ export function AutoRunsHistoryPanel({ prompts, assistantPromptVersion, decompositionPromptVersion, + showAssistantMode, + showDecompositionMode, + showProgressMode, onLog }: AutoRunsHistoryPanelProps) { const [filters, setFilters] = useState({ ...DEFAULT_FILTERS, fromLocal: defaultFromDateValue() }); + const [leftTab, setLeftTab] = useState("settings"); const [history, setHistory] = useState(null); const [runDetail, setRunDetail] = useState(null); const [dialog, setDialog] = useState(null); + const [annotations, setAnnotations] = useState([]); + const [annotationDecisionFilter, setAnnotationDecisionFilter] = useState("all"); + const [manualDecisionSchema, setManualDecisionSchema] = useState | null>(null); + const [availableManualDecisions, setAvailableManualDecisions] = useState([]); + const [selectedAnnotationId, setSelectedAnnotationId] = useState(""); const [selectedRunId, setSelectedRunId] = useState(""); const [selectedCaseId, setSelectedCaseId] = useState(""); + const [autoGenSettings, setAutoGenSettings] = useState(DEFAULT_AUTOGEN_SETTINGS); + const [autoGenHistory, setAutoGenHistory] = useState([]); + const [postAnalysis, setPostAnalysis] = useState(null); + const [autoGenBusy, setAutoGenBusy] = useState(false); + const [postAnalysisBusy, setPostAnalysisBusy] = useState(false); + const [autogenHistoryBusy, setAutogenHistoryBusy] = useState(false); const [historyBusy, setHistoryBusy] = useState(false); const [detailBusy, setDetailBusy] = useState(false); const [dialogBusy, setDialogBusy] = useState(false); + const [annotationsBusy, setAnnotationsBusy] = useState(false); const [errorText, setErrorText] = useState(""); - const [showAssistantMode, setShowAssistantMode] = useState(true); - const [showDecompositionMode, setShowDecompositionMode] = useState(true); - const [showProgressMode, setShowProgressMode] = useState(true); + const [commentModal, setCommentModal] = useState({ + open: false, + messageIndex: -1, + rating: 3, + comment: "", + manualCaseDecision: DEFAULT_MANUAL_DECISION, + annotationAuthor: "manual_reviewer", + saving: false, + error: "" + }); + + const initialLoadDoneRef = useRef(false); const activeRunSummary: AutoRunSummary | null = - history?.items.find((item) => item.run_id === selectedRunId) ?? null; + history?.items.find((item) => item.run_id === selectedRunId) ?? runDetail?.run ?? null; const activeCase = runDetail ? getSelectedCase(runDetail.cases, selectedCaseId) : null; + const selectedAnnotation = annotations.find((item) => item.annotation_id === selectedAnnotationId) ?? null; + const modalMessage = dialog?.messages.find((item) => item.message_index === commentModal.messageIndex) ?? null; + + const annotationsAverageRating = useMemo(() => { + if (annotations.length === 0) return null; + const avg = annotations.reduce((acc, item) => acc + item.rating, 0) / annotations.length; + return Number(avg.toFixed(2)); + }, [annotations]); + + const runSelectItems = useMemo(() => { + const list = [...(history?.items ?? [])]; + if (selectedRunId && !list.some((item) => item.run_id === selectedRunId) && runDetail?.run) { + list.unshift(runDetail.run); + } + return list; + }, [history?.items, runDetail?.run, selectedRunId]); const log = useCallback( (message: string) => { @@ -160,6 +247,121 @@ export function AutoRunsHistoryPanel({ [onLog] ); + const loadAnnotations = useCallback(async () => { + setAnnotationsBusy(true); + try { + const payload = await apiClient.loadAutoRunAnnotations({ + limit: 800, + manual_case_decision: annotationDecisionFilter + }); + setAnnotations(payload.items); + setManualDecisionSchema(payload.manual_case_decision_schema ?? null); + setAvailableManualDecisions(payload.available_manual_case_decisions ?? []); + setSelectedAnnotationId((prev) => { + if (payload.items.length === 0) return ""; + if (payload.items.some((item) => item.annotation_id === prev)) return prev; + return payload.items[0].annotation_id; + }); + } catch (error) { + log(`Annotations load error: ${error instanceof Error ? error.message : String(error)}`); + } finally { + setAnnotationsBusy(false); + } + }, [annotationDecisionFilter, log]); + + const loadAutoGenHistory = useCallback(async () => { + setAutogenHistoryBusy(true); + try { + const payload = await apiClient.loadAutoRunAutogenHistory({ limit: 180 }); + setAutoGenHistory(payload.items); + } catch (error) { + log(`Autogen history load error: ${error instanceof Error ? error.message : String(error)}`); + } finally { + setAutogenHistoryBusy(false); + } + }, [log]); + + const loadPostAnalysis = useCallback(async () => { + setPostAnalysisBusy(true); + try { + const payload = await apiClient.loadAutoRunPostAnalysis({ + run_id: selectedRunId || undefined, + limit_per_queue: 30, + annotation_limit: 1500, + from: localInputToIso(filters.fromLocal), + to: localInputToIso(filters.toLocal), + target: filters.target, + mode: filters.mode, + use_mock: filters.useMock, + prompt_contains: filters.promptContains.trim() || undefined + }); + setPostAnalysis(payload); + } catch (error) { + log(`Post-analysis load error: ${error instanceof Error ? error.message : String(error)}`); + setPostAnalysis(null); + } finally { + setPostAnalysisBusy(false); + } + }, [filters.fromLocal, filters.mode, filters.promptContains, filters.target, filters.toLocal, filters.useMock, log, selectedRunId]); + + const generateAutogenBatch = useCallback(async () => { + setAutoGenBusy(true); + setErrorText(""); + try { + const promptFingerprint = [ + prompts.systemPrompt, + prompts.developerPrompt, + prompts.domainPrompt, + prompts.schemaNotes, + prompts.fewShotExamples + ] + .join("\n") + .slice(0, 900); + const payload = await apiClient.generateAutoRunQuestions({ + mode: autoGenSettings.mode, + count: autoGenSettings.count, + domain: autoGenSettings.domain.trim() || undefined, + persist_to_eval_cases: autoGenSettings.persistToEvalCases, + generated_by: autoGenSettings.generatedBy.trim() || undefined, + context: { + llm_provider: connection.llmProvider, + model: connection.model, + assistant_prompt_version: assistantPromptVersion, + decomposition_prompt_version: decompositionPromptVersion, + prompt_fingerprint: promptFingerprint + } + }); + log( + `Generated ${payload.generation.count} questions (${payload.generation.mode}) id=${payload.generation.generation_id}` + + (payload.generation.saved_case_set_file ? ` saved=${payload.generation.saved_case_set_file}` : "") + ); + await loadAutoGenHistory(); + } catch (error) { + const message = error instanceof Error ? error.message : String(error); + setErrorText(`Автогенерация: ${message}`); + log(`Autogen generate error: ${message}`); + } finally { + setAutoGenBusy(false); + } + }, [ + assistantPromptVersion, + autoGenSettings.count, + autoGenSettings.domain, + autoGenSettings.generatedBy, + autoGenSettings.mode, + autoGenSettings.persistToEvalCases, + connection.llmProvider, + connection.model, + decompositionPromptVersion, + loadAutoGenHistory, + log, + prompts.developerPrompt, + prompts.domainPrompt, + prompts.fewShotExamples, + prompts.schemaNotes, + prompts.systemPrompt + ]); + const loadCaseDialog = useCallback( async (runId: string, caseId: string) => { setDialogBusy(true); @@ -169,8 +371,8 @@ export function AutoRunsHistoryPanel({ } catch (error) { const message = error instanceof Error ? error.message : String(error); setErrorText(`Диалог кейса: ${message}`); - log(`Dialog load error for ${runId}/${caseId}: ${message}`); setDialog(null); + log(`Dialog load error for ${runId}/${caseId}: ${message}`); } finally { setDialogBusy(false); } @@ -188,6 +390,7 @@ export function AutoRunsHistoryPanel({ (preferredCaseId && payload.cases.some((item) => item.case_id === preferredCaseId) ? preferredCaseId : "") || payload.cases[0]?.case_id || ""; + setSelectedRunId(runId); setSelectedCaseId(nextCaseId); if (nextCaseId) { await loadCaseDialog(runId, nextCaseId); @@ -197,9 +400,9 @@ export function AutoRunsHistoryPanel({ } catch (error) { const message = error instanceof Error ? error.message : String(error); setErrorText(`Детализация прогона: ${message}`); - log(`Run detail load error for ${runId}: ${message}`); setRunDetail(null); setDialog(null); + log(`Run detail load error for ${runId}: ${message}`); } finally { setDetailBusy(false); } @@ -222,8 +425,7 @@ export function AutoRunsHistoryPanel({ limit: filters.limit }); setHistory(payload); - const hasRuns = payload.items.length > 0; - if (!hasRuns) { + if (payload.items.length === 0) { setSelectedRunId(""); setSelectedCaseId(""); setRunDetail(null); @@ -238,8 +440,8 @@ export function AutoRunsHistoryPanel({ keepSelection && preferredRunId && payload.items.some((item) => item.run_id === preferredRunId) ? preferredRunId : payload.items[0].run_id; - setSelectedRunId(nextRunId); await loadRunDetail(nextRunId, keepSelection ? preferredCaseId : undefined); + void loadPostAnalysis(); } catch (error) { const message = error instanceof Error ? error.message : String(error); setErrorText(`История прогонов: ${message}`); @@ -248,176 +450,486 @@ export function AutoRunsHistoryPanel({ setHistoryBusy(false); } }, - [filters.fromLocal, filters.limit, filters.mode, filters.promptContains, filters.target, filters.toLocal, filters.useMock, loadRunDetail, log] + [ + filters.fromLocal, + filters.limit, + filters.mode, + filters.promptContains, + filters.target, + filters.toLocal, + filters.useMock, + loadPostAnalysis, + loadRunDetail, + log + ] + ); + + const openCommentModal = useCallback((message: AutoRunDialogMessage) => { + if (message.role !== "assistant") return; + setCommentModal({ + open: true, + messageIndex: message.message_index, + rating: message.annotation?.rating ?? 3, + comment: message.annotation?.comment ?? "", + manualCaseDecision: message.annotation?.manual_case_decision ?? DEFAULT_MANUAL_DECISION, + annotationAuthor: message.annotation?.annotation_author ?? autoGenSettings.generatedBy, + saving: false, + error: "" + }); + }, [autoGenSettings.generatedBy]); + + const closeCommentModal = useCallback(() => { + setCommentModal((prev) => { + if (prev.saving) return prev; + return { + open: false, + messageIndex: -1, + rating: 3, + comment: "", + manualCaseDecision: DEFAULT_MANUAL_DECISION, + annotationAuthor: autoGenSettings.generatedBy, + saving: false, + error: "" + }; + }); + }, [autoGenSettings.generatedBy]); + + const submitCommentModal = useCallback(async () => { + if (!selectedRunId || !selectedCaseId || commentModal.messageIndex < 0) return; + if (!commentModal.comment.trim()) { + setCommentModal((prev) => ({ ...prev, error: "Добавьте комментарий." })); + return; + } + setCommentModal((prev) => ({ ...prev, saving: true, error: "" })); + try { + await apiClient.saveAutoRunAnnotation({ + run_id: selectedRunId, + case_id: selectedCaseId, + message_index: commentModal.messageIndex, + rating: commentModal.rating, + comment: commentModal.comment.trim(), + manual_case_decision: commentModal.manualCaseDecision, + annotation_author: commentModal.annotationAuthor.trim() || undefined + }); + + await Promise.all([loadRunDetail(selectedRunId, selectedCaseId), loadAnnotations(), loadPostAnalysis()]); + closeCommentModal(); + } catch (error) { + setCommentModal((prev) => ({ + ...prev, + saving: false, + error: error instanceof Error ? error.message : String(error) + })); + } + }, [ + closeCommentModal, + commentModal.annotationAuthor, + commentModal.comment, + commentModal.manualCaseDecision, + commentModal.messageIndex, + commentModal.rating, + loadAnnotations, + loadPostAnalysis, + loadRunDetail, + selectedCaseId, + selectedRunId + ]); + + const openAnnotationContext = useCallback( + async (annotation: AutoRunAnnotationRecord) => { + setSelectedAnnotationId(annotation.annotation_id); + setLeftTab("settings"); + await loadRunDetail(annotation.run_id, annotation.case_id); + if (!history?.items.some((item) => item.run_id === annotation.run_id)) { + setErrorText("Комментарий относится к прогону вне текущего фильтра. Детали загружены напрямую."); + } + }, + [history?.items, loadRunDetail] ); useEffect(() => { + if (initialLoadDoneRef.current) return; + initialLoadDoneRef.current = true; void loadHistory({ keepSelection: false }); - }, [loadHistory]); + void loadAutoGenHistory(); + void loadPostAnalysis(); + }, [loadAutoGenHistory, loadHistory, loadPostAnalysis]); - const dynamicColumns = useMemo(() => { - const columns = ["minmax(290px, 340px)", "minmax(300px, 360px)", "minmax(420px, 1fr)"]; - if (showAssistantMode) columns.push("minmax(280px, 320px)"); - if (showDecompositionMode) columns.push("minmax(280px, 320px)"); - if (showProgressMode) columns.push("minmax(280px, 320px)"); - return columns.join(" "); - }, [showAssistantMode, showDecompositionMode, showProgressMode]); + useEffect(() => { + if (!initialLoadDoneRef.current) return; + void loadAnnotations(); + }, [annotationDecisionFilter, loadAnnotations]); return ( - - - -
- } + className="autoruns-frame" + title="" + hideHeader > -
+
-

Настройки выборки

-
- - - - - - - -
- - {(history?.available.prompt_versions ?? []).map((item) => ( - -
- -
-

Контур генерации

-
-
- Провайдер: - {connection.llmProvider} -
-
- Модель: - {connection.model || "n/a"} -
-
- Prompt assistant: - {assistantPromptVersion} -
-
- Prompt decomposition: - {decompositionPromptVersion} -
-
+ {leftTab === "settings" ? ( + <> +

Настройки выборки

+
+ + + + + + + +
+ + {(history?.available.prompt_versions ?? []).map((item) => ( + -
- Дублирование главного промпта (read-only) -