import assert from 'node:assert/strict' import { readFileSync } from 'node:fs' import test from 'node:test' import { fileURLToPath } from 'node:url' import { buildAiAdviceViewModel, buildAttachmentInsightViewModel, buildAttachmentRiskCards, buildClaimSummaryRiskCards, buildItemClaimRiskState, extractRiskTagsFromText, filterRiskCardsByBusinessStage, resolveRiskTags, resolveRiskTagTone } from '../src/views/scripts/travelRequestDetailInsights.js' import { buildExpenseDraftIssues, buildExpenseItemViewModel, buildDraftBlockingIssues, rebuildExpenseItems, buildStandardAdjustmentMap, isApplicationDocumentRequest } from '../src/views/scripts/travelRequestDetailExpenseModel.js' import { buildEmployeeProfileAdviceItems, buildTravelReceiptMaterialPrompts } from '../src/views/scripts/travelRequestDetailAdviceModel.js' import { buildStandardAdjustmentPayload, filterSubmitterResolvedRiskCards, isRiskCardMissingExpenseNote } from '../src/views/scripts/travelRequestDetailStandardAdjustment.js' const detailViewTemplate = readFileSync( fileURLToPath(new URL('../src/views/TravelRequestDetailView.vue', import.meta.url)), 'utf8' ) const detailViewComponentScript = readFileSync( fileURLToPath(new URL('../src/views/scripts/TravelRequestDetailView.js', import.meta.url)), 'utf8' ) const detailViewInsights = readFileSync( fileURLToPath(new URL('../src/views/scripts/travelRequestDetailInsights.js', import.meta.url)), 'utf8' ) const detailAiAdviceModelScript = readFileSync( fileURLToPath(new URL('../src/views/scripts/travelRequestDetailAiAdviceModel.js', import.meta.url)), 'utf8' ) const detailExpenseModelScript = readFileSync( fileURLToPath(new URL('../src/views/scripts/travelRequestDetailExpenseModel.js', import.meta.url)), 'utf8' ) const detailViewSetupScript = readFileSync( fileURLToPath(new URL('../src/views/scripts/travelRequestDetailSetup.js', import.meta.url)), 'utf8' ) const detailSmartEntryScript = readFileSync( fileURLToPath(new URL('../src/views/scripts/travelRequestDetailSmartEntryRecognition.js', import.meta.url)), 'utf8' ) const detailAttachmentPreviewScript = readFileSync( fileURLToPath(new URL('../src/views/scripts/useTravelRequestDetailAttachmentPreview.js', import.meta.url)), 'utf8' ) const detailExpenseEditorScript = readFileSync( fileURLToPath(new URL('../src/views/scripts/useTravelRequestDetailExpenseEditor.js', import.meta.url)), 'utf8' ) const detailRiskSubmitScript = readFileSync( fileURLToPath(new URL('../src/views/scripts/useTravelRequestDetailRiskSubmit.js', import.meta.url)), 'utf8' ) const detailApprovalFlowScript = readFileSync( fileURLToPath(new URL('../src/views/scripts/useTravelRequestDetailApprovalFlow.js', import.meta.url)), 'utf8' ) const detailEmployeeRiskProfileScript = readFileSync( fileURLToPath(new URL('../src/views/scripts/useTravelRequestEmployeeRiskProfile.js', import.meta.url)), 'utf8' ) const detailViewScript = [ detailViewComponentScript, detailViewSetupScript, detailSmartEntryScript, detailAttachmentPreviewScript, detailExpenseEditorScript, detailRiskSubmitScript, detailApprovalFlowScript, detailEmployeeRiskProfileScript, detailExpenseModelScript ].join('\n') const detailViewStyle = readFileSync( fileURLToPath(new URL('../src/assets/styles/views/travel-request-detail-view.css', import.meta.url)), 'utf8' ) const requestsComposableScript = readFileSync( fileURLToPath(new URL('../src/composables/useRequests.js', import.meta.url)), 'utf8' ) const approvalCenterTemplate = readFileSync( fileURLToPath(new URL('../src/views/ApprovalCenterView.vue', import.meta.url)), 'utf8' ) const approvalCenterScript = readFileSync( fileURLToPath(new URL('../src/views/scripts/ApprovalCenterView.js', import.meta.url)), 'utf8' ) const returnReasonDialog = readFileSync( fileURLToPath(new URL('../src/components/shared/ReturnReasonDialog.vue', import.meta.url)), 'utf8' ) const stageRiskAdviceCard = readFileSync( fileURLToPath(new URL('../src/components/travel/StageRiskAdviceCard.vue', import.meta.url)), 'utf8' ) const stageRiskAdviceStyles = readFileSync( fileURLToPath(new URL('../src/assets/styles/components/stage-risk-advice-card.css', import.meta.url)), 'utf8' ) const detailHeroTemplate = readFileSync( fileURLToPath(new URL('../src/components/travel/TravelRequestDetailHero.vue', import.meta.url)), 'utf8' ) const relatedApplicationCardTemplate = readFileSync( fileURLToPath(new URL('../src/components/travel/TravelRequestRelatedApplicationCard.vue', import.meta.url)), 'utf8' ) const progressCardTemplate = readFileSync( fileURLToPath(new URL('../src/components/travel/TravelRequestProgressCard.vue', import.meta.url)), 'utf8' ) const attachmentMeta = { file_name: 'taxi-invoice.pdf', media_type: 'application/pdf', previewable: true, document_info: { document_type: 'taxi_receipt', document_type_label: '出租车/网约车票据', fields: [ { label: '金额', value: '121.54' }, { label: '日期', value: '2026-03-04' } ] }, requirement_check: { matches: false, message: '附件类型与当前费用项目不匹配。' }, analysis: { severity: 'high', label: '高风险', headline: '票据类型不匹配', summary: '交通票据挂在办公用品费明细下。', points: ['票据识别为出租车/网约车票据', '当前费用项目为办公用品费'], suggestion: '把费用项目调整为交通费,或更换为办公用品票据。' } } test('attachment insight exposes recognition fields and rule basis', () => { const insight = buildAttachmentInsightViewModel(attachmentMeta, { name: '办公用品费', itemType: 'office' }) assert.equal(insight.documentTypeLabel, '出租车/网约车票据') assert.equal(insight.requirementLabel, '不符合当前费用类型') assert.deepEqual(insight.fields, ['金额:121.54', '日期:2026-03-04']) assert.ok(insight.ruleBasis.some((item) => item.includes('附件类型与当前费用项目不匹配'))) }) test('AI advice card splits every attachment risk point with basis and suggestion', () => { const riskCards = buildAttachmentRiskCards({ expenseItems: [ { id: 'item-1', name: '办公用品费', invoiceId: 'taxi-invoice.pdf' } ], attachmentMetaByItemId: { 'item-1': attachmentMeta } }) const advice = buildAiAdviceViewModel({ completionItems: [], riskCards }) assert.equal(riskCards.length, 2) assert.equal(advice.badge, '优先整改') assert.equal(advice.riskCards.length, 2) assert.ok(advice.riskCards.every((card) => card.ruleBasis.length > 0)) assert.ok(advice.riskCards.every((card) => card.suggestion.includes('费用项目调整为交通费'))) }) test('risk cards carry severity and domain tags for statistics', () => { const hotelRisk = { tone: 'high', title: '住宿超标待说明', risk: '住宿标准:北京酒店 800 元/晚超出报销标准。' } const trafficRisk = { tone: 'medium', title: '交通票据提醒', risk: '火车票说明格式待调整。' } assert.deepEqual(resolveRiskTags(hotelRisk), ['#high_risk', '#hotel']) assert.deepEqual(resolveRiskTags(trafficRisk), ['#middle_risk', '#traffic']) assert.equal(resolveRiskTagTone('#hotel'), 'hotel') assert.deepEqual(extractRiskTagsFromText('超标说明:#high_risk #hotel 原因'), ['#high_risk', '#hotel']) }) test('AI advice splits claim attachment risk flags into specific points', () => { const riskCards = buildAttachmentRiskCards({ claimRiskFlags: [ { source: 'attachment_analysis', severity: 'medium', label: '中风险', message: '费用明细第 2 条:日期字段:未识别到开票日期。', summary: '当前附件可见部分内容,但金额、用途、日期或附件类型仍有缺失或不一致。', points: [ '日期字段:未识别到开票日期或业务发生日期。', '金额字段:附件识别金额 300.00 元与报销金额 88.00 元不一致。' ] } ] }) assert.equal(riskCards.length, 2) assert.equal(riskCards[0].risk, '日期字段:未识别到开票日期或业务发生日期。') assert.equal(riskCards[1].risk, '金额字段:附件识别金额 300.00 元与报销金额 88.00 元不一致。') assert.ok(riskCards[0].ruleBasis.some((item) => item.includes('风险汇总'))) }) test('AI advice keeps visible risk flags when backend uses tone instead of severity', () => { const riskCards = buildAttachmentRiskCards({ claimRiskFlags: [ { source: 'submission_review', tone: 'medium', label: '中风险', message: '直属领导缺失,当前单据需审批环节补充分配。' } ] }) assert.equal(riskCards.length, 1) assert.equal(riskCards[0].tone, 'medium') assert.equal(riskCards[0].risk, '直属领导缺失,当前单据需审批环节补充分配。') assert.ok(riskCards[0].ruleBasis.some((item) => item.includes('审批链校验'))) assert.ok(riskCards[0].suggestion.includes('员工档案')) }) test('risk card badge only shows severity while title keeps business risk name', () => { const riskCards = buildAttachmentRiskCards({ claimRiskFlags: [ { source: 'attachment_analysis', severity: 'high', label: '票据日期超出差旅行程高风险', message: '酒店发票日期为 2 月,晚于已批准 6 月差旅行程范围。' } ] }) assert.equal(riskCards.length, 1) assert.equal(riskCards[0].tone, 'high') assert.equal(riskCards[0].label, '高风险') assert.equal(riskCards[0].title, '票据日期超出差旅行程高风险') }) test('AI advice falls back to claim risk summary instead of showing an empty risk area', () => { const riskCards = buildClaimSummaryRiskCards({ riskSummary: '自动检测发现 1 条中风险附件,已随单流转给审批人复核。' }) assert.equal(riskCards.length, 1) assert.equal(riskCards[0].businessStage, 'reimbursement') assert.equal(riskCards[0].tone, 'medium') assert.equal(riskCards[0].label, '中风险') assert.match(riskCards[0].risk, /中风险附件/) assert.ok(riskCards[0].ruleBasis.some((item) => item.includes('风险汇总'))) assert.ok(riskCards[0].suggestion.includes('附件预览')) }) test('risk cards carry structured business stage for approval advice filtering', () => { const applicationCards = buildAttachmentRiskCards({ businessStage: 'expense_application', claimRiskFlags: [ { source: 'policy_review', severity: 'medium', label: '预算风险', message: '申请金额可能占用预算余额,需要预算管理者复核。' } ] }) const reimbursementCards = buildAttachmentRiskCards({ businessStage: 'expense_application', claimRiskFlags: [ { source: 'attachment_analysis', business_stage: 'reimbursement', severity: 'high', label: '票据风险', message: '报销票据城市与行程城市不一致。' } ] }) const legacyAttachmentCards = buildAttachmentRiskCards({ businessStage: 'expense_application', claimRiskFlags: [ { source: 'attachment_analysis', severity: 'medium', label: '附件风险', message: '差旅附件暂未识别到有效票据信息,请重新上传清晰附件或人工补录。' } ] }) const summaryCards = buildClaimSummaryRiskCards({ documentTypeCode: 'application', riskSummary: '预算余额不足,申请需要复核。' }) const attachmentSummaryCards = buildClaimSummaryRiskCards({ documentTypeCode: 'application', businessStage: 'expense_application', riskSummary: '差旅附件暂未识别到有效票据信息,请重新上传清晰附件或人工补录。' }) assert.equal(applicationCards.length, 1) assert.equal(applicationCards[0].businessStage, 'expense_application') assert.equal(reimbursementCards.length, 1) assert.equal(reimbursementCards[0].businessStage, 'reimbursement') assert.equal(legacyAttachmentCards.length, 1) assert.equal(legacyAttachmentCards[0].businessStage, 'reimbursement') assert.deepEqual( filterRiskCardsByBusinessStage( [...applicationCards, ...reimbursementCards, ...legacyAttachmentCards], 'expense_application' ).map((card) => card.risk), ['申请金额可能占用预算余额,需要预算管理者复核。'] ) assert.equal(summaryCards.length, 1) assert.equal(summaryCards[0].businessStage, 'expense_application') assert.equal(attachmentSummaryCards.length, 1) assert.equal(attachmentSummaryCards[0].businessStage, 'reimbursement') assert.deepEqual(filterRiskCardsByBusinessStage(attachmentSummaryCards, 'expense_application'), []) }) test('stage risk advice card focuses on document risks without profile or budget boards', () => { assert.match(stageRiskAdviceCard, /employee-risk-decision-panel/) assert.match(stageRiskAdviceCard, /综合审核结论/) assert.match(stageRiskAdviceCard, /是否建议通过/) assert.match(stageRiskAdviceCard, /\{\{ decisionBadgeLabel \}\}/) assert.match(stageRiskAdviceCard, /employee-risk-review-summary/) assert.match(stageRiskAdviceCard, /reviewSummaryItems/) assert.match(stageRiskAdviceCard, /风险概览/) assert.match(stageRiskAdviceCard, /重点依据/) assert.match(stageRiskAdviceCard, /审核建议/) assert.match(stageRiskAdviceCard, /stageRiskFactSummary/) assert.match(stageRiskAdviceCard, /stageReviewBasisSummary/) assert.match(stageRiskAdviceCard, /compactEvidenceItems/) assert.match(stageRiskAdviceCard, /stageBasisTitle/) assert.match(stageRiskAdviceCard, /stageBasisHint/) assert.match(stageRiskAdviceCard, /employee-risk-profile-section/) assert.match(stageRiskAdviceCard, /employee-risk-profile-list/) assert.match(stageRiskAdviceCard, /
{ const riskCards = buildAttachmentRiskCards({ claimRiskFlags: [ { source: 'manual_approval', severity: 'info', label: '领导审批通过', message: '同意' }, { source: 'finance_approval', severity: 'info', label: '财务审核通过', message: '周晓彤 已完成财务审核,进入归档入账。' }, { source: 'application_link_sync', event_type: 'expense_application_reimbursement_deleted', severity: 'warning', label: '关联报销单已删除', message: '关联报销单 RDELETE01 已删除,申请单已回到待关联状态。' } ] }) assert.deepEqual(riskCards, []) assert.deepEqual(buildClaimSummaryRiskCards({ riskSummary: '同意' }), []) assert.deepEqual(buildClaimSummaryRiskCards({ riskSummary: '周晓彤 已完成财务审核,进入归档入账。' }), []) }) test('expense row risk state falls back to claim item risk flags', () => { const state = buildItemClaimRiskState( { id: 'hotel-item', name: '住宿费' }, [ { source: 'attachment_analysis', item_id: 'hotel-item', severity: 'high', label: '高风险', message: '费用明细第 2 条:住宿标准:当前酒店识别金额约 880.00 元/晚。', summary: '当前住宿票据金额超过规则中心差旅住宿标准。', points: ['住宿标准:当前酒店识别金额约 880.00 元/晚。'] } ] ) assert.equal(state.tone, 'high') assert.equal(state.label, '高风险') assert.match(state.summary, /住宿票据金额超过/) assert.deepEqual(state.points, ['住宿标准:当前酒店识别金额约 880.00 元/晚。']) }) test('attachment risk cards do not duplicate claim fallback flags for the same item', () => { const riskCards = buildAttachmentRiskCards({ expenseItems: [ { id: 'hotel-item', name: '住宿费', invoiceId: 'hotel-risk.png' } ], attachmentMetaByItemId: { 'hotel-item': { analysis: { severity: 'high', label: '高风险', headline: 'AI提示:住宿金额超出报销标准', summary: '当前住宿票据金额超过规则中心差旅住宿标准。', points: ['住宿标准:当前酒店识别金额约 880.00 元/晚。'], suggestion: '请补充超标说明。' } } }, claimRiskFlags: [ { source: 'attachment_analysis', item_id: 'hotel-item', severity: 'high', label: '高风险', message: '费用明细第 1 条:住宿标准:当前酒店识别金额约 880.00 元/晚。', summary: '当前住宿票据金额超过规则中心差旅住宿标准。', points: ['住宿标准:当前酒店识别金额约 880.00 元/晚。'] } ] }) assert.equal(riskCards.length, 1) assert.equal(riskCards[0].risk, '住宿标准:当前酒店识别金额约 880.00 元/晚。') }) test('AI advice hides generic auto review summaries when a specific hotel over-standard risk exists', () => { const riskCards = buildAttachmentRiskCards({ expenseItems: [ { id: 'hotel-over-standard-item', name: '住宿票', itemType: 'hotel_ticket', invoiceId: 'hotel-over-standard.png' } ], attachmentMetaByItemId: { 'hotel-over-standard-item': { analysis: { severity: 'high', label: '高风险', headline: 'AI提示:住宿金额超出报销标准', summary: '当前住宿票据金额超过规则中心差旅住宿标准。', points: ['住宿标准:P5在上海的住宿标准为 250.00 元/晚,当前酒店识别金额约 362.00 元/晚。'], suggestion: '请补充超标说明。' } } }, claimRiskFlags: [ { source: 'submission_review', severity: 'high', label: '自动检测重点复核', message: '自动检测发现 1 条高风险附件,已随单流转给审批人重点复核。' }, { source: 'submission_review', severity: 'high', label: '住宿超标待说明', message: 'P5 职级在上海的住宿标准为 250.00 元/晚,当前酒店识别金额约 362.00 元/晚。 当前未识别到超标说明,请先补充原因。' }, { source: 'submission_review', severity: 'medium', label: '自动检测重点复核', message: '自动检测发现需审批重点关注事项:存在高风险票据,需审批人重点复核;住宿金额超出当前职级差标,且未补充超标说明。' } ] }) assert.equal(riskCards.length, 1) assert.equal(riskCards[0].title, '住宿金额超出报销标准') assert.equal(riskCards[0].tone, 'high') }) test('AI advice view model exposes grouped completion and risk sections', () => { const advice = buildAiAdviceViewModel({ completionItems: ['补充业务地点', '补充报销金额'], riskCards: [ { id: 'risk-1', tone: 'high', label: '高风险', title: '票据类型不匹配', risk: '交通票据挂在办公用品费明细下。', ruleBasis: ['附件类型与当前费用项目不匹配。'], suggestion: '把费用项目调整为交通费。' } ] }) assert.equal(advice.sections.length, 2) assert.deepEqual( advice.sections.map((section) => ({ title: section.title, kind: section.kind })), [ { title: '建议补充字段', kind: 'completion' }, { title: '已知存在风险(1项)', kind: 'risk' } ] ) assert.deepEqual(advice.sections[0].items, ['补充业务地点', '补充报销金额']) assert.equal(advice.sections[1].items.length, 1) }) test('AI advice view model sorts and displays every risk card', () => { const riskCards = [ { id: 'risk-1', tone: 'medium', label: '中风险', title: '中风险一', risk: '中风险一。' }, { id: 'risk-2', tone: 'high', label: '高风险', title: '高风险一', risk: '高风险一。' }, { id: 'risk-3', tone: 'medium', label: '中风险', title: '中风险二', risk: '中风险二。' }, { id: 'risk-4', tone: 'high', label: '高风险', title: '高风险二', risk: '高风险二。' } ] const advice = buildAiAdviceViewModel({ completionItems: [], riskCards }) const riskSection = advice.sections.find((section) => section.kind === 'risk') assert.equal(advice.riskCards.length, 4) assert.equal(riskSection.items.length, 4) assert.equal(riskSection.hiddenCount, undefined) assert.deepEqual(riskSection.items.map((item) => item.id), ['risk-2', 'risk-4', 'risk-1', 'risk-3']) }) test('AI advice hides lower severity duplicate route explanation risks', () => { const advice = buildAiAdviceViewModel({ riskCards: [ { id: 'route-high', tone: 'high', label: '高风险', title: '多城市行程待说明', risk: '检测到本次差旅涉及 深圳 多个目的地,但当前报销事由未说明中转、多地拜访或改签原因。', itemIds: ['train-transfer', 'train-transfer-return'] }, { id: 'route-medium', tone: 'medium', label: '中风险', title: '多城市行程缺少说明中风险', risk: '本次报销识别到多城市行程(上海、武汉、深圳),但事由中未说明中转、多地拜访或改签原因。', itemIds: ['train-transfer', 'train-transfer-return'] }, { id: 'hotel-high', tone: 'high', label: '高风险', title: '住宿金额超出报销标准', risk: '住宿金额超出当前职级报销标准。', itemIds: ['hotel-item'] } ] }) const riskSection = advice.sections.find((section) => section.kind === 'risk') assert.deepEqual(advice.riskCards.map((item) => item.id), ['route-high', 'hotel-high']) assert.deepEqual(riskSection.items.map((item) => item.id), ['route-high', 'hotel-high']) assert.equal(riskSection.totalCount, 2) }) test('AI advice view model omits empty sections', () => { const readyAdvice = buildAiAdviceViewModel({ completionItems: [], riskCards: [] }) const completionOnlyAdvice = buildAiAdviceViewModel({ completionItems: ['补充业务地点'], riskCards: [] }) const riskOnlyAdvice = buildAiAdviceViewModel({ completionItems: [], riskCards: [ { id: 'risk-1', tone: 'medium', label: '中风险', title: '说明不完整', risk: '缺少业务背景。', ruleBasis: ['系统预审规则命中该风险提示。'], suggestion: '补充说明。' } ] }) assert.deepEqual(readyAdvice.sections, []) assert.equal(readyAdvice.badge, '可以提交') assert.deepEqual(completionOnlyAdvice.sections.map((section) => section.title), ['建议补充字段']) assert.deepEqual(riskOnlyAdvice.sections.map((section) => section.title), ['已知存在风险(1项)']) }) test('AI advice separates material prompts and profile advice from risk cards', () => { const advice = buildAiAdviceViewModel({ completionItems: [], materialPrompts: ['当前包含 1 条住宿费用明细,但暂未关联住宿发票或酒店水单。'], profileAdviceItems: ['历史退单建议:近 90 天存在 1 次退单或退回记录。'], riskCards: [] }) assert.equal(advice.riskCards.length, 0) assert.equal(advice.badge, '建议关注') assert.deepEqual( advice.sections.map((section) => ({ kind: section.kind, title: section.title })), [ { kind: 'material', title: '材料补充提示' }, { kind: 'profile', title: '历史操作建议' } ] ) }) test('AI advice template renders grouped section titles with completion before risk', () => { assert.match(detailViewTemplate, /v-if="showAiAdvicePanel" class="detail-card panel validation-card"/) assert.match(detailViewTemplate, /

\{\{ aiAdviceTitle \}\}<\/h3>/) assert.match(detailViewTemplate, /

\{\{ aiAdviceHint \}\}<\/p>/) assert.doesNotMatch(detailViewScript, /AI预审已完成,请按风险提示补充原因或进入下一步。/) assert.match(detailViewScript, /businessStage: currentBusinessStage/) assert.match(detailViewScript, /filterRiskCardsByBusinessStage/) assert.match(detailViewScript, /const summaryRiskCards = filterRiskCardsByBusinessStage/) assert.match(detailViewScript, /buildClaimSummaryRiskCards\(\{/) assert.match(detailViewScript, /const canViewApprovalRiskAdvice = computed/) assert.match(detailViewScript, /!isCurrentApplicant\.value/) assert.match(detailViewScript, /const hasVisibleRiskCards = computed/) assert.match(detailViewScript, /const showCompactSafeAdvice = computed/) assert.match(detailViewScript, /const showAiAdvicePanel = computed\(\(\) => \(/) assert.match(detailViewScript, /isCurrentApplicant\.value && hasVisibleRiskCards\.value/) assert.match(detailViewScript, /isApplicationDocument\.value \? '申请风险提示' : '风险提示'/) assert.match(detailViewScript, /return isEditableRequest\.value \? 'AI建议' : '风险提示'/) assert.doesNotMatch(detailViewScript, /return '报销风险提示'/) assert.match(detailViewScript, /canViewApprovalRiskAdvice\.value && aiAdvice\.value\.riskCards\.length > 0/) assert.match(detailViewScript, /buildTravelReceiptMaterialPrompts\(request\.value, expenseItems\.value\)/) assert.match(detailViewScript, /buildEmployeeProfileAdviceItems\(employeeRiskProfile\.value\)/) assert.match(detailViewScript, /fetchEmployeeLatestProfile\(employeeId/) assert.doesNotMatch(detailViewScript, /hasAiPreReviewResult\.value/) assert.match(detailViewTemplate, /v-if="aiAdvice\.sections\.length" class="validation-sections"/) assert.match(detailViewTemplate, /v-for="section in aiAdvice\.sections"/) assert.match(detailViewTemplate, /validation-section--\$\{section\.kind\}/) assert.match(detailViewTemplate, /

\{\{ section\.title \}\}<\/h4>/) assert.match(detailViewTemplate, /v-if="section\.kind !== 'risk'" class="validation-list"/) assert.match(detailViewTemplate, /v-else class="risk-advice-list"/) assert.ok( detailViewTemplate.indexOf("section.kind !== 'risk'") < detailViewTemplate.indexOf('risk-advice-card') ) }) test('AI advice risk section keeps compact risk prompt styling', () => { assert.match(detailViewTemplate, /class="\['risk-advice-card', card\.tone, \{ 'is-highlighted': isHighlightedRiskCard\(card\) \}\]"/) assert.match(detailViewTemplate, /class="risk-advice-point"/) assert.match(detailViewTemplate, /class="risk-advice-compact-meta"/) assert.match(detailViewTemplate, /\{\{ card\.ruleBasis\[0\] \}\}/) assert.doesNotMatch(detailViewTemplate, /risk-advice-detail-grid/) assert.doesNotMatch(detailViewTemplate, /
风险事实<\/dt>/) assert.doesNotMatch(detailViewTemplate, /section\.hiddenCount/) assert.doesNotMatch(detailViewTemplate, /risk-advice-more/) assert.doesNotMatch(detailViewTemplate, /card\.tags\?\.length/) assert.doesNotMatch(detailViewTemplate, /risk-card-tag-list/) assert.doesNotMatch(detailViewTemplate, /risk-note-tag/) assert.match(detailViewScript, /tags: resolveRiskTags\(card\)/) assert.match(detailAiAdviceModelScript, /const sortedRiskCards = sortRiskCardsByTone\(normalizedRiskCards\)/) assert.doesNotMatch(detailViewInsights, /visibleRiskCards/) assert.doesNotMatch(detailViewInsights, /hiddenCount/) assert.match(detailViewStyle, /\.validation-card \{\s*border: 1px solid #e5e7eb;/) assert.match(detailViewStyle, /\.validation-section--risk \.risk-advice-list \{\s*display: grid;\s*gap: 8px;\s*max-height: 360px;/) assert.match(detailViewStyle, /\.validation-section--risk \.risk-advice-card \{\s*position: relative;\s*display: grid;\s*grid-template-columns: minmax\(0, 1\.1fr\) minmax\(220px, \.9fr\);/) assert.match(detailViewStyle, /\.validation-section--risk \.risk-advice-card\.low/) assert.match(detailViewStyle, /\.risk-advice-card\.low/) assert.doesNotMatch(detailViewStyle, /\.risk-note-tag/) assert.match(detailViewStyle, /\.risk-advice-compact-meta span,\s*\.risk-advice-compact-meta em \{\s*margin: 0;/) assert.doesNotMatch(detailViewStyle, /\.risk-advice-detail-grid/) assert.doesNotMatch(detailViewStyle, /\.risk-advice-more/) }) test('expense rows show a major-risk warning icon before time', () => { assert.match(detailViewTemplate, /'has-major-risk': hasExpenseRiskIndicator\(item\)/) assert.match(detailViewTemplate, /class="expense-time-content"/) assert.match(detailViewTemplate, /class="expense-risk-indicator"/) assert.match(detailViewTemplate, /class="expense-risk-indicator-placeholder"/) assert.match(detailViewTemplate, /@click="focusExpenseRisk\(item\)"/) assert.match(detailViewStyle, /\.expense-time-content \{/) assert.match(detailViewStyle, /\.expense-risk-indicator \{/) assert.match(detailViewStyle, /\.expense-risk-indicator,\s*\.expense-risk-indicator-placeholder \{/) assert.match(detailViewScript, /function hasExpenseRiskIndicator\(item\)/) assert.match(detailViewScript, /buildItemClaimRiskState\(item, resolveClaimRiskFlags\(\)\)/) }) test('expense risk indicator can focus and flash related risk card', () => { assert.match(detailViewTemplate, /:id="resolveRiskCardDomId\(card\)"/) assert.match(detailViewTemplate, /:data-risk-card-id="card\.id"/) assert.match(detailViewTemplate, /'is-highlighted': isHighlightedRiskCard\(card\)/) assert.match(detailViewScript, /async function focusExpenseRisk\(item\)/) assert.match(detailViewScript, /document\.getElementById\(resolveRiskCardDomId\(card\)\)/) assert.match(detailViewScript, /scrollIntoView\(\{ behavior: 'smooth', block: 'nearest', inline: 'nearest' \}\)/) assert.match(detailViewStyle, /\.validation-section--risk \.risk-advice-card\.is-highlighted/) assert.match(detailViewStyle, /@keyframes risk-card-flash/) }) test('route-level risk cards keep related item ids for every affected expense row', () => { const riskCards = buildAttachmentRiskCards({ expenseItems: [ { id: 'travel-item-1', name: '火车票', category: '火车票' }, { id: 'travel-item-2', name: '火车票', category: '火车票' }, { id: 'travel-item-3', name: '火车票', category: '火车票' } ], claimRiskFlags: [ { source: 'submission_review', severity: 'high', label: '多城市行程待说明', message: '检测到本次差旅涉及 深圳 多个目的地,但当前报销事由未说明中转、多地拜访或改签原因。', item_ids: ['travel-item-2', 'travel-item-3'] } ] }) assert.equal(riskCards.length, 1) assert.deepEqual(riskCards[0].itemIds, ['travel-item-2', 'travel-item-3']) assert.equal(riskCards[0].title, '多城市行程待说明') assert.match(detailViewScript, /cardItemIds\.includes\(itemId\)/) }) test('claim risk cards expose related expense explanations to reviewers', () => { const riskCards = buildAttachmentRiskCards({ expenseItems: [ { id: 'hotel-row', name: '住宿票', desc: '上海喜来登酒店', itemNote: '时间紧,没有合适的酒店,只能住宿超过金额的酒店。' }, { id: 'route-extra-out', name: '火车票', desc: '上海-深圳', itemNote: '中间去深圳,公司要求。' }, { id: 'route-extra-back', name: '火车票', desc: '深圳-上海', itemNote: '中间去深圳,公司要求。' } ], claimRiskFlags: [ { source: 'submission_review', severity: 'high', label: '多城市行程待说明', message: '检测到本次差旅涉及 深圳 多个目的地,但当前报销事由未说明中转、多地拜访或改签原因。', item_ids: ['route-extra-out', 'route-extra-back'] } ] }) assert.equal(riskCards.length, 1) assert.deepEqual(riskCards[0].itemIds, ['route-extra-out', 'route-extra-back']) assert.match(riskCards[0].risk, /用户已在相关费用明细补充异常说明/) assert.doesNotMatch(riskCards[0].risk, /未说明/) assert.match(riskCards[0].suggestion, /用户已在费用明细补充异常说明/) assert.match(riskCards[0].suggestion, /上海-深圳:中间去深圳,公司要求/) assert.match(riskCards[0].relatedExplanationSummary, /深圳-上海:中间去深圳,公司要求/) assert.ok(riskCards[0].ruleBasis.some((item) => item.includes('用户已补充异常说明'))) }) test('claim risk cards infer hotel explanations when risk flag has no item ids', () => { const riskCards = buildAttachmentRiskCards({ expenseItems: [ { id: 'hotel-row', name: '住宿票', desc: '上海喜来登酒店', itemType: 'hotel_ticket', itemNote: '时间紧,没有合适的酒店,只能住宿超过金额的酒店。' }, { id: 'route-row', name: '火车票', desc: '上海-深圳', itemType: 'train_ticket', itemNote: '中间去深圳,公司要求。' } ], claimRiskFlags: [ { source: 'submission_review', severity: 'high', label: '住宿金额超出报销标准', message: '住宿标准:P5在上海的住宿标准为 250.00 元/晚,票据识别金额 1086.00 元 / 3 晚,约 362.00 元/晚,超出 112.00 元/晚。' } ] }) assert.equal(riskCards.length, 1) assert.deepEqual(riskCards[0].itemIds, ['hotel-row']) assert.match(riskCards[0].relatedExplanationSummary, /上海喜来登酒店:时间紧,没有合适的酒店/) assert.doesNotMatch(riskCards[0].relatedExplanationSummary, /上海-深圳/) assert.ok(riskCards[0].ruleBasis.some((item) => item.includes('用户已补充异常说明'))) }) test('legacy route-level risk cards infer affected travel rows when backend has no item ids', () => { const riskCards = buildAttachmentRiskCards({ expenseItems: [ { id: 'train-outbound', name: '火车票', category: '火车票', desc: '武汉-上海', detail: '起始地-目的地', invoiceId: 'outbound.png' }, { id: 'train-transfer', name: '火车票', category: '火车票', desc: '上海-深圳', detail: '起始地-目的地', invoiceId: 'transfer.png' }, { id: 'train-transfer-return', name: '火车票', category: '火车票', desc: '深圳-上海', detail: '起始地-目的地', invoiceId: 'transfer-return.png' }, { id: 'train-return', name: '火车票', category: '火车票', desc: '上海-武汉', detail: '起始地-目的地', invoiceId: 'return.png' }, { id: 'allowance-row', name: '出差补贴', category: '出差补贴', desc: '系统自动计算', detail: '直辖市/特区' } ], claimRiskFlags: [ { source: 'submission_review', severity: 'high', label: '多城市行程待说明', message: '检测到本次差旅涉及 深圳 多个目的地,但当前报销事由未说明中转、多地拜访或改签原因。' } ] }) assert.equal(riskCards.length, 1) assert.deepEqual(riskCards[0].itemIds, ['train-transfer', 'train-transfer-return']) }) test('route-level risk cards narrow broad backend item ids to abnormal route rows', () => { const expenseItems = [ { id: 'train-outbound', name: '火车票', category: '火车票', desc: '武汉-上海', detail: '起始地-目的地', invoiceId: 'outbound.png' }, { id: 'train-transfer', name: '火车票', category: '火车票', desc: '上海-深圳', detail: '起始地-目的地', invoiceId: 'transfer.png' }, { id: 'train-transfer-return', name: '火车票', category: '火车票', desc: '深圳-上海', detail: '起始地-目的地', invoiceId: 'transfer-return.png' }, { id: 'train-return', name: '火车票', category: '火车票', desc: '上海-武汉', detail: '起始地-目的地', invoiceId: 'return.png' } ] const riskCards = buildAttachmentRiskCards({ expenseItems, claimRiskFlags: [ { source: 'submission_review', severity: 'high', label: '多城市行程待说明', message: '本次报销识别到多城市行程(上海、武汉、深圳),但事由中未说明中转、多地拜访或改签原因。', item_ids: expenseItems.map((item) => item.id) } ] }) assert.equal(riskCards.length, 1) assert.deepEqual(riskCards[0].itemIds, ['train-transfer', 'train-transfer-return']) }) test('AI advice shows only the latest manual return while preserving return count context', () => { const riskCards = buildAttachmentRiskCards({ claimRiskFlags: [ { source: 'manual_return', severity: 'medium', label: '人工退回', message: '第一次退回:缺少附件。', reason: '缺少附件。', return_count: 1, return_stage: '直属领导审批', risk_points: ['附件缺失或不清晰'] }, { source: 'manual_return', severity: 'medium', label: '人工退回', message: '第二次退回:超标说明不完整。', reason: '超标说明不完整。', return_count: 2, return_stage: '财务审批', risk_points: ['超出制度标准或缺少超标说明'] } ] }) assert.equal(riskCards.length, 1) assert.equal(riskCards[0].risk, '第二次退回:超标说明不完整。') assert.ok(riskCards[0].ruleBasis.some((item) => item.includes('累计退回 2 次'))) assert.ok(riskCards[0].ruleBasis.some((item) => item.includes('财务审批'))) }) test('expense attachment actions keep preview as the only recognition entry point', () => { assert.match(detailViewTemplate, /:aria-label="resolveAttachmentPreviewTitle\(item\)"/) assert.match(detailViewScript, /return fileName \? `预览附件:\$\{fileName\}` : '预览附件'/) assert.match(detailViewScript, /\.filter\(\(item\) => canPreviewAttachment\(item\)\)/) assert.match(detailViewScript, /function hasStoredAttachmentReference\(item\) \{[\s\S]*return String\(item\?\.invoiceId \|\| ''\)\.includes\('\/'\)/) assert.match(detailViewScript, /if \(metadata\) \{[\s\S]*return metadata\.previewable !== false[\s\S]*return true/) assert.match(detailViewScript, /原件尚未保存到单据中,请重新上传后预览/) assert.doesNotMatch(detailViewTemplate, /aria-label="识别附件"/) assert.doesNotMatch(detailViewTemplate, /点击识别按钮/) assert.doesNotMatch(detailViewScript, /recognizeExpenseAttachment/) assert.doesNotMatch(detailViewScript, /recognizingExpenseId/) }) test('expense detail table shows the amount total below detail rows', () => { assert.match(detailViewTemplate, /]*class="detail-expense-table"/) assert.match(detailViewTemplate, /当前还没有费用明细/) assert.match(detailViewTemplate, /通过智能录入上传票据后由系统自动归集/) assert.doesNotMatch(detailViewTemplate, /增加明细/) assert.doesNotMatch(detailViewTemplate, /handleAddExpenseItem/) assert.doesNotMatch(detailViewScript, /handleAddExpenseItem/) assert.doesNotMatch(detailViewTemplate, /class="total-row"/) assert.match(detailViewTemplate, /class="expense-total-under-table"[\s\S]*金额合计[\s\S]*\{\{ expenseTotal \}\}/) assert.doesNotMatch(detailViewTemplate, /\{\{ uploadedExpenseCount \}\} 项已关联票据/) assert.doesNotMatch(detailViewTemplate, /\{\{ expenseSummaryText \}\}/) }) test('related application information is shown above expense details for reimbursement check', () => { assert.ok( detailViewTemplate.indexOf('/) assert.match(relatedApplicationCardTemplate, /

关联单据信息<\/h3>/) assert.match(relatedApplicationCardTemplate, /展示本次报销关联的前置申请/) assert.match(relatedApplicationCardTemplate, /relatedApplicationFactItems/) assert.match(relatedApplicationCardTemplate, /暂未识别到关联申请单/) assert.match(detailViewScript, /buildRelatedApplicationFactItems/) assert.match(detailExpenseModelScript, /label:\s*'关联单据'/) assert.match(detailExpenseModelScript, /label:\s*'已归档'/) assert.match(detailViewStyle, /\.related-application-empty/) assert.doesNotMatch(detailViewTemplate, /v-if="canEditDetailNote" class="detail-note-editor"/) assert.doesNotMatch(detailViewTemplate, /v-model="detailNoteEditorView"/) }) test('split detail page header cards keep their scoped styles', () => { assert.match( detailHeroTemplate, /