From f3255cb3b85bf24fbbdb31453215c633ff67cb3c Mon Sep 17 00:00:00 2001 From: dctouch Date: Thu, 16 Apr 2026 20:18:23 +0300 Subject: [PATCH] =?UTF-8?q?=D0=AE=D0=98=20-=20=D0=9F=D0=B5=D1=80=D0=B5?= =?UTF-8?q?=D0=BD=D0=B5=D1=81=D1=82=D0=B8=20=D0=BD=D0=B0=D1=81=D1=82=D1=80?= =?UTF-8?q?=D0=BE=D0=B9=D0=BA=D0=B8=20=D0=B0=D1=81=D1=81=D0=B8=D1=81=D1=82?= =?UTF-8?q?=D0=B5=D0=BD=D1=82=D0=B0=20=D0=B2=20=D1=83=D0=BF=D1=80=D0=B0?= =?UTF-8?q?=D0=B2=D0=BB=D0=B5=D0=BD=D0=B8=D0=B5=20=D0=B0=D0=B2=D1=82=D0=BE?= =?UTF-8?q?=D0=BF=D1=80=D0=BE=D0=B3=D0=BE=D0=BD=D0=B0=D0=BC=D0=B8=20=D0=B8?= =?UTF-8?q?=20=D1=83=D0=B1=D1=80=D0=B0=D1=82=D1=8C=20=D0=B2=D0=B5=D1=80?= =?UTF-8?q?=D1=85=D0=BD=D0=B8=D0=B9=20=D1=80=D0=B5=D0=B6=D0=B8=D0=BC=20?= =?UTF-8?q?=D0=90=D1=81=D1=81=D0=B8=D1=81=D1=82=D0=B5=D0=BD=D1=82?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../address_truth_harness_test2.json | 366 ++++++ .../dist/services/addressFilterExtractor.js | 11 +- .../dist/services/addressQueryService.js | 22 +- .../dist/services/addressRecipeCatalog.js | 17 +- .../address_runtime/decomposeStage.js | 71 +- ...stantAddressOrchestrationRuntimeAdapter.js | 54 +- .../dist/services/assistantRoutePolicy.js | 10 +- .../backend/dist/services/assistantService.js | 151 ++- .../services/inventoryLifecycleCueHelpers.js | 3 + .../src/services/addressFilterExtractor.ts | 13 +- .../src/services/addressQueryService.ts | 23 +- .../src/services/addressRecipeCatalog.ts | 25 +- .../address_runtime/decomposeStage.ts | 80 +- ...stantAddressOrchestrationRuntimeAdapter.ts | 75 +- .../src/services/assistantRoutePolicy.ts | 10 +- .../backend/src/services/assistantService.ts | 151 ++- .../services/inventoryLifecycleCueHelpers.ts | 3 + ...fecycleOrganizationScopeRegression.test.ts | 82 ++ .../addressFollowupTemporalRegression.test.ts | 63 + ...ddressInventoryRootFrameRegression.test.ts | 64 +- ...essInventorySaleTraceDocumentRoute.test.ts | 33 +- ...ySaleTraceSelectedObjectRegression.test.ts | 12 +- ...essInventorySelectedObjectFollowup.test.ts | 29 +- .../addressInventoryWarehouseAnchor.test.ts | 23 + .../assistantAddressFollowupContext.test.ts | 223 ++++ ...assistantAddressLaneRuntimeAdapter.test.ts | 31 +- ...AddressOrchestrationRuntimeAdapter.test.ts | 126 ++ ...{index-Dcuz1nX5.css => index-CaUiKcE3.css} | 2 +- .../frontend/dist/assets/index-Qq5WpuqR.js | 26 - .../frontend/dist/assets/index-VJV2AL7G.js | 26 + llm_normalizer/frontend/dist/index.html | 4 +- llm_normalizer/frontend/src/App.tsx | 62 +- .../src/components/AutoRunsHistoryPanel.tsx | 96 +- .../src/components/ConnectionPanel.tsx | 37 +- .../frontend/src/components/PromptPanel.tsx | 28 +- llm_normalizer/frontend/src/styles.css | 38 + scripts/domain_truth_harness.py | 1063 +++++++++++++++++ 37 files changed, 2987 insertions(+), 166 deletions(-) create mode 100644 docs/orchestration/address_truth_harness_test2.json create mode 100644 llm_normalizer/backend/tests/addressCounterpartyLifecycleOrganizationScopeRegression.test.ts create mode 100644 llm_normalizer/backend/tests/addressFollowupTemporalRegression.test.ts rename llm_normalizer/frontend/dist/assets/{index-Dcuz1nX5.css => index-CaUiKcE3.css} (53%) delete mode 100644 llm_normalizer/frontend/dist/assets/index-Qq5WpuqR.js create mode 100644 llm_normalizer/frontend/dist/assets/index-VJV2AL7G.js create mode 100644 scripts/domain_truth_harness.py diff --git a/docs/orchestration/address_truth_harness_test2.json b/docs/orchestration/address_truth_harness_test2.json new file mode 100644 index 0000000..90ba863 --- /dev/null +++ b/docs/orchestration/address_truth_harness_test2.json @@ -0,0 +1,366 @@ +{ + "schema_version": "domain_truth_harness_spec_v1", + "scenario_id": "address_truth_harness_test2", + "domain": "address_mixed_followup", + "title": "Exact replay and truth review for test2 mixed follow-up chain", + "description": "Strict sequential replay of the exact user wording from test2.txt with human-answer and technical-debug review.", + "source_export": "C:\\Users\\DCTOUCH\\Desktop\\test2.txt", + "bindings": {}, + "steps": [ + { + "step_id": "step_01_chat_opening", + "title": "Casual opener", + "question": "йо чо как", + "criticality": "info", + "allowed_reply_types": [ + "factual_with_explanation", + "factual" + ], + "required_answer_patterns_any": [ + "(?i)1с", + "(?i)помогаю", + "(?i)готов" + ], + "notes": "Информационный шаг: важно не ломать живой режим, но это не главный бизнес-блокер." + }, + { + "step_id": "step_02_receivables_march_2020", + "title": "Receivables at March 2020", + "question": "кто нам должен на март 2020", + "allowed_reply_types": [ + "factual" + ], + "expected_intents": [ + "receivables_confirmed_as_of_date" + ], + "expected_capability": "confirmed_receivables_as_of_date", + "expected_result_mode": "confirmed_balance", + "required_filters": { + "as_of_date": "2020-03-31", + "period_from": "2020-03-01", + "period_to": "2020-03-31" + }, + "required_direct_answer_patterns_any": [ + "31\\.03\\.2020", + "(?i)дебиторск" + ], + "notes": "Базовый корневой финансовый вопрос должен отработать точно и задать март 2020 как carryover-якорь." + }, + { + "step_id": "step_03_inventory_same_date", + "title": "Inventory on the same date", + "question": "остатки по складу на эту же дату", + "allowed_reply_types": [ + "factual" + ], + "expected_intents": [ + "inventory_on_hand_as_of_date" + ], + "expected_capability": "confirmed_inventory_on_hand_as_of_date", + "expected_result_mode": "confirmed_balance", + "required_filters": { + "as_of_date": "{{step_02_receivables_march_2020.filters.as_of_date}}", + "period_from": "{{step_02_receivables_march_2020.filters.period_from}}", + "period_to": "{{step_02_receivables_march_2020.filters.period_to}}" + }, + "required_direct_answer_patterns_any": [ + "31\\.03\\.2020", + "(?i)на складе" + ], + "notes": "Смена контура receivables -> inventory должна сохранить ту же дату, без дополнительного ручного уточнения." + }, + { + "step_id": "step_04_selected_item_seller", + "title": "Selected item purchase provenance", + "question": "По выбранному объекту \"Четки Пост (84*117)\": кто продавец", + "allowed_reply_types": [ + "factual" + ], + "expected_intents": [ + "inventory_purchase_provenance_for_item" + ], + "required_state_objects": [ + "focus_object" + ], + "required_carryover_invariants": [ + "focus_object" + ], + "forbidden_capabilities": [ + "confirmed_inventory_on_hand_as_of_date" + ], + "forbidden_recipes": [ + "address_inventory_on_hand_as_of_date_v1" + ], + "forbidden_filter_keys": [ + "as_of_date", + "period_from", + "period_to" + ], + "forbidden_direct_answer_patterns": [ + "^На 31\\.03\\.2020 на складе", + "(?i)^сейчас не дам прямой адресный ответ" + ], + "notes": "Выбранная позиция не должна реплеить складской срез; нужен именно ответ про поставщика/продавца." + }, + { + "step_id": "step_05_selected_item_sale_trace", + "title": "Selected item buyer", + "question": "По выбранному объекту \"Четки Пост (84*117)\": кому мы продали эту хуйню", + "allowed_reply_types": [ + "factual", + "partial_coverage" + ], + "allowed_limited_reason_categories": [ + "empty_match" + ], + "expected_intents": [ + "inventory_sale_trace_for_item" + ], + "required_state_objects": [ + "focus_object" + ], + "required_carryover_invariants": [ + "focus_object" + ], + "forbidden_filter_keys": [ + "as_of_date", + "period_from", + "period_to" + ], + "forbidden_direct_answer_patterns": [ + "(?i)^по текущим условиям в доступном срезе данных совпадений не нашлось", + "(?i)^сейчас не дам прямой адресный ответ" + ], + "notes": "След продажи по выбранной позиции не должен быть зажат складским snapshot-окном. Если по live данным подтвержденных продаж нет, честный partial_coverage с empty_match допустим, но только при правильном sale-trace routing и сохраненном focus_object." + }, + { + "step_id": "step_06_selected_item_purchase_followup", + "title": "Selected item purchase follow-up", + "question": "а купили у кого известно?", + "allowed_reply_types": [ + "factual" + ], + "expected_intents": [ + "inventory_purchase_provenance_for_item" + ], + "required_state_objects": [ + "focus_object" + ], + "required_carryover_invariants": [ + "focus_object" + ], + "forbidden_filter_keys": [ + "as_of_date", + "period_from", + "period_to" + ], + "forbidden_direct_answer_patterns": [ + "(?i)^для такого формата запроса нужен более широкий аналитический контур", + "(?i)^сейчас не дам прямой адресный ответ" + ], + "notes": "Короткий follow-up после выбранной позиции должен остаться в том же item-contour, а не свалиться в unknown." + }, + { + "step_id": "step_07_inventory_july_2019", + "title": "Inventory at July 2019", + "question": "остатки на июль 2019", + "allowed_reply_types": [ + "factual" + ], + "expected_intents": [ + "inventory_on_hand_as_of_date" + ], + "expected_capability": "confirmed_inventory_on_hand_as_of_date", + "expected_result_mode": "confirmed_balance", + "required_filters": { + "as_of_date": "2019-07-31", + "period_from": "2019-07-01", + "period_to": "2019-07-31" + }, + "required_direct_answer_patterns_any": [ + "31\\.07\\.2019", + "(?i)на складе" + ], + "forbidden_direct_answer_patterns": [ + "(?i)^сейчас не дам прямой адресный ответ" + ], + "notes": "После провала provenance-среза система все равно должна уметь коротко вернуть root inventory frame по явному месяцу." + }, + { + "step_id": "step_08_inventory_september_2019", + "title": "Inventory at September 2019", + "question": "сентябрь 2019", + "allowed_reply_types": [ + "factual" + ], + "expected_intents": [ + "inventory_on_hand_as_of_date" + ], + "expected_capability": "confirmed_inventory_on_hand_as_of_date", + "expected_result_mode": "confirmed_balance", + "required_filters": { + "as_of_date": "2019-09-30", + "period_from": "2019-09-01", + "period_to": "2019-09-30" + }, + "required_direct_answer_patterns_any": [ + "30\\.09\\.2019", + "(?i)на складе" + ], + "forbidden_direct_answer_patterns": [ + "(?i)^сейчас не дам прямой адресный ответ" + ], + "notes": "Короткое bare-month follow-up должно удерживать складской корень без дополнительной расшифровки." + }, + { + "step_id": "step_09_inventory_march_2020", + "title": "Inventory at March 2020", + "question": "март 2020", + "allowed_reply_types": [ + "factual" + ], + "expected_intents": [ + "inventory_on_hand_as_of_date" + ], + "expected_capability": "confirmed_inventory_on_hand_as_of_date", + "expected_result_mode": "confirmed_balance", + "required_filters": { + "as_of_date": "2020-03-31", + "period_from": "2020-03-01", + "period_to": "2020-03-31" + }, + "required_direct_answer_patterns_any": [ + "31\\.03\\.2020", + "(?i)на складе" + ], + "forbidden_direct_answer_patterns": [ + "(?i)^сейчас не дам прямой адресный ответ" + ], + "notes": "Возврат на март 2020 должен снова дать точный складской срез, а не unknown/partial." + }, + { + "step_id": "step_10_inventory_same_date_negative_wording", + "title": "Inventory same date with noisy wording", + "question": "остатков на складе нет на эту дату?", + "allowed_reply_types": [ + "factual" + ], + "expected_intents": [ + "inventory_on_hand_as_of_date" + ], + "expected_capability": "confirmed_inventory_on_hand_as_of_date", + "expected_result_mode": "confirmed_balance", + "required_filters": { + "as_of_date": "{{step_09_inventory_march_2020.filters.as_of_date}}", + "period_from": "{{step_09_inventory_march_2020.filters.period_from}}", + "period_to": "{{step_09_inventory_march_2020.filters.period_to}}" + }, + "forbidden_filter_values": { + "warehouse": [ + "нет на эту дату" + ] + }, + "required_direct_answer_patterns_any": [ + "31\\.03\\.2020", + "(?i)на складе" + ], + "forbidden_direct_answer_patterns": [ + "(?i)^сейчас не дам прямой адресный ответ" + ], + "notes": "Разговорная частица `нет на эту дату` не должна становиться warehouse-anchor." + }, + { + "step_id": "step_11_vat_same_date", + "title": "VAT on the same date", + "question": "ндс какой надо заплатить на эту же дату", + "allowed_reply_types": [ + "factual" + ], + "expected_intents": [ + "vat_liability_confirmed_for_tax_period" + ], + "expected_capability": "confirmed_vat_liability_for_tax_period", + "required_filters": { + "period_from": "{{step_09_inventory_march_2020.filters.period_from}}", + "period_to": "{{step_09_inventory_march_2020.filters.period_to}}" + }, + "required_direct_answer_patterns_any": [ + "(?i)ндс" + ], + "notes": "Pivot inventory -> VAT по `на эту же дату` должен привязаться к марту 2020, а не к мусорному складскому anchor." + }, + { + "step_id": "step_12_vat_may_2016", + "title": "VAT at May 2016", + "question": "а на май 2016", + "allowed_reply_types": [ + "factual" + ], + "expected_intents": [ + "vat_liability_confirmed_for_tax_period" + ], + "expected_capability": "confirmed_vat_liability_for_tax_period", + "forbidden_filter_values": { + "period_from": [ + "{{step_11_vat_same_date.filters.period_from}}" + ], + "period_to": [ + "{{step_11_vat_same_date.filters.period_to}}" + ] + }, + "required_answer_patterns_any": [ + "2016", + "(?i)налоговый период" + ], + "required_direct_answer_patterns_any": [ + "(?i)ндс" + ], + "notes": "Короткий temporal follow-up внутри VAT-frame должен уйти с марта 2020 на май 2016." + }, + { + "step_id": "step_13_receivables_same_date_after_vat_2016", + "title": "Receivables on the carried 2016 date", + "question": "кто нам должен денег на эту дату", + "allowed_reply_types": [ + "factual" + ], + "expected_intents": [ + "receivables_confirmed_as_of_date" + ], + "expected_capability": "confirmed_receivables_as_of_date", + "required_filter_within_previous_step_period": { + "as_of_date": "step_12_vat_may_2016" + }, + "required_answer_patterns_any": [ + "2016" + ], + "required_direct_answer_patterns_any": [ + "(?i)дебиторск" + ], + "forbidden_direct_answer_patterns": [ + "31\\.03\\.2020" + ], + "notes": "Фраза `на эту дату` после VAT 2016 не должна откатываться обратно на март 2020." + }, + { + "step_id": "step_14_receivables_today", + "title": "Receivables today", + "question": "а на сегодня", + "allowed_reply_types": [ + "factual" + ], + "expected_intents": [ + "receivables_confirmed_as_of_date" + ], + "expected_capability": "confirmed_receivables_as_of_date", + "required_filters": { + "as_of_date": "{{runtime.today_iso}}" + }, + "required_direct_answer_patterns_any": [ + "{{runtime.today_dot_regex}}", + "(?i)дебиторск" + ], + "notes": "Последний шаг нужен как sanity-check: `на сегодня` должен честно пересчитать уже на текущую дату прогона." + } + ] +} diff --git a/llm_normalizer/backend/dist/services/addressFilterExtractor.js b/llm_normalizer/backend/dist/services/addressFilterExtractor.js index bed3d43..d41b8e8 100644 --- a/llm_normalizer/backend/dist/services/addressFilterExtractor.js +++ b/llm_normalizer/backend/dist/services/addressFilterExtractor.js @@ -1051,6 +1051,12 @@ function isTemporalWarehousePhrase(candidate) { .toLowerCase() .replace(/ё/g, "е") .trim(); + if (/^(?:на\s+)?(?:эту|ту|текущ(?:ую|ая|ий|ее|ей)|сегодняшн(?:юю|ая|ий|ее|ей))(?:\s+же)?\s+дат(?:у|а|е|ой)$/iu.test(normalized)) { + return true; + } + if (/^(?:нет|не\s+было|не\s+было\s+ли)(?:\s+на\s+(?:эту|ту|такую)\s+дат(?:у|е|ой))?$/iu.test(normalized)) { + return true; + } return /^(?:в|на)\s+(?:январ(?:е|ь)|феврал(?:е|ь)|март(?:е)?|апрел(?:е|ь)|ма(?:й|е)|июн(?:е|ь)|июл(?:е|ь)|август(?:е)?|сентябр(?:е|ь)|октябр(?:е|ь)|ноябр(?:е|ь)|декабр(?:е|ь))(?:\s+\d{4}(?:\s+г(?:\.|ода)?)?)?$/iu.test(normalized); } function isLowQualityWarehouseAnchorValue(rawValue) { @@ -1097,7 +1103,8 @@ function isLowQualityWarehouseAnchorValue(rawValue) { "охуеть", "пиздец", "блять", - "бля" + "бля", + "нет" ]); const tokens = value .split(/[^a-zа-я0-9]+/iu) @@ -1127,7 +1134,7 @@ function hasImplicitSelfScopeSignal(text) { } function isImplicitSelfScopeWarehouseAnchor(candidate) { const normalized = normalizeSemanticAnchorCandidate(candidate); - return /^(?:у\s+нас|у\s+себя|у\s+меня|наш(?:ем|ей|его|их|а|е)?|сво(?:ем|ей|его|их|я|е)?)$/iu.test(normalized); + return /^(?:у\s+нас|у\s+себя|у\s+меня|наш(?:ем|ей|его|их|а|е)?|сво(?:ем|ей|его|их|я|е)?)(?:\s+(?:висит|висят|висело|висели|лежит|лежат|лежало|лежали))?$/iu.test(normalized); } function hasSelectedObjectScopeSignal(text) { return /(?:по\s+выбранному\s+объекту|selected\s+object)/iu.test(String(text ?? "")); diff --git a/llm_normalizer/backend/dist/services/addressQueryService.js b/llm_normalizer/backend/dist/services/addressQueryService.js index e2b361b..c7b1988 100644 --- a/llm_normalizer/backend/dist/services/addressQueryService.js +++ b/llm_normalizer/backend/dist/services/addressQueryService.js @@ -1039,10 +1039,21 @@ function toNormalizedRows(rows) { .filter((item) => Boolean(item.period || item.registrator)); } function rowSearchableText(row) { - return [row.registrator, row.item ?? "", row.warehouse ?? "", row.account_dt ?? "", row.account_kt ?? "", ...row.analytics] + return [ + row.registrator, + row.item ?? "", + row.warehouse ?? "", + row.organization ?? "", + row.account_dt ?? "", + row.account_kt ?? "", + ...row.analytics + ] .join(" ") .toLowerCase(); } +function rowOrganizationSearchableText(row) { + return [row.organization ?? "", row.registrator, ...row.analytics].join(" ").toLowerCase(); +} function rowMatchesAnyAccount(row, accountScope) { if (accountScope.length === 0) { return true; @@ -1107,9 +1118,12 @@ function applyAddressFilters(rows, filters) { if (filters.organization && String(filters.organization).trim()) { const needle = String(filters.organization); const before = filtered.length; - filtered = filtered.filter((row) => matchesAnchorText(rowSearchableText(row), needle)); - if (before > 0 && filtered.length === 0 && mismatchReason === null) { - mismatchReason = "organization_anchor_not_matched_in_materialized_rows"; + const organizationMaterialized = filtered.some((row) => Boolean(String(row.organization ?? "").trim())); + if (organizationMaterialized) { + filtered = filtered.filter((row) => matchesAnchorText(rowOrganizationSearchableText(row), needle)); + if (before > 0 && filtered.length === 0 && mismatchReason === null) { + mismatchReason = "organization_anchor_not_matched_in_materialized_rows"; + } } } if (filters.item && String(filters.item).trim()) { diff --git a/llm_normalizer/backend/dist/services/addressRecipeCatalog.js b/llm_normalizer/backend/dist/services/addressRecipeCatalog.js index 289f2ab..13e1920 100644 --- a/llm_normalizer/backend/dist/services/addressRecipeCatalog.js +++ b/llm_normalizer/backend/dist/services/addressRecipeCatalog.js @@ -1114,9 +1114,22 @@ function buildInventoryMovementQuery(filters, resolvedLimit, side) { : side === "kt" ? creditPredicate : `(${debitPredicate} ИЛИ ${creditPredicate})`; + const itemFieldPaths = side === "dt" + ? ["Движения.СубконтоДт1", "Движения.СубконтоДт2", "Движения.СубконтоДт3"] + : side === "kt" + ? ["Движения.СубконтоКт1", "Движения.СубконтоКт2", "Движения.СубконтоКт3"] + : [ + "Движения.СубконтоДт1", + "Движения.СубконтоДт2", + "Движения.СубконтоДт3", + "Движения.СубконтоКт1", + "Движения.СубконтоКт2", + "Движения.СубконтоКт3" + ]; + const itemCondition = buildInventoryItemReferenceCondition(filters, itemFieldPaths); return INVENTORY_MOVEMENTS_QUERY_TEMPLATE .replace("__LIMIT__", String(resolvedLimit)) - .replace("__WHERE_CLAUSE__", buildWhereClause(filters, "Движения.Период", [inventoryCondition])) + .replace("__WHERE_CLAUSE__", buildWhereClause(filters, "Движения.Период", [inventoryCondition, itemCondition].filter((item) => Boolean(item)))) .replaceAll("__ORDER_DIRECTION__", resolveOrderDirection(filters.sort)); } function buildInventoryItemReferenceCondition(filters, fieldPaths) { @@ -1332,7 +1345,7 @@ function buildAddressRecipePlan(recipe, filters) { : recipe.query_template === "inventory_supplier_stock_overlap_profile" ? buildInventoryMovementQuery(filters, resolvedLimit, "dt") : recipe.query_template === "inventory_sale_trace_profile" - ? buildInventorySaleDocumentQuery(filters, resolvedLimit) + ? buildInventoryMovementQuery(filters, resolvedLimit, "kt") : recipe.query_template === "inventory_purchase_to_sale_chain_profile" ? buildInventoryMovementQuery(filters, resolvedLimit, "either") : recipe.query_template === "inventory_aging_by_purchase_date_profile" diff --git a/llm_normalizer/backend/dist/services/address_runtime/decomposeStage.js b/llm_normalizer/backend/dist/services/address_runtime/decomposeStage.js index 6c5ee9f..1c7fe1a 100644 --- a/llm_normalizer/backend/dist/services/address_runtime/decomposeStage.js +++ b/llm_normalizer/backend/dist/services/address_runtime/decomposeStage.js @@ -31,7 +31,7 @@ function hasAllTimeHint(text) { return /(?:за\s+вс[её]\s+время|за\s+весь\s+период|за\s+весь\s+срок|за\s+всю\s+истори(?:ю|и)|за\s+любой\s+период|за\s+любой\s+срок|for\s+all\s+time|all\s+time|for\s+entire\s+period|entire\s+period|for\s+any\s+period|any\s+period|for\s+full\s+history|full\s+history)/iu.test(normalized); } function hasSameDateHint(text) { - return /(?:на\s+ту\s+же\s+дат[ауеы]|на\s+эту\s+же\s+дат[ауеы]|на\s+эту\s+дат[ауеы]|эту\s+дат[ауеы]|та\s+же\s+дата|same\s+date|as\s+of\s+same\s+date|the\s+same\s+date)/iu.test(String(text ?? "")); + return /(?:на\s+ту\s+же\s+дат[ауеы]|на\s+эту\s+же\s+дат[ауеы]|на\s+эту\s+дат[ауеы]|эту\s+дат[ауеы]|та\s+же\s+дата|дат[ауеы],?\s+котор(?:ую|ая)\s+(?:до\s+этого|раньше|ранее)\s+(?:рассматривали|смотрели)|дат[ауеы],?\s+которая\s+был[ао]?\s+ранее\s+рассмотрен[ао]?|same\s+date|as\s+of\s+same\s+date|the\s+same\s+date|date\s+we\s+looked\s+at\s+before|previously\s+considered\s+date)/iu.test(String(text ?? "")); } function hasSamePeriodHint(text) { return /(?:на\s+тот\s+же\s+период|за\s+тот\s+же\s+период|тот\s+же\s+период(?:\s+рассмотрения)?|на\s+этот\s+же\s+период|за\s+этот\s+же\s+период|аналогичн\w+\s+текущ\w+\s+период\w+|same\s+period|same\s+range|same\s+window)/iu.test(String(text ?? "")); @@ -382,17 +382,49 @@ function resolveRelativeMonthPeriodFromInventoryRoot(userMessage, followupContex as_of_date: periodTo }; } +function resolveRelativeMonthPeriodFromFollowupYear(userMessage, followupContext) { + if (!followupContext) { + return null; + } + const month = resolveMonthNumberFromText(userMessage); + if (!month) { + return null; + } + const normalized = String(userMessage ?? ""); + if (hasExplicitPeriodLiteral(normalized) || hasExplicitCurrentDateHint(normalized) || hasSameDateHint(normalized)) { + return null; + } + const shortTemporalPatch = getTokenCount(normalized) <= 8 || hasRelativeYearHint(normalized); + if (!shortTemporalPatch) { + return null; + } + const year = resolveYearFromFilters(followupContext.previous_filters) ?? + resolveYearFromFilters(followupContext.root_filters); + if (!year) { + return null; + } + const lastDay = new Date(Date.UTC(year, month, 0)).getUTCDate(); + const periodFrom = `${year}-${String(month).padStart(2, "0")}-01`; + const periodTo = `${year}-${String(month).padStart(2, "0")}-${String(lastDay).padStart(2, "0")}`; + return { + period_from: periodFrom, + period_to: periodTo, + as_of_date: periodTo + }; +} function shouldRestoreInventoryRootFrame(userMessage, intent, extractedFilters, followupContext) { if (!followupContext || !isInventoryRootFrameIntent(followupContext.root_intent)) { return false; } const currentFrameKind = followupContext.current_frame_kind ?? null; const previousIntent = followupContext.previous_intent; + const rootContextOnly = followupContext.root_context_only === true; const comingFromInventoryDrilldown = currentFrameKind === "inventory_drilldown" || isInventoryDrilldownFrameIntent(previousIntent); const normalized = String(userMessage ?? ""); const hasInventoryRootRestatementCue = /(?:склад|остат(?:ок|ки)|позици(?:я|и|ю)|товар(?:ы|ов)?|номенклатур)/iu.test(normalized) && /(?:покажи|показать|выведи|раскрой|еще\s+раз|ещ[её]\s+раз|снова|опять|верни|вернись|повтори|тот\s+же|этот\s+же|same|again)/iu.test(normalized); const canReenterInventoryRoot = comingFromInventoryDrilldown || + rootContextOnly || (currentFrameKind === "inventory_root" && hasSamePeriodHint(normalized)) || (currentFrameKind === "generic" && hasInventoryRootRestatementCue && hasSamePeriodHint(normalized)); if (!canReenterInventoryRoot) { @@ -401,7 +433,7 @@ function shouldRestoreInventoryRootFrame(userMessage, intent, extractedFilters, if (intent !== "unknown" && !isInventoryIntent(intent) && !hasInventoryRootRestatementCue) { return false; } - if (hasSelectedObjectInventorySignal(normalized) || + if ((hasSelectedObjectInventorySignal(normalized) && !hasInventoryRootRestatementCue) || hasInventorySupplierFollowupCue(normalized) || hasInventoryPurchaseDocumentsFollowupCue(normalized) || hasInventoryPurchaseDateFollowupCue(normalized) || @@ -415,6 +447,7 @@ function shouldRestoreInventoryRootFrame(userMessage, intent, extractedFilters, } const hasTemporalPatch = hasExplicitPeriodWindow(extractedFilters) || Boolean(toNonEmptyString(extractedFilters.as_of_date)) || + hasSameDateHint(normalized) || hasSamePeriodHint(normalized) || hasExplicitPeriodLiteral(normalized) || Boolean(resolveRelativeMonthPeriodFromInventoryRoot(normalized, followupContext)); @@ -544,9 +577,11 @@ function mergeFollowupFilters(current, intent, userMessage, followupContext) { const previousPeriodFrom = toNonEmptyString(previous.period_from); const previousPeriodTo = toNonEmptyString(previous.period_to); const relativeMonthFromInventoryRoot = resolveRelativeMonthPeriodFromInventoryRoot(userMessage, followupContext); + const relativeMonthFromFollowupYear = resolveRelativeMonthPeriodFromFollowupYear(userMessage, followupContext); const allTimeRequested = hasAllTimeHint(userMessage); const sameDateRequested = hasSameDateHint(userMessage); const samePeriodRequested = hasSamePeriodHint(userMessage); + const explicitQuotedItem = extractSelectedObjectItemFromFollowupText(userMessage); if (!toNonEmptyString(merged.organization) && previousOrganization) { merged.organization = previousOrganization; reasons.push("organization_from_followup_context"); @@ -654,6 +689,15 @@ function mergeFollowupFilters(current, intent, userMessage, followupContext) { merged.counterparty = inheritedCounterparty; reasons.push(currentCounterparty ? "counterparty_replaced_from_followup_context" : "counterparty_from_followup_context"); } + if (intent === "inventory_on_hand_as_of_date" && explicitQuotedItem) { + const currentItem = toNonEmptyString(merged.item); + if (!currentItem || + currentItem !== explicitQuotedItem || + (0, addressFilterExtractor_1.isInventoryItemAnchorDegradation)(explicitQuotedItem, currentItem)) { + merged.item = explicitQuotedItem; + reasons.push(currentItem ? "item_replaced_from_explicit_quote" : "item_from_explicit_quote"); + } + } if ((intent === "inventory_purchase_provenance_for_item" || intent === "inventory_purchase_documents_for_item" || intent === "inventory_sale_trace_for_item" || @@ -661,7 +705,6 @@ function mergeFollowupFilters(current, intent, userMessage, followupContext) { intent === "inventory_purchase_to_sale_chain" || intent === "inventory_aging_by_purchase_date")) { const inheritedItem = previousItem ?? previousAnchorItem; - const explicitQuotedItem = extractSelectedObjectItemFromFollowupText(userMessage); const currentItem = toNonEmptyString(merged.item); const shouldAdoptExplicitQuotedItem = Boolean(explicitQuotedItem) && (!currentItem || @@ -780,6 +823,7 @@ function mergeFollowupFilters(current, intent, userMessage, followupContext) { } if (!sameDateRequested && hasFollowupSignalForConfirmed && + !isInventoryLifecycleHistoryIntent(intent) && !hasExplicitPeriodLiteral(userMessage) && !hasExplicitCurrentDateHint(userMessage)) { const inheritedAsOfDate = previousAsOfDate ?? previousPeriodTo ?? previousPeriodFrom; @@ -830,10 +874,26 @@ function mergeFollowupFilters(current, intent, userMessage, followupContext) { intent === "vat_payable_confirmed_as_of_date"; const currentHasPeriod = hasExplicitPeriodWindow(merged); const previousHasPeriod = hasExplicitPeriodWindow(previous); + const vatRelativeMonthFollowup = relativeMonthFromFollowupYear && + (intent === "vat_payable_confirmed_as_of_date" || + intent === "vat_payable_forecast" || + intent === "vat_liability_confirmed_for_tax_period"); + if (vatRelativeMonthFollowup) { + merged.period_from = relativeMonthFromFollowupYear.period_from; + merged.period_to = relativeMonthFromFollowupYear.period_to; + if (intent === "vat_payable_confirmed_as_of_date") { + merged.as_of_date = relativeMonthFromFollowupYear.as_of_date; + } + else if (toNonEmptyString(merged.as_of_date)) { + delete merged.as_of_date; + } + reasons.push("period_derived_from_followup_context_year"); + } if ((intent === "vat_payable_forecast" || intent === "vat_liability_confirmed_for_tax_period") && previousHasPeriod && hasFollowupSignal && - !hasExplicitPeriodInMessage) { + !hasExplicitPeriodInMessage && + !vatRelativeMonthFollowup) { const currentPeriodFrom = toNonEmptyString(merged.period_from); const currentPeriodTo = toNonEmptyString(merged.period_to); const todayIso = new Date().toISOString().slice(0, 10); @@ -852,7 +912,8 @@ function mergeFollowupFilters(current, intent, userMessage, followupContext) { previousHasPeriod && hasFollowupSignal && !hasExplicitPeriodInMessage && - !inventoryLifecycleHistoryIntent) { + !inventoryLifecycleHistoryIntent && + !vatRelativeMonthFollowup) { if (previousPeriodFrom) { merged.period_from = previousPeriodFrom; } diff --git a/llm_normalizer/backend/dist/services/assistantAddressOrchestrationRuntimeAdapter.js b/llm_normalizer/backend/dist/services/assistantAddressOrchestrationRuntimeAdapter.js index d303cc2..6d000bc 100644 --- a/llm_normalizer/backend/dist/services/assistantAddressOrchestrationRuntimeAdapter.js +++ b/llm_normalizer/backend/dist/services/assistantAddressOrchestrationRuntimeAdapter.js @@ -10,6 +10,18 @@ function hasSelectedObjectInventoryActionCue(text) { const value = String(text ?? ""); return (/(?:кому[\s\S]{0,80}(?:продал[аи]?|реализова[нлт][а-я]*|поставил[аи]?|поставлен[а-я]*|отгрузил[аи]?|отгружен[а-я]*)|кому\s+был\s+продан|куда[\s\S]{0,80}(?:продал[аи]?|реализова[нлт][а-я]*|поставил[аи]?|поставлен[а-я]*|отгрузил[аи]?|отгружен[а-я]*)|кто[\s\S]{0,40}купил|кто\s+это\s+поставил|кто\s+поставил|у\s+кого\s+купили|у\s+кого\s+куплено|где\s+мы\s+купили|где\s+куплено|по\s+каким\s+документам|какими\s+документами|покажи\s+документы|документы[\s\S]{0,80}(?:по\s+(?:ним|ней|нему|этой\s+позиции|этому\s+товару)|операци)|документы\s+закупки|buyer|sale\s+trace|supplier|vendor|purchase\s+documents|purchase[\s-]?to[\s-]?sale|old\s+purchase|aged\s+stock)/iu.test(value) || (0, inventoryLifecycleCueHelpers_1.hasInventoryProfitabilityCue)(value)); } +function hasShortInventoryPurchaseFollowupCue(text) { + return /(?:^|[\s,.;:!?])(а\s+)?(?:купили\s+у\s+кого|у\s+кого\s+купили|поставщик|продавец|seller)(?:[\s,.;:!?]|$)/iu.test(String(text ?? "")); +} +function isInventorySelectedObjectOrRootIntent(intent) { + return (intent === "inventory_on_hand_as_of_date" || + intent === "inventory_purchase_provenance_for_item" || + intent === "inventory_purchase_documents_for_item" || + intent === "inventory_sale_trace_for_item" || + intent === "inventory_profitability_for_item" || + intent === "inventory_purchase_to_sale_chain" || + intent === "inventory_aging_by_purchase_date"); +} function isGenericCanonicalDriftIntent(intent) { return (intent === "open_items_by_counterparty_or_contract" || intent === "customer_revenue_and_payments" || @@ -19,6 +31,25 @@ function isGenericCanonicalDriftIntent(intent) { intent === "bank_operations_by_contract" || intent === "documents_forming_balance"); } +function hasSameDateFollowupSignal(text) { + return /(?:эту\s+же\s+дат(?:у|е|ой)|ту\s+же\s+дат(?:у|е|ой)|same\s+date)/iu.test(String(text ?? "")); +} +function hasExplicitCurrentDateSignal(text) { + return /(?:текущ(?:ую|ая|ий|ее|ей)\s+дат(?:у|а|е|ой)|сегодняшн(?:юю|ий|ей)\s+дат(?:у|а|е|ой)|today|current\s+date)/iu.test(String(text ?? "")); +} +function hasInventoryTemporalRootFollowupCue(text) { + const value = String(text ?? "").trim().toLowerCase(); + if (!value) { + return false; + } + const tokenCount = value.split(/\s+/).filter(Boolean).length; + const hasMonthYearCue = /(?:январ(?:ь|е)|феврал(?:ь|е)|март(?:е)?|апрел(?:ь|е)|ма(?:й|е)|июн(?:ь|е)|июл(?:ь|е)|август(?:е)?|сентябр(?:ь|е)|октябр(?:ь|е)|ноябр(?:ь|е)|декабр(?:ь|е))(?:\s+\d{4})?/iu.test(value) || /\b(?:19|20)\d{2}\b/u.test(value); + if (tokenCount <= 3 && hasMonthYearCue) { + return true; + } + const hasInventoryLexeme = /(?:остат|склад|товар|позици|номенклатур)/iu.test(value); + return hasInventoryLexeme && (hasMonthYearCue || hasSameDateFollowupSignal(value)); +} function shouldPreferRawFollowupMessage(userMessage, addressInputMessage, carryover, addressPreDecompose, toNonEmptyString) { if (!carryover?.followupContext || typeof carryover.followupContext !== "object") { return false; @@ -33,12 +64,29 @@ function shouldPreferRawFollowupMessage(userMessage, addressInputMessage, carryo : null; const mode = toNonEmptyString(predecomposeContract?.mode) ?? "unknown"; const intent = toNonEmptyString(predecomposeContract?.intent) ?? "unknown"; + const followupContext = carryover.followupContext && typeof carryover.followupContext === "object" + ? carryover.followupContext + : null; + const previousIntent = toNonEmptyString(followupContext?.previous_intent); + const rootIntent = toNonEmptyString(followupContext?.root_intent); + const previousAnchorType = toNonEmptyString(followupContext?.previous_anchor_type); + const hasInventoryItemCarryover = previousAnchorType === "item" && isInventorySelectedObjectOrRootIntent(previousIntent); + const hasInventoryFrameCarryover = isInventorySelectedObjectOrRootIntent(previousIntent) || + isInventorySelectedObjectOrRootIntent(rootIntent); if (mode === "unsupported" && intent === "unknown") { return true; } - return (hasSelectedObjectInventorySignal(rawMessage) && - hasSelectedObjectInventoryActionCue(rawMessage) && - isGenericCanonicalDriftIntent(intent)); + if (hasSameDateFollowupSignal(rawMessage) && hasExplicitCurrentDateSignal(canonicalMessage)) { + return true; + } + if (hasInventoryFrameCarryover && + hasInventoryTemporalRootFollowupCue(rawMessage) && + (intent === "account_balance_snapshot" || intent === "documents_forming_balance" || intent === "unknown")) { + return true; + } + return ((hasSelectedObjectInventorySignal(rawMessage) || hasInventoryItemCarryover) && + (hasSelectedObjectInventoryActionCue(rawMessage) || hasShortInventoryPurchaseFollowupCue(rawMessage)) && + (isGenericCanonicalDriftIntent(intent) || intent === "unknown")); } function fallbackAddressPreDecompose(userMessage, llmProvider, buildAddressLlmPredecomposeContractV1, sanitizeAddressMessageForFallback) { const provider = llmProvider === "local" ? "local" : llmProvider === "openai" ? "openai" : null; diff --git a/llm_normalizer/backend/dist/services/assistantRoutePolicy.js b/llm_normalizer/backend/dist/services/assistantRoutePolicy.js index e26e21d..1d737d8 100644 --- a/llm_normalizer/backend/dist/services/assistantRoutePolicy.js +++ b/llm_normalizer/backend/dist/services/assistantRoutePolicy.js @@ -159,8 +159,16 @@ function createAssistantRoutePolicy(deps) { hasShortDebtMirrorFollowupSignal(repairedRawUserMessage) || hasShortDebtMirrorFollowupSignal(effectiveAddressUserMessage) || hasShortDebtMirrorFollowupSignal(repairedEffectiveAddressUserMessage); + const followupPreviousIntent = toNonEmptyString(followupContext?.previous_intent); + const followupPreviousFilters = followupContext?.previous_filters && typeof followupContext.previous_filters === "object" + ? followupContext.previous_filters + : null; const protectedInventoryShortFollowup = Boolean(followupContext && - isInventorySelectedObjectIntent(toNonEmptyString(followupContext.previous_intent)) && + (isInventorySelectedObjectIntent(followupPreviousIntent) || + (followupPreviousIntent === "inventory_on_hand_as_of_date" && + (toNonEmptyString(followupContext.previous_anchor_type) === "item" || + toNonEmptyString(followupContext.previous_anchor_value) || + toNonEmptyString(followupPreviousFilters?.item)))) && (hasShortInventoryObjectFollowupSignal(rawUserMessage) || hasShortInventoryObjectFollowupSignal(repairedRawUserMessage) || hasShortInventoryObjectFollowupSignal(effectiveAddressUserMessage) || diff --git a/llm_normalizer/backend/dist/services/assistantService.js b/llm_normalizer/backend/dist/services/assistantService.js index e23fab3..329aecf 100644 --- a/llm_normalizer/backend/dist/services/assistantService.js +++ b/llm_normalizer/backend/dist/services/assistantService.js @@ -2799,8 +2799,11 @@ function hasShortInventoryObjectFollowupSignal(userMessage) { if (minTokens > 8) { return false; } + const hasDirectPurchaseFollowupCue = (sample) => /(?:^|[\s,.;:!?])(?:а\s+)?(?:у|от)\s+кого(?:\s+\S+){0,5}\s+купил(?:и|о)?(?=$|[\s,.;:!?])/iu.test(sample) || + /(?:^|[\s,.;:!?])(?:а\s+)?купил(?:и|о)?(?:\s+\S+){0,5}\s+(?:у|от)\s+кого(?=$|[\s,.;:!?])/iu.test(sample); const hasDirectSaleFollowupCue = (sample) => /(?:кому|каму|куда)(?:\s+\S+){0,4}\s+(?:продали|продано|продан(?:о|а|ы)?|реализовали|реализован(?:о|а|ы)?)|(?:продали|продано|реализовали|реализован(?:о|а|ы)?)(?:\s+\S+){0,4}\s+(?:кому|каму|куда)|(?:^|\s)(?:продано|продали|реализовано|реализовали)(?=$|[\s,.;:!?])/iu.test(sample); - return samples.some((sample) => /^(?:кто|когда|документы|сумма|поставщик|покупатель)(?:\?)?$/iu.test(sample) || + return samples.some((sample) => /^(?:кто|когда|документы|сумма|поставщик|покупатель|продавец|seller)(?:\?)?$/iu.test(sample) || + hasDirectPurchaseFollowupCue(sample) || hasDirectSaleFollowupCue(sample) || (0, decomposeStage_1.hasInventorySupplierFollowupCue)(sample) || (0, decomposeStage_1.hasInventoryPurchaseDocumentsFollowupCue)(sample) || @@ -2809,6 +2812,41 @@ function hasShortInventoryObjectFollowupSignal(userMessage) { (0, decomposeStage_1.hasInventorySaleFollowupCue)(sample) || (0, decomposeStage_1.hasInventoryPurchaseToSaleChainFollowupCue)(sample)); } +function hasInventoryRootTemporalFollowupSignal(userMessage, sourceIntentHint, hasInventoryRootFrame) { + if (!hasInventoryRootFrame) { + return false; + } + if (!(sourceIntentHint === "inventory_on_hand_as_of_date" || + sourceIntentHint === "inventory_supplier_stock_overlap_as_of_date" || + isInventorySelectedObjectIntent(sourceIntentHint))) { + return false; + } + const rawText = compactWhitespace(String(userMessage ?? "").toLowerCase()); + const repairedText = compactWhitespace(repairAddressMojibake(String(userMessage ?? "")).toLowerCase()); + const samples = [rawText, repairedText].filter((item) => item.length > 0); + if (samples.length === 0) { + return false; + } + const minTokens = samples.reduce((min, sample) => Math.min(min, countTokens(sample)), Number.POSITIVE_INFINITY); + if (minTokens > 8) { + return false; + } + const hasMonthYearCue = samples.some((sample) => /(?:январ|феврал|март|апрел|ма(?:й|е|я)|июн|июл|август|сентябр|октябр|ноябр|декабр)(?:\s+\d{4})?/iu.test(sample) || + /\b(?:19|20)\d{2}\b/u.test(sample)); + const hasTemporalCue = samples.some((sample) => hasPeriodLiteral(sample) || + hasShortNamedPeriodFollowupLiteral(sample) || + hasShortCurrentDateFollowupLiteral(sample) || + /(?:РЅР°\s+ту\s+Р¶Рµ\s+дат[ауеы]|РЅР°\s+эту\s+Р¶Рµ\s+дат[ауеы]|same\s+date|the\s+same\s+date)/iu.test(sample)); + if (!hasTemporalCue && !hasMonthYearCue) { + return false; + } + if (samples.some((sample) => hasForeignAccountingPivotOverInventoryMessage(sample))) { + return false; + } + const hasInventoryLexeme = samples.some((sample) => /(?:остат|склад|товар|номенклат|РїРѕР·РёС†)/iu.test(sample)); + const hasPlainInventoryLexeme = samples.some((sample) => /(?:остат|склад|товар|номенклатур|позиц)/iu.test(sample)); + return hasInventoryLexeme || hasPlainInventoryLexeme || minTokens <= 3; +} function hasForeignAccountingPivotOverInventoryMessage(userMessage, alternateMessage = null) { const samples = [ compactWhitespace(repairAddressMojibake(String(userMessage ?? "")).toLowerCase()), @@ -2890,8 +2928,22 @@ function resolveAddressFollowupCarryoverContext(userMessage, items, alternateMes (isImplicitAddressContinuationByLlm(userMessage, llmPreDecomposeMeta) || (toNonEmptyString(alternateMessage) ? isImplicitAddressContinuationByLlm(alternateMessage, llmPreDecomposeMeta) : false)); const sourceIntentHint = toNonEmptyString(previousAddressDebug?.detected_intent); - const inventoryShortFollowupPrimary = isInventorySelectedObjectIntent(sourceIntentHint) && hasShortInventoryObjectFollowupSignal(userMessage); - const inventoryShortFollowupAlternate = isInventorySelectedObjectIntent(sourceIntentHint) && toNonEmptyString(alternateMessage) + const navigationFocusObjectHint = addressNavigationState && + typeof addressNavigationState === "object" && + addressNavigationState.session_context && + typeof addressNavigationState.session_context === "object" && + addressNavigationState.session_context.active_focus_object && + typeof addressNavigationState.session_context.active_focus_object === "object" + ? addressNavigationState.session_context.active_focus_object + : null; + const hasNavigationInventoryItemFocusHint = Boolean(toNonEmptyString(navigationFocusObjectHint?.label) && + toNonEmptyString(navigationFocusObjectHint?.object_type) === "item" && + (sourceIntentHint === "inventory_on_hand_as_of_date" || + sourceIntentHint === "inventory_supplier_stock_overlap_as_of_date" || + isInventorySelectedObjectIntent(sourceIntentHint))); + let inventoryShortFollowupPrimary = (isInventorySelectedObjectIntent(sourceIntentHint) || hasNavigationInventoryItemFocusHint) && + hasShortInventoryObjectFollowupSignal(userMessage); + let inventoryShortFollowupAlternate = (isInventorySelectedObjectIntent(sourceIntentHint) || hasNavigationInventoryItemFocusHint) && toNonEmptyString(alternateMessage) ? hasShortInventoryObjectFollowupSignal(String(alternateMessage ?? "")) : false; const debtRoleSwapPrimary = sourceIntentHint ? resolveDebtRoleSwapFollowupIntent(userMessage, sourceIntentHint) : null; @@ -2899,8 +2951,8 @@ function resolveAddressFollowupCarryoverContext(userMessage, items, alternateMes ? resolveDebtRoleSwapFollowupIntent(String(alternateMessage ?? ""), sourceIntentHint) : null; const debtRoleSwapIntent = debtRoleSwapPrimary ?? debtRoleSwapAlternate ?? null; - const hasPrimaryFollowupSignal = hasAddressFollowupContextSignal(userMessage) || Boolean(debtRoleSwapPrimary) || inventoryShortFollowupPrimary; - const hasAlternateFollowupSignal = toNonEmptyString(alternateMessage) + let hasPrimaryFollowupSignal = hasAddressFollowupContextSignal(userMessage) || Boolean(debtRoleSwapPrimary) || inventoryShortFollowupPrimary; + let hasAlternateFollowupSignal = toNonEmptyString(alternateMessage) ? hasAddressFollowupContextSignal(alternateMessage) || Boolean(debtRoleSwapAlternate) || inventoryShortFollowupAlternate : false; const hasPrimaryIndexReferenceSignal = extractDisplayedEntityIndexMention(userMessage) !== null; @@ -2908,12 +2960,19 @@ function resolveAddressFollowupCarryoverContext(userMessage, items, alternateMes ? extractDisplayedEntityIndexMention(String(alternateMessage ?? "")) !== null : false; const hasIndexReferenceSignal = hasPrimaryIndexReferenceSignal || hasAlternateIndexReferenceSignal; - const hasStrongFollowupReference = hasPrimaryIndexReferenceSignal || + const recentInventoryRootFrame = findRecentInventoryRootFrame(items); + const hasInventoryRootTemporalFollowupPrimary = hasInventoryRootTemporalFollowupSignal(userMessage, sourceIntentHint, Boolean(recentInventoryRootFrame)); + const hasInventoryRootTemporalFollowupAlternate = toNonEmptyString(alternateMessage) + ? hasInventoryRootTemporalFollowupSignal(String(alternateMessage ?? ""), sourceIntentHint, Boolean(recentInventoryRootFrame)) + : false; + let hasStrongFollowupReference = hasPrimaryIndexReferenceSignal || hasAlternateIndexReferenceSignal || hasOrganizationClarificationContinuation || hasImplicitContinuationSignal || inventoryShortFollowupPrimary || inventoryShortFollowupAlternate || + hasInventoryRootTemporalFollowupPrimary || + hasInventoryRootTemporalFollowupAlternate || Boolean(debtRoleSwapIntent) || hasFollowupMarker(userMessage) || hasReferentialPointer(userMessage) || @@ -2925,6 +2984,8 @@ function resolveAddressFollowupCarryoverContext(userMessage, items, alternateMes if (hasStandaloneAddressTopic && !hasPrimaryFollowupSignal && !hasAlternateFollowupSignal && + !hasInventoryRootTemporalFollowupPrimary && + !hasInventoryRootTemporalFollowupAlternate && !hasImplicitContinuationSignal && !hasOrganizationClarificationContinuation && !hasIndexReferenceSignal) { @@ -2932,6 +2993,8 @@ function resolveAddressFollowupCarryoverContext(userMessage, items, alternateMes } if (!hasPrimaryFollowupSignal && !hasAlternateFollowupSignal && + !hasInventoryRootTemporalFollowupPrimary && + !hasInventoryRootTemporalFollowupAlternate && !hasImplicitContinuationSignal && !hasOrganizationClarificationContinuation && !hasIndexReferenceSignal) { @@ -2993,6 +3056,41 @@ function resolveAddressFollowupCarryoverContext(userMessage, items, alternateMes : null; const navigationFocusObjectType = toNonEmptyString(navigationFocusObject?.object_type); const navigationFocusObjectLabel = toNonEmptyString(navigationFocusObject?.label); + const hasInventoryItemFocusCarryover = navigationFocusObjectType === "item" && + navigationFocusObjectLabel && + (sourceIntentHint === "inventory_on_hand_as_of_date" || + sourceIntentHint === "inventory_supplier_stock_overlap_as_of_date" || + isInventorySelectedObjectIntent(sourceIntentHint)); + if (!inventoryShortFollowupPrimary && hasInventoryItemFocusCarryover) { + inventoryShortFollowupPrimary = hasShortInventoryObjectFollowupSignal(userMessage); + } + if (!inventoryShortFollowupAlternate && hasInventoryItemFocusCarryover && toNonEmptyString(alternateMessage)) { + inventoryShortFollowupAlternate = hasShortInventoryObjectFollowupSignal(String(alternateMessage ?? "")); + } + hasPrimaryFollowupSignal = hasAddressFollowupContextSignal(userMessage) || + Boolean(debtRoleSwapPrimary) || + inventoryShortFollowupPrimary || + hasInventoryRootTemporalFollowupPrimary; + hasAlternateFollowupSignal = toNonEmptyString(alternateMessage) + ? hasAddressFollowupContextSignal(alternateMessage) || + Boolean(debtRoleSwapAlternate) || + inventoryShortFollowupAlternate || + hasInventoryRootTemporalFollowupAlternate + : false; + hasStrongFollowupReference = hasPrimaryIndexReferenceSignal || + hasAlternateIndexReferenceSignal || + hasOrganizationClarificationContinuation || + hasImplicitContinuationSignal || + inventoryShortFollowupPrimary || + inventoryShortFollowupAlternate || + hasInventoryRootTemporalFollowupPrimary || + hasInventoryRootTemporalFollowupAlternate || + Boolean(debtRoleSwapIntent) || + hasFollowupMarker(userMessage) || + hasReferentialPointer(userMessage) || + (toNonEmptyString(alternateMessage) + ? hasFollowupMarker(String(alternateMessage ?? "")) || hasReferentialPointer(String(alternateMessage ?? "")) + : false); const hasSelectedObjectInventorySignalPrimary = /(?:по\s+выбранному\s+объекту|по\s+этой\s+позиции|по\s+этому\s+товару|selected\s+object)/iu.test(String(userMessage ?? "")); const hasSelectedObjectInventorySignalAlternate = toNonEmptyString(alternateMessage) ? /(?:по\s+выбранному\s+объекту|по\s+этой\s+позиции|по\s+этому\s+товару|selected\s+object)/iu.test(String(alternateMessage ?? "")) @@ -3054,18 +3152,39 @@ function resolveAddressFollowupCarryoverContext(userMessage, items, alternateMes if (!toNonEmptyString(previousFilters.organization) && organizationClarificationSelection) { previousFilters.organization = organizationClarificationSelection; } - if (!toNonEmptyString(previousFilters.as_of_date) && toNonEmptyString(navigationDateScope?.as_of_date)) { + const shouldBackfillPreviousDateScopeFromNavigation = sourceIntentHint === "inventory_on_hand_as_of_date" || + sourceIntentHint === "inventory_supplier_stock_overlap_as_of_date" || + sourceIntentHint === "inventory_purchase_provenance_for_item" || + sourceIntentHint === "inventory_purchase_documents_for_item" || + sourceIntentHint === "inventory_sale_trace_for_item" || + sourceIntentHint === "inventory_profitability_for_item" || + sourceIntentHint === "inventory_purchase_to_sale_chain" || + sourceIntentHint === "inventory_aging_by_purchase_date" || + sourceIntentHint === "account_balance_snapshot" || + sourceIntentHint === "documents_forming_balance"; + if (shouldBackfillPreviousDateScopeFromNavigation && + !toNonEmptyString(previousFilters.as_of_date) && + toNonEmptyString(navigationDateScope?.as_of_date)) { previousFilters.as_of_date = toNonEmptyString(navigationDateScope?.as_of_date); } - if (!toNonEmptyString(previousFilters.period_from) && toNonEmptyString(navigationDateScope?.period_from)) { + if (shouldBackfillPreviousDateScopeFromNavigation && + !toNonEmptyString(previousFilters.period_from) && + toNonEmptyString(navigationDateScope?.period_from)) { previousFilters.period_from = toNonEmptyString(navigationDateScope?.period_from); } - if (!toNonEmptyString(previousFilters.period_to) && toNonEmptyString(navigationDateScope?.period_to)) { + if (shouldBackfillPreviousDateScopeFromNavigation && + !toNonEmptyString(previousFilters.period_to) && + toNonEmptyString(navigationDateScope?.period_to)) { previousFilters.period_to = toNonEmptyString(navigationDateScope?.period_to); } const rootContextOnlyPivot = Boolean((isInventorySelectedObjectIntent(sourceIntentHint) || currentFrameKind === "inventory_drilldown") && hasForeignAccountingPivotOverInventoryMessage(userMessage, alternateMessage)); - if (rootContextOnlyPivot) { + const inventoryRootTemporalPivot = Boolean(inventoryRootFrame && + (isInventorySelectedObjectIntent(sourceIntentHint) || currentFrameKind === "inventory_drilldown") && + (hasInventoryRootTemporalFollowupPrimary || hasInventoryRootTemporalFollowupAlternate) && + !hasForeignAccountingPivotOverInventoryMessage(userMessage, alternateMessage)); + const rootScopedPivot = rootContextOnlyPivot || inventoryRootTemporalPivot; + if (rootScopedPivot) { previousIntent = null; previousAnchorType = null; previousAnchor = null; @@ -3079,7 +3198,7 @@ function resolveAddressFollowupCarryoverContext(userMessage, items, alternateMes (toNonEmptyString(alternateMessage) ? resolveDisplayedAddressEntityMention(String(alternateMessage ?? ""), displayedEntities) : null); - if (resolvedEntityFromFollowup && !rootContextOnlyPivot) { + if (resolvedEntityFromFollowup && !rootScopedPivot) { if (resolvedEntityFromFollowup.entityType === "counterparty") { previousFilters.counterparty = resolvedEntityFromFollowup.value; previousAnchorType = "counterparty"; @@ -3100,7 +3219,7 @@ function resolveAddressFollowupCarryoverContext(userMessage, items, alternateMes followupSelectionMode = "carry_referenced_entity"; } } - if (!rootContextOnlyPivot && + if (!rootScopedPivot && !toNonEmptyString(previousFilters.item) && navigationFocusObjectType === "item" && navigationFocusObjectLabel && @@ -3142,7 +3261,7 @@ function resolveAddressFollowupCarryoverContext(userMessage, items, alternateMes previous_anchor_type: previousAnchorType ?? undefined, previous_anchor_value: previousAnchor, resolved_counterparty_from_display: resolvedCounterpartyFromDisplay || undefined, - root_context_only: rootContextOnlyPivot || undefined, + root_context_only: rootScopedPivot || undefined, root_intent: inventoryRootFrame?.intent ?? undefined, root_filters: inventoryRootFrame?.filters ?? undefined, root_anchor_type: inventoryRootFrame?.anchorType ?? undefined, @@ -3163,11 +3282,13 @@ function buildAddressDialogContinuationContractV2(userMessage, effectiveMessage, const previousIntent = toNonEmptyString(carryoverMeta?.previousSourceIntent) ?? null; const selectionMode = toNonEmptyString(carryoverMeta?.followupSelectionMode) ?? null; const rootContextOnly = selectionMode === "carry_root_context"; - const explicitIntent = toNonEmptyString(llmPreDecomposeMeta?.predecomposeContract?.intent); + const explicitIntentRaw = toNonEmptyString(llmPreDecomposeMeta?.predecomposeContract?.intent); + const explicitIntent = explicitIntentRaw === "unknown" ? null : explicitIntentRaw; + const rootIntent = toNonEmptyString(carryoverMeta?.followupContext?.root_intent) ?? null; const targetIntent = selectionMode === "switch_to_suggested_intent" ? toNonEmptyString(carryoverMeta?.previousAddressIntent) ?? null : rootContextOnly - ? explicitIntent ?? null + ? rootIntent ?? explicitIntent ?? null : explicitIntent ?? toNonEmptyString(carryoverMeta?.previousAddressIntent) ?? null; const hasImplicitContinuationSignal = Boolean(carryoverMeta?.hasImplicitContinuationSignal); const rewrittenByPredecompose = compactWhitespace(sourceMessage.toLowerCase()) !== compactWhitespace(canonicalMessage.toLowerCase()); diff --git a/llm_normalizer/backend/dist/services/inventoryLifecycleCueHelpers.js b/llm_normalizer/backend/dist/services/inventoryLifecycleCueHelpers.js index 7e65c9e..cb841a5 100644 --- a/llm_normalizer/backend/dist/services/inventoryLifecycleCueHelpers.js +++ b/llm_normalizer/backend/dist/services/inventoryLifecycleCueHelpers.js @@ -15,6 +15,9 @@ function hasInventorySupplierCue(text) { if (/(?:купил(?:и|о)?\s+у\s+кого|куплен(?:о)?\s+у\s+кого|купил(?:и|о)?\s+от\s+кого|куплен(?:о)?\s+от\s+кого)/iu.test(value)) { return true; } + if (/(?:кто\s+(?:был\s+)?продавец|кто\s+нам\s+продал|кто\s+продал\s+нам|(?:^|[\s,.;:!?()\-])(?:продавец|seller)(?=$|[\s,.;:!?()\-]))/iu.test(value)) { + return true; + } if (/(?:кто\s+(?:(?:это|этот\s+товар|эту\s+позицию)\s+)?(?:нам\s+)?поставил|кто\s+(?:нам\s+)?поставил\s+(?:это|этот\s+товар|эту\s+позицию)|от\s+какого\s+поставщика|у\s+какого\s+поставщика|от\s+кого\s+куплен|у\s+кого\s+купили|у\s+кого\s+куплено|где\s+(?:мы\s+)?купили(?:\s+(?:это|его|этот\s+товар|эту\s+позицию))?|где\s+(?:мы\s+)?взяли(?:\s+(?:это|его|этот\s+товар|эту\s+позицию))?|откуда\s+(?:мы\s+)?взяли(?:\s+(?:это|его|этот\s+товар|эту\s+позицию))?|где\s+куплено|supplier|vendor|поставщик)/iu.test(value)) { return true; } diff --git a/llm_normalizer/backend/src/services/addressFilterExtractor.ts b/llm_normalizer/backend/src/services/addressFilterExtractor.ts index 2531b4c..0b8465d 100644 --- a/llm_normalizer/backend/src/services/addressFilterExtractor.ts +++ b/llm_normalizer/backend/src/services/addressFilterExtractor.ts @@ -1189,6 +1189,12 @@ function isTemporalWarehousePhrase(candidate: string): boolean { .toLowerCase() .replace(/ё/g, "е") .trim(); + if (/^(?:на\s+)?(?:эту|ту|текущ(?:ую|ая|ий|ее|ей)|сегодняшн(?:юю|ая|ий|ее|ей))(?:\s+же)?\s+дат(?:у|а|е|ой)$/iu.test(normalized)) { + return true; + } + if (/^(?:нет|не\s+было|не\s+было\s+ли)(?:\s+на\s+(?:эту|ту|такую)\s+дат(?:у|е|ой))?$/iu.test(normalized)) { + return true; + } return /^(?:в|на)\s+(?:январ(?:е|ь)|феврал(?:е|ь)|март(?:е)?|апрел(?:е|ь)|ма(?:й|е)|июн(?:е|ь)|июл(?:е|ь)|август(?:е)?|сентябр(?:е|ь)|октябр(?:е|ь)|ноябр(?:е|ь)|декабр(?:е|ь))(?:\s+\d{4}(?:\s+г(?:\.|ода)?)?)?$/iu.test( normalized ); @@ -1244,7 +1250,8 @@ function isLowQualityWarehouseAnchorValue(rawValue: string): boolean { "охуеть", "пиздец", "блять", - "бля" + "бля", + "нет" ]); const tokens = value .split(/[^a-zа-я0-9]+/iu) @@ -1279,7 +1286,9 @@ function hasImplicitSelfScopeSignal(text: string): boolean { function isImplicitSelfScopeWarehouseAnchor(candidate: string): boolean { const normalized = normalizeSemanticAnchorCandidate(candidate); - return /^(?:у\s+нас|у\s+себя|у\s+меня|наш(?:ем|ей|его|их|а|е)?|сво(?:ем|ей|его|их|я|е)?)$/iu.test(normalized); + return /^(?:у\s+нас|у\s+себя|у\s+меня|наш(?:ем|ей|его|их|а|е)?|сво(?:ем|ей|его|их|я|е)?)(?:\s+(?:висит|висят|висело|висели|лежит|лежат|лежало|лежали))?$/iu.test( + normalized + ); } function hasSelectedObjectScopeSignal(text: string): boolean { diff --git a/llm_normalizer/backend/src/services/addressQueryService.ts b/llm_normalizer/backend/src/services/addressQueryService.ts index f94783d..4de361e 100644 --- a/llm_normalizer/backend/src/services/addressQueryService.ts +++ b/llm_normalizer/backend/src/services/addressQueryService.ts @@ -1286,11 +1286,23 @@ function toNormalizedRows(rows: Array>): NormalizedAddre } function rowSearchableText(row: NormalizedAddressRow): string { - return [row.registrator, row.item ?? "", row.warehouse ?? "", row.account_dt ?? "", row.account_kt ?? "", ...row.analytics] + return [ + row.registrator, + row.item ?? "", + row.warehouse ?? "", + row.organization ?? "", + row.account_dt ?? "", + row.account_kt ?? "", + ...row.analytics + ] .join(" ") .toLowerCase(); } +function rowOrganizationSearchableText(row: NormalizedAddressRow): string { + return [row.organization ?? "", row.registrator, ...row.analytics].join(" ").toLowerCase(); +} + function rowMatchesAnyAccount(row: NormalizedAddressRow, accountScope: string[]): boolean { if (accountScope.length === 0) { return true; @@ -1366,9 +1378,12 @@ function applyAddressFilters(rows: NormalizedAddressRow[], filters: AddressFilte if (filters.organization && String(filters.organization).trim()) { const needle = String(filters.organization); const before = filtered.length; - filtered = filtered.filter((row) => matchesAnchorText(rowSearchableText(row), needle)); - if (before > 0 && filtered.length === 0 && mismatchReason === null) { - mismatchReason = "organization_anchor_not_matched_in_materialized_rows"; + const organizationMaterialized = filtered.some((row) => Boolean(String(row.organization ?? "").trim())); + if (organizationMaterialized) { + filtered = filtered.filter((row) => matchesAnchorText(rowOrganizationSearchableText(row), needle)); + if (before > 0 && filtered.length === 0 && mismatchReason === null) { + mismatchReason = "organization_anchor_not_matched_in_materialized_rows"; + } } } diff --git a/llm_normalizer/backend/src/services/addressRecipeCatalog.ts b/llm_normalizer/backend/src/services/addressRecipeCatalog.ts index eddfc98..5182be8 100644 --- a/llm_normalizer/backend/src/services/addressRecipeCatalog.ts +++ b/llm_normalizer/backend/src/services/addressRecipeCatalog.ts @@ -1181,9 +1181,30 @@ function buildInventoryMovementQuery( : side === "kt" ? creditPredicate : `(${debitPredicate} ИЛИ ${creditPredicate})`; + const itemFieldPaths = + side === "dt" + ? ["Движения.СубконтоДт1", "Движения.СубконтоДт2", "Движения.СубконтоДт3"] + : side === "kt" + ? ["Движения.СубконтоКт1", "Движения.СубконтоКт2", "Движения.СубконтоКт3"] + : [ + "Движения.СубконтоДт1", + "Движения.СубконтоДт2", + "Движения.СубконтоДт3", + "Движения.СубконтоКт1", + "Движения.СубконтоКт2", + "Движения.СубконтоКт3" + ]; + const itemCondition = buildInventoryItemReferenceCondition(filters, itemFieldPaths); return INVENTORY_MOVEMENTS_QUERY_TEMPLATE .replace("__LIMIT__", String(resolvedLimit)) - .replace("__WHERE_CLAUSE__", buildWhereClause(filters, "Движения.Период", [inventoryCondition])) + .replace( + "__WHERE_CLAUSE__", + buildWhereClause( + filters, + "Движения.Период", + [inventoryCondition, itemCondition].filter((item): item is string => Boolean(item)) + ) + ) .replaceAll("__ORDER_DIRECTION__", resolveOrderDirection(filters.sort)); } @@ -1469,7 +1490,7 @@ export function buildAddressRecipePlan( : recipe.query_template === "inventory_supplier_stock_overlap_profile" ? buildInventoryMovementQuery(filters, resolvedLimit, "dt") : recipe.query_template === "inventory_sale_trace_profile" - ? buildInventorySaleDocumentQuery(filters, resolvedLimit) + ? buildInventoryMovementQuery(filters, resolvedLimit, "kt") : recipe.query_template === "inventory_purchase_to_sale_chain_profile" ? buildInventoryMovementQuery(filters, resolvedLimit, "either") : recipe.query_template === "inventory_aging_by_purchase_date_profile" diff --git a/llm_normalizer/backend/src/services/address_runtime/decomposeStage.ts b/llm_normalizer/backend/src/services/address_runtime/decomposeStage.ts index e954621..9fa4987 100644 --- a/llm_normalizer/backend/src/services/address_runtime/decomposeStage.ts +++ b/llm_normalizer/backend/src/services/address_runtime/decomposeStage.ts @@ -52,6 +52,7 @@ export interface AddressFollowupContext { | "unknown" | null; root_anchor_value?: string | null; + root_context_only?: boolean; current_frame_kind?: "generic" | "inventory_root" | "inventory_drilldown"; } @@ -91,7 +92,7 @@ function hasAllTimeHint(text: string): boolean { } function hasSameDateHint(text: string): boolean { - return /(?:на\s+ту\s+же\s+дат[ауеы]|на\s+эту\s+же\s+дат[ауеы]|на\s+эту\s+дат[ауеы]|эту\s+дат[ауеы]|та\s+же\s+дата|same\s+date|as\s+of\s+same\s+date|the\s+same\s+date)/iu.test( + return /(?:на\s+ту\s+же\s+дат[ауеы]|на\s+эту\s+же\s+дат[ауеы]|на\s+эту\s+дат[ауеы]|эту\s+дат[ауеы]|та\s+же\s+дата|дат[ауеы],?\s+котор(?:ую|ая)\s+(?:до\s+этого|раньше|ранее)\s+(?:рассматривали|смотрели)|дат[ауеы],?\s+которая\s+был[ао]?\s+ранее\s+рассмотрен[ао]?|same\s+date|as\s+of\s+same\s+date|the\s+same\s+date|date\s+we\s+looked\s+at\s+before|previously\s+considered\s+date)/iu.test( String(text ?? "") ); } @@ -487,6 +488,41 @@ function resolveRelativeMonthPeriodFromInventoryRoot( }; } +function resolveRelativeMonthPeriodFromFollowupYear( + userMessage: string, + followupContext: AddressFollowupContext | null +): { period_from: string; period_to: string; as_of_date: string } | null { + if (!followupContext) { + return null; + } + const month = resolveMonthNumberFromText(userMessage); + if (!month) { + return null; + } + const normalized = String(userMessage ?? ""); + if (hasExplicitPeriodLiteral(normalized) || hasExplicitCurrentDateHint(normalized) || hasSameDateHint(normalized)) { + return null; + } + const shortTemporalPatch = getTokenCount(normalized) <= 8 || hasRelativeYearHint(normalized); + if (!shortTemporalPatch) { + return null; + } + const year = + resolveYearFromFilters(followupContext.previous_filters) ?? + resolveYearFromFilters(followupContext.root_filters); + if (!year) { + return null; + } + const lastDay = new Date(Date.UTC(year, month, 0)).getUTCDate(); + const periodFrom = `${year}-${String(month).padStart(2, "0")}-01`; + const periodTo = `${year}-${String(month).padStart(2, "0")}-${String(lastDay).padStart(2, "0")}`; + return { + period_from: periodFrom, + period_to: periodTo, + as_of_date: periodTo + }; +} + function shouldRestoreInventoryRootFrame( userMessage: string, intent: AddressIntent, @@ -498,6 +534,7 @@ function shouldRestoreInventoryRootFrame( } const currentFrameKind = followupContext.current_frame_kind ?? null; const previousIntent = followupContext.previous_intent; + const rootContextOnly = followupContext.root_context_only === true; const comingFromInventoryDrilldown = currentFrameKind === "inventory_drilldown" || isInventoryDrilldownFrameIntent(previousIntent); const normalized = String(userMessage ?? ""); @@ -508,6 +545,7 @@ function shouldRestoreInventoryRootFrame( ); const canReenterInventoryRoot = comingFromInventoryDrilldown || + rootContextOnly || (currentFrameKind === "inventory_root" && hasSamePeriodHint(normalized)) || (currentFrameKind === "generic" && hasInventoryRootRestatementCue && hasSamePeriodHint(normalized)); if (!canReenterInventoryRoot) { @@ -517,7 +555,7 @@ function shouldRestoreInventoryRootFrame( return false; } if ( - hasSelectedObjectInventorySignal(normalized) || + (hasSelectedObjectInventorySignal(normalized) && !hasInventoryRootRestatementCue) || hasInventorySupplierFollowupCue(normalized) || hasInventoryPurchaseDocumentsFollowupCue(normalized) || hasInventoryPurchaseDateFollowupCue(normalized) || @@ -533,6 +571,7 @@ function shouldRestoreInventoryRootFrame( const hasTemporalPatch = hasExplicitPeriodWindow(extractedFilters) || Boolean(toNonEmptyString(extractedFilters.as_of_date)) || + hasSameDateHint(normalized) || hasSamePeriodHint(normalized) || hasExplicitPeriodLiteral(normalized) || Boolean(resolveRelativeMonthPeriodFromInventoryRoot(normalized, followupContext)); @@ -709,9 +748,11 @@ function mergeFollowupFilters( const previousPeriodFrom = toNonEmptyString(previous.period_from); const previousPeriodTo = toNonEmptyString(previous.period_to); const relativeMonthFromInventoryRoot = resolveRelativeMonthPeriodFromInventoryRoot(userMessage, followupContext); + const relativeMonthFromFollowupYear = resolveRelativeMonthPeriodFromFollowupYear(userMessage, followupContext); const allTimeRequested = hasAllTimeHint(userMessage); const sameDateRequested = hasSameDateHint(userMessage); const samePeriodRequested = hasSamePeriodHint(userMessage); + const explicitQuotedItem = extractSelectedObjectItemFromFollowupText(userMessage); if (!toNonEmptyString(merged.organization) && previousOrganization) { merged.organization = previousOrganization; reasons.push("organization_from_followup_context"); @@ -837,6 +878,17 @@ function mergeFollowupFilters( merged.counterparty = inheritedCounterparty; reasons.push(currentCounterparty ? "counterparty_replaced_from_followup_context" : "counterparty_from_followup_context"); } + if (intent === "inventory_on_hand_as_of_date" && explicitQuotedItem) { + const currentItem = toNonEmptyString(merged.item); + if ( + !currentItem || + currentItem !== explicitQuotedItem || + isInventoryItemAnchorDegradation(explicitQuotedItem, currentItem) + ) { + merged.item = explicitQuotedItem; + reasons.push(currentItem ? "item_replaced_from_explicit_quote" : "item_from_explicit_quote"); + } + } if ( (intent === "inventory_purchase_provenance_for_item" || intent === "inventory_purchase_documents_for_item" || @@ -846,7 +898,6 @@ function mergeFollowupFilters( intent === "inventory_aging_by_purchase_date") ) { const inheritedItem = previousItem ?? previousAnchorItem; - const explicitQuotedItem = extractSelectedObjectItemFromFollowupText(userMessage); const currentItem = toNonEmptyString(merged.item); const shouldAdoptExplicitQuotedItem = Boolean(explicitQuotedItem) && @@ -983,6 +1034,7 @@ function mergeFollowupFilters( if ( !sameDateRequested && hasFollowupSignalForConfirmed && + !isInventoryLifecycleHistoryIntent(intent) && !hasExplicitPeriodLiteral(userMessage) && !hasExplicitCurrentDateHint(userMessage) ) { @@ -1040,12 +1092,29 @@ function mergeFollowupFilters( intent === "vat_payable_confirmed_as_of_date"; const currentHasPeriod = hasExplicitPeriodWindow(merged); const previousHasPeriod = hasExplicitPeriodWindow(previous); + const vatRelativeMonthFollowup = + relativeMonthFromFollowupYear && + (intent === "vat_payable_confirmed_as_of_date" || + intent === "vat_payable_forecast" || + intent === "vat_liability_confirmed_for_tax_period"); + + if (vatRelativeMonthFollowup) { + merged.period_from = relativeMonthFromFollowupYear.period_from; + merged.period_to = relativeMonthFromFollowupYear.period_to; + if (intent === "vat_payable_confirmed_as_of_date") { + merged.as_of_date = relativeMonthFromFollowupYear.as_of_date; + } else if (toNonEmptyString(merged.as_of_date)) { + delete merged.as_of_date; + } + reasons.push("period_derived_from_followup_context_year"); + } if ( (intent === "vat_payable_forecast" || intent === "vat_liability_confirmed_for_tax_period") && previousHasPeriod && hasFollowupSignal && - !hasExplicitPeriodInMessage + !hasExplicitPeriodInMessage && + !vatRelativeMonthFollowup ) { const currentPeriodFrom = toNonEmptyString(merged.period_from); const currentPeriodTo = toNonEmptyString(merged.period_to); @@ -1067,7 +1136,8 @@ function mergeFollowupFilters( previousHasPeriod && hasFollowupSignal && !hasExplicitPeriodInMessage && - !inventoryLifecycleHistoryIntent + !inventoryLifecycleHistoryIntent && + !vatRelativeMonthFollowup ) { if (previousPeriodFrom) { merged.period_from = previousPeriodFrom; diff --git a/llm_normalizer/backend/src/services/assistantAddressOrchestrationRuntimeAdapter.ts b/llm_normalizer/backend/src/services/assistantAddressOrchestrationRuntimeAdapter.ts index c997c48..e58244f 100644 --- a/llm_normalizer/backend/src/services/assistantAddressOrchestrationRuntimeAdapter.ts +++ b/llm_normalizer/backend/src/services/assistantAddressOrchestrationRuntimeAdapter.ts @@ -77,6 +77,24 @@ function hasSelectedObjectInventoryActionCue(text: string | null): boolean { ); } +function hasShortInventoryPurchaseFollowupCue(text: string | null): boolean { + return /(?:^|[\s,.;:!?])(а\s+)?(?:купили\s+у\s+кого|у\s+кого\s+купили|поставщик|продавец|seller)(?:[\s,.;:!?]|$)/iu.test( + String(text ?? "") + ); +} + +function isInventorySelectedObjectOrRootIntent(intent: string | null): boolean { + return ( + intent === "inventory_on_hand_as_of_date" || + intent === "inventory_purchase_provenance_for_item" || + intent === "inventory_purchase_documents_for_item" || + intent === "inventory_sale_trace_for_item" || + intent === "inventory_profitability_for_item" || + intent === "inventory_purchase_to_sale_chain" || + intent === "inventory_aging_by_purchase_date" + ); +} + function isGenericCanonicalDriftIntent(intent: string | null): boolean { return ( intent === "open_items_by_counterparty_or_contract" || @@ -89,6 +107,33 @@ function isGenericCanonicalDriftIntent(intent: string | null): boolean { ); } +function hasSameDateFollowupSignal(text: string | null): boolean { + return /(?:эту\s+же\s+дат(?:у|е|ой)|ту\s+же\s+дат(?:у|е|ой)|same\s+date)/iu.test(String(text ?? "")); +} + +function hasExplicitCurrentDateSignal(text: string | null): boolean { + return /(?:текущ(?:ую|ая|ий|ее|ей)\s+дат(?:у|а|е|ой)|сегодняшн(?:юю|ий|ей)\s+дат(?:у|а|е|ой)|today|current\s+date)/iu.test( + String(text ?? "") + ); +} + +function hasInventoryTemporalRootFollowupCue(text: string | null): boolean { + const value = String(text ?? "").trim().toLowerCase(); + if (!value) { + return false; + } + const tokenCount = value.split(/\s+/).filter(Boolean).length; + const hasMonthYearCue = + /(?:январ(?:ь|е)|феврал(?:ь|е)|март(?:е)?|апрел(?:ь|е)|ма(?:й|е)|июн(?:ь|е)|июл(?:ь|е)|август(?:е)?|сентябр(?:ь|е)|октябр(?:ь|е)|ноябр(?:ь|е)|декабр(?:ь|е))(?:\s+\d{4})?/iu.test( + value + ) || /\b(?:19|20)\d{2}\b/u.test(value); + if (tokenCount <= 3 && hasMonthYearCue) { + return true; + } + const hasInventoryLexeme = /(?:остат|склад|товар|позици|номенклатур)/iu.test(value); + return hasInventoryLexeme && (hasMonthYearCue || hasSameDateFollowupSignal(value)); +} + function shouldPreferRawFollowupMessage( userMessage: string, addressInputMessage: string, @@ -112,15 +157,39 @@ function shouldPreferRawFollowupMessage( : null; const mode = toNonEmptyString(predecomposeContract?.mode) ?? "unknown"; const intent = toNonEmptyString(predecomposeContract?.intent) ?? "unknown"; + const followupContext = + carryover.followupContext && typeof carryover.followupContext === "object" + ? (carryover.followupContext as Record) + : null; + const previousIntent = toNonEmptyString(followupContext?.previous_intent); + const rootIntent = toNonEmptyString(followupContext?.root_intent); + const previousAnchorType = toNonEmptyString(followupContext?.previous_anchor_type); + const hasInventoryItemCarryover = + previousAnchorType === "item" && isInventorySelectedObjectOrRootIntent(previousIntent); + const hasInventoryFrameCarryover = + isInventorySelectedObjectOrRootIntent(previousIntent) || + isInventorySelectedObjectOrRootIntent(rootIntent); if (mode === "unsupported" && intent === "unknown") { return true; } + if (hasSameDateFollowupSignal(rawMessage) && hasExplicitCurrentDateSignal(canonicalMessage)) { + return true; + } + + if ( + hasInventoryFrameCarryover && + hasInventoryTemporalRootFollowupCue(rawMessage) && + (intent === "account_balance_snapshot" || intent === "documents_forming_balance" || intent === "unknown") + ) { + return true; + } + return ( - hasSelectedObjectInventorySignal(rawMessage) && - hasSelectedObjectInventoryActionCue(rawMessage) && - isGenericCanonicalDriftIntent(intent) + (hasSelectedObjectInventorySignal(rawMessage) || hasInventoryItemCarryover) && + (hasSelectedObjectInventoryActionCue(rawMessage) || hasShortInventoryPurchaseFollowupCue(rawMessage)) && + (isGenericCanonicalDriftIntent(intent) || intent === "unknown") ); } diff --git a/llm_normalizer/backend/src/services/assistantRoutePolicy.ts b/llm_normalizer/backend/src/services/assistantRoutePolicy.ts index cdca213..07afd48 100644 --- a/llm_normalizer/backend/src/services/assistantRoutePolicy.ts +++ b/llm_normalizer/backend/src/services/assistantRoutePolicy.ts @@ -193,8 +193,16 @@ export function createAssistantRoutePolicy(deps) { hasShortDebtMirrorFollowupSignal(repairedRawUserMessage) || hasShortDebtMirrorFollowupSignal(effectiveAddressUserMessage) || hasShortDebtMirrorFollowupSignal(repairedEffectiveAddressUserMessage); + const followupPreviousIntent = toNonEmptyString(followupContext?.previous_intent); + const followupPreviousFilters = followupContext?.previous_filters && typeof followupContext.previous_filters === "object" + ? followupContext.previous_filters + : null; const protectedInventoryShortFollowup = Boolean(followupContext && - isInventorySelectedObjectIntent(toNonEmptyString(followupContext.previous_intent)) && + (isInventorySelectedObjectIntent(followupPreviousIntent) || + (followupPreviousIntent === "inventory_on_hand_as_of_date" && + (toNonEmptyString(followupContext.previous_anchor_type) === "item" || + toNonEmptyString(followupContext.previous_anchor_value) || + toNonEmptyString(followupPreviousFilters?.item)))) && (hasShortInventoryObjectFollowupSignal(rawUserMessage) || hasShortInventoryObjectFollowupSignal(repairedRawUserMessage) || hasShortInventoryObjectFollowupSignal(effectiveAddressUserMessage) || diff --git a/llm_normalizer/backend/src/services/assistantService.ts b/llm_normalizer/backend/src/services/assistantService.ts index 43c2057..1d749d7 100644 --- a/llm_normalizer/backend/src/services/assistantService.ts +++ b/llm_normalizer/backend/src/services/assistantService.ts @@ -2757,8 +2757,11 @@ function hasShortInventoryObjectFollowupSignal(userMessage) { if (minTokens > 8) { return false; } + const hasDirectPurchaseFollowupCue = (sample) => /(?:^|[\s,.;:!?])(?:а\s+)?(?:у|от)\s+кого(?:\s+\S+){0,5}\s+купил(?:и|о)?(?=$|[\s,.;:!?])/iu.test(sample) || + /(?:^|[\s,.;:!?])(?:а\s+)?купил(?:и|о)?(?:\s+\S+){0,5}\s+(?:у|от)\s+кого(?=$|[\s,.;:!?])/iu.test(sample); const hasDirectSaleFollowupCue = (sample) => /(?:кому|каму|куда)(?:\s+\S+){0,4}\s+(?:продали|продано|продан(?:о|а|ы)?|реализовали|реализован(?:о|а|ы)?)|(?:продали|продано|реализовали|реализован(?:о|а|ы)?)(?:\s+\S+){0,4}\s+(?:кому|каму|куда)|(?:^|\s)(?:продано|продали|реализовано|реализовали)(?=$|[\s,.;:!?])/iu.test(sample); - return samples.some((sample) => /^(?:кто|когда|документы|сумма|поставщик|покупатель)(?:\?)?$/iu.test(sample) || + return samples.some((sample) => /^(?:кто|когда|документы|сумма|поставщик|покупатель|продавец|seller)(?:\?)?$/iu.test(sample) || + hasDirectPurchaseFollowupCue(sample) || hasDirectSaleFollowupCue(sample) || (0, decomposeStage_1.hasInventorySupplierFollowupCue)(sample) || (0, decomposeStage_1.hasInventoryPurchaseDocumentsFollowupCue)(sample) || @@ -2767,6 +2770,41 @@ function hasShortInventoryObjectFollowupSignal(userMessage) { (0, decomposeStage_1.hasInventorySaleFollowupCue)(sample) || (0, decomposeStage_1.hasInventoryPurchaseToSaleChainFollowupCue)(sample)); } +function hasInventoryRootTemporalFollowupSignal(userMessage, sourceIntentHint, hasInventoryRootFrame) { + if (!hasInventoryRootFrame) { + return false; + } + if (!(sourceIntentHint === "inventory_on_hand_as_of_date" || + sourceIntentHint === "inventory_supplier_stock_overlap_as_of_date" || + isInventorySelectedObjectIntent(sourceIntentHint))) { + return false; + } + const rawText = compactWhitespace(String(userMessage ?? "").toLowerCase()); + const repairedText = compactWhitespace(repairAddressMojibake(String(userMessage ?? "")).toLowerCase()); + const samples = [rawText, repairedText].filter((item) => item.length > 0); + if (samples.length === 0) { + return false; + } + const minTokens = samples.reduce((min, sample) => Math.min(min, countTokens(sample)), Number.POSITIVE_INFINITY); + if (minTokens > 8) { + return false; + } + const hasMonthYearCue = samples.some((sample) => /(?:январ|феврал|март|апрел|ма(?:й|е|я)|июн|июл|август|сентябр|октябр|ноябр|декабр)(?:\s+\d{4})?/iu.test(sample) || + /\b(?:19|20)\d{2}\b/u.test(sample)); + const hasTemporalCue = samples.some((sample) => hasPeriodLiteral(sample) || + hasShortNamedPeriodFollowupLiteral(sample) || + hasShortCurrentDateFollowupLiteral(sample) || + /(?:РЅР°\s+ту\s+Р¶Рµ\s+дат[ауеы]|РЅР°\s+эту\s+Р¶Рµ\s+дат[ауеы]|same\s+date|the\s+same\s+date)/iu.test(sample)); + if (!hasTemporalCue && !hasMonthYearCue) { + return false; + } + if (samples.some((sample) => hasForeignAccountingPivotOverInventoryMessage(sample))) { + return false; + } + const hasInventoryLexeme = samples.some((sample) => /(?:остат|склад|товар|номенклат|РїРѕР·РёС†)/iu.test(sample)); + const hasPlainInventoryLexeme = samples.some((sample) => /(?:остат|склад|товар|номенклатур|позиц)/iu.test(sample)); + return hasInventoryLexeme || hasPlainInventoryLexeme || minTokens <= 3; +} function hasForeignAccountingPivotOverInventoryMessage(userMessage, alternateMessage = null) { const samples = [ compactWhitespace(repairAddressMojibake(String(userMessage ?? "")).toLowerCase()), @@ -2848,8 +2886,22 @@ function resolveAddressFollowupCarryoverContext(userMessage, items, alternateMes (isImplicitAddressContinuationByLlm(userMessage, llmPreDecomposeMeta) || (toNonEmptyString(alternateMessage) ? isImplicitAddressContinuationByLlm(alternateMessage, llmPreDecomposeMeta) : false)); const sourceIntentHint = toNonEmptyString(previousAddressDebug?.detected_intent); - const inventoryShortFollowupPrimary = isInventorySelectedObjectIntent(sourceIntentHint) && hasShortInventoryObjectFollowupSignal(userMessage); - const inventoryShortFollowupAlternate = isInventorySelectedObjectIntent(sourceIntentHint) && toNonEmptyString(alternateMessage) + const navigationFocusObjectHint = addressNavigationState && + typeof addressNavigationState === "object" && + addressNavigationState.session_context && + typeof addressNavigationState.session_context === "object" && + addressNavigationState.session_context.active_focus_object && + typeof addressNavigationState.session_context.active_focus_object === "object" + ? addressNavigationState.session_context.active_focus_object + : null; + const hasNavigationInventoryItemFocusHint = Boolean(toNonEmptyString(navigationFocusObjectHint?.label) && + toNonEmptyString(navigationFocusObjectHint?.object_type) === "item" && + (sourceIntentHint === "inventory_on_hand_as_of_date" || + sourceIntentHint === "inventory_supplier_stock_overlap_as_of_date" || + isInventorySelectedObjectIntent(sourceIntentHint))); + let inventoryShortFollowupPrimary = (isInventorySelectedObjectIntent(sourceIntentHint) || hasNavigationInventoryItemFocusHint) && + hasShortInventoryObjectFollowupSignal(userMessage); + let inventoryShortFollowupAlternate = (isInventorySelectedObjectIntent(sourceIntentHint) || hasNavigationInventoryItemFocusHint) && toNonEmptyString(alternateMessage) ? hasShortInventoryObjectFollowupSignal(String(alternateMessage ?? "")) : false; const debtRoleSwapPrimary = sourceIntentHint ? resolveDebtRoleSwapFollowupIntent(userMessage, sourceIntentHint) : null; @@ -2857,8 +2909,8 @@ function resolveAddressFollowupCarryoverContext(userMessage, items, alternateMes ? resolveDebtRoleSwapFollowupIntent(String(alternateMessage ?? ""), sourceIntentHint) : null; const debtRoleSwapIntent = debtRoleSwapPrimary ?? debtRoleSwapAlternate ?? null; - const hasPrimaryFollowupSignal = hasAddressFollowupContextSignal(userMessage) || Boolean(debtRoleSwapPrimary) || inventoryShortFollowupPrimary; - const hasAlternateFollowupSignal = toNonEmptyString(alternateMessage) + let hasPrimaryFollowupSignal = hasAddressFollowupContextSignal(userMessage) || Boolean(debtRoleSwapPrimary) || inventoryShortFollowupPrimary; + let hasAlternateFollowupSignal = toNonEmptyString(alternateMessage) ? hasAddressFollowupContextSignal(alternateMessage) || Boolean(debtRoleSwapAlternate) || inventoryShortFollowupAlternate : false; const hasPrimaryIndexReferenceSignal = extractDisplayedEntityIndexMention(userMessage) !== null; @@ -2866,12 +2918,19 @@ function resolveAddressFollowupCarryoverContext(userMessage, items, alternateMes ? extractDisplayedEntityIndexMention(String(alternateMessage ?? "")) !== null : false; const hasIndexReferenceSignal = hasPrimaryIndexReferenceSignal || hasAlternateIndexReferenceSignal; - const hasStrongFollowupReference = hasPrimaryIndexReferenceSignal || + const recentInventoryRootFrame = findRecentInventoryRootFrame(items); + const hasInventoryRootTemporalFollowupPrimary = hasInventoryRootTemporalFollowupSignal(userMessage, sourceIntentHint, Boolean(recentInventoryRootFrame)); + const hasInventoryRootTemporalFollowupAlternate = toNonEmptyString(alternateMessage) + ? hasInventoryRootTemporalFollowupSignal(String(alternateMessage ?? ""), sourceIntentHint, Boolean(recentInventoryRootFrame)) + : false; + let hasStrongFollowupReference = hasPrimaryIndexReferenceSignal || hasAlternateIndexReferenceSignal || hasOrganizationClarificationContinuation || hasImplicitContinuationSignal || inventoryShortFollowupPrimary || inventoryShortFollowupAlternate || + hasInventoryRootTemporalFollowupPrimary || + hasInventoryRootTemporalFollowupAlternate || Boolean(debtRoleSwapIntent) || hasFollowupMarker(userMessage) || hasReferentialPointer(userMessage) || @@ -2883,6 +2942,8 @@ function resolveAddressFollowupCarryoverContext(userMessage, items, alternateMes if (hasStandaloneAddressTopic && !hasPrimaryFollowupSignal && !hasAlternateFollowupSignal && + !hasInventoryRootTemporalFollowupPrimary && + !hasInventoryRootTemporalFollowupAlternate && !hasImplicitContinuationSignal && !hasOrganizationClarificationContinuation && !hasIndexReferenceSignal) { @@ -2890,6 +2951,8 @@ function resolveAddressFollowupCarryoverContext(userMessage, items, alternateMes } if (!hasPrimaryFollowupSignal && !hasAlternateFollowupSignal && + !hasInventoryRootTemporalFollowupPrimary && + !hasInventoryRootTemporalFollowupAlternate && !hasImplicitContinuationSignal && !hasOrganizationClarificationContinuation && !hasIndexReferenceSignal) { @@ -2951,6 +3014,41 @@ function resolveAddressFollowupCarryoverContext(userMessage, items, alternateMes : null; const navigationFocusObjectType = toNonEmptyString(navigationFocusObject?.object_type); const navigationFocusObjectLabel = toNonEmptyString(navigationFocusObject?.label); + const hasInventoryItemFocusCarryover = navigationFocusObjectType === "item" && + navigationFocusObjectLabel && + (sourceIntentHint === "inventory_on_hand_as_of_date" || + sourceIntentHint === "inventory_supplier_stock_overlap_as_of_date" || + isInventorySelectedObjectIntent(sourceIntentHint)); + if (!inventoryShortFollowupPrimary && hasInventoryItemFocusCarryover) { + inventoryShortFollowupPrimary = hasShortInventoryObjectFollowupSignal(userMessage); + } + if (!inventoryShortFollowupAlternate && hasInventoryItemFocusCarryover && toNonEmptyString(alternateMessage)) { + inventoryShortFollowupAlternate = hasShortInventoryObjectFollowupSignal(String(alternateMessage ?? "")); + } + hasPrimaryFollowupSignal = hasAddressFollowupContextSignal(userMessage) || + Boolean(debtRoleSwapPrimary) || + inventoryShortFollowupPrimary || + hasInventoryRootTemporalFollowupPrimary; + hasAlternateFollowupSignal = toNonEmptyString(alternateMessage) + ? hasAddressFollowupContextSignal(alternateMessage) || + Boolean(debtRoleSwapAlternate) || + inventoryShortFollowupAlternate || + hasInventoryRootTemporalFollowupAlternate + : false; + hasStrongFollowupReference = hasPrimaryIndexReferenceSignal || + hasAlternateIndexReferenceSignal || + hasOrganizationClarificationContinuation || + hasImplicitContinuationSignal || + inventoryShortFollowupPrimary || + inventoryShortFollowupAlternate || + hasInventoryRootTemporalFollowupPrimary || + hasInventoryRootTemporalFollowupAlternate || + Boolean(debtRoleSwapIntent) || + hasFollowupMarker(userMessage) || + hasReferentialPointer(userMessage) || + (toNonEmptyString(alternateMessage) + ? hasFollowupMarker(String(alternateMessage ?? "")) || hasReferentialPointer(String(alternateMessage ?? "")) + : false); const hasSelectedObjectInventorySignalPrimary = /(?:по\s+выбранному\s+объекту|по\s+этой\s+позиции|по\s+этому\s+товару|selected\s+object)/iu.test(String(userMessage ?? "")); const hasSelectedObjectInventorySignalAlternate = toNonEmptyString(alternateMessage) ? /(?:по\s+выбранному\s+объекту|по\s+этой\s+позиции|по\s+этому\s+товару|selected\s+object)/iu.test(String(alternateMessage ?? "")) @@ -3012,18 +3110,39 @@ function resolveAddressFollowupCarryoverContext(userMessage, items, alternateMes if (!toNonEmptyString(previousFilters.organization) && organizationClarificationSelection) { previousFilters.organization = organizationClarificationSelection; } - if (!toNonEmptyString(previousFilters.as_of_date) && toNonEmptyString(navigationDateScope?.as_of_date)) { + const shouldBackfillPreviousDateScopeFromNavigation = sourceIntentHint === "inventory_on_hand_as_of_date" || + sourceIntentHint === "inventory_supplier_stock_overlap_as_of_date" || + sourceIntentHint === "inventory_purchase_provenance_for_item" || + sourceIntentHint === "inventory_purchase_documents_for_item" || + sourceIntentHint === "inventory_sale_trace_for_item" || + sourceIntentHint === "inventory_profitability_for_item" || + sourceIntentHint === "inventory_purchase_to_sale_chain" || + sourceIntentHint === "inventory_aging_by_purchase_date" || + sourceIntentHint === "account_balance_snapshot" || + sourceIntentHint === "documents_forming_balance"; + if (shouldBackfillPreviousDateScopeFromNavigation && + !toNonEmptyString(previousFilters.as_of_date) && + toNonEmptyString(navigationDateScope?.as_of_date)) { previousFilters.as_of_date = toNonEmptyString(navigationDateScope?.as_of_date); } - if (!toNonEmptyString(previousFilters.period_from) && toNonEmptyString(navigationDateScope?.period_from)) { + if (shouldBackfillPreviousDateScopeFromNavigation && + !toNonEmptyString(previousFilters.period_from) && + toNonEmptyString(navigationDateScope?.period_from)) { previousFilters.period_from = toNonEmptyString(navigationDateScope?.period_from); } - if (!toNonEmptyString(previousFilters.period_to) && toNonEmptyString(navigationDateScope?.period_to)) { + if (shouldBackfillPreviousDateScopeFromNavigation && + !toNonEmptyString(previousFilters.period_to) && + toNonEmptyString(navigationDateScope?.period_to)) { previousFilters.period_to = toNonEmptyString(navigationDateScope?.period_to); } const rootContextOnlyPivot = Boolean((isInventorySelectedObjectIntent(sourceIntentHint) || currentFrameKind === "inventory_drilldown") && hasForeignAccountingPivotOverInventoryMessage(userMessage, alternateMessage)); - if (rootContextOnlyPivot) { + const inventoryRootTemporalPivot = Boolean(inventoryRootFrame && + (isInventorySelectedObjectIntent(sourceIntentHint) || currentFrameKind === "inventory_drilldown") && + (hasInventoryRootTemporalFollowupPrimary || hasInventoryRootTemporalFollowupAlternate) && + !hasForeignAccountingPivotOverInventoryMessage(userMessage, alternateMessage)); + const rootScopedPivot = rootContextOnlyPivot || inventoryRootTemporalPivot; + if (rootScopedPivot) { previousIntent = null; previousAnchorType = null; previousAnchor = null; @@ -3037,7 +3156,7 @@ function resolveAddressFollowupCarryoverContext(userMessage, items, alternateMes (toNonEmptyString(alternateMessage) ? resolveDisplayedAddressEntityMention(String(alternateMessage ?? ""), displayedEntities) : null); - if (resolvedEntityFromFollowup && !rootContextOnlyPivot) { + if (resolvedEntityFromFollowup && !rootScopedPivot) { if (resolvedEntityFromFollowup.entityType === "counterparty") { previousFilters.counterparty = resolvedEntityFromFollowup.value; previousAnchorType = "counterparty"; @@ -3058,7 +3177,7 @@ function resolveAddressFollowupCarryoverContext(userMessage, items, alternateMes followupSelectionMode = "carry_referenced_entity"; } } - if (!rootContextOnlyPivot && + if (!rootScopedPivot && !toNonEmptyString(previousFilters.item) && navigationFocusObjectType === "item" && navigationFocusObjectLabel && @@ -3100,7 +3219,7 @@ function resolveAddressFollowupCarryoverContext(userMessage, items, alternateMes previous_anchor_type: previousAnchorType ?? undefined, previous_anchor_value: previousAnchor, resolved_counterparty_from_display: resolvedCounterpartyFromDisplay || undefined, - root_context_only: rootContextOnlyPivot || undefined, + root_context_only: rootScopedPivot || undefined, root_intent: inventoryRootFrame?.intent ?? undefined, root_filters: inventoryRootFrame?.filters ?? undefined, root_anchor_type: inventoryRootFrame?.anchorType ?? undefined, @@ -3121,11 +3240,13 @@ function buildAddressDialogContinuationContractV2(userMessage, effectiveMessage, const previousIntent = toNonEmptyString(carryoverMeta?.previousSourceIntent) ?? null; const selectionMode = toNonEmptyString(carryoverMeta?.followupSelectionMode) ?? null; const rootContextOnly = selectionMode === "carry_root_context"; - const explicitIntent = toNonEmptyString(llmPreDecomposeMeta?.predecomposeContract?.intent); + const explicitIntentRaw = toNonEmptyString(llmPreDecomposeMeta?.predecomposeContract?.intent); + const explicitIntent = explicitIntentRaw === "unknown" ? null : explicitIntentRaw; + const rootIntent = toNonEmptyString(carryoverMeta?.followupContext?.root_intent) ?? null; const targetIntent = selectionMode === "switch_to_suggested_intent" ? toNonEmptyString(carryoverMeta?.previousAddressIntent) ?? null : rootContextOnly - ? explicitIntent ?? null + ? rootIntent ?? explicitIntent ?? null : explicitIntent ?? toNonEmptyString(carryoverMeta?.previousAddressIntent) ?? null; const hasImplicitContinuationSignal = Boolean(carryoverMeta?.hasImplicitContinuationSignal); const rewrittenByPredecompose = compactWhitespace(sourceMessage.toLowerCase()) !== compactWhitespace(canonicalMessage.toLowerCase()); diff --git a/llm_normalizer/backend/src/services/inventoryLifecycleCueHelpers.ts b/llm_normalizer/backend/src/services/inventoryLifecycleCueHelpers.ts index 5dd2881..c202c7d 100644 --- a/llm_normalizer/backend/src/services/inventoryLifecycleCueHelpers.ts +++ b/llm_normalizer/backend/src/services/inventoryLifecycleCueHelpers.ts @@ -11,6 +11,9 @@ export function hasInventorySupplierCue(text: string): boolean { if (/(?:купил(?:и|о)?\s+у\s+кого|куплен(?:о)?\s+у\s+кого|купил(?:и|о)?\s+от\s+кого|куплен(?:о)?\s+от\s+кого)/iu.test(value)) { return true; } + if (/(?:кто\s+(?:был\s+)?продавец|кто\s+нам\s+продал|кто\s+продал\s+нам|(?:^|[\s,.;:!?()\-])(?:продавец|seller)(?=$|[\s,.;:!?()\-]))/iu.test(value)) { + return true; + } if ( /(?:кто\s+(?:(?:это|этот\s+товар|эту\s+позицию)\s+)?(?:нам\s+)?поставил|кто\s+(?:нам\s+)?поставил\s+(?:это|этот\s+товар|эту\s+позицию)|от\s+какого\s+поставщика|у\s+какого\s+поставщика|от\s+кого\s+куплен|у\s+кого\s+купили|у\s+кого\s+куплено|где\s+(?:мы\s+)?купили(?:\s+(?:это|его|этот\s+товар|эту\s+позицию))?|где\s+(?:мы\s+)?взяли(?:\s+(?:это|его|этот\s+товар|эту\s+позицию))?|откуда\s+(?:мы\s+)?взяли(?:\s+(?:это|его|этот\s+товар|эту\s+позицию))?|где\s+куплено|supplier|vendor|поставщик)/iu.test( value diff --git a/llm_normalizer/backend/tests/addressCounterpartyLifecycleOrganizationScopeRegression.test.ts b/llm_normalizer/backend/tests/addressCounterpartyLifecycleOrganizationScopeRegression.test.ts new file mode 100644 index 0000000..6d172cf --- /dev/null +++ b/llm_normalizer/backend/tests/addressCounterpartyLifecycleOrganizationScopeRegression.test.ts @@ -0,0 +1,82 @@ +import { afterEach, describe, expect, it, vi } from "vitest"; + +const { executeAddressMcpQueryMock } = vi.hoisted(() => ({ + executeAddressMcpQueryMock: vi.fn() +})); + +vi.mock("../src/services/addressMcpClient", async () => { + const actual = await vi.importActual( + "../src/services/addressMcpClient" + ); + return { + ...actual, + executeAddressMcpQuery: executeAddressMcpQueryMock + }; +}); + +import { AddressQueryService } from "../src/services/addressQueryService"; + +afterEach(() => { + executeAddressMcpQueryMock.mockReset(); + vi.restoreAllMocks(); +}); + +describe("counterparty lifecycle organization scope regressions", () => { + it("keeps carried organization scope without false empty-match when rows omit organization column", async () => { + executeAddressMcpQueryMock.mockResolvedValueOnce({ + fetched_rows: 3, + matched_rows: 3, + raw_rows: [ + { + Period: "2020-01-15T00:00:00Z", + Registrator: "CP_CUSTOMER_ACTIVITY", + AccountDt: "62.01", + AccountKt: "90.01", + Amount: 12, + Counterparty: 'ООО "Ромашка"' + }, + { + Period: "2020-02-20T00:00:00Z", + Registrator: "CP_CUSTOMER_ACTIVITY", + AccountDt: "62.01", + AccountKt: "90.01", + Amount: 8, + Counterparty: 'ООО "Ландыш"' + }, + { + Period: "2020-03-10T00:00:00Z", + Registrator: "CP_CUSTOMER_ACTIVITY", + AccountDt: "62.01", + AccountKt: "90.01", + Amount: 4, + Counterparty: 'ООО "Ромашка"' + } + ], + rows: [], + error: null + }); + + const service = new AddressQueryService(); + const result = await service.tryHandle("какие клиенты заплатили больше всего денег за все время", { + followupContext: { + previous_intent: "vat_payable_forecast", + previous_filters: { + organization: 'ООО "Альтернатива Плюс"', + period_from: "2020-01-01", + period_to: "2020-03-31" + }, + previous_anchor_type: "organization", + previous_anchor_value: 'ООО "Альтернатива Плюс"' + } + }); + + expect(result?.handled).toBe(true); + expect(result?.reply_type).toBe("factual"); + expect(result?.response_type).toBe("FACTUAL_LIST"); + expect(result?.debug.detected_intent).toBe("counterparty_activity_lifecycle"); + expect(result?.debug.extracted_filters?.organization).toBe('ООО "Альтернатива Плюс"'); + expect(result?.debug.match_failure_reason).toBeNull(); + expect(result?.debug.rows_matched).toBe(3); + expect(String(result?.reply_text ?? "")).toContain('ООО "Ромашка"'); + }); +}); diff --git a/llm_normalizer/backend/tests/addressFollowupTemporalRegression.test.ts b/llm_normalizer/backend/tests/addressFollowupTemporalRegression.test.ts new file mode 100644 index 0000000..8df3f7d --- /dev/null +++ b/llm_normalizer/backend/tests/addressFollowupTemporalRegression.test.ts @@ -0,0 +1,63 @@ +import { describe, expect, it } from "vitest"; +import { runAddressDecomposeStage } from "../src/services/address_runtime/decomposeStage"; + +describe("address follow-up temporal regressions", () => { + it("replaces VAT month-only follow-up with the previous follow-up year", () => { + const result = runAddressDecomposeStage("а на март", { + previous_intent: "vat_payable_confirmed_as_of_date", + previous_filters: { + period_from: "2019-05-01", + period_to: "2019-05-31", + as_of_date: "2019-05-31" + }, + previous_anchor_type: "unknown", + previous_anchor_value: null + }); + + expect(result).not.toBeNull(); + expect(result?.intent.intent).toBe("vat_payable_confirmed_as_of_date"); + expect(result?.filters.extracted_filters.period_from).toBe("2019-03-01"); + expect(result?.filters.extracted_filters.period_to).toBe("2019-03-31"); + expect(result?.filters.extracted_filters.as_of_date).toBe("2019-03-31"); + expect(result?.baseReasons).toContain("period_derived_from_followup_context_year"); + }); + + it("replaces VAT month-only follow-up with a later month from the same inherited year", () => { + const result = runAddressDecomposeStage("на сентябрь", { + previous_intent: "vat_payable_confirmed_as_of_date", + previous_filters: { + period_from: "2019-05-01", + period_to: "2019-05-31", + as_of_date: "2019-05-31" + }, + previous_anchor_type: "unknown", + previous_anchor_value: null + }); + + expect(result).not.toBeNull(); + expect(result?.intent.intent).toBe("vat_payable_confirmed_as_of_date"); + expect(result?.filters.extracted_filters.period_from).toBe("2019-09-01"); + expect(result?.filters.extracted_filters.period_to).toBe("2019-09-30"); + expect(result?.filters.extracted_filters.as_of_date).toBe("2019-09-30"); + expect(result?.baseReasons).toContain("period_derived_from_followup_context_year"); + }); + + it("keeps same-date inventory pivot anchored to the previous VAT date", () => { + const result = runAddressDecomposeStage("какие остатки по складу на эту же дату", { + previous_intent: "vat_payable_confirmed_as_of_date", + previous_filters: { + period_from: "2019-09-01", + period_to: "2019-09-30", + as_of_date: "2019-09-30" + }, + previous_anchor_type: "unknown", + previous_anchor_value: null + }); + + expect(result).not.toBeNull(); + expect(result?.intent.intent).toBe("inventory_on_hand_as_of_date"); + expect(result?.filters.extracted_filters.as_of_date).toBe("2019-09-30"); + expect(result?.filters.extracted_filters.warehouse).toBeUndefined(); + expect(result?.baseReasons).toContain("as_of_date_from_followup_context"); + }); +}); diff --git a/llm_normalizer/backend/tests/addressInventoryRootFrameRegression.test.ts b/llm_normalizer/backend/tests/addressInventoryRootFrameRegression.test.ts index a111084..d78e472 100644 --- a/llm_normalizer/backend/tests/addressInventoryRootFrameRegression.test.ts +++ b/llm_normalizer/backend/tests/addressInventoryRootFrameRegression.test.ts @@ -56,7 +56,6 @@ describe("inventory root frame regressions", () => { expect(result).not.toBeNull(); expect(result?.intent.intent).toBe("inventory_purchase_provenance_for_item"); expect(result?.filters.extracted_filters.item).toBe("Зеркало для инвалидов поворотное травмобезопасное"); - expect(result?.filters.extracted_filters.as_of_date).toBe("2022-02-28"); expect( result?.baseReasons?.includes("intent_adjusted_to_inventory_followup_context") || result?.intent.reasons.includes("inventory_selected_object_provenance_signal_detected") @@ -111,4 +110,67 @@ describe("inventory root frame regressions", () => { expect(result?.intent.intent).not.toBe("bank_operations_by_counterparty"); expect(result?.filters.extracted_filters.item).toBe("Четки Пост (84*117)"); }); + it("restores stock snapshot intent for selected-object restatement on the previously reviewed date", () => { + const result = runAddressDecomposeStage( + 'По выбранному объекту "Столешница 600*3050*26 альмандин": покажи еще раз остатки на дату которую до этого рассматривали', + { + previous_intent: "inventory_purchase_documents_for_item", + previous_filters: { + item: "Столешница 600*3050*26 альмандин", + organization: 'ООО "Альтернатива Плюс"', + as_of_date: "2019-05-31" + }, + previous_anchor_type: "item", + previous_anchor_value: "Столешница 600*3050*26 альмандин", + root_intent: "inventory_on_hand_as_of_date", + root_filters: { + organization: 'ООО "Альтернатива Плюс"', + period_from: "2019-05-01", + period_to: "2019-05-31", + as_of_date: "2019-05-31" + }, + root_anchor_type: "organization", + root_anchor_value: 'ООО "Альтернатива Плюс"', + current_frame_kind: "inventory_drilldown" + } + ); + + expect(result).not.toBeNull(); + expect(result?.intent.intent).toBe("inventory_on_hand_as_of_date"); + expect(result?.intent.reasons).toContain("intent_restored_to_inventory_root_frame"); + expect(result?.filters.extracted_filters.item).toBe("Столешница 600*3050*26 альмандин"); + expect(result?.filters.extracted_filters.organization).toBe('ООО "Альтернатива Плюс"'); + expect(result?.filters.extracted_filters.as_of_date).toBe("2019-05-31"); + expect(result?.baseReasons).toContain("address_followup_context_applied"); + }); + + it("restores inventory root frame for root-context-only month follow-up after drilldown", () => { + const result = runAddressDecomposeStage("остатки на июль 2019", { + previous_filters: { + organization: 'ООО "Альтернатива Плюс"', + as_of_date: "2020-03-31", + period_from: "2020-03-01", + period_to: "2020-03-31" + }, + previous_anchor_type: null, + previous_anchor_value: null, + root_intent: "inventory_on_hand_as_of_date", + root_filters: { + organization: 'ООО "Альтернатива Плюс"', + as_of_date: "2020-03-31", + period_from: "2020-03-01", + period_to: "2020-03-31" + }, + root_context_only: true, + current_frame_kind: "inventory_root" + }); + + expect(result).not.toBeNull(); + expect(result?.intent.intent).toBe("inventory_on_hand_as_of_date"); + expect(result?.intent.reasons).toContain("intent_restored_to_inventory_root_frame"); + expect(result?.filters.extracted_filters.organization).toBe('ООО "Альтернатива Плюс"'); + expect(result?.filters.extracted_filters.period_from).toBe("2019-07-01"); + expect(result?.filters.extracted_filters.period_to).toBe("2019-07-31"); + expect(result?.filters.extracted_filters.as_of_date).toBe("2019-07-31"); + }); }); diff --git a/llm_normalizer/backend/tests/addressInventorySaleTraceDocumentRoute.test.ts b/llm_normalizer/backend/tests/addressInventorySaleTraceDocumentRoute.test.ts index bce5a1d..60a5a26 100644 --- a/llm_normalizer/backend/tests/addressInventorySaleTraceDocumentRoute.test.ts +++ b/llm_normalizer/backend/tests/addressInventorySaleTraceDocumentRoute.test.ts @@ -21,8 +21,8 @@ afterEach(() => { vi.restoreAllMocks(); }); -describe("inventory sale trace document route", () => { - it("uses document sales route with native item resolution for selected-object buyer follow-up", async () => { +describe("inventory sale trace movement route", () => { + it("uses 41.01 movement route with native item resolution for selected-object buyer follow-up", async () => { executeAddressMcpQueryMock.mockResolvedValueOnce({ fetched_rows: 2, matched_rows: 2, @@ -30,25 +30,27 @@ describe("inventory sale trace document route", () => { { Period: "2015-02-25T12:00:00Z", Registrator: "Реализация товаров и услуг 00000000012 от 25.02.2015 12:00:00", - AccountDt: "", + AccountDt: "62.01", AccountKt: "41.01", Amount: 12605435.66, Quantity: 40, - Item: "Рабочая станция универсального специалиста (индивидуальное изготовление)", - Counterparty: "Комитет государственных услуг г. Москвы", - Contract: "Гос.контракт № 42/15 от 20.02.2015г. Силино окна", + SubcontoKt1: "Рабочая станция универсального специалиста (индивидуальное изготовление)", + SubcontoKt3: "Основной склад", + SubcontoDt1: "Комитет государственных услуг г. Москвы", + SubcontoDt2: "Гос.контракт № 42/15 от 20.02.2015 г. Силино окна", Organization: "ООО \\Альтернатива Плюс\\" }, { Period: "2015-02-09T12:00:14Z", Registrator: "Реализация товаров и услуг 00000000004 от 09.02.2015 12:00:14", - AccountDt: "", + AccountDt: "62.01", AccountKt: "41.01", Amount: 16421320.17, Quantity: 51, - Item: "Рабочая станция универсального специалиста (индивидуальное изготовление)", - Counterparty: "Комитет государственных услуг г. Москвы", - Contract: "Гос.контракт № 17/15 от 02.02.2015г.", + SubcontoKt1: "Рабочая станция универсального специалиста (индивидуальное изготовление)", + SubcontoKt3: "Основной склад", + SubcontoDt1: "Комитет государственных услуг г. Москвы", + SubcontoDt2: "Гос.контракт № 17/15 от 02.02.2015 г.", Organization: "ООО \\Альтернатива Плюс\\" } ], @@ -82,11 +84,12 @@ describe("inventory sale trace document route", () => { expect(executeAddressMcpQueryMock).toHaveBeenCalledTimes(1); const query = String(executeAddressMcpQueryMock.mock.calls[0]?.[0]?.query ?? ""); - expect(query).toContain("Документ.РеализацияТоваровУслуг.Товары КАК Товары"); - expect(query).toContain("Товары.Номенклатура В (ВЫБРАТЬ Номенклатура.Ссылка"); - expect(query).toContain('Номенклатура.Наименование = "Рабочая станция универсального специалиста (индивидуальное изготовление)"'); - expect(query).toContain("ПРЕДСТАВЛЕНИЕ(Товары.Ссылка.Контрагент) КАК Контрагент"); - expect(query).toContain("ПРЕДСТАВЛЕНИЕ(Товары.Ссылка.Организация) КАК Организация"); + expect(query).toContain("РегистрБухгалтерии.Хозрасчетный.ДвиженияССубконто КАК Движения"); + expect(query).toContain('ПОДСТРОКА(ЕСТЬNULL(Движения.СчетКт.Код, ""), 1, 5) = "41.01"'); + expect(query).toContain("Движения.СубконтоКт1 В (ВЫБРАТЬ Номенклатура.Ссылка"); + expect(query).toContain( + 'Номенклатура.Наименование = "Рабочая станция универсального специалиста (индивидуальное изготовление)"' + ); expect(query).not.toContain("2016-06-30"); expect(query).not.toContain("2016-06-01"); expect(query).not.toContain('ПРЕДСТАВЛЕНИЕ(Движения.Организация) = "ООО \\Альтернатива Плюс\\"'); diff --git a/llm_normalizer/backend/tests/addressInventorySaleTraceSelectedObjectRegression.test.ts b/llm_normalizer/backend/tests/addressInventorySaleTraceSelectedObjectRegression.test.ts index 9810e42..0f79e88 100644 --- a/llm_normalizer/backend/tests/addressInventorySaleTraceSelectedObjectRegression.test.ts +++ b/llm_normalizer/backend/tests/addressInventorySaleTraceSelectedObjectRegression.test.ts @@ -25,13 +25,14 @@ describe("inventory sale trace selected-object regressions", () => { const saleRow = { Period: "2021-04-15T00:00:00Z", Registrator: "Реализация товаров и услуг 00000000201 от 15.04.2021 0:00:00", - AccountDt: "", + AccountDt: "62.01", AccountKt: "41.01", Amount: 165.83, Quantity: 1, - Item: "Кромка с клеем 33 дуб ниагара 137 м", - Counterparty: "ООО \\Покупатель\\", - Contract: "Договор реализации № 17 от 14.04.2021", + SubcontoKt1: "Кромка с клеем 33 дуб ниагара 137 м", + SubcontoKt3: "Основной склад", + SubcontoDt1: "ООО \\Покупатель\\", + SubcontoDt2: "Договор реализации № 17 от 14.04.2021", Organization: "ООО \\Альтернатива Плюс\\" }; @@ -70,7 +71,7 @@ describe("inventory sale trace selected-object regressions", () => { expect(executeAddressMcpQueryMock).toHaveBeenCalledTimes(1); const query = String(executeAddressMcpQueryMock.mock.calls[0]?.[0]?.query ?? ""); - expect(query).toContain("Документ.РеализацияТоваровУслуг.Товары КАК Товары"); + expect(query).toContain("РегистрБухгалтерии.Хозрасчетный.ДвиженияССубконто КАК Движения"); expect(query).toContain('Номенклатура.Наименование = "Кромка с клеем 33 дуб ниагара 137 м"'); expect(query).not.toContain('Номенклатура.Наименование = "Кромка"'); }); @@ -132,6 +133,7 @@ describe("inventory sale trace selected-object regressions", () => { expect(executeAddressMcpQueryMock).toHaveBeenCalledTimes(1); const query = String(executeAddressMcpQueryMock.mock.calls[0]?.[0]?.query ?? ""); + expect(query).toContain("РегистрБухгалтерии.Хозрасчетный.ДвиженияССубконто КАК Движения"); expect(query).toContain('Номенклатура.Наименование = "Кромка с клеем 33 дуб ниагара 137 м"'); expect(query).not.toContain('Номенклатура.Наименование = "Кромка"'); }); diff --git a/llm_normalizer/backend/tests/addressInventorySelectedObjectFollowup.test.ts b/llm_normalizer/backend/tests/addressInventorySelectedObjectFollowup.test.ts index 74b4337..6e7dda1 100644 --- a/llm_normalizer/backend/tests/addressInventorySelectedObjectFollowup.test.ts +++ b/llm_normalizer/backend/tests/addressInventorySelectedObjectFollowup.test.ts @@ -193,7 +193,7 @@ describe("inventory selected-object follow-up", () => { expect(result?.response_type).toBe("FACTUAL_SUMMARY"); expect(result?.debug.detected_intent).toBe("inventory_purchase_provenance_for_item"); expect(result?.debug.extracted_filters?.item).toBe("Столешница 600*3050*26 дуб ниагара"); - expect(result?.debug.extracted_filters?.as_of_date).toBe("2019-03-31"); + expect(result?.debug.extracted_filters?.as_of_date).toBeUndefined(); expect(String(result?.reply_text ?? "")).toContain("Торговый дом \\Союз МСК\\"); }); @@ -242,7 +242,7 @@ describe("inventory selected-object follow-up", () => { expect(result?.response_type).toBe("FACTUAL_SUMMARY"); expect(result?.debug.detected_intent).toBe("inventory_purchase_provenance_for_item"); expect(result?.debug.extracted_filters?.item).toBe("Рабочая станция универсального специалиста (индивидуальное изготовление)"); - expect(result?.debug.extracted_filters?.as_of_date).toBe("2016-07-31"); + expect(result?.debug.extracted_filters?.as_of_date).toBeUndefined(); expect(String(result?.reply_text ?? "")).toContain("ООО \\Производство мебели\\"); }); @@ -292,7 +292,7 @@ describe("inventory selected-object follow-up", () => { expect(result?.debug.detected_intent).toBe("inventory_purchase_provenance_for_item"); expect(result?.debug.selected_recipe).toBe("address_inventory_purchase_provenance_for_item_v1"); expect(result?.debug.extracted_filters?.item).toBe("Конструкция трансформер рабочей станции 1300*900*2000"); - expect(result?.debug.extracted_filters?.as_of_date).toBe("2020-06-30"); + expect(result?.debug.extracted_filters?.as_of_date).toBeUndefined(); expect(result?.debug.extracted_filters?.period_from).toBeUndefined(); expect(result?.debug.extracted_filters?.period_to).toBeUndefined(); expect(result?.debug.capability_id).toBe("inventory_inventory_purchase_provenance_for_item"); @@ -345,7 +345,7 @@ describe("inventory selected-object follow-up", () => { expect(result?.response_type).toBe("FACTUAL_SUMMARY"); expect(result?.debug.detected_intent).toBe("inventory_purchase_provenance_for_item"); expect(result?.debug.extracted_filters?.item).toBe("Рабочая станция универсального специалиста (индивидуальное изготовление)"); - expect(result?.debug.extracted_filters?.as_of_date).toBe("2016-05-31"); + expect(result?.debug.extracted_filters?.as_of_date).toBeUndefined(); expect(String(result?.reply_text ?? "")).toContain("ООО \\Производство мебели\\"); }); @@ -394,7 +394,7 @@ describe("inventory selected-object follow-up", () => { expect(result?.response_type).toBe("FACTUAL_SUMMARY"); expect(result?.debug.detected_intent).toBe("inventory_purchase_provenance_for_item"); expect(result?.debug.extracted_filters?.item).toBe("Рабочая станция универсального специалиста (индивидуальное изготовление)"); - expect(result?.debug.extracted_filters?.as_of_date).toBe("2016-05-31"); + expect(result?.debug.extracted_filters?.as_of_date).toBeUndefined(); expect(String(result?.reply_text ?? "")).toContain("ООО \\Производство мебели\\"); }); @@ -444,7 +444,7 @@ describe("inventory selected-object follow-up", () => { expect(result?.debug.detected_intent).toBe("inventory_purchase_provenance_for_item"); expect(result?.debug.selected_recipe).toBe("address_inventory_purchase_provenance_for_item_v1"); expect(result?.debug.extracted_filters?.item).toBe("Рабочая станция универсального специалиста (индивидуальное изготовление)"); - expect(result?.debug.extracted_filters?.as_of_date).toBe("2016-05-31"); + expect(result?.debug.extracted_filters?.as_of_date).toBeUndefined(); expect(String(result?.reply_text ?? "")).toContain("ООО \\Производство мебели\\"); }); @@ -491,7 +491,7 @@ describe("inventory selected-object follow-up", () => { expect(result?.debug.detected_intent).toBe("inventory_purchase_documents_for_item"); expect(result?.debug.selected_recipe).toBe("address_inventory_purchase_documents_for_item_v1"); expect(result?.debug.extracted_filters?.item).toBe("Столешница 600*3050*26 дуб ниагара"); - expect(result?.debug.extracted_filters?.as_of_date).toBe("2019-03-31"); + expect(result?.debug.extracted_filters?.as_of_date).toBeUndefined(); expect(String(result?.reply_text ?? "")).toContain("Поступление товаров и услуг 00000000077"); }); @@ -537,7 +537,7 @@ describe("inventory selected-object follow-up", () => { expect(result?.debug.detected_intent).toBe("inventory_sale_trace_for_item"); expect(result?.debug.selected_recipe).toBe("address_inventory_sale_trace_for_item_v1"); expect(result?.debug.extracted_filters?.item).toBe("Четки Пост (84*117)"); - expect(result?.debug.extracted_filters?.as_of_date).toBe("2020-03-31"); + expect(result?.debug.extracted_filters?.as_of_date).toBeUndefined(); expect(String(result?.reply_text ?? "").split("\n")[0]).toContain("ИП Покупатель"); expect(String(result?.reply_text ?? "")).toContain("Документы выбытия"); }); @@ -583,7 +583,7 @@ describe("inventory selected-object follow-up", () => { expect(result?.debug.detected_intent).toBe("inventory_sale_trace_for_item"); expect(result?.debug.selected_recipe).toBe("address_inventory_sale_trace_for_item_v1"); expect(result?.debug.extracted_filters?.item).toBe("Пуф арий"); - expect(result?.debug.extracted_filters?.as_of_date).toBe("2020-05-31"); + expect(result?.debug.extracted_filters?.as_of_date).toBeUndefined(); expect(result?.debug.reasons).toContain("inventory_selected_object_sale_trace_signal_detected"); expect(String(result?.reply_text ?? "").split("\n")[0]).toContain("ООО \\Ромашка\\"); expect(String(result?.reply_text ?? "")).toContain("Документы выбытия"); @@ -631,7 +631,7 @@ describe("inventory selected-object follow-up", () => { expect(result?.debug.detected_intent).toBe("inventory_sale_trace_for_item"); expect(result?.debug.selected_recipe).toBe("address_inventory_sale_trace_for_item_v1"); expect(result?.debug.extracted_filters?.item).toBe("Кромка с клеем 33 альмандин 137 м"); - expect(result?.debug.extracted_filters?.as_of_date).toBe("2020-03-31"); + expect(result?.debug.extracted_filters?.as_of_date).toBeUndefined(); expect(String(result?.reply_text ?? "")).toContain("ООО \\Покупатель\\"); }); @@ -678,10 +678,7 @@ describe("inventory selected-object follow-up", () => { expect(result?.handled).toBe(true); expect(result?.response_type).toBe("FACTUAL_LIST"); expect(result?.debug.detected_intent).toBe("inventory_sale_trace_for_item"); - expect(result?.debug.reasons).toContain("lifecycle_execution_detached_from_snapshot_date"); - expect(result?.debug.reasons).toContain("as_of_date_cleared_for_history_recovery"); - expect(result?.debug.limitations).toContain("lifecycle_execution_detached_from_snapshot_date"); - expect(result?.debug.limitations).toContain("as_of_date_cleared_for_history_recovery"); + expect(result?.debug.extracted_filters?.as_of_date).toBeUndefined(); expect(executeAddressMcpQueryMock).toHaveBeenCalledTimes(1); const query = String(executeAddressMcpQueryMock.mock.calls[0]?.[0]?.query ?? ""); expect(query).not.toContain("2019-03-31"); @@ -808,7 +805,7 @@ describe("inventory selected-object follow-up", () => { expect(result?.response_type).toBe("FACTUAL_SUMMARY"); expect(result?.debug.detected_intent).toBe("inventory_purchase_provenance_for_item"); expect(result?.debug.extracted_filters?.item).toBe("Кресло орион"); - expect(result?.debug.extracted_filters?.as_of_date).toBe("2020-03-31"); + expect(result?.debug.extracted_filters?.as_of_date).toBeUndefined(); expect(result?.debug.extracted_filters?.period_from).toBeUndefined(); expect(result?.debug.extracted_filters?.period_to).toBeUndefined(); expect(result?.debug.reasons ?? []).not.toContain("lifecycle_execution_detached_from_snapshot_date"); @@ -820,7 +817,7 @@ describe("inventory selected-object follow-up", () => { const query = String(executeAddressMcpQueryMock.mock.calls[0]?.[0]?.query ?? ""); expect(query).toContain("Документ.ПоступлениеТоваровУслуг.Товары КАК Товары"); expect(query).toContain("Товары.Номенклатура В (ВЫБРАТЬ Номенклатура.Ссылка"); - expect(query).toContain("Товары.Ссылка.Дата <= ДАТАВРЕМЯ(2020, 3, 31, 23, 59, 59)"); + expect(query).not.toContain("2020-03-31"); expect(query).toContain("ПРЕДСТАВЛЕНИЕ(Товары.Ссылка.Организация) КАК Организация"); }); }); diff --git a/llm_normalizer/backend/tests/addressInventoryWarehouseAnchor.test.ts b/llm_normalizer/backend/tests/addressInventoryWarehouseAnchor.test.ts index f14d49b..e39711e 100644 --- a/llm_normalizer/backend/tests/addressInventoryWarehouseAnchor.test.ts +++ b/llm_normalizer/backend/tests/addressInventoryWarehouseAnchor.test.ts @@ -62,6 +62,29 @@ describe("inventory warehouse anchor extraction", () => { "inventory_on_hand_as_of_date" ).extracted_filters; + expect(filters.warehouse).toBeUndefined(); + }); + it("does not materialize self-scope slang tail as warehouse anchor in stock snapshot wording from the run", () => { + const result = extractAddressFilters("что там на складе у нас висит", "inventory_on_hand_as_of_date"); + + expect(result.extracted_filters.warehouse).toBeUndefined(); + expect(result.warnings).toContain("warehouse_self_scope_detected"); + expect(result.semantic_frame?.scope_kind).toBe("implicit_self_scope"); + expect(result.semantic_frame?.anchor_kind).toBe("self_scope"); + expect(result.semantic_frame?.anchor_value).toBeNull(); + }); + it("does not materialize same-date phrasing as warehouse anchor in stock follow-up", () => { + const filters = extractAddressFilters("какие остатки по складу на эту же дату", "inventory_on_hand_as_of_date").extracted_filters; + + expect(filters.warehouse).toBeUndefined(); + }); + + it("does not materialize current-date phrasing as warehouse anchor in stock follow-up", () => { + const filters = extractAddressFilters( + "получить остатки по складу на текущую дату", + "inventory_on_hand_as_of_date" + ).extracted_filters; + expect(filters.warehouse).toBeUndefined(); }); }); diff --git a/llm_normalizer/backend/tests/assistantAddressFollowupContext.test.ts b/llm_normalizer/backend/tests/assistantAddressFollowupContext.test.ts index a911ee5..a3d73d0 100644 --- a/llm_normalizer/backend/tests/assistantAddressFollowupContext.test.ts +++ b/llm_normalizer/backend/tests/assistantAddressFollowupContext.test.ts @@ -2376,5 +2376,228 @@ describe("assistant address follow-up carryover", () => { expect(normalizerService.normalize).not.toHaveBeenCalled(); }); + it("treats short inventory month follow-up after drilldown as continuation of the recent inventory root frame", async () => { + const calls: Array<{ message: string; options?: any }> = []; + const followupMessage = "остатки на июль 2019"; + const itemLabel = "Четки Пост (84*117)"; + const organization = 'ООО "Альтернатива Плюс"'; + + const inventoryResult = { + handled: true, + reply_text: "Подтвержденный складской срез на 31.07.2019 собран.", + reply_type: "factual", + response_type: "FACTUAL_SUMMARY", + debug: { + detected_mode: "address_query", + detected_intent: "inventory_on_hand_as_of_date", + detected_intent_confidence: "high", + extracted_filters: { + as_of_date: "2019-07-31", + period_from: "2019-07-01", + period_to: "2019-07-31", + organization + }, + selected_recipe: "address_inventory_on_hand_as_of_date_v1", + reasons: ["address_action_detected", "address_entity_detected", "address_followup_context_applied"] + } + } as any; + + const addressQueryService = { + tryHandle: vi.fn(async (message: string, options?: any) => { + calls.push({ message, options }); + if (message === followupMessage && options?.followupContext) { + return inventoryResult; + } + return null; + }) + } as any; + + const normalizerService = { + normalize: vi.fn(async () => ({ + assistant_reply: "normalizer_fallback_should_not_be_used", + reply_type: "partial_coverage", + debug: {} + })) + } as any; + + const sessions = new AssistantSessionStore(); + const service = new AssistantService( + normalizerService, + sessions as any, + {} as any, + { persistSession: vi.fn() } as any, + addressQueryService + ); + + const sessionId = `asst-address-followup-inventory-root-month-${Date.now()}`; + sessions.appendItem(sessionId, { + message_id: "msg-inventory-root-seed-july", + session_id: sessionId, + role: "assistant", + text: "inventory root seed", + reply_type: "factual", + created_at: "2026-04-15T19:00:00.000Z", + trace_id: "address-root-seed-july", + debug: { + detected_mode: "address_query", + detected_intent: "inventory_on_hand_as_of_date", + extracted_filters: { + as_of_date: "2020-03-31", + period_from: "2020-03-01", + period_to: "2020-03-31", + organization + }, + selected_recipe: "address_inventory_on_hand_as_of_date_v1" + } + } as any); + sessions.appendItem(sessionId, { + message_id: "msg-inventory-drilldown-seed-july", + session_id: sessionId, + role: "assistant", + text: "inventory provenance seed", + reply_type: "factual", + created_at: "2026-04-15T19:01:00.000Z", + trace_id: "address-drilldown-seed-july", + debug: { + detected_mode: "address_query", + detected_intent: "inventory_purchase_provenance_for_item", + extracted_filters: { + item: itemLabel, + organization + }, + selected_recipe: "address_inventory_purchase_provenance_for_item_v1", + anchor_type: "item", + anchor_value_raw: itemLabel, + anchor_value_resolved: itemLabel + } + } as any); + + const second = await service.handleMessage({ + session_id: sessionId, + user_message: followupMessage, + useMock: true + } as any); + + expect(second.ok).toBe(true); + expect(second.reply_type).toBe("factual"); + expect(calls).toHaveLength(1); + expect(calls[0].message).toBe(followupMessage); + expect(calls[0].options?.followupContext?.previous_intent).toBeUndefined(); + expect(calls[0].options?.followupContext?.previous_filters?.item).toBeUndefined(); + expect(calls[0].options?.followupContext?.root_context_only).toBe(true); + expect(calls[0].options?.followupContext?.root_intent).toBe("inventory_on_hand_as_of_date"); + expect(calls[0].options?.followupContext?.root_filters?.organization).toBe(organization); + expect(calls[0].options?.followupContext?.root_filters?.as_of_date).toBe("2020-03-31"); + expect(calls[0].options?.followupContext?.current_frame_kind).toBe("inventory_root"); + expect(normalizerService.normalize).not.toHaveBeenCalled(); + }); + + it("treats colloquial supplier follow-up from an inventory root slice as continuation of the focused item", async () => { + const calls: Array<{ message: string; options?: any }> = []; + const followupMessage = "у кого мы модуль прямоугольный купили кстати"; + const itemLabel = "Модуль прямоугольый 1400*110*750"; + const organization = 'ООО "Альтернатива Плюс"'; + + const provenanceResult = { + handled: true, + reply_text: `По позиции ${itemLabel} подтвержден поставщик: Торговый дом "Союз".`, + reply_type: "factual", + response_type: "FACTUAL_SUMMARY", + debug: { + detected_mode: "address_query", + detected_intent: "inventory_purchase_provenance_for_item", + detected_intent_confidence: "medium", + extracted_filters: { + item: itemLabel, + organization, + as_of_date: "2020-05-31" + }, + selected_recipe: "address_inventory_purchase_provenance_for_item_v1", + reasons: ["address_action_detected", "address_entity_detected", "address_followup_context_applied"] + } + } as any; + + const addressQueryService = { + tryHandle: vi.fn(async (message: string, options?: any) => { + calls.push({ message, options }); + if (message === followupMessage && options?.followupContext) { + return provenanceResult; + } + return null; + }) + } as any; + + const normalizerService = { + normalize: vi.fn(async () => ({ + assistant_reply: "normalizer_fallback_should_not_be_used", + reply_type: "partial_coverage", + debug: {} + })) + } as any; + + const sessions = new AssistantSessionStore(); + const service = new AssistantService( + normalizerService, + sessions as any, + {} as any, + { persistSession: vi.fn() } as any, + addressQueryService + ); + + const sessionId = `asst-address-followup-root-item-focus-${Date.now()}`; + sessions.appendItem(sessionId, { + message_id: "msg-inventory-root-seed", + session_id: sessionId, + role: "assistant", + text: `${itemLabel} был в остатках на 31.05.2020.`, + reply_type: "factual", + created_at: "2026-04-15T19:00:00.000Z", + trace_id: "address-root-seed", + debug: { + detected_mode: "address_query", + detected_intent: "inventory_on_hand_as_of_date", + extracted_filters: { + organization, + as_of_date: "2020-05-31", + period_from: "2020-05-01", + period_to: "2020-05-31" + }, + selected_recipe: "address_inventory_on_hand_as_of_date_v1" + } + } as any); + sessions.setAddressNavigationState(sessionId, { + session_id: sessionId, + session_context: { + active_focus_object: { + object_type: "item", + label: itemLabel + }, + organization_scope: organization, + date_scope: { + as_of_date: "2020-05-31", + period_from: "2020-05-01", + period_to: "2020-05-31" + } + } + } as any); + + const response = await service.handleMessage({ + session_id: sessionId, + user_message: followupMessage, + useMock: true + } as any); + + expect(response.ok).toBe(true); + expect(response.reply_type).toBe("factual"); + expect(calls).toHaveLength(1); + expect(calls[0].message).toBe(followupMessage); + expect(calls[0].options?.followupContext?.previous_intent).toBe("inventory_on_hand_as_of_date"); + expect(calls[0].options?.followupContext?.previous_anchor_type).toBe("item"); + expect(calls[0].options?.followupContext?.previous_anchor_value).toBe(itemLabel); + expect(calls[0].options?.followupContext?.previous_filters?.item).toBe(itemLabel); + expect(calls[0].options?.followupContext?.previous_filters?.organization).toBe(organization); + expect(calls[0].options?.followupContext?.previous_filters?.as_of_date).toBe("2020-05-31"); + expect(normalizerService.normalize).not.toHaveBeenCalled(); + }); }); diff --git a/llm_normalizer/backend/tests/assistantAddressLaneRuntimeAdapter.test.ts b/llm_normalizer/backend/tests/assistantAddressLaneRuntimeAdapter.test.ts index ded2e4b..341de93 100644 --- a/llm_normalizer/backend/tests/assistantAddressLaneRuntimeAdapter.test.ts +++ b/llm_normalizer/backend/tests/assistantAddressLaneRuntimeAdapter.test.ts @@ -34,8 +34,8 @@ describe("assistant address lane runtime adapter", () => { const runAddressLaneAttempt = vi.fn(async () => factualLane()); const result = await runAssistantAddressLaneRuntime({ - userMessage: "сырой вопрос", - addressInputMessage: "нормализованный вопрос", + userMessage: "raw question", + addressInputMessage: "normalized question", carryover, shouldPreferContextualLane: true, canRetryWithRawUserMessage: true, @@ -44,24 +44,24 @@ describe("assistant address lane runtime adapter", () => { }); expect(result.handled).toBe(true); - expect(result.selection?.messageUsed).toBe("нормализованный вопрос"); + expect(result.selection?.messageUsed).toBe("normalized question"); expect(result.selection?.carryMeta).toBe(carryover); expect(result.retryAudit.attempted).toBe(false); expect(runAddressLaneAttempt).toHaveBeenCalledTimes(1); - expect(runAddressLaneAttempt).toHaveBeenCalledWith("нормализованный вопрос", carryover); + expect(runAddressLaneAttempt).toHaveBeenCalledWith("normalized question", carryover, null); }); it("retries with raw message after limited result and returns factual retry", async () => { const carryover: AssistantAddressFollowupCarryoverLike = { followupContext: { scope: "ctx" } }; const runAddressLaneAttempt = vi .fn() - .mockResolvedValueOnce(limitedLane("empty_match")) // primary - .mockResolvedValueOnce(limitedLane("empty_match")) // contextual - .mockResolvedValueOnce(factualLane()); // raw contextual retry + .mockResolvedValueOnce(limitedLane("empty_match")) + .mockResolvedValueOnce(limitedLane("empty_match")) + .mockResolvedValueOnce(factualLane()); const result = await runAssistantAddressLaneRuntime({ - userMessage: "сырой вопрос", - addressInputMessage: "нормализованный вопрос", + userMessage: "raw question", + addressInputMessage: "normalized question", carryover, shouldPreferContextualLane: false, canRetryWithRawUserMessage: true, @@ -70,7 +70,7 @@ describe("assistant address lane runtime adapter", () => { }); expect(result.handled).toBe(true); - expect(result.selection?.messageUsed).toBe("сырой вопрос"); + expect(result.selection?.messageUsed).toBe("raw question"); expect(result.selection?.carryMeta).toBe(carryover); expect(result.retryAudit.attempted).toBe(true); expect(result.retryAudit.reason).toBe("limited_result_retry_with_raw_message"); @@ -83,12 +83,12 @@ describe("assistant address lane runtime adapter", () => { it("returns pending limited result when retry is disabled", async () => { const runAddressLaneAttempt = vi .fn() - .mockResolvedValueOnce(limitedLane("missing_anchor")) // primary - .mockResolvedValueOnce(unhandledLane()); // contextual fallback + .mockResolvedValueOnce(limitedLane("missing_anchor")) + .mockResolvedValueOnce(unhandledLane()); const result = await runAssistantAddressLaneRuntime({ - userMessage: "сырой вопрос", - addressInputMessage: "нормализованный вопрос", + userMessage: "raw question", + addressInputMessage: "normalized question", carryover: { followupContext: { scope: "ctx" } }, shouldPreferContextualLane: false, canRetryWithRawUserMessage: false, @@ -97,9 +97,8 @@ describe("assistant address lane runtime adapter", () => { }); expect(result.handled).toBe(true); - expect(result.selection?.messageUsed).toBe("нормализованный вопрос"); + expect(result.selection?.messageUsed).toBe("normalized question"); expect(result.selection?.addressLane.debug?.limited_reason_category).toBe("missing_anchor"); expect(result.retryAudit.attempted).toBe(false); }); }); - diff --git a/llm_normalizer/backend/tests/assistantAddressOrchestrationRuntimeAdapter.test.ts b/llm_normalizer/backend/tests/assistantAddressOrchestrationRuntimeAdapter.test.ts index e6e342b..160d0ea 100644 --- a/llm_normalizer/backend/tests/assistantAddressOrchestrationRuntimeAdapter.test.ts +++ b/llm_normalizer/backend/tests/assistantAddressOrchestrationRuntimeAdapter.test.ts @@ -178,6 +178,80 @@ describe("assistant address orchestration runtime adapter", () => { ); }); + it("prefers raw inventory temporal root follow-up over account-balance canonical drift", async () => { + const resolveAddressFollowupCarryoverContext = vi.fn(() => ({ + followupContext: { + previous_intent: "inventory_purchase_provenance_for_item", + previous_filters: { + item: "Четки Пост (84*117)", + organization: "ООО \\Альтернатива Плюс\\" + }, + previous_anchor_type: "item", + previous_anchor_value: "Четки Пост (84*117)", + root_intent: "inventory_on_hand_as_of_date", + root_filters: { + as_of_date: "2020-03-31", + period_from: "2020-03-01", + period_to: "2020-03-31", + organization: "ООО \\Альтернатива Плюс\\" + }, + current_frame_kind: "inventory_drilldown" + } + })); + const resolveAssistantOrchestrationDecision = vi.fn(() => ({ + runAddressLane: true, + livingMode: "address_data", + livingReason: "address_lane_triggered", + toolGateDecision: "run_address_lane", + toolGateReason: "followup_context_detected", + orchestrationContract: { schema_version: "assistant_orchestration_contract_v1" } + })); + const buildAddressLlmPredecomposeContractV1 = vi.fn( + ({ sourceMessage, canonicalMessage }: { sourceMessage: string; canonicalMessage: string }) => ({ + schema_version: "address_llm_predecompose_contract_v1", + source_message: sourceMessage, + canonical_message: canonicalMessage, + mode: "address_query", + intent: canonicalMessage === sourceMessage ? "inventory_on_hand_as_of_date" : "account_balance_snapshot" + }) + ); + const rawMessage = "остатки на июль 2019"; + + const output = await buildAssistantAddressOrchestrationRuntime( + buildInput({ + userMessage: rawMessage, + runAddressLlmPreDecompose: vi.fn(async () => ({ + attempted: true, + applied: true, + effectiveMessage: "проверить остатки по счетам на июль 2019 года", + reason: "normalized_fragment_applied", + predecomposeContract: { + mode: "address_query", + intent: "account_balance_snapshot" + } + })), + buildAddressLlmPredecomposeContractV1, + resolveAddressFollowupCarryoverContext, + resolveAssistantOrchestrationDecision + }) + ); + + expect(output.addressInputMessage).toBe(rawMessage); + expect(output.addressPreDecompose.applied).toBe(false); + expect(output.addressPreDecompose.reason).toBe("followup_raw_message_preferred_over_llm_rewrite"); + expect(buildAddressLlmPredecomposeContractV1).toHaveBeenCalledWith({ + sourceMessage: rawMessage, + canonicalMessage: rawMessage + }); + expect(resolveAddressFollowupCarryoverContext).toHaveBeenCalledTimes(2); + expect(resolveAssistantOrchestrationDecision).toHaveBeenCalledWith( + expect.objectContaining({ + rawUserMessage: rawMessage, + effectiveAddressUserMessage: rawMessage + }) + ); + }); + it("prefers raw selected-object inventory action over generic canonical drift intent", async () => { const resolveAddressFollowupCarryoverContext = vi.fn(() => ({ followupContext: { @@ -441,4 +515,56 @@ describe("assistant address orchestration runtime adapter", () => { }) ); }); + + it("prefers raw same-date follow-up over canonical rewrite to current date", async () => { + const resolveAddressFollowupCarryoverContext = vi.fn(() => ({ + followupContext: { + previous_intent: "vat_payable_confirmed_as_of_date", + previous_filters: { + as_of_date: "2019-09-30", + period_from: "2019-09-01", + period_to: "2019-09-30" + } + } + })); + const resolveAssistantOrchestrationDecision = vi.fn(() => ({ + runAddressLane: true, + livingMode: "address_data", + livingReason: "address_lane_triggered", + toolGateDecision: "run_address_lane", + toolGateReason: "followup_context_detected", + orchestrationContract: { schema_version: "assistant_orchestration_contract_v1" } + })); + + const rawMessage = "какие остатки по складу на эту же дату"; + + const output = await buildAssistantAddressOrchestrationRuntime( + buildInput({ + userMessage: rawMessage, + runAddressLlmPreDecompose: vi.fn(async () => ({ + attempted: true, + applied: true, + effectiveMessage: "получить остатки по складу на текущую дату", + reason: "normalized_fragment_applied", + predecomposeContract: { + mode: "address_query", + intent: "inventory_on_hand_as_of_date" + } + })), + resolveAddressFollowupCarryoverContext, + resolveAssistantOrchestrationDecision + }) + ); + + expect(output.addressInputMessage).toBe(rawMessage); + expect(output.addressPreDecompose.applied).toBe(false); + expect(output.addressPreDecompose.reason).toBe("followup_raw_message_preferred_over_llm_rewrite"); + expect(resolveAddressFollowupCarryoverContext).toHaveBeenCalledTimes(2); + expect(resolveAssistantOrchestrationDecision).toHaveBeenCalledWith( + expect.objectContaining({ + rawUserMessage: rawMessage, + effectiveAddressUserMessage: rawMessage + }) + ); + }); }); diff --git a/llm_normalizer/frontend/dist/assets/index-Dcuz1nX5.css b/llm_normalizer/frontend/dist/assets/index-CaUiKcE3.css similarity index 53% rename from llm_normalizer/frontend/dist/assets/index-Dcuz1nX5.css rename to llm_normalizer/frontend/dist/assets/index-CaUiKcE3.css index 01e4d50..e517791 100644 --- a/llm_normalizer/frontend/dist/assets/index-Dcuz1nX5.css +++ b/llm_normalizer/frontend/dist/assets/index-CaUiKcE3.css @@ -1 +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-assistant-chip: 18, 18, 18;--rgb-assistant-chip-hover: 44, 44, 44;--rgb-assistant-chip-selected: 228, 142, 92;--rgb-assistant-chip-selected-text: 18, 18, 18;--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;--mode-column-width: 440px;--mode-toggle-width: 188px;--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: var(--mode-column-width)}*{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,.layout-grid.layout-grid-mode-columns{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;max-width:72%;overflow-x:auto;overflow-y:hidden;padding-bottom:2px}.mode-switch-row .tab{white-space:nowrap}.mode-switch-row.mode-switch-row-right .tab{width:var(--mode-toggle-width);min-width:var(--mode-toggle-width);text-align:center}.mode-columns{display:flex;gap:12px;width:100%;min-height:0;flex:1 1 auto;overflow-x:auto;overflow-y:hidden;padding-bottom:4px}.mode-col{flex:0 0 var(--mode-column-width);width:var(--mode-column-width);min-height:0;height:100%;display:flex}.mode-col.mode-col-wide,.mode-col.mode-col-xwide{flex-basis:var(--mode-column-width);width:var(--mode-column-width)}.mode-col .panel-frame{width:100%;height:100%}.mode-col .panel-body{min-height:0;overflow:auto}.mode-columns-empty{min-width:360px;border-radius:14px;background:rgb(var(--rgb-surface-main));color:var(--text-muted);padding:14px}.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-toolbar{display:grid;gap:8px}.assistant-toolbar-actions{display:grid;grid-template-columns:repeat(3,minmax(0,1fr));align-items:center;gap:8px}.assistant-toolbar-meta{display:flex;align-items:center;justify-content:space-between;gap:8px}.assistant-toolbar-meta-right{margin-left:auto;display:flex;align-items:center;justify-content:flex-end;gap:8px;min-width:0;flex-wrap:wrap}.assistant-live-status{color:var(--text-muted);font-size:.8rem;white-space:nowrap}.assistant-toolbar-error{margin:0}.assistant-copy-btn{width:100%;justify-self:stretch;background:transparent;border-color:transparent;color:var(--text-main);font-size:.6rem;line-height:1.1;white-space:nowrap;text-align:center;letter-spacing:0;padding:6px 8px;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;overflow-y:auto;overflow-x:hidden;scrollbar-gutter:stable}.assistant-input-textarea,.autoruns-personality-prompt{overflow-y:auto;overflow-x:hidden;scrollbar-gutter:stable both-edges;border-bottom-right-radius:6px}.assistant-input-textarea::-webkit-scrollbar-corner,.autoruns-personality-prompt::-webkit-scrollbar-corner{background:rgb(var(--rgb-surface-horizontal))}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}.prompt-manager-grid{display:grid;grid-template-columns: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{flex:1 1 auto;min-height:0;overflow:auto;overscroll-behavior:contain;display:flex;flex-direction:column;gap:8px;padding:10px;border:none;border-radius:12px;background:rgb(var(--rgb-surface-horizontal))}.assistant-chat-list .assistant-msg:first-child{margin-top:auto}.assistant-empty{padding:18px;text-align:center}.assistant-msg{border:none;border-radius:12px;background:rgb(var(--rgb-surface-focus));padding:8px 10px;display:grid;gap:6px;min-width:0}.assistant-msg.user{margin-left:12%;border-color:transparent;background:rgb(var(--rgb-active));color:rgb(var(--rgb-active-text))}.assistant-msg.assistant{margin-right:12%;border-color:transparent}.assistant-msg.user .assistant-msg-head{color:rgba(var(--rgb-active-text),.9)}.assistant-msg.user .assistant-msg-body{color:rgb(var(--rgb-active-text))}.assistant-msg-head{display:flex;align-items:center;justify-content:space-between;gap:8px;margin-bottom:0;font-size:.74rem;color:var(--text-muted)}.assistant-msg-head-main{flex:1 1 auto;min-width:0;display:flex;align-items:center;justify-content:space-between;gap:8px}.assistant-msg-head-actions{display:inline-flex;align-items:center;justify-content:center;flex:0 0 auto}.assistant-comment-btn{cursor:pointer}.assistant-comment-btn:disabled{opacity:.42;cursor:not-allowed}.assistant-msg.user .assistant-comment-btn{color:rgba(var(--rgb-active-text),.92)}.assistant-msg-body{display:grid;gap:10px;line-height:1.35;font-size:.84rem;min-width:0}.assistant-msg-block{display:grid;gap:4px;min-width:0}.assistant-msg-block.selectable{cursor:pointer;padding:8px 10px;border-radius:12px;transition:background .18s ease,color .18s ease}.assistant-msg-block.selectable:hover,.assistant-msg-block.selectable:focus-visible{background:rgba(var(--rgb-active),.18);outline:none}.assistant-msg-block.selectable.active{background:rgb(var(--rgb-active));color:rgb(var(--rgb-active-text))}.assistant-msg-block.selectable.active:hover,.assistant-msg-block.selectable.active:focus-visible{background:rgb(var(--rgb-active))}.assistant-msg-block.selectable.active .assistant-msg-line,.assistant-msg-block.selectable.active .assistant-msg-line strong{color:rgb(var(--rgb-active-text))}.assistant-msg-line{margin:0;white-space:pre-wrap;word-break:break-word;overflow-wrap:anywhere}.assistant-msg-line.heading{font-weight:700;letter-spacing:.01em}.assistant-msg-line.numbered{margin-top:2px}.assistant-msg-line strong{font-weight:800}.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:0;display:grid;gap:8px;flex:0 0 auto}.assistant-compose-context{display:grid;gap:6px;padding:10px 12px;border-radius:var(--radius-md);background:rgb(var(--rgb-surface-horizontal))}.assistant-compose-context-label{color:var(--text-muted);font-size:.74rem;font-weight:700;letter-spacing:.01em}.assistant-compose-context-pill{display:inline-flex;align-items:center;gap:8px;max-width:100%;width:fit-content;min-height:38px;padding:0 10px 0 12px;border-radius:999px;background:rgb(var(--rgb-assistant-chip-selected));color:rgb(var(--rgb-assistant-chip-selected-text))}.assistant-compose-context-pill-text{min-width:0;max-width:min(100%,460px);overflow:hidden;text-overflow:ellipsis;white-space:nowrap;font-size:.82rem;font-weight:700}.assistant-compose-context-clear{display:inline-flex;align-items:center;justify-content:center;width:24px;height:24px;min-width:24px;padding:0;border-radius:999px;background:#0000002e;color:inherit;font-size:.96rem;line-height:1}.assistant-compose-context-clear:hover{background:#00000047}.assistant-send-row{align-items:center;margin-top:2px}.assistant-send-btn{margin-left:auto}.assistant-comments-frame .panel-body{display:flex;flex-direction:column;min-height:0;overflow:hidden}.assistant-comments-shell{display:grid;gap:8px;min-height:0;height:100%}.assistant-comments-toolbar{display:flex;align-items:center;justify-content:space-between;gap:8px}.assistant-comments-list{display:grid;gap:8px;overflow:auto;min-height:0;padding-right:2px}.assistant-comment-item{border:none;border-radius:10px;background:rgb(var(--rgb-surface-horizontal));padding:8px;display:grid;gap:6px}.assistant-comment-head{display:flex;align-items:center;justify-content:space-between;gap:8px;font-size:.75rem}.assistant-comment-item p{margin:0;white-space:pre-wrap;font-size:.8rem}.assistant-comment-meta{display:flex;flex-wrap:wrap;gap:8px;color:var(--text-muted);font-size:.74rem}.app-root-autoruns .assistant-panel-frame .panel-header{position:sticky;top:-12px;z-index:8;margin:-12px -12px 0;padding:12px 12px 10px;background:rgb(var(--rgb-surface-main))}.app-root-autoruns .assistant-panel-frame{overflow:visible}.app-root-autoruns .assistant-panel-frame .panel-body{flex:1 1 auto;padding:0 12px 12px;display:flex;flex-direction:column;min-height:0;overflow:hidden}.app-root-autoruns .assistant-panel-frame .assistant-live-shell{flex:1 1 auto;min-height:0;padding:12px;border-radius:14px;background:rgb(var(--rgb-background));display:flex;flex-direction:column;gap:10px}.app-root-autoruns .assistant-panel-frame .assistant-chat-list{overflow-y:auto}.app-root-autoruns .assistant-panel-frame .panel-header h2{margin:0;font-size:.95rem}.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-stack{display:grid;grid-template-columns:minmax(0,1fr);gap:12px}.runtime-details{display:grid;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(--mode-column-width);width:var(--mode-column-width);height:100%;min-height:0;overflow-y:auto;overflow-x:hidden;border:none;border-radius:14px;background:rgb(var(--rgb-surface-main));padding:12px;scrollbar-gutter:stable}.autoruns-assistant-live-col{background:rgb(var(--rgb-surface-main));padding:12px;overflow:hidden;scrollbar-gutter:auto}.autoruns-assistant-live-col .panel-frame{height:100%;background:rgb(var(--rgb-surface-main))}.autoruns-col h3{margin:0;font-size:.95rem}.autoruns-col h4{margin:12px 0 8px;font-size:.82rem;color:var(--text-muted)}.autoruns-col-header{position:sticky;top:-12px;z-index:8;margin:-12px -12px 10px;padding:12px 12px 10px;background:rgb(var(--rgb-surface-main))}.autoruns-col-header .tab-row{margin:8px 0 0}.autoruns-col-header .autoruns-dialog-toolbar{margin-top:8px}.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;transition:background-color .2s ease,box-shadow .2s ease}.autoruns-run-item.selected{background:rgb(var(--rgb-active));color:rgb(var(--rgb-active-text));box-shadow:none}.autoruns-run-item.selected .autoruns-run-meta{color:rgba(var(--rgb-active-text),.95)}.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-run-id-row{display:flex;align-items:center;justify-content:space-between;gap:8px;min-width:0}.autoruns-run-id-row>span{min-width:0;overflow-wrap:anywhere;word-break:break-word}.autoruns-copy-run-id-btn{border:none;background:transparent;color:rgb(var(--rgb-text-main));width:16px;height:16px;min-width:16px;min-height:16px;padding:0;display:inline-flex;align-items:center;justify-content:center;border-radius:4px;opacity:.92;cursor:pointer}.autoruns-copy-run-id-btn:hover{color:rgb(var(--rgb-text-main));opacity:1;background:transparent;box-shadow:none;transform:none}.autoruns-copy-run-id-btn:focus-visible{outline:1px solid rgba(var(--rgb-text-main),.7);outline-offset:1px}.autoruns-copy-icon-svg{width:.82rem;height:.82rem;fill:none;stroke:currentColor;stroke-width:1.75;stroke-linecap:round;stroke-linejoin:round}.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{background:rgb(var(--rgb-active));color:rgb(var(--rgb-active-text));box-shadow:none}.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;min-width:0;overflow:hidden}.autoruns-msg header,.autoruns-msg footer{display:flex;justify-content:space-between;align-items:flex-start;flex-wrap:wrap;min-width:0;gap:8px;font-size:.74rem;color:var(--text-muted)}.autoruns-msg-head-actions{display:flex;align-items:center;justify-content:flex-end;flex:1 1 auto;min-width:0;flex-wrap:wrap;gap:8px}.autoruns-msg-head-actions>span{min-width:0;overflow-wrap:anywhere;word-break:break-word}.autoruns-msg-case-tag{display:inline-flex;align-items:center;border-radius:999px;padding:2px 8px;font-size:.7rem;line-height:1;color:rgb(var(--rgb-active-text));background:rgba(var(--rgb-active),.24)}.autoruns-msg p{margin:0;white-space:pre-wrap;line-height:1.35;font-size:.84rem;overflow-wrap:anywhere;word-break:break-word}.autoruns-msg footer span{min-width:0;overflow-wrap:anywhere;word-break:break-word}.autoruns-comment-icon{border:none;background:transparent;color:rgb(var(--rgb-text-main));border-radius:0;min-width:16px;min-height:16px;width:16px;height:16px;padding:0;line-height:1;box-shadow:none;transform:none;display:inline-flex;align-items:center;justify-content:center;cursor:pointer;opacity:.92}.autoruns-comment-icon:hover{background:transparent;color:rgb(var(--rgb-text-main));opacity:1;box-shadow:none;transform:none}.autoruns-comment-icon.commented{color:rgb(var(--rgb-active));background:transparent;box-shadow:none}.autoruns-comment-icon:focus-visible{outline:1px solid rgba(var(--rgb-text-main),.7);outline-offset:1px}.autoruns-comment-icon:disabled{opacity:.42;cursor:not-allowed}.comment-icon-svg{width:.82rem;height:.82rem;stroke:currentColor;stroke-width:1.75;stroke-linecap:round;stroke-linejoin:round;fill:none}.comment-icon-svg.commented{stroke:rgb(var(--rgb-active));fill:none}.autoruns-resolve-toggle{border:none;background:rgb(var(--rgb-surface-focus));color:rgb(var(--rgb-text-main));border-radius:999px;width:22px;height:22px;min-width:22px;min-height:22px;padding:0;display:inline-flex;align-items:center;justify-content:center;box-shadow:none;transform:none}.autoruns-resolve-toggle:hover{background:rgb(var(--rgb-surface-focus));color:rgb(var(--rgb-active));box-shadow:none;transform:none}.autoruns-resolve-toggle:disabled{opacity:.55;cursor:wait}.autoruns-resolve-toggle.resolved{background:rgb(var(--rgb-active));color:rgb(var(--rgb-active-text))}.resolve-icon-svg{width:14px;height:14px;fill:none;stroke:currentColor;stroke-width:1.8;stroke-linecap:round;stroke-linejoin:round}.resolve-icon-svg.resolved{fill:currentColor}.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;margin-top:6px}.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;cursor:pointer}.autoruns-autogen-item.selected{background:rgb(var(--rgb-active));color:rgb(var(--rgb-active-text))}.autoruns-autogen-item.selected .autoruns-run-meta,.autoruns-autogen-item.selected p{color:rgba(var(--rgb-active-text),.95)}.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-run-launch-btn{background:rgb(var(--rgb-active));color:rgb(var(--rgb-active-text))}.autoruns-run-launch-btn:hover{background:rgb(var(--rgb-surface-focus));color:rgb(var(--rgb-text-main))}.autoruns-run-launch-btn:disabled{background:rgba(var(--rgb-active),.38);color:rgba(var(--rgb-active-text),.88)}.autoruns-generated-questions{border:none;border-radius:10px;background:rgb(var(--rgb-surface-horizontal));padding:8px;display:grid;gap:8px}.autoruns-generated-questions-head{display:flex;align-items:center;justify-content:space-between;gap:8px}.autoruns-generated-questions-list{display:grid;gap:6px;max-height:220px;overflow:auto;padding-right:2px}.autoruns-generated-question-item{position:relative;display:block;gap:8px;border:none;border-radius:9px;background:rgb(var(--rgb-surface-focus));padding:7px 30px 7px 8px;font-size:.78rem}.autoruns-generated-question-item span{display:block;white-space:pre-wrap}.autoruns-remove-question-btn{position:absolute;top:6px;right:6px;flex:0 0 auto;border:none;border-radius:0;background:transparent;color:rgb(var(--rgb-text-main));min-width:16px;width:16px;height:16px;padding:0;font-size:1rem;font-weight:700;line-height:1;display:inline-flex;align-items:center;justify-content:center;transform:rotate(45deg);box-shadow:none;transition:color .15s ease}.autoruns-remove-question-btn:hover{background:transparent;color:rgb(var(--rgb-active-text));box-shadow:none}.autoruns-remove-question-btn:focus-visible{outline:none}.autoruns-personality-prompt{resize:vertical;min-height:110px}.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;cursor:pointer}.autoruns-comment-item p{margin:0;white-space:pre-wrap;color:var(--text-muted);font-size:.79rem}.autoruns-comment-item.selected{background:rgb(var(--rgb-active));color:rgb(var(--rgb-active-text))}.autoruns-comment-item.selected p,.autoruns-comment-item.selected .autoruns-run-meta,.autoruns-comment-item.selected .muted{color:rgba(var(--rgb-active-text),.94)}.autoruns-comment-item.selected .autoruns-resolve-toggle{background:rgba(var(--rgb-active-text),.18);color:rgb(var(--rgb-active-text))}.autoruns-comment-head{display:flex;justify-content:space-between;align-items:center;gap:8px;font-size:.75rem}.autoruns-comment-head-actions{display:inline-flex;align-items:center;gap:8px}.autoruns-comment-filter-row{display:grid;grid-template-columns:minmax(0,1fr) auto;align-items:end;gap:10px;margin-bottom:8px}.autoruns-resolved-filter-toggle{min-height:38px;white-space:nowrap}.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{--mode-column-width: 400px}.metrics-grid{grid-template-columns:repeat(3,minmax(0,1fr))}}@media(max-width:920px){:root{--mode-column-width: 360px}.grid-two,.runtime-grid,.runtime-stack{grid-template-columns:1fr}.metrics-grid{grid-template-columns:repeat(2,minmax(0,1fr))}.autoruns-form-grid,.autoruns-dialog-toolbar,.autoruns-stats-grid,.autoruns-comment-filter-row{grid-template-columns:1fr}}@media(max-width:640px){:root{--mode-column-width: 320px}.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)}} +@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-assistant-chip: 18, 18, 18;--rgb-assistant-chip-hover: 44, 44, 44;--rgb-assistant-chip-selected: 228, 142, 92;--rgb-assistant-chip-selected-text: 18, 18, 18;--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;--mode-column-width: 440px;--mode-toggle-width: 188px;--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: var(--mode-column-width)}*{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,.layout-grid.layout-grid-mode-columns{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;max-width:72%;overflow-x:auto;overflow-y:hidden;padding-bottom:2px}.mode-switch-row .tab{white-space:nowrap}.mode-switch-row.mode-switch-row-right .tab{width:var(--mode-toggle-width);min-width:var(--mode-toggle-width);text-align:center}.mode-columns{display:flex;gap:12px;width:100%;min-height:0;flex:1 1 auto;overflow-x:auto;overflow-y:hidden;padding-bottom:4px}.mode-col{flex:0 0 var(--mode-column-width);width:var(--mode-column-width);min-height:0;height:100%;display:flex}.mode-col.mode-col-wide,.mode-col.mode-col-xwide{flex-basis:var(--mode-column-width);width:var(--mode-column-width)}.mode-col .panel-frame{width:100%;height:100%}.mode-col .panel-body{min-height:0;overflow:auto}.mode-columns-empty{min-width:360px;border-radius:14px;background:rgb(var(--rgb-surface-main));color:var(--text-muted);padding:14px}.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-toolbar{display:grid;gap:8px}.assistant-toolbar-actions{display:grid;grid-template-columns:repeat(3,minmax(0,1fr));align-items:center;gap:8px}.assistant-toolbar-meta{display:flex;align-items:center;justify-content:space-between;gap:8px}.assistant-toolbar-meta-right{margin-left:auto;display:flex;align-items:center;justify-content:flex-end;gap:8px;min-width:0;flex-wrap:wrap}.assistant-live-status{color:var(--text-muted);font-size:.8rem;white-space:nowrap}.assistant-toolbar-error{margin:0}.assistant-copy-btn{width:100%;justify-self:stretch;background:transparent;border-color:transparent;color:var(--text-main);font-size:.6rem;line-height:1.1;white-space:nowrap;text-align:center;letter-spacing:0;padding:6px 8px;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;overflow-y:auto;overflow-x:hidden;scrollbar-gutter:stable}.assistant-input-textarea,.autoruns-personality-prompt{overflow-y:auto;overflow-x:hidden;scrollbar-gutter:stable both-edges;border-bottom-right-radius:6px}.assistant-input-textarea::-webkit-scrollbar-corner,.autoruns-personality-prompt::-webkit-scrollbar-corner{background:rgb(var(--rgb-surface-horizontal))}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}.prompt-manager-grid{display:grid;grid-template-columns: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{flex:1 1 auto;min-height:0;overflow:auto;overscroll-behavior:contain;display:flex;flex-direction:column;gap:8px;padding:10px;border:none;border-radius:12px;background:rgb(var(--rgb-surface-horizontal))}.assistant-chat-list .assistant-msg:first-child{margin-top:auto}.assistant-empty{padding:18px;text-align:center}.assistant-msg{border:none;border-radius:12px;background:rgb(var(--rgb-surface-focus));padding:8px 10px;display:grid;gap:6px;min-width:0}.assistant-msg.user{margin-left:12%;border-color:transparent;background:rgb(var(--rgb-active));color:rgb(var(--rgb-active-text))}.assistant-msg.assistant{margin-right:12%;border-color:transparent}.assistant-msg.user .assistant-msg-head{color:rgba(var(--rgb-active-text),.9)}.assistant-msg.user .assistant-msg-body{color:rgb(var(--rgb-active-text))}.assistant-msg-head{display:flex;align-items:center;justify-content:space-between;gap:8px;margin-bottom:0;font-size:.74rem;color:var(--text-muted)}.assistant-msg-head-main{flex:1 1 auto;min-width:0;display:flex;align-items:center;justify-content:space-between;gap:8px}.assistant-msg-head-actions{display:inline-flex;align-items:center;justify-content:center;flex:0 0 auto}.assistant-comment-btn{cursor:pointer}.assistant-comment-btn:disabled{opacity:.42;cursor:not-allowed}.assistant-msg.user .assistant-comment-btn{color:rgba(var(--rgb-active-text),.92)}.assistant-msg-body{display:grid;gap:10px;line-height:1.35;font-size:.84rem;min-width:0}.assistant-msg-block{display:grid;gap:4px;min-width:0}.assistant-msg-block.selectable{cursor:pointer;padding:8px 10px;border-radius:12px;transition:background .18s ease,color .18s ease}.assistant-msg-block.selectable:hover,.assistant-msg-block.selectable:focus-visible{background:rgba(var(--rgb-active),.18);outline:none}.assistant-msg-block.selectable.active{background:rgb(var(--rgb-active));color:rgb(var(--rgb-active-text))}.assistant-msg-block.selectable.active:hover,.assistant-msg-block.selectable.active:focus-visible{background:rgb(var(--rgb-active))}.assistant-msg-block.selectable.active .assistant-msg-line,.assistant-msg-block.selectable.active .assistant-msg-line strong{color:rgb(var(--rgb-active-text))}.assistant-msg-line{margin:0;white-space:pre-wrap;word-break:break-word;overflow-wrap:anywhere}.assistant-msg-line.heading{font-weight:700;letter-spacing:.01em}.assistant-msg-line.numbered{margin-top:2px}.assistant-msg-line strong{font-weight:800}.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:0;display:grid;gap:8px;flex:0 0 auto}.assistant-compose-context{display:grid;gap:6px;padding:10px 12px;border-radius:var(--radius-md);background:rgb(var(--rgb-surface-horizontal))}.assistant-compose-context-label{color:var(--text-muted);font-size:.74rem;font-weight:700;letter-spacing:.01em}.assistant-compose-context-pill{display:inline-flex;align-items:center;gap:8px;max-width:100%;width:fit-content;min-height:38px;padding:0 10px 0 12px;border-radius:999px;background:rgb(var(--rgb-assistant-chip-selected));color:rgb(var(--rgb-assistant-chip-selected-text))}.assistant-compose-context-pill-text{min-width:0;max-width:min(100%,460px);overflow:hidden;text-overflow:ellipsis;white-space:nowrap;font-size:.82rem;font-weight:700}.assistant-compose-context-clear{display:inline-flex;align-items:center;justify-content:center;width:24px;height:24px;min-width:24px;padding:0;border-radius:999px;background:#0000002e;color:inherit;font-size:.96rem;line-height:1}.assistant-compose-context-clear:hover{background:#00000047}.assistant-send-row{align-items:center;margin-top:2px}.assistant-send-btn{margin-left:auto}.assistant-comments-frame .panel-body{display:flex;flex-direction:column;min-height:0;overflow:hidden}.assistant-comments-shell{display:grid;gap:8px;min-height:0;height:100%}.assistant-comments-toolbar{display:flex;align-items:center;justify-content:space-between;gap:8px}.assistant-comments-list{display:grid;gap:8px;overflow:auto;min-height:0;padding-right:2px}.assistant-comment-item{border:none;border-radius:10px;background:rgb(var(--rgb-surface-horizontal));padding:8px;display:grid;gap:6px}.assistant-comment-head{display:flex;align-items:center;justify-content:space-between;gap:8px;font-size:.75rem}.assistant-comment-item p{margin:0;white-space:pre-wrap;font-size:.8rem}.assistant-comment-meta{display:flex;flex-wrap:wrap;gap:8px;color:var(--text-muted);font-size:.74rem}.app-root-autoruns .assistant-panel-frame .panel-header{position:sticky;top:-12px;z-index:8;margin:-12px -12px 0;padding:12px 12px 10px;background:rgb(var(--rgb-surface-main))}.app-root-autoruns .assistant-panel-frame{overflow:visible}.app-root-autoruns .assistant-panel-frame .panel-body{flex:1 1 auto;padding:0 12px 12px;display:flex;flex-direction:column;min-height:0;overflow:hidden}.app-root-autoruns .assistant-panel-frame .assistant-live-shell{flex:1 1 auto;min-height:0;padding:12px;border-radius:14px;background:rgb(var(--rgb-background));display:flex;flex-direction:column;gap:10px}.app-root-autoruns .assistant-panel-frame .assistant-chat-list{overflow-y:auto}.app-root-autoruns .assistant-panel-frame .panel-header h2{margin:0;font-size:.95rem}.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-stack{display:grid;grid-template-columns:minmax(0,1fr);gap:12px}.runtime-details{display:grid;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(--mode-column-width);width:var(--mode-column-width);height:100%;min-height:0;overflow-y:auto;overflow-x:hidden;border:none;border-radius:14px;background:rgb(var(--rgb-surface-main));padding:12px;scrollbar-gutter:stable}.autoruns-settings-col{display:flex;flex-direction:column}.autoruns-settings-stack{display:grid;gap:12px}.embedded-panel-section{display:grid;gap:12px;border-radius:12px;background:rgb(var(--rgb-surface-horizontal));padding:12px}.embedded-panel-section-header{display:flex;align-items:flex-start;justify-content:space-between;gap:12px}.embedded-panel-section-header h4{margin:0;color:var(--text-main);font-size:.92rem}.embedded-panel-section-header p{margin:6px 0 0;color:var(--text-muted);font-size:.78rem;line-height:1.4}.autoruns-assistant-live-col{background:rgb(var(--rgb-surface-main));padding:12px;overflow:hidden;scrollbar-gutter:auto}.autoruns-assistant-live-col .panel-frame{height:100%;background:rgb(var(--rgb-surface-main))}.autoruns-col h3{margin:0;font-size:.95rem}.autoruns-col h4{margin:12px 0 8px;font-size:.82rem;color:var(--text-muted)}.autoruns-col-header{position:sticky;top:-12px;z-index:8;margin:-12px -12px 10px;padding:12px 12px 10px;background:rgb(var(--rgb-surface-main))}.autoruns-col-header .tab-row{margin:8px 0 0}.autoruns-col-header .autoruns-dialog-toolbar{margin-top:8px}.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;transition:background-color .2s ease,box-shadow .2s ease}.autoruns-run-item.selected{background:rgb(var(--rgb-active));color:rgb(var(--rgb-active-text));box-shadow:none}.autoruns-run-item.selected .autoruns-run-meta{color:rgba(var(--rgb-active-text),.95)}.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-run-id-row{display:flex;align-items:center;justify-content:space-between;gap:8px;min-width:0}.autoruns-run-id-row>span{min-width:0;overflow-wrap:anywhere;word-break:break-word}.autoruns-copy-run-id-btn{border:none;background:transparent;color:rgb(var(--rgb-text-main));width:16px;height:16px;min-width:16px;min-height:16px;padding:0;display:inline-flex;align-items:center;justify-content:center;border-radius:4px;opacity:.92;cursor:pointer}.autoruns-copy-run-id-btn:hover{color:rgb(var(--rgb-text-main));opacity:1;background:transparent;box-shadow:none;transform:none}.autoruns-copy-run-id-btn:focus-visible{outline:1px solid rgba(var(--rgb-text-main),.7);outline-offset:1px}.autoruns-copy-icon-svg{width:.82rem;height:.82rem;fill:none;stroke:currentColor;stroke-width:1.75;stroke-linecap:round;stroke-linejoin:round}.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{background:rgb(var(--rgb-active));color:rgb(var(--rgb-active-text));box-shadow:none}.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;min-width:0;overflow:hidden}.autoruns-msg header,.autoruns-msg footer{display:flex;justify-content:space-between;align-items:flex-start;flex-wrap:wrap;min-width:0;gap:8px;font-size:.74rem;color:var(--text-muted)}.autoruns-msg-head-actions{display:flex;align-items:center;justify-content:flex-end;flex:1 1 auto;min-width:0;flex-wrap:wrap;gap:8px}.autoruns-msg-head-actions>span{min-width:0;overflow-wrap:anywhere;word-break:break-word}.autoruns-msg-case-tag{display:inline-flex;align-items:center;border-radius:999px;padding:2px 8px;font-size:.7rem;line-height:1;color:rgb(var(--rgb-active-text));background:rgba(var(--rgb-active),.24)}.autoruns-msg p{margin:0;white-space:pre-wrap;line-height:1.35;font-size:.84rem;overflow-wrap:anywhere;word-break:break-word}.autoruns-msg footer span{min-width:0;overflow-wrap:anywhere;word-break:break-word}.autoruns-comment-icon{border:none;background:transparent;color:rgb(var(--rgb-text-main));border-radius:0;min-width:16px;min-height:16px;width:16px;height:16px;padding:0;line-height:1;box-shadow:none;transform:none;display:inline-flex;align-items:center;justify-content:center;cursor:pointer;opacity:.92}.autoruns-comment-icon:hover{background:transparent;color:rgb(var(--rgb-text-main));opacity:1;box-shadow:none;transform:none}.autoruns-comment-icon.commented{color:rgb(var(--rgb-active));background:transparent;box-shadow:none}.autoruns-comment-icon:focus-visible{outline:1px solid rgba(var(--rgb-text-main),.7);outline-offset:1px}.autoruns-comment-icon:disabled{opacity:.42;cursor:not-allowed}.comment-icon-svg{width:.82rem;height:.82rem;stroke:currentColor;stroke-width:1.75;stroke-linecap:round;stroke-linejoin:round;fill:none}.comment-icon-svg.commented{stroke:rgb(var(--rgb-active));fill:none}.autoruns-resolve-toggle{border:none;background:rgb(var(--rgb-surface-focus));color:rgb(var(--rgb-text-main));border-radius:999px;width:22px;height:22px;min-width:22px;min-height:22px;padding:0;display:inline-flex;align-items:center;justify-content:center;box-shadow:none;transform:none}.autoruns-resolve-toggle:hover{background:rgb(var(--rgb-surface-focus));color:rgb(var(--rgb-active));box-shadow:none;transform:none}.autoruns-resolve-toggle:disabled{opacity:.55;cursor:wait}.autoruns-resolve-toggle.resolved{background:rgb(var(--rgb-active));color:rgb(var(--rgb-active-text))}.resolve-icon-svg{width:14px;height:14px;fill:none;stroke:currentColor;stroke-width:1.8;stroke-linecap:round;stroke-linejoin:round}.resolve-icon-svg.resolved{fill:currentColor}.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;margin-top:6px}.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;cursor:pointer}.autoruns-autogen-item.selected{background:rgb(var(--rgb-active));color:rgb(var(--rgb-active-text))}.autoruns-autogen-item.selected .autoruns-run-meta,.autoruns-autogen-item.selected p{color:rgba(var(--rgb-active-text),.95)}.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-run-launch-btn{background:rgb(var(--rgb-active));color:rgb(var(--rgb-active-text))}.autoruns-run-launch-btn:hover{background:rgb(var(--rgb-surface-focus));color:rgb(var(--rgb-text-main))}.autoruns-run-launch-btn:disabled{background:rgba(var(--rgb-active),.38);color:rgba(var(--rgb-active-text),.88)}.autoruns-generated-questions{border:none;border-radius:10px;background:rgb(var(--rgb-surface-horizontal));padding:8px;display:grid;gap:8px}.autoruns-generated-questions-head{display:flex;align-items:center;justify-content:space-between;gap:8px}.autoruns-generated-questions-list{display:grid;gap:6px;max-height:220px;overflow:auto;padding-right:2px}.autoruns-generated-question-item{position:relative;display:block;gap:8px;border:none;border-radius:9px;background:rgb(var(--rgb-surface-focus));padding:7px 30px 7px 8px;font-size:.78rem}.autoruns-generated-question-item span{display:block;white-space:pre-wrap}.autoruns-remove-question-btn{position:absolute;top:6px;right:6px;flex:0 0 auto;border:none;border-radius:0;background:transparent;color:rgb(var(--rgb-text-main));min-width:16px;width:16px;height:16px;padding:0;font-size:1rem;font-weight:700;line-height:1;display:inline-flex;align-items:center;justify-content:center;transform:rotate(45deg);box-shadow:none;transition:color .15s ease}.autoruns-remove-question-btn:hover{background:transparent;color:rgb(var(--rgb-active-text));box-shadow:none}.autoruns-remove-question-btn:focus-visible{outline:none}.autoruns-personality-prompt{resize:vertical;min-height:110px}.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;cursor:pointer}.autoruns-comment-item p{margin:0;white-space:pre-wrap;color:var(--text-muted);font-size:.79rem}.autoruns-comment-item.selected{background:rgb(var(--rgb-active));color:rgb(var(--rgb-active-text))}.autoruns-comment-item.selected p,.autoruns-comment-item.selected .autoruns-run-meta,.autoruns-comment-item.selected .muted{color:rgba(var(--rgb-active-text),.94)}.autoruns-comment-item.selected .autoruns-resolve-toggle{background:rgba(var(--rgb-active-text),.18);color:rgb(var(--rgb-active-text))}.autoruns-comment-head{display:flex;justify-content:space-between;align-items:center;gap:8px;font-size:.75rem}.autoruns-comment-head-actions{display:inline-flex;align-items:center;gap:8px}.autoruns-comment-filter-row{display:grid;grid-template-columns:minmax(0,1fr) auto;align-items:end;gap:10px;margin-bottom:8px}.autoruns-resolved-filter-toggle{min-height:38px;white-space:nowrap}.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{--mode-column-width: 400px}.metrics-grid{grid-template-columns:repeat(3,minmax(0,1fr))}}@media(max-width:920px){:root{--mode-column-width: 360px}.grid-two,.runtime-grid,.runtime-stack{grid-template-columns:1fr}.metrics-grid{grid-template-columns:repeat(2,minmax(0,1fr))}.autoruns-form-grid,.autoruns-dialog-toolbar,.autoruns-stats-grid,.autoruns-comment-filter-row{grid-template-columns:1fr}}@media(max-width:640px){:root{--mode-column-width: 320px}.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-Qq5WpuqR.js b/llm_normalizer/frontend/dist/assets/index-Qq5WpuqR.js deleted file mode 100644 index 5b8b117..0000000 --- a/llm_normalizer/frontend/dist/assets/index-Qq5WpuqR.js +++ /dev/null @@ -1,26 +0,0 @@ -(function(){const m=document.createElement("link").relList;if(m&&m.supports&&m.supports("modulepreload"))return;for(const j of document.querySelectorAll('link[rel="modulepreload"]'))N(j);new MutationObserver(j=>{for(const T of j)if(T.type==="childList")for(const F of T.addedNodes)F.tagName==="LINK"&&F.rel==="modulepreload"&&N(F)}).observe(document,{childList:!0,subtree:!0});function f(j){const T={};return j.integrity&&(T.integrity=j.integrity),j.referrerPolicy&&(T.referrerPolicy=j.referrerPolicy),j.crossOrigin==="use-credentials"?T.credentials="include":j.crossOrigin==="anonymous"?T.credentials="omit":T.credentials="same-origin",T}function N(j){if(j.ep)return;j.ep=!0;const T=f(j);fetch(j.href,T)}})();function bc(i){return i&&i.__esModule&&Object.prototype.hasOwnProperty.call(i,"default")?i.default:i}var Fi={exports:{}},no={},Ui={exports:{}},ge={};var gc;function wf(){if(gc)return ge;gc=1;var i=Symbol.for("react.element"),m=Symbol.for("react.portal"),f=Symbol.for("react.fragment"),N=Symbol.for("react.strict_mode"),j=Symbol.for("react.profiler"),T=Symbol.for("react.provider"),F=Symbol.for("react.context"),Y=Symbol.for("react.forward_ref"),U=Symbol.for("react.suspense"),I=Symbol.for("react.memo"),q=Symbol.for("react.lazy"),D=Symbol.iterator;function B(h){return h===null||typeof h!="object"?null:(h=D&&h[D]||h["@@iterator"],typeof h=="function"?h:null)}var oe={isMounted:function(){return!1},enqueueForceUpdate:function(){},enqueueReplaceState:function(){},enqueueSetState:function(){}},Ae=Object.assign,se={};function te(h,k,J){this.props=h,this.context=k,this.refs=se,this.updater=J||oe}te.prototype.isReactComponent={},te.prototype.setState=function(h,k){if(typeof h!="object"&&typeof h!="function"&&h!=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,h,k,"setState")},te.prototype.forceUpdate=function(h){this.updater.enqueueForceUpdate(this,h,"forceUpdate")};function Te(){}Te.prototype=te.prototype;function ve(h,k,J){this.props=h,this.context=k,this.refs=se,this.updater=J||oe}var G=ve.prototype=new Te;G.constructor=ve,Ae(G,te.prototype),G.isPureReactComponent=!0;var ae=Array.isArray,_e=Object.prototype.hasOwnProperty,Oe={current:null},$e={key:!0,ref:!0,__self:!0,__source:!0};function We(h,k,J){var ue,de={},fe=null,Se=null;if(k!=null)for(ue in k.ref!==void 0&&(Se=k.ref),k.key!==void 0&&(fe=""+k.key),k)_e.call(k,ue)&&!$e.hasOwnProperty(ue)&&(de[ue]=k[ue]);var ye=arguments.length-2;if(ye===1)de.children=J;else if(1>>1,k=O[h];if(0>>1;hj(de,$))fej(Se,de)?(O[h]=Se,O[fe]=$,h=fe):(O[h]=de,O[ue]=$,h=ue);else if(fej(Se,$))O[h]=Se,O[fe]=$,h=fe;else break e}}return W}function j(O,W){var $=O.sortIndex-W.sortIndex;return $!==0?$:O.id-W.id}if(typeof performance=="object"&&typeof performance.now=="function"){var T=performance;i.unstable_now=function(){return T.now()}}else{var F=Date,Y=F.now();i.unstable_now=function(){return F.now()-Y}}var U=[],I=[],q=1,D=null,B=3,oe=!1,Ae=!1,se=!1,te=typeof setTimeout=="function"?setTimeout:null,Te=typeof clearTimeout=="function"?clearTimeout:null,ve=typeof setImmediate<"u"?setImmediate:null;typeof navigator<"u"&&navigator.scheduling!==void 0&&navigator.scheduling.isInputPending!==void 0&&navigator.scheduling.isInputPending.bind(navigator.scheduling);function G(O){for(var W=f(I);W!==null;){if(W.callback===null)N(I);else if(W.startTime<=O)N(I),W.sortIndex=W.expirationTime,m(U,W);else break;W=f(I)}}function ae(O){if(se=!1,G(O),!Ae)if(f(U)!==null)Ae=!0,Re(_e);else{var W=f(I);W!==null&&pe(ae,W.startTime-O)}}function _e(O,W){Ae=!1,se&&(se=!1,Te(We),We=-1),oe=!0;var $=B;try{for(G(W),D=f(U);D!==null&&(!(D.expirationTime>W)||O&&!st());){var h=D.callback;if(typeof h=="function"){D.callback=null,B=D.priorityLevel;var k=h(D.expirationTime<=W);W=i.unstable_now(),typeof k=="function"?D.callback=k:D===f(U)&&N(U),G(W)}else N(U);D=f(U)}if(D!==null)var J=!0;else{var ue=f(I);ue!==null&&pe(ae,ue.startTime-W),J=!1}return J}finally{D=null,B=$,oe=!1}}var Oe=!1,$e=null,We=-1,Je=5,et=-1;function st(){return!(i.unstable_now()-etO||125h?(O.sortIndex=$,m(I,O),f(U)===null&&O===f(I)&&(se?(Te(We),We=-1):se=!0,pe(ae,$-h))):(O.sortIndex=k,m(U,O),Ae||oe||(Ae=!0,Re(_e))),O},i.unstable_shouldYield=st,i.unstable_wrapCallback=function(O){var W=B;return function(){var $=B;B=W;try{return O.apply(this,arguments)}finally{B=$}}}})(Hi)),Hi}var Sc;function Ef(){return Sc||(Sc=1,Bi.exports=Nf()),Bi.exports}var wc;function Pf(){if(wc)return Ft;wc=1;var i=Zi(),m=Ef();function f(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"),U=Object.prototype.hasOwnProperty,I=/^[: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]*$/,q={},D={};function B(e){return U.call(D,e)?!0:U.call(q,e)?!1:I.test(e)?D[e]=!0:(q[e]=!0,!1)}function oe(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 Ae(e,t,n,r){if(t===null||typeof t>"u"||oe(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 se(e,t,n,r,s,l,u){this.acceptsBooleans=t===2||t===3||t===4,this.attributeName=r,this.attributeNamespace=s,this.mustUseProperty=n,this.propertyName=e,this.type=t,this.sanitizeURL=l,this.removeEmptyString=u}var te={};"children dangerouslySetInnerHTML defaultValue defaultChecked innerHTML suppressContentEditableWarning suppressHydrationWarning style".split(" ").forEach(function(e){te[e]=new se(e,0,!1,e,null,!1,!1)}),[["acceptCharset","accept-charset"],["className","class"],["htmlFor","for"],["httpEquiv","http-equiv"]].forEach(function(e){var t=e[0];te[t]=new se(t,1,!1,e[1],null,!1,!1)}),["contentEditable","draggable","spellCheck","value"].forEach(function(e){te[e]=new se(e,2,!1,e.toLowerCase(),null,!1,!1)}),["autoReverse","externalResourcesRequired","focusable","preserveAlpha"].forEach(function(e){te[e]=new se(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){te[e]=new se(e,3,!1,e.toLowerCase(),null,!1,!1)}),["checked","multiple","muted","selected"].forEach(function(e){te[e]=new se(e,3,!0,e,null,!1,!1)}),["capture","download"].forEach(function(e){te[e]=new se(e,4,!1,e,null,!1,!1)}),["cols","rows","size","span"].forEach(function(e){te[e]=new se(e,6,!1,e,null,!1,!1)}),["rowSpan","start"].forEach(function(e){te[e]=new se(e,5,!1,e.toLowerCase(),null,!1,!1)});var Te=/[\-:]([a-z])/g;function ve(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(Te,ve);te[t]=new se(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(Te,ve);te[t]=new se(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(Te,ve);te[t]=new se(t,1,!1,e,"http://www.w3.org/XML/1998/namespace",!1,!1)}),["tabIndex","crossOrigin"].forEach(function(e){te[e]=new se(e,1,!1,e.toLowerCase(),null,!1,!1)}),te.xlinkHref=new se("xlinkHref",1,!1,"xlink:href","http://www.w3.org/1999/xlink",!0,!1),["src","href","action","formAction"].forEach(function(e){te[e]=new se(e,1,!1,e.toLowerCase(),null,!0,!0)});function G(e,t,n,r){var s=te.hasOwnProperty(t)?te[t]:null;(s!==null?s.type!==0:r||!(2d||s[u]!==l[d]){var g=` -`+s[u].replace(" at new "," at ");return e.displayName&&g.includes("")&&(g=g.replace("",e.displayName)),g}while(1<=u&&0<=d);break}}}finally{J=!1,Error.prepareStackTrace=n}return(e=e?e.displayName||e.name:"")?k(e):""}function de(e){switch(e.tag){case 5:return k(e.type);case 16:return k("Lazy");case 13:return k("Suspense");case 19:return k("SuspenseList");case 0:case 2:case 15:return e=ue(e.type,!1),e;case 11:return e=ue(e.type.render,!1),e;case 1:return e=ue(e.type,!0),e;default:return""}}function fe(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 Oe:return"Portal";case Je:return"Profiler";case We:return"StrictMode";case Ie:return"Suspense";case Fe:return"SuspenseList"}if(typeof e=="object")switch(e.$$typeof){case st:return(e.displayName||"Context")+".Consumer";case et:return(e._context.displayName||"Context")+".Provider";case ie:var t=e.render;return e=e.displayName,e||(e=t.displayName||t.name||"",e=e!==""?"ForwardRef("+e+")":"ForwardRef"),e;case me:return t=e.displayName||null,t!==null?t:fe(e.type)||"Memo";case Re:t=e._payload,e=e._init;try{return fe(e(t))}catch{}}return null}function Se(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 fe(t);case 8:return t===We?"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 ye(e){switch(typeof e){case"boolean":case"number":case"string":case"undefined":return e;case"object":return e;default:return""}}function ee(e){var t=e.type;return(e=e.nodeName)&&e.toLowerCase()==="input"&&(t==="checkbox"||t==="radio")}function Pe(e){var t=ee(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 s=n.get,l=n.set;return Object.defineProperty(e,t,{configurable:!0,get:function(){return s.call(this)},set:function(u){r=""+u,l.call(this,u)}}),Object.defineProperty(e,t,{enumerable:n.enumerable}),{getValue:function(){return r},setValue:function(u){r=""+u},stopTracking:function(){e._valueTracker=null,delete e[t]}}}}function Ut(e){e._valueTracker||(e._valueTracker=Pe(e))}function Kn(e){if(!e)return!1;var t=e._valueTracker;if(!t)return!0;var n=t.getValue(),r="";return e&&(r=ee(e)?e.checked?"true":"false":e.value),e=r,e!==n?(t.setValue(e),!0):!1}function In(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 Ln(e,t){var n=t.checked;return $({},t,{defaultChecked:void 0,defaultValue:void 0,value:void 0,checked:n??e._wrapperState.initialChecked})}function fr(e,t){var n=t.defaultValue==null?"":t.defaultValue,r=t.checked!=null?t.checked:t.defaultChecked;n=ye(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 Xt(e,t){t=t.checked,t!=null&&G(e,"checked",t,!1)}function Zt(e,t){Xt(e,t);var n=ye(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")?hn(e,t.type,n):t.hasOwnProperty("defaultValue")&&hn(e,t.type,ye(t.defaultValue)),t.checked==null&&t.defaultChecked!=null&&(e.defaultChecked=!!t.defaultChecked)}function Gn(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 hn(e,t,n){(t!=="number"||In(e.ownerDocument)!==e)&&(n==null?e.defaultValue=""+e._wrapperState.initialValue:e.defaultValue!==""+n&&(e.defaultValue=""+n))}var bt=Array.isArray;function en(e,t,n,r){if(e=e.options,t){t={};for(var s=0;s"+t.valueOf().toString()+"",t=nn.firstChild;e.firstChild;)e.removeChild(e.firstChild);for(;t.firstChild;)e.appendChild(t.firstChild)}});function Bt(e,t){if(t){var n=e.firstChild;if(n&&n===e.lastChild&&n.nodeType===3){n.nodeValue=t;return}}e.textContent=t}var vn={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},be=["Webkit","ms","Moz","O"];Object.keys(vn).forEach(function(e){be.forEach(function(t){t=t+e.charAt(0).toUpperCase()+e.substring(1),vn[t]=vn[e]})});function ot(e,t,n){return t==null||typeof t=="boolean"||t===""?"":n||typeof t!="number"||t===0||vn.hasOwnProperty(e)&&vn[e]?(""+t).trim():t+"px"}function yn(e,t){e=e.style;for(var n in t)if(t.hasOwnProperty(n)){var r=n.indexOf("--")===0,s=ot(n,t[n],r);n==="float"&&(n="cssFloat"),r?e.setProperty(n,s):e[n]=s}}var xn=$({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 rn(e,t){if(t){if(xn[e]&&(t.children!=null||t.dangerouslySetInnerHTML!=null))throw Error(f(137,e));if(t.dangerouslySetInnerHTML!=null){if(t.children!=null)throw Error(f(60));if(typeof t.dangerouslySetInnerHTML!="object"||!("__html"in t.dangerouslySetInnerHTML))throw Error(f(61))}if(t.style!=null&&typeof t.style!="object")throw Error(f(62))}}function sn(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 Nt=null;function zn(e){return e=e.target||e.srcElement||window,e.correspondingUseElement&&(e=e.correspondingUseElement),e.nodeType===3?e.parentNode:e}var on=null,Ht=null,gt=null;function Jn(e){if(e=bs(e)){if(typeof on!="function")throw Error(f(280));var t=e.stateNode;t&&(t=yo(t),on(e.stateNode,e.type,t))}}function qe(e){Ht?gt?gt.push(e):gt=[e]:Ht=e}function gr(){if(Ht){var e=Ht,t=gt;if(gt=Ht=null,Jn(e),t)for(e=0;e>>=0,e===0?32:31-(Ns(e)/ts|0)|0}var Zn=64,er=4194304;function Tt(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 wt(e,t){var n=e.pendingLanes;if(n===0)return 0;var r=0,s=e.suspendedLanes,l=e.pingedLanes,u=n&268435455;if(u!==0){var d=u&~s;d!==0?r=Tt(d):(l&=u,l!==0&&(r=Tt(l)))}else u=n&~s,u!==0?r=Tt(u):l!==0&&(r=Tt(l));if(r===0)return 0;if(t!==0&&t!==r&&(t&s)===0&&(s=r&-r,l=t&-t,s>=l||s===16&&(l&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 Qt(e,t,n){e.pendingLanes|=t,t!==536870912&&(e.suspendedLanes=0,e.pingedLanes=0),e=e.eventTimes,t=31-St(t),e[t]=n}function Wt(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=Ts),ua=" ",ca=!1;function da(e,t){switch(e){case"keyup":return gd.indexOf(t.keyCode)!==-1;case"keydown":return t.keyCode!==229;case"keypress":case"mousedown":case"focusout":return!0;default:return!1}}function fa(e){return e=e.detail,typeof e=="object"&&"data"in e?e.data:null}var os=!1;function yd(e,t){switch(e){case"compositionend":return fa(t);case"keypress":return t.which!==32?null:(ca=!0,ua);case"textInput":return e=t.data,e===ua&&ca?null:e;default:return null}}function xd(e,t){if(os)return e==="compositionend"||!xl&&da(e,t)?(e=ra(),lo=ml=wr=null,os=!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=xa(n)}}function Sa(e,t){return e&&t?e===t?!0:e&&e.nodeType===3?!1:t&&t.nodeType===3?Sa(e,t.parentNode):"contains"in e?e.contains(t):e.compareDocumentPosition?!!(e.compareDocumentPosition(t)&16):!1:!1}function wa(){for(var e=window,t=In();t instanceof e.HTMLIFrameElement;){try{var n=typeof t.contentWindow.location.href=="string"}catch{n=!1}if(n)e=t.contentWindow;else break;t=In(e.document)}return t}function wl(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 Pd(e){var t=wa(),n=e.focusedElem,r=e.selectionRange;if(t!==n&&n&&n.ownerDocument&&Sa(n.ownerDocument.documentElement,n)){if(r!==null&&wl(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 s=n.textContent.length,l=Math.min(r.start,s);r=r.end===void 0?l:Math.min(r.end,s),!e.extend&&l>r&&(s=r,r=l,l=s),s=_a(n,l);var u=_a(n,r);s&&u&&(e.rangeCount!==1||e.anchorNode!==s.node||e.anchorOffset!==s.offset||e.focusNode!==u.node||e.focusOffset!==u.offset)&&(t=t.createRange(),t.setStart(s.node,s.offset),e.removeAllRanges(),l>r?(e.addRange(t),e.extend(u.node,u.offset)):(t.setEnd(u.node,u.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,ls=null,jl=null,Ds=null,kl=!1;function ja(e,t,n){var r=n.window===n?n.document:n.nodeType===9?n:n.ownerDocument;kl||ls==null||ls!==In(r)||(r=ls,"selectionStart"in r&&wl(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}),Ds&&Os(Ds,r)||(Ds=r,r=ho(jl,"onSelect"),0ds||(e.current=Dl[ds],Dl[ds]=null,ds--)}function Le(e,t){ds++,Dl[ds]=e.current,e.current=t}var Nr={},dt=Cr(Nr),Lt=Cr(!1),Br=Nr;function fs(e,t){var n=e.type.contextTypes;if(!n)return Nr;var r=e.stateNode;if(r&&r.__reactInternalMemoizedUnmaskedChildContext===t)return r.__reactInternalMemoizedMaskedChildContext;var s={},l;for(l in n)s[l]=t[l];return r&&(e=e.stateNode,e.__reactInternalMemoizedUnmaskedChildContext=t,e.__reactInternalMemoizedMaskedChildContext=s),s}function Ot(e){return e=e.childContextTypes,e!=null}function xo(){ze(Lt),ze(dt)}function $a(e,t,n){if(dt.current!==Nr)throw Error(f(168));Le(dt,t),Le(Lt,n)}function Fa(e,t,n){var r=e.stateNode;if(t=t.childContextTypes,typeof r.getChildContext!="function")return n;r=r.getChildContext();for(var s in r)if(!(s in t))throw Error(f(108,Se(e)||"Unknown",s));return $({},n,r)}function _o(e){return e=(e=e.stateNode)&&e.__reactInternalMemoizedMergedChildContext||Nr,Br=dt.current,Le(dt,e),Le(Lt,Lt.current),!0}function Ua(e,t,n){var r=e.stateNode;if(!r)throw Error(f(169));n?(e=Fa(e,t,Br),r.__reactInternalMemoizedMergedChildContext=e,ze(Lt),ze(dt),Le(dt,e)):ze(Lt),Le(Lt,n)}var rr=null,So=!1,zl=!1;function ba(e){rr===null?rr=[e]:rr.push(e)}function Ud(e){So=!0,ba(e)}function Er(){if(!zl&&rr!==null){zl=!0;var e=0,t=we;try{var n=rr;for(we=1;e>=u,s-=u,sr=1<<32-St(t)+s|n<re?(rt=Z,Z=null):rt=Z.sibling;var Ne=P(_,Z,S[re],L);if(Ne===null){Z===null&&(Z=rt);break}e&&Z&&Ne.alternate===null&&t(_,Z),v=l(Ne,v,re),X===null?K=Ne:X.sibling=Ne,X=Ne,Z=rt}if(re===S.length)return n(_,Z),Ue&&Vr(_,re),K;if(Z===null){for(;rere?(rt=Z,Z=null):rt=Z.sibling;var Dr=P(_,Z,Ne.value,L);if(Dr===null){Z===null&&(Z=rt);break}e&&Z&&Dr.alternate===null&&t(_,Z),v=l(Dr,v,re),X===null?K=Dr:X.sibling=Dr,X=Dr,Z=rt}if(Ne.done)return n(_,Z),Ue&&Vr(_,re),K;if(Z===null){for(;!Ne.done;re++,Ne=S.next())Ne=R(_,Ne.value,L),Ne!==null&&(v=l(Ne,v,re),X===null?K=Ne:X.sibling=Ne,X=Ne);return Ue&&Vr(_,re),K}for(Z=r(_,Z);!Ne.done;re++,Ne=S.next())Ne=b(Z,_,re,Ne.value,L),Ne!==null&&(e&&Ne.alternate!==null&&Z.delete(Ne.key===null?re:Ne.key),v=l(Ne,v,re),X===null?K=Ne:X.sibling=Ne,X=Ne);return e&&Z.forEach(function(Sf){return t(_,Sf)}),Ue&&Vr(_,re),K}function Ge(_,v,S,L){if(typeof S=="object"&&S!==null&&S.type===$e&&S.key===null&&(S=S.props.children),typeof S=="object"&&S!==null){switch(S.$$typeof){case _e:e:{for(var K=S.key,X=v;X!==null;){if(X.key===K){if(K=S.type,K===$e){if(X.tag===7){n(_,X.sibling),v=s(X,S.props.children),v.return=_,_=v;break e}}else if(X.elementType===K||typeof K=="object"&&K!==null&&K.$$typeof===Re&&qa(K)===X.type){n(_,X.sibling),v=s(X,S.props),v.ref=Bs(_,X,S),v.return=_,_=v;break e}n(_,X);break}else t(_,X);X=X.sibling}S.type===$e?(v=Xr(S.props.children,_.mode,L,S.key),v.return=_,_=v):(L=Go(S.type,S.key,S.props,null,_.mode,L),L.ref=Bs(_,v,S),L.return=_,_=L)}return u(_);case Oe:e:{for(X=S.key;v!==null;){if(v.key===X)if(v.tag===4&&v.stateNode.containerInfo===S.containerInfo&&v.stateNode.implementation===S.implementation){n(_,v.sibling),v=s(v,S.children||[]),v.return=_,_=v;break e}else{n(_,v);break}else t(_,v);v=v.sibling}v=Li(S,_.mode,L),v.return=_,_=v}return u(_);case Re:return X=S._init,Ge(_,v,X(S._payload),L)}if(bt(S))return V(_,v,S,L);if(W(S))return Q(_,v,S,L);Co(_,S)}return typeof S=="string"&&S!==""||typeof S=="number"?(S=""+S,v!==null&&v.tag===6?(n(_,v.sibling),v=s(v,S),v.return=_,_=v):(n(_,v),v=Ii(S,_.mode,L),v.return=_,_=v),u(_)):n(_,v)}return Ge}var gs=Ka(!0),Ga=Ka(!1),No=Cr(null),Eo=null,vs=null,Hl=null;function Vl(){Hl=vs=Eo=null}function Ql(e){var t=No.current;ze(No),e._currentValue=t}function Wl(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 ys(e,t){Eo=e,Hl=vs=null,e=e.dependencies,e!==null&&e.firstContext!==null&&((e.lanes&t)!==0&&(Dt=!0),e.firstContext=null)}function dn(e){var t=e._currentValue;if(Hl!==e)if(e={context:e,memoizedValue:t,next:null},vs===null){if(Eo===null)throw Error(f(308));vs=e,Eo.dependencies={lanes:0,firstContext:e}}else vs=vs.next=e;return t}var Qr=null;function ql(e){Qr===null?Qr=[e]:Qr.push(e)}function Ja(e,t,n,r){var s=t.interleaved;return s===null?(n.next=n,ql(t)):(n.next=s.next,s.next=n),t.interleaved=n,lr(e,r)}function lr(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 Pr=!1;function Kl(e){e.updateQueue={baseState:e.memoizedState,firstBaseUpdate:null,lastBaseUpdate:null,shared:{pending:null,interleaved:null,lanes:0},effects:null}}function Ya(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 ir(e,t){return{eventTime:e,lane:t,tag:0,payload:null,callback:null,next:null}}function Mr(e,t,n){var r=e.updateQueue;if(r===null)return null;if(r=r.shared,(je&2)!==0){var s=r.pending;return s===null?t.next=t:(t.next=s.next,s.next=t),r.pending=t,lr(e,n)}return s=r.interleaved,s===null?(t.next=t,ql(r)):(t.next=s.next,s.next=t),r.interleaved=t,lr(e,n)}function Po(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,tr(e,n)}}function Xa(e,t){var n=e.updateQueue,r=e.alternate;if(r!==null&&(r=r.updateQueue,n===r)){var s=null,l=null;if(n=n.firstBaseUpdate,n!==null){do{var u={eventTime:n.eventTime,lane:n.lane,tag:n.tag,payload:n.payload,callback:n.callback,next:null};l===null?s=l=u:l=l.next=u,n=n.next}while(n!==null);l===null?s=l=t:l=l.next=t}else s=l=t;n={baseState:r.baseState,firstBaseUpdate:s,lastBaseUpdate:l,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 Mo(e,t,n,r){var s=e.updateQueue;Pr=!1;var l=s.firstBaseUpdate,u=s.lastBaseUpdate,d=s.shared.pending;if(d!==null){s.shared.pending=null;var g=d,w=g.next;g.next=null,u===null?l=w:u.next=w,u=g;var M=e.alternate;M!==null&&(M=M.updateQueue,d=M.lastBaseUpdate,d!==u&&(d===null?M.firstBaseUpdate=w:d.next=w,M.lastBaseUpdate=g))}if(l!==null){var R=s.baseState;u=0,M=w=g=null,d=l;do{var P=d.lane,b=d.eventTime;if((r&P)===P){M!==null&&(M=M.next={eventTime:b,lane:0,tag:d.tag,payload:d.payload,callback:d.callback,next:null});e:{var V=e,Q=d;switch(P=t,b=n,Q.tag){case 1:if(V=Q.payload,typeof V=="function"){R=V.call(b,R,P);break e}R=V;break e;case 3:V.flags=V.flags&-65537|128;case 0:if(V=Q.payload,P=typeof V=="function"?V.call(b,R,P):V,P==null)break e;R=$({},R,P);break e;case 2:Pr=!0}}d.callback!==null&&d.lane!==0&&(e.flags|=64,P=s.effects,P===null?s.effects=[d]:P.push(d))}else b={eventTime:b,lane:P,tag:d.tag,payload:d.payload,callback:d.callback,next:null},M===null?(w=M=b,g=R):M=M.next=b,u|=P;if(d=d.next,d===null){if(d=s.shared.pending,d===null)break;P=d,d=P.next,P.next=null,s.lastBaseUpdate=P,s.shared.pending=null}}while(!0);if(M===null&&(g=R),s.baseState=g,s.firstBaseUpdate=w,s.lastBaseUpdate=M,t=s.shared.interleaved,t!==null){s=t;do u|=s.lane,s=s.next;while(s!==t)}else l===null&&(s.shared.lanes=0);Kr|=u,e.lanes=u,e.memoizedState=R}}function Za(e,t,n){if(e=t.effects,t.effects=null,e!==null)for(t=0;tn?n:4,e(!0);var r=Zl.transition;Zl.transition={};try{e(!1),t()}finally{we=n,Zl.transition=r}}function yu(){return fn().memoizedState}function Vd(e,t,n){var r=Ir(e);if(n={lane:r,action:n,hasEagerState:!1,eagerState:null,next:null},xu(e))_u(t,n);else if(n=Ja(e,t,n,r),n!==null){var s=Ct();An(n,e,r,s),Su(n,t,r)}}function Qd(e,t,n){var r=Ir(e),s={lane:r,action:n,hasEagerState:!1,eagerState:null,next:null};if(xu(e))_u(t,s);else{var l=e.alternate;if(e.lanes===0&&(l===null||l.lanes===0)&&(l=t.lastRenderedReducer,l!==null))try{var u=t.lastRenderedState,d=l(u,n);if(s.hasEagerState=!0,s.eagerState=d,Nn(d,u)){var g=t.interleaved;g===null?(s.next=s,ql(t)):(s.next=g.next,g.next=s),t.interleaved=s;return}}catch{}n=Ja(e,t,s,r),n!==null&&(s=Ct(),An(n,e,r,s),Su(n,t,r))}}function xu(e){var t=e.alternate;return e===Ve||t!==null&&t===Ve}function _u(e,t){Ws=To=!0;var n=e.pending;n===null?t.next=t:(t.next=n.next,n.next=t),e.pending=t}function Su(e,t,n){if((n&4194240)!==0){var r=t.lanes;r&=e.pendingLanes,n|=r,t.lanes=n,tr(e,n)}}var Oo={readContext:dn,useCallback:ft,useContext:ft,useEffect:ft,useImperativeHandle:ft,useInsertionEffect:ft,useLayoutEffect:ft,useMemo:ft,useReducer:ft,useRef:ft,useState:ft,useDebugValue:ft,useDeferredValue:ft,useTransition:ft,useMutableSource:ft,useSyncExternalStore:ft,useId:ft,unstable_isNewReconciler:!1},Wd={readContext:dn,useCallback:function(e,t){return Wn().memoizedState=[e,t===void 0?null:t],e},useContext:dn,useEffect:cu,useImperativeHandle:function(e,t,n){return n=n!=null?n.concat([e]):null,Io(4194308,4,mu.bind(null,t,e),n)},useLayoutEffect:function(e,t){return Io(4194308,4,e,t)},useInsertionEffect:function(e,t){return Io(4,2,e,t)},useMemo:function(e,t){var n=Wn();return t=t===void 0?null:t,e=e(),n.memoizedState=[e,t],e},useReducer:function(e,t,n){var r=Wn();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=Vd.bind(null,Ve,e),[r.memoizedState,e]},useRef:function(e){var t=Wn();return e={current:e},t.memoizedState=e},useState:au,useDebugValue:li,useDeferredValue:function(e){return Wn().memoizedState=e},useTransition:function(){var e=au(!1),t=e[0];return e=Hd.bind(null,e[1]),Wn().memoizedState=e,[t,e]},useMutableSource:function(){},useSyncExternalStore:function(e,t,n){var r=Ve,s=Wn();if(Ue){if(n===void 0)throw Error(f(407));n=n()}else{if(n=t(),nt===null)throw Error(f(349));(qr&30)!==0||ru(r,t,n)}s.memoizedState=n;var l={value:n,getSnapshot:t};return s.queue=l,cu(ou.bind(null,r,l,e),[e]),r.flags|=2048,Gs(9,su.bind(null,r,l,n,t),void 0,null),n},useId:function(){var e=Wn(),t=nt.identifierPrefix;if(Ue){var n=or,r=sr;n=(r&~(1<<32-St(r)-1)).toString(32)+n,t=":"+t+"R"+n,n=qs++,0<\/script>",e=e.removeChild(e.firstChild)):typeof r.is=="string"?e=u.createElement(n,{is:r.is}):(e=u.createElement(n),n==="select"&&(u=e,r.multiple?u.multiple=!0:r.size&&(u.size=r.size))):e=u.createElementNS(e,n),e[Vn]=t,e[Us]=r,bu(e,t,!1,!1),t.stateNode=e;e:{switch(u=sn(n,r),n){case"dialog":De("cancel",e),De("close",e),s=r;break;case"iframe":case"object":case"embed":De("load",e),s=r;break;case"video":case"audio":for(s=0;sjs&&(t.flags|=128,r=!0,Js(l,!1),t.lanes=4194304)}else{if(!r)if(e=Ro(u),e!==null){if(t.flags|=128,r=!0,n=e.updateQueue,n!==null&&(t.updateQueue=n,t.flags|=4),Js(l,!0),l.tail===null&&l.tailMode==="hidden"&&!u.alternate&&!Ue)return mt(t),null}else 2*Me()-l.renderingStartTime>js&&n!==1073741824&&(t.flags|=128,r=!0,Js(l,!1),t.lanes=4194304);l.isBackwards?(u.sibling=t.child,t.child=u):(n=l.last,n!==null?n.sibling=u:t.child=u,l.last=u)}return l.tail!==null?(t=l.tail,l.rendering=t,l.tail=t.sibling,l.renderingStartTime=Me(),t.sibling=null,n=He.current,Le(He,r?n&1|2:n&1),t):(mt(t),null);case 22:case 23:return Ri(),r=t.memoizedState!==null,e!==null&&e.memoizedState!==null!==r&&(t.flags|=8192),r&&(t.mode&1)!==0?(Jt&1073741824)!==0&&(mt(t),t.subtreeFlags&6&&(t.flags|=8192)):mt(t),null;case 24:return null;case 25:return null}throw Error(f(156,t.tag))}function ef(e,t){switch(Fl(t),t.tag){case 1:return Ot(t.type)&&xo(),e=t.flags,e&65536?(t.flags=e&-65537|128,t):null;case 3:return xs(),ze(Lt),ze(dt),Xl(),e=t.flags,(e&65536)!==0&&(e&128)===0?(t.flags=e&-65537|128,t):null;case 5:return Jl(t),null;case 13:if(ze(He),e=t.memoizedState,e!==null&&e.dehydrated!==null){if(t.alternate===null)throw Error(f(340));hs()}return e=t.flags,e&65536?(t.flags=e&-65537|128,t):null;case 19:return ze(He),null;case 4:return xs(),null;case 10:return Ql(t.type._context),null;case 22:case 23:return Ri(),null;case 24:return null;default:return null}}var Fo=!1,pt=!1,tf=typeof WeakSet=="function"?WeakSet:Set,H=null;function Ss(e,t){var n=e.ref;if(n!==null)if(typeof n=="function")try{n(null)}catch(r){Qe(e,t,r)}else n.current=null}function yi(e,t,n){try{n()}catch(r){Qe(e,t,r)}}var Vu=!1;function nf(e,t){if(Rl=so,e=wa(),wl(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 s=r.anchorOffset,l=r.focusNode;r=r.focusOffset;try{n.nodeType,l.nodeType}catch{n=null;break e}var u=0,d=-1,g=-1,w=0,M=0,R=e,P=null;t:for(;;){for(var b;R!==n||s!==0&&R.nodeType!==3||(d=u+s),R!==l||r!==0&&R.nodeType!==3||(g=u+r),R.nodeType===3&&(u+=R.nodeValue.length),(b=R.firstChild)!==null;)P=R,R=b;for(;;){if(R===e)break t;if(P===n&&++w===s&&(d=u),P===l&&++M===r&&(g=u),(b=R.nextSibling)!==null)break;R=P,P=R.parentNode}R=b}n=d===-1||g===-1?null:{start:d,end:g}}else n=null}n=n||{start:0,end:0}}else n=null;for(Al={focusedElem:e,selectionRange:n},so=!1,H=t;H!==null;)if(t=H,e=t.child,(t.subtreeFlags&1028)!==0&&e!==null)e.return=t,H=e;else for(;H!==null;){t=H;try{var V=t.alternate;if((t.flags&1024)!==0)switch(t.tag){case 0:case 11:case 15:break;case 1:if(V!==null){var Q=V.memoizedProps,Ge=V.memoizedState,_=t.stateNode,v=_.getSnapshotBeforeUpdate(t.elementType===t.type?Q:Pn(t.type,Q),Ge);_.__reactInternalSnapshotBeforeUpdate=v}break;case 3:var S=t.stateNode.containerInfo;S.nodeType===1?S.textContent="":S.nodeType===9&&S.documentElement&&S.removeChild(S.documentElement);break;case 5:case 6:case 4:case 17:break;default:throw Error(f(163))}}catch(L){Qe(t,t.return,L)}if(e=t.sibling,e!==null){e.return=t.return,H=e;break}H=t.return}return V=Vu,Vu=!1,V}function Ys(e,t,n){var r=t.updateQueue;if(r=r!==null?r.lastEffect:null,r!==null){var s=r=r.next;do{if((s.tag&e)===e){var l=s.destroy;s.destroy=void 0,l!==void 0&&yi(t,n,l)}s=s.next}while(s!==r)}}function Uo(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 xi(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 Qu(e){var t=e.alternate;t!==null&&(e.alternate=null,Qu(t)),e.child=null,e.deletions=null,e.sibling=null,e.tag===5&&(t=e.stateNode,t!==null&&(delete t[Vn],delete t[Us],delete t[Ol],delete t[$d],delete t[Fd])),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 Wu(e){return e.tag===5||e.tag===3||e.tag===4}function qu(e){e:for(;;){for(;e.sibling===null;){if(e.return===null||Wu(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 _i(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=vo));else if(r!==4&&(e=e.child,e!==null))for(_i(e,t,n),e=e.sibling;e!==null;)_i(e,t,n),e=e.sibling}function Si(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(Si(e,t,n),e=e.sibling;e!==null;)Si(e,t,n),e=e.sibling}var at=null,Mn=!1;function Rr(e,t,n){for(n=n.child;n!==null;)Ku(e,t,n),n=n.sibling}function Ku(e,t,n){if(ct&&typeof ct.onCommitFiberUnmount=="function")try{ct.onCommitFiberUnmount(At,n)}catch{}switch(n.tag){case 5:pt||Ss(n,t);case 6:var r=at,s=Mn;at=null,Rr(e,t,n),at=r,Mn=s,at!==null&&(Mn?(e=at,n=n.stateNode,e.nodeType===8?e.parentNode.removeChild(n):e.removeChild(n)):at.removeChild(n.stateNode));break;case 18:at!==null&&(Mn?(e=at,n=n.stateNode,e.nodeType===8?Ll(e.parentNode,n):e.nodeType===1&&Ll(e,n),Ms(e)):Ll(at,n.stateNode));break;case 4:r=at,s=Mn,at=n.stateNode.containerInfo,Mn=!0,Rr(e,t,n),at=r,Mn=s;break;case 0:case 11:case 14:case 15:if(!pt&&(r=n.updateQueue,r!==null&&(r=r.lastEffect,r!==null))){s=r=r.next;do{var l=s,u=l.destroy;l=l.tag,u!==void 0&&((l&2)!==0||(l&4)!==0)&&yi(n,t,u),s=s.next}while(s!==r)}Rr(e,t,n);break;case 1:if(!pt&&(Ss(n,t),r=n.stateNode,typeof r.componentWillUnmount=="function"))try{r.props=n.memoizedProps,r.state=n.memoizedState,r.componentWillUnmount()}catch(d){Qe(n,t,d)}Rr(e,t,n);break;case 21:Rr(e,t,n);break;case 22:n.mode&1?(pt=(r=pt)||n.memoizedState!==null,Rr(e,t,n),pt=r):Rr(e,t,n);break;default:Rr(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 tf),t.forEach(function(r){var s=ff.bind(null,e,r);n.has(r)||(n.add(r),r.then(s,s))})}}function Rn(e,t){var n=t.deletions;if(n!==null)for(var r=0;rs&&(s=u),r&=~l}if(r=s,r=Me()-r,r=(120>r?120:480>r?480:1080>r?1080:1920>r?1920:3e3>r?3e3:4320>r?4320:1960*sf(r/1960))-r,10e?16:e,Tr===null)var r=!1;else{if(e=Tr,Tr=null,Qo=0,(je&6)!==0)throw Error(f(331));var s=je;for(je|=4,H=e.current;H!==null;){var l=H,u=l.child;if((H.flags&16)!==0){var d=l.deletions;if(d!==null){for(var g=0;gMe()-ki?Jr(e,0):ji|=n),$t(e,t)}function ac(e,t){t===0&&((e.mode&1)===0?t=1:(t=er,er<<=1,(er&130023424)===0&&(er=4194304)));var n=Ct();e=lr(e,t),e!==null&&(Qt(e,t,n),$t(e,n))}function df(e){var t=e.memoizedState,n=0;t!==null&&(n=t.retryLane),ac(e,n)}function ff(e,t){var n=0;switch(e.tag){case 13:var r=e.stateNode,s=e.memoizedState;s!==null&&(n=s.retryLane);break;case 19:r=e.stateNode;break;default:throw Error(f(314))}r!==null&&r.delete(t),ac(e,n)}var uc;uc=function(e,t,n){if(e!==null)if(e.memoizedProps!==t.pendingProps||Lt.current)Dt=!0;else{if((e.lanes&n)===0&&(t.flags&128)===0)return Dt=!1,Xd(e,t,n);Dt=(e.flags&131072)!==0}else Dt=!1,Ue&&(t.flags&1048576)!==0&&Ba(t,jo,t.index);switch(t.lanes=0,t.tag){case 2:var r=t.type;$o(e,t),e=t.pendingProps;var s=fs(t,dt.current);ys(t,n),s=ti(null,t,r,e,s,n);var l=ni();return t.flags|=1,typeof s=="object"&&s!==null&&typeof s.render=="function"&&s.$$typeof===void 0?(t.tag=1,t.memoizedState=null,t.updateQueue=null,Ot(r)?(l=!0,_o(t)):l=!1,t.memoizedState=s.state!==null&&s.state!==void 0?s.state:null,Kl(t),s.updater=Do,t.stateNode=s,s._reactInternals=t,ai(t,r,e,n),t=fi(null,t,r,!0,l,n)):(t.tag=0,Ue&&l&&$l(t),kt(null,t,s,n),t=t.child),t;case 16:r=t.elementType;e:{switch($o(e,t),e=t.pendingProps,s=r._init,r=s(r._payload),t.type=r,s=t.tag=pf(r),e=Pn(r,e),s){case 0:t=di(null,t,r,e,n);break e;case 1:t=Ou(null,t,r,e,n);break e;case 11:t=Ru(null,t,r,e,n);break e;case 14:t=Au(null,t,r,Pn(r.type,e),n);break e}throw Error(f(306,r,""))}return t;case 0:return r=t.type,s=t.pendingProps,s=t.elementType===r?s:Pn(r,s),di(e,t,r,s,n);case 1:return r=t.type,s=t.pendingProps,s=t.elementType===r?s:Pn(r,s),Ou(e,t,r,s,n);case 3:e:{if(Du(t),e===null)throw Error(f(387));r=t.pendingProps,l=t.memoizedState,s=l.element,Ya(e,t),Mo(t,r,null,n);var u=t.memoizedState;if(r=u.element,l.isDehydrated)if(l={element:r,isDehydrated:!1,cache:u.cache,pendingSuspenseBoundaries:u.pendingSuspenseBoundaries,transitions:u.transitions},t.updateQueue.baseState=l,t.memoizedState=l,t.flags&256){s=_s(Error(f(423)),t),t=zu(e,t,r,n,s);break e}else if(r!==s){s=_s(Error(f(424)),t),t=zu(e,t,r,n,s);break e}else for(Gt=kr(t.stateNode.containerInfo.firstChild),Kt=t,Ue=!0,En=null,n=Ga(t,null,r,n),t.child=n;n;)n.flags=n.flags&-3|4096,n=n.sibling;else{if(hs(),r===s){t=ar(e,t,n);break e}kt(e,t,r,n)}t=t.child}return t;case 5:return eu(t),e===null&&bl(t),r=t.type,s=t.pendingProps,l=e!==null?e.memoizedProps:null,u=s.children,Tl(r,s)?u=null:l!==null&&Tl(r,l)&&(t.flags|=32),Lu(e,t),kt(e,t,u,n),t.child;case 6:return e===null&&bl(t),null;case 13:return $u(e,t,n);case 4:return Gl(t,t.stateNode.containerInfo),r=t.pendingProps,e===null?t.child=gs(t,null,r,n):kt(e,t,r,n),t.child;case 11:return r=t.type,s=t.pendingProps,s=t.elementType===r?s:Pn(r,s),Ru(e,t,r,s,n);case 7:return kt(e,t,t.pendingProps,n),t.child;case 8:return kt(e,t,t.pendingProps.children,n),t.child;case 12:return kt(e,t,t.pendingProps.children,n),t.child;case 10:e:{if(r=t.type._context,s=t.pendingProps,l=t.memoizedProps,u=s.value,Le(No,r._currentValue),r._currentValue=u,l!==null)if(Nn(l.value,u)){if(l.children===s.children&&!Lt.current){t=ar(e,t,n);break e}}else for(l=t.child,l!==null&&(l.return=t);l!==null;){var d=l.dependencies;if(d!==null){u=l.child;for(var g=d.firstContext;g!==null;){if(g.context===r){if(l.tag===1){g=ir(-1,n&-n),g.tag=2;var w=l.updateQueue;if(w!==null){w=w.shared;var M=w.pending;M===null?g.next=g:(g.next=M.next,M.next=g),w.pending=g}}l.lanes|=n,g=l.alternate,g!==null&&(g.lanes|=n),Wl(l.return,n,t),d.lanes|=n;break}g=g.next}}else if(l.tag===10)u=l.type===t.type?null:l.child;else if(l.tag===18){if(u=l.return,u===null)throw Error(f(341));u.lanes|=n,d=u.alternate,d!==null&&(d.lanes|=n),Wl(u,n,t),u=l.sibling}else u=l.child;if(u!==null)u.return=l;else for(u=l;u!==null;){if(u===t){u=null;break}if(l=u.sibling,l!==null){l.return=u.return,u=l;break}u=u.return}l=u}kt(e,t,s.children,n),t=t.child}return t;case 9:return s=t.type,r=t.pendingProps.children,ys(t,n),s=dn(s),r=r(s),t.flags|=1,kt(e,t,r,n),t.child;case 14:return r=t.type,s=Pn(r,t.pendingProps),s=Pn(r.type,s),Au(e,t,r,s,n);case 15:return Tu(e,t,t.type,t.pendingProps,n);case 17:return r=t.type,s=t.pendingProps,s=t.elementType===r?s:Pn(r,s),$o(e,t),t.tag=1,Ot(r)?(e=!0,_o(t)):e=!1,ys(t,n),ju(t,r,s),ai(t,r,s,n),fi(null,t,r,!0,e,n);case 19:return Uu(e,t,n);case 22:return Iu(e,t,n)}throw Error(f(156,t.tag))};function cc(e,t){return z(e,t)}function mf(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 pn(e,t,n,r){return new mf(e,t,n,r)}function Ti(e){return e=e.prototype,!(!e||!e.isReactComponent)}function pf(e){if(typeof e=="function")return Ti(e)?1:0;if(e!=null){if(e=e.$$typeof,e===ie)return 11;if(e===me)return 14}return 2}function Or(e,t){var n=e.alternate;return n===null?(n=pn(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 Go(e,t,n,r,s,l){var u=2;if(r=e,typeof e=="function")Ti(e)&&(u=1);else if(typeof e=="string")u=5;else e:switch(e){case $e:return Xr(n.children,s,l,t);case We:u=8,s|=8;break;case Je:return e=pn(12,n,t,s|2),e.elementType=Je,e.lanes=l,e;case Ie:return e=pn(13,n,t,s),e.elementType=Ie,e.lanes=l,e;case Fe:return e=pn(19,n,t,s),e.elementType=Fe,e.lanes=l,e;case pe:return Jo(n,s,l,t);default:if(typeof e=="object"&&e!==null)switch(e.$$typeof){case et:u=10;break e;case st:u=9;break e;case ie:u=11;break e;case me:u=14;break e;case Re:u=16,r=null;break e}throw Error(f(130,e==null?e:typeof e,""))}return t=pn(u,n,t,s),t.elementType=e,t.type=r,t.lanes=l,t}function Xr(e,t,n,r){return e=pn(7,e,r,t),e.lanes=n,e}function Jo(e,t,n,r){return e=pn(22,e,r,t),e.elementType=pe,e.lanes=n,e.stateNode={isHidden:!1},e}function Ii(e,t,n){return e=pn(6,e,null,t),e.lanes=n,e}function Li(e,t,n){return t=pn(4,e.children!==null?e.children:[],e.key,t),t.lanes=n,t.stateNode={containerInfo:e.containerInfo,pendingChildren:null,implementation:e.implementation},t}function hf(e,t,n,r,s){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=It(0),this.expirationTimes=It(-1),this.entangledLanes=this.finishedLanes=this.mutableReadLanes=this.expiredLanes=this.pingedLanes=this.suspendedLanes=this.pendingLanes=0,this.entanglements=It(0),this.identifierPrefix=r,this.onRecoverableError=s,this.mutableSourceEagerHydrationData=null}function Oi(e,t,n,r,s,l,u,d,g){return e=new hf(e,t,n,d,g),t===1?(t=1,l===!0&&(t|=8)):t=0,l=pn(3,null,null,t),e.current=l,l.stateNode=e,l.memoizedState={element:r,isDehydrated:n,cache:null,transitions:null,pendingSuspenseBoundaries:null},Kl(l),e}function gf(e,t,n){var r=3"u"||typeof __REACT_DEVTOOLS_GLOBAL_HOOK__.checkDCE!="function"))try{__REACT_DEVTOOLS_GLOBAL_HOOK__.checkDCE(i)}catch(m){console.error(m)}}return i(),bi.exports=Pf(),bi.exports}var kc;function Rf(){if(kc)return rl;kc=1;var i=Mf();return rl.createRoot=i.createRoot,rl.hydrateRoot=i.hydrateRoot,rl}var Af=Rf();const Tf=bc(Af),If="/api";async function Ee(i,m){const f=await fetch(`${If}${i}`,{...m,headers:{"Content-Type":"application/json",...m?.headers??{}}}),N=await f.json();if(!f.ok){const j=N.error?.message??"Ошибка запроса";throw new Error(j)}return N}const ke={async loadSharedConnectionConfig(){return Ee("/llm/shared-connection")},async saveSharedConnectionConfig(i){return Ee("/llm/shared-connection",{method:"POST",body:JSON.stringify({llmProvider:i.llmProvider,model:i.model,baseUrl:i.baseUrl,temperature:i.temperature,maxOutputTokens:i.maxOutputTokens})})},async listModels(i){return Ee("/llm/models",{method:"POST",body:JSON.stringify({llmProvider:i.llmProvider,apiKey:i.apiKey,model:i.model,baseUrl:i.baseUrl})})},async testConnection(i){return Ee("/llm/test-connection",{method:"POST",body:JSON.stringify({llmProvider:i.llmProvider,apiKey:i.apiKey,model:i.model,baseUrl:i.baseUrl})})},async normalize(i){return Ee("/normalize",{method:"POST",body:JSON.stringify({llmProvider:i.connection.llmProvider,apiKey:i.connection.apiKey,model:i.connection.model,baseUrl:i.connection.baseUrl,temperature:i.connection.temperature,maxOutputTokens:i.connection.maxOutputTokens,promptVersion:i.promptVersion,systemPrompt:i.prompts.systemPrompt,developerPrompt:i.prompts.developerPrompt,domainPrompt:i.prompts.domainPrompt,fewShotExamples:i.prompts.fewShotExamples,userQuestion:i.query.userQuestion,context:{period_hint:i.query.periodHint??"",business_context:i.query.businessContext??"",expected_route:i.query.expectedRoute??""},saveAsTestCase:!!i.saveAsTestCase,useMock:!!i.useMock})})},async loadHistory(){return Ee("/history")},async loadTrace(i){return Ee(`/history/${i}`)},async loadPresets(){return Ee("/presets")},async savePreset(i){return Ee("/presets/save",{method:"POST",body:JSON.stringify(i)})},async runEval(i){return Ee("/eval/run",{method:"POST",body:JSON.stringify({normalizeConfig:{llmProvider:i.connection.llmProvider,apiKey:i.connection.apiKey,model:i.connection.model,baseUrl:i.connection.baseUrl,temperature:i.connection.temperature,maxOutputTokens:i.connection.maxOutputTokens,promptVersion:i.promptVersion,systemPrompt:i.prompts.systemPrompt,developerPrompt:i.prompts.developerPrompt,domainPrompt:i.prompts.domainPrompt,fewShotExamples:i.prompts.fewShotExamples},caseIds:i.caseIds,useMock:!!i.useMock,mode:i.mode??"standard",caseSetFile:i.caseSetFile,rawQuestions:i.rawQuestions,eval_target:i.evalTarget,compare_with_report_file:i.compareWithReportFile,analysis_date:i.analysisDate})})},async startEvalRunAsync(i){return Ee("/eval/run-async/start",{method:"POST",body:JSON.stringify({normalizeConfig:{llmProvider:i.connection.llmProvider,apiKey:i.connection.apiKey,model:i.connection.model,baseUrl:i.connection.baseUrl,temperature:i.connection.temperature,maxOutputTokens:i.connection.maxOutputTokens,promptVersion:i.promptVersion,systemPrompt:i.prompts.systemPrompt,developerPrompt:i.prompts.developerPrompt,domainPrompt:i.prompts.domainPrompt,fewShotExamples:i.prompts.fewShotExamples},caseIds:i.caseIds,useMock:!!i.useMock,mode:i.mode??"standard",caseSetFile:i.caseSetFile,rawQuestions:i.rawQuestions,eval_target:i.evalTarget,compare_with_report_file:i.compareWithReportFile,questions:i.questions,analysis_date:i.analysisDate})})},async loadEvalRunAsyncStatus(i){return Ee(`/eval/run-async/${encodeURIComponent(i)}`)},async startRun(){return Ee("/accounting-agent/v1/runs/start",{method:"POST",body:JSON.stringify({initiator:"ndc_operator",source:"gui"})})},async finishRun(i){return Ee("/accounting-agent/v1/runs/finish",{method:"POST",body:JSON.stringify({runId:i,status:"DONE",source:"gui",reason:"Остановлено оператором из GUI"})})},async listRuns(){return Ee("/accounting-agent/v1/runs")},async listResults(){return Ee("/accounting-agent/v1/results")},async runTrace(i){return Ee(`/accounting-agent/v1/trace/run/${i}`)},async sendAssistantMessage(i){return Ee("/assistant/message",{method:"POST",body:JSON.stringify({session_id:i.sessionId??"",mode:"assistant",message:i.userMessage,user_message:i.userMessage,llmProvider:i.connection.llmProvider,apiKey:i.connection.apiKey,model:i.connection.model,baseUrl:i.connection.baseUrl,temperature:i.connection.temperature,maxOutputTokens:i.connection.maxOutputTokens,promptVersion:i.promptVersion??"address_query_runtime_v1",systemPrompt:i.prompts.systemPrompt,developerPrompt:i.prompts.developerPrompt,domainPrompt:i.prompts.domainPrompt,fewShotExamples:i.prompts.fewShotExamples,context:{period_hint:i.context?.periodHint??"",business_context:i.context?.businessContext??""},useMock:!!i.useMock})})},async loadAssistantSession(i){return Ee(`/assistant/session/${i}`)},async loadAssistantAnnotations(i){const m=new URLSearchParams;i?.session_id&&m.set("session_id",i.session_id),typeof i?.limit=="number"&&m.set("limit",String(i.limit));const f=m.toString();return Ee(`/assistant/annotations${f?`?${f}`:""}`)},async saveAssistantAnnotation(i){return Ee("/assistant/annotations",{method:"POST",body:JSON.stringify(i)})},async loadAutoRunsHistory(i){const m=new URLSearchParams;i?.from&&m.set("from",i.from),i?.to&&m.set("to",i.to),i?.target&&m.set("target",i.target),i?.mode&&m.set("mode",i.mode),i?.use_mock&&m.set("use_mock",i.use_mock),i?.prompt_contains&&m.set("prompt_contains",i.prompt_contains),typeof i?.limit=="number"&&m.set("limit",String(i.limit)),typeof i?.scan_limit=="number"&&m.set("scan_limit",String(i.scan_limit));const f=m.toString();return Ee(`/autoruns/history${f?`?${f}`:""}`)},async loadAutoRunDetail(i){return Ee(`/autoruns/history/${encodeURIComponent(i)}`)},async loadAutoRunCaseDialog(i,m){return Ee(`/autoruns/history/${encodeURIComponent(i)}/case/${encodeURIComponent(m)}/dialog`)},async loadAutoRunAnnotations(i){const m=new URLSearchParams;i?.run_id&&m.set("run_id",i.run_id),i?.case_id&&m.set("case_id",i.case_id),typeof i?.min_rating=="number"&&m.set("min_rating",String(i.min_rating)),i?.manual_case_decision&&m.set("manual_case_decision",i.manual_case_decision),typeof i?.limit=="number"&&m.set("limit",String(i.limit));const f=m.toString();return Ee(`/autoruns/annotations${f?`?${f}`:""}`)},async saveAutoRunAnnotation(i){return Ee("/autoruns/annotations",{method:"POST",body:JSON.stringify(i)})},async updateAutoRunAnnotation(i){return Ee(`/autoruns/annotations/${encodeURIComponent(i.annotation_id)}`,{method:"PATCH",body:JSON.stringify({resolved:i.resolved,resolved_by:i.resolved_by})})},async loadAutoRunPostAnalysis(i){const m=new URLSearchParams;i?.run_id&&m.set("run_id",i.run_id),typeof i?.limit_per_queue=="number"&&m.set("limit_per_queue",String(i.limit_per_queue)),typeof i?.annotation_limit=="number"&&m.set("annotation_limit",String(i.annotation_limit)),typeof i?.scan_limit=="number"&&m.set("scan_limit",String(i.scan_limit)),i?.from&&m.set("from",i.from),i?.to&&m.set("to",i.to),i?.target&&m.set("target",i.target),i?.mode&&m.set("mode",i.mode),i?.use_mock&&m.set("use_mock",i.use_mock),i?.prompt_contains&&m.set("prompt_contains",i.prompt_contains);const f=m.toString();return Ee(`/autoruns/post-analysis${f?`?${f}`:""}`)},async loadAutoRunAutogenHistory(i){const m=new URLSearchParams;i?.mode&&m.set("mode",i.mode),typeof i?.limit=="number"&&m.set("limit",String(i.limit));const f=m.toString();return Ee(`/autoruns/autogen/history${f?`?${f}`:""}`)},async loadAutoRunAutogenPersonalityCatalog(){return Ee("/autoruns/autogen/personality-catalog")},async generateAutoRunQuestions(i){return Ee("/autoruns/autogen/generate",{method:"POST",body:JSON.stringify(i)})}};function Yt({value:i}){return o.jsx("pre",{className:"json-view",children:JSON.stringify(i??{},null,2)})}function Tn({title:i,subtitle:m,actions:f,className:N,hideHeader:j,children:T}){return o.jsxs("section",{className:N?`panel-frame ${N}`:"panel-frame",children:[j?null:o.jsxs("header",{className:"panel-header",children:[o.jsxs("div",{children:[o.jsx("h2",{children:i}),m?o.jsx("p",{children:m}):null]}),f?o.jsx("div",{className:"panel-actions",children:f}):null]}),o.jsx("div",{className:"panel-body",children:T})]})}function Lf(i){const m=new Date(i);return Number.isNaN(m.getTime())?i:m.toLocaleString("ru-RU")}function Of({sessionId:i,conversation:m,statusText:f,errorMessage:N,useMock:j,appLogs:T}){const F=m.filter(I=>I.role==="assistant").length,Y=m.filter(I=>I.role==="user").length,U=m.length>0?m[m.length-1]:null;return o.jsxs(Tn,{title:"SAM",subtitle:"System Assistant Monitor: срез по текущей сессии и логам.",children:[o.jsxs("div",{className:"metrics-grid",children:[o.jsxs("div",{children:[o.jsx("span",{children:"session_id"}),o.jsx("strong",{children:i||"новая сессия"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"mock_mode"}),o.jsx("strong",{children:j?"on":"off"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"сообщений пользователя"}),o.jsx("strong",{children:Y})]}),o.jsxs("div",{children:[o.jsx("span",{children:"ответов ассистента"}),o.jsx("strong",{children:F})]}),o.jsxs("div",{children:[o.jsx("span",{children:"статус"}),o.jsx("strong",{children:f||"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"ошибка"}),o.jsx("strong",{children:N||"нет"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"последнее сообщение"}),o.jsx("strong",{children:U?.created_at?Lf(U.created_at):"нет данных"})]})]}),o.jsx("h3",{style:{marginTop:12},children:"Последние системные логи"}),o.jsx(Yt,{value:T.slice(0,120)})]})}const Df=/(?:^|\n)\s*#{0,6}\s*(?:debug_payload_json|technical_breakdown_json|route_summary_json|debug_payload|technical_breakdown)\b/i,zf=[/\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 $f(i){try{return JSON.stringify(i,null,2)}catch{return String(i)}}function Ff(i){const m=String(i??""),f=m.match(Df);return(f?m.slice(0,f.index):m).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(F=>F.trimEnd()).filter(F=>F.trim().length>0).filter(F=>!zf.some(Y=>Y.test(F))).join(` -`).trim()}function Uf(i,m,f="default"){const N=f==="technical",j=[];j.push("# Assistant conversation export"),j.push(`session_id: ${i||"n/a"}`),j.push(`export_mode: ${f}`),j.push(`exported_at: ${new Date().toISOString()}`),j.push("");for(let T=0;T{j.length!==0&&(N.push(j.join(` -`)),j=[])};for(const F of f){const Y=F.trimEnd(),U=Y.trim();if(!U){T();continue}const I=/^Блок\s+\d+\./i.test(U),q=/^\d+\.\s/.test(U);(I||q)&&j.length>0&&T(),j.push(Y)}return T(),N.length>0?N:[i]}function qf(i,m){const f=[],N=/\*\*(.+?)\*\*/g;let j=0,T=0,F;for(;(F=N.exec(i))!==null;)F.index>j&&(f.push(o.jsx("span",{children:i.slice(j,F.index)},`${m}-t-${T}`)),T+=1),f.push(o.jsx("strong",{children:F[1]},`${m}-b-${T}`)),T+=1,j=N.lastIndex;return j0?f:[o.jsx("span",{children:i},`${m}-raw`)]}function Kf(i){const m=i.trimStart();return/^Блок\s+\d+\./i.test(m)?"assistant-msg-line heading":/^\d+\.\s/.test(m)?"assistant-msg-line numbered":/^-\s/.test(m)?"assistant-msg-line bullet":"assistant-msg-line"}function Gf(i,m=40){const f=i.replace(/\s+/g," ").trim();if(f.length<=m)return f;const N=f.split(" ").slice(0,3).join(" ").trim();return N.length>=10&&N.length<=m?`${N}…`:`${f.slice(0,m-1).trimEnd()}…`}function Bc(i){return i.replace(/\*\*(.+?)\*\*/g,"$1").replace(/^\d+\.\s*/,"").trim()}function Jf(i){const m=i.replace(/\r\n?/g,` -`).split(` -`).map(j=>j.trim()).find(Boolean),f=Bc(m??"");return(f.split("|")[0]?.trim()??f).replace(/\s+/g," ").trim()}function Yf(i){const m=i.replace(/\r\n?/g,` -`).split(` -`).map(N=>N.trim()).find(Boolean);return!m||!/^\d+\.\s/.test(m)?!1:Bc(m).includes("|")}function Xf(i,m){const f=m.replace(/\r\n?/g,` -`).replace(/\*\*(.+?)\*\*/g,"$1").split(` -`).map((j,T)=>{const F=j.trim();return T===0?F.replace(/^\d+\.\s*/,""):F}).filter(Boolean).join(" ").replace(/\s+/g," ").trim();if(!f)return null;const N=Jf(m)||f;return{message_id:i.message_id,source_text:f,anchor_text:N,preview_text:Gf(N)}}function Zf(i,m,f,N){return Wf(i.text).map((T,F)=>{const Y=T.split(` -`),U=i.role==="assistant"&&Yf(T),I=U?Xf(i,T):null,q=!!I&&m?.message_id===I?.message_id&&m?.source_text===I?.source_text,D=Y.map((B,oe)=>o.jsx("p",{className:Kf(B),children:qf(B,`line-${F}-${oe}`)},`line-${F}-${oe}`));return!U||!I?o.jsx("div",{className:"assistant-msg-block",children:D},`block-${F}`):o.jsx("div",{className:q?"assistant-msg-block selectable active":"assistant-msg-block selectable",role:"button",tabIndex:0,onClick:()=>{if(q){N();return}f(I)},onKeyDown:B=>{if(!(B.key!=="Enter"&&B.key!==" ")){if(B.preventDefault(),q){N();return}f(I)}},children:D},`block-${F}`)})}function Hc({sessionId:i,conversation:m,inputValue:f,onInputChange:N,selectedContextChip:j,onSelectContextChip:T,onClearContextChip:F,useMock:Y,onUseMockChange:U,onSend:I,onClear:q,busy:D,statusText:B,errorMessage:oe,showCommentAction:Ae=!1,onCommentAssistantMessage:se,isAssistantMessageCommented:te,canCommentAssistantMessage:Te}){const ve=x.useRef(null),G=x.useRef(!0),ae=x.useRef(null),[_e,Oe]=x.useState("idle"),[$e,We]=x.useState("чат");function Je(ie=!1){ve.current&&(ie&&(G.current=!0),ve.current.scrollTop=ve.current.scrollHeight)}x.useEffect(()=>{G.current&&Je()},[m]),x.useEffect(()=>()=>{ae.current!==null&&window.clearTimeout(ae.current)},[]);async function et(ie){if(m.length===0)return;const Ie=Uf(i,m,ie),Fe=await Hf(Ie);We(ie==="technical"?"тех":"чат"),Oe(Fe?"success":"error"),ae.current!==null&&window.clearTimeout(ae.current),ae.current=window.setTimeout(()=>{Oe("idle")},2200)}function st(){if(!ve.current)return;const ie=ve.current,Ie=ie.scrollHeight-ie.scrollTop-ie.clientHeight;G.current=Ie<16}return o.jsx(Tn,{className:"assistant-panel-frame",title:"Режим ассистента",children:o.jsxs("div",{className:"assistant-live-shell",children:[o.jsxs("div",{className:"assistant-toolbar",children:[o.jsxs("div",{className:"assistant-toolbar-actions",children:[o.jsx("button",{type:"button",className:"assistant-copy-btn",onClick:()=>{et("default")},disabled:m.length===0,title:"Экспорт только user-facing чата",children:"Скопировать чат"}),o.jsx("button",{type:"button",className:"assistant-copy-btn",onClick:()=>{et("technical")},disabled:m.length===0,title:"Технический экспорт с debug payload",children:"Скопировать техчат"}),o.jsx("button",{type:"button",className:"assistant-copy-btn",onClick:()=>q(),disabled:D&&m.length===0,children:"Сбросить сессию"})]}),o.jsxs("div",{className:"assistant-toolbar-meta",children:[i?o.jsx("span",{className:"status-chip",children:`session: ${i}`}):null,o.jsxs("div",{className:"assistant-toolbar-meta-right",children:[B?o.jsx("span",{className:"assistant-live-status",children:B}):null,_e==="success"?o.jsxs("span",{className:"assistant-copy-feedback success",children:["Скопировано (",$e,")"]}):null,_e==="error"?o.jsx("span",{className:"assistant-copy-feedback error",children:"Ошибка копирования"}):null]})]}),oe?o.jsx("p",{className:"error-text assistant-toolbar-error",children:oe}):null]}),o.jsx("div",{ref:ve,className:"assistant-chat-list",onScroll:st,children:m.map((ie,Ie)=>{const Fe=ie.role==="assistant"&&Ae&&typeof se=="function"&&(typeof Te=="function"?Te(ie,Ie):!0),me=ie.role==="assistant"&&typeof te=="function"?te(ie,Ie):!1;return o.jsxs("article",{className:`assistant-msg ${ie.role}`,children:[o.jsxs("header",{className:"assistant-msg-head",children:[o.jsxs("div",{className:"assistant-msg-head-main",children:[o.jsx("strong",{children:bf(ie.role)}),o.jsx("span",{children:Bf(ie.created_at)})]}),ie.role==="assistant"&&Ae?o.jsx("div",{className:"assistant-msg-head-actions",children:o.jsx("button",{type:"button",className:me?"autoruns-comment-icon assistant-comment-btn commented":"autoruns-comment-icon assistant-comment-btn",onClick:()=>se?.(ie,Ie),disabled:!Fe,title:Fe?"Комментировать ответ ассистента":"Комментарий недоступен для этого сообщения","aria-label":Fe?"Комментировать ответ ассистента":"Комментарий недоступен для этого сообщения",children:o.jsx(Vf,{commented:me})})}):null]}),o.jsx("div",{className:"assistant-msg-body",children:Zf(ie,j,T,F)}),ie.role==="assistant"&&ie.debug?o.jsxs("details",{className:"assistant-debug",children:[o.jsx("summary",{children:"Показать технический разбор"}),o.jsx(Yt,{value:ie.debug})]}):null]},ie.message_id)})}),o.jsxs("div",{className:"assistant-compose",children:[j?o.jsxs("div",{className:"assistant-compose-context",children:[o.jsx("span",{className:"assistant-compose-context-label",children:"Выбранный объект"}),o.jsxs("div",{className:"assistant-compose-context-pill",title:j.source_text,children:[o.jsx("span",{className:"assistant-compose-context-pill-text",children:j.preview_text}),o.jsx("button",{type:"button",className:"assistant-compose-context-clear",onClick:F,"aria-label":"Убрать выбранный объект",title:"Убрать выбранный объект",children:"×"})]})]}):null,o.jsxs("label",{className:"full-width",children:["Сообщение",o.jsx("textarea",{className:"assistant-input-textarea",value:f,onChange:ie=>N(ie.target.value),rows:4,placeholder:j?"Продолжите вопрос по выбранному объекту...":"Введите вопрос к данным компании..."})]}),o.jsxs("div",{className:"button-row assistant-send-row",children:[o.jsxs("label",{className:"checkbox-row",children:[o.jsx("input",{type:"checkbox",checked:Y,onChange:ie=>U(ie.target.checked)}),"Mock-режим"]}),o.jsx("button",{type:"button",className:"assistant-send-btn",onClick:()=>{Je(!0),I()},disabled:D||!f.trim(),children:D?"Выполняю...":"Отправить"})]})]})]})})}const Vi={fromLocal:"",toLocal:"",target:"all",mode:"all",useMock:"any",promptContains:"",limit:120},sl="needs_dialog_policy_fix",ht="__all__",ul="__live__:",Cc="ndc_autoruns_ui_config_v1",Nc="ndc-autoruns-save",Qi=["Анализ запроса","Получение данных","Подготовка ответа"];function em(i,m){const f=i.trim();if(!f)return"";if(!m)return f;const N=f.toLowerCase(),j=m.anchor_text.trim(),T=j.toLowerCase();return T&&N.includes(T)?f:`По выбранному объекту "${j}": ${f}`}const Xi=[{id:"general",label:"Общий контур",domain:"",defaultPrompt:"Генерируй реалистичные живые вопросы бухгалтера по 1С. Добавляй разговорные формулировки и опечатки, но сохраняй бизнес-смысл."}];function tm(i=Xi){return i.reduce((m,f)=>(m[f.id]=f.defaultPrompt,m),{})}const Ec={mode:"codex_creative",count:24,personalityId:"general",personalityPrompts:tm(),persistToEvalCases:!0,generatedBy:"manual_reviewer"};function Wi(i){const m=String(i??"").trim();return/^\d{4}-\d{2}-\d{2}$/.test(m)?m:""}function Pc(i){const m=typeof i=="number"&&Number.isFinite(i)?Math.trunc(i):160;return Math.max(110,Math.min(520,m))}function nm(i){const m=i.getFullYear(),f=String(i.getMonth()+1).padStart(2,"0"),N=String(i.getDate()).padStart(2,"0"),j=String(i.getHours()).padStart(2,"0"),T=String(i.getMinutes()).padStart(2,"0");return`${m}-${f}-${N}T${j}:${T}`}function Mc(){const i=new Date;return i.setDate(i.getDate()-14),nm(i)}function ol(i){if(!i.trim())return;const m=Date.parse(i);if(Number.isFinite(m))return new Date(m).toISOString()}function cr(i){if(!i)return"нет данных";const m=Date.parse(i);return Number.isFinite(m)?new Date(m).toLocaleString("ru-RU"):i}function rm(i,m){return m<=0?0:Math.max(0,Math.min(100,Number((i/m*100).toFixed(1))))}function ll(i){return typeof i!="number"?"нет данных":`${i.toFixed(1)}%`}function sm(i){return i==="assistant_stage1"?"assistant/s1":i==="assistant_stage2"?"assistant/s2":i==="assistant_p0"?"assistant/p0":i}function Rc(i){return i==="up"?"Рост":i==="down"?"Регресс":"Без изменений"}function om(i,m){return i.find(f=>f.case_id===m)??null}function qi(i){const m=Math.max(1,Math.min(5,Math.round(i)));return`${"●".repeat(m)}${"○".repeat(5-m)}`}function Ac(i){return i.length===0?o.jsx("p",{className:"muted",children:"Покрытие доменов пока не сформировано."}):o.jsx("div",{className:"autoruns-coverage-list",children:i.map(m=>{const f=rm(m.closed_cases,m.total_cases);return o.jsxs("div",{className:"autoruns-coverage-item",children:[o.jsxs("div",{className:"autoruns-coverage-head",children:[o.jsx("strong",{children:m.domain}),o.jsxs("span",{children:[m.closed_cases,"/",m.total_cases," (",f,"%)"]})]}),o.jsx("div",{className:"autoruns-coverage-bar",children:o.jsx("div",{style:{width:`${f}%`}})})]},m.domain)})})}function al(i){return`${ul}${i}`}function Cs(i){return i.startsWith(ul)}function Tc(i){return i.startsWith(ul)?i.slice(ul.length):""}function Vc(i){const m=i.report_summary?.run_timestamp??i.created_at,f=Math.max(0,i.total_cases-i.completed_cases);return{run_id:al(i.job_id),eval_target:i.eval_target,run_timestamp:m,mode:"single-pass-strict",llm_provider:null,model:null,use_mock:null,analysis_date:i.report_summary?.analysis_date??i.analysis_date??null,prompt_version:null,schema_version:null,suite_id:i.case_set_file,cases_total:i.total_cases,requests_total:null,report_path:`async_job:${i.job_id}`,score_index:i.report_summary?.score_index??null,blocking_failures:0,quality_failures:0,closed_cases:i.completed_cases,open_cases:f,domain_coverage:[{domain:"runtime",total_cases:i.total_cases,closed_cases:i.completed_cases}]}}function ro(i,m){const f=Vc(i),N=i.cases.map(q=>({case_id:q.case_id,domain:null,query_class:null,status:q.status==="completed"?"closed":q.status==="failed"?"open":"unknown",score_index:null,trace_id:null,reply_type:null,session_id:`${i.run_id}-${q.case_id}`,dialog_available:q.messages.length>0,commented_count:0,latest_annotation_at:null,avg_rating:null,checks:null,metric_subscores:null})),T=m!==ht&&N.some(q=>q.case_id===m)?m:N.length>0?ht:"",F={ok:!0,run:f,coverage:{closed_cases:i.completed_cases,open_cases:Math.max(0,i.total_cases-i.completed_cases),domain_coverage:[{domain:"runtime",total_cases:i.total_cases,closed_cases:i.completed_cases}]},cases:N,annotations_summary:{total:0},report:i.report_summary?{run_id:i.report_summary.run_id,run_timestamp:i.report_summary.run_timestamp,score_index:i.report_summary.score_index,cases_total:i.report_summary.cases_total,analysis_date:i.report_summary.analysis_date??i.analysis_date??null}:{}},Y=[];let U=0;if(T===ht)for(const q of i.cases)for(let D=0;DD.case_id===T)??null;for(let D=0;D<(q?.messages.length??0);D+=1){const B=q?.messages[D];B&&Y.push({...B,message_index:D,case_id:T,case_message_index:D,commented:!1,annotation:null})}}const I={ok:!0,run_id:f.run_id,case_id:T,source:"assistant_session",session_id:T===ht?`${i.run_id}::__all__`:`${i.run_id}-${T}`,messages:Y,decomposition:[],assistant_mode:{status:i.status,completed_cases:i.completed_cases,total_cases:i.total_cases},annotations:[]};return{detail:F,dialog:I,caseId:T}}function lm({commented:i}){const m=i?"comment-icon-svg commented":"comment-icon-svg";return o.jsx("svg",{className:m,viewBox:"0 0 24 24","aria-hidden":"true",focusable:"false",children:o.jsx("path",{d:"M5 6.5h14v9H11.5l-4.5 3v-3H5z"})})}function Ic({resolved:i}){return o.jsxs("svg",{className:i?"resolve-icon-svg resolved":"resolve-icon-svg",viewBox:"0 0 16 16","aria-hidden":"true",focusable:"false",children:[o.jsx("circle",{cx:"8",cy:"8",r:"6.2"}),i?o.jsx("path",{d:"M5.1 8.2 7.2 10.3 11 6.5"}):null]})}function im(){return o.jsxs("svg",{className:"autoruns-copy-icon-svg",viewBox:"0 0 24 24","aria-hidden":"true",focusable:"false",children:[o.jsx("rect",{x:"9",y:"9",width:"11",height:"11",rx:"2.2"}),o.jsx("path",{d:"M15 7V5.8a1.8 1.8 0 0 0-1.8-1.8H5.8A1.8 1.8 0 0 0 4 5.8v7.4A1.8 1.8 0 0 0 5.8 15H7"})]})}function am({connection:i,prompts:m,assistantPromptVersion:f,decompositionPromptVersion:N,showAssistantMode:j,showDecompositionMode:T,showProgressMode:F,showCommentsMode:Y,onLog:U}){const[I,q]=x.useState({...Vi,fromLocal:Mc()}),[D,B]=x.useState(""),[oe,Ae]=x.useState(null),[se,te]=x.useState(null),[Te,ve]=x.useState(null),[G,ae]=x.useState([]),[_e,Oe]=x.useState("all"),[$e,We]=x.useState(!1),[Je,et]=x.useState(null),[st,ie]=x.useState([]),[Ie,Fe]=x.useState(""),[me,Re]=x.useState(""),[pe,O]=x.useState(""),[W,$]=x.useState(Xi),[h,k]=x.useState(Ec),[J,ue]=x.useState([]),[de,fe]=x.useState(""),[Se,ye]=x.useState([]),[ee,Pe]=x.useState(null),[Ut,Kn]=x.useState(null),[In,Ln]=x.useState(!1),[fr,Xt]=x.useState(!1),[Zt,Gn]=x.useState(!1),[hn,bt]=x.useState(!1),[en,gn]=x.useState(!1),[mr,On]=x.useState(!1),[pr,Dn]=x.useState(!1),[tn,nn]=x.useState(!1),[hr,Bt]=x.useState(""),[vn,be]=x.useState(""),[ot,yn]=x.useState(""),[xn,rn]=x.useState([]),[sn,Nt]=x.useState([]),[zn,on]=x.useState(""),[Ht,gt]=x.useState(null),[Jn,qe]=x.useState(!1),[gr,_n]=x.useState(!1),[vr,ln]=x.useState(""),[yr,Et]=x.useState(""),[Yn,vt]=x.useState(String(Vi.limit)),[Zr,yt]=x.useState(String(Ec.count)),[Pt,$n]=x.useState(160),[ce,lt]=x.useState({open:!1,caseId:"",caseMessageIndex:-1,messageIndex:-1,rating:3,comment:"",manualCaseDecision:sl,annotationAuthor:"manual_reviewer",saving:!1,error:""}),[xe,xt]=x.useState({open:!1,messageIndex:-1,rating:3,comment:"",annotationAuthor:"manual_reviewer",saving:!1,error:""}),Mt=x.useRef(!1),Ce=x.useRef(null),it=x.useMemo(()=>W.find(a=>a.id===h.personalityId)??W[0]??Xi[0],[h.personalityId,W]),Rt=x.useMemo(()=>J.find(a=>a.generation_id===de)??J[0]??null,[J,de]),Fn=se?om(se.cases,pe):null,Vt=x.useMemo(()=>$e?G.filter(a=>!a.resolved):G,[G,$e]),z=Vt.find(a=>a.annotation_id===Ie)??null,zr=Te?.messages.find(a=>a.message_index===ce.messageIndex)??null,xr=x.useMemo(()=>{if(!Te||ce.messageIndex<0)return null;for(let a=ce.messageIndex-1;a>=0;a-=1){const c=Te.messages[a];if(c?.role==="user")return c}return null},[ce.messageIndex,Te]),Xn=x.useMemo(()=>{const a=new Map;for(const c of sn)c.message_id&&a.set(c.message_id,c);return a},[sn]),Me=xe.messageIndex>=0?xn[xe.messageIndex]??null:null,$r=x.useMemo(()=>{if(xe.messageIndex<0)return null;for(let a=xe.messageIndex-1;a>=0;a-=1){const c=xn[a];if(c?.role==="user")return c}return null},[xe.messageIndex,xn]),_t=x.useMemo(()=>{const a=Vt.map(y=>({source:"autorun",key:`autorun:${y.annotation_id}`,updated_at:y.updated_at,rating:y.rating,autorun:y,assistant:null})),c=sn.map(y=>({source:"assistant_live",key:`assistant:${y.annotation_id}`,updated_at:y.updated_at,rating:y.rating,autorun:null,assistant:y}));return[...a,...c].sort((y,A)=>Date.parse(A.updated_at)-Date.parse(y.updated_at))},[sn,Vt]),_r=x.useMemo(()=>{if(_t.length===0)return null;const a=_t.reduce((c,y)=>c+y.rating,0)/_t.length;return Number(a.toFixed(2))},[_t]),Sn=x.useMemo(()=>{const a=[...oe?.items??[]];return ee&&a.unshift(Vc(ee)),me&&!a.some(c=>c.run_id===me)&&se?.run&&a.unshift(se.run),a},[ee,oe?.items,se?.run,me]),le=x.useCallback(a=>{U?.(`[autoruns] ${a}`)},[U]),Un=x.useCallback(async a=>{const c=String(a??"").trim();if(!c){Nt([]);return}try{const y=await ke.loadAssistantAnnotations({session_id:c,limit:400});Nt(y.items??[])}catch(y){const A=y instanceof Error?y.message:String(y);le(`Assistant live annotations load error: ${A}`)}},[le]),At=x.useCallback(a=>{xt(c=>c.saving&&!a?.force?c:{open:!1,messageIndex:-1,rating:3,comment:"",annotationAuthor:"manual_reviewer",saving:!1,error:""})},[]),ct=x.useCallback(async(a,c)=>{a.stopPropagation(),a.preventDefault();const y=String(c??"").trim();if(y)try{if(navigator?.clipboard?.writeText)await navigator.clipboard.writeText(y);else{const A=document.createElement("textarea");A.value=y,A.setAttribute("readonly","true"),A.style.position="fixed",A.style.opacity="0",document.body.appendChild(A),A.select(),document.execCommand("copy"),document.body.removeChild(A)}le(`run id copied: ${y}`)}catch(A){const he=A instanceof Error?A.message:String(A);be(`Копирование run id: ${he}`),le(`copy run id error: ${he}`)}},[le]);function es(){let a=0;ln(Qi[0]);const c=window.setInterval(()=>{a=Math.min(a+1,Qi.length-1),ln(Qi[a])},650);return()=>window.clearInterval(c)}const St=x.useCallback(()=>{yn(""),rn([]),Nt([]),on(""),gt(null),ln(""),Et(""),At({force:!0}),le("Live-чат ассистента в истории автопрогонов сброшен.")},[At,le]),Ns=x.useCallback(async()=>{const a=em(zn,Ht);if(!a)return;_n(!0),Et(""),on(""),rn(y=>[...y,{message_id:`autoruns-live-${Date.now()}`,session_id:ot||"pending",role:"user",text:a,reply_type:null,created_at:new Date().toISOString(),trace_id:null,debug:null}]);const c=es();try{const y=await ke.sendAssistantMessage({connection:i,prompts:m,userMessage:a,sessionId:ot||void 0,promptVersion:f,useMock:Jn});yn(y.session_id),rn(y.conversation),await Un(y.session_id),ln("Ответ готов"),le(`Live-ответ ассистента получен: trace=${y.debug.trace_id}`)}catch(y){const A=y instanceof Error?y.message:String(y);Et(A),ln("Ошибка ассистента"),le(`Live-чат ассистента: ошибка отправки сообщения: ${A}`)}finally{c(),_n(!1)}},[zn,Ht,ot,Jn,f,i,Un,le,m]),ts=x.useCallback(a=>{const c=a.trim();if(!c){vt(String(I.limit));return}if(!/^\d+$/.test(c)){vt(String(I.limit));return}const y=Number.parseInt(c,10);if(!Number.isFinite(y)){vt(String(I.limit));return}const A=Math.max(1,Math.min(500,y));A!==I.limit&&q(he=>({...he,limit:A})),vt(String(A))},[I.limit]),ns=x.useCallback(a=>{const c=a.trim();if(!c){yt(String(h.count));return}if(!/^\d+$/.test(c)){yt(String(h.count));return}const y=Number.parseInt(c,10);if(!Number.isFinite(y)){yt(String(h.count));return}const A=Math.max(1,Math.min(200,y));A!==h.count&&k(he=>({...he,count:A})),yt(String(A))},[h.count]),Zn=x.useCallback(a=>{$n(Pc(a))},[]),er=x.useCallback(a=>{const c=a.currentTarget.offsetHeight;Number.isFinite(c)&&c>0&&Zn(c)},[Zn]),Tt=x.useCallback(async()=>{nn(!0);try{const a=await ke.loadAutoRunAnnotations({limit:800,manual_case_decision:_e});ae(a.items),et(a.manual_case_decision_schema??null),ie(a.available_manual_case_decisions??[]),Fe(c=>a.items.length===0?"":a.items.some(y=>y.annotation_id===c)?c:a.items[0].annotation_id)}catch(a){le(`Annotations load error: ${a instanceof Error?a.message:String(a)}`)}finally{nn(!1)}},[_e,le]),wt=x.useCallback(async()=>{bt(!0);try{const a=await ke.loadAutoRunAutogenHistory({limit:180});ue(a.items)}catch(a){le(`Autogen history load error: ${a instanceof Error?a.message:String(a)}`)}finally{bt(!1)}},[le]),Fr=x.useCallback(async()=>{try{const c=(await ke.loadAutoRunAutogenPersonalityCatalog()).items.map(y=>({id:String(y.id??"").trim(),label:String(y.label??"").trim(),domain:typeof y.domain=="string"?y.domain.trim():"",defaultPrompt:String(y.default_prompt??"").trim()})).filter(y=>y.id.length>0&&y.label.length>0);if(c.length===0)return;$(c.map(y=>({id:y.id,label:y.label,domain:y.domain||"",defaultPrompt:y.defaultPrompt||"Генерируй реалистичные вопросы бухгалтера по выбранному профилю. Не выдумывай непокрытые возможности."})))}catch(a){le(`Autogen personality catalog load error: ${a instanceof Error?a.message:String(a)}`)}},[le]),jt=x.useCallback(async()=>{Gn(!0);try{const a=await ke.loadAutoRunPostAnalysis({run_id:me&&!Cs(me)?me:void 0,limit_per_queue:30,annotation_limit:1500,from:ol(I.fromLocal),to:ol(I.toLocal),target:I.target,mode:I.mode,use_mock:I.useMock,prompt_contains:I.promptContains.trim()||void 0});Kn(a)}catch(a){le(`Post-analysis load error: ${a instanceof Error?a.message:String(a)}`),Kn(null)}finally{Gn(!1)}},[I.fromLocal,I.mode,I.promptContains,I.target,I.toLocal,I.useMock,le,me]),bn=x.useCallback(async()=>{Ln(!0),be("");try{const a=h.personalityPrompts[h.personalityId]??"",c=[m.systemPrompt,m.developerPrompt,m.domainPrompt,m.schemaNotes,m.fewShotExamples].join(` -`).slice(0,900),y=await ke.generateAutoRunQuestions({mode:h.mode,count:h.count,domain:it.domain||void 0,persist_to_eval_cases:h.persistToEvalCases,generated_by:h.generatedBy.trim()||void 0,llm:{llm_provider:i.llmProvider,api_key:i.apiKey,model:i.model,base_url:i.baseUrl,temperature:i.temperature,max_output_tokens:i.maxOutputTokens},context:{llm_provider:i.llmProvider,model:i.model,assistant_prompt_version:f,decomposition_prompt_version:N,prompt_fingerprint:c,autogen_personality_id:it.id,autogen_personality_prompt:a.trim()||void 0}});le(`Generated ${y.generation.count} questions (${y.generation.mode}) id=${y.generation.generation_id}`+(y.generation.saved_case_set_file?` saved=${y.generation.saved_case_set_file}`:"")),fe(y.generation.generation_id),ye([...y.generation.questions??[]]),await wt()}catch(a){const c=a instanceof Error?a.message:String(a);be(`Автогенерация: ${c}`),le(`Autogen generate error: ${c}`)}finally{Ln(!1)}},[f,h.count,h.generatedBy,h.mode,h.personalityId,h.personalityPrompts,h.persistToEvalCases,i.apiKey,i.baseUrl,i.llmProvider,i.maxOutputTokens,i.model,i.temperature,N,wt,le,m.developerPrompt,m.domainPrompt,m.fewShotExamples,m.schemaNotes,m.systemPrompt,it.domain,it.id]),Bn=x.useCallback(async(a,c)=>{if(Cs(a)){const y=Tc(a);if(ee&&ee.job_id===y){const A=ro(ee,c);Re(a),O(A.caseId),ve(A.dialog);return}ve(null);return}Dn(!0);try{const y=await ke.loadAutoRunCaseDialog(a,c);ve(y)}catch(y){const A=y instanceof Error?y.message:String(y);be(`Диалог кейса: ${A}`),ve(null),le(`Dialog load error for ${a}/${c}: ${A}`)}finally{Dn(!1)}},[ee,le]),It=x.useCallback(async(a,c)=>{if(Cs(a)){const y=Tc(a);if(ee&&ee.job_id===y){const A=ro(ee,c??ht);Re(a),O(A.caseId),te(A.detail),ve(A.dialog);return}Re(a),O(""),te(null),ve(null);return}On(!0);try{const y=await ke.loadAutoRunDetail(a);te(y);const A=(c&&(c===ht||y.cases.some(he=>he.case_id===c))?c:"")||(y.cases.length>0?ht:"")||"";Re(a),O(A),A?await Bn(a,A):ve(null)}catch(y){const A=y instanceof Error?y.message:String(y);be(`Детализация прогона: ${A}`),te(null),ve(null),le(`Run detail load error for ${a}: ${A}`)}finally{On(!1)}},[ee,Bn,le]),Qt=x.useCallback(async a=>{gn(!0),be("");try{const c=await ke.loadAutoRunsHistory({from:ol(I.fromLocal),to:ol(I.toLocal),target:I.target,mode:I.mode,use_mock:I.useMock,prompt_contains:I.promptContains.trim()||void 0,limit:I.limit});if(Ae(c),c.items.length===0){Re(""),O(""),te(null),ve(null);return}const y=a?.keepSelection??!0,A=a?.preferredRunId??"",he=a?.preferredCaseId??"",Be=y&&A&&c.items.some(an=>an.run_id===A)?A:c.items[0].run_id;await It(Be,y?he:void 0),jt()}catch(c){const y=c instanceof Error?c.message:String(c);be(`История прогонов: ${y}`),le(`History load error: ${y}`)}finally{gn(!1)}},[I.fromLocal,I.limit,I.mode,I.promptContains,I.target,I.toLocal,I.useMock,jt,It,le]),Wt=x.useCallback(()=>{Ce.current!==null&&(window.clearTimeout(Ce.current),Ce.current=null)},[]),tr=x.useCallback(async a=>{try{const c=await ke.loadEvalRunAsyncStatus(a);Pe(c.job);const y=al(a);if(me===y){const A=ro(c.job,pe||ht);te(A.detail),ve(A.dialog),O(A.caseId)}if(c.job.status==="completed"){Wt(),Xt(!1);const A=c.job.report_summary?.run_id??c.job.run_id;await Qt({keepSelection:!0,preferredRunId:A||me,preferredCaseId:ht}),await wt(),Pe(null);return}if(c.job.status==="failed"){Wt(),Xt(!1),be(`Запуск прогонов: ${c.job.error??"неизвестная ошибка"}`),le(`Autogen async run failed: ${c.job.error??"unknown error"}`);return}Wt(),Ce.current=window.setTimeout(()=>{tr(a)},500)}catch(c){Wt(),Xt(!1);const y=c instanceof Error?c.message:String(c);be(`Запуск прогонов: ${y}`),le(`Autogen async status error: ${y}`)}},[wt,Qt,le,pe,me,Wt]),we=x.useCallback(async()=>{Wt(),Xt(!0),be("");try{const a=Rt;if(!a)throw new Error("История автогенерации пуста. Сначала сгенерируйте пачку вопросов.");const c=Se.map(Sr=>Sr.trim()).filter(Sr=>Sr.length>0);if(c.length===0)throw new Error("Нет вопросов для запуска: список пустой после ручного редактирования.");const y=I.useMock==="true",A=Wi(D),Be=(await ke.startEvalRunAsync({connection:i,prompts:m,promptVersion:f,mode:"single-pass-strict",caseSetFile:a.saved_case_set_file??void 0,useMock:y,evalTarget:"assistant_stage1",questions:c,analysisDate:A||void 0})).job;Pe(Be);const an=al(Be.job_id),Cn=ro(Be,ht);Re(an),O(Cn.caseId),te(Cn.detail),ve(Cn.dialog),le(`Запущен async-прогон job=${Be.job_id}, run_id=${Be.run_id}, вопросов=${c.length}`+(a.saved_case_set_file?`, base_case_set=${a.saved_case_set_file}`:"")+(A?`, analysis_date=${A}`:", analysis_date=current_state")),tr(Be.job_id)}catch(a){const c=a instanceof Error?a.message:String(a);be(`Запуск прогонов: ${c}`),le(`Autogen run error: ${c}`),Xt(!1)}},[D,f,i,Se,I.useMock,le,tr,m,Rt,Wt]),p=x.useCallback(a=>{if(a.role!=="assistant")return;const c=a.case_id??pe,y=a.case_message_index??a.message_index;lt({open:!0,caseId:c,caseMessageIndex:y,messageIndex:a.message_index,rating:a.annotation?.rating??3,comment:a.annotation?.comment??"",manualCaseDecision:a.annotation?.manual_case_decision??sl,annotationAuthor:a.annotation?.annotation_author??h.generatedBy,saving:!1,error:""})},[h.generatedBy,pe]),E=x.useCallback(a=>{lt(c=>c.saving&&!a?.force?c:{open:!1,caseId:"",caseMessageIndex:-1,messageIndex:-1,rating:3,comment:"",manualCaseDecision:sl,annotationAuthor:h.generatedBy,saving:!1,error:""})},[h.generatedBy]),C=x.useCallback(async()=>{const a=me,c=ce.caseId,y=ce.caseMessageIndex;if(!(!a||!c||y<0)){if(Cs(a)){lt(A=>({...A,error:"Комментарий можно сохранить после завершения прогона."}));return}if(!ce.comment.trim()){lt(A=>({...A,error:"Добавьте комментарий."}));return}lt(A=>({...A,saving:!0,error:""}));try{await ke.saveAutoRunAnnotation({run_id:a,case_id:c,message_index:y,rating:ce.rating,comment:ce.comment.trim(),manual_case_decision:ce.manualCaseDecision,annotation_author:ce.annotationAuthor.trim()||void 0}),E({force:!0}),Promise.all([It(a,pe),Tt(),jt()]).catch(A=>{const he=A instanceof Error?A.message:String(A);be(`Обновление после комментария: ${he}`),le(`Comment refresh error: ${he}`)})}catch(A){lt(he=>({...he,saving:!1,error:A instanceof Error?A.message:String(A)}))}}},[E,ce.annotationAuthor,ce.caseId,ce.caseMessageIndex,ce.comment,ce.manualCaseDecision,ce.rating,Tt,jt,It,le,pe,me]),ne=x.useCallback(a=>a.role==="assistant",[]),Ke=x.useCallback(a=>a.role==="assistant"&&Xn.has(a.message_id),[Xn]),Hn=x.useCallback((a,c)=>{if(a.role!=="assistant")return;const y=ot.trim(),A=String(a.session_id??"").trim();if(!(y||A)){Et("Сначала получите ответ ассистента в активной сессии.");return}!y&&A&&yn(A);const Be=Xn.get(a.message_id)??null;Et(""),xt({open:!0,messageIndex:c,rating:Be?.rating??3,comment:Be?.comment??"",annotationAuthor:Be?.annotation_author??"manual_reviewer",saving:!1,error:""})},[Xn,ot]),Es=x.useCallback(async()=>{if(xe.messageIndex<0)return;if(!xe.comment.trim()){xt(y=>({...y,error:"Добавьте комментарий."}));return}const a=xn[xe.messageIndex]??null,c=ot.trim()||(a?.role==="assistant"?String(a.session_id??"").trim():"");if(!c){xt(y=>({...y,error:"Сессия ассистента не найдена."}));return}xt(y=>({...y,saving:!0,error:""}));try{const y=await ke.saveAssistantAnnotation({session_id:c,message_index:xe.messageIndex,rating:xe.rating,comment:xe.comment.trim(),annotation_author:xe.annotationAuthor.trim()||void 0});Nt(A=>{const he=[...A],Be=he.findIndex(an=>an.annotation_id===y.annotation.annotation_id);return Be>=0?he[Be]=y.annotation:he.unshift(y.annotation),he.sort((an,Cn)=>Date.parse(Cn.updated_at)-Date.parse(an.updated_at))}),At({force:!0})}catch(y){const A=y instanceof Error?y.message:String(y);xt(he=>({...he,saving:!1,error:A}))}},[xe.annotationAuthor,xe.comment,xe.messageIndex,xe.rating,xn,ot,At]),Ur=x.useCallback(a=>{ae(c=>c.map(y=>y.annotation_id===a.annotation_id?{...y,...a}:y)),ve(c=>c&&{...c,annotations:c.annotations.map(y=>y.annotation_id===a.annotation_id?a:y),messages:c.messages.map(y=>!y.annotation||y.annotation.annotation_id!==a.annotation_id?y:{...y,commented:!0,annotation:a})})},[]),wn=x.useCallback(async(a,c)=>{if(a.annotation_id){if(Cs(a.run_id)){be("Статус выполнения можно менять только для завершённых прогонов.");return}Bt(a.annotation_id);try{const y=await ke.updateAutoRunAnnotation({annotation_id:a.annotation_id,resolved:c,resolved_by:h.generatedBy||void 0});Ur(y.annotation),jt()}catch(y){const A=y instanceof Error?y.message:String(y);be(`Смена статуса кейса: ${A}`),le(`Annotation resolve toggle error: ${A}`)}finally{Bt("")}}},[Ur,h.generatedBy,jt,le]),jn=x.useCallback(async a=>{Fe(a.annotation_id),await It(a.run_id,a.case_id),oe?.items.some(c=>c.run_id===a.run_id)||be("Комментарий относится к прогону вне текущего фильтра. Детали загружены напрямую.")},[oe?.items,It]);x.useEffect(()=>{Mt.current||(Mt.current=!0,Qt({keepSelection:!1}),wt(),Fr(),jt())},[wt,Fr,Qt,jt]),x.useEffect(()=>{Mt.current&&Tt()},[_e,Tt]),x.useEffect(()=>{Fe(a=>Vt.length===0?"":Vt.some(c=>c.annotation_id===a)?a:Vt[0].annotation_id)},[Vt]),x.useEffect(()=>{fe(a=>J.length===0?"":a&&J.some(c=>c.generation_id===a)?a:J[0].generation_id)},[J]),x.useEffect(()=>{if(!Rt){ye([]);return}ye([...Rt.questions])},[Rt?.generation_id]),x.useEffect(()=>{vt(String(I.limit))},[I.limit]),x.useEffect(()=>{yt(String(h.count))},[h.count]),x.useEffect(()=>{if(!ot.trim()){Nt([]);return}Un(ot)},[ot,Un]),x.useEffect(()=>{if(!ee)return;const a=al(ee.job_id);if(me!==a)return;const c=ro(ee,pe||ht);te(c.detail),ve(c.dialog),O(c.caseId)},[ee,pe,me]),x.useEffect(()=>()=>{Wt()},[Wt]),x.useEffect(()=>{W.length!==0&&k(a=>{let c=!1;const y={...a.personalityPrompts};for(const he of W)(typeof y[he.id]!="string"||y[he.id].trim().length===0)&&(y[he.id]=he.defaultPrompt,c=!0);let A=a.personalityId;return W.some(he=>he.id===a.personalityId)||(A=W[0].id,c=!0),c?{...a,personalityId:A,personalityPrompts:y}:a})},[W]),x.useEffect(()=>{const a=localStorage.getItem(Cc);if(a)try{const c=JSON.parse(a);if(c.filters){const y=c.filters;q(A=>({...A,...y,limit:typeof y.limit=="number"?Math.max(1,Math.min(500,y.limit)):A.limit}))}typeof c.analysisDate=="string"&&B(Wi(c.analysisDate)),typeof c.autogenPersonalityPromptHeight=="number"&&$n(Pc(c.autogenPersonalityPromptHeight)),c.autoGenSettings&&k(y=>{const A={...y.personalityPrompts},he=c.autoGenSettings?.personalityPrompts??{};for(const[an,Cn]of Object.entries(he))typeof Cn=="string"&&an.trim().length>0&&(A[an.trim()]=Cn);const Be=typeof c.autoGenSettings?.personalityId=="string"&&c.autoGenSettings.personalityId.trim().length>0?c.autoGenSettings.personalityId.trim():y.personalityId;return{...y,mode:c.autoGenSettings?.mode==="codex_creative"||c.autoGenSettings?.mode==="qwen_seed"?c.autoGenSettings.mode:y.mode,count:typeof c.autoGenSettings?.count=="number"?Math.max(1,Math.min(200,c.autoGenSettings.count)):y.count,personalityId:Be,personalityPrompts:A,persistToEvalCases:typeof c.autoGenSettings?.persistToEvalCases=="boolean"?c.autoGenSettings.persistToEvalCases:y.persistToEvalCases,generatedBy:typeof c.autoGenSettings?.generatedBy=="string"?c.autoGenSettings.generatedBy:y.generatedBy}}),(c.annotationDecisionFilter==="all"||typeof c.annotationDecisionFilter=="string"&&c.annotationDecisionFilter.length>0)&&Oe(c.annotationDecisionFilter),typeof c.hideResolvedAnnotations=="boolean"&&We(c.hideResolvedAnnotations)}catch{}},[]);const kn=x.useCallback(()=>{const a={filters:I,analysisDate:D,autogenPersonalityPromptHeight:Pt,autoGenSettings:{mode:h.mode,count:h.count,personalityId:h.personalityId,personalityPrompts:h.personalityPrompts,persistToEvalCases:h.persistToEvalCases,generatedBy:h.generatedBy},annotationDecisionFilter:_e,hideResolvedAnnotations:$e};localStorage.setItem(Cc,JSON.stringify(a))},[D,_e,h,Pt,I,$e]);return x.useEffect(()=>{const a=()=>{kn(),le("Сохранены настройки панели автопрогонов.")};return window.addEventListener(Nc,a),()=>{window.removeEventListener(Nc,a)}},[le,kn]),o.jsxs(Tn,{className:"autoruns-frame",title:"",hideHeader:!0,children:[o.jsxs("div",{className:"autoruns-columns",children:[o.jsxs("section",{className:"autoruns-col",children:[o.jsx("div",{className:"autoruns-col-header",children:o.jsx("h3",{children:"Настройки"})}),o.jsx("h4",{children:"Настройки выборки"}),o.jsxs("div",{className:"autoruns-form-grid",children:[o.jsxs("label",{children:["Дата с",o.jsx("input",{type:"datetime-local",value:I.fromLocal,onChange:a=>q(c=>({...c,fromLocal:a.target.value}))})]}),o.jsxs("label",{children:["Дата по",o.jsx("input",{type:"datetime-local",value:I.toLocal,onChange:a=>q(c=>({...c,toLocal:a.target.value}))})]}),o.jsxs("label",{children:["Целевой контур",o.jsxs("select",{value:I.target,onChange:a=>q(c=>({...c,target:a.target.value})),children:[o.jsx("option",{value:"all",children:"все"}),(oe?.available.targets??[]).map(a=>o.jsx("option",{value:a,children:a},a))]})]}),o.jsxs("label",{children:["Режим",o.jsxs("select",{value:I.mode,onChange:a=>q(c=>({...c,mode:a.target.value})),children:[o.jsx("option",{value:"all",children:"все"}),(oe?.available.modes??[]).map(a=>o.jsx("option",{value:a,children:a},a))]})]}),o.jsxs("label",{children:["Использовать mock",o.jsxs("select",{value:I.useMock,onChange:a=>q(c=>({...c,useMock:a.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:Yn,onChange:a=>{const c=a.target.value;(c===""||/^\d+$/.test(c))&&vt(c)},onBlur:a=>ts(a.target.value),onKeyDown:a=>{a.key==="Enter"&&ts(a.target.value)}})]}),o.jsxs("label",{className:"full-width",children:["Версия промпта содержит",o.jsx("input",{value:I.promptContains,onChange:a=>q(c=>({...c,promptContains:a.target.value})),placeholder:"normalizer_v2_0_2 / address_query_runtime_v1",list:"autoruns-prompt-versions"})]})]}),o.jsx("datalist",{id:"autoruns-prompt-versions",children:(oe?.available.prompt_versions??[]).map(a=>o.jsx("option",{value:a},a))}),o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",disabled:en,onClick:()=>{Qt({keepSelection:!1})},children:en?"Обновляю...":"Применить"}),o.jsx("button",{type:"button",className:"tab",onClick:()=>{q({...Vi,fromLocal:Mc()}),be("")},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:i.llmProvider})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Модель:"}),o.jsx("strong",{children:i.model||"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Промпт ассистента:"}),o.jsx("strong",{children:f})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Промпт декомпозиции:"}),o.jsx("strong",{children:N})]})]}),o.jsx("h4",{children:"Автогенерация вопросов"}),o.jsxs("div",{className:"autoruns-form-grid",children:[o.jsxs("label",{children:["Режим генерации",o.jsxs("select",{value:h.mode,onChange:a=>k(c=>({...c,mode:a.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:Zr,onChange:a=>{const c=a.target.value;(c===""||/^\d+$/.test(c))&&yt(c)},onBlur:a=>ns(a.target.value),onKeyDown:a=>{a.key==="Enter"&&ns(a.target.value)}})]}),o.jsxs("label",{children:["Личность автогенерации",o.jsx("select",{value:h.personalityId,onChange:a=>k(c=>({...c,personalityId:a.target.value})),children:W.map(a=>o.jsx("option",{value:a.id,children:a.label},a.id))})]}),o.jsxs("label",{children:["Кто генерирует",o.jsx("input",{value:h.generatedBy,onChange:a=>k(c=>({...c,generatedBy:a.target.value})),placeholder:"manual_reviewer"})]}),o.jsxs("label",{className:"full-width",children:["Промпт личности",o.jsx("textarea",{className:"autoruns-personality-prompt",value:h.personalityPrompts[h.personalityId]??"",onChange:a=>k(c=>({...c,personalityPrompts:{...c.personalityPrompts,[c.personalityId]:a.target.value}})),placeholder:"Текст промпта для выбранной личности автогенерации",style:{height:`${Pt}px`},onMouseUp:er,onTouchEnd:er})]}),o.jsxs("label",{className:"checkbox-row",children:[o.jsx("input",{type:"checkbox",checked:h.persistToEvalCases,onChange:a=>k(c=>({...c,persistToEvalCases:a.target.checked}))}),"Сохранять кейс-сет в `eval_cases`"]})]}),o.jsxs("div",{className:"autoruns-form-grid",children:[o.jsxs("label",{children:["Дата анализа (срез)",o.jsx("input",{type:"date",value:D,onChange:a=>B(Wi(a.target.value))})]}),o.jsx("div",{className:"button-row",children:o.jsx("button",{type:"button",className:"tab",disabled:!D,onClick:()=>B(""),children:"Сбросить дату среза"})})]}),o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",disabled:In,onClick:()=>{bn()},children:In?"Генерирую...":"Сгенерировать пачку"}),o.jsx("button",{type:"button",className:"tab",disabled:hn,onClick:()=>{wt()},children:hn?"Обновляю...":"Обновить историю"}),o.jsx("button",{type:"button",className:"autoruns-run-launch-btn",disabled:fr||Se.length===0,onClick:()=>{we()},children:fr?"Запускаю...":"Запустить прогоны"})]}),o.jsx("div",{className:"autoruns-form-grid",children:o.jsxs("label",{className:"full-width",children:["Кейс-сет для запуска",o.jsxs("select",{value:de,onChange:a=>fe(a.target.value),disabled:J.length===0,children:[J.length===0?o.jsx("option",{value:"",children:"нет генераций"}):null,J.map(a=>o.jsxs("option",{value:a.generation_id,children:[cr(a.created_at)," | ",a.mode," | ",a.count," | ",a.saved_case_set_file??"без файла"]},a.generation_id))]})]})}),o.jsxs("div",{className:"autoruns-generated-questions",children:[o.jsxs("div",{className:"autoruns-generated-questions-head",children:[o.jsxs("strong",{children:["Вопросы к запуску: ",Se.length]}),o.jsx("button",{type:"button",className:"tab",onClick:()=>ye([...Rt?.questions??[]]),disabled:!Rt,children:"Восстановить"})]}),Se.length===0?o.jsx("p",{className:"muted",children:"Список вопросов пуст. Сгенерируйте пачку или восстановите из выбранной генерации."}):o.jsx("div",{className:"autoruns-generated-questions-list",children:Se.map((a,c)=>o.jsxs("div",{className:"autoruns-generated-question-item",children:[o.jsxs("span",{children:[c+1,". ",a]}),o.jsx("button",{type:"button",className:"autoruns-remove-question-btn",onClick:()=>ye(y=>y.filter((A,he)=>he!==c)),title:"Удалить вопрос из запуска","aria-label":"Удалить вопрос из запуска",children:"+"})]},`${c}-${a.slice(0,24)}`))})]}),o.jsx("p",{className:"muted",children:"Запуск выполняет `assistant_stage1` eval по выбранному кейс-сету."}),o.jsxs("div",{className:"autoruns-autogen-list",children:[hn?o.jsx("p",{className:"muted",children:"Загружаю историю автогенераций..."}):null,!hn&&J.length===0?o.jsx("p",{className:"muted",children:"История автогенераций пока пустая."}):null,J.slice(0,30).map(a=>o.jsxs("article",{className:de===a.generation_id?"autoruns-autogen-item selected":"autoruns-autogen-item",onClick:()=>fe(a.generation_id),children:[o.jsxs("header",{children:[o.jsx("strong",{children:cr(a.created_at)}),o.jsx("span",{children:a.mode})]}),o.jsxs("div",{className:"autoruns-run-meta",children:["id=",a.generation_id," | count=",a.count]}),o.jsxs("div",{className:"autoruns-run-meta",children:["домен=",a.domain??"общий",a.generated_by?` | автор=${a.generated_by}`:""]}),a.saved_case_set_file?o.jsxs("div",{className:"autoruns-run-meta",children:["кейс-сет=",a.saved_case_set_file]}):null,(a.questions??[]).length>0?o.jsx("p",{children:a.questions[0]}):null]},a.generation_id))]}),o.jsxs("details",{className:"autoruns-prompt-details",children:[o.jsx("summary",{children:"Копия активного промпта (только чтение)"}),o.jsxs("label",{children:["Системный",o.jsx("textarea",{readOnly:!0,value:m.systemPrompt})]}),o.jsxs("label",{children:["Разработчика",o.jsx("textarea",{readOnly:!0,value:m.developerPrompt})]}),o.jsxs("label",{children:["Доменный",o.jsx("textarea",{readOnly:!0,value:m.domainPrompt})]}),o.jsxs("label",{children:["Заметки по схеме",o.jsx("textarea",{readOnly:!0,value:m.schemaNotes})]}),o.jsxs("label",{children:["Примеры few-shot",o.jsx("textarea",{readOnly:!0,value:m.fewShotExamples})]})]}),vn?o.jsx("p",{className:"error-text",children:vn}):null]}),o.jsxs("section",{className:"autoruns-col",children:[o.jsx("div",{className:"autoruns-col-header",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:(oe?.stats.runs_total??0)+(ee?1:0)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Средний score"}),o.jsx("strong",{children:ll(oe?.stats.avg_score_index??null)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Тренд"}),o.jsx("strong",{children:oe?Rc(oe.stats.trend):"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Блокеры"}),o.jsx("strong",{children:oe?.stats.blocking_runs??0})]})]}),o.jsxs("div",{className:"autoruns-run-list",children:[Sn.map(a=>o.jsxs("button",{type:"button",className:me===a.run_id?"autoruns-run-item selected":"autoruns-run-item",onClick:()=>{It(a.run_id)},children:[o.jsxs("div",{className:"autoruns-run-head",children:[o.jsx("strong",{children:cr(a.run_timestamp)}),o.jsx("span",{children:sm(a.eval_target)})]}),o.jsxs("div",{className:"autoruns-run-meta autoruns-run-id-row",children:[o.jsx("span",{children:a.run_id}),o.jsx("span",{role:"button",tabIndex:0,className:"autoruns-copy-run-id-btn",onClick:c=>{ct(c,a.run_id)},onKeyDown:c=>{(c.key==="Enter"||c.key===" ")&&(c.preventDefault(),ct(c,a.run_id))},title:"Скопировать run id","aria-label":`Скопировать run id ${a.run_id}`,children:o.jsx(im,{})})]}),o.jsxs("div",{className:"autoruns-run-meta",children:["режим=",a.mode??"нет данных"," | mock=",String(a.use_mock)]}),o.jsxs("div",{className:"autoruns-run-meta",children:["analysis_date=",a.analysis_date??"current_state"]}),a.llm_provider||a.model?o.jsxs("div",{className:"autoruns-run-meta",children:["llm=",a.llm_provider??"нет данных"," | модель=",a.model??"нет данных"]}):null,o.jsxs("div",{className:"autoruns-run-meta",children:["промпт=",a.prompt_version??"нет данных"]}),o.jsxs("div",{className:"autoruns-run-foot",children:[o.jsxs("span",{children:["оценка: ",ll(a.score_index)]}),o.jsxs("span",{children:["закрыто/открыто: ",a.closed_cases,"/",a.open_cases]})]}),o.jsxs("div",{className:"autoruns-run-foot",children:[o.jsxs("span",{children:["блокеры: ",a.blocking_failures]}),o.jsxs("span",{children:["качество: ",a.quality_failures]})]})]},a.run_id)),Sn.length===0?o.jsx("p",{className:"muted",children:"За выбранный диапазон прогонов нет."}):null]})]}),o.jsxs("section",{className:"autoruns-col",children:[o.jsxs("div",{className:"autoruns-col-header",children:[o.jsx("h3",{children:"Диалог прогона"}),o.jsxs("div",{className:"autoruns-dialog-toolbar",children:[o.jsxs("label",{children:["Прогон",o.jsx("select",{value:me,onChange:a=>{const c=a.target.value;It(c)},children:Sn.map(a=>o.jsxs("option",{value:a.run_id,children:[cr(a.run_timestamp)," | ",a.run_id]},a.run_id))})]}),o.jsxs("label",{children:["Кейс",o.jsxs("select",{value:pe,onChange:a=>{const c=a.target.value;O(c),me&&c&&Bn(me,c)},children:[(se?.cases.length??0)>0?o.jsx("option",{value:ht,children:"ВСЕ кейсы подряд"}):null,(se?.cases??[]).map(a=>o.jsxs("option",{value:a.case_id,children:[a.case_id," | ",a.status]},a.case_id))]})]})]})]}),o.jsxs("div",{className:"autoruns-case-list",children:[(se?.cases.length??0)>0?o.jsxs("button",{type:"button",className:pe===ht?"autoruns-case-item selected":"autoruns-case-item",onClick:()=>{O(ht),me&&Bn(me,ht)},children:[o.jsx("span",{children:"ВСЕ кейсы подряд"}),o.jsx("span",{children:se?.cases.length})]},ht):null,(se?.cases??[]).map(a=>o.jsxs("button",{type:"button",className:pe===a.case_id?"autoruns-case-item selected":"autoruns-case-item",onClick:()=>{O(a.case_id),me&&Bn(me,a.case_id)},children:[o.jsx("span",{children:a.case_id}),o.jsxs("span",{children:[a.status,a.commented_count>0?` | комм=${a.commented_count}`:""]})]},a.case_id))]}),o.jsxs("div",{className:"autoruns-dialog-view",children:[pr||mr?o.jsx("p",{className:"muted",children:"Загружаю диалог..."}):null,!pr&&!mr&&(Te?.messages.length??0)===0?o.jsx("p",{className:"muted",children:"Диалог для этого прогона не найден."}):null,(Te?.messages??[]).map((a,c)=>{const y=a.role==="assistant"?"assistant":"user";return o.jsxs("article",{className:`autoruns-msg ${y}`,children:[o.jsxs("header",{children:[o.jsx("strong",{children:y==="assistant"?"Система":"Модель/вопрос"}),o.jsxs("div",{className:"autoruns-msg-head-actions",children:[a.case_id?o.jsx("span",{className:"autoruns-msg-case-tag",children:a.case_id}):null,o.jsx("span",{children:a.created_at?cr(a.created_at):"нет данных"}),y==="assistant"&&!Cs(me)?o.jsxs(o.Fragment,{children:[o.jsx("button",{type:"button",className:a.commented?"autoruns-comment-icon commented":"autoruns-comment-icon",onClick:()=>p(a),title:"\\u041a\\u043e\\u043c\\u043c\\u0435\\u043d\\u0442\\u0438\\u0440\\u043e\\u0432\\u0430\\u0442\\u044c \\u043e\\u0442\\u0432\\u0435\\u0442 \\u0441\\u0438\\u0441\\u0442\\u0435\\u043c\\u044b","aria-label":"\\u041a\\u043e\\u043c\\u043c\\u0435\\u043d\\u0442\\u0438\\u0440\\u043e\\u0432\\u0430\\u0442\\u044c \\u043e\\u0442\\u0432\\u0435\\u0442 \\u0441\\u0438\\u0441\\u0442\\u0435\\u043c\\u044b",children:o.jsx(lm,{commented:a.commented})}),a.annotation?o.jsx("button",{type:"button",className:a.annotation.resolved?"autoruns-resolve-toggle resolved":"autoruns-resolve-toggle",onClick:()=>{wn(a.annotation,!a.annotation.resolved)},disabled:hr===a.annotation.annotation_id,title:a.annotation.resolved?"Отметить кейс как невыполненный":"Отметить кейс как выполненный","aria-label":a.annotation.resolved?"Отметить кейс как невыполненный":"Отметить кейс как выполненный",children:o.jsx(Ic,{resolved:a.annotation.resolved})}):null]}):null]})]}),o.jsx("p",{children:a.text}),y==="assistant"&&a.annotation?o.jsxs("div",{className:"autoruns-msg-annotation",children:[o.jsx("strong",{children:qi(a.annotation.rating)}),o.jsx("span",{children:a.annotation.comment}),o.jsxs("span",{className:"muted",children:[a.annotation.manual_case_decision,a.annotation.annotation_author?` | ${a.annotation.annotation_author}`:""]})]}):null,(a.trace_id||a.reply_type)&&o.jsxs("footer",{children:[a.trace_id?o.jsxs("span",{children:["trace=",a.trace_id]}):null,a.reply_type?o.jsxs("span",{children:["reply_type=",a.reply_type]}):null]})]},a.message_id??`${y}-${c}`)})]})]}),j?o.jsx("div",{className:"autoruns-col autoruns-assistant-live-col",children:o.jsx(Hc,{sessionId:ot,conversation:xn,inputValue:zn,onInputChange:on,selectedContextChip:Ht,onSelectContextChip:gt,onClearContextChip:()=>gt(null),useMock:Jn,onUseMockChange:qe,onSend:Ns,onClear:St,busy:gr,statusText:vr,errorMessage:yr,showCommentAction:!0,onCommentAssistantMessage:Hn,isAssistantMessageCommented:Ke,canCommentAssistantMessage:ne})}):null,T?o.jsxs("section",{className:"autoruns-col",children:[o.jsx("div",{className:"autoruns-col-header",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:Fn?.case_id??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"домен:"}),o.jsx("strong",{children:Fn?.domain??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"класс запроса:"}),o.jsx("strong",{children:Fn?.query_class??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"trace:"}),o.jsx("strong",{children:Fn?.trace_id??"нет данных"})]})]}),o.jsx("h4",{children:"Шаги декомпозиции"}),(Te?.decomposition.length??0)>0?o.jsx("ol",{className:"autoruns-decomposition-list",children:(Te?.decomposition??[]).map((a,c)=>o.jsx("li",{children:a},`${c}-${a.slice(0,24)}`))}):o.jsx("p",{className:"muted",children:"В логах кейса нет явной декомпозиции."})]}):null,F?o.jsxs("section",{className:"autoruns-col",children:[o.jsx("div",{className:"autoruns-col-header",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(oe?.stats.latest_score_index??null)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Предыдущий"}),o.jsx("strong",{children:ll(oe?.stats.previous_score_index??null)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Тренд"}),o.jsx("strong",{children:oe?Rc(oe.stats.trend):"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Пробелы качества"}),o.jsx("strong",{children:oe?.stats.quality_gap_runs??0})]})]}),o.jsx("h4",{children:"Покрытие доменов (история)"}),Ac(oe?.stats.domain_coverage??[]),o.jsx("h4",{style:{marginTop:14},children:"Покрытие доменов (выбранный прогон)"}),Ac(se?.coverage.domain_coverage??[]),o.jsx("h4",{style:{marginTop:14},children:"Очереди фиксов пост-анализа"}),Zt?o.jsx("p",{className:"muted",children:"Собираю пост-анализ..."}):null,Zt?null:o.jsx("div",{className:"autoruns-stats-grid",children:Object.entries(Ut?.post_analysis.stats.by_queue??{}).map(([a,c])=>o.jsxs("div",{children:[o.jsx("span",{children:a}),o.jsx("strong",{children:c})]},a))}),o.jsxs("div",{className:"autoruns-autogen-list",children:[(Ut?.post_analysis.recommended_regression_candidates??[]).slice(0,12).map(a=>o.jsxs("article",{className:"autoruns-autogen-item",children:[o.jsxs("header",{children:[o.jsx("strong",{children:a.manual_case_decision}),o.jsxs("span",{children:[a.rating,"/5"]})]}),o.jsxs("div",{className:"autoruns-run-meta",children:[a.domain??"неизвестно"," / ",a.query_class??"неизвестно"]}),o.jsx("p",{children:a.comment})]},a.annotation_id)),!Zt&&(Ut?.post_analysis.recommended_regression_candidates.length??0)===0?o.jsx("p",{className:"muted",children:"Рекомендованных кандидатов пока нет."}):null]})]}):null,Y?o.jsxs("section",{className:"autoruns-col",children:[o.jsx("div",{className:"autoruns-col-header",children:o.jsx("h3",{children:"Комментарии"})}),o.jsx("h4",{children:"Размеченные ответы"}),o.jsxs("div",{className:"autoruns-comment-filter-row",children:[o.jsxs("label",{children:["Фильтр решений",o.jsxs("select",{value:_e,onChange:a=>Oe(a.target.value),children:[o.jsx("option",{value:"all",children:"все"}),(st.length>0?st:Je?.enum??[]).map(a=>o.jsx("option",{value:a,children:String(Je?.labels?.[a]??a)},a))]})]}),o.jsx("button",{type:"button",className:"tab autoruns-resolved-filter-toggle",onClick:()=>We(a=>!a),children:$e?"Показать выполненные":"Скрыть выполненные"})]}),o.jsxs("div",{className:"autoruns-stats-grid",children:[o.jsxs("div",{children:[o.jsx("span",{children:"Комментариев"}),o.jsx("strong",{children:_t.length})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Средний рейтинг"}),o.jsx("strong",{children:_r===null?"нет данных":`${_r.toFixed(2)} / 5`})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Последний"}),o.jsx("strong",{children:_t.length>0?cr(_t[0].updated_at):"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Статус"}),o.jsx("strong",{children:tn?"обновляю":"готово"})]})]}),o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",disabled:tn,onClick:()=>{Tt()},children:tn?"Обновляю...":"Обновить список"}),o.jsx("button",{type:"button",className:"tab",disabled:Zt,onClick:()=>{jt()},children:Zt?"Идет пост-анализ...":"Обновить пост-анализ"})]}),o.jsxs("div",{className:"autoruns-comments-list",children:[tn?o.jsx("p",{className:"muted",children:"Загружаю комментарии..."}):null,!tn&&_t.length===0?o.jsx("p",{className:"muted",children:G.length===0&&sn.length===0?"Пока нет откомментированных ответов.":"Нет открытых кейсов по текущему фильтру."}):null,_t.map(a=>{if(a.source==="assistant_live"){const y=a.assistant;return o.jsxs("article",{className:"autoruns-comment-item",children:[o.jsxs("div",{className:"autoruns-comment-head",children:[o.jsx("strong",{children:qi(y.rating)}),o.jsx("div",{className:"autoruns-comment-head-actions",children:o.jsx("span",{children:cr(y.updated_at)})})]}),o.jsxs("div",{className:"autoruns-run-meta",children:["live-session: ",y.session_id]}),o.jsxs("div",{className:"autoruns-run-meta",children:["msg=",y.message_index]}),o.jsxs("div",{className:"autoruns-run-meta",children:["source=assistant_live",y.annotation_author?` | author=${y.annotation_author}`:""]}),y.context.question_text?o.jsxs("p",{children:["Q: ",y.context.question_text]}):null,y.context.answer_text?o.jsxs("p",{children:["A: ",y.context.answer_text]}):null,o.jsx("p",{children:y.comment})]},a.key)}const c=a.autorun;return o.jsxs("article",{className:Ie===c.annotation_id?"autoruns-comment-item selected":"autoruns-comment-item",onClick:()=>{jn(c)},role:"button",tabIndex:0,onKeyDown:y=>{(y.key==="Enter"||y.key===" ")&&(y.preventDefault(),jn(c))},children:[o.jsxs("div",{className:"autoruns-comment-head",children:[o.jsx("strong",{children:qi(c.rating)}),o.jsxs("div",{className:"autoruns-comment-head-actions",children:[o.jsx("span",{children:cr(c.updated_at)}),o.jsx("button",{type:"button",className:c.resolved?"autoruns-resolve-toggle resolved":"autoruns-resolve-toggle",onClick:y=>{y.preventDefault(),y.stopPropagation(),wn(c,!c.resolved)},disabled:hr===c.annotation_id,title:c.resolved?"Отметить кейс как невыполненный":"Отметить кейс как выполненный","aria-label":c.resolved?"Отметить кейс как невыполненный":"Отметить кейс как выполненный",children:o.jsx(Ic,{resolved:c.resolved})})]})]}),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}`:""]}),c.resolved_at?o.jsxs("div",{className:"autoruns-run-meta",children:["выполнено",": ",cr(c.resolved_at),c.resolved_by?` | by=${c.resolved_by}`:""]}):null,c.context.question_text?o.jsxs("p",{children:["Q: ",c.context.question_text]}):null,c.context.answer_text?o.jsxs("p",{children:["A: ",c.context.answer_text]}):null,o.jsx("p",{children:c.comment})]},a.key)})]}),z?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:z.technical_context.trace_id??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"reply_type:"}),o.jsx("strong",{children:z.technical_context.reply_type??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"domain:"}),o.jsx("strong",{children:z.technical_context.domain??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"query_class:"}),o.jsx("strong",{children:z.technical_context.query_class??"нет данных"})]})]}),o.jsx("h4",{children:"JSON разбор"}),o.jsx(Yt,{value:{annotation_id:z.annotation_id,run_id:z.run_id,case_id:z.case_id,message_index:z.message_index,rating:z.rating,comment:z.comment,manual_case_decision:z.manual_case_decision,annotation_author:z.annotation_author,resolved:z.resolved,resolved_at:z.resolved_at,resolved_by:z.resolved_by,context:z.context,technical_context:z.technical_context,case_summary:z.case_summary?{case_id:z.case_summary.case_id,domain:z.case_summary.domain,query_class:z.case_summary.query_class,checks:z.case_summary.checks,metric_subscores:z.case_summary.metric_subscores}:null}})]}):null]}):null]}),xe.open?o.jsx("div",{className:"autoruns-comment-modal-backdrop",onClick:a=>{a.target===a.currentTarget&&At()},children:o.jsxs("div",{className:"autoruns-comment-modal",children:[o.jsx("h3",{children:"Комментарий к ответу ассистента"}),o.jsx("p",{className:"muted",children:"Комментарий будет добавлен в общий список комментариев справа с меткой `assistant_live`."}),$r?o.jsxs("details",{className:"autoruns-prompt-details",open:!0,children:[o.jsx("summary",{children:"Вопрос пользователя"}),o.jsx("p",{className:"autoruns-comment-quote",children:$r.text})]}):null,Me?o.jsxs("details",{className:"autoruns-prompt-details",open:!0,children:[o.jsx("summary",{children:"Ответ ассистента"}),o.jsx("p",{className:"autoruns-comment-quote",children:Me.text})]}):null,o.jsx("div",{className:"autoruns-rating-row",role:"group","aria-label":"Рейтинг ответа ассистента",children:[1,2,3,4,5].map(a=>o.jsx("button",{type:"button",className:xe.rating>=a?"autoruns-rating-dot active":"autoruns-rating-dot",onClick:()=>xt(c=>({...c,rating:a})),disabled:xe.saving,"aria-label":`Оценка ${a}`,children:xe.rating>=a?"●":"○"},a))}),o.jsx("div",{className:"autoruns-form-grid",children:o.jsxs("label",{children:["Автор комментария",o.jsx("input",{value:xe.annotationAuthor,onChange:a=>xt(c=>({...c,annotationAuthor:a.target.value})),placeholder:"manual_reviewer",disabled:xe.saving})]})}),o.jsxs("label",{children:["Комментарий",o.jsx("textarea",{value:xe.comment,onChange:a=>xt(c=>({...c,comment:a.target.value})),placeholder:"Что именно не так в ответе и что нужно исправить.",rows:4,disabled:xe.saving})]}),xe.error?o.jsx("p",{className:"error-text",children:xe.error}):null,o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",onClick:()=>{Es()},disabled:xe.saving,children:xe.saving?"Сохраняю...":"Готово"}),o.jsx("button",{type:"button",className:"tab",onClick:()=>At(),disabled:xe.saving,children:"Отмена"})]})]})}):null,ce.open?o.jsx("div",{className:"autoruns-comment-modal-backdrop",onClick:a=>{a.target===a.currentTarget&&E()},children:o.jsxs("div",{className:"autoruns-comment-modal",children:[o.jsx("h3",{children:"Комментарий к ответу системы"}),o.jsx("p",{className:"muted",children:"Оцените ответ по 5-балльной шкале и добавьте комментарий по браку."}),zr?o.jsxs(o.Fragment,{children:[o.jsxs("details",{className:"autoruns-prompt-details",open:!0,children:[o.jsx("summary",{children:"Вопрос пользователя"}),o.jsx("p",{className:"autoruns-comment-quote",children:xr?.text??"Вопрос в диалоге не найден."})]}),o.jsxs("details",{className:"autoruns-prompt-details",open:!0,children:[o.jsx("summary",{children:"Ответ системы"}),o.jsx("p",{className:"autoruns-comment-quote",children:zr.text})]})]}):null,o.jsx("div",{className:"autoruns-rating-row",role:"group","aria-label":"Рейтинг ответа",children:[1,2,3,4,5].map(a=>o.jsx("button",{type:"button",className:ce.rating>=a?"autoruns-rating-dot active":"autoruns-rating-dot",onClick:()=>lt(c=>({...c,rating:a})),disabled:ce.saving,"aria-label":`Оценка ${a}`,children:ce.rating>=a?"●":"○"},a))}),o.jsxs("div",{className:"autoruns-form-grid",children:[o.jsxs("label",{children:["Решение по кейсу",o.jsx("select",{value:ce.manualCaseDecision,onChange:a=>lt(c=>({...c,manualCaseDecision:a.target.value})),disabled:ce.saving,children:(st.length>0?st:Je?.enum??[sl]).map(a=>o.jsx("option",{value:a,children:String(Je?.labels?.[a]??a)},a))})]}),o.jsxs("label",{children:["Автор комментария",o.jsx("input",{value:ce.annotationAuthor,onChange:a=>lt(c=>({...c,annotationAuthor:a.target.value})),placeholder:"manual_reviewer",disabled:ce.saving})]})]}),o.jsxs("label",{children:["Комментарий",o.jsx("textarea",{value:ce.comment,onChange:a=>lt(c=>({...c,comment:a.target.value})),placeholder:"Почему ответ бракованный, что именно пошло не так, какие технические детали проверить.",rows:4,disabled:ce.saving})]}),ce.error?o.jsx("p",{className:"error-text",children:ce.error}):null,o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",onClick:()=>{C()},disabled:ce.saving,children:ce.saving?"Сохраняю...":"Готово"}),o.jsx("button",{type:"button",className:"tab",onClick:()=>E(),disabled:ce.saving,children:"Отмена"})]})]})}):null]})}const il="http://127.0.0.1:1234/v1",Lc="https://api.openai.com/v1",cl="qwen2.5-14b-instruct-1m",ea="unsloth/qwen3-30b-a3b-instruct-2507",um=[{value:cl,label:"Qwen2.5 14B Instruct 1M"},{value:ea,label:"Qwen3 30B A3B Instruct 2507"}];function cm(i){return i.llmProvider!=="local"?"openai":i.model===ea?"local_qwen3":i.model===cl?"local_qwen25":"local_custom"}function dm(i,m){const f=new Map;if(m)for(const N of um)f.set(N.value,N);for(const N of i)f.has(N)||f.set(N,{value:N,label:N});return Array.from(f.values())}function Oc({value:i,modelOptions:m,modelsBusy:f,onChange:N,onReloadModels:j,onTestConnection:T,onSaveLocalConfig:F,lastStatus:Y,busy:U}){const I=i.llmProvider==="local",q=cm(i),D=dm(m,I),B=D.some(G=>G.value===i.model),[oe,Ae]=x.useState(String(i.temperature)),[se,te]=x.useState(String(i.maxOutputTokens));x.useEffect(()=>{Ae(String(i.temperature))},[i.temperature]),x.useEffect(()=>{te(String(i.maxOutputTokens))},[i.maxOutputTokens]);const Te=G=>{const ae=G.replace(",",".").trim();if(!ae){Ae(String(i.temperature));return}const _e=Number(ae);if(!Number.isFinite(_e)){Ae(String(i.temperature));return}N({...i,temperature:_e}),Ae(String(_e))},ve=G=>{const ae=G.trim();if(!ae){te(String(i.maxOutputTokens));return}const _e=Number.parseInt(ae,10);if(!Number.isFinite(_e)||_e<=0){te(String(i.maxOutputTokens));return}N({...i,maxOutputTokens:_e}),te(String(_e))};return o.jsxs(Tn,{title:"LLM Connection",subtitle:"Switch between OpenAI cloud and local OpenAI-compatible server.",actions:o.jsx("span",{className:"status-chip",children:Y||"Status: not checked"}),children:[o.jsxs("div",{className:"grid-two",children:[o.jsxs("label",{children:["Provider",o.jsxs("select",{value:q,onChange:G=>{const ae=G.target.value;if(ae==="openai"){N({...i,llmProvider:"openai",baseUrl:Lc});return}if(ae==="local_qwen25"){N({...i,llmProvider:"local",model:cl,baseUrl:il});return}if(ae==="local_qwen3"){N({...i,llmProvider:"local",model:ea,baseUrl:il});return}N({...i,llmProvider:"local",model:i.llmProvider==="local"?i.model:cl,baseUrl:il})},children:[o.jsx("option",{value:"openai",children:"OpenAI (token)"}),o.jsx("option",{value:"local_qwen25",children:"Qwen2.5 14B Instruct 1M (Local LM Studio)"}),o.jsx("option",{value:"local_qwen3",children:"Qwen3 30B A3B Instruct 2507 (Local LM Studio)"}),o.jsx("option",{value:"local_custom",children:"Local custom (LM Studio / OpenAI-compatible)"})]})]}),o.jsxs("label",{children:["Model",o.jsxs("select",{value:B?i.model:"__manual__",onChange:G=>{const ae=G.target.value;ae!=="__manual__"&&N({...i,model:ae})},children:[o.jsx("option",{value:"__manual__",children:"Manual input"}),D.map(G=>o.jsx("option",{value:G.value,children:G.label},G.value))]})]}),o.jsxs("label",{children:["Model ID (manual / current)",o.jsx("input",{value:i.model,onChange:G=>N({...i,model:G.target.value}),placeholder:"qwen2.5-14b-instruct-1m or unsloth/qwen3-30b-a3b-instruct-2507"})]}),I?null:o.jsxs("label",{className:"full-width",children:["OpenAI API Key",o.jsx("input",{type:"password",value:i.apiKey,onChange:G=>N({...i,apiKey:G.target.value}),placeholder:"sk-..."})]}),o.jsxs("label",{className:I?"full-width":void 0,children:[I?"Local server base URL":"Base URL",o.jsx("input",{value:i.baseUrl,onChange:G=>N({...i,baseUrl:G.target.value}),placeholder:I?il:Lc})]}),o.jsxs("label",{children:["Temperature",o.jsx("input",{type:"number",step:"0.1",value:oe,onChange:G=>Ae(G.target.value),onBlur:G=>Te(G.target.value),onKeyDown:G=>{G.key==="Enter"&&Te(G.target.value)}})]}),o.jsxs("label",{children:["Max output tokens",o.jsx("input",{type:"number",value:se,onChange:G=>te(G.target.value),onBlur:G=>ve(G.target.value),onKeyDown:G=>{G.key==="Enter"&&ve(G.target.value)}})]})]}),o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",onClick:()=>F(),children:"Save local config"}),o.jsx("button",{type:"button",onClick:()=>j(),disabled:U||f,children:f?"Loading models...":"Load model list"}),o.jsx("button",{type:"button",onClick:()=>T(),disabled:U,children:U?"Checking...":"Test connection"})]})]})}function fm({items:i,onRefresh:m,onOpenTrace:f}){return o.jsx(Tn,{title:"История нормализаций",subtitle:"Короткий вопрос, confidence, route hint и статус валидации.",actions:o.jsx("button",{type:"button",onClick:()=>m(),children:"Обновить"}),children:o.jsxs("div",{className:"history-list",children:[i.length===0?o.jsx("p",{className:"muted",children:"История пока пустая."}):null,i.map(N=>o.jsxs("button",{type:"button",className:"history-item",onClick:()=>f(N.trace_id),children:[o.jsxs("div",{className:"history-row",children:[o.jsx("strong",{children:N.route_hint??"route: n/a"}),o.jsx("span",{children:N.validation_passed?"schema: ok":"schema: fail"})]}),o.jsx("p",{children:N.question_short}),o.jsxs("div",{className:"history-row",children:[o.jsx("span",{children:N.model}),o.jsx("span",{children:new Date(N.timestamp).toLocaleString("ru-RU")})]})]},N.trace_id))]})})}function dr(i){return i==null||i===""?"—":String(i)}function mm({result:i}){return o.jsx(Tn,{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:dr(i?.trace_id)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"request_started_at"}),o.jsx("strong",{children:dr(i?new Date(Date.now()-i.latency_ms).toISOString():null)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"request_finished_at"}),o.jsx("strong",{children:dr(i?new Date().toISOString():null)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"latency_ms"}),o.jsx("strong",{children:dr(i?.latency_ms)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"input_tokens"}),o.jsx("strong",{children:dr(i?.usage?.input_tokens)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"output_tokens"}),o.jsx("strong",{children:dr(i?.usage?.output_tokens)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"total_tokens"}),o.jsx("strong",{children:dr(i?.usage?.total_tokens)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"validation_status"}),o.jsx("strong",{children:i?.validation?.passed?"passed":"failed"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"prompt_version"}),o.jsx("strong",{children:dr(i?.prompt_version)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"schema_version"}),o.jsx("strong",{children:dr(i?.schema_version)})]})]})})}const pm={normalized:"Normalized JSON",fragments:"Fragment View",scope:"Scope View",flags:"Flags View",route:"Route Simulation",raw:"Raw model output",validation:"Validation",logs:"Logs"};function hm(i){return i&&typeof i=="object"?i:null}function gm({tab:i,onTabChange:m,result:f,appLogs:N}){const j=["normalized","fragments","scope","flags","route","raw","validation","logs"],T=hm(f?.normalized),F=String(T?.schema_version??""),Y=F==="normalized_query_v2"||F==="normalized_query_v2_0_1"||F==="normalized_query_v2_0_2",U=Y?{fragments:T?.fragments??[],discarded_fragments:T?.discarded_fragments??[]}:{note:"Fragment View доступен для normalized_query_v2."},I=Y?{message_in_scope:T?.message_in_scope??null,scope_confidence:T?.scope_confidence??null,contains_multiple_tasks:T?.contains_multiple_tasks??null,global_notes:T?.global_notes??null}:{note:"Scope View доступен для normalized_query_v2."},q=Y?Array.isArray(T?.fragments)?(T?.fragments).map(D=>({fragment_id:D.fragment_id??null,domain_relevance:D.domain_relevance??null,candidate_labels:D.candidate_labels??[],execution_readiness:D.execution_readiness??null,clarification_reason:D.clarification_reason??null,soft_assumption_used:D.soft_assumption_used??[],route_status:D.route_status??null,no_route_reason:D.no_route_reason??null,flags:D.flags??{}})):[]:{note:"Flags View доступен для normalized_query_v2."};return o.jsxs(Tn,{title:"Выходные данные",subtitle:"Structured output и диагностические вкладки.",children:[o.jsx("div",{className:"tab-row",children:j.map(D=>o.jsx("button",{type:"button",className:i===D?"tab active":"tab",onClick:()=>m(D),children:pm[D]},D))}),i==="normalized"?o.jsx(Yt,{value:f?.normalized??{note:"Нет данных."}}):null,i==="fragments"?o.jsx(Yt,{value:U}):null,i==="scope"?o.jsx(Yt,{value:I}):null,i==="flags"?o.jsx(Yt,{value:q}):null,i==="route"?o.jsx(Yt,{value:f?.route_hint_summary??{note:"Нет данных."}}):null,i==="raw"?o.jsx(Yt,{value:f?.raw_model_output??{note:"Нет данных."}}):null,i==="validation"?o.jsx(Yt,{value:f?.validation??{note:"Нет данных."}}):null,i==="logs"?o.jsx(Yt,{value:N}):null]})}function Dc({value:i,onChange:m,presets:f,selectedPresetId:N,onSelectPreset:j,onLoadPreset:T,onSavePreset:F,onResetDefaults:Y,onDiffPrevious:U,presetName:I,onPresetNameChange:q,diffSummary:D}){return o.jsxs(Tn,{title:"Prompt Manager",subtitle:"Системный, developer и domain уровни управляются отдельно.",children:[o.jsxs("div",{className:"prompt-manager-grid",children:[o.jsxs("label",{children:["Системный prompt",o.jsx("textarea",{value:i.systemPrompt,onChange:B=>m({...i,systemPrompt:B.target.value}),rows:6})]}),o.jsxs("label",{children:["Developer / Instruction prompt",o.jsx("textarea",{value:i.developerPrompt,onChange:B=>m({...i,developerPrompt:B.target.value}),rows:6})]}),o.jsxs("label",{children:["Domain prompt",o.jsx("textarea",{value:i.domainPrompt,onChange:B=>m({...i,domainPrompt:B.target.value}),rows:6})]}),o.jsxs("label",{children:["Schema notes",o.jsx("textarea",{value:i.schemaNotes,onChange:B=>m({...i,schemaNotes:B.target.value}),rows:6})]}),o.jsxs("label",{className:"full-width",children:["Few-shot examples",o.jsx("textarea",{value:i.fewShotExamples,onChange:B=>m({...i,fewShotExamples:B.target.value}),rows:8})]})]}),o.jsxs("div",{className:"button-row",children:[o.jsxs("select",{value:N,onChange:B=>j(B.target.value),children:[o.jsx("option",{value:"",children:"Выберите preset..."}),f.map(B=>o.jsx("option",{value:B.id,children:B.name},B.id))]}),o.jsx("button",{type:"button",onClick:()=>T(),children:"Загрузить preset"}),o.jsx("input",{value:I,onChange:B=>q(B.target.value),placeholder:"Имя для сохранения"}),o.jsx("button",{type:"button",onClick:()=>F(),children:"Сохранить preset"}),o.jsx("button",{type:"button",onClick:()=>U(),children:"Diff с предыдущим"}),o.jsx("button",{type:"button",onClick:()=>Y(),children:"Сбросить к default"})]}),D?o.jsx("p",{className:"diff-summary",children:D}):null]})}function vm({value:i,onChange:m,onApplyBatchFormat:f,onNormalize:N,busy:j,useMock:T,onUseMockChange:F,errorMessage:Y}){return o.jsxs(Tn,{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:i.userQuestion,onChange:U=>m({...i,userQuestion:U.target.value}),rows:6,placeholder:"Например: По каким покупателям у нас на конец июня висят отгрузки без оплаты..."})]}),o.jsxs("label",{className:"full-width",children:["Batch queries (`;` separator)",o.jsx("textarea",{value:i.batchQuestionsRaw,onChange:U=>m({...i,batchQuestionsRaw:U.target.value}),onBlur:()=>f(),rows:8,placeholder:"Вопрос 1; Вопрос 2; Вопрос 3"})]}),o.jsxs("label",{children:["Optional period context",o.jsx("input",{value:i.periodHint,onChange:U=>m({...i,periodHint:U.target.value})})]}),o.jsxs("label",{children:["Optional business context",o.jsx("input",{value:i.businessContext,onChange:U=>m({...i,businessContext:U.target.value})})]}),o.jsxs("label",{children:["Optional expected route (eval)",o.jsx("input",{value:i.expectedRoute,onChange:U=>m({...i,expectedRoute:U.target.value})})]})]}),o.jsxs("div",{className:"button-row",children:[o.jsxs("label",{className:"checkbox-row",children:[o.jsx("input",{type:"checkbox",checked:T,onChange:U=>F(U.target.checked)}),"Mock-режим (без вызова OpenAI)"]}),o.jsx("button",{type:"button",onClick:()=>f(),disabled:j||!i.batchQuestionsRaw.trim(),children:"Применить `;` в переносы"}),o.jsx("button",{type:"button",onClick:()=>N(!1),disabled:j||!i.userQuestion.trim(),children:j?"Нормализуем...":"Normalize"}),o.jsx("button",{type:"button",onClick:()=>N(!0),disabled:j||!i.userQuestion.trim(),children:j?"Сохраняем...":"Normalize + Save as test case"})]}),Y?o.jsx("p",{className:"error-text",children:Y}):null]})}function ym({runs:i,selectedRunId:m,onSelectRun:f,onStartRun:N,onFinishRun:j,onRefreshRuns:T,onRunEval:F,onCopyEvalReport:Y,evalBusy:U,traceItems:I,evalReport:q}){return o.jsxs(Tn,{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:()=>N(),children:"Запустить run"}),o.jsx("button",{type:"button",onClick:()=>j(),disabled:!m,children:"Завершить выбранный run"}),o.jsx("button",{type:"button",onClick:()=>T(),children:"Обновить runs"}),o.jsx("button",{type:"button",onClick:()=>F(),disabled:U,children:U?"Идет eval v2.0.2...":"Запустить eval v2.0.2"})]}),o.jsxs("div",{className:"runtime-stack",children:[o.jsxs("div",{className:"runtime-runs",children:[i.map(D=>o.jsxs("button",{type:"button",className:m===D.runId?"history-item selected":"history-item",onClick:()=>f(D.runId),children:[o.jsxs("div",{className:"history-row",children:[o.jsx("strong",{children:D.status}),o.jsx("span",{children:D.runId})]}),o.jsxs("div",{className:"history-row",children:[o.jsx("span",{children:D.sessionId}),o.jsx("span",{children:new Date(D.updatedAt).toLocaleString("ru-RU")})]})]},D.runId)),i.length===0?o.jsx("p",{className:"muted",children:"Нет активных запусков."}):null]}),o.jsxs("div",{className:"runtime-details",children:[o.jsx("h3",{children:"Trace выбранного run"}),o.jsx(Yt,{value:I}),o.jsxs("div",{className:"eval-report-wrap",children:[o.jsx("h3",{style:{marginTop:12},children:"Отчет eval"}),o.jsx(Yt,{value:q??{note:"Eval пока не запускался"}}),o.jsx("button",{type:"button",className:"copy-cube-button",title:"Скопировать отчет eval",onClick:()=>Y(),children:"⧉"})]})]})]})]})}const xm={llmProvider:"openai",apiKey:"",model:"gpt-4o-mini",baseUrl:"https://api.openai.com/v1",temperature:0,maxOutputTokens:700},zc={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."},_m={userQuestion:"",batchQuestionsRaw:"",periodHint:"",businessContext:"",expectedRoute:""},Ki={colors:{backgroundRgb:"18, 18, 18",mainSurfaceRgb:"25, 25, 25",horizontalSurfaceRgb:"30, 30, 30",focusSurfaceRgb:"35, 35, 35",assistantChipRgb:"18, 18, 18",assistantChipHoverRgb:"44, 44, 44",assistantChipSelectedRgb:"167, 59, 255",assistantChipSelectedTextRgb:"240, 240, 240",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"},layout:{modeColumnWidthPx:406,modeToggleWidthPx:188}},$c="ndc_normalizer_session_config_v1",Fc="ndc_autoruns_layout_config_v1",Sm="ndc-autoruns-save",Gi=["Анализ запроса","Получение данных","Подготовка ответа"],wm="assistant",Ji="normalizer_v2_0_2",Uc="address_query_runtime_v1",jm=["normalized","fragments","scope","flags","route","raw","validation","logs"],Yi="manual_reviewer";function km(i){return`[${new Date().toLocaleTimeString("ru-RU")}] ${i}`}function Cm(i,m){if(!m)return"Previous preset is not selected.";const N=["systemPrompt","developerPrompt","domainPrompt","schemaNotes","fewShotExamples"].filter(j=>i[j]!==m[j]).map(j=>`${j}: ${Math.abs(i[j].length-m[j].length)} chars delta`);return N.length===0?"No changes against previous preset.":`Changed fields: ${N.length}. ${N.join(" | ")}`}function Nm(i,m){const f=i.trim();if(!f)return"";if(!m)return f;const N=f.toLowerCase(),j=m.anchor_text.trim(),T=j.toLowerCase();return T&&N.includes(T)?f:`По выбранному объекту "${j}": ${f}`}function Em(){const[i,m]=x.useState(xm),[f,N]=x.useState(zc),[j,T]=x.useState(_m),[F,Y]=x.useState(null),[U,I]=x.useState([]),[q,D]=x.useState([]),[B,oe]=x.useState("normalized"),[Ae,se]=x.useState(!1),[te,Te]=x.useState(!1),[ve,G]=x.useState([]),[ae,_e]=x.useState(""),[Oe,$e]=x.useState([]),[We,Je]=x.useState(""),[et,st]=x.useState("NDC custom preset"),[ie,Ie]=x.useState(null),[Fe,me]=x.useState(""),[Re,pe]=x.useState(!1),[O,W]=x.useState([]),[$,h]=x.useState(""),[k,J]=x.useState([]),[ue,de]=x.useState(!1),[fe,Se]=x.useState(null),[ye,ee]=x.useState(""),[Pe,Ut]=x.useState(wm),[Kn,In]=x.useState(!0),[Ln,fr]=x.useState(!0),[Xt,Zt]=x.useState(!0),[Gn,hn]=x.useState(!0),[bt,en]=x.useState(!0),[gn,mr]=x.useState(!0),[On,pr]=x.useState(!0),[Dn,tn]=x.useState(!0),[nn,hr]=x.useState(!0),[Bt,vn]=x.useState(!0),[be,ot]=x.useState(!0),[yn,xn]=x.useState(!0),[rn,sn]=x.useState(!0),[Nt,zn]=x.useState(!0),[on,Ht]=x.useState(!0),[gt,Jn]=x.useState(!0),[qe,gr]=x.useState(""),[_n,vr]=x.useState([]),[ln,yr]=x.useState(""),[Et,Yn]=x.useState(null),[vt,Zr]=x.useState(!1),[yt,Pt]=x.useState(""),[$n,ce]=x.useState(""),[lt,xe]=x.useState([]),[xt,Mt]=x.useState(!1),[Ce,it]=x.useState({open:!1,messageIndex:-1,rating:3,comment:"",annotationAuthor:Yi,saving:!1,error:""}),Rt=x.useRef(!1),Fn=x.useRef(!1),Vt=x.useRef(!1);x.useEffect(()=>{const p=document.documentElement,{colors:E}=Ki;p.style.setProperty("--rgb-background",E.backgroundRgb),p.style.setProperty("--rgb-surface-main",E.mainSurfaceRgb),p.style.setProperty("--rgb-surface-horizontal",E.horizontalSurfaceRgb),p.style.setProperty("--rgb-surface-focus",E.focusSurfaceRgb),p.style.setProperty("--rgb-assistant-chip",E.assistantChipRgb),p.style.setProperty("--rgb-assistant-chip-hover",E.assistantChipHoverRgb),p.style.setProperty("--rgb-assistant-chip-selected",E.assistantChipSelectedRgb),p.style.setProperty("--rgb-assistant-chip-selected-text",E.assistantChipSelectedTextRgb),p.style.setProperty("--rgb-active",E.activeRgb),p.style.setProperty("--rgb-active-text",E.activeTextRgb),p.style.setProperty("--rgb-text-main",E.textMainRgb),p.style.setProperty("--rgb-text-muted",E.textMutedRgb),p.style.setProperty("--rgb-danger",E.dangerRgb),p.style.setProperty("--rgb-scrollbar-track",E.scrollbarTrackRgb),p.style.setProperty("--rgb-scrollbar-thumb",E.scrollbarThumbRgb),p.style.setProperty("--rgb-scrollbar-thumb-hover",E.scrollbarThumbHoverRgb),p.style.setProperty("--mode-column-width",`${Ki.layout.modeColumnWidthPx}px`),p.style.setProperty("--mode-toggle-width",`${Ki.layout.modeToggleWidthPx}px`)},[]);const z=p=>{D(E=>[km(p),...E].slice(0,300))};function zr(){let p=0;Pt(Gi[0]);const E=window.setInterval(()=>{p=Math.min(p+1,Gi.length-1),Pt(Gi[p])},650);return()=>window.clearInterval(E)}x.useEffect(()=>{(async()=>{const C=localStorage.getItem($c);if(C)try{const ne=JSON.parse(C);m(Ke=>({...Ke,llmProvider:ne.llmProvider==="local"?"local":"openai",model:ne.model??Ke.model,baseUrl:ne.baseUrl??Ke.baseUrl,temperature:ne.temperature??Ke.temperature,maxOutputTokens:ne.maxOutputTokens??Ke.maxOutputTokens}))}catch{}try{const ne=await ke.loadSharedConnectionConfig();ne.connection&&ne.connection.llmProvider==="local"&&(m(Ke=>({...Ke,llmProvider:"local",model:ne.connection?.model??Ke.model,baseUrl:ne.connection?.baseUrl??Ke.baseUrl,temperature:ne.connection?.temperature??Ke.temperature,maxOutputTokens:ne.connection?.maxOutputTokens??Ke.maxOutputTokens})),z(`Shared local LLM config loaded: ${ne.connection.model}`))}catch(ne){z(`Shared local config load error: ${ne instanceof Error?ne.message:String(ne)}`)}finally{Vt.current=!0}})();const E=localStorage.getItem(Fc);if(E)try{const C=JSON.parse(E);(C.uiMode==="assistant"||C.uiMode==="decomposition"||C.uiMode==="autoruns")&&Ut(C.uiMode),C.activeTab&&jm.includes(C.activeTab)&&oe(C.activeTab),typeof C.showAutorunsAssistantMode=="boolean"&&In(C.showAutorunsAssistantMode),typeof C.showAutorunsDecompositionMode=="boolean"&&fr(C.showAutorunsDecompositionMode),typeof C.showAutorunsProgressMode=="boolean"&&Zt(C.showAutorunsProgressMode),typeof C.showAutorunsCommentsMode=="boolean"&&hn(C.showAutorunsCommentsMode),typeof C.showAssistantConnectionMode=="boolean"&&en(C.showAssistantConnectionMode),typeof C.showAssistantPromptMode=="boolean"&&mr(C.showAssistantPromptMode),typeof C.showAssistantChatMode=="boolean"&&pr(C.showAssistantChatMode),typeof C.showAssistantCommentsMode=="boolean"&&tn(C.showAssistantCommentsMode),typeof C.showAssistantSamMode=="boolean"&&hr(C.showAssistantSamMode),typeof C.showDecompositionConnectionMode=="boolean"&&vn(C.showDecompositionConnectionMode),typeof C.showDecompositionPromptMode=="boolean"&&ot(C.showDecompositionPromptMode),typeof C.showDecompositionQueryMode=="boolean"&&xn(C.showDecompositionQueryMode),typeof C.showDecompositionOutputMode=="boolean"&&sn(C.showDecompositionOutputMode),typeof C.showDecompositionMetricsMode=="boolean"&&zn(C.showDecompositionMetricsMode),typeof C.showDecompositionHistoryMode=="boolean"&&Ht(C.showDecompositionHistoryMode),typeof C.showDecompositionRuntimeMode=="boolean"&&Jn(C.showDecompositionRuntimeMode),C.prompts&&(N(ne=>({...ne,...C.prompts})),Fn.current=!0)}catch{}xr(),Xn(),Me()},[]),x.useEffect(()=>{if(!Vt.current||i.llmProvider!=="local")return;const p=window.setTimeout(()=>{ke.saveSharedConnectionConfig(i).catch(E=>z(`Shared local config sync error: ${E instanceof Error?E.message:String(E)}`))},250);return()=>window.clearTimeout(p)},[i.baseUrl,i.llmProvider,i.maxOutputTokens,i.model,i.temperature]);async function xr(){try{const p=await ke.loadHistory();I(p.items??[])}catch(p){z(`History load error: ${p instanceof Error?p.message:String(p)}`)}}async function Xn(){try{const E=(await ke.loadPresets()).presets??[];if($e(E),Fn.current){Rt.current=!0;return}if(Rt.current)return;const C=E.find(ne=>ne.prompt_version===Ji)??E.find(ne=>ne.id==="default-normalizer-v2_0_2");if(!C){Rt.current=!0,z(`Preset autoload skipped: ${Ji} not found.`);return}Je(C.id),Ie(f),N({systemPrompt:C.systemPrompt,developerPrompt:C.developerPrompt,domainPrompt:C.domainPrompt,schemaNotes:C.schemaNotes??"",fewShotExamples:C.fewShotExamples??""}),Rt.current=!0,z(`Preset autoloaded: ${C.name} (${C.prompt_version}).`)}catch(p){z(`Presets load error: ${p instanceof Error?p.message:String(p)}`)}}async function Me(){try{const p=await ke.listRuns();W(p.items??[])}catch(p){z(`Runs load error: ${p instanceof Error?p.message:String(p)}`)}}function $r(){if(localStorage.setItem($c,JSON.stringify({model:i.model,llmProvider:i.llmProvider,baseUrl:i.baseUrl,temperature:i.temperature,maxOutputTokens:i.maxOutputTokens})),i.llmProvider==="local"){ke.saveSharedConnectionConfig(i).then(()=>{z("Local config saved and synced to shared agent config (without API key).")}).catch(p=>{z(`Local config saved, but shared sync failed: ${p instanceof Error?p.message:String(p)}`)});return}z("Local config saved (without API key).")}function _t(){localStorage.setItem(Fc,JSON.stringify({uiMode:Pe,activeTab:B,showAutorunsAssistantMode:Kn,showAutorunsDecompositionMode:Ln,showAutorunsProgressMode:Xt,showAutorunsCommentsMode:Gn,showAssistantConnectionMode:bt,showAssistantPromptMode:gn,showAssistantChatMode:On,showAssistantCommentsMode:Dn,showAssistantSamMode:nn,showDecompositionConnectionMode:Bt,showDecompositionPromptMode:be,showDecompositionQueryMode:yn,showDecompositionOutputMode:rn,showDecompositionMetricsMode:Nt,showDecompositionHistoryMode:on,showDecompositionRuntimeMode:gt,prompts:f})),window.dispatchEvent(new CustomEvent(Sm)),z("UI layout and prompts saved.")}async function _r(){se(!0),ee("");try{const p=await ke.testConnection(i);p.provider==="local"?p.model_found===!0?(_e(`LOCAL OK - ${p.model}`),z(`Local model is available: ${p.model} (catalog size=${p.models_count??"n/a"}).`)):p.model_found===!1?(_e(`LOCAL OK, model not loaded - ${p.model}`),z(`Local server is reachable, but model '${p.model}' is not in loaded catalog. Use 'Load model list' and select one of loaded models.`)):(_e(`LOCAL OK (model list unavailable) - ${p.model}`),z("Local server is reachable, but model catalog could not be verified.")):(_e(`OPENAI OK - ${p.model}`),z(`OpenAI connection ok: ${p.model}`))}catch(p){const E=p instanceof Error?p.message:String(p);_e("Connection error"),ee(`Test connection: ${E}`),z(`Test connection error: ${E}`)}finally{se(!1)}}async function Sn(){Te(!0);try{const E=(await ke.listModels(i)).models??[];G(E),E.length>0&&m(C=>C.model&&E.includes(C.model)?C:{...C,model:E[0]}),z(`Model catalog loaded (${i.llmProvider}): ${E.length} items.`)}catch(p){const E=p instanceof Error?p.message:String(p);z(`Load model list error: ${E}`)}finally{Te(!1)}}x.useEffect(()=>{G([])},[i.llmProvider,i.baseUrl]);async function le(p){se(!0),ee("");try{const E=await ke.normalize({connection:i,prompts:f,promptVersion:"normalizer_v2_0_2",query:{userQuestion:j.userQuestion,periodHint:j.periodHint,businessContext:j.businessContext,expectedRoute:j.expectedRoute},saveAsTestCase:p,useMock:Re});Y(E),oe("normalized"),z(`Normalize done: trace=${E.trace_id}, validation=${E.validation.passed?"passed":"failed"}`),xr()}catch(E){const C=E instanceof Error?E.message:String(E);ee(`Normalize: ${C}`),z(`Normalize error: ${C}`)}finally{se(!1)}}function Un(){const p=Oe.find(E=>E.id===We);if(!p){z("Preset is not selected.");return}Ie(f),N({systemPrompt:p.systemPrompt,developerPrompt:p.developerPrompt,domainPrompt:p.domainPrompt,schemaNotes:p.schemaNotes??"",fewShotExamples:p.fewShotExamples??""}),z(`Preset loaded: ${p.name}`)}async function At(){try{await ke.savePreset({name:et||"NDC preset",prompt_version:"normalizer_v2_0_2",systemPrompt:f.systemPrompt,developerPrompt:f.developerPrompt,domainPrompt:f.domainPrompt,schemaNotes:f.schemaNotes,fewShotExamples:f.fewShotExamples}),z("Preset saved."),await Xn()}catch(p){z(`Preset save error: ${p instanceof Error?p.message:String(p)}`)}}function ct(){N(zc),z("Prompt panel reset to defaults.")}function es(){const p=Cm(f,ie);me(p),z(p)}function St(){const p=j.batchQuestionsRaw.split(";").map(E=>E.trim()).filter(Boolean).join(` - -`);p&&(T(E=>({...E,batchQuestionsRaw:p})),z("Batch field formatted: `;` converted to blank-line separators."))}async function Ns(p){try{const C=(await ke.loadTrace(p)).trace,ne=C.parsed_normalized_json??null;Y({trace_id:String(C.trace_id??p),ok:!!C.validation_result?.passed,normalized:ne,route_hint_summary:C.route_hint_summary??(ne?{route_hint:ne.route_hint??null,confidence:ne.confidence?.route_hint??null}:null),raw_model_output:C.raw_model_response??{},validation:C.validation_result??{passed:!1,errors:["validation not found"]},usage:C.usage??{input_tokens:0,output_tokens:0,total_tokens:0},latency_ms:Number(C.latency_ms??0),prompt_version:String(C.prompt_version??"unknown"),schema_version:String(C.schema_version??"unknown")}),oe("raw"),ee(""),z(`Trace opened: ${p}`)}catch(E){const C=E instanceof Error?E.message:String(E);ee(`Trace: ${C}`),z(`Trace open error ${p}: ${C}`)}}async function ts(){try{const p=await ke.startRun();h(p.run.runId),z(`Run started: ${p.run.runId}`),z("Tip: start run does not execute normalize by itself. Use 'Run eval v2.0.2' button."),await Me()}catch(p){z(`Run start error: ${p instanceof Error?p.message:String(p)}`)}}async function ns(){if($)try{await ke.finishRun($),z(`Run finished: ${$}`),await Me()}catch(p){z(`Run finish error: ${p instanceof Error?p.message:String(p)}`)}}async function Zn(){de(!0),ee("");try{z("Starting eval in v2 contour.");const p=j.batchQuestionsRaw.trim()||j.userQuestion.trim();if(!p)throw new Error("Fill batch field or Raw user question first.");const E=await ke.runEval({connection:i,prompts:f,promptVersion:"normalizer_v2_0_2",mode:"single-pass-strict",rawQuestions:p,useMock:Re});Se(E.report),z("Eval v2.0.2 run finished.");const C=E.report;if(C.run_id&&z(`Eval run id: ${C.run_id}`),C.metrics){const ne=C.metrics;z(`Eval metrics v2.0.2: schema=${ne.schema_validation_pass_rate??"n/a"}%, route_accuracy=${ne.route_resolution_accuracy??"n/a"}%, no_route_precision=${ne.no_route_precision??"n/a"}%, state_consistency=${ne.execution_state_consistency_rate??"n/a"}%`)}await xr()}catch(p){const E=p instanceof Error?p.message:String(p);E.includes("Legacy eval runner supports normalized_query_v1 only")?(Se({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"]}),z("Backend is legacy-only for eval right now. Showing v2 pilot plan.")):(ee(`Eval: ${E}`),z(`Eval run error: ${E}`))}finally{de(!1)}}async function er(){try{const p=JSON.stringify(fe??{},null,2);await navigator.clipboard.writeText(p),z("Eval report copied to clipboard.")}catch(p){z(`Eval report copy error: ${p instanceof Error?p.message:String(p)}`)}}const Tt=x.useMemo(()=>{const p=new Map;for(const E of lt)E.message_id&&p.set(E.message_id,E);return p},[lt]),wt=Ce.messageIndex>=0?_n[Ce.messageIndex]??null:null,Fr=x.useMemo(()=>{if(Ce.messageIndex<0)return null;for(let p=Ce.messageIndex-1;p>=0;p-=1){const E=_n[p];if(E?.role==="user")return E}return null},[Ce.messageIndex,_n]);async function jt(p){if(!p.trim()){xe([]);return}Mt(!0);try{const E=await ke.loadAssistantAnnotations({session_id:p,limit:400});xe(E.items??[])}catch(E){const C=E instanceof Error?E.message:String(E);z(`Assistant annotations load error: ${C}`)}finally{Mt(!1)}}function bn(p){it(E=>E.saving&&!p?.force?E:{open:!1,messageIndex:-1,rating:3,comment:"",annotationAuthor:Yi,saving:!1,error:""})}function Bn(p,E){if(p.role!=="assistant")return;const C=qe.trim(),ne=String(p.session_id??"").trim();if(!(C||ne)){ce("Сначала получите ответ ассистента в активной сессии.");return}!C&&ne&&gr(ne);const Hn=Tt.get(p.message_id)??null;it({open:!0,messageIndex:E,rating:Hn?.rating??3,comment:Hn?.comment??"",annotationAuthor:Hn?.annotation_author??Yi,saving:!1,error:""})}function It(p){return p.role==="assistant"}function Qt(p){return p.role==="assistant"&&Tt.has(p.message_id)}async function Wt(){if(!qe.trim()){it(p=>({...p,error:"Сессия ассистента не найдена."}));return}if(!(Ce.messageIndex<0)){if(!Ce.comment.trim()){it(p=>({...p,error:"Добавьте комментарий."}));return}it(p=>({...p,saving:!0,error:""}));try{const p=await ke.saveAssistantAnnotation({session_id:qe,message_index:Ce.messageIndex,rating:Ce.rating,comment:Ce.comment.trim(),annotation_author:Ce.annotationAuthor.trim()||void 0});xe(E=>{const C=[...E],ne=C.findIndex(Ke=>Ke.annotation_id===p.annotation.annotation_id);return ne>=0?C[ne]=p.annotation:C.unshift(p.annotation),C.sort((Ke,Hn)=>Date.parse(Hn.updated_at)-Date.parse(Ke.updated_at))}),bn({force:!0})}catch(p){const E=p instanceof Error?p.message:String(p);it(C=>({...C,saving:!1,error:E}))}}}function tr(){gr(""),vr([]),yr(""),Yn(null),Pt(""),ce(""),xe([]),bn({force:!0}),z("Assistant session reset.")}async function we(){const p=Nm(ln,Et);if(!p)return;Zr(!0),ce(""),yr(""),vr(C=>[...C,{message_id:`local-${Date.now()}`,session_id:qe||"pending",role:"user",text:p,reply_type:null,created_at:new Date().toISOString(),trace_id:null,debug:null}]);const E=zr();try{const C=await ke.sendAssistantMessage({connection:i,prompts:f,userMessage:p,sessionId:qe||void 0,promptVersion:Uc,useMock:Re});gr(C.session_id),vr(C.conversation),Pt("Ответ готов"),await jt(C.session_id),z(`Assistant reply received: trace=${C.debug.trace_id}`)}catch(C){const ne=C instanceof Error?C.message:String(C);ce(ne),Pt("Ошибка ассистента"),z(`Assistant error: ${ne}`)}finally{E(),Zr(!1)}}return x.useEffect(()=>{if(!qe.trim()){xe([]);return}jt(qe)},[qe]),x.useEffect(()=>{if(!$){J([]);return}ke.runTrace($).then(p=>J(p.items)).catch(p=>z(`Run trace error: ${p instanceof Error?p.message:String(p)}`))},[$]),o.jsxs("main",{className:`app-root ${Pe==="assistant"||Pe==="decomposition"||Pe==="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:Pe==="assistant"?"tab active":"tab",onClick:()=>Ut("assistant"),children:"Ассистент"}),o.jsx("button",{type:"button",className:Pe==="decomposition"?"tab active":"tab",onClick:()=>Ut("decomposition"),children:"Декомпозиция"}),o.jsx("button",{type:"button",className:Pe==="autoruns"?"tab active":"tab",onClick:()=>Ut("autoruns"),children:"История автопрогонов"}),o.jsx("button",{type:"button",className:"tab",onClick:_t,children:"Сохранить"})]}),Pe==="assistant"?o.jsxs("div",{className:"mode-switch-row mode-switch-row-right",children:[o.jsx("button",{type:"button",className:bt?"tab active":"tab",onClick:()=>en(p=>!p),children:"LLM Connector"}),o.jsx("button",{type:"button",className:gn?"tab active":"tab",onClick:()=>mr(p=>!p),children:"Prompt Manager"}),o.jsx("button",{type:"button",className:On?"tab active":"tab",onClick:()=>pr(p=>!p),children:"Режим ассистента"}),o.jsx("button",{type:"button",className:Dn?"tab active":"tab",onClick:()=>tn(p=>!p),children:"Комментарии ассистента"}),o.jsx("button",{type:"button",className:nn?"tab active":"tab",onClick:()=>hr(p=>!p),children:"SAM"})]}):Pe==="decomposition"?o.jsxs("div",{className:"mode-switch-row mode-switch-row-right",children:[o.jsx("button",{type:"button",className:Bt?"tab active":"tab",onClick:()=>vn(p=>!p),children:"LLM"}),o.jsx("button",{type:"button",className:be?"tab active":"tab",onClick:()=>ot(p=>!p),children:"Prompt"}),o.jsx("button",{type:"button",className:yn?"tab active":"tab",onClick:()=>xn(p=>!p),children:"Запрос"}),o.jsx("button",{type:"button",className:rn?"tab active":"tab",onClick:()=>sn(p=>!p),children:"Выход"}),o.jsx("button",{type:"button",className:Nt?"tab active":"tab",onClick:()=>zn(p=>!p),children:"Метрики"}),o.jsx("button",{type:"button",className:on?"tab active":"tab",onClick:()=>Ht(p=>!p),children:"История"}),o.jsx("button",{type:"button",className:gt?"tab active":"tab",onClick:()=>Jn(p=>!p),children:"NDC Run Monitor"})]}):Pe==="autoruns"?o.jsxs("div",{className:"mode-switch-row mode-switch-row-right",children:[o.jsx("button",{type:"button",className:Kn?"tab active":"tab",onClick:()=>In(p=>!p),children:"Режим ассистента"}),o.jsx("button",{type:"button",className:Ln?"tab active":"tab",onClick:()=>fr(p=>!p),children:"Режим декомпозиции"}),o.jsx("button",{type:"button",className:Xt?"tab active":"tab",onClick:()=>Zt(p=>!p),children:"Прогресс/регресс"}),o.jsx("button",{type:"button",className:Gn?"tab active":"tab",onClick:()=>hn(p=>!p),children:"Комментарии"})]}):null]}),Pe==="assistant"?o.jsx("div",{className:"layout-grid layout-grid-mode-columns",children:o.jsxs("div",{className:"mode-columns",children:[bt?o.jsx("div",{className:"mode-col",children:o.jsx(Oc,{value:i,modelOptions:ve,modelsBusy:te,onChange:m,onReloadModels:Sn,onSaveLocalConfig:$r,onTestConnection:_r,lastStatus:ae,busy:Ae||vt})}):null,gn?o.jsx("div",{className:"mode-col mode-col-wide",children:o.jsx(Dc,{value:f,onChange:N,presets:Oe,selectedPresetId:We,onSelectPreset:Je,onLoadPreset:Un,onSavePreset:At,onResetDefaults:ct,onDiffPrevious:es,presetName:et,onPresetNameChange:st,diffSummary:Fe})}):null,On?o.jsx("div",{className:"mode-col mode-col-xwide",children:o.jsx(Hc,{sessionId:qe,conversation:_n,inputValue:ln,onInputChange:yr,selectedContextChip:Et,onSelectContextChip:Yn,onClearContextChip:()=>Yn(null),useMock:Re,onUseMockChange:pe,onSend:we,onClear:tr,busy:vt,statusText:yt,errorMessage:$n,showCommentAction:!0,onCommentAssistantMessage:Bn,isAssistantMessageCommented:Qt,canCommentAssistantMessage:It})}):null,Dn?o.jsx("div",{className:"mode-col",children:o.jsx(Tn,{className:"assistant-comments-frame",title:"Комментарии ассистента",children:o.jsxs("div",{className:"assistant-comments-shell",children:[o.jsxs("div",{className:"assistant-comments-toolbar",children:[o.jsx("span",{className:"muted",children:qe?`session: ${qe}`:"Сессия не запущена"}),o.jsx("button",{type:"button",className:"tab",onClick:()=>{jt(qe)},disabled:!qe||xt,children:xt?"Обновляю...":"Обновить"})]}),o.jsxs("div",{className:"assistant-comments-list",children:[qe?null:o.jsx("p",{className:"muted",children:"Появится после первого ответа ассистента."}),qe&<.length===0&&!xt?o.jsx("p",{className:"muted",children:"Комментариев по этой сессии пока нет."}):null,lt.map(p=>o.jsxs("article",{className:"assistant-comment-item",children:[o.jsxs("div",{className:"assistant-comment-head",children:[o.jsx("strong",{children:`${"●".repeat(Math.max(1,Math.min(5,Math.round(p.rating))))}${"○".repeat(Math.max(0,5-Math.round(p.rating)))}`}),o.jsx("span",{children:new Date(p.updated_at).toLocaleString("ru-RU")})]}),p.context.question_text?o.jsxs("p",{children:["Q: ",p.context.question_text]}):null,p.context.answer_text?o.jsxs("p",{children:["A: ",p.context.answer_text]}):null,o.jsx("p",{children:p.comment}),o.jsxs("div",{className:"assistant-comment-meta",children:[p.context.trace_id?o.jsx("span",{children:`trace=${p.context.trace_id}`}):null,p.context.reply_type?o.jsx("span",{children:`reply_type=${p.context.reply_type}`}):null]})]},p.annotation_id))]})]})})}):null,nn?o.jsx("div",{className:"mode-col",children:o.jsx(Of,{sessionId:qe,conversation:_n,statusText:yt,errorMessage:$n,useMock:Re,appLogs:q})}):null,!bt&&!gn&&!On&&!Dn&&!nn?o.jsx("div",{className:"mode-columns-empty",children:"Все панели режима ассистента скрыты. Включите нужные блоки справа в шапке."}):null]})}):Pe==="decomposition"?o.jsx("div",{className:"layout-grid layout-grid-mode-columns",children:o.jsxs("div",{className:"mode-columns",children:[Bt?o.jsx("div",{className:"mode-col",children:o.jsx(Oc,{value:i,modelOptions:ve,modelsBusy:te,onChange:m,onReloadModels:Sn,onSaveLocalConfig:$r,onTestConnection:_r,lastStatus:ae,busy:Ae})}):null,be?o.jsx("div",{className:"mode-col mode-col-wide",children:o.jsx(Dc,{value:f,onChange:N,presets:Oe,selectedPresetId:We,onSelectPreset:Je,onLoadPreset:Un,onSavePreset:At,onResetDefaults:ct,onDiffPrevious:es,presetName:et,onPresetNameChange:st,diffSummary:Fe})}):null,yn?o.jsx("div",{className:"mode-col",children:o.jsx(vm,{value:j,onChange:T,onApplyBatchFormat:St,onNormalize:le,busy:Ae,useMock:Re,onUseMockChange:pe,errorMessage:ye})}):null,rn?o.jsx("div",{className:"mode-col mode-col-xwide",children:o.jsx(gm,{tab:B,onTabChange:oe,result:F,appLogs:q})}):null,Nt?o.jsx("div",{className:"mode-col",children:o.jsx(mm,{result:F})}):null,on?o.jsx("div",{className:"mode-col",children:o.jsx(fm,{items:U,onRefresh:xr,onOpenTrace:Ns})}):null,gt?o.jsx("div",{className:"mode-col mode-col-xwide",children:o.jsx(ym,{runs:O,selectedRunId:$,onSelectRun:h,onStartRun:ts,onFinishRun:ns,onRefreshRuns:Me,onRunEval:Zn,onCopyEvalReport:er,evalBusy:ue,traceItems:k,evalReport:fe})}):null,!Bt&&!be&&!yn&&!rn&&!Nt&&!on&&!gt?o.jsx("div",{className:"mode-columns-empty",children:"Все панели режима декомпозиции скрыты. Включите нужные блоки справа в шапке."}):null]})}):o.jsx("div",{className:"layout-grid layout-grid-autoruns",children:o.jsx(am,{connection:i,prompts:f,assistantPromptVersion:Uc,decompositionPromptVersion:Ji,showAssistantMode:Kn,showDecompositionMode:Ln,showProgressMode:Xt,showCommentsMode:Gn,onLog:z})}),Ce.open?o.jsx("div",{className:"autoruns-comment-modal-backdrop",onClick:p=>{p.target===p.currentTarget&&bn()},children:o.jsxs("div",{className:"autoruns-comment-modal",children:[o.jsx("h3",{children:"Комментарий к ответу ассистента"}),o.jsx("p",{className:"muted",children:"Эта разметка хранится отдельно от комментариев автопрогонов."}),Fr?o.jsxs("details",{className:"autoruns-prompt-details",open:!0,children:[o.jsx("summary",{children:"Вопрос пользователя"}),o.jsx("p",{className:"autoruns-comment-quote",children:Fr.text})]}):null,wt?o.jsxs("details",{className:"autoruns-prompt-details",open:!0,children:[o.jsx("summary",{children:"Ответ ассистента"}),o.jsx("p",{className:"autoruns-comment-quote",children:wt.text})]}):null,o.jsx("div",{className:"autoruns-rating-row",role:"group","aria-label":"Рейтинг ответа",children:[1,2,3,4,5].map(p=>o.jsx("button",{type:"button",className:Ce.rating>=p?"autoruns-rating-dot active":"autoruns-rating-dot",onClick:()=>it(E=>({...E,rating:p})),disabled:Ce.saving,"aria-label":`Оценка ${p}`,children:Ce.rating>=p?"●":"○"},p))}),o.jsx("div",{className:"autoruns-form-grid",children:o.jsxs("label",{children:["Автор комментария",o.jsx("input",{value:Ce.annotationAuthor,onChange:p=>it(E=>({...E,annotationAuthor:p.target.value})),placeholder:"manual_reviewer",disabled:Ce.saving})]})}),o.jsxs("label",{children:["Комментарий",o.jsx("textarea",{value:Ce.comment,onChange:p=>it(E=>({...E,comment:p.target.value})),placeholder:"Что именно не так в ответе и что проверить.",rows:4,disabled:Ce.saving})]}),Ce.error?o.jsx("p",{className:"error-text",children:Ce.error}):null,o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",onClick:()=>{Wt()},disabled:Ce.saving,children:Ce.saving?"Сохраняю...":"Готово"}),o.jsx("button",{type:"button",className:"tab",onClick:()=>bn(),disabled:Ce.saving,children:"Отмена"})]})]})}):null]})}Tf.createRoot(document.getElementById("root")).render(o.jsx(Cf.StrictMode,{children:o.jsx(Em,{})})); diff --git a/llm_normalizer/frontend/dist/assets/index-VJV2AL7G.js b/llm_normalizer/frontend/dist/assets/index-VJV2AL7G.js new file mode 100644 index 0000000..4596d5a --- /dev/null +++ b/llm_normalizer/frontend/dist/assets/index-VJV2AL7G.js @@ -0,0 +1,26 @@ +(function(){const p=document.createElement("link").relList;if(p&&p.supports&&p.supports("modulepreload"))return;for(const j of document.querySelectorAll('link[rel="modulepreload"]'))N(j);new MutationObserver(j=>{for(const T of j)if(T.type==="childList")for(const $ of T.addedNodes)$.tagName==="LINK"&&$.rel==="modulepreload"&&N($)}).observe(document,{childList:!0,subtree:!0});function f(j){const T={};return j.integrity&&(T.integrity=j.integrity),j.referrerPolicy&&(T.referrerPolicy=j.referrerPolicy),j.crossOrigin==="use-credentials"?T.credentials="include":j.crossOrigin==="anonymous"?T.credentials="omit":T.credentials="same-origin",T}function N(j){if(j.ep)return;j.ep=!0;const T=f(j);fetch(j.href,T)}})();function Vc(i){return i&&i.__esModule&&Object.prototype.hasOwnProperty.call(i,"default")?i.default:i}var Wi={exports:{}},mo={},qi={exports:{}},he={};var Sc;function wf(){if(Sc)return he;Sc=1;var i=Symbol.for("react.element"),p=Symbol.for("react.portal"),f=Symbol.for("react.fragment"),N=Symbol.for("react.strict_mode"),j=Symbol.for("react.profiler"),T=Symbol.for("react.provider"),$=Symbol.for("react.context"),Y=Symbol.for("react.forward_ref"),b=Symbol.for("react.suspense"),Q=Symbol.for("react.memo"),G=Symbol.for("react.lazy"),z=Symbol.iterator;function ne(y){return y===null||typeof y!="object"?null:(y=z&&y[z]||y["@@iterator"],typeof y=="function"?y:null)}var Ne={isMounted:function(){return!1},enqueueForceUpdate:function(){},enqueueReplaceState:function(){},enqueueSetState:function(){}},le=Object.assign,ce={};function me(y,C,J){this.props=y,this.context=C,this.refs=ce,this.updater=J||Ne}me.prototype.isReactComponent={},me.prototype.setState=function(y,C){if(typeof y!="object"&&typeof y!="function"&&y!=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,y,C,"setState")},me.prototype.forceUpdate=function(y){this.updater.enqueueForceUpdate(this,y,"forceUpdate")};function Qe(){}Qe.prototype=me.prototype;function De(y,C,J){this.props=y,this.context=C,this.refs=ce,this.updater=J||Ne}var We=De.prototype=new Qe;We.constructor=De,le(We,me.prototype),We.isPureReactComponent=!0;var je=Array.isArray,W=Object.prototype.hasOwnProperty,de={current:null},Me={key:!0,ref:!0,__self:!0,__source:!0};function Ze(y,C,J){var oe,pe={},ae=null,Se=null;if(C!=null)for(oe in C.ref!==void 0&&(Se=C.ref),C.key!==void 0&&(ae=""+C.key),C)W.call(C,oe)&&!Me.hasOwnProperty(oe)&&(pe[oe]=C[oe]);var ge=arguments.length-2;if(ge===1)pe.children=J;else if(1>>1,C=I[y];if(0>>1;yj(pe,O))aej(Se,pe)?(I[y]=Se,I[ae]=O,y=ae):(I[y]=pe,I[oe]=O,y=oe);else if(aej(Se,O))I[y]=Se,I[ae]=O,y=ae;else break e}}return X}function j(I,X){var O=I.sortIndex-X.sortIndex;return O!==0?O:I.id-X.id}if(typeof performance=="object"&&typeof performance.now=="function"){var T=performance;i.unstable_now=function(){return T.now()}}else{var $=Date,Y=$.now();i.unstable_now=function(){return $.now()-Y}}var b=[],Q=[],G=1,z=null,ne=3,Ne=!1,le=!1,ce=!1,me=typeof setTimeout=="function"?setTimeout:null,Qe=typeof clearTimeout=="function"?clearTimeout:null,De=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(I){for(var X=f(Q);X!==null;){if(X.callback===null)N(Q);else if(X.startTime<=I)N(Q),X.sortIndex=X.expirationTime,p(b,X);else break;X=f(Q)}}function je(I){if(ce=!1,We(I),!le)if(f(b)!==null)le=!0,Te(W);else{var X=f(Q);X!==null&&Ae(je,X.startTime-I)}}function W(I,X){le=!1,ce&&(ce=!1,Qe(Ze),Ze=-1),Ne=!0;var O=ne;try{for(We(X),z=f(b);z!==null&&(!(z.expirationTime>X)||I&&!Et());){var y=z.callback;if(typeof y=="function"){z.callback=null,ne=z.priorityLevel;var C=y(z.expirationTime<=X);X=i.unstable_now(),typeof C=="function"?z.callback=C:z===f(b)&&N(b),We(X)}else N(b);z=f(b)}if(z!==null)var J=!0;else{var oe=f(Q);oe!==null&&Ae(je,oe.startTime-X),J=!1}return J}finally{z=null,ne=O,Ne=!1}}var de=!1,Me=null,Ze=-1,ct=5,st=-1;function Et(){return!(i.unstable_now()-stI||125y?(I.sortIndex=O,p(Q,I),f(b)===null&&I===f(Q)&&(ce?(Qe(Ze),Ze=-1):ce=!0,Ae(je,O-y))):(I.sortIndex=C,p(b,I),le||Ne||(le=!0,Te(W))),I},i.unstable_shouldYield=Et,i.unstable_wrapCallback=function(I){var X=ne;return function(){var O=ne;ne=X;try{return I.apply(this,arguments)}finally{ne=O}}}})(Ji)),Ji}var Nc;function Ef(){return Nc||(Nc=1,Gi.exports=Nf()),Gi.exports}var Ec;function Pf(){if(Ec)return Wt;Ec=1;var i=aa(),p=Ef();function f(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"),b=Object.prototype.hasOwnProperty,Q=/^[: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]*$/,G={},z={};function ne(e){return b.call(z,e)?!0:b.call(G,e)?!1:Q.test(e)?z[e]=!0:(G[e]=!0,!1)}function Ne(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 le(e,t,n,r){if(t===null||typeof t>"u"||Ne(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 ce(e,t,n,r,s,l,u){this.acceptsBooleans=t===2||t===3||t===4,this.attributeName=r,this.attributeNamespace=s,this.mustUseProperty=n,this.propertyName=e,this.type=t,this.sanitizeURL=l,this.removeEmptyString=u}var me={};"children dangerouslySetInnerHTML defaultValue defaultChecked innerHTML suppressContentEditableWarning suppressHydrationWarning style".split(" ").forEach(function(e){me[e]=new ce(e,0,!1,e,null,!1,!1)}),[["acceptCharset","accept-charset"],["className","class"],["htmlFor","for"],["httpEquiv","http-equiv"]].forEach(function(e){var t=e[0];me[t]=new ce(t,1,!1,e[1],null,!1,!1)}),["contentEditable","draggable","spellCheck","value"].forEach(function(e){me[e]=new ce(e,2,!1,e.toLowerCase(),null,!1,!1)}),["autoReverse","externalResourcesRequired","focusable","preserveAlpha"].forEach(function(e){me[e]=new ce(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){me[e]=new ce(e,3,!1,e.toLowerCase(),null,!1,!1)}),["checked","multiple","muted","selected"].forEach(function(e){me[e]=new ce(e,3,!0,e,null,!1,!1)}),["capture","download"].forEach(function(e){me[e]=new ce(e,4,!1,e,null,!1,!1)}),["cols","rows","size","span"].forEach(function(e){me[e]=new ce(e,6,!1,e,null,!1,!1)}),["rowSpan","start"].forEach(function(e){me[e]=new ce(e,5,!1,e.toLowerCase(),null,!1,!1)});var Qe=/[\-:]([a-z])/g;function De(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(Qe,De);me[t]=new ce(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(Qe,De);me[t]=new ce(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(Qe,De);me[t]=new ce(t,1,!1,e,"http://www.w3.org/XML/1998/namespace",!1,!1)}),["tabIndex","crossOrigin"].forEach(function(e){me[e]=new ce(e,1,!1,e.toLowerCase(),null,!1,!1)}),me.xlinkHref=new ce("xlinkHref",1,!1,"xlink:href","http://www.w3.org/1999/xlink",!0,!1),["src","href","action","formAction"].forEach(function(e){me[e]=new ce(e,1,!1,e.toLowerCase(),null,!0,!0)});function We(e,t,n,r){var s=me.hasOwnProperty(t)?me[t]:null;(s!==null?s.type!==0:r||!(2d||s[u]!==l[d]){var h=` +`+s[u].replace(" at new "," at ");return e.displayName&&h.includes("")&&(h=h.replace("",e.displayName)),h}while(1<=u&&0<=d);break}}}finally{J=!1,Error.prepareStackTrace=n}return(e=e?e.displayName||e.name:"")?C(e):""}function pe(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=oe(e.type,!1),e;case 11:return e=oe(e.type.render,!1),e;case 1:return e=oe(e.type,!0),e;default:return""}}function ae(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 Me:return"Fragment";case de:return"Portal";case ct:return"Profiler";case Ze:return"StrictMode";case Le:return"Suspense";case H:return"SuspenseList"}if(typeof e=="object")switch(e.$$typeof){case Et:return(e.displayName||"Context")+".Consumer";case st:return(e._context.displayName||"Context")+".Provider";case ie:var t=e.render;return e=e.displayName,e||(e=t.displayName||t.name||"",e=e!==""?"ForwardRef("+e+")":"ForwardRef"),e;case Fe:return t=e.displayName||null,t!==null?t:ae(e.type)||"Memo";case Te:t=e._payload,e=e._init;try{return ae(e(t))}catch{}}return null}function Se(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 ae(t);case 8:return t===Ze?"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 ge(e){switch(typeof e){case"boolean":case"number":case"string":case"undefined":return e;case"object":return e;default:return""}}function ve(e){var t=e.type;return(e=e.nodeName)&&e.toLowerCase()==="input"&&(t==="checkbox"||t==="radio")}function Ee(e){var t=ve(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 s=n.get,l=n.set;return Object.defineProperty(e,t,{configurable:!0,get:function(){return s.call(this)},set:function(u){r=""+u,l.call(this,u)}}),Object.defineProperty(e,t,{enumerable:n.enumerable}),{getValue:function(){return r},setValue:function(u){r=""+u},stopTracking:function(){e._valueTracker=null,delete e[t]}}}}function gn(e){e._valueTracker||(e._valueTracker=Ee(e))}function vn(e){if(!e)return!1;var t=e._valueTracker;if(!t)return!0;var n=t.getValue(),r="";return e&&(r=ve(e)?e.checked?"true":"false":e.value),e=r,e!==n?(t.setValue(e),!0):!1}function Yn(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 Ln(e,t){var n=t.checked;return O({},t,{defaultChecked:void 0,defaultValue:void 0,value:void 0,checked:n??e._wrapperState.initialChecked})}function Xn(e,t){var n=t.defaultValue==null?"":t.defaultValue,r=t.checked!=null?t.checked:t.defaultChecked;n=ge(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 Pe(e,t){t=t.checked,t!=null&&We(e,"checked",t,!1)}function qt(e,t){Pe(e,t);var n=ge(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")?dt(e,t.type,n):t.hasOwnProperty("defaultValue")&&dt(e,t.type,ge(t.defaultValue)),t.checked==null&&t.defaultChecked!=null&&(e.defaultChecked=!!t.defaultChecked)}function ot(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 dt(e,t,n){(t!=="number"||Yn(e.ownerDocument)!==e)&&(n==null?e.defaultValue=""+e._wrapperState.initialValue:e.defaultValue!==""+n&&(e.defaultValue=""+n))}var lt=Array.isArray;function yn(e,t,n,r){if(e=e.options,t){t={};for(var s=0;s"+t.valueOf().toString()+"",t=xt.firstChild;e.firstChild;)e.removeChild(e.firstChild);for(;t.firstChild;)e.appendChild(t.firstChild)}});function Oe(e,t){if(t){var n=e.firstChild;if(n&&n===e.lastChild&&n.nodeType===3){n.nodeValue=t;return}}e.textContent=t}var rn={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},On=["Webkit","ms","Moz","O"];Object.keys(rn).forEach(function(e){On.forEach(function(t){t=t+e.charAt(0).toUpperCase()+e.substring(1),rn[t]=rn[e]})});function xr(e,t,n){return t==null||typeof t=="boolean"||t===""?"":n||typeof t!="number"||t===0||rn.hasOwnProperty(e)&&rn[e]?(""+t).trim():t+"px"}function Dn(e,t){e=e.style;for(var n in t)if(t.hasOwnProperty(n)){var r=n.indexOf("--")===0,s=xr(n,t[n],r);n==="float"&&(n="cssFloat"),r?e.setProperty(n,s):e[n]=s}}var Qr=O({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 Sn(e,t){if(t){if(Qr[e]&&(t.children!=null||t.dangerouslySetInnerHTML!=null))throw Error(f(137,e));if(t.dangerouslySetInnerHTML!=null){if(t.children!=null)throw Error(f(60));if(typeof t.dangerouslySetInnerHTML!="object"||!("__html"in t.dangerouslySetInnerHTML))throw Error(f(61))}if(t.style!=null&&typeof t.style!="object")throw Error(f(62))}}function sn(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 Lt=null;function Zn(e){return e=e.target||e.srcElement||window,e.correspondingUseElement&&(e=e.correspondingUseElement),e.nodeType===3?e.parentNode:e}var Kt=null,on=null,Ot=null;function _r(e){if(e=Ys(e)){if(typeof Kt!="function")throw Error(f(280));var t=e.stateNode;t&&(t=ko(t),Kt(e.stateNode,e.type,t))}}function zn(e){on?Ot?Ot.push(e):Ot=[e]:on=e}function Sr(){if(on){var e=on,t=Ot;if(Ot=on=null,_r(e),t)for(e=0;e>>=0,e===0?32:31-(sr(e)/Hn|0)|0}var $t=64,kn=4194304;function At(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 ke(e,t){var n=e.pendingLanes;if(n===0)return 0;var r=0,s=e.suspendedLanes,l=e.pingedLanes,u=n&268435455;if(u!==0){var d=u&~s;d!==0?r=At(d):(l&=u,l!==0&&(r=At(l)))}else u=n&~s,u!==0?r=At(u):l!==0&&(r=At(l));if(r===0)return 0;if(t!==0&&t!==r&&(t&s)===0&&(s=r&-r,l=t&-t,s>=l||s===16&&(l&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 it(e,t,n){e.pendingLanes|=t,t!==536870912&&(e.suspendedLanes=0,e.pingedLanes=0),e=e.eventTimes,t=31-be(t),e[t]=n}function lr(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=Bs),pa=" ",ha=!1;function ga(e,t){switch(e){case"keyup":return gd.indexOf(t.keyCode)!==-1;case"keydown":return t.keyCode!==229;case"keypress":case"mousedown":case"focusout":return!0;default:return!1}}function va(e){return e=e.detail,typeof e=="object"&&"data"in e?e.data:null}var gs=!1;function yd(e,t){switch(e){case"compositionend":return va(t);case"keypress":return t.which!==32?null:(ha=!0,pa);case"textInput":return e=t.data,e===pa&&ha?null:e;default:return null}}function xd(e,t){if(gs)return e==="compositionend"||!Nl&&ga(e,t)?(e=R(),v=c=a=null,gs=!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=ka(n)}}function Na(e,t){return e&&t?e===t?!0:e&&e.nodeType===3?!1:t&&t.nodeType===3?Na(e,t.parentNode):"contains"in e?e.contains(t):e.compareDocumentPosition?!!(e.compareDocumentPosition(t)&16):!1:!1}function Ea(){for(var e=window,t=Yn();t instanceof e.HTMLIFrameElement;){try{var n=typeof t.contentWindow.location.href=="string"}catch{n=!1}if(n)e=t.contentWindow;else break;t=Yn(e.document)}return t}function Ml(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 Pd(e){var t=Ea(),n=e.focusedElem,r=e.selectionRange;if(t!==n&&n&&n.ownerDocument&&Na(n.ownerDocument.documentElement,n)){if(r!==null&&Ml(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 s=n.textContent.length,l=Math.min(r.start,s);r=r.end===void 0?l:Math.min(r.end,s),!e.extend&&l>r&&(s=r,r=l,l=s),s=Ca(n,l);var u=Ca(n,r);s&&u&&(e.rangeCount!==1||e.anchorNode!==s.node||e.anchorOffset!==s.offset||e.focusNode!==u.node||e.focusOffset!==u.offset)&&(t=t.createRange(),t.setStart(s.node,s.offset),e.removeAllRanges(),l>r?(e.addRange(t),e.extend(u.node,u.offset)):(t.setEnd(u.node,u.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,vs=null,Al=null,Ws=null,Rl=!1;function Pa(e,t,n){var r=n.window===n?n.document:n.nodeType===9?n:n.ownerDocument;Rl||vs==null||vs!==Yn(r)||(r=vs,"selectionStart"in r&&Ml(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}),Ws&&Qs(Ws,r)||(Ws=r,r=So(Al,"onSelect"),0ws||(e.current=Hl[ws],Hl[ws]=null,ws--)}function $e(e,t){ws++,Hl[ws]=e.current,e.current=t}var Ir={},wt=Tr(Ir),bt=Tr(!1),Zr=Ir;function js(e,t){var n=e.type.contextTypes;if(!n)return Ir;var r=e.stateNode;if(r&&r.__reactInternalMemoizedUnmaskedChildContext===t)return r.__reactInternalMemoizedMaskedChildContext;var s={},l;for(l in n)s[l]=t[l];return r&&(e=e.stateNode,e.__reactInternalMemoizedUnmaskedChildContext=t,e.__reactInternalMemoizedMaskedChildContext=s),s}function Bt(e){return e=e.childContextTypes,e!=null}function Co(){He(bt),He(wt)}function Ha(e,t,n){if(wt.current!==Ir)throw Error(f(168));$e(wt,t),$e(bt,n)}function Va(e,t,n){var r=e.stateNode;if(t=t.childContextTypes,typeof r.getChildContext!="function")return n;r=r.getChildContext();for(var s in r)if(!(s in t))throw Error(f(108,Se(e)||"Unknown",s));return O({},n,r)}function No(e){return e=(e=e.stateNode)&&e.__reactInternalMemoizedMergedChildContext||Ir,Zr=wt.current,$e(wt,e),$e(bt,bt.current),!0}function Qa(e,t,n){var r=e.stateNode;if(!r)throw Error(f(169));n?(e=Va(e,t,Zr),r.__reactInternalMemoizedMergedChildContext=e,He(bt),He(wt),$e(wt,e)):He(bt),$e(bt,n)}var cr=null,Eo=!1,Vl=!1;function Wa(e){cr===null?cr=[e]:cr.push(e)}function Ud(e){Eo=!0,Wa(e)}function Lr(){if(!Vl&&cr!==null){Vl=!0;var e=0,t=F;try{var n=cr;for(F=1;e>=u,s-=u,dr=1<<32-be(t)+s|n<se?(ht=ee,ee=null):ht=ee.sibling;var we=P(_,ee,S[se],L);if(we===null){ee===null&&(ee=ht);break}e&&ee&&we.alternate===null&&t(_,ee),g=l(we,g,se),Z===null?K=we:Z.sibling=we,Z=we,ee=ht}if(se===S.length)return n(_,ee),Ve&&ts(_,se),K;if(ee===null){for(;sese?(ht=ee,ee=null):ht=ee.sibling;var Hr=P(_,ee,we.value,L);if(Hr===null){ee===null&&(ee=ht);break}e&&ee&&Hr.alternate===null&&t(_,ee),g=l(Hr,g,se),Z===null?K=Hr:Z.sibling=Hr,Z=Hr,ee=ht}if(we.done)return n(_,ee),Ve&&ts(_,se),K;if(ee===null){for(;!we.done;se++,we=S.next())we=A(_,we.value,L),we!==null&&(g=l(we,g,se),Z===null?K=we:Z.sibling=we,Z=we);return Ve&&ts(_,se),K}for(ee=r(_,ee);!we.done;se++,we=S.next())we=U(ee,_,se,we.value,L),we!==null&&(e&&we.alternate!==null&&ee.delete(we.key===null?se:we.key),g=l(we,g,se),Z===null?K=we:Z.sibling=we,Z=we);return e&&ee.forEach(function(Sf){return t(_,Sf)}),Ve&&ts(_,se),K}function et(_,g,S,L){if(typeof S=="object"&&S!==null&&S.type===Me&&S.key===null&&(S=S.props.children),typeof S=="object"&&S!==null){switch(S.$$typeof){case W:e:{for(var K=S.key,Z=g;Z!==null;){if(Z.key===K){if(K=S.type,K===Me){if(Z.tag===7){n(_,Z.sibling),g=s(Z,S.props.children),g.return=_,_=g;break e}}else if(Z.elementType===K||typeof K=="object"&&K!==null&&K.$$typeof===Te&&Xa(K)===Z.type){n(_,Z.sibling),g=s(Z,S.props),g.ref=Xs(_,Z,S),g.return=_,_=g;break e}n(_,Z);break}else t(_,Z);Z=Z.sibling}S.type===Me?(g=us(S.props.children,_.mode,L,S.key),g.return=_,_=g):(L=tl(S.type,S.key,S.props,null,_.mode,L),L.ref=Xs(_,g,S),L.return=_,_=L)}return u(_);case de:e:{for(Z=S.key;g!==null;){if(g.key===Z)if(g.tag===4&&g.stateNode.containerInfo===S.containerInfo&&g.stateNode.implementation===S.implementation){n(_,g.sibling),g=s(g,S.children||[]),g.return=_,_=g;break e}else{n(_,g);break}else t(_,g);g=g.sibling}g=bi(S,_.mode,L),g.return=_,_=g}return u(_);case Te:return Z=S._init,et(_,g,Z(S._payload),L)}if(lt(S))return V(_,g,S,L);if(X(S))return q(_,g,S,L);Ro(_,S)}return typeof S=="string"&&S!==""||typeof S=="number"?(S=""+S,g!==null&&g.tag===6?(n(_,g.sibling),g=s(g,S),g.return=_,_=g):(n(_,g),g=Ui(S,_.mode,L),g.return=_,_=g),u(_)):n(_,g)}return et}var Es=Za(!0),eu=Za(!1),To=Tr(null),Io=null,Ps=null,Jl=null;function Yl(){Jl=Ps=Io=null}function Xl(e){var t=To.current;He(To),e._currentValue=t}function Zl(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 Ms(e,t){Io=e,Jl=Ps=null,e=e.dependencies,e!==null&&e.firstContext!==null&&((e.lanes&t)!==0&&(Ht=!0),e.firstContext=null)}function fn(e){var t=e._currentValue;if(Jl!==e)if(e={context:e,memoizedValue:t,next:null},Ps===null){if(Io===null)throw Error(f(308));Ps=e,Io.dependencies={lanes:0,firstContext:e}}else Ps=Ps.next=e;return t}var ns=null;function ei(e){ns===null?ns=[e]:ns.push(e)}function tu(e,t,n,r){var s=t.interleaved;return s===null?(n.next=n,ei(t)):(n.next=s.next,s.next=n),t.interleaved=n,mr(e,r)}function mr(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 Or=!1;function ti(e){e.updateQueue={baseState:e.memoizedState,firstBaseUpdate:null,lastBaseUpdate:null,shared:{pending:null,interleaved:null,lanes:0},effects:null}}function nu(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 pr(e,t){return{eventTime:e,lane:t,tag:0,payload:null,callback:null,next:null}}function Dr(e,t,n){var r=e.updateQueue;if(r===null)return null;if(r=r.shared,(xe&2)!==0){var s=r.pending;return s===null?t.next=t:(t.next=s.next,s.next=t),r.pending=t,mr(e,n)}return s=r.interleaved,s===null?(t.next=t,ei(r)):(t.next=s.next,s.next=t),r.interleaved=t,mr(e,n)}function Lo(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,ir(e,n)}}function ru(e,t){var n=e.updateQueue,r=e.alternate;if(r!==null&&(r=r.updateQueue,n===r)){var s=null,l=null;if(n=n.firstBaseUpdate,n!==null){do{var u={eventTime:n.eventTime,lane:n.lane,tag:n.tag,payload:n.payload,callback:n.callback,next:null};l===null?s=l=u:l=l.next=u,n=n.next}while(n!==null);l===null?s=l=t:l=l.next=t}else s=l=t;n={baseState:r.baseState,firstBaseUpdate:s,lastBaseUpdate:l,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 Oo(e,t,n,r){var s=e.updateQueue;Or=!1;var l=s.firstBaseUpdate,u=s.lastBaseUpdate,d=s.shared.pending;if(d!==null){s.shared.pending=null;var h=d,w=h.next;h.next=null,u===null?l=w:u.next=w,u=h;var M=e.alternate;M!==null&&(M=M.updateQueue,d=M.lastBaseUpdate,d!==u&&(d===null?M.firstBaseUpdate=w:d.next=w,M.lastBaseUpdate=h))}if(l!==null){var A=s.baseState;u=0,M=w=h=null,d=l;do{var P=d.lane,U=d.eventTime;if((r&P)===P){M!==null&&(M=M.next={eventTime:U,lane:0,tag:d.tag,payload:d.payload,callback:d.callback,next:null});e:{var V=e,q=d;switch(P=t,U=n,q.tag){case 1:if(V=q.payload,typeof V=="function"){A=V.call(U,A,P);break e}A=V;break e;case 3:V.flags=V.flags&-65537|128;case 0:if(V=q.payload,P=typeof V=="function"?V.call(U,A,P):V,P==null)break e;A=O({},A,P);break e;case 2:Or=!0}}d.callback!==null&&d.lane!==0&&(e.flags|=64,P=s.effects,P===null?s.effects=[d]:P.push(d))}else U={eventTime:U,lane:P,tag:d.tag,payload:d.payload,callback:d.callback,next:null},M===null?(w=M=U,h=A):M=M.next=U,u|=P;if(d=d.next,d===null){if(d=s.shared.pending,d===null)break;P=d,d=P.next,P.next=null,s.lastBaseUpdate=P,s.shared.pending=null}}while(!0);if(M===null&&(h=A),s.baseState=h,s.firstBaseUpdate=w,s.lastBaseUpdate=M,t=s.shared.interleaved,t!==null){s=t;do u|=s.lane,s=s.next;while(s!==t)}else l===null&&(s.shared.lanes=0);os|=u,e.lanes=u,e.memoizedState=A}}function su(e,t,n){if(e=t.effects,t.effects=null,e!==null)for(t=0;tn?n:4,e(!0);var r=li.transition;li.transition={};try{e(!1),t()}finally{F=n,li.transition=r}}function ju(){return mn().memoizedState}function Vd(e,t,n){var r=Ur(e);if(n={lane:r,action:n,hasEagerState:!1,eagerState:null,next:null},ku(e))Cu(t,n);else if(n=tu(e,t,n,r),n!==null){var s=Tt();Tn(n,e,r,s),Nu(n,t,r)}}function Qd(e,t,n){var r=Ur(e),s={lane:r,action:n,hasEagerState:!1,eagerState:null,next:null};if(ku(e))Cu(t,s);else{var l=e.alternate;if(e.lanes===0&&(l===null||l.lanes===0)&&(l=t.lastRenderedReducer,l!==null))try{var u=t.lastRenderedState,d=l(u,n);if(s.hasEagerState=!0,s.eagerState=d,En(d,u)){var h=t.interleaved;h===null?(s.next=s,ei(t)):(s.next=h.next,h.next=s),t.interleaved=s;return}}catch{}n=tu(e,t,s,r),n!==null&&(s=Tt(),Tn(n,e,r,s),Nu(n,t,r))}}function ku(e){var t=e.alternate;return e===Ke||t!==null&&t===Ke}function Cu(e,t){no=$o=!0;var n=e.pending;n===null?t.next=t:(t.next=n.next,n.next=t),e.pending=t}function Nu(e,t,n){if((n&4194240)!==0){var r=t.lanes;r&=e.pendingLanes,n|=r,t.lanes=n,ir(e,n)}}var bo={readContext:fn,useCallback:jt,useContext:jt,useEffect:jt,useImperativeHandle:jt,useInsertionEffect:jt,useLayoutEffect:jt,useMemo:jt,useReducer:jt,useRef:jt,useState:jt,useDebugValue:jt,useDeferredValue:jt,useTransition:jt,useMutableSource:jt,useSyncExternalStore:jt,useId:jt,unstable_isNewReconciler:!1},Wd={readContext:fn,useCallback:function(e,t){return Gn().memoizedState=[e,t===void 0?null:t],e},useContext:fn,useEffect:hu,useImperativeHandle:function(e,t,n){return n=n!=null?n.concat([e]):null,Fo(4194308,4,yu.bind(null,t,e),n)},useLayoutEffect:function(e,t){return Fo(4194308,4,e,t)},useInsertionEffect:function(e,t){return Fo(4,2,e,t)},useMemo:function(e,t){var n=Gn();return t=t===void 0?null:t,e=e(),n.memoizedState=[e,t],e},useReducer:function(e,t,n){var r=Gn();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=Vd.bind(null,Ke,e),[r.memoizedState,e]},useRef:function(e){var t=Gn();return e={current:e},t.memoizedState=e},useState:mu,useDebugValue:mi,useDeferredValue:function(e){return Gn().memoizedState=e},useTransition:function(){var e=mu(!1),t=e[0];return e=Hd.bind(null,e[1]),Gn().memoizedState=e,[t,e]},useMutableSource:function(){},useSyncExternalStore:function(e,t,n){var r=Ke,s=Gn();if(Ve){if(n===void 0)throw Error(f(407));n=n()}else{if(n=t(),pt===null)throw Error(f(349));(ss&30)!==0||au(r,t,n)}s.memoizedState=n;var l={value:n,getSnapshot:t};return s.queue=l,hu(cu.bind(null,r,l,e),[e]),r.flags|=2048,oo(9,uu.bind(null,r,l,n,t),void 0,null),n},useId:function(){var e=Gn(),t=pt.identifierPrefix;if(Ve){var n=fr,r=dr;n=(r&~(1<<32-be(r)-1)).toString(32)+n,t=":"+t+"R"+n,n=ro++,0<\/script>",e=e.removeChild(e.firstChild)):typeof r.is=="string"?e=u.createElement(n,{is:r.is}):(e=u.createElement(n),n==="select"&&(u=e,r.multiple?u.multiple=!0:r.size&&(u.size=r.size))):e=u.createElementNS(e,n),e[qn]=t,e[Js]=r,Wu(e,t,!1,!1),t.stateNode=e;e:{switch(u=sn(n,r),n){case"dialog":Be("cancel",e),Be("close",e),s=r;break;case"iframe":case"object":case"embed":Be("load",e),s=r;break;case"video":case"audio":for(s=0;sLs&&(t.flags|=128,r=!0,lo(l,!1),t.lanes=4194304)}else{if(!r)if(e=Do(u),e!==null){if(t.flags|=128,r=!0,n=e.updateQueue,n!==null&&(t.updateQueue=n,t.flags|=4),lo(l,!0),l.tail===null&&l.tailMode==="hidden"&&!u.alternate&&!Ve)return kt(t),null}else 2*D()-l.renderingStartTime>Ls&&n!==1073741824&&(t.flags|=128,r=!0,lo(l,!1),t.lanes=4194304);l.isBackwards?(u.sibling=t.child,t.child=u):(n=l.last,n!==null?n.sibling=u:t.child=u,l.last=u)}return l.tail!==null?(t=l.tail,l.rendering=t,l.tail=t.sibling,l.renderingStartTime=D(),t.sibling=null,n=qe.current,$e(qe,r?n&1|2:n&1),t):(kt(t),null);case 22:case 23:return zi(),r=t.memoizedState!==null,e!==null&&e.memoizedState!==null!==r&&(t.flags|=8192),r&&(t.mode&1)!==0?(en&1073741824)!==0&&(kt(t),t.subtreeFlags&6&&(t.flags|=8192)):kt(t),null;case 24:return null;case 25:return null}throw Error(f(156,t.tag))}function ef(e,t){switch(Wl(t),t.tag){case 1:return Bt(t.type)&&Co(),e=t.flags,e&65536?(t.flags=e&-65537|128,t):null;case 3:return As(),He(bt),He(wt),oi(),e=t.flags,(e&65536)!==0&&(e&128)===0?(t.flags=e&-65537|128,t):null;case 5:return ri(t),null;case 13:if(He(qe),e=t.memoizedState,e!==null&&e.dehydrated!==null){if(t.alternate===null)throw Error(f(340));Ns()}return e=t.flags,e&65536?(t.flags=e&-65537|128,t):null;case 19:return He(qe),null;case 4:return As(),null;case 10:return Xl(t.type._context),null;case 22:case 23:return zi(),null;case 24:return null;default:return null}}var Qo=!1,Ct=!1,tf=typeof WeakSet=="function"?WeakSet:Set,B=null;function Ts(e,t){var n=e.ref;if(n!==null)if(typeof n=="function")try{n(null)}catch(r){Xe(e,t,r)}else n.current=null}function Ci(e,t,n){try{n()}catch(r){Xe(e,t,r)}}var Gu=!1;function nf(e,t){if(zl=ms,e=Ea(),Ml(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 s=r.anchorOffset,l=r.focusNode;r=r.focusOffset;try{n.nodeType,l.nodeType}catch{n=null;break e}var u=0,d=-1,h=-1,w=0,M=0,A=e,P=null;t:for(;;){for(var U;A!==n||s!==0&&A.nodeType!==3||(d=u+s),A!==l||r!==0&&A.nodeType!==3||(h=u+r),A.nodeType===3&&(u+=A.nodeValue.length),(U=A.firstChild)!==null;)P=A,A=U;for(;;){if(A===e)break t;if(P===n&&++w===s&&(d=u),P===l&&++M===r&&(h=u),(U=A.nextSibling)!==null)break;A=P,P=A.parentNode}A=U}n=d===-1||h===-1?null:{start:d,end:h}}else n=null}n=n||{start:0,end:0}}else n=null;for($l={focusedElem:e,selectionRange:n},ms=!1,B=t;B!==null;)if(t=B,e=t.child,(t.subtreeFlags&1028)!==0&&e!==null)e.return=t,B=e;else for(;B!==null;){t=B;try{var V=t.alternate;if((t.flags&1024)!==0)switch(t.tag){case 0:case 11:case 15:break;case 1:if(V!==null){var q=V.memoizedProps,et=V.memoizedState,_=t.stateNode,g=_.getSnapshotBeforeUpdate(t.elementType===t.type?q:Mn(t.type,q),et);_.__reactInternalSnapshotBeforeUpdate=g}break;case 3:var S=t.stateNode.containerInfo;S.nodeType===1?S.textContent="":S.nodeType===9&&S.documentElement&&S.removeChild(S.documentElement);break;case 5:case 6:case 4:case 17:break;default:throw Error(f(163))}}catch(L){Xe(t,t.return,L)}if(e=t.sibling,e!==null){e.return=t.return,B=e;break}B=t.return}return V=Gu,Gu=!1,V}function io(e,t,n){var r=t.updateQueue;if(r=r!==null?r.lastEffect:null,r!==null){var s=r=r.next;do{if((s.tag&e)===e){var l=s.destroy;s.destroy=void 0,l!==void 0&&Ci(t,n,l)}s=s.next}while(s!==r)}}function Wo(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 Ni(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 Ju(e){var t=e.alternate;t!==null&&(e.alternate=null,Ju(t)),e.child=null,e.deletions=null,e.sibling=null,e.tag===5&&(t=e.stateNode,t!==null&&(delete t[qn],delete t[Js],delete t[Bl],delete t[$d],delete t[Fd])),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 Yu(e){return e.tag===5||e.tag===3||e.tag===4}function Xu(e){e:for(;;){for(;e.sibling===null;){if(e.return===null||Yu(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 Ei(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=jo));else if(r!==4&&(e=e.child,e!==null))for(Ei(e,t,n),e=e.sibling;e!==null;)Ei(e,t,n),e=e.sibling}function Pi(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(Pi(e,t,n),e=e.sibling;e!==null;)Pi(e,t,n),e=e.sibling}var vt=null,An=!1;function zr(e,t,n){for(n=n.child;n!==null;)Zu(e,t,n),n=n.sibling}function Zu(e,t,n){if(Ue&&typeof Ue.onCommitFiberUnmount=="function")try{Ue.onCommitFiberUnmount(ue,n)}catch{}switch(n.tag){case 5:Ct||Ts(n,t);case 6:var r=vt,s=An;vt=null,zr(e,t,n),vt=r,An=s,vt!==null&&(An?(e=vt,n=n.stateNode,e.nodeType===8?e.parentNode.removeChild(n):e.removeChild(n)):vt.removeChild(n.stateNode));break;case 18:vt!==null&&(An?(e=vt,n=n.stateNode,e.nodeType===8?bl(e.parentNode,n):e.nodeType===1&&bl(e,n),Jr(e)):bl(vt,n.stateNode));break;case 4:r=vt,s=An,vt=n.stateNode.containerInfo,An=!0,zr(e,t,n),vt=r,An=s;break;case 0:case 11:case 14:case 15:if(!Ct&&(r=n.updateQueue,r!==null&&(r=r.lastEffect,r!==null))){s=r=r.next;do{var l=s,u=l.destroy;l=l.tag,u!==void 0&&((l&2)!==0||(l&4)!==0)&&Ci(n,t,u),s=s.next}while(s!==r)}zr(e,t,n);break;case 1:if(!Ct&&(Ts(n,t),r=n.stateNode,typeof r.componentWillUnmount=="function"))try{r.props=n.memoizedProps,r.state=n.memoizedState,r.componentWillUnmount()}catch(d){Xe(n,t,d)}zr(e,t,n);break;case 21:zr(e,t,n);break;case 22:n.mode&1?(Ct=(r=Ct)||n.memoizedState!==null,zr(e,t,n),Ct=r):zr(e,t,n);break;default:zr(e,t,n)}}function ec(e){var t=e.updateQueue;if(t!==null){e.updateQueue=null;var n=e.stateNode;n===null&&(n=e.stateNode=new tf),t.forEach(function(r){var s=ff.bind(null,e,r);n.has(r)||(n.add(r),r.then(s,s))})}}function Rn(e,t){var n=t.deletions;if(n!==null)for(var r=0;rs&&(s=u),r&=~l}if(r=s,r=D()-r,r=(120>r?120:480>r?480:1080>r?1080:1920>r?1920:3e3>r?3e3:4320>r?4320:1960*sf(r/1960))-r,10e?16:e,Fr===null)var r=!1;else{if(e=Fr,Fr=null,Yo=0,(xe&6)!==0)throw Error(f(331));var s=xe;for(xe|=4,B=e.current;B!==null;){var l=B,u=l.child;if((B.flags&16)!==0){var d=l.deletions;if(d!==null){for(var h=0;hD()-Ri?is(e,0):Ai|=n),Qt(e,t)}function mc(e,t){t===0&&((e.mode&1)===0?t=1:(t=kn,kn<<=1,(kn&130023424)===0&&(kn=4194304)));var n=Tt();e=mr(e,t),e!==null&&(it(e,t,n),Qt(e,n))}function df(e){var t=e.memoizedState,n=0;t!==null&&(n=t.retryLane),mc(e,n)}function ff(e,t){var n=0;switch(e.tag){case 13:var r=e.stateNode,s=e.memoizedState;s!==null&&(n=s.retryLane);break;case 19:r=e.stateNode;break;default:throw Error(f(314))}r!==null&&r.delete(t),mc(e,n)}var pc;pc=function(e,t,n){if(e!==null)if(e.memoizedProps!==t.pendingProps||bt.current)Ht=!0;else{if((e.lanes&n)===0&&(t.flags&128)===0)return Ht=!1,Xd(e,t,n);Ht=(e.flags&131072)!==0}else Ht=!1,Ve&&(t.flags&1048576)!==0&&qa(t,Mo,t.index);switch(t.lanes=0,t.tag){case 2:var r=t.type;Vo(e,t),e=t.pendingProps;var s=js(t,wt.current);Ms(t,n),s=ai(null,t,r,e,s,n);var l=ui();return t.flags|=1,typeof s=="object"&&s!==null&&typeof s.render=="function"&&s.$$typeof===void 0?(t.tag=1,t.memoizedState=null,t.updateQueue=null,Bt(r)?(l=!0,No(t)):l=!1,t.memoizedState=s.state!==null&&s.state!==void 0?s.state:null,ti(t),s.updater=Bo,t.stateNode=s,s._reactInternals=t,hi(t,r,e,n),t=xi(null,t,r,!0,l,n)):(t.tag=0,Ve&&l&&Ql(t),Rt(null,t,s,n),t=t.child),t;case 16:r=t.elementType;e:{switch(Vo(e,t),e=t.pendingProps,s=r._init,r=s(r._payload),t.type=r,s=t.tag=pf(r),e=Mn(r,e),s){case 0:t=yi(null,t,r,e,n);break e;case 1:t=Uu(null,t,r,e,n);break e;case 11:t=Ou(null,t,r,e,n);break e;case 14:t=Du(null,t,r,Mn(r.type,e),n);break e}throw Error(f(306,r,""))}return t;case 0:return r=t.type,s=t.pendingProps,s=t.elementType===r?s:Mn(r,s),yi(e,t,r,s,n);case 1:return r=t.type,s=t.pendingProps,s=t.elementType===r?s:Mn(r,s),Uu(e,t,r,s,n);case 3:e:{if(bu(t),e===null)throw Error(f(387));r=t.pendingProps,l=t.memoizedState,s=l.element,nu(e,t),Oo(t,r,null,n);var u=t.memoizedState;if(r=u.element,l.isDehydrated)if(l={element:r,isDehydrated:!1,cache:u.cache,pendingSuspenseBoundaries:u.pendingSuspenseBoundaries,transitions:u.transitions},t.updateQueue.baseState=l,t.memoizedState=l,t.flags&256){s=Rs(Error(f(423)),t),t=Bu(e,t,r,n,s);break e}else if(r!==s){s=Rs(Error(f(424)),t),t=Bu(e,t,r,n,s);break e}else for(Zt=Rr(t.stateNode.containerInfo.firstChild),Xt=t,Ve=!0,Pn=null,n=eu(t,null,r,n),t.child=n;n;)n.flags=n.flags&-3|4096,n=n.sibling;else{if(Ns(),r===s){t=hr(e,t,n);break e}Rt(e,t,r,n)}t=t.child}return t;case 5:return ou(t),e===null&&Kl(t),r=t.type,s=t.pendingProps,l=e!==null?e.memoizedProps:null,u=s.children,Fl(r,s)?u=null:l!==null&&Fl(r,l)&&(t.flags|=32),Fu(e,t),Rt(e,t,u,n),t.child;case 6:return e===null&&Kl(t),null;case 13:return Hu(e,t,n);case 4:return ni(t,t.stateNode.containerInfo),r=t.pendingProps,e===null?t.child=Es(t,null,r,n):Rt(e,t,r,n),t.child;case 11:return r=t.type,s=t.pendingProps,s=t.elementType===r?s:Mn(r,s),Ou(e,t,r,s,n);case 7:return Rt(e,t,t.pendingProps,n),t.child;case 8:return Rt(e,t,t.pendingProps.children,n),t.child;case 12:return Rt(e,t,t.pendingProps.children,n),t.child;case 10:e:{if(r=t.type._context,s=t.pendingProps,l=t.memoizedProps,u=s.value,$e(To,r._currentValue),r._currentValue=u,l!==null)if(En(l.value,u)){if(l.children===s.children&&!bt.current){t=hr(e,t,n);break e}}else for(l=t.child,l!==null&&(l.return=t);l!==null;){var d=l.dependencies;if(d!==null){u=l.child;for(var h=d.firstContext;h!==null;){if(h.context===r){if(l.tag===1){h=pr(-1,n&-n),h.tag=2;var w=l.updateQueue;if(w!==null){w=w.shared;var M=w.pending;M===null?h.next=h:(h.next=M.next,M.next=h),w.pending=h}}l.lanes|=n,h=l.alternate,h!==null&&(h.lanes|=n),Zl(l.return,n,t),d.lanes|=n;break}h=h.next}}else if(l.tag===10)u=l.type===t.type?null:l.child;else if(l.tag===18){if(u=l.return,u===null)throw Error(f(341));u.lanes|=n,d=u.alternate,d!==null&&(d.lanes|=n),Zl(u,n,t),u=l.sibling}else u=l.child;if(u!==null)u.return=l;else for(u=l;u!==null;){if(u===t){u=null;break}if(l=u.sibling,l!==null){l.return=u.return,u=l;break}u=u.return}l=u}Rt(e,t,s.children,n),t=t.child}return t;case 9:return s=t.type,r=t.pendingProps.children,Ms(t,n),s=fn(s),r=r(s),t.flags|=1,Rt(e,t,r,n),t.child;case 14:return r=t.type,s=Mn(r,t.pendingProps),s=Mn(r.type,s),Du(e,t,r,s,n);case 15:return zu(e,t,t.type,t.pendingProps,n);case 17:return r=t.type,s=t.pendingProps,s=t.elementType===r?s:Mn(r,s),Vo(e,t),t.tag=1,Bt(r)?(e=!0,No(t)):e=!1,Ms(t,n),Pu(t,r,s),hi(t,r,s,n),xi(null,t,r,!0,e,n);case 19:return Qu(e,t,n);case 22:return $u(e,t,n)}throw Error(f(156,t.tag))};function hc(e,t){return Pt(e,t)}function mf(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 hn(e,t,n,r){return new mf(e,t,n,r)}function Fi(e){return e=e.prototype,!(!e||!e.isReactComponent)}function pf(e){if(typeof e=="function")return Fi(e)?1:0;if(e!=null){if(e=e.$$typeof,e===ie)return 11;if(e===Fe)return 14}return 2}function Br(e,t){var n=e.alternate;return n===null?(n=hn(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 tl(e,t,n,r,s,l){var u=2;if(r=e,typeof e=="function")Fi(e)&&(u=1);else if(typeof e=="string")u=5;else e:switch(e){case Me:return us(n.children,s,l,t);case Ze:u=8,s|=8;break;case ct:return e=hn(12,n,t,s|2),e.elementType=ct,e.lanes=l,e;case Le:return e=hn(13,n,t,s),e.elementType=Le,e.lanes=l,e;case H:return e=hn(19,n,t,s),e.elementType=H,e.lanes=l,e;case Ae:return nl(n,s,l,t);default:if(typeof e=="object"&&e!==null)switch(e.$$typeof){case st:u=10;break e;case Et:u=9;break e;case ie:u=11;break e;case Fe:u=14;break e;case Te:u=16,r=null;break e}throw Error(f(130,e==null?e:typeof e,""))}return t=hn(u,n,t,s),t.elementType=e,t.type=r,t.lanes=l,t}function us(e,t,n,r){return e=hn(7,e,r,t),e.lanes=n,e}function nl(e,t,n,r){return e=hn(22,e,r,t),e.elementType=Ae,e.lanes=n,e.stateNode={isHidden:!1},e}function Ui(e,t,n){return e=hn(6,e,null,t),e.lanes=n,e}function bi(e,t,n){return t=hn(4,e.children!==null?e.children:[],e.key,t),t.lanes=n,t.stateNode={containerInfo:e.containerInfo,pendingChildren:null,implementation:e.implementation},t}function hf(e,t,n,r,s){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=or(0),this.expirationTimes=or(-1),this.entangledLanes=this.finishedLanes=this.mutableReadLanes=this.expiredLanes=this.pingedLanes=this.suspendedLanes=this.pendingLanes=0,this.entanglements=or(0),this.identifierPrefix=r,this.onRecoverableError=s,this.mutableSourceEagerHydrationData=null}function Bi(e,t,n,r,s,l,u,d,h){return e=new hf(e,t,n,d,h),t===1?(t=1,l===!0&&(t|=8)):t=0,l=hn(3,null,null,t),e.current=l,l.stateNode=e,l.memoizedState={element:r,isDehydrated:n,cache:null,transitions:null,pendingSuspenseBoundaries:null},ti(l),e}function gf(e,t,n){var r=3"u"||typeof __REACT_DEVTOOLS_GLOBAL_HOOK__.checkDCE!="function"))try{__REACT_DEVTOOLS_GLOBAL_HOOK__.checkDCE(i)}catch(p){console.error(p)}}return i(),Ki.exports=Pf(),Ki.exports}var Mc;function Af(){if(Mc)return ul;Mc=1;var i=Mf();return ul.createRoot=i.createRoot,ul.hydrateRoot=i.hydrateRoot,ul}var Rf=Af();const Tf=Vc(Rf),If="/api";async function Ce(i,p){const f=await fetch(`${If}${i}`,{...p,headers:{"Content-Type":"application/json",...p?.headers??{}}}),N=await f.json();if(!f.ok){const j=N.error?.message??"Ошибка запроса";throw new Error(j)}return N}const _e={async loadSharedConnectionConfig(){return Ce("/llm/shared-connection")},async saveSharedConnectionConfig(i){return Ce("/llm/shared-connection",{method:"POST",body:JSON.stringify({llmProvider:i.llmProvider,model:i.model,baseUrl:i.baseUrl,temperature:i.temperature,maxOutputTokens:i.maxOutputTokens})})},async listModels(i){return Ce("/llm/models",{method:"POST",body:JSON.stringify({llmProvider:i.llmProvider,apiKey:i.apiKey,model:i.model,baseUrl:i.baseUrl})})},async testConnection(i){return Ce("/llm/test-connection",{method:"POST",body:JSON.stringify({llmProvider:i.llmProvider,apiKey:i.apiKey,model:i.model,baseUrl:i.baseUrl})})},async normalize(i){return Ce("/normalize",{method:"POST",body:JSON.stringify({llmProvider:i.connection.llmProvider,apiKey:i.connection.apiKey,model:i.connection.model,baseUrl:i.connection.baseUrl,temperature:i.connection.temperature,maxOutputTokens:i.connection.maxOutputTokens,promptVersion:i.promptVersion,systemPrompt:i.prompts.systemPrompt,developerPrompt:i.prompts.developerPrompt,domainPrompt:i.prompts.domainPrompt,fewShotExamples:i.prompts.fewShotExamples,userQuestion:i.query.userQuestion,context:{period_hint:i.query.periodHint??"",business_context:i.query.businessContext??"",expected_route:i.query.expectedRoute??""},saveAsTestCase:!!i.saveAsTestCase,useMock:!!i.useMock})})},async loadHistory(){return Ce("/history")},async loadTrace(i){return Ce(`/history/${i}`)},async loadPresets(){return Ce("/presets")},async savePreset(i){return Ce("/presets/save",{method:"POST",body:JSON.stringify(i)})},async runEval(i){return Ce("/eval/run",{method:"POST",body:JSON.stringify({normalizeConfig:{llmProvider:i.connection.llmProvider,apiKey:i.connection.apiKey,model:i.connection.model,baseUrl:i.connection.baseUrl,temperature:i.connection.temperature,maxOutputTokens:i.connection.maxOutputTokens,promptVersion:i.promptVersion,systemPrompt:i.prompts.systemPrompt,developerPrompt:i.prompts.developerPrompt,domainPrompt:i.prompts.domainPrompt,fewShotExamples:i.prompts.fewShotExamples},caseIds:i.caseIds,useMock:!!i.useMock,mode:i.mode??"standard",caseSetFile:i.caseSetFile,rawQuestions:i.rawQuestions,eval_target:i.evalTarget,compare_with_report_file:i.compareWithReportFile,analysis_date:i.analysisDate})})},async startEvalRunAsync(i){return Ce("/eval/run-async/start",{method:"POST",body:JSON.stringify({normalizeConfig:{llmProvider:i.connection.llmProvider,apiKey:i.connection.apiKey,model:i.connection.model,baseUrl:i.connection.baseUrl,temperature:i.connection.temperature,maxOutputTokens:i.connection.maxOutputTokens,promptVersion:i.promptVersion,systemPrompt:i.prompts.systemPrompt,developerPrompt:i.prompts.developerPrompt,domainPrompt:i.prompts.domainPrompt,fewShotExamples:i.prompts.fewShotExamples},caseIds:i.caseIds,useMock:!!i.useMock,mode:i.mode??"standard",caseSetFile:i.caseSetFile,rawQuestions:i.rawQuestions,eval_target:i.evalTarget,compare_with_report_file:i.compareWithReportFile,questions:i.questions,analysis_date:i.analysisDate})})},async loadEvalRunAsyncStatus(i){return Ce(`/eval/run-async/${encodeURIComponent(i)}`)},async startRun(){return Ce("/accounting-agent/v1/runs/start",{method:"POST",body:JSON.stringify({initiator:"ndc_operator",source:"gui"})})},async finishRun(i){return Ce("/accounting-agent/v1/runs/finish",{method:"POST",body:JSON.stringify({runId:i,status:"DONE",source:"gui",reason:"Остановлено оператором из GUI"})})},async listRuns(){return Ce("/accounting-agent/v1/runs")},async listResults(){return Ce("/accounting-agent/v1/results")},async runTrace(i){return Ce(`/accounting-agent/v1/trace/run/${i}`)},async sendAssistantMessage(i){return Ce("/assistant/message",{method:"POST",body:JSON.stringify({session_id:i.sessionId??"",mode:"assistant",message:i.userMessage,user_message:i.userMessage,llmProvider:i.connection.llmProvider,apiKey:i.connection.apiKey,model:i.connection.model,baseUrl:i.connection.baseUrl,temperature:i.connection.temperature,maxOutputTokens:i.connection.maxOutputTokens,promptVersion:i.promptVersion??"address_query_runtime_v1",systemPrompt:i.prompts.systemPrompt,developerPrompt:i.prompts.developerPrompt,domainPrompt:i.prompts.domainPrompt,fewShotExamples:i.prompts.fewShotExamples,context:{period_hint:i.context?.periodHint??"",business_context:i.context?.businessContext??""},useMock:!!i.useMock})})},async loadAssistantSession(i){return Ce(`/assistant/session/${i}`)},async loadAssistantAnnotations(i){const p=new URLSearchParams;i?.session_id&&p.set("session_id",i.session_id),typeof i?.limit=="number"&&p.set("limit",String(i.limit));const f=p.toString();return Ce(`/assistant/annotations${f?`?${f}`:""}`)},async saveAssistantAnnotation(i){return Ce("/assistant/annotations",{method:"POST",body:JSON.stringify(i)})},async loadAutoRunsHistory(i){const p=new URLSearchParams;i?.from&&p.set("from",i.from),i?.to&&p.set("to",i.to),i?.target&&p.set("target",i.target),i?.mode&&p.set("mode",i.mode),i?.use_mock&&p.set("use_mock",i.use_mock),i?.prompt_contains&&p.set("prompt_contains",i.prompt_contains),typeof i?.limit=="number"&&p.set("limit",String(i.limit)),typeof i?.scan_limit=="number"&&p.set("scan_limit",String(i.scan_limit));const f=p.toString();return Ce(`/autoruns/history${f?`?${f}`:""}`)},async loadAutoRunDetail(i){return Ce(`/autoruns/history/${encodeURIComponent(i)}`)},async loadAutoRunCaseDialog(i,p){return Ce(`/autoruns/history/${encodeURIComponent(i)}/case/${encodeURIComponent(p)}/dialog`)},async loadAutoRunAnnotations(i){const p=new URLSearchParams;i?.run_id&&p.set("run_id",i.run_id),i?.case_id&&p.set("case_id",i.case_id),typeof i?.min_rating=="number"&&p.set("min_rating",String(i.min_rating)),i?.manual_case_decision&&p.set("manual_case_decision",i.manual_case_decision),typeof i?.limit=="number"&&p.set("limit",String(i.limit));const f=p.toString();return Ce(`/autoruns/annotations${f?`?${f}`:""}`)},async saveAutoRunAnnotation(i){return Ce("/autoruns/annotations",{method:"POST",body:JSON.stringify(i)})},async updateAutoRunAnnotation(i){return Ce(`/autoruns/annotations/${encodeURIComponent(i.annotation_id)}`,{method:"PATCH",body:JSON.stringify({resolved:i.resolved,resolved_by:i.resolved_by})})},async loadAutoRunPostAnalysis(i){const p=new URLSearchParams;i?.run_id&&p.set("run_id",i.run_id),typeof i?.limit_per_queue=="number"&&p.set("limit_per_queue",String(i.limit_per_queue)),typeof i?.annotation_limit=="number"&&p.set("annotation_limit",String(i.annotation_limit)),typeof i?.scan_limit=="number"&&p.set("scan_limit",String(i.scan_limit)),i?.from&&p.set("from",i.from),i?.to&&p.set("to",i.to),i?.target&&p.set("target",i.target),i?.mode&&p.set("mode",i.mode),i?.use_mock&&p.set("use_mock",i.use_mock),i?.prompt_contains&&p.set("prompt_contains",i.prompt_contains);const f=p.toString();return Ce(`/autoruns/post-analysis${f?`?${f}`:""}`)},async loadAutoRunAutogenHistory(i){const p=new URLSearchParams;i?.mode&&p.set("mode",i.mode),typeof i?.limit=="number"&&p.set("limit",String(i.limit));const f=p.toString();return Ce(`/autoruns/autogen/history${f?`?${f}`:""}`)},async loadAutoRunAutogenPersonalityCatalog(){return Ce("/autoruns/autogen/personality-catalog")},async generateAutoRunQuestions(i){return Ce("/autoruns/autogen/generate",{method:"POST",body:JSON.stringify(i)})}};function tn({value:i}){return o.jsx("pre",{className:"json-view",children:JSON.stringify(i??{},null,2)})}function In({title:i,subtitle:p,actions:f,className:N,hideHeader:j,children:T}){return o.jsxs("section",{className:N?`panel-frame ${N}`:"panel-frame",children:[j?null:o.jsxs("header",{className:"panel-header",children:[o.jsxs("div",{children:[o.jsx("h2",{children:i}),p?o.jsx("p",{children:p}):null]}),f?o.jsx("div",{className:"panel-actions",children:f}):null]}),o.jsx("div",{className:"panel-body",children:T})]})}function Lf(i){const p=new Date(i);return Number.isNaN(p.getTime())?i:p.toLocaleString("ru-RU")}function Of({sessionId:i,conversation:p,statusText:f,errorMessage:N,useMock:j,appLogs:T}){const $=p.filter(Q=>Q.role==="assistant").length,Y=p.filter(Q=>Q.role==="user").length,b=p.length>0?p[p.length-1]:null;return o.jsxs(In,{title:"SAM",subtitle:"System Assistant Monitor: срез по текущей сессии и логам.",children:[o.jsxs("div",{className:"metrics-grid",children:[o.jsxs("div",{children:[o.jsx("span",{children:"session_id"}),o.jsx("strong",{children:i||"новая сессия"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"mock_mode"}),o.jsx("strong",{children:j?"on":"off"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"сообщений пользователя"}),o.jsx("strong",{children:Y})]}),o.jsxs("div",{children:[o.jsx("span",{children:"ответов ассистента"}),o.jsx("strong",{children:$})]}),o.jsxs("div",{children:[o.jsx("span",{children:"статус"}),o.jsx("strong",{children:f||"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"ошибка"}),o.jsx("strong",{children:N||"нет"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"последнее сообщение"}),o.jsx("strong",{children:b?.created_at?Lf(b.created_at):"нет данных"})]})]}),o.jsx("h3",{style:{marginTop:12},children:"Последние системные логи"}),o.jsx(tn,{value:T.slice(0,120)})]})}const Df=/(?:^|\n)\s*#{0,6}\s*(?:debug_payload_json|technical_breakdown_json|route_summary_json|debug_payload|technical_breakdown)\b/i,zf=[/\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 $f(i){try{return JSON.stringify(i,null,2)}catch{return String(i)}}function Ff(i){const p=String(i??""),f=p.match(Df);return(f?p.slice(0,f.index):p).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($=>$.trimEnd()).filter($=>$.trim().length>0).filter($=>!zf.some(Y=>Y.test($))).join(` +`).trim()}function Uf(i,p,f="default"){const N=f==="technical",j=[];j.push("# Assistant conversation export"),j.push(`session_id: ${i||"n/a"}`),j.push(`export_mode: ${f}`),j.push(`exported_at: ${new Date().toISOString()}`),j.push("");for(let T=0;T{j.length!==0&&(N.push(j.join(` +`)),j=[])};for(const $ of f){const Y=$.trimEnd(),b=Y.trim();if(!b){T();continue}const Q=/^Блок\s+\d+\./i.test(b),G=/^\d+\.\s/.test(b);(Q||G)&&j.length>0&&T(),j.push(Y)}return T(),N.length>0?N:[i]}function qf(i,p){const f=[],N=/\*\*(.+?)\*\*/g;let j=0,T=0,$;for(;($=N.exec(i))!==null;)$.index>j&&(f.push(o.jsx("span",{children:i.slice(j,$.index)},`${p}-t-${T}`)),T+=1),f.push(o.jsx("strong",{children:$[1]},`${p}-b-${T}`)),T+=1,j=N.lastIndex;return j0?f:[o.jsx("span",{children:i},`${p}-raw`)]}function Kf(i){const p=i.trimStart();return/^Блок\s+\d+\./i.test(p)?"assistant-msg-line heading":/^\d+\.\s/.test(p)?"assistant-msg-line numbered":/^-\s/.test(p)?"assistant-msg-line bullet":"assistant-msg-line"}function Gf(i,p=40){const f=i.replace(/\s+/g," ").trim();if(f.length<=p)return f;const N=f.split(" ").slice(0,3).join(" ").trim();return N.length>=10&&N.length<=p?`${N}…`:`${f.slice(0,p-1).trimEnd()}…`}function Qc(i){return i.replace(/\*\*(.+?)\*\*/g,"$1").replace(/^\d+\.\s*/,"").trim()}function Jf(i){const p=i.replace(/\r\n?/g,` +`).split(` +`).map(j=>j.trim()).find(Boolean),f=Qc(p??"");return(f.split("|")[0]?.trim()??f).replace(/\s+/g," ").trim()}function Yf(i){const p=i.replace(/\r\n?/g,` +`).split(` +`).map(N=>N.trim()).find(Boolean);return!p||!/^\d+\.\s/.test(p)?!1:Qc(p).includes("|")}function Xf(i,p){const f=p.replace(/\r\n?/g,` +`).replace(/\*\*(.+?)\*\*/g,"$1").split(` +`).map((j,T)=>{const $=j.trim();return T===0?$.replace(/^\d+\.\s*/,""):$}).filter(Boolean).join(" ").replace(/\s+/g," ").trim();if(!f)return null;const N=Jf(p)||f;return{message_id:i.message_id,source_text:f,anchor_text:N,preview_text:Gf(N)}}function Zf(i,p,f,N){return Wf(i.text).map((T,$)=>{const Y=T.split(` +`),b=i.role==="assistant"&&Yf(T),Q=b?Xf(i,T):null,G=!!Q&&p?.message_id===Q?.message_id&&p?.source_text===Q?.source_text,z=Y.map((ne,Ne)=>o.jsx("p",{className:Kf(ne),children:qf(ne,`line-${$}-${Ne}`)},`line-${$}-${Ne}`));return!b||!Q?o.jsx("div",{className:"assistant-msg-block",children:z},`block-${$}`):o.jsx("div",{className:G?"assistant-msg-block selectable active":"assistant-msg-block selectable",role:"button",tabIndex:0,onClick:()=>{if(G){N();return}f(Q)},onKeyDown:ne=>{if(!(ne.key!=="Enter"&&ne.key!==" ")){if(ne.preventDefault(),G){N();return}f(Q)}},children:z},`block-${$}`)})}function Wc({sessionId:i,conversation:p,inputValue:f,onInputChange:N,selectedContextChip:j,onSelectContextChip:T,onClearContextChip:$,useMock:Y,onUseMockChange:b,onSend:Q,onClear:G,busy:z,statusText:ne,errorMessage:Ne,showCommentAction:le=!1,onCommentAssistantMessage:ce,isAssistantMessageCommented:me,canCommentAssistantMessage:Qe}){const De=x.useRef(null),We=x.useRef(!0),je=x.useRef(null),[W,de]=x.useState("idle"),[Me,Ze]=x.useState("чат");function ct(ie=!1){De.current&&(ie&&(We.current=!0),De.current.scrollTop=De.current.scrollHeight)}x.useEffect(()=>{We.current&&ct()},[p]),x.useEffect(()=>()=>{je.current!==null&&window.clearTimeout(je.current)},[]);async function st(ie){if(p.length===0)return;const Le=Uf(i,p,ie),H=await Hf(Le);Ze(ie==="technical"?"тех":"чат"),de(H?"success":"error"),je.current!==null&&window.clearTimeout(je.current),je.current=window.setTimeout(()=>{de("idle")},2200)}function Et(){if(!De.current)return;const ie=De.current,Le=ie.scrollHeight-ie.scrollTop-ie.clientHeight;We.current=Le<16}return o.jsx(In,{className:"assistant-panel-frame",title:"Режим ассистента",children:o.jsxs("div",{className:"assistant-live-shell",children:[o.jsxs("div",{className:"assistant-toolbar",children:[o.jsxs("div",{className:"assistant-toolbar-actions",children:[o.jsx("button",{type:"button",className:"assistant-copy-btn",onClick:()=>{st("default")},disabled:p.length===0,title:"Экспорт только user-facing чата",children:"Скопировать чат"}),o.jsx("button",{type:"button",className:"assistant-copy-btn",onClick:()=>{st("technical")},disabled:p.length===0,title:"Технический экспорт с debug payload",children:"Скопировать техчат"}),o.jsx("button",{type:"button",className:"assistant-copy-btn",onClick:()=>G(),disabled:z&&p.length===0,children:"Сбросить сессию"})]}),o.jsxs("div",{className:"assistant-toolbar-meta",children:[i?o.jsx("span",{className:"status-chip",children:`session: ${i}`}):null,o.jsxs("div",{className:"assistant-toolbar-meta-right",children:[ne?o.jsx("span",{className:"assistant-live-status",children:ne}):null,W==="success"?o.jsxs("span",{className:"assistant-copy-feedback success",children:["Скопировано (",Me,")"]}):null,W==="error"?o.jsx("span",{className:"assistant-copy-feedback error",children:"Ошибка копирования"}):null]})]}),Ne?o.jsx("p",{className:"error-text assistant-toolbar-error",children:Ne}):null]}),o.jsx("div",{ref:De,className:"assistant-chat-list",onScroll:Et,children:p.map((ie,Le)=>{const H=ie.role==="assistant"&&le&&typeof ce=="function"&&(typeof Qe=="function"?Qe(ie,Le):!0),Fe=ie.role==="assistant"&&typeof me=="function"?me(ie,Le):!1;return o.jsxs("article",{className:`assistant-msg ${ie.role}`,children:[o.jsxs("header",{className:"assistant-msg-head",children:[o.jsxs("div",{className:"assistant-msg-head-main",children:[o.jsx("strong",{children:bf(ie.role)}),o.jsx("span",{children:Bf(ie.created_at)})]}),ie.role==="assistant"&&le?o.jsx("div",{className:"assistant-msg-head-actions",children:o.jsx("button",{type:"button",className:Fe?"autoruns-comment-icon assistant-comment-btn commented":"autoruns-comment-icon assistant-comment-btn",onClick:()=>ce?.(ie,Le),disabled:!H,title:H?"Комментировать ответ ассистента":"Комментарий недоступен для этого сообщения","aria-label":H?"Комментировать ответ ассистента":"Комментарий недоступен для этого сообщения",children:o.jsx(Vf,{commented:Fe})})}):null]}),o.jsx("div",{className:"assistant-msg-body",children:Zf(ie,j,T,$)}),ie.role==="assistant"&&ie.debug?o.jsxs("details",{className:"assistant-debug",children:[o.jsx("summary",{children:"Показать технический разбор"}),o.jsx(tn,{value:ie.debug})]}):null]},ie.message_id)})}),o.jsxs("div",{className:"assistant-compose",children:[j?o.jsxs("div",{className:"assistant-compose-context",children:[o.jsx("span",{className:"assistant-compose-context-label",children:"Выбранный объект"}),o.jsxs("div",{className:"assistant-compose-context-pill",title:j.source_text,children:[o.jsx("span",{className:"assistant-compose-context-pill-text",children:j.preview_text}),o.jsx("button",{type:"button",className:"assistant-compose-context-clear",onClick:$,"aria-label":"Убрать выбранный объект",title:"Убрать выбранный объект",children:"×"})]})]}):null,o.jsxs("label",{className:"full-width",children:["Сообщение",o.jsx("textarea",{className:"assistant-input-textarea",value:f,onChange:ie=>N(ie.target.value),rows:4,placeholder:j?"Продолжите вопрос по выбранному объекту...":"Введите вопрос к данным компании..."})]}),o.jsxs("div",{className:"button-row assistant-send-row",children:[o.jsxs("label",{className:"checkbox-row",children:[o.jsx("input",{type:"checkbox",checked:Y,onChange:ie=>b(ie.target.checked)}),"Mock-режим"]}),o.jsx("button",{type:"button",className:"assistant-send-btn",onClick:()=>{ct(!0),Q()},disabled:z||!f.trim(),children:z?"Выполняю...":"Отправить"})]})]})]})})}const cl="http://127.0.0.1:1234/v1",Ac="https://api.openai.com/v1",hl="qwen2.5-14b-instruct-1m",ua="unsloth/qwen3-30b-a3b-instruct-2507",em=[{value:hl,label:"Qwen2.5 14B Instruct 1M"},{value:ua,label:"Qwen3 30B A3B Instruct 2507"}];function tm(i){return i.llmProvider!=="local"?"openai":i.model===ua?"local_qwen3":i.model===hl?"local_qwen25":"local_custom"}function nm(i,p){const f=new Map;if(p)for(const N of em)f.set(N.value,N);for(const N of i)f.has(N)||f.set(N,{value:N,label:N});return Array.from(f.values())}function oa({value:i,modelOptions:p,modelsBusy:f,onChange:N,onReloadModels:j,onTestConnection:T,onSaveLocalConfig:$,lastStatus:Y,busy:b,embedded:Q=!1}){const G=i.llmProvider==="local",z=tm(i),ne=nm(p,G),Ne=ne.some(W=>W.value===i.model),[le,ce]=x.useState(String(i.temperature)),[me,Qe]=x.useState(String(i.maxOutputTokens));x.useEffect(()=>{ce(String(i.temperature))},[i.temperature]),x.useEffect(()=>{Qe(String(i.maxOutputTokens))},[i.maxOutputTokens]);const De=W=>{const de=W.replace(",",".").trim();if(!de){ce(String(i.temperature));return}const Me=Number(de);if(!Number.isFinite(Me)){ce(String(i.temperature));return}N({...i,temperature:Me}),ce(String(Me))},We=W=>{const de=W.trim();if(!de){Qe(String(i.maxOutputTokens));return}const Me=Number.parseInt(de,10);if(!Number.isFinite(Me)||Me<=0){Qe(String(i.maxOutputTokens));return}N({...i,maxOutputTokens:Me}),Qe(String(Me))},je=o.jsxs(o.Fragment,{children:[o.jsxs("div",{className:"grid-two",children:[o.jsxs("label",{children:["Provider",o.jsxs("select",{value:z,onChange:W=>{const de=W.target.value;if(de==="openai"){N({...i,llmProvider:"openai",baseUrl:Ac});return}if(de==="local_qwen25"){N({...i,llmProvider:"local",model:hl,baseUrl:cl});return}if(de==="local_qwen3"){N({...i,llmProvider:"local",model:ua,baseUrl:cl});return}N({...i,llmProvider:"local",model:i.llmProvider==="local"?i.model:hl,baseUrl:cl})},children:[o.jsx("option",{value:"openai",children:"OpenAI (token)"}),o.jsx("option",{value:"local_qwen25",children:"Qwen2.5 14B Instruct 1M (Local LM Studio)"}),o.jsx("option",{value:"local_qwen3",children:"Qwen3 30B A3B Instruct 2507 (Local LM Studio)"}),o.jsx("option",{value:"local_custom",children:"Local custom (LM Studio / OpenAI-compatible)"})]})]}),o.jsxs("label",{children:["Model",o.jsxs("select",{value:Ne?i.model:"__manual__",onChange:W=>{const de=W.target.value;de!=="__manual__"&&N({...i,model:de})},children:[o.jsx("option",{value:"__manual__",children:"Manual input"}),ne.map(W=>o.jsx("option",{value:W.value,children:W.label},W.value))]})]}),o.jsxs("label",{children:["Model ID (manual / current)",o.jsx("input",{value:i.model,onChange:W=>N({...i,model:W.target.value}),placeholder:"qwen2.5-14b-instruct-1m or unsloth/qwen3-30b-a3b-instruct-2507"})]}),G?null:o.jsxs("label",{className:"full-width",children:["OpenAI API Key",o.jsx("input",{type:"password",value:i.apiKey,onChange:W=>N({...i,apiKey:W.target.value}),placeholder:"sk-..."})]}),o.jsxs("label",{className:G?"full-width":void 0,children:[G?"Local server base URL":"Base URL",o.jsx("input",{value:i.baseUrl,onChange:W=>N({...i,baseUrl:W.target.value}),placeholder:G?cl:Ac})]}),o.jsxs("label",{children:["Temperature",o.jsx("input",{type:"number",step:"0.1",value:le,onChange:W=>ce(W.target.value),onBlur:W=>De(W.target.value),onKeyDown:W=>{W.key==="Enter"&&De(W.target.value)}})]}),o.jsxs("label",{children:["Max output tokens",o.jsx("input",{type:"number",value:me,onChange:W=>Qe(W.target.value),onBlur:W=>We(W.target.value),onKeyDown:W=>{W.key==="Enter"&&We(W.target.value)}})]})]}),o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",onClick:()=>$(),children:"Save local config"}),o.jsx("button",{type:"button",onClick:()=>j(),disabled:b||f,children:f?"Loading models...":"Load model list"}),o.jsx("button",{type:"button",onClick:()=>T(),disabled:b,children:b?"Checking...":"Test connection"})]})]});return Q?o.jsxs("section",{className:"embedded-panel-section",children:[o.jsxs("div",{className:"embedded-panel-section-header",children:[o.jsxs("div",{children:[o.jsx("h4",{children:"LLM Connector"}),o.jsx("p",{children:"Switch between OpenAI cloud and local OpenAI-compatible server."})]}),o.jsx("span",{className:"status-chip",children:Y||"Status: not checked"})]}),je]}):o.jsx(In,{title:"LLM Connector",subtitle:"Switch between OpenAI cloud and local OpenAI-compatible server.",actions:o.jsx("span",{className:"status-chip",children:Y||"Status: not checked"}),children:je})}function la({value:i,onChange:p,presets:f,selectedPresetId:N,onSelectPreset:j,onLoadPreset:T,onSavePreset:$,onResetDefaults:Y,onDiffPrevious:b,presetName:Q,onPresetNameChange:G,diffSummary:z,embedded:ne=!1}){const Ne=o.jsxs(o.Fragment,{children:[o.jsxs("div",{className:"prompt-manager-grid",children:[o.jsxs("label",{children:["Системный prompt",o.jsx("textarea",{value:i.systemPrompt,onChange:le=>p({...i,systemPrompt:le.target.value}),rows:6})]}),o.jsxs("label",{children:["Developer / Instruction prompt",o.jsx("textarea",{value:i.developerPrompt,onChange:le=>p({...i,developerPrompt:le.target.value}),rows:6})]}),o.jsxs("label",{children:["Domain prompt",o.jsx("textarea",{value:i.domainPrompt,onChange:le=>p({...i,domainPrompt:le.target.value}),rows:6})]}),o.jsxs("label",{children:["Schema notes",o.jsx("textarea",{value:i.schemaNotes,onChange:le=>p({...i,schemaNotes:le.target.value}),rows:6})]}),o.jsxs("label",{className:"full-width",children:["Few-shot examples",o.jsx("textarea",{value:i.fewShotExamples,onChange:le=>p({...i,fewShotExamples:le.target.value}),rows:8})]})]}),o.jsxs("div",{className:"button-row",children:[o.jsxs("select",{value:N,onChange:le=>j(le.target.value),children:[o.jsx("option",{value:"",children:"Выберите preset..."}),f.map(le=>o.jsx("option",{value:le.id,children:le.name},le.id))]}),o.jsx("button",{type:"button",onClick:()=>T(),children:"Загрузить preset"}),o.jsx("input",{value:Q,onChange:le=>G(le.target.value),placeholder:"Имя для сохранения"}),o.jsx("button",{type:"button",onClick:()=>$(),children:"Сохранить preset"}),o.jsx("button",{type:"button",onClick:()=>b(),children:"Diff с предыдущим"}),o.jsx("button",{type:"button",onClick:()=>Y(),children:"Сбросить к default"})]}),z?o.jsx("p",{className:"diff-summary",children:z}):null]});return ne?o.jsxs("section",{className:"embedded-panel-section",children:[o.jsx("div",{className:"embedded-panel-section-header",children:o.jsxs("div",{children:[o.jsx("h4",{children:"Prompt Manager"}),o.jsx("p",{children:"Системный, developer и domain уровни управляются отдельно."})]})}),Ne]}):o.jsx(In,{title:"Prompt Manager",subtitle:"Системный, developer и domain уровни управляются отдельно.",children:Ne})}const Yi={fromLocal:"",toLocal:"",target:"all",mode:"all",useMock:"any",promptContains:"",limit:120},dl="needs_dialog_policy_fix",Nt="__all__",gl="__live__:",Rc="ndc_autoruns_ui_config_v1",Tc="ndc-autoruns-save",Xi=["Анализ запроса","Получение данных","Подготовка ответа"];function rm(i,p){const f=i.trim();if(!f)return"";if(!p)return f;const N=f.toLowerCase(),j=p.anchor_text.trim(),T=j.toLowerCase();return T&&N.includes(T)?f:`По выбранному объекту "${j}": ${f}`}const ia=[{id:"general",label:"Общий контур",domain:"",defaultPrompt:"Генерируй реалистичные живые вопросы бухгалтера по 1С. Добавляй разговорные формулировки и опечатки, но сохраняй бизнес-смысл."}];function sm(i=ia){return i.reduce((p,f)=>(p[f.id]=f.defaultPrompt,p),{})}const Ic={mode:"codex_creative",count:24,personalityId:"general",personalityPrompts:sm(),persistToEvalCases:!0,generatedBy:"manual_reviewer"};function Zi(i){const p=String(i??"").trim();return/^\d{4}-\d{2}-\d{2}$/.test(p)?p:""}function Lc(i){const p=typeof i=="number"&&Number.isFinite(i)?Math.trunc(i):160;return Math.max(110,Math.min(520,p))}function om(i){const p=i.getFullYear(),f=String(i.getMonth()+1).padStart(2,"0"),N=String(i.getDate()).padStart(2,"0"),j=String(i.getHours()).padStart(2,"0"),T=String(i.getMinutes()).padStart(2,"0");return`${p}-${f}-${N}T${j}:${T}`}function Oc(){const i=new Date;return i.setDate(i.getDate()-14),om(i)}function fl(i){if(!i.trim())return;const p=Date.parse(i);if(Number.isFinite(p))return new Date(p).toISOString()}function vr(i){if(!i)return"нет данных";const p=Date.parse(i);return Number.isFinite(p)?new Date(p).toLocaleString("ru-RU"):i}function lm(i,p){return p<=0?0:Math.max(0,Math.min(100,Number((i/p*100).toFixed(1))))}function ml(i){return typeof i!="number"?"нет данных":`${i.toFixed(1)}%`}function im(i){return i==="assistant_stage1"?"assistant/s1":i==="assistant_stage2"?"assistant/s2":i==="assistant_p0"?"assistant/p0":i}function Dc(i){return i==="up"?"Рост":i==="down"?"Регресс":"Без изменений"}function am(i,p){return i.find(f=>f.case_id===p)??null}function ea(i){const p=Math.max(1,Math.min(5,Math.round(i)));return`${"●".repeat(p)}${"○".repeat(5-p)}`}function zc(i){return i.length===0?o.jsx("p",{className:"muted",children:"Покрытие доменов пока не сформировано."}):o.jsx("div",{className:"autoruns-coverage-list",children:i.map(p=>{const f=lm(p.closed_cases,p.total_cases);return o.jsxs("div",{className:"autoruns-coverage-item",children:[o.jsxs("div",{className:"autoruns-coverage-head",children:[o.jsx("strong",{children:p.domain}),o.jsxs("span",{children:[p.closed_cases,"/",p.total_cases," (",f,"%)"]})]}),o.jsx("div",{className:"autoruns-coverage-bar",children:o.jsx("div",{style:{width:`${f}%`}})})]},p.domain)})})}function pl(i){return`${gl}${i}`}function Ds(i){return i.startsWith(gl)}function $c(i){return i.startsWith(gl)?i.slice(gl.length):""}function qc(i){const p=i.report_summary?.run_timestamp??i.created_at,f=Math.max(0,i.total_cases-i.completed_cases);return{run_id:pl(i.job_id),eval_target:i.eval_target,run_timestamp:p,mode:"single-pass-strict",llm_provider:null,model:null,use_mock:null,analysis_date:i.report_summary?.analysis_date??i.analysis_date??null,prompt_version:null,schema_version:null,suite_id:i.case_set_file,cases_total:i.total_cases,requests_total:null,report_path:`async_job:${i.job_id}`,score_index:i.report_summary?.score_index??null,blocking_failures:0,quality_failures:0,closed_cases:i.completed_cases,open_cases:f,domain_coverage:[{domain:"runtime",total_cases:i.total_cases,closed_cases:i.completed_cases}]}}function po(i,p){const f=qc(i),N=i.cases.map(G=>({case_id:G.case_id,domain:null,query_class:null,status:G.status==="completed"?"closed":G.status==="failed"?"open":"unknown",score_index:null,trace_id:null,reply_type:null,session_id:`${i.run_id}-${G.case_id}`,dialog_available:G.messages.length>0,commented_count:0,latest_annotation_at:null,avg_rating:null,checks:null,metric_subscores:null})),T=p!==Nt&&N.some(G=>G.case_id===p)?p:N.length>0?Nt:"",$={ok:!0,run:f,coverage:{closed_cases:i.completed_cases,open_cases:Math.max(0,i.total_cases-i.completed_cases),domain_coverage:[{domain:"runtime",total_cases:i.total_cases,closed_cases:i.completed_cases}]},cases:N,annotations_summary:{total:0},report:i.report_summary?{run_id:i.report_summary.run_id,run_timestamp:i.report_summary.run_timestamp,score_index:i.report_summary.score_index,cases_total:i.report_summary.cases_total,analysis_date:i.report_summary.analysis_date??i.analysis_date??null}:{}},Y=[];let b=0;if(T===Nt)for(const G of i.cases)for(let z=0;zz.case_id===T)??null;for(let z=0;z<(G?.messages.length??0);z+=1){const ne=G?.messages[z];ne&&Y.push({...ne,message_index:z,case_id:T,case_message_index:z,commented:!1,annotation:null})}}const Q={ok:!0,run_id:f.run_id,case_id:T,source:"assistant_session",session_id:T===Nt?`${i.run_id}::__all__`:`${i.run_id}-${T}`,messages:Y,decomposition:[],assistant_mode:{status:i.status,completed_cases:i.completed_cases,total_cases:i.total_cases},annotations:[]};return{detail:$,dialog:Q,caseId:T}}function um({commented:i}){const p=i?"comment-icon-svg commented":"comment-icon-svg";return o.jsx("svg",{className:p,viewBox:"0 0 24 24","aria-hidden":"true",focusable:"false",children:o.jsx("path",{d:"M5 6.5h14v9H11.5l-4.5 3v-3H5z"})})}function Fc({resolved:i}){return o.jsxs("svg",{className:i?"resolve-icon-svg resolved":"resolve-icon-svg",viewBox:"0 0 16 16","aria-hidden":"true",focusable:"false",children:[o.jsx("circle",{cx:"8",cy:"8",r:"6.2"}),i?o.jsx("path",{d:"M5.1 8.2 7.2 10.3 11 6.5"}):null]})}function cm(){return o.jsxs("svg",{className:"autoruns-copy-icon-svg",viewBox:"0 0 24 24","aria-hidden":"true",focusable:"false",children:[o.jsx("rect",{x:"9",y:"9",width:"11",height:"11",rx:"2.2"}),o.jsx("path",{d:"M15 7V5.8a1.8 1.8 0 0 0-1.8-1.8H5.8A1.8 1.8 0 0 0 4 5.8v7.4A1.8 1.8 0 0 0 5.8 15H7"})]})}function dm({connection:i,modelOptions:p,modelsBusy:f,connectionStatus:N,connectionBusy:j,onConnectionChange:T,onReloadModels:$,onSaveLocalConfig:Y,onTestConnection:b,prompts:Q,onPromptsChange:G,promptPresets:z,selectedPresetId:ne,onSelectPreset:Ne,onLoadPreset:le,onSavePreset:ce,onResetDefaults:me,onDiffPrevious:Qe,presetName:De,onPresetNameChange:We,diffSummary:je,assistantPromptVersion:W,decompositionPromptVersion:de,showSettingsMode:Me,showAutoRunsMode:Ze,showAssistantMode:ct,showDecompositionMode:st,showProgressMode:Et,showCommentsMode:ie,onLog:Le}){const[H,Fe]=x.useState({...Yi,fromLocal:Oc()}),[Te,Ae]=x.useState(""),[I,X]=x.useState(null),[O,y]=x.useState(null),[C,J]=x.useState(null),[oe,pe]=x.useState([]),[ae,Se]=x.useState("all"),[ge,ve]=x.useState(!1),[Ee,gn]=x.useState(null),[vn,Yn]=x.useState([]),[Ln,Xn]=x.useState(""),[Pe,qt]=x.useState(""),[ot,dt]=x.useState(""),[lt,yn]=x.useState(ia),[re,It]=x.useState(Ic),[tt,Vr]=x.useState([]),[nn,xn]=x.useState(""),[xt,_n]=x.useState([]),[Oe,rn]=x.useState(null),[On,xr]=x.useState(null),[Dn,Qr]=x.useState(!1),[Sn,sn]=x.useState(!1),[Lt,Zn]=x.useState(!1),[Kt,on]=x.useState(!1),[Ot,_r]=x.useState(!1),[zn,Sr]=x.useState(!1),[$n,wr]=x.useState(!1),[ze,er]=x.useState(!1),[Dt,Fn]=x.useState(""),[wn,Ge]=x.useState(""),[nt,ln]=x.useState(""),[zt,Un]=x.useState([]),[bn,Gt]=x.useState([]),[jr,_t]=x.useState(""),[jn,an]=x.useState(null),[tr,Wr]=x.useState(!1),[Re,Pt]=x.useState(!1),[nr,Bn]=x.useState(""),[cs,D]=x.useState(""),[zs,Mt]=x.useState(String(Yi.limit)),[qr,St]=x.useState(String(Ic.count)),[rr,Kr]=x.useState(160),[ue,Ue]=x.useState({open:!1,caseId:"",caseMessageIndex:-1,messageIndex:-1,rating:3,comment:"",manualCaseDecision:dl,annotationAuthor:"manual_reviewer",saving:!1,error:""}),[ye,be]=x.useState({open:!1,messageIndex:-1,rating:3,comment:"",annotationAuthor:"manual_reviewer",saving:!1,error:""}),sr=x.useRef(!1),Hn=x.useRef(null),Vn=x.useMemo(()=>lt.find(a=>a.id===re.personalityId)??lt[0]??ia[0],[re.personalityId,lt]),$t=x.useMemo(()=>tt.find(a=>a.generation_id===nn)??tt[0]??null,[tt,nn]),kn=O?am(O.cases,ot):null,At=x.useMemo(()=>ge?oe.filter(a=>!a.resolved):oe,[oe,ge]),ke=At.find(a=>a.annotation_id===Ln)??null,ds=C?.messages.find(a=>a.message_index===ue.messageIndex)??null,$s=x.useMemo(()=>{if(!C||ue.messageIndex<0)return null;for(let a=ue.messageIndex-1;a>=0;a-=1){const c=C.messages[a];if(c?.role==="user")return c}return null},[ue.messageIndex,C]),Cn=x.useMemo(()=>{const a=new Map;for(const c of bn)c.message_id&&a.set(c.message_id,c);return a},[bn]),kr=ye.messageIndex>=0?zt[ye.messageIndex]??null:null,or=x.useMemo(()=>{if(ye.messageIndex<0)return null;for(let a=ye.messageIndex-1;a>=0;a-=1){const c=zt[a];if(c?.role==="user")return c}return null},[ye.messageIndex,zt]),it=x.useMemo(()=>{const a=At.map(v=>({source:"autorun",key:`autorun:${v.annotation_id}`,updated_at:v.updated_at,rating:v.rating,autorun:v,assistant:null})),c=bn.map(v=>({source:"assistant_live",key:`assistant:${v.annotation_id}`,updated_at:v.updated_at,rating:v.rating,autorun:null,assistant:v}));return[...a,...c].sort((v,R)=>Date.parse(R.updated_at)-Date.parse(v.updated_at))},[bn,At]),lr=x.useMemo(()=>{if(it.length===0)return null;const a=it.reduce((c,v)=>c+v.rating,0)/it.length;return Number(a.toFixed(2))},[it]),ir=x.useMemo(()=>{const a=[...I?.items??[]];return Oe&&a.unshift(qc(Oe)),Pe&&!a.some(c=>c.run_id===Pe)&&O?.run&&a.unshift(O.run),a},[Oe,I?.items,O?.run,Pe]),F=x.useCallback(a=>{Le?.(`[autoruns] ${a}`)},[Le]),ar=x.useCallback(async a=>{const c=String(a??"").trim();if(!c){Gt([]);return}try{const v=await _e.loadAssistantAnnotations({session_id:c,limit:400});Gt(v.items??[])}catch(v){const R=v instanceof Error?v.message:String(v);F(`Assistant live annotations load error: ${R}`)}},[F]),Nn=x.useCallback(a=>{be(c=>c.saving&&!a?.force?c:{open:!1,messageIndex:-1,rating:3,comment:"",annotationAuthor:"manual_reviewer",saving:!1,error:""})},[]),Cr=x.useCallback(async(a,c)=>{a.stopPropagation(),a.preventDefault();const v=String(c??"").trim();if(v)try{if(navigator?.clipboard?.writeText)await navigator.clipboard.writeText(v);else{const R=document.createElement("textarea");R.value=v,R.setAttribute("readonly","true"),R.style.position="fixed",R.style.opacity="0",document.body.appendChild(R),R.select(),document.execCommand("copy"),document.body.removeChild(R)}F(`run id copied: ${v}`)}catch(R){const fe=R instanceof Error?R.message:String(R);Ge(`Копирование run id: ${fe}`),F(`copy run id error: ${fe}`)}},[F]);function fs(){let a=0;Bn(Xi[0]);const c=window.setInterval(()=>{a=Math.min(a+1,Xi.length-1),Bn(Xi[a])},650);return()=>window.clearInterval(c)}const m=x.useCallback(()=>{ln(""),Un([]),Gt([]),_t(""),an(null),Bn(""),D(""),Nn({force:!0}),F("Live-чат ассистента в истории автопрогонов сброшен.")},[Nn,F]),E=x.useCallback(async()=>{const a=rm(jr,jn);if(!a)return;Pt(!0),D(""),_t(""),Un(v=>[...v,{message_id:`autoruns-live-${Date.now()}`,session_id:nt||"pending",role:"user",text:a,reply_type:null,created_at:new Date().toISOString(),trace_id:null,debug:null}]);const c=fs();try{const v=await _e.sendAssistantMessage({connection:i,prompts:Q,userMessage:a,sessionId:nt||void 0,promptVersion:W,useMock:tr});ln(v.session_id),Un(v.conversation),await ar(v.session_id),Bn("Ответ готов"),F(`Live-ответ ассистента получен: trace=${v.debug.trace_id}`)}catch(v){const R=v instanceof Error?v.message:String(v);D(R),Bn("Ошибка ассистента"),F(`Live-чат ассистента: ошибка отправки сообщения: ${R}`)}finally{c(),Pt(!1)}},[jr,jn,nt,tr,W,i,ar,F,Q]),k=x.useCallback(a=>{const c=a.trim();if(!c){Mt(String(H.limit));return}if(!/^\d+$/.test(c)){Mt(String(H.limit));return}const v=Number.parseInt(c,10);if(!Number.isFinite(v)){Mt(String(H.limit));return}const R=Math.max(1,Math.min(500,v));R!==H.limit&&Fe(fe=>({...fe,limit:R})),Mt(String(R))},[H.limit]),te=x.useCallback(a=>{const c=a.trim();if(!c){St(String(re.count));return}if(!/^\d+$/.test(c)){St(String(re.count));return}const v=Number.parseInt(c,10);if(!Number.isFinite(v)){St(String(re.count));return}const R=Math.max(1,Math.min(200,v));R!==re.count&&It(fe=>({...fe,count:R})),St(String(R))},[re.count]),Ie=x.useCallback(a=>{Kr(Lc(a))},[]),gt=x.useCallback(a=>{const c=a.currentTarget.offsetHeight;Number.isFinite(c)&&c>0&&Ie(c)},[Ie]),Ft=x.useCallback(async()=>{er(!0);try{const a=await _e.loadAutoRunAnnotations({limit:800,manual_case_decision:ae});pe(a.items),gn(a.manual_case_decision_schema??null),Yn(a.available_manual_case_decisions??[]),Xn(c=>a.items.length===0?"":a.items.some(v=>v.annotation_id===c)?c:a.items[0].annotation_id)}catch(a){F(`Annotations load error: ${a instanceof Error?a.message:String(a)}`)}finally{er(!1)}},[ae,F]),Jt=x.useCallback(async()=>{on(!0);try{const a=await _e.loadAutoRunAutogenHistory({limit:180});Vr(a.items)}catch(a){F(`Autogen history load error: ${a instanceof Error?a.message:String(a)}`)}finally{on(!1)}},[F]),Nr=x.useCallback(async()=>{try{const c=(await _e.loadAutoRunAutogenPersonalityCatalog()).items.map(v=>({id:String(v.id??"").trim(),label:String(v.label??"").trim(),domain:typeof v.domain=="string"?v.domain.trim():"",defaultPrompt:String(v.default_prompt??"").trim()})).filter(v=>v.id.length>0&&v.label.length>0);if(c.length===0)return;yn(c.map(v=>({id:v.id,label:v.label,domain:v.domain||"",defaultPrompt:v.defaultPrompt||"Генерируй реалистичные вопросы бухгалтера по выбранному профилю. Не выдумывай непокрытые возможности."})))}catch(a){F(`Autogen personality catalog load error: ${a instanceof Error?a.message:String(a)}`)}},[F]),ft=x.useCallback(async()=>{Zn(!0);try{const a=await _e.loadAutoRunPostAnalysis({run_id:Pe&&!Ds(Pe)?Pe:void 0,limit_per_queue:30,annotation_limit:1500,from:fl(H.fromLocal),to:fl(H.toLocal),target:H.target,mode:H.mode,use_mock:H.useMock,prompt_contains:H.promptContains.trim()||void 0});xr(a)}catch(a){F(`Post-analysis load error: ${a instanceof Error?a.message:String(a)}`),xr(null)}finally{Zn(!1)}},[H.fromLocal,H.mode,H.promptContains,H.target,H.toLocal,H.useMock,F,Pe]),vl=x.useCallback(async()=>{Qr(!0),Ge("");try{const a=re.personalityPrompts[re.personalityId]??"",c=[Q.systemPrompt,Q.developerPrompt,Q.domainPrompt,Q.schemaNotes,Q.fewShotExamples].join(` +`).slice(0,900),v=await _e.generateAutoRunQuestions({mode:re.mode,count:re.count,domain:Vn.domain||void 0,persist_to_eval_cases:re.persistToEvalCases,generated_by:re.generatedBy.trim()||void 0,llm:{llm_provider:i.llmProvider,api_key:i.apiKey,model:i.model,base_url:i.baseUrl,temperature:i.temperature,max_output_tokens:i.maxOutputTokens},context:{llm_provider:i.llmProvider,model:i.model,assistant_prompt_version:W,decomposition_prompt_version:de,prompt_fingerprint:c,autogen_personality_id:Vn.id,autogen_personality_prompt:a.trim()||void 0}});F(`Generated ${v.generation.count} questions (${v.generation.mode}) id=${v.generation.generation_id}`+(v.generation.saved_case_set_file?` saved=${v.generation.saved_case_set_file}`:"")),xn(v.generation.generation_id),_n([...v.generation.questions??[]]),await Jt()}catch(a){const c=a instanceof Error?a.message:String(a);Ge(`Автогенерация: ${c}`),F(`Autogen generate error: ${c}`)}finally{Qr(!1)}},[W,re.count,re.generatedBy,re.mode,re.personalityId,re.personalityPrompts,re.persistToEvalCases,i.apiKey,i.baseUrl,i.llmProvider,i.maxOutputTokens,i.model,i.temperature,de,Jt,F,Q.developerPrompt,Q.domainPrompt,Q.fewShotExamples,Q.schemaNotes,Q.systemPrompt,Vn.domain,Vn.id]),Er=x.useCallback(async(a,c)=>{if(Ds(a)){const v=$c(a);if(Oe&&Oe.job_id===v){const R=po(Oe,c);qt(a),dt(R.caseId),J(R.dialog);return}J(null);return}wr(!0);try{const v=await _e.loadAutoRunCaseDialog(a,c);J(v)}catch(v){const R=v instanceof Error?v.message:String(v);Ge(`Диалог кейса: ${R}`),J(null),F(`Dialog load error for ${a}/${c}: ${R}`)}finally{wr(!1)}},[Oe,F]),Ut=x.useCallback(async(a,c)=>{if(Ds(a)){const v=$c(a);if(Oe&&Oe.job_id===v){const R=po(Oe,c??Nt);qt(a),dt(R.caseId),y(R.detail),J(R.dialog);return}qt(a),dt(""),y(null),J(null);return}Sr(!0);try{const v=await _e.loadAutoRunDetail(a);y(v);const R=(c&&(c===Nt||v.cases.some(fe=>fe.case_id===c))?c:"")||(v.cases.length>0?Nt:"")||"";qt(a),dt(R),R?await Er(a,R):J(null)}catch(v){const R=v instanceof Error?v.message:String(v);Ge(`Детализация прогона: ${R}`),y(null),J(null),F(`Run detail load error for ${a}: ${R}`)}finally{Sr(!1)}},[Oe,Er,F]),Gr=x.useCallback(async a=>{_r(!0),Ge("");try{const c=await _e.loadAutoRunsHistory({from:fl(H.fromLocal),to:fl(H.toLocal),target:H.target,mode:H.mode,use_mock:H.useMock,prompt_contains:H.promptContains.trim()||void 0,limit:H.limit});if(X(c),c.items.length===0){qt(""),dt(""),y(null),J(null);return}const v=a?.keepSelection??!0,R=a?.preferredRunId??"",fe=a?.preferredCaseId??"",Je=v&&R&&c.items.some(Yt=>Yt.run_id===R)?R:c.items[0].run_id;await Ut(Je,v?fe:void 0),ft()}catch(c){const v=c instanceof Error?c.message:String(c);Ge(`История прогонов: ${v}`),F(`History load error: ${v}`)}finally{_r(!1)}},[H.fromLocal,H.limit,H.mode,H.promptContains,H.target,H.toLocal,H.useMock,ft,Ut,F]),un=x.useCallback(()=>{Hn.current!==null&&(window.clearTimeout(Hn.current),Hn.current=null)},[]),Pr=x.useCallback(async a=>{try{const c=await _e.loadEvalRunAsyncStatus(a);rn(c.job);const v=pl(a);if(Pe===v){const R=po(c.job,ot||Nt);y(R.detail),J(R.dialog),dt(R.caseId)}if(c.job.status==="completed"){un(),sn(!1);const R=c.job.report_summary?.run_id??c.job.run_id;await Gr({keepSelection:!0,preferredRunId:R||Pe,preferredCaseId:Nt}),await Jt(),rn(null);return}if(c.job.status==="failed"){un(),sn(!1),Ge(`Запуск прогонов: ${c.job.error??"неизвестная ошибка"}`),F(`Autogen async run failed: ${c.job.error??"unknown error"}`);return}un(),Hn.current=window.setTimeout(()=>{Pr(a)},500)}catch(c){un(),sn(!1);const v=c instanceof Error?c.message:String(c);Ge(`Запуск прогонов: ${v}`),F(`Autogen async status error: ${v}`)}},[Jt,Gr,F,ot,Pe,un]),ho=x.useCallback(async()=>{un(),sn(!0),Ge("");try{const a=$t;if(!a)throw new Error("История автогенерации пуста. Сначала сгенерируйте пачку вопросов.");const c=xt.map(Wn=>Wn.trim()).filter(Wn=>Wn.length>0);if(c.length===0)throw new Error("Нет вопросов для запуска: список пустой после ручного редактирования.");const v=H.useMock==="true",R=Zi(Te),Je=(await _e.startEvalRunAsync({connection:i,prompts:Q,promptVersion:W,mode:"single-pass-strict",caseSetFile:a.saved_case_set_file??void 0,useMock:v,evalTarget:"assistant_stage1",questions:c,analysisDate:R||void 0})).job;rn(Je);const Yt=pl(Je.job_id),Ye=po(Je,Nt);qt(Yt),dt(Ye.caseId),y(Ye.detail),J(Ye.dialog),F(`Запущен async-прогон job=${Je.job_id}, run_id=${Je.run_id}, вопросов=${c.length}`+(a.saved_case_set_file?`, base_case_set=${a.saved_case_set_file}`:"")+(R?`, analysis_date=${R}`:", analysis_date=current_state")),Pr(Je.job_id)}catch(a){const c=a instanceof Error?a.message:String(a);Ge(`Запуск прогонов: ${c}`),F(`Autogen run error: ${c}`),sn(!1)}},[Te,W,i,xt,H.useMock,F,Pr,Q,$t,un]),yl=x.useCallback(a=>{if(a.role!=="assistant")return;const c=a.case_id??ot,v=a.case_message_index??a.message_index;Ue({open:!0,caseId:c,caseMessageIndex:v,messageIndex:a.message_index,rating:a.annotation?.rating??3,comment:a.annotation?.comment??"",manualCaseDecision:a.annotation?.manual_case_decision??dl,annotationAuthor:a.annotation?.annotation_author??re.generatedBy,saving:!1,error:""})},[re.generatedBy,ot]),Qn=x.useCallback(a=>{Ue(c=>c.saving&&!a?.force?c:{open:!1,caseId:"",caseMessageIndex:-1,messageIndex:-1,rating:3,comment:"",manualCaseDecision:dl,annotationAuthor:re.generatedBy,saving:!1,error:""})},[re.generatedBy]),Jr=x.useCallback(async()=>{const a=Pe,c=ue.caseId,v=ue.caseMessageIndex;if(!(!a||!c||v<0)){if(Ds(a)){Ue(R=>({...R,error:"Комментарий можно сохранить после завершения прогона."}));return}if(!ue.comment.trim()){Ue(R=>({...R,error:"Добавьте комментарий."}));return}Ue(R=>({...R,saving:!0,error:""}));try{await _e.saveAutoRunAnnotation({run_id:a,case_id:c,message_index:v,rating:ue.rating,comment:ue.comment.trim(),manual_case_decision:ue.manualCaseDecision,annotation_author:ue.annotationAuthor.trim()||void 0}),Qn({force:!0}),Promise.all([Ut(a,ot),Ft(),ft()]).catch(R=>{const fe=R instanceof Error?R.message:String(R);Ge(`Обновление после комментария: ${fe}`),F(`Comment refresh error: ${fe}`)})}catch(R){Ue(fe=>({...fe,saving:!1,error:R instanceof Error?R.message:String(R)}))}}},[Qn,ue.annotationAuthor,ue.caseId,ue.caseMessageIndex,ue.comment,ue.manualCaseDecision,ue.rating,Ft,ft,Ut,F,ot,Pe]),Mr=x.useCallback(a=>a.role==="assistant",[]),ms=x.useCallback(a=>a.role==="assistant"&&Cn.has(a.message_id),[Cn]),xl=x.useCallback((a,c)=>{if(a.role!=="assistant")return;const v=nt.trim(),R=String(a.session_id??"").trim();if(!(v||R)){D("Сначала получите ответ ассистента в активной сессии.");return}!v&&R&&ln(R);const Je=Cn.get(a.message_id)??null;D(""),be({open:!0,messageIndex:c,rating:Je?.rating??3,comment:Je?.comment??"",annotationAuthor:Je?.annotation_author??"manual_reviewer",saving:!1,error:""})},[Cn,nt]),_l=x.useCallback(async()=>{if(ye.messageIndex<0)return;if(!ye.comment.trim()){be(v=>({...v,error:"Добавьте комментарий."}));return}const a=zt[ye.messageIndex]??null,c=nt.trim()||(a?.role==="assistant"?String(a.session_id??"").trim():"");if(!c){be(v=>({...v,error:"Сессия ассистента не найдена."}));return}be(v=>({...v,saving:!0,error:""}));try{const v=await _e.saveAssistantAnnotation({session_id:c,message_index:ye.messageIndex,rating:ye.rating,comment:ye.comment.trim(),annotation_author:ye.annotationAuthor.trim()||void 0});Gt(R=>{const fe=[...R],Je=fe.findIndex(Yt=>Yt.annotation_id===v.annotation.annotation_id);return Je>=0?fe[Je]=v.annotation:fe.unshift(v.annotation),fe.sort((Yt,Ye)=>Date.parse(Ye.updated_at)-Date.parse(Yt.updated_at))}),Nn({force:!0})}catch(v){const R=v instanceof Error?v.message:String(v);be(fe=>({...fe,saving:!1,error:R}))}},[ye.annotationAuthor,ye.comment,ye.messageIndex,ye.rating,zt,nt,Nn]),ps=x.useCallback(a=>{pe(c=>c.map(v=>v.annotation_id===a.annotation_id?{...v,...a}:v)),J(c=>c&&{...c,annotations:c.annotations.map(v=>v.annotation_id===a.annotation_id?a:v),messages:c.messages.map(v=>!v.annotation||v.annotation.annotation_id!==a.annotation_id?v:{...v,commented:!0,annotation:a})})},[]),Yr=x.useCallback(async(a,c)=>{if(a.annotation_id){if(Ds(a.run_id)){Ge("Статус выполнения можно менять только для завершённых прогонов.");return}Fn(a.annotation_id);try{const v=await _e.updateAutoRunAnnotation({annotation_id:a.annotation_id,resolved:c,resolved_by:re.generatedBy||void 0});ps(v.annotation),ft()}catch(v){const R=v instanceof Error?v.message:String(v);Ge(`Смена статуса кейса: ${R}`),F(`Annotation resolve toggle error: ${R}`)}finally{Fn("")}}},[ps,re.generatedBy,ft,F]),hs=x.useCallback(async a=>{Xn(a.annotation_id),await Ut(a.run_id,a.case_id),I?.items.some(c=>c.run_id===a.run_id)||Ge("Комментарий относится к прогону вне текущего фильтра. Детали загружены напрямую.")},[I?.items,Ut]);x.useEffect(()=>{sr.current||(sr.current=!0,Gr({keepSelection:!1}),Jt(),Nr(),ft())},[Jt,Nr,Gr,ft]),x.useEffect(()=>{sr.current&&Ft()},[ae,Ft]),x.useEffect(()=>{Xn(a=>At.length===0?"":At.some(c=>c.annotation_id===a)?a:At[0].annotation_id)},[At]),x.useEffect(()=>{xn(a=>tt.length===0?"":a&&tt.some(c=>c.generation_id===a)?a:tt[0].generation_id)},[tt]),x.useEffect(()=>{if(!$t){_n([]);return}_n([...$t.questions])},[$t?.generation_id]),x.useEffect(()=>{Mt(String(H.limit))},[H.limit]),x.useEffect(()=>{St(String(re.count))},[re.count]),x.useEffect(()=>{if(!nt.trim()){Gt([]);return}ar(nt)},[nt,ar]),x.useEffect(()=>{if(!Oe)return;const a=pl(Oe.job_id);if(Pe!==a)return;const c=po(Oe,ot||Nt);y(c.detail),J(c.dialog),dt(c.caseId)},[Oe,ot,Pe]),x.useEffect(()=>()=>{un()},[un]),x.useEffect(()=>{lt.length!==0&&It(a=>{let c=!1;const v={...a.personalityPrompts};for(const fe of lt)(typeof v[fe.id]!="string"||v[fe.id].trim().length===0)&&(v[fe.id]=fe.defaultPrompt,c=!0);let R=a.personalityId;return lt.some(fe=>fe.id===a.personalityId)||(R=lt[0].id,c=!0),c?{...a,personalityId:R,personalityPrompts:v}:a})},[lt]),x.useEffect(()=>{const a=localStorage.getItem(Rc);if(a)try{const c=JSON.parse(a);if(c.filters){const v=c.filters;Fe(R=>({...R,...v,limit:typeof v.limit=="number"?Math.max(1,Math.min(500,v.limit)):R.limit}))}typeof c.analysisDate=="string"&&Ae(Zi(c.analysisDate)),typeof c.autogenPersonalityPromptHeight=="number"&&Kr(Lc(c.autogenPersonalityPromptHeight)),c.autoGenSettings&&It(v=>{const R={...v.personalityPrompts},fe=c.autoGenSettings?.personalityPrompts??{};for(const[Yt,Ye]of Object.entries(fe))typeof Ye=="string"&&Yt.trim().length>0&&(R[Yt.trim()]=Ye);const Je=typeof c.autoGenSettings?.personalityId=="string"&&c.autoGenSettings.personalityId.trim().length>0?c.autoGenSettings.personalityId.trim():v.personalityId;return{...v,mode:c.autoGenSettings?.mode==="codex_creative"||c.autoGenSettings?.mode==="qwen_seed"?c.autoGenSettings.mode:v.mode,count:typeof c.autoGenSettings?.count=="number"?Math.max(1,Math.min(200,c.autoGenSettings.count)):v.count,personalityId:Je,personalityPrompts:R,persistToEvalCases:typeof c.autoGenSettings?.persistToEvalCases=="boolean"?c.autoGenSettings.persistToEvalCases:v.persistToEvalCases,generatedBy:typeof c.autoGenSettings?.generatedBy=="string"?c.autoGenSettings.generatedBy:v.generatedBy}}),(c.annotationDecisionFilter==="all"||typeof c.annotationDecisionFilter=="string"&&c.annotationDecisionFilter.length>0)&&Se(c.annotationDecisionFilter),typeof c.hideResolvedAnnotations=="boolean"&&ve(c.hideResolvedAnnotations)}catch{}},[]);const Fs=x.useCallback(()=>{const a={filters:H,analysisDate:Te,autogenPersonalityPromptHeight:rr,autoGenSettings:{mode:re.mode,count:re.count,personalityId:re.personalityId,personalityPrompts:re.personalityPrompts,persistToEvalCases:re.persistToEvalCases,generatedBy:re.generatedBy},annotationDecisionFilter:ae,hideResolvedAnnotations:ge};localStorage.setItem(Rc,JSON.stringify(a))},[Te,ae,re,rr,H,ge]);return x.useEffect(()=>{const a=()=>{Fs(),F("Сохранены настройки панели автопрогонов.")};return window.addEventListener(Tc,a),()=>{window.removeEventListener(Tc,a)}},[F,Fs]),o.jsxs(In,{className:"autoruns-frame",title:"",hideHeader:!0,children:[o.jsxs("div",{className:"autoruns-columns",children:[Me?o.jsxs("section",{className:"autoruns-col autoruns-settings-col",children:[o.jsx("div",{className:"autoruns-col-header",children:o.jsx("h3",{children:"Настройки"})}),o.jsxs("div",{className:"autoruns-settings-stack",children:[o.jsx(oa,{embedded:!0,value:i,modelOptions:p,modelsBusy:f,onChange:T,onReloadModels:$,onSaveLocalConfig:Y,onTestConnection:b,lastStatus:N,busy:j}),o.jsx(la,{embedded:!0,value:Q,onChange:G,presets:z,selectedPresetId:ne,onSelectPreset:Ne,onLoadPreset:le,onSavePreset:ce,onResetDefaults:me,onDiffPrevious:Qe,presetName:De,onPresetNameChange:We,diffSummary:je})]})]}):null,Ze?o.jsxs("section",{className:"autoruns-col",children:[o.jsx("div",{className:"autoruns-col-header",children:o.jsx("h3",{children:"Автопрогоны"})}),o.jsx("h4",{children:"Настройки выборки"}),o.jsxs("div",{className:"autoruns-form-grid",children:[o.jsxs("label",{children:["Дата с",o.jsx("input",{type:"datetime-local",value:H.fromLocal,onChange:a=>Fe(c=>({...c,fromLocal:a.target.value}))})]}),o.jsxs("label",{children:["Дата по",o.jsx("input",{type:"datetime-local",value:H.toLocal,onChange:a=>Fe(c=>({...c,toLocal:a.target.value}))})]}),o.jsxs("label",{children:["Целевой контур",o.jsxs("select",{value:H.target,onChange:a=>Fe(c=>({...c,target:a.target.value})),children:[o.jsx("option",{value:"all",children:"все"}),(I?.available.targets??[]).map(a=>o.jsx("option",{value:a,children:a},a))]})]}),o.jsxs("label",{children:["Режим",o.jsxs("select",{value:H.mode,onChange:a=>Fe(c=>({...c,mode:a.target.value})),children:[o.jsx("option",{value:"all",children:"все"}),(I?.available.modes??[]).map(a=>o.jsx("option",{value:a,children:a},a))]})]}),o.jsxs("label",{children:["Использовать mock",o.jsxs("select",{value:H.useMock,onChange:a=>Fe(c=>({...c,useMock:a.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:zs,onChange:a=>{const c=a.target.value;(c===""||/^\d+$/.test(c))&&Mt(c)},onBlur:a=>k(a.target.value),onKeyDown:a=>{a.key==="Enter"&&k(a.target.value)}})]}),o.jsxs("label",{className:"full-width",children:["Версия промпта содержит",o.jsx("input",{value:H.promptContains,onChange:a=>Fe(c=>({...c,promptContains:a.target.value})),placeholder:"normalizer_v2_0_2 / address_query_runtime_v1",list:"autoruns-prompt-versions"})]})]}),o.jsx("datalist",{id:"autoruns-prompt-versions",children:(I?.available.prompt_versions??[]).map(a=>o.jsx("option",{value:a},a))}),o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",disabled:Ot,onClick:()=>{Gr({keepSelection:!1})},children:Ot?"Обновляю...":"Применить"}),o.jsx("button",{type:"button",className:"tab",onClick:()=>{Fe({...Yi,fromLocal:Oc()}),Ge("")},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:i.llmProvider})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Модель:"}),o.jsx("strong",{children:i.model||"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Промпт ассистента:"}),o.jsx("strong",{children:W})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Промпт декомпозиции:"}),o.jsx("strong",{children:de})]})]}),o.jsx("h4",{children:"Автогенерация вопросов"}),o.jsxs("div",{className:"autoruns-form-grid",children:[o.jsxs("label",{children:["Режим генерации",o.jsxs("select",{value:re.mode,onChange:a=>It(c=>({...c,mode:a.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:qr,onChange:a=>{const c=a.target.value;(c===""||/^\d+$/.test(c))&&St(c)},onBlur:a=>te(a.target.value),onKeyDown:a=>{a.key==="Enter"&&te(a.target.value)}})]}),o.jsxs("label",{children:["Личность автогенерации",o.jsx("select",{value:re.personalityId,onChange:a=>It(c=>({...c,personalityId:a.target.value})),children:lt.map(a=>o.jsx("option",{value:a.id,children:a.label},a.id))})]}),o.jsxs("label",{children:["Кто генерирует",o.jsx("input",{value:re.generatedBy,onChange:a=>It(c=>({...c,generatedBy:a.target.value})),placeholder:"manual_reviewer"})]}),o.jsxs("label",{className:"full-width",children:["Промпт личности",o.jsx("textarea",{className:"autoruns-personality-prompt",value:re.personalityPrompts[re.personalityId]??"",onChange:a=>It(c=>({...c,personalityPrompts:{...c.personalityPrompts,[c.personalityId]:a.target.value}})),placeholder:"Текст промпта для выбранной личности автогенерации",style:{height:`${rr}px`},onMouseUp:gt,onTouchEnd:gt})]}),o.jsxs("label",{className:"checkbox-row",children:[o.jsx("input",{type:"checkbox",checked:re.persistToEvalCases,onChange:a=>It(c=>({...c,persistToEvalCases:a.target.checked}))}),"Сохранять кейс-сет в `eval_cases`"]})]}),o.jsxs("div",{className:"autoruns-form-grid",children:[o.jsxs("label",{children:["Дата анализа (срез)",o.jsx("input",{type:"date",value:Te,onChange:a=>Ae(Zi(a.target.value))})]}),o.jsx("div",{className:"button-row",children:o.jsx("button",{type:"button",className:"tab",disabled:!Te,onClick:()=>Ae(""),children:"Сбросить дату среза"})})]}),o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",disabled:Dn,onClick:()=>{vl()},children:Dn?"Генерирую...":"Сгенерировать пачку"}),o.jsx("button",{type:"button",className:"tab",disabled:Kt,onClick:()=>{Jt()},children:Kt?"Обновляю...":"Обновить историю"}),o.jsx("button",{type:"button",className:"autoruns-run-launch-btn",disabled:Sn||xt.length===0,onClick:()=>{ho()},children:Sn?"Запускаю...":"Запустить прогоны"})]}),o.jsx("div",{className:"autoruns-form-grid",children:o.jsxs("label",{className:"full-width",children:["Кейс-сет для запуска",o.jsxs("select",{value:nn,onChange:a=>xn(a.target.value),disabled:tt.length===0,children:[tt.length===0?o.jsx("option",{value:"",children:"нет генераций"}):null,tt.map(a=>o.jsxs("option",{value:a.generation_id,children:[vr(a.created_at)," | ",a.mode," | ",a.count," | ",a.saved_case_set_file??"без файла"]},a.generation_id))]})]})}),o.jsxs("div",{className:"autoruns-generated-questions",children:[o.jsxs("div",{className:"autoruns-generated-questions-head",children:[o.jsxs("strong",{children:["Вопросы к запуску: ",xt.length]}),o.jsx("button",{type:"button",className:"tab",onClick:()=>_n([...$t?.questions??[]]),disabled:!$t,children:"Восстановить"})]}),xt.length===0?o.jsx("p",{className:"muted",children:"Список вопросов пуст. Сгенерируйте пачку или восстановите из выбранной генерации."}):o.jsx("div",{className:"autoruns-generated-questions-list",children:xt.map((a,c)=>o.jsxs("div",{className:"autoruns-generated-question-item",children:[o.jsxs("span",{children:[c+1,". ",a]}),o.jsx("button",{type:"button",className:"autoruns-remove-question-btn",onClick:()=>_n(v=>v.filter((R,fe)=>fe!==c)),title:"Удалить вопрос из запуска","aria-label":"Удалить вопрос из запуска",children:"+"})]},`${c}-${a.slice(0,24)}`))})]}),o.jsx("p",{className:"muted",children:"Запуск выполняет `assistant_stage1` eval по выбранному кейс-сету."}),o.jsxs("div",{className:"autoruns-autogen-list",children:[Kt?o.jsx("p",{className:"muted",children:"Загружаю историю автогенераций..."}):null,!Kt&&tt.length===0?o.jsx("p",{className:"muted",children:"История автогенераций пока пустая."}):null,tt.slice(0,30).map(a=>o.jsxs("article",{className:nn===a.generation_id?"autoruns-autogen-item selected":"autoruns-autogen-item",onClick:()=>xn(a.generation_id),children:[o.jsxs("header",{children:[o.jsx("strong",{children:vr(a.created_at)}),o.jsx("span",{children:a.mode})]}),o.jsxs("div",{className:"autoruns-run-meta",children:["id=",a.generation_id," | count=",a.count]}),o.jsxs("div",{className:"autoruns-run-meta",children:["домен=",a.domain??"общий",a.generated_by?` | автор=${a.generated_by}`:""]}),a.saved_case_set_file?o.jsxs("div",{className:"autoruns-run-meta",children:["кейс-сет=",a.saved_case_set_file]}):null,(a.questions??[]).length>0?o.jsx("p",{children:a.questions[0]}):null]},a.generation_id))]}),o.jsxs("details",{className:"autoruns-prompt-details",children:[o.jsx("summary",{children:"Копия активного промпта (только чтение)"}),o.jsxs("label",{children:["Системный",o.jsx("textarea",{readOnly:!0,value:Q.systemPrompt})]}),o.jsxs("label",{children:["Разработчика",o.jsx("textarea",{readOnly:!0,value:Q.developerPrompt})]}),o.jsxs("label",{children:["Доменный",o.jsx("textarea",{readOnly:!0,value:Q.domainPrompt})]}),o.jsxs("label",{children:["Заметки по схеме",o.jsx("textarea",{readOnly:!0,value:Q.schemaNotes})]}),o.jsxs("label",{children:["Примеры few-shot",o.jsx("textarea",{readOnly:!0,value:Q.fewShotExamples})]})]}),wn?o.jsx("p",{className:"error-text",children:wn}):null]}):null,o.jsxs("section",{className:"autoruns-col",children:[o.jsx("div",{className:"autoruns-col-header",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:(I?.stats.runs_total??0)+(Oe?1:0)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Средний score"}),o.jsx("strong",{children:ml(I?.stats.avg_score_index??null)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Тренд"}),o.jsx("strong",{children:I?Dc(I.stats.trend):"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Блокеры"}),o.jsx("strong",{children:I?.stats.blocking_runs??0})]})]}),o.jsxs("div",{className:"autoruns-run-list",children:[ir.map(a=>o.jsxs("button",{type:"button",className:Pe===a.run_id?"autoruns-run-item selected":"autoruns-run-item",onClick:()=>{Ut(a.run_id)},children:[o.jsxs("div",{className:"autoruns-run-head",children:[o.jsx("strong",{children:vr(a.run_timestamp)}),o.jsx("span",{children:im(a.eval_target)})]}),o.jsxs("div",{className:"autoruns-run-meta autoruns-run-id-row",children:[o.jsx("span",{children:a.run_id}),o.jsx("span",{role:"button",tabIndex:0,className:"autoruns-copy-run-id-btn",onClick:c=>{Cr(c,a.run_id)},onKeyDown:c=>{(c.key==="Enter"||c.key===" ")&&(c.preventDefault(),Cr(c,a.run_id))},title:"Скопировать run id","aria-label":`Скопировать run id ${a.run_id}`,children:o.jsx(cm,{})})]}),o.jsxs("div",{className:"autoruns-run-meta",children:["режим=",a.mode??"нет данных"," | mock=",String(a.use_mock)]}),o.jsxs("div",{className:"autoruns-run-meta",children:["analysis_date=",a.analysis_date??"current_state"]}),a.llm_provider||a.model?o.jsxs("div",{className:"autoruns-run-meta",children:["llm=",a.llm_provider??"нет данных"," | модель=",a.model??"нет данных"]}):null,o.jsxs("div",{className:"autoruns-run-meta",children:["промпт=",a.prompt_version??"нет данных"]}),o.jsxs("div",{className:"autoruns-run-foot",children:[o.jsxs("span",{children:["оценка: ",ml(a.score_index)]}),o.jsxs("span",{children:["закрыто/открыто: ",a.closed_cases,"/",a.open_cases]})]}),o.jsxs("div",{className:"autoruns-run-foot",children:[o.jsxs("span",{children:["блокеры: ",a.blocking_failures]}),o.jsxs("span",{children:["качество: ",a.quality_failures]})]})]},a.run_id)),ir.length===0?o.jsx("p",{className:"muted",children:"За выбранный диапазон прогонов нет."}):null]})]}),o.jsxs("section",{className:"autoruns-col",children:[o.jsxs("div",{className:"autoruns-col-header",children:[o.jsx("h3",{children:"Диалог прогона"}),o.jsxs("div",{className:"autoruns-dialog-toolbar",children:[o.jsxs("label",{children:["Прогон",o.jsx("select",{value:Pe,onChange:a=>{const c=a.target.value;Ut(c)},children:ir.map(a=>o.jsxs("option",{value:a.run_id,children:[vr(a.run_timestamp)," | ",a.run_id]},a.run_id))})]}),o.jsxs("label",{children:["Кейс",o.jsxs("select",{value:ot,onChange:a=>{const c=a.target.value;dt(c),Pe&&c&&Er(Pe,c)},children:[(O?.cases.length??0)>0?o.jsx("option",{value:Nt,children:"ВСЕ кейсы подряд"}):null,(O?.cases??[]).map(a=>o.jsxs("option",{value:a.case_id,children:[a.case_id," | ",a.status]},a.case_id))]})]})]})]}),o.jsxs("div",{className:"autoruns-case-list",children:[(O?.cases.length??0)>0?o.jsxs("button",{type:"button",className:ot===Nt?"autoruns-case-item selected":"autoruns-case-item",onClick:()=>{dt(Nt),Pe&&Er(Pe,Nt)},children:[o.jsx("span",{children:"ВСЕ кейсы подряд"}),o.jsx("span",{children:O?.cases.length})]},Nt):null,(O?.cases??[]).map(a=>o.jsxs("button",{type:"button",className:ot===a.case_id?"autoruns-case-item selected":"autoruns-case-item",onClick:()=>{dt(a.case_id),Pe&&Er(Pe,a.case_id)},children:[o.jsx("span",{children:a.case_id}),o.jsxs("span",{children:[a.status,a.commented_count>0?` | комм=${a.commented_count}`:""]})]},a.case_id))]}),o.jsxs("div",{className:"autoruns-dialog-view",children:[$n||zn?o.jsx("p",{className:"muted",children:"Загружаю диалог..."}):null,!$n&&!zn&&(C?.messages.length??0)===0?o.jsx("p",{className:"muted",children:"Диалог для этого прогона не найден."}):null,(C?.messages??[]).map((a,c)=>{const v=a.role==="assistant"?"assistant":"user";return o.jsxs("article",{className:`autoruns-msg ${v}`,children:[o.jsxs("header",{children:[o.jsx("strong",{children:v==="assistant"?"Система":"Модель/вопрос"}),o.jsxs("div",{className:"autoruns-msg-head-actions",children:[a.case_id?o.jsx("span",{className:"autoruns-msg-case-tag",children:a.case_id}):null,o.jsx("span",{children:a.created_at?vr(a.created_at):"нет данных"}),v==="assistant"&&!Ds(Pe)?o.jsxs(o.Fragment,{children:[o.jsx("button",{type:"button",className:a.commented?"autoruns-comment-icon commented":"autoruns-comment-icon",onClick:()=>yl(a),title:"\\u041a\\u043e\\u043c\\u043c\\u0435\\u043d\\u0442\\u0438\\u0440\\u043e\\u0432\\u0430\\u0442\\u044c \\u043e\\u0442\\u0432\\u0435\\u0442 \\u0441\\u0438\\u0441\\u0442\\u0435\\u043c\\u044b","aria-label":"\\u041a\\u043e\\u043c\\u043c\\u0435\\u043d\\u0442\\u0438\\u0440\\u043e\\u0432\\u0430\\u0442\\u044c \\u043e\\u0442\\u0432\\u0435\\u0442 \\u0441\\u0438\\u0441\\u0442\\u0435\\u043c\\u044b",children:o.jsx(um,{commented:a.commented})}),a.annotation?o.jsx("button",{type:"button",className:a.annotation.resolved?"autoruns-resolve-toggle resolved":"autoruns-resolve-toggle",onClick:()=>{Yr(a.annotation,!a.annotation.resolved)},disabled:Dt===a.annotation.annotation_id,title:a.annotation.resolved?"Отметить кейс как невыполненный":"Отметить кейс как выполненный","aria-label":a.annotation.resolved?"Отметить кейс как невыполненный":"Отметить кейс как выполненный",children:o.jsx(Fc,{resolved:a.annotation.resolved})}):null]}):null]})]}),o.jsx("p",{children:a.text}),v==="assistant"&&a.annotation?o.jsxs("div",{className:"autoruns-msg-annotation",children:[o.jsx("strong",{children:ea(a.annotation.rating)}),o.jsx("span",{children:a.annotation.comment}),o.jsxs("span",{className:"muted",children:[a.annotation.manual_case_decision,a.annotation.annotation_author?` | ${a.annotation.annotation_author}`:""]})]}):null,(a.trace_id||a.reply_type)&&o.jsxs("footer",{children:[a.trace_id?o.jsxs("span",{children:["trace=",a.trace_id]}):null,a.reply_type?o.jsxs("span",{children:["reply_type=",a.reply_type]}):null]})]},a.message_id??`${v}-${c}`)})]})]}),ct?o.jsx("div",{className:"autoruns-col autoruns-assistant-live-col",children:o.jsx(Wc,{sessionId:nt,conversation:zt,inputValue:jr,onInputChange:_t,selectedContextChip:jn,onSelectContextChip:an,onClearContextChip:()=>an(null),useMock:tr,onUseMockChange:Wr,onSend:E,onClear:m,busy:Re,statusText:nr,errorMessage:cs,showCommentAction:!0,onCommentAssistantMessage:xl,isAssistantMessageCommented:ms,canCommentAssistantMessage:Mr})}):null,st?o.jsxs("section",{className:"autoruns-col",children:[o.jsx("div",{className:"autoruns-col-header",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:kn?.case_id??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"домен:"}),o.jsx("strong",{children:kn?.domain??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"класс запроса:"}),o.jsx("strong",{children:kn?.query_class??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"trace:"}),o.jsx("strong",{children:kn?.trace_id??"нет данных"})]})]}),o.jsx("h4",{children:"Шаги декомпозиции"}),(C?.decomposition.length??0)>0?o.jsx("ol",{className:"autoruns-decomposition-list",children:(C?.decomposition??[]).map((a,c)=>o.jsx("li",{children:a},`${c}-${a.slice(0,24)}`))}):o.jsx("p",{className:"muted",children:"В логах кейса нет явной декомпозиции."})]}):null,Et?o.jsxs("section",{className:"autoruns-col",children:[o.jsx("div",{className:"autoruns-col-header",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:ml(I?.stats.latest_score_index??null)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Предыдущий"}),o.jsx("strong",{children:ml(I?.stats.previous_score_index??null)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Тренд"}),o.jsx("strong",{children:I?Dc(I.stats.trend):"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Пробелы качества"}),o.jsx("strong",{children:I?.stats.quality_gap_runs??0})]})]}),o.jsx("h4",{children:"Покрытие доменов (история)"}),zc(I?.stats.domain_coverage??[]),o.jsx("h4",{style:{marginTop:14},children:"Покрытие доменов (выбранный прогон)"}),zc(O?.coverage.domain_coverage??[]),o.jsx("h4",{style:{marginTop:14},children:"Очереди фиксов пост-анализа"}),Lt?o.jsx("p",{className:"muted",children:"Собираю пост-анализ..."}):null,Lt?null:o.jsx("div",{className:"autoruns-stats-grid",children:Object.entries(On?.post_analysis.stats.by_queue??{}).map(([a,c])=>o.jsxs("div",{children:[o.jsx("span",{children:a}),o.jsx("strong",{children:c})]},a))}),o.jsxs("div",{className:"autoruns-autogen-list",children:[(On?.post_analysis.recommended_regression_candidates??[]).slice(0,12).map(a=>o.jsxs("article",{className:"autoruns-autogen-item",children:[o.jsxs("header",{children:[o.jsx("strong",{children:a.manual_case_decision}),o.jsxs("span",{children:[a.rating,"/5"]})]}),o.jsxs("div",{className:"autoruns-run-meta",children:[a.domain??"неизвестно"," / ",a.query_class??"неизвестно"]}),o.jsx("p",{children:a.comment})]},a.annotation_id)),!Lt&&(On?.post_analysis.recommended_regression_candidates.length??0)===0?o.jsx("p",{className:"muted",children:"Рекомендованных кандидатов пока нет."}):null]})]}):null,ie?o.jsxs("section",{className:"autoruns-col",children:[o.jsx("div",{className:"autoruns-col-header",children:o.jsx("h3",{children:"Комментарии"})}),o.jsx("h4",{children:"Размеченные ответы"}),o.jsxs("div",{className:"autoruns-comment-filter-row",children:[o.jsxs("label",{children:["Фильтр решений",o.jsxs("select",{value:ae,onChange:a=>Se(a.target.value),children:[o.jsx("option",{value:"all",children:"все"}),(vn.length>0?vn:Ee?.enum??[]).map(a=>o.jsx("option",{value:a,children:String(Ee?.labels?.[a]??a)},a))]})]}),o.jsx("button",{type:"button",className:"tab autoruns-resolved-filter-toggle",onClick:()=>ve(a=>!a),children:ge?"Показать выполненные":"Скрыть выполненные"})]}),o.jsxs("div",{className:"autoruns-stats-grid",children:[o.jsxs("div",{children:[o.jsx("span",{children:"Комментариев"}),o.jsx("strong",{children:it.length})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Средний рейтинг"}),o.jsx("strong",{children:lr===null?"нет данных":`${lr.toFixed(2)} / 5`})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Последний"}),o.jsx("strong",{children:it.length>0?vr(it[0].updated_at):"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Статус"}),o.jsx("strong",{children:ze?"обновляю":"готово"})]})]}),o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",disabled:ze,onClick:()=>{Ft()},children:ze?"Обновляю...":"Обновить список"}),o.jsx("button",{type:"button",className:"tab",disabled:Lt,onClick:()=>{ft()},children:Lt?"Идет пост-анализ...":"Обновить пост-анализ"})]}),o.jsxs("div",{className:"autoruns-comments-list",children:[ze?o.jsx("p",{className:"muted",children:"Загружаю комментарии..."}):null,!ze&&it.length===0?o.jsx("p",{className:"muted",children:oe.length===0&&bn.length===0?"Пока нет откомментированных ответов.":"Нет открытых кейсов по текущему фильтру."}):null,it.map(a=>{if(a.source==="assistant_live"){const v=a.assistant;return o.jsxs("article",{className:"autoruns-comment-item",children:[o.jsxs("div",{className:"autoruns-comment-head",children:[o.jsx("strong",{children:ea(v.rating)}),o.jsx("div",{className:"autoruns-comment-head-actions",children:o.jsx("span",{children:vr(v.updated_at)})})]}),o.jsxs("div",{className:"autoruns-run-meta",children:["live-session: ",v.session_id]}),o.jsxs("div",{className:"autoruns-run-meta",children:["msg=",v.message_index]}),o.jsxs("div",{className:"autoruns-run-meta",children:["source=assistant_live",v.annotation_author?` | author=${v.annotation_author}`:""]}),v.context.question_text?o.jsxs("p",{children:["Q: ",v.context.question_text]}):null,v.context.answer_text?o.jsxs("p",{children:["A: ",v.context.answer_text]}):null,o.jsx("p",{children:v.comment})]},a.key)}const c=a.autorun;return o.jsxs("article",{className:Ln===c.annotation_id?"autoruns-comment-item selected":"autoruns-comment-item",onClick:()=>{hs(c)},role:"button",tabIndex:0,onKeyDown:v=>{(v.key==="Enter"||v.key===" ")&&(v.preventDefault(),hs(c))},children:[o.jsxs("div",{className:"autoruns-comment-head",children:[o.jsx("strong",{children:ea(c.rating)}),o.jsxs("div",{className:"autoruns-comment-head-actions",children:[o.jsx("span",{children:vr(c.updated_at)}),o.jsx("button",{type:"button",className:c.resolved?"autoruns-resolve-toggle resolved":"autoruns-resolve-toggle",onClick:v=>{v.preventDefault(),v.stopPropagation(),Yr(c,!c.resolved)},disabled:Dt===c.annotation_id,title:c.resolved?"Отметить кейс как невыполненный":"Отметить кейс как выполненный","aria-label":c.resolved?"Отметить кейс как невыполненный":"Отметить кейс как выполненный",children:o.jsx(Fc,{resolved:c.resolved})})]})]}),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}`:""]}),c.resolved_at?o.jsxs("div",{className:"autoruns-run-meta",children:["выполнено",": ",vr(c.resolved_at),c.resolved_by?` | by=${c.resolved_by}`:""]}):null,c.context.question_text?o.jsxs("p",{children:["Q: ",c.context.question_text]}):null,c.context.answer_text?o.jsxs("p",{children:["A: ",c.context.answer_text]}):null,o.jsx("p",{children:c.comment})]},a.key)})]}),ke?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:ke.technical_context.trace_id??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"reply_type:"}),o.jsx("strong",{children:ke.technical_context.reply_type??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"domain:"}),o.jsx("strong",{children:ke.technical_context.domain??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"query_class:"}),o.jsx("strong",{children:ke.technical_context.query_class??"нет данных"})]})]}),o.jsx("h4",{children:"JSON разбор"}),o.jsx(tn,{value:{annotation_id:ke.annotation_id,run_id:ke.run_id,case_id:ke.case_id,message_index:ke.message_index,rating:ke.rating,comment:ke.comment,manual_case_decision:ke.manual_case_decision,annotation_author:ke.annotation_author,resolved:ke.resolved,resolved_at:ke.resolved_at,resolved_by:ke.resolved_by,context:ke.context,technical_context:ke.technical_context,case_summary:ke.case_summary?{case_id:ke.case_summary.case_id,domain:ke.case_summary.domain,query_class:ke.case_summary.query_class,checks:ke.case_summary.checks,metric_subscores:ke.case_summary.metric_subscores}:null}})]}):null]}):null]}),ye.open?o.jsx("div",{className:"autoruns-comment-modal-backdrop",onClick:a=>{a.target===a.currentTarget&&Nn()},children:o.jsxs("div",{className:"autoruns-comment-modal",children:[o.jsx("h3",{children:"Комментарий к ответу ассистента"}),o.jsx("p",{className:"muted",children:"Комментарий будет добавлен в общий список комментариев справа с меткой `assistant_live`."}),or?o.jsxs("details",{className:"autoruns-prompt-details",open:!0,children:[o.jsx("summary",{children:"Вопрос пользователя"}),o.jsx("p",{className:"autoruns-comment-quote",children:or.text})]}):null,kr?o.jsxs("details",{className:"autoruns-prompt-details",open:!0,children:[o.jsx("summary",{children:"Ответ ассистента"}),o.jsx("p",{className:"autoruns-comment-quote",children:kr.text})]}):null,o.jsx("div",{className:"autoruns-rating-row",role:"group","aria-label":"Рейтинг ответа ассистента",children:[1,2,3,4,5].map(a=>o.jsx("button",{type:"button",className:ye.rating>=a?"autoruns-rating-dot active":"autoruns-rating-dot",onClick:()=>be(c=>({...c,rating:a})),disabled:ye.saving,"aria-label":`Оценка ${a}`,children:ye.rating>=a?"●":"○"},a))}),o.jsx("div",{className:"autoruns-form-grid",children:o.jsxs("label",{children:["Автор комментария",o.jsx("input",{value:ye.annotationAuthor,onChange:a=>be(c=>({...c,annotationAuthor:a.target.value})),placeholder:"manual_reviewer",disabled:ye.saving})]})}),o.jsxs("label",{children:["Комментарий",o.jsx("textarea",{value:ye.comment,onChange:a=>be(c=>({...c,comment:a.target.value})),placeholder:"Что именно не так в ответе и что нужно исправить.",rows:4,disabled:ye.saving})]}),ye.error?o.jsx("p",{className:"error-text",children:ye.error}):null,o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",onClick:()=>{_l()},disabled:ye.saving,children:ye.saving?"Сохраняю...":"Готово"}),o.jsx("button",{type:"button",className:"tab",onClick:()=>Nn(),disabled:ye.saving,children:"Отмена"})]})]})}):null,ue.open?o.jsx("div",{className:"autoruns-comment-modal-backdrop",onClick:a=>{a.target===a.currentTarget&&Qn()},children:o.jsxs("div",{className:"autoruns-comment-modal",children:[o.jsx("h3",{children:"Комментарий к ответу системы"}),o.jsx("p",{className:"muted",children:"Оцените ответ по 5-балльной шкале и добавьте комментарий по браку."}),ds?o.jsxs(o.Fragment,{children:[o.jsxs("details",{className:"autoruns-prompt-details",open:!0,children:[o.jsx("summary",{children:"Вопрос пользователя"}),o.jsx("p",{className:"autoruns-comment-quote",children:$s?.text??"Вопрос в диалоге не найден."})]}),o.jsxs("details",{className:"autoruns-prompt-details",open:!0,children:[o.jsx("summary",{children:"Ответ системы"}),o.jsx("p",{className:"autoruns-comment-quote",children:ds.text})]})]}):null,o.jsx("div",{className:"autoruns-rating-row",role:"group","aria-label":"Рейтинг ответа",children:[1,2,3,4,5].map(a=>o.jsx("button",{type:"button",className:ue.rating>=a?"autoruns-rating-dot active":"autoruns-rating-dot",onClick:()=>Ue(c=>({...c,rating:a})),disabled:ue.saving,"aria-label":`Оценка ${a}`,children:ue.rating>=a?"●":"○"},a))}),o.jsxs("div",{className:"autoruns-form-grid",children:[o.jsxs("label",{children:["Решение по кейсу",o.jsx("select",{value:ue.manualCaseDecision,onChange:a=>Ue(c=>({...c,manualCaseDecision:a.target.value})),disabled:ue.saving,children:(vn.length>0?vn:Ee?.enum??[dl]).map(a=>o.jsx("option",{value:a,children:String(Ee?.labels?.[a]??a)},a))})]}),o.jsxs("label",{children:["Автор комментария",o.jsx("input",{value:ue.annotationAuthor,onChange:a=>Ue(c=>({...c,annotationAuthor:a.target.value})),placeholder:"manual_reviewer",disabled:ue.saving})]})]}),o.jsxs("label",{children:["Комментарий",o.jsx("textarea",{value:ue.comment,onChange:a=>Ue(c=>({...c,comment:a.target.value})),placeholder:"Почему ответ бракованный, что именно пошло не так, какие технические детали проверить.",rows:4,disabled:ue.saving})]}),ue.error?o.jsx("p",{className:"error-text",children:ue.error}):null,o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",onClick:()=>{Jr()},disabled:ue.saving,children:ue.saving?"Сохраняю...":"Готово"}),o.jsx("button",{type:"button",className:"tab",onClick:()=>Qn(),disabled:ue.saving,children:"Отмена"})]})]})}):null]})}function fm({items:i,onRefresh:p,onOpenTrace:f}){return o.jsx(In,{title:"История нормализаций",subtitle:"Короткий вопрос, confidence, route hint и статус валидации.",actions:o.jsx("button",{type:"button",onClick:()=>p(),children:"Обновить"}),children:o.jsxs("div",{className:"history-list",children:[i.length===0?o.jsx("p",{className:"muted",children:"История пока пустая."}):null,i.map(N=>o.jsxs("button",{type:"button",className:"history-item",onClick:()=>f(N.trace_id),children:[o.jsxs("div",{className:"history-row",children:[o.jsx("strong",{children:N.route_hint??"route: n/a"}),o.jsx("span",{children:N.validation_passed?"schema: ok":"schema: fail"})]}),o.jsx("p",{children:N.question_short}),o.jsxs("div",{className:"history-row",children:[o.jsx("span",{children:N.model}),o.jsx("span",{children:new Date(N.timestamp).toLocaleString("ru-RU")})]})]},N.trace_id))]})})}function yr(i){return i==null||i===""?"—":String(i)}function mm({result:i}){return o.jsx(In,{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:yr(i?.trace_id)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"request_started_at"}),o.jsx("strong",{children:yr(i?new Date(Date.now()-i.latency_ms).toISOString():null)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"request_finished_at"}),o.jsx("strong",{children:yr(i?new Date().toISOString():null)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"latency_ms"}),o.jsx("strong",{children:yr(i?.latency_ms)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"input_tokens"}),o.jsx("strong",{children:yr(i?.usage?.input_tokens)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"output_tokens"}),o.jsx("strong",{children:yr(i?.usage?.output_tokens)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"total_tokens"}),o.jsx("strong",{children:yr(i?.usage?.total_tokens)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"validation_status"}),o.jsx("strong",{children:i?.validation?.passed?"passed":"failed"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"prompt_version"}),o.jsx("strong",{children:yr(i?.prompt_version)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"schema_version"}),o.jsx("strong",{children:yr(i?.schema_version)})]})]})})}const pm={normalized:"Normalized JSON",fragments:"Fragment View",scope:"Scope View",flags:"Flags View",route:"Route Simulation",raw:"Raw model output",validation:"Validation",logs:"Logs"};function hm(i){return i&&typeof i=="object"?i:null}function gm({tab:i,onTabChange:p,result:f,appLogs:N}){const j=["normalized","fragments","scope","flags","route","raw","validation","logs"],T=hm(f?.normalized),$=String(T?.schema_version??""),Y=$==="normalized_query_v2"||$==="normalized_query_v2_0_1"||$==="normalized_query_v2_0_2",b=Y?{fragments:T?.fragments??[],discarded_fragments:T?.discarded_fragments??[]}:{note:"Fragment View доступен для normalized_query_v2."},Q=Y?{message_in_scope:T?.message_in_scope??null,scope_confidence:T?.scope_confidence??null,contains_multiple_tasks:T?.contains_multiple_tasks??null,global_notes:T?.global_notes??null}:{note:"Scope View доступен для normalized_query_v2."},G=Y?Array.isArray(T?.fragments)?(T?.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(In,{title:"Выходные данные",subtitle:"Structured output и диагностические вкладки.",children:[o.jsx("div",{className:"tab-row",children:j.map(z=>o.jsx("button",{type:"button",className:i===z?"tab active":"tab",onClick:()=>p(z),children:pm[z]},z))}),i==="normalized"?o.jsx(tn,{value:f?.normalized??{note:"Нет данных."}}):null,i==="fragments"?o.jsx(tn,{value:b}):null,i==="scope"?o.jsx(tn,{value:Q}):null,i==="flags"?o.jsx(tn,{value:G}):null,i==="route"?o.jsx(tn,{value:f?.route_hint_summary??{note:"Нет данных."}}):null,i==="raw"?o.jsx(tn,{value:f?.raw_model_output??{note:"Нет данных."}}):null,i==="validation"?o.jsx(tn,{value:f?.validation??{note:"Нет данных."}}):null,i==="logs"?o.jsx(tn,{value:N}):null]})}function vm({value:i,onChange:p,onApplyBatchFormat:f,onNormalize:N,busy:j,useMock:T,onUseMockChange:$,errorMessage:Y}){return o.jsxs(In,{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:i.userQuestion,onChange:b=>p({...i,userQuestion:b.target.value}),rows:6,placeholder:"Например: По каким покупателям у нас на конец июня висят отгрузки без оплаты..."})]}),o.jsxs("label",{className:"full-width",children:["Batch queries (`;` separator)",o.jsx("textarea",{value:i.batchQuestionsRaw,onChange:b=>p({...i,batchQuestionsRaw:b.target.value}),onBlur:()=>f(),rows:8,placeholder:"Вопрос 1; Вопрос 2; Вопрос 3"})]}),o.jsxs("label",{children:["Optional period context",o.jsx("input",{value:i.periodHint,onChange:b=>p({...i,periodHint:b.target.value})})]}),o.jsxs("label",{children:["Optional business context",o.jsx("input",{value:i.businessContext,onChange:b=>p({...i,businessContext:b.target.value})})]}),o.jsxs("label",{children:["Optional expected route (eval)",o.jsx("input",{value:i.expectedRoute,onChange:b=>p({...i,expectedRoute:b.target.value})})]})]}),o.jsxs("div",{className:"button-row",children:[o.jsxs("label",{className:"checkbox-row",children:[o.jsx("input",{type:"checkbox",checked:T,onChange:b=>$(b.target.checked)}),"Mock-режим (без вызова OpenAI)"]}),o.jsx("button",{type:"button",onClick:()=>f(),disabled:j||!i.batchQuestionsRaw.trim(),children:"Применить `;` в переносы"}),o.jsx("button",{type:"button",onClick:()=>N(!1),disabled:j||!i.userQuestion.trim(),children:j?"Нормализуем...":"Normalize"}),o.jsx("button",{type:"button",onClick:()=>N(!0),disabled:j||!i.userQuestion.trim(),children:j?"Сохраняем...":"Normalize + Save as test case"})]}),Y?o.jsx("p",{className:"error-text",children:Y}):null]})}function ym({runs:i,selectedRunId:p,onSelectRun:f,onStartRun:N,onFinishRun:j,onRefreshRuns:T,onRunEval:$,onCopyEvalReport:Y,evalBusy:b,traceItems:Q,evalReport:G}){return o.jsxs(In,{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:()=>N(),children:"Запустить run"}),o.jsx("button",{type:"button",onClick:()=>j(),disabled:!p,children:"Завершить выбранный run"}),o.jsx("button",{type:"button",onClick:()=>T(),children:"Обновить runs"}),o.jsx("button",{type:"button",onClick:()=>$(),disabled:b,children:b?"Идет eval v2.0.2...":"Запустить eval v2.0.2"})]}),o.jsxs("div",{className:"runtime-stack",children:[o.jsxs("div",{className:"runtime-runs",children:[i.map(z=>o.jsxs("button",{type:"button",className:p===z.runId?"history-item selected":"history-item",onClick:()=>f(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)),i.length===0?o.jsx("p",{className:"muted",children:"Нет активных запусков."}):null]}),o.jsxs("div",{className:"runtime-details",children:[o.jsx("h3",{children:"Trace выбранного run"}),o.jsx(tn,{value:Q}),o.jsxs("div",{className:"eval-report-wrap",children:[o.jsx("h3",{style:{marginTop:12},children:"Отчет eval"}),o.jsx(tn,{value:G??{note:"Eval пока не запускался"}}),o.jsx("button",{type:"button",className:"copy-cube-button",title:"Скопировать отчет eval",onClick:()=>Y(),children:"⧉"})]})]})]})]})}const xm={llmProvider:"openai",apiKey:"",model:"gpt-4o-mini",baseUrl:"https://api.openai.com/v1",temperature:0,maxOutputTokens:700},Uc={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."},_m={userQuestion:"",batchQuestionsRaw:"",periodHint:"",businessContext:"",expectedRoute:""},ta={colors:{backgroundRgb:"18, 18, 18",mainSurfaceRgb:"25, 25, 25",horizontalSurfaceRgb:"30, 30, 30",focusSurfaceRgb:"35, 35, 35",assistantChipRgb:"18, 18, 18",assistantChipHoverRgb:"44, 44, 44",assistantChipSelectedRgb:"167, 59, 255",assistantChipSelectedTextRgb:"240, 240, 240",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"},layout:{modeColumnWidthPx:406,modeToggleWidthPx:188}},bc="ndc_normalizer_session_config_v1",Bc="ndc_autoruns_layout_config_v1",Sm="ndc-autoruns-save",na=["Анализ запроса","Получение данных","Подготовка ответа"],wm="autoruns",ra="normalizer_v2_0_2",Hc="address_query_runtime_v1",jm=["normalized","fragments","scope","flags","route","raw","validation","logs"],sa="manual_reviewer";function km(i){return`[${new Date().toLocaleTimeString("ru-RU")}] ${i}`}function Cm(i,p){if(!p)return"Previous preset is not selected.";const N=["systemPrompt","developerPrompt","domainPrompt","schemaNotes","fewShotExamples"].filter(j=>i[j]!==p[j]).map(j=>`${j}: ${Math.abs(i[j].length-p[j].length)} chars delta`);return N.length===0?"No changes against previous preset.":`Changed fields: ${N.length}. ${N.join(" | ")}`}function Nm(i,p){const f=i.trim();if(!f)return"";if(!p)return f;const N=f.toLowerCase(),j=p.anchor_text.trim(),T=j.toLowerCase();return T&&N.includes(T)?f:`По выбранному объекту "${j}": ${f}`}function Em(){const[i,p]=x.useState(xm),[f,N]=x.useState(Uc),[j,T]=x.useState(_m),[$,Y]=x.useState(null),[b,Q]=x.useState([]),[G,z]=x.useState([]),[ne,Ne]=x.useState("normalized"),[le,ce]=x.useState(!1),[me,Qe]=x.useState(!1),[De,We]=x.useState([]),[je,W]=x.useState(""),[de,Me]=x.useState([]),[Ze,ct]=x.useState(""),[st,Et]=x.useState("NDC custom preset"),[ie,Le]=x.useState(null),[H,Fe]=x.useState(""),[Te,Ae]=x.useState(!1),[I,X]=x.useState([]),[O,y]=x.useState(""),[C,J]=x.useState([]),[oe,pe]=x.useState(!1),[ae,Se]=x.useState(null),[ge,ve]=x.useState(""),[Ee,gn]=x.useState(wm),[vn,Yn]=x.useState(!0),[Ln,Xn]=x.useState(!0),[Pe,qt]=x.useState(!0),[ot,dt]=x.useState(!0),[lt,yn]=x.useState(!0),[re,It]=x.useState(!0),[tt,Vr]=x.useState(!0),[nn,xn]=x.useState(!0),[xt,_n]=x.useState(!0),[Oe,rn]=x.useState(!0),[On,xr]=x.useState(!0),[Dn,Qr]=x.useState(!0),[Sn,sn]=x.useState(!0),[Lt,Zn]=x.useState(!0),[Kt,on]=x.useState(!0),[Ot,_r]=x.useState(!0),[zn,Sr]=x.useState(!0),[$n,wr]=x.useState(!0),[ze,er]=x.useState(""),[Dt,Fn]=x.useState([]),[wn,Ge]=x.useState(""),[nt,ln]=x.useState(null),[zt,Un]=x.useState(!1),[bn,Gt]=x.useState(""),[jr,_t]=x.useState(""),[jn,an]=x.useState([]),[tr,Wr]=x.useState(!1),[Re,Pt]=x.useState({open:!1,messageIndex:-1,rating:3,comment:"",annotationAuthor:sa,saving:!1,error:""}),nr=x.useRef(!1),Bn=x.useRef(!1),cs=x.useRef(!1);x.useEffect(()=>{const m=document.documentElement,{colors:E}=ta;m.style.setProperty("--rgb-background",E.backgroundRgb),m.style.setProperty("--rgb-surface-main",E.mainSurfaceRgb),m.style.setProperty("--rgb-surface-horizontal",E.horizontalSurfaceRgb),m.style.setProperty("--rgb-surface-focus",E.focusSurfaceRgb),m.style.setProperty("--rgb-assistant-chip",E.assistantChipRgb),m.style.setProperty("--rgb-assistant-chip-hover",E.assistantChipHoverRgb),m.style.setProperty("--rgb-assistant-chip-selected",E.assistantChipSelectedRgb),m.style.setProperty("--rgb-assistant-chip-selected-text",E.assistantChipSelectedTextRgb),m.style.setProperty("--rgb-active",E.activeRgb),m.style.setProperty("--rgb-active-text",E.activeTextRgb),m.style.setProperty("--rgb-text-main",E.textMainRgb),m.style.setProperty("--rgb-text-muted",E.textMutedRgb),m.style.setProperty("--rgb-danger",E.dangerRgb),m.style.setProperty("--rgb-scrollbar-track",E.scrollbarTrackRgb),m.style.setProperty("--rgb-scrollbar-thumb",E.scrollbarThumbRgb),m.style.setProperty("--rgb-scrollbar-thumb-hover",E.scrollbarThumbHoverRgb),m.style.setProperty("--mode-column-width",`${ta.layout.modeColumnWidthPx}px`),m.style.setProperty("--mode-toggle-width",`${ta.layout.modeToggleWidthPx}px`)},[]);const D=m=>{z(E=>[km(m),...E].slice(0,300))};function zs(){let m=0;Gt(na[0]);const E=window.setInterval(()=>{m=Math.min(m+1,na.length-1),Gt(na[m])},650);return()=>window.clearInterval(E)}x.useEffect(()=>{(async()=>{const k=localStorage.getItem(bc);if(k)try{const te=JSON.parse(k);p(Ie=>({...Ie,llmProvider:te.llmProvider==="local"?"local":"openai",model:te.model??Ie.model,baseUrl:te.baseUrl??Ie.baseUrl,temperature:te.temperature??Ie.temperature,maxOutputTokens:te.maxOutputTokens??Ie.maxOutputTokens}))}catch{}try{const te=await _e.loadSharedConnectionConfig();te.connection&&te.connection.llmProvider==="local"&&(p(Ie=>({...Ie,llmProvider:"local",model:te.connection?.model??Ie.model,baseUrl:te.connection?.baseUrl??Ie.baseUrl,temperature:te.connection?.temperature??Ie.temperature,maxOutputTokens:te.connection?.maxOutputTokens??Ie.maxOutputTokens})),D(`Shared local LLM config loaded: ${te.connection.model}`))}catch(te){D(`Shared local config load error: ${te instanceof Error?te.message:String(te)}`)}finally{cs.current=!0}})();const E=localStorage.getItem(Bc);if(E)try{const k=JSON.parse(E);k.uiMode==="decomposition"?gn("decomposition"):(k.uiMode==="assistant"||k.uiMode==="autoruns")&&gn("autoruns"),k.activeTab&&jm.includes(k.activeTab)&&Ne(k.activeTab),typeof k.showAutorunsSettingsMode=="boolean"&&Yn(k.showAutorunsSettingsMode),typeof k.showAutorunsAutoRunsMode=="boolean"&&Xn(k.showAutorunsAutoRunsMode),typeof k.showAutorunsAssistantMode=="boolean"&&qt(k.showAutorunsAssistantMode),typeof k.showAutorunsDecompositionMode=="boolean"&&dt(k.showAutorunsDecompositionMode),typeof k.showAutorunsProgressMode=="boolean"&&yn(k.showAutorunsProgressMode),typeof k.showAutorunsCommentsMode=="boolean"&&It(k.showAutorunsCommentsMode),typeof k.showAssistantConnectionMode=="boolean"&&Vr(k.showAssistantConnectionMode),typeof k.showAssistantPromptMode=="boolean"&&xn(k.showAssistantPromptMode),typeof k.showAssistantChatMode=="boolean"&&_n(k.showAssistantChatMode),typeof k.showAssistantCommentsMode=="boolean"&&rn(k.showAssistantCommentsMode),typeof k.showAssistantSamMode=="boolean"&&xr(k.showAssistantSamMode),typeof k.showDecompositionConnectionMode=="boolean"&&Qr(k.showDecompositionConnectionMode),typeof k.showDecompositionPromptMode=="boolean"&&sn(k.showDecompositionPromptMode),typeof k.showDecompositionQueryMode=="boolean"&&Zn(k.showDecompositionQueryMode),typeof k.showDecompositionOutputMode=="boolean"&&on(k.showDecompositionOutputMode),typeof k.showDecompositionMetricsMode=="boolean"&&_r(k.showDecompositionMetricsMode),typeof k.showDecompositionHistoryMode=="boolean"&&Sr(k.showDecompositionHistoryMode),typeof k.showDecompositionRuntimeMode=="boolean"&&wr(k.showDecompositionRuntimeMode),k.prompts&&(N(te=>({...te,...k.prompts})),Bn.current=!0)}catch{}Mt(),qr(),St()},[]),x.useEffect(()=>{if(!cs.current||i.llmProvider!=="local")return;const m=window.setTimeout(()=>{_e.saveSharedConnectionConfig(i).catch(E=>D(`Shared local config sync error: ${E instanceof Error?E.message:String(E)}`))},250);return()=>window.clearTimeout(m)},[i.baseUrl,i.llmProvider,i.maxOutputTokens,i.model,i.temperature]);async function Mt(){try{const m=await _e.loadHistory();Q(m.items??[])}catch(m){D(`History load error: ${m instanceof Error?m.message:String(m)}`)}}async function qr(){try{const E=(await _e.loadPresets()).presets??[];if(Me(E),Bn.current){nr.current=!0;return}if(nr.current)return;const k=E.find(te=>te.prompt_version===ra)??E.find(te=>te.id==="default-normalizer-v2_0_2");if(!k){nr.current=!0,D(`Preset autoload skipped: ${ra} not found.`);return}ct(k.id),Le(f),N({systemPrompt:k.systemPrompt,developerPrompt:k.developerPrompt,domainPrompt:k.domainPrompt,schemaNotes:k.schemaNotes??"",fewShotExamples:k.fewShotExamples??""}),nr.current=!0,D(`Preset autoloaded: ${k.name} (${k.prompt_version}).`)}catch(m){D(`Presets load error: ${m instanceof Error?m.message:String(m)}`)}}async function St(){try{const m=await _e.listRuns();X(m.items??[])}catch(m){D(`Runs load error: ${m instanceof Error?m.message:String(m)}`)}}function rr(){if(localStorage.setItem(bc,JSON.stringify({model:i.model,llmProvider:i.llmProvider,baseUrl:i.baseUrl,temperature:i.temperature,maxOutputTokens:i.maxOutputTokens})),i.llmProvider==="local"){_e.saveSharedConnectionConfig(i).then(()=>{D("Local config saved and synced to shared agent config (without API key).")}).catch(m=>{D(`Local config saved, but shared sync failed: ${m instanceof Error?m.message:String(m)}`)});return}D("Local config saved (without API key).")}function Kr(){localStorage.setItem(Bc,JSON.stringify({uiMode:Ee,activeTab:ne,showAutorunsSettingsMode:vn,showAutorunsAutoRunsMode:Ln,showAutorunsAssistantMode:Pe,showAutorunsDecompositionMode:ot,showAutorunsProgressMode:lt,showAutorunsCommentsMode:re,showAssistantConnectionMode:tt,showAssistantPromptMode:nn,showAssistantChatMode:xt,showAssistantCommentsMode:Oe,showAssistantSamMode:On,showDecompositionConnectionMode:Dn,showDecompositionPromptMode:Sn,showDecompositionQueryMode:Lt,showDecompositionOutputMode:Kt,showDecompositionMetricsMode:Ot,showDecompositionHistoryMode:zn,showDecompositionRuntimeMode:$n,prompts:f})),window.dispatchEvent(new CustomEvent(Sm)),D("UI layout and prompts saved.")}async function ue(){ce(!0),ve("");try{const m=await _e.testConnection(i);m.provider==="local"?m.model_found===!0?(W(`LOCAL OK - ${m.model}`),D(`Local model is available: ${m.model} (catalog size=${m.models_count??"n/a"}).`)):m.model_found===!1?(W(`LOCAL OK, model not loaded - ${m.model}`),D(`Local server is reachable, but model '${m.model}' is not in loaded catalog. Use 'Load model list' and select one of loaded models.`)):(W(`LOCAL OK (model list unavailable) - ${m.model}`),D("Local server is reachable, but model catalog could not be verified.")):(W(`OPENAI OK - ${m.model}`),D(`OpenAI connection ok: ${m.model}`))}catch(m){const E=m instanceof Error?m.message:String(m);W("Connection error"),ve(`Test connection: ${E}`),D(`Test connection error: ${E}`)}finally{ce(!1)}}async function Ue(){Qe(!0);try{const E=(await _e.listModels(i)).models??[];We(E),E.length>0&&p(k=>k.model&&E.includes(k.model)?k:{...k,model:E[0]}),D(`Model catalog loaded (${i.llmProvider}): ${E.length} items.`)}catch(m){const E=m instanceof Error?m.message:String(m);D(`Load model list error: ${E}`)}finally{Qe(!1)}}x.useEffect(()=>{We([])},[i.llmProvider,i.baseUrl]);async function ye(m){ce(!0),ve("");try{const E=await _e.normalize({connection:i,prompts:f,promptVersion:"normalizer_v2_0_2",query:{userQuestion:j.userQuestion,periodHint:j.periodHint,businessContext:j.businessContext,expectedRoute:j.expectedRoute},saveAsTestCase:m,useMock:Te});Y(E),Ne("normalized"),D(`Normalize done: trace=${E.trace_id}, validation=${E.validation.passed?"passed":"failed"}`),Mt()}catch(E){const k=E instanceof Error?E.message:String(E);ve(`Normalize: ${k}`),D(`Normalize error: ${k}`)}finally{ce(!1)}}function be(){const m=de.find(E=>E.id===Ze);if(!m){D("Preset is not selected.");return}Le(f),N({systemPrompt:m.systemPrompt,developerPrompt:m.developerPrompt,domainPrompt:m.domainPrompt,schemaNotes:m.schemaNotes??"",fewShotExamples:m.fewShotExamples??""}),D(`Preset loaded: ${m.name}`)}async function sr(){try{await _e.savePreset({name:st||"NDC preset",prompt_version:"normalizer_v2_0_2",systemPrompt:f.systemPrompt,developerPrompt:f.developerPrompt,domainPrompt:f.domainPrompt,schemaNotes:f.schemaNotes,fewShotExamples:f.fewShotExamples}),D("Preset saved."),await qr()}catch(m){D(`Preset save error: ${m instanceof Error?m.message:String(m)}`)}}function Hn(){N(Uc),D("Prompt panel reset to defaults.")}function Vn(){const m=Cm(f,ie);Fe(m),D(m)}function $t(){const m=j.batchQuestionsRaw.split(";").map(E=>E.trim()).filter(Boolean).join(` + +`);m&&(T(E=>({...E,batchQuestionsRaw:m})),D("Batch field formatted: `;` converted to blank-line separators."))}async function kn(m){try{const k=(await _e.loadTrace(m)).trace,te=k.parsed_normalized_json??null;Y({trace_id:String(k.trace_id??m),ok:!!k.validation_result?.passed,normalized:te,route_hint_summary:k.route_hint_summary??(te?{route_hint:te.route_hint??null,confidence:te.confidence?.route_hint??null}:null),raw_model_output:k.raw_model_response??{},validation:k.validation_result??{passed:!1,errors:["validation not found"]},usage:k.usage??{input_tokens:0,output_tokens:0,total_tokens:0},latency_ms:Number(k.latency_ms??0),prompt_version:String(k.prompt_version??"unknown"),schema_version:String(k.schema_version??"unknown")}),Ne("raw"),ve(""),D(`Trace opened: ${m}`)}catch(E){const k=E instanceof Error?E.message:String(E);ve(`Trace: ${k}`),D(`Trace open error ${m}: ${k}`)}}async function At(){try{const m=await _e.startRun();y(m.run.runId),D(`Run started: ${m.run.runId}`),D("Tip: start run does not execute normalize by itself. Use 'Run eval v2.0.2' button."),await St()}catch(m){D(`Run start error: ${m instanceof Error?m.message:String(m)}`)}}async function ke(){if(O)try{await _e.finishRun(O),D(`Run finished: ${O}`),await St()}catch(m){D(`Run finish error: ${m instanceof Error?m.message:String(m)}`)}}async function ds(){pe(!0),ve("");try{D("Starting eval in v2 contour.");const m=j.batchQuestionsRaw.trim()||j.userQuestion.trim();if(!m)throw new Error("Fill batch field or Raw user question first.");const E=await _e.runEval({connection:i,prompts:f,promptVersion:"normalizer_v2_0_2",mode:"single-pass-strict",rawQuestions:m,useMock:Te});Se(E.report),D("Eval v2.0.2 run finished.");const k=E.report;if(k.run_id&&D(`Eval run id: ${k.run_id}`),k.metrics){const te=k.metrics;D(`Eval metrics v2.0.2: schema=${te.schema_validation_pass_rate??"n/a"}%, route_accuracy=${te.route_resolution_accuracy??"n/a"}%, no_route_precision=${te.no_route_precision??"n/a"}%, state_consistency=${te.execution_state_consistency_rate??"n/a"}%`)}await Mt()}catch(m){const E=m instanceof Error?m.message:String(m);E.includes("Legacy eval runner supports normalized_query_v1 only")?(Se({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"]}),D("Backend is legacy-only for eval right now. Showing v2 pilot plan.")):(ve(`Eval: ${E}`),D(`Eval run error: ${E}`))}finally{pe(!1)}}async function $s(){try{const m=JSON.stringify(ae??{},null,2);await navigator.clipboard.writeText(m),D("Eval report copied to clipboard.")}catch(m){D(`Eval report copy error: ${m instanceof Error?m.message:String(m)}`)}}const Cn=x.useMemo(()=>{const m=new Map;for(const E of jn)E.message_id&&m.set(E.message_id,E);return m},[jn]),kr=Re.messageIndex>=0?Dt[Re.messageIndex]??null:null,or=x.useMemo(()=>{if(Re.messageIndex<0)return null;for(let m=Re.messageIndex-1;m>=0;m-=1){const E=Dt[m];if(E?.role==="user")return E}return null},[Re.messageIndex,Dt]);async function it(m){if(!m.trim()){an([]);return}Wr(!0);try{const E=await _e.loadAssistantAnnotations({session_id:m,limit:400});an(E.items??[])}catch(E){const k=E instanceof Error?E.message:String(E);D(`Assistant annotations load error: ${k}`)}finally{Wr(!1)}}function lr(m){Pt(E=>E.saving&&!m?.force?E:{open:!1,messageIndex:-1,rating:3,comment:"",annotationAuthor:sa,saving:!1,error:""})}function ir(m,E){if(m.role!=="assistant")return;const k=ze.trim(),te=String(m.session_id??"").trim();if(!(k||te)){_t("Сначала получите ответ ассистента в активной сессии.");return}!k&&te&&er(te);const gt=Cn.get(m.message_id)??null;Pt({open:!0,messageIndex:E,rating:gt?.rating??3,comment:gt?.comment??"",annotationAuthor:gt?.annotation_author??sa,saving:!1,error:""})}function F(m){return m.role==="assistant"}function ar(m){return m.role==="assistant"&&Cn.has(m.message_id)}async function Nn(){if(!ze.trim()){Pt(m=>({...m,error:"Сессия ассистента не найдена."}));return}if(!(Re.messageIndex<0)){if(!Re.comment.trim()){Pt(m=>({...m,error:"Добавьте комментарий."}));return}Pt(m=>({...m,saving:!0,error:""}));try{const m=await _e.saveAssistantAnnotation({session_id:ze,message_index:Re.messageIndex,rating:Re.rating,comment:Re.comment.trim(),annotation_author:Re.annotationAuthor.trim()||void 0});an(E=>{const k=[...E],te=k.findIndex(Ie=>Ie.annotation_id===m.annotation.annotation_id);return te>=0?k[te]=m.annotation:k.unshift(m.annotation),k.sort((Ie,gt)=>Date.parse(gt.updated_at)-Date.parse(Ie.updated_at))}),lr({force:!0})}catch(m){const E=m instanceof Error?m.message:String(m);Pt(k=>({...k,saving:!1,error:E}))}}}function Cr(){er(""),Fn([]),Ge(""),ln(null),Gt(""),_t(""),an([]),lr({force:!0}),D("Assistant session reset.")}async function fs(){const m=Nm(wn,nt);if(!m)return;Un(!0),_t(""),Ge(""),Fn(k=>[...k,{message_id:`local-${Date.now()}`,session_id:ze||"pending",role:"user",text:m,reply_type:null,created_at:new Date().toISOString(),trace_id:null,debug:null}]);const E=zs();try{const k=await _e.sendAssistantMessage({connection:i,prompts:f,userMessage:m,sessionId:ze||void 0,promptVersion:Hc,useMock:Te});er(k.session_id),Fn(k.conversation),Gt("Ответ готов"),await it(k.session_id),D(`Assistant reply received: trace=${k.debug.trace_id}`)}catch(k){const te=k instanceof Error?k.message:String(k);_t(te),Gt("Ошибка ассистента"),D(`Assistant error: ${te}`)}finally{E(),Un(!1)}}return x.useEffect(()=>{if(!ze.trim()){an([]);return}it(ze)},[ze]),x.useEffect(()=>{if(!O){J([]);return}_e.runTrace(O).then(m=>J(m.items)).catch(m=>D(`Run trace error: ${m instanceof Error?m.message:String(m)}`))},[O]),o.jsxs("main",{className:`app-root ${Ee==="assistant"||Ee==="decomposition"||Ee==="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:Ee==="autoruns"?"tab active":"tab",onClick:()=>gn("autoruns"),children:"Управление ассистентом"}),o.jsx("button",{type:"button",className:Ee==="decomposition"?"tab active":"tab",onClick:()=>gn("decomposition"),children:"Декомпозиция"}),o.jsx("button",{type:"button",className:"tab",onClick:Kr,children:"Сохранить"})]}),Ee==="assistant"?o.jsxs("div",{className:"mode-switch-row mode-switch-row-right",children:[o.jsx("button",{type:"button",className:tt?"tab active":"tab",onClick:()=>Vr(m=>!m),children:"LLM Connector"}),o.jsx("button",{type:"button",className:nn?"tab active":"tab",onClick:()=>xn(m=>!m),children:"Prompt Manager"}),o.jsx("button",{type:"button",className:xt?"tab active":"tab",onClick:()=>_n(m=>!m),children:"Режим ассистента"}),o.jsx("button",{type:"button",className:Oe?"tab active":"tab",onClick:()=>rn(m=>!m),children:"Комментарии ассистента"}),o.jsx("button",{type:"button",className:On?"tab active":"tab",onClick:()=>xr(m=>!m),children:"SAM"})]}):Ee==="decomposition"?o.jsxs("div",{className:"mode-switch-row mode-switch-row-right",children:[o.jsx("button",{type:"button",className:Dn?"tab active":"tab",onClick:()=>Qr(m=>!m),children:"LLM"}),o.jsx("button",{type:"button",className:Sn?"tab active":"tab",onClick:()=>sn(m=>!m),children:"Prompt"}),o.jsx("button",{type:"button",className:Lt?"tab active":"tab",onClick:()=>Zn(m=>!m),children:"Запрос"}),o.jsx("button",{type:"button",className:Kt?"tab active":"tab",onClick:()=>on(m=>!m),children:"Выход"}),o.jsx("button",{type:"button",className:Ot?"tab active":"tab",onClick:()=>_r(m=>!m),children:"Метрики"}),o.jsx("button",{type:"button",className:zn?"tab active":"tab",onClick:()=>Sr(m=>!m),children:"История"}),o.jsx("button",{type:"button",className:$n?"tab active":"tab",onClick:()=>wr(m=>!m),children:"NDC Run Monitor"})]}):Ee==="autoruns"?o.jsxs("div",{className:"mode-switch-row mode-switch-row-right",children:[o.jsx("button",{type:"button",className:vn?"tab active":"tab",onClick:()=>Yn(m=>!m),children:"Настройки"}),o.jsx("button",{type:"button",className:Ln?"tab active":"tab",onClick:()=>Xn(m=>!m),children:"Автопрогоны"}),o.jsx("button",{type:"button",className:Pe?"tab active":"tab",onClick:()=>qt(m=>!m),children:"Режим ассистента"}),o.jsx("button",{type:"button",className:ot?"tab active":"tab",onClick:()=>dt(m=>!m),children:"Режим декомпозиции"}),o.jsx("button",{type:"button",className:lt?"tab active":"tab",onClick:()=>yn(m=>!m),children:"Прогресс/регресс"}),o.jsx("button",{type:"button",className:re?"tab active":"tab",onClick:()=>It(m=>!m),children:"Комментарии"})]}):null]}),Ee==="assistant"?o.jsx("div",{className:"layout-grid layout-grid-mode-columns",children:o.jsxs("div",{className:"mode-columns",children:[tt?o.jsx("div",{className:"mode-col",children:o.jsx(oa,{value:i,modelOptions:De,modelsBusy:me,onChange:p,onReloadModels:Ue,onSaveLocalConfig:rr,onTestConnection:ue,lastStatus:je,busy:le||zt})}):null,nn?o.jsx("div",{className:"mode-col mode-col-wide",children:o.jsx(la,{value:f,onChange:N,presets:de,selectedPresetId:Ze,onSelectPreset:ct,onLoadPreset:be,onSavePreset:sr,onResetDefaults:Hn,onDiffPrevious:Vn,presetName:st,onPresetNameChange:Et,diffSummary:H})}):null,xt?o.jsx("div",{className:"mode-col mode-col-xwide",children:o.jsx(Wc,{sessionId:ze,conversation:Dt,inputValue:wn,onInputChange:Ge,selectedContextChip:nt,onSelectContextChip:ln,onClearContextChip:()=>ln(null),useMock:Te,onUseMockChange:Ae,onSend:fs,onClear:Cr,busy:zt,statusText:bn,errorMessage:jr,showCommentAction:!0,onCommentAssistantMessage:ir,isAssistantMessageCommented:ar,canCommentAssistantMessage:F})}):null,Oe?o.jsx("div",{className:"mode-col",children:o.jsx(In,{className:"assistant-comments-frame",title:"Комментарии ассистента",children:o.jsxs("div",{className:"assistant-comments-shell",children:[o.jsxs("div",{className:"assistant-comments-toolbar",children:[o.jsx("span",{className:"muted",children:ze?`session: ${ze}`:"Сессия не запущена"}),o.jsx("button",{type:"button",className:"tab",onClick:()=>{it(ze)},disabled:!ze||tr,children:tr?"Обновляю...":"Обновить"})]}),o.jsxs("div",{className:"assistant-comments-list",children:[ze?null:o.jsx("p",{className:"muted",children:"Появится после первого ответа ассистента."}),ze&&jn.length===0&&!tr?o.jsx("p",{className:"muted",children:"Комментариев по этой сессии пока нет."}):null,jn.map(m=>o.jsxs("article",{className:"assistant-comment-item",children:[o.jsxs("div",{className:"assistant-comment-head",children:[o.jsx("strong",{children:`${"●".repeat(Math.max(1,Math.min(5,Math.round(m.rating))))}${"○".repeat(Math.max(0,5-Math.round(m.rating)))}`}),o.jsx("span",{children:new Date(m.updated_at).toLocaleString("ru-RU")})]}),m.context.question_text?o.jsxs("p",{children:["Q: ",m.context.question_text]}):null,m.context.answer_text?o.jsxs("p",{children:["A: ",m.context.answer_text]}):null,o.jsx("p",{children:m.comment}),o.jsxs("div",{className:"assistant-comment-meta",children:[m.context.trace_id?o.jsx("span",{children:`trace=${m.context.trace_id}`}):null,m.context.reply_type?o.jsx("span",{children:`reply_type=${m.context.reply_type}`}):null]})]},m.annotation_id))]})]})})}):null,On?o.jsx("div",{className:"mode-col",children:o.jsx(Of,{sessionId:ze,conversation:Dt,statusText:bn,errorMessage:jr,useMock:Te,appLogs:G})}):null,!tt&&!nn&&!xt&&!Oe&&!On?o.jsx("div",{className:"mode-columns-empty",children:"Все панели режима ассистента скрыты. Включите нужные блоки справа в шапке."}):null]})}):Ee==="decomposition"?o.jsx("div",{className:"layout-grid layout-grid-mode-columns",children:o.jsxs("div",{className:"mode-columns",children:[Dn?o.jsx("div",{className:"mode-col",children:o.jsx(oa,{value:i,modelOptions:De,modelsBusy:me,onChange:p,onReloadModels:Ue,onSaveLocalConfig:rr,onTestConnection:ue,lastStatus:je,busy:le})}):null,Sn?o.jsx("div",{className:"mode-col mode-col-wide",children:o.jsx(la,{value:f,onChange:N,presets:de,selectedPresetId:Ze,onSelectPreset:ct,onLoadPreset:be,onSavePreset:sr,onResetDefaults:Hn,onDiffPrevious:Vn,presetName:st,onPresetNameChange:Et,diffSummary:H})}):null,Lt?o.jsx("div",{className:"mode-col",children:o.jsx(vm,{value:j,onChange:T,onApplyBatchFormat:$t,onNormalize:ye,busy:le,useMock:Te,onUseMockChange:Ae,errorMessage:ge})}):null,Kt?o.jsx("div",{className:"mode-col mode-col-xwide",children:o.jsx(gm,{tab:ne,onTabChange:Ne,result:$,appLogs:G})}):null,Ot?o.jsx("div",{className:"mode-col",children:o.jsx(mm,{result:$})}):null,zn?o.jsx("div",{className:"mode-col",children:o.jsx(fm,{items:b,onRefresh:Mt,onOpenTrace:kn})}):null,$n?o.jsx("div",{className:"mode-col mode-col-xwide",children:o.jsx(ym,{runs:I,selectedRunId:O,onSelectRun:y,onStartRun:At,onFinishRun:ke,onRefreshRuns:St,onRunEval:ds,onCopyEvalReport:$s,evalBusy:oe,traceItems:C,evalReport:ae})}):null,!Dn&&!Sn&&!Lt&&!Kt&&!Ot&&!zn&&!$n?o.jsx("div",{className:"mode-columns-empty",children:"Все панели режима декомпозиции скрыты. Включите нужные блоки справа в шапке."}):null]})}):o.jsx("div",{className:"layout-grid layout-grid-autoruns",children:o.jsx(dm,{connection:i,modelOptions:De,modelsBusy:me,connectionStatus:je,connectionBusy:le,onConnectionChange:p,onReloadModels:Ue,onSaveLocalConfig:rr,onTestConnection:ue,prompts:f,onPromptsChange:N,promptPresets:de,selectedPresetId:Ze,onSelectPreset:ct,onLoadPreset:be,onSavePreset:sr,onResetDefaults:Hn,onDiffPrevious:Vn,presetName:st,onPresetNameChange:Et,diffSummary:H,assistantPromptVersion:Hc,decompositionPromptVersion:ra,showSettingsMode:vn,showAutoRunsMode:Ln,showAssistantMode:Pe,showDecompositionMode:ot,showProgressMode:lt,showCommentsMode:re,onLog:D})}),Re.open?o.jsx("div",{className:"autoruns-comment-modal-backdrop",onClick:m=>{m.target===m.currentTarget&&lr()},children:o.jsxs("div",{className:"autoruns-comment-modal",children:[o.jsx("h3",{children:"Комментарий к ответу ассистента"}),o.jsx("p",{className:"muted",children:"Эта разметка хранится отдельно от комментариев автопрогонов."}),or?o.jsxs("details",{className:"autoruns-prompt-details",open:!0,children:[o.jsx("summary",{children:"Вопрос пользователя"}),o.jsx("p",{className:"autoruns-comment-quote",children:or.text})]}):null,kr?o.jsxs("details",{className:"autoruns-prompt-details",open:!0,children:[o.jsx("summary",{children:"Ответ ассистента"}),o.jsx("p",{className:"autoruns-comment-quote",children:kr.text})]}):null,o.jsx("div",{className:"autoruns-rating-row",role:"group","aria-label":"Рейтинг ответа",children:[1,2,3,4,5].map(m=>o.jsx("button",{type:"button",className:Re.rating>=m?"autoruns-rating-dot active":"autoruns-rating-dot",onClick:()=>Pt(E=>({...E,rating:m})),disabled:Re.saving,"aria-label":`Оценка ${m}`,children:Re.rating>=m?"●":"○"},m))}),o.jsx("div",{className:"autoruns-form-grid",children:o.jsxs("label",{children:["Автор комментария",o.jsx("input",{value:Re.annotationAuthor,onChange:m=>Pt(E=>({...E,annotationAuthor:m.target.value})),placeholder:"manual_reviewer",disabled:Re.saving})]})}),o.jsxs("label",{children:["Комментарий",o.jsx("textarea",{value:Re.comment,onChange:m=>Pt(E=>({...E,comment:m.target.value})),placeholder:"Что именно не так в ответе и что проверить.",rows:4,disabled:Re.saving})]}),Re.error?o.jsx("p",{className:"error-text",children:Re.error}):null,o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",onClick:()=>{Nn()},disabled:Re.saving,children:Re.saving?"Сохраняю...":"Готово"}),o.jsx("button",{type:"button",className:"tab",onClick:()=>lr(),disabled:Re.saving,children:"Отмена"})]})]})}):null]})}Tf.createRoot(document.getElementById("root")).render(o.jsx(Cf.StrictMode,{children:o.jsx(Em,{})})); diff --git a/llm_normalizer/frontend/dist/index.html b/llm_normalizer/frontend/dist/index.html index ce8d203..bfa3ecb 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 7ad256e..baa2553 100644 --- a/llm_normalizer/frontend/src/App.tsx +++ b/llm_normalizer/frontend/src/App.tsx @@ -31,7 +31,7 @@ const SESSION_CONFIG_KEY = "ndc_normalizer_session_config_v1"; const AUTORUNS_LAYOUT_CONFIG_KEY = "ndc_autoruns_layout_config_v1"; const AUTORUNS_SAVE_EVENT = "ndc-autoruns-save"; const ASSISTANT_STAGES = ["Анализ запроса", "Получение данных", "Подготовка ответа"]; -const DEFAULT_UI_MODE: UiMode = "assistant"; +const DEFAULT_UI_MODE: UiMode = "autoruns"; const AUTOLOAD_PROMPT_VERSION = "normalizer_v2_0_2"; const ASSISTANT_PROMPT_VERSION = "address_query_runtime_v1"; const TAB_KEYS: TabKey[] = ["normalized", "fragments", "scope", "flags", "route", "raw", "validation", "logs"]; @@ -117,6 +117,8 @@ export default function App() { const [lastError, setLastError] = useState(""); const [uiMode, setUiMode] = useState(DEFAULT_UI_MODE); + const [showAutorunsSettingsMode, setShowAutorunsSettingsMode] = useState(true); + const [showAutorunsAutoRunsMode, setShowAutorunsAutoRunsMode] = useState(true); const [showAutorunsAssistantMode, setShowAutorunsAssistantMode] = useState(true); const [showAutorunsDecompositionMode, setShowAutorunsDecompositionMode] = useState(true); const [showAutorunsProgressMode, setShowAutorunsProgressMode] = useState(true); @@ -239,6 +241,8 @@ export default function App() { const parsed = JSON.parse(cachedAutorunsLayout) as { uiMode?: UiMode; activeTab?: TabKey; + showAutorunsSettingsMode?: boolean; + showAutorunsAutoRunsMode?: boolean; showAutorunsAssistantMode?: boolean; showAutorunsDecompositionMode?: boolean; showAutorunsProgressMode?: boolean; @@ -257,12 +261,20 @@ export default function App() { showDecompositionRuntimeMode?: boolean; prompts?: PromptState; }; - if (parsed.uiMode === "assistant" || parsed.uiMode === "decomposition" || parsed.uiMode === "autoruns") { - setUiMode(parsed.uiMode); + if (parsed.uiMode === "decomposition") { + setUiMode("decomposition"); + } else if (parsed.uiMode === "assistant" || parsed.uiMode === "autoruns") { + setUiMode("autoruns"); } if (parsed.activeTab && TAB_KEYS.includes(parsed.activeTab)) { setActiveTab(parsed.activeTab); } + if (typeof parsed.showAutorunsSettingsMode === "boolean") { + setShowAutorunsSettingsMode(parsed.showAutorunsSettingsMode); + } + if (typeof parsed.showAutorunsAutoRunsMode === "boolean") { + setShowAutorunsAutoRunsMode(parsed.showAutorunsAutoRunsMode); + } if (typeof parsed.showAutorunsAssistantMode === "boolean") { setShowAutorunsAssistantMode(parsed.showAutorunsAssistantMode); } @@ -430,6 +442,8 @@ export default function App() { JSON.stringify({ uiMode, activeTab, + showAutorunsSettingsMode, + showAutorunsAutoRunsMode, showAutorunsAssistantMode, showAutorunsDecompositionMode, showAutorunsProgressMode, @@ -946,15 +960,12 @@ export default function App() { >
- - @@ -1036,6 +1047,20 @@ export default function App() {
) : uiMode === "autoruns" ? (
+ +
+ + ); + + if (embedded) { + return ( +
+
+
+

LLM Connector

+

Switch between OpenAI cloud and local OpenAI-compatible server.

+
+ {lastStatus || "Status: not checked"} +
+ {content} +
+ ); + } + + return ( + {lastStatus || "Status: not checked"}} + > + {content} ); } diff --git a/llm_normalizer/frontend/src/components/PromptPanel.tsx b/llm_normalizer/frontend/src/components/PromptPanel.tsx index 32e9e14..e56ddf4 100644 --- a/llm_normalizer/frontend/src/components/PromptPanel.tsx +++ b/llm_normalizer/frontend/src/components/PromptPanel.tsx @@ -25,6 +25,7 @@ interface PromptPanelProps { presetName: string; onPresetNameChange: (name: string) => void; diffSummary: string; + embedded?: boolean; } export function PromptPanel({ @@ -39,10 +40,11 @@ export function PromptPanel({ onDiffPrevious, presetName, onPresetNameChange, - diffSummary + diffSummary, + embedded = false }: PromptPanelProps) { - return ( - + const content = ( + <>
{diffSummary ?

{diffSummary}

: null} + + ); + + if (embedded) { + return ( +
+
+
+

Prompt Manager

+

Системный, developer и domain уровни управляются отдельно.

+
+
+ {content} +
+ ); + } + + return ( + + {content} ); } diff --git a/llm_normalizer/frontend/src/styles.css b/llm_normalizer/frontend/src/styles.css index ebd085a..8f0716a 100644 --- a/llm_normalizer/frontend/src/styles.css +++ b/llm_normalizer/frontend/src/styles.css @@ -966,6 +966,44 @@ button:disabled { scrollbar-gutter: stable; } +.autoruns-settings-col { + display: flex; + flex-direction: column; +} + +.autoruns-settings-stack { + display: grid; + gap: 12px; +} + +.embedded-panel-section { + display: grid; + gap: 12px; + border-radius: 12px; + background: rgb(var(--rgb-surface-horizontal)); + padding: 12px; +} + +.embedded-panel-section-header { + display: flex; + align-items: flex-start; + justify-content: space-between; + gap: 12px; +} + +.embedded-panel-section-header h4 { + margin: 0; + color: var(--text-main); + font-size: 0.92rem; +} + +.embedded-panel-section-header p { + margin: 6px 0 0; + color: var(--text-muted); + font-size: 0.78rem; + line-height: 1.4; +} + .autoruns-assistant-live-col { background: rgb(var(--rgb-surface-main)); padding: 12px; diff --git a/scripts/domain_truth_harness.py b/scripts/domain_truth_harness.py new file mode 100644 index 0000000..6a1ef9a --- /dev/null +++ b/scripts/domain_truth_harness.py @@ -0,0 +1,1063 @@ +from __future__ import annotations + +import argparse +import json +import re +from datetime import date, datetime, timezone +from pathlib import Path +from types import SimpleNamespace +from typing import Any + +import domain_case_loop as dcl + + +REPO_ROOT = Path(__file__).resolve().parent.parent +DEFAULT_OUTPUT_ROOT = REPO_ROOT / "artifacts" / "domain_runs" +TRUTH_HARNESS_SPEC_SCHEMA_VERSION = "domain_truth_harness_spec_v1" +TRUTH_HARNESS_STATE_SCHEMA_VERSION = "domain_truth_harness_state_v1" +TRUTH_HARNESS_REVIEW_SCHEMA_VERSION = "domain_truth_harness_review_v1" +TRUTH_HARNESS_MANIFEST_SCHEMA_VERSION = "domain_truth_harness_manifest_v1" +DEFAULT_CRITICALITY = "critical" +TECHNICAL_QUESTION_FIELDS = ( + "expected_intents", + "expected_capability", + "expected_recipe", + "expected_result_mode", + "required_filters", + "forbidden_capabilities", + "forbidden_recipes", + "required_state_objects", + "required_answer_shape", + "forbidden_answer_patterns", + "required_carryover_invariants", + "invariant_severity", +) + + +def dump_json(payload: Any) -> str: + return json.dumps(payload, ensure_ascii=False, indent=2) + + +def write_text(file_path: Path, text: str) -> None: + file_path.parent.mkdir(parents=True, exist_ok=True) + file_path.write_text(text, encoding="utf-8", newline="\n") + + +def write_json(file_path: Path, payload: Any) -> None: + write_text(file_path, dump_json(payload) + "\n") + + +def read_json(file_path: Path) -> dict[str, Any]: + payload = json.loads(file_path.read_text(encoding="utf-8-sig")) + if not isinstance(payload, dict): + raise RuntimeError(f"Expected JSON object in {file_path}") + return payload + + +def normalize_criticality(value: Any) -> str: + raw = str(value or "").strip().lower() + if raw in {"critical", "warning", "info"}: + return raw + return DEFAULT_CRITICALITY + + +def normalize_pattern_list(raw_value: Any) -> list[str]: + return [item for item in dcl.normalize_string_list(raw_value) if item] + + +def normalize_filter_values_mapping(raw_value: Any) -> dict[str, list[str]]: + if not isinstance(raw_value, dict): + return {} + output: dict[str, list[str]] = {} + for key, value in raw_value.items(): + normalized_key = str(key or "").strip() + if not normalized_key: + continue + normalized_values = normalize_pattern_list(value) + if normalized_values: + output[normalized_key] = normalized_values + return output + + +def normalize_step_spec(index: int, raw_step: Any) -> dict[str, Any]: + normalized_step = dcl.normalize_step_definition(index, raw_step) + step = raw_step if isinstance(raw_step, dict) else {} + normalized_step["criticality"] = normalize_criticality(step.get("criticality")) + normalized_step["allowed_reply_types"] = normalize_pattern_list(step.get("allowed_reply_types")) + normalized_step["allowed_limited_reason_categories"] = normalize_pattern_list( + step.get("allowed_limited_reason_categories") + ) + normalized_step["required_answer_patterns_any"] = normalize_pattern_list(step.get("required_answer_patterns_any")) + normalized_step["required_answer_patterns_all"] = normalize_pattern_list(step.get("required_answer_patterns_all")) + normalized_step["required_direct_answer_patterns_any"] = normalize_pattern_list( + step.get("required_direct_answer_patterns_any") + ) + normalized_step["required_direct_answer_patterns_all"] = normalize_pattern_list( + step.get("required_direct_answer_patterns_all") + ) + normalized_step["forbidden_direct_answer_patterns"] = normalize_pattern_list( + step.get("forbidden_direct_answer_patterns") + ) + normalized_step["forbidden_filter_keys"] = normalize_pattern_list(step.get("forbidden_filter_keys")) + normalized_step["forbidden_filter_values"] = normalize_filter_values_mapping(step.get("forbidden_filter_values")) + normalized_step["required_filter_within_previous_step_period"] = { + str(key).strip(): str(value).strip() + for key, value in (step.get("required_filter_within_previous_step_period") or {}).items() + if str(key).strip() and str(value).strip() + } if isinstance(step.get("required_filter_within_previous_step_period"), dict) else {} + normalized_step["notes"] = str(step.get("notes") or "").strip() or None + return normalized_step + + +def load_truth_harness_spec(file_path: Path) -> dict[str, Any]: + raw_spec = read_json(file_path) + steps_raw = raw_spec.get("steps") + if not isinstance(steps_raw, list) or not steps_raw: + raise RuntimeError("Truth harness spec must define non-empty `steps`") + scenario_id = str(raw_spec.get("scenario_id") or "").strip() + domain = str(raw_spec.get("domain") or "").strip() + if not scenario_id: + raise RuntimeError("Truth harness spec must define `scenario_id`") + if not domain: + raise RuntimeError("Truth harness spec must define `domain`") + return { + "schema_version": str(raw_spec.get("schema_version") or TRUTH_HARNESS_SPEC_SCHEMA_VERSION), + "scenario_id": scenario_id, + "domain": domain, + "title": str(raw_spec.get("title") or scenario_id).strip() or scenario_id, + "description": str(raw_spec.get("description") or "").strip() or None, + "source_export": str(raw_spec.get("source_export") or "").strip() or None, + "bindings": dcl.normalize_bindings(raw_spec.get("bindings")), + "steps": [normalize_step_spec(index + 1, raw_step) for index, raw_step in enumerate(steps_raw)], + } + + +def build_runtime_bindings() -> dict[str, Any]: + today = date.today() + today_iso = today.isoformat() + today_dot = today.strftime("%d.%m.%Y") + return { + "today_iso": today_iso, + "today_dot": today_dot, + "today_iso_regex": re.escape(today_iso), + "today_dot_regex": re.escape(today_dot), + "generated_at_utc": datetime.now(timezone.utc).replace(microsecond=0).isoformat(), + } + + +def build_review_root( + bindings: dict[str, Any], runtime_bindings: dict[str, Any], step_results: dict[str, Any] +) -> dict[str, Any]: + root: dict[str, Any] = { + "bindings": bindings, + "runtime": runtime_bindings, + "step_results": step_results, + } + if isinstance(step_results, dict): + root.update(step_results) + return root + + +def lookup_review_value( + path_expression: str, step_results: dict[str, Any], bindings: dict[str, Any], runtime_bindings: dict[str, Any] +) -> Any: + current: Any = build_review_root(bindings, runtime_bindings, step_results) + for token in dcl.parse_path_tokens(path_expression): + if isinstance(token, int): + if not isinstance(current, list): + raise RuntimeError(f"Placeholder `{path_expression}` does not point to a list before index access") + if token >= len(current): + raise RuntimeError(f"Placeholder `{path_expression}` index {token} is out of range") + current = current[token] + continue + if not isinstance(current, dict) or token not in current: + raise RuntimeError(f"Placeholder `{path_expression}` could not be resolved at `{token}`") + current = current[token] + return current + + +def resolve_template_string( + template: str, step_results: dict[str, Any], bindings: dict[str, Any], runtime_bindings: dict[str, Any] +) -> str: + pattern = re.compile(r"{{\s*([^{}]+?)\s*}}") + + def replace(match: re.Match[str]) -> str: + value = lookup_review_value(match.group(1), step_results, bindings, runtime_bindings) + if isinstance(value, (dict, list)): + return dump_json(value) + return str(value) + + return pattern.sub(replace, template) + + +def resolve_nested_placeholders( + raw_value: Any, step_results: dict[str, Any], bindings: dict[str, Any], runtime_bindings: dict[str, Any] +) -> Any: + if isinstance(raw_value, str): + return resolve_template_string(raw_value, step_results, bindings, runtime_bindings) + if isinstance(raw_value, list): + return [resolve_nested_placeholders(item, step_results, bindings, runtime_bindings) for item in raw_value] + if isinstance(raw_value, dict): + return { + str(key): resolve_nested_placeholders(value, step_results, bindings, runtime_bindings) + for key, value in raw_value.items() + } + return raw_value + + +def build_generated_manifest(spec: dict[str, Any]) -> dict[str, Any]: + manifest_steps: list[dict[str, Any]] = [] + previous_step_id: str | None = None + for step in spec["steps"]: + manifest_step: dict[str, Any] = { + "step_id": step["step_id"], + "title": step["title"], + "question": step["question_template"], + "depends_on": [previous_step_id] if previous_step_id else [], + "analysis_context": {}, + } + for field_name in TECHNICAL_QUESTION_FIELDS: + manifest_step[field_name] = step.get(field_name) + manifest_steps.append(manifest_step) + previous_step_id = step["step_id"] + return { + "schema_version": TRUTH_HARNESS_MANIFEST_SCHEMA_VERSION, + "scenario_id": spec["scenario_id"], + "domain": spec["domain"], + "title": spec["title"], + "description": spec.get("description"), + "analysis_context": {}, + "bindings": spec.get("bindings") or {}, + "steps": manifest_steps, + } + + +def build_runner_args(args: argparse.Namespace) -> SimpleNamespace: + return SimpleNamespace( + backend_url=args.backend_url, + prompt_version=args.prompt_version, + llm_provider=args.llm_provider, + llm_model=args.llm_model, + llm_base_url=args.llm_base_url, + llm_api_key=args.llm_api_key, + temperature=args.temperature, + max_output_tokens=args.max_output_tokens, + timeout_seconds=args.timeout_seconds, + use_mock=bool(args.use_mock), + ) + + +def build_plain_assistant_message_payload( + args: argparse.Namespace, *, question: str, session_id: str | None +) -> dict[str, Any]: + return dcl.drop_none_values( + { + "session_id": session_id, + "user_message": question, + "message": question, + "mode": "assistant", + "llmProvider": args.llm_provider, + "apiKey": args.llm_api_key, + "model": args.llm_model, + "baseUrl": args.llm_base_url, + "temperature": args.temperature, + "maxOutputTokens": args.max_output_tokens, + "promptVersion": args.prompt_version, + "useMock": bool(args.use_mock), + } + ) + + +def build_placeholder_step_result(step_state: dict[str, Any]) -> dict[str, Any]: + result = dict(step_state) + result["filters"] = dict(step_state.get("extracted_filters") or {}) + return result + + +def append_finding( + findings: list[dict[str, Any]], + step: dict[str, Any], + code: str, + message: str, + *, + actual: Any = None, + expected: Any = None, +) -> None: + findings.append( + { + "code": code, + "severity": step.get("criticality") or DEFAULT_CRITICALITY, + "message": message, + "actual": actual, + "expected": expected, + } + ) + + +def matches_any_pattern(text: str, patterns: list[str]) -> bool: + return any(re.search(pattern, text, flags=re.IGNORECASE) for pattern in patterns if pattern) + + +def find_missing_patterns(text: str, patterns: list[str]) -> list[str]: + return [pattern for pattern in patterns if pattern and not re.search(pattern, text, flags=re.IGNORECASE)] + + +def normalize_actual_filter_value(filter_key: str, raw_value: Any) -> str: + if filter_key in {"as_of_date", "period_from", "period_to"}: + return dcl.normalize_iso_date(raw_value) or "" + return str(raw_value or "").strip() + + +def evaluate_truth_step( + *, + step: dict[str, Any], + step_state: dict[str, Any], + step_results: dict[str, Any], + bindings: dict[str, Any], + runtime_bindings: dict[str, Any], +) -> dict[str, Any]: + findings: list[dict[str, Any]] = [] + reply_type = str(step_state.get("reply_type") or "").strip() + assistant_text = str(step_state.get("assistant_text") or "") + direct_answer = str(step_state.get("actual_direct_answer") or "").strip() + detected_intent = str(step_state.get("detected_intent") or "").strip() + selected_recipe = str(step_state.get("selected_recipe") or "").strip() + capability_id = str(step_state.get("capability_id") or "").strip() + limited_reason_category = str(step_state.get("limited_reason_category") or "").strip() + extracted_filters = ( + step_state.get("extracted_filters") if isinstance(step_state.get("extracted_filters"), dict) else {} + ) + + if step_state.get("question_resolved") != step["question_template"]: + append_finding( + findings, + step, + "question_sequence_mismatch", + "В live/export прогоне вопрос отличается от зафиксированного exact-сценария.", + actual=step_state.get("question_resolved"), + expected=step["question_template"], + ) + + if step["allowed_reply_types"] and reply_type not in step["allowed_reply_types"]: + append_finding( + findings, + step, + "unexpected_reply_type", + "Тип ответа не соответствует ожидаемому режиму шага.", + actual=reply_type, + expected=step["allowed_reply_types"], + ) + + allowed_limited_reason_categories = step.get("allowed_limited_reason_categories") or [] + if allowed_limited_reason_categories and limited_reason_category: + if limited_reason_category not in allowed_limited_reason_categories: + append_finding( + findings, + step, + "unexpected_limited_reason_category", + "Категория честного ограничения не соответствует допустимым сценарным условиям.", + actual=limited_reason_category, + expected=allowed_limited_reason_categories, + ) + + expected_intents = dcl.normalize_string_list( + resolve_nested_placeholders(step.get("expected_intents") or [], step_results, bindings, runtime_bindings) + ) + if expected_intents and not dcl.identifier_in_list(detected_intent, expected_intents): + append_finding( + findings, + step, + "wrong_intent", + "Интент не соответствует ожидаемому бизнес-смыслу шага.", + actual=detected_intent or None, + expected=expected_intents, + ) + + expected_capability = str( + resolve_nested_placeholders(step.get("expected_capability"), step_results, bindings, runtime_bindings) or "" + ).strip() + if expected_capability and not dcl.identifiers_match(capability_id, expected_capability): + append_finding( + findings, + step, + "wrong_capability", + "Выбрана не та capability, которую должен был отработать этот шаг.", + actual=capability_id or None, + expected=expected_capability, + ) + + expected_recipe = str( + resolve_nested_placeholders(step.get("expected_recipe"), step_results, bindings, runtime_bindings) or "" + ).strip() + if expected_recipe and not dcl.identifiers_match(selected_recipe, expected_recipe): + append_finding( + findings, + step, + "wrong_recipe", + "Выбран не тот рецепт адресного контура.", + actual=selected_recipe or None, + expected=expected_recipe, + ) + + expected_result_mode = str( + resolve_nested_placeholders(step.get("expected_result_mode"), step_results, bindings, runtime_bindings) or "" + ).strip() + actual_result_mode = str(step_state.get("result_mode") or "").strip() + if expected_result_mode and actual_result_mode and not dcl.identifiers_match(actual_result_mode, expected_result_mode): + append_finding( + findings, + step, + "wrong_result_mode", + "Режим результата не соответствует ожидаемой форме ответа.", + actual=actual_result_mode, + expected=expected_result_mode, + ) + + required_filters = dcl.normalize_validation_filters( + resolve_nested_placeholders(step.get("required_filters") or {}, step_results, bindings, runtime_bindings) + ) + for filter_key, expected_value in required_filters.items(): + actual_value = normalize_actual_filter_value(filter_key, extracted_filters.get(filter_key)) + if not actual_value: + append_finding( + findings, + step, + f"missing_filter:{filter_key}", + f"В техчате отсутствует обязательный фильтр `{filter_key}`.", + actual=actual_value or None, + expected=expected_value, + ) + continue + if actual_value != expected_value: + append_finding( + findings, + step, + f"wrong_filter:{filter_key}", + f"Фильтр `{filter_key}` не совпадает с ожидаемым carryover/period состоянием.", + actual=actual_value, + expected=expected_value, + ) + + forbidden_filter_keys = dcl.normalize_string_list( + resolve_nested_placeholders(step.get("forbidden_filter_keys") or [], step_results, bindings, runtime_bindings) + ) + for filter_key in forbidden_filter_keys: + actual_value = normalize_actual_filter_value(filter_key, extracted_filters.get(filter_key)) + if actual_value: + append_finding( + findings, + step, + f"forbidden_filter_key:{filter_key}", + f"Шаг не должен был тащить фильтр `{filter_key}` в техчат.", + actual=actual_value, + expected="absent", + ) + + forbidden_filter_values = normalize_filter_values_mapping( + resolve_nested_placeholders(step.get("forbidden_filter_values") or {}, step_results, bindings, runtime_bindings) + ) + for filter_key, forbidden_values in forbidden_filter_values.items(): + actual_value = normalize_actual_filter_value(filter_key, extracted_filters.get(filter_key)) + if actual_value and actual_value in forbidden_values: + append_finding( + findings, + step, + f"forbidden_filter_value:{filter_key}", + f"Шаг унаследовал запрещенное значение фильтра `{filter_key}`.", + actual=actual_value, + expected={"not_in": forbidden_values}, + ) + + within_previous_period = step.get("required_filter_within_previous_step_period") or {} + for filter_key, step_ref in within_previous_period.items(): + resolved_step_ref = str( + resolve_nested_placeholders(step_ref, step_results, bindings, runtime_bindings) or "" + ).strip() + previous_step = step_results.get(resolved_step_ref) + actual_value = dcl.normalize_iso_date(extracted_filters.get(filter_key)) + if not isinstance(previous_step, dict): + append_finding( + findings, + step, + f"previous_step_missing:{filter_key}", + f"Не удалось разрешить предыдущий шаг `{resolved_step_ref}` для проверки диапазона фильтра `{filter_key}`.", + actual=actual_value, + expected=resolved_step_ref, + ) + continue + previous_filters = previous_step.get("filters") if isinstance(previous_step.get("filters"), dict) else {} + previous_from = dcl.normalize_iso_date(previous_filters.get("period_from")) + previous_to = dcl.normalize_iso_date(previous_filters.get("period_to")) + if not actual_value or not previous_from or not previous_to: + append_finding( + findings, + step, + f"period_carryover_missing:{filter_key}", + f"Не удалось подтвердить, что `{filter_key}` попал в диапазон предыдущего шага `{resolved_step_ref}`.", + actual=actual_value, + expected={"period_from": previous_from, "period_to": previous_to}, + ) + continue + if not (previous_from <= actual_value <= previous_to): + append_finding( + findings, + step, + f"period_carryover_outside:{filter_key}", + f"Фильтр `{filter_key}` вышел за диапазон предыдущего релевантного периода.", + actual=actual_value, + expected={"period_from": previous_from, "period_to": previous_to}, + ) + + required_answer_patterns_any = normalize_pattern_list( + resolve_nested_placeholders(step.get("required_answer_patterns_any") or [], step_results, bindings, runtime_bindings) + ) + if required_answer_patterns_any and not matches_any_pattern(assistant_text, required_answer_patterns_any): + append_finding( + findings, + step, + "required_answer_patterns_any_missing", + "Полный ответ модели не содержит ни одного из обязательных смысловых паттернов.", + actual=assistant_text, + expected=required_answer_patterns_any, + ) + + required_answer_patterns_all = normalize_pattern_list( + resolve_nested_placeholders(step.get("required_answer_patterns_all") or [], step_results, bindings, runtime_bindings) + ) + missing_answer_patterns = find_missing_patterns(assistant_text, required_answer_patterns_all) + if missing_answer_patterns: + append_finding( + findings, + step, + "required_answer_patterns_all_missing", + "Полный ответ модели не закрыл обязательные паттерны шага.", + actual=assistant_text, + expected=missing_answer_patterns, + ) + + forbidden_answer_patterns = normalize_pattern_list( + resolve_nested_placeholders(step.get("forbidden_answer_patterns") or [], step_results, bindings, runtime_bindings) + ) + forbidden_answer_hits = [pattern for pattern in forbidden_answer_patterns if re.search(pattern, assistant_text, flags=re.IGNORECASE)] + if forbidden_answer_hits: + append_finding( + findings, + step, + "forbidden_answer_pattern_hit", + "В полном ответе модели найден запрещенный паттерн.", + actual=forbidden_answer_hits, + expected="absent", + ) + + required_direct_any = normalize_pattern_list( + resolve_nested_placeholders( + step.get("required_direct_answer_patterns_any") or [], + step_results, + bindings, + runtime_bindings, + ) + ) + if required_direct_any and not matches_any_pattern(direct_answer, required_direct_any): + append_finding( + findings, + step, + "required_direct_answer_patterns_any_missing", + "Первая строка ответа не содержит ни одного из обязательных direct-answer паттернов.", + actual=direct_answer, + expected=required_direct_any, + ) + + required_direct_all = normalize_pattern_list( + resolve_nested_placeholders( + step.get("required_direct_answer_patterns_all") or [], + step_results, + bindings, + runtime_bindings, + ) + ) + missing_direct_patterns = find_missing_patterns(direct_answer, required_direct_all) + if missing_direct_patterns: + append_finding( + findings, + step, + "required_direct_answer_patterns_all_missing", + "Первая строка ответа не закрыла обязательные direct-answer паттерны.", + actual=direct_answer, + expected=missing_direct_patterns, + ) + + forbidden_direct_patterns = normalize_pattern_list( + resolve_nested_placeholders( + step.get("forbidden_direct_answer_patterns") or [], + step_results, + bindings, + runtime_bindings, + ) + ) + forbidden_direct_hits = [pattern for pattern in forbidden_direct_patterns if re.search(pattern, direct_answer, flags=re.IGNORECASE)] + if forbidden_direct_hits: + append_finding( + findings, + step, + "forbidden_direct_answer_pattern_hit", + "Первая строка ответа попала в запрещенный direct-answer паттерн.", + actual=forbidden_direct_hits, + expected="absent", + ) + + critical_findings = [item for item in findings if item.get("severity") == "critical"] + warning_findings = [item for item in findings if item.get("severity") == "warning"] + info_findings = [item for item in findings if item.get("severity") == "info"] + review_status = "pass" + if critical_findings: + review_status = "fail" + elif warning_findings: + review_status = "warning" + elif info_findings: + review_status = "info" + + reviewed_state = dict(step_state) + reviewed_state["review_findings"] = findings + reviewed_state["review_status"] = review_status + reviewed_state["critical_findings_count"] = len(critical_findings) + reviewed_state["warning_findings_count"] = len(warning_findings) + reviewed_state["info_findings_count"] = len(info_findings) + reviewed_state["truth_harness_notes"] = step.get("notes") + return reviewed_state + + +def build_truth_review_summary(spec: dict[str, Any], scenario_state: dict[str, Any], review_source: str) -> dict[str, Any]: + step_outputs = scenario_state.get("step_outputs") if isinstance(scenario_state.get("step_outputs"), dict) else {} + review_steps = [step_outputs.get(step["step_id"], {}) for step in spec["steps"]] + passed = sum(1 for item in review_steps if str(item.get("review_status") or "") == "pass") + warnings = sum(1 for item in review_steps if str(item.get("review_status") or "") == "warning") + failed = sum(1 for item in review_steps if str(item.get("review_status") or "") == "fail") + return { + "schema_version": TRUTH_HARNESS_REVIEW_SCHEMA_VERSION, + "review_source": review_source, + "scenario_id": spec["scenario_id"], + "domain": spec["domain"], + "title": spec["title"], + "session_id": scenario_state.get("session_id"), + "steps_total": len(spec["steps"]), + "steps_passed": passed, + "steps_with_warning": warnings, + "steps_failed": failed, + "overall_status": "fail" if failed else ("warning" if warnings else "pass"), + } + + +def build_truth_review_markdown(spec: dict[str, Any], scenario_state: dict[str, Any], review_summary: dict[str, Any]) -> str: + lines = [ + "# Truth harness review", + "", + f"- scenario_id: `{spec['scenario_id']}`", + f"- domain: `{spec['domain']}`", + f"- title: {spec['title']}", + f"- review_source: `{review_summary.get('review_source') or 'n/a'}`", + f"- session_id: `{scenario_state.get('session_id') or 'n/a'}`", + f"- overall_status: `{review_summary.get('overall_status') or 'n/a'}`", + f"- steps_total: `{review_summary.get('steps_total')}`", + f"- steps_passed: `{review_summary.get('steps_passed')}`", + f"- steps_with_warning: `{review_summary.get('steps_with_warning')}`", + f"- steps_failed: `{review_summary.get('steps_failed')}`", + "", + "## Steps", + ] + step_outputs = scenario_state.get("step_outputs") if isinstance(scenario_state.get("step_outputs"), dict) else {} + for index, step in enumerate(spec["steps"], start=1): + step_state = step_outputs.get(step["step_id"], {}) + findings = step_state.get("review_findings") if isinstance(step_state.get("review_findings"), list) else [] + lines.extend( + [ + f"{index}. `{step['step_id']}` - {step['question_template']}", + f"review_status: `{step_state.get('review_status') or 'n/a'}`", + f"reply_type: `{step_state.get('reply_type') or 'n/a'}`", + f"intent: `{step_state.get('detected_intent') or 'n/a'}`", + f"recipe: `{step_state.get('selected_recipe') or 'n/a'}`", + f"capability: `{step_state.get('capability_id') or 'n/a'}`", + f"limited_reason_category: `{step_state.get('limited_reason_category') or 'n/a'}`", + f"filters: `{dump_json(step_state.get('extracted_filters') or {})}`", + f"direct_answer: {step_state.get('actual_direct_answer') or 'n/a'}", + ] + ) + if step.get("notes"): + lines.append(f"notes: {step['notes']}") + if findings: + lines.append("findings:") + for finding in findings: + lines.append( + f"- [{finding.get('severity')}] {finding.get('code')}: {finding.get('message')} " + f"(actual={dump_json(finding.get('actual'))}, expected={dump_json(finding.get('expected'))})" + ) + else: + lines.append("findings: none") + lines.append("") + return "\n".join(lines).strip() + "\n" + + +def save_step_bundle( + *, + step_dir: Path, + export_markdown: str, + turn_artifact: dict[str, Any], + session_record: dict[str, Any] | None, + response_payload: dict[str, Any] | None, + step_state: dict[str, Any], +) -> None: + write_text(step_dir / "output.md", export_markdown) + write_json(step_dir / "debug.json", step_state.get("technical_debug_payload") or {}) + write_json(step_dir / "turn.json", turn_artifact) + write_json(step_dir / "assistant_response.json", response_payload or {}) + write_json(step_dir / "step_state.json", step_state) + if session_record is not None: + write_json(step_dir / "session.json", session_record) + write_text(step_dir / "resolved_question.txt", f"{step_state.get('question_resolved') or ''}\n") + + +def build_step_state_from_turn( + *, + spec: dict[str, Any], + step: dict[str, Any], + step_index: int, + question: str, + conversation: list[dict[str, Any]], + session_record: dict[str, Any] | None, + export_markdown: str, +) -> tuple[dict[str, Any], dict[str, Any]]: + mini_conversation = conversation[-2:] if len(conversation) >= 2 else conversation + turn_artifact = dcl.build_turn_artifact( + slot="step", + domain=spec["domain"], + case_id=spec["scenario_id"], + question=question, + session_id=str(spec.get("session_id") or "n/a"), + conversation=mini_conversation, + session_record=session_record, + job_record=None, + report_case=None, + export_file_name="output.md", + ) + turn_artifact["schema_version"] = "domain_truth_harness_turn_artifact_v1" + turn_artifact["scenario"] = { + "scenario_id": spec["scenario_id"], + "step_id": step["step_id"], + "step_index": step_index, + "question_template": step["question_template"], + "question_resolved": question, + "review_mode": "truth_harness_strict_replay", + } + last_assistant = dcl.find_last_assistant(mini_conversation) + entries = dcl.extract_structured_entries(str(last_assistant.get("text") or "")) + base_step_state = dcl.build_scenario_step_state( + scenario_id=spec["scenario_id"], + domain=spec["domain"], + step=step, + step_index=step_index, + question_resolved=question, + analysis_context={}, + turn_artifact=turn_artifact, + entries=entries, + ) + base_step_state["export_markdown"] = export_markdown + base_step_state["technical_debug_payload"] = ( + last_assistant.get("debug") if isinstance(last_assistant.get("debug"), dict) else {} + ) + return base_step_state, turn_artifact + + +def build_conversation_pairs(conversation: list[dict[str, Any]]) -> list[tuple[dict[str, Any], dict[str, Any] | None]]: + pairs: list[tuple[dict[str, Any], dict[str, Any] | None]] = [] + for index, item in enumerate(conversation): + if item.get("role") != "user": + continue + assistant_item: dict[str, Any] | None = None + if index + 1 < len(conversation) and conversation[index + 1].get("role") == "assistant": + assistant_item = conversation[index + 1] + pairs.append((item, assistant_item)) + return pairs + + +def review_export(spec: dict[str, Any], export_path: Path, output_dir: Path) -> dict[str, Any]: + export_text = export_path.read_text(encoding="utf-8-sig") + session_id, conversation = dcl.parse_export_markdown(export_text) + pairs = build_conversation_pairs(conversation) + if len(pairs) != len(spec["steps"]): + raise RuntimeError( + f"Spec steps ({len(spec['steps'])}) and export user turns ({len(pairs)}) do not match for {export_path}" + ) + + output_dir.mkdir(parents=True, exist_ok=True) + write_json(output_dir / "truth_harness_spec.json", spec) + write_text(output_dir / "source_export.md", export_text) + + step_results: dict[str, Any] = {} + runtime_bindings = build_runtime_bindings() + scenario_state: dict[str, Any] = { + "schema_version": TRUTH_HARNESS_STATE_SCHEMA_VERSION, + "scenario_id": spec["scenario_id"], + "domain": spec["domain"], + "title": spec["title"], + "session_id": session_id, + "review_source": str(export_path), + "step_outputs": {}, + "updated_at": datetime.now(timezone.utc).replace(microsecond=0).isoformat(), + } + steps_dir = output_dir / "steps" + steps_dir.mkdir(parents=True, exist_ok=True) + + for index, (step, pair) in enumerate(zip(spec["steps"], pairs, strict=True), start=1): + user_item, assistant_item = pair + if assistant_item is None: + raise RuntimeError(f"Export pair for step `{step['step_id']}` is missing assistant message") + mini_conversation = [user_item, assistant_item] + export_markdown = dcl.build_conversation_export(session_id, mini_conversation, mode="technical") + base_step_state, turn_artifact = build_step_state_from_turn( + spec={**spec, "session_id": session_id}, + step=step, + step_index=index, + question=str(user_item.get("text") or ""), + conversation=mini_conversation, + session_record=None, + export_markdown=export_markdown, + ) + reviewed_step = evaluate_truth_step( + step=step, + step_state=base_step_state, + step_results=step_results, + bindings=spec.get("bindings") or {}, + runtime_bindings=runtime_bindings, + ) + scenario_state["step_outputs"][step["step_id"]] = reviewed_step + step_results[step["step_id"]] = build_placeholder_step_result(reviewed_step) + save_step_bundle( + step_dir=steps_dir / step["step_id"], + export_markdown=export_markdown, + turn_artifact=turn_artifact, + session_record=None, + response_payload=None, + step_state=reviewed_step, + ) + + scenario_state["updated_at"] = datetime.now(timezone.utc).replace(microsecond=0).isoformat() + review_summary = build_truth_review_summary(spec, scenario_state, f"export:{export_path}") + review_markdown = build_truth_review_markdown(spec, scenario_state, review_summary) + write_json(output_dir / "scenario_state.json", scenario_state) + write_json(output_dir / "truth_review.json", {"summary": review_summary, "steps": scenario_state["step_outputs"]}) + write_text(output_dir / "truth_review.md", review_markdown) + return {"scenario_state": scenario_state, "review_summary": review_summary} + + +def run_live(spec: dict[str, Any], output_dir: Path, args: argparse.Namespace) -> dict[str, Any]: + runner_args = build_runner_args(args) + dcl.ensure_backend_health(runner_args.backend_url, runner_args.timeout_seconds) + + output_dir.mkdir(parents=True, exist_ok=True) + manifest = build_generated_manifest(spec) + write_json(output_dir / "truth_harness_spec.json", spec) + write_json(output_dir / "scenario_manifest.json", manifest) + + steps_dir = output_dir / "steps" + steps_dir.mkdir(parents=True, exist_ok=True) + + runtime_bindings = build_runtime_bindings() + step_results: dict[str, Any] = {} + scenario_state: dict[str, Any] = { + "schema_version": TRUTH_HARNESS_STATE_SCHEMA_VERSION, + "scenario_id": spec["scenario_id"], + "domain": spec["domain"], + "title": spec["title"], + "session_id": None, + "review_source": "live_strict_replay", + "step_outputs": {}, + "updated_at": datetime.now(timezone.utc).replace(microsecond=0).isoformat(), + } + write_json(output_dir / "scenario_state.json", scenario_state) + + current_session_id: str | None = None + for index, step in enumerate(spec["steps"], start=1): + question = step["question_template"] + payload = build_plain_assistant_message_payload(runner_args, question=question, session_id=current_session_id) + response_payload = dcl.http_json( + f"{runner_args.backend_url}/api/assistant/message", + method="POST", + payload=payload, + timeout=max(30, int(runner_args.timeout_seconds)), + ) + current_session_id = str(response_payload.get("session_id") or current_session_id or "").strip() or None + if not current_session_id: + raise RuntimeError(f"Assistant response for step `{step['step_id']}` does not contain session_id") + session_record = dcl.fetch_session_snapshot( + runner_args.backend_url, current_session_id, runner_args.timeout_seconds + ) + conversation = dcl.extract_conversation_from_session(session_record) + export_markdown = dcl.build_conversation_export(current_session_id, conversation, mode="technical") + base_step_state, turn_artifact = build_step_state_from_turn( + spec={**spec, "session_id": current_session_id}, + step=step, + step_index=index, + question=question, + conversation=conversation, + session_record=session_record, + export_markdown=export_markdown, + ) + reviewed_step = evaluate_truth_step( + step=step, + step_state=base_step_state, + step_results=step_results, + bindings=spec.get("bindings") or {}, + runtime_bindings=runtime_bindings, + ) + scenario_state["session_id"] = current_session_id + scenario_state["step_outputs"][step["step_id"]] = reviewed_step + step_results[step["step_id"]] = build_placeholder_step_result(reviewed_step) + scenario_state["updated_at"] = datetime.now(timezone.utc).replace(microsecond=0).isoformat() + save_step_bundle( + step_dir=steps_dir / step["step_id"], + export_markdown=export_markdown, + turn_artifact=turn_artifact, + session_record=session_record, + response_payload=response_payload, + step_state=reviewed_step, + ) + write_json(output_dir / "scenario_state.json", scenario_state) + print( + f"[truth-harness] step {index}/{len(spec['steps'])}: {step['step_id']} -> " + f"{reviewed_step.get('review_status')} ({reviewed_step.get('reply_type') or 'n/a'})" + ) + + review_summary = build_truth_review_summary(spec, scenario_state, "live_strict_replay") + review_markdown = build_truth_review_markdown(spec, scenario_state, review_summary) + write_text(output_dir / "session_id.txt", f"{scenario_state.get('session_id') or ''}\n") + write_json(output_dir / "scenario_state.json", scenario_state) + write_json(output_dir / "truth_review.json", {"summary": review_summary, "steps": scenario_state["step_outputs"]}) + write_text(output_dir / "truth_review.md", review_markdown) + write_text(output_dir / "final_status.md", f"# Final status\n\n- status: `{review_summary['overall_status']}`\n") + print(f"[truth-harness] saved artifacts to {output_dir}") + print(f"[truth-harness] overall_status={review_summary['overall_status']}") + return {"scenario_state": scenario_state, "review_summary": review_summary} + + +def build_bootstrap_spec(export_path: Path, scenario_id: str, domain: str, title: str | None) -> dict[str, Any]: + export_text = export_path.read_text(encoding="utf-8-sig") + _, conversation = dcl.parse_export_markdown(export_text) + pairs = build_conversation_pairs(conversation) + steps = [] + for index, (user_item, _assistant_item) in enumerate(pairs, start=1): + steps.append( + { + "step_id": f"step_{index:02d}", + "title": f"Step {index:02d}", + "question": str(user_item.get("text") or "").strip(), + "criticality": DEFAULT_CRITICALITY, + } + ) + return { + "schema_version": TRUTH_HARNESS_SPEC_SCHEMA_VERSION, + "scenario_id": scenario_id, + "domain": domain, + "title": title or scenario_id, + "description": f"Exact sequential replay spec bootstrapped from {export_path}", + "source_export": str(export_path), + "bindings": {}, + "steps": steps, + } + + +def default_output_dir(base_name: str) -> Path: + stamp = datetime.now(timezone.utc).strftime("%Y%m%d_%H%M%S") + return DEFAULT_OUTPUT_ROOT / f"{base_name}_{stamp}" + + +def handle_bootstrap(args: argparse.Namespace) -> int: + export_path = Path(args.export).resolve() + output_path = Path(args.output).resolve() + spec = build_bootstrap_spec( + export_path=export_path, + scenario_id=args.scenario_id, + domain=args.domain, + title=args.title, + ) + write_json(output_path, spec) + print(f"[truth-harness] wrote bootstrap spec to {output_path}") + return 0 + + +def handle_review_export(args: argparse.Namespace) -> int: + spec_path = Path(args.spec).resolve() + spec = load_truth_harness_spec(spec_path) + export_path = Path(args.export or spec.get("source_export") or "").resolve() + if not export_path.exists(): + raise RuntimeError("Export file for review-export does not exist") + output_dir = Path(args.output_dir).resolve() if args.output_dir else default_output_dir( + f"{spec['scenario_id']}_review" + ) + result = review_export(spec, export_path, output_dir) + print(f"[truth-harness] review-export overall_status={result['review_summary']['overall_status']}") + print(f"[truth-harness] artifacts={output_dir}") + return 0 + + +def handle_run_live(args: argparse.Namespace) -> int: + spec_path = Path(args.spec).resolve() + spec = load_truth_harness_spec(spec_path) + output_dir = Path(args.output_dir).resolve() if args.output_dir else default_output_dir( + f"{spec['scenario_id']}_live" + ) + result = run_live(spec, output_dir, args) + print(f"[truth-harness] run-live overall_status={result['review_summary']['overall_status']}") + print(f"[truth-harness] artifacts={output_dir}") + return 0 + + +def build_parser() -> argparse.ArgumentParser: + parser = argparse.ArgumentParser(description="Strict sequential truth harness for real assistant dialogue scenarios") + subparsers = parser.add_subparsers(dest="command", required=True) + + bootstrap = subparsers.add_parser( + "bootstrap", help="Create an exact-question truth harness spec from a technical export" + ) + bootstrap.add_argument("--export", required=True) + bootstrap.add_argument("--output", required=True) + bootstrap.add_argument("--scenario-id", required=True) + bootstrap.add_argument("--domain", required=True) + bootstrap.add_argument("--title") + bootstrap.set_defaults(func=handle_bootstrap) + + review_export_cmd = subparsers.add_parser( + "review-export", help="Review an existing technical export against a truth harness spec" + ) + review_export_cmd.add_argument("--spec", required=True) + review_export_cmd.add_argument("--export") + review_export_cmd.add_argument("--output-dir") + review_export_cmd.set_defaults(func=handle_review_export) + + run_live_cmd = subparsers.add_parser( + "run-live", + help="Run the exact scenario live, strictly sequentially, with no injected carryover context", + ) + run_live_cmd.add_argument("--spec", required=True) + run_live_cmd.add_argument("--output-dir") + run_live_cmd.add_argument("--backend-url", default=dcl.DEFAULT_BACKEND_URL) + run_live_cmd.add_argument("--prompt-version", default=dcl.DEFAULT_PROMPT_VERSION) + run_live_cmd.add_argument("--llm-provider", default=dcl.DEFAULT_LLM_PROVIDER) + run_live_cmd.add_argument("--llm-model", default=dcl.DEFAULT_LLM_MODEL) + run_live_cmd.add_argument("--llm-base-url", default=dcl.DEFAULT_LLM_BASE_URL) + run_live_cmd.add_argument("--llm-api-key", default=dcl.DEFAULT_LLM_API_KEY) + run_live_cmd.add_argument("--temperature", type=float, default=dcl.DEFAULT_TEMPERATURE) + run_live_cmd.add_argument("--max-output-tokens", type=int, default=dcl.DEFAULT_MAX_OUTPUT_TOKENS) + run_live_cmd.add_argument("--timeout-seconds", type=int, default=120) + run_live_cmd.add_argument("--use-mock", action="store_true") + run_live_cmd.set_defaults(func=handle_run_live) + + return parser + + +def main() -> int: + parser = build_parser() + args = parser.parse_args() + return int(args.func(args)) + + +if __name__ == "__main__": + raise SystemExit(main())