extern crate markdown; use markdown::{ mdast::{Definition, Node, Root}, to_html, to_html_with_options, to_mdast, unist::Position, CompileOptions, Constructs, Options, ParseOptions, }; use pretty_assertions::assert_eq; #[test] fn definition() -> Result<(), String> { let danger = Options { compile: CompileOptions { allow_dangerous_html: true, allow_dangerous_protocol: true, ..CompileOptions::default() }, ..Options::default() }; assert_eq!( to_html("[foo]: /url \"title\"\n\n[foo]"), "
", "should support link definitions" ); assert_eq!( to_html("[foo]:\n\n/url\n\n[foo]"), "[foo]:
\n/url
\n[foo]
", "should not support blank lines before destination" ); assert_eq!( to_html(" [foo]: \n /url \n 'the title' \n\n[foo]"), "", "should support whitespace and line endings in definitions" ); assert_eq!( to_html("[a]:b 'c'\n\n[a]"), "", "should support no whitespace after `:` in definitions" ); assert_eq!( to_html("[Foo*bar\\]]:my_(url) 'title (with parens)'\n\n[Foo*bar\\]]"), "", "should support complex definitions (1)" ); assert_eq!( to_html("[Foo bar]:\n[foo]: /url 'title
\nwith blank line'
\n[foo]
", "should not support blank lines in titles" ); assert_eq!( to_html("[foo]:\n/url\n\n[foo]"), "", "should support definitions w/o title" ); assert_eq!( to_html("[foo]:\n\n[foo]"), "[foo]:
\n[foo]
", "should not support definitions w/o destination" ); assert_eq!( to_html("[foo]: <>\n\n[foo]"), "", "should support definitions w/ explicit empty destinations" ); assert_eq!( to_html_with_options("[foo]:[foo]:
[foo]
", "should not support definitions w/ no whitespace between destination and title" ); assert_eq!( to_html("[foo]: /url\\bar\\*baz \"foo\\\"bar\\baz\"\n\n[foo]"), "", "should support character escapes in destinations and titles" ); assert_eq!( to_html("[foo]\n\n[foo]: url"), "\n", "should support a link before a definition" ); assert_eq!( to_html("[foo]: first\n[foo]: second\n\n[foo]"), "", "should match w/ the first definition" ); assert_eq!( to_html("[FOO]: /url\n\n[Foo]"), "", "should match w/ case-insensitive (1)" ); assert_eq!( to_html("[ΑΓΩ]: /φου\n\n[αγω]"), "", "should match w/ case-insensitive (2)" ); assert_eq!( to_html("[ı]: a\n\n[I]"), "", "should match w/ undotted turkish i (1)" ); assert_eq!( to_html("[I]: a\n\n[ı]"), "", "should match w/ undotted turkish i (2)" ); // Ref:[İ]
", "should *not* match w/ dotted turkish i (1)" ); // Ref:[i]
", "should *not* match w/ dotted turkish i (2)" ); assert_eq!( to_html("[foo]: /url"), "", "should not contribute anything w/o reference (1)" ); assert_eq!( to_html("[\nfoo\n]: /url\nbar"), "bar
", "should not contribute anything w/o reference (2)" ); assert_eq!( to_html("[foo]: /url \"title\" \n\n[foo]"), "", "should support whitespace after title" ); assert_eq!( to_html("[foo]: /url\n\"title\" \n\n[foo]"), "", "should support whitespace after title on a separate line" ); assert_eq!( to_html("[foo]: /url \"title\" ok"), "[foo]: /url "title" ok
", "should not support non-whitespace content after definitions (1)" ); assert_eq!( to_html("[foo]: /url\n\"title\" ok"), ""title" ok
", "should not support non-whitespace content after definitions (2)" ); assert_eq!( to_html(" [foo]: /url \"title\"\n\n[foo]"), "[foo]: /url "title"\n
\n[foo]
", "should prefer indented code over definitions" ); assert_eq!( to_html("```\n[foo]: /url\n```\n\n[foo]"), "[foo]: /url\n
\n[foo]
", "should not support definitions in fenced code" ); assert_eq!( to_html("Foo\n[bar]: /baz\n\n[bar]"), "Foo\n[bar]: /baz
\n[bar]
", "should not support definitions in paragraphs" ); assert_eq!( to_html("# [Foo]\n[foo]: /url\n> bar"), "\n", "should not support definitions in headings" ); assert_eq!( to_html("[foo]: /url\nbar\n===\n[foo]"), "bar
\n
===\nfoo
", "should not support setext heading underlines after definitions" ); assert_eq!( to_html( "[foo]: /foo-url \"foo\"\n[bar]: /bar-url\n \"bar\"\n[baz]: /baz-url\n\n[foo],\n[bar],\n[baz]"), "", "should support definitions after definitions" ); assert_eq!( to_html("> [foo]: /url\n\n[foo]"), "\n\n", "should support definitions in block quotes (1)" ); assert_eq!( to_html("> [a]: <> 'b\n> c'"), "
\n", "should support definitions in block quotes (2)" ); assert_eq!( to_html("> [a]\n\n[a]: b (c\n)"), "
\n\n\n", "should support definitions in block quotes (3)" ); // Extra assert_eq!( to_html("[\\[\\+\\]]: example.com\n\nLink: [\\[\\+\\]]."), "
Link: [+].
", "should match w/ character escapes" ); assert_eq!( to_html("[x]: \\\" \\(\\)\\\"\n\n[x]"), "", "should support character escapes & references in unenclosed destinations" ); assert_eq!( to_html("[x]: <\\> \\+\\>>\n\n[x]"), "", "should support character escapes & references in enclosed destinations" ); assert_eq!( to_html("[x]: <\n\n[x]"), "[x]: <
\n[x]
", "should not support a line ending at start of enclosed destination" ); assert_eq!( to_html("[x]:[x]
", "should not support a line ending in enclosed destination" ); assert_eq!( to_html("[x]: \u{000b}a\n\n[x]"), "[x]: \u{000b}a
\n[x]
", "should not support ascii control characters at the start of destination" ); assert_eq!( to_html("[x]: a\u{000b}b\n\n[x]"), "[x]: a\u{000b}b
\n[x]
", "should not support ascii control characters in destination" ); assert_eq!( to_html("[x]: <\u{000b}a>\n\n[x]"), "", "should support ascii control characters at the start of enclosed destination" ); assert_eq!( to_html("[x]: \n\n[x]"), "", "should support ascii control characters in enclosed destinations" ); assert_eq!( to_html("[x]: a \"\\\"\"\n\n[x]"), "", "should support character escapes at the start of a title" ); assert_eq!( to_html("[x]: a \"'\"\n\n[x]"), "", "should support double quoted titles" ); assert_eq!( to_html("[x]: a '\"'\n\n[x]"), "", "should support single quoted titles" ); assert_eq!( to_html("[x]: a (\"')\n\n[x]"), "", "should support paren enclosed titles" ); assert_eq!( to_html("[x]: a(()\n\n[x]"), "[x]: a(()
\n[x]
", "should not support more opening than closing parens in the destination" ); assert_eq!( to_html("[x]: a(())\n\n[x]"), "", "should support balanced opening and closing parens in the destination" ); assert_eq!( to_html("[x]: a())\n\n[x]"), "[x]: a())
\n[x]
", "should not support more closing than opening parens in the destination" ); assert_eq!( to_html("[x]: a \t\n\n[x]"), "", "should support trailing whitespace after a destination" ); assert_eq!( to_html("[x]: a \"X\" \t\n\n[x]"), "", "should support trailing whitespace after a title" ); assert_eq!( to_html("[&©&]: example.com/&©& \"&©&\"\n\n[&©&]"), "", "should support character references in definitions" ); assert_eq!( to_html("[x]:\nexample.com\n\n[x]"), "", "should support a line ending before a destination" ); assert_eq!( to_html("[x]: \t\nexample.com\n\n[x]"), "", "should support whitespace before a destination" ); // See:[a]
\n[a]: <b
[a]
\n[a]: b(c
", "should not support an extra left paren (`(`) in a raw destination" ); assert_eq!( to_html("[a]\n\n[a]: b)c"), "[a]
\n[a]: b)c
", "should not support an extra right paren (`)`) in a raw destination" ); assert_eq!( to_html("[a]\n\n[a]: b)c"), "[a]
\n[a]: b)c
", "should not support an extra right paren (`)`) in a raw destination" ); assert_eq!( to_html("[a]\n\n[a]: a(1(2(3(4()))))b"), "\n", "should support 4 or more sets of parens in a raw destination (link resources don’t)" ); assert_eq!( to_html("[a]\n\n[a]: aaa)"), "[a]
\n[a]: aaa)
", "should not support a final (unbalanced) right paren in a raw destination" ); assert_eq!( to_html("[a]\n\n[a]: aaa) \"a\""), "[a]
\n[a]: aaa) "a"
", "should not support a final (unbalanced) right paren in a raw destination “before” a title" ); assert_eq!( to_html(" [a]: b \"c\"\n [d]: e\n [f]: g \"h\"\n [i]: j\n\t[k]: l (m)\n\t n [k] o"), "n k o
", "should support subsequent indented definitions" ); assert_eq!( to_html("[a\n b]: c\n\n[a\n b]"), "", "should support line prefixes in definition labels" ); assert_eq!( to_html("[a]: )\n\n[a]"), "[a]: )
\n[a]
", "should not support definitions w/ only a closing paren as a raw destination" ); assert_eq!( to_html("[a]: )b\n\n[a]"), "[a]: )b
\n[a]
", "should not support definitions w/ closing paren + more text as a raw destination" ); assert_eq!( to_html("[a]: b)\n\n[a]"), "[a]: b)
\n[a]
", "should not support definitions w/ text + a closing paren as a raw destination" ); assert_eq!( to_html("[\na\n=\n]: b"), "]: b
", "should prefer setext headings over definition labels" ); assert_eq!( to_html("[a]: b '\nc\n=\n'"), "'
", "should prefer setext headings over definition titles" ); assert_eq!( to_html("[\n***\n]: b"), "[
\n]: b
", "should prefer thematic breaks over definition labels" ); assert_eq!( to_html("[a]: b '\n***\n'"), "[a]: b '
\n'
", "should prefer thematic breaks over definition titles" ); assert_eq!( to_html("[\n```\n]: b"), "[
\n]: b\n
\n",
"should prefer code (fenced) over definition labels"
);
assert_eq!(
to_html("[a]: b '\n```\n'"),
"[a]: b '
\n'\n
\n",
"should prefer code (fenced) over definition titles"
);
assert_eq!(
to_html_with_options(
"[foo]: /url \"title\"",
&Options {
parse: ParseOptions {
constructs: Constructs {
definition: false,
..Constructs::default()
},
..ParseOptions::default()
},
..Options::default()
}
)?,
"[foo]: /url "title"
", "should support turning off definitions" ); assert_eq!( to_mdast("[a]: 'c'", &ParseOptions::default())?, Node::Root(Root { children: vec![Node::Definition(Definition { url: "b".into(), identifier: "a".into(), label: Some("a".into()), title: Some("c".into()), position: Some(Position::new(1, 1, 0, 1, 13, 12)) })], position: Some(Position::new(1, 1, 0, 1, 13, 12)) }), "should support definitions as `Definition`s in mdast" ); Ok(()) }