feat(web): 工作台 AI 模式报销预审与文档查询模型拆分
- 新增 aiApplicationPrecheckModel/aiDocumentQueryModel/aiApplicationPreviewActions/aiConversationHtmlRenderer 四个独立模型与服务,按职责从主组件拆出 - PersonalWorkbenchAiMode 接入拆分后的预审、文档查询与 HTML 渲染逻辑,配合 markdown 工具增强结构化展示 - 文档中心与归档筛选、风险可见性、申请预览等工具同步适配,补充对应单元测试 - 新增 AI 文档卡片背景资源
This commit is contained in:
100
web/tests/ai-conversation-html-renderer.test.mjs
Normal file
100
web/tests/ai-conversation-html-renderer.test.mjs
Normal file
@@ -0,0 +1,100 @@
|
||||
import assert from 'node:assert/strict'
|
||||
import test from 'node:test'
|
||||
|
||||
import { renderAiConversationHtml } from '../src/utils/aiConversationHtmlRenderer.js'
|
||||
|
||||
test('AI conversation renderer turns business copy into spacious semantic HTML', () => {
|
||||
const rendered = renderAiConversationHtml([
|
||||
'### 出差申请办理确认',
|
||||
'',
|
||||
'**我已在您的输入中提取到关键信息**,如下表所示:',
|
||||
'',
|
||||
'> **前置查询结果**:我已查询您名下可关联的差旅申请单,当前未查到可关联单据。',
|
||||
'',
|
||||
'> **需要您确认**:发起新的出差申请属于业务操作,需要您手动确认后我再继续办理。',
|
||||
'',
|
||||
'点击下方 **确认发起出差申请** 后,我会继续完成:',
|
||||
'',
|
||||
'- **单据重叠核查**:检查同一时间段是否已有申请单,避免重复申请。',
|
||||
'- **预算与审批预审**:查看部门预算影响,判断是否可能增加预算管理者审核。'
|
||||
].join('\n'))
|
||||
|
||||
assert.match(rendered, /<div class="ai-html-flow">/)
|
||||
assert.match(rendered, /<h3 class="ai-html-title">出差申请办理确认<\/h3>/)
|
||||
assert.match(rendered, /<section class="ai-html-focus-grid" aria-label="重点信息">/)
|
||||
assert.match(rendered, /<article class="ai-html-focus-card">[\s\S]*前置查询结果[\s\S]*当前未查到可关联单据/)
|
||||
assert.match(rendered, /<article class="ai-html-focus-card">[\s\S]*需要您确认[\s\S]*需要您手动确认后我再继续办理/)
|
||||
assert.match(rendered, /<ul class="ai-html-steps">[\s\S]*单据重叠核查[\s\S]*预算与审批预审/)
|
||||
assert.doesNotMatch(rendered, /<blockquote>/)
|
||||
assert.doesNotMatch(rendered, /<ul>\s*<li><strong>/)
|
||||
})
|
||||
|
||||
test('AI conversation renderer supports tables and escapes unsafe HTML', () => {
|
||||
const rendered = renderAiConversationHtml([
|
||||
'### 查询结果',
|
||||
'',
|
||||
'| 字段 | 内容 |',
|
||||
'| --- | --- |',
|
||||
'| 事由 | 辅助 <script>alert(1)</script> 部署 |',
|
||||
'| 地点 | 上海 |'
|
||||
].join('\n'))
|
||||
|
||||
assert.match(rendered, /<div class="ai-html-table-wrap">/)
|
||||
assert.match(rendered, /<th>字段<\/th>/)
|
||||
assert.match(rendered, /<script>alert\(1\)<\/script>/)
|
||||
assert.doesNotMatch(rendered, /<script>/)
|
||||
})
|
||||
|
||||
test('AI conversation renderer renders application detail action links as buttons', () => {
|
||||
const rendered = renderAiConversationHtml([
|
||||
'| 单据编号 | 操作 |',
|
||||
'| --- | --- |',
|
||||
'| AP-OVERLAP | [查看](#ai-open-application-detail:AP-OVERLAP) |'
|
||||
].join('\n'))
|
||||
|
||||
assert.match(rendered, /class="ai-html-action-link ai-html-action-link-application"/)
|
||||
assert.match(rendered, /data-ai-action="open-application-detail"/)
|
||||
assert.match(rendered, /href="#ai-open-application-detail:AP-OVERLAP"/)
|
||||
assert.doesNotMatch(rendered, /target="_blank"[\s\S]{0,120}#ai-open-application-detail/)
|
||||
})
|
||||
|
||||
test('AI conversation renderer renders document detail action links as buttons', () => {
|
||||
const rendered = renderAiConversationHtml('[查看单据](#ai-open-document-detail:CL-20260221001)')
|
||||
|
||||
assert.match(rendered, /class="ai-html-action-link ai-html-action-link-document"/)
|
||||
assert.match(rendered, /data-ai-action="open-document-detail"/)
|
||||
assert.match(rendered, /href="#ai-open-document-detail:CL-20260221001"/)
|
||||
assert.doesNotMatch(rendered, /target="_blank"[\s\S]{0,120}#ai-open-document-detail/)
|
||||
})
|
||||
|
||||
test('AI conversation renderer renders images as html and rejects unsafe image sources', () => {
|
||||
const rendered = renderAiConversationHtml([
|
||||
'### 图片材料',
|
||||
'',
|
||||
'',
|
||||
'',
|
||||
'内联图片:)'
|
||||
].join('\n'))
|
||||
|
||||
assert.match(rendered, /<figure class="ai-html-image-frame">/)
|
||||
assert.match(rendered, /<img class="ai-html-image" src="https:\/\/example\.com\/receipt\.png" alt="票据预览" loading="lazy" \/>/)
|
||||
assert.match(rendered, /<figcaption class="ai-html-image-caption">票据预览<\/figcaption>/)
|
||||
assert.doesNotMatch(rendered, /javascript:alert/)
|
||||
})
|
||||
|
||||
test('AI conversation renderer keeps separated step bullets in one numbered sequence', () => {
|
||||
const rendered = renderAiConversationHtml([
|
||||
'点击下方 **确认发起出差申请** 后,我会继续完成:',
|
||||
'',
|
||||
'- **单据重叠核查**:检查同一时间段是否已有申请单,避免重复申请。',
|
||||
'',
|
||||
'- **预算与审批预审**:查看部门预算影响,判断是否可能增加预算管理者审核。',
|
||||
'',
|
||||
'- **申请表生成**:预审完成后,再展示完整申请表并自动预填已识别信息。'
|
||||
].join('\n'))
|
||||
|
||||
assert.equal((rendered.match(/class="ai-html-steps"/g) || []).length, 1)
|
||||
assert.match(rendered, /<span class="ai-html-step-index">1<\/span>[\s\S]*单据重叠核查/)
|
||||
assert.match(rendered, /<span class="ai-html-step-index">2<\/span>[\s\S]*预算与审批预审/)
|
||||
assert.match(rendered, /<span class="ai-html-step-index">3<\/span>[\s\S]*申请表生成/)
|
||||
})
|
||||
Reference in New Issue
Block a user