From bbc257fd6c644ef4ea3186e9d11d0221ae0510b5 Mon Sep 17 00:00:00 2001 From: dctouch Date: Thu, 21 May 2026 09:00:08 +0300 Subject: [PATCH] =?UTF-8?q?=D0=A3=D1=81=D0=B8=D0=BB=D0=B8=D1=82=D1=8C=20an?= =?UTF-8?q?swer=20contract=20=D0=B8=20=D0=B0=D0=B3=D0=B5=D0=BD=D1=82=D0=BD?= =?UTF-8?q?=D1=8B=D0=B9=20=D0=B0=D1=83=D0=B4=D0=B8=D1=82=20=D0=B4=D0=BB?= =?UTF-8?q?=D1=8F=20phase105?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../backend/dist/routes/autoRuns.js | 9 +- .../services/address_runtime/composeStage.js | 79 +++++++- .../counterpartyAnalyticsReplyBuilders.js | 50 ++++- .../address_runtime/inventoryReplyBuilders.js | 8 +- .../assistantMcpDiscoveryResponseCandidate.js | 23 +++ .../assistantMcpDiscoveryTurnInputAdapter.js | 135 +++++++++++--- .../services/assistantTurnMeaningPolicy.js | 125 ++++++++++--- .../backend/dist/services/routeHintAdapter.js | 28 ++- llm_normalizer/backend/src/routes/autoRuns.ts | 9 +- .../services/address_runtime/composeStage.ts | 92 ++++++++- .../counterpartyAnalyticsReplyBuilders.ts | 53 +++++- .../address_runtime/inventoryReplyBuilders.ts | 8 +- .../assistantMcpDiscoveryResponseCandidate.ts | 34 ++++ .../assistantMcpDiscoveryTurnInputAdapter.ts | 176 +++++++++++++----- .../services/assistantTurnMeaningPolicy.ts | 89 ++++++++- .../backend/src/services/routeHintAdapter.ts | 35 +++- .../tests/addressQueryRuntimeM23.test.ts | 3 + .../addressReplyBuildersRegression.test.ts | 82 ++++++++ ...istantMcpDiscoveryTurnInputAdapter.test.ts | 101 ++++++++++ .../tests/assistantTurnMeaningPolicy.test.ts | 28 +++ ...counterpartyAnalyticsReplyBuilders.test.ts | 3 +- .../backend/tests/routeHintAdapter.test.ts | 59 ++++++ .../frontend/dist/assets/index-6meEannb.js | 24 --- .../frontend/dist/assets/index-9w9i5XPJ.js | 25 +++ ...{index-BURU4_Sm.css => index-DW_SonhM.css} | 2 +- llm_normalizer/frontend/dist/index.html | 4 +- .../src/components/AutoRunsHistoryPanel.tsx | 128 +++++++++++-- llm_normalizer/frontend/src/state/types.ts | 1 + llm_normalizer/frontend/src/styles.css | 15 ++ .../frontend/src/utils/conversationExport.ts | 141 ++++++++++++++ scripts/domain_case_loop.py | 1 + scripts/domain_truth_harness.py | 2 + scripts/review_assistant_stage1_run.py | 65 +++++++ scripts/test_review_assistant_stage1_run.py | 80 ++++++++ 34 files changed, 1533 insertions(+), 184 deletions(-) delete mode 100644 llm_normalizer/frontend/dist/assets/index-6meEannb.js create mode 100644 llm_normalizer/frontend/dist/assets/index-9w9i5XPJ.js rename llm_normalizer/frontend/dist/assets/{index-BURU4_Sm.css => index-DW_SonhM.css} (54%) diff --git a/llm_normalizer/backend/dist/routes/autoRuns.js b/llm_normalizer/backend/dist/routes/autoRuns.js index d248270..4718224 100644 --- a/llm_normalizer/backend/dist/routes/autoRuns.js +++ b/llm_normalizer/backend/dist/routes/autoRuns.js @@ -1008,7 +1008,8 @@ function loadSessionDialog(runId, caseId) { text: toStringSafe(item.text) ?? "", created_at: toStringSafe(item.created_at), trace_id: toStringSafe(item.trace_id), - reply_type: toStringSafe(item.reply_type) + reply_type: toStringSafe(item.reply_type), + debug: item.debug ?? null })); const turns = toArray(record.turns) .map((item) => toRecord(item)) @@ -1076,7 +1077,8 @@ function buildFallbackDialog(run, caseId) { text: userText, created_at: null, trace_id: null, - reply_type: null + reply_type: null, + debug: null }, { message_id: null, @@ -1084,7 +1086,8 @@ function buildFallbackDialog(run, caseId) { text: assistantSummaryParts.join("\n"), created_at: null, trace_id: toStringSafe(targetCase.trace_id), - reply_type: toStringSafe(targetCase.reply_type) + reply_type: toStringSafe(targetCase.reply_type), + debug: null } ], decomposition: [], diff --git a/llm_normalizer/backend/dist/services/address_runtime/composeStage.js b/llm_normalizer/backend/dist/services/address_runtime/composeStage.js index 343bfe3..4208c9a 100644 --- a/llm_normalizer/backend/dist/services/address_runtime/composeStage.js +++ b/llm_normalizer/backend/dist/services/address_runtime/composeStage.js @@ -365,6 +365,63 @@ function bankOperationEvidenceLine(rows, preferredDirection = null) { } return `Основание 1С: ${parts.join("; ")}.`; } +function classifyBankOperationSemanticBucket(row) { + const text = [ + row.registrator, + row.operation_kind, + row.payment_purpose, + row.contract, + row.comment + ] + .map((item) => String(item ?? "").toLowerCase()) + .join(" "); + if (/(?:комисс|тариф|эквайр|обслуживан)/iu.test(text)) { + return "commission"; + } + if (/(?:депозит|кредит|займ|овердрафт|процент|ссуд)/iu.test(text)) { + return "deposit_or_credit"; + } + if (/(?:налог|ндс|взнос|бюджет|фнс|пфр|страхов)/iu.test(text)) { + return "tax_or_budget"; + } + if (/(?:возврат|перевод|перечислен|переброс|пополн|инкасс|перенос)/iu.test(text)) { + return "transfer_or_return"; + } + return "other"; +} +function bankOperationSemanticBucketLabel(bucket) { + if (bucket === "commission") { + return "комиссии и банковое обслуживание"; + } + if (bucket === "deposit_or_credit") { + return "депозиты, кредиты или проценты"; + } + if (bucket === "tax_or_budget") { + return "налоги и бюджетные платежи"; + } + if (bucket === "transfer_or_return") { + return "переводы, возвраты или перебросы"; + } + return "прочие банковские операции"; +} +function summarizeBankOperationSemantics(rows) { + if (rows.length === 0) { + return null; + } + const counts = new Map(); + for (const row of rows) { + const bucket = classifyBankOperationSemanticBucket(row); + counts.set(bucket, (counts.get(bucket) ?? 0) + 1); + } + const ranked = Array.from(counts.entries()) + .sort((left, right) => right[1] - left[1]) + .slice(0, 3); + if (ranked.length === 0) { + return null; + } + const parts = ranked.map(([bucket, count]) => `${bankOperationSemanticBucketLabel(bucket)} — ${count}`); + return `По смыслу это скорее финансовый/банковский контур: ${parts.join("; ")}.`; +} function bankRoleBoundaryLine(userMessage, rows) { const incomingBoundary = hasBankIncomingRoleBoundaryQuestion(userMessage); const outgoingBoundary = hasBankOutgoingRoleBoundaryQuestion(userMessage); @@ -3931,13 +3988,31 @@ function composeFactualReplyBody(intent, rows, options = {}) { .filter((item) => Boolean(item))); const counterparty = resolvePreferredCounterpartyDisplayLabel(options.counterpartyHint, rowCounterparties); const roleBoundary = bankRoleBoundaryLine(options.userMessage, rows); - const visibleRows = rows.slice(0, Math.min(rows.length, 5)); + const visibleRows = [...rows] + .sort((left, right) => Math.abs(right.amount ?? 0) - Math.abs(left.amount ?? 0) || + (String(right.period ?? "").localeCompare(String(left.period ?? ""), "ru"))) + .slice(0, Math.min(rows.length, 5)); + const semanticSummary = summarizeBankOperationSemantics(rows); + const compactEvidenceRows = visibleRows.map((row, index) => { + const direction = bankOperationDirectionLabel(bankOperationDirection(row)); + const amount = formatMoneyRub(row.amount ?? 0); + const period = row.period ? formatDateRu(row.period) : "дата не указана"; + const operationKind = String(row.operation_kind ?? "").trim(); + const paymentPurpose = String(row.payment_purpose ?? "").trim(); + const detail = operationKind || paymentPurpose + ? ` | ${[operationKind, paymentPurpose].filter(Boolean).join("; ")}` + : ""; + return `${index + 1}. ${period} | ${direction} | ${amount}${detail}`; + }); const lines = [ `Коротко: найдено банковских операций${counterparty ? ` по ${counterparty}` : " по контрагенту"} — ${rows.length}.`, summarizeBankOperationDirections(rows), roleBoundary ?? "Показываю подтвержденные банковские операции из текущего среза.", bankOperationEvidenceLine(rows, preferredBankEvidenceDirection(options.userMessage)), - ...formatTopRows(visibleRows, visibleRows.length) + ...(semanticSummary ? [semanticSummary] : []), + "Примеры строк 1С:", + ...compactEvidenceRows, + "Следующий шаг: могу отдельно разложить назначения платежа, договоры или отделить банковский контур от клиентского/поставщицкого." ]; if (rows.length > visibleRows.length) { lines.push(`Показаны первые ${visibleRows.length} из ${rows.length}; полный список остается в подтвержденном срезе.`); diff --git a/llm_normalizer/backend/dist/services/address_runtime/counterpartyAnalyticsReplyBuilders.js b/llm_normalizer/backend/dist/services/address_runtime/counterpartyAnalyticsReplyBuilders.js index 1893574..7735b39 100644 --- a/llm_normalizer/backend/dist/services/address_runtime/counterpartyAnalyticsReplyBuilders.js +++ b/llm_normalizer/backend/dist/services/address_runtime/counterpartyAnalyticsReplyBuilders.js @@ -30,6 +30,23 @@ function findFocusedCounterpartyValuePoint(profileRows, counterpartyHint, deps) } return profileRows.length === 1 ? profileRows[0] : null; } +function hasProfitAmbiguityCue(normalizedQuestion) { + return /(?:заработ|прибыл|прибыль|доход|выручк)/iu.test(normalizedQuestion); +} +function buildCashflowBoundaryLine(isSupplier) { + return isSupplier + ? "Граница ответа: это подтвержденный денежный поток по поставщику, а не итоговая задолженность." + : "Граница ответа: это подтвержденный денежный поток по поступлениям, а не чистая прибыль."; +} +function buildCashflowNextStepLine(isSupplier, normalizedQuestion) { + if (isSupplier) { + return "Следующий шаг: могу отдельно показать остаток долга, просрочку или расшифровку по документам."; + } + if (hasProfitAmbiguityCue(normalizedQuestion)) { + return "Следующий шаг: могу отдельно проверить чистую прибыль по закрытию 90/91/99."; + } + return "Следующий шаг: могу разложить поток по месяцам, документам или контрагентам."; +} function composeCounterpartyAnalyticsReply(intent, rows, options = {}, deps) { if (intent === "counterparty_population_and_roles") { const rowsByMarker = groupRowsByMarker(rows); @@ -396,6 +413,8 @@ function composeCounterpartyAnalyticsReply(intent, rows, options = {}, deps) { /(?:клиент|заказчик|покупател|контрагент|customer|client|counterparty|buyer)/iu.test(normalizedQuestion); const semanticSingleBestCounterparty = focus === "top_by_total" && hasSingleBestCounterpartyCue && !asksExplicitRankingList; const effectiveLimit = asksSingleBestCounterparty || semanticSingleBestCounterparty ? 1 : limit; + const cashflowBoundaryLine = buildCashflowBoundaryLine(isSupplier); + const cashflowNextStepLine = buildCashflowNextStepLine(isSupplier, normalizedQuestion); const byCounterparty = new Map(); const byYear = new Map(); const deals = []; @@ -490,10 +509,12 @@ function composeCounterpartyAnalyticsReply(intent, rows, options = {}, deps) { ? `за период ${deps.formatDateRu(options.periodFrom)}..${deps.formatDateRu(options.periodTo)}` : "за доступное время"; const directAnswerLine = isSupplier - ? `Оборот по ${focusedCounterparty.name} ${periodLabel}: ${deps.formatMoneyRub(focusedCounterparty.total)} по ${focusedCounterparty.ops} подтвержденным исходящим операциям. Это денежный поток по поставщику, а не итоговая задолженность.` - : `Оборот по ${focusedCounterparty.name} ${periodLabel}: ${deps.formatMoneyRub(focusedCounterparty.total)} по ${focusedCounterparty.ops} подтвержденным входящим операциям. Это денежный поток от клиента, а не чистая прибыль.`; + ? `Оборот по ${focusedCounterparty.name} ${periodLabel}: ${deps.formatMoneyRub(focusedCounterparty.total)} по ${focusedCounterparty.ops} подтвержденным исходящим операциям.` + : `Оборот по ${focusedCounterparty.name} ${periodLabel}: ${deps.formatMoneyRub(focusedCounterparty.total)} по ${focusedCounterparty.ops} подтвержденным входящим операциям.`; const summaryLines = [ directAnswerLine, + cashflowBoundaryLine, + ...(cashflowNextStepLine ? [cashflowNextStepLine] : []), "", "Подтверждение:", `- Контрагент в выборке: ${focusedCounterparty.name}.`, @@ -511,11 +532,10 @@ function composeCounterpartyAnalyticsReply(intent, rows, options = {}, deps) { const periodLine = options.periodFrom && options.periodTo ? `За период ${deps.formatDateRu(options.periodFrom)}..${deps.formatDateRu(options.periodTo)} подтверждено ${deps.formatMoneyRub(totalFlow)} ${isSupplier ? "исходящих выплат" : "входящих поступлений"}.` : `За все доступное время подтверждено ${deps.formatMoneyRub(totalFlow)} ${isSupplier ? "исходящих выплат" : "входящих поступлений"}.`; - const directAnswerLine = isSupplier - ? periodLine - : `${periodLine} Это денежный поток от клиентов, а не чистая прибыль.`; const summaryLines = [ - directAnswerLine, + periodLine, + cashflowBoundaryLine, + ...(cashflowNextStepLine ? [cashflowNextStepLine] : []), "", "Подтверждение:", `- Операций в выборке: ${totalOperations}.`, @@ -538,11 +558,17 @@ function composeCounterpartyAnalyticsReply(intent, rows, options = {}, deps) { const strongestYear = visible[0]; const directAnswerLine = isSupplier ? `Самый крупный год по подтвержденным выплатам: ${strongestYear.year} (${deps.formatMoneyRub(strongestYear.total)} по ${strongestYear.ops} операциям).` - : `Самый доходный год по подтвержденным поступлениям: ${strongestYear.year} (${deps.formatMoneyRub(strongestYear.total)} по ${strongestYear.ops} операциям). Это денежный поток, а не чистая прибыль.`; + : `Самый доходный год по подтвержденным поступлениям: ${strongestYear.year} (${deps.formatMoneyRub(strongestYear.total)} по ${strongestYear.ops} операциям).`; const heading = isSupplier ? `Топ-${visible.length} лет по сумме выплат:` : `Топ-${visible.length} лет по сумме поступлений:`; lines.unshift(heading); + if (!isSupplier) { + lines.unshift(cashflowBoundaryLine); + if (cashflowNextStepLine) { + lines.unshift(cashflowNextStepLine); + } + } lines.unshift(directAnswerLine); lines.push(...visible.map((item, index) => `${index + 1}. ${item.year} | сумма: ${deps.formatMoneyRub(item.total)} | операций: ${item.ops} | контрагентов: ${item.counterparties.size} | максимальная разовая сумма: ${deps.formatMoneyRub(item.maxSingle)}`)); } @@ -622,11 +648,17 @@ function composeCounterpartyAnalyticsReply(intent, rows, options = {}, deps) { const directAnswerLine = singleCandidateOnly ? isSupplier ? `В выбранном срезе найден один поставщик: ${leadingCounterparty.name} (${deps.formatMoneyRub(leadingCounterparty.total)} по ${leadingCounterparty.ops} операциям). Это не полноценный сравнительный рейтинг.` - : `В выбранном срезе найден один клиент: ${leadingCounterparty.name} (${deps.formatMoneyRub(leadingCounterparty.total)} по ${leadingCounterparty.ops} операциям). Это не полноценный сравнительный рейтинг; сумма является денежным потоком, а не чистой прибылью.` + : `В выбранном срезе найден один клиент: ${leadingCounterparty.name} (${deps.formatMoneyRub(leadingCounterparty.total)} по ${leadingCounterparty.ops} операциям). Это не полноценный сравнительный рейтинг.` : isSupplier ? `Крупнейший поставщик по подтвержденным выплатам ${rankingPeriodLabel}: ${leadingCounterparty.name} (${deps.formatMoneyRub(leadingCounterparty.total)} по ${leadingCounterparty.ops} операциям).` - : `Самый доходный клиент ${rankingPeriodLabel} по подтвержденным поступлениям: ${leadingCounterparty.name} (${deps.formatMoneyRub(leadingCounterparty.total)} по ${leadingCounterparty.ops} операциям). Это денежный поток, а не чистая прибыль.`; + : `Самый доходный клиент ${rankingPeriodLabel} по подтвержденным поступлениям: ${leadingCounterparty.name} (${deps.formatMoneyRub(leadingCounterparty.total)} по ${leadingCounterparty.ops} операциям).`; lines.unshift(directAnswerLine); + if (!isSupplier) { + lines.splice(1, 0, cashflowBoundaryLine); + if (cashflowNextStepLine) { + lines.splice(2, 0, cashflowNextStepLine); + } + } } lines.push(...visible.map((item, index) => { const avgCheck = item.ops > 0 ? item.total / item.ops : 0; diff --git a/llm_normalizer/backend/dist/services/address_runtime/inventoryReplyBuilders.js b/llm_normalizer/backend/dist/services/address_runtime/inventoryReplyBuilders.js index b4d80b4..363f7a8 100644 --- a/llm_normalizer/backend/dist/services/address_runtime/inventoryReplyBuilders.js +++ b/llm_normalizer/backend/dist/services/address_runtime/inventoryReplyBuilders.js @@ -87,10 +87,9 @@ function composeInventoryReply(intent, rows, options, deps) { const positions = deps.buildInventoryOnHandAggregate(rows, asOfDate); const uniqueItems = deps.uniqueStrings(positions.map((item) => item.item)); const uniqueWarehouses = deps.uniqueStrings(positions.map((item) => String(item.warehouse ?? "").trim()).filter((item) => item.length > 0)); - const totalQuantity = positions.reduce((sum, item) => sum + item.quantity, 0); const totalAmount = positions.reduce((sum, item) => sum + item.amount, 0); const directAnswerLine = positions.length > 0 - ? `На ${deps.formatDateRu(asOfDate)} на складе подтверждено ${deps.formatNumberWithDots(positions.length)} позиций с остатком на ${deps.formatMoneyRub(totalAmount)}.` + ? `На ${deps.formatDateRu(asOfDate)} на складе подтверждено ${deps.formatNumberWithDots(positions.length)} позиций на ${deps.formatMoneyRub(totalAmount)}.` : `На ${deps.formatDateRu(asOfDate)} подтвержденных товарных остатков по счету 41.01 не найдено.`; const lines = [directAnswerLine]; if (positions.length > 0) { @@ -115,11 +114,14 @@ function composeInventoryReply(intent, rows, options, deps) { `Позиции с остатком: ${deps.formatNumberWithDots(positions.length)}.`, `Уникальных товаров: ${deps.formatNumberWithDots(uniqueItems.length)}.`, `Уникальных складов: ${deps.formatNumberWithDots(uniqueWarehouses.length)}.`, - `Суммарное количество: ${deps.formatNumberWithDots(totalQuantity, 3)}.` + "Общее количество не свожу в один управленческий показатель, потому что в остатках смешаны разнородные позиции." ]); if (rows.length !== positions.length) { lines.push(`- Проверенных строк движения: ${deps.formatNumberWithDots(rows.length)}.`); } + if (positions.length > 0) { + lines.push("- Следующий шаг: могу раскрыть полный список, разложить остатки по складам или сравнить с другой датой."); + } return positions.length > 0 ? (0, replyContracts_1.buildFactualListReply)(lines, (0, replyContracts_1.buildConfirmedBalanceSemantics)("strong")) : (0, replyContracts_1.buildFactualSummaryReply)(lines, (0, replyContracts_1.buildConfirmedBalanceSemantics)("medium")); diff --git a/llm_normalizer/backend/dist/services/assistantMcpDiscoveryResponseCandidate.js b/llm_normalizer/backend/dist/services/assistantMcpDiscoveryResponseCandidate.js index 7e6fd8c..e26b86d 100644 --- a/llm_normalizer/backend/dist/services/assistantMcpDiscoveryResponseCandidate.js +++ b/llm_normalizer/backend/dist/services/assistantMcpDiscoveryResponseCandidate.js @@ -942,6 +942,29 @@ function buildCompactBusinessOverviewReply(entryPoint, draft) { return joinBusinessReplyLines(lines); } if (rankingNeed) { + const explicitPeriodRankingOverview = period && + !/(?:все\s+доступное|все\s+время|all\s+time)/iu.test(period) && + (incomingAmount || outgoingAmount || netAmount); + if (explicitPeriodRankingOverview) { + lines.push(`Коротко: ${organizationPrefix}${period} денежная картина подтверждена по найденным строкам 1С.`); + lines.push(`Деньги: входящие ${incomingAmount ?? "0 руб."}, исходящие ${outgoingAmount ?? "0 руб."}, расчетное операционное нетто ${sentenceAmount(netAmount) ?? netAmount ?? "0 руб."}.`); + if (customerName && customerAmount) { + lines.push(topCustomerLooksFinancial + ? `Топ входящих: 1. ${customerName} — ${sentenceAmount(customerAmount) ?? customerAmount}. Это финансовый/банковский контур, не считаю его клиентской выручкой без назначения платежа.${nonFinancialCustomer ? ` 2. Крупнейший небанковский входящий контрагент: ${nonFinancialCustomer}.` : ""}` + : `Крупнейший входящий контрагент: ${customerName} — ${sentenceAmount(customerAmount) ?? customerAmount}.`); + } + if (topSupplier) { + lines.push(topSupplierLooksFinancial + ? `Топ исходящих: 1. ${topSupplier}. Это финансовый/банковский контур, не считаю его обычным поставщиком без назначения платежа и договора.${nonFinancialSupplier ? ` 2. Крупнейший небанковский получатель исходящих денег: ${nonFinancialSupplier}.` : ""}` + : `Крупнейший получатель исходящих денег: ${topSupplier}.`); + } + lines.push(`Вывод: по движению денег период ${netDirection}; это не чистая прибыль и не бухгалтерский финрезультат.`); + if (requestedFinancialBoundaryLine) { + lines.push(requestedFinancialBoundaryLine); + } + lines.push("Следующий шаг: могу отдельно посчитать чистую прибыль через закрытие 90/91/99 или разложить этот период по контрагентам."); + return joinBusinessReplyLines(lines); + } const incomingLeader = strongestIncomingYear(overview); const canRankYearlyNet = !limitLine; const netLeader = canRankYearlyNet ? strongestNetYear(overview) : null; diff --git a/llm_normalizer/backend/dist/services/assistantMcpDiscoveryTurnInputAdapter.js b/llm_normalizer/backend/dist/services/assistantMcpDiscoveryTurnInputAdapter.js index 6913407..cdab88a 100644 --- a/llm_normalizer/backend/dist/services/assistantMcpDiscoveryTurnInputAdapter.js +++ b/llm_normalizer/backend/dist/services/assistantMcpDiscoveryTurnInputAdapter.js @@ -1139,13 +1139,38 @@ function buildAssistantMcpDiscoveryTurnInput(input) { const rawEffectiveText = toNonEmptyString(input.effectiveMessage); const repairedUserText = rawUserText ? (0, addressTextRepair_1.repairAddressMojibakeText)(rawUserText) : null; const repairedEffectiveText = rawEffectiveText ? (0, addressTextRepair_1.repairAddressMojibakeText)(rawEffectiveText) : null; + const rawUserSignalSourceText = repairedUserText ?? rawUserText ?? ""; const rawSignalSourceText = `${repairedUserText ?? rawUserText ?? ""} ${repairedEffectiveText ?? rawEffectiveText ?? ""}`.trim(); const rawEntitySourceText = repairedUserText ?? rawUserText ?? repairedEffectiveText ?? rawEffectiveText ?? rawSignalSourceText; + const rawUserEntitySourceText = rawUserSignalSourceText || rawEntitySourceText; + const rawUserTextOnly = compactLower(rawUserSignalSourceText); + const rawAssistantEntityCandidates = collectEntityCandidates(assistantTurnMeaning?.explicit_entity_candidates); + const rawUserPrimaryBusinessOverviewSignal = hasBusinessOverviewSignal(rawUserTextOnly); + const rawUserLifecyclePivotTextSignal = !rawUserPrimaryBusinessOverviewSignal && hasLifecycleSignal(rawUserTextOnly); + const rawUserBidirectionalValueFlowPivotTextSignal = !rawUserPrimaryBusinessOverviewSignal && + !rawUserLifecyclePivotTextSignal && + hasBidirectionalValueFlowSignal(rawUserTextOnly); + const rawUserScopedEntityCandidate = rawUserSignalSourceText + ? rawScopedEntityCandidateFromText(rawUserEntitySourceText) + : null; + const rawUserCounterpartyBidirectionalOverride = Boolean(rawUserBidirectionalValueFlowPivotTextSignal && + (rawUserScopedEntityCandidate || + predecomposeEntities.counterparty || + rawAssistantEntityCandidates.find((candidate) => !isInvalidEntityCandidate(candidate)))); const rawText = compactLower(rawSignalSourceText); const rawReferentialDocumentExclusionSignal = hasReferentialDocumentExclusionFollowupSignal(repairedUserText ?? rawUserText ?? ""); - const rawPrimaryBusinessOverviewSignal = hasBusinessOverviewSignal(rawText); + const rawPrimaryBusinessOverviewSignal = hasBusinessOverviewSignal(rawText) && !rawUserCounterpartyBidirectionalOverride; const explicitVatQuestionSignal = hasExplicitVatQuestionSignal(rawText); const explicitVatMovementEvidenceSignal = hasExplicitVatMovementEvidenceSignal(rawText); + const rawLifecyclePivotTextSignal = !rawPrimaryBusinessOverviewSignal && hasLifecycleSignal(rawText); + const rawBidirectionalValueFlowPivotTextSignal = !rawPrimaryBusinessOverviewSignal && + !rawLifecyclePivotTextSignal && + hasBidirectionalValueFlowSignal(rawText); + const rawValueFlowPivotTextSignal = !rawPrimaryBusinessOverviewSignal && + !rawLifecyclePivotTextSignal && + (hasValueFlowSignal(rawText) || + hasValueRankingSignal(rawText) || + rawBidirectionalValueFlowPivotTextSignal); const explicitVatSuppressesBusinessOverviewContinuation = Boolean(explicitVatQuestionSignal && !rawPrimaryBusinessOverviewSignal); const businessOverviewContinuationSignal = hasBusinessOverviewFollowupSeed(followupSeed) && hasBusinessOverviewContinuationSignal(rawText) && @@ -1163,7 +1188,7 @@ function buildAssistantMcpDiscoveryTurnInput(input) { hasMetadataSignal(rawText); const rawEntityResolutionSignal = !rawLifecycleSignal && !rawValueFlowSignal && !rawMetadataSignal && hasEntityResolutionSignal(rawText); const rawPayoutSignal = rawValueFlowSignal && !rawBidirectionalValueFlowSignal && hasPayoutSignal(rawText); - const rawValueFlowAggregateQuestionSignal = rawValueFlowSignal && hasValueFlowAggregateQuestionSignal(rawText); + const rawValueFlowAggregateQuestionSignal = (rawValueFlowSignal || rawValueFlowPivotTextSignal) && hasValueFlowAggregateQuestionSignal(rawText); const monthlyAggregationSignal = hasMonthlyAggregationSignal(rawText); const rawAllTimeScopeSignal = hasAllTimeScopeHint(rawText); const dateScopeSignalText = stripNegatedTaxDateScopeClauses(rawText); @@ -1216,6 +1241,32 @@ function buildAssistantMcpDiscoveryTurnInput(input) { : profitMarginBusinessOverviewSignal ? "profit_margin_boundary" : "broad_evaluation"; + const assistantTurnMeaningDateScope = toNonEmptyString(assistantTurnMeaning?.explicit_date_scope); + const rawAssistantTurnMeaningOrganizationScope = toNonEmptyString(assistantTurnMeaning?.explicit_organization_scope); + const assistantTurnMeaningOrganizationScope = isReferentialOrganizationPlaceholder(rawAssistantTurnMeaningOrganizationScope) + ? null + : rawAssistantTurnMeaningOrganizationScope; + const rawOrganizationMentionSignal = hasOrganizationScopeSignalUtf8(rawText); + const rawOrganizationScope = extractOrganizationScopeFromRawText(rawUserText ?? rawEffectiveText ?? rawSignalSourceText); + const currentTurnFreshOrganizationScope = predecomposeEntities.organization ?? rawOrganizationScope; + const currentTurnOrganizationScope = currentTurnFreshOrganizationScope ?? assistantTurnMeaningOrganizationScope; + const predecomposeOrganizationMirrorsCounterparty = sameScopedName(predecomposeEntities.counterparty, predecomposeEntities.organization); + const organizationMirrorsPredecomposeCounterpartyForPivot = Boolean(sameScopedName(predecomposeEntities.counterparty, assistantTurnMeaningOrganizationScope) || + sameScopedName(predecomposeEntities.counterparty, currentTurnOrganizationScope) || + predecomposeOrganizationMirrorsCounterparty); + const normalizedPredecomposeCounterpartyForPivot = organizationMirrorsPredecomposeCounterpartyForPivot + ? null + : normalizeFollowupCounterpartyCandidate(predecomposeEntities.counterparty); + const rawExplicitCounterpartyPivotCandidate = rawScopedEntityCandidate ?? + rawAssistantEntityCandidates.find((candidate) => !isInvalidEntityCandidate(candidate) && + !sameScopedName(candidate, currentTurnOrganizationScope)) ?? + normalizedPredecomposeCounterpartyForPivot ?? + null; + const businessOverviewCounterpartyValueFlowPivot = Boolean(businessOverviewContinuationSignal && + !rawPrimaryBusinessOverviewSignal && + rawValueFlowPivotTextSignal && + rawExplicitCounterpartyPivotCandidate && + (rawTopicSwitchSignal || rawValueFlowAggregateQuestionSignal)); const businessOverviewUnsupportedFamily = inventoryReserveBusinessOverviewSignal ? "inventory_reserve_liquidation_boundary" : debtDueDateBusinessOverviewSignal @@ -1225,8 +1276,8 @@ function buildAssistantMcpDiscoveryTurnInput(input) { : profitMarginBusinessOverviewSignal ? "profit_margin_boundary" : "broad_business_evaluation"; - const businessOverviewSignal = rawBusinessOverviewSignal || - seededBusinessOverviewSignal; + const businessOverviewSignal = !businessOverviewCounterpartyValueFlowPivot && + (rawBusinessOverviewSignal || seededBusinessOverviewSignal); const businessOverviewSeparateCounterpartySignal = Boolean(businessOverviewSignal && hasBusinessOverviewSeparateCounterpartySignal(rawText)); const businessOverviewSeparateCounterpartyCandidate = businessOverviewSeparateCounterpartySignal ? businessOverviewSeparateCounterpartyCandidateFromText(rawText) @@ -1244,15 +1295,6 @@ function buildAssistantMcpDiscoveryTurnInput(input) { hasSimpleMovementLanePivotSignal(rawText) || hasMovementEvidenceFollowupSignal(rawText) || hasPronounMovementEvidenceFollowupSignal(rawText); - const assistantTurnMeaningDateScope = toNonEmptyString(assistantTurnMeaning?.explicit_date_scope); - const rawAssistantTurnMeaningOrganizationScope = toNonEmptyString(assistantTurnMeaning?.explicit_organization_scope); - const assistantTurnMeaningOrganizationScope = isReferentialOrganizationPlaceholder(rawAssistantTurnMeaningOrganizationScope) - ? null - : rawAssistantTurnMeaningOrganizationScope; - const rawOrganizationMentionSignal = hasOrganizationScopeSignalUtf8(rawText); - const rawOrganizationScope = extractOrganizationScopeFromRawText(rawUserText ?? rawEffectiveText ?? rawSignalSourceText); - const currentTurnFreshOrganizationScope = predecomposeEntities.organization ?? rawOrganizationScope; - const currentTurnOrganizationScope = currentTurnFreshOrganizationScope ?? assistantTurnMeaningOrganizationScope; const followupCounterpartyIsMetadataOrganizationScope = Boolean(followupSeed.subjectResolutionOptional && followupSeed.counterparty && (followupSeed.metadataScopeHint || @@ -1288,7 +1330,6 @@ function buildAssistantMcpDiscoveryTurnInput(input) { const rawOpenScopeValueFlowOrganizationSignal = Boolean(rawValueFlowSignal && !rawBidirectionalValueFlowSignal && explicitOrganizationScopeSignal); - const predecomposeOrganizationMirrorsCounterparty = sameScopedName(predecomposeEntities.counterparty, predecomposeEntities.organization); const organizationMirrorsPredecomposeCounterparty = Boolean((rawBidirectionalValueFlowSignal || hasValueRankingSignal(rawText) || rawOpenScopeValueFlowOrganizationSignal || @@ -1564,14 +1605,25 @@ function buildAssistantMcpDiscoveryTurnInput(input) { const lifecycleSignal = !businessOverviewSignal && (rawLifecycleSignal || seededDomain === "counterparty_lifecycle"); const bidirectionalValueFlowSignal = !businessOverviewSignal && !lifecycleSignal && - (rawBidirectionalValueFlowSignal || seededAction === "net_value_flow"); + ((businessOverviewCounterpartyValueFlowPivot + ? rawBidirectionalValueFlowPivotTextSignal + : rawBidirectionalValueFlowSignal) || + seededAction === "net_value_flow"); const valueFlowSignal = !businessOverviewSignal && !lifecycleSignal && !metadataGroundedMovementLaneApplicable && - (rawValueFlowSignal || seededDomain === "counterparty_value"); + ((businessOverviewCounterpartyValueFlowPivot + ? rawValueFlowPivotTextSignal + : rawValueFlowSignal) || + seededDomain === "counterparty_value"); const payoutSignal = valueFlowSignal && !bidirectionalValueFlowSignal && - (rawPayoutSignal || seededAction === "payout"); + ((businessOverviewCounterpartyValueFlowPivot + ? rawValueFlowPivotTextSignal && + !rawBidirectionalValueFlowPivotTextSignal && + hasPayoutSignal(rawText) + : rawPayoutSignal) || + seededAction === "payout"); const semanticDataNeed = metadataAmbiguityLaneClarificationApplicable ? "metadata lane clarification" : semanticNeedFor({ @@ -1579,17 +1631,37 @@ function buildAssistantMcpDiscoveryTurnInput(input) { ? "movements" : businessOverviewSignal ? "business_overview" - : rawDomain ?? seededDomain, + : lifecycleSignal + ? "counterparty_lifecycle" + : valueFlowSignal + ? "counterparty_value" + : rawDomain ?? seededDomain, action: explicitVatMovementEvidenceSignal ? "list_movements" : businessOverviewSignal ? businessOverviewActionFamily - : rawAction ?? seededAction, + : lifecycleSignal + ? "activity_duration" + : valueFlowSignal + ? bidirectionalValueFlowSignal + ? "net_value_flow" + : payoutSignal + ? "payout" + : rawAction ?? seededAction ?? "turnover" + : rawAction ?? seededAction, unsupported: explicitVatMovementEvidenceSignal ? "movement_evidence" : businessOverviewSignal ? businessOverviewUnsupportedFamily - : unsupported ?? seededUnsupported, + : lifecycleSignal + ? "counterparty_lifecycle" + : valueFlowSignal + ? bidirectionalValueFlowSignal + ? "counterparty_bidirectional_value_flow_or_netting" + : payoutSignal + ? "counterparty_payouts_or_outflow" + : seededUnsupported ?? "counterparty_value_or_turnover" + : unsupported ?? seededUnsupported, lifecycleSignal, valueFlowSignal, metadataSignal: rawMetadataSignal || effectiveMetadataFollowupSeedApplicable, @@ -1853,16 +1925,16 @@ function buildAssistantMcpDiscoveryTurnInput(input) { subject_resolution_optional: metadataScopedLaneWithoutSubject || undefined, unsupported_but_understood_family: businessOverviewSignal ? businessOverviewUnsupportedFamily - : unsupported ?? - (lifecycleSignal - ? "counterparty_lifecycle" - : valueFlowSignal - ? bidirectionalValueFlowSignal - ? "counterparty_bidirectional_value_flow_or_netting" - : payoutSignal - ? "counterparty_payouts_or_outflow" - : seededUnsupported ?? "counterparty_value_or_turnover" - : metadataGroundedMovementLaneApplicable + : lifecycleSignal + ? "counterparty_lifecycle" + : valueFlowSignal + ? bidirectionalValueFlowSignal + ? "counterparty_bidirectional_value_flow_or_netting" + : payoutSignal + ? "counterparty_payouts_or_outflow" + : seededUnsupported ?? "counterparty_value_or_turnover" + : unsupported ?? + (metadataGroundedMovementLaneApplicable ? "movement_evidence" : metadataGroundedDocumentLaneApplicable ? "document_evidence" @@ -2137,6 +2209,9 @@ function buildAssistantMcpDiscoveryTurnInput(input) { if (businessOverviewSignal) { pushReason(reasonCodes, "mcp_discovery_broad_business_evaluation_route_candidate"); } + if (businessOverviewCounterpartyValueFlowPivot) { + pushReason(reasonCodes, "mcp_discovery_business_overview_followup_pivoted_to_counterparty_value_flow"); + } if (businessOverviewContinuationSignal) { pushReason(reasonCodes, "mcp_discovery_business_overview_continuation_from_followup_context"); } diff --git a/llm_normalizer/backend/dist/services/assistantTurnMeaningPolicy.js b/llm_normalizer/backend/dist/services/assistantTurnMeaningPolicy.js index 2182271..01c2fc3 100644 --- a/llm_normalizer/backend/dist/services/assistantTurnMeaningPolicy.js +++ b/llm_normalizer/backend/dist/services/assistantTurnMeaningPolicy.js @@ -133,6 +133,57 @@ function detectCounterpartyTurnoverFamily(text) { entity }; } +function detectScopedCounterpartyEntity(text) { + const patterns = [ + /(?:^|[\s,.;:!?])(?:\u043f\u043e|\u0443|\u0434\u043b\u044f|by|for)\s+(.+?)(?=$|[,.;:!?]|\s+(?:\u0437\u0430|\u043d\u0430|\u0432|\u0432\u043e|\u043a|\u043f\u043e|\u0441\u043a\u043e\u043b\u044c\u043a\u043e|\u0441\u043a\u043e\u043a|\u043a\u0430\u043a|\u043a\u0430\u043a\u043e\u0435|\u043a\u0430\u043a\u043e\u0439|\u043a\u0430\u043a\u0430\u044f|\u043a\u0430\u043a\u0438\u0435|\u043f\u043e\u043b\u0443\u0447\p{L}*|\u0437\u0430\u043f\u043b\u0430\u0442\p{L}*|\u043d\u0435\u0442\u0442\u043e|\u0441\u0430\u043b\u044c\u0434\u043e|\u0434\u0435\u043d\u0435\u0433|\u0434\u0435\u043d\u0435\u0436\p{L}*|\u043f\u043b\u0430\u0442[\u0435\u0451]\u0436\p{L}*|\u0438\u0441\u0445\u043e\u0434\p{L}*|\u0432\u0445\u043e\u0434\p{L}*)(?=$|[\s,.;:!?]))/iu, + /(?:^|[\s,.;:!?])(?:\u043f\u043e|\u0443|\u0434\u043b\u044f|by|for)\s+([\p{L}\d._-]{2,})(?=$|[\s,.;:!?])/iu + ]; + const ignored = new Set([ + "\u0433\u043e\u0434", + "\u0433\u043e\u0434\u0430", + "\u043f\u0435\u0440\u0438\u043e\u0434", + "\u043f\u0435\u0440\u0438\u043e\u0434\u0430", + "\u043c\u0435\u0441\u044f\u0446", + "\u043c\u0435\u0441\u044f\u0446\u0430", + "\u043a\u0432\u0430\u0440\u0442\u0430\u043b", + "\u043a\u0432\u0430\u0440\u0442\u0430\u043b\u0430", + "\u0434\u0435\u043d\u044c\u0433\u0438", + "\u043d\u0435\u0442\u0442\u043e", + "\u0441\u0430\u043b\u044c\u0434\u043e", + "year", + "period", + "month", + "quarter", + "net" + ]); + for (const pattern of patterns) { + const rawEntity = text.match(pattern)?.[1]?.trim() ?? ""; + if (!rawEntity) { + continue; + } + const entity = rawEntity.replace(/^["'«»]+|["'«»]+$/gu, "").trim(); + if (entity.length >= 2 && !ignored.has(entity)) { + return entity; + } + } + return null; +} +function detectCounterpartyBidirectionalValueFlowFamily(text) { + const hasNetCue = /(?:\u043d\u0435\u0442\u0442\u043e|\u0441\u0430\u043b\u044c\u0434\u043e|net\s+(?:flow|cash|payment)|cash\s+net)/iu.test(text); + const hasIncomingCue = /(?:\u043f\u043e\u043b\u0443\u0447\p{L}*|\u0432\u0445\u043e\u0434\p{L}*|\u043f\u043e\u0441\u0442\u0443\u043f\p{L}*|received|incoming)/iu.test(text); + const hasOutgoingCue = /(?:\u0437\u0430\u043f\u043b\u0430\u0442\p{L}*|\u0438\u0441\u0445\u043e\u0434\p{L}*|\u0441\u043f\u0438\u0441\u0430\u043d\p{L}*|paid|outgoing|payment)/iu.test(text); + if (!(hasNetCue || (hasIncomingCue && hasOutgoingCue))) { + return null; + } + const entity = detectScopedCounterpartyEntity(text); + if (!entity) { + return null; + } + return { + family: "counterparty_bidirectional_value_flow_or_netting", + entity + }; +} function hasExplicitCounterpartyValueObject(text) { return /(?:\u043a\u043b\u0438\u0435\u043d\u0442|\u043f\u043e\u043a\u0443\u043f\u0430\u0442\u0435\u043b|\u0437\u0430\u043a\u0430\u0437\u0447\u0438\u043a|\u043a\u043e\u043d\u0442\u0440\u0430\u0433\u0435\u043d\u0442|\u043f\u043e\u0441\u0442\u0430\u0432\u0449\u0438\u043a|\u0434\u043e\u0433\u043e\u0432\u043e\u0440|\u043a\u043e\u043d\u0442\u0440\u0430\u043a\u0442|\u0442\u043e\u0432\u0430\u0440|\u043d\u043e\u043c\u0435\u043d\u043a\u043b\u0430\u0442\u0443\u0440|\u0441\u0434\u0435\u043b\u043a|customer|client|counterparty|supplier|vendor|contract|item|product|deal)/iu.test(text); } @@ -279,14 +330,14 @@ function detectBroadBusinessEvaluation(text) { } return null; } -function buildEntityCandidates(counterpartyTurnover) { - if (!counterpartyTurnover?.entity) { +function buildEntityCandidates(entityFamily) { + if (!entityFamily?.entity) { return []; } return [ { type: "counterparty", - value: counterpartyTurnover.entity, + value: entityFamily.entity, source: "current_turn_loose_entity_tail" } ]; @@ -299,22 +350,30 @@ function createAssistantTurnMeaningPolicy(deps = {}) { const effectiveText = normalizeTurnText(effectiveMessage, deps); const joinedText = fallbackCompactWhitespace(`${rawText} ${effectiveText}`); const supportedIntent = detectSupportedIntent(joinedText, deps); + const counterpartyBidirectionalValueFlow = detectCounterpartyBidirectionalValueFlowFamily(joinedText); const counterpartyTurnover = detectCounterpartyTurnoverFamily(joinedText); const selectedObjectInventoryExact = hasSelectedObjectInventoryExactSignal(joinedText); - const broadBusinessEvaluation = selectedObjectInventoryExact ? null : detectBroadBusinessEvaluation(joinedText); + const broadBusinessEvaluation = selectedObjectInventoryExact || counterpartyBidirectionalValueFlow?.family + ? null + : detectBroadBusinessEvaluation(joinedText); const llmIntent = toNonEmptyString(input?.llmPreDecomposeMeta?.predecomposeContract?.intent, deps); const explicitIntentCandidate = broadBusinessEvaluation?.family ? null : supportedIntent?.intent ?? (llmIntent && llmIntent !== "unknown" ? llmIntent : null); const unsupportedFamily = broadBusinessEvaluation?.family ? broadBusinessEvaluation.family - : !explicitIntentCandidate && counterpartyTurnover?.family - ? counterpartyTurnover.family - : null; + : !explicitIntentCandidate && counterpartyBidirectionalValueFlow?.family + ? counterpartyBidirectionalValueFlow.family + : !explicitIntentCandidate && counterpartyTurnover?.family + ? counterpartyTurnover.family + : null; const reasonCodes = []; if (supportedIntent?.reason) { reasonCodes.push(supportedIntent.reason); } + if (counterpartyBidirectionalValueFlow?.family) { + reasonCodes.push("counterparty_bidirectional_value_flow_current_turn_signal"); + } if (counterpartyTurnover?.family) { reasonCodes.push("counterparty_turnover_current_turn_signal"); } @@ -338,32 +397,39 @@ function createAssistantTurnMeaningPolicy(deps = {}) { ? "inventory" : broadBusinessEvaluation?.family ? "business_summary" - : explicitIntentCandidate?.includes("counterparty") - ? "counterparty" - : counterpartyTurnover?.family + : counterpartyBidirectionalValueFlow?.family + ? "counterparty_value" + : explicitIntentCandidate?.includes("counterparty") ? "counterparty" - : null; + : counterpartyTurnover?.family + ? "counterparty" + : null; const askedActionFamily = explicitIntentCandidate === "receivables_confirmed_as_of_date" || explicitIntentCandidate === "payables_confirmed_as_of_date" || explicitIntentCandidate === "inventory_on_hand_as_of_date" ? "confirmed_snapshot" : broadBusinessEvaluation?.family ? "broad_evaluation" - : explicitIntentCandidate === "customer_revenue_and_payments" || - explicitIntentCandidate === "supplier_payouts_profile" - ? "counterparty_value_or_turnover" - : explicitIntentCandidate === "vat_liability_confirmed_for_tax_period" - ? "confirmed_tax_period" - : explicitIntentCandidate === "vat_payable_confirmed_as_of_date" - ? "confirmed_snapshot" - : explicitIntentCandidate === "vat_payable_forecast" - ? "forecast" - : explicitIntentCandidate === "list_documents_by_counterparty" - ? "list_documents" - : counterpartyTurnover?.family - ? "counterparty_value_or_turnover" - : null; - const staleReplayForbidden = Boolean(unsupportedFamily || broadBusinessEvaluation?.family || (counterpartyTurnover?.entity && !explicitIntentCandidate)); + : counterpartyBidirectionalValueFlow?.family + ? "net_value_flow" + : explicitIntentCandidate === "customer_revenue_and_payments" || + explicitIntentCandidate === "supplier_payouts_profile" + ? "counterparty_value_or_turnover" + : explicitIntentCandidate === "vat_liability_confirmed_for_tax_period" + ? "confirmed_tax_period" + : explicitIntentCandidate === "vat_payable_confirmed_as_of_date" + ? "confirmed_snapshot" + : explicitIntentCandidate === "vat_payable_forecast" + ? "forecast" + : explicitIntentCandidate === "list_documents_by_counterparty" + ? "list_documents" + : counterpartyTurnover?.family + ? "counterparty_value_or_turnover" + : null; + const staleReplayForbidden = Boolean(unsupportedFamily || + broadBusinessEvaluation?.family || + (counterpartyBidirectionalValueFlow?.entity && !explicitIntentCandidate) || + (counterpartyTurnover?.entity && !explicitIntentCandidate)); return { schema_version: "assistant_turn_meaning_v1", raw_message: rawMessage, @@ -373,10 +439,13 @@ function createAssistantTurnMeaningPolicy(deps = {}) { asked_domain_family: askedDomainFamily, asked_action_family: askedActionFamily, explicit_intent_candidate: explicitIntentCandidate, - explicit_entity_candidates: broadBusinessEvaluation?.family ? [] : buildEntityCandidates(counterpartyTurnover), + explicit_entity_candidates: broadBusinessEvaluation?.family + ? [] + : buildEntityCandidates(counterpartyBidirectionalValueFlow ?? counterpartyTurnover), meaning_confidence: broadBusinessEvaluation?.family ? "medium" - : supportedIntent?.confidence ?? (counterpartyTurnover?.family ? "medium" : "low"), + : supportedIntent?.confidence ?? + (counterpartyBidirectionalValueFlow?.family || counterpartyTurnover?.family ? "medium" : "low"), intent_override_strength: explicitIntentCandidate ? "explicit_current_turn_intent" : staleReplayForbidden diff --git a/llm_normalizer/backend/dist/services/routeHintAdapter.js b/llm_normalizer/backend/dist/services/routeHintAdapter.js index 3d04d68..33edeab 100644 --- a/llm_normalizer/backend/dist/services/routeHintAdapter.js +++ b/llm_normalizer/backend/dist/services/routeHintAdapter.js @@ -31,6 +31,8 @@ function toRouteHintSummaryV1(normalized) { } const ACCOUNT_HINT_PATTERN = /(?:\b(?:account|acct|schet|счет|сч)\s*[:#]?\s*(?:[1-9][0-9](?:[./-][0-9]{1,2})?)|\b(?:19|20|21|23|25|26|28|29|44|51|60|62|68)\b)/i; const PERIOD_PATTERN = /\b20\d{2}(?:[-./](?:0[1-9]|1[0-2]))?\b/i; +const BIDIRECTIONAL_VALUE_FLOW_PATTERN = /(?:\b(?:receive(?:d)?|received|get|got|incoming|inflow|paid|payment|payments|outgoing|outflow|net|netto|cash\s*flow)\b|получ(?:ить|ил[аи]?|ено|аем|или)|поступ(?:ил[аи]?|ление|ления)|заплат(?:ить|ил[аи]?|или)|оплат(?:ить|ил[аи]?|ы|или)|входящ(?:ий|ие|их)|исходящ(?:ий|ие|их)|нетто|сальдо)/iu; +const COUNTERPARTY_SCOPE_PATTERN = /(?:\b(?:counterparty|supplier|customer|vendor|client|bank)\b|контрагент|поставщик|покупател|клиент|заказчик|банк|сбербанк|по\s+(?:ип|ооо|пао|зао|оао|группа)\b)/iu; const SYMPTOM_MARKER_PATTERN = /(?:\bsymptom\b|\banomaly\b|\bproblem\b|\bissue\b|\btail\b|\bhanging\b|\bblocked\b|\bincomplete\b|remains?\s+open|not\s+(?:confirmed|observed|resolved|closed)|не\s+(?:подтвержден|закрыт|наблюдается)|хвост|сбой|проблем)/i; const LIFECYCLE_MARKER_PATTERN = /(?:\blifecycle\b|\bchain\b|\btransition\b|\bstep\b|\btrace\b|цепоч|этап|переход|связк|где\s+разрыв)/i; const CHAIN_BREAK_PATTERN = /(?:\bbreak\b|\bbroken\b|\bgap\b|missing\s+(?:transition|step|link)|chain\s+break|разрыв|обрыв|нет\s+переход|не\s+дошл|не\s+наблюд)/i; @@ -54,6 +56,13 @@ exports.ROUTE_DISCIPLINE_RULE_TABLE = [ forbidden_fallback: ["store_canonical", "hybrid_store_plus_live"], description: "Ranking and period summary queries require analytical batch path." }, + { + query_class: "bidirectional_value_flow", + required_route: "hybrid_store_plus_live", + allowed_fallback: ["no_route"], + forbidden_fallback: ["store_canonical"], + description: "Scoped bidirectional value-flow questions require hybrid evidence path." + }, { query_class: "symptom_first", required_route: "hybrid_store_plus_live", @@ -155,6 +164,17 @@ function hasAmbiguitySignal(fragment, lowerText) { function hasAccountOrPeriodAnchor(fragment, lowerText) { return fragment.account_hints.length > 0 || ACCOUNT_HINT_PATTERN.test(lowerText) || PERIOD_PATTERN.test(lowerText); } +function hasBidirectionalValueFlowSignal(fragment, lowerText) { + if (fragment.flags.asks_for_ranking_or_top || fragment.flags.asks_for_period_summary) { + return false; + } + return BIDIRECTIONAL_VALUE_FLOW_PATTERN.test(lowerText); +} +function hasCounterpartyScopeSignal(fragment, lowerText) { + return (COUNTERPARTY_SCOPE_PATTERN.test(lowerText) || + fragment.entity_hints.some((hint) => hint.trim().length > 0) || + fragment.candidate_labels.includes("cross_entity")); +} function resolveRouteClass(fragment) { const lowerText = mergedFragmentText(fragment); const symptomSignal = hasSymptomSignal(fragment, lowerText); @@ -164,12 +184,17 @@ function resolveRouteClass(fragment) { const causalSignal = hasCausalSignal(lowerText); const ambiguitySignal = hasAmbiguitySignal(fragment, lowerText); const accountOrPeriodAnchor = hasAccountOrPeriodAnchor(fragment, lowerText); + const bidirectionalValueFlowSignal = hasBidirectionalValueFlowSignal(fragment, lowerText); + const counterpartyScopeSignal = hasCounterpartyScopeSignal(fragment, lowerText); if (fragment.flags.asks_for_exact_object_trace) { return ROUTE_DISCIPLINE_RULE_MAP.get("exact_object_trace"); } if (fragment.flags.asks_for_ranking_or_top || fragment.flags.asks_for_period_summary) { return ROUTE_DISCIPLINE_RULE_MAP.get("ranking_or_period_summary"); } + if (bidirectionalValueFlowSignal && counterpartyScopeSignal) { + return ROUTE_DISCIPLINE_RULE_MAP.get("bidirectional_value_flow"); + } if (ambiguitySignal && (symptomSignal || lifecycleSignal || chainBreakSignal || periodImpactSignal || causalSignal)) { return ROUTE_DISCIPLINE_RULE_MAP.get("mixed_ambiguity"); } @@ -205,7 +230,8 @@ function shouldPromoteFromNoRoute(fragment, rule) { hasLifecycleSignal(fragment, lowerText) || hasChainBreakSignal(lowerText) || hasPeriodImpactSignal(lowerText) || - hasCausalSignal(lowerText); + hasCausalSignal(lowerText) || + (hasBidirectionalValueFlowSignal(fragment, lowerText) && hasCounterpartyScopeSignal(fragment, lowerText)); const hasAnchor = hasAccountOrPeriodAnchor(fragment, lowerText) || fragment.candidate_labels.includes("cross_entity") || DOMAIN_LEXICAL_ANCHOR_PATTERN.test(lowerText); diff --git a/llm_normalizer/backend/src/routes/autoRuns.ts b/llm_normalizer/backend/src/routes/autoRuns.ts index aeb71bd..46c8726 100644 --- a/llm_normalizer/backend/src/routes/autoRuns.ts +++ b/llm_normalizer/backend/src/routes/autoRuns.ts @@ -1287,7 +1287,8 @@ function loadSessionDialog(runId: string, caseId: string): { text: toStringSafe(item.text) ?? "", created_at: toStringSafe(item.created_at), trace_id: toStringSafe(item.trace_id), - reply_type: toStringSafe(item.reply_type) + reply_type: toStringSafe(item.reply_type), + debug: item.debug ?? null })); const turns = toArray(record.turns) @@ -1364,7 +1365,8 @@ function buildFallbackDialog(run: IndexedRun, caseId: string): { text: userText, created_at: null, trace_id: null, - reply_type: null + reply_type: null, + debug: null }, { message_id: null, @@ -1372,7 +1374,8 @@ function buildFallbackDialog(run: IndexedRun, caseId: string): { text: assistantSummaryParts.join("\n"), created_at: null, trace_id: toStringSafe(targetCase.trace_id), - reply_type: toStringSafe(targetCase.reply_type) + reply_type: toStringSafe(targetCase.reply_type), + debug: null } ], decomposition: [], diff --git a/llm_normalizer/backend/src/services/address_runtime/composeStage.ts b/llm_normalizer/backend/src/services/address_runtime/composeStage.ts index ab81064..ef887e4 100644 --- a/llm_normalizer/backend/src/services/address_runtime/composeStage.ts +++ b/llm_normalizer/backend/src/services/address_runtime/composeStage.ts @@ -524,6 +524,73 @@ function bankOperationEvidenceLine( return `Основание 1С: ${parts.join("; ")}.`; } +type BankOperationSemanticBucket = + | "commission" + | "deposit_or_credit" + | "tax_or_budget" + | "transfer_or_return" + | "other"; + +function classifyBankOperationSemanticBucket(row: ComposeStageRow): BankOperationSemanticBucket { + const text = [ + row.registrator, + row.operation_kind, + row.payment_purpose, + row.contract, + row.comment + ] + .map((item) => String(item ?? "").toLowerCase()) + .join(" "); + + if (/(?:комисс|тариф|эквайр|обслуживан)/iu.test(text)) { + return "commission"; + } + if (/(?:депозит|кредит|займ|овердрафт|процент|ссуд)/iu.test(text)) { + return "deposit_or_credit"; + } + if (/(?:налог|ндс|взнос|бюджет|фнс|пфр|страхов)/iu.test(text)) { + return "tax_or_budget"; + } + if (/(?:возврат|перевод|перечислен|переброс|пополн|инкасс|перенос)/iu.test(text)) { + return "transfer_or_return"; + } + return "other"; +} + +function bankOperationSemanticBucketLabel(bucket: BankOperationSemanticBucket): string { + if (bucket === "commission") { + return "комиссии и банковое обслуживание"; + } + if (bucket === "deposit_or_credit") { + return "депозиты, кредиты или проценты"; + } + if (bucket === "tax_or_budget") { + return "налоги и бюджетные платежи"; + } + if (bucket === "transfer_or_return") { + return "переводы, возвраты или перебросы"; + } + return "прочие банковские операции"; +} + +function summarizeBankOperationSemantics(rows: ComposeStageRow[]): string | null { + if (rows.length === 0) { + return null; + } + const counts = new Map(); + for (const row of rows) { + const bucket = classifyBankOperationSemanticBucket(row); + counts.set(bucket, (counts.get(bucket) ?? 0) + 1); + } + const ranked = Array.from(counts.entries()) + .sort((left, right) => right[1] - left[1]) + .slice(0, 3); + if (ranked.length === 0) { + return null; + } + const parts = ranked.map(([bucket, count]) => `${bankOperationSemanticBucketLabel(bucket)} — ${count}`); + return `По смыслу это скорее финансовый/банковский контур: ${parts.join("; ")}.`; +} function bankRoleBoundaryLine(userMessage: string | null | undefined, rows: ComposeStageRow[]): string | null { const incomingBoundary = hasBankIncomingRoleBoundaryQuestion(userMessage); const outgoingBoundary = hasBankOutgoingRoleBoundaryQuestion(userMessage); @@ -5013,13 +5080,34 @@ function composeFactualReplyBody( ); const counterparty = resolvePreferredCounterpartyDisplayLabel(options.counterpartyHint, rowCounterparties); const roleBoundary = bankRoleBoundaryLine(options.userMessage, rows); - const visibleRows = rows.slice(0, Math.min(rows.length, 5)); + const visibleRows = [...rows] + .sort( + (left, right) => + Math.abs(right.amount ?? 0) - Math.abs(left.amount ?? 0) || + (String(right.period ?? "").localeCompare(String(left.period ?? ""), "ru")) + ) + .slice(0, Math.min(rows.length, 5)); + const semanticSummary = summarizeBankOperationSemantics(rows); + const compactEvidenceRows = visibleRows.map((row, index) => { + const direction = bankOperationDirectionLabel(bankOperationDirection(row)); + const amount = formatMoneyRub(row.amount ?? 0); + const period = row.period ? formatDateRu(row.period) : "дата не указана"; + const operationKind = String(row.operation_kind ?? "").trim(); + const paymentPurpose = String(row.payment_purpose ?? "").trim(); + const detail = operationKind || paymentPurpose + ? ` | ${[operationKind, paymentPurpose].filter(Boolean).join("; ")}` + : ""; + return `${index + 1}. ${period} | ${direction} | ${amount}${detail}`; + }); const lines = [ `Коротко: найдено банковских операций${counterparty ? ` по ${counterparty}` : " по контрагенту"} — ${rows.length}.`, summarizeBankOperationDirections(rows), roleBoundary ?? "Показываю подтвержденные банковские операции из текущего среза.", bankOperationEvidenceLine(rows, preferredBankEvidenceDirection(options.userMessage)), - ...formatTopRows(visibleRows, visibleRows.length) + ...(semanticSummary ? [semanticSummary] : []), + "Примеры строк 1С:", + ...compactEvidenceRows, + "Следующий шаг: могу отдельно разложить назначения платежа, договоры или отделить банковский контур от клиентского/поставщицкого." ]; if (rows.length > visibleRows.length) { lines.push(`Показаны первые ${visibleRows.length} из ${rows.length}; полный список остается в подтвержденном срезе.`); diff --git a/llm_normalizer/backend/src/services/address_runtime/counterpartyAnalyticsReplyBuilders.ts b/llm_normalizer/backend/src/services/address_runtime/counterpartyAnalyticsReplyBuilders.ts index 3737b98..4c685be 100644 --- a/llm_normalizer/backend/src/services/address_runtime/counterpartyAnalyticsReplyBuilders.ts +++ b/llm_normalizer/backend/src/services/address_runtime/counterpartyAnalyticsReplyBuilders.ts @@ -120,6 +120,26 @@ function findFocusedCounterpartyValuePoint( return profileRows.length === 1 ? profileRows[0] : null; } +function hasProfitAmbiguityCue(normalizedQuestion: string): boolean { + return /(?:заработ|прибыл|прибыль|доход|выручк)/iu.test(normalizedQuestion); +} + +function buildCashflowBoundaryLine(isSupplier: boolean): string { + return isSupplier + ? "Граница ответа: это подтвержденный денежный поток по поставщику, а не итоговая задолженность." + : "Граница ответа: это подтвержденный денежный поток по поступлениям, а не чистая прибыль."; +} + +function buildCashflowNextStepLine(isSupplier: boolean, normalizedQuestion: string): string | null { + if (isSupplier) { + return "Следующий шаг: могу отдельно показать остаток долга, просрочку или расшифровку по документам."; + } + if (hasProfitAmbiguityCue(normalizedQuestion)) { + return "Следующий шаг: могу отдельно проверить чистую прибыль по закрытию 90/91/99."; + } + return "Следующий шаг: могу разложить поток по месяцам, документам или контрагентам."; +} + export function composeCounterpartyAnalyticsReply( intent: AddressIntent, rows: ComposeStageRow[], @@ -546,6 +566,8 @@ export function composeCounterpartyAnalyticsReply( const semanticSingleBestCounterparty = focus === "top_by_total" && hasSingleBestCounterpartyCue && !asksExplicitRankingList; const effectiveLimit = asksSingleBestCounterparty || semanticSingleBestCounterparty ? 1 : limit; + const cashflowBoundaryLine = buildCashflowBoundaryLine(isSupplier); + const cashflowNextStepLine = buildCashflowNextStepLine(isSupplier, normalizedQuestion); const byCounterparty = new Map(); const byYear = new Map(); @@ -655,10 +677,12 @@ export function composeCounterpartyAnalyticsReply( ? `за период ${deps.formatDateRu(options.periodFrom)}..${deps.formatDateRu(options.periodTo)}` : "за доступное время"; const directAnswerLine = isSupplier - ? `Оборот по ${focusedCounterparty.name} ${periodLabel}: ${deps.formatMoneyRub(focusedCounterparty.total)} по ${focusedCounterparty.ops} подтвержденным исходящим операциям. Это денежный поток по поставщику, а не итоговая задолженность.` - : `Оборот по ${focusedCounterparty.name} ${periodLabel}: ${deps.formatMoneyRub(focusedCounterparty.total)} по ${focusedCounterparty.ops} подтвержденным входящим операциям. Это денежный поток от клиента, а не чистая прибыль.`; + ? `Оборот по ${focusedCounterparty.name} ${periodLabel}: ${deps.formatMoneyRub(focusedCounterparty.total)} по ${focusedCounterparty.ops} подтвержденным исходящим операциям.` + : `Оборот по ${focusedCounterparty.name} ${periodLabel}: ${deps.formatMoneyRub(focusedCounterparty.total)} по ${focusedCounterparty.ops} подтвержденным входящим операциям.`; const summaryLines = [ directAnswerLine, + cashflowBoundaryLine, + ...(cashflowNextStepLine ? [cashflowNextStepLine] : []), "", "Подтверждение:", `- Контрагент в выборке: ${focusedCounterparty.name}.`, @@ -678,11 +702,10 @@ export function composeCounterpartyAnalyticsReply( options.periodFrom && options.periodTo ? `За период ${deps.formatDateRu(options.periodFrom)}..${deps.formatDateRu(options.periodTo)} подтверждено ${deps.formatMoneyRub(totalFlow)} ${isSupplier ? "исходящих выплат" : "входящих поступлений"}.` : `За все доступное время подтверждено ${deps.formatMoneyRub(totalFlow)} ${isSupplier ? "исходящих выплат" : "входящих поступлений"}.`; - const directAnswerLine = isSupplier - ? periodLine - : `${periodLine} Это денежный поток от клиентов, а не чистая прибыль.`; const summaryLines = [ - directAnswerLine, + periodLine, + cashflowBoundaryLine, + ...(cashflowNextStepLine ? [cashflowNextStepLine] : []), "", "Подтверждение:", `- Операций в выборке: ${totalOperations}.`, @@ -709,11 +732,17 @@ export function composeCounterpartyAnalyticsReply( const strongestYear = visible[0]; const directAnswerLine = isSupplier ? `Самый крупный год по подтвержденным выплатам: ${strongestYear.year} (${deps.formatMoneyRub(strongestYear.total)} по ${strongestYear.ops} операциям).` - : `Самый доходный год по подтвержденным поступлениям: ${strongestYear.year} (${deps.formatMoneyRub(strongestYear.total)} по ${strongestYear.ops} операциям). Это денежный поток, а не чистая прибыль.`; + : `Самый доходный год по подтвержденным поступлениям: ${strongestYear.year} (${deps.formatMoneyRub(strongestYear.total)} по ${strongestYear.ops} операциям).`; const heading = isSupplier ? `Топ-${visible.length} лет по сумме выплат:` : `Топ-${visible.length} лет по сумме поступлений:`; lines.unshift(heading); + if (!isSupplier) { + lines.unshift(cashflowBoundaryLine); + if (cashflowNextStepLine) { + lines.unshift(cashflowNextStepLine); + } + } lines.unshift(directAnswerLine); lines.push( ...visible.map( @@ -829,11 +858,17 @@ export function composeCounterpartyAnalyticsReply( const directAnswerLine = singleCandidateOnly ? isSupplier ? `В выбранном срезе найден один поставщик: ${leadingCounterparty.name} (${deps.formatMoneyRub(leadingCounterparty.total)} по ${leadingCounterparty.ops} операциям). Это не полноценный сравнительный рейтинг.` - : `В выбранном срезе найден один клиент: ${leadingCounterparty.name} (${deps.formatMoneyRub(leadingCounterparty.total)} по ${leadingCounterparty.ops} операциям). Это не полноценный сравнительный рейтинг; сумма является денежным потоком, а не чистой прибылью.` + : `В выбранном срезе найден один клиент: ${leadingCounterparty.name} (${deps.formatMoneyRub(leadingCounterparty.total)} по ${leadingCounterparty.ops} операциям). Это не полноценный сравнительный рейтинг.` : isSupplier ? `Крупнейший поставщик по подтвержденным выплатам ${rankingPeriodLabel}: ${leadingCounterparty.name} (${deps.formatMoneyRub(leadingCounterparty.total)} по ${leadingCounterparty.ops} операциям).` - : `Самый доходный клиент ${rankingPeriodLabel} по подтвержденным поступлениям: ${leadingCounterparty.name} (${deps.formatMoneyRub(leadingCounterparty.total)} по ${leadingCounterparty.ops} операциям). Это денежный поток, а не чистая прибыль.`; + : `Самый доходный клиент ${rankingPeriodLabel} по подтвержденным поступлениям: ${leadingCounterparty.name} (${deps.formatMoneyRub(leadingCounterparty.total)} по ${leadingCounterparty.ops} операциям).`; lines.unshift(directAnswerLine); + if (!isSupplier) { + lines.splice(1, 0, cashflowBoundaryLine); + if (cashflowNextStepLine) { + lines.splice(2, 0, cashflowNextStepLine); + } + } } lines.push( ...visible.map((item, index) => { diff --git a/llm_normalizer/backend/src/services/address_runtime/inventoryReplyBuilders.ts b/llm_normalizer/backend/src/services/address_runtime/inventoryReplyBuilders.ts index 92aa057..abb36f6 100644 --- a/llm_normalizer/backend/src/services/address_runtime/inventoryReplyBuilders.ts +++ b/llm_normalizer/backend/src/services/address_runtime/inventoryReplyBuilders.ts @@ -175,11 +175,10 @@ export function composeInventoryReply( const uniqueWarehouses = deps.uniqueStrings( positions.map((item) => String(item.warehouse ?? "").trim()).filter((item) => item.length > 0) ); - const totalQuantity = positions.reduce((sum, item) => sum + item.quantity, 0); const totalAmount = positions.reduce((sum, item) => sum + item.amount, 0); const directAnswerLine = positions.length > 0 - ? `На ${deps.formatDateRu(asOfDate)} на складе подтверждено ${deps.formatNumberWithDots(positions.length)} позиций с остатком на ${deps.formatMoneyRub(totalAmount)}.` + ? `На ${deps.formatDateRu(asOfDate)} на складе подтверждено ${deps.formatNumberWithDots(positions.length)} позиций на ${deps.formatMoneyRub(totalAmount)}.` : `На ${deps.formatDateRu(asOfDate)} подтвержденных товарных остатков по счету 41.01 не найдено.`; const lines: string[] = [directAnswerLine]; @@ -213,11 +212,14 @@ export function composeInventoryReply( `Позиции с остатком: ${deps.formatNumberWithDots(positions.length)}.`, `Уникальных товаров: ${deps.formatNumberWithDots(uniqueItems.length)}.`, `Уникальных складов: ${deps.formatNumberWithDots(uniqueWarehouses.length)}.`, - `Суммарное количество: ${deps.formatNumberWithDots(totalQuantity, 3)}.` + "Общее количество не свожу в один управленческий показатель, потому что в остатках смешаны разнородные позиции." ]); if (rows.length !== positions.length) { lines.push(`- Проверенных строк движения: ${deps.formatNumberWithDots(rows.length)}.`); } + if (positions.length > 0) { + lines.push("- Следующий шаг: могу раскрыть полный список, разложить остатки по складам или сравнить с другой датой."); + } return positions.length > 0 ? buildFactualListReply(lines, buildConfirmedBalanceSemantics("strong")) diff --git a/llm_normalizer/backend/src/services/assistantMcpDiscoveryResponseCandidate.ts b/llm_normalizer/backend/src/services/assistantMcpDiscoveryResponseCandidate.ts index bdbd69c..b8455bb 100644 --- a/llm_normalizer/backend/src/services/assistantMcpDiscoveryResponseCandidate.ts +++ b/llm_normalizer/backend/src/services/assistantMcpDiscoveryResponseCandidate.ts @@ -1128,6 +1128,40 @@ function buildCompactBusinessOverviewReply( } if (rankingNeed) { + const explicitPeriodRankingOverview = + period && + !/(?:все\s+доступное|все\s+время|all\s+time)/iu.test(period) && + (incomingAmount || outgoingAmount || netAmount); + if (explicitPeriodRankingOverview) { + lines.push( + `Коротко: ${organizationPrefix}${period} денежная картина подтверждена по найденным строкам 1С.` + ); + lines.push( + `Деньги: входящие ${incomingAmount ?? "0 руб."}, исходящие ${outgoingAmount ?? "0 руб."}, расчетное операционное нетто ${sentenceAmount(netAmount) ?? netAmount ?? "0 руб."}.` + ); + if (customerName && customerAmount) { + lines.push( + topCustomerLooksFinancial + ? `Топ входящих: 1. ${customerName} — ${sentenceAmount(customerAmount) ?? customerAmount}. Это финансовый/банковский контур, не считаю его клиентской выручкой без назначения платежа.${nonFinancialCustomer ? ` 2. Крупнейший небанковский входящий контрагент: ${nonFinancialCustomer}.` : ""}` + : `Крупнейший входящий контрагент: ${customerName} — ${sentenceAmount(customerAmount) ?? customerAmount}.` + ); + } + if (topSupplier) { + lines.push( + topSupplierLooksFinancial + ? `Топ исходящих: 1. ${topSupplier}. Это финансовый/банковский контур, не считаю его обычным поставщиком без назначения платежа и договора.${nonFinancialSupplier ? ` 2. Крупнейший небанковский получатель исходящих денег: ${nonFinancialSupplier}.` : ""}` + : `Крупнейший получатель исходящих денег: ${topSupplier}.` + ); + } + lines.push( + `Вывод: по движению денег период ${netDirection}; это не чистая прибыль и не бухгалтерский финрезультат.` + ); + if (requestedFinancialBoundaryLine) { + lines.push(requestedFinancialBoundaryLine); + } + lines.push("Следующий шаг: могу отдельно посчитать чистую прибыль через закрытие 90/91/99 или разложить этот период по контрагентам."); + return joinBusinessReplyLines(lines); + } const incomingLeader = strongestIncomingYear(overview); const canRankYearlyNet = !limitLine; const netLeader = canRankYearlyNet ? strongestNetYear(overview) : null; diff --git a/llm_normalizer/backend/src/services/assistantMcpDiscoveryTurnInputAdapter.ts b/llm_normalizer/backend/src/services/assistantMcpDiscoveryTurnInputAdapter.ts index e1353b3..39fc051 100644 --- a/llm_normalizer/backend/src/services/assistantMcpDiscoveryTurnInputAdapter.ts +++ b/llm_normalizer/backend/src/services/assistantMcpDiscoveryTurnInputAdapter.ts @@ -1600,15 +1600,48 @@ export function buildAssistantMcpDiscoveryTurnInput( const rawEffectiveText = toNonEmptyString(input.effectiveMessage); const repairedUserText = rawUserText ? repairAddressMojibakeText(rawUserText) : null; const repairedEffectiveText = rawEffectiveText ? repairAddressMojibakeText(rawEffectiveText) : null; + const rawUserSignalSourceText = repairedUserText ?? rawUserText ?? ""; const rawSignalSourceText = `${repairedUserText ?? rawUserText ?? ""} ${repairedEffectiveText ?? rawEffectiveText ?? ""}`.trim(); const rawEntitySourceText = repairedUserText ?? rawUserText ?? repairedEffectiveText ?? rawEffectiveText ?? rawSignalSourceText; + const rawUserEntitySourceText = rawUserSignalSourceText || rawEntitySourceText; + const rawUserTextOnly = compactLower(rawUserSignalSourceText); + const rawAssistantEntityCandidates = collectEntityCandidates(assistantTurnMeaning?.explicit_entity_candidates); + const rawUserPrimaryBusinessOverviewSignal = hasBusinessOverviewSignal(rawUserTextOnly); + const rawUserLifecyclePivotTextSignal = + !rawUserPrimaryBusinessOverviewSignal && hasLifecycleSignal(rawUserTextOnly); + const rawUserBidirectionalValueFlowPivotTextSignal = + !rawUserPrimaryBusinessOverviewSignal && + !rawUserLifecyclePivotTextSignal && + hasBidirectionalValueFlowSignal(rawUserTextOnly); + const rawUserScopedEntityCandidate = rawUserSignalSourceText + ? rawScopedEntityCandidateFromText(rawUserEntitySourceText) + : null; + const rawUserCounterpartyBidirectionalOverride = Boolean( + rawUserBidirectionalValueFlowPivotTextSignal && + (rawUserScopedEntityCandidate || + predecomposeEntities.counterparty || + rawAssistantEntityCandidates.find((candidate) => !isInvalidEntityCandidate(candidate))) + ); const rawText = compactLower(rawSignalSourceText); const rawReferentialDocumentExclusionSignal = hasReferentialDocumentExclusionFollowupSignal( repairedUserText ?? rawUserText ?? "" ); - const rawPrimaryBusinessOverviewSignal = hasBusinessOverviewSignal(rawText); + const rawPrimaryBusinessOverviewSignal = + hasBusinessOverviewSignal(rawText) && !rawUserCounterpartyBidirectionalOverride; const explicitVatQuestionSignal = hasExplicitVatQuestionSignal(rawText); const explicitVatMovementEvidenceSignal = hasExplicitVatMovementEvidenceSignal(rawText); + const rawLifecyclePivotTextSignal = + !rawPrimaryBusinessOverviewSignal && hasLifecycleSignal(rawText); + const rawBidirectionalValueFlowPivotTextSignal = + !rawPrimaryBusinessOverviewSignal && + !rawLifecyclePivotTextSignal && + hasBidirectionalValueFlowSignal(rawText); + const rawValueFlowPivotTextSignal = + !rawPrimaryBusinessOverviewSignal && + !rawLifecyclePivotTextSignal && + (hasValueFlowSignal(rawText) || + hasValueRankingSignal(rawText) || + rawBidirectionalValueFlowPivotTextSignal); const explicitVatSuppressesBusinessOverviewContinuation = Boolean( explicitVatQuestionSignal && !rawPrimaryBusinessOverviewSignal ); @@ -1634,7 +1667,7 @@ export function buildAssistantMcpDiscoveryTurnInput( !rawLifecycleSignal && !rawValueFlowSignal && !rawMetadataSignal && hasEntityResolutionSignal(rawText); const rawPayoutSignal = rawValueFlowSignal && !rawBidirectionalValueFlowSignal && hasPayoutSignal(rawText); const rawValueFlowAggregateQuestionSignal = - rawValueFlowSignal && hasValueFlowAggregateQuestionSignal(rawText); + (rawValueFlowSignal || rawValueFlowPivotTextSignal) && hasValueFlowAggregateQuestionSignal(rawText); const monthlyAggregationSignal = hasMonthlyAggregationSignal(rawText); const rawAllTimeScopeSignal = hasAllTimeScopeHint(rawText); const dateScopeSignalText = stripNegatedTaxDateScopeClauses(rawText); @@ -1702,6 +1735,47 @@ export function buildAssistantMcpDiscoveryTurnInput( : profitMarginBusinessOverviewSignal ? "profit_margin_boundary" : "broad_evaluation"; + const assistantTurnMeaningDateScope = toNonEmptyString(assistantTurnMeaning?.explicit_date_scope); + const rawAssistantTurnMeaningOrganizationScope = toNonEmptyString(assistantTurnMeaning?.explicit_organization_scope); + const assistantTurnMeaningOrganizationScope = isReferentialOrganizationPlaceholder( + rawAssistantTurnMeaningOrganizationScope + ) + ? null + : rawAssistantTurnMeaningOrganizationScope; + const rawOrganizationMentionSignal = hasOrganizationScopeSignalUtf8(rawText); + const rawOrganizationScope = extractOrganizationScopeFromRawText(rawUserText ?? rawEffectiveText ?? rawSignalSourceText); + const currentTurnFreshOrganizationScope = predecomposeEntities.organization ?? rawOrganizationScope; + const currentTurnOrganizationScope = + currentTurnFreshOrganizationScope ?? assistantTurnMeaningOrganizationScope; + const predecomposeOrganizationMirrorsCounterparty = sameScopedName( + predecomposeEntities.counterparty, + predecomposeEntities.organization + ); + const organizationMirrorsPredecomposeCounterpartyForPivot = Boolean( + sameScopedName(predecomposeEntities.counterparty, assistantTurnMeaningOrganizationScope) || + sameScopedName(predecomposeEntities.counterparty, currentTurnOrganizationScope) || + predecomposeOrganizationMirrorsCounterparty + ); + const normalizedPredecomposeCounterpartyForPivot = + organizationMirrorsPredecomposeCounterpartyForPivot + ? null + : normalizeFollowupCounterpartyCandidate(predecomposeEntities.counterparty); + const rawExplicitCounterpartyPivotCandidate = + rawScopedEntityCandidate ?? + rawAssistantEntityCandidates.find( + (candidate) => + !isInvalidEntityCandidate(candidate) && + !sameScopedName(candidate, currentTurnOrganizationScope) + ) ?? + normalizedPredecomposeCounterpartyForPivot ?? + null; + const businessOverviewCounterpartyValueFlowPivot = Boolean( + businessOverviewContinuationSignal && + !rawPrimaryBusinessOverviewSignal && + rawValueFlowPivotTextSignal && + rawExplicitCounterpartyPivotCandidate && + (rawTopicSwitchSignal || rawValueFlowAggregateQuestionSignal) + ); const businessOverviewUnsupportedFamily = inventoryReserveBusinessOverviewSignal ? "inventory_reserve_liquidation_boundary" : debtDueDateBusinessOverviewSignal @@ -1712,8 +1786,8 @@ export function buildAssistantMcpDiscoveryTurnInput( ? "profit_margin_boundary" : "broad_business_evaluation"; const businessOverviewSignal = - rawBusinessOverviewSignal || - seededBusinessOverviewSignal; + !businessOverviewCounterpartyValueFlowPivot && + (rawBusinessOverviewSignal || seededBusinessOverviewSignal); const businessOverviewSeparateCounterpartySignal = Boolean( businessOverviewSignal && hasBusinessOverviewSeparateCounterpartySignal(rawText) ); @@ -1735,18 +1809,6 @@ export function buildAssistantMcpDiscoveryTurnInput( hasSimpleMovementLanePivotSignal(rawText) || hasMovementEvidenceFollowupSignal(rawText) || hasPronounMovementEvidenceFollowupSignal(rawText); - const assistantTurnMeaningDateScope = toNonEmptyString(assistantTurnMeaning?.explicit_date_scope); - const rawAssistantTurnMeaningOrganizationScope = toNonEmptyString(assistantTurnMeaning?.explicit_organization_scope); - const assistantTurnMeaningOrganizationScope = isReferentialOrganizationPlaceholder( - rawAssistantTurnMeaningOrganizationScope - ) - ? null - : rawAssistantTurnMeaningOrganizationScope; - const rawOrganizationMentionSignal = hasOrganizationScopeSignalUtf8(rawText); - const rawOrganizationScope = extractOrganizationScopeFromRawText(rawUserText ?? rawEffectiveText ?? rawSignalSourceText); - const currentTurnFreshOrganizationScope = predecomposeEntities.organization ?? rawOrganizationScope; - const currentTurnOrganizationScope = - currentTurnFreshOrganizationScope ?? assistantTurnMeaningOrganizationScope; const followupCounterpartyIsMetadataOrganizationScope = Boolean( followupSeed.subjectResolutionOptional && followupSeed.counterparty && @@ -1792,10 +1854,6 @@ export function buildAssistantMcpDiscoveryTurnInput( !rawBidirectionalValueFlowSignal && explicitOrganizationScopeSignal ); - const predecomposeOrganizationMirrorsCounterparty = sameScopedName( - predecomposeEntities.counterparty, - predecomposeEntities.organization - ); const organizationMirrorsPredecomposeCounterparty = Boolean( (rawBidirectionalValueFlowSignal || hasValueRankingSignal(rawText) || @@ -2130,16 +2188,27 @@ export function buildAssistantMcpDiscoveryTurnInput( const bidirectionalValueFlowSignal = !businessOverviewSignal && !lifecycleSignal && - (rawBidirectionalValueFlowSignal || seededAction === "net_value_flow"); + ((businessOverviewCounterpartyValueFlowPivot + ? rawBidirectionalValueFlowPivotTextSignal + : rawBidirectionalValueFlowSignal) || + seededAction === "net_value_flow"); const valueFlowSignal = !businessOverviewSignal && !lifecycleSignal && !metadataGroundedMovementLaneApplicable && - (rawValueFlowSignal || seededDomain === "counterparty_value"); + ((businessOverviewCounterpartyValueFlowPivot + ? rawValueFlowPivotTextSignal + : rawValueFlowSignal) || + seededDomain === "counterparty_value"); const payoutSignal = valueFlowSignal && !bidirectionalValueFlowSignal && - (rawPayoutSignal || seededAction === "payout"); + ((businessOverviewCounterpartyValueFlowPivot + ? rawValueFlowPivotTextSignal && + !rawBidirectionalValueFlowPivotTextSignal && + hasPayoutSignal(rawText) + : rawPayoutSignal) || + seededAction === "payout"); const semanticDataNeed = metadataAmbiguityLaneClarificationApplicable ? "metadata lane clarification" : semanticNeedFor({ @@ -2147,17 +2216,37 @@ export function buildAssistantMcpDiscoveryTurnInput( ? "movements" : businessOverviewSignal ? "business_overview" + : lifecycleSignal + ? "counterparty_lifecycle" + : valueFlowSignal + ? "counterparty_value" : rawDomain ?? seededDomain, action: explicitVatMovementEvidenceSignal ? "list_movements" : businessOverviewSignal ? businessOverviewActionFamily + : lifecycleSignal + ? "activity_duration" + : valueFlowSignal + ? bidirectionalValueFlowSignal + ? "net_value_flow" + : payoutSignal + ? "payout" + : rawAction ?? seededAction ?? "turnover" : rawAction ?? seededAction, unsupported: explicitVatMovementEvidenceSignal ? "movement_evidence" : businessOverviewSignal ? businessOverviewUnsupportedFamily - : unsupported ?? seededUnsupported, + : lifecycleSignal + ? "counterparty_lifecycle" + : valueFlowSignal + ? bidirectionalValueFlowSignal + ? "counterparty_bidirectional_value_flow_or_netting" + : payoutSignal + ? "counterparty_payouts_or_outflow" + : seededUnsupported ?? "counterparty_value_or_turnover" + : unsupported ?? seededUnsupported, lifecycleSignal, valueFlowSignal, metadataSignal: rawMetadataSignal || effectiveMetadataFollowupSeedApplicable, @@ -2469,30 +2558,30 @@ export function buildAssistantMcpDiscoveryTurnInput( unsupported_but_understood_family: businessOverviewSignal ? businessOverviewUnsupportedFamily - : unsupported ?? - (lifecycleSignal - ? "counterparty_lifecycle" + : lifecycleSignal + ? "counterparty_lifecycle" : valueFlowSignal ? bidirectionalValueFlowSignal ? "counterparty_bidirectional_value_flow_or_netting" : payoutSignal ? "counterparty_payouts_or_outflow" : seededUnsupported ?? "counterparty_value_or_turnover" - : metadataGroundedMovementLaneApplicable - ? "movement_evidence" - : metadataGroundedDocumentLaneApplicable - ? "document_evidence" - : explicitVatMovementEvidenceSignal - ? "movement_evidence" - : metadataAmbiguityLaneClarificationApplicable - ? "metadata_lane_choice_clarification" - : entityResolutionSignal - ? "entity_resolution" - : rawMetadataSignal || effectiveMetadataFollowupSeedApplicable - ? "1c_metadata_surface" - : followupDiscoverySeedApplicable - ? seededUnsupported - : null), + : unsupported ?? + (metadataGroundedMovementLaneApplicable + ? "movement_evidence" + : metadataGroundedDocumentLaneApplicable + ? "document_evidence" + : explicitVatMovementEvidenceSignal + ? "movement_evidence" + : metadataAmbiguityLaneClarificationApplicable + ? "metadata_lane_choice_clarification" + : entityResolutionSignal + ? "entity_resolution" + : rawMetadataSignal || effectiveMetadataFollowupSeedApplicable + ? "1c_metadata_surface" + : followupDiscoverySeedApplicable + ? seededUnsupported + : null), stale_replay_forbidden: Boolean( assistantTurnMeaning?.stale_replay_forbidden || businessOverviewSignal || @@ -2763,6 +2852,9 @@ export function buildAssistantMcpDiscoveryTurnInput( if (businessOverviewSignal) { pushReason(reasonCodes, "mcp_discovery_broad_business_evaluation_route_candidate"); } + if (businessOverviewCounterpartyValueFlowPivot) { + pushReason(reasonCodes, "mcp_discovery_business_overview_followup_pivoted_to_counterparty_value_flow"); + } if (businessOverviewContinuationSignal) { pushReason(reasonCodes, "mcp_discovery_business_overview_continuation_from_followup_context"); } diff --git a/llm_normalizer/backend/src/services/assistantTurnMeaningPolicy.ts b/llm_normalizer/backend/src/services/assistantTurnMeaningPolicy.ts index 710c3bd..ce364bb 100644 --- a/llm_normalizer/backend/src/services/assistantTurnMeaningPolicy.ts +++ b/llm_normalizer/backend/src/services/assistantTurnMeaningPolicy.ts @@ -138,6 +138,62 @@ function detectCounterpartyTurnoverFamily(text) { }; } +function detectScopedCounterpartyEntity(text) { + const patterns = [ + /(?:^|[\s,.;:!?])(?:\u043f\u043e|\u0443|\u0434\u043b\u044f|by|for)\s+(.+?)(?=$|[,.;:!?]|\s+(?:\u0437\u0430|\u043d\u0430|\u0432|\u0432\u043e|\u043a|\u043f\u043e|\u0441\u043a\u043e\u043b\u044c\u043a\u043e|\u0441\u043a\u043e\u043a|\u043a\u0430\u043a|\u043a\u0430\u043a\u043e\u0435|\u043a\u0430\u043a\u043e\u0439|\u043a\u0430\u043a\u0430\u044f|\u043a\u0430\u043a\u0438\u0435|\u043f\u043e\u043b\u0443\u0447\p{L}*|\u0437\u0430\u043f\u043b\u0430\u0442\p{L}*|\u043d\u0435\u0442\u0442\u043e|\u0441\u0430\u043b\u044c\u0434\u043e|\u0434\u0435\u043d\u0435\u0433|\u0434\u0435\u043d\u0435\u0436\p{L}*|\u043f\u043b\u0430\u0442[\u0435\u0451]\u0436\p{L}*|\u0438\u0441\u0445\u043e\u0434\p{L}*|\u0432\u0445\u043e\u0434\p{L}*)(?=$|[\s,.;:!?]))/iu, + /(?:^|[\s,.;:!?])(?:\u043f\u043e|\u0443|\u0434\u043b\u044f|by|for)\s+([\p{L}\d._-]{2,})(?=$|[\s,.;:!?])/iu + ]; + const ignored = new Set([ + "\u0433\u043e\u0434", + "\u0433\u043e\u0434\u0430", + "\u043f\u0435\u0440\u0438\u043e\u0434", + "\u043f\u0435\u0440\u0438\u043e\u0434\u0430", + "\u043c\u0435\u0441\u044f\u0446", + "\u043c\u0435\u0441\u044f\u0446\u0430", + "\u043a\u0432\u0430\u0440\u0442\u0430\u043b", + "\u043a\u0432\u0430\u0440\u0442\u0430\u043b\u0430", + "\u0434\u0435\u043d\u044c\u0433\u0438", + "\u043d\u0435\u0442\u0442\u043e", + "\u0441\u0430\u043b\u044c\u0434\u043e", + "year", + "period", + "month", + "quarter", + "net" + ]); + for (const pattern of patterns) { + const rawEntity = text.match(pattern)?.[1]?.trim() ?? ""; + if (!rawEntity) { + continue; + } + const entity = rawEntity.replace(/^["'«»]+|["'«»]+$/gu, "").trim(); + if (entity.length >= 2 && !ignored.has(entity)) { + return entity; + } + } + return null; +} + +function detectCounterpartyBidirectionalValueFlowFamily(text) { + const hasNetCue = + /(?:\u043d\u0435\u0442\u0442\u043e|\u0441\u0430\u043b\u044c\u0434\u043e|net\s+(?:flow|cash|payment)|cash\s+net)/iu.test(text); + const hasIncomingCue = + /(?:\u043f\u043e\u043b\u0443\u0447\p{L}*|\u0432\u0445\u043e\u0434\p{L}*|\u043f\u043e\u0441\u0442\u0443\u043f\p{L}*|received|incoming)/iu.test(text); + const hasOutgoingCue = + /(?:\u0437\u0430\u043f\u043b\u0430\u0442\p{L}*|\u0438\u0441\u0445\u043e\u0434\p{L}*|\u0441\u043f\u0438\u0441\u0430\u043d\p{L}*|paid|outgoing|payment)/iu.test(text); + if (!(hasNetCue || (hasIncomingCue && hasOutgoingCue))) { + return null; + } + const entity = detectScopedCounterpartyEntity(text); + if (!entity) { + return null; + } + return { + family: "counterparty_bidirectional_value_flow_or_netting", + entity + }; +} + function hasExplicitCounterpartyValueObject(text) { return /(?:\u043a\u043b\u0438\u0435\u043d\u0442|\u043f\u043e\u043a\u0443\u043f\u0430\u0442\u0435\u043b|\u0437\u0430\u043a\u0430\u0437\u0447\u0438\u043a|\u043a\u043e\u043d\u0442\u0440\u0430\u0433\u0435\u043d\u0442|\u043f\u043e\u0441\u0442\u0430\u0432\u0449\u0438\u043a|\u0434\u043e\u0433\u043e\u0432\u043e\u0440|\u043a\u043e\u043d\u0442\u0440\u0430\u043a\u0442|\u0442\u043e\u0432\u0430\u0440|\u043d\u043e\u043c\u0435\u043d\u043a\u043b\u0430\u0442\u0443\u0440|\u0441\u0434\u0435\u043b\u043a|customer|client|counterparty|supplier|vendor|contract|item|product|deal)/iu.test( text @@ -379,14 +435,14 @@ function detectBroadBusinessEvaluation(text) { return null; } -function buildEntityCandidates(counterpartyTurnover) { - if (!counterpartyTurnover?.entity) { +function buildEntityCandidates(entityFamily) { + if (!entityFamily?.entity) { return []; } return [ { type: "counterparty", - value: counterpartyTurnover.entity, + value: entityFamily.entity, source: "current_turn_loose_entity_tail" } ]; @@ -400,9 +456,13 @@ export function createAssistantTurnMeaningPolicy(deps = {}) { const effectiveText = normalizeTurnText(effectiveMessage, deps); const joinedText = fallbackCompactWhitespace(`${rawText} ${effectiveText}`); const supportedIntent = detectSupportedIntent(joinedText, deps); + const counterpartyBidirectionalValueFlow = detectCounterpartyBidirectionalValueFlowFamily(joinedText); const counterpartyTurnover = detectCounterpartyTurnoverFamily(joinedText); const selectedObjectInventoryExact = hasSelectedObjectInventoryExactSignal(joinedText); - const broadBusinessEvaluation = selectedObjectInventoryExact ? null : detectBroadBusinessEvaluation(joinedText); + const broadBusinessEvaluation = + selectedObjectInventoryExact || counterpartyBidirectionalValueFlow?.family + ? null + : detectBroadBusinessEvaluation(joinedText); const llmIntent = toNonEmptyString(input?.llmPreDecomposeMeta?.predecomposeContract?.intent, deps); const explicitIntentCandidate = broadBusinessEvaluation?.family @@ -410,6 +470,8 @@ export function createAssistantTurnMeaningPolicy(deps = {}) { : supportedIntent?.intent ?? (llmIntent && llmIntent !== "unknown" ? llmIntent : null); const unsupportedFamily = broadBusinessEvaluation?.family ? broadBusinessEvaluation.family + : !explicitIntentCandidate && counterpartyBidirectionalValueFlow?.family + ? counterpartyBidirectionalValueFlow.family : !explicitIntentCandidate && counterpartyTurnover?.family ? counterpartyTurnover.family : null; @@ -417,6 +479,9 @@ export function createAssistantTurnMeaningPolicy(deps = {}) { if (supportedIntent?.reason) { reasonCodes.push(supportedIntent.reason); } + if (counterpartyBidirectionalValueFlow?.family) { + reasonCodes.push("counterparty_bidirectional_value_flow_current_turn_signal"); + } if (counterpartyTurnover?.family) { reasonCodes.push("counterparty_turnover_current_turn_signal"); } @@ -443,6 +508,8 @@ export function createAssistantTurnMeaningPolicy(deps = {}) { ? "inventory" : broadBusinessEvaluation?.family ? "business_summary" + : counterpartyBidirectionalValueFlow?.family + ? "counterparty_value" : explicitIntentCandidate?.includes("counterparty") ? "counterparty" : counterpartyTurnover?.family @@ -455,6 +522,8 @@ export function createAssistantTurnMeaningPolicy(deps = {}) { ? "confirmed_snapshot" : broadBusinessEvaluation?.family ? "broad_evaluation" + : counterpartyBidirectionalValueFlow?.family + ? "net_value_flow" : explicitIntentCandidate === "customer_revenue_and_payments" || explicitIntentCandidate === "supplier_payouts_profile" ? "counterparty_value_or_turnover" @@ -470,7 +539,10 @@ export function createAssistantTurnMeaningPolicy(deps = {}) { ? "counterparty_value_or_turnover" : null; const staleReplayForbidden = Boolean( - unsupportedFamily || broadBusinessEvaluation?.family || (counterpartyTurnover?.entity && !explicitIntentCandidate) + unsupportedFamily || + broadBusinessEvaluation?.family || + (counterpartyBidirectionalValueFlow?.entity && !explicitIntentCandidate) || + (counterpartyTurnover?.entity && !explicitIntentCandidate) ); return { schema_version: "assistant_turn_meaning_v1", @@ -481,10 +553,13 @@ export function createAssistantTurnMeaningPolicy(deps = {}) { asked_domain_family: askedDomainFamily, asked_action_family: askedActionFamily, explicit_intent_candidate: explicitIntentCandidate, - explicit_entity_candidates: broadBusinessEvaluation?.family ? [] : buildEntityCandidates(counterpartyTurnover), + explicit_entity_candidates: broadBusinessEvaluation?.family + ? [] + : buildEntityCandidates(counterpartyBidirectionalValueFlow ?? counterpartyTurnover), meaning_confidence: broadBusinessEvaluation?.family ? "medium" - : supportedIntent?.confidence ?? (counterpartyTurnover?.family ? "medium" : "low"), + : supportedIntent?.confidence ?? + (counterpartyBidirectionalValueFlow?.family || counterpartyTurnover?.family ? "medium" : "low"), intent_override_strength: explicitIntentCandidate ? "explicit_current_turn_intent" : staleReplayForbidden diff --git a/llm_normalizer/backend/src/services/routeHintAdapter.ts b/llm_normalizer/backend/src/services/routeHintAdapter.ts index b436023..d3975c8 100644 --- a/llm_normalizer/backend/src/services/routeHintAdapter.ts +++ b/llm_normalizer/backend/src/services/routeHintAdapter.ts @@ -46,6 +46,7 @@ type V2FamilyFragment = V2Family["fragments"][number]; type RouteQueryClass = | "exact_object_trace" | "ranking_or_period_summary" + | "bidirectional_value_flow" | "symptom_first" | "lifecycle_first" | "chain_break" @@ -66,6 +67,10 @@ interface RouteDisciplineRule { const ACCOUNT_HINT_PATTERN = /(?:\b(?:account|acct|schet|счет|сч)\s*[:#]?\s*(?:[1-9][0-9](?:[./-][0-9]{1,2})?)|\b(?:19|20|21|23|25|26|28|29|44|51|60|62|68)\b)/i; const PERIOD_PATTERN = /\b20\d{2}(?:[-./](?:0[1-9]|1[0-2]))?\b/i; +const BIDIRECTIONAL_VALUE_FLOW_PATTERN = + /(?:\b(?:receive(?:d)?|received|get|got|incoming|inflow|paid|payment|payments|outgoing|outflow|net|netto|cash\s*flow)\b|получ(?:ить|ил[аи]?|ено|аем|или)|поступ(?:ил[аи]?|ление|ления)|заплат(?:ить|ил[аи]?|или)|оплат(?:ить|ил[аи]?|ы|или)|входящ(?:ий|ие|их)|исходящ(?:ий|ие|их)|нетто|сальдо)/iu; +const COUNTERPARTY_SCOPE_PATTERN = + /(?:\b(?:counterparty|supplier|customer|vendor|client|bank)\b|контрагент|поставщик|покупател|клиент|заказчик|банк|сбербанк|по\s+(?:ип|ооо|пао|зао|оао|группа)\b)/iu; const SYMPTOM_MARKER_PATTERN = /(?:\bsymptom\b|\banomaly\b|\bproblem\b|\bissue\b|\btail\b|\bhanging\b|\bblocked\b|\bincomplete\b|remains?\s+open|not\s+(?:confirmed|observed|resolved|closed)|не\s+(?:подтвержден|закрыт|наблюдается)|хвост|сбой|проблем)/i; const LIFECYCLE_MARKER_PATTERN = @@ -96,6 +101,13 @@ export const ROUTE_DISCIPLINE_RULE_TABLE: RouteDisciplineRule[] = [ forbidden_fallback: ["store_canonical", "hybrid_store_plus_live"], description: "Ranking and period summary queries require analytical batch path." }, + { + query_class: "bidirectional_value_flow", + required_route: "hybrid_store_plus_live", + allowed_fallback: ["no_route"], + forbidden_fallback: ["store_canonical"], + description: "Scoped bidirectional value-flow questions require hybrid evidence path." + }, { query_class: "symptom_first", required_route: "hybrid_store_plus_live", @@ -218,6 +230,21 @@ function hasAccountOrPeriodAnchor(fragment: V2FamilyFragment, lowerText: string) return fragment.account_hints.length > 0 || ACCOUNT_HINT_PATTERN.test(lowerText) || PERIOD_PATTERN.test(lowerText); } +function hasBidirectionalValueFlowSignal(fragment: V2FamilyFragment, lowerText: string): boolean { + if (fragment.flags.asks_for_ranking_or_top || fragment.flags.asks_for_period_summary) { + return false; + } + return BIDIRECTIONAL_VALUE_FLOW_PATTERN.test(lowerText); +} + +function hasCounterpartyScopeSignal(fragment: V2FamilyFragment, lowerText: string): boolean { + return ( + COUNTERPARTY_SCOPE_PATTERN.test(lowerText) || + fragment.entity_hints.some((hint) => hint.trim().length > 0) || + fragment.candidate_labels.includes("cross_entity") + ); +} + function resolveRouteClass(fragment: V2FamilyFragment): RouteDisciplineRule { const lowerText = mergedFragmentText(fragment); const symptomSignal = hasSymptomSignal(fragment, lowerText); @@ -227,6 +254,8 @@ function resolveRouteClass(fragment: V2FamilyFragment): RouteDisciplineRule { const causalSignal = hasCausalSignal(lowerText); const ambiguitySignal = hasAmbiguitySignal(fragment, lowerText); const accountOrPeriodAnchor = hasAccountOrPeriodAnchor(fragment, lowerText); + const bidirectionalValueFlowSignal = hasBidirectionalValueFlowSignal(fragment, lowerText); + const counterpartyScopeSignal = hasCounterpartyScopeSignal(fragment, lowerText); if (fragment.flags.asks_for_exact_object_trace) { return ROUTE_DISCIPLINE_RULE_MAP.get("exact_object_trace")!; @@ -234,6 +263,9 @@ function resolveRouteClass(fragment: V2FamilyFragment): RouteDisciplineRule { if (fragment.flags.asks_for_ranking_or_top || fragment.flags.asks_for_period_summary) { return ROUTE_DISCIPLINE_RULE_MAP.get("ranking_or_period_summary")!; } + if (bidirectionalValueFlowSignal && counterpartyScopeSignal) { + return ROUTE_DISCIPLINE_RULE_MAP.get("bidirectional_value_flow")!; + } if (ambiguitySignal && (symptomSignal || lifecycleSignal || chainBreakSignal || periodImpactSignal || causalSignal)) { return ROUTE_DISCIPLINE_RULE_MAP.get("mixed_ambiguity")!; } @@ -272,7 +304,8 @@ function shouldPromoteFromNoRoute(fragment: V2FamilyFragment, rule: RouteDiscipl hasLifecycleSignal(fragment, lowerText) || hasChainBreakSignal(lowerText) || hasPeriodImpactSignal(lowerText) || - hasCausalSignal(lowerText); + hasCausalSignal(lowerText) || + (hasBidirectionalValueFlowSignal(fragment, lowerText) && hasCounterpartyScopeSignal(fragment, lowerText)); const hasAnchor = hasAccountOrPeriodAnchor(fragment, lowerText) || diff --git a/llm_normalizer/backend/tests/addressQueryRuntimeM23.test.ts b/llm_normalizer/backend/tests/addressQueryRuntimeM23.test.ts index be0c47e..c039601 100644 --- a/llm_normalizer/backend/tests/addressQueryRuntimeM23.test.ts +++ b/llm_normalizer/backend/tests/addressQueryRuntimeM23.test.ts @@ -587,6 +587,7 @@ describe("address compose stage utf8 headers", () => { expect(reply.text).toContain("не подтвержденная клиентская выручка"); expect(reply.text).toContain("Сводка по направлению"); expect(reply.text).toContain("Основание 1С"); + expect(reply.text).toContain("По смыслу это скорее финансовый/банковский контур"); expect(reply.text).toContain("вид операции/назначение платежа/договор"); }); @@ -613,6 +614,8 @@ describe("address compose stage utf8 headers", () => { expect(reply.text).toContain("Сводка по направлению"); expect(reply.text).toContain("Это не обычный поставщик автоматически"); + expect(reply.text).toContain("Примеры строк 1С:"); + expect(reply.text).toContain("Следующий шаг: могу отдельно разложить назначения платежа"); expect(reply.text).toContain("Показаны первые 5 из 8"); expect(reply.text).not.toContain("00000000007"); }); diff --git a/llm_normalizer/backend/tests/addressReplyBuildersRegression.test.ts b/llm_normalizer/backend/tests/addressReplyBuildersRegression.test.ts index 38b439a..f434dec 100644 --- a/llm_normalizer/backend/tests/addressReplyBuildersRegression.test.ts +++ b/llm_normalizer/backend/tests/addressReplyBuildersRegression.test.ts @@ -443,4 +443,86 @@ describe("address reply builders regressions", () => { expect(result.text.split("\n")[0]).toContain("1.000,00"); expect(result.text).not.toContain("встречных остатков"); }); + it("avoids mixed stock quantity as a fake management KPI in on-hand snapshot answers", () => { + const result = composeInventoryReply( + "inventory_on_hand_as_of_date", + [ + { + amount: 695360, + quantity: 3, + item: "Рабочая станция", + warehouse: "Основной склад", + organization: 'ООО "Альтернатива Плюс"', + period: "2017-06-30", + registrator: "Остатки" + } as any, + { + amount: 295526.51, + quantity: 317000, + item: "Лифлеты", + warehouse: "Основной склад", + organization: 'ООО "Альтернатива Плюс"', + period: "2017-06-30", + registrator: "Остатки" + } as any + ], + { + userMessage: "какие остатки на июнь 2017", + asOfDate: "2017-06-30" + }, + { + resolvePayablesAsOfDate: () => "2017-06-30", + buildInventoryOnHandAggregate: () => [ + { + item: "Рабочая станция", + warehouse: "Основной склад", + organization: 'ООО "Альтернатива Плюс"', + quantity: 3, + amount: 695360, + operations: 1, + firstPeriod: "2017-06-30", + lastPeriod: "2017-06-30", + sourceRefs: [] + }, + { + item: "Лифлеты", + warehouse: "Основной склад", + organization: 'ООО "Альтернатива Плюс"', + quantity: 317000, + amount: 295526.51, + operations: 1, + firstPeriod: "2017-06-30", + lastPeriod: "2017-06-30", + sourceRefs: [] + } + ], + uniqueStrings: (values: string[]) => Array.from(new Set(values)), + formatDateRu: (value: string) => value, + formatNumberWithDots: (value: number, fractionDigits = 0) => value.toFixed(fractionDigits), + formatMoneyRub: (value: number) => `${value} ₽`, + isInventoryPurchaseMovement: () => false, + summarizeInventoryTraceRows: () => ({ + item: null, + warehouses: [], + organizations: [], + counterparties: [], + documents: [], + firstPeriod: null, + lastPeriod: null, + totalAmount: 0 + }), + formatInventoryTraceRows: () => [], + hasInventoryPurchaseDateActionFocus: () => false, + inventoryTraceDateLabel: () => "", + extractInventoryCounterpartyCandidates: () => [], + buildInventoryAgingByItemAggregate: () => [], + formatInventoryAgingRows: () => [], + isInventorySaleMovement: () => false + } + ); + + expect(result?.text).not.toContain("Суммарное количество"); + expect(result?.text).toContain("Общее количество не свожу в один управленческий показатель"); + expect(result?.text).toContain("Следующий шаг: могу раскрыть полный список"); + }); }); diff --git a/llm_normalizer/backend/tests/assistantMcpDiscoveryTurnInputAdapter.test.ts b/llm_normalizer/backend/tests/assistantMcpDiscoveryTurnInputAdapter.test.ts index d907012..7b0f4fc 100644 --- a/llm_normalizer/backend/tests/assistantMcpDiscoveryTurnInputAdapter.test.ts +++ b/llm_normalizer/backend/tests/assistantMcpDiscoveryTurnInputAdapter.test.ts @@ -3342,6 +3342,107 @@ describe("assistant MCP discovery turn input adapter", () => { expect(result.reason_codes).not.toContain("mcp_discovery_counterparty_from_followup_context"); }); + it("pivots a business-overview follow-up with explicit counterparty net-flow wording back to counterparty value flow", () => { + const orgName = + "\u041e\u041e\u041e \u0410\u043b\u044c\u0442\u0435\u0440\u043d\u0430\u0442\u0438\u0432\u0430 \u041f\u043b\u044e\u0441"; + const counterpartyName = "\u0413\u0440\u0443\u043f\u043f\u0430 \u0421\u0412\u041a"; + const result = buildAssistantMcpDiscoveryTurnInput({ + userMessage: + "\u0410 \u0442\u0435\u043f\u0435\u0440\u044c \u043f\u043e \u0413\u0440\u0443\u043f\u043f\u0430 \u0421\u0412\u041a \u0437\u0430 2020: \u0441\u043a\u043e\u043b\u044c\u043a\u043e \u0434\u0435\u043d\u0435\u0433 \u043f\u043e\u043b\u0443\u0447\u0438\u043b\u0438, \u0441\u043a\u043e\u043b\u044c\u043a\u043e \u0437\u0430\u043f\u043b\u0430\u0442\u0438\u043b\u0438 \u0438 \u043a\u0430\u043a\u043e\u0435 \u043d\u0435\u0442\u0442\u043e?", + assistantTurnMeaning: { + asked_domain_family: "unknown", + asked_action_family: "unknown" + }, + followupContext: { + previous_discovery_pilot_scope: "business_overview_route_template_v1", + previous_filters: { + organization: orgName, + period_from: "2020-01-01", + period_to: "2020-12-31" + } + } + }); + + expect(result.adapter_status).toBe("ready"); + expect(result.should_run_discovery).toBe(true); + expect(result.semantic_data_need).toBe("counterparty value-flow evidence"); + expect(result.data_need_graph?.business_fact_family).toBe("value_flow"); + expect(result.data_need_graph?.subject_candidates).toEqual([counterpartyName]); + expect(result.turn_meaning_ref).toMatchObject({ + asked_domain_family: "counterparty_value", + asked_action_family: "net_value_flow", + explicit_entity_candidates: [counterpartyName], + explicit_organization_scope: orgName, + explicit_date_scope: "2020", + unsupported_but_understood_family: "counterparty_bidirectional_value_flow_or_netting", + stale_replay_forbidden: true + }); + expect(result.reason_codes).toContain( + "mcp_discovery_business_overview_followup_pivoted_to_counterparty_value_flow" + ); + expect(result.reason_codes).toContain("mcp_discovery_business_overview_continuation_from_followup_context"); + expect(result.reason_codes).toContain("mcp_discovery_counterparty_from_raw_scope"); + expect(result.reason_codes).not.toContain("mcp_discovery_broad_business_evaluation_route_candidate"); + expect(result.reason_codes).not.toContain("mcp_discovery_business_overview_suppressed_stale_counterparty"); + }); + + it("survives poisoned business-overview turn meaning when predecompose still carries the counterparty", () => { + const orgName = + "\u041e\u041e\u041e \u0410\u043b\u044c\u0442\u0435\u0440\u043d\u0430\u0442\u0438\u0432\u0430 \u041f\u043b\u044e\u0441"; + const counterpartyName = "\u0413\u0440\u0443\u043f\u043f\u0430 \u0421\u0412\u041a"; + const result = buildAssistantMcpDiscoveryTurnInput({ + userMessage: + "\u0410 \u0442\u0435\u043f\u0435\u0440\u044c \u043f\u043e \u0413\u0440\u0443\u043f\u043f\u0430 \u0421\u0412\u041a \u0437\u0430 2020: \u0441\u043a\u043e\u043b\u044c\u043a\u043e \u0434\u0435\u043d\u0435\u0433 \u043f\u043e\u043b\u0443\u0447\u0438\u043b\u0438, \u0441\u043a\u043e\u043b\u044c\u043a\u043e \u0437\u0430\u043f\u043b\u0430\u0442\u0438\u043b\u0438 \u0438 \u043a\u0430\u043a\u043e\u0435 \u043d\u0435\u0442\u0442\u043e?", + effectiveMessage: + "\u041e\u043f\u0440\u0435\u0434\u0435\u043b\u0438\u0442\u044c \u0441\u0443\u043c\u043c\u0443 \u043f\u043e\u043b\u0443\u0447\u0435\u043d\u043d\u044b\u0445 \u0441\u0440\u0435\u0434\u0441\u0442\u0432, \u0441\u0443\u043c\u043c\u0443 \u0432\u044b\u043f\u043b\u0430\u0447\u0435\u043d\u043d\u044b\u0445 \u0441\u0440\u0435\u0434\u0441\u0442\u0432 \u0438 \u0447\u0438\u0441\u0442\u0443\u044e \u043f\u0440\u0438\u0431\u044b\u043b\u044c (\u043d\u0435\u0442\u0442\u043e) \u0434\u043b\u044f \u0433\u0440\u0443\u043f\u043f\u044b \u043a\u043e\u043c\u043f\u0430\u043d\u0438\u0439 \u0421\u0412\u041a \u0437\u0430 2020 \u0433\u043e\u0434.", + assistantTurnMeaning: { + asked_domain_family: "business_overview", + asked_action_family: "profit_margin_boundary", + explicit_date_scope: "2020", + unsupported_but_understood_family: "profit_margin_boundary", + stale_replay_forbidden: true + }, + predecomposeContract: { + entities: { + counterparty: counterpartyName + }, + period: { + scope: "range", + period_from: "2020-01-01", + period_to: "2020-12-31" + } + }, + followupContext: { + previous_discovery_pilot_scope: "business_overview_route_template_v1", + previous_filters: { + organization: orgName, + period_from: "2020-01-01", + period_to: "2020-12-31" + } + } + }); + + expect(result.adapter_status).toBe("ready"); + expect(result.should_run_discovery).toBe(true); + expect(result.semantic_data_need).toBe("counterparty value-flow evidence"); + expect(result.data_need_graph?.business_fact_family).toBe("value_flow"); + expect(result.data_need_graph?.subject_candidates).toEqual([counterpartyName]); + expect(result.turn_meaning_ref).toMatchObject({ + asked_domain_family: "counterparty_value", + asked_action_family: "net_value_flow", + explicit_entity_candidates: [counterpartyName], + explicit_organization_scope: orgName, + explicit_date_scope: "2020", + unsupported_but_understood_family: "counterparty_bidirectional_value_flow_or_netting", + stale_replay_forbidden: true + }); + expect(result.reason_codes).toContain( + "mcp_discovery_business_overview_followup_pivoted_to_counterparty_value_flow" + ); + expect(result.reason_codes).toContain("mcp_discovery_counterparty_from_predecompose"); + expect(result.reason_codes).not.toContain("mcp_discovery_broad_business_evaluation_route_candidate"); + }); + it("lets an explicit VAT follow-up stay on the exact VAT route instead of stale business overview", () => { const orgName = "\u041e\u041e\u041e \u0410\u043b\u044c\u0442\u0435\u0440\u043d\u0430\u0442\u0438\u0432\u0430 \u041f\u043b\u044e\u0441"; diff --git a/llm_normalizer/backend/tests/assistantTurnMeaningPolicy.test.ts b/llm_normalizer/backend/tests/assistantTurnMeaningPolicy.test.ts index 22831f5..20eae08 100644 --- a/llm_normalizer/backend/tests/assistantTurnMeaningPolicy.test.ts +++ b/llm_normalizer/backend/tests/assistantTurnMeaningPolicy.test.ts @@ -57,6 +57,34 @@ describe("assistantTurnMeaningPolicy", () => { ]); }); + it("keeps counterparty net-flow wording in counterparty semantics even if effective text mentions profit", () => { + const policy = buildPolicy({ + resolveAddressIntent: () => ({ intent: "unknown", confidence: "low" }) + }); + + const meaning = policy.resolveAssistantTurnMeaning({ + rawUserMessage: + "\u0410 \u0442\u0435\u043f\u0435\u0440\u044c \u043f\u043e \u0413\u0440\u0443\u043f\u043f\u0430 \u0421\u0412\u041a \u0437\u0430 2020: \u0441\u043a\u043e\u043b\u044c\u043a\u043e \u0434\u0435\u043d\u0435\u0433 \u043f\u043e\u043b\u0443\u0447\u0438\u043b\u0438, \u0441\u043a\u043e\u043b\u044c\u043a\u043e \u0437\u0430\u043f\u043b\u0430\u0442\u0438\u043b\u0438 \u0438 \u043a\u0430\u043a\u043e\u0435 \u043d\u0435\u0442\u0442\u043e?", + effectiveAddressUserMessage: + "\u041e\u043f\u0440\u0435\u0434\u0435\u043b\u0438\u0442\u044c \u0441\u0443\u043c\u043c\u0443 \u043f\u043e\u043b\u0443\u0447\u0435\u043d\u043d\u044b\u0445 \u0441\u0440\u0435\u0434\u0441\u0442\u0432, \u0441\u0443\u043c\u043c\u0443 \u0432\u044b\u043f\u043b\u0430\u0447\u0435\u043d\u043d\u044b\u0445 \u0441\u0440\u0435\u0434\u0441\u0442\u0432 \u0438 \u0447\u0438\u0441\u0442\u0443\u044e \u043f\u0440\u0438\u0431\u044b\u043b\u044c (\u043d\u0435\u0442\u0442\u043e) \u0434\u043b\u044f \u0413\u0440\u0443\u043f\u043f\u0430 \u0421\u0412\u041a \u0437\u0430 2020 \u0433\u043e\u0434." + }); + + expect(meaning.explicit_intent_candidate).toBeNull(); + expect(meaning.asked_domain_family).toBe("counterparty_value"); + expect(meaning.asked_action_family).toBe("net_value_flow"); + expect(meaning.unsupported_but_understood_family).toBe("counterparty_bidirectional_value_flow_or_netting"); + expect(meaning.explicit_entity_candidates).toEqual([ + { + type: "counterparty", + value: "\u0433\u0440\u0443\u043f\u043f\u0430 \u0441\u0432\u043a", + source: "current_turn_loose_entity_tail" + } + ]); + expect(meaning.reason_codes).toContain("counterparty_bidirectional_value_flow_current_turn_signal"); + expect(meaning.reason_codes).not.toContain("broad_business_evaluation_current_turn_signal"); + expect(meaning.stale_replay_forbidden).toBe(true); + }); + it("ignores temporal tail words in all-time revenue ranking questions", () => { const policy = buildPolicy({ resolveAddressIntent: (text: string) => diff --git a/llm_normalizer/backend/tests/counterpartyAnalyticsReplyBuilders.test.ts b/llm_normalizer/backend/tests/counterpartyAnalyticsReplyBuilders.test.ts index 772142d..2ef757b 100644 --- a/llm_normalizer/backend/tests/counterpartyAnalyticsReplyBuilders.test.ts +++ b/llm_normalizer/backend/tests/counterpartyAnalyticsReplyBuilders.test.ts @@ -109,7 +109,8 @@ describe("counterparty analytics reply builders", () => { expect(reply.responseType).toBe("FACTUAL_SUMMARY"); expect(reply.text).toContain("Оборот по СВК за доступное время: 3.500,00"); expect(reply.text).toContain("по 2 подтвержденным входящим операциям"); - expect(reply.text).toContain("Это денежный поток от клиента, а не чистая прибыль"); + expect(reply.text).toContain("Граница ответа: это подтвержденный денежный поток по поступлениям, а не чистая прибыль."); + expect(reply.text).toContain("Следующий шаг: могу разложить поток по месяцам, документам или контрагентам."); expect(reply.text).not.toContain("Самый доходный клиент"); expect(reply.text).not.toContain("Топ-"); }); diff --git a/llm_normalizer/backend/tests/routeHintAdapter.test.ts b/llm_normalizer/backend/tests/routeHintAdapter.test.ts index 67bdd7b..1ff8b1f 100644 --- a/llm_normalizer/backend/tests/routeHintAdapter.test.ts +++ b/llm_normalizer/backend/tests/routeHintAdapter.test.ts @@ -225,4 +225,63 @@ describe("routeHintAdapter", () => { } expect(summary.decisions[0]?.route).toBe("store_canonical"); }); + + it("routes counterparty received-paid-net wording to hybrid instead of canonical fact lookup", () => { + const summary = toRouteHintSummary({ + schema_version: "normalized_query_v2_0_2", + user_message_raw: "А теперь по Группа СВК за 2020: сколько денег получили, сколько заплатили и какое нетто?", + message_in_scope: true, + scope_confidence: "high", + contains_multiple_tasks: false, + fragments: [ + { + fragment_id: "F1", + raw_fragment_text: "А теперь по Группа СВК за 2020: сколько денег получили, сколько заплатили и какое нетто?", + normalized_fragment_text: + "Определить сумму полученных средств, сумму выплаченных средств и чистый остаток (нетто) для контрагента Группа СВК за период 2020 года.", + domain_relevance: "in_scope", + business_scope: "company_specific_accounting", + entity_hints: ["Группа СВК"], + account_hints: [], + document_hints: [], + register_hints: [], + time_scope: { + type: "explicit", + value: "2020", + confidence: "high" + }, + flags: { + has_multi_entity_scope: false, + asks_for_chain_explanation: false, + asks_for_ranking_or_top: false, + asks_for_period_summary: false, + asks_for_rule_check: false, + asks_for_anomaly_scan: false, + asks_for_exact_object_trace: false, + asks_for_evidence: false, + mentions_period_close_context: false + }, + candidate_labels: ["simple_factual"], + confidence: "high", + execution_readiness: "executable", + clarification_reason: null, + soft_assumption_used: [], + route_status: "routed", + no_route_reason: null + } + ], + discarded_fragments: [], + global_notes: { + needs_clarification: false, + clarification_reason: null + } + }); + + expect(summary.mode).toBe("deterministic_v2"); + if (summary.mode !== "deterministic_v2") { + throw new Error("Expected deterministic_v2 summary"); + } + expect(summary.decisions[0]?.route).toBe("hybrid_store_plus_live"); + expect(summary.decisions[0]?.reason).toContain("bidirectional_value_flow"); + }); }); diff --git a/llm_normalizer/frontend/dist/assets/index-6meEannb.js b/llm_normalizer/frontend/dist/assets/index-6meEannb.js deleted file mode 100644 index 53bd011..0000000 --- a/llm_normalizer/frontend/dist/assets/index-6meEannb.js +++ /dev/null @@ -1,24 +0,0 @@ -(function(){const g=document.createElement("link").relList;if(g&&g.supports&&g.supports("modulepreload"))return;for(const E of document.querySelectorAll('link[rel="modulepreload"]'))R(E);new MutationObserver(E=>{for(const L of E)if(L.type==="childList")for(const B of L.addedNodes)B.tagName==="LINK"&&B.rel==="modulepreload"&&R(B)}).observe(document,{childList:!0,subtree:!0});function p(E){const L={};return E.integrity&&(L.integrity=E.integrity),E.referrerPolicy&&(L.referrerPolicy=E.referrerPolicy),E.crossOrigin==="use-credentials"?L.credentials="include":E.crossOrigin==="anonymous"?L.credentials="omit":L.credentials="same-origin",L}function R(E){if(E.ep)return;E.ep=!0;const L=p(E);fetch(E.href,L)}})();function yd(i){return i&&i.__esModule&&Object.prototype.hasOwnProperty.call(i,"default")?i.default:i}var Ai={exports:{}},Fo={},Mi={exports:{}},pe={};var Wc;function wf(){if(Wc)return pe;Wc=1;var i=Symbol.for("react.element"),g=Symbol.for("react.portal"),p=Symbol.for("react.fragment"),R=Symbol.for("react.strict_mode"),E=Symbol.for("react.profiler"),L=Symbol.for("react.provider"),B=Symbol.for("react.context"),ie=Symbol.for("react.forward_ref"),ne=Symbol.for("react.suspense"),z=Symbol.for("react.memo"),X=Symbol.for("react.lazy"),Z=Symbol.iterator;function te(y){return y===null||typeof y!="object"?null:(y=Z&&y[Z]||y["@@iterator"],typeof y=="function"?y:null)}var Re={isMounted:function(){return!1},enqueueForceUpdate:function(){},enqueueReplaceState:function(){},enqueueSetState:function(){}},de=Object.assign,fe={};function he(y,k,se){this.props=y,this.context=k,this.refs=fe,this.updater=se||Re}he.prototype.isReactComponent={},he.prototype.setState=function(y,k){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,k,"setState")},he.prototype.forceUpdate=function(y){this.updater.enqueueForceUpdate(this,y,"forceUpdate")};function He(){}He.prototype=he.prototype;function Ye(y,k,se){this.props=y,this.context=k,this.refs=fe,this.updater=se||Re}var Ve=Ye.prototype=new He;Ve.constructor=Ye,de(Ve,he.prototype),Ve.isPureReactComponent=!0;var Me=Array.isArray,$=Object.prototype.hasOwnProperty,ae={current:null},Ne={key:!0,ref:!0,__self:!0,__source:!0};function be(y,k,se){var ue,ce={},ge=null,_e=null;if(k!=null)for(ue in k.ref!==void 0&&(_e=k.ref),k.key!==void 0&&(ge=""+k.key),k)$.call(k,ue)&&!Ne.hasOwnProperty(ue)&&(ce[ue]=k[ue]);var Se=arguments.length-2;if(Se===1)ce.children=se;else if(1>>1,k=M[y];if(0>>1;yE(ce,I))geE(_e,ce)?(M[y]=_e,M[ge]=I,y=ge):(M[y]=ce,M[ue]=I,y=ue);else if(geE(_e,I))M[y]=_e,M[ge]=I,y=ge;else break e}}return F}function E(M,F){var I=M.sortIndex-F.sortIndex;return I!==0?I:M.id-F.id}if(typeof performance=="object"&&typeof performance.now=="function"){var L=performance;i.unstable_now=function(){return L.now()}}else{var B=Date,ie=B.now();i.unstable_now=function(){return B.now()-ie}}var ne=[],z=[],X=1,Z=null,te=3,Re=!1,de=!1,fe=!1,he=typeof setTimeout=="function"?setTimeout:null,He=typeof clearTimeout=="function"?clearTimeout:null,Ye=typeof setImmediate<"u"?setImmediate:null;typeof navigator<"u"&&navigator.scheduling!==void 0&&navigator.scheduling.isInputPending!==void 0&&navigator.scheduling.isInputPending.bind(navigator.scheduling);function Ve(M){for(var F=p(z);F!==null;){if(F.callback===null)R(z);else if(F.startTime<=M)R(z),F.sortIndex=F.expirationTime,g(ne,F);else break;F=p(z)}}function Me(M){if(fe=!1,Ve(M),!de)if(p(ne)!==null)de=!0,re($);else{var F=p(z);F!==null&&ee(Me,F.startTime-M)}}function $(M,F){de=!1,fe&&(fe=!1,He(be),be=-1),Re=!0;var I=te;try{for(Ve(F),Z=p(ne);Z!==null&&(!(Z.expirationTime>F)||M&&!Kt());){var y=Z.callback;if(typeof y=="function"){Z.callback=null,te=Z.priorityLevel;var k=y(Z.expirationTime<=F);F=i.unstable_now(),typeof k=="function"?Z.callback=k:Z===p(ne)&&R(ne),Ve(F)}else R(ne);Z=p(ne)}if(Z!==null)var se=!0;else{var ue=p(z);ue!==null&&ee(Me,ue.startTime-F),se=!1}return se}finally{Z=null,te=I,Re=!1}}var ae=!1,Ne=null,be=-1,yt=5,dt=-1;function Kt(){return!(i.unstable_now()-dtM||125y?(M.sortIndex=I,g(z,M),p(ne)===null&&M===p(z)&&(fe?(He(be),be=-1):fe=!0,ee(Me,I-y))):(M.sortIndex=k,g(ne,M),de||Re||(de=!0,re($))),M},i.unstable_shouldYield=Kt,i.unstable_wrapCallback=function(M){var F=te;return function(){var I=te;te=F;try{return M.apply(this,arguments)}finally{te=I}}}})(Di)),Di}var Xc;function Ef(){return Xc||(Xc=1,Li.exports=Nf()),Li.exports}var Zc;function Pf(){if(Zc)return qt;Zc=1;var i=Vi(),g=Ef();function p(e){for(var t="https://reactjs.org/docs/error-decoder.html?invariant="+e,n=1;n"u"||typeof window.document>"u"||typeof window.document.createElement>"u"),ne=Object.prototype.hasOwnProperty,z=/^[: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]*$/,X={},Z={};function te(e){return ne.call(Z,e)?!0:ne.call(X,e)?!1:z.test(e)?Z[e]=!0:(X[e]=!0,!1)}function Re(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 de(e,t,n,r){if(t===null||typeof t>"u"||Re(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 fe(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 he={};"children dangerouslySetInnerHTML defaultValue defaultChecked innerHTML suppressContentEditableWarning suppressHydrationWarning style".split(" ").forEach(function(e){he[e]=new fe(e,0,!1,e,null,!1,!1)}),[["acceptCharset","accept-charset"],["className","class"],["htmlFor","for"],["httpEquiv","http-equiv"]].forEach(function(e){var t=e[0];he[t]=new fe(t,1,!1,e[1],null,!1,!1)}),["contentEditable","draggable","spellCheck","value"].forEach(function(e){he[e]=new fe(e,2,!1,e.toLowerCase(),null,!1,!1)}),["autoReverse","externalResourcesRequired","focusable","preserveAlpha"].forEach(function(e){he[e]=new fe(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){he[e]=new fe(e,3,!1,e.toLowerCase(),null,!1,!1)}),["checked","multiple","muted","selected"].forEach(function(e){he[e]=new fe(e,3,!0,e,null,!1,!1)}),["capture","download"].forEach(function(e){he[e]=new fe(e,4,!1,e,null,!1,!1)}),["cols","rows","size","span"].forEach(function(e){he[e]=new fe(e,6,!1,e,null,!1,!1)}),["rowSpan","start"].forEach(function(e){he[e]=new fe(e,5,!1,e.toLowerCase(),null,!1,!1)});var He=/[\-:]([a-z])/g;function Ye(e){return e[1].toUpperCase()}"accent-height alignment-baseline arabic-form baseline-shift cap-height clip-path clip-rule color-interpolation color-interpolation-filters color-profile color-rendering dominant-baseline enable-background fill-opacity fill-rule flood-color flood-opacity font-family font-size font-size-adjust font-stretch font-style font-variant font-weight glyph-name glyph-orientation-horizontal glyph-orientation-vertical horiz-adv-x horiz-origin-x image-rendering letter-spacing lighting-color marker-end marker-mid marker-start overline-position overline-thickness paint-order panose-1 pointer-events rendering-intent shape-rendering stop-color stop-opacity strikethrough-position strikethrough-thickness stroke-dasharray stroke-dashoffset stroke-linecap stroke-linejoin stroke-miterlimit stroke-opacity stroke-width text-anchor text-decoration text-rendering underline-position underline-thickness unicode-bidi unicode-range units-per-em v-alphabetic v-hanging v-ideographic v-mathematical vector-effect vert-adv-y vert-origin-x vert-origin-y word-spacing writing-mode xmlns:xlink x-height".split(" ").forEach(function(e){var t=e.replace(He,Ye);he[t]=new fe(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(He,Ye);he[t]=new fe(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(He,Ye);he[t]=new fe(t,1,!1,e,"http://www.w3.org/XML/1998/namespace",!1,!1)}),["tabIndex","crossOrigin"].forEach(function(e){he[e]=new fe(e,1,!1,e.toLowerCase(),null,!1,!1)}),he.xlinkHref=new fe("xlinkHref",1,!1,"xlink:href","http://www.w3.org/1999/xlink",!0,!1),["src","href","action","formAction"].forEach(function(e){he[e]=new fe(e,1,!1,e.toLowerCase(),null,!0,!0)});function Ve(e,t,n,r){var s=he.hasOwnProperty(t)?he[t]:null;(s!==null?s.type!==0:r||!(2d||s[u]!==l[d]){var m=` -`+s[u].replace(" at new "," at ");return e.displayName&&m.includes("")&&(m=m.replace("",e.displayName)),m}while(1<=u&&0<=d);break}}}finally{se=!1,Error.prepareStackTrace=n}return(e=e?e.displayName||e.name:"")?k(e):""}function ce(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 ge(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 Ne:return"Fragment";case ae:return"Portal";case yt:return"Profiler";case be:return"StrictMode";case b:return"Suspense";case Ie:return"SuspenseList"}if(typeof e=="object")switch(e.$$typeof){case Kt:return(e.displayName||"Context")+".Consumer";case dt:return(e._context.displayName||"Context")+".Provider";case ft:var t=e.render;return e=e.displayName,e||(e=t.displayName||t.name||"",e=e!==""?"ForwardRef("+e+")":"ForwardRef"),e;case Ke:return t=e.displayName||null,t!==null?t:ge(e.type)||"Memo";case re:t=e._payload,e=e._init;try{return ge(e(t))}catch{}}return null}function _e(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 ge(t);case 8:return t===be?"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 Se(e){switch(typeof e){case"boolean":case"number":case"string":case"undefined":return e;case"object":return e;default:return""}}function ke(e){var t=e.type;return(e=e.nodeName)&&e.toLowerCase()==="input"&&(t==="checkbox"||t==="radio")}function pt(e){var t=ke(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 ln(e){e._valueTracker||(e._valueTracker=pt(e))}function vr(e){if(!e)return!1;var t=e._valueTracker;if(!t)return!0;var n=t.getValue(),r="";return e&&(r=ke(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 xn(e,t){var n=t.checked;return I({},t,{defaultChecked:void 0,defaultValue:void 0,value:void 0,checked:n??e._wrapperState.initialChecked})}function Te(e,t){var n=t.defaultValue==null?"":t.defaultValue,r=t.checked!=null?t.checked:t.defaultChecked;n=Se(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 Jt(e,t){t=t.checked,t!=null&&Ve(e,"checked",t,!1)}function lt(e,t){Jt(e,t);var n=Se(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")?Pt(e,t.type,n):t.hasOwnProperty("defaultValue")&&Pt(e,t.type,Se(t.defaultValue)),t.checked==null&&t.defaultChecked!=null&&(e.defaultChecked=!!t.defaultChecked)}function xt(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 Pt(e,t,n){(t!=="number"||In(e.ownerDocument)!==e)&&(n==null?e.defaultValue=""+e._wrapperState.initialValue:e.defaultValue!==""+n&&(e.defaultValue=""+n))}var _n=Array.isArray;function W(e,t,n,r){if(e=e.options,t){t={};for(var s=0;s"+t.valueOf().toString()+"",t=Dn.firstChild;e.firstChild;)e.removeChild(e.firstChild);for(;t.firstChild;)e.appendChild(t.firstChild)}});function Rt(e,t){if(t){var n=e.firstChild;if(n&&n===e.lastChild&&n.nodeType===3){n.nodeValue=t;return}}e.textContent=t}var Xe={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},_s=["Webkit","ms","Moz","O"];Object.keys(Xe).forEach(function(e){_s.forEach(function(t){t=t+e.charAt(0).toUpperCase()+e.substring(1),Xe[t]=Xe[e]})});function tt(e,t,n){return t==null||typeof t=="boolean"||t===""?"":n||typeof t!="number"||t===0||Xe.hasOwnProperty(e)&&Xe[e]?(""+t).trim():t+"px"}function er(e,t){e=e.style;for(var n in t)if(t.hasOwnProperty(n)){var r=n.indexOf("--")===0,s=tt(n,t[n],r);n==="float"&&(n="cssFloat"),r?e.setProperty(n,s):e[n]=s}}var xr=I({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 an(e,t){if(t){if(xr[e]&&(t.children!=null||t.dangerouslySetInnerHTML!=null))throw Error(p(137,e));if(t.dangerouslySetInnerHTML!=null){if(t.children!=null)throw Error(p(60));if(typeof t.dangerouslySetInnerHTML!="object"||!("__html"in t.dangerouslySetInnerHTML))throw Error(p(61))}if(t.style!=null&&typeof t.style!="object")throw Error(p(62))}}function On(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 Ut=null;function Sn(e){return e=e.target||e.srcElement||window,e.correspondingUseElement&&(e=e.correspondingUseElement),e.nodeType===3?e.parentNode:e}var Yt=null,q=null,At=null;function tr(e){if(e=jo(e)){if(typeof Yt!="function")throw Error(p(280));var t=e.stateNode;t&&(t=dl(t),Yt(e.stateNode,e.type,t))}}function Hr(e){q?At?At.push(e):At=[e]:q=e}function Vr(){if(q){var e=q,t=At;if(At=q=null,tr(e),t)for(e=0;e>>=0,e===0?32:31-(nr(e)/aa|0)|0}var js=64,wn=4194304;function rr(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 Zr(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=rr(d):(l&=u,l!==0&&(r=rr(l)))}else u=n&~s,u!==0?r=rr(u):l!==0&&(r=rr(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 wr(e,t,n){e.pendingLanes|=t,t!==536870912&&(e.suspendedLanes=0,e.pingedLanes=0),e=e.eventTimes,t=31-Zt(t),e[t]=n}function kr(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=Tr),mo=" ",rl=!1;function ho(e,t){switch(e){case"keyup":return el.indexOf(t.keyCode)!==-1;case"keydown":return t.keyCode!==229;case"keypress":case"mousedown":case"focusout":return!0;default:return!1}}function go(e){return e=e.detail,typeof e=="object"&&"data"in e?e.data:null}var ar=!1;function sl(e,t){switch(e){case"compositionend":return go(t);case"keypress":return t.which!==32?null:(rl=!0,mo);case"textInput":return e=t.data,e===mo&&rl?null:e;default:return null}}function a(e,t){if(ar)return e==="compositionend"||!Ds&&ho(e,t)?(e=os(),or=Nr=It=null,ar=!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=Ji(n)}}function Xi(e,t){return e&&t?e===t?!0:e&&e.nodeType===3?!1:t&&t.nodeType===3?Xi(e,t.parentNode):"contains"in e?e.contains(t):e.compareDocumentPosition?!!(e.compareDocumentPosition(t)&16):!1:!1}function Zi(){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 ma(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=Zi(),n=e.focusedElem,r=e.selectionRange;if(t!==n&&n&&n.ownerDocument&&Xi(n.ownerDocument.documentElement,n)){if(r!==null&&ma(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=Yi(n,l);var u=Yi(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,zs=null,ha=null,xo=null,ga=!1;function eu(e,t,n){var r=n.window===n?n.document:n.nodeType===9?n:n.ownerDocument;ga||zs==null||zs!==In(r)||(r=zs,"selectionStart"in r&&ma(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}),xo&&yo(xo,r)||(xo=r,r=il(ha,"onSelect"),0Bs||(e.current=Pa[Bs],Pa[Bs]=null,Bs--)}function Oe(e,t){Bs++,Pa[Bs]=e.current,e.current=t}var Ir={},kt=Mr(Ir),Qt=Mr(!1),us=Ir;function Qs(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 Ht(e){return e=e.childContextTypes,e!=null}function fl(){Fe(Qt),Fe(kt)}function hu(e,t,n){if(kt.current!==Ir)throw Error(p(168));Oe(kt,t),Oe(Qt,n)}function gu(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(p(108,_e(e)||"Unknown",s));return I({},n,r)}function pl(e){return e=(e=e.stateNode)&&e.__reactInternalMemoizedMergedChildContext||Ir,us=kt.current,Oe(kt,e),Oe(Qt,Qt.current),!0}function vu(e,t,n){var r=e.stateNode;if(!r)throw Error(p(169));n?(e=gu(e,t,us),r.__reactInternalMemoizedMergedChildContext=e,Fe(Qt),Fe(kt),Oe(kt,e)):Fe(Qt),Oe(Qt,n)}var cr=null,ml=!1,Ta=!1;function yu(e){cr===null?cr=[e]:cr.push(e)}function Ud(e){ml=!0,yu(e)}function Lr(){if(!Ta&&cr!==null){Ta=!0;var e=0,t=ve;try{var n=cr;for(ve=1;e>=u,s-=u,dr=1<<32-Zt(t)+s|n<oe?(ct=J,J=null):ct=J.sibling;var Ce=j(x,J,_[oe],A);if(Ce===null){J===null&&(J=ct);break}e&&J&&Ce.alternate===null&&t(x,J),v=l(Ce,v,oe),K===null?V=Ce:K.sibling=Ce,K=Ce,J=ct}if(oe===_.length)return n(x,J),Ue&&ds(x,oe),V;if(J===null){for(;oe<_.length;oe++)J=P(x,_[oe],A),J!==null&&(v=l(J,v,oe),K===null?V=J:K.sibling=J,K=J);return Ue&&ds(x,oe),V}for(J=r(x,J);oe<_.length;oe++)ct=D(J,x,oe,_[oe],A),ct!==null&&(e&&ct.alternate!==null&&J.delete(ct.key===null?oe:ct.key),v=l(ct,v,oe),K===null?V=ct:K.sibling=ct,K=ct);return e&&J.forEach(function(Qr){return t(x,Qr)}),Ue&&ds(x,oe),V}function Q(x,v,_,A){var V=F(_);if(typeof V!="function")throw Error(p(150));if(_=V.call(_),_==null)throw Error(p(151));for(var K=V=null,J=v,oe=v=0,ct=null,Ce=_.next();J!==null&&!Ce.done;oe++,Ce=_.next()){J.index>oe?(ct=J,J=null):ct=J.sibling;var Qr=j(x,J,Ce.value,A);if(Qr===null){J===null&&(J=ct);break}e&&J&&Qr.alternate===null&&t(x,J),v=l(Qr,v,oe),K===null?V=Qr:K.sibling=Qr,K=Qr,J=ct}if(Ce.done)return n(x,J),Ue&&ds(x,oe),V;if(J===null){for(;!Ce.done;oe++,Ce=_.next())Ce=P(x,Ce.value,A),Ce!==null&&(v=l(Ce,v,oe),K===null?V=Ce:K.sibling=Ce,K=Ce);return Ue&&ds(x,oe),V}for(J=r(x,J);!Ce.done;oe++,Ce=_.next())Ce=D(J,x,oe,Ce.value,A),Ce!==null&&(e&&Ce.alternate!==null&&J.delete(Ce.key===null?oe:Ce.key),v=l(Ce,v,oe),K===null?V=Ce:K.sibling=Ce,K=Ce);return e&&J.forEach(function(Sf){return t(x,Sf)}),Ue&&ds(x,oe),V}function Je(x,v,_,A){if(typeof _=="object"&&_!==null&&_.type===Ne&&_.key===null&&(_=_.props.children),typeof _=="object"&&_!==null){switch(_.$$typeof){case $:e:{for(var V=_.key,K=v;K!==null;){if(K.key===V){if(V=_.type,V===Ne){if(K.tag===7){n(x,K.sibling),v=s(K,_.props.children),v.return=x,x=v;break e}}else if(K.elementType===V||typeof V=="object"&&V!==null&&V.$$typeof===re&&ju(V)===K.type){n(x,K.sibling),v=s(K,_.props),v.ref=Co(x,K,_),v.return=x,x=v;break e}n(x,K);break}else t(x,K);K=K.sibling}_.type===Ne?(v=xs(_.props.children,x.mode,A,_.key),v.return=x,x=v):(A=Bl(_.type,_.key,_.props,null,x.mode,A),A.ref=Co(x,v,_),A.return=x,x=A)}return u(x);case ae:e:{for(K=_.key;v!==null;){if(v.key===K)if(v.tag===4&&v.stateNode.containerInfo===_.containerInfo&&v.stateNode.implementation===_.implementation){n(x,v.sibling),v=s(v,_.children||[]),v.return=x,x=v;break e}else{n(x,v);break}else t(x,v);v=v.sibling}v=Ni(_,x.mode,A),v.return=x,x=v}return u(x);case re:return K=_._init,Je(x,v,K(_._payload),A)}if(_n(_))return U(x,v,_,A);if(F(_))return Q(x,v,_,A);yl(x,_)}return typeof _=="string"&&_!==""||typeof _=="number"?(_=""+_,v!==null&&v.tag===6?(n(x,v.sibling),v=s(v,_),v.return=x,x=v):(n(x,v),v=Ci(_,x.mode,A),v.return=x,x=v),u(x)):n(x,v)}return Je}var Ws=Cu(!0),Nu=Cu(!1),xl=Mr(null),_l=null,qs=null,Da=null;function Oa(){Da=qs=_l=null}function za(e){var t=xl.current;Fe(xl),e._currentValue=t}function $a(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 Ks(e,t){_l=e,Da=qs=null,e=e.dependencies,e!==null&&e.firstContext!==null&&((e.lanes&t)!==0&&(Vt=!0),e.firstContext=null)}function mn(e){var t=e._currentValue;if(Da!==e)if(e={context:e,memoizedValue:t,next:null},qs===null){if(_l===null)throw Error(p(308));qs=e,_l.dependencies={lanes:0,firstContext:e}}else qs=qs.next=e;return t}var fs=null;function Fa(e){fs===null?fs=[e]:fs.push(e)}function Eu(e,t,n,r){var s=t.interleaved;return s===null?(n.next=n,Fa(t)):(n.next=s.next,s.next=n),t.interleaved=n,pr(e,r)}function pr(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 Dr=!1;function Ua(e){e.updateQueue={baseState:e.memoizedState,firstBaseUpdate:null,lastBaseUpdate:null,shared:{pending:null,interleaved:null,lanes:0},effects:null}}function Pu(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 mr(e,t){return{eventTime:e,lane:t,tag:0,payload:null,callback:null,next:null}}function Or(e,t,n){var r=e.updateQueue;if(r===null)return null;if(r=r.shared,(we&2)!==0){var s=r.pending;return s===null?t.next=t:(t.next=s.next,s.next=t),r.pending=t,pr(e,n)}return s=r.interleaved,s===null?(t.next=t,Fa(r)):(t.next=s.next,s.next=t),r.interleaved=t,pr(e,n)}function Sl(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,Cs(e,n)}}function Tu(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 wl(e,t,n,r){var s=e.updateQueue;Dr=!1;var l=s.firstBaseUpdate,u=s.lastBaseUpdate,d=s.shared.pending;if(d!==null){s.shared.pending=null;var m=d,w=m.next;m.next=null,u===null?l=w:u.next=w,u=m;var N=e.alternate;N!==null&&(N=N.updateQueue,d=N.lastBaseUpdate,d!==u&&(d===null?N.firstBaseUpdate=w:d.next=w,N.lastBaseUpdate=m))}if(l!==null){var P=s.baseState;u=0,N=w=m=null,d=l;do{var j=d.lane,D=d.eventTime;if((r&j)===j){N!==null&&(N=N.next={eventTime:D,lane:0,tag:d.tag,payload:d.payload,callback:d.callback,next:null});e:{var U=e,Q=d;switch(j=t,D=n,Q.tag){case 1:if(U=Q.payload,typeof U=="function"){P=U.call(D,P,j);break e}P=U;break e;case 3:U.flags=U.flags&-65537|128;case 0:if(U=Q.payload,j=typeof U=="function"?U.call(D,P,j):U,j==null)break e;P=I({},P,j);break e;case 2:Dr=!0}}d.callback!==null&&d.lane!==0&&(e.flags|=64,j=s.effects,j===null?s.effects=[d]:j.push(d))}else D={eventTime:D,lane:j,tag:d.tag,payload:d.payload,callback:d.callback,next:null},N===null?(w=N=D,m=P):N=N.next=D,u|=j;if(d=d.next,d===null){if(d=s.shared.pending,d===null)break;j=d,d=j.next,j.next=null,s.lastBaseUpdate=j,s.shared.pending=null}}while(!0);if(N===null&&(m=P),s.baseState=m,s.firstBaseUpdate=w,s.lastBaseUpdate=N,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);hs|=u,e.lanes=u,e.memoizedState=P}}function Ru(e,t,n){if(e=t.effects,t.effects=null,e!==null)for(t=0;tn?n:4,e(!0);var r=Va.transition;Va.transition={};try{e(!1),t()}finally{ve=n,Va.transition=r}}function Ku(){return hn().memoizedState}function Hd(e,t,n){var r=Ur(e);if(n={lane:r,action:n,hasEagerState:!1,eagerState:null,next:null},Ju(e))Yu(t,n);else if(n=Eu(e,t,n,r),n!==null){var s=$t();Mn(n,e,r,s),Xu(n,t,r)}}function Vd(e,t,n){var r=Ur(e),s={lane:r,action:n,hasEagerState:!1,eagerState:null,next:null};if(Ju(e))Yu(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 m=t.interleaved;m===null?(s.next=s,Fa(t)):(s.next=m.next,m.next=s),t.interleaved=s;return}}catch{}n=Eu(e,t,s,r),n!==null&&(s=$t(),Mn(n,e,r,s),Xu(n,t,r))}}function Ju(e){var t=e.alternate;return e===Qe||t!==null&&t===Qe}function Yu(e,t){To=Cl=!0;var n=e.pending;n===null?t.next=t:(t.next=n.next,n.next=t),e.pending=t}function Xu(e,t,n){if((n&4194240)!==0){var r=t.lanes;r&=e.pendingLanes,n|=r,t.lanes=n,Cs(e,n)}}var Pl={readContext:mn,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},Gd={readContext:mn,useCallback:function(e,t){return Xn().memoizedState=[e,t===void 0?null:t],e},useContext:mn,useEffect:bu,useImperativeHandle:function(e,t,n){return n=n!=null?n.concat([e]):null,Nl(4194308,4,Hu.bind(null,t,e),n)},useLayoutEffect:function(e,t){return Nl(4194308,4,e,t)},useInsertionEffect:function(e,t){return Nl(4,2,e,t)},useMemo:function(e,t){var n=Xn();return t=t===void 0?null:t,e=e(),n.memoizedState=[e,t],e},useReducer:function(e,t,n){var r=Xn();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=Hd.bind(null,Qe,e),[r.memoizedState,e]},useRef:function(e){var t=Xn();return e={current:e},t.memoizedState=e},useState:Fu,useDebugValue:Xa,useDeferredValue:function(e){return Xn().memoizedState=e},useTransition:function(){var e=Fu(!1),t=e[0];return e=Qd.bind(null,e[1]),Xn().memoizedState=e,[t,e]},useMutableSource:function(){},useSyncExternalStore:function(e,t,n){var r=Qe,s=Xn();if(Ue){if(n===void 0)throw Error(p(407));n=n()}else{if(n=t(),ut===null)throw Error(p(349));(ms&30)!==0||Lu(r,t,n)}s.memoizedState=n;var l={value:n,getSnapshot:t};return s.queue=l,bu(Ou.bind(null,r,l,e),[e]),r.flags|=2048,Mo(9,Du.bind(null,r,l,n,t),void 0,null),n},useId:function(){var e=Xn(),t=ut.identifierPrefix;if(Ue){var n=fr,r=dr;n=(r&~(1<<32-Zt(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[Jn]=t,e[ko]=r,yc(e,t,!1,!1),t.stateNode=e;e:{switch(u=On(n,r),n){case"dialog":$e("cancel",e),$e("close",e),s=r;break;case"iframe":case"object":case"embed":$e("load",e),s=r;break;case"video":case"audio":for(s=0;s<_o.length;s++)$e(_o[s],e);s=r;break;case"source":$e("error",e),s=r;break;case"img":case"image":case"link":$e("error",e),$e("load",e),s=r;break;case"details":$e("toggle",e),s=r;break;case"input":Te(e,r),s=xn(e,r),$e("invalid",e);break;case"option":s=r;break;case"select":e._wrapperState={wasMultiple:!!r.multiple},s=I({},r,{value:void 0}),$e("invalid",e);break;case"textarea":yr(e,r),s=mt(e,r),$e("invalid",e);break;default:s=r}an(n,s),d=s;for(l in d)if(d.hasOwnProperty(l)){var m=d[l];l==="style"?er(e,m):l==="dangerouslySetInnerHTML"?(m=m?m.__html:void 0,m!=null&&me(e,m)):l==="children"?typeof m=="string"?(n!=="textarea"||m!=="")&&Rt(e,m):typeof m=="number"&&Rt(e,""+m):l!=="suppressContentEditableWarning"&&l!=="suppressHydrationWarning"&&l!=="autoFocus"&&(E.hasOwnProperty(l)?m!=null&&l==="onScroll"&&$e("scroll",e):m!=null&&Ve(e,l,m,u))}switch(n){case"input":ln(e),xt(e,r,!1);break;case"textarea":ln(e),Ge(e);break;case"option":r.value!=null&&e.setAttribute("value",""+Se(r.value));break;case"select":e.multiple=!!r.multiple,l=r.value,l!=null?W(e,!!r.multiple,l,!1):r.defaultValue!=null&&W(e,!!r.multiple,r.defaultValue,!0);break;default:typeof s.onClick=="function"&&(e.onclick=cl)}switch(n){case"button":case"input":case"select":case"textarea":r=!!r.autoFocus;break e;case"img":r=!0;break e;default:r=!1}}r&&(t.flags|=4)}t.ref!==null&&(t.flags|=512,t.flags|=2097152)}return Ct(t),null;case 6:if(e&&t.stateNode!=null)_c(e,t,e.memoizedProps,r);else{if(typeof r!="string"&&t.stateNode===null)throw Error(p(166));if(n=ps(Po.current),ps(Yn.current),vl(t)){if(r=t.stateNode,n=t.memoizedProps,r[Jn]=t,(l=r.nodeValue!==n)&&(e=rn,e!==null))switch(e.tag){case 3:ul(r.nodeValue,n,(e.mode&1)!==0);break;case 5:e.memoizedProps.suppressHydrationWarning!==!0&&ul(r.nodeValue,n,(e.mode&1)!==0)}l&&(t.flags|=4)}else r=(n.nodeType===9?n:n.ownerDocument).createTextNode(r),r[Jn]=t,t.stateNode=r}return Ct(t),null;case 13:if(Fe(Be),r=t.memoizedState,e===null||e.memoizedState!==null&&e.memoizedState.dehydrated!==null){if(Ue&&sn!==null&&(t.mode&1)!==0&&(t.flags&128)===0)ku(),Gs(),t.flags|=98560,l=!1;else if(l=vl(t),r!==null&&r.dehydrated!==null){if(e===null){if(!l)throw Error(p(318));if(l=t.memoizedState,l=l!==null?l.dehydrated:null,!l)throw Error(p(317));l[Jn]=t}else Gs(),(t.flags&128)===0&&(t.memoizedState=null),t.flags|=4;Ct(t),l=!1}else Pn!==null&&(_i(Pn),Pn=null),l=!0;if(!l)return t.flags&65536?t:null}return(t.flags&128)!==0?(t.lanes=n,t):(r=r!==null,r!==(e!==null&&e.memoizedState!==null)&&r&&(t.child.flags|=8192,(t.mode&1)!==0&&(e===null||(Be.current&1)!==0?ot===0&&(ot=3):ki())),t.updateQueue!==null&&(t.flags|=4),Ct(t),null);case 4:return Js(),ui(e,t),e===null&&So(t.stateNode.containerInfo),Ct(t),null;case 10:return za(t.type._context),Ct(t),null;case 17:return Ht(t.type)&&fl(),Ct(t),null;case 19:if(Fe(Be),l=t.memoizedState,l===null)return Ct(t),null;if(r=(t.flags&128)!==0,u=l.rendering,u===null)if(r)Io(l,!1);else{if(ot!==0||e!==null&&(e.flags&128)!==0)for(e=t.child;e!==null;){if(u=kl(e),u!==null){for(t.flags|=128,Io(l,!1),r=u.updateQueue,r!==null&&(t.updateQueue=r,t.flags|=4),t.subtreeFlags=0,r=n,n=t.child;n!==null;)l=n,e=r,l.flags&=14680066,u=l.alternate,u===null?(l.childLanes=0,l.lanes=e,l.child=null,l.subtreeFlags=0,l.memoizedProps=null,l.memoizedState=null,l.updateQueue=null,l.dependencies=null,l.stateNode=null):(l.childLanes=u.childLanes,l.lanes=u.lanes,l.child=u.child,l.subtreeFlags=0,l.deletions=null,l.memoizedProps=u.memoizedProps,l.memoizedState=u.memoizedState,l.updateQueue=u.updateQueue,l.type=u.type,e=u.dependencies,l.dependencies=e===null?null:{lanes:e.lanes,firstContext:e.firstContext}),n=n.sibling;return Oe(Be,Be.current&1|2),t.child}e=e.sibling}l.tail!==null&&ze()>eo&&(t.flags|=128,r=!0,Io(l,!1),t.lanes=4194304)}else{if(!r)if(e=kl(u),e!==null){if(t.flags|=128,r=!0,n=e.updateQueue,n!==null&&(t.updateQueue=n,t.flags|=4),Io(l,!0),l.tail===null&&l.tailMode==="hidden"&&!u.alternate&&!Ue)return Ct(t),null}else 2*ze()-l.renderingStartTime>eo&&n!==1073741824&&(t.flags|=128,r=!0,Io(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=ze(),t.sibling=null,n=Be.current,Oe(Be,r?n&1|2:n&1),t):(Ct(t),null);case 22:case 23:return wi(),r=t.memoizedState!==null,e!==null&&e.memoizedState!==null!==r&&(t.flags|=8192),r&&(t.mode&1)!==0?(on&1073741824)!==0&&(Ct(t),t.subtreeFlags&6&&(t.flags|=8192)):Ct(t),null;case 24:return null;case 25:return null}throw Error(p(156,t.tag))}function ef(e,t){switch(Aa(t),t.tag){case 1:return Ht(t.type)&&fl(),e=t.flags,e&65536?(t.flags=e&-65537|128,t):null;case 3:return Js(),Fe(Qt),Fe(kt),Ha(),e=t.flags,(e&65536)!==0&&(e&128)===0?(t.flags=e&-65537|128,t):null;case 5:return Ba(t),null;case 13:if(Fe(Be),e=t.memoizedState,e!==null&&e.dehydrated!==null){if(t.alternate===null)throw Error(p(340));Gs()}return e=t.flags,e&65536?(t.flags=e&-65537|128,t):null;case 19:return Fe(Be),null;case 4:return Js(),null;case 10:return za(t.type._context),null;case 22:case 23:return wi(),null;case 24:return null;default:return null}}var Ml=!1,Nt=!1,tf=typeof WeakSet=="function"?WeakSet:Set,O=null;function Xs(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 ci(e,t,n){try{n()}catch(r){qe(e,t,r)}}var Sc=!1;function nf(e,t){if(wa=rs,e=Zi(),ma(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,m=-1,w=0,N=0,P=e,j=null;t:for(;;){for(var D;P!==n||s!==0&&P.nodeType!==3||(d=u+s),P!==l||r!==0&&P.nodeType!==3||(m=u+r),P.nodeType===3&&(u+=P.nodeValue.length),(D=P.firstChild)!==null;)j=P,P=D;for(;;){if(P===e)break t;if(j===n&&++w===s&&(d=u),j===l&&++N===r&&(m=u),(D=P.nextSibling)!==null)break;P=j,j=P.parentNode}P=D}n=d===-1||m===-1?null:{start:d,end:m}}else n=null}n=n||{start:0,end:0}}else n=null;for(ka={focusedElem:e,selectionRange:n},rs=!1,O=t;O!==null;)if(t=O,e=t.child,(t.subtreeFlags&1028)!==0&&e!==null)e.return=t,O=e;else for(;O!==null;){t=O;try{var U=t.alternate;if((t.flags&1024)!==0)switch(t.tag){case 0:case 11:case 15:break;case 1:if(U!==null){var Q=U.memoizedProps,Je=U.memoizedState,x=t.stateNode,v=x.getSnapshotBeforeUpdate(t.elementType===t.type?Q:Tn(t.type,Q),Je);x.__reactInternalSnapshotBeforeUpdate=v}break;case 3:var _=t.stateNode.containerInfo;_.nodeType===1?_.textContent="":_.nodeType===9&&_.documentElement&&_.removeChild(_.documentElement);break;case 5:case 6:case 4:case 17:break;default:throw Error(p(163))}}catch(A){qe(t,t.return,A)}if(e=t.sibling,e!==null){e.return=t.return,O=e;break}O=t.return}return U=Sc,Sc=!1,U}function Lo(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 Il(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 di(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 wc(e){var t=e.alternate;t!==null&&(e.alternate=null,wc(t)),e.child=null,e.deletions=null,e.sibling=null,e.tag===5&&(t=e.stateNode,t!==null&&(delete t[Jn],delete t[ko],delete t[Ea],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 kc(e){return e.tag===5||e.tag===3||e.tag===4}function jc(e){e:for(;;){for(;e.sibling===null;){if(e.return===null||kc(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 fi(e,t,n){var r=e.tag;if(r===5||r===6)e=e.stateNode,t?n.nodeType===8?n.parentNode.insertBefore(e,t):n.insertBefore(e,t):(n.nodeType===8?(t=n.parentNode,t.insertBefore(e,n)):(t=n,t.appendChild(e)),n=n._reactRootContainer,n!=null||t.onclick!==null||(t.onclick=cl));else if(r!==4&&(e=e.child,e!==null))for(fi(e,t,n),e=e.sibling;e!==null;)fi(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 gt=null,Rn=!1;function zr(e,t,n){for(n=n.child;n!==null;)Cc(e,t,n),n=n.sibling}function Cc(e,t,n){if(un&&typeof un.onCommitFiberUnmount=="function")try{un.onCommitFiberUnmount(Yr,n)}catch{}switch(n.tag){case 5:Nt||Xs(n,t);case 6:var r=gt,s=Rn;gt=null,zr(e,t,n),gt=r,Rn=s,gt!==null&&(Rn?(e=gt,n=n.stateNode,e.nodeType===8?e.parentNode.removeChild(n):e.removeChild(n)):gt.removeChild(n.stateNode));break;case 18:gt!==null&&(Rn?(e=gt,n=n.stateNode,e.nodeType===8?Na(e.parentNode,n):e.nodeType===1&&Na(e,n),ns(e)):Na(gt,n.stateNode));break;case 4:r=gt,s=Rn,gt=n.stateNode.containerInfo,Rn=!0,zr(e,t,n),gt=r,Rn=s;break;case 0:case 11:case 14:case 15:if(!Nt&&(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(!Nt&&(Xs(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)}zr(e,t,n);break;case 21:zr(e,t,n);break;case 22:n.mode&1?(Nt=(r=Nt)||n.memoizedState!==null,zr(e,t,n),Nt=r):zr(e,t,n);break;default:zr(e,t,n)}}function Nc(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 An(e,t){var n=t.deletions;if(n!==null)for(var r=0;rs&&(s=u),r&=~l}if(r=s,r=ze()-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,$l=0,(we&6)!==0)throw Error(p(331));var s=we;for(we|=4,O=e.current;O!==null;){var l=O,u=l.child;if((O.flags&16)!==0){var d=l.deletions;if(d!==null){for(var m=0;mze()-gi?vs(e,0):hi|=n),Wt(e,t)}function Fc(e,t){t===0&&((e.mode&1)===0?t=1:(t=wn,wn<<=1,(wn&130023424)===0&&(wn=4194304)));var n=$t();e=pr(e,t),e!==null&&(wr(e,t,n),Wt(e,n))}function df(e){var t=e.memoizedState,n=0;t!==null&&(n=t.retryLane),Fc(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(p(314))}r!==null&&r.delete(t),Fc(e,n)}var Uc;Uc=function(e,t,n){if(e!==null)if(e.memoizedProps!==t.pendingProps||Qt.current)Vt=!0;else{if((e.lanes&n)===0&&(t.flags&128)===0)return Vt=!1,Xd(e,t,n);Vt=(e.flags&131072)!==0}else Vt=!1,Ue&&(t.flags&1048576)!==0&&xu(t,gl,t.index);switch(t.lanes=0,t.tag){case 2:var r=t.type;Al(e,t),e=t.pendingProps;var s=Qs(t,kt.current);Ks(t,n),s=Wa(null,t,r,e,s,n);var l=qa();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,Ht(r)?(l=!0,pl(t)):l=!1,t.memoizedState=s.state!==null&&s.state!==void 0?s.state:null,Ua(t),s.updater=Tl,t.stateNode=s,s._reactInternals=t,ei(t,r,e,n),t=si(null,t,r,!0,l,n)):(t.tag=0,Ue&&l&&Ra(t),zt(null,t,s,n),t=t.child),t;case 16:r=t.elementType;e:{switch(Al(e,t),e=t.pendingProps,s=r._init,r=s(r._payload),t.type=r,s=t.tag=mf(r),e=Tn(r,e),s){case 0:t=ri(null,t,r,e,n);break e;case 1:t=fc(null,t,r,e,n);break e;case 11:t=ac(null,t,r,e,n);break e;case 14:t=ic(null,t,r,Tn(r.type,e),n);break e}throw Error(p(306,r,""))}return t;case 0:return r=t.type,s=t.pendingProps,s=t.elementType===r?s:Tn(r,s),ri(e,t,r,s,n);case 1:return r=t.type,s=t.pendingProps,s=t.elementType===r?s:Tn(r,s),fc(e,t,r,s,n);case 3:e:{if(pc(t),e===null)throw Error(p(387));r=t.pendingProps,l=t.memoizedState,s=l.element,Pu(e,t),wl(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=Ys(Error(p(423)),t),t=mc(e,t,r,n,s);break e}else if(r!==s){s=Ys(Error(p(424)),t),t=mc(e,t,r,n,s);break e}else for(sn=Ar(t.stateNode.containerInfo.firstChild),rn=t,Ue=!0,Pn=null,n=Nu(t,null,r,n),t.child=n;n;)n.flags=n.flags&-3|4096,n=n.sibling;else{if(Gs(),r===s){t=hr(e,t,n);break e}zt(e,t,r,n)}t=t.child}return t;case 5:return Au(t),e===null&&Ia(t),r=t.type,s=t.pendingProps,l=e!==null?e.memoizedProps:null,u=s.children,ja(r,s)?u=null:l!==null&&ja(r,l)&&(t.flags|=32),dc(e,t),zt(e,t,u,n),t.child;case 6:return e===null&&Ia(t),null;case 13:return hc(e,t,n);case 4:return ba(t,t.stateNode.containerInfo),r=t.pendingProps,e===null?t.child=Ws(t,null,r,n):zt(e,t,r,n),t.child;case 11:return r=t.type,s=t.pendingProps,s=t.elementType===r?s:Tn(r,s),ac(e,t,r,s,n);case 7:return zt(e,t,t.pendingProps,n),t.child;case 8:return zt(e,t,t.pendingProps.children,n),t.child;case 12:return zt(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,Oe(xl,r._currentValue),r._currentValue=u,l!==null)if(En(l.value,u)){if(l.children===s.children&&!Qt.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 m=d.firstContext;m!==null;){if(m.context===r){if(l.tag===1){m=mr(-1,n&-n),m.tag=2;var w=l.updateQueue;if(w!==null){w=w.shared;var N=w.pending;N===null?m.next=m:(m.next=N.next,N.next=m),w.pending=m}}l.lanes|=n,m=l.alternate,m!==null&&(m.lanes|=n),$a(l.return,n,t),d.lanes|=n;break}m=m.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(p(341));u.lanes|=n,d=u.alternate,d!==null&&(d.lanes|=n),$a(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}zt(e,t,s.children,n),t=t.child}return t;case 9:return s=t.type,r=t.pendingProps.children,Ks(t,n),s=mn(s),r=r(s),t.flags|=1,zt(e,t,r,n),t.child;case 14:return r=t.type,s=Tn(r,t.pendingProps),s=Tn(r.type,s),ic(e,t,r,s,n);case 15:return uc(e,t,t.type,t.pendingProps,n);case 17:return r=t.type,s=t.pendingProps,s=t.elementType===r?s:Tn(r,s),Al(e,t),t.tag=1,Ht(r)?(e=!0,pl(t)):e=!1,Ks(t,n),ec(t,r,s),ei(t,r,s,n),si(null,t,r,!0,e,n);case 19:return vc(e,t,n);case 22:return cc(e,t,n)}throw Error(p(156,t.tag))};function bc(e,t){return Xt(e,t)}function pf(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 vn(e,t,n,r){return new pf(e,t,n,r)}function ji(e){return e=e.prototype,!(!e||!e.isReactComponent)}function mf(e){if(typeof e=="function")return ji(e)?1:0;if(e!=null){if(e=e.$$typeof,e===ft)return 11;if(e===Ke)return 14}return 2}function Br(e,t){var n=e.alternate;return n===null?(n=vn(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 Bl(e,t,n,r,s,l){var u=2;if(r=e,typeof e=="function")ji(e)&&(u=1);else if(typeof e=="string")u=5;else e:switch(e){case Ne:return xs(n.children,s,l,t);case be:u=8,s|=8;break;case yt:return e=vn(12,n,t,s|2),e.elementType=yt,e.lanes=l,e;case b:return e=vn(13,n,t,s),e.elementType=b,e.lanes=l,e;case Ie:return e=vn(19,n,t,s),e.elementType=Ie,e.lanes=l,e;case ee:return Ql(n,s,l,t);default:if(typeof e=="object"&&e!==null)switch(e.$$typeof){case dt:u=10;break e;case Kt:u=9;break e;case ft:u=11;break e;case Ke:u=14;break e;case re:u=16,r=null;break e}throw Error(p(130,e==null?e:typeof e,""))}return t=vn(u,n,t,s),t.elementType=e,t.type=r,t.lanes=l,t}function xs(e,t,n,r){return e=vn(7,e,r,t),e.lanes=n,e}function Ql(e,t,n,r){return e=vn(22,e,r,t),e.elementType=ee,e.lanes=n,e.stateNode={isHidden:!1},e}function Ci(e,t,n){return e=vn(6,e,null,t),e.lanes=n,e}function Ni(e,t,n){return t=vn(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=Hn(0),this.expirationTimes=Hn(-1),this.entangledLanes=this.finishedLanes=this.mutableReadLanes=this.expiredLanes=this.pingedLanes=this.suspendedLanes=this.pendingLanes=0,this.entanglements=Hn(0),this.identifierPrefix=r,this.onRecoverableError=s,this.mutableSourceEagerHydrationData=null}function Ei(e,t,n,r,s,l,u,d,m){return e=new hf(e,t,n,d,m),t===1?(t=1,l===!0&&(t|=8)):t=0,l=vn(3,null,null,t),e.current=l,l.stateNode=e,l.memoizedState={element:r,isDehydrated:n,cache:null,transitions:null,pendingSuspenseBoundaries:null},Ua(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(g){console.error(g)}}return i(),Ii.exports=Pf(),Ii.exports}var td;function Rf(){if(td)return Jl;td=1;var i=Tf();return Jl.createRoot=i.createRoot,Jl.hydrateRoot=i.hydrateRoot,Jl}var Af=Rf();const Mf=yd(Af),If="/api";async function xe(i,g){const p=await fetch(`${If}${i}`,{...g,headers:{"Content-Type":"application/json",...g?.headers??{}}}),R=await p.json();if(!p.ok){const E=R.error?.message??"Ошибка запроса";throw new Error(E)}return R}const Pe={async loadSharedConnectionConfig(){return xe("/llm/shared-connection")},async saveSharedConnectionConfig(i){return xe("/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 xe("/llm/models",{method:"POST",body:JSON.stringify({llmProvider:i.llmProvider,apiKey:i.apiKey,model:i.model,baseUrl:i.baseUrl})})},async testConnection(i){return xe("/llm/test-connection",{method:"POST",body:JSON.stringify({llmProvider:i.llmProvider,apiKey:i.apiKey,model:i.model,baseUrl:i.baseUrl})})},async normalize(i){return xe("/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 xe("/history")},async loadTrace(i){return xe(`/history/${i}`)},async loadPresets(){return xe("/presets")},async savePreset(i){return xe("/presets/save",{method:"POST",body:JSON.stringify(i)})},async runEval(i){return xe("/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 xe("/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,scenarioQuestions:i.scenarioQuestions,scenarioTitle:i.scenarioTitle,analysis_date:i.analysisDate})})},async loadEvalRunAsyncStatus(i){return xe(`/eval/run-async/${encodeURIComponent(i)}`)},async cancelEvalRunAsync(i){return xe(`/eval/run-async/${encodeURIComponent(i)}/cancel`,{method:"POST"})},async startRun(){return xe("/accounting-agent/v1/runs/start",{method:"POST",body:JSON.stringify({initiator:"ndc_operator",source:"gui"})})},async finishRun(i){return xe("/accounting-agent/v1/runs/finish",{method:"POST",body:JSON.stringify({runId:i,status:"DONE",source:"gui",reason:"Остановлено оператором из GUI"})})},async listRuns(){return xe("/accounting-agent/v1/runs")},async listResults(){return xe("/accounting-agent/v1/results")},async runTrace(i){return xe(`/accounting-agent/v1/trace/run/${i}`)},async sendAssistantMessage(i){return xe("/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 xe(`/assistant/session/${i}`)},async saveAutoRunAssistantSession(i){return xe("/autoruns/autogen/save-assistant-session",{method:"POST",body:JSON.stringify(i)})},async loadAssistantAnnotations(i){const g=new URLSearchParams;i?.session_id&&g.set("session_id",i.session_id),typeof i?.limit=="number"&&g.set("limit",String(i.limit));const p=g.toString();return xe(`/assistant/annotations${p?`?${p}`:""}`)},async saveAssistantAnnotation(i){return xe("/assistant/annotations",{method:"POST",body:JSON.stringify(i)})},async loadAutoRunsHistory(i){const g=new URLSearchParams;i?.from&&g.set("from",i.from),i?.to&&g.set("to",i.to),i?.target&&g.set("target",i.target),i?.mode&&g.set("mode",i.mode),i?.use_mock&&g.set("use_mock",i.use_mock),i?.prompt_contains&&g.set("prompt_contains",i.prompt_contains),typeof i?.limit=="number"&&g.set("limit",String(i.limit)),typeof i?.scan_limit=="number"&&g.set("scan_limit",String(i.scan_limit));const p=g.toString();return xe(`/autoruns/history${p?`?${p}`:""}`)},async loadAutoRunDetail(i){return xe(`/autoruns/history/${encodeURIComponent(i)}`)},async loadAutoRunCaseDialog(i,g){return xe(`/autoruns/history/${encodeURIComponent(i)}/case/${encodeURIComponent(g)}/dialog`)},async loadAutoRunAnnotations(i){const g=new URLSearchParams;i?.run_id&&g.set("run_id",i.run_id),i?.case_id&&g.set("case_id",i.case_id),typeof i?.min_rating=="number"&&g.set("min_rating",String(i.min_rating)),i?.manual_case_decision&&g.set("manual_case_decision",i.manual_case_decision),typeof i?.limit=="number"&&g.set("limit",String(i.limit));const p=g.toString();return xe(`/autoruns/annotations${p?`?${p}`:""}`)},async saveAutoRunAnnotation(i){return xe("/autoruns/annotations",{method:"POST",body:JSON.stringify(i)})},async updateAutoRunAnnotation(i){return xe(`/autoruns/annotations/${encodeURIComponent(i.annotation_id)}`,{method:"PATCH",body:JSON.stringify({resolved:i.resolved,resolved_by:i.resolved_by})})},async loadAutoRunPostAnalysis(i){const g=new URLSearchParams;i?.run_id&&g.set("run_id",i.run_id),typeof i?.limit_per_queue=="number"&&g.set("limit_per_queue",String(i.limit_per_queue)),typeof i?.annotation_limit=="number"&&g.set("annotation_limit",String(i.annotation_limit)),typeof i?.scan_limit=="number"&&g.set("scan_limit",String(i.scan_limit)),i?.from&&g.set("from",i.from),i?.to&&g.set("to",i.to),i?.target&&g.set("target",i.target),i?.mode&&g.set("mode",i.mode),i?.use_mock&&g.set("use_mock",i.use_mock),i?.prompt_contains&&g.set("prompt_contains",i.prompt_contains);const p=g.toString();return xe(`/autoruns/post-analysis${p?`?${p}`:""}`)},async loadAutoRunAutogenHistory(i){const g=new URLSearchParams;i?.mode&&g.set("mode",i.mode),typeof i?.limit=="number"&&g.set("limit",String(i.limit));const p=g.toString();return xe(`/autoruns/autogen/history${p?`?${p}`:""}`)},async loadAutoRunAutogenPersonalityCatalog(){return xe("/autoruns/autogen/personality-catalog")},async updateAutoRunAutogenQuestions(i){return xe(`/autoruns/autogen/history/${encodeURIComponent(i.generation_id)}/questions`,{method:"PATCH",body:JSON.stringify({questions:i.questions})})},async deleteAutoRunAutogenHistoryRecord(i){return xe(`/autoruns/autogen/history/${encodeURIComponent(i)}`,{method:"DELETE"})},async generateAutoRunQuestions(i){return xe("/autoruns/autogen/generate",{method:"POST",body:JSON.stringify(i)})}},Lf=/(?:^|\n)\s*#{0,6}\s*(?:debug_payload_json|technical_breakdown_json|route_summary_json|debug_payload|technical_breakdown)\b/i,Df=[/\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 Of(i){try{return JSON.stringify(i,null,2)}catch{return String(i)}}function zf(i){const g=String(i??""),p=g.match(Lf);return(p?g.slice(0,p.index):g).replace(/###\s*(?:debug_payload_json|technical_breakdown_json|route_summary_json)[\s\S]*?(?:```[\s\S]*?```|$)/gi,"").replace(/(?:^|\n)\s*#{0,6}\s*(?:debug_payload_json|technical_breakdown_json|route_summary_json)\b[\s\S]*$/gi,"").split(/\r?\n/g).map(B=>B.trimEnd()).filter(B=>B.trim().length>0).filter(B=>!Df.some(ie=>ie.test(B))).join(` -`).trim()}function $f(i,g,p="default"){const R=p==="technical",E=[];E.push("# Assistant conversation export"),E.push(`session_id: ${i||"n/a"}`),E.push(`export_mode: ${p}`),E.push(`exported_at: ${new Date().toISOString()}`),E.push("");for(let L=0;L{E.length!==0&&(R.push(E.join(` -`)),E=[])};for(const B of p){const ie=B.trimEnd(),ne=ie.trim();if(!ne){L();continue}const z=/^Блок\s+\d+\./i.test(ne),X=/^\d+\.\s/.test(ne);(z||X)&&E.length>0&&L(),E.push(ie)}return L(),R.length>0?R:[i]}function Vf(i,g){const p=[],R=/\*\*(.+?)\*\*/g;let E=0,L=0,B;for(;(B=R.exec(i))!==null;)B.index>E&&(p.push(o.jsx("span",{children:i.slice(E,B.index)},`${g}-t-${L}`)),L+=1),p.push(o.jsx("strong",{children:B[1]},`${g}-b-${L}`)),L+=1,E=R.lastIndex;return E0?p:[o.jsx("span",{children:i},`${g}-raw`)]}function Gf(i){const g=i.trimStart();return/^Блок\s+\d+\./i.test(g)?"assistant-msg-line heading":/^\d+\.\s/.test(g)?"assistant-msg-line numbered":/^-\s/.test(g)?"assistant-msg-line bullet":"assistant-msg-line"}function Wf(i,g=40){const p=i.replace(/\s+/g," ").trim();if(p.length<=g)return p;const R=p.split(" ").slice(0,3).join(" ").trim();return R.length>=10&&R.length<=g?`${R}…`:`${p.slice(0,g-1).trimEnd()}…`}function _d(i){return i.replace(/\*\*(.+?)\*\*/g,"$1").replace(/^\d+\.\s*/,"").trim()}function qf(i){const g=i.replace(/\r\n?/g,` -`).split(` -`).map(E=>E.trim()).find(Boolean),p=_d(g??"");return(p.split("|")[0]?.trim()??p).replace(/\s+/g," ").trim()}function Kf(i){const g=i.replace(/\r\n?/g,` -`).split(` -`).map(R=>R.trim()).find(Boolean);return!g||!/^\d+\.\s/.test(g)?!1:_d(g).includes("|")}function Jf(i,g){const p=g.replace(/\r\n?/g,` -`).replace(/\*\*(.+?)\*\*/g,"$1").split(` -`).map((E,L)=>{const B=E.trim();return L===0?B.replace(/^\d+\.\s*/,""):B}).filter(Boolean).join(" ").replace(/\s+/g," ").trim();if(!p)return null;const R=qf(g)||p;return{message_id:i.message_id,source_text:p,anchor_text:R,preview_text:Wf(R)}}function Yf(i,g,p,R){return Hf(i.text).map((L,B)=>{const ie=L.split(` -`),ne=i.role==="assistant"&&Kf(L),z=ne?Jf(i,L):null,X=!!z&&g?.message_id===z?.message_id&&g?.source_text===z?.source_text,Z=ie.map((te,Re)=>o.jsx("p",{className:Gf(te),children:Vf(te,`line-${B}-${Re}`)},`line-${B}-${Re}`));return!ne||!z?o.jsx("div",{className:"assistant-msg-block",children:Z},`block-${B}`):o.jsx("div",{className:X?"assistant-msg-block selectable active":"assistant-msg-block selectable",role:"button",tabIndex:0,onClick:()=>{if(X){R();return}p(z)},onKeyDown:te=>{if(!(te.key!=="Enter"&&te.key!==" ")){if(te.preventDefault(),X){R();return}p(z)}},children:Z},`block-${B}`)})}function Xf({sessionId:i,conversation:g,inputValue:p,onInputChange:R,selectedContextChip:E,onSelectContextChip:L,onClearContextChip:B,useMock:ie,onUseMockChange:ne,onSend:z,onClear:X,onSaveSession:Z,busy:te,saveBusy:Re=!1,saveDisabled:de=!1,statusText:fe,errorMessage:he,showSaveAction:He=!1,showCommentAction:Ye=!1,onCommentAssistantMessage:Ve,isAssistantMessageCommented:Me,canCommentAssistantMessage:$}){const ae=h.useRef(null),Ne=h.useRef(!0),be=h.useRef(null),[yt,dt]=h.useState("idle"),[Kt,ft]=h.useState("чат");function b(re=!1){ae.current&&(re&&(Ne.current=!0),ae.current.scrollTop=ae.current.scrollHeight)}h.useEffect(()=>{Ne.current&&b()},[g]),h.useEffect(()=>()=>{be.current!==null&&window.clearTimeout(be.current)},[]);async function Ie(re){if(g.length===0)return;const ee=$f(i,g,re),M=await bf(ee);ft(re==="technical"?"тех":"чат"),dt(M?"success":"error"),be.current!==null&&window.clearTimeout(be.current),be.current=window.setTimeout(()=>{dt("idle")},2200)}function Ke(){if(!ae.current)return;const re=ae.current,ee=re.scrollHeight-re.scrollTop-re.clientHeight;Ne.current=ee<16}return o.jsx(oa,{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:()=>{Ie("default")},disabled:g.length===0,title:"Экспорт только user-facing чата",children:"Скопировать чат"}),o.jsx("button",{type:"button",className:"assistant-copy-btn",onClick:()=>{Ie("technical")},disabled:g.length===0,title:"Технический экспорт с debug payload",children:"Скопировать техчат"}),He?o.jsx("button",{type:"button",className:"assistant-copy-btn",onClick:()=>Z?.(),disabled:Re||de,children:Re?"Сохраняю...":"Сохранить"}):null,o.jsx("button",{type:"button",className:"assistant-copy-btn",onClick:()=>X(),disabled:te&&g.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:[fe?o.jsx("span",{className:"assistant-live-status",children:fe}):null,yt==="success"?o.jsxs("span",{className:"assistant-copy-feedback success",children:["Скопировано (",Kt,")"]}):null,yt==="error"?o.jsx("span",{className:"assistant-copy-feedback error",children:"Ошибка копирования"}):null]})]}),he?o.jsx("p",{className:"error-text assistant-toolbar-error",children:he}):null]}),o.jsx("div",{ref:ae,className:"assistant-chat-list",onScroll:Ke,children:g.map((re,ee)=>{const M=re.role==="assistant"&&Ye&&typeof Ve=="function"&&(typeof $=="function"?$(re,ee):!0),F=re.role==="assistant"&&typeof Me=="function"?Me(re,ee):!1;return o.jsxs("article",{className:`assistant-msg ${re.role}`,children:[o.jsxs("header",{className:"assistant-msg-head",children:[o.jsxs("div",{className:"assistant-msg-head-main",children:[o.jsx("strong",{children:Ff(re.role)}),o.jsx("span",{children:Uf(re.created_at)})]}),re.role==="assistant"&&Ye?o.jsx("div",{className:"assistant-msg-head-actions",children:o.jsx("button",{type:"button",className:F?"autoruns-comment-icon assistant-comment-btn commented":"autoruns-comment-icon assistant-comment-btn",onClick:()=>Ve?.(re,ee),disabled:!M,title:M?"Комментировать ответ ассистента":"Комментарий недоступен для этого сообщения","aria-label":M?"Комментировать ответ ассистента":"Комментарий недоступен для этого сообщения",children:o.jsx(Bf,{commented:F})})}):null]}),o.jsx("div",{className:"assistant-msg-body",children:Yf(re,E,L,B)}),re.role==="assistant"&&re.debug?o.jsxs("details",{className:"assistant-debug",children:[o.jsx("summary",{children:"Показать технический разбор"}),o.jsx(xd,{value:re.debug})]}):null]},re.message_id)})}),o.jsxs("div",{className:"assistant-compose",children:[E?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:E.source_text,children:[o.jsx("span",{className:"assistant-compose-context-pill-text",children:E.preview_text}),o.jsx("button",{type:"button",className:"assistant-compose-context-clear",onClick:B,"aria-label":"Убрать выбранный объект",title:"Убрать выбранный объект",children:"×"})]})]}):null,o.jsxs("label",{className:"full-width",children:["Сообщение",o.jsx("textarea",{className:"assistant-input-textarea",value:p,onChange:re=>R(re.target.value),rows:4,placeholder:E?"Продолжите вопрос по выбранному объекту...":"Введите вопрос к данным компании..."})]}),o.jsxs("div",{className:"button-row assistant-send-row",children:[o.jsxs("label",{className:"checkbox-row",children:[o.jsx("input",{type:"checkbox",checked:ie,onChange:re=>ne(re.target.checked)}),"Mock-режим"]}),o.jsx("button",{type:"button",className:"assistant-send-btn",onClick:()=>{b(!0),z()},disabled:te||!p.trim(),children:te?"Выполняю...":"Отправить"})]})]})]})})}const Yl="http://127.0.0.1:1234/v1",nd="https://api.openai.com/v1",ra="qwen2.5-14b-instruct-1m",Gi="unsloth/qwen3-30b-a3b-instruct-2507",Zf=[{value:ra,label:"Qwen2.5 14B Instruct 1M"},{value:Gi,label:"Qwen3 30B A3B Instruct 2507"}];function ep(i){return i.llmProvider!=="local"?"openai":i.model===Gi?"local_qwen3":i.model===ra?"local_qwen25":"local_custom"}function tp(i,g){const p=new Map;if(g)for(const R of Zf)p.set(R.value,R);for(const R of i)p.has(R)||p.set(R,{value:R,label:R});return Array.from(p.values())}function np({value:i,modelOptions:g,modelsBusy:p,onChange:R,onReloadModels:E,onTestConnection:L,onSaveLocalConfig:B,lastStatus:ie,busy:ne,embedded:z=!1}){const X=i.llmProvider==="local",Z=ep(i),te=tp(g,X),Re=te.some($=>$.value===i.model),[de,fe]=h.useState(String(i.temperature)),[he,He]=h.useState(String(i.maxOutputTokens));h.useEffect(()=>{fe(String(i.temperature))},[i.temperature]),h.useEffect(()=>{He(String(i.maxOutputTokens))},[i.maxOutputTokens]);const Ye=$=>{const ae=$.replace(",",".").trim();if(!ae){fe(String(i.temperature));return}const Ne=Number(ae);if(!Number.isFinite(Ne)){fe(String(i.temperature));return}R({...i,temperature:Ne}),fe(String(Ne))},Ve=$=>{const ae=$.trim();if(!ae){He(String(i.maxOutputTokens));return}const Ne=Number.parseInt(ae,10);if(!Number.isFinite(Ne)||Ne<=0){He(String(i.maxOutputTokens));return}R({...i,maxOutputTokens:Ne}),He(String(Ne))},Me=o.jsxs(o.Fragment,{children:[o.jsxs("div",{className:"grid-two",children:[o.jsxs("label",{children:["Provider",o.jsxs("select",{value:Z,onChange:$=>{const ae=$.target.value;if(ae==="openai"){R({...i,llmProvider:"openai",baseUrl:nd});return}if(ae==="local_qwen25"){R({...i,llmProvider:"local",model:ra,baseUrl:Yl});return}if(ae==="local_qwen3"){R({...i,llmProvider:"local",model:Gi,baseUrl:Yl});return}R({...i,llmProvider:"local",model:i.llmProvider==="local"?i.model:ra,baseUrl:Yl})},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:Re?i.model:"__manual__",onChange:$=>{const ae=$.target.value;ae!=="__manual__"&&R({...i,model:ae})},children:[o.jsx("option",{value:"__manual__",children:"Manual input"}),te.map($=>o.jsx("option",{value:$.value,children:$.label},$.value))]})]}),o.jsxs("label",{children:["Model ID (manual / current)",o.jsx("input",{value:i.model,onChange:$=>R({...i,model:$.target.value}),placeholder:"qwen2.5-14b-instruct-1m or unsloth/qwen3-30b-a3b-instruct-2507"})]}),X?null:o.jsxs("label",{className:"full-width",children:["OpenAI API Key",o.jsx("input",{type:"password",value:i.apiKey,onChange:$=>R({...i,apiKey:$.target.value}),placeholder:"sk-..."})]}),o.jsxs("label",{className:X?"full-width":void 0,children:[X?"Local server base URL":"Base URL",o.jsx("input",{value:i.baseUrl,onChange:$=>R({...i,baseUrl:$.target.value}),placeholder:X?Yl:nd})]}),o.jsxs("label",{children:["Temperature",o.jsx("input",{type:"number",step:"0.1",value:de,onChange:$=>fe($.target.value),onBlur:$=>Ye($.target.value),onKeyDown:$=>{$.key==="Enter"&&Ye($.target.value)}})]}),o.jsxs("label",{children:["Max output tokens",o.jsx("input",{type:"number",value:he,onChange:$=>He($.target.value),onBlur:$=>Ve($.target.value),onKeyDown:$=>{$.key==="Enter"&&Ve($.target.value)}})]})]}),o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",onClick:()=>B(),children:"Save local config"}),o.jsx("button",{type:"button",onClick:()=>E(),disabled:ne||p,children:p?"Loading models...":"Load model list"}),o.jsx("button",{type:"button",onClick:()=>L(),disabled:ne,children:ne?"Checking...":"Test connection"})]})]});return z?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:ie||"Status: not checked"})]}),Me]}):o.jsx(oa,{title:"LLM Connector",subtitle:"Switch between OpenAI cloud and local OpenAI-compatible server.",actions:o.jsx("span",{className:"status-chip",children:ie||"Status: not checked"}),children:Me})}function rp({value:i,onChange:g,presets:p,selectedPresetId:R,onSelectPreset:E,onLoadPreset:L,onSavePreset:B,onResetDefaults:ie,onDiffPrevious:ne,presetName:z,onPresetNameChange:X,diffSummary:Z,embedded:te=!1}){const Re=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:de=>g({...i,systemPrompt:de.target.value}),rows:6})]}),o.jsxs("label",{children:["Developer / Instruction prompt",o.jsx("textarea",{value:i.developerPrompt,onChange:de=>g({...i,developerPrompt:de.target.value}),rows:6})]}),o.jsxs("label",{children:["Domain prompt",o.jsx("textarea",{value:i.domainPrompt,onChange:de=>g({...i,domainPrompt:de.target.value}),rows:6})]}),o.jsxs("label",{children:["Schema notes",o.jsx("textarea",{value:i.schemaNotes,onChange:de=>g({...i,schemaNotes:de.target.value}),rows:6})]}),o.jsxs("label",{className:"full-width",children:["Few-shot examples",o.jsx("textarea",{value:i.fewShotExamples,onChange:de=>g({...i,fewShotExamples:de.target.value}),rows:8})]})]}),o.jsxs("div",{className:"button-row",children:[o.jsxs("select",{value:R,onChange:de=>E(de.target.value),children:[o.jsx("option",{value:"",children:"Выберите preset..."}),p.map(de=>o.jsx("option",{value:de.id,children:de.name},de.id))]}),o.jsx("button",{type:"button",onClick:()=>L(),children:"Загрузить preset"}),o.jsx("input",{value:z,onChange:de=>X(de.target.value),placeholder:"Имя для сохранения"}),o.jsx("button",{type:"button",onClick:()=>B(),children:"Сохранить preset"}),o.jsx("button",{type:"button",onClick:()=>ne(),children:"Diff с предыдущим"}),o.jsx("button",{type:"button",onClick:()=>ie(),children:"Сбросить к default"})]}),Z?o.jsx("p",{className:"diff-summary",children:Z}):null]});return te?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 уровни управляются отдельно."})]})}),Re]}):o.jsx(oa,{title:"Prompt Manager",subtitle:"Системный, developer и domain уровни управляются отдельно.",children:Re})}const Oi={fromLocal:"",toLocal:"",target:"all",mode:"all",useMock:"any",promptContains:"",limit:120},Xl="needs_dialog_policy_fix",Et="__all__",sa="__live__:",rd="ndc_autoruns_ui_config_v1",sd="ndc-autoruns-save",zi=["Анализ запроса","Получение данных","Подготовка ответа"];function sp(i,g){const p=i.trim();if(!p)return"";if(!g)return p;const R=p.toLowerCase(),E=g.anchor_text.trim(),L=E.toLowerCase();return L&&R.includes(L)?p:`По выбранному объекту "${E}": ${p}`}const Hi=[{id:"general",label:"Общий контур",domain:"",defaultPrompt:"Генерируй реалистичные живые вопросы бухгалтера по 1С. Добавляй разговорные формулировки и опечатки, но сохраняй бизнес-смысл."}];function op(i=Hi){return i.reduce((g,p)=>(g[p.id]=p.defaultPrompt,g),{})}const od={mode:"codex_creative",count:24,personalityId:"general",personalityPrompts:op(),persistToEvalCases:!0,generatedBy:"manual_reviewer"};function $i(i){const g=String(i??"").trim();return/^\d{4}-\d{2}-\d{2}$/.test(g)?g:""}function ld(i){const g=typeof i=="number"&&Number.isFinite(i)?Math.trunc(i):160;return Math.max(110,Math.min(520,g))}function lp(i){const g=i.getFullYear(),p=String(i.getMonth()+1).padStart(2,"0"),R=String(i.getDate()).padStart(2,"0"),E=String(i.getHours()).padStart(2,"0"),L=String(i.getMinutes()).padStart(2,"0");return`${g}-${p}-${R}T${E}:${L}`}function ad(){const i=new Date;return i.setDate(i.getDate()-14),lp(i)}function Zl(i){if(!i.trim())return;const g=Date.parse(i);if(Number.isFinite(g))return new Date(g).toISOString()}function yn(i){if(!i)return"нет данных";const g=Date.parse(i);return Number.isFinite(g)?new Date(g).toLocaleString("ru-RU"):i}function id(i){const g=typeof i.case_message_index=="number"?i.case_message_index:typeof i.message_index=="number"?i.message_index:null;if(g===null||g<0)return null;const p=Math.floor(g/2)+1,R=String(p).padStart(3,"0"),E=i.role==="assistant"?"ответ":"вопрос";return`${R} ${E}`}function Fi(i){return i==="saved_user_sessions"?"Пользовательские сессии":i}function Sd(i){return i?i.context?.agent_run===!0||i.context?.saved_case_set_kind==="agent_semantic_scenario"?!0:typeof i.title=="string"&&i.title.trim().toUpperCase().startsWith("AGENT"):!1}function Ui(i){const g=i.title??yn(i.created_at);return Sd(i)&&!g.trim().toUpperCase().startsWith("AGENT")?`AGENT | ${g}`:g}function ap(i){const g=i[i.length-1];return`Ручная сессия ${yn(g?.created_at??new Date().toISOString())}`}function ip(i,g){return g<=0?0:Math.max(0,Math.min(100,Number((i/g*100).toFixed(1))))}function ea(i){return typeof i!="number"?"нет данных":`${i.toFixed(1)}%`}function up(i){return i==="assistant_stage1"?"assistant/s1":i==="assistant_stage2"?"assistant/s2":i==="assistant_p0"?"assistant/p0":i}function ud(i){return i==="up"?"Рост":i==="down"?"Регресс":"Без изменений"}function bi(i){const g=Math.max(1,Math.min(5,Math.round(i)));return`${"●".repeat(g)}${"○".repeat(5-g)}`}function cd(i){return i.length===0?o.jsx("p",{className:"muted",children:"Покрытие доменов пока не сформировано."}):o.jsx("div",{className:"autoruns-coverage-list",children:i.map(g=>{const p=ip(g.closed_cases,g.total_cases);return o.jsxs("div",{className:"autoruns-coverage-item",children:[o.jsxs("div",{className:"autoruns-coverage-head",children:[o.jsx("strong",{children:g.domain}),o.jsxs("span",{children:[g.closed_cases,"/",g.total_cases," (",p,"%)"]})]}),o.jsx("div",{className:"autoruns-coverage-bar",children:o.jsx("div",{style:{width:`${p}%`}})})]},g.domain)})})}function na(i){return`${sa}${i}`}function no(i){return i.startsWith(sa)}function dd(i){return i.startsWith(sa)?i.slice(sa.length):""}function wd(i){const g=i.report_summary?.run_timestamp??i.created_at,p=Math.max(0,i.total_cases-i.completed_cases);return{run_id:na(i.job_id),eval_target:i.eval_target,run_timestamp:g,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:p,domain_coverage:[{domain:"runtime",total_cases:i.total_cases,closed_cases:i.completed_cases}]}}function Uo(i,g){const p=wd(i),R=i.cases.map(X=>({case_id:X.case_id,domain:null,query_class:null,status:X.status==="completed"?"closed":X.status==="failed"?"open":"unknown",score_index:null,trace_id:null,reply_type:null,session_id:`${i.run_id}-${X.case_id}`,dialog_available:X.messages.length>0,commented_count:0,latest_annotation_at:null,avg_rating:null,checks:null,metric_subscores:null})),L=g!==Et&&R.some(X=>X.case_id===g)?g:R.length>0?Et:"",B={ok:!0,run:p,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:R,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}:{}},ie=[];let ne=0;if(L===Et)for(const X of i.cases)for(let Z=0;ZZ.case_id===L)??null;for(let Z=0;Z<(X?.messages.length??0);Z+=1){const te=X?.messages[Z];te&&ie.push({...te,message_index:Z,case_id:L,case_message_index:Z,commented:!1,annotation:null})}}const z={ok:!0,run_id:p.run_id,case_id:L,source:"assistant_session",session_id:L===Et?`${i.run_id}::__all__`:`${i.run_id}-${L}`,messages:ie,decomposition:[],assistant_mode:{status:i.status,completed_cases:i.completed_cases,total_cases:i.total_cases},annotations:[]};return{detail:B,dialog:z,caseId:L}}function cp({commented:i}){const g=i?"comment-icon-svg commented":"comment-icon-svg";return o.jsx("svg",{className:g,viewBox:"0 0 24 24","aria-hidden":"true",focusable:"false",children:o.jsx("path",{d:"M5 6.5h14v9H11.5l-4.5 3v-3H5z"})})}function fd({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 pd(){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 md(){return o.jsxs("svg",{className:"autoruns-question-grip-svg",viewBox:"0 0 16 16","aria-hidden":"true",focusable:"false",children:[o.jsx("circle",{cx:"4",cy:"4",r:"1"}),o.jsx("circle",{cx:"8",cy:"4",r:"1"}),o.jsx("circle",{cx:"12",cy:"4",r:"1"}),o.jsx("circle",{cx:"4",cy:"8",r:"1"}),o.jsx("circle",{cx:"8",cy:"8",r:"1"}),o.jsx("circle",{cx:"12",cy:"8",r:"1"}),o.jsx("circle",{cx:"4",cy:"12",r:"1"}),o.jsx("circle",{cx:"8",cy:"12",r:"1"}),o.jsx("circle",{cx:"12",cy:"12",r:"1"})]})}function dp({expanded:i}){return o.jsx("svg",{className:i?"autoruns-card-chevron-svg expanded":"autoruns-card-chevron-svg",viewBox:"0 0 16 16","aria-hidden":"true",focusable:"false",children:o.jsx("path",{d:"M3.5 6.2 8 10.4l4.5-4.2"})})}function fp(){return o.jsx("svg",{className:"autoruns-card-launch-svg",viewBox:"0 0 16 16","aria-hidden":"true",focusable:"false",children:o.jsx("path",{d:"M5 3.8 12 8l-7 4.2Z"})})}function pp(){return o.jsx("svg",{className:"autoruns-card-stop-svg",viewBox:"0 0 16 16","aria-hidden":"true",focusable:"false",children:o.jsx("rect",{x:"4.2",y:"4.2",width:"7.6",height:"7.6",rx:"0.8"})})}function ta({expanded:i}){return o.jsx("svg",{className:i?"autoruns-group-chevron-svg expanded":"autoruns-group-chevron-svg",viewBox:"0 0 16 16","aria-hidden":"true",focusable:"false",children:o.jsx("path",{d:"M3.5 6.2 8 10.4l4.5-4.2"})})}function mp({connection:i,modelOptions:g,modelsBusy:p,connectionStatus:R,connectionBusy:E,onConnectionChange:L,onReloadModels:B,onSaveLocalConfig:ie,onTestConnection:ne,prompts:z,onPromptsChange:X,promptPresets:Z,selectedPresetId:te,onSelectPreset:Re,onLoadPreset:de,onSavePreset:fe,onResetDefaults:he,onDiffPrevious:He,presetName:Ye,onPresetNameChange:Ve,diffSummary:Me,assistantPromptVersion:$,decompositionPromptVersion:ae,showSettingsMode:Ne,showAutoRunsMode:be,showAssistantMode:yt,showProgressMode:dt,showCommentsMode:Kt,onLog:ft}){const[b,Ie]=h.useState({...Oi,fromLocal:ad()}),[Ke,re]=h.useState(""),[ee,M]=h.useState(null),[F,I]=h.useState(null),[y,k]=h.useState(null),[se,ue]=h.useState([]),[ce,ge]=h.useState("all"),[_e,Se]=h.useState(!1),[ke,pt]=h.useState(null),[ln,vr]=h.useState([]),[In,xn]=h.useState(""),[Te,Jt]=h.useState(""),[lt,xt]=h.useState(""),[Pt,_n]=h.useState(Hi),[W,mt]=h.useState(od),[yr,Ln]=h.useState([]),[Ge,Tt]=h.useState(""),[Ft,Dn]=h.useState(""),[me,Rt]=h.useState([]),[Xe,_s]=h.useState(!1),[tt,er]=h.useState(null),[xr,an]=h.useState(""),[On,Ut]=h.useState(null),[Sn,Yt]=h.useState(null),[q,At]=h.useState(null),[tr,Hr]=h.useState(null),[Vr,Gr]=h.useState(!1),[zn,bt]=h.useState(!1),[$n,Fn]=h.useState(!1),[Un,bn]=h.useState(!1),[C,Y]=h.useState(!1),[H,Ee]=h.useState(!1),[at,bo]=h.useState(!1),[Wr,Bo]=h.useState(!1),[Bn,ro]=h.useState(""),[so,We]=h.useState(""),[nt,Ss]=h.useState(""),[Xt,ws]=h.useState([]),[qr,Kr]=h.useState([]),[ze,oo]=h.useState(""),[Jr,ks]=h.useState(null),[_r,la]=h.useState(!1),[lo,Yr]=h.useState(!1),[un,Xr]=h.useState(""),[Zt,nr]=h.useState(""),[aa,Sr]=h.useState(String(Oi.limit)),[js,wn]=h.useState(String(od.count)),[rr,Zr]=h.useState(160),[Qo,Qn]=h.useState(""),[es,en]=h.useState(!1),[Hn,wr]=h.useState(!0),[kr,Cs]=h.useState(!0),[ve,ao]=h.useState(!0),[sr,Ns]=h.useState(!0),[ye,tn]=h.useState({open:!1,caseId:"",caseMessageIndex:-1,messageIndex:-1,rating:3,comment:"",manualCaseDecision:Xl,annotationAuthor:"manual_reviewer",saving:!1,error:""}),[je,nn]=h.useState({open:!1,messageIndex:-1,rating:3,comment:"",annotationAuthor:"manual_reviewer",saving:!1,error:""}),[_t,St]=h.useState({open:!1,title:"",saving:!1,error:""}),[Ze,Bt]=h.useState({open:!1,generationId:"",questionIndex:-1,questionText:"",saving:!1,error:""}),[Mt,Vn]=h.useState({open:!1,generationId:"",title:"",saving:!1,error:""}),cn=h.useRef(!1),Es=h.useRef(null),ts=h.useRef(null),Le=W.mode==="saved_user_sessions",Ps=h.useMemo(()=>Pt.find(a=>a.id===W.personalityId)??Pt[0]??Hi[0],[W.personalityId,Pt]),ht=h.useMemo(()=>yr.filter(a=>a.mode===W.mode),[yr,W.mode]),Ae=h.useMemo(()=>ht.find(a=>a.generation_id===Ge)??ht[0]??null,[Ge,ht]),Gn=h.useMemo(()=>_e?se.filter(a=>!a.resolved):se,[se,_e]),De=Gn.find(a=>a.annotation_id===In)??null,jr=y?.messages.find(a=>a.message_index===ye.messageIndex)??null,ns=h.useMemo(()=>{if(!y||ye.messageIndex<0)return null;for(let a=ye.messageIndex-1;a>=0;a-=1){const c=y.messages[a];if(c?.role==="user")return c}return null},[ye.messageIndex,y]),kn=h.useMemo(()=>{const a=new Map;for(const c of qr)c.message_id&&a.set(c.message_id,c);return a},[qr]),rs=je.messageIndex>=0?Xt[je.messageIndex]??null:null,Ho=h.useMemo(()=>{if(je.messageIndex<0)return null;for(let a=je.messageIndex-1;a>=0;a-=1){const c=Xt[a];if(c?.role==="user")return c}return null},[je.messageIndex,Xt]),jn=h.useMemo(()=>{const a=Gn.map(f=>({source:"autorun",key:`autorun:${f.annotation_id}`,updated_at:f.updated_at,rating:f.rating,autorun:f,assistant:null})),c=qr.map(f=>({source:"assistant_live",key:`assistant:${f.annotation_id}`,updated_at:f.updated_at,rating:f.rating,autorun:null,assistant:f}));return[...a,...c].sort((f,S)=>Date.parse(S.updated_at)-Date.parse(f.updated_at))},[qr,Gn]),Ts=h.useMemo(()=>{if(jn.length===0)return null;const a=jn.reduce((c,f)=>c+f.rating,0)/jn.length;return Number(a.toFixed(2))},[jn]),Cr=h.useMemo(()=>{const a=[...ee?.items??[]];return q&&a.unshift(wd(q)),Te&&!a.some(c=>c.run_id===Te)&&F?.run&&a.unshift(F.run),a},[q,ee?.items,F?.run,Te]),G=h.useCallback(a=>{ft?.(`[autoruns] ${a}`)},[ft]),ss=h.useCallback(async a=>{const c=String(a??"").trim();if(!c){Kr([]);return}try{const f=await Pe.loadAssistantAnnotations({session_id:c,limit:400});Kr(f.items??[])}catch(f){const S=f instanceof Error?f.message:String(f);G(`Assistant live annotations load error: ${S}`)}},[G]),It=h.useCallback(a=>{nn(c=>c.saving&&!a?.force?c:{open:!1,messageIndex:-1,rating:3,comment:"",annotationAuthor:"manual_reviewer",saving:!1,error:""})},[]),Nr=h.useCallback(a=>{St(c=>c.saving&&!a?.force?c:{open:!1,title:"",saving:!1,error:""})},[]),or=h.useCallback(a=>{Bt(c=>c.saving&&!a?.force?c:{open:!1,generationId:"",questionIndex:-1,questionText:"",saving:!1,error:""})},[]),os=h.useCallback(a=>{Vn(c=>c.saving&&!a?.force?c:{open:!1,generationId:"",title:"",saving:!1,error:""})},[]),lr=h.useCallback(async(a,c,f)=>{a.stopPropagation(),a.preventDefault();const S=String(c??"").trim();if(S)try{if(navigator?.clipboard?.writeText)await navigator.clipboard.writeText(S);else{const T=document.createElement("textarea");T.value=S,T.setAttribute("readonly","true"),T.style.position="fixed",T.style.opacity="0",document.body.appendChild(T),T.select(),document.execCommand("copy"),document.body.removeChild(T)}G(`${f} copied: ${S}`)}catch(T){const le=T instanceof Error?T.message:String(T);We(`Копирование ${f}: ${le}`),G(`copy ${f} error: ${le}`)}},[G]);function Rs(){let a=0;Xr(zi[0]);const c=window.setInterval(()=>{a=Math.min(a+1,zi.length-1),Xr(zi[a])},650);return()=>window.clearInterval(c)}const Vo=h.useCallback(()=>{Ss(""),ws([]),Kr([]),oo(""),ks(null),Xr(""),nr(""),It({force:!0}),G("Live-чат ассистента в истории автопрогонов сброшен.")},[It,G]),Lt=h.useCallback(async()=>{const a=sp(ze,Jr);if(!a)return;Yr(!0),nr(""),oo(""),ws(f=>[...f,{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=Rs();try{const f=await Pe.sendAssistantMessage({connection:i,prompts:z,userMessage:a,sessionId:nt||void 0,promptVersion:$,useMock:_r});Ss(f.session_id),ws(f.conversation),await ss(f.session_id),Xr("Ответ готов"),G(`Live-ответ ассистента получен: trace=${f.debug.trace_id}`)}catch(f){const S=f instanceof Error?f.message:String(f);nr(S),Xr("Ошибка ассистента"),G(`Live-чат ассистента: ошибка отправки сообщения: ${S}`)}finally{c(),Yr(!1)}},[ze,Jr,nt,_r,$,i,ss,G,z]),Er=h.useCallback(()=>{if(!nt.trim()||Xt.length===0){nr("Сначала получите хотя бы один ответ в живой сессии ассистента.");return}nr(""),St({open:!0,title:ap(Xt),saving:!1,error:""})},[Xt,nt]),io=h.useCallback(async()=>{const a=nt.trim(),c=_t.title.trim();if(!a){St(f=>({...f,error:"Активная сессия ассистента не найдена."}));return}if(!c){St(f=>({...f,error:"Укажите название сессии."}));return}St(f=>({...f,saving:!0,error:""}));try{const f=[z.systemPrompt,z.developerPrompt,z.domainPrompt,z.schemaNotes,z.fewShotExamples].join("||"),S=await Pe.saveAutoRunAssistantSession({session_id:a,title:c,generated_by:W.generatedBy.trim()||void 0,context:{llm_provider:i.llmProvider,model:i.model,assistant_prompt_version:$,decomposition_prompt_version:ae,prompt_fingerprint:f}});Ln(T=>[S.generation,...T.filter(le=>le.generation_id!==S.generation.generation_id)]),mt(T=>({...T,mode:"saved_user_sessions"})),Tt(S.generation.generation_id),Nr({force:!0}),G(`Живая сессия сохранена в автопрогоны: ${S.generation.generation_id}`)}catch(f){const S=f instanceof Error?f.message:String(f);St(T=>({...T,saving:!1,error:S})),G(`Assistant live save error: ${S}`)}},[_t.title,nt,$,W.generatedBy,Nr,i.llmProvider,i.model,ae,G,z.developerPrompt,z.domainPrompt,z.fewShotExamples,z.schemaNotes,z.systemPrompt]),Pr=h.useCallback(a=>{const c=a.trim();if(!c){Sr(String(b.limit));return}if(!/^\d+$/.test(c)){Sr(String(b.limit));return}const f=Number.parseInt(c,10);if(!Number.isFinite(f)){Sr(String(b.limit));return}const S=Math.max(1,Math.min(500,f));S!==b.limit&&Ie(T=>({...T,limit:S})),Sr(String(S))},[b.limit]),Go=h.useCallback(a=>{const c=a.trim();if(!c){wn(String(W.count));return}if(!/^\d+$/.test(c)){wn(String(W.count));return}const f=Number.parseInt(c,10);if(!Number.isFinite(f)){wn(String(W.count));return}const S=Math.max(1,Math.min(200,f));S!==W.count&&mt(T=>({...T,count:S})),wn(String(S))},[W.count]),As=h.useCallback(a=>{Zr(ld(a))},[]),Ms=h.useCallback(a=>{const c=a.currentTarget.offsetHeight;Number.isFinite(c)&&c>0&&As(c)},[As]),Cn=h.useCallback(async()=>{Bo(!0);try{const a=await Pe.loadAutoRunAnnotations({limit:800,manual_case_decision:ce});ue(a.items),pt(a.manual_case_decision_schema??null),vr(a.available_manual_case_decisions??[]),xn(c=>a.items.length===0?"":a.items.some(f=>f.annotation_id===c)?c:a.items[0].annotation_id)}catch(a){G(`Annotations load error: ${a instanceof Error?a.message:String(a)}`)}finally{Bo(!1)}},[ce,G]),Dt=h.useCallback(async()=>{bn(!0);try{const a=await Pe.loadAutoRunAutogenHistory({limit:180});Ln(a.items)}catch(a){G(`Autogen history load error: ${a instanceof Error?a.message:String(a)}`)}finally{bn(!1)}},[G]),uo=h.useCallback(async()=>{try{const c=(await Pe.loadAutoRunAutogenPersonalityCatalog()).items.map(f=>({id:String(f.id??"").trim(),label:String(f.label??"").trim(),domain:typeof f.domain=="string"?f.domain.trim():"",defaultPrompt:String(f.default_prompt??"").trim()})).filter(f=>f.id.length>0&&f.label.length>0);if(c.length===0)return;_n(c.map(f=>({id:f.id,label:f.label,domain:f.domain||"",defaultPrompt:f.defaultPrompt||"Генерируй реалистичные вопросы бухгалтера по выбранному профилю. Не выдумывай непокрытые возможности."})))}catch(a){G(`Autogen personality catalog load error: ${a instanceof Error?a.message:String(a)}`)}},[G]),Nn=h.useCallback(async()=>{Fn(!0);try{const a=await Pe.loadAutoRunPostAnalysis({run_id:Te&&!no(Te)?Te:void 0,limit_per_queue:30,annotation_limit:1500,from:Zl(b.fromLocal),to:Zl(b.toLocal),target:b.target,mode:b.mode,use_mock:b.useMock,prompt_contains:b.promptContains.trim()||void 0});Hr(a)}catch(a){G(`Post-analysis load error: ${a instanceof Error?a.message:String(a)}`),Hr(null)}finally{Fn(!1)}},[b.fromLocal,b.mode,b.promptContains,b.target,b.toLocal,b.useMock,G,Te]),ia=h.useCallback(async()=>{Gr(!0),We("");try{if(W.mode==="saved_user_sessions")throw new Error("Пользовательские сессии сохраняются из живого чата, а не генерируются автоматически.");const a=W.personalityPrompts[W.personalityId]??"",c=[z.systemPrompt,z.developerPrompt,z.domainPrompt,z.schemaNotes,z.fewShotExamples].join(` -`).slice(0,900),f=await Pe.generateAutoRunQuestions({mode:W.mode,count:W.count,domain:Ps.domain||void 0,persist_to_eval_cases:W.persistToEvalCases,generated_by:W.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:$,decomposition_prompt_version:ae,prompt_fingerprint:c,autogen_personality_id:Ps.id,autogen_personality_prompt:a.trim()||void 0}});G(`Generated ${f.generation.count} questions (${f.generation.mode}) id=${f.generation.generation_id}`+(f.generation.saved_case_set_file?` saved=${f.generation.saved_case_set_file}`:"")),Tt(f.generation.generation_id),Rt([...f.generation.questions??[]]),await Dt()}catch(a){const c=a instanceof Error?a.message:String(a);We(`Автогенерация: ${c}`),G(`Autogen generate error: ${c}`)}finally{Gr(!1)}},[$,W.count,W.generatedBy,W.mode,W.personalityId,W.personalityPrompts,W.persistToEvalCases,i.apiKey,i.baseUrl,i.llmProvider,i.maxOutputTokens,i.model,i.temperature,ae,Dt,G,z.developerPrompt,z.domainPrompt,z.fewShotExamples,z.schemaNotes,z.systemPrompt,Ps.domain,Ps.id]),ls=h.useCallback(async(a,c)=>{if(no(a)){const f=dd(a);if(q&&q.job_id===f){const S=Uo(q,c);Jt(a),xt(S.caseId),k(S.dialog);return}k(null);return}bo(!0);try{const f=await Pe.loadAutoRunCaseDialog(a,c);k(f)}catch(f){const S=f instanceof Error?f.message:String(f);We(`Диалог кейса: ${S}`),k(null),G(`Dialog load error for ${a}/${c}: ${S}`)}finally{bo(!1)}},[q,G]),dn=h.useCallback(async(a,c)=>{if(no(a)){const f=dd(a);if(q&&q.job_id===f){const S=Uo(q,c??Et);Jt(a),xt(S.caseId),I(S.detail),k(S.dialog);return}Jt(a),xt(""),I(null),k(null);return}Ee(!0);try{const f=await Pe.loadAutoRunDetail(a);I(f);const S=(c&&(c===Et||f.cases.some(T=>T.case_id===c))?c:"")||(f.cases.length>0?Et:"")||"";Jt(a),xt(S),S?await ls(a,S):k(null)}catch(f){const S=f instanceof Error?f.message:String(f);We(`Детализация прогона: ${S}`),I(null),k(null),G(`Run detail load error for ${a}: ${S}`)}finally{Ee(!1)}},[q,ls,G]),Wn=h.useCallback(async a=>{Y(!0),We("");try{const c=await Pe.loadAutoRunsHistory({from:Zl(b.fromLocal),to:Zl(b.toLocal),target:b.target,mode:b.mode,use_mock:b.useMock,prompt_contains:b.promptContains.trim()||void 0,limit:b.limit});if(M(c),c.items.length===0){Jt(""),xt(""),I(null),k(null);return}const f=a?.keepSelection??!0,S=a?.preferredRunId??"",T=a?.preferredCaseId??"",le=f&&S&&c.items.some(rt=>rt.run_id===S)?S:c.items[0].run_id;await dn(le,f?T:void 0),Nn()}catch(c){const f=c instanceof Error?c.message:String(c);We(`История прогонов: ${f}`),G(`History load error: ${f}`)}finally{Y(!1)}},[b.fromLocal,b.limit,b.mode,b.promptContains,b.target,b.toLocal,b.useMock,Nn,dn,G]),Ot=h.useCallback(()=>{Es.current!==null&&(window.clearTimeout(Es.current),Es.current=null)},[]),co=h.useCallback(async a=>{try{const c=await Pe.loadEvalRunAsyncStatus(a);At(c.job);const f=na(a);if(Te===f){const S=Uo(c.job,lt||Et);I(S.detail),k(S.dialog),xt(S.caseId)}if(c.job.status==="completed"){Ot(),bt(!1),en(!1),Qn("");const S=c.job.report_summary?.run_id??c.job.run_id;await Wn({keepSelection:!0,preferredRunId:S||Te,preferredCaseId:Et}),await Dt(),At(null);return}if(c.job.status==="failed"){Ot(),bt(!1),en(!1),Qn(""),We(`Запуск прогонов: ${c.job.error??"неизвестная ошибка"}`),G(`Autogen async run failed: ${c.job.error??"unknown error"}`);return}if(c.job.status==="canceled"){Ot(),bt(!1),en(!1),Qn(""),At(null),await Wn({keepSelection:!1}),await Dt(),G(`Autogen async run canceled: job=${c.job.job_id}`);return}Ot(),Es.current=window.setTimeout(()=>{co(a)},500)}catch(c){Ot(),bt(!1),en(!1),Qn("");const f=c instanceof Error?c.message:String(c);We(`Запуск прогонов: ${f}`),G(`Autogen async status error: ${f}`)}},[Dt,Wn,G,lt,Te,Ot]),Wo=h.useCallback(async(a,c)=>{Ot(),bt(!0),We("");try{const f=a??Ae;if(!f)throw new Error("История автогенерации пуста. Сначала сгенерируйте пачку вопросов.");const T=(c??(Ae?.generation_id===f.generation_id?me:f.questions)).map(Os=>Os.trim()).filter(Os=>Os.length>0);if(T.length===0)throw new Error("Нет вопросов для запуска: список пустой после ручного редактирования.");const le=b.useMock==="true",rt=$i(Ke),wt=f.mode==="saved_user_sessions",ir=(await Pe.startEvalRunAsync({connection:i,prompts:z,promptVersion:$,mode:"single-pass-strict",caseSetFile:wt?void 0:f.saved_case_set_file??void 0,useMock:le,evalTarget:"assistant_stage1",questions:wt?void 0:T,scenarioQuestions:wt?T:void 0,scenarioTitle:wt?f.title??void 0:void 0,analysisDate:wt?void 0:rt||void 0})).job;Qn(f.generation_id),en(!1),At(ir);const vo=na(ir.job_id),as=Uo(ir,Et);Jt(vo),xt(as.caseId),I(as.detail),k(as.dialog),G(`Запущен async-прогон job=${ir.job_id}, run_id=${ir.run_id}, вопросов=${T.length}`+(f.saved_case_set_file?`, base_case_set=${f.saved_case_set_file}`:"")+(wt?", replay_mode=saved_user_session_scenario":rt?`, analysis_date=${rt}`:", analysis_date=current_state")),co(ir.job_id)}catch(f){const S=f instanceof Error?f.message:String(f);We(`Запуск прогонов: ${S}`),G(`Autogen run error: ${S}`),bt(!1),en(!1),Qn("")}},[Ke,$,i,me,b.useMock,G,co,z,Ae,Ot]),qo=h.useCallback(async()=>{const a=q?.job_id??"";if(!a){bt(!1),en(!1),Qn(""),At(null),Ot();return}en(!0),We("");try{const c=await Pe.cancelEvalRunAsync(a);Ot(),At(null),bt(!1),en(!1),Qn(""),await Wn({keepSelection:!1}),await Dt(),G(`Autogen async run stopped: job=${c.job.job_id}`)}catch(c){const f=c instanceof Error?c.message:String(c);en(!1),We(`Остановка прогона: ${f}`),G(`Autogen stop error: ${f}`)}},[q,Dt,Wn,G,Ot]),Ko=h.useCallback(a=>{if(a.role!=="assistant")return;const c=a.case_id??lt,f=a.case_message_index??a.message_index;tn({open:!0,caseId:c,caseMessageIndex:f,messageIndex:a.message_index,rating:a.annotation?.rating??3,comment:a.annotation?.comment??"",manualCaseDecision:a.annotation?.manual_case_decision??Xl,annotationAuthor:a.annotation?.annotation_author??W.generatedBy,saving:!1,error:""})},[W.generatedBy,lt]),Is=h.useCallback(a=>{tn(c=>c.saving&&!a?.force?c:{open:!1,caseId:"",caseMessageIndex:-1,messageIndex:-1,rating:3,comment:"",manualCaseDecision:Xl,annotationAuthor:W.generatedBy,saving:!1,error:""})},[W.generatedBy]),ua=h.useCallback(async()=>{const a=Te,c=ye.caseId,f=ye.caseMessageIndex;if(!(!a||!c||f<0)){if(no(a)){tn(S=>({...S,error:"Комментарий можно сохранить после завершения прогона."}));return}if(!ye.comment.trim()){tn(S=>({...S,error:"Добавьте комментарий."}));return}tn(S=>({...S,saving:!0,error:""}));try{await Pe.saveAutoRunAnnotation({run_id:a,case_id:c,message_index:f,rating:ye.rating,comment:ye.comment.trim(),manual_case_decision:ye.manualCaseDecision,annotation_author:ye.annotationAuthor.trim()||void 0}),Is({force:!0}),Promise.all([dn(a,lt),Cn(),Nn()]).catch(S=>{const T=S instanceof Error?S.message:String(S);We(`Обновление после комментария: ${T}`),G(`Comment refresh error: ${T}`)})}catch(S){tn(T=>({...T,saving:!1,error:S instanceof Error?S.message:String(S)}))}}},[Is,ye.annotationAuthor,ye.caseId,ye.caseMessageIndex,ye.comment,ye.manualCaseDecision,ye.rating,Cn,Nn,dn,G,lt,Te]),ca=h.useCallback(a=>a.role==="assistant",[]),da=h.useCallback(a=>a.role==="assistant"&&kn.has(a.message_id),[kn]),fo=h.useCallback((a,c)=>{if(a.role!=="assistant")return;const f=nt.trim(),S=String(a.session_id??"").trim();if(!(f||S)){nr("Сначала получите ответ ассистента в активной сессии.");return}!f&&S&&Ss(S);const le=kn.get(a.message_id)??null;nr(""),nn({open:!0,messageIndex:c,rating:le?.rating??3,comment:le?.comment??"",annotationAuthor:le?.annotation_author??"manual_reviewer",saving:!1,error:""})},[kn,nt]),fa=h.useCallback(async()=>{if(je.messageIndex<0)return;if(!je.comment.trim()){nn(f=>({...f,error:"Добавьте комментарий."}));return}const a=Xt[je.messageIndex]??null,c=nt.trim()||(a?.role==="assistant"?String(a.session_id??"").trim():"");if(!c){nn(f=>({...f,error:"Сессия ассистента не найдена."}));return}nn(f=>({...f,saving:!0,error:""}));try{const f=await Pe.saveAssistantAnnotation({session_id:c,message_index:je.messageIndex,rating:je.rating,comment:je.comment.trim(),annotation_author:je.annotationAuthor.trim()||void 0});Kr(S=>{const T=[...S],le=T.findIndex(rt=>rt.annotation_id===f.annotation.annotation_id);return le>=0?T[le]=f.annotation:T.unshift(f.annotation),T.sort((rt,wt)=>Date.parse(wt.updated_at)-Date.parse(rt.updated_at))}),It({force:!0})}catch(f){const S=f instanceof Error?f.message:String(f);nn(T=>({...T,saving:!1,error:S}))}},[je.annotationAuthor,je.comment,je.messageIndex,je.rating,Xt,nt,It]);h.useCallback(a=>{if(!Ae||Ae.mode!=="saved_user_sessions")return;const c=me[a]??"";Bt({open:!0,generationId:Ae.generation_id,questionIndex:a,questionText:c,saving:!1,error:""})},[me,Ae]);const pa=h.useCallback(async()=>{const a=Ze.generationId,c=Ze.questionIndex;if(!a||c<0)return;const f=me.filter((S,T)=>T!==c);if(f.length===0){Bt(S=>({...S,error:"Нельзя удалить последний вопрос из сохраненной сессии."}));return}Bt(S=>({...S,saving:!0,error:""}));try{const S=await Pe.updateAutoRunAutogenQuestions({generation_id:a,questions:f});Ln(T=>T.map(le=>le.generation_id===a?S.generation:le)),Rt(S.generation.questions),or({force:!0}),G(`Обновлена сохраненная сессия: ${a}`)}catch(S){const T=S instanceof Error?S.message:String(S);Bt(le=>({...le,saving:!1,error:T})),G(`Saved session question delete error: ${T}`)}},[or,me,G,Ze.generationId,Ze.questionIndex]),qn=h.useCallback(async(a,c)=>{const f=Ae?.generation_id??"",S=c?.revertQuestions??me;if(Rt(a),!f)return!0;_s(!0);try{const T=await Pe.updateAutoRunAutogenQuestions({generation_id:f,questions:a});return Ln(le=>le.map(rt=>rt.generation_id===f?T.generation:rt)),Rt([...T.generation.questions??[]]),c?.successLog&&G(c.successLog),!0}catch(T){const le=T instanceof Error?T.message:String(T);return Rt(S),We(`Вопросы к запуску: ${le}`),G(`Autogen questions update error: ${le}`),!1}finally{_s(!1)}},[me,G,Ae]),po=h.useCallback(a=>{er(a),an(me[a]??"")},[me]),Kn=h.useCallback(()=>{er(null),an("")},[]),Ls=h.useCallback(async a=>{if(a===null)return;const c=me[a]??"",f=xr.trim();if(!f||f===c){Kn();return}const S=me.map((le,rt)=>rt===a?f:le);await qn(S,{successLog:`Список вопросов обновлен: ${Ae?.generation_id??"local"}`,revertQuestions:me})&&Kn()},[me,xr,Ae,Kn,qn]),Jo=h.useCallback(()=>{Ls(tt)},[Ls,tt]),Yo=h.useCallback(a=>{if(a.key==="Enter"){a.preventDefault(),Ls(tt);return}a.key==="Escape"&&(a.preventDefault(),Kn())},[Ls,tt,Kn]),Xo=h.useCallback(async()=>{const a=[...me,"Новый вопрос"],c=a.length-1;await qn(a,{successLog:`В список добавлен вопрос: ${Ae?.generation_id??"local"}`,revertQuestions:me})&&(er(c),an(a[c]))},[me,Ae,qn]),Zo=h.useCallback(async a=>{if(me.length<=1){We("В списке должен остаться хотя бы один вопрос.");return}const c=me.filter((S,T)=>T!==a);await qn(c,{successLog:`Из списка удален вопрос: ${Ae?.generation_id??"local"}`,revertQuestions:me})&&(er(S=>S===null?S:S===a?null:S>a?S-1:S),an(""))},[me,Ae,qn]),el=h.useCallback((a,c)=>{if(Xe){a.preventDefault();return}Ut(c),Yt(c),a.dataTransfer.effectAllowed="move",a.dataTransfer.setData("text/plain",String(c))},[Xe]),Ds=h.useCallback((a,c)=>{a.preventDefault(),Sn!==c&&Yt(c),a.dataTransfer.dropEffect="move"},[Sn]),Tr=h.useCallback(async(a,c)=>{a.preventDefault();const f=On;if(Yt(null),Ut(null),f===null||f===c)return;const S=[...me],[T]=S.splice(f,1);S.splice(c,0,T),await qn(S,{successLog:`Порядок вопросов обновлен: ${Ae?.generation_id??"local"}`,revertQuestions:me})},[On,me,Ae,qn]),tl=h.useCallback(()=>{Ut(null),Yt(null)},[]),nl=h.useCallback(a=>{Tt(a),Dn(c=>c===a?"":a)},[]),mo=h.useCallback(a=>{Vn({open:!0,generationId:a.generation_id,title:a.title??`${Fi(a.mode)} ${yn(a.created_at)}`,saving:!1,error:""})},[]),rl=h.useCallback(async()=>{const a=Mt.generationId.trim();if(a){Vn(c=>({...c,saving:!0,error:""}));try{const c=await Pe.deleteAutoRunAutogenHistoryRecord(a);Ln(f=>f.filter(S=>S.generation_id!==c.generation_id)),os({force:!0}),G(`Удален набор автопрогона: ${c.generation_id}`+(c.deleted_files.length>0?`, files=${c.deleted_files.length}`:""))}catch(c){const f=c instanceof Error?c.message:String(c);Vn(S=>({...S,saving:!1,error:f})),G(`Autogen record delete error: ${f}`)}}},[Mt.generationId,os,G]),ho=h.useCallback(a=>{ue(c=>c.map(f=>f.annotation_id===a.annotation_id?{...f,...a}:f)),k(c=>c&&{...c,annotations:c.annotations.map(f=>f.annotation_id===a.annotation_id?a:f),messages:c.messages.map(f=>!f.annotation||f.annotation.annotation_id!==a.annotation_id?f:{...f,commented:!0,annotation:a})})},[]),go=h.useCallback(async(a,c)=>{if(a.annotation_id){if(no(a.run_id)){We("Статус выполнения можно менять только для завершённых прогонов.");return}ro(a.annotation_id);try{const f=await Pe.updateAutoRunAnnotation({annotation_id:a.annotation_id,resolved:c,resolved_by:W.generatedBy||void 0});ho(f.annotation),Nn()}catch(f){const S=f instanceof Error?f.message:String(f);We(`Смена статуса кейса: ${S}`),G(`Annotation resolve toggle error: ${S}`)}finally{ro("")}}},[ho,W.generatedBy,Nn,G]),ar=h.useCallback(async a=>{xn(a.annotation_id),await dn(a.run_id,a.case_id),ee?.items.some(c=>c.run_id===a.run_id)||We("Комментарий относится к прогону вне текущего фильтра. Детали загружены напрямую.")},[ee?.items,dn]);h.useEffect(()=>{cn.current||(cn.current=!0,Wn({keepSelection:!1}),Dt(),uo(),Nn())},[Dt,uo,Wn,Nn]),h.useEffect(()=>{cn.current&&Cn()},[ce,Cn]),h.useEffect(()=>{xn(a=>Gn.length===0?"":Gn.some(c=>c.annotation_id===a)?a:Gn[0].annotation_id)},[Gn]),h.useEffect(()=>{Tt(a=>ht.length===0?"":a&&ht.some(c=>c.generation_id===a)?a:ht[0].generation_id)},[ht]),h.useEffect(()=>{if(!Ae){Rt([]),Kn(),Ut(null),Yt(null);return}Rt([...Ae.questions]),Kn(),Ut(null),Yt(null)},[Ae,Kn]),h.useEffect(()=>{if(tt===null)return;const a=window.setTimeout(()=>{ts.current?.focus(),ts.current?.select()},0);return()=>window.clearTimeout(a)},[tt]),h.useEffect(()=>{if(!Le){Dn("");return}Ft&&!ht.some(a=>a.generation_id===Ft)&&Dn("")},[Ft,Le,ht]),h.useEffect(()=>{Sr(String(b.limit))},[b.limit]),h.useEffect(()=>{wn(String(W.count))},[W.count]),h.useEffect(()=>{if(!nt.trim()){Kr([]);return}ss(nt)},[nt,ss]),h.useEffect(()=>{if(!q)return;const a=na(q.job_id);if(Te!==a)return;const c=Uo(q,lt||Et);I(c.detail),k(c.dialog),xt(c.caseId)},[q,lt,Te]),h.useEffect(()=>()=>{Ot()},[Ot]),h.useEffect(()=>{Pt.length!==0&&mt(a=>{let c=!1;const f={...a.personalityPrompts};for(const T of Pt)(typeof f[T.id]!="string"||f[T.id].trim().length===0)&&(f[T.id]=T.defaultPrompt,c=!0);let S=a.personalityId;return Pt.some(T=>T.id===a.personalityId)||(S=Pt[0].id,c=!0),c?{...a,personalityId:S,personalityPrompts:f}:a})},[Pt]),h.useEffect(()=>{const a=localStorage.getItem(rd);if(a)try{const c=JSON.parse(a);if(c.filters){const f=c.filters;Ie(S=>({...S,...f,limit:typeof f.limit=="number"?Math.max(1,Math.min(500,f.limit)):S.limit}))}typeof c.analysisDate=="string"&&re($i(c.analysisDate)),typeof c.autogenPersonalityPromptHeight=="number"&&Zr(ld(c.autogenPersonalityPromptHeight)),c.groupsExpanded&&(typeof c.groupsExpanded.filters=="boolean"&&wr(c.groupsExpanded.filters),typeof c.groupsExpanded.generationContext=="boolean"&&Cs(c.groupsExpanded.generationContext),typeof c.groupsExpanded.autogen=="boolean"&&ao(c.groupsExpanded.autogen),typeof c.groupsExpanded.savedSessions=="boolean"&&Ns(c.groupsExpanded.savedSessions)),c.autoGenSettings&&mt(f=>{const S={...f.personalityPrompts},T=c.autoGenSettings?.personalityPrompts??{};for(const[rt,wt]of Object.entries(T))typeof wt=="string"&&rt.trim().length>0&&(S[rt.trim()]=wt);const le=typeof c.autoGenSettings?.personalityId=="string"&&c.autoGenSettings.personalityId.trim().length>0?c.autoGenSettings.personalityId.trim():f.personalityId;return{...f,mode:c.autoGenSettings?.mode==="codex_creative"||c.autoGenSettings?.mode==="qwen_seed"||c.autoGenSettings?.mode==="saved_user_sessions"?c.autoGenSettings.mode:f.mode,count:typeof c.autoGenSettings?.count=="number"?Math.max(1,Math.min(200,c.autoGenSettings.count)):f.count,personalityId:le,personalityPrompts:S,persistToEvalCases:typeof c.autoGenSettings?.persistToEvalCases=="boolean"?c.autoGenSettings.persistToEvalCases:f.persistToEvalCases,generatedBy:typeof c.autoGenSettings?.generatedBy=="string"?c.autoGenSettings.generatedBy:f.generatedBy}}),(c.annotationDecisionFilter==="all"||typeof c.annotationDecisionFilter=="string"&&c.annotationDecisionFilter.length>0)&&ge(c.annotationDecisionFilter),typeof c.hideResolvedAnnotations=="boolean"&&Se(c.hideResolvedAnnotations)}catch{}},[]);const sl=h.useCallback(()=>{const a={filters:b,analysisDate:Ke,autogenPersonalityPromptHeight:rr,groupsExpanded:{filters:Hn,generationContext:kr,autogen:ve,savedSessions:sr},autoGenSettings:{mode:W.mode,count:W.count,personalityId:W.personalityId,personalityPrompts:W.personalityPrompts,persistToEvalCases:W.persistToEvalCases,generatedBy:W.generatedBy},annotationDecisionFilter:ce,hideResolvedAnnotations:_e};localStorage.setItem(rd,JSON.stringify(a))},[Ke,ce,W,ve,rr,b,Hn,kr,_e,sr]);return h.useEffect(()=>{const a=()=>{sl(),G("Сохранены настройки панели автопрогонов.")};return window.addEventListener(sd,a),()=>{window.removeEventListener(sd,a)}},[G,sl]),o.jsxs(oa,{className:"autoruns-frame",title:"",hideHeader:!0,children:[o.jsxs("div",{className:"autoruns-columns",children:[Ne?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(np,{embedded:!0,value:i,modelOptions:g,modelsBusy:p,onChange:L,onReloadModels:B,onSaveLocalConfig:ie,onTestConnection:ne,lastStatus:R,busy:E}),o.jsx(rp,{embedded:!0,value:z,onChange:X,presets:Z,selectedPresetId:te,onSelectPreset:Re,onLoadPreset:de,onSavePreset:fe,onResetDefaults:he,onDiffPrevious:He,presetName:Ye,onPresetNameChange:Ve,diffSummary:Me})]})]}):null,be?o.jsxs("section",{className:"autoruns-col",children:[o.jsx("div",{className:"autoruns-col-header",children:o.jsx("h3",{children:"Автопрогоны"})}),o.jsxs("div",{className:"autoruns-group-heading",children:[o.jsx("h4",{children:"Настройки выборки"}),o.jsx("button",{type:"button",className:"autoruns-group-toggle",onClick:()=>wr(a=>!a),"aria-label":Hn?"Скрыть группу настройки выборки":"Показать группу настройки выборки",title:Hn?"Скрыть группу":"Показать группу",children:o.jsx(ta,{expanded:Hn})})]}),Hn?o.jsxs(o.Fragment,{children:[o.jsxs("div",{className:"autoruns-form-grid",children:[o.jsxs("label",{children:["Дата с",o.jsx("input",{type:"datetime-local",value:b.fromLocal,onChange:a=>Ie(c=>({...c,fromLocal:a.target.value}))})]}),o.jsxs("label",{children:["Дата по",o.jsx("input",{type:"datetime-local",value:b.toLocal,onChange:a=>Ie(c=>({...c,toLocal:a.target.value}))})]}),o.jsxs("label",{children:["Целевой контур",o.jsxs("select",{value:b.target,onChange:a=>Ie(c=>({...c,target:a.target.value})),children:[o.jsx("option",{value:"all",children:"все"}),(ee?.available.targets??[]).map(a=>o.jsx("option",{value:a,children:a},a))]})]}),o.jsxs("label",{children:["Режим",o.jsxs("select",{value:b.mode,onChange:a=>Ie(c=>({...c,mode:a.target.value})),children:[o.jsx("option",{value:"all",children:"все"}),(ee?.available.modes??[]).map(a=>o.jsx("option",{value:a,children:a},a))]})]}),o.jsxs("label",{children:["Использовать mock",o.jsxs("select",{value:b.useMock,onChange:a=>Ie(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:aa,onChange:a=>{const c=a.target.value;(c===""||/^\d+$/.test(c))&&Sr(c)},onBlur:a=>Pr(a.target.value),onKeyDown:a=>{a.key==="Enter"&&Pr(a.target.value)}})]}),o.jsxs("label",{className:"full-width",children:["Версия промпта содержит",o.jsx("input",{value:b.promptContains,onChange:a=>Ie(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:(ee?.available.prompt_versions??[]).map(a=>o.jsx("option",{value:a},a))}),o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",disabled:C,onClick:()=>{Wn({keepSelection:!1})},children:C?"Обновляю...":"Применить"}),o.jsx("button",{type:"button",className:"tab",onClick:()=>{Ie({...Oi,fromLocal:ad()}),We("")},children:"Сбросить фильтры"})]})]}):null,o.jsxs("div",{className:"autoruns-group-heading",children:[o.jsx("h4",{children:"Контур генерации"}),o.jsx("button",{type:"button",className:"autoruns-group-toggle",onClick:()=>Cs(a=>!a),"aria-label":kr?"Скрыть группу контура генерации":"Показать группу контура генерации",title:kr?"Скрыть группу":"Показать группу",children:o.jsx(ta,{expanded:kr})})]}),kr?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:$})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Промпт декомпозиции:"}),o.jsx("strong",{children:ae})]})]}):null,o.jsxs("div",{className:"autoruns-group-heading",children:[o.jsx("h4",{children:"Автопрогоны"}),o.jsx("button",{type:"button",className:"autoruns-group-toggle",onClick:()=>ao(a=>!a),"aria-label":ve?"Скрыть группу автопрогонов":"Показать группу автопрогонов",title:ve?"Скрыть группу":"Показать группу",children:o.jsx(ta,{expanded:ve})})]}),ve?o.jsxs(o.Fragment,{children:[o.jsxs("div",{className:"autoruns-form-grid",children:[o.jsxs("label",{children:["Режимы",o.jsxs("select",{value:W.mode,onChange:a=>mt(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.jsx("option",{value:"saved_user_sessions",children:"Пользовательские сессии"})]})]}),Le?null:o.jsxs(o.Fragment,{children:[o.jsxs("label",{children:["Кол-во",o.jsx("input",{type:"number",min:1,max:200,value:js,onChange:a=>{const c=a.target.value;(c===""||/^\d+$/.test(c))&&wn(c)},onBlur:a=>Go(a.target.value),onKeyDown:a=>{a.key==="Enter"&&Go(a.target.value)}})]}),o.jsxs("label",{children:["Личность автогенерации",o.jsx("select",{value:W.personalityId,onChange:a=>mt(c=>({...c,personalityId:a.target.value})),children:Pt.map(a=>o.jsx("option",{value:a.id,children:a.label},a.id))})]}),o.jsxs("label",{children:["Кто генерирует",o.jsx("input",{value:W.generatedBy,onChange:a=>mt(c=>({...c,generatedBy:a.target.value})),placeholder:"manual_reviewer"})]}),o.jsxs("label",{className:"full-width",children:["Промпт личности",o.jsx("textarea",{className:"autoruns-personality-prompt",value:W.personalityPrompts[W.personalityId]??"",onChange:a=>mt(c=>({...c,personalityPrompts:{...c.personalityPrompts,[c.personalityId]:a.target.value}})),placeholder:"Текст промпта для выбранной личности автогенерации",style:{height:`${rr}px`},onMouseUp:Ms,onTouchEnd:Ms})]}),o.jsxs("label",{className:"checkbox-row",children:[o.jsx("input",{type:"checkbox",checked:W.persistToEvalCases,onChange:a=>mt(c=>({...c,persistToEvalCases:a.target.checked}))}),"Сохранять кейс-сет в `eval_cases`"]})]})]}),Le?null:o.jsxs("div",{className:"autoruns-form-grid",children:[o.jsxs("label",{children:["Дата анализа (срез)",o.jsx("input",{type:"date",value:Ke,onChange:a=>re($i(a.target.value))})]}),o.jsx("div",{className:"button-row",children:o.jsx("button",{type:"button",className:"tab",disabled:!Ke,onClick:()=>re(""),children:"Сбросить дату среза"})})]}),o.jsxs("div",{className:"button-row",children:[Le?null:o.jsxs(o.Fragment,{children:[o.jsx("button",{type:"button",disabled:Vr,onClick:()=>{ia()},children:Vr?"Генерирую...":"Сгенерировать пачку"}),o.jsx("button",{type:"button",className:"tab",disabled:Un,onClick:()=>{Dt()},children:Un?"Обновляю...":"Обновить историю"})]}),o.jsx("button",{type:"button",className:"autoruns-run-launch-btn",style:Le?{display:"none"}:void 0,disabled:es||!zn&&(me.length===0||!Ae),onClick:()=>{zn?qo():Wo()},children:zn?es?"Останавливаю...":"Остановить прогон":"Запустить прогон"})]}),o.jsx("div",{className:"autoruns-form-grid",children:o.jsxs("label",{className:"full-width",children:[Le?"Сохраненная сессия":"Кейс-сет для запуска",o.jsxs("select",{value:Ge,onChange:a=>Tt(a.target.value),disabled:ht.length===0,children:[ht.length===0?o.jsx("option",{value:"",children:Le?"нет сохраненных сессий":"нет генераций"}):null,ht.map(a=>o.jsxs("option",{value:a.generation_id,children:[yn(a.created_at)," | ",Ui(a)??Fi(a.mode)," | ",a.count]},a.generation_id))]})]})}),o.jsxs(o.Fragment,{children:[o.jsxs("div",{className:"autoruns-generated-questions",style:Le?{display:"none"}:void 0,children:[o.jsx("div",{className:"autoruns-generated-questions-head",children:o.jsxs("strong",{children:["Вопросы к запуску: ",me.length]})}),me.length===0?o.jsx("p",{className:"muted",children:Le?"Список вопросов пуст. Сначала сохраните живую пользовательскую сессию.":"Список вопросов пуст. Сгенерируйте пачку или добавьте вопрос вручную."}):o.jsx("div",{className:"autoruns-generated-questions-list",children:me.map((a,c)=>o.jsxs("div",{className:["autoruns-generated-question-item",Sn===c?"drag-over":"",On===c?"dragging":"",tt===c?"editing":""].filter(Boolean).join(" "),onDragOver:f=>Ds(f,c),onDrop:f=>{Tr(f,c)},children:[o.jsx("button",{type:"button",className:"autoruns-question-grip-btn",draggable:!Xe&&tt!==c,disabled:Xe||tt===c,onDragStart:f=>el(f,c),onDragEnd:tl,title:"Перетащить вопрос","aria-label":`Перетащить вопрос ${c+1}`,children:o.jsx(md,{})}),tt===c?o.jsxs(o.Fragment,{children:[o.jsx("input",{ref:ts,className:"autoruns-generated-question-input",value:xr,onChange:f=>an(f.target.value),onBlur:Jo,onKeyDown:Yo,placeholder:"Текст вопроса",disabled:Xe}),o.jsx("button",{type:"button",className:"autoruns-remove-question-btn",onMouseDown:f=>f.preventDefault(),onClick:()=>{Zo(c)},title:"Удалить вопрос","aria-label":`Удалить вопрос ${c+1}`,disabled:Xe,children:"×"})]}):o.jsxs("button",{type:"button",className:"autoruns-generated-question-text",onDoubleClick:()=>po(c),title:"Двойной клик для редактирования",children:[c+1,". ",a]})]},`${c}-${a.slice(0,24)}`))}),o.jsx("button",{type:"button",className:"autoruns-add-question-btn",onClick:()=>{Xo()},disabled:!Ae||Xe,children:"+"})]}),Le?null:o.jsx("p",{className:"muted",children:"Запуск выполняет `assistant_stage1` eval по выбранному кейс-сету."})]})]}):null,o.jsxs("div",{className:"autoruns-group-heading",children:[o.jsx("h4",{children:Le?"Сохраненные пользовательские сессии":"История автогенераций"}),o.jsx("button",{type:"button",className:"autoruns-group-toggle",onClick:()=>Ns(a=>!a),"aria-label":sr?Le?"Скрыть группу сохраненных пользовательских сессий":"Скрыть группу истории автогенераций":Le?"Показать группу сохраненных пользовательских сессий":"Показать группу истории автогенераций",title:sr?"Скрыть группу":"Показать группу",children:o.jsx(ta,{expanded:sr})})]}),sr?o.jsxs("div",{className:"autoruns-autogen-list",children:[Un?o.jsx("p",{className:"muted",children:Le?"Загружаю сохраненные пользовательские сессии...":"Загружаю историю автогенераций..."}):null,!Un&&ht.length===0?o.jsx("p",{className:"muted",children:Le?"Сохраненные пользовательские сессии пока пусты.":"История автогенераций пока пустая."}):null,ht.slice(0,30).map(a=>{const c=zn&&Qo===a.generation_id,f=zn&&Qo!==a.generation_id;return o.jsxs("article",{className:["autoruns-autogen-item",Ge===a.generation_id?"selected":"",Ft===a.generation_id?"expanded":"",Le?"saved-session":""].filter(Boolean).join(" "),onClick:Le?void 0:()=>Tt(a.generation_id),children:[Le?o.jsxs("div",{className:"autoruns-saved-session-topbar",children:[o.jsx("button",{type:"button",className:"autoruns-saved-session-icon-btn",disabled:es||f,onClick:S=>{if(S.preventDefault(),S.stopPropagation(),c){qo();return}Tt(a.generation_id),Wo(a,Ge===a.generation_id?me:a.questions)},title:c?"Остановить прогон":"Запустить прогон","aria-label":`${c?"Остановить прогон":"Запустить прогон"} для ${Ui(a)}`,children:c?o.jsx(pp,{}):o.jsx(fp,{})}),o.jsx("button",{type:"button",className:"autoruns-autogen-delete-btn",onClick:S=>{S.preventDefault(),S.stopPropagation(),mo(a)},title:"Удалить сохраненный набор","aria-label":`Удалить набор ${a.generation_id}`,children:"×"})]}):null,o.jsxs("header",{children:[o.jsx("strong",{children:Ui(a)}),o.jsx("div",{className:"autoruns-autogen-card-actions",children:o.jsx("button",{type:"button",className:"autoruns-autogen-delete-btn",onClick:S=>{S.preventDefault(),S.stopPropagation(),mo(a)},title:"Удалить сохраненный набор","aria-label":`Удалить набор ${a.generation_id}`,children:"×"})})]}),o.jsxs("div",{className:"autoruns-saved-session-meta",children:[o.jsx("strong",{className:"autoruns-saved-session-meta-label",children:"Дата и время"}),o.jsxs("span",{className:"autoruns-saved-session-meta-value",children:["| ",yn(a.created_at)]})]}),o.jsxs("div",{className:"autoruns-saved-session-meta autoruns-saved-session-meta-id",children:[o.jsxs("div",{className:"autoruns-saved-session-meta-id-main",children:[o.jsx("strong",{className:"autoruns-saved-session-meta-label",children:"ID"}),o.jsxs("span",{className:"autoruns-saved-session-meta-value",children:["| ",a.generation_id]})]}),o.jsx("span",{role:"button",tabIndex:0,className:"autoruns-copy-run-id-btn",onClick:S=>{lr(S,a.generation_id,"set id")},onKeyDown:S=>{(S.key==="Enter"||S.key===" ")&&(S.preventDefault(),lr(S,a.generation_id,"set id"))},title:"Скопировать id набора","aria-label":`Скопировать id набора ${a.generation_id}`,children:o.jsx(pd,{})})]}),o.jsxs("div",{className:"autoruns-saved-session-meta",children:[o.jsx("strong",{className:"autoruns-saved-session-meta-label",children:"Режим"}),o.jsxs("span",{className:"autoruns-saved-session-meta-value",children:["| ",Fi(a.mode)]})]}),o.jsxs("div",{className:"autoruns-saved-session-meta",children:[o.jsx("strong",{className:"autoruns-saved-session-meta-label",children:"Тип"}),o.jsxs("span",{className:"autoruns-saved-session-meta-value",children:["| ",Sd(a)?"АГЕНТНЫЙ ПРОГОН":"АВТОПРОГОН"]})]}),Le?o.jsxs(o.Fragment,{children:[o.jsxs("div",{className:"autoruns-saved-session-meta",children:[o.jsx("strong",{className:"autoruns-saved-session-meta-label",children:"Количество вопросов"}),o.jsxs("span",{className:"autoruns-saved-session-meta-value",children:["| ",a.questions.length]})]}),o.jsx("div",{className:"autoruns-saved-session-footer",children:o.jsx("button",{type:"button",className:"autoruns-saved-session-icon-btn",onClick:S=>{S.preventDefault(),S.stopPropagation(),nl(a.generation_id)},title:Ft===a.generation_id?"Скрыть вопросы":"Показать вопросы","aria-label":Ft===a.generation_id?"Скрыть вопросы":"Показать вопросы",children:o.jsx(dp,{expanded:Ft===a.generation_id})})}),o.jsx("div",{className:Ft===a.generation_id?"autoruns-saved-session-questions expanded":"autoruns-saved-session-questions",children:o.jsxs("div",{className:"autoruns-generated-questions autoruns-generated-questions-embedded",children:[o.jsx("div",{className:"autoruns-generated-questions-head",children:o.jsxs("strong",{children:["Вопросы к запуску:"," ",Ge===a.generation_id?me.length:a.questions.length]})}),(Ge===a.generation_id?me:a.questions).length===0?o.jsx("p",{className:"muted",children:"Список вопросов пуст."}):o.jsx("div",{className:"autoruns-generated-questions-list",children:(Ge===a.generation_id?me:a.questions).map((S,T)=>o.jsxs("div",{className:["autoruns-generated-question-item",Sn===T&&Ge===a.generation_id?"drag-over":"",On===T&&Ge===a.generation_id?"dragging":"",tt===T&&Ge===a.generation_id?"editing":""].filter(Boolean).join(" "),onDragOver:le=>Ge===a.generation_id?Ds(le,T):void 0,onDrop:le=>Ge===a.generation_id?void Tr(le,T):void 0,children:[o.jsx("button",{type:"button",className:"autoruns-question-grip-btn",draggable:Ge===a.generation_id&&!Xe&&tt!==T,disabled:Ge!==a.generation_id||Xe||tt===T,onDragStart:le=>{Tt(a.generation_id),el(le,T)},onDragEnd:tl,title:"Перетащить вопрос","aria-label":`Перетащить вопрос ${T+1}`,children:o.jsx(md,{})}),Ge===a.generation_id&&tt===T?o.jsxs(o.Fragment,{children:[o.jsx("input",{ref:ts,className:"autoruns-generated-question-input",value:xr,onChange:le=>an(le.target.value),onBlur:Jo,onKeyDown:Yo,placeholder:"Текст вопроса",disabled:Xe}),o.jsx("button",{type:"button",className:"autoruns-remove-question-btn",onMouseDown:le=>le.preventDefault(),onClick:()=>{Zo(T)},title:"Удалить вопрос","aria-label":`Удалить вопрос ${T+1}`,disabled:Xe,children:"×"})]}):o.jsxs("button",{type:"button",className:"autoruns-generated-question-text",onDoubleClick:()=>{Tt(a.generation_id),po(T)},title:"Двойной клик для редактирования",children:[T+1,". ",S]})]},`${a.generation_id}-${T}-${S.slice(0,24)}`))}),o.jsx("button",{type:"button",className:"autoruns-add-question-btn",onClick:()=>{Tt(a.generation_id),Xo()},disabled:Ge!==a.generation_id||Xe,children:"+"})]})})]}):null]},a.generation_id)})]}):null,o.jsxs("details",{className:"autoruns-prompt-details",children:[o.jsx("summary",{children:"Копия активного промпта (только чтение)"}),o.jsxs("label",{children:["Системный",o.jsx("textarea",{readOnly:!0,value:z.systemPrompt})]}),o.jsxs("label",{children:["Разработчика",o.jsx("textarea",{readOnly:!0,value:z.developerPrompt})]}),o.jsxs("label",{children:["Доменный",o.jsx("textarea",{readOnly:!0,value:z.domainPrompt})]}),o.jsxs("label",{children:["Заметки по схеме",o.jsx("textarea",{readOnly:!0,value:z.schemaNotes})]}),o.jsxs("label",{children:["Примеры few-shot",o.jsx("textarea",{readOnly:!0,value:z.fewShotExamples})]})]}),so?o.jsx("p",{className:"error-text",children:so}):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:(ee?.stats.runs_total??0)+(q?1:0)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Средний score"}),o.jsx("strong",{children:ea(ee?.stats.avg_score_index??null)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Тренд"}),o.jsx("strong",{children:ee?ud(ee.stats.trend):"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Блокеры"}),o.jsx("strong",{children:ee?.stats.blocking_runs??0})]})]}),o.jsxs("div",{className:"autoruns-run-list",children:[Cr.map(a=>o.jsxs("button",{type:"button",className:Te===a.run_id?"autoruns-run-item selected":"autoruns-run-item",onClick:()=>{dn(a.run_id)},children:[o.jsxs("div",{className:"autoruns-run-head",children:[o.jsx("strong",{children:yn(a.run_timestamp)}),o.jsx("span",{children:up(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=>{lr(c,a.run_id,"run id")},onKeyDown:c=>{(c.key==="Enter"||c.key===" ")&&(c.preventDefault(),lr(c,a.run_id,"run id"))},title:"Скопировать run id","aria-label":`Скопировать run id ${a.run_id}`,children:o.jsx(pd,{})})]}),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:["оценка: ",ea(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)),Cr.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:Te,onChange:a=>{const c=a.target.value;dn(c)},children:Cr.map(a=>o.jsxs("option",{value:a.run_id,children:[yn(a.run_timestamp)," | ",a.run_id]},a.run_id))})]}),o.jsxs("label",{children:["Кейс",o.jsxs("select",{value:lt,onChange:a=>{const c=a.target.value;xt(c),Te&&c&&ls(Te,c)},children:[(F?.cases.length??0)>0?o.jsx("option",{value:Et,children:"ВСЕ кейсы подряд"}):null,(F?.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:[(F?.cases.length??0)>0?o.jsxs("button",{type:"button",className:lt===Et?"autoruns-case-item selected":"autoruns-case-item",onClick:()=>{xt(Et),Te&&ls(Te,Et)},children:[o.jsx("span",{children:"ВСЕ кейсы подряд"}),o.jsx("span",{children:F?.cases.length})]},Et):null,(F?.cases??[]).map(a=>o.jsxs("button",{type:"button",className:lt===a.case_id?"autoruns-case-item selected":"autoruns-case-item",onClick:()=>{xt(a.case_id),Te&&ls(Te,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:[at||H?o.jsx("p",{className:"muted",children:"Загружаю диалог..."}):null,!at&&!H&&(y?.messages.length??0)===0?o.jsx("p",{className:"muted",children:"Диалог для этого прогона не найден."}):null,(y?.messages??[]).map((a,c)=>{const f=a.role==="assistant"?"assistant":"user";return o.jsxs("article",{className:`autoruns-msg ${f}`,children:[o.jsxs("header",{children:[o.jsx("strong",{children:f==="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,id(a)?o.jsx("span",{className:"autoruns-msg-case-tag",children:id(a)}):null,o.jsx("span",{children:a.created_at?yn(a.created_at):"нет данных"}),f==="assistant"&&!no(Te)?o.jsxs(o.Fragment,{children:[o.jsx("button",{type:"button",className:a.commented?"autoruns-comment-icon commented":"autoruns-comment-icon",onClick:()=>Ko(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(cp,{commented:a.commented})}),a.annotation?o.jsx("button",{type:"button",className:a.annotation.resolved?"autoruns-resolve-toggle resolved":"autoruns-resolve-toggle",onClick:()=>{go(a.annotation,!a.annotation.resolved)},disabled:Bn===a.annotation.annotation_id,title:a.annotation.resolved?"Отметить кейс как невыполненный":"Отметить кейс как выполненный","aria-label":a.annotation.resolved?"Отметить кейс как невыполненный":"Отметить кейс как выполненный",children:o.jsx(fd,{resolved:a.annotation.resolved})}):null]}):null]})]}),o.jsx("p",{children:a.text}),f==="assistant"&&a.annotation?o.jsxs("div",{className:"autoruns-msg-annotation",children:[o.jsx("strong",{children:bi(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??`${f}-${c}`)})]})]}),yt?o.jsx("div",{className:"autoruns-col autoruns-assistant-live-col",children:o.jsx(Xf,{sessionId:nt,conversation:Xt,inputValue:ze,onInputChange:oo,selectedContextChip:Jr,onSelectContextChip:ks,onClearContextChip:()=>ks(null),useMock:_r,onUseMockChange:la,onSend:Lt,onClear:Vo,onSaveSession:Er,busy:lo,saveBusy:_t.saving,saveDisabled:!nt.trim()||Xt.length===0||lo,statusText:un,errorMessage:Zt,showSaveAction:!0,showCommentAction:!0,onCommentAssistantMessage:fo,isAssistantMessageCommented:da,canCommentAssistantMessage:ca})}):null,dt?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:ea(ee?.stats.latest_score_index??null)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Предыдущий"}),o.jsx("strong",{children:ea(ee?.stats.previous_score_index??null)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Тренд"}),o.jsx("strong",{children:ee?ud(ee.stats.trend):"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Пробелы качества"}),o.jsx("strong",{children:ee?.stats.quality_gap_runs??0})]})]}),o.jsx("h4",{children:"Покрытие доменов (история)"}),cd(ee?.stats.domain_coverage??[]),o.jsx("h4",{style:{marginTop:14},children:"Покрытие доменов (выбранный прогон)"}),cd(F?.coverage.domain_coverage??[]),o.jsx("h4",{style:{marginTop:14},children:"Очереди фиксов пост-анализа"}),$n?o.jsx("p",{className:"muted",children:"Собираю пост-анализ..."}):null,$n?null:o.jsx("div",{className:"autoruns-stats-grid",children:Object.entries(tr?.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:[(tr?.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)),!$n&&(tr?.post_analysis.recommended_regression_candidates.length??0)===0?o.jsx("p",{className:"muted",children:"Рекомендованных кандидатов пока нет."}):null]})]}):null,Kt?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:ce,onChange:a=>ge(a.target.value),children:[o.jsx("option",{value:"all",children:"все"}),(ln.length>0?ln:ke?.enum??[]).map(a=>o.jsx("option",{value:a,children:String(ke?.labels?.[a]??a)},a))]})]}),o.jsx("button",{type:"button",className:"tab autoruns-resolved-filter-toggle",onClick:()=>Se(a=>!a),children:_e?"Показать выполненные":"Скрыть выполненные"})]}),o.jsxs("div",{className:"autoruns-stats-grid",children:[o.jsxs("div",{children:[o.jsx("span",{children:"Комментариев"}),o.jsx("strong",{children:jn.length})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Средний рейтинг"}),o.jsx("strong",{children:Ts===null?"нет данных":`${Ts.toFixed(2)} / 5`})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Последний"}),o.jsx("strong",{children:jn.length>0?yn(jn[0].updated_at):"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Статус"}),o.jsx("strong",{children:Wr?"обновляю":"готово"})]})]}),o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",disabled:Wr,onClick:()=>{Cn()},children:Wr?"Обновляю...":"Обновить список"}),o.jsx("button",{type:"button",className:"tab",disabled:$n,onClick:()=>{Nn()},children:$n?"Идет пост-анализ...":"Обновить пост-анализ"})]}),o.jsxs("div",{className:"autoruns-comments-list",children:[Wr?o.jsx("p",{className:"muted",children:"Загружаю комментарии..."}):null,!Wr&&jn.length===0?o.jsx("p",{className:"muted",children:se.length===0&&qr.length===0?"Пока нет откомментированных ответов.":"Нет открытых кейсов по текущему фильтру."}):null,jn.map(a=>{if(a.source==="assistant_live"){const f=a.assistant;return o.jsxs("article",{className:"autoruns-comment-item",children:[o.jsxs("div",{className:"autoruns-comment-head",children:[o.jsx("strong",{children:bi(f.rating)}),o.jsx("div",{className:"autoruns-comment-head-actions",children:o.jsx("span",{children:yn(f.updated_at)})})]}),o.jsxs("div",{className:"autoruns-run-meta",children:["live-session: ",f.session_id]}),o.jsxs("div",{className:"autoruns-run-meta",children:["msg=",f.message_index]}),o.jsxs("div",{className:"autoruns-run-meta",children:["source=assistant_live",f.annotation_author?` | author=${f.annotation_author}`:""]}),f.context.question_text?o.jsxs("p",{children:["Q: ",f.context.question_text]}):null,f.context.answer_text?o.jsxs("p",{children:["A: ",f.context.answer_text]}):null,o.jsx("p",{children:f.comment})]},a.key)}const c=a.autorun;return o.jsxs("article",{className:In===c.annotation_id?"autoruns-comment-item selected":"autoruns-comment-item",onClick:()=>{ar(c)},role:"button",tabIndex:0,onKeyDown:f=>{(f.key==="Enter"||f.key===" ")&&(f.preventDefault(),ar(c))},children:[o.jsxs("div",{className:"autoruns-comment-head",children:[o.jsx("strong",{children:bi(c.rating)}),o.jsxs("div",{className:"autoruns-comment-head-actions",children:[o.jsx("span",{children:yn(c.updated_at)}),o.jsx("button",{type:"button",className:c.resolved?"autoruns-resolve-toggle resolved":"autoruns-resolve-toggle",onClick:f=>{f.preventDefault(),f.stopPropagation(),go(c,!c.resolved)},disabled:Bn===c.annotation_id,title:c.resolved?"Отметить кейс как невыполненный":"Отметить кейс как выполненный","aria-label":c.resolved?"Отметить кейс как невыполненный":"Отметить кейс как выполненный",children:o.jsx(fd,{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:["выполнено",": ",yn(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)})]}),De?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:De.technical_context.trace_id??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"reply_type:"}),o.jsx("strong",{children:De.technical_context.reply_type??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"domain:"}),o.jsx("strong",{children:De.technical_context.domain??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"query_class:"}),o.jsx("strong",{children:De.technical_context.query_class??"нет данных"})]})]}),o.jsx("h4",{children:"JSON разбор"}),o.jsx(xd,{value:{annotation_id:De.annotation_id,run_id:De.run_id,case_id:De.case_id,message_index:De.message_index,rating:De.rating,comment:De.comment,manual_case_decision:De.manual_case_decision,annotation_author:De.annotation_author,resolved:De.resolved,resolved_at:De.resolved_at,resolved_by:De.resolved_by,context:De.context,technical_context:De.technical_context,case_summary:De.case_summary?{case_id:De.case_summary.case_id,domain:De.case_summary.domain,query_class:De.case_summary.query_class,checks:De.case_summary.checks,metric_subscores:De.case_summary.metric_subscores}:null}})]}):null]}):null]}),_t.open?o.jsx("div",{className:"autoruns-comment-modal-backdrop",onClick:a=>{a.target===a.currentTarget&&Nr()},children:o.jsxs("div",{className:"autoruns-comment-modal",children:[o.jsx("h3",{children:"Сохранить ручную сессию"}),o.jsx("p",{className:"muted",children:"Технический чат будет сохранен в автопрогоны как пользовательская multi-turn сессия."}),o.jsxs("label",{children:["Название",o.jsx("input",{value:_t.title,onChange:a=>St(c=>({...c,title:a.target.value})),placeholder:"Например: НДС и склад на март 2020",disabled:_t.saving})]}),_t.error?o.jsx("p",{className:"error-text",children:_t.error}):null,o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",onClick:()=>{io()},disabled:_t.saving,children:_t.saving?"Сохраняю...":"Сохранить"}),o.jsx("button",{type:"button",className:"tab",onClick:()=>Nr(),disabled:_t.saving,children:"Отмена"})]})]})}):null,Ze.open?o.jsx("div",{className:"autoruns-comment-modal-backdrop",onClick:a=>{a.target===a.currentTarget&&or()},children:o.jsxs("div",{className:"autoruns-comment-modal",children:[o.jsx("h3",{children:"Удалить вопрос"}),o.jsx("p",{className:"muted",children:"Действительно удалить вопрос из сохраненной пользовательской сессии?"}),o.jsx("p",{className:"autoruns-comment-quote",children:Ze.questionText}),Ze.error?o.jsx("p",{className:"error-text",children:Ze.error}):null,o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",onClick:()=>{pa()},disabled:Ze.saving,children:Ze.saving?"Удаляю...":"Да"}),o.jsx("button",{type:"button",className:"tab",onClick:()=>or(),disabled:Ze.saving,children:"Нет"})]})]})}):null,Mt.open?o.jsx("div",{className:"autoruns-comment-modal-backdrop",onClick:a=>{a.target===a.currentTarget&&os()},children:o.jsxs("div",{className:"autoruns-comment-modal",children:[o.jsx("h3",{children:"Удалить сохраненный набор"}),o.jsx("p",{className:"muted",children:"Будет удалена карточка истории и связанный файл кейс-сета на бэке."}),o.jsx("p",{className:"autoruns-comment-quote",children:Mt.title}),Mt.error?o.jsx("p",{className:"error-text",children:Mt.error}):null,o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",onClick:()=>{rl()},disabled:Mt.saving,children:Mt.saving?"Удаляю...":"Да"}),o.jsx("button",{type:"button",className:"tab",onClick:()=>os(),disabled:Mt.saving,children:"Нет"})]})]})}):null,je.open?o.jsx("div",{className:"autoruns-comment-modal-backdrop",onClick:a=>{a.target===a.currentTarget&&It()},children:o.jsxs("div",{className:"autoruns-comment-modal",children:[o.jsx("h3",{children:"Комментарий к ответу ассистента"}),o.jsx("p",{className:"muted",children:"Комментарий будет добавлен в общий список комментариев справа с меткой `assistant_live`."}),Ho?o.jsxs("details",{className:"autoruns-prompt-details",open:!0,children:[o.jsx("summary",{children:"Вопрос пользователя"}),o.jsx("p",{className:"autoruns-comment-quote",children:Ho.text})]}):null,rs?o.jsxs("details",{className:"autoruns-prompt-details",open:!0,children:[o.jsx("summary",{children:"Ответ ассистента"}),o.jsx("p",{className:"autoruns-comment-quote",children:rs.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:je.rating>=a?"autoruns-rating-dot active":"autoruns-rating-dot",onClick:()=>nn(c=>({...c,rating:a})),disabled:je.saving,"aria-label":`Оценка ${a}`,children:je.rating>=a?"●":"○"},a))}),o.jsx("div",{className:"autoruns-form-grid",children:o.jsxs("label",{children:["Автор комментария",o.jsx("input",{value:je.annotationAuthor,onChange:a=>nn(c=>({...c,annotationAuthor:a.target.value})),placeholder:"manual_reviewer",disabled:je.saving})]})}),o.jsxs("label",{children:["Комментарий",o.jsx("textarea",{value:je.comment,onChange:a=>nn(c=>({...c,comment:a.target.value})),placeholder:"Что именно не так в ответе и что нужно исправить.",rows:4,disabled:je.saving})]}),je.error?o.jsx("p",{className:"error-text",children:je.error}):null,o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",onClick:()=>{fa()},disabled:je.saving,children:je.saving?"Сохраняю...":"Готово"}),o.jsx("button",{type:"button",className:"tab",onClick:()=>It(),disabled:je.saving,children:"Отмена"})]})]})}):null,ye.open?o.jsx("div",{className:"autoruns-comment-modal-backdrop",onClick:a=>{a.target===a.currentTarget&&Is()},children:o.jsxs("div",{className:"autoruns-comment-modal",children:[o.jsx("h3",{children:"Комментарий к ответу системы"}),o.jsx("p",{className:"muted",children:"Оцените ответ по 5-балльной шкале и добавьте комментарий по браку."}),jr?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:ns?.text??"Вопрос в диалоге не найден."})]}),o.jsxs("details",{className:"autoruns-prompt-details",open:!0,children:[o.jsx("summary",{children:"Ответ системы"}),o.jsx("p",{className:"autoruns-comment-quote",children:jr.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:()=>tn(c=>({...c,rating:a})),disabled:ye.saving,"aria-label":`Оценка ${a}`,children:ye.rating>=a?"●":"○"},a))}),o.jsxs("div",{className:"autoruns-form-grid",children:[o.jsxs("label",{children:["Решение по кейсу",o.jsx("select",{value:ye.manualCaseDecision,onChange:a=>tn(c=>({...c,manualCaseDecision:a.target.value})),disabled:ye.saving,children:(ln.length>0?ln:ke?.enum??[Xl]).map(a=>o.jsx("option",{value:a,children:String(ke?.labels?.[a]??a)},a))})]}),o.jsxs("label",{children:["Автор комментария",o.jsx("input",{value:ye.annotationAuthor,onChange:a=>tn(c=>({...c,annotationAuthor:a.target.value})),placeholder:"manual_reviewer",disabled:ye.saving})]})]}),o.jsxs("label",{children:["Комментарий",o.jsx("textarea",{value:ye.comment,onChange:a=>tn(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:()=>{ua()},disabled:ye.saving,children:ye.saving?"Сохраняю...":"Готово"}),o.jsx("button",{type:"button",className:"tab",onClick:()=>Is(),disabled:ye.saving,children:"Отмена"})]})]})}):null]})}const hp={llmProvider:"openai",apiKey:"",model:"gpt-4o-mini",baseUrl:"https://api.openai.com/v1",temperature:0,maxOutputTokens:700},hd={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."},gp={userQuestion:"",batchQuestionsRaw:"",periodHint:"",businessContext:"",expectedRoute:""},Bi={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}},gd="ndc_normalizer_session_config_v1",vd="ndc_autoruns_layout_config_v1",vp="ndc-autoruns-save",yp="autoruns",Qi="normalizer_v2_0_2",xp="address_query_runtime_v1",_p=["normalized","fragments","scope","flags","route","raw","validation","logs"];function Sp(i){return`[${new Date().toLocaleTimeString("ru-RU")}] ${i}`}function wp(i,g){if(!g)return"Previous preset is not selected.";const R=["systemPrompt","developerPrompt","domainPrompt","schemaNotes","fewShotExamples"].filter(E=>i[E]!==g[E]).map(E=>`${E}: ${Math.abs(i[E].length-g[E].length)} chars delta`);return R.length===0?"No changes against previous preset.":`Changed fields: ${R.length}. ${R.join(" | ")}`}function kp(){const[i,g]=h.useState(hp),[p,R]=h.useState(hd),[E,L]=h.useState(gp),[B,ie]=h.useState(null),[ne,z]=h.useState([]),[X,Z]=h.useState([]),[te,Re]=h.useState("normalized"),[de,fe]=h.useState(!1),[he,He]=h.useState(!1),[Ye,Ve]=h.useState([]),[Me,$]=h.useState(""),[ae,Ne]=h.useState([]),[be,yt]=h.useState(""),[dt,Kt]=h.useState("NDC custom preset"),[ft,b]=h.useState(null),[Ie,Ke]=h.useState(""),[re,ee]=h.useState(!1),[M,F]=h.useState([]),[I,y]=h.useState(""),[k,se]=h.useState([]),[ue,ce]=h.useState(!1),[ge,_e]=h.useState(null),[Se,ke]=h.useState(""),[pt,ln]=h.useState(yp),[vr,In]=h.useState(!0),[xn,Te]=h.useState(!0),[Jt,lt]=h.useState(!0),[xt,Pt]=h.useState(!0),[_n,W]=h.useState(!0),[mt,yr]=h.useState(!0),[Ln,Ge]=h.useState(!0),[Tt,Ft]=h.useState(!0),[Dn,me]=h.useState(!0),[Rt,Xe]=h.useState(!0),[_s,tt]=h.useState(!0),[er,xr]=h.useState(!0),[an,On]=h.useState(!0),Ut=h.useRef(!1),Sn=h.useRef(!1),Yt=h.useRef(!1);h.useEffect(()=>{const C=document.documentElement,{colors:Y}=Bi;C.style.setProperty("--rgb-background",Y.backgroundRgb),C.style.setProperty("--rgb-surface-main",Y.mainSurfaceRgb),C.style.setProperty("--rgb-surface-horizontal",Y.horizontalSurfaceRgb),C.style.setProperty("--rgb-surface-focus",Y.focusSurfaceRgb),C.style.setProperty("--rgb-assistant-chip",Y.assistantChipRgb),C.style.setProperty("--rgb-assistant-chip-hover",Y.assistantChipHoverRgb),C.style.setProperty("--rgb-assistant-chip-selected",Y.assistantChipSelectedRgb),C.style.setProperty("--rgb-assistant-chip-selected-text",Y.assistantChipSelectedTextRgb),C.style.setProperty("--rgb-active",Y.activeRgb),C.style.setProperty("--rgb-active-text",Y.activeTextRgb),C.style.setProperty("--rgb-text-main",Y.textMainRgb),C.style.setProperty("--rgb-text-muted",Y.textMutedRgb),C.style.setProperty("--rgb-danger",Y.dangerRgb),C.style.setProperty("--rgb-scrollbar-track",Y.scrollbarTrackRgb),C.style.setProperty("--rgb-scrollbar-thumb",Y.scrollbarThumbRgb),C.style.setProperty("--rgb-scrollbar-thumb-hover",Y.scrollbarThumbHoverRgb),C.style.setProperty("--mode-column-width",`${Bi.layout.modeColumnWidthPx}px`),C.style.setProperty("--mode-toggle-width",`${Bi.layout.modeToggleWidthPx}px`)},[]);const q=C=>{Z(Y=>[Sp(C),...Y].slice(0,300))};h.useEffect(()=>{(async()=>{const H=localStorage.getItem(gd);if(H)try{const Ee=JSON.parse(H);g(at=>({...at,llmProvider:Ee.llmProvider==="local"?"local":"openai",model:Ee.model??at.model,baseUrl:Ee.baseUrl??at.baseUrl,temperature:Ee.temperature??at.temperature,maxOutputTokens:Ee.maxOutputTokens??at.maxOutputTokens}))}catch{}try{const Ee=await Pe.loadSharedConnectionConfig();Ee.connection&&Ee.connection.llmProvider==="local"&&(g(at=>({...at,llmProvider:"local",model:Ee.connection?.model??at.model,baseUrl:Ee.connection?.baseUrl??at.baseUrl,temperature:Ee.connection?.temperature??at.temperature,maxOutputTokens:Ee.connection?.maxOutputTokens??at.maxOutputTokens})),q(`Shared local LLM config loaded: ${Ee.connection.model}`))}catch(Ee){q(`Shared local config load error: ${Ee instanceof Error?Ee.message:String(Ee)}`)}finally{Yt.current=!0}})();const Y=localStorage.getItem(vd);if(Y)try{const H=JSON.parse(Y);(H.uiMode==="assistant"||H.uiMode==="autoruns"||H.uiMode==="decomposition")&&ln("autoruns"),H.activeTab&&_p.includes(H.activeTab)&&Re(H.activeTab),typeof H.showAutorunsSettingsMode=="boolean"&&In(H.showAutorunsSettingsMode),typeof H.showAutorunsAutoRunsMode=="boolean"&&Te(H.showAutorunsAutoRunsMode),typeof H.showAutorunsAssistantMode=="boolean"&<(H.showAutorunsAssistantMode),typeof H.showAutorunsDecompositionMode=="boolean"&&Pt(H.showAutorunsDecompositionMode),typeof H.showAutorunsProgressMode=="boolean"&&W(H.showAutorunsProgressMode),typeof H.showAutorunsCommentsMode=="boolean"&&yr(H.showAutorunsCommentsMode),typeof H.showDecompositionConnectionMode=="boolean"&&Ge(H.showDecompositionConnectionMode),typeof H.showDecompositionPromptMode=="boolean"&&Ft(H.showDecompositionPromptMode),typeof H.showDecompositionQueryMode=="boolean"&&me(H.showDecompositionQueryMode),typeof H.showDecompositionOutputMode=="boolean"&&Xe(H.showDecompositionOutputMode),typeof H.showDecompositionMetricsMode=="boolean"&&tt(H.showDecompositionMetricsMode),typeof H.showDecompositionHistoryMode=="boolean"&&xr(H.showDecompositionHistoryMode),typeof H.showDecompositionRuntimeMode=="boolean"&&On(H.showDecompositionRuntimeMode),H.prompts&&(R(Ee=>({...Ee,...H.prompts})),Sn.current=!0)}catch{}At(),tr(),Hr()},[]),h.useEffect(()=>{if(!Yt.current||i.llmProvider!=="local")return;const C=window.setTimeout(()=>{Pe.saveSharedConnectionConfig(i).catch(Y=>q(`Shared local config sync error: ${Y instanceof Error?Y.message:String(Y)}`))},250);return()=>window.clearTimeout(C)},[i.baseUrl,i.llmProvider,i.maxOutputTokens,i.model,i.temperature]);async function At(){try{const C=await Pe.loadHistory();z(C.items??[])}catch(C){q(`History load error: ${C instanceof Error?C.message:String(C)}`)}}async function tr(){try{const Y=(await Pe.loadPresets()).presets??[];if(Ne(Y),Sn.current){Ut.current=!0;return}if(Ut.current)return;const H=Y.find(Ee=>Ee.prompt_version===Qi)??Y.find(Ee=>Ee.id==="default-normalizer-v2_0_2");if(!H){Ut.current=!0,q(`Preset autoload skipped: ${Qi} not found.`);return}yt(H.id),b(p),R({systemPrompt:H.systemPrompt,developerPrompt:H.developerPrompt,domainPrompt:H.domainPrompt,schemaNotes:H.schemaNotes??"",fewShotExamples:H.fewShotExamples??""}),Ut.current=!0,q(`Preset autoloaded: ${H.name} (${H.prompt_version}).`)}catch(C){q(`Presets load error: ${C instanceof Error?C.message:String(C)}`)}}async function Hr(){try{const C=await Pe.listRuns();F(C.items??[])}catch(C){q(`Runs load error: ${C instanceof Error?C.message:String(C)}`)}}function Vr(){if(localStorage.setItem(gd,JSON.stringify({model:i.model,llmProvider:i.llmProvider,baseUrl:i.baseUrl,temperature:i.temperature,maxOutputTokens:i.maxOutputTokens})),i.llmProvider==="local"){Pe.saveSharedConnectionConfig(i).then(()=>{q("Local config saved and synced to shared agent config (without API key).")}).catch(C=>{q(`Local config saved, but shared sync failed: ${C instanceof Error?C.message:String(C)}`)});return}q("Local config saved (without API key).")}function Gr(){localStorage.setItem(vd,JSON.stringify({uiMode:pt,activeTab:te,showAutorunsSettingsMode:vr,showAutorunsAutoRunsMode:xn,showAutorunsAssistantMode:Jt,showAutorunsDecompositionMode:xt,showAutorunsProgressMode:_n,showAutorunsCommentsMode:mt,showDecompositionConnectionMode:Ln,showDecompositionPromptMode:Tt,showDecompositionQueryMode:Dn,showDecompositionOutputMode:Rt,showDecompositionMetricsMode:_s,showDecompositionHistoryMode:er,showDecompositionRuntimeMode:an,prompts:p})),window.dispatchEvent(new CustomEvent(vp)),q("UI layout and prompts saved.")}async function zn(){fe(!0),ke("");try{const C=await Pe.testConnection(i);C.provider==="local"?C.model_found===!0?($(`LOCAL OK - ${C.model}`),q(`Local model is available: ${C.model} (catalog size=${C.models_count??"n/a"}).`)):C.model_found===!1?($(`LOCAL OK, model not loaded - ${C.model}`),q(`Local server is reachable, but model '${C.model}' is not in loaded catalog. Use 'Load model list' and select one of loaded models.`)):($(`LOCAL OK (model list unavailable) - ${C.model}`),q("Local server is reachable, but model catalog could not be verified.")):($(`OPENAI OK - ${C.model}`),q(`OpenAI connection ok: ${C.model}`))}catch(C){const Y=C instanceof Error?C.message:String(C);$("Connection error"),ke(`Test connection: ${Y}`),q(`Test connection error: ${Y}`)}finally{fe(!1)}}async function bt(){He(!0);try{const Y=(await Pe.listModels(i)).models??[];Ve(Y),Y.length>0&&g(H=>H.model&&Y.includes(H.model)?H:{...H,model:Y[0]}),q(`Model catalog loaded (${i.llmProvider}): ${Y.length} items.`)}catch(C){const Y=C instanceof Error?C.message:String(C);q(`Load model list error: ${Y}`)}finally{He(!1)}}h.useEffect(()=>{Ve([])},[i.llmProvider,i.baseUrl]);function $n(){const C=ae.find(Y=>Y.id===be);if(!C){q("Preset is not selected.");return}b(p),R({systemPrompt:C.systemPrompt,developerPrompt:C.developerPrompt,domainPrompt:C.domainPrompt,schemaNotes:C.schemaNotes??"",fewShotExamples:C.fewShotExamples??""}),q(`Preset loaded: ${C.name}`)}async function Fn(){try{await Pe.savePreset({name:dt||"NDC preset",prompt_version:"normalizer_v2_0_2",systemPrompt:p.systemPrompt,developerPrompt:p.developerPrompt,domainPrompt:p.domainPrompt,schemaNotes:p.schemaNotes,fewShotExamples:p.fewShotExamples}),q("Preset saved."),await tr()}catch(C){q(`Preset save error: ${C instanceof Error?C.message:String(C)}`)}}function Un(){R(hd),q("Prompt panel reset to defaults.")}function bn(){const C=wp(p,ft);Ke(C),q(C)}return h.useEffect(()=>{if(!I){se([]);return}Pe.runTrace(I).then(C=>se(C.items)).catch(C=>q(`Run trace error: ${C instanceof Error?C.message:String(C)}`))},[I]),o.jsxs("main",{className:"app-root 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:"tab active",onClick:()=>ln("autoruns"),children:"Управление ассистентом"}),o.jsx("button",{type:"button",className:"tab",onClick:Gr,children:"Сохранить"})]}),o.jsxs("div",{className:"mode-switch-row mode-switch-row-right",children:[o.jsx("button",{type:"button",className:vr?"tab active":"tab",onClick:()=>In(C=>!C),children:"Настройки"}),o.jsx("button",{type:"button",className:xn?"tab active":"tab",onClick:()=>Te(C=>!C),children:"Автопрогоны"}),o.jsx("button",{type:"button",className:Jt?"tab active":"tab",onClick:()=>lt(C=>!C),children:"Режим ассистента"}),o.jsx("button",{type:"button",className:_n?"tab active":"tab",onClick:()=>W(C=>!C),children:"Прогресс/регресс"}),o.jsx("button",{type:"button",className:mt?"tab active":"tab",onClick:()=>yr(C=>!C),children:"Комментарии"})]})]}),o.jsx("div",{className:"layout-grid layout-grid-autoruns",children:o.jsx(mp,{connection:i,modelOptions:Ye,modelsBusy:he,connectionStatus:Me,connectionBusy:de,onConnectionChange:g,onReloadModels:bt,onSaveLocalConfig:Vr,onTestConnection:zn,prompts:p,onPromptsChange:R,promptPresets:ae,selectedPresetId:be,onSelectPreset:yt,onLoadPreset:$n,onSavePreset:Fn,onResetDefaults:Un,onDiffPrevious:bn,presetName:dt,onPresetNameChange:Kt,diffSummary:Ie,assistantPromptVersion:xp,decompositionPromptVersion:Qi,showSettingsMode:vr,showAutoRunsMode:xn,showAssistantMode:Jt,showProgressMode:_n,showCommentsMode:mt,onLog:q})})]})}Mf.createRoot(document.getElementById("root")).render(o.jsx(Cf.StrictMode,{children:o.jsx(kp,{})})); diff --git a/llm_normalizer/frontend/dist/assets/index-9w9i5XPJ.js b/llm_normalizer/frontend/dist/assets/index-9w9i5XPJ.js new file mode 100644 index 0000000..8cb35a9 --- /dev/null +++ b/llm_normalizer/frontend/dist/assets/index-9w9i5XPJ.js @@ -0,0 +1,25 @@ +(function(){const m=document.createElement("link").relList;if(m&&m.supports&&m.supports("modulepreload"))return;for(const C of document.querySelectorAll('link[rel="modulepreload"]'))S(C);new MutationObserver(C=>{for(const M of C)if(M.type==="childList")for(const z of M.addedNodes)z.tagName==="LINK"&&z.rel==="modulepreload"&&S(z)}).observe(document,{childList:!0,subtree:!0});function p(C){const M={};return C.integrity&&(M.integrity=C.integrity),C.referrerPolicy&&(M.referrerPolicy=C.referrerPolicy),C.crossOrigin==="use-credentials"?M.credentials="include":C.crossOrigin==="anonymous"?M.credentials="omit":M.credentials="same-origin",M}function S(C){if(C.ep)return;C.ep=!0;const M=p(C);fetch(C.href,M)}})();function kd(a){return a&&a.__esModule&&Object.prototype.hasOwnProperty.call(a,"default")?a.default:a}var Fi={exports:{}},Bo={},bi={exports:{}},ge={};var Xc;function Cf(){if(Xc)return ge;Xc=1;var a=Symbol.for("react.element"),m=Symbol.for("react.portal"),p=Symbol.for("react.fragment"),S=Symbol.for("react.strict_mode"),C=Symbol.for("react.profiler"),M=Symbol.for("react.provider"),z=Symbol.for("react.context"),ae=Symbol.for("react.forward_ref"),ne=Symbol.for("react.suspense"),F=Symbol.for("react.memo"),X=Symbol.for("react.lazy"),Z=Symbol.iterator;function te(y){return y===null||typeof y!="object"?null:(y=Z&&y[Z]||y["@@iterator"],typeof y=="function"?y:null)}var Me={isMounted:function(){return!1},enqueueForceUpdate:function(){},enqueueReplaceState:function(){},enqueueSetState:function(){}},pe=Object.assign,me={};function ye(y,k,se){this.props=y,this.context=k,this.refs=me,this.updater=se||Me}ye.prototype.isReactComponent={},ye.prototype.setState=function(y,k){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,k,"setState")},ye.prototype.forceUpdate=function(y){this.updater.enqueueForceUpdate(this,y,"forceUpdate")};function Qe(){}Qe.prototype=ye.prototype;function Xe(y,k,se){this.props=y,this.context=k,this.refs=me,this.updater=se||Me}var He=Xe.prototype=new Qe;He.constructor=Xe,pe(He,ye.prototype),He.isPureReactComponent=!0;var Ae=Array.isArray,b=Object.prototype.hasOwnProperty,le={current:null},Ee={key:!0,ref:!0,__self:!0,__source:!0};function be(y,k,se){var ue,de={},xe=null,we=null;if(k!=null)for(ue in k.ref!==void 0&&(we=k.ref),k.key!==void 0&&(xe=""+k.key),k)b.call(k,ue)&&!Ee.hasOwnProperty(ue)&&(de[ue]=k[ue]);var je=arguments.length-2;if(je===1)de.children=se;else if(1>>1,k=I[y];if(0>>1;yC(de,L))xeC(we,de)?(I[y]=we,I[xe]=L,y=xe):(I[y]=de,I[ue]=L,y=ue);else if(xeC(we,L))I[y]=we,I[xe]=L,y=xe;else break e}}return O}function C(I,O){var L=I.sortIndex-O.sortIndex;return L!==0?L:I.id-O.id}if(typeof performance=="object"&&typeof performance.now=="function"){var M=performance;a.unstable_now=function(){return M.now()}}else{var z=Date,ae=z.now();a.unstable_now=function(){return z.now()-ae}}var ne=[],F=[],X=1,Z=null,te=3,Me=!1,pe=!1,me=!1,ye=typeof setTimeout=="function"?setTimeout:null,Qe=typeof clearTimeout=="function"?clearTimeout:null,Xe=typeof setImmediate<"u"?setImmediate:null;typeof navigator<"u"&&navigator.scheduling!==void 0&&navigator.scheduling.isInputPending!==void 0&&navigator.scheduling.isInputPending.bind(navigator.scheduling);function He(I){for(var O=p(F);O!==null;){if(O.callback===null)S(F);else if(O.startTime<=I)S(F),O.sortIndex=O.expirationTime,m(ne,O);else break;O=p(F)}}function Ae(I){if(me=!1,He(I),!pe)if(p(ne)!==null)pe=!0,re(b);else{var O=p(F);O!==null&&ee(Ae,O.startTime-I)}}function b(I,O){pe=!1,me&&(me=!1,Qe(be),be=-1),Me=!0;var L=te;try{for(He(O),Z=p(ne);Z!==null&&(!(Z.expirationTime>O)||I&&!Kt());){var y=Z.callback;if(typeof y=="function"){Z.callback=null,te=Z.priorityLevel;var k=y(Z.expirationTime<=O);O=a.unstable_now(),typeof k=="function"?Z.callback=k:Z===p(ne)&&S(ne),He(O)}else S(ne);Z=p(ne)}if(Z!==null)var se=!0;else{var ue=p(F);ue!==null&&ee(Ae,ue.startTime-O),se=!1}return se}finally{Z=null,te=L,Me=!1}}var le=!1,Ee=null,be=-1,St=5,mt=-1;function Kt(){return!(a.unstable_now()-mtI||125y?(I.sortIndex=L,m(F,I),p(ne)===null&&I===p(F)&&(me?(Qe(be),be=-1):me=!0,ee(Ae,L-y))):(I.sortIndex=k,m(ne,I),pe||Me||(pe=!0,re(b))),I},a.unstable_shouldYield=Kt,a.unstable_wrapCallback=function(I){var O=te;return function(){var L=te;te=O;try{return I.apply(this,arguments)}finally{te=L}}}})(Qi)),Qi}var rd;function Rf(){return rd||(rd=1,Bi.exports=Tf()),Bi.exports}var sd;function Mf(){if(sd)return qt;sd=1;var a=Zi(),m=Rf();function p(e){for(var t="https://reactjs.org/docs/error-decoder.html?invariant="+e,n=1;n"u"||typeof window.document>"u"||typeof window.document.createElement>"u"),ne=Object.prototype.hasOwnProperty,F=/^[: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]*$/,X={},Z={};function te(e){return ne.call(Z,e)?!0:ne.call(X,e)?!1:F.test(e)?Z[e]=!0:(X[e]=!0,!1)}function Me(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 pe(e,t,n,r){if(t===null||typeof t>"u"||Me(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 me(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 ye={};"children dangerouslySetInnerHTML defaultValue defaultChecked innerHTML suppressContentEditableWarning suppressHydrationWarning style".split(" ").forEach(function(e){ye[e]=new me(e,0,!1,e,null,!1,!1)}),[["acceptCharset","accept-charset"],["className","class"],["htmlFor","for"],["httpEquiv","http-equiv"]].forEach(function(e){var t=e[0];ye[t]=new me(t,1,!1,e[1],null,!1,!1)}),["contentEditable","draggable","spellCheck","value"].forEach(function(e){ye[e]=new me(e,2,!1,e.toLowerCase(),null,!1,!1)}),["autoReverse","externalResourcesRequired","focusable","preserveAlpha"].forEach(function(e){ye[e]=new me(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){ye[e]=new me(e,3,!1,e.toLowerCase(),null,!1,!1)}),["checked","multiple","muted","selected"].forEach(function(e){ye[e]=new me(e,3,!0,e,null,!1,!1)}),["capture","download"].forEach(function(e){ye[e]=new me(e,4,!1,e,null,!1,!1)}),["cols","rows","size","span"].forEach(function(e){ye[e]=new me(e,6,!1,e,null,!1,!1)}),["rowSpan","start"].forEach(function(e){ye[e]=new me(e,5,!1,e.toLowerCase(),null,!1,!1)});var Qe=/[\-:]([a-z])/g;function Xe(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,Xe);ye[t]=new me(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,Xe);ye[t]=new me(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,Xe);ye[t]=new me(t,1,!1,e,"http://www.w3.org/XML/1998/namespace",!1,!1)}),["tabIndex","crossOrigin"].forEach(function(e){ye[e]=new me(e,1,!1,e.toLowerCase(),null,!1,!1)}),ye.xlinkHref=new me("xlinkHref",1,!1,"xlink:href","http://www.w3.org/1999/xlink",!0,!1),["src","href","action","formAction"].forEach(function(e){ye[e]=new me(e,1,!1,e.toLowerCase(),null,!0,!0)});function He(e,t,n,r){var s=ye.hasOwnProperty(t)?ye[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{se=!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 xe(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 Ee:return"Fragment";case le:return"Portal";case St:return"Profiler";case be:return"StrictMode";case B:return"Suspense";case Ie:return"SuspenseList"}if(typeof e=="object")switch(e.$$typeof){case Kt:return(e.displayName||"Context")+".Consumer";case mt:return(e._context.displayName||"Context")+".Provider";case ht:var t=e.render;return e=e.displayName,e||(e=t.displayName||t.name||"",e=e!==""?"ForwardRef("+e+")":"ForwardRef"),e;case Je:return t=e.displayName||null,t!==null?t:xe(e.type)||"Memo";case re:t=e._payload,e=e._init;try{return xe(e(t))}catch{}}return null}function we(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 xe(t);case 8:return t===be?"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 je(e){switch(typeof e){case"boolean":case"number":case"string":case"undefined":return e;case"object":return e;default:return""}}function Ce(e){var t=e.type;return(e=e.nodeName)&&e.toLowerCase()==="input"&&(t==="checkbox"||t==="radio")}function gt(e){var t=Ce(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 sn(e){e._valueTracker||(e._valueTracker=gt(e))}function yr(e){if(!e)return!1;var t=e._valueTracker;if(!t)return!0;var n=t.getValue(),r="";return e&&(r=Ce(e)?e.checked?"true":"false":e.value),e=r,e!==n?(t.setValue(e),!0):!1}function Rn(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 vn(e,t){var n=t.checked;return L({},t,{defaultChecked:void 0,defaultValue:void 0,value:void 0,checked:n??e._wrapperState.initialChecked})}function Se(e,t){var n=t.defaultValue==null?"":t.defaultValue,r=t.checked!=null?t.checked:t.defaultChecked;n=je(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 Jt(e,t){t=t.checked,t!=null&&He(e,"checked",t,!1)}function Ve(e,t){Jt(e,t);var n=je(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")?Tt(e,t.type,n):t.hasOwnProperty("defaultValue")&&Tt(e,t.type,je(t.defaultValue)),t.checked==null&&t.defaultChecked!=null&&(e.defaultChecked=!!t.defaultChecked)}function wt(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 Tt(e,t,n){(t!=="number"||Rn(e.ownerDocument)!==e)&&(n==null?e.defaultValue=""+e._wrapperState.initialValue:e.defaultValue!==""+n&&(e.defaultValue=""+n))}var yn=Array.isArray;function W(e,t,n,r){if(e=e.options,t){t={};for(var s=0;s"+t.valueOf().toString()+"",t=An.firstChild;e.firstChild;)e.removeChild(e.firstChild);for(;t.firstChild;)e.appendChild(t.firstChild)}});function Mt(e,t){if(t){var n=e.firstChild;if(n&&n===e.lastChild&&n.nodeType===3){n.nodeValue=t;return}}e.textContent=t}var Ze={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},ys=["Webkit","ms","Moz","O"];Object.keys(Ze).forEach(function(e){ys.forEach(function(t){t=t+e.charAt(0).toUpperCase()+e.substring(1),Ze[t]=Ze[e]})});function st(e,t,n){return t==null||typeof t=="boolean"||t===""?"":n||typeof t!="number"||t===0||Ze.hasOwnProperty(e)&&Ze[e]?(""+t).trim():t+"px"}function er(e,t){e=e.style;for(var n in t)if(t.hasOwnProperty(n)){var r=n.indexOf("--")===0,s=st(n,t[n],r);n==="float"&&(n="cssFloat"),r?e.setProperty(n,s):e[n]=s}}var _r=L({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 on(e,t){if(t){if(_r[e]&&(t.children!=null||t.dangerouslySetInnerHTML!=null))throw Error(p(137,e));if(t.dangerouslySetInnerHTML!=null){if(t.children!=null)throw Error(p(60));if(typeof t.dangerouslySetInnerHTML!="object"||!("__html"in t.dangerouslySetInnerHTML))throw Error(p(61))}if(t.style!=null&&typeof t.style!="object")throw Error(p(62))}}function In(e,t){if(e.indexOf("-")===-1)return typeof t.is=="string";switch(e){case"annotation-xml":case"color-profile":case"font-face":case"font-face-src":case"font-face-uri":case"font-face-format":case"font-face-name":case"missing-glyph":return!1;default:return!0}}var Ft=null;function xn(e){return e=e.target||e.srcElement||window,e.correspondingUseElement&&(e=e.correspondingUseElement),e.nodeType===3?e.parentNode:e}var Yt=null,q=null,At=null;function tr(e){if(e=Eo(e)){if(typeof Yt!="function")throw Error(p(280));var t=e.stateNode;t&&(t=pl(t),Yt(e.stateNode,e.type,t))}}function Kr(e){q?At?At.push(e):At=[e]:q=e}function Jr(){if(q){var e=q,t=At;if(At=q=null,tr(e),t)for(e=0;e>>=0,e===0?32:31-(Ko(e)/Jo|0)|0}var Un=64,_s=4194304;function Zt(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 Ss(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=Zt(d):(l&=u,l!==0&&(r=Zt(l)))}else u=n&~s,u!==0?r=Zt(u):l!==0&&(r=Zt(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 Cr(e,t,n){e.pendingLanes|=t,t!==536870912&&(e.suspendedLanes=0,e.pingedLanes=0),e=e.eventTimes,t=31-Xt(t),e[t]=n}function Qn(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=Mr),po=" ",mo=!1;function ho(e,t){switch(e){case"keyup":return Kn.indexOf(t.keyCode)!==-1;case"keydown":return t.keyCode!==229;case"keypress":case"mousedown":case"focusout":return!0;default:return!1}}function go(e){return e=e.detail,typeof e=="object"&&"data"in e?e.data:null}var ir=!1;function ya(e,t){switch(e){case"compositionend":return go(t);case"keypress":return t.which!==32?null:(mo=!0,po);case"textInput":return e=t.data,e===po&&mo?null:e;default:return null}}function ll(e,t){if(ir)return e==="compositionend"||!Rr&&ho(e,t)?(e=V(),Er=Es=ut=null,ir=!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=tu(n)}}function ru(e,t){return e&&t?e===t?!0:e&&e.nodeType===3?!1:t&&t.nodeType===3?ru(e,t.parentNode):"contains"in e?e.contains(t):e.compareDocumentPosition?!!(e.compareDocumentPosition(t)&16):!1:!1}function su(){for(var e=window,t=Rn();t instanceof e.HTMLIFrameElement;){try{var n=typeof t.contentWindow.location.href=="string"}catch{n=!1}if(n)e=t.contentWindow;else break;t=Rn(e.document)}return t}function wa(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 Md(e){var t=su(),n=e.focusedElem,r=e.selectionRange;if(t!==n&&n&&n.ownerDocument&&ru(n.ownerDocument.documentElement,n)){if(r!==null&&wa(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=nu(n,l);var u=nu(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,Ms=null,ja=null,wo=null,ka=!1;function ou(e,t,n){var r=n.window===n?n.document:n.nodeType===9?n:n.ownerDocument;ka||Ms==null||Ms!==Rn(r)||(r=Ms,"selectionStart"in r&&wa(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}),wo&&So(wo,r)||(wo=r,r=cl(ja,"onSelect"),0Os||(e.current=Oa[Os],Oa[Os]=null,Os--)}function De(e,t){Os++,Oa[Os]=e.current,e.current=t}var zr={},kt=$r(zr),Qt=$r(!1),as=zr;function $s(e,t){var n=e.type.contextTypes;if(!n)return zr;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 Ht(e){return e=e.childContextTypes,e!=null}function ml(){ze(Qt),ze(kt)}function _u(e,t,n){if(kt.current!==zr)throw Error(p(168));De(kt,t),De(Qt,n)}function Su(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(p(108,we(e)||"Unknown",s));return L({},n,r)}function hl(e){return e=(e=e.stateNode)&&e.__reactInternalMemoizedMergedChildContext||zr,as=kt.current,De(kt,e),De(Qt,Qt.current),!0}function wu(e,t,n){var r=e.stateNode;if(!r)throw Error(p(169));n?(e=Su(e,t,as),r.__reactInternalMemoizedMergedChildContext=e,ze(Qt),ze(kt),De(kt,e)):ze(Qt),De(Qt,n)}var cr=null,gl=!1,$a=!1;function ju(e){cr===null?cr=[e]:cr.push(e)}function Qd(e){gl=!0,ju(e)}function Fr(){if(!$a&&cr!==null){$a=!0;var e=0,t=fe;try{var n=cr;for(fe=1;e>=u,s-=u,dr=1<<32-Xt(t)+s|n<oe?(pt=J,J=null):pt=J.sibling;var Ne=N(x,J,_[oe],R);if(Ne===null){J===null&&(J=pt);break}e&&J&&Ne.alternate===null&&t(x,J),v=l(Ne,v,oe),K===null?G=Ne:K.sibling=Ne,K=Ne,J=pt}if(oe===_.length)return n(x,J),Fe&&us(x,oe),G;if(J===null){for(;oe<_.length;oe++)J=T(x,_[oe],R),J!==null&&(v=l(J,v,oe),K===null?G=J:K.sibling=J,K=J);return Fe&&us(x,oe),G}for(J=r(x,J);oe<_.length;oe++)pt=D(J,x,oe,_[oe],R),pt!==null&&(e&&pt.alternate!==null&&J.delete(pt.key===null?oe:pt.key),v=l(pt,v,oe),K===null?G=pt:K.sibling=pt,K=pt);return e&&J.forEach(function(qr){return t(x,qr)}),Fe&&us(x,oe),G}function H(x,v,_,R){var G=O(_);if(typeof G!="function")throw Error(p(150));if(_=G.call(_),_==null)throw Error(p(151));for(var K=G=null,J=v,oe=v=0,pt=null,Ne=_.next();J!==null&&!Ne.done;oe++,Ne=_.next()){J.index>oe?(pt=J,J=null):pt=J.sibling;var qr=N(x,J,Ne.value,R);if(qr===null){J===null&&(J=pt);break}e&&J&&qr.alternate===null&&t(x,J),v=l(qr,v,oe),K===null?G=qr:K.sibling=qr,K=qr,J=pt}if(Ne.done)return n(x,J),Fe&&us(x,oe),G;if(J===null){for(;!Ne.done;oe++,Ne=_.next())Ne=T(x,Ne.value,R),Ne!==null&&(v=l(Ne,v,oe),K===null?G=Ne:K.sibling=Ne,K=Ne);return Fe&&us(x,oe),G}for(J=r(x,J);!Ne.done;oe++,Ne=_.next())Ne=D(J,x,oe,Ne.value,R),Ne!==null&&(e&&Ne.alternate!==null&&J.delete(Ne.key===null?oe:Ne.key),v=l(Ne,v,oe),K===null?G=Ne:K.sibling=Ne,K=Ne);return e&&J.forEach(function(kf){return t(x,kf)}),Fe&&us(x,oe),G}function Ye(x,v,_,R){if(typeof _=="object"&&_!==null&&_.type===Ee&&_.key===null&&(_=_.props.children),typeof _=="object"&&_!==null){switch(_.$$typeof){case b:e:{for(var G=_.key,K=v;K!==null;){if(K.key===G){if(G=_.type,G===Ee){if(K.tag===7){n(x,K.sibling),v=s(K,_.props.children),v.return=x,x=v;break e}}else if(K.elementType===G||typeof G=="object"&&G!==null&&G.$$typeof===re&&Tu(G)===K.type){n(x,K.sibling),v=s(K,_.props),v.ref=Po(x,K,_),v.return=x,x=v;break e}n(x,K);break}else t(x,K);K=K.sibling}_.type===Ee?(v=vs(_.props.children,x.mode,R,_.key),v.return=x,x=v):(R=Hl(_.type,_.key,_.props,null,x.mode,R),R.ref=Po(x,v,_),R.return=x,x=R)}return u(x);case le:e:{for(K=_.key;v!==null;){if(v.key===K)if(v.tag===4&&v.stateNode.containerInfo===_.containerInfo&&v.stateNode.implementation===_.implementation){n(x,v.sibling),v=s(v,_.children||[]),v.return=x,x=v;break e}else{n(x,v);break}else t(x,v);v=v.sibling}v=Li(_,x.mode,R),v.return=x,x=v}return u(x);case re:return K=_._init,Ye(x,v,K(_._payload),R)}if(yn(_))return U(x,v,_,R);if(O(_))return H(x,v,_,R);_l(x,_)}return typeof _=="string"&&_!==""||typeof _=="number"?(_=""+_,v!==null&&v.tag===6?(n(x,v.sibling),v=s(v,_),v.return=x,x=v):(n(x,v),v=Ii(_,x.mode,R),v.return=x,x=v),u(x)):n(x,v)}return Ye}var Us=Ru(!0),Mu=Ru(!1),Sl=$r(null),wl=null,Bs=null,Qa=null;function Ha(){Qa=Bs=wl=null}function Va(e){var t=Sl.current;ze(Sl),e._currentValue=t}function Ga(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 Qs(e,t){wl=e,Qa=Bs=null,e=e.dependencies,e!==null&&e.firstContext!==null&&((e.lanes&t)!==0&&(Vt=!0),e.firstContext=null)}function fn(e){var t=e._currentValue;if(Qa!==e)if(e={context:e,memoizedValue:t,next:null},Bs===null){if(wl===null)throw Error(p(308));Bs=e,wl.dependencies={lanes:0,firstContext:e}}else Bs=Bs.next=e;return t}var cs=null;function Wa(e){cs===null?cs=[e]:cs.push(e)}function Au(e,t,n,r){var s=t.interleaved;return s===null?(n.next=n,Wa(t)):(n.next=s.next,s.next=n),t.interleaved=n,pr(e,r)}function pr(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 br=!1;function qa(e){e.updateQueue={baseState:e.memoizedState,firstBaseUpdate:null,lastBaseUpdate:null,shared:{pending:null,interleaved:null,lanes:0},effects:null}}function Iu(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 mr(e,t){return{eventTime:e,lane:t,tag:0,payload:null,callback:null,next:null}}function Ur(e,t,n){var r=e.updateQueue;if(r===null)return null;if(r=r.shared,(ke&2)!==0){var s=r.pending;return s===null?t.next=t:(t.next=s.next,s.next=t),r.pending=t,pr(e,n)}return s=r.interleaved,s===null?(t.next=t,Wa(r)):(t.next=s.next,s.next=t),r.interleaved=t,pr(e,n)}function jl(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,es(e,n)}}function Lu(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 kl(e,t,n,r){var s=e.updateQueue;br=!1;var l=s.firstBaseUpdate,u=s.lastBaseUpdate,d=s.shared.pending;if(d!==null){s.shared.pending=null;var h=d,j=h.next;h.next=null,u===null?l=j:u.next=j,u=h;var P=e.alternate;P!==null&&(P=P.updateQueue,d=P.lastBaseUpdate,d!==u&&(d===null?P.firstBaseUpdate=j:d.next=j,P.lastBaseUpdate=h))}if(l!==null){var T=s.baseState;u=0,P=j=h=null,d=l;do{var N=d.lane,D=d.eventTime;if((r&N)===N){P!==null&&(P=P.next={eventTime:D,lane:0,tag:d.tag,payload:d.payload,callback:d.callback,next:null});e:{var U=e,H=d;switch(N=t,D=n,H.tag){case 1:if(U=H.payload,typeof U=="function"){T=U.call(D,T,N);break e}T=U;break e;case 3:U.flags=U.flags&-65537|128;case 0:if(U=H.payload,N=typeof U=="function"?U.call(D,T,N):U,N==null)break e;T=L({},T,N);break e;case 2:br=!0}}d.callback!==null&&d.lane!==0&&(e.flags|=64,N=s.effects,N===null?s.effects=[d]:N.push(d))}else D={eventTime:D,lane:N,tag:d.tag,payload:d.payload,callback:d.callback,next:null},P===null?(j=P=D,h=T):P=P.next=D,u|=N;if(d=d.next,d===null){if(d=s.shared.pending,d===null)break;N=d,d=N.next,N.next=null,s.lastBaseUpdate=N,s.shared.pending=null}}while(!0);if(P===null&&(h=T),s.baseState=h,s.firstBaseUpdate=j,s.lastBaseUpdate=P,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);ps|=u,e.lanes=u,e.memoizedState=T}}function Du(e,t,n){if(e=t.effects,t.effects=null,e!==null)for(t=0;tn?n:4,e(!0);var r=Za.transition;Za.transition={};try{e(!1),t()}finally{fe=n,Za.transition=r}}function ec(){return pn().memoizedState}function Wd(e,t,n){var r=Vr(e);if(n={lane:r,action:n,hasEagerState:!1,eagerState:null,next:null},tc(e))nc(t,n);else if(n=Au(e,t,n,r),n!==null){var s=$t();Tn(n,e,r,s),rc(n,t,r)}}function qd(e,t,n){var r=Vr(e),s={lane:r,action:n,hasEagerState:!1,eagerState:null,next:null};if(tc(e))nc(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,kn(d,u)){var h=t.interleaved;h===null?(s.next=s,Wa(t)):(s.next=h.next,h.next=s),t.interleaved=s;return}}catch{}n=Au(e,t,s,r),n!==null&&(s=$t(),Tn(n,e,r,s),rc(n,t,r))}}function tc(e){var t=e.alternate;return e===Be||t!==null&&t===Be}function nc(e,t){Ao=El=!0;var n=e.pending;n===null?t.next=t:(t.next=n.next,n.next=t),e.pending=t}function rc(e,t,n){if((n&4194240)!==0){var r=t.lanes;r&=e.pendingLanes,n|=r,t.lanes=n,es(e,n)}}var Rl={readContext:fn,useCallback:Ct,useContext:Ct,useEffect:Ct,useImperativeHandle:Ct,useInsertionEffect:Ct,useLayoutEffect:Ct,useMemo:Ct,useReducer:Ct,useRef:Ct,useState:Ct,useDebugValue:Ct,useDeferredValue:Ct,useTransition:Ct,useMutableSource:Ct,useSyncExternalStore:Ct,useId:Ct,unstable_isNewReconciler:!1},Kd={readContext:fn,useCallback:function(e,t){return Xn().memoizedState=[e,t===void 0?null:t],e},useContext:fn,useEffect:Gu,useImperativeHandle:function(e,t,n){return n=n!=null?n.concat([e]):null,Pl(4194308,4,Ku.bind(null,t,e),n)},useLayoutEffect:function(e,t){return Pl(4194308,4,e,t)},useInsertionEffect:function(e,t){return Pl(4,2,e,t)},useMemo:function(e,t){var n=Xn();return t=t===void 0?null:t,e=e(),n.memoizedState=[e,t],e},useReducer:function(e,t,n){var r=Xn();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=Wd.bind(null,Be,e),[r.memoizedState,e]},useRef:function(e){var t=Xn();return e={current:e},t.memoizedState=e},useState:Hu,useDebugValue:li,useDeferredValue:function(e){return Xn().memoizedState=e},useTransition:function(){var e=Hu(!1),t=e[0];return e=Gd.bind(null,e[1]),Xn().memoizedState=e,[t,e]},useMutableSource:function(){},useSyncExternalStore:function(e,t,n){var r=Be,s=Xn();if(Fe){if(n===void 0)throw Error(p(407));n=n()}else{if(n=t(),ft===null)throw Error(p(349));(fs&30)!==0||Fu(r,t,n)}s.memoizedState=n;var l={value:n,getSnapshot:t};return s.queue=l,Gu(Uu.bind(null,r,l,e),[e]),r.flags|=2048,Do(9,bu.bind(null,r,l,n,t),void 0,null),n},useId:function(){var e=Xn(),t=ft.identifierPrefix;if(Fe){var n=fr,r=dr;n=(r&~(1<<32-Xt(r)-1)).toString(32)+n,t=":"+t+"R"+n,n=Io++,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[Jn]=t,e[No]=r,jc(e,t,!1,!1),t.stateNode=e;e:{switch(u=In(n,r),n){case"dialog":$e("cancel",e),$e("close",e),s=r;break;case"iframe":case"object":case"embed":$e("load",e),s=r;break;case"video":case"audio":for(s=0;sqs&&(t.flags|=128,r=!0,Oo(l,!1),t.lanes=4194304)}else{if(!r)if(e=Cl(u),e!==null){if(t.flags|=128,r=!0,n=e.updateQueue,n!==null&&(t.updateQueue=n,t.flags|=4),Oo(l,!0),l.tail===null&&l.tailMode==="hidden"&&!u.alternate&&!Fe)return Nt(t),null}else 2*Oe()-l.renderingStartTime>qs&&n!==1073741824&&(t.flags|=128,r=!0,Oo(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=Oe(),t.sibling=null,n=Ue.current,De(Ue,r?n&1|2:n&1),t):(Nt(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?(rn&1073741824)!==0&&(Nt(t),t.subtreeFlags&6&&(t.flags|=8192)):Nt(t),null;case 24:return null;case 25:return null}throw Error(p(156,t.tag))}function rf(e,t){switch(Fa(t),t.tag){case 1:return Ht(t.type)&&ml(),e=t.flags,e&65536?(t.flags=e&-65537|128,t):null;case 3:return Hs(),ze(Qt),ze(kt),Xa(),e=t.flags,(e&65536)!==0&&(e&128)===0?(t.flags=e&-65537|128,t):null;case 5:return Ja(t),null;case 13:if(ze(Ue),e=t.memoizedState,e!==null&&e.dehydrated!==null){if(t.alternate===null)throw Error(p(340));bs()}return e=t.flags,e&65536?(t.flags=e&-65537|128,t):null;case 19:return ze(Ue),null;case 4:return Hs(),null;case 10:return Va(t.type._context),null;case 22:case 23:return Ri(),null;case 24:return null;default:return null}}var Ll=!1,Et=!1,sf=typeof WeakSet=="function"?WeakSet:Set,$=null;function Gs(e,t){var n=e.ref;if(n!==null)if(typeof n=="function")try{n(null)}catch(r){Ke(e,t,r)}else n.current=null}function yi(e,t,n){try{n()}catch(r){Ke(e,t,r)}}var Nc=!1;function of(e,t){if(Ra=an,e=su(),wa(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,j=0,P=0,T=e,N=null;t:for(;;){for(var D;T!==n||s!==0&&T.nodeType!==3||(d=u+s),T!==l||r!==0&&T.nodeType!==3||(h=u+r),T.nodeType===3&&(u+=T.nodeValue.length),(D=T.firstChild)!==null;)N=T,T=D;for(;;){if(T===e)break t;if(N===n&&++j===s&&(d=u),N===l&&++P===r&&(h=u),(D=T.nextSibling)!==null)break;T=N,N=T.parentNode}T=D}n=d===-1||h===-1?null:{start:d,end:h}}else n=null}n=n||{start:0,end:0}}else n=null;for(Ma={focusedElem:e,selectionRange:n},an=!1,$=t;$!==null;)if(t=$,e=t.child,(t.subtreeFlags&1028)!==0&&e!==null)e.return=t,$=e;else for(;$!==null;){t=$;try{var U=t.alternate;if((t.flags&1024)!==0)switch(t.tag){case 0:case 11:case 15:break;case 1:if(U!==null){var H=U.memoizedProps,Ye=U.memoizedState,x=t.stateNode,v=x.getSnapshotBeforeUpdate(t.elementType===t.type?H:Nn(t.type,H),Ye);x.__reactInternalSnapshotBeforeUpdate=v}break;case 3:var _=t.stateNode.containerInfo;_.nodeType===1?_.textContent="":_.nodeType===9&&_.documentElement&&_.removeChild(_.documentElement);break;case 5:case 6:case 4:case 17:break;default:throw Error(p(163))}}catch(R){Ke(t,t.return,R)}if(e=t.sibling,e!==null){e.return=t.return,$=e;break}$=t.return}return U=Nc,Nc=!1,U}function $o(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 Dl(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 Ec(e){var t=e.alternate;t!==null&&(e.alternate=null,Ec(t)),e.child=null,e.deletions=null,e.sibling=null,e.tag===5&&(t=e.stateNode,t!==null&&(delete t[Jn],delete t[No],delete t[Da],delete t[Ud],delete t[Bd])),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 Pc(e){return e.tag===5||e.tag===3||e.tag===4}function Tc(e){e:for(;;){for(;e.sibling===null;){if(e.return===null||Pc(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=fl));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 xt=null,En=!1;function Br(e,t,n){for(n=n.child;n!==null;)Rc(e,t,n),n=n.sibling}function Rc(e,t,n){if(Ut&&typeof Ut.onCommitFiberUnmount=="function")try{Ut.onCommitFiberUnmount(jr,n)}catch{}switch(n.tag){case 5:Et||Gs(n,t);case 6:var r=xt,s=En;xt=null,Br(e,t,n),xt=r,En=s,xt!==null&&(En?(e=xt,n=n.stateNode,e.nodeType===8?e.parentNode.removeChild(n):e.removeChild(n)):xt.removeChild(n.stateNode));break;case 18:xt!==null&&(En?(e=xt,n=n.stateNode,e.nodeType===8?La(e.parentNode,n):e.nodeType===1&&La(e,n),tt(e)):La(xt,n.stateNode));break;case 4:r=xt,s=En,xt=n.stateNode.containerInfo,En=!0,Br(e,t,n),xt=r,En=s;break;case 0:case 11:case 14:case 15:if(!Et&&(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)}Br(e,t,n);break;case 1:if(!Et&&(Gs(n,t),r=n.stateNode,typeof r.componentWillUnmount=="function"))try{r.props=n.memoizedProps,r.state=n.memoizedState,r.componentWillUnmount()}catch(d){Ke(n,t,d)}Br(e,t,n);break;case 21:Br(e,t,n);break;case 22:n.mode&1?(Et=(r=Et)||n.memoizedState!==null,Br(e,t,n),Et=r):Br(e,t,n);break;default:Br(e,t,n)}}function Mc(e){var t=e.updateQueue;if(t!==null){e.updateQueue=null;var n=e.stateNode;n===null&&(n=e.stateNode=new sf),t.forEach(function(r){var s=hf.bind(null,e,r);n.has(r)||(n.add(r),r.then(s,s))})}}function Pn(e,t){var n=t.deletions;if(n!==null)for(var r=0;rs&&(s=u),r&=~l}if(r=s,r=Oe()-r,r=(120>r?120:480>r?480:1080>r?1080:1920>r?1920:3e3>r?3e3:4320>r?4320:1960*af(r/1960))-r,10e?16:e,Hr===null)var r=!1;else{if(e=Hr,Hr=null,bl=0,(ke&6)!==0)throw Error(p(331));var s=ke;for(ke|=4,$=e.current;$!==null;){var l=$,u=l.child;if(($.flags&16)!==0){var d=l.deletions;if(d!==null){for(var h=0;hOe()-ki?hs(e,0):ji|=n),Wt(e,t)}function Hc(e,t){t===0&&((e.mode&1)===0?t=1:(t=_s,_s<<=1,(_s&130023424)===0&&(_s=4194304)));var n=$t();e=pr(e,t),e!==null&&(Cr(e,t,n),Wt(e,n))}function mf(e){var t=e.memoizedState,n=0;t!==null&&(n=t.retryLane),Hc(e,n)}function hf(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(p(314))}r!==null&&r.delete(t),Hc(e,n)}var Vc;Vc=function(e,t,n){if(e!==null)if(e.memoizedProps!==t.pendingProps||Qt.current)Vt=!0;else{if((e.lanes&n)===0&&(t.flags&128)===0)return Vt=!1,tf(e,t,n);Vt=(e.flags&131072)!==0}else Vt=!1,Fe&&(t.flags&1048576)!==0&&ku(t,yl,t.index);switch(t.lanes=0,t.tag){case 2:var r=t.type;Il(e,t),e=t.pendingProps;var s=$s(t,kt.current);Qs(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,Ht(r)?(l=!0,hl(t)):l=!1,t.memoizedState=s.state!==null&&s.state!==void 0?s.state:null,qa(t),s.updater=Ml,t.stateNode=s,s._reactInternals=t,ii(t,r,e,n),t=fi(null,t,r,!0,l,n)):(t.tag=0,Fe&&l&&za(t),Ot(null,t,s,n),t=t.child),t;case 16:r=t.elementType;e:{switch(Il(e,t),e=t.pendingProps,s=r._init,r=s(r._payload),t.type=r,s=t.tag=vf(r),e=Nn(r,e),s){case 0:t=di(null,t,r,e,n);break e;case 1:t=vc(null,t,r,e,n);break e;case 11:t=fc(null,t,r,e,n);break e;case 14:t=pc(null,t,r,Nn(r.type,e),n);break e}throw Error(p(306,r,""))}return t;case 0:return r=t.type,s=t.pendingProps,s=t.elementType===r?s:Nn(r,s),di(e,t,r,s,n);case 1:return r=t.type,s=t.pendingProps,s=t.elementType===r?s:Nn(r,s),vc(e,t,r,s,n);case 3:e:{if(yc(t),e===null)throw Error(p(387));r=t.pendingProps,l=t.memoizedState,s=l.element,Iu(e,t),kl(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=Vs(Error(p(423)),t),t=xc(e,t,r,n,s);break e}else if(r!==s){s=Vs(Error(p(424)),t),t=xc(e,t,r,n,s);break e}else for(nn=Or(t.stateNode.containerInfo.firstChild),tn=t,Fe=!0,Cn=null,n=Mu(t,null,r,n),t.child=n;n;)n.flags=n.flags&-3|4096,n=n.sibling;else{if(bs(),r===s){t=hr(e,t,n);break e}Ot(e,t,r,n)}t=t.child}return t;case 5:return Ou(t),e===null&&Ua(t),r=t.type,s=t.pendingProps,l=e!==null?e.memoizedProps:null,u=s.children,Aa(r,s)?u=null:l!==null&&Aa(r,l)&&(t.flags|=32),gc(e,t),Ot(e,t,u,n),t.child;case 6:return e===null&&Ua(t),null;case 13:return _c(e,t,n);case 4:return Ka(t,t.stateNode.containerInfo),r=t.pendingProps,e===null?t.child=Us(t,null,r,n):Ot(e,t,r,n),t.child;case 11:return r=t.type,s=t.pendingProps,s=t.elementType===r?s:Nn(r,s),fc(e,t,r,s,n);case 7:return Ot(e,t,t.pendingProps,n),t.child;case 8:return Ot(e,t,t.pendingProps.children,n),t.child;case 12:return Ot(e,t,t.pendingProps.children,n),t.child;case 10:e:{if(r=t.type._context,s=t.pendingProps,l=t.memoizedProps,u=s.value,De(Sl,r._currentValue),r._currentValue=u,l!==null)if(kn(l.value,u)){if(l.children===s.children&&!Qt.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=mr(-1,n&-n),h.tag=2;var j=l.updateQueue;if(j!==null){j=j.shared;var P=j.pending;P===null?h.next=h:(h.next=P.next,P.next=h),j.pending=h}}l.lanes|=n,h=l.alternate,h!==null&&(h.lanes|=n),Ga(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(p(341));u.lanes|=n,d=u.alternate,d!==null&&(d.lanes|=n),Ga(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}Ot(e,t,s.children,n),t=t.child}return t;case 9:return s=t.type,r=t.pendingProps.children,Qs(t,n),s=fn(s),r=r(s),t.flags|=1,Ot(e,t,r,n),t.child;case 14:return r=t.type,s=Nn(r,t.pendingProps),s=Nn(r.type,s),pc(e,t,r,s,n);case 15:return mc(e,t,t.type,t.pendingProps,n);case 17:return r=t.type,s=t.pendingProps,s=t.elementType===r?s:Nn(r,s),Il(e,t),t.tag=1,Ht(r)?(e=!0,hl(t)):e=!1,Qs(t,n),oc(t,r,s),ii(t,r,s,n),fi(null,t,r,!0,e,n);case 19:return wc(e,t,n);case 22:return hc(e,t,n)}throw Error(p(156,t.tag))};function Gc(e,t){return qo(e,t)}function gf(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 gf(e,t,n,r)}function Ai(e){return e=e.prototype,!(!e||!e.isReactComponent)}function vf(e){if(typeof e=="function")return Ai(e)?1:0;if(e!=null){if(e=e.$$typeof,e===ht)return 11;if(e===Je)return 14}return 2}function Wr(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 Hl(e,t,n,r,s,l){var u=2;if(r=e,typeof e=="function")Ai(e)&&(u=1);else if(typeof e=="string")u=5;else e:switch(e){case Ee:return vs(n.children,s,l,t);case be:u=8,s|=8;break;case St:return e=hn(12,n,t,s|2),e.elementType=St,e.lanes=l,e;case B:return e=hn(13,n,t,s),e.elementType=B,e.lanes=l,e;case Ie:return e=hn(19,n,t,s),e.elementType=Ie,e.lanes=l,e;case ee:return Vl(n,s,l,t);default:if(typeof e=="object"&&e!==null)switch(e.$$typeof){case mt:u=10;break e;case Kt:u=9;break e;case ht:u=11;break e;case Je:u=14;break e;case re:u=16,r=null;break e}throw Error(p(130,e==null?e:typeof e,""))}return t=hn(u,n,t,s),t.elementType=e,t.type=r,t.lanes=l,t}function vs(e,t,n,r){return e=hn(7,e,r,t),e.lanes=n,e}function Vl(e,t,n,r){return e=hn(22,e,r,t),e.elementType=ee,e.lanes=n,e.stateNode={isHidden:!1},e}function Ii(e,t,n){return e=hn(6,e,null,t),e.lanes=n,e}function Li(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 yf(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=js(0),this.expirationTimes=js(-1),this.entangledLanes=this.finishedLanes=this.mutableReadLanes=this.expiredLanes=this.pingedLanes=this.suspendedLanes=this.pendingLanes=0,this.entanglements=js(0),this.identifierPrefix=r,this.onRecoverableError=s,this.mutableSourceEagerHydrationData=null}function Di(e,t,n,r,s,l,u,d,h){return e=new yf(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},qa(l),e}function xf(e,t,n){var r=3"u"||typeof __REACT_DEVTOOLS_GLOBAL_HOOK__.checkDCE!="function"))try{__REACT_DEVTOOLS_GLOBAL_HOOK__.checkDCE(a)}catch(m){console.error(m)}}return a(),Ui.exports=Mf(),Ui.exports}var ld;function If(){if(ld)return Xl;ld=1;var a=Af();return Xl.createRoot=a.createRoot,Xl.hydrateRoot=a.hydrateRoot,Xl}var Lf=If();const Df=kd(Lf),Of="/api";async function _e(a,m){const p=await fetch(`${Of}${a}`,{...m,headers:{"Content-Type":"application/json",...m?.headers??{}}}),S=await p.json();if(!p.ok){const C=S.error?.message??"Ошибка запроса";throw new Error(C)}return S}const Te={async loadSharedConnectionConfig(){return _e("/llm/shared-connection")},async saveSharedConnectionConfig(a){return _e("/llm/shared-connection",{method:"POST",body:JSON.stringify({llmProvider:a.llmProvider,model:a.model,baseUrl:a.baseUrl,temperature:a.temperature,maxOutputTokens:a.maxOutputTokens})})},async listModels(a){return _e("/llm/models",{method:"POST",body:JSON.stringify({llmProvider:a.llmProvider,apiKey:a.apiKey,model:a.model,baseUrl:a.baseUrl})})},async testConnection(a){return _e("/llm/test-connection",{method:"POST",body:JSON.stringify({llmProvider:a.llmProvider,apiKey:a.apiKey,model:a.model,baseUrl:a.baseUrl})})},async normalize(a){return _e("/normalize",{method:"POST",body:JSON.stringify({llmProvider:a.connection.llmProvider,apiKey:a.connection.apiKey,model:a.connection.model,baseUrl:a.connection.baseUrl,temperature:a.connection.temperature,maxOutputTokens:a.connection.maxOutputTokens,promptVersion:a.promptVersion,systemPrompt:a.prompts.systemPrompt,developerPrompt:a.prompts.developerPrompt,domainPrompt:a.prompts.domainPrompt,fewShotExamples:a.prompts.fewShotExamples,userQuestion:a.query.userQuestion,context:{period_hint:a.query.periodHint??"",business_context:a.query.businessContext??"",expected_route:a.query.expectedRoute??""},saveAsTestCase:!!a.saveAsTestCase,useMock:!!a.useMock})})},async loadHistory(){return _e("/history")},async loadTrace(a){return _e(`/history/${a}`)},async loadPresets(){return _e("/presets")},async savePreset(a){return _e("/presets/save",{method:"POST",body:JSON.stringify(a)})},async runEval(a){return _e("/eval/run",{method:"POST",body:JSON.stringify({normalizeConfig:{llmProvider:a.connection.llmProvider,apiKey:a.connection.apiKey,model:a.connection.model,baseUrl:a.connection.baseUrl,temperature:a.connection.temperature,maxOutputTokens:a.connection.maxOutputTokens,promptVersion:a.promptVersion,systemPrompt:a.prompts.systemPrompt,developerPrompt:a.prompts.developerPrompt,domainPrompt:a.prompts.domainPrompt,fewShotExamples:a.prompts.fewShotExamples},caseIds:a.caseIds,useMock:!!a.useMock,mode:a.mode??"standard",caseSetFile:a.caseSetFile,rawQuestions:a.rawQuestions,eval_target:a.evalTarget,compare_with_report_file:a.compareWithReportFile,analysis_date:a.analysisDate})})},async startEvalRunAsync(a){return _e("/eval/run-async/start",{method:"POST",body:JSON.stringify({normalizeConfig:{llmProvider:a.connection.llmProvider,apiKey:a.connection.apiKey,model:a.connection.model,baseUrl:a.connection.baseUrl,temperature:a.connection.temperature,maxOutputTokens:a.connection.maxOutputTokens,promptVersion:a.promptVersion,systemPrompt:a.prompts.systemPrompt,developerPrompt:a.prompts.developerPrompt,domainPrompt:a.prompts.domainPrompt,fewShotExamples:a.prompts.fewShotExamples},caseIds:a.caseIds,useMock:!!a.useMock,mode:a.mode??"standard",caseSetFile:a.caseSetFile,rawQuestions:a.rawQuestions,eval_target:a.evalTarget,compare_with_report_file:a.compareWithReportFile,questions:a.questions,scenarioQuestions:a.scenarioQuestions,scenarioTitle:a.scenarioTitle,analysis_date:a.analysisDate})})},async loadEvalRunAsyncStatus(a){return _e(`/eval/run-async/${encodeURIComponent(a)}`)},async cancelEvalRunAsync(a){return _e(`/eval/run-async/${encodeURIComponent(a)}/cancel`,{method:"POST"})},async startRun(){return _e("/accounting-agent/v1/runs/start",{method:"POST",body:JSON.stringify({initiator:"ndc_operator",source:"gui"})})},async finishRun(a){return _e("/accounting-agent/v1/runs/finish",{method:"POST",body:JSON.stringify({runId:a,status:"DONE",source:"gui",reason:"Остановлено оператором из GUI"})})},async listRuns(){return _e("/accounting-agent/v1/runs")},async listResults(){return _e("/accounting-agent/v1/results")},async runTrace(a){return _e(`/accounting-agent/v1/trace/run/${a}`)},async sendAssistantMessage(a){return _e("/assistant/message",{method:"POST",body:JSON.stringify({session_id:a.sessionId??"",mode:"assistant",message:a.userMessage,user_message:a.userMessage,llmProvider:a.connection.llmProvider,apiKey:a.connection.apiKey,model:a.connection.model,baseUrl:a.connection.baseUrl,temperature:a.connection.temperature,maxOutputTokens:a.connection.maxOutputTokens,promptVersion:a.promptVersion??"address_query_runtime_v1",systemPrompt:a.prompts.systemPrompt,developerPrompt:a.prompts.developerPrompt,domainPrompt:a.prompts.domainPrompt,fewShotExamples:a.prompts.fewShotExamples,context:{period_hint:a.context?.periodHint??"",business_context:a.context?.businessContext??""},useMock:!!a.useMock})})},async loadAssistantSession(a){return _e(`/assistant/session/${a}`)},async saveAutoRunAssistantSession(a){return _e("/autoruns/autogen/save-assistant-session",{method:"POST",body:JSON.stringify(a)})},async loadAssistantAnnotations(a){const m=new URLSearchParams;a?.session_id&&m.set("session_id",a.session_id),typeof a?.limit=="number"&&m.set("limit",String(a.limit));const p=m.toString();return _e(`/assistant/annotations${p?`?${p}`:""}`)},async saveAssistantAnnotation(a){return _e("/assistant/annotations",{method:"POST",body:JSON.stringify(a)})},async loadAutoRunsHistory(a){const m=new URLSearchParams;a?.from&&m.set("from",a.from),a?.to&&m.set("to",a.to),a?.target&&m.set("target",a.target),a?.mode&&m.set("mode",a.mode),a?.use_mock&&m.set("use_mock",a.use_mock),a?.prompt_contains&&m.set("prompt_contains",a.prompt_contains),typeof a?.limit=="number"&&m.set("limit",String(a.limit)),typeof a?.scan_limit=="number"&&m.set("scan_limit",String(a.scan_limit));const p=m.toString();return _e(`/autoruns/history${p?`?${p}`:""}`)},async loadAutoRunDetail(a){return _e(`/autoruns/history/${encodeURIComponent(a)}`)},async loadAutoRunCaseDialog(a,m){return _e(`/autoruns/history/${encodeURIComponent(a)}/case/${encodeURIComponent(m)}/dialog`)},async loadAutoRunAnnotations(a){const m=new URLSearchParams;a?.run_id&&m.set("run_id",a.run_id),a?.case_id&&m.set("case_id",a.case_id),typeof a?.min_rating=="number"&&m.set("min_rating",String(a.min_rating)),a?.manual_case_decision&&m.set("manual_case_decision",a.manual_case_decision),typeof a?.limit=="number"&&m.set("limit",String(a.limit));const p=m.toString();return _e(`/autoruns/annotations${p?`?${p}`:""}`)},async saveAutoRunAnnotation(a){return _e("/autoruns/annotations",{method:"POST",body:JSON.stringify(a)})},async updateAutoRunAnnotation(a){return _e(`/autoruns/annotations/${encodeURIComponent(a.annotation_id)}`,{method:"PATCH",body:JSON.stringify({resolved:a.resolved,resolved_by:a.resolved_by})})},async loadAutoRunPostAnalysis(a){const m=new URLSearchParams;a?.run_id&&m.set("run_id",a.run_id),typeof a?.limit_per_queue=="number"&&m.set("limit_per_queue",String(a.limit_per_queue)),typeof a?.annotation_limit=="number"&&m.set("annotation_limit",String(a.annotation_limit)),typeof a?.scan_limit=="number"&&m.set("scan_limit",String(a.scan_limit)),a?.from&&m.set("from",a.from),a?.to&&m.set("to",a.to),a?.target&&m.set("target",a.target),a?.mode&&m.set("mode",a.mode),a?.use_mock&&m.set("use_mock",a.use_mock),a?.prompt_contains&&m.set("prompt_contains",a.prompt_contains);const p=m.toString();return _e(`/autoruns/post-analysis${p?`?${p}`:""}`)},async loadAutoRunAutogenHistory(a){const m=new URLSearchParams;a?.mode&&m.set("mode",a.mode),typeof a?.limit=="number"&&m.set("limit",String(a.limit));const p=m.toString();return _e(`/autoruns/autogen/history${p?`?${p}`:""}`)},async loadAutoRunAutogenPersonalityCatalog(){return _e("/autoruns/autogen/personality-catalog")},async updateAutoRunAutogenQuestions(a){return _e(`/autoruns/autogen/history/${encodeURIComponent(a.generation_id)}/questions`,{method:"PATCH",body:JSON.stringify({questions:a.questions})})},async deleteAutoRunAutogenHistoryRecord(a){return _e(`/autoruns/autogen/history/${encodeURIComponent(a)}`,{method:"DELETE"})},async generateAutoRunQuestions(a){return _e("/autoruns/autogen/generate",{method:"POST",body:JSON.stringify(a)})}},$f=/(?:^|\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 vr(a){try{return JSON.stringify(a,null,2)}catch{return String(a)}}function Ff(a){return a==="assistant"?"assistant":"user"}function Cd(a){const m=String(a??""),p=m.match($f);return(p?m.slice(0,p.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(z=>z.trimEnd()).filter(z=>z.trim().length>0).filter(z=>!zf.some(ae=>ae.test(z))).join(` +`).trim()}function bf(a,m,p="default"){const S=p==="technical",C=[];C.push("# Assistant conversation export"),C.push(`session_id: ${a||"n/a"}`),C.push(`export_mode: ${p}`),C.push(`exported_at: ${new Date().toISOString()}`),C.push("");for(let M=0;M0&&(S.push("### decomposition_json"),S.push("```json"),S.push(vr(a.decomposition)),S.push("```"),S.push("")),a.assistantMode&&(S.push("### assistant_mode_json"),S.push("```json"),S.push(vr(a.assistantMode)),S.push("```"),S.push("")),(a.annotations??[]).length>0&&(S.push("### annotations_json"),S.push("```json"),S.push(vr(a.annotations)),S.push("```"),S.push("")),a.runSummary&&(S.push("### run_summary_json"),S.push("```json"),S.push(vr(a.runSummary)),S.push("```"),S.push("")),a.coverage&&(S.push("### coverage_json"),S.push("```json"),S.push(vr(a.coverage)),S.push("```"),S.push("")),a.report&&(S.push("### run_report_json"),S.push("```json"),S.push(vr(a.report)),S.push("```"),S.push(""))),S.join(` +`)}function Nd({value:a}){return o.jsx("pre",{className:"json-view",children:JSON.stringify(a??{},null,2)})}function aa({title:a,subtitle:m,actions:p,className:S,hideHeader:C,children:M}){return o.jsxs("section",{className:S?`panel-frame ${S}`:"panel-frame",children:[C?null:o.jsxs("header",{className:"panel-header",children:[o.jsxs("div",{children:[o.jsx("h2",{children:a}),m?o.jsx("p",{children:m}):null]}),p?o.jsx("div",{className:"panel-actions",children:p}):null]}),o.jsx("div",{className:"panel-body",children:M})]})}function Bf(a){return a==="assistant"?"Ассистент":"Вы"}function Qf(a){const m=new Date(a);return Number.isNaN(m.getTime())?a:m.toLocaleTimeString("ru-RU")}async function Hf(a){if(navigator.clipboard&&window.isSecureContext)try{return await navigator.clipboard.writeText(a),!0}catch{}const m=document.createElement("textarea");m.value=a,m.setAttribute("readonly","true"),m.style.position="fixed",m.style.opacity="0",m.style.pointerEvents="none",document.body.appendChild(m),m.select();let p=!1;try{p=document.execCommand("copy")}catch{p=!1}finally{document.body.removeChild(m)}return p}function Vf({commented:a}){const m=a?"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 Gf(a){return a.replace(/\r\n?/g,` +`).replace(/([^\n])\s+(Блок\s+\d+\.)/gi,`$1 + +$2`).replace(/([^\n])\s+(\d+\.\s)/g,`$1 +$2`)}function Wf(a){const p=Gf(a).split(` +`),S=[];let C=[];const M=()=>{C.length!==0&&(S.push(C.join(` +`)),C=[])};for(const z of p){const ae=z.trimEnd(),ne=ae.trim();if(!ne){M();continue}const F=/^Блок\s+\d+\./i.test(ne),X=/^\d+\.\s/.test(ne);(F||X)&&C.length>0&&M(),C.push(ae)}return M(),S.length>0?S:[a]}function qf(a,m){const p=[],S=/\*\*(.+?)\*\*/g;let C=0,M=0,z;for(;(z=S.exec(a))!==null;)z.index>C&&(p.push(o.jsx("span",{children:a.slice(C,z.index)},`${m}-t-${M}`)),M+=1),p.push(o.jsx("strong",{children:z[1]},`${m}-b-${M}`)),M+=1,C=S.lastIndex;return C0?p:[o.jsx("span",{children:a},`${m}-raw`)]}function Kf(a){const m=a.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 Jf(a,m=40){const p=a.replace(/\s+/g," ").trim();if(p.length<=m)return p;const S=p.split(" ").slice(0,3).join(" ").trim();return S.length>=10&&S.length<=m?`${S}…`:`${p.slice(0,m-1).trimEnd()}…`}function Ed(a){return a.replace(/\*\*(.+?)\*\*/g,"$1").replace(/^\d+\.\s*/,"").trim()}function Yf(a){const m=a.replace(/\r\n?/g,` +`).split(` +`).map(C=>C.trim()).find(Boolean),p=Ed(m??"");return(p.split("|")[0]?.trim()??p).replace(/\s+/g," ").trim()}function Xf(a){const m=a.replace(/\r\n?/g,` +`).split(` +`).map(S=>S.trim()).find(Boolean);return!m||!/^\d+\.\s/.test(m)?!1:Ed(m).includes("|")}function Zf(a,m){const p=m.replace(/\r\n?/g,` +`).replace(/\*\*(.+?)\*\*/g,"$1").split(` +`).map((C,M)=>{const z=C.trim();return M===0?z.replace(/^\d+\.\s*/,""):z}).filter(Boolean).join(" ").replace(/\s+/g," ").trim();if(!p)return null;const S=Yf(m)||p;return{message_id:a.message_id,source_text:p,anchor_text:S,preview_text:Jf(S)}}function ep(a,m,p,S){return Wf(a.text).map((M,z)=>{const ae=M.split(` +`),ne=a.role==="assistant"&&Xf(M),F=ne?Zf(a,M):null,X=!!F&&m?.message_id===F?.message_id&&m?.source_text===F?.source_text,Z=ae.map((te,Me)=>o.jsx("p",{className:Kf(te),children:qf(te,`line-${z}-${Me}`)},`line-${z}-${Me}`));return!ne||!F?o.jsx("div",{className:"assistant-msg-block",children:Z},`block-${z}`):o.jsx("div",{className:X?"assistant-msg-block selectable active":"assistant-msg-block selectable",role:"button",tabIndex:0,onClick:()=>{if(X){S();return}p(F)},onKeyDown:te=>{if(!(te.key!=="Enter"&&te.key!==" ")){if(te.preventDefault(),X){S();return}p(F)}},children:Z},`block-${z}`)})}function tp({sessionId:a,conversation:m,inputValue:p,onInputChange:S,selectedContextChip:C,onSelectContextChip:M,onClearContextChip:z,useMock:ae,onUseMockChange:ne,onSend:F,onClear:X,onSaveSession:Z,busy:te,saveBusy:Me=!1,saveDisabled:pe=!1,statusText:me,errorMessage:ye,showSaveAction:Qe=!1,showCommentAction:Xe=!1,onCommentAssistantMessage:He,isAssistantMessageCommented:Ae,canCommentAssistantMessage:b}){const le=g.useRef(null),Ee=g.useRef(!0),be=g.useRef(null),[St,mt]=g.useState("idle"),[Kt,ht]=g.useState("чат");function B(re=!1){le.current&&(re&&(Ee.current=!0),le.current.scrollTop=le.current.scrollHeight)}g.useEffect(()=>{Ee.current&&B()},[m]),g.useEffect(()=>()=>{be.current!==null&&window.clearTimeout(be.current)},[]);async function Ie(re){if(m.length===0)return;const ee=bf(a,m,re),I=await Hf(ee);ht(re==="technical"?"тех":"чат"),mt(I?"success":"error"),be.current!==null&&window.clearTimeout(be.current),be.current=window.setTimeout(()=>{mt("idle")},2200)}function Je(){if(!le.current)return;const re=le.current,ee=re.scrollHeight-re.scrollTop-re.clientHeight;Ee.current=ee<16}return o.jsx(aa,{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:()=>{Ie("default")},disabled:m.length===0,title:"Экспорт только user-facing чата",children:"Скопировать чат"}),o.jsx("button",{type:"button",className:"assistant-copy-btn",onClick:()=>{Ie("technical")},disabled:m.length===0,title:"Технический экспорт с debug payload",children:"Скопировать техчат"}),Qe?o.jsx("button",{type:"button",className:"assistant-copy-btn",onClick:()=>Z?.(),disabled:Me||pe,children:Me?"Сохраняю...":"Сохранить"}):null,o.jsx("button",{type:"button",className:"assistant-copy-btn",onClick:()=>X(),disabled:te&&m.length===0,children:"Сбросить сессию"})]}),o.jsxs("div",{className:"assistant-toolbar-meta",children:[a?o.jsx("span",{className:"status-chip",children:`session: ${a}`}):null,o.jsxs("div",{className:"assistant-toolbar-meta-right",children:[me?o.jsx("span",{className:"assistant-live-status",children:me}):null,St==="success"?o.jsxs("span",{className:"assistant-copy-feedback success",children:["Скопировано (",Kt,")"]}):null,St==="error"?o.jsx("span",{className:"assistant-copy-feedback error",children:"Ошибка копирования"}):null]})]}),ye?o.jsx("p",{className:"error-text assistant-toolbar-error",children:ye}):null]}),o.jsx("div",{ref:le,className:"assistant-chat-list",onScroll:Je,children:m.map((re,ee)=>{const I=re.role==="assistant"&&Xe&&typeof He=="function"&&(typeof b=="function"?b(re,ee):!0),O=re.role==="assistant"&&typeof Ae=="function"?Ae(re,ee):!1;return o.jsxs("article",{className:`assistant-msg ${re.role}`,children:[o.jsxs("header",{className:"assistant-msg-head",children:[o.jsxs("div",{className:"assistant-msg-head-main",children:[o.jsx("strong",{children:Bf(re.role)}),o.jsx("span",{children:Qf(re.created_at)})]}),re.role==="assistant"&&Xe?o.jsx("div",{className:"assistant-msg-head-actions",children:o.jsx("button",{type:"button",className:O?"autoruns-comment-icon assistant-comment-btn commented":"autoruns-comment-icon assistant-comment-btn",onClick:()=>He?.(re,ee),disabled:!I,title:I?"Комментировать ответ ассистента":"Комментарий недоступен для этого сообщения","aria-label":I?"Комментировать ответ ассистента":"Комментарий недоступен для этого сообщения",children:o.jsx(Vf,{commented:O})})}):null]}),o.jsx("div",{className:"assistant-msg-body",children:ep(re,C,M,z)}),re.role==="assistant"&&re.debug?o.jsxs("details",{className:"assistant-debug",children:[o.jsx("summary",{children:"Показать технический разбор"}),o.jsx(Nd,{value:re.debug})]}):null]},re.message_id)})}),o.jsxs("div",{className:"assistant-compose",children:[C?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:C.source_text,children:[o.jsx("span",{className:"assistant-compose-context-pill-text",children:C.preview_text}),o.jsx("button",{type:"button",className:"assistant-compose-context-clear",onClick:z,"aria-label":"Убрать выбранный объект",title:"Убрать выбранный объект",children:"×"})]})]}):null,o.jsxs("label",{className:"full-width",children:["Сообщение",o.jsx("textarea",{className:"assistant-input-textarea",value:p,onChange:re=>S(re.target.value),rows:4,placeholder:C?"Продолжите вопрос по выбранному объекту...":"Введите вопрос к данным компании..."})]}),o.jsxs("div",{className:"button-row assistant-send-row",children:[o.jsxs("label",{className:"checkbox-row",children:[o.jsx("input",{type:"checkbox",checked:ae,onChange:re=>ne(re.target.checked)}),"Mock-режим"]}),o.jsx("button",{type:"button",className:"assistant-send-btn",onClick:()=>{B(!0),F()},disabled:te||!p.trim(),children:te?"Выполняю...":"Отправить"})]})]})]})})}const Zl="http://127.0.0.1:1234/v1",ad="https://api.openai.com/v1",oa="qwen2.5-14b-instruct-1m",eu="unsloth/qwen3-30b-a3b-instruct-2507",np=[{value:oa,label:"Qwen2.5 14B Instruct 1M"},{value:eu,label:"Qwen3 30B A3B Instruct 2507"}];function rp(a){return a.llmProvider!=="local"?"openai":a.model===eu?"local_qwen3":a.model===oa?"local_qwen25":"local_custom"}function sp(a,m){const p=new Map;if(m)for(const S of np)p.set(S.value,S);for(const S of a)p.has(S)||p.set(S,{value:S,label:S});return Array.from(p.values())}function op({value:a,modelOptions:m,modelsBusy:p,onChange:S,onReloadModels:C,onTestConnection:M,onSaveLocalConfig:z,lastStatus:ae,busy:ne,embedded:F=!1}){const X=a.llmProvider==="local",Z=rp(a),te=sp(m,X),Me=te.some(b=>b.value===a.model),[pe,me]=g.useState(String(a.temperature)),[ye,Qe]=g.useState(String(a.maxOutputTokens));g.useEffect(()=>{me(String(a.temperature))},[a.temperature]),g.useEffect(()=>{Qe(String(a.maxOutputTokens))},[a.maxOutputTokens]);const Xe=b=>{const le=b.replace(",",".").trim();if(!le){me(String(a.temperature));return}const Ee=Number(le);if(!Number.isFinite(Ee)){me(String(a.temperature));return}S({...a,temperature:Ee}),me(String(Ee))},He=b=>{const le=b.trim();if(!le){Qe(String(a.maxOutputTokens));return}const Ee=Number.parseInt(le,10);if(!Number.isFinite(Ee)||Ee<=0){Qe(String(a.maxOutputTokens));return}S({...a,maxOutputTokens:Ee}),Qe(String(Ee))},Ae=o.jsxs(o.Fragment,{children:[o.jsxs("div",{className:"grid-two",children:[o.jsxs("label",{children:["Provider",o.jsxs("select",{value:Z,onChange:b=>{const le=b.target.value;if(le==="openai"){S({...a,llmProvider:"openai",baseUrl:ad});return}if(le==="local_qwen25"){S({...a,llmProvider:"local",model:oa,baseUrl:Zl});return}if(le==="local_qwen3"){S({...a,llmProvider:"local",model:eu,baseUrl:Zl});return}S({...a,llmProvider:"local",model:a.llmProvider==="local"?a.model:oa,baseUrl:Zl})},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:Me?a.model:"__manual__",onChange:b=>{const le=b.target.value;le!=="__manual__"&&S({...a,model:le})},children:[o.jsx("option",{value:"__manual__",children:"Manual input"}),te.map(b=>o.jsx("option",{value:b.value,children:b.label},b.value))]})]}),o.jsxs("label",{children:["Model ID (manual / current)",o.jsx("input",{value:a.model,onChange:b=>S({...a,model:b.target.value}),placeholder:"qwen2.5-14b-instruct-1m or unsloth/qwen3-30b-a3b-instruct-2507"})]}),X?null:o.jsxs("label",{className:"full-width",children:["OpenAI API Key",o.jsx("input",{type:"password",value:a.apiKey,onChange:b=>S({...a,apiKey:b.target.value}),placeholder:"sk-..."})]}),o.jsxs("label",{className:X?"full-width":void 0,children:[X?"Local server base URL":"Base URL",o.jsx("input",{value:a.baseUrl,onChange:b=>S({...a,baseUrl:b.target.value}),placeholder:X?Zl:ad})]}),o.jsxs("label",{children:["Temperature",o.jsx("input",{type:"number",step:"0.1",value:pe,onChange:b=>me(b.target.value),onBlur:b=>Xe(b.target.value),onKeyDown:b=>{b.key==="Enter"&&Xe(b.target.value)}})]}),o.jsxs("label",{children:["Max output tokens",o.jsx("input",{type:"number",value:ye,onChange:b=>Qe(b.target.value),onBlur:b=>He(b.target.value),onKeyDown:b=>{b.key==="Enter"&&He(b.target.value)}})]})]}),o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",onClick:()=>z(),children:"Save local config"}),o.jsx("button",{type:"button",onClick:()=>C(),disabled:ne||p,children:p?"Loading models...":"Load model list"}),o.jsx("button",{type:"button",onClick:()=>M(),disabled:ne,children:ne?"Checking...":"Test connection"})]})]});return F?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:ae||"Status: not checked"})]}),Ae]}):o.jsx(aa,{title:"LLM Connector",subtitle:"Switch between OpenAI cloud and local OpenAI-compatible server.",actions:o.jsx("span",{className:"status-chip",children:ae||"Status: not checked"}),children:Ae})}function lp({value:a,onChange:m,presets:p,selectedPresetId:S,onSelectPreset:C,onLoadPreset:M,onSavePreset:z,onResetDefaults:ae,onDiffPrevious:ne,presetName:F,onPresetNameChange:X,diffSummary:Z,embedded:te=!1}){const Me=o.jsxs(o.Fragment,{children:[o.jsxs("div",{className:"prompt-manager-grid",children:[o.jsxs("label",{children:["Системный prompt",o.jsx("textarea",{value:a.systemPrompt,onChange:pe=>m({...a,systemPrompt:pe.target.value}),rows:6})]}),o.jsxs("label",{children:["Developer / Instruction prompt",o.jsx("textarea",{value:a.developerPrompt,onChange:pe=>m({...a,developerPrompt:pe.target.value}),rows:6})]}),o.jsxs("label",{children:["Domain prompt",o.jsx("textarea",{value:a.domainPrompt,onChange:pe=>m({...a,domainPrompt:pe.target.value}),rows:6})]}),o.jsxs("label",{children:["Schema notes",o.jsx("textarea",{value:a.schemaNotes,onChange:pe=>m({...a,schemaNotes:pe.target.value}),rows:6})]}),o.jsxs("label",{className:"full-width",children:["Few-shot examples",o.jsx("textarea",{value:a.fewShotExamples,onChange:pe=>m({...a,fewShotExamples:pe.target.value}),rows:8})]})]}),o.jsxs("div",{className:"button-row",children:[o.jsxs("select",{value:S,onChange:pe=>C(pe.target.value),children:[o.jsx("option",{value:"",children:"Выберите preset..."}),p.map(pe=>o.jsx("option",{value:pe.id,children:pe.name},pe.id))]}),o.jsx("button",{type:"button",onClick:()=>M(),children:"Загрузить preset"}),o.jsx("input",{value:F,onChange:pe=>X(pe.target.value),placeholder:"Имя для сохранения"}),o.jsx("button",{type:"button",onClick:()=>z(),children:"Сохранить preset"}),o.jsx("button",{type:"button",onClick:()=>ne(),children:"Diff с предыдущим"}),o.jsx("button",{type:"button",onClick:()=>ae(),children:"Сбросить к default"})]}),Z?o.jsx("p",{className:"diff-summary",children:Z}):null]});return te?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 уровни управляются отдельно."})]})}),Me]}):o.jsx(aa,{title:"Prompt Manager",subtitle:"Системный, developer и domain уровни управляются отдельно.",children:Me})}const Hi={fromLocal:"",toLocal:"",target:"all",mode:"all",useMock:"any",promptContains:"",limit:120},ea="needs_dialog_policy_fix",Pt="__all__",la="__live__:",id="ndc_autoruns_ui_config_v1",ud="ndc-autoruns-save",Vi=["Анализ запроса","Получение данных","Подготовка ответа"];function ap(a,m){const p=a.trim();if(!p)return"";if(!m)return p;const S=p.toLowerCase(),C=m.anchor_text.trim(),M=C.toLowerCase();return M&&S.includes(M)?p:`По выбранному объекту "${C}": ${p}`}const Xi=[{id:"general",label:"Общий контур",domain:"",defaultPrompt:"Генерируй реалистичные живые вопросы бухгалтера по 1С. Добавляй разговорные формулировки и опечатки, но сохраняй бизнес-смысл."}];function ip(a=Xi){return a.reduce((m,p)=>(m[p.id]=p.defaultPrompt,m),{})}const cd={mode:"codex_creative",count:24,personalityId:"general",personalityPrompts:ip(),persistToEvalCases:!0,generatedBy:"manual_reviewer"};function Gi(a){const m=String(a??"").trim();return/^\d{4}-\d{2}-\d{2}$/.test(m)?m:""}function dd(a){const m=typeof a=="number"&&Number.isFinite(a)?Math.trunc(a):160;return Math.max(110,Math.min(520,m))}function up(a){const m=a.getFullYear(),p=String(a.getMonth()+1).padStart(2,"0"),S=String(a.getDate()).padStart(2,"0"),C=String(a.getHours()).padStart(2,"0"),M=String(a.getMinutes()).padStart(2,"0");return`${m}-${p}-${S}T${C}:${M}`}function fd(){const a=new Date;return a.setDate(a.getDate()-14),up(a)}function ta(a){if(!a.trim())return;const m=Date.parse(a);if(Number.isFinite(m))return new Date(m).toISOString()}function gn(a){if(!a)return"нет данных";const m=Date.parse(a);return Number.isFinite(m)?new Date(m).toLocaleString("ru-RU"):a}function pd(a){const m=typeof a.case_message_index=="number"?a.case_message_index:typeof a.message_index=="number"?a.message_index:null;if(m===null||m<0)return null;const p=Math.floor(m/2)+1,S=String(p).padStart(3,"0"),C=a.role==="assistant"?"ответ":"вопрос";return`${S} ${C}`}function Wi(a){return a==="saved_user_sessions"?"Пользовательские сессии":a}function Pd(a){return a?a.context?.agent_run===!0||a.context?.saved_case_set_kind==="agent_semantic_scenario"?!0:typeof a.title=="string"&&a.title.trim().toUpperCase().startsWith("AGENT"):!1}function qi(a){const m=a.title??gn(a.created_at);return Pd(a)&&!m.trim().toUpperCase().startsWith("AGENT")?`AGENT | ${m}`:m}function cp(a){const m=a[a.length-1];return`Ручная сессия ${gn(m?.created_at??new Date().toISOString())}`}function dp(a,m){return m<=0?0:Math.max(0,Math.min(100,Number((a/m*100).toFixed(1))))}function na(a){return typeof a!="number"?"нет данных":`${a.toFixed(1)}%`}function fp(a){return a==="assistant_stage1"?"assistant/s1":a==="assistant_stage2"?"assistant/s2":a==="assistant_p0"?"assistant/p0":a}function md(a){return a==="up"?"Рост":a==="down"?"Регресс":"Без изменений"}function Ki(a){const m=Math.max(1,Math.min(5,Math.round(a)));return`${"●".repeat(m)}${"○".repeat(5-m)}`}function hd(a){return a.length===0?o.jsx("p",{className:"muted",children:"Покрытие доменов пока не сформировано."}):o.jsx("div",{className:"autoruns-coverage-list",children:a.map(m=>{const p=dp(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," (",p,"%)"]})]}),o.jsx("div",{className:"autoruns-coverage-bar",children:o.jsx("div",{style:{width:`${p}%`}})})]},m.domain)})})}function sa(a){return`${la}${a}`}function Js(a){return a.startsWith(la)}function gd(a){return a.startsWith(la)?a.slice(la.length):""}function Td(a){const m=a.report_summary?.run_timestamp??a.created_at,p=Math.max(0,a.total_cases-a.completed_cases);return{run_id:sa(a.job_id),eval_target:a.eval_target,run_timestamp:m,mode:"single-pass-strict",llm_provider:null,model:null,use_mock:null,analysis_date:a.report_summary?.analysis_date??a.analysis_date??null,prompt_version:null,schema_version:null,suite_id:a.case_set_file,cases_total:a.total_cases,requests_total:null,report_path:`async_job:${a.job_id}`,score_index:a.report_summary?.score_index??null,blocking_failures:0,quality_failures:0,closed_cases:a.completed_cases,open_cases:p,domain_coverage:[{domain:"runtime",total_cases:a.total_cases,closed_cases:a.completed_cases}]}}function Qo(a,m){const p=Td(a),S=a.cases.map(X=>({case_id:X.case_id,domain:null,query_class:null,status:X.status==="completed"?"closed":X.status==="failed"?"open":"unknown",score_index:null,trace_id:null,reply_type:null,session_id:`${a.run_id}-${X.case_id}`,dialog_available:X.messages.length>0,commented_count:0,latest_annotation_at:null,avg_rating:null,checks:null,metric_subscores:null})),M=m!==Pt&&S.some(X=>X.case_id===m)?m:S.length>0?Pt:"",z={ok:!0,run:p,coverage:{closed_cases:a.completed_cases,open_cases:Math.max(0,a.total_cases-a.completed_cases),domain_coverage:[{domain:"runtime",total_cases:a.total_cases,closed_cases:a.completed_cases}]},cases:S,annotations_summary:{total:0},report:a.report_summary?{run_id:a.report_summary.run_id,run_timestamp:a.report_summary.run_timestamp,score_index:a.report_summary.score_index,cases_total:a.report_summary.cases_total,analysis_date:a.report_summary.analysis_date??a.analysis_date??null}:{}},ae=[];let ne=0;if(M===Pt)for(const X of a.cases)for(let Z=0;ZZ.case_id===M)??null;for(let Z=0;Z<(X?.messages.length??0);Z+=1){const te=X?.messages[Z];te&&ae.push({...te,message_index:Z,case_id:M,case_message_index:Z,commented:!1,annotation:null})}}const F={ok:!0,run_id:p.run_id,case_id:M,source:"assistant_session",session_id:M===Pt?`${a.run_id}::__all__`:`${a.run_id}-${M}`,messages:ae,decomposition:[],assistant_mode:{status:a.status,completed_cases:a.completed_cases,total_cases:a.total_cases},annotations:[]};return{detail:z,dialog:F,caseId:M}}function pp({commented:a}){const m=a?"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 vd({resolved:a}){return o.jsxs("svg",{className:a?"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"}),a?o.jsx("path",{d:"M5.1 8.2 7.2 10.3 11 6.5"}):null]})}function yd(){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 xd(){return o.jsxs("svg",{className:"autoruns-question-grip-svg",viewBox:"0 0 16 16","aria-hidden":"true",focusable:"false",children:[o.jsx("circle",{cx:"4",cy:"4",r:"1"}),o.jsx("circle",{cx:"8",cy:"4",r:"1"}),o.jsx("circle",{cx:"12",cy:"4",r:"1"}),o.jsx("circle",{cx:"4",cy:"8",r:"1"}),o.jsx("circle",{cx:"8",cy:"8",r:"1"}),o.jsx("circle",{cx:"12",cy:"8",r:"1"}),o.jsx("circle",{cx:"4",cy:"12",r:"1"}),o.jsx("circle",{cx:"8",cy:"12",r:"1"}),o.jsx("circle",{cx:"12",cy:"12",r:"1"})]})}function mp({expanded:a}){return o.jsx("svg",{className:a?"autoruns-card-chevron-svg expanded":"autoruns-card-chevron-svg",viewBox:"0 0 16 16","aria-hidden":"true",focusable:"false",children:o.jsx("path",{d:"M3.5 6.2 8 10.4l4.5-4.2"})})}function hp(){return o.jsx("svg",{className:"autoruns-card-launch-svg",viewBox:"0 0 16 16","aria-hidden":"true",focusable:"false",children:o.jsx("path",{d:"M5 3.8 12 8l-7 4.2Z"})})}function gp(){return o.jsx("svg",{className:"autoruns-card-stop-svg",viewBox:"0 0 16 16","aria-hidden":"true",focusable:"false",children:o.jsx("rect",{x:"4.2",y:"4.2",width:"7.6",height:"7.6",rx:"0.8"})})}async function _d(a){if(navigator.clipboard&&window.isSecureContext)try{return await navigator.clipboard.writeText(a),!0}catch{}const m=document.createElement("textarea");m.value=a,m.setAttribute("readonly","true"),m.style.position="fixed",m.style.opacity="0",m.style.pointerEvents="none",document.body.appendChild(m),m.select();let p=!1;try{p=document.execCommand("copy")}catch{p=!1}finally{document.body.removeChild(m)}return p}function ra({expanded:a}){return o.jsx("svg",{className:a?"autoruns-group-chevron-svg expanded":"autoruns-group-chevron-svg",viewBox:"0 0 16 16","aria-hidden":"true",focusable:"false",children:o.jsx("path",{d:"M3.5 6.2 8 10.4l4.5-4.2"})})}function vp({connection:a,modelOptions:m,modelsBusy:p,connectionStatus:S,connectionBusy:C,onConnectionChange:M,onReloadModels:z,onSaveLocalConfig:ae,onTestConnection:ne,prompts:F,onPromptsChange:X,promptPresets:Z,selectedPresetId:te,onSelectPreset:Me,onLoadPreset:pe,onSavePreset:me,onResetDefaults:ye,onDiffPrevious:Qe,presetName:Xe,onPresetNameChange:He,diffSummary:Ae,assistantPromptVersion:b,decompositionPromptVersion:le,showSettingsMode:Ee,showAutoRunsMode:be,showAssistantMode:St,showProgressMode:mt,showCommentsMode:Kt,onLog:ht}){const[B,Ie]=g.useState({...Hi,fromLocal:fd()}),[Je,re]=g.useState(""),[ee,I]=g.useState(null),[O,L]=g.useState(null),[y,k]=g.useState(null),[se,ue]=g.useState([]),[de,xe]=g.useState("all"),[we,je]=g.useState(!1),[Ce,gt]=g.useState(null),[sn,yr]=g.useState([]),[Rn,vn]=g.useState(""),[Se,Jt]=g.useState(""),[Ve,wt]=g.useState(""),[Tt,yn]=g.useState(Xi),[W,vt]=g.useState(cd),[xr,Mn]=g.useState([]),[Ge,Rt]=g.useState(""),[zt,An]=g.useState(""),[ve,Mt]=g.useState([]),[Ze,ys]=g.useState(!1),[st,er]=g.useState(null),[_r,on]=g.useState(""),[In,Ft]=g.useState(null),[xn,Yt]=g.useState(null),[q,At]=g.useState(null),[tr,Kr]=g.useState(null),[Jr,Yr]=g.useState(!1),[Ln,bt]=g.useState(!1),[Dn,On]=g.useState(!1),[$n,zn]=g.useState(!1),[E,Y]=g.useState(!1),[Q,Pe]=g.useState(!1),[et,Ho]=g.useState(!1),[Xr,Vo]=g.useState(!1),[Fn,Ys]=g.useState(""),Sr=g.useRef(null),[Go,Xs]=g.useState("idle"),[Wo,qo]=g.useState("чат"),[Zs,We]=g.useState(""),[it,Oe]=g.useState(""),[ln,Zr]=g.useState([]),[wr,bn]=g.useState([]),[eo,xs]=g.useState(""),[jr,Ut]=g.useState(null),[to,Xt]=g.useState(!1),[Ko,Jo]=g.useState(!1),[ia,Un]=g.useState(""),[_s,Zt]=g.useState(""),[Ss,kr]=g.useState(String(Hi.limit)),[ua,Bn]=g.useState(String(cd.count)),[ws,js]=g.useState(160),[Cr,Qn]=g.useState(""),[es,fe]=g.useState(!1),[nr,no]=g.useState(!0),[Hn,ro]=g.useState(!0),[rr,so]=g.useState(!0),[Vn,ts]=g.useState(!0),[ce,ot]=g.useState({open:!1,caseId:"",caseMessageIndex:-1,messageIndex:-1,rating:3,comment:"",manualCaseDecision:ea,annotationAuthor:"manual_reviewer",saving:!1,error:""}),[he,It]=g.useState({open:!1,messageIndex:-1,rating:3,comment:"",annotationAuthor:"manual_reviewer",saving:!1,error:""}),[yt,jt]=g.useState({open:!1,title:"",saving:!1,error:""}),[en,Nr]=g.useState({open:!1,generationId:"",questionIndex:-1,questionText:"",saving:!1,error:""}),[Lt,ks]=g.useState({open:!1,generationId:"",title:"",saving:!1,error:""}),Cs=g.useRef(!1),sr=g.useRef(null),ns=g.useRef(null),qe=W.mode==="saved_user_sessions",Gn=g.useMemo(()=>Tt.find(i=>i.id===W.personalityId)??Tt[0]??Xi[0],[W.personalityId,Tt]),tt=g.useMemo(()=>xr.filter(i=>i.mode===W.mode),[xr,W.mode]),Re=g.useMemo(()=>tt.find(i=>i.generation_id===Ge)??tt[0]??null,[Ge,tt]),an=g.useMemo(()=>we?se.filter(i=>!i.resolved):se,[se,we]),Le=an.find(i=>i.annotation_id===Rn)??null,Yo=y?.messages.find(i=>i.message_index===ce.messageIndex)??null,oo=g.useMemo(()=>{if(!y||ce.messageIndex<0)return null;for(let i=ce.messageIndex-1;i>=0;i-=1){const c=y.messages[i];if(c?.role==="user")return c}return null},[ce.messageIndex,y]),or=g.useMemo(()=>{const i=new Map;for(const c of wr)c.message_id&&i.set(c.message_id,c);return i},[wr]),Ns=he.messageIndex>=0?ln[he.messageIndex]??null:null,lo=g.useMemo(()=>{if(he.messageIndex<0)return null;for(let i=he.messageIndex-1;i>=0;i-=1){const c=ln[i];if(c?.role==="user")return c}return null},[he.messageIndex,ln]),ut=g.useMemo(()=>{const i=an.map(f=>({source:"autorun",key:`autorun:${f.annotation_id}`,updated_at:f.updated_at,rating:f.rating,autorun:f,assistant:null})),c=wr.map(f=>({source:"assistant_live",key:`assistant:${f.annotation_id}`,updated_at:f.updated_at,rating:f.rating,autorun:null,assistant:f}));return[...i,...c].sort((f,w)=>Date.parse(w.updated_at)-Date.parse(f.updated_at))},[wr,an]),Es=g.useMemo(()=>{if(ut.length===0)return null;const i=ut.reduce((c,f)=>c+f.rating,0)/ut.length;return Number(i.toFixed(2))},[ut]),Er=g.useMemo(()=>{const i=[...ee?.items??[]];return q&&i.unshift(Td(q)),Se&&!i.some(c=>c.run_id===Se)&&O?.run&&i.unshift(O.run),i},[q,ee?.items,O?.run,Se]),V=g.useCallback(i=>{ht?.(`[autoruns] ${i}`)},[ht]),lr=g.useCallback(async i=>{const c=String(i??"").trim();if(!c){bn([]);return}try{const f=await Te.loadAssistantAnnotations({session_id:c,limit:400});bn(f.items??[])}catch(f){const w=f instanceof Error?f.message:String(f);V(`Assistant live annotations load error: ${w}`)}},[V]),_n=g.useCallback(i=>{It(c=>c.saving&&!i?.force?c:{open:!1,messageIndex:-1,rating:3,comment:"",annotationAuthor:"manual_reviewer",saving:!1,error:""})},[]),rs=g.useCallback(i=>{jt(c=>c.saving&&!i?.force?c:{open:!1,title:"",saving:!1,error:""})},[]),ct=g.useCallback(i=>{Nr(c=>c.saving&&!i?.force?c:{open:!1,generationId:"",questionIndex:-1,questionText:"",saving:!1,error:""})},[]),Sn=g.useCallback(i=>{ks(c=>c.saving&&!i?.force?c:{open:!1,generationId:"",title:"",saving:!1,error:""})},[]);g.useEffect(()=>()=>{Sr.current!==null&&window.clearTimeout(Sr.current)},[]);const Pr=g.useCallback(async(i,c,f)=>{i.stopPropagation(),i.preventDefault();const w=String(c??"").trim();if(w)try{await _d(w),V(`${f} copied: ${w}`)}catch(A){const ie=A instanceof Error?A.message:String(A);We(`Копирование ${f}: ${ie}`),V(`copy ${f} error: ${ie}`)}},[V]),Tr=g.useCallback(async i=>{if(!y||y.messages.length===0||!Se)return;const c=Uf({runId:Se,caseId:Ve||y.case_id||"n/a",sessionId:y.session_id,source:y.source,messages:y.messages,decomposition:y.decomposition,assistantMode:y.assistant_mode,annotations:y.annotations,runSummary:O?.run??null,coverage:O?.coverage??null,report:O?.report??null},i),f=await _d(c);qo(i==="technical"?"тех":"чат"),Xs(f?"success":"error"),Sr.current!==null&&window.clearTimeout(Sr.current),Sr.current=window.setTimeout(()=>{Xs("idle")},2200),V(f?`Dialog ${i==="technical"?"technical":"chat"} copied: run=${Se} case=${Ve||y.case_id}`:`Dialog copy failed: run=${Se} case=${Ve||y.case_id}`)},[y,V,O,Ve,Se]);function ca(){let i=0;Un(Vi[0]);const c=window.setInterval(()=>{i=Math.min(i+1,Vi.length-1),Un(Vi[i])},650);return()=>window.clearInterval(c)}const ao=g.useCallback(()=>{Oe(""),Zr([]),bn([]),xs(""),Ut(null),Un(""),Zt(""),_n({force:!0}),V("Live-чат ассистента в истории автопрогонов сброшен.")},[_n,V]),io=g.useCallback(async()=>{const i=ap(eo,jr);if(!i)return;Jo(!0),Zt(""),xs(""),Zr(f=>[...f,{message_id:`autoruns-live-${Date.now()}`,session_id:it||"pending",role:"user",text:i,reply_type:null,created_at:new Date().toISOString(),trace_id:null,debug:null}]);const c=ca();try{const f=await Te.sendAssistantMessage({connection:a,prompts:F,userMessage:i,sessionId:it||void 0,promptVersion:b,useMock:to});Oe(f.session_id),Zr(f.conversation),await lr(f.session_id),Un("Ответ готов"),V(`Live-ответ ассистента получен: trace=${f.debug.trace_id}`)}catch(f){const w=f instanceof Error?f.message:String(f);Zt(w),Un("Ошибка ассистента"),V(`Live-чат ассистента: ошибка отправки сообщения: ${w}`)}finally{c(),Jo(!1)}},[eo,jr,it,to,b,a,lr,V,F]),ss=g.useCallback(()=>{if(!it.trim()||ln.length===0){Zt("Сначала получите хотя бы один ответ в живой сессии ассистента.");return}Zt(""),jt({open:!0,title:cp(ln),saving:!1,error:""})},[ln,it]),Ps=g.useCallback(async()=>{const i=it.trim(),c=yt.title.trim();if(!i){jt(f=>({...f,error:"Активная сессия ассистента не найдена."}));return}if(!c){jt(f=>({...f,error:"Укажите название сессии."}));return}jt(f=>({...f,saving:!0,error:""}));try{const f=[F.systemPrompt,F.developerPrompt,F.domainPrompt,F.schemaNotes,F.fewShotExamples].join("||"),w=await Te.saveAutoRunAssistantSession({session_id:i,title:c,generated_by:W.generatedBy.trim()||void 0,context:{llm_provider:a.llmProvider,model:a.model,assistant_prompt_version:b,decomposition_prompt_version:le,prompt_fingerprint:f}});Mn(A=>[w.generation,...A.filter(ie=>ie.generation_id!==w.generation.generation_id)]),vt(A=>({...A,mode:"saved_user_sessions"})),Rt(w.generation.generation_id),rs({force:!0}),V(`Живая сессия сохранена в автопрогоны: ${w.generation.generation_id}`)}catch(f){const w=f instanceof Error?f.message:String(f);jt(A=>({...A,saving:!1,error:w})),V(`Assistant live save error: ${w}`)}},[yt.title,it,b,W.generatedBy,rs,a.llmProvider,a.model,le,V,F.developerPrompt,F.domainPrompt,F.fewShotExamples,F.schemaNotes,F.systemPrompt]),uo=g.useCallback(i=>{const c=i.trim();if(!c){kr(String(B.limit));return}if(!/^\d+$/.test(c)){kr(String(B.limit));return}const f=Number.parseInt(c,10);if(!Number.isFinite(f)){kr(String(B.limit));return}const w=Math.max(1,Math.min(500,f));w!==B.limit&&Ie(A=>({...A,limit:w})),kr(String(w))},[B.limit]),Xo=g.useCallback(i=>{const c=i.trim();if(!c){Bn(String(W.count));return}if(!/^\d+$/.test(c)){Bn(String(W.count));return}const f=Number.parseInt(c,10);if(!Number.isFinite(f)){Bn(String(W.count));return}const w=Math.max(1,Math.min(200,f));w!==W.count&&vt(A=>({...A,count:w})),Bn(String(w))},[W.count]),Zo=g.useCallback(i=>{js(dd(i))},[]),el=g.useCallback(i=>{const c=i.currentTarget.offsetHeight;Number.isFinite(c)&&c>0&&Zo(c)},[Zo]),ar=g.useCallback(async()=>{Vo(!0);try{const i=await Te.loadAutoRunAnnotations({limit:800,manual_case_decision:de});ue(i.items),gt(i.manual_case_decision_schema??null),yr(i.available_manual_case_decisions??[]),vn(c=>i.items.length===0?"":i.items.some(f=>f.annotation_id===c)?c:i.items[0].annotation_id)}catch(i){V(`Annotations load error: ${i instanceof Error?i.message:String(i)}`)}finally{Vo(!1)}},[de,V]),un=g.useCallback(async()=>{zn(!0);try{const i=await Te.loadAutoRunAutogenHistory({limit:180});Mn(i.items)}catch(i){V(`Autogen history load error: ${i instanceof Error?i.message:String(i)}`)}finally{zn(!1)}},[V]),tl=g.useCallback(async()=>{try{const c=(await Te.loadAutoRunAutogenPersonalityCatalog()).items.map(f=>({id:String(f.id??"").trim(),label:String(f.label??"").trim(),domain:typeof f.domain=="string"?f.domain.trim():"",defaultPrompt:String(f.default_prompt??"").trim()})).filter(f=>f.id.length>0&&f.label.length>0);if(c.length===0)return;yn(c.map(f=>({id:f.id,label:f.label,domain:f.domain||"",defaultPrompt:f.defaultPrompt||"Генерируй реалистичные вопросы бухгалтера по выбранному профилю. Не выдумывай непокрытые возможности."})))}catch(i){V(`Autogen personality catalog load error: ${i instanceof Error?i.message:String(i)}`)}},[V]),wn=g.useCallback(async()=>{On(!0);try{const i=await Te.loadAutoRunPostAnalysis({run_id:Se&&!Js(Se)?Se:void 0,limit_per_queue:30,annotation_limit:1500,from:ta(B.fromLocal),to:ta(B.toLocal),target:B.target,mode:B.mode,use_mock:B.useMock,prompt_contains:B.promptContains.trim()||void 0});Kr(i)}catch(i){V(`Post-analysis load error: ${i instanceof Error?i.message:String(i)}`),Kr(null)}finally{On(!1)}},[B.fromLocal,B.mode,B.promptContains,B.target,B.toLocal,B.useMock,V,Se]),da=g.useCallback(async()=>{Yr(!0),We("");try{if(W.mode==="saved_user_sessions")throw new Error("Пользовательские сессии сохраняются из живого чата, а не генерируются автоматически.");const i=W.personalityPrompts[W.personalityId]??"",c=[F.systemPrompt,F.developerPrompt,F.domainPrompt,F.schemaNotes,F.fewShotExamples].join(` +`).slice(0,900),f=await Te.generateAutoRunQuestions({mode:W.mode,count:W.count,domain:Gn.domain||void 0,persist_to_eval_cases:W.persistToEvalCases,generated_by:W.generatedBy.trim()||void 0,llm:{llm_provider:a.llmProvider,api_key:a.apiKey,model:a.model,base_url:a.baseUrl,temperature:a.temperature,max_output_tokens:a.maxOutputTokens},context:{llm_provider:a.llmProvider,model:a.model,assistant_prompt_version:b,decomposition_prompt_version:le,prompt_fingerprint:c,autogen_personality_id:Gn.id,autogen_personality_prompt:i.trim()||void 0}});V(`Generated ${f.generation.count} questions (${f.generation.mode}) id=${f.generation.generation_id}`+(f.generation.saved_case_set_file?` saved=${f.generation.saved_case_set_file}`:"")),Rt(f.generation.generation_id),Mt([...f.generation.questions??[]]),await un()}catch(i){const c=i instanceof Error?i.message:String(i);We(`Автогенерация: ${c}`),V(`Autogen generate error: ${c}`)}finally{Yr(!1)}},[b,W.count,W.generatedBy,W.mode,W.personalityId,W.personalityPrompts,W.persistToEvalCases,a.apiKey,a.baseUrl,a.llmProvider,a.maxOutputTokens,a.model,a.temperature,le,un,V,F.developerPrompt,F.domainPrompt,F.fewShotExamples,F.schemaNotes,F.systemPrompt,Gn.domain,Gn.id]),os=g.useCallback(async(i,c)=>{if(Js(i)){const f=gd(i);if(q&&q.job_id===f){const w=Qo(q,c);Jt(i),wt(w.caseId),k(w.dialog);return}k(null);return}Ho(!0);try{const f=await Te.loadAutoRunCaseDialog(i,c);k(f)}catch(f){const w=f instanceof Error?f.message:String(f);We(`Диалог кейса: ${w}`),k(null),V(`Dialog load error for ${i}/${c}: ${w}`)}finally{Ho(!1)}},[q,V]),jn=g.useCallback(async(i,c)=>{if(Js(i)){const f=gd(i);if(q&&q.job_id===f){const w=Qo(q,c??Pt);Jt(i),wt(w.caseId),L(w.detail),k(w.dialog);return}Jt(i),wt(""),L(null),k(null);return}Pe(!0);try{const f=await Te.loadAutoRunDetail(i);L(f);const w=(c&&(c===Pt||f.cases.some(A=>A.case_id===c))?c:"")||(f.cases.length>0?Pt:"")||"";Jt(i),wt(w),w?await os(i,w):k(null)}catch(f){const w=f instanceof Error?f.message:String(f);We(`Детализация прогона: ${w}`),L(null),k(null),V(`Run detail load error for ${i}: ${w}`)}finally{Pe(!1)}},[q,os,V]),Wn=g.useCallback(async i=>{Y(!0),We("");try{const c=await Te.loadAutoRunsHistory({from:ta(B.fromLocal),to:ta(B.toLocal),target:B.target,mode:B.mode,use_mock:B.useMock,prompt_contains:B.promptContains.trim()||void 0,limit:B.limit});if(I(c),c.items.length===0){Jt(""),wt(""),L(null),k(null);return}const f=i?.keepSelection??!0,w=i?.preferredRunId??"",A=i?.preferredCaseId??"",ie=f&&w&&c.items.some(nt=>nt.run_id===w)?w:c.items[0].run_id;await jn(ie,f?A:void 0),wn()}catch(c){const f=c instanceof Error?c.message:String(c);We(`История прогонов: ${f}`),V(`History load error: ${f}`)}finally{Y(!1)}},[B.fromLocal,B.limit,B.mode,B.promptContains,B.target,B.toLocal,B.useMock,wn,jn,V]),Dt=g.useCallback(()=>{sr.current!==null&&(window.clearTimeout(sr.current),sr.current=null)},[]),co=g.useCallback(async i=>{try{const c=await Te.loadEvalRunAsyncStatus(i);At(c.job);const f=sa(i);if(Se===f){const w=Qo(c.job,Ve||Pt);L(w.detail),k(w.dialog),wt(w.caseId)}if(c.job.status==="completed"){Dt(),bt(!1),fe(!1),Qn("");const w=c.job.report_summary?.run_id??c.job.run_id;await Wn({keepSelection:!0,preferredRunId:w||Se,preferredCaseId:Pt}),await un(),At(null);return}if(c.job.status==="failed"){Dt(),bt(!1),fe(!1),Qn(""),We(`Запуск прогонов: ${c.job.error??"неизвестная ошибка"}`),V(`Autogen async run failed: ${c.job.error??"unknown error"}`);return}if(c.job.status==="canceled"){Dt(),bt(!1),fe(!1),Qn(""),At(null),await Wn({keepSelection:!1}),await un(),V(`Autogen async run canceled: job=${c.job.job_id}`);return}Dt(),sr.current=window.setTimeout(()=>{co(i)},500)}catch(c){Dt(),bt(!1),fe(!1),Qn("");const f=c instanceof Error?c.message:String(c);We(`Запуск прогонов: ${f}`),V(`Autogen async status error: ${f}`)}},[un,Wn,V,Ve,Se,Dt]),nl=g.useCallback(async(i,c)=>{Dt(),bt(!0),We("");try{const f=i??Re;if(!f)throw new Error("История автогенерации пуста. Сначала сгенерируйте пачку вопросов.");const A=(c??(Re?.generation_id===f.generation_id?ve:f.questions)).map(_o=>_o.trim()).filter(_o=>_o.length>0);if(A.length===0)throw new Error("Нет вопросов для запуска: список пустой после ручного редактирования.");const ie=B.useMock==="true",nt=Gi(Je),Bt=f.mode==="saved_user_sessions",Lr=(await Te.startEvalRunAsync({connection:a,prompts:F,promptVersion:b,mode:"single-pass-strict",caseSetFile:Bt?void 0:f.saved_case_set_file??void 0,useMock:ie,evalTarget:"assistant_stage1",questions:Bt?void 0:A,scenarioQuestions:Bt?A:void 0,scenarioTitle:Bt?f.title??void 0:void 0,analysisDate:Bt?void 0:nt||void 0})).job;Qn(f.generation_id),fe(!1),At(Lr);const Sa=sa(Lr.job_id),xo=Qo(Lr,Pt);Jt(Sa),wt(xo.caseId),L(xo.detail),k(xo.dialog),V(`Запущен async-прогон job=${Lr.job_id}, run_id=${Lr.run_id}, вопросов=${A.length}`+(f.saved_case_set_file?`, base_case_set=${f.saved_case_set_file}`:"")+(Bt?", replay_mode=saved_user_session_scenario":nt?`, analysis_date=${nt}`:", analysis_date=current_state")),co(Lr.job_id)}catch(f){const w=f instanceof Error?f.message:String(f);We(`Запуск прогонов: ${w}`),V(`Autogen run error: ${w}`),bt(!1),fe(!1),Qn("")}},[Je,b,a,ve,B.useMock,V,co,F,Re,Dt]),Ts=g.useCallback(async()=>{const i=q?.job_id??"";if(!i){bt(!1),fe(!1),Qn(""),At(null),Dt();return}fe(!0),We("");try{const c=await Te.cancelEvalRunAsync(i);Dt(),At(null),bt(!1),fe(!1),Qn(""),await Wn({keepSelection:!1}),await un(),V(`Autogen async run stopped: job=${c.job.job_id}`)}catch(c){const f=c instanceof Error?c.message:String(c);fe(!1),We(`Остановка прогона: ${f}`),V(`Autogen stop error: ${f}`)}},[q,un,Wn,V,Dt]),fa=g.useCallback(i=>{if(i.role!=="assistant")return;const c=i.case_id??Ve,f=i.case_message_index??i.message_index;ot({open:!0,caseId:c,caseMessageIndex:f,messageIndex:i.message_index,rating:i.annotation?.rating??3,comment:i.annotation?.comment??"",manualCaseDecision:i.annotation?.manual_case_decision??ea,annotationAuthor:i.annotation?.annotation_author??W.generatedBy,saving:!1,error:""})},[W.generatedBy,Ve]),Rs=g.useCallback(i=>{ot(c=>c.saving&&!i?.force?c:{open:!1,caseId:"",caseMessageIndex:-1,messageIndex:-1,rating:3,comment:"",manualCaseDecision:ea,annotationAuthor:W.generatedBy,saving:!1,error:""})},[W.generatedBy]),pa=g.useCallback(async()=>{const i=Se,c=ce.caseId,f=ce.caseMessageIndex;if(!(!i||!c||f<0)){if(Js(i)){ot(w=>({...w,error:"Комментарий можно сохранить после завершения прогона."}));return}if(!ce.comment.trim()){ot(w=>({...w,error:"Добавьте комментарий."}));return}ot(w=>({...w,saving:!0,error:""}));try{await Te.saveAutoRunAnnotation({run_id:i,case_id:c,message_index:f,rating:ce.rating,comment:ce.comment.trim(),manual_case_decision:ce.manualCaseDecision,annotation_author:ce.annotationAuthor.trim()||void 0}),Rs({force:!0}),Promise.all([jn(i,Ve),ar(),wn()]).catch(w=>{const A=w instanceof Error?w.message:String(w);We(`Обновление после комментария: ${A}`),V(`Comment refresh error: ${A}`)})}catch(w){ot(A=>({...A,saving:!1,error:w instanceof Error?w.message:String(w)}))}}},[Rs,ce.annotationAuthor,ce.caseId,ce.caseMessageIndex,ce.comment,ce.manualCaseDecision,ce.rating,ar,wn,jn,V,Ve,Se]),rl=g.useCallback(i=>i.role==="assistant",[]),ma=g.useCallback(i=>i.role==="assistant"&&or.has(i.message_id),[or]),ha=g.useCallback((i,c)=>{if(i.role!=="assistant")return;const f=it.trim(),w=String(i.session_id??"").trim();if(!(f||w)){Zt("Сначала получите ответ ассистента в активной сессии.");return}!f&&w&&Oe(w);const ie=or.get(i.message_id)??null;Zt(""),It({open:!0,messageIndex:c,rating:ie?.rating??3,comment:ie?.comment??"",annotationAuthor:ie?.annotation_author??"manual_reviewer",saving:!1,error:""})},[or,it]),ga=g.useCallback(async()=>{if(he.messageIndex<0)return;if(!he.comment.trim()){It(f=>({...f,error:"Добавьте комментарий."}));return}const i=ln[he.messageIndex]??null,c=it.trim()||(i?.role==="assistant"?String(i.session_id??"").trim():"");if(!c){It(f=>({...f,error:"Сессия ассистента не найдена."}));return}It(f=>({...f,saving:!0,error:""}));try{const f=await Te.saveAssistantAnnotation({session_id:c,message_index:he.messageIndex,rating:he.rating,comment:he.comment.trim(),annotation_author:he.annotationAuthor.trim()||void 0});bn(w=>{const A=[...w],ie=A.findIndex(nt=>nt.annotation_id===f.annotation.annotation_id);return ie>=0?A[ie]=f.annotation:A.unshift(f.annotation),A.sort((nt,Bt)=>Date.parse(Bt.updated_at)-Date.parse(nt.updated_at))}),_n({force:!0})}catch(f){const w=f instanceof Error?f.message:String(f);It(A=>({...A,saving:!1,error:w}))}},[he.annotationAuthor,he.comment,he.messageIndex,he.rating,ln,it,_n]);g.useCallback(i=>{if(!Re||Re.mode!=="saved_user_sessions")return;const c=ve[i]??"";Nr({open:!0,generationId:Re.generation_id,questionIndex:i,questionText:c,saving:!1,error:""})},[ve,Re]);const va=g.useCallback(async()=>{const i=en.generationId,c=en.questionIndex;if(!i||c<0)return;const f=ve.filter((w,A)=>A!==c);if(f.length===0){Nr(w=>({...w,error:"Нельзя удалить последний вопрос из сохраненной сессии."}));return}Nr(w=>({...w,saving:!0,error:""}));try{const w=await Te.updateAutoRunAutogenQuestions({generation_id:i,questions:f});Mn(A=>A.map(ie=>ie.generation_id===i?w.generation:ie)),Mt(w.generation.questions),ct({force:!0}),V(`Обновлена сохраненная сессия: ${i}`)}catch(w){const A=w instanceof Error?w.message:String(w);Nr(ie=>({...ie,saving:!1,error:A})),V(`Saved session question delete error: ${A}`)}},[ct,ve,V,en.generationId,en.questionIndex]),qn=g.useCallback(async(i,c)=>{const f=Re?.generation_id??"",w=c?.revertQuestions??ve;if(Mt(i),!f)return!0;ys(!0);try{const A=await Te.updateAutoRunAutogenQuestions({generation_id:f,questions:i});return Mn(ie=>ie.map(nt=>nt.generation_id===f?A.generation:nt)),Mt([...A.generation.questions??[]]),c?.successLog&&V(c.successLog),!0}catch(A){const ie=A instanceof Error?A.message:String(A);return Mt(w),We(`Вопросы к запуску: ${ie}`),V(`Autogen questions update error: ${ie}`),!1}finally{ys(!1)}},[ve,V,Re]),sl=g.useCallback(i=>{er(i),on(ve[i]??"")},[ve]),Kn=g.useCallback(()=>{er(null),on("")},[]),Rr=g.useCallback(async i=>{if(i===null)return;const c=ve[i]??"",f=_r.trim();if(!f||f===c){Kn();return}const w=ve.map((ie,nt)=>nt===i?f:ie);await qn(w,{successLog:`Список вопросов обновлен: ${Re?.generation_id??"local"}`,revertQuestions:ve})&&Kn()},[ve,_r,Re,Kn,qn]),Mr=g.useCallback(()=>{Rr(st)},[Rr,st]),ol=g.useCallback(i=>{if(i.key==="Enter"){i.preventDefault(),Rr(st);return}i.key==="Escape"&&(i.preventDefault(),Kn())},[Rr,st,Kn]),fo=g.useCallback(async()=>{const i=[...ve,"Новый вопрос"],c=i.length-1;await qn(i,{successLog:`В список добавлен вопрос: ${Re?.generation_id??"local"}`,revertQuestions:ve})&&(er(c),on(i[c]))},[ve,Re,qn]),po=g.useCallback(async i=>{if(ve.length<=1){We("В списке должен остаться хотя бы один вопрос.");return}const c=ve.filter((w,A)=>A!==i);await qn(c,{successLog:`Из списка удален вопрос: ${Re?.generation_id??"local"}`,revertQuestions:ve})&&(er(w=>w===null?w:w===i?null:w>i?w-1:w),on(""))},[ve,Re,qn]),mo=g.useCallback((i,c)=>{if(Ze){i.preventDefault();return}Ft(c),Yt(c),i.dataTransfer.effectAllowed="move",i.dataTransfer.setData("text/plain",String(c))},[Ze]),ho=g.useCallback((i,c)=>{i.preventDefault(),xn!==c&&Yt(c),i.dataTransfer.dropEffect="move"},[xn]),go=g.useCallback(async(i,c)=>{i.preventDefault();const f=In;if(Yt(null),Ft(null),f===null||f===c)return;const w=[...ve],[A]=w.splice(f,1);w.splice(c,0,A),await qn(w,{successLog:`Порядок вопросов обновлен: ${Re?.generation_id??"local"}`,revertQuestions:ve})},[In,ve,Re,qn]),ir=g.useCallback(()=>{Ft(null),Yt(null)},[]),ya=g.useCallback(i=>{Rt(i),An(c=>c===i?"":i)},[]),ll=g.useCallback(i=>{ks({open:!0,generationId:i.generation_id,title:i.title??`${Wi(i.mode)} ${gn(i.created_at)}`,saving:!1,error:""})},[]),xa=g.useCallback(async()=>{const i=Lt.generationId.trim();if(i){ks(c=>({...c,saving:!0,error:""}));try{const c=await Te.deleteAutoRunAutogenHistoryRecord(i);Mn(f=>f.filter(w=>w.generation_id!==c.generation_id)),Sn({force:!0}),V(`Удален набор автопрогона: ${c.generation_id}`+(c.deleted_files.length>0?`, files=${c.deleted_files.length}`:""))}catch(c){const f=c instanceof Error?c.message:String(c);ks(w=>({...w,saving:!1,error:f})),V(`Autogen record delete error: ${f}`)}}},[Lt.generationId,Sn,V]),vo=g.useCallback(i=>{ue(c=>c.map(f=>f.annotation_id===i.annotation_id?{...f,...i}:f)),k(c=>c&&{...c,annotations:c.annotations.map(f=>f.annotation_id===i.annotation_id?i:f),messages:c.messages.map(f=>!f.annotation||f.annotation.annotation_id!==i.annotation_id?f:{...f,commented:!0,annotation:i})})},[]),yo=g.useCallback(async(i,c)=>{if(i.annotation_id){if(Js(i.run_id)){We("Статус выполнения можно менять только для завершённых прогонов.");return}Ys(i.annotation_id);try{const f=await Te.updateAutoRunAnnotation({annotation_id:i.annotation_id,resolved:c,resolved_by:W.generatedBy||void 0});vo(f.annotation),wn()}catch(f){const w=f instanceof Error?f.message:String(f);We(`Смена статуса кейса: ${w}`),V(`Annotation resolve toggle error: ${w}`)}finally{Ys("")}}},[vo,W.generatedBy,wn,V]),Ar=g.useCallback(async i=>{vn(i.annotation_id),await jn(i.run_id,i.case_id),ee?.items.some(c=>c.run_id===i.run_id)||We("Комментарий относится к прогону вне текущего фильтра. Детали загружены напрямую.")},[ee?.items,jn]);g.useEffect(()=>{Cs.current||(Cs.current=!0,Wn({keepSelection:!1}),un(),tl(),wn())},[un,tl,Wn,wn]),g.useEffect(()=>{Cs.current&&ar()},[de,ar]),g.useEffect(()=>{vn(i=>an.length===0?"":an.some(c=>c.annotation_id===i)?i:an[0].annotation_id)},[an]),g.useEffect(()=>{Rt(i=>tt.length===0?"":i&&tt.some(c=>c.generation_id===i)?i:tt[0].generation_id)},[tt]),g.useEffect(()=>{if(!Re){Mt([]),Kn(),Ft(null),Yt(null);return}Mt([...Re.questions]),Kn(),Ft(null),Yt(null)},[Re,Kn]),g.useEffect(()=>{if(st===null)return;const i=window.setTimeout(()=>{ns.current?.focus(),ns.current?.select()},0);return()=>window.clearTimeout(i)},[st]),g.useEffect(()=>{if(!qe){An("");return}zt&&!tt.some(i=>i.generation_id===zt)&&An("")},[zt,qe,tt]),g.useEffect(()=>{kr(String(B.limit))},[B.limit]),g.useEffect(()=>{Bn(String(W.count))},[W.count]),g.useEffect(()=>{if(!it.trim()){bn([]);return}lr(it)},[it,lr]),g.useEffect(()=>{if(!q)return;const i=sa(q.job_id);if(Se!==i)return;const c=Qo(q,Ve||Pt);L(c.detail),k(c.dialog),wt(c.caseId)},[q,Ve,Se]),g.useEffect(()=>()=>{Dt()},[Dt]),g.useEffect(()=>{Tt.length!==0&&vt(i=>{let c=!1;const f={...i.personalityPrompts};for(const A of Tt)(typeof f[A.id]!="string"||f[A.id].trim().length===0)&&(f[A.id]=A.defaultPrompt,c=!0);let w=i.personalityId;return Tt.some(A=>A.id===i.personalityId)||(w=Tt[0].id,c=!0),c?{...i,personalityId:w,personalityPrompts:f}:i})},[Tt]),g.useEffect(()=>{const i=localStorage.getItem(id);if(i)try{const c=JSON.parse(i);if(c.filters){const f=c.filters;Ie(w=>({...w,...f,limit:typeof f.limit=="number"?Math.max(1,Math.min(500,f.limit)):w.limit}))}typeof c.analysisDate=="string"&&re(Gi(c.analysisDate)),typeof c.autogenPersonalityPromptHeight=="number"&&js(dd(c.autogenPersonalityPromptHeight)),c.groupsExpanded&&(typeof c.groupsExpanded.filters=="boolean"&&no(c.groupsExpanded.filters),typeof c.groupsExpanded.generationContext=="boolean"&&ro(c.groupsExpanded.generationContext),typeof c.groupsExpanded.autogen=="boolean"&&so(c.groupsExpanded.autogen),typeof c.groupsExpanded.savedSessions=="boolean"&&ts(c.groupsExpanded.savedSessions)),c.autoGenSettings&&vt(f=>{const w={...f.personalityPrompts},A=c.autoGenSettings?.personalityPrompts??{};for(const[nt,Bt]of Object.entries(A))typeof Bt=="string"&&nt.trim().length>0&&(w[nt.trim()]=Bt);const ie=typeof c.autoGenSettings?.personalityId=="string"&&c.autoGenSettings.personalityId.trim().length>0?c.autoGenSettings.personalityId.trim():f.personalityId;return{...f,mode:c.autoGenSettings?.mode==="codex_creative"||c.autoGenSettings?.mode==="qwen_seed"||c.autoGenSettings?.mode==="saved_user_sessions"?c.autoGenSettings.mode:f.mode,count:typeof c.autoGenSettings?.count=="number"?Math.max(1,Math.min(200,c.autoGenSettings.count)):f.count,personalityId:ie,personalityPrompts:w,persistToEvalCases:typeof c.autoGenSettings?.persistToEvalCases=="boolean"?c.autoGenSettings.persistToEvalCases:f.persistToEvalCases,generatedBy:typeof c.autoGenSettings?.generatedBy=="string"?c.autoGenSettings.generatedBy:f.generatedBy}}),(c.annotationDecisionFilter==="all"||typeof c.annotationDecisionFilter=="string"&&c.annotationDecisionFilter.length>0)&&xe(c.annotationDecisionFilter),typeof c.hideResolvedAnnotations=="boolean"&&je(c.hideResolvedAnnotations)}catch{}},[]);const Ir=g.useCallback(()=>{const i={filters:B,analysisDate:Je,autogenPersonalityPromptHeight:ws,groupsExpanded:{filters:nr,generationContext:Hn,autogen:rr,savedSessions:Vn},autoGenSettings:{mode:W.mode,count:W.count,personalityId:W.personalityId,personalityPrompts:W.personalityPrompts,persistToEvalCases:W.persistToEvalCases,generatedBy:W.generatedBy},annotationDecisionFilter:de,hideResolvedAnnotations:we};localStorage.setItem(id,JSON.stringify(i))},[Je,de,W,rr,ws,B,nr,Hn,we,Vn]);return g.useEffect(()=>{const i=()=>{Ir(),V("Сохранены настройки панели автопрогонов.")};return window.addEventListener(ud,i),()=>{window.removeEventListener(ud,i)}},[V,Ir]),o.jsxs(aa,{className:"autoruns-frame",title:"",hideHeader:!0,children:[o.jsxs("div",{className:"autoruns-columns",children:[Ee?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(op,{embedded:!0,value:a,modelOptions:m,modelsBusy:p,onChange:M,onReloadModels:z,onSaveLocalConfig:ae,onTestConnection:ne,lastStatus:S,busy:C}),o.jsx(lp,{embedded:!0,value:F,onChange:X,presets:Z,selectedPresetId:te,onSelectPreset:Me,onLoadPreset:pe,onSavePreset:me,onResetDefaults:ye,onDiffPrevious:Qe,presetName:Xe,onPresetNameChange:He,diffSummary:Ae})]})]}):null,be?o.jsxs("section",{className:"autoruns-col",children:[o.jsx("div",{className:"autoruns-col-header",children:o.jsx("h3",{children:"Автопрогоны"})}),o.jsxs("div",{className:"autoruns-group-heading",children:[o.jsx("h4",{children:"Настройки выборки"}),o.jsx("button",{type:"button",className:"autoruns-group-toggle",onClick:()=>no(i=>!i),"aria-label":nr?"Скрыть группу настройки выборки":"Показать группу настройки выборки",title:nr?"Скрыть группу":"Показать группу",children:o.jsx(ra,{expanded:nr})})]}),nr?o.jsxs(o.Fragment,{children:[o.jsxs("div",{className:"autoruns-form-grid",children:[o.jsxs("label",{children:["Дата с",o.jsx("input",{type:"datetime-local",value:B.fromLocal,onChange:i=>Ie(c=>({...c,fromLocal:i.target.value}))})]}),o.jsxs("label",{children:["Дата по",o.jsx("input",{type:"datetime-local",value:B.toLocal,onChange:i=>Ie(c=>({...c,toLocal:i.target.value}))})]}),o.jsxs("label",{children:["Целевой контур",o.jsxs("select",{value:B.target,onChange:i=>Ie(c=>({...c,target:i.target.value})),children:[o.jsx("option",{value:"all",children:"все"}),(ee?.available.targets??[]).map(i=>o.jsx("option",{value:i,children:i},i))]})]}),o.jsxs("label",{children:["Режим",o.jsxs("select",{value:B.mode,onChange:i=>Ie(c=>({...c,mode:i.target.value})),children:[o.jsx("option",{value:"all",children:"все"}),(ee?.available.modes??[]).map(i=>o.jsx("option",{value:i,children:i},i))]})]}),o.jsxs("label",{children:["Использовать mock",o.jsxs("select",{value:B.useMock,onChange:i=>Ie(c=>({...c,useMock:i.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:Ss,onChange:i=>{const c=i.target.value;(c===""||/^\d+$/.test(c))&&kr(c)},onBlur:i=>uo(i.target.value),onKeyDown:i=>{i.key==="Enter"&&uo(i.target.value)}})]}),o.jsxs("label",{className:"full-width",children:["Версия промпта содержит",o.jsx("input",{value:B.promptContains,onChange:i=>Ie(c=>({...c,promptContains:i.target.value})),placeholder:"normalizer_v2_0_2 / address_query_runtime_v1",list:"autoruns-prompt-versions"})]})]}),o.jsx("datalist",{id:"autoruns-prompt-versions",children:(ee?.available.prompt_versions??[]).map(i=>o.jsx("option",{value:i},i))}),o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",disabled:E,onClick:()=>{Wn({keepSelection:!1})},children:E?"Обновляю...":"Применить"}),o.jsx("button",{type:"button",className:"tab",onClick:()=>{Ie({...Hi,fromLocal:fd()}),We("")},children:"Сбросить фильтры"})]})]}):null,o.jsxs("div",{className:"autoruns-group-heading",children:[o.jsx("h4",{children:"Контур генерации"}),o.jsx("button",{type:"button",className:"autoruns-group-toggle",onClick:()=>ro(i=>!i),"aria-label":Hn?"Скрыть группу контура генерации":"Показать группу контура генерации",title:Hn?"Скрыть группу":"Показать группу",children:o.jsx(ra,{expanded:Hn})})]}),Hn?o.jsxs("div",{className:"autoruns-meta-list",children:[o.jsxs("div",{children:[o.jsx("span",{children:"Провайдер:"}),o.jsx("strong",{children:a.llmProvider})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Модель:"}),o.jsx("strong",{children:a.model||"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Промпт ассистента:"}),o.jsx("strong",{children:b})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Промпт декомпозиции:"}),o.jsx("strong",{children:le})]})]}):null,o.jsxs("div",{className:"autoruns-group-heading",children:[o.jsx("h4",{children:"Автопрогоны"}),o.jsx("button",{type:"button",className:"autoruns-group-toggle",onClick:()=>so(i=>!i),"aria-label":rr?"Скрыть группу автопрогонов":"Показать группу автопрогонов",title:rr?"Скрыть группу":"Показать группу",children:o.jsx(ra,{expanded:rr})})]}),rr?o.jsxs(o.Fragment,{children:[o.jsxs("div",{className:"autoruns-form-grid",children:[o.jsxs("label",{children:["Режимы",o.jsxs("select",{value:W.mode,onChange:i=>vt(c=>({...c,mode:i.target.value})),children:[o.jsx("option",{value:"codex_creative",children:"codex_creative"}),o.jsx("option",{value:"qwen_seed",children:"qwen_seed"}),o.jsx("option",{value:"saved_user_sessions",children:"Пользовательские сессии"})]})]}),qe?null:o.jsxs(o.Fragment,{children:[o.jsxs("label",{children:["Кол-во",o.jsx("input",{type:"number",min:1,max:200,value:ua,onChange:i=>{const c=i.target.value;(c===""||/^\d+$/.test(c))&&Bn(c)},onBlur:i=>Xo(i.target.value),onKeyDown:i=>{i.key==="Enter"&&Xo(i.target.value)}})]}),o.jsxs("label",{children:["Личность автогенерации",o.jsx("select",{value:W.personalityId,onChange:i=>vt(c=>({...c,personalityId:i.target.value})),children:Tt.map(i=>o.jsx("option",{value:i.id,children:i.label},i.id))})]}),o.jsxs("label",{children:["Кто генерирует",o.jsx("input",{value:W.generatedBy,onChange:i=>vt(c=>({...c,generatedBy:i.target.value})),placeholder:"manual_reviewer"})]}),o.jsxs("label",{className:"full-width",children:["Промпт личности",o.jsx("textarea",{className:"autoruns-personality-prompt",value:W.personalityPrompts[W.personalityId]??"",onChange:i=>vt(c=>({...c,personalityPrompts:{...c.personalityPrompts,[c.personalityId]:i.target.value}})),placeholder:"Текст промпта для выбранной личности автогенерации",style:{height:`${ws}px`},onMouseUp:el,onTouchEnd:el})]}),o.jsxs("label",{className:"checkbox-row",children:[o.jsx("input",{type:"checkbox",checked:W.persistToEvalCases,onChange:i=>vt(c=>({...c,persistToEvalCases:i.target.checked}))}),"Сохранять кейс-сет в `eval_cases`"]})]})]}),qe?null:o.jsxs("div",{className:"autoruns-form-grid",children:[o.jsxs("label",{children:["Дата анализа (срез)",o.jsx("input",{type:"date",value:Je,onChange:i=>re(Gi(i.target.value))})]}),o.jsx("div",{className:"button-row",children:o.jsx("button",{type:"button",className:"tab",disabled:!Je,onClick:()=>re(""),children:"Сбросить дату среза"})})]}),o.jsxs("div",{className:"button-row",children:[qe?null:o.jsxs(o.Fragment,{children:[o.jsx("button",{type:"button",disabled:Jr,onClick:()=>{da()},children:Jr?"Генерирую...":"Сгенерировать пачку"}),o.jsx("button",{type:"button",className:"tab",disabled:$n,onClick:()=>{un()},children:$n?"Обновляю...":"Обновить историю"})]}),o.jsx("button",{type:"button",className:"autoruns-run-launch-btn",style:qe?{display:"none"}:void 0,disabled:es||!Ln&&(ve.length===0||!Re),onClick:()=>{Ln?Ts():nl()},children:Ln?es?"Останавливаю...":"Остановить прогон":"Запустить прогон"})]}),o.jsx("div",{className:"autoruns-form-grid",children:o.jsxs("label",{className:"full-width",children:[qe?"Сохраненная сессия":"Кейс-сет для запуска",o.jsxs("select",{value:Ge,onChange:i=>Rt(i.target.value),disabled:tt.length===0,children:[tt.length===0?o.jsx("option",{value:"",children:qe?"нет сохраненных сессий":"нет генераций"}):null,tt.map(i=>o.jsxs("option",{value:i.generation_id,children:[gn(i.created_at)," | ",qi(i)??Wi(i.mode)," | ",i.count]},i.generation_id))]})]})}),o.jsxs(o.Fragment,{children:[o.jsxs("div",{className:"autoruns-generated-questions",style:qe?{display:"none"}:void 0,children:[o.jsx("div",{className:"autoruns-generated-questions-head",children:o.jsxs("strong",{children:["Вопросы к запуску: ",ve.length]})}),ve.length===0?o.jsx("p",{className:"muted",children:qe?"Список вопросов пуст. Сначала сохраните живую пользовательскую сессию.":"Список вопросов пуст. Сгенерируйте пачку или добавьте вопрос вручную."}):o.jsx("div",{className:"autoruns-generated-questions-list",children:ve.map((i,c)=>o.jsxs("div",{className:["autoruns-generated-question-item",xn===c?"drag-over":"",In===c?"dragging":"",st===c?"editing":""].filter(Boolean).join(" "),onDragOver:f=>ho(f,c),onDrop:f=>{go(f,c)},children:[o.jsx("button",{type:"button",className:"autoruns-question-grip-btn",draggable:!Ze&&st!==c,disabled:Ze||st===c,onDragStart:f=>mo(f,c),onDragEnd:ir,title:"Перетащить вопрос","aria-label":`Перетащить вопрос ${c+1}`,children:o.jsx(xd,{})}),st===c?o.jsxs(o.Fragment,{children:[o.jsx("input",{ref:ns,className:"autoruns-generated-question-input",value:_r,onChange:f=>on(f.target.value),onBlur:Mr,onKeyDown:ol,placeholder:"Текст вопроса",disabled:Ze}),o.jsx("button",{type:"button",className:"autoruns-remove-question-btn",onMouseDown:f=>f.preventDefault(),onClick:()=>{po(c)},title:"Удалить вопрос","aria-label":`Удалить вопрос ${c+1}`,disabled:Ze,children:"×"})]}):o.jsxs("button",{type:"button",className:"autoruns-generated-question-text",onDoubleClick:()=>sl(c),title:"Двойной клик для редактирования",children:[c+1,". ",i]})]},`${c}-${i.slice(0,24)}`))}),o.jsx("button",{type:"button",className:"autoruns-add-question-btn",onClick:()=>{fo()},disabled:!Re||Ze,children:"+"})]}),qe?null:o.jsx("p",{className:"muted",children:"Запуск выполняет `assistant_stage1` eval по выбранному кейс-сету."})]})]}):null,o.jsxs("div",{className:"autoruns-group-heading",children:[o.jsx("h4",{children:qe?"Сохраненные пользовательские сессии":"История автогенераций"}),o.jsx("button",{type:"button",className:"autoruns-group-toggle",onClick:()=>ts(i=>!i),"aria-label":Vn?qe?"Скрыть группу сохраненных пользовательских сессий":"Скрыть группу истории автогенераций":qe?"Показать группу сохраненных пользовательских сессий":"Показать группу истории автогенераций",title:Vn?"Скрыть группу":"Показать группу",children:o.jsx(ra,{expanded:Vn})})]}),Vn?o.jsxs("div",{className:"autoruns-autogen-list",children:[$n?o.jsx("p",{className:"muted",children:qe?"Загружаю сохраненные пользовательские сессии...":"Загружаю историю автогенераций..."}):null,!$n&&tt.length===0?o.jsx("p",{className:"muted",children:qe?"Сохраненные пользовательские сессии пока пусты.":"История автогенераций пока пустая."}):null,tt.slice(0,30).map(i=>{const c=Ln&&Cr===i.generation_id,f=Ln&&Cr!==i.generation_id;return o.jsxs("article",{className:["autoruns-autogen-item",Ge===i.generation_id?"selected":"",zt===i.generation_id?"expanded":"",qe?"saved-session":""].filter(Boolean).join(" "),onClick:qe?void 0:()=>Rt(i.generation_id),children:[qe?o.jsxs("div",{className:"autoruns-saved-session-topbar",children:[o.jsx("button",{type:"button",className:"autoruns-saved-session-icon-btn",disabled:es||f,onClick:w=>{if(w.preventDefault(),w.stopPropagation(),c){Ts();return}Rt(i.generation_id),nl(i,Ge===i.generation_id?ve:i.questions)},title:c?"Остановить прогон":"Запустить прогон","aria-label":`${c?"Остановить прогон":"Запустить прогон"} для ${qi(i)}`,children:c?o.jsx(gp,{}):o.jsx(hp,{})}),o.jsx("button",{type:"button",className:"autoruns-autogen-delete-btn",onClick:w=>{w.preventDefault(),w.stopPropagation(),ll(i)},title:"Удалить сохраненный набор","aria-label":`Удалить набор ${i.generation_id}`,children:"×"})]}):null,o.jsxs("header",{children:[o.jsx("strong",{children:qi(i)}),o.jsx("div",{className:"autoruns-autogen-card-actions",children:o.jsx("button",{type:"button",className:"autoruns-autogen-delete-btn",onClick:w=>{w.preventDefault(),w.stopPropagation(),ll(i)},title:"Удалить сохраненный набор","aria-label":`Удалить набор ${i.generation_id}`,children:"×"})})]}),o.jsxs("div",{className:"autoruns-saved-session-meta",children:[o.jsx("strong",{className:"autoruns-saved-session-meta-label",children:"Дата и время"}),o.jsxs("span",{className:"autoruns-saved-session-meta-value",children:["| ",gn(i.created_at)]})]}),o.jsxs("div",{className:"autoruns-saved-session-meta autoruns-saved-session-meta-id",children:[o.jsxs("div",{className:"autoruns-saved-session-meta-id-main",children:[o.jsx("strong",{className:"autoruns-saved-session-meta-label",children:"ID"}),o.jsxs("span",{className:"autoruns-saved-session-meta-value",children:["| ",i.generation_id]})]}),o.jsx("span",{role:"button",tabIndex:0,className:"autoruns-copy-run-id-btn",onClick:w=>{Pr(w,i.generation_id,"set id")},onKeyDown:w=>{(w.key==="Enter"||w.key===" ")&&(w.preventDefault(),Pr(w,i.generation_id,"set id"))},title:"Скопировать id набора","aria-label":`Скопировать id набора ${i.generation_id}`,children:o.jsx(yd,{})})]}),o.jsxs("div",{className:"autoruns-saved-session-meta",children:[o.jsx("strong",{className:"autoruns-saved-session-meta-label",children:"Режим"}),o.jsxs("span",{className:"autoruns-saved-session-meta-value",children:["| ",Wi(i.mode)]})]}),o.jsxs("div",{className:"autoruns-saved-session-meta",children:[o.jsx("strong",{className:"autoruns-saved-session-meta-label",children:"Тип"}),o.jsxs("span",{className:"autoruns-saved-session-meta-value",children:["| ",Pd(i)?"АГЕНТНЫЙ ПРОГОН":"АВТОПРОГОН"]})]}),qe?o.jsxs(o.Fragment,{children:[o.jsxs("div",{className:"autoruns-saved-session-meta",children:[o.jsx("strong",{className:"autoruns-saved-session-meta-label",children:"Количество вопросов"}),o.jsxs("span",{className:"autoruns-saved-session-meta-value",children:["| ",i.questions.length]})]}),o.jsx("div",{className:"autoruns-saved-session-footer",children:o.jsx("button",{type:"button",className:"autoruns-saved-session-icon-btn",onClick:w=>{w.preventDefault(),w.stopPropagation(),ya(i.generation_id)},title:zt===i.generation_id?"Скрыть вопросы":"Показать вопросы","aria-label":zt===i.generation_id?"Скрыть вопросы":"Показать вопросы",children:o.jsx(mp,{expanded:zt===i.generation_id})})}),o.jsx("div",{className:zt===i.generation_id?"autoruns-saved-session-questions expanded":"autoruns-saved-session-questions",children:o.jsxs("div",{className:"autoruns-generated-questions autoruns-generated-questions-embedded",children:[o.jsx("div",{className:"autoruns-generated-questions-head",children:o.jsxs("strong",{children:["Вопросы к запуску:"," ",Ge===i.generation_id?ve.length:i.questions.length]})}),(Ge===i.generation_id?ve:i.questions).length===0?o.jsx("p",{className:"muted",children:"Список вопросов пуст."}):o.jsx("div",{className:"autoruns-generated-questions-list",children:(Ge===i.generation_id?ve:i.questions).map((w,A)=>o.jsxs("div",{className:["autoruns-generated-question-item",xn===A&&Ge===i.generation_id?"drag-over":"",In===A&&Ge===i.generation_id?"dragging":"",st===A&&Ge===i.generation_id?"editing":""].filter(Boolean).join(" "),onDragOver:ie=>Ge===i.generation_id?ho(ie,A):void 0,onDrop:ie=>Ge===i.generation_id?void go(ie,A):void 0,children:[o.jsx("button",{type:"button",className:"autoruns-question-grip-btn",draggable:Ge===i.generation_id&&!Ze&&st!==A,disabled:Ge!==i.generation_id||Ze||st===A,onDragStart:ie=>{Rt(i.generation_id),mo(ie,A)},onDragEnd:ir,title:"Перетащить вопрос","aria-label":`Перетащить вопрос ${A+1}`,children:o.jsx(xd,{})}),Ge===i.generation_id&&st===A?o.jsxs(o.Fragment,{children:[o.jsx("input",{ref:ns,className:"autoruns-generated-question-input",value:_r,onChange:ie=>on(ie.target.value),onBlur:Mr,onKeyDown:ol,placeholder:"Текст вопроса",disabled:Ze}),o.jsx("button",{type:"button",className:"autoruns-remove-question-btn",onMouseDown:ie=>ie.preventDefault(),onClick:()=>{po(A)},title:"Удалить вопрос","aria-label":`Удалить вопрос ${A+1}`,disabled:Ze,children:"×"})]}):o.jsxs("button",{type:"button",className:"autoruns-generated-question-text",onDoubleClick:()=>{Rt(i.generation_id),sl(A)},title:"Двойной клик для редактирования",children:[A+1,". ",w]})]},`${i.generation_id}-${A}-${w.slice(0,24)}`))}),o.jsx("button",{type:"button",className:"autoruns-add-question-btn",onClick:()=>{Rt(i.generation_id),fo()},disabled:Ge!==i.generation_id||Ze,children:"+"})]})})]}):null]},i.generation_id)})]}):null,o.jsxs("details",{className:"autoruns-prompt-details",children:[o.jsx("summary",{children:"Копия активного промпта (только чтение)"}),o.jsxs("label",{children:["Системный",o.jsx("textarea",{readOnly:!0,value:F.systemPrompt})]}),o.jsxs("label",{children:["Разработчика",o.jsx("textarea",{readOnly:!0,value:F.developerPrompt})]}),o.jsxs("label",{children:["Доменный",o.jsx("textarea",{readOnly:!0,value:F.domainPrompt})]}),o.jsxs("label",{children:["Заметки по схеме",o.jsx("textarea",{readOnly:!0,value:F.schemaNotes})]}),o.jsxs("label",{children:["Примеры few-shot",o.jsx("textarea",{readOnly:!0,value:F.fewShotExamples})]})]}),Zs?o.jsx("p",{className:"error-text",children:Zs}):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:(ee?.stats.runs_total??0)+(q?1:0)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Средний score"}),o.jsx("strong",{children:na(ee?.stats.avg_score_index??null)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Тренд"}),o.jsx("strong",{children:ee?md(ee.stats.trend):"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Блокеры"}),o.jsx("strong",{children:ee?.stats.blocking_runs??0})]})]}),o.jsxs("div",{className:"autoruns-run-list",children:[Er.map(i=>o.jsxs("button",{type:"button",className:Se===i.run_id?"autoruns-run-item selected":"autoruns-run-item",onClick:()=>{jn(i.run_id)},children:[o.jsxs("div",{className:"autoruns-run-head",children:[o.jsx("strong",{children:gn(i.run_timestamp)}),o.jsx("span",{children:fp(i.eval_target)})]}),o.jsxs("div",{className:"autoruns-run-meta autoruns-run-id-row",children:[o.jsx("span",{children:i.run_id}),o.jsx("span",{role:"button",tabIndex:0,className:"autoruns-copy-run-id-btn",onClick:c=>{Pr(c,i.run_id,"run id")},onKeyDown:c=>{(c.key==="Enter"||c.key===" ")&&(c.preventDefault(),Pr(c,i.run_id,"run id"))},title:"Скопировать run id","aria-label":`Скопировать run id ${i.run_id}`,children:o.jsx(yd,{})})]}),o.jsxs("div",{className:"autoruns-run-meta",children:["режим=",i.mode??"нет данных"," | mock=",String(i.use_mock)]}),o.jsxs("div",{className:"autoruns-run-meta",children:["analysis_date=",i.analysis_date??"current_state"]}),i.llm_provider||i.model?o.jsxs("div",{className:"autoruns-run-meta",children:["llm=",i.llm_provider??"нет данных"," | модель=",i.model??"нет данных"]}):null,o.jsxs("div",{className:"autoruns-run-meta",children:["промпт=",i.prompt_version??"нет данных"]}),o.jsxs("div",{className:"autoruns-run-foot",children:[o.jsxs("span",{children:["оценка: ",na(i.score_index)]}),o.jsxs("span",{children:["закрыто/открыто: ",i.closed_cases,"/",i.open_cases]})]}),o.jsxs("div",{className:"autoruns-run-foot",children:[o.jsxs("span",{children:["блокеры: ",i.blocking_failures]}),o.jsxs("span",{children:["качество: ",i.quality_failures]})]})]},i.run_id)),Er.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:Se,onChange:i=>{const c=i.target.value;jn(c)},children:Er.map(i=>o.jsxs("option",{value:i.run_id,children:[gn(i.run_timestamp)," | ",i.run_id]},i.run_id))})]}),o.jsxs("label",{children:["Кейс",o.jsxs("select",{value:Ve,onChange:i=>{const c=i.target.value;wt(c),Se&&c&&os(Se,c)},children:[(O?.cases.length??0)>0?o.jsx("option",{value:Pt,children:"ВСЕ кейсы подряд"}):null,(O?.cases??[]).map(i=>o.jsxs("option",{value:i.case_id,children:[i.case_id," | ",i.status]},i.case_id))]})]}),o.jsxs("div",{className:"autoruns-dialog-copy-actions",children:[o.jsx("button",{type:"button",className:"assistant-copy-btn",onClick:()=>{Tr("default")},disabled:et||Q||(y?.messages.length??0)===0,title:"Скопировать question-answer диалог текущего прогона",children:"Скопировать чат"}),o.jsx("button",{type:"button",className:"assistant-copy-btn",onClick:()=>{Tr("technical")},disabled:et||Q||(y?.messages.length??0)===0,title:"Скопировать диалог вместе с debug JSON и метаданными прогона",children:"Скопировать техчат"}),o.jsxs("div",{className:"autoruns-dialog-copy-status",children:[Go==="success"?o.jsxs("span",{className:"assistant-copy-feedback success",children:["Скопировано (",Wo,")"]}):null,Go==="error"?o.jsx("span",{className:"assistant-copy-feedback error",children:"Ошибка копирования"}):null]})]})]})]}),o.jsxs("div",{className:"autoruns-case-list",children:[(O?.cases.length??0)>0?o.jsxs("button",{type:"button",className:Ve===Pt?"autoruns-case-item selected":"autoruns-case-item",onClick:()=>{wt(Pt),Se&&os(Se,Pt)},children:[o.jsx("span",{children:"ВСЕ кейсы подряд"}),o.jsx("span",{children:O?.cases.length})]},Pt):null,(O?.cases??[]).map(i=>o.jsxs("button",{type:"button",className:Ve===i.case_id?"autoruns-case-item selected":"autoruns-case-item",onClick:()=>{wt(i.case_id),Se&&os(Se,i.case_id)},children:[o.jsx("span",{children:i.case_id}),o.jsxs("span",{children:[i.status,i.commented_count>0?` | комм=${i.commented_count}`:""]})]},i.case_id))]}),o.jsxs("div",{className:"autoruns-dialog-view",children:[et||Q?o.jsx("p",{className:"muted",children:"Загружаю диалог..."}):null,!et&&!Q&&(y?.messages.length??0)===0?o.jsx("p",{className:"muted",children:"Диалог для этого прогона не найден."}):null,(y?.messages??[]).map((i,c)=>{const f=i.role==="assistant"?"assistant":"user";return o.jsxs("article",{className:`autoruns-msg ${f}`,children:[o.jsxs("header",{children:[o.jsx("strong",{children:f==="assistant"?"Система":"Модель/вопрос"}),o.jsxs("div",{className:"autoruns-msg-head-actions",children:[i.case_id?o.jsx("span",{className:"autoruns-msg-case-tag",children:i.case_id}):null,pd(i)?o.jsx("span",{className:"autoruns-msg-case-tag",children:pd(i)}):null,o.jsx("span",{children:i.created_at?gn(i.created_at):"нет данных"}),f==="assistant"&&!Js(Se)?o.jsxs(o.Fragment,{children:[o.jsx("button",{type:"button",className:i.commented?"autoruns-comment-icon commented":"autoruns-comment-icon",onClick:()=>fa(i),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(pp,{commented:i.commented})}),i.annotation?o.jsx("button",{type:"button",className:i.annotation.resolved?"autoruns-resolve-toggle resolved":"autoruns-resolve-toggle",onClick:()=>{yo(i.annotation,!i.annotation.resolved)},disabled:Fn===i.annotation.annotation_id,title:i.annotation.resolved?"Отметить кейс как невыполненный":"Отметить кейс как выполненный","aria-label":i.annotation.resolved?"Отметить кейс как невыполненный":"Отметить кейс как выполненный",children:o.jsx(vd,{resolved:i.annotation.resolved})}):null]}):null]})]}),o.jsx("p",{children:i.text}),f==="assistant"&&i.annotation?o.jsxs("div",{className:"autoruns-msg-annotation",children:[o.jsx("strong",{children:Ki(i.annotation.rating)}),o.jsx("span",{children:i.annotation.comment}),o.jsxs("span",{className:"muted",children:[i.annotation.manual_case_decision,i.annotation.annotation_author?` | ${i.annotation.annotation_author}`:""]})]}):null,(i.trace_id||i.reply_type)&&o.jsxs("footer",{children:[i.trace_id?o.jsxs("span",{children:["trace=",i.trace_id]}):null,i.reply_type?o.jsxs("span",{children:["reply_type=",i.reply_type]}):null]})]},i.message_id??`${f}-${c}`)})]})]}),St?o.jsx("div",{className:"autoruns-col autoruns-assistant-live-col",children:o.jsx(tp,{sessionId:it,conversation:ln,inputValue:eo,onInputChange:xs,selectedContextChip:jr,onSelectContextChip:Ut,onClearContextChip:()=>Ut(null),useMock:to,onUseMockChange:Xt,onSend:io,onClear:ao,onSaveSession:ss,busy:Ko,saveBusy:yt.saving,saveDisabled:!it.trim()||ln.length===0||Ko,statusText:ia,errorMessage:_s,showSaveAction:!0,showCommentAction:!0,onCommentAssistantMessage:ha,isAssistantMessageCommented:ma,canCommentAssistantMessage:rl})}):null,mt?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:na(ee?.stats.latest_score_index??null)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Предыдущий"}),o.jsx("strong",{children:na(ee?.stats.previous_score_index??null)})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Тренд"}),o.jsx("strong",{children:ee?md(ee.stats.trend):"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Пробелы качества"}),o.jsx("strong",{children:ee?.stats.quality_gap_runs??0})]})]}),o.jsx("h4",{children:"Покрытие доменов (история)"}),hd(ee?.stats.domain_coverage??[]),o.jsx("h4",{style:{marginTop:14},children:"Покрытие доменов (выбранный прогон)"}),hd(O?.coverage.domain_coverage??[]),o.jsx("h4",{style:{marginTop:14},children:"Очереди фиксов пост-анализа"}),Dn?o.jsx("p",{className:"muted",children:"Собираю пост-анализ..."}):null,Dn?null:o.jsx("div",{className:"autoruns-stats-grid",children:Object.entries(tr?.post_analysis.stats.by_queue??{}).map(([i,c])=>o.jsxs("div",{children:[o.jsx("span",{children:i}),o.jsx("strong",{children:c})]},i))}),o.jsxs("div",{className:"autoruns-autogen-list",children:[(tr?.post_analysis.recommended_regression_candidates??[]).slice(0,12).map(i=>o.jsxs("article",{className:"autoruns-autogen-item",children:[o.jsxs("header",{children:[o.jsx("strong",{children:i.manual_case_decision}),o.jsxs("span",{children:[i.rating,"/5"]})]}),o.jsxs("div",{className:"autoruns-run-meta",children:[i.domain??"неизвестно"," / ",i.query_class??"неизвестно"]}),o.jsx("p",{children:i.comment})]},i.annotation_id)),!Dn&&(tr?.post_analysis.recommended_regression_candidates.length??0)===0?o.jsx("p",{className:"muted",children:"Рекомендованных кандидатов пока нет."}):null]})]}):null,Kt?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:de,onChange:i=>xe(i.target.value),children:[o.jsx("option",{value:"all",children:"все"}),(sn.length>0?sn:Ce?.enum??[]).map(i=>o.jsx("option",{value:i,children:String(Ce?.labels?.[i]??i)},i))]})]}),o.jsx("button",{type:"button",className:"tab autoruns-resolved-filter-toggle",onClick:()=>je(i=>!i),children:we?"Показать выполненные":"Скрыть выполненные"})]}),o.jsxs("div",{className:"autoruns-stats-grid",children:[o.jsxs("div",{children:[o.jsx("span",{children:"Комментариев"}),o.jsx("strong",{children:ut.length})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Средний рейтинг"}),o.jsx("strong",{children:Es===null?"нет данных":`${Es.toFixed(2)} / 5`})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Последний"}),o.jsx("strong",{children:ut.length>0?gn(ut[0].updated_at):"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"Статус"}),o.jsx("strong",{children:Xr?"обновляю":"готово"})]})]}),o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",disabled:Xr,onClick:()=>{ar()},children:Xr?"Обновляю...":"Обновить список"}),o.jsx("button",{type:"button",className:"tab",disabled:Dn,onClick:()=>{wn()},children:Dn?"Идет пост-анализ...":"Обновить пост-анализ"})]}),o.jsxs("div",{className:"autoruns-comments-list",children:[Xr?o.jsx("p",{className:"muted",children:"Загружаю комментарии..."}):null,!Xr&&ut.length===0?o.jsx("p",{className:"muted",children:se.length===0&&wr.length===0?"Пока нет откомментированных ответов.":"Нет открытых кейсов по текущему фильтру."}):null,ut.map(i=>{if(i.source==="assistant_live"){const f=i.assistant;return o.jsxs("article",{className:"autoruns-comment-item",children:[o.jsxs("div",{className:"autoruns-comment-head",children:[o.jsx("strong",{children:Ki(f.rating)}),o.jsx("div",{className:"autoruns-comment-head-actions",children:o.jsx("span",{children:gn(f.updated_at)})})]}),o.jsxs("div",{className:"autoruns-run-meta",children:["live-session: ",f.session_id]}),o.jsxs("div",{className:"autoruns-run-meta",children:["msg=",f.message_index]}),o.jsxs("div",{className:"autoruns-run-meta",children:["source=assistant_live",f.annotation_author?` | author=${f.annotation_author}`:""]}),f.context.question_text?o.jsxs("p",{children:["Q: ",f.context.question_text]}):null,f.context.answer_text?o.jsxs("p",{children:["A: ",f.context.answer_text]}):null,o.jsx("p",{children:f.comment})]},i.key)}const c=i.autorun;return o.jsxs("article",{className:Rn===c.annotation_id?"autoruns-comment-item selected":"autoruns-comment-item",onClick:()=>{Ar(c)},role:"button",tabIndex:0,onKeyDown:f=>{(f.key==="Enter"||f.key===" ")&&(f.preventDefault(),Ar(c))},children:[o.jsxs("div",{className:"autoruns-comment-head",children:[o.jsx("strong",{children:Ki(c.rating)}),o.jsxs("div",{className:"autoruns-comment-head-actions",children:[o.jsx("span",{children:gn(c.updated_at)}),o.jsx("button",{type:"button",className:c.resolved?"autoruns-resolve-toggle resolved":"autoruns-resolve-toggle",onClick:f=>{f.preventDefault(),f.stopPropagation(),yo(c,!c.resolved)},disabled:Fn===c.annotation_id,title:c.resolved?"Отметить кейс как невыполненный":"Отметить кейс как выполненный","aria-label":c.resolved?"Отметить кейс как невыполненный":"Отметить кейс как выполненный",children:o.jsx(vd,{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:["выполнено",": ",gn(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})]},i.key)})]}),Le?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:Le.technical_context.trace_id??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"reply_type:"}),o.jsx("strong",{children:Le.technical_context.reply_type??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"domain:"}),o.jsx("strong",{children:Le.technical_context.domain??"нет данных"})]}),o.jsxs("div",{children:[o.jsx("span",{children:"query_class:"}),o.jsx("strong",{children:Le.technical_context.query_class??"нет данных"})]})]}),o.jsx("h4",{children:"JSON разбор"}),o.jsx(Nd,{value:{annotation_id:Le.annotation_id,run_id:Le.run_id,case_id:Le.case_id,message_index:Le.message_index,rating:Le.rating,comment:Le.comment,manual_case_decision:Le.manual_case_decision,annotation_author:Le.annotation_author,resolved:Le.resolved,resolved_at:Le.resolved_at,resolved_by:Le.resolved_by,context:Le.context,technical_context:Le.technical_context,case_summary:Le.case_summary?{case_id:Le.case_summary.case_id,domain:Le.case_summary.domain,query_class:Le.case_summary.query_class,checks:Le.case_summary.checks,metric_subscores:Le.case_summary.metric_subscores}:null}})]}):null]}):null]}),yt.open?o.jsx("div",{className:"autoruns-comment-modal-backdrop",onClick:i=>{i.target===i.currentTarget&&rs()},children:o.jsxs("div",{className:"autoruns-comment-modal",children:[o.jsx("h3",{children:"Сохранить ручную сессию"}),o.jsx("p",{className:"muted",children:"Технический чат будет сохранен в автопрогоны как пользовательская multi-turn сессия."}),o.jsxs("label",{children:["Название",o.jsx("input",{value:yt.title,onChange:i=>jt(c=>({...c,title:i.target.value})),placeholder:"Например: НДС и склад на март 2020",disabled:yt.saving})]}),yt.error?o.jsx("p",{className:"error-text",children:yt.error}):null,o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",onClick:()=>{Ps()},disabled:yt.saving,children:yt.saving?"Сохраняю...":"Сохранить"}),o.jsx("button",{type:"button",className:"tab",onClick:()=>rs(),disabled:yt.saving,children:"Отмена"})]})]})}):null,en.open?o.jsx("div",{className:"autoruns-comment-modal-backdrop",onClick:i=>{i.target===i.currentTarget&&ct()},children:o.jsxs("div",{className:"autoruns-comment-modal",children:[o.jsx("h3",{children:"Удалить вопрос"}),o.jsx("p",{className:"muted",children:"Действительно удалить вопрос из сохраненной пользовательской сессии?"}),o.jsx("p",{className:"autoruns-comment-quote",children:en.questionText}),en.error?o.jsx("p",{className:"error-text",children:en.error}):null,o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",onClick:()=>{va()},disabled:en.saving,children:en.saving?"Удаляю...":"Да"}),o.jsx("button",{type:"button",className:"tab",onClick:()=>ct(),disabled:en.saving,children:"Нет"})]})]})}):null,Lt.open?o.jsx("div",{className:"autoruns-comment-modal-backdrop",onClick:i=>{i.target===i.currentTarget&&Sn()},children:o.jsxs("div",{className:"autoruns-comment-modal",children:[o.jsx("h3",{children:"Удалить сохраненный набор"}),o.jsx("p",{className:"muted",children:"Будет удалена карточка истории и связанный файл кейс-сета на бэке."}),o.jsx("p",{className:"autoruns-comment-quote",children:Lt.title}),Lt.error?o.jsx("p",{className:"error-text",children:Lt.error}):null,o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",onClick:()=>{xa()},disabled:Lt.saving,children:Lt.saving?"Удаляю...":"Да"}),o.jsx("button",{type:"button",className:"tab",onClick:()=>Sn(),disabled:Lt.saving,children:"Нет"})]})]})}):null,he.open?o.jsx("div",{className:"autoruns-comment-modal-backdrop",onClick:i=>{i.target===i.currentTarget&&_n()},children:o.jsxs("div",{className:"autoruns-comment-modal",children:[o.jsx("h3",{children:"Комментарий к ответу ассистента"}),o.jsx("p",{className:"muted",children:"Комментарий будет добавлен в общий список комментариев справа с меткой `assistant_live`."}),lo?o.jsxs("details",{className:"autoruns-prompt-details",open:!0,children:[o.jsx("summary",{children:"Вопрос пользователя"}),o.jsx("p",{className:"autoruns-comment-quote",children:lo.text})]}):null,Ns?o.jsxs("details",{className:"autoruns-prompt-details",open:!0,children:[o.jsx("summary",{children:"Ответ ассистента"}),o.jsx("p",{className:"autoruns-comment-quote",children:Ns.text})]}):null,o.jsx("div",{className:"autoruns-rating-row",role:"group","aria-label":"Рейтинг ответа ассистента",children:[1,2,3,4,5].map(i=>o.jsx("button",{type:"button",className:he.rating>=i?"autoruns-rating-dot active":"autoruns-rating-dot",onClick:()=>It(c=>({...c,rating:i})),disabled:he.saving,"aria-label":`Оценка ${i}`,children:he.rating>=i?"●":"○"},i))}),o.jsx("div",{className:"autoruns-form-grid",children:o.jsxs("label",{children:["Автор комментария",o.jsx("input",{value:he.annotationAuthor,onChange:i=>It(c=>({...c,annotationAuthor:i.target.value})),placeholder:"manual_reviewer",disabled:he.saving})]})}),o.jsxs("label",{children:["Комментарий",o.jsx("textarea",{value:he.comment,onChange:i=>It(c=>({...c,comment:i.target.value})),placeholder:"Что именно не так в ответе и что нужно исправить.",rows:4,disabled:he.saving})]}),he.error?o.jsx("p",{className:"error-text",children:he.error}):null,o.jsxs("div",{className:"button-row",children:[o.jsx("button",{type:"button",onClick:()=>{ga()},disabled:he.saving,children:he.saving?"Сохраняю...":"Готово"}),o.jsx("button",{type:"button",className:"tab",onClick:()=>_n(),disabled:he.saving,children:"Отмена"})]})]})}):null,ce.open?o.jsx("div",{className:"autoruns-comment-modal-backdrop",onClick:i=>{i.target===i.currentTarget&&Rs()},children:o.jsxs("div",{className:"autoruns-comment-modal",children:[o.jsx("h3",{children:"Комментарий к ответу системы"}),o.jsx("p",{className:"muted",children:"Оцените ответ по 5-балльной шкале и добавьте комментарий по браку."}),Yo?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:oo?.text??"Вопрос в диалоге не найден."})]}),o.jsxs("details",{className:"autoruns-prompt-details",open:!0,children:[o.jsx("summary",{children:"Ответ системы"}),o.jsx("p",{className:"autoruns-comment-quote",children:Yo.text})]})]}):null,o.jsx("div",{className:"autoruns-rating-row",role:"group","aria-label":"Рейтинг ответа",children:[1,2,3,4,5].map(i=>o.jsx("button",{type:"button",className:ce.rating>=i?"autoruns-rating-dot active":"autoruns-rating-dot",onClick:()=>ot(c=>({...c,rating:i})),disabled:ce.saving,"aria-label":`Оценка ${i}`,children:ce.rating>=i?"●":"○"},i))}),o.jsxs("div",{className:"autoruns-form-grid",children:[o.jsxs("label",{children:["Решение по кейсу",o.jsx("select",{value:ce.manualCaseDecision,onChange:i=>ot(c=>({...c,manualCaseDecision:i.target.value})),disabled:ce.saving,children:(sn.length>0?sn:Ce?.enum??[ea]).map(i=>o.jsx("option",{value:i,children:String(Ce?.labels?.[i]??i)},i))})]}),o.jsxs("label",{children:["Автор комментария",o.jsx("input",{value:ce.annotationAuthor,onChange:i=>ot(c=>({...c,annotationAuthor:i.target.value})),placeholder:"manual_reviewer",disabled:ce.saving})]})]}),o.jsxs("label",{children:["Комментарий",o.jsx("textarea",{value:ce.comment,onChange:i=>ot(c=>({...c,comment:i.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:()=>{pa()},disabled:ce.saving,children:ce.saving?"Сохраняю...":"Готово"}),o.jsx("button",{type:"button",className:"tab",onClick:()=>Rs(),disabled:ce.saving,children:"Отмена"})]})]})}):null]})}const yp={llmProvider:"openai",apiKey:"",model:"gpt-4o-mini",baseUrl:"https://api.openai.com/v1",temperature:0,maxOutputTokens:700},Sd={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."},xp={userQuestion:"",batchQuestionsRaw:"",periodHint:"",businessContext:"",expectedRoute:""},Ji={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}},wd="ndc_normalizer_session_config_v1",jd="ndc_autoruns_layout_config_v1",_p="ndc-autoruns-save",Sp="autoruns",Yi="normalizer_v2_0_2",wp="address_query_runtime_v1",jp=["normalized","fragments","scope","flags","route","raw","validation","logs"];function kp(a){return`[${new Date().toLocaleTimeString("ru-RU")}] ${a}`}function Cp(a,m){if(!m)return"Previous preset is not selected.";const S=["systemPrompt","developerPrompt","domainPrompt","schemaNotes","fewShotExamples"].filter(C=>a[C]!==m[C]).map(C=>`${C}: ${Math.abs(a[C].length-m[C].length)} chars delta`);return S.length===0?"No changes against previous preset.":`Changed fields: ${S.length}. ${S.join(" | ")}`}function Np(){const[a,m]=g.useState(yp),[p,S]=g.useState(Sd),[C,M]=g.useState(xp),[z,ae]=g.useState(null),[ne,F]=g.useState([]),[X,Z]=g.useState([]),[te,Me]=g.useState("normalized"),[pe,me]=g.useState(!1),[ye,Qe]=g.useState(!1),[Xe,He]=g.useState([]),[Ae,b]=g.useState(""),[le,Ee]=g.useState([]),[be,St]=g.useState(""),[mt,Kt]=g.useState("NDC custom preset"),[ht,B]=g.useState(null),[Ie,Je]=g.useState(""),[re,ee]=g.useState(!1),[I,O]=g.useState([]),[L,y]=g.useState(""),[k,se]=g.useState([]),[ue,de]=g.useState(!1),[xe,we]=g.useState(null),[je,Ce]=g.useState(""),[gt,sn]=g.useState(Sp),[yr,Rn]=g.useState(!0),[vn,Se]=g.useState(!0),[Jt,Ve]=g.useState(!0),[wt,Tt]=g.useState(!0),[yn,W]=g.useState(!0),[vt,xr]=g.useState(!0),[Mn,Ge]=g.useState(!0),[Rt,zt]=g.useState(!0),[An,ve]=g.useState(!0),[Mt,Ze]=g.useState(!0),[ys,st]=g.useState(!0),[er,_r]=g.useState(!0),[on,In]=g.useState(!0),Ft=g.useRef(!1),xn=g.useRef(!1),Yt=g.useRef(!1);g.useEffect(()=>{const E=document.documentElement,{colors:Y}=Ji;E.style.setProperty("--rgb-background",Y.backgroundRgb),E.style.setProperty("--rgb-surface-main",Y.mainSurfaceRgb),E.style.setProperty("--rgb-surface-horizontal",Y.horizontalSurfaceRgb),E.style.setProperty("--rgb-surface-focus",Y.focusSurfaceRgb),E.style.setProperty("--rgb-assistant-chip",Y.assistantChipRgb),E.style.setProperty("--rgb-assistant-chip-hover",Y.assistantChipHoverRgb),E.style.setProperty("--rgb-assistant-chip-selected",Y.assistantChipSelectedRgb),E.style.setProperty("--rgb-assistant-chip-selected-text",Y.assistantChipSelectedTextRgb),E.style.setProperty("--rgb-active",Y.activeRgb),E.style.setProperty("--rgb-active-text",Y.activeTextRgb),E.style.setProperty("--rgb-text-main",Y.textMainRgb),E.style.setProperty("--rgb-text-muted",Y.textMutedRgb),E.style.setProperty("--rgb-danger",Y.dangerRgb),E.style.setProperty("--rgb-scrollbar-track",Y.scrollbarTrackRgb),E.style.setProperty("--rgb-scrollbar-thumb",Y.scrollbarThumbRgb),E.style.setProperty("--rgb-scrollbar-thumb-hover",Y.scrollbarThumbHoverRgb),E.style.setProperty("--mode-column-width",`${Ji.layout.modeColumnWidthPx}px`),E.style.setProperty("--mode-toggle-width",`${Ji.layout.modeToggleWidthPx}px`)},[]);const q=E=>{Z(Y=>[kp(E),...Y].slice(0,300))};g.useEffect(()=>{(async()=>{const Q=localStorage.getItem(wd);if(Q)try{const Pe=JSON.parse(Q);m(et=>({...et,llmProvider:Pe.llmProvider==="local"?"local":"openai",model:Pe.model??et.model,baseUrl:Pe.baseUrl??et.baseUrl,temperature:Pe.temperature??et.temperature,maxOutputTokens:Pe.maxOutputTokens??et.maxOutputTokens}))}catch{}try{const Pe=await Te.loadSharedConnectionConfig();Pe.connection&&Pe.connection.llmProvider==="local"&&(m(et=>({...et,llmProvider:"local",model:Pe.connection?.model??et.model,baseUrl:Pe.connection?.baseUrl??et.baseUrl,temperature:Pe.connection?.temperature??et.temperature,maxOutputTokens:Pe.connection?.maxOutputTokens??et.maxOutputTokens})),q(`Shared local LLM config loaded: ${Pe.connection.model}`))}catch(Pe){q(`Shared local config load error: ${Pe instanceof Error?Pe.message:String(Pe)}`)}finally{Yt.current=!0}})();const Y=localStorage.getItem(jd);if(Y)try{const Q=JSON.parse(Y);(Q.uiMode==="assistant"||Q.uiMode==="autoruns"||Q.uiMode==="decomposition")&&sn("autoruns"),Q.activeTab&&jp.includes(Q.activeTab)&&Me(Q.activeTab),typeof Q.showAutorunsSettingsMode=="boolean"&&Rn(Q.showAutorunsSettingsMode),typeof Q.showAutorunsAutoRunsMode=="boolean"&&Se(Q.showAutorunsAutoRunsMode),typeof Q.showAutorunsAssistantMode=="boolean"&&Ve(Q.showAutorunsAssistantMode),typeof Q.showAutorunsDecompositionMode=="boolean"&&Tt(Q.showAutorunsDecompositionMode),typeof Q.showAutorunsProgressMode=="boolean"&&W(Q.showAutorunsProgressMode),typeof Q.showAutorunsCommentsMode=="boolean"&&xr(Q.showAutorunsCommentsMode),typeof Q.showDecompositionConnectionMode=="boolean"&&Ge(Q.showDecompositionConnectionMode),typeof Q.showDecompositionPromptMode=="boolean"&&zt(Q.showDecompositionPromptMode),typeof Q.showDecompositionQueryMode=="boolean"&&ve(Q.showDecompositionQueryMode),typeof Q.showDecompositionOutputMode=="boolean"&&Ze(Q.showDecompositionOutputMode),typeof Q.showDecompositionMetricsMode=="boolean"&&st(Q.showDecompositionMetricsMode),typeof Q.showDecompositionHistoryMode=="boolean"&&_r(Q.showDecompositionHistoryMode),typeof Q.showDecompositionRuntimeMode=="boolean"&&In(Q.showDecompositionRuntimeMode),Q.prompts&&(S(Pe=>({...Pe,...Q.prompts})),xn.current=!0)}catch{}At(),tr(),Kr()},[]),g.useEffect(()=>{if(!Yt.current||a.llmProvider!=="local")return;const E=window.setTimeout(()=>{Te.saveSharedConnectionConfig(a).catch(Y=>q(`Shared local config sync error: ${Y instanceof Error?Y.message:String(Y)}`))},250);return()=>window.clearTimeout(E)},[a.baseUrl,a.llmProvider,a.maxOutputTokens,a.model,a.temperature]);async function At(){try{const E=await Te.loadHistory();F(E.items??[])}catch(E){q(`History load error: ${E instanceof Error?E.message:String(E)}`)}}async function tr(){try{const Y=(await Te.loadPresets()).presets??[];if(Ee(Y),xn.current){Ft.current=!0;return}if(Ft.current)return;const Q=Y.find(Pe=>Pe.prompt_version===Yi)??Y.find(Pe=>Pe.id==="default-normalizer-v2_0_2");if(!Q){Ft.current=!0,q(`Preset autoload skipped: ${Yi} not found.`);return}St(Q.id),B(p),S({systemPrompt:Q.systemPrompt,developerPrompt:Q.developerPrompt,domainPrompt:Q.domainPrompt,schemaNotes:Q.schemaNotes??"",fewShotExamples:Q.fewShotExamples??""}),Ft.current=!0,q(`Preset autoloaded: ${Q.name} (${Q.prompt_version}).`)}catch(E){q(`Presets load error: ${E instanceof Error?E.message:String(E)}`)}}async function Kr(){try{const E=await Te.listRuns();O(E.items??[])}catch(E){q(`Runs load error: ${E instanceof Error?E.message:String(E)}`)}}function Jr(){if(localStorage.setItem(wd,JSON.stringify({model:a.model,llmProvider:a.llmProvider,baseUrl:a.baseUrl,temperature:a.temperature,maxOutputTokens:a.maxOutputTokens})),a.llmProvider==="local"){Te.saveSharedConnectionConfig(a).then(()=>{q("Local config saved and synced to shared agent config (without API key).")}).catch(E=>{q(`Local config saved, but shared sync failed: ${E instanceof Error?E.message:String(E)}`)});return}q("Local config saved (without API key).")}function Yr(){localStorage.setItem(jd,JSON.stringify({uiMode:gt,activeTab:te,showAutorunsSettingsMode:yr,showAutorunsAutoRunsMode:vn,showAutorunsAssistantMode:Jt,showAutorunsDecompositionMode:wt,showAutorunsProgressMode:yn,showAutorunsCommentsMode:vt,showDecompositionConnectionMode:Mn,showDecompositionPromptMode:Rt,showDecompositionQueryMode:An,showDecompositionOutputMode:Mt,showDecompositionMetricsMode:ys,showDecompositionHistoryMode:er,showDecompositionRuntimeMode:on,prompts:p})),window.dispatchEvent(new CustomEvent(_p)),q("UI layout and prompts saved.")}async function Ln(){me(!0),Ce("");try{const E=await Te.testConnection(a);E.provider==="local"?E.model_found===!0?(b(`LOCAL OK - ${E.model}`),q(`Local model is available: ${E.model} (catalog size=${E.models_count??"n/a"}).`)):E.model_found===!1?(b(`LOCAL OK, model not loaded - ${E.model}`),q(`Local server is reachable, but model '${E.model}' is not in loaded catalog. Use 'Load model list' and select one of loaded models.`)):(b(`LOCAL OK (model list unavailable) - ${E.model}`),q("Local server is reachable, but model catalog could not be verified.")):(b(`OPENAI OK - ${E.model}`),q(`OpenAI connection ok: ${E.model}`))}catch(E){const Y=E instanceof Error?E.message:String(E);b("Connection error"),Ce(`Test connection: ${Y}`),q(`Test connection error: ${Y}`)}finally{me(!1)}}async function bt(){Qe(!0);try{const Y=(await Te.listModels(a)).models??[];He(Y),Y.length>0&&m(Q=>Q.model&&Y.includes(Q.model)?Q:{...Q,model:Y[0]}),q(`Model catalog loaded (${a.llmProvider}): ${Y.length} items.`)}catch(E){const Y=E instanceof Error?E.message:String(E);q(`Load model list error: ${Y}`)}finally{Qe(!1)}}g.useEffect(()=>{He([])},[a.llmProvider,a.baseUrl]);function Dn(){const E=le.find(Y=>Y.id===be);if(!E){q("Preset is not selected.");return}B(p),S({systemPrompt:E.systemPrompt,developerPrompt:E.developerPrompt,domainPrompt:E.domainPrompt,schemaNotes:E.schemaNotes??"",fewShotExamples:E.fewShotExamples??""}),q(`Preset loaded: ${E.name}`)}async function On(){try{await Te.savePreset({name:mt||"NDC preset",prompt_version:"normalizer_v2_0_2",systemPrompt:p.systemPrompt,developerPrompt:p.developerPrompt,domainPrompt:p.domainPrompt,schemaNotes:p.schemaNotes,fewShotExamples:p.fewShotExamples}),q("Preset saved."),await tr()}catch(E){q(`Preset save error: ${E instanceof Error?E.message:String(E)}`)}}function $n(){S(Sd),q("Prompt panel reset to defaults.")}function zn(){const E=Cp(p,ht);Je(E),q(E)}return g.useEffect(()=>{if(!L){se([]);return}Te.runTrace(L).then(E=>se(E.items)).catch(E=>q(`Run trace error: ${E instanceof Error?E.message:String(E)}`))},[L]),o.jsxs("main",{className:"app-root 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:"tab active",onClick:()=>sn("autoruns"),children:"Управление ассистентом"}),o.jsx("button",{type:"button",className:"tab",onClick:Yr,children:"Сохранить"})]}),o.jsxs("div",{className:"mode-switch-row mode-switch-row-right",children:[o.jsx("button",{type:"button",className:yr?"tab active":"tab",onClick:()=>Rn(E=>!E),children:"Настройки"}),o.jsx("button",{type:"button",className:vn?"tab active":"tab",onClick:()=>Se(E=>!E),children:"Автопрогоны"}),o.jsx("button",{type:"button",className:Jt?"tab active":"tab",onClick:()=>Ve(E=>!E),children:"Режим ассистента"}),o.jsx("button",{type:"button",className:yn?"tab active":"tab",onClick:()=>W(E=>!E),children:"Прогресс/регресс"}),o.jsx("button",{type:"button",className:vt?"tab active":"tab",onClick:()=>xr(E=>!E),children:"Комментарии"})]})]}),o.jsx("div",{className:"layout-grid layout-grid-autoruns",children:o.jsx(vp,{connection:a,modelOptions:Xe,modelsBusy:ye,connectionStatus:Ae,connectionBusy:pe,onConnectionChange:m,onReloadModels:bt,onSaveLocalConfig:Jr,onTestConnection:Ln,prompts:p,onPromptsChange:S,promptPresets:le,selectedPresetId:be,onSelectPreset:St,onLoadPreset:Dn,onSavePreset:On,onResetDefaults:$n,onDiffPrevious:zn,presetName:mt,onPresetNameChange:Kt,diffSummary:Ie,assistantPromptVersion:wp,decompositionPromptVersion:Yi,showSettingsMode:yr,showAutoRunsMode:vn,showAssistantMode:Jt,showProgressMode:yn,showCommentsMode:vt,onLog:q})})]})}Df.createRoot(document.getElementById("root")).render(o.jsx(Pf.StrictMode,{children:o.jsx(Np,{})})); diff --git a/llm_normalizer/frontend/dist/assets/index-BURU4_Sm.css b/llm_normalizer/frontend/dist/assets/index-DW_SonhM.css similarity index 54% rename from llm_normalizer/frontend/dist/assets/index-BURU4_Sm.css rename to llm_normalizer/frontend/dist/assets/index-DW_SonhM.css index a2b52ec..c48f7d6 100644 --- a/llm_normalizer/frontend/dist/assets/index-BURU4_Sm.css +++ b/llm_normalizer/frontend/dist/assets/index-DW_SonhM.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-emerald: 126, 211, 154;--rgb-warning: 255, 196, 116;--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-group-heading{display:flex;align-items:center;justify-content:space-between;gap:8px;margin:12px 0 8px}.autoruns-group-heading h4{margin:0}.autoruns-group-toggle{width:20px;min-width:20px;height:20px;padding:0;border:none;border-radius:6px;background:transparent;color:var(--text-muted);display:inline-flex;align-items:center;justify-content:center}.autoruns-group-toggle:hover{background:rgba(var(--rgb-background),.28);color:rgb(var(--rgb-text-main))}.autoruns-group-chevron-svg{width:14px;height:14px;stroke:currentColor;fill:none;stroke-width:1.6;stroke-linecap:round;stroke-linejoin:round;transform:rotate(-90deg);transition:transform .18s ease}.autoruns-group-chevron-svg.expanded{transform:rotate(0)}.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;overflow:hidden}.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.saved-session{cursor:default;gap:8px}.autoruns-autogen-item.saved-session header{display:grid;gap:4px}.autoruns-autogen-item.saved-session header strong{font-size:.84rem}.autoruns-autogen-item.saved-session header span{color:var(--text-muted);font-size:.74rem}.autoruns-autogen-item.saved-session header .autoruns-autogen-card-actions{display:none}.autoruns-saved-session-meta{display:flex;align-items:center;gap:6px;flex-wrap:wrap;font-size:.74rem;line-height:1.35}.autoruns-saved-session-meta-label{font-size:inherit;font-weight:700;color:#fff}.autoruns-saved-session-meta-value{color:#ffffffd1}.autoruns-saved-session-meta-id{width:100%;justify-content:space-between;align-items:flex-start;flex-wrap:nowrap;gap:8px}.autoruns-saved-session-meta-id-main{display:flex;align-items:center;gap:6px;min-width:0;flex:1 1 auto;flex-wrap:wrap}.autoruns-saved-session-meta-id-main .autoruns-saved-session-meta-value{overflow-wrap:anywhere;word-break:break-word}.autoruns-saved-session-meta-id .autoruns-copy-run-id-btn{color:#ffffffd1;margin-left:auto;align-self:flex-start}.autoruns-saved-session-meta-id .autoruns-copy-run-id-btn:hover{color:rgb(var(--rgb-active))}.autoruns-saved-session-topbar{display:flex;align-items:center;justify-content:space-between;gap:8px;min-height:18px}.autoruns-saved-session-footer{display:flex;align-items:center;justify-content:flex-start;gap:10px;min-height:20px;margin-top:2px}.autoruns-saved-session-icon-btn{width:20px;min-width:20px;height:20px;padding:0;border:none;border-radius:6px;background:transparent;color:var(--text-muted);display:inline-flex;align-items:center;justify-content:center}.autoruns-saved-session-icon-btn:hover:not(:disabled){background:rgba(var(--rgb-background),.28);color:rgb(var(--rgb-text-main))}.autoruns-autogen-item.saved-session .autoruns-saved-session-icon-btn,.autoruns-autogen-item.saved-session .autoruns-autogen-delete-btn{color:#fff}.autoruns-autogen-item.saved-session .autoruns-saved-session-icon-btn:hover:not(:disabled),.autoruns-autogen-item.saved-session .autoruns-autogen-delete-btn:hover:not(:disabled){background:rgba(var(--rgb-background),.28);color:rgb(var(--rgb-active));box-shadow:none}.autoruns-autogen-item.saved-session .autoruns-autogen-delete-btn{border-radius:6px}.autoruns-card-chevron-svg{width:14px;height:14px;stroke:currentColor;fill:none;stroke-width:1.6;stroke-linecap:round;stroke-linejoin:round;transition:transform .18s ease}.autoruns-card-launch-svg,.autoruns-card-stop-svg{width:14px;height:14px;fill:currentColor;stroke:none}.autoruns-card-chevron-svg.expanded{transform:rotate(180deg)}.autoruns-saved-session-questions{max-height:0;overflow:hidden;opacity:0;transition:max-height .24s ease,opacity .18s ease,margin-top .24s ease;margin-top:0}.autoruns-saved-session-questions.expanded{max-height:520px;opacity:1;margin-top:4px}.autoruns-generated-questions-embedded{margin-top:2px}.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:grid;grid-template-columns:22px minmax(0,1fr) auto;align-items:start;gap:8px;border:none;border-radius:9px;background:rgb(var(--rgb-surface-focus));padding:7px 8px;font-size:.78rem;transition:background .15s ease,outline-color .15s ease,opacity .15s ease}.autoruns-generated-question-item.drag-over{outline:1px solid rgba(var(--rgb-active),.75)}.autoruns-generated-question-item.dragging{opacity:.72}.autoruns-generated-question-item.editing{background:rgb(var(--rgb-active));color:rgb(var(--rgb-active-text))}.autoruns-question-grip-btn{width:18px;min-width:18px;height:18px;padding:0;border:none;border-radius:6px;background:transparent;color:var(--text-muted);display:inline-flex;align-items:center;justify-content:center;cursor:grab;margin-top:1px}.autoruns-question-grip-btn:hover:not(:disabled){color:rgb(var(--rgb-text-main));background:rgba(var(--rgb-background),.3)}.autoruns-generated-question-item.editing .autoruns-question-grip-btn{color:rgba(var(--rgb-active-text),.9)}.autoruns-generated-question-item.editing .autoruns-question-grip-btn:hover:not(:disabled){color:rgb(var(--rgb-active-text));background:rgba(var(--rgb-active-text),.14)}.autoruns-question-grip-btn:disabled{cursor:default;opacity:.45}.autoruns-question-grip-svg{width:14px;height:14px;fill:currentColor}.autoruns-generated-question-text{border:none;background:transparent;color:rgb(var(--rgb-text-main));padding:0;margin:0;text-align:left;font:inherit;white-space:pre-wrap;line-height:1.4;cursor:text}.autoruns-generated-question-text:hover{color:rgb(var(--rgb-active))}.autoruns-generated-question-input{width:100%;min-width:0;border:none;border-radius:8px;background:rgba(var(--rgb-background),.55);color:rgb(var(--rgb-text-main));padding:6px 8px;font:inherit;line-height:1.4}.autoruns-generated-question-input:focus{outline:none}.autoruns-generated-question-item.editing .autoruns-generated-question-input{background:rgba(var(--rgb-active-text),.14);color:rgb(var(--rgb-active-text))}.autoruns-generated-question-item.editing .autoruns-generated-question-input::placeholder{color:rgba(var(--rgb-active-text),.78)}.autoruns-add-question-btn{width:100%;min-height:30px;border-radius:8px;border:none;background:rgb(var(--rgb-surface-focus));color:rgb(var(--rgb-text-main));font-size:1.1rem;font-weight:700;line-height:1}.autoruns-add-question-btn:hover:not(:disabled){background:rgb(var(--rgb-active));color:rgb(var(--rgb-active-text))}.autoruns-add-question-btn:disabled{opacity:.5;cursor:default}.autoruns-remove-question-btn{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;box-shadow:none;transition:color .15s ease;align-self:start}.autoruns-remove-question-btn:hover{background:transparent;color:rgb(var(--rgb-active));box-shadow:none}.autoruns-generated-question-item.editing .autoruns-remove-question-btn{color:rgb(var(--rgb-active-text))}.autoruns-generated-question-item.editing .autoruns-remove-question-btn:hover{color:rgba(var(--rgb-active-text),.82)}.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))}.autoruns-autogen-card-actions{display:inline-flex;align-items:center;gap:8px}.autoruns-autogen-delete-btn{border:none;border-radius:0;width:22px;height:22px;min-width:22px;padding:0;background:transparent;color:rgb(var(--rgb-text-main));display:inline-flex;align-items:center;justify-content:center;font-size:1rem;font-weight:700;line-height:1;box-shadow:none;transition:color .15s ease}.autoruns-autogen-delete-btn:hover{background:transparent;color:rgb(var(--rgb-active));box-shadow:none}.autoruns-agent-acceptance{display:grid;gap:6px;margin-top:6px}.autoruns-agent-pill-row{display:flex;flex-wrap:wrap;gap:6px}.autoruns-agent-pill{display:inline-flex;align-items:center;justify-content:center;min-height:24px;padding:0 8px;border-radius:999px;background:rgb(var(--rgb-surface-focus));color:rgb(var(--rgb-text-main));font-size:.73rem;line-height:1;border:1px solid rgba(var(--rgb-text-main),.1)}.autoruns-agent-pill.neutral{color:var(--text-muted)}.autoruns-agent-pill.status-accepted{color:rgb(var(--rgb-emerald));border-color:rgba(var(--rgb-emerald),.32)}.autoruns-agent-pill.status-partial{color:rgb(var(--rgb-warning));border-color:rgba(var(--rgb-warning),.32)}.autoruns-agent-pill.status-blocked{color:rgb(var(--rgb-danger));border-color:rgba(var(--rgb-danger),.32)}.autoruns-agent-pill.status-unknown{color:var(--text-muted)}@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-emerald: 126, 211, 154;--rgb-warning: 255, 196, 116;--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-group-heading{display:flex;align-items:center;justify-content:space-between;gap:8px;margin:12px 0 8px}.autoruns-group-heading h4{margin:0}.autoruns-group-toggle{width:20px;min-width:20px;height:20px;padding:0;border:none;border-radius:6px;background:transparent;color:var(--text-muted);display:inline-flex;align-items:center;justify-content:center}.autoruns-group-toggle:hover{background:rgba(var(--rgb-background),.28);color:rgb(var(--rgb-text-main))}.autoruns-group-chevron-svg{width:14px;height:14px;stroke:currentColor;fill:none;stroke-width:1.6;stroke-linecap:round;stroke-linejoin:round;transform:rotate(-90deg);transition:transform .18s ease}.autoruns-group-chevron-svg.expanded{transform:rotate(0)}.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-dialog-copy-actions{grid-column:1 / -1;display:grid;grid-template-columns:repeat(2,minmax(0,1fr));gap:8px;align-items:center}.autoruns-dialog-copy-status{grid-column:1 / -1;display:flex;justify-content:flex-end;min-height:18px}.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;overflow:hidden}.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.saved-session{cursor:default;gap:8px}.autoruns-autogen-item.saved-session header{display:grid;gap:4px}.autoruns-autogen-item.saved-session header strong{font-size:.84rem}.autoruns-autogen-item.saved-session header span{color:var(--text-muted);font-size:.74rem}.autoruns-autogen-item.saved-session header .autoruns-autogen-card-actions{display:none}.autoruns-saved-session-meta{display:flex;align-items:center;gap:6px;flex-wrap:wrap;font-size:.74rem;line-height:1.35}.autoruns-saved-session-meta-label{font-size:inherit;font-weight:700;color:#fff}.autoruns-saved-session-meta-value{color:#ffffffd1}.autoruns-saved-session-meta-id{width:100%;justify-content:space-between;align-items:flex-start;flex-wrap:nowrap;gap:8px}.autoruns-saved-session-meta-id-main{display:flex;align-items:center;gap:6px;min-width:0;flex:1 1 auto;flex-wrap:wrap}.autoruns-saved-session-meta-id-main .autoruns-saved-session-meta-value{overflow-wrap:anywhere;word-break:break-word}.autoruns-saved-session-meta-id .autoruns-copy-run-id-btn{color:#ffffffd1;margin-left:auto;align-self:flex-start}.autoruns-saved-session-meta-id .autoruns-copy-run-id-btn:hover{color:rgb(var(--rgb-active))}.autoruns-saved-session-topbar{display:flex;align-items:center;justify-content:space-between;gap:8px;min-height:18px}.autoruns-saved-session-footer{display:flex;align-items:center;justify-content:flex-start;gap:10px;min-height:20px;margin-top:2px}.autoruns-saved-session-icon-btn{width:20px;min-width:20px;height:20px;padding:0;border:none;border-radius:6px;background:transparent;color:var(--text-muted);display:inline-flex;align-items:center;justify-content:center}.autoruns-saved-session-icon-btn:hover:not(:disabled){background:rgba(var(--rgb-background),.28);color:rgb(var(--rgb-text-main))}.autoruns-autogen-item.saved-session .autoruns-saved-session-icon-btn,.autoruns-autogen-item.saved-session .autoruns-autogen-delete-btn{color:#fff}.autoruns-autogen-item.saved-session .autoruns-saved-session-icon-btn:hover:not(:disabled),.autoruns-autogen-item.saved-session .autoruns-autogen-delete-btn:hover:not(:disabled){background:rgba(var(--rgb-background),.28);color:rgb(var(--rgb-active));box-shadow:none}.autoruns-autogen-item.saved-session .autoruns-autogen-delete-btn{border-radius:6px}.autoruns-card-chevron-svg{width:14px;height:14px;stroke:currentColor;fill:none;stroke-width:1.6;stroke-linecap:round;stroke-linejoin:round;transition:transform .18s ease}.autoruns-card-launch-svg,.autoruns-card-stop-svg{width:14px;height:14px;fill:currentColor;stroke:none}.autoruns-card-chevron-svg.expanded{transform:rotate(180deg)}.autoruns-saved-session-questions{max-height:0;overflow:hidden;opacity:0;transition:max-height .24s ease,opacity .18s ease,margin-top .24s ease;margin-top:0}.autoruns-saved-session-questions.expanded{max-height:520px;opacity:1;margin-top:4px}.autoruns-generated-questions-embedded{margin-top:2px}.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:grid;grid-template-columns:22px minmax(0,1fr) auto;align-items:start;gap:8px;border:none;border-radius:9px;background:rgb(var(--rgb-surface-focus));padding:7px 8px;font-size:.78rem;transition:background .15s ease,outline-color .15s ease,opacity .15s ease}.autoruns-generated-question-item.drag-over{outline:1px solid rgba(var(--rgb-active),.75)}.autoruns-generated-question-item.dragging{opacity:.72}.autoruns-generated-question-item.editing{background:rgb(var(--rgb-active));color:rgb(var(--rgb-active-text))}.autoruns-question-grip-btn{width:18px;min-width:18px;height:18px;padding:0;border:none;border-radius:6px;background:transparent;color:var(--text-muted);display:inline-flex;align-items:center;justify-content:center;cursor:grab;margin-top:1px}.autoruns-question-grip-btn:hover:not(:disabled){color:rgb(var(--rgb-text-main));background:rgba(var(--rgb-background),.3)}.autoruns-generated-question-item.editing .autoruns-question-grip-btn{color:rgba(var(--rgb-active-text),.9)}.autoruns-generated-question-item.editing .autoruns-question-grip-btn:hover:not(:disabled){color:rgb(var(--rgb-active-text));background:rgba(var(--rgb-active-text),.14)}.autoruns-question-grip-btn:disabled{cursor:default;opacity:.45}.autoruns-question-grip-svg{width:14px;height:14px;fill:currentColor}.autoruns-generated-question-text{border:none;background:transparent;color:rgb(var(--rgb-text-main));padding:0;margin:0;text-align:left;font:inherit;white-space:pre-wrap;line-height:1.4;cursor:text}.autoruns-generated-question-text:hover{color:rgb(var(--rgb-active))}.autoruns-generated-question-input{width:100%;min-width:0;border:none;border-radius:8px;background:rgba(var(--rgb-background),.55);color:rgb(var(--rgb-text-main));padding:6px 8px;font:inherit;line-height:1.4}.autoruns-generated-question-input:focus{outline:none}.autoruns-generated-question-item.editing .autoruns-generated-question-input{background:rgba(var(--rgb-active-text),.14);color:rgb(var(--rgb-active-text))}.autoruns-generated-question-item.editing .autoruns-generated-question-input::placeholder{color:rgba(var(--rgb-active-text),.78)}.autoruns-add-question-btn{width:100%;min-height:30px;border-radius:8px;border:none;background:rgb(var(--rgb-surface-focus));color:rgb(var(--rgb-text-main));font-size:1.1rem;font-weight:700;line-height:1}.autoruns-add-question-btn:hover:not(:disabled){background:rgb(var(--rgb-active));color:rgb(var(--rgb-active-text))}.autoruns-add-question-btn:disabled{opacity:.5;cursor:default}.autoruns-remove-question-btn{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;box-shadow:none;transition:color .15s ease;align-self:start}.autoruns-remove-question-btn:hover{background:transparent;color:rgb(var(--rgb-active));box-shadow:none}.autoruns-generated-question-item.editing .autoruns-remove-question-btn{color:rgb(var(--rgb-active-text))}.autoruns-generated-question-item.editing .autoruns-remove-question-btn:hover{color:rgba(var(--rgb-active-text),.82)}.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))}.autoruns-autogen-card-actions{display:inline-flex;align-items:center;gap:8px}.autoruns-autogen-delete-btn{border:none;border-radius:0;width:22px;height:22px;min-width:22px;padding:0;background:transparent;color:rgb(var(--rgb-text-main));display:inline-flex;align-items:center;justify-content:center;font-size:1rem;font-weight:700;line-height:1;box-shadow:none;transition:color .15s ease}.autoruns-autogen-delete-btn:hover{background:transparent;color:rgb(var(--rgb-active));box-shadow:none}.autoruns-agent-acceptance{display:grid;gap:6px;margin-top:6px}.autoruns-agent-pill-row{display:flex;flex-wrap:wrap;gap:6px}.autoruns-agent-pill{display:inline-flex;align-items:center;justify-content:center;min-height:24px;padding:0 8px;border-radius:999px;background:rgb(var(--rgb-surface-focus));color:rgb(var(--rgb-text-main));font-size:.73rem;line-height:1;border:1px solid rgba(var(--rgb-text-main),.1)}.autoruns-agent-pill.neutral{color:var(--text-muted)}.autoruns-agent-pill.status-accepted{color:rgb(var(--rgb-emerald));border-color:rgba(var(--rgb-emerald),.32)}.autoruns-agent-pill.status-partial{color:rgb(var(--rgb-warning));border-color:rgba(var(--rgb-warning),.32)}.autoruns-agent-pill.status-blocked{color:rgb(var(--rgb-danger));border-color:rgba(var(--rgb-danger),.32)}.autoruns-agent-pill.status-unknown{color:var(--text-muted)}@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/index.html b/llm_normalizer/frontend/dist/index.html index 6d8ddbb..c729903 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/components/AutoRunsHistoryPanel.tsx b/llm_normalizer/frontend/src/components/AutoRunsHistoryPanel.tsx index 7e4f974..13e08d8 100644 --- a/llm_normalizer/frontend/src/components/AutoRunsHistoryPanel.tsx +++ b/llm_normalizer/frontend/src/components/AutoRunsHistoryPanel.tsx @@ -21,6 +21,7 @@ import type { ManualCaseDecision, PromptState } from "../state/types"; +import { buildAutoRunDialogExportForCopy, type ConversationExportMode } from "../utils/conversationExport"; import { AssistantPanel } from "./AssistantPanel"; import { ConnectionPanel } from "./ConnectionPanel"; import { JsonView } from "./JsonView"; @@ -620,6 +621,37 @@ function CardStopIcon() { ); } +async function writeTextToClipboard(text: string): Promise { + if (navigator.clipboard && window.isSecureContext) { + try { + await navigator.clipboard.writeText(text); + return true; + } catch { + // Fall back to the legacy path below. + } + } + + const textarea = document.createElement("textarea"); + textarea.value = text; + textarea.setAttribute("readonly", "true"); + textarea.style.position = "fixed"; + textarea.style.opacity = "0"; + textarea.style.pointerEvents = "none"; + document.body.appendChild(textarea); + textarea.select(); + + let copied = false; + try { + copied = document.execCommand("copy"); + } catch { + copied = false; + } finally { + document.body.removeChild(textarea); + } + + return copied; +} + function GroupChevronIcon({ expanded }: { expanded: boolean }) { return (
+ + +
+ {dialogCopyState === "success" ? Скопировано ({dialogCopyModeLabel}) : null} + {dialogCopyState === "error" ? Ошибка копирования : null} +
+
diff --git a/llm_normalizer/frontend/src/state/types.ts b/llm_normalizer/frontend/src/state/types.ts index e21f5bb..d27c2f9 100644 --- a/llm_normalizer/frontend/src/state/types.ts +++ b/llm_normalizer/frontend/src/state/types.ts @@ -248,6 +248,7 @@ export interface AutoRunDialogMessage { created_at: string | null; trace_id: string | null; reply_type: string | null; + debug?: unknown | null; message_index: number; case_id?: string | null; case_message_index?: number | null; diff --git a/llm_normalizer/frontend/src/styles.css b/llm_normalizer/frontend/src/styles.css index b54b0cf..30fa73d 100644 --- a/llm_normalizer/frontend/src/styles.css +++ b/llm_normalizer/frontend/src/styles.css @@ -1263,6 +1263,21 @@ button:disabled { gap: 8px; } +.autoruns-dialog-copy-actions { + grid-column: 1 / -1; + display: grid; + grid-template-columns: repeat(2, minmax(0, 1fr)); + gap: 8px; + align-items: center; +} + +.autoruns-dialog-copy-status { + grid-column: 1 / -1; + display: flex; + justify-content: flex-end; + min-height: 18px; +} + .autoruns-case-list { margin-top: 8px; display: grid; diff --git a/llm_normalizer/frontend/src/utils/conversationExport.ts b/llm_normalizer/frontend/src/utils/conversationExport.ts index d93c079..c11b8e5 100644 --- a/llm_normalizer/frontend/src/utils/conversationExport.ts +++ b/llm_normalizer/frontend/src/utils/conversationExport.ts @@ -10,6 +10,33 @@ export interface ConversationExportItem { debug?: unknown | null; } +export interface AutoRunDialogExportItem { + message_id: string | null; + role: string; + text: string; + reply_type: string | null; + created_at: string | null; + trace_id: string | null; + message_index: number; + case_id?: string | null; + case_message_index?: number | null; + debug?: unknown | null; +} + +export interface AutoRunDialogExportPayload { + runId: string; + caseId: string; + sessionId: string; + source: string; + messages: AutoRunDialogExportItem[]; + decomposition?: string[]; + assistantMode?: unknown | null; + annotations?: unknown[]; + runSummary?: unknown | null; + coverage?: unknown | null; + report?: unknown | null; +} + const DEBUG_SECTION_PATTERN = /(?:^|\n)\s*#{0,6}\s*(?:debug_payload_json|technical_breakdown_json|route_summary_json|debug_payload|technical_breakdown)\b/i; @@ -28,6 +55,10 @@ function stringifyDebug(value: unknown): string { } } +function normalizeRole(value: string): "user" | "assistant" { + return value === "assistant" ? "assistant" : "user"; +} + export function sanitizeConversationExportText(value: string): string { const raw = String(value ?? ""); const cutMatch = raw.match(DEBUG_SECTION_PATTERN); @@ -83,3 +114,113 @@ export function buildConversationExportForCopy( return lines.join("\n"); } + +export function buildAutoRunDialogExportForCopy( + payload: AutoRunDialogExportPayload, + mode: ConversationExportMode = "default" +): string { + const includeDebug = mode === "technical"; + const lines: string[] = []; + lines.push("# Autorun dialog export"); + lines.push(`run_id: ${payload.runId || "n/a"}`); + lines.push(`case_id: ${payload.caseId || "n/a"}`); + lines.push(`session_id: ${payload.sessionId || "n/a"}`); + lines.push(`source: ${payload.source || "n/a"}`); + lines.push(`export_mode: ${mode}`); + lines.push(`exported_at: ${new Date().toISOString()}`); + lines.push(""); + + for (let index = 0; index < payload.messages.length; index += 1) { + const item = payload.messages[index]; + const role = normalizeRole(item.role); + const safeText = sanitizeConversationExportText(item.text || ""); + lines.push(`## ${index + 1}. ${role}`); + lines.push(`message_index: ${item.message_index}`); + if (item.case_id) { + lines.push(`case_id: ${item.case_id}`); + } + if (typeof item.case_message_index === "number") { + lines.push(`case_message_index: ${item.case_message_index}`); + } + if (item.created_at) { + lines.push(`created_at: ${item.created_at}`); + } + if (includeDebug) { + lines.push(`reply_type: ${item.reply_type ?? "n/a"}`); + if (item.trace_id) { + lines.push(`trace_id: ${item.trace_id}`); + } + } + lines.push(""); + lines.push(safeText || "(empty)"); + lines.push(""); + + if (includeDebug && role === "assistant" && item.debug) { + lines.push("### technical_debug_payload_json"); + lines.push("```json"); + lines.push(stringifyDebug(item.debug)); + lines.push("```"); + lines.push(""); + } + } + + if (!includeDebug) { + return lines.join("\n"); + } + + lines.push("### dialog_messages_json"); + lines.push("```json"); + lines.push(stringifyDebug(payload.messages)); + lines.push("```"); + lines.push(""); + + if ((payload.decomposition ?? []).length > 0) { + lines.push("### decomposition_json"); + lines.push("```json"); + lines.push(stringifyDebug(payload.decomposition)); + lines.push("```"); + lines.push(""); + } + + if (payload.assistantMode) { + lines.push("### assistant_mode_json"); + lines.push("```json"); + lines.push(stringifyDebug(payload.assistantMode)); + lines.push("```"); + lines.push(""); + } + + if ((payload.annotations ?? []).length > 0) { + lines.push("### annotations_json"); + lines.push("```json"); + lines.push(stringifyDebug(payload.annotations)); + lines.push("```"); + lines.push(""); + } + + if (payload.runSummary) { + lines.push("### run_summary_json"); + lines.push("```json"); + lines.push(stringifyDebug(payload.runSummary)); + lines.push("```"); + lines.push(""); + } + + if (payload.coverage) { + lines.push("### coverage_json"); + lines.push("```json"); + lines.push(stringifyDebug(payload.coverage)); + lines.push("```"); + lines.push(""); + } + + if (payload.report) { + lines.push("### run_report_json"); + lines.push("```json"); + lines.push(stringifyDebug(payload.report)); + lines.push("```"); + lines.push(""); + } + + return lines.join("\n"); +} diff --git a/scripts/domain_case_loop.py b/scripts/domain_case_loop.py index a73ea39..c71be67 100644 --- a/scripts/domain_case_loop.py +++ b/scripts/domain_case_loop.py @@ -184,6 +184,7 @@ DEFAULT_INVARIANT_SEVERITY: dict[str, str] = { "top_level_noise_present": "P0", "business_direct_answer_missing": "P0", "technical_garbage_in_answer": "P0", + "counterparty_value_flow_misrouted_to_company_profit": "P0", "answer_layering_noise": "P1", "business_answer_too_verbose": "P1", } diff --git a/scripts/domain_truth_harness.py b/scripts/domain_truth_harness.py index 06d0810..14f6e81 100644 --- a/scripts/domain_truth_harness.py +++ b/scripts/domain_truth_harness.py @@ -322,6 +322,7 @@ def append_finding( BUSINESS_REVIEW_FINDING_MESSAGES = { "technical_garbage_in_answer": "User-facing answer leaked internal runtime or MCP identifiers.", "business_direct_answer_missing": "The answer did not put the direct business answer first.", + "counterparty_value_flow_misrouted_to_company_profit": "Counterparty received/paid/net flow question was answered with company profit instead of counterparty cashflow.", "answer_layering_noise": "The answer opened with scaffolding or report framing instead of a clean business result.", "business_answer_too_verbose": "The answer is too verbose for a direct business question.", } @@ -329,6 +330,7 @@ BUSINESS_REVIEW_FINDING_MESSAGES = { BUSINESS_REVIEW_FINDING_SEVERITY = { "technical_garbage_in_answer": "critical", "business_direct_answer_missing": "critical", + "counterparty_value_flow_misrouted_to_company_profit": "critical", "answer_layering_noise": "critical", "business_answer_too_verbose": "warning", } diff --git a/scripts/review_assistant_stage1_run.py b/scripts/review_assistant_stage1_run.py index 238d676..3aa1d4b 100644 --- a/scripts/review_assistant_stage1_run.py +++ b/scripts/review_assistant_stage1_run.py @@ -111,6 +111,30 @@ SAFE_FINANCIAL_BOUNDARY_MARKERS = ( "без назначения платеж", "без договора", ) +COUNTERPARTY_VALUE_FLOW_QUESTION_RE = re.compile( + "(?:\u0441\u043a\u043e\u043b\u044c\u043a\u043e\\s+.*\u043f\u043e\u043b\u0443\u0447|\u0441\u043a\u043e\u043b\u044c\u043a\u043e\\s+.*\u0437\u0430\u043f\u043b\u0430\u0442|\u043a\u0430\u043a\u043e\u0435\\s+\u043d\u0435\u0442\u0442\u043e|\u043a\u0430\u043a\u043e\u0435\\s+\u0441\u0430\u043b\u044c\u0434\u043e)", + re.IGNORECASE, +) +COUNTERPARTY_SCOPE_QUESTION_RE = re.compile( + "(?:\u043a\u043e\u043d\u0442\u0440\u0430\u0433\u0435\u043d\u0442|\u043f\u043e\u0441\u0442\u0430\u0432\u0449\u0438\u043a|\u043f\u043e\u043a\u0443\u043f\u0430\u0442\u0435\u043b|\u043a\u043b\u0438\u0435\u043d\u0442|\u0441\u0432\u043a|\u0447\u0435\u043f\u0443\u0440\u043d\u043e\u0432)", + re.IGNORECASE, +) +COUNTERPARTY_VALUE_FLOW_ANSWER_RE = re.compile( + "(?:\u043f\u043e\u043b\u0443\u0447\u0438\u043b\u0438|\u043f\u043e\u043b\u0443\u0447\u0435\u043d\u043e|\u0437\u0430\u043f\u043b\u0430\u0442\u0438\u043b\u0438|\u043e\u043f\u043b\u0430\u0442\u0438\u043b\u0438|\u043d\u0435\u0442\u0442\u043e|\u0441\u0430\u043b\u044c\u0434\u043e|\u0432\\s+\u043d\u0430\u0448\u0443\\s+\u0441\u0442\u043e\u0440\u043e\u043d\u0443|\u0438\u0441\u0445\u043e\u0434\u044f\u0449\u0438\u0445\\s+\u043f\u043b\u0430\u0442\u0435\u0436\u0435\u0439|\u0432\u0445\u043e\u0434\u044f\u0449\u0438\u0445\\s+\u043f\u043e\u0441\u0442\u0443\u043f\u043b\u0435\u043d\u0438\u0439)", + re.IGNORECASE, +) +COUNTERPARTY_VALUE_FLOW_REQUIRED_ANSWER_RE = re.compile( + "(?:\u043f\u043e\u043b\u0443\u0447\u0438\u043b\u0438|\u043f\u043e\u043b\u0443\u0447\u0435\u043d\u043e|\u0437\u0430\u043f\u043b\u0430\u0442\u0438\u043b\u0438|\u043e\u043f\u043b\u0430\u0442\u0438\u043b\u0438|\u0438\u0441\u0445\u043e\u0434\u044f\u0449\u0438\u0445\\s+\u043f\u043b\u0430\u0442\u0435\u0436\u0435\u0439|\u0432\u0445\u043e\u0434\u044f\u0449\u0438\u0445\\s+\u043f\u043e\u0441\u0442\u0443\u043f\u043b\u0435\u043d\u0438\u0439)", + re.IGNORECASE, +) +COUNTERPARTY_ANSWER_SCOPE_RE = re.compile( + "(?:\u043a\u043e\u043d\u0442\u0440\u0430\u0433\u0435\u043d\u0442|\u043f\u043e\u0441\u0442\u0430\u0432\u0449\u0438\u043a|\u043f\u043e\u043a\u0443\u043f\u0430\u0442\u0435\u043b|\u043a\u043b\u0438\u0435\u043d\u0442|\u0441\u0432\u043a|\u0447\u0435\u043f\u0443\u0440\u043d\u043e\u0432)", + re.IGNORECASE, +) +COMPANY_PROFIT_ANSWER_RE = re.compile( + "(?:\u0447\u0438\u0441\u0442\u0430\u044f\\s+\u043f\u0440\u0438\u0431\u044b\u043b\u044c|\u043f\u0440\u0438\u0431\u044b\u043b\u044c\u044e|90/91/99|\u0444\u0438\u043d\u0440\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442|\u0443\u0431\u044b\u0442\u043e\u043a)", + re.IGNORECASE, +) def now_iso() -> str: @@ -335,6 +359,7 @@ def build_step_for_pair(pair: dict[str, Any]) -> dict[str, Any]: "answer_layering_noise": "P1", "business_answer_too_verbose": "P1", "bank_counterparty_misclassified_as_business_partner": "P1", + "counterparty_value_flow_misrouted_to_company_profit": "P0", }, } @@ -349,12 +374,41 @@ def marker_hits(text: str, markers: tuple[str, ...]) -> list[str]: return [marker for marker in markers if marker and marker.casefold() in lowered] +def detect_counterparty_value_flow_profit_mismatch(question: str, assistant_text: str) -> dict[str, Any] | None: + question_text = str(question or "") + answer_text = str(assistant_text or "") + question_flow_match = COUNTERPARTY_VALUE_FLOW_QUESTION_RE.search(question_text) + question_scope_match = COUNTERPARTY_SCOPE_QUESTION_RE.search(question_text) + if not question_flow_match or not question_scope_match: + return None + + profit_match = COMPANY_PROFIT_ANSWER_RE.search(answer_text) + if not profit_match: + return None + + value_flow_match = COUNTERPARTY_VALUE_FLOW_ANSWER_RE.search(answer_text) + required_flow_match = COUNTERPARTY_VALUE_FLOW_REQUIRED_ANSWER_RE.search(answer_text) + answer_scope_match = COUNTERPARTY_ANSWER_SCOPE_RE.search(answer_text) + if required_flow_match and answer_scope_match: + return None + + return { + "question_flow_hit": question_flow_match.group(0), + "question_scope_hit": question_scope_match.group(0), + "profit_hit": profit_match.group(0), + "value_flow_hit": value_flow_match.group(0) if value_flow_match else None, + "required_flow_hit": required_flow_match.group(0) if required_flow_match else None, + "answer_scope_hit": answer_scope_match.group(0) if answer_scope_match else None, + } + + def augment_gui_business_review(step_state: dict[str, Any]) -> dict[str, Any]: review = ( dict(step_state.get("business_first_review")) if isinstance(step_state.get("business_first_review"), dict) else {} ) + question = str(step_state.get("question_resolved") or step_state.get("question_template") or "") assistant_text = str(step_state.get("assistant_text") or "") issue_codes = [str(item) for item in review.get("issue_codes", []) if str(item).strip()] root_layers = [str(item) for item in review.get("suggested_root_cause_layers", []) if str(item).strip()] @@ -378,6 +432,17 @@ def augment_gui_business_review(step_state: dict[str, Any]) -> dict[str, Any]: if "business_semantic_role_gap" not in root_layers: root_layers.append("business_semantic_role_gap") + mismatch_details = detect_counterparty_value_flow_profit_mismatch(question, assistant_text) + if mismatch_details: + issue_code = "counterparty_value_flow_misrouted_to_company_profit" + if issue_code not in issue_codes: + issue_codes.append(issue_code) + if "followup_action_resolution_gap" not in root_layers: + root_layers.append("followup_action_resolution_gap") + if "answer_shape_mismatch" not in root_layers: + root_layers.append("answer_shape_mismatch") + review["semantic_mismatch_details"] = mismatch_details + review["technical_garbage_present"] = bool(technical_hits) review["technical_garbage_hits"] = technical_hits review["issue_codes"] = issue_codes diff --git a/scripts/test_review_assistant_stage1_run.py b/scripts/test_review_assistant_stage1_run.py index 3898005..c84a65a 100644 --- a/scripts/test_review_assistant_stage1_run.py +++ b/scripts/test_review_assistant_stage1_run.py @@ -170,6 +170,86 @@ class AssistantStage1RunReviewTests(unittest.TestCase): self.assertIn("technical_garbage_in_answer", review["summary"]["issue_counts"]) self.assertIn("bank_counterparty_misclassified_as_business_partner", review["summary"]["issue_counts"]) + def test_review_flags_counterparty_net_flow_answer_that_slips_into_company_profit(self) -> None: + with tempfile.TemporaryDirectory() as tmp: + root = Path(tmp) + sessions_dir = root / "sessions" + reports_dir = root / "reports" + run_id = "assistant-stage1-counterparty-profit-slip" + session_file = sessions_dir / f"{run_id}-SAVED-001.json" + report_file = reports_dir / f"{run_id}.md" + write_json( + session_file, + session_payload( + [ + { + "role": "user", + "text": "А теперь по Группа СВК за 2020: сколько денег получили, сколько заплатили и какое нетто?", + }, + { + "role": "assistant", + "text": "Нет, денежное операционное нетто не стоит считать чистой прибылью. " + "По закрытию 90/91/99 подтвержден учетный убыток −7 136 815,85 ₽.", + "reply_type": "factual_with_explanation", + "message_id": "a-counterparty-profit-slip", + "trace_id": "trace-counterparty-profit-slip", + "debug": {}, + }, + ] + ), + ) + report_file.parent.mkdir(parents=True, exist_ok=True) + report_file.write_text(f"# Assistant Stage 1 Eval Run\n\n- run_id: {run_id}\n", encoding="utf-8") + + review = reviewer.build_run_review( + run_id=run_id, + session_files=[session_file], + report_path=report_file, + ) + + self.assertEqual(review["summary"]["overall_business_status"], "fail") + self.assertIn("counterparty_value_flow_misrouted_to_company_profit", review["summary"]["issue_counts"]) + target_by_issue = {item["issue_code"]: item for item in review["repair_targets"]} + self.assertEqual(target_by_issue["counterparty_value_flow_misrouted_to_company_profit"]["severity"], "P0") + + def test_review_does_not_flag_counterparty_net_flow_when_received_paid_answer_is_present(self) -> None: + with tempfile.TemporaryDirectory() as tmp: + root = Path(tmp) + sessions_dir = root / "sessions" + reports_dir = root / "reports" + run_id = "assistant-stage1-counterparty-net-clean" + session_file = sessions_dir / f"{run_id}-SAVED-001.json" + report_file = reports_dir / f"{run_id}.md" + write_json( + session_file, + session_payload( + [ + { + "role": "user", + "text": "А теперь по Группа СВК за 2020: сколько денег получили, сколько заплатили и какое нетто?", + }, + { + "role": "assistant", + "text": "По Группа СВК за 2020 получили 12 093 465 ₽, заплатили 0 ₽, денежное нетто +12 093 465 ₽ в нашу сторону.", + "reply_type": "factual", + "message_id": "a-counterparty-net-clean", + "trace_id": "trace-counterparty-net-clean", + "debug": {}, + }, + ] + ), + ) + report_file.parent.mkdir(parents=True, exist_ok=True) + report_file.write_text(f"# Assistant Stage 1 Eval Run\n\n- run_id: {run_id}\n", encoding="utf-8") + + review = reviewer.build_run_review( + run_id=run_id, + session_files=[session_file], + report_path=report_file, + ) + + self.assertNotIn("counterparty_value_flow_misrouted_to_company_profit", review["summary"]["issue_counts"]) + def test_question_quality_treats_short_natural_followups_as_contextual(self) -> None: pairs = [ {"pair_index": 1, "user": {"text": "приветик - че как там дела"}},