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 { buildExpenseItemViewModel, buildDraftBlockingIssues, rebuildExpenseItems, buildStandardAdjustmentMap, isApplicationDocumentRequest } from '../src/views/scripts/travelRequestDetailExpenseModel.js' import { buildEmployeeProfileAdviceItems, buildTravelReceiptMaterialPrompts } from '../src/views/scripts/travelRequestDetailAdviceModel.js' import { buildStandardAdjustmentPayload, filterSubmitterResolvedRiskCards } from '../src/views/scripts/travelRequestDetailStandardAdjustment.js' const detailViewTemplate = readFileSync( fileURLToPath(new URL('../src/views/TravelRequestDetailView.vue', import.meta.url)), 'utf8' ) const detailViewScript = 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 detailExpenseModelScript = readFileSync( fileURLToPath(new URL('../src/views/scripts/travelRequestDetailExpenseModel.js', import.meta.url)), 'utf8' ) 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 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 exposes direct reviewer action suggestion', () => { assert.match(stageRiskAdviceCard, /class="employee-risk-action"/) assert.match(stageRiskAdviceCard, /\{\{ decisionAction \}\}/) assert.match(stageRiskAdviceCard, /compactEvidenceItems/) assert.match(stageRiskAdviceCard, /compactAdviceItems/) assert.ok( stageRiskAdviceCard.indexOf('class="employee-risk-advice-list"') < stageRiskAdviceCard.indexOf('class="employee-risk-action"') ) assert.match(stageRiskAdviceCard, /employee-risk-tone-pill/) assert.match(stageRiskAdviceCard, /\.employee-risk-ai-note \{[\s\S]*grid-template-columns: minmax\(0, 1fr\);/) assert.match(stageRiskAdviceCard, /\.employee-risk-action \{[\s\S]*align-items: center;[\s\S]*justify-content: center;[\s\S]*text-align: center;/) assert.match(stageRiskAdviceCard, /建议退回补充票据、行程说明或超标原因/) assert.match(stageRiskAdviceCard, /可按权限继续审批/) }) test('AI advice ignores approval opinions and flow logs as risks', () => { const riskCards = buildAttachmentRiskCards({ claimRiskFlags: [ { source: 'manual_approval', severity: 'info', label: '领导审批通过', message: '同意' }, { source: 'finance_approval', severity: 'info', label: '财务审核通过', message: '周晓彤 已完成财务审核,进入归档入账。' } ] }) 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, '第 1 条:AI提示:住宿金额超出报销标准') 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 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, /
\{\{ 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 && !isApplicationDocument\.value && hasVisibleRiskCards\.value/) assert.match(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, /