mirror of
https://github.com/nexu-io/open-design.git
synced 2026-06-01 03:14:35 +07:00
166 lines
7 KiB
TypeScript
166 lines
7 KiB
TypeScript
import { describe, expect, it } from 'vitest';
|
|
|
|
import { renderMarkdownToSafeHtml } from '../../src/artifacts/markdown';
|
|
|
|
describe('renderMarkdownToSafeHtml', () => {
|
|
it('renders common markdown blocks', () => {
|
|
const md = [
|
|
'# Title',
|
|
'',
|
|
'Paragraph with **bold** and *italic* and `code`.',
|
|
'',
|
|
'- one',
|
|
'- two',
|
|
'',
|
|
'1. first',
|
|
'2. second',
|
|
'',
|
|
'> note line',
|
|
'',
|
|
'```',
|
|
'const x = 1 < 2;',
|
|
'```',
|
|
].join('\n');
|
|
|
|
const out = renderMarkdownToSafeHtml(md);
|
|
expect(out).toContain('<h1>Title</h1>');
|
|
expect(out).toContain('<p>Paragraph with <strong>bold</strong> and <em>italic</em> and <code>code</code>.</p>');
|
|
expect(out).toContain('<ul><li>one</li><li>two</li></ul>');
|
|
expect(out).toContain('<ol><li>first</li><li>second</li></ol>');
|
|
expect(out).toContain('<blockquote>note line</blockquote>');
|
|
expect(out).toContain('<pre><code>const x = 1 < 2;</code></pre>');
|
|
});
|
|
|
|
it('escapes raw html', () => {
|
|
const out = renderMarkdownToSafeHtml('<script>alert(1)</script>');
|
|
expect(out).toContain('<script>alert(1)</script>');
|
|
expect(out).not.toContain('<script>');
|
|
});
|
|
|
|
it('renders safe links with target attributes', () => {
|
|
const out = renderMarkdownToSafeHtml('[Open](https://example.com)');
|
|
expect(out).toContain('<a href="https://example.com" rel="noreferrer noopener" target="_blank">Open</a>');
|
|
});
|
|
|
|
it('keeps underscores inside href intact', () => {
|
|
const out = renderMarkdownToSafeHtml('[x](https://example.com/a_b_c)');
|
|
expect(out).toContain('<a href="https://example.com/a_b_c" rel="noreferrer noopener" target="_blank">x</a>');
|
|
expect(out).not.toContain('<em>b</em>');
|
|
});
|
|
|
|
it('escapes raw html inside link text', () => {
|
|
const out = renderMarkdownToSafeHtml('[<img src=x onerror=alert(1)>](https://example.com)');
|
|
expect(out).toContain('<img src=x onerror=alert(1)>');
|
|
expect(out).not.toContain('<img ');
|
|
});
|
|
|
|
it('keeps markdown emphasis markers literal inside inline code', () => {
|
|
const out = renderMarkdownToSafeHtml('Use `**literal**` and `_literal_` as code.');
|
|
expect(out).toContain('<code>**literal**</code>');
|
|
expect(out).toContain('<code>_literal_</code>');
|
|
expect(out).not.toContain('<code><strong>literal</strong></code>');
|
|
expect(out).not.toContain('<code><em>literal</em></code>');
|
|
});
|
|
|
|
it('does not render unsafe link protocols', () => {
|
|
const out = renderMarkdownToSafeHtml('[Bad](javascript:alert(1))');
|
|
expect(out).toContain('<p>Bad)</p>');
|
|
expect(out).not.toContain('javascript:');
|
|
expect(out).not.toContain('<a ');
|
|
});
|
|
|
|
it('strips trailing punctuation from URLs', () => {
|
|
// Common cases: URL followed by ", }, ), ., ,, ;
|
|
const cases: [string, string][] = [
|
|
['[Link](https://example.com")', 'https://example.com'],
|
|
['[Link](https://example.com}")', 'https://example.com'],
|
|
['[Link](https://example.com")} )', 'https://example.com'],
|
|
['[Link](https://example.com.)', 'https://example.com'],
|
|
['[Link](https://example.com,)', 'https://example.com'],
|
|
['[Link](https://example.com;)', 'https://example.com'],
|
|
['[Link](https://example.com:8080/path)",', 'https://example.com:8080/path'],
|
|
// URLs inside JSON-like output
|
|
['[Link](https://github.com/user/repo")}', 'https://github.com/user/repo'],
|
|
['[Link](https://github.com/user/repo.git")}', 'https://github.com/user/repo.git'],
|
|
// Realistic sentence
|
|
['Check out [the repo](https://github.com/nexu-io/open-design).', 'https://github.com/nexu-io/open-design'],
|
|
];
|
|
for (const [md, expectedHref] of cases) {
|
|
const out = renderMarkdownToSafeHtml(md);
|
|
expect(out).toContain(`href="${expectedHref}"`);
|
|
}
|
|
});
|
|
|
|
it('renders a GFM pipe table with header and body rows', () => {
|
|
const md = [
|
|
'| 字段 | 类型 | 说明 |',
|
|
'|---|---|---|',
|
|
'| id | string | 主键 |',
|
|
'| name | string | 简称 |',
|
|
].join('\n');
|
|
const out = renderMarkdownToSafeHtml(md);
|
|
expect(out).toContain('<div class="md-table-wrap"><table class="md-table">');
|
|
expect(out).toContain('<thead><tr><th>字段</th><th>类型</th><th>说明</th></tr></thead>');
|
|
expect(out).toContain('<td>id</td><td>string</td><td>主键</td>');
|
|
expect(out).toContain('<td>name</td><td>string</td><td>简称</td>');
|
|
expect(out).not.toContain('<p>| 字段');
|
|
});
|
|
|
|
it('honors :--- / ---: / :---: column alignment markers', () => {
|
|
const md = [
|
|
'| L | C | R |',
|
|
'|:---|:---:|---:|',
|
|
'| a | b | c |',
|
|
].join('\n');
|
|
const out = renderMarkdownToSafeHtml(md);
|
|
expect(out).toContain('<th style="text-align:left">L</th>');
|
|
expect(out).toContain('<th style="text-align:center">C</th>');
|
|
expect(out).toContain('<th style="text-align:right">R</th>');
|
|
expect(out).toContain('<td style="text-align:left">a</td>');
|
|
expect(out).toContain('<td style="text-align:center">b</td>');
|
|
expect(out).toContain('<td style="text-align:right">c</td>');
|
|
});
|
|
|
|
it('formats inline markup inside table cells', () => {
|
|
const md = ['| key | value |', '|---|---|', '| `id` | **bold** |'].join('\n');
|
|
const out = renderMarkdownToSafeHtml(md);
|
|
expect(out).toContain('<td><code>id</code></td>');
|
|
expect(out).toContain('<td><strong>bold</strong></td>');
|
|
});
|
|
|
|
it('keeps escaped pipes \\| as literal | inside a cell', () => {
|
|
const md = ['| a | b |', '|---|---|', '| x \\| y | z |'].join('\n');
|
|
const out = renderMarkdownToSafeHtml(md);
|
|
expect(out).toContain('<td>x | y</td>');
|
|
expect(out).toContain('<td>z</td>');
|
|
});
|
|
|
|
it('pads short rows with empty cells', () => {
|
|
const md = ['| a | b | c |', '|---|---|---|', '| x | y |'].join('\n');
|
|
const out = renderMarkdownToSafeHtml(md);
|
|
expect(out).toContain('<tr><td>x</td><td>y</td><td></td></tr>');
|
|
});
|
|
|
|
it('breaks a preceding paragraph at a table start without a blank line', () => {
|
|
const md = ['Some paragraph', '| a | b |', '|---|---|', '| 1 | 2 |'].join('\n');
|
|
const out = renderMarkdownToSafeHtml(md);
|
|
expect(out).toContain('<p>Some paragraph</p>');
|
|
expect(out).toContain('<div class="md-table-wrap"><table class="md-table">');
|
|
});
|
|
|
|
it('does not promote a stray pipe-containing paragraph to a table', () => {
|
|
const md = 'Just a line with a | pipe in it.';
|
|
const out = renderMarkdownToSafeHtml(md);
|
|
expect(out).toContain('<p>Just a line with a | pipe in it.</p>');
|
|
expect(out).not.toContain('<table');
|
|
});
|
|
|
|
it('treats pipes inside a backtick code span as cell content, not column boundaries', () => {
|
|
// Common TypeScript-style union-type cell — the `|` between `"ready"`
|
|
// and `"done"` lives inside the backtick code span and must not split
|
|
// the cell. Without this, the row collapses from 2 columns to 3.
|
|
const md = ['| status | type |', '|---|---|', '| ok | `"ready" \| "done"` |'].join('\n');
|
|
const out = renderMarkdownToSafeHtml(md);
|
|
expect(out).toContain('<tr><td>ok</td><td><code>"ready" | "done"</code></td></tr>');
|
|
});
|
|
});
|