import { mount } from '@vue/test-utils'; import katex from 'katex'; import Vue, { nextTick } from 'vue'; import markdownTableJson from 'test_fixtures/blob/notebook/markdown-table.json'; import basicJson from 'test_fixtures/blob/notebook/basic.json'; import mathJson from 'test_fixtures/blob/notebook/math.json'; import MarkdownComponent from '~/notebook/cells/markdown.vue'; import Prompt from '~/notebook/cells/prompt.vue'; const Component = Vue.extend(MarkdownComponent); window.katex = katex; function buildCellComponent(cell, relativePath = '', hidePrompt) { return mount(Component, { propsData: { cell, hidePrompt, }, provide: { relativeRawPath: relativePath, }, }); } function buildMarkdownComponent(markdownContent, relativePath = '') { return buildCellComponent( { cell_type: 'markdown', metadata: {}, source: markdownContent, }, relativePath, ); } describe('Markdown component', () => { let wrapper; let cell; let json; beforeEach(async () => { json = basicJson; // eslint-disable-next-line prefer-destructuring cell = json.cells[1]; wrapper = buildCellComponent(cell); await nextTick(); }); const findPrompt = () => wrapper.findComponent(Prompt); it('renders a prompt by default', () => { expect(findPrompt().exists()).toBe(true); }); it('does not render a prompt if hidePrompt is true', () => { wrapper = buildCellComponent(cell, '', true); expect(findPrompt().exists()).toBe(false); }); it('does not render the markdown text', () => { expect(wrapper.vm.$el.querySelector('.markdown').innerHTML.trim()).not.toEqual( cell.source.join(''), ); }); it('renders the markdown HTML', () => { expect(wrapper.vm.$el.querySelector('.markdown h1')).not.toBeNull(); }); it('sanitizes Markdown output', async () => { Object.assign(cell, { source: [ '[XSS](data:text/html;base64,PHNjcmlwdD5hbGVydChkb2N1bWVudC5kb21haW4pPC9zY3JpcHQ+Cg==)\n', ], }); await nextTick(); expect(wrapper.vm.$el.querySelector('a').getAttribute('href')).toBeNull(); }); it('sanitizes HTML', async () => { const findLink = () => wrapper.vm.$el.querySelector('.xss-link'); Object.assign(cell, { source: ['XSS\n'], }); await nextTick(); expect(findLink().dataset.remote).toBeUndefined(); expect(findLink().dataset.type).toBeUndefined(); }); describe('When parsing images', () => { it.each([ [ 'for relative images in root folder, it does', '![](local_image.png)\n', 'src="/raw/local_image', ], [ 'for relative images in child folders, it does', '![](data/local_image.png)\n', 'src="/raw/data', ], ["for embedded images, it doesn't", '![](data:image/jpeg;base64)\n', 'src="data:'], ["for images urls, it doesn't", '![](http://image.png)\n', 'src="http:'], ])('%s', async ([testMd, mustContain]) => { wrapper = buildMarkdownComponent([testMd], '/raw/'); await nextTick(); expect(wrapper.vm.$el.innerHTML).toContain(mustContain); }); }); describe('tables', () => { beforeEach(() => { json = markdownTableJson; }); it('renders images and text', async () => { wrapper = buildCellComponent(json.cells[0]); await nextTick(); const images = wrapper.vm.$el.querySelectorAll('img'); expect(images.length).toBe(5); const columns = wrapper.vm.$el.querySelectorAll('td'); expect(columns.length).toBe(6); expect(columns[0].textContent).toEqual('Hello '); expect(columns[1].textContent).toEqual('Test '); expect(columns[2].textContent).toEqual('World '); expect(columns[3].textContent).toEqual('Fake '); expect(columns[4].textContent).toEqual('External image: '); expect(columns[5].textContent).toEqual('Empty'); expect(columns[0].innerHTML).toContain(''); expect(columns[4].innerHTML).toContain(' { wrapper = buildMarkdownComponent([ "- list with inline $a=2$ inline formula $a' + b = c$\n", '\n', ]); await nextTick(); // expect one list with a katex formula in it expect(wrapper.vm.$el.querySelectorAll('li')).toHaveLength(1); expect(wrapper.vm.$el.querySelectorAll('li .katex')).toHaveLength(2); }); it('renders math formula with less-than-operator < in it', async () => { wrapper = buildMarkdownComponent([ '- list with inline $a=2$ inline formula $a + b < c$\n', '\n', ]); await nextTick(); // expect one list with a katex formula in it expect(wrapper.vm.$el.querySelectorAll('li')).toHaveLength(1); expect(wrapper.vm.$el.querySelectorAll('li .katex')).toHaveLength(2); }); it('renders math formula with greater-than-operator > in it', async () => { wrapper = buildMarkdownComponent([ '- list with inline $a=2$ inline formula $a + b > c$\n', '\n', ]); await nextTick(); // expect one list with a katex formula in it expect(wrapper.vm.$el.querySelectorAll('li')).toHaveLength(1); expect(wrapper.vm.$el.querySelectorAll('li .katex')).toHaveLength(2); }); }); });