import { describe, expect, it } from 'vitest';
import { render, screen } from '@testing-library/react';
import { ChatMessage } from '@/pages/Chat/ChatMessage';
import type { RawMessage } from '@/stores/chat';
import { appendBusinessResponseGuidance } from '../../shared/business-guidance';
describe('ChatMessage attachment dedupe', () => {
it('keeps attachment-only assistant replies visible even when process attachments are suppressed', () => {
const message: RawMessage = {
role: 'assistant',
content: [],
_attachedFiles: [
{
fileName: 'artifact.png',
mimeType: 'image/png',
fileSize: 0,
preview: '/tmp/artifact.png',
filePath: '/tmp/artifact.png',
source: 'tool-result',
},
],
};
render(
,
);
expect(screen.getByAltText('artifact.png')).toBeInTheDocument();
});
});
describe('ChatMessage LaTeX rendering', () => {
it('renders inline `$...$` math with KaTeX', () => {
const message: RawMessage = {
role: 'assistant',
content: 'Mass-energy equivalence: $E=mc^2$ is famous.',
};
const { container } = render();
expect(container.querySelector('.katex')).not.toBeNull();
});
it('renders display `$$...$$` math as a block', () => {
const message: RawMessage = {
role: 'assistant',
content: 'Definite integral:\n\n$$\n\\int_0^1 x\\,dx = \\frac{1}{2}\n$$\n',
};
const { container } = render();
expect(container.querySelector('.katex-display')).not.toBeNull();
});
it('renders `\\(...\\)` inline math (OpenAI-style escaping)', () => {
const message: RawMessage = {
role: 'assistant',
content: 'Quadratic formula: \\(x = \\frac{-b \\pm \\sqrt{b^2-4ac}}{2a}\\).',
};
const { container } = render();
expect(container.querySelector('.katex')).not.toBeNull();
expect(container.querySelector('.katex-display')).toBeNull();
});
it('renders `\\[...\\]` block math (OpenAI-style escaping)', () => {
const message: RawMessage = {
role: 'assistant',
content: 'Sum formula:\n\n\\[\\sum_{i=1}^n i = \\frac{n(n+1)}{2}\\]',
};
const { container } = render();
expect(container.querySelector('.katex-display')).not.toBeNull();
});
it('does not rewrite `\\(` inside code fences', () => {
const message: RawMessage = {
role: 'assistant',
content: 'Code sample:\n\n```\nprintf("\\(hello\\)")\n```\n',
};
const { container } = render();
expect(container.textContent).toContain('\\(hello\\)');
expect(container.querySelector('.katex')).toBeNull();
});
});
describe('ChatMessage business answer rendering', () => {
it('renders a business summary panel for structured operational replies', () => {
const message: RawMessage = {
role: 'assistant',
content: [
'状态:需处理,携程渠道房态与 PMS 不一致。',
'依据:携程 0508 房型显示可售 3 间,PMS 为 0 间。',
'影响:可能产生超售风险。',
'下一步:请先暂停该房型自动售卖,并复核渠道登录态。',
].join('\n'),
};
render();
const panel = screen.getByTestId('business-answer-panel');
expect(panel).toHaveTextContent('业务摘要');
expect(panel).toHaveTextContent('需处理,携程渠道房态与 PMS 不一致');
expect(panel).toHaveTextContent('携程 0508 房型显示可售 3 间');
expect(panel).toHaveTextContent('请先暂停该房型自动售卖');
});
it('does not add the business panel to ordinary assistant chat', () => {
const message: RawMessage = {
role: 'assistant',
content: '你好,我可以帮你处理问题。',
};
render();
expect(screen.queryByTestId('business-answer-panel')).not.toBeInTheDocument();
});
});
describe('ChatMessage markdown rendering', () => {
it('hides legacy injected business guidance from visible user messages', () => {
const message: RawMessage = {
role: 'user',
content: appendBusinessResponseGuidance('会话中的 markdown 结构化,表格也可以加强 UI 渲染'),
};
render();
expect(screen.getByText('会话中的 markdown 结构化,表格也可以加强 UI 渲染')).toBeInTheDocument();
expect(screen.queryByText(/YINIAN_BUSINESS_RESPONSE_GUIDANCE/)).not.toBeInTheDocument();
expect(screen.queryByText(/请按智念业务员工/)).not.toBeInTheDocument();
});
it('renders markdown tables inside the enhanced table shell', () => {
const message: RawMessage = {
role: 'assistant',
content: [
'| 渠道 | 房型 | 状态 |',
'| --- | --- | --- |',
'| 携程 | 豪华大床房 | 需复核 |',
].join('\n'),
};
render();
expect(screen.getByTestId('markdown-table-scroll')).toBeInTheDocument();
expect(screen.getByTestId('markdown-table')).toHaveTextContent('豪华大床房');
});
});