import { parse, ParserOptions, TextModes } from '../src/parse' import { ErrorCodes } from '../src/errors' import { CommentNode, ElementNode, ElementTypes, ExpressionNode, Namespaces, NodeTypes, Position, TextNode, AttributeNode } from '../src/ast' describe('compiler: parse', () => { describe('Text', () => { test('simple text', () => { const ast = parse('some text') const text = ast.children[0] as TextNode expect(text).toStrictEqual({ type: NodeTypes.TEXT, content: 'some text', isEmpty: false, loc: { start: { offset: 0, line: 1, column: 1 }, end: { offset: 9, line: 1, column: 10 }, source: 'some text' } }) }) test('simple text with invalid end tag', () => { const ast = parse('some text', { onError: () => {} }) const text = ast.children[0] as TextNode expect(text).toStrictEqual({ type: NodeTypes.TEXT, content: 'some text', isEmpty: false, loc: { start: { offset: 0, line: 1, column: 1 }, end: { offset: 9, line: 1, column: 10 }, source: 'some text' } }) }) test('text with interpolation', () => { const ast = parse('some {{ foo + bar }} text') const text1 = ast.children[0] as TextNode const text2 = ast.children[2] as TextNode expect(text1).toStrictEqual({ type: NodeTypes.TEXT, content: 'some ', isEmpty: false, loc: { start: { offset: 0, line: 1, column: 1 }, end: { offset: 5, line: 1, column: 6 }, source: 'some ' } }) expect(text2).toStrictEqual({ type: NodeTypes.TEXT, content: ' text', isEmpty: false, loc: { start: { offset: 20, line: 1, column: 21 }, end: { offset: 25, line: 1, column: 26 }, source: ' text' } }) }) test('text with interpolation which has `<`', () => { const ast = parse('some {{ ad }} text') const text1 = ast.children[0] as TextNode const text2 = ast.children[2] as TextNode expect(text1).toStrictEqual({ type: NodeTypes.TEXT, content: 'some ', isEmpty: false, loc: { start: { offset: 0, line: 1, column: 1 }, end: { offset: 5, line: 1, column: 6 }, source: 'some ' } }) expect(text2).toStrictEqual({ type: NodeTypes.TEXT, content: ' text', isEmpty: false, loc: { start: { offset: 21, line: 1, column: 22 }, end: { offset: 26, line: 1, column: 27 }, source: ' text' } }) }) test('text with mix of tags and interpolations', () => { const ast = parse('some {{ foo < bar + foo }} text') const text1 = ast.children[0] as TextNode const text2 = (ast.children[1] as ElementNode).children![1] as TextNode expect(text1).toStrictEqual({ type: NodeTypes.TEXT, content: 'some ', isEmpty: false, loc: { start: { offset: 0, line: 1, column: 1 }, end: { offset: 5, line: 1, column: 6 }, source: 'some ' } }) expect(text2).toStrictEqual({ type: NodeTypes.TEXT, content: ' text', isEmpty: false, loc: { start: { offset: 32, line: 1, column: 33 }, end: { offset: 37, line: 1, column: 38 }, source: ' text' } }) }) test('lonly "<" don\'t separate nodes', () => { const ast = parse('a < b', { onError: err => { if (err.code !== ErrorCodes.INVALID_FIRST_CHARACTER_OF_TAG_NAME) { throw err } } }) const text = ast.children[0] as TextNode expect(text).toStrictEqual({ type: NodeTypes.TEXT, content: 'a < b', isEmpty: false, loc: { start: { offset: 0, line: 1, column: 1 }, end: { offset: 5, line: 1, column: 6 }, source: 'a < b' } }) }) test('lonly "{{" don\'t separate nodes', () => { const ast = parse('a {{ b', { onError: error => { if (error.code !== ErrorCodes.X_MISSING_INTERPOLATION_END) { throw error } } }) const text = ast.children[0] as TextNode expect(text).toStrictEqual({ type: NodeTypes.TEXT, content: 'a {{ b', isEmpty: false, loc: { start: { offset: 0, line: 1, column: 1 }, end: { offset: 6, line: 1, column: 7 }, source: 'a {{ b' } }) }) test('HTML entities compatibility in text (https://html.spec.whatwg.org/multipage/parsing.html#named-character-reference-state).', () => { const spy = jest.fn() const ast = parse('&ersand;', { namedCharacterReferences: { amp: '&' }, onError: spy }) const text = ast.children[0] as TextNode expect(text).toStrictEqual({ type: NodeTypes.TEXT, content: '&ersand;', isEmpty: false, loc: { start: { offset: 0, line: 1, column: 1 }, end: { offset: 11, line: 1, column: 12 }, source: '&ersand;' } }) expect(spy.mock.calls).toMatchObject([ [ { code: ErrorCodes.MISSING_SEMICOLON_AFTER_CHARACTER_REFERENCE, loc: { offset: 4, line: 1, column: 5 } } ] ]) }) test('HTML entities compatibility in attribute (https://html.spec.whatwg.org/multipage/parsing.html#named-character-reference-state).', () => { const spy = jest.fn() const ast = parse( '
', { namedCharacterReferences: { amp: '&', 'amp;': '&' }, onError: spy } ) const element = ast.children[0] as ElementNode const text1 = (element.props[0] as AttributeNode).value const text2 = (element.props[1] as AttributeNode).value const text3 = (element.props[2] as AttributeNode).value expect(text1).toStrictEqual({ type: NodeTypes.TEXT, content: '&ersand;', isEmpty: false, loc: { start: { offset: 7, line: 1, column: 8 }, end: { offset: 20, line: 1, column: 21 }, source: '"&ersand;"' } }) expect(text2).toStrictEqual({ type: NodeTypes.TEXT, content: '&ersand;', isEmpty: false, loc: { start: { offset: 23, line: 1, column: 24 }, end: { offset: 37, line: 1, column: 38 }, source: '"&ersand;"' } }) expect(text3).toStrictEqual({ type: NodeTypes.TEXT, content: '&!', isEmpty: false, loc: { start: { offset: 40, line: 1, column: 41 }, end: { offset: 47, line: 1, column: 48 }, source: '"&!"' } }) expect(spy.mock.calls).toMatchObject([ [ { code: ErrorCodes.MISSING_SEMICOLON_AFTER_CHARACTER_REFERENCE, loc: { offset: 45, line: 1, column: 46 } } ] ]) }) test('Some control character reference should be replaced.', () => { const spy = jest.fn() const ast = parse('†', { onError: spy }) const text = ast.children[0] as TextNode expect(text).toStrictEqual({ type: NodeTypes.TEXT, content: '†', isEmpty: false, loc: { start: { offset: 0, line: 1, column: 1 }, end: { offset: 6, line: 1, column: 7 }, source: '†' } }) expect(spy.mock.calls).toMatchObject([ [ { code: ErrorCodes.CONTROL_CHARACTER_REFERENCE, loc: { offset: 0, line: 1, column: 1 } } ] ]) }) }) describe('Interpolation', () => { test('simple interpolation', () => { const ast = parse('{{message}}') const interpolation = ast.children[0] as ExpressionNode expect(interpolation).toStrictEqual({ type: NodeTypes.EXPRESSION, content: 'message', isStatic: false, loc: { start: { offset: 0, line: 1, column: 1 }, end: { offset: 11, line: 1, column: 12 }, source: '{{message}}' } }) }) test('it can have tag-like notation', () => { const ast = parse('{{ a { const ast = parse('{{ ad }}') const interpolation1 = ast.children[0] as ExpressionNode const interpolation2 = ast.children[1] as ExpressionNode expect(interpolation1).toStrictEqual({ type: NodeTypes.EXPRESSION, content: 'ad', isStatic: false, loc: { start: { offset: 9, line: 1, column: 10 }, end: { offset: 18, line: 1, column: 19 }, source: '{{ c>d }}' } }) }) test('it can have tag-like notation (3)', () => { const ast = parse('
{{ "
" }}') const element = ast.children[0] as ElementNode const interpolation = element.children[0] as ExpressionNode expect(interpolation).toStrictEqual({ type: NodeTypes.EXPRESSION, content: '""', isStatic: false, loc: { start: { offset: 5, line: 1, column: 6 }, end: { offset: 19, line: 1, column: 20 }, source: '{{ "" }}' } }) }) }) describe('Comment', () => { test('empty comment', () => { const ast = parse('') const comment = ast.children[0] as CommentNode expect(comment).toStrictEqual({ type: NodeTypes.COMMENT, content: '', loc: { start: { offset: 0, line: 1, column: 1 }, end: { offset: 7, line: 1, column: 8 }, source: '' } }) }) test('simple comment', () => { const ast = parse('') const comment = ast.children[0] as CommentNode expect(comment).toStrictEqual({ type: NodeTypes.COMMENT, content: 'abc', loc: { start: { offset: 0, line: 1, column: 1 }, end: { offset: 10, line: 1, column: 11 }, source: '' } }) }) test('two comments', () => { const ast = parse('') const comment1 = ast.children[0] as CommentNode const comment2 = ast.children[1] as CommentNode expect(comment1).toStrictEqual({ type: NodeTypes.COMMENT, content: 'abc', loc: { start: { offset: 0, line: 1, column: 1 }, end: { offset: 10, line: 1, column: 11 }, source: '' } }) expect(comment2).toStrictEqual({ type: NodeTypes.COMMENT, content: 'def', loc: { start: { offset: 10, line: 1, column: 11 }, end: { offset: 20, line: 1, column: 21 }, source: '' } }) }) }) describe('Element', () => { test('simple div', () => { const ast = parse('
hello
') const element = ast.children[0] as ElementNode expect(element).toStrictEqual({ type: NodeTypes.ELEMENT, ns: Namespaces.HTML, tag: 'div', tagType: ElementTypes.ELEMENT, codegenNode: undefined, props: [], isSelfClosing: false, children: [ { type: NodeTypes.TEXT, content: 'hello', isEmpty: false, loc: { start: { offset: 5, line: 1, column: 6 }, end: { offset: 10, line: 1, column: 11 }, source: 'hello' } } ], loc: { start: { offset: 0, line: 1, column: 1 }, end: { offset: 16, line: 1, column: 17 }, source: '
hello
' } }) }) test('empty', () => { const ast = parse('
') const element = ast.children[0] as ElementNode expect(element).toStrictEqual({ type: NodeTypes.ELEMENT, ns: Namespaces.HTML, tag: 'div', tagType: ElementTypes.ELEMENT, codegenNode: undefined, props: [], isSelfClosing: false, children: [], loc: { start: { offset: 0, line: 1, column: 1 }, end: { offset: 11, line: 1, column: 12 }, source: '
' } }) }) test('self closing', () => { const ast = parse('
after') const element = ast.children[0] as ElementNode expect(element).toStrictEqual({ type: NodeTypes.ELEMENT, ns: Namespaces.HTML, tag: 'div', tagType: ElementTypes.ELEMENT, codegenNode: undefined, props: [], isSelfClosing: true, children: [], loc: { start: { offset: 0, line: 1, column: 1 }, end: { offset: 6, line: 1, column: 7 }, source: '
' } }) }) test('void element', () => { const ast = parse('after', { isVoidTag: tag => tag === 'img' }) const element = ast.children[0] as ElementNode expect(element).toStrictEqual({ type: NodeTypes.ELEMENT, ns: Namespaces.HTML, tag: 'img', tagType: ElementTypes.ELEMENT, codegenNode: undefined, props: [], isSelfClosing: false, children: [], loc: { start: { offset: 0, line: 1, column: 1 }, end: { offset: 5, line: 1, column: 6 }, source: '' } }) }) test('attribute with no value', () => { const ast = parse('
') const element = ast.children[0] as ElementNode expect(element).toStrictEqual({ type: NodeTypes.ELEMENT, ns: Namespaces.HTML, tag: 'div', tagType: ElementTypes.ELEMENT, codegenNode: undefined, props: [ { type: NodeTypes.ATTRIBUTE, name: 'id', value: undefined, loc: { start: { offset: 5, line: 1, column: 6 }, end: { offset: 7, line: 1, column: 8 }, source: 'id' } } ], isSelfClosing: false, children: [], loc: { start: { offset: 0, line: 1, column: 1 }, end: { offset: 14, line: 1, column: 15 }, source: '
' } }) }) test('attribute with empty value, double quote', () => { const ast = parse('
') const element = ast.children[0] as ElementNode expect(element).toStrictEqual({ type: NodeTypes.ELEMENT, ns: Namespaces.HTML, tag: 'div', tagType: ElementTypes.ELEMENT, codegenNode: undefined, props: [ { type: NodeTypes.ATTRIBUTE, name: 'id', value: { type: NodeTypes.TEXT, content: '', isEmpty: true, loc: { start: { offset: 8, line: 1, column: 9 }, end: { offset: 10, line: 1, column: 11 }, source: '""' } }, loc: { start: { offset: 5, line: 1, column: 6 }, end: { offset: 10, line: 1, column: 11 }, source: 'id=""' } } ], isSelfClosing: false, children: [], loc: { start: { offset: 0, line: 1, column: 1 }, end: { offset: 17, line: 1, column: 18 }, source: '
' } }) }) test('attribute with empty value, single quote', () => { const ast = parse("
") const element = ast.children[0] as ElementNode expect(element).toStrictEqual({ type: NodeTypes.ELEMENT, ns: Namespaces.HTML, tag: 'div', tagType: ElementTypes.ELEMENT, codegenNode: undefined, props: [ { type: NodeTypes.ATTRIBUTE, name: 'id', value: { type: NodeTypes.TEXT, content: '', isEmpty: true, loc: { start: { offset: 8, line: 1, column: 9 }, end: { offset: 10, line: 1, column: 11 }, source: "''" } }, loc: { start: { offset: 5, line: 1, column: 6 }, end: { offset: 10, line: 1, column: 11 }, source: "id=''" } } ], isSelfClosing: false, children: [], loc: { start: { offset: 0, line: 1, column: 1 }, end: { offset: 17, line: 1, column: 18 }, source: "
" } }) }) test('attribute with value, double quote', () => { const ast = parse('
') const element = ast.children[0] as ElementNode expect(element).toStrictEqual({ type: NodeTypes.ELEMENT, ns: Namespaces.HTML, tag: 'div', tagType: ElementTypes.ELEMENT, codegenNode: undefined, props: [ { type: NodeTypes.ATTRIBUTE, name: 'id', value: { type: NodeTypes.TEXT, content: ">'", isEmpty: false, loc: { start: { offset: 8, line: 1, column: 9 }, end: { offset: 12, line: 1, column: 13 }, source: '">\'"' } }, loc: { start: { offset: 5, line: 1, column: 6 }, end: { offset: 12, line: 1, column: 13 }, source: 'id=">\'"' } } ], isSelfClosing: false, children: [], loc: { start: { offset: 0, line: 1, column: 1 }, end: { offset: 19, line: 1, column: 20 }, source: '
' } }) }) test('attribute with value, single quote', () => { const ast = parse("
") const element = ast.children[0] as ElementNode expect(element).toStrictEqual({ type: NodeTypes.ELEMENT, ns: Namespaces.HTML, tag: 'div', tagType: ElementTypes.ELEMENT, codegenNode: undefined, props: [ { type: NodeTypes.ATTRIBUTE, name: 'id', value: { type: NodeTypes.TEXT, content: '>"', isEmpty: false, loc: { start: { offset: 8, line: 1, column: 9 }, end: { offset: 12, line: 1, column: 13 }, source: "'>\"'" } }, loc: { start: { offset: 5, line: 1, column: 6 }, end: { offset: 12, line: 1, column: 13 }, source: "id='>\"'" } } ], isSelfClosing: false, children: [], loc: { start: { offset: 0, line: 1, column: 1 }, end: { offset: 19, line: 1, column: 20 }, source: "
" } }) }) test('attribute with value, unquoted', () => { const ast = parse('
') const element = ast.children[0] as ElementNode expect(element).toStrictEqual({ type: NodeTypes.ELEMENT, ns: Namespaces.HTML, tag: 'div', tagType: ElementTypes.ELEMENT, codegenNode: undefined, props: [ { type: NodeTypes.ATTRIBUTE, name: 'id', value: { type: NodeTypes.TEXT, content: 'a/', isEmpty: false, loc: { start: { offset: 8, line: 1, column: 9 }, end: { offset: 10, line: 1, column: 11 }, source: 'a/' } }, loc: { start: { offset: 5, line: 1, column: 6 }, end: { offset: 10, line: 1, column: 11 }, source: 'id=a/' } } ], isSelfClosing: false, children: [], loc: { start: { offset: 0, line: 1, column: 1 }, end: { offset: 17, line: 1, column: 18 }, source: '
' } }) }) test('multiple attributes', () => { const ast = parse('
') const element = ast.children[0] as ElementNode expect(element).toStrictEqual({ type: NodeTypes.ELEMENT, ns: Namespaces.HTML, tag: 'div', tagType: ElementTypes.ELEMENT, codegenNode: undefined, props: [ { type: NodeTypes.ATTRIBUTE, name: 'id', value: { type: NodeTypes.TEXT, content: 'a', isEmpty: false, loc: { start: { offset: 8, line: 1, column: 9 }, end: { offset: 9, line: 1, column: 10 }, source: 'a' } }, loc: { start: { offset: 5, line: 1, column: 6 }, end: { offset: 9, line: 1, column: 10 }, source: 'id=a' } }, { type: NodeTypes.ATTRIBUTE, name: 'class', value: { type: NodeTypes.TEXT, content: 'c', isEmpty: false, loc: { start: { offset: 16, line: 1, column: 17 }, end: { offset: 19, line: 1, column: 20 }, source: '"c"' } }, loc: { start: { offset: 10, line: 1, column: 11 }, end: { offset: 19, line: 1, column: 20 }, source: 'class="c"' } }, { type: NodeTypes.ATTRIBUTE, name: 'inert', value: undefined, loc: { start: { offset: 20, line: 1, column: 21 }, end: { offset: 25, line: 1, column: 26 }, source: 'inert' } }, { type: NodeTypes.ATTRIBUTE, name: 'style', value: { type: NodeTypes.TEXT, content: '', isEmpty: true, loc: { start: { offset: 32, line: 1, column: 33 }, end: { offset: 34, line: 1, column: 35 }, source: "''" } }, loc: { start: { offset: 26, line: 1, column: 27 }, end: { offset: 34, line: 1, column: 35 }, source: "style=''" } } ], isSelfClosing: false, children: [], loc: { start: { offset: 0, line: 1, column: 1 }, end: { offset: 41, line: 1, column: 42 }, source: '
' } }) }) test('directive with no value', () => { const ast = parse('
') const directive = (ast.children[0] as ElementNode).props[0] expect(directive).toStrictEqual({ type: NodeTypes.DIRECTIVE, name: 'if', arg: undefined, modifiers: [], exp: undefined, loc: { start: { offset: 5, line: 1, column: 6 }, end: { offset: 9, line: 1, column: 10 }, source: 'v-if' } }) }) test('directive with value', () => { const ast = parse('
') const directive = (ast.children[0] as ElementNode).props[0] expect(directive).toStrictEqual({ type: NodeTypes.DIRECTIVE, name: 'if', arg: undefined, modifiers: [], exp: { type: NodeTypes.EXPRESSION, content: 'a', isStatic: false, loc: { start: { offset: 10, line: 1, column: 11 }, end: { offset: 13, line: 1, column: 14 }, source: '"a"' } }, loc: { start: { offset: 5, line: 1, column: 6 }, end: { offset: 13, line: 1, column: 14 }, source: 'v-if="a"' } }) }) test('directive with argument', () => { const ast = parse('
') const directive = (ast.children[0] as ElementNode).props[0] expect(directive).toStrictEqual({ type: NodeTypes.DIRECTIVE, name: 'on', arg: { type: 4, content: 'click', isStatic: true, loc: { source: 'click', start: { column: 11, line: 1, offset: 10 }, end: { column: 16, line: 1, offset: 15 } } }, modifiers: [], exp: undefined, loc: { start: { offset: 5, line: 1, column: 6 }, end: { offset: 15, line: 1, column: 16 }, source: 'v-on:click' } }) }) test('directive with a modifier', () => { const ast = parse('
') const directive = (ast.children[0] as ElementNode).props[0] expect(directive).toStrictEqual({ type: NodeTypes.DIRECTIVE, name: 'on', arg: undefined, modifiers: ['enter'], exp: undefined, loc: { start: { offset: 5, line: 1, column: 6 }, end: { offset: 15, line: 1, column: 16 }, source: 'v-on.enter' } }) }) test('directive with two modifiers', () => { const ast = parse('
') const directive = (ast.children[0] as ElementNode).props[0] expect(directive).toStrictEqual({ type: NodeTypes.DIRECTIVE, name: 'on', arg: undefined, modifiers: ['enter', 'exact'], exp: undefined, loc: { start: { offset: 5, line: 1, column: 6 }, end: { offset: 21, line: 1, column: 22 }, source: 'v-on.enter.exact' } }) }) test('directive with argument and modifiers', () => { const ast = parse('
') const directive = (ast.children[0] as ElementNode).props[0] expect(directive).toStrictEqual({ type: NodeTypes.DIRECTIVE, name: 'on', arg: { type: 4, content: 'click', isStatic: true, loc: { source: 'click', start: { column: 11, line: 1, offset: 10 }, end: { column: 16, line: 1, offset: 15 } } }, modifiers: ['enter', 'exact'], exp: undefined, loc: { start: { offset: 5, line: 1, column: 6 }, end: { offset: 27, line: 1, column: 28 }, source: 'v-on:click.enter.exact' } }) }) test('v-bind shorthand', () => { const ast = parse('
') const directive = (ast.children[0] as ElementNode).props[0] expect(directive).toStrictEqual({ type: NodeTypes.DIRECTIVE, name: 'bind', arg: { type: 4, content: 'a', isStatic: true, loc: { source: 'a', start: { column: 7, line: 1, offset: 6 }, end: { column: 8, line: 1, offset: 7 } } }, modifiers: [], exp: { type: NodeTypes.EXPRESSION, content: 'b', isStatic: false, loc: { start: { offset: 8, line: 1, column: 9 }, end: { offset: 9, line: 1, column: 10 }, source: 'b' } }, loc: { start: { offset: 5, line: 1, column: 6 }, end: { offset: 9, line: 1, column: 10 }, source: ':a=b' } }) }) test('v-bind shorthand with modifier', () => { const ast = parse('
') const directive = (ast.children[0] as ElementNode).props[0] expect(directive).toStrictEqual({ type: NodeTypes.DIRECTIVE, name: 'bind', arg: { type: 4, content: 'a', isStatic: true, loc: { source: 'a', start: { column: 7, line: 1, offset: 6 }, end: { column: 8, line: 1, offset: 7 } } }, modifiers: ['sync'], exp: { type: NodeTypes.EXPRESSION, content: 'b', isStatic: false, loc: { start: { offset: 13, line: 1, column: 14 }, end: { offset: 14, line: 1, column: 15 }, source: 'b' } }, loc: { start: { offset: 5, line: 1, column: 6 }, end: { offset: 14, line: 1, column: 15 }, source: ':a.sync=b' } }) }) test('v-on shorthand', () => { const ast = parse('
') const directive = (ast.children[0] as ElementNode).props[0] expect(directive).toStrictEqual({ type: NodeTypes.DIRECTIVE, name: 'on', arg: { type: 4, content: 'a', isStatic: true, loc: { source: 'a', start: { column: 7, line: 1, offset: 6 }, end: { column: 8, line: 1, offset: 7 } } }, modifiers: [], exp: { type: NodeTypes.EXPRESSION, content: 'b', isStatic: false, loc: { start: { offset: 8, line: 1, column: 9 }, end: { offset: 9, line: 1, column: 10 }, source: 'b' } }, loc: { start: { offset: 5, line: 1, column: 6 }, end: { offset: 9, line: 1, column: 10 }, source: '@a=b' } }) }) test('v-on shorthand with modifier', () => { const ast = parse('
') const directive = (ast.children[0] as ElementNode).props[0] expect(directive).toStrictEqual({ type: NodeTypes.DIRECTIVE, name: 'on', arg: { type: 4, content: 'a', isStatic: true, loc: { source: 'a', start: { column: 7, line: 1, offset: 6 }, end: { column: 8, line: 1, offset: 7 } } }, modifiers: ['enter'], exp: { type: NodeTypes.EXPRESSION, content: 'b', isStatic: false, loc: { start: { offset: 14, line: 1, column: 15 }, end: { offset: 15, line: 1, column: 16 }, source: 'b' } }, loc: { start: { offset: 5, line: 1, column: 6 }, end: { offset: 15, line: 1, column: 16 }, source: '@a.enter=b' } }) }) test('end tags are case-insensitive.', () => { const ast = parse('
hello
after') const element = ast.children[0] as ElementNode const text = element.children[0] as TextNode expect(text).toStrictEqual({ type: NodeTypes.TEXT, content: 'hello', isEmpty: false, loc: { start: { offset: 5, line: 1, column: 6 }, end: { offset: 10, line: 1, column: 11 }, source: 'hello' } }) }) }) test('self closing single tag', () => { const ast = parse('
') expect(ast.children).toHaveLength(1) expect(ast.children[0]).toMatchObject({ tag: 'div' }) }) test('self closing multiple tag', () => { const ast = parse( `
\n` + `

` ) expect(ast).toMatchSnapshot() expect(ast.children).toHaveLength(2) expect(ast.children[0]).toMatchObject({ tag: 'div' }) expect(ast.children[1]).toMatchObject({ tag: 'p' }) }) test('valid html', () => { const ast = parse( `

\n` + `

\n` + ` \n` + `

` ) expect(ast).toMatchSnapshot() expect(ast.children).toHaveLength(1) const el = ast.children[0] as any expect(el).toMatchObject({ tag: 'div' }) expect(el.children).toHaveLength(2) expect(el.children[0]).toMatchObject({ tag: 'p' }) expect(el.children[1]).toMatchObject({ type: NodeTypes.COMMENT }) }) test('invalid html', () => { expect(() => { parse(`
\n\n
\n`) }).toThrow('End tag was not found. (3:1)') const spy = jest.fn() const ast = parse(`
\n\n
\n`, { onError: spy }) expect(spy.mock.calls).toMatchObject([ [ { code: ErrorCodes.X_MISSING_END_TAG, loc: { offset: 13, line: 3, column: 1 } } ], [ { code: ErrorCodes.X_INVALID_END_TAG, loc: { offset: 20, line: 4, column: 1 } } ] ]) expect(ast).toMatchSnapshot() }) test('parse with correct location info', () => { const [foo, bar, but, baz] = parse( 'foo \n is {{ bar }} but {{ baz }}' ).children let offset = 0 expect(foo.loc.start).toEqual({ line: 1, column: 1, offset }) offset += foo.loc.source.length expect(foo.loc.end).toEqual({ line: 2, column: 4, offset }) expect(bar.loc.start).toEqual({ line: 2, column: 4, offset }) offset += bar.loc.source.length expect(bar.loc.end).toEqual({ line: 2, column: 13, offset }) expect(but.loc.start).toEqual({ line: 2, column: 13, offset }) offset += but.loc.source.length expect(but.loc.end).toEqual({ line: 2, column: 18, offset }) expect(baz.loc.start).toEqual({ line: 2, column: 18, offset }) offset += baz.loc.source.length expect(baz.loc.end).toEqual({ line: 2, column: 27, offset }) }) describe('namedCharacterReferences option', () => { test('use the given map', () => { const ast: any = parse('&∪︀', { namedCharacterReferences: { 'cups;': '\u222A\uFE00' // UNION with serifs }, onError: () => {} // Ignore errors }) expect(ast.children.length).toBe(1) expect(ast.children[0].type).toBe(NodeTypes.TEXT) expect(ast.children[0].content).toBe('&\u222A\uFE00') }) }) describe('Errors', () => { const patterns: { [key: string]: Array<{ code: string errors: Array<{ type: ErrorCodes; loc: Position }> options?: Partial }> } = { ABRUPT_CLOSING_OF_EMPTY_COMMENT: [ { code: '', errors: [ { type: ErrorCodes.ABRUPT_CLOSING_OF_EMPTY_COMMENT, loc: { offset: 10, line: 1, column: 11 } } ] }, { code: '', errors: [ { type: ErrorCodes.ABRUPT_CLOSING_OF_EMPTY_COMMENT, loc: { offset: 10, line: 1, column: 11 } } ] }, { code: '', errors: [] } ], ABSENCE_OF_DIGITS_IN_NUMERIC_CHARACTER_REFERENCE: [ { code: '', errors: [ { type: ErrorCodes.ABSENCE_OF_DIGITS_IN_NUMERIC_CHARACTER_REFERENCE, loc: { offset: 10, line: 1, column: 11 } } ] }, { code: '', errors: [ { type: ErrorCodes.ABSENCE_OF_DIGITS_IN_NUMERIC_CHARACTER_REFERENCE, loc: { offset: 10, line: 1, column: 11 } } ] }, { code: '', errors: [] }, { code: '', errors: [] }, { code: '', errors: [ { type: ErrorCodes.ABSENCE_OF_DIGITS_IN_NUMERIC_CHARACTER_REFERENCE, loc: { offset: 16, line: 1, column: 17 } } ] }, { code: '', errors: [ { type: ErrorCodes.ABSENCE_OF_DIGITS_IN_NUMERIC_CHARACTER_REFERENCE, loc: { offset: 16, line: 1, column: 17 } } ] }, { code: '', errors: [] }, { code: '', errors: [] } ], CDATA_IN_HTML_CONTENT: [ { code: '', errors: [ { type: ErrorCodes.CDATA_IN_HTML_CONTENT, loc: { offset: 10, line: 1, column: 11 } } ] }, { code: '', errors: [] } ], CHARACTER_REFERENCE_OUTSIDE_UNICODE_RANGE: [ { code: '', errors: [ { type: ErrorCodes.CHARACTER_REFERENCE_OUTSIDE_UNICODE_RANGE, loc: { offset: 10, line: 1, column: 11 } } ] } ], CONTROL_CHARACTER_REFERENCE: [ { code: '', errors: [ { type: ErrorCodes.CONTROL_CHARACTER_REFERENCE, loc: { offset: 10, line: 1, column: 11 } } ] }, { code: '', errors: [ { type: ErrorCodes.CONTROL_CHARACTER_REFERENCE, loc: { offset: 10, line: 1, column: 11 } } ] } ], DUPLICATE_ATTRIBUTE: [ { code: '', errors: [ { type: ErrorCodes.DUPLICATE_ATTRIBUTE, loc: { offset: 21, line: 1, column: 22 } } ] } ], END_TAG_WITH_ATTRIBUTES: [ { code: '', errors: [ { type: ErrorCodes.END_TAG_WITH_ATTRIBUTES, loc: { offset: 21, line: 1, column: 22 } } ] } ], END_TAG_WITH_TRAILING_SOLIDUS: [ { code: '', errors: [ { type: ErrorCodes.END_TAG_WITH_TRAILING_SOLIDUS, loc: { offset: 20, line: 1, column: 21 } } ] } ], EOF_BEFORE_TAG_NAME: [ { code: '