2024-09-11 08:49:35 -07:00
|
|
|
import MdToHtml, { LinkRenderingType } from '@joplin/renderer/MdToHtml';
|
2020-11-07 15:59:37 +00:00
|
|
|
const { filename } = require('@joplin/lib/path-utils');
|
2021-08-12 16:54:10 +01:00
|
|
|
import { setupDatabaseAndSynchronizer, switchClient } from '@joplin/lib/testing/test-utils';
|
2021-01-22 17:41:11 +00:00
|
|
|
import shim from '@joplin/lib/shim';
|
2024-07-16 11:25:23 -07:00
|
|
|
import { RenderOptions } from '@joplin/renderer/types';
|
|
|
|
import { isResourceUrl, resourceUrlToId } from '@joplin/lib/models/utils/resourceUtils';
|
2020-11-07 15:59:37 +00:00
|
|
|
const { themeStyle } = require('@joplin/lib/theme');
|
2020-02-13 23:59:23 +00:00
|
|
|
|
2024-04-05 12:16:49 +01:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
2020-11-12 19:13:28 +00:00
|
|
|
function newTestMdToHtml(options: any = null) {
|
2020-03-23 00:47:25 +00:00
|
|
|
options = {
|
|
|
|
ResourceModel: {
|
2024-07-16 11:25:23 -07:00
|
|
|
isResourceUrl: isResourceUrl,
|
|
|
|
urlToId: resourceUrlToId,
|
2020-03-23 00:47:25 +00:00
|
|
|
},
|
|
|
|
fsDriver: shim.fsDriver(),
|
|
|
|
...options,
|
|
|
|
};
|
|
|
|
|
2021-01-23 15:51:19 +00:00
|
|
|
return new MdToHtml(options);
|
2020-03-23 00:47:25 +00:00
|
|
|
}
|
|
|
|
|
2023-02-20 12:02:29 -03:00
|
|
|
describe('MdToHtml', () => {
|
2020-02-13 23:59:23 +00:00
|
|
|
|
2022-11-15 10:23:50 +00:00
|
|
|
beforeEach(async () => {
|
2020-02-13 23:59:23 +00:00
|
|
|
await setupDatabaseAndSynchronizer(1);
|
|
|
|
await switchClient(1);
|
|
|
|
});
|
|
|
|
|
2020-12-01 18:05:24 +00:00
|
|
|
it('should convert from Markdown to Html', (async () => {
|
2020-02-13 23:59:23 +00:00
|
|
|
const basePath = `${__dirname}/md_to_html`;
|
|
|
|
const files = await shim.fsDriver().readDirStats(basePath);
|
2020-03-23 00:47:25 +00:00
|
|
|
const mdToHtml = newTestMdToHtml();
|
2020-02-13 23:59:23 +00:00
|
|
|
|
|
|
|
for (let i = 0; i < files.length; i++) {
|
|
|
|
const mdFilename = files[i].path;
|
|
|
|
if (mdFilename.indexOf('.md') < 0) continue;
|
|
|
|
|
|
|
|
const mdFilePath = `${basePath}/${mdFilename}`;
|
|
|
|
const htmlPath = `${basePath}/${filename(mdFilePath)}.html`;
|
|
|
|
|
2023-05-17 16:09:17 +01:00
|
|
|
// if (mdFilename !== 'sanitize_9.md') continue;
|
2020-02-13 23:59:23 +00:00
|
|
|
|
2024-04-05 12:16:49 +01:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
2024-07-16 11:25:23 -07:00
|
|
|
const mdToHtmlOptions: RenderOptions = {
|
2020-02-13 23:59:23 +00:00
|
|
|
bodyOnly: true,
|
|
|
|
};
|
|
|
|
|
2020-03-23 00:47:25 +00:00
|
|
|
if (mdFilename === 'checkbox_alternative.md') {
|
|
|
|
mdToHtmlOptions.plugins = {
|
|
|
|
checkbox: {
|
2020-10-21 00:23:55 +01:00
|
|
|
checkboxRenderingType: 2,
|
2020-03-23 00:47:25 +00:00
|
|
|
},
|
|
|
|
};
|
2024-05-30 00:34:52 -07:00
|
|
|
} else if (mdFilename.startsWith('sourcemap_')) {
|
|
|
|
mdToHtmlOptions.mapsToLine = true;
|
2024-07-16 11:25:23 -07:00
|
|
|
} else if (mdFilename.startsWith('resource_')) {
|
|
|
|
mdToHtmlOptions.resources = {};
|
2020-03-23 00:47:25 +00:00
|
|
|
}
|
|
|
|
|
2020-02-13 23:59:23 +00:00
|
|
|
const markdown = await shim.fsDriver().readFile(mdFilePath);
|
|
|
|
let expectedHtml = await shim.fsDriver().readFile(htmlPath);
|
|
|
|
|
|
|
|
const result = await mdToHtml.render(markdown, null, mdToHtmlOptions);
|
|
|
|
let actualHtml = result.html;
|
|
|
|
|
2021-08-12 22:07:57 +01:00
|
|
|
expectedHtml = expectedHtml.replace(/\r?\n/g, '\n');
|
|
|
|
actualHtml = actualHtml.replace(/\r?\n/g, '\n');
|
2020-02-13 23:59:23 +00:00
|
|
|
|
|
|
|
if (actualHtml !== expectedHtml) {
|
2022-04-26 12:52:48 +01:00
|
|
|
const msg: string[] = [
|
|
|
|
'',
|
|
|
|
`Error converting file: ${mdFilename}`,
|
|
|
|
'--------------------------------- Got:',
|
|
|
|
actualHtml,
|
|
|
|
'--------------------------------- Raw:',
|
|
|
|
actualHtml.split('\n'),
|
2023-10-31 16:53:47 +00:00
|
|
|
'--------------------------------- Expected (Lines)',
|
2022-04-26 12:52:48 +01:00
|
|
|
expectedHtml.split('\n'),
|
2023-10-31 16:53:47 +00:00
|
|
|
'--------------------------------- Expected (Text)',
|
|
|
|
expectedHtml,
|
2022-04-26 12:52:48 +01:00
|
|
|
'--------------------------------------------',
|
|
|
|
'',
|
|
|
|
];
|
|
|
|
|
2023-02-16 10:55:24 +00:00
|
|
|
// eslint-disable-next-line no-console
|
2022-04-26 12:52:48 +01:00
|
|
|
console.info(msg.join('\n'));
|
2020-02-13 23:59:23 +00:00
|
|
|
|
|
|
|
expect(false).toBe(true);
|
|
|
|
// return;
|
|
|
|
} else {
|
|
|
|
expect(true).toBe(true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}));
|
|
|
|
|
2020-12-01 18:05:24 +00:00
|
|
|
it('should return enabled plugin assets', (async () => {
|
2024-04-05 12:16:49 +01:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
2020-11-12 19:13:28 +00:00
|
|
|
const pluginOptions: any = {};
|
2020-03-23 00:47:25 +00:00
|
|
|
const pluginNames = MdToHtml.pluginNames();
|
|
|
|
|
|
|
|
for (const n of pluginNames) pluginOptions[n] = { enabled: false };
|
|
|
|
|
|
|
|
{
|
|
|
|
const mdToHtml = newTestMdToHtml({ pluginOptions: pluginOptions });
|
|
|
|
const assets = await mdToHtml.allAssets(themeStyle(1));
|
|
|
|
expect(assets.length).toBe(1); // Base note style should always be returned
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
pluginOptions['checkbox'].enabled = true;
|
|
|
|
const mdToHtml = newTestMdToHtml({ pluginOptions: pluginOptions });
|
|
|
|
|
|
|
|
const assets = await mdToHtml.allAssets(themeStyle(1));
|
|
|
|
expect(assets.length).toBe(2);
|
|
|
|
expect(assets[1].mime).toBe('text/css');
|
|
|
|
|
|
|
|
const content = await shim.fsDriver().readFile(assets[1].path);
|
|
|
|
expect(content.indexOf('joplin-checklist') >= 0).toBe(true);
|
|
|
|
}
|
|
|
|
}));
|
|
|
|
|
2020-12-01 18:05:24 +00:00
|
|
|
it('should wrapped the rendered Markdown', (async () => {
|
2020-03-23 00:47:25 +00:00
|
|
|
const mdToHtml = newTestMdToHtml();
|
|
|
|
|
|
|
|
// In this case, the HTML contains both the style and
|
|
|
|
// the rendered markdown wrapped in a DIV.
|
|
|
|
const result = await mdToHtml.render('just **testing**');
|
2020-10-21 00:23:55 +01:00
|
|
|
expect(result.cssStrings.length).toBeGreaterThan(0);
|
2020-03-23 00:47:25 +00:00
|
|
|
expect(result.html.indexOf('rendered-md') >= 0).toBe(true);
|
|
|
|
}));
|
|
|
|
|
2020-12-01 18:05:24 +00:00
|
|
|
it('should return the rendered body only', (async () => {
|
2020-03-23 00:47:25 +00:00
|
|
|
const mdToHtml = newTestMdToHtml();
|
|
|
|
|
2020-11-05 16:58:23 +00:00
|
|
|
// In this case, the HTML contains only the rendered markdown, with
|
|
|
|
// no wrapper and no style. The style is instead in the cssStrings
|
|
|
|
// property.
|
|
|
|
{
|
|
|
|
const result = await mdToHtml.render('just **testing**', null, { bodyOnly: true });
|
|
|
|
expect(result.cssStrings.length).toBeGreaterThan(0);
|
|
|
|
expect(result.html.trim()).toBe('just <strong>testing</strong>');
|
|
|
|
}
|
|
|
|
|
|
|
|
// But it should not remove the wrapping <p> tags if there's more
|
|
|
|
// than one line
|
|
|
|
{
|
|
|
|
const result = await mdToHtml.render('one\n\ntwo', null, { bodyOnly: true });
|
|
|
|
expect(result.html.trim()).toBe('<p>one</p>\n<p>two</p>');
|
|
|
|
}
|
2020-03-23 00:47:25 +00:00
|
|
|
}));
|
|
|
|
|
2021-05-17 20:30:48 +02:00
|
|
|
it('should render an empty string', (async () => {
|
|
|
|
const mdToHtml = newTestMdToHtml();
|
|
|
|
const result = await mdToHtml.render('', null, { splitted: true });
|
|
|
|
// The TinyMCE component checks for this exact string to apply a hack,
|
|
|
|
// so make sure it doesn't change from version to version.
|
|
|
|
expect(result.html).toBe('<div id="rendered-md"></div>');
|
|
|
|
}));
|
|
|
|
|
2020-12-01 18:05:24 +00:00
|
|
|
it('should split HTML and CSS', (async () => {
|
2020-03-23 00:47:25 +00:00
|
|
|
const mdToHtml = newTestMdToHtml();
|
|
|
|
|
2020-11-05 16:58:23 +00:00
|
|
|
// It is similar to the bodyOnly option, excepts that the rendered
|
|
|
|
// Markdown is wrapped in a DIV
|
2020-03-23 00:47:25 +00:00
|
|
|
const result = await mdToHtml.render('just **testing**', null, { splitted: true });
|
2020-10-21 00:23:55 +01:00
|
|
|
expect(result.cssStrings.length).toBeGreaterThan(0);
|
2020-03-23 00:47:25 +00:00
|
|
|
expect(result.html.trim()).toBe('<div id="rendered-md"><p>just <strong>testing</strong></p>\n</div>');
|
|
|
|
}));
|
2020-03-09 23:24:57 +00:00
|
|
|
|
2021-01-02 16:53:59 +00:00
|
|
|
it('should render links correctly', (async () => {
|
|
|
|
const testCases = [
|
|
|
|
// 0: input
|
|
|
|
// 1: output with linkify = off
|
|
|
|
// 2: output with linkify = on
|
|
|
|
[
|
|
|
|
'https://example.com',
|
|
|
|
'https://example.com',
|
|
|
|
'<a data-from-md title=\'https://example.com\' href=\'https://example.com\'>https://example.com</a>',
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'file://C:\\AUTOEXEC.BAT',
|
|
|
|
'file://C:\\AUTOEXEC.BAT',
|
|
|
|
'<a data-from-md title=\'file://C:%5CAUTOEXEC.BAT\' href=\'file://C:%5CAUTOEXEC.BAT\'>file://C:\\AUTOEXEC.BAT</a>',
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'example.com',
|
|
|
|
'example.com',
|
|
|
|
'example.com',
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'oo.ps',
|
|
|
|
'oo.ps',
|
|
|
|
'oo.ps',
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'test@example.com',
|
|
|
|
'test@example.com',
|
|
|
|
'test@example.com',
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'<https://example.com>',
|
|
|
|
'<a data-from-md title=\'https://example.com\' href=\'https://example.com\'>https://example.com</a>',
|
|
|
|
'<a data-from-md title=\'https://example.com\' href=\'https://example.com\'>https://example.com</a>',
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'[ok](https://example.com)',
|
|
|
|
'<a data-from-md title=\'https://example.com\' href=\'https://example.com\'>ok</a>',
|
|
|
|
'<a data-from-md title=\'https://example.com\' href=\'https://example.com\'>ok</a>',
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'[bla.pdf](file:///Users/tessus/Downloads/bla.pdf)',
|
|
|
|
'<a data-from-md title=\'file:///Users/tessus/Downloads/bla.pdf\' href=\'file:///Users/tessus/Downloads/bla.pdf\'>bla.pdf</a>',
|
|
|
|
'<a data-from-md title=\'file:///Users/tessus/Downloads/bla.pdf\' href=\'file:///Users/tessus/Downloads/bla.pdf\'>bla.pdf</a>',
|
|
|
|
],
|
|
|
|
];
|
|
|
|
|
|
|
|
const mdToHtmlLinkifyOn = newTestMdToHtml({
|
|
|
|
pluginOptions: {
|
|
|
|
linkify: { enabled: true },
|
2024-09-11 08:49:35 -07:00
|
|
|
link_open: {
|
|
|
|
linkRenderingType: LinkRenderingType.HrefHandler,
|
|
|
|
},
|
2021-01-02 16:53:59 +00:00
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
const mdToHtmlLinkifyOff = newTestMdToHtml({
|
|
|
|
pluginOptions: {
|
|
|
|
linkify: { enabled: false },
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2024-09-11 08:49:35 -07:00
|
|
|
const renderOptions = {
|
|
|
|
bodyOnly: true,
|
|
|
|
plainResourceRendering: true,
|
|
|
|
linkRenderingType: LinkRenderingType.HrefHandler,
|
|
|
|
};
|
|
|
|
|
2021-01-02 16:53:59 +00:00
|
|
|
for (const testCase of testCases) {
|
|
|
|
const [input, expectedLinkifyOff, expectedLinkifyOn] = testCase;
|
|
|
|
|
|
|
|
{
|
2024-09-11 08:49:35 -07:00
|
|
|
const actual = await mdToHtmlLinkifyOn.render(input, null, renderOptions);
|
2021-01-02 16:53:59 +00:00
|
|
|
|
|
|
|
expect(actual.html).toBe(expectedLinkifyOn);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
2024-09-11 08:49:35 -07:00
|
|
|
const actual = await mdToHtmlLinkifyOff.render(input, null, renderOptions);
|
2021-01-02 16:53:59 +00:00
|
|
|
|
|
|
|
expect(actual.html).toBe(expectedLinkifyOff);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}));
|
2020-03-09 23:24:57 +00:00
|
|
|
|
2024-09-11 08:49:35 -07:00
|
|
|
it.each([
|
|
|
|
'[test](http://example.com/)',
|
|
|
|
'[test](mailto:test@example.com)',
|
|
|
|
])('should add onclick handlers to links when linkRenderingType is JavaScriptHandler (%j)', async (markdown) => {
|
|
|
|
const mdToHtml = newTestMdToHtml();
|
|
|
|
|
|
|
|
const renderWithoutOnClickOptions = {
|
|
|
|
bodyOnly: true,
|
|
|
|
linkRenderingType: LinkRenderingType.HrefHandler,
|
|
|
|
};
|
|
|
|
expect(
|
|
|
|
(await mdToHtml.render(markdown, undefined, renderWithoutOnClickOptions)).html,
|
|
|
|
).not.toContain('onclick');
|
|
|
|
|
|
|
|
const renderWithOnClickOptions = {
|
|
|
|
bodyOnly: true,
|
|
|
|
linkRenderingType: LinkRenderingType.JavaScriptHandler,
|
|
|
|
};
|
|
|
|
expect(
|
|
|
|
(await mdToHtml.render(markdown, undefined, renderWithOnClickOptions)).html,
|
|
|
|
).toMatch(/<a data-from-md .*onclick=['"].*['"].*>/);
|
|
|
|
});
|
|
|
|
|
2021-11-03 21:10:46 +09:00
|
|
|
it('should return attributes of line numbers', (async () => {
|
|
|
|
const mdToHtml = newTestMdToHtml();
|
|
|
|
|
|
|
|
// Mapping information between source lines and html elements is
|
|
|
|
// annotated.
|
|
|
|
{
|
|
|
|
const input = '# Head\nFruits\n- Apple\n';
|
|
|
|
const result = await mdToHtml.render(input, null, { bodyOnly: true, mapsToLine: true });
|
2022-04-10 19:31:17 +09:00
|
|
|
expect(result.html.trim()).toBe('<h1 id="head" class="maps-to-line" source-line="0" source-line-end="1">Head</h1>\n' +
|
|
|
|
'<p class="maps-to-line" source-line="1" source-line-end="2">Fruits</p>\n' +
|
2023-08-22 11:58:53 +01:00
|
|
|
'<ul>\n<li class="maps-to-line" source-line="2" source-line-end="3">Apple</li>\n</ul>',
|
2021-11-03 21:10:46 +09:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}));
|
2023-07-10 04:03:52 -07:00
|
|
|
|
|
|
|
it('should attach source blocks to block KaTeX', async () => {
|
|
|
|
const mdToHtml = newTestMdToHtml();
|
|
|
|
|
|
|
|
const katex = [
|
|
|
|
'3 + 3',
|
|
|
|
'\n\\int_0^1 x dx\n\n',
|
|
|
|
'\n\\int_0^1 x dx\n3 + 3\n',
|
|
|
|
'\n\t2^{3^4}\n\t3 + 3\n',
|
|
|
|
'3\n4',
|
|
|
|
];
|
|
|
|
const surroundingTextChoices = [
|
|
|
|
['', ''],
|
|
|
|
['Test', ''],
|
|
|
|
['Test', 'Test!'],
|
|
|
|
['Test\n\n', '\n\nTest!'],
|
|
|
|
];
|
|
|
|
|
|
|
|
const tests = [];
|
|
|
|
for (const texSource of katex) {
|
|
|
|
for (const [start, end] of surroundingTextChoices) {
|
|
|
|
tests.push([texSource, `${start}\n$$${texSource}$$\n${end}`]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const [tex, input] of tests) {
|
|
|
|
const html = await mdToHtml.render(input, null, { bodyOnly: true });
|
|
|
|
|
|
|
|
const opening = '<pre class="joplin-source" data-joplin-language="katex" data-joplin-source-open="$$ " data-joplin-source-close=" $$ ">';
|
|
|
|
const closing = '</pre>';
|
|
|
|
|
|
|
|
// Remove any single leading and trailing newlines, those are included in data-joplin-source-open
|
|
|
|
// and data-joplin-source-close.
|
|
|
|
const trimmedTex = tex.replace(/^[\n]/, '').replace(/[\n]$/, '');
|
|
|
|
expect(html.html).toContain(opening + trimmedTex + closing);
|
|
|
|
}
|
|
|
|
});
|
2023-12-06 11:23:08 -08:00
|
|
|
|
|
|
|
it('should render inline KaTeX after a numbered equation', async () => {
|
|
|
|
const mdToHtml = newTestMdToHtml();
|
|
|
|
|
|
|
|
// This test is intended to verify that inline KaTeX renders correctly
|
|
|
|
// after creating a numbered equation with \begin{align}...\end{align}.
|
|
|
|
//
|
|
|
|
// See https://github.com/laurent22/joplin/issues/9455 for details.
|
|
|
|
|
|
|
|
const markdown = [
|
|
|
|
'$$',
|
|
|
|
'\\begin{align}\\text{Block}\\end{align}',
|
|
|
|
'$$',
|
|
|
|
'',
|
|
|
|
'$\\text{Inline}$',
|
|
|
|
].join('\n');
|
|
|
|
const { html } = await mdToHtml.render(markdown, null, { bodyOnly: true });
|
|
|
|
|
|
|
|
// Because we don't control the output of KaTeX, this test should be as general as
|
|
|
|
// possible while still verifying that rendering (without an error) occurs.
|
|
|
|
|
|
|
|
// Should have rendered the inline and block content without errors
|
|
|
|
expect(html).toContain('Inline</span>');
|
|
|
|
expect(html).toContain('Block</span>');
|
|
|
|
});
|
2024-10-15 08:37:15 -07:00
|
|
|
|
|
|
|
it('should sanitize KaTeX errors', async () => {
|
|
|
|
const markdown = '$\\a<svg>$';
|
|
|
|
const renderResult = await newTestMdToHtml().render(markdown, null, { bodyOnly: true });
|
|
|
|
|
|
|
|
// Should not contain the HTML in unsanitized form
|
|
|
|
expect(renderResult.html).not.toContain('<svg>');
|
|
|
|
});
|
2020-02-13 23:59:23 +00:00
|
|
|
});
|