ГЛОБАЛЬНЫЙ РЕФАКТОРИНГ АРХИТЕКТУРЫ - Этап 4.2: унифицирован fallback-ответ в 5-блочный контракт и обновлены регрессии

This commit is contained in:
dctouch 2026-04-12 01:13:39 +03:00
parent 5a8edfb4f3
commit 522575c7cc
4 changed files with 109 additions and 22 deletions

View File

@ -2651,7 +2651,23 @@ Implemented in current pass (Stage 4.1 answer-layer contract rollout, 2026-04-12
- focused Stage 4 answer pack: `5 files / 30 tests passed`;
- `npm --prefix llm_normalizer/backend run build` passed.
Status: In progress (Stage 4.1 completed; continue with quality loop on real runs/manual comments)
Implemented in current pass (Stage 4.2 boundary fallback contract alignment, 2026-04-12):
1. Reworked boundary fallback replies to the same Stage 4 human-centric contract shape:
- `Коротко`
- `Что именно проверено`
- `Что найдено`
- `Что пока не доказано`
- `Что могу сделать сейчас`
2. Preserved fallback semantics and safety behavior:
- soft refusal and clarification flow remain deterministic;
- no route/runtime contract changes.
3. Updated boundary regression assertions to the unified Stage 4 answer shape:
- `assistantBoundaryFallbackReply`.
4. Validation snapshot:
- focused Stage 4 pack passed: `4 files / 16 tests` (`assistantBoundaryFallbackReply`, `assistantSoftPolicyReply`, `assistantAnswerPolicyV11`, `assistantWave17RunRegression20260411`);
- `npm --prefix llm_normalizer/backend run build` passed.
Status: In progress (Stage 4.1-4.2 completed; continue with quality loop on real runs/manual comments)
## Stage 5 (P3): Quality Loop Driven By GUI Markup

View File

@ -1635,6 +1635,13 @@ function buildBoundaryQuickActionLine(capabilities) {
}
return `Что могу сделать сейчас: ${actions.join("; ")}.`;
}
function buildBoundaryQuickActionItems(capabilities) {
const actions = capabilities
.slice(0, 3)
.map((item) => item.replace(/:\s*/u, " — ").trim())
.filter((item) => item.length > 0);
return uniqueStrings(actions, 3);
}
function buildNaturalClarificationHints(input) {
const hints = [];
if (input.missingAnchors.period) {
@ -1686,16 +1693,27 @@ function shouldUseBoundaryFallbackReply(input) {
function buildBoundaryFallbackReply(input) {
const nearbyCapabilities = pickBoundaryCapabilityLines(input.userMessage, 3);
const quickActionLine = buildBoundaryQuickActionLine(nearbyCapabilities);
const quickActionItems = buildBoundaryQuickActionItems(nearbyCapabilities);
if (input.focusDomain === null) {
const heading = pickDeterministicBoundaryVariant(input.userMessage, [
"По этому запросу у меня нет надежного доменного покрытия, поэтому даю мягкий отказ вместо технического шаблона.",
"Сейчас у меня нет надежного доменного маршрута по этому запросу, поэтому даю мягкий отказ вместо шаблонной технички."
]);
return sanitizeUserFacingReply([
heading,
nearbyCapabilities.length > 0 ? `Что могу сделать рядом по смыслу:\n${formatList(nearbyCapabilities)}` : "",
quickActionLine ?? "",
"Переформулируй вопрос через один из вариантов выше, и я сразу перейду к проверке по данным 1С."
`Коротко: ${ensureSentence(heading)}`,
`Что именно проверено:\n${formatList([
"Проверен доступный контур 1С и возможность маршрутизации запроса.",
"Надежный доменный путь для текущей формулировки не подтвержден."
])}`,
`Что найдено:\n${formatList(nearbyCapabilities.length > 0
? nearbyCapabilities
: ["Близких поддерживаемых сценариев по текущей формулировке не найдено."])}`,
`Что пока не доказано:\n${formatList([
"Запрос не подтвержден в поддерживаемом контуре как надежный бухгалтерский сценарий."
])}`,
`Что могу сделать сейчас:\n${formatList(quickActionItems.length > 0
? quickActionItems
: [quickActionLine ?? "Переформулируйте вопрос через период, счет или объект, и я сразу продолжу проверку."])}`
]
.filter(Boolean)
.join("\n\n"));
@ -1709,10 +1727,20 @@ function buildBoundaryFallbackReply(input) {
`По сценарию ${formatNarrativeDomainLabel(input.focusDomain)} пока не хватает подтвержденной опоры для надежного вывода.`
]);
return sanitizeUserFacingReply([
domainHeading,
clarificationHints.length > 0 ? `Чтобы сразу перейти к проверке, уточни:\n${formatList(clarificationHints)}` : "",
nearbyCapabilities.length > 0 ? `Если удобнее, могу начать с близкого сценария:\n${formatList(nearbyCapabilities.slice(0, 2))}` : "",
quickActionLine ?? ""
`Коротко: ${ensureSentence(domainHeading)}`,
`Что именно проверено:\n${formatList([
`Проверен сценарий ${formatNarrativeDomainLabel(input.focusDomain)} в текущем контуре 1С.`,
"Оценена достаточность подтвержденной опоры для прямого вывода."
])}`,
`Что найдено:\n${formatList(nearbyCapabilities.length > 0
? nearbyCapabilities.slice(0, 2)
: ["Подтвержденных близких сценариев в текущем проходе не найдено."])}`,
`Что пока не доказано:\n${formatList(clarificationHints.length > 0
? clarificationHints
: ["Без дополнительного ориентира (период, объект или контрагент) вывод останется частичным."])}`,
`Что могу сделать сейчас:\n${formatList(quickActionItems.length > 0
? quickActionItems
: [quickActionLine ?? "Добавьте один уточняющий ориентир, и я продолжу проверку без смены контура."])}`
]
.filter(Boolean)
.join("\n\n"));

View File

@ -1932,6 +1932,14 @@ function buildBoundaryQuickActionLine(capabilities: string[]): string | null {
return `Что могу сделать сейчас: ${actions.join("; ")}.`;
}
function buildBoundaryQuickActionItems(capabilities: string[]): string[] {
const actions = capabilities
.slice(0, 3)
.map((item) => item.replace(/:\s*/u, " ").trim())
.filter((item) => item.length > 0);
return uniqueStrings(actions, 3);
}
function buildNaturalClarificationHints(input: {
missingAnchors: MissingAnchors;
coverageReport: RequirementCoverageReport;
@ -2006,6 +2014,7 @@ function buildBoundaryFallbackReply(input: {
}): string {
const nearbyCapabilities = pickBoundaryCapabilityLines(input.userMessage, 3);
const quickActionLine = buildBoundaryQuickActionLine(nearbyCapabilities);
const quickActionItems = buildBoundaryQuickActionItems(nearbyCapabilities);
if (input.focusDomain === null) {
const heading = pickDeterministicBoundaryVariant(
input.userMessage,
@ -2016,10 +2025,24 @@ function buildBoundaryFallbackReply(input: {
);
return sanitizeUserFacingReply(
[
heading,
nearbyCapabilities.length > 0 ? `Что могу сделать рядом по смыслу:\n${formatList(nearbyCapabilities)}` : "",
quickActionLine ?? "",
"Переформулируй вопрос через один из вариантов выше, и я сразу перейду к проверке по данным 1С."
`Коротко: ${ensureSentence(heading)}`,
`Что именно проверено:\n${formatList([
"Проверен доступный контур 1С и возможность маршрутизации запроса.",
"Надежный доменный путь для текущей формулировки не подтвержден."
])}`,
`Что найдено:\n${formatList(
nearbyCapabilities.length > 0
? nearbyCapabilities
: ["Близких поддерживаемых сценариев по текущей формулировке не найдено."]
)}`,
`Что пока не доказано:\n${formatList([
"Запрос не подтвержден в поддерживаемом контуре как надежный бухгалтерский сценарий."
])}`,
`Что могу сделать сейчас:\n${formatList(
quickActionItems.length > 0
? quickActionItems
: [quickActionLine ?? "Переформулируйте вопрос через период, счет или объект, и я сразу продолжу проверку."]
)}`
]
.filter(Boolean)
.join("\n\n")
@ -2039,10 +2062,26 @@ function buildBoundaryFallbackReply(input: {
);
return sanitizeUserFacingReply(
[
domainHeading,
clarificationHints.length > 0 ? `Чтобы сразу перейти к проверке, уточни:\n${formatList(clarificationHints)}` : "",
nearbyCapabilities.length > 0 ? `Если удобнее, могу начать с близкого сценария:\n${formatList(nearbyCapabilities.slice(0, 2))}` : "",
quickActionLine ?? ""
`Коротко: ${ensureSentence(domainHeading)}`,
`Что именно проверено:\n${formatList([
`Проверен сценарий ${formatNarrativeDomainLabel(input.focusDomain)} в текущем контуре 1С.`,
"Оценена достаточность подтвержденной опоры для прямого вывода."
])}`,
`Что найдено:\n${formatList(
nearbyCapabilities.length > 0
? nearbyCapabilities.slice(0, 2)
: ["Подтвержденных близких сценариев в текущем проходе не найдено."]
)}`,
`Что пока не доказано:\n${formatList(
clarificationHints.length > 0
? clarificationHints
: ["Без дополнительного ориентира (период, объект или контрагент) вывод останется частичным."]
)}`,
`Что могу сделать сейчас:\n${formatList(
quickActionItems.length > 0
? quickActionItems
: [quickActionLine ?? "Добавьте один уточняющий ориентир, и я продолжу проверку без смены контура."]
)}`
]
.filter(Boolean)
.join("\n\n")

View File

@ -53,7 +53,9 @@ describe("assistant boundary fallback reply", () => {
expect(output.reply_type).toBe("clarification_required");
expect(output.assistant_reply).toMatch(/мягкий отказ/i);
expect(output.assistant_reply).toContain("Что могу сделать рядом по смыслу:");
expect(output.assistant_reply).toContain("Что именно проверено:");
expect(output.assistant_reply).toContain("Что найдено:");
expect(output.assistant_reply).toContain("Что пока не доказано:");
expect(output.assistant_reply).toContain("Что могу сделать сейчас:");
expect(output.assistant_reply).not.toContain("Что сломано:");
});
@ -78,8 +80,9 @@ describe("assistant boundary fallback reply", () => {
expect(output.reply_type).toBe("clarification_required");
expect(output.assistant_reply).toMatch(/не могу надежно ответить по сценарию|не хватает подтвержденной опоры/i);
expect(output.assistant_reply).toContain("Чтобы сразу перейти к проверке, уточни:");
expect(output.assistant_reply).toContain("Если удобнее, могу начать с близкого сценария:");
expect(output.assistant_reply).toContain("Что именно проверено:");
expect(output.assistant_reply).toContain("Что найдено:");
expect(output.assistant_reply).toContain("Что пока не доказано:");
expect(output.assistant_reply).toContain("Что могу сделать сейчас:");
expect(output.assistant_reply).not.toContain("Что сломано:");
});
@ -161,8 +164,9 @@ describe("assistant boundary fallback reply", () => {
});
expect(output.reply_type).toBe("partial_coverage");
expect(output.assistant_reply).toContain("Что могу сделать рядом по смыслу:");
expect(output.assistant_reply).toContain("Что найдено:");
expect(output.assistant_reply).toContain("Что пока не доказано:");
expect(output.assistant_reply).toContain("Что могу сделать сейчас:");
expect(output.assistant_reply).not.toContain("Что сломано:");
});
});
});