2018-03-18 10:46:28 -07:00
|
|
|
|
{-# LANGUAGE NoImplicitPrelude #-}
|
2013-01-23 08:47:43 -08:00
|
|
|
|
{-# LANGUAGE OverloadedStrings #-}
|
2019-02-04 22:52:31 +01:00
|
|
|
|
{- |
|
|
|
|
|
Module : Tests.Readers.Markdown
|
|
|
|
|
Copyright : © 2006-2019 John MacFarlane
|
|
|
|
|
License : GNU GPL, version 2 or above
|
|
|
|
|
|
|
|
|
|
Maintainer : John MacFarlane <jgm@berkeley.edu>
|
|
|
|
|
Stability : alpha
|
|
|
|
|
Portability : portable
|
|
|
|
|
|
|
|
|
|
Tests for the Markdown reader.
|
|
|
|
|
-}
|
2011-01-26 22:09:09 -08:00
|
|
|
|
module Tests.Readers.Markdown (tests) where
|
|
|
|
|
|
2018-03-18 10:46:28 -07:00
|
|
|
|
import Prelude
|
2017-06-10 18:26:44 +02:00
|
|
|
|
import Data.Text (Text, unpack)
|
|
|
|
|
import qualified Data.Text as T
|
2017-03-14 17:05:36 +01:00
|
|
|
|
import Test.Tasty
|
2011-01-26 22:09:09 -08:00
|
|
|
|
import Tests.Helpers
|
|
|
|
|
import Text.Pandoc
|
2017-03-04 13:03:41 +01:00
|
|
|
|
import Text.Pandoc.Arbitrary ()
|
|
|
|
|
import Text.Pandoc.Builder
|
2011-01-26 22:09:09 -08:00
|
|
|
|
|
2017-06-10 18:26:44 +02:00
|
|
|
|
markdown :: Text -> Pandoc
|
2017-01-15 20:42:00 +01:00
|
|
|
|
markdown = purely $ readMarkdown def { readerExtensions =
|
|
|
|
|
disableExtension Ext_smart pandocExtensions }
|
2011-01-26 22:09:09 -08:00
|
|
|
|
|
2017-06-10 18:26:44 +02:00
|
|
|
|
markdownSmart :: Text -> Pandoc
|
2017-01-14 18:27:06 +01:00
|
|
|
|
markdownSmart = purely $ readMarkdown def { readerExtensions =
|
2017-01-15 20:42:00 +01:00
|
|
|
|
enableExtension Ext_smart pandocExtensions }
|
2011-07-25 23:49:45 -07:00
|
|
|
|
|
2017-06-10 18:26:44 +02:00
|
|
|
|
markdownCDL :: Text -> Pandoc
|
2017-01-14 13:06:27 +01:00
|
|
|
|
markdownCDL = purely $ readMarkdown def { readerExtensions = enableExtension
|
2017-01-15 20:42:00 +01:00
|
|
|
|
Ext_compact_definition_lists pandocExtensions }
|
2014-07-20 16:33:59 -07:00
|
|
|
|
|
2017-06-10 18:26:44 +02:00
|
|
|
|
markdownGH :: Text -> Pandoc
|
2017-01-15 20:42:00 +01:00
|
|
|
|
markdownGH = purely $ readMarkdown def {
|
|
|
|
|
readerExtensions = githubMarkdownExtensions }
|
2014-09-26 13:32:08 +04:00
|
|
|
|
|
2012-02-05 13:23:06 -08:00
|
|
|
|
infix 4 =:
|
2011-01-26 22:09:09 -08:00
|
|
|
|
(=:) :: ToString c
|
2017-06-10 18:26:44 +02:00
|
|
|
|
=> String -> (Text, c) -> TestTree
|
2011-01-26 22:09:09 -08:00
|
|
|
|
(=:) = test markdown
|
|
|
|
|
|
2017-06-10 18:26:44 +02:00
|
|
|
|
testBareLink :: (Text, Inlines) -> TestTree
|
2013-01-15 12:28:31 -08:00
|
|
|
|
testBareLink (inp, ils) =
|
2016-12-01 12:47:05 -05:00
|
|
|
|
test (purely $ readMarkdown def{ readerExtensions =
|
2017-01-14 13:06:27 +01:00
|
|
|
|
extensionsFromList [Ext_autolink_bare_uris, Ext_raw_html] })
|
2017-06-10 18:26:44 +02:00
|
|
|
|
(unpack inp) (inp, doc $ para ils)
|
2013-01-15 12:28:31 -08:00
|
|
|
|
|
|
|
|
|
autolink :: String -> Inlines
|
2018-09-19 14:49:46 -07:00
|
|
|
|
autolink = autolinkWith ("",["uri"],[])
|
2016-10-26 12:18:58 +02:00
|
|
|
|
|
|
|
|
|
autolinkWith :: Attr -> String -> Inlines
|
|
|
|
|
autolinkWith attr s = linkWith attr s "" (str s)
|
2013-01-15 12:28:31 -08:00
|
|
|
|
|
2017-06-10 18:26:44 +02:00
|
|
|
|
bareLinkTests :: [(Text, Inlines)]
|
2013-01-15 12:28:31 -08:00
|
|
|
|
bareLinkTests =
|
|
|
|
|
[ ("http://google.com is a search engine.",
|
|
|
|
|
autolink "http://google.com" <> " is a search engine.")
|
2013-09-01 15:18:56 -07:00
|
|
|
|
, ("<a href=\"http://foo.bar.baz\">http://foo.bar.baz</a>",
|
|
|
|
|
rawInline "html" "<a href=\"http://foo.bar.baz\">" <>
|
|
|
|
|
"http://foo.bar.baz" <> rawInline "html" "</a>")
|
2013-01-15 12:28:31 -08:00
|
|
|
|
, ("Try this query: http://google.com?search=fish&time=hour.",
|
|
|
|
|
"Try this query: " <> autolink "http://google.com?search=fish&time=hour" <> ".")
|
|
|
|
|
, ("HTTPS://GOOGLE.COM,",
|
|
|
|
|
autolink "HTTPS://GOOGLE.COM" <> ",")
|
|
|
|
|
, ("http://el.wikipedia.org/wiki/Τεχνολογία,",
|
|
|
|
|
autolink "http://el.wikipedia.org/wiki/Τεχνολογία" <> ",")
|
|
|
|
|
, ("doi:10.1000/182,",
|
|
|
|
|
autolink "doi:10.1000/182" <> ",")
|
|
|
|
|
, ("git://github.com/foo/bar.git,",
|
|
|
|
|
autolink "git://github.com/foo/bar.git" <> ",")
|
|
|
|
|
, ("file:///Users/joe/joe.txt, and",
|
|
|
|
|
autolink "file:///Users/joe/joe.txt" <> ", and")
|
|
|
|
|
, ("mailto:someone@somedomain.com.",
|
|
|
|
|
autolink "mailto:someone@somedomain.com" <> ".")
|
|
|
|
|
, ("Use http: this is not a link!",
|
|
|
|
|
"Use http: this is not a link!")
|
|
|
|
|
, ("(http://google.com).",
|
|
|
|
|
"(" <> autolink "http://google.com" <> ").")
|
|
|
|
|
, ("http://en.wikipedia.org/wiki/Sprite_(computer_graphics)",
|
|
|
|
|
autolink "http://en.wikipedia.org/wiki/Sprite_(computer_graphics)")
|
|
|
|
|
, ("http://en.wikipedia.org/wiki/Sprite_[computer_graphics]",
|
2018-09-19 14:49:46 -07:00
|
|
|
|
linkWith ("",["uri"],[])
|
|
|
|
|
"http://en.wikipedia.org/wiki/Sprite_%5Bcomputer_graphics%5D" ""
|
Percent-encode more special characters in URLs.
HTML, LaTeX writers adjusted.
The special characters are '<','>','|','"','{','}','[',']','^', '`'.
Closes #1640, #2377.
2015-10-11 17:06:26 -07:00
|
|
|
|
(str "http://en.wikipedia.org/wiki/Sprite_[computer_graphics]"))
|
2013-01-15 12:28:31 -08:00
|
|
|
|
, ("http://en.wikipedia.org/wiki/Sprite_{computer_graphics}",
|
2018-09-19 14:49:46 -07:00
|
|
|
|
linkWith ("",["uri"],[])
|
|
|
|
|
"http://en.wikipedia.org/wiki/Sprite_%7Bcomputer_graphics%7D" ""
|
Percent-encode more special characters in URLs.
HTML, LaTeX writers adjusted.
The special characters are '<','>','|','"','{','}','[',']','^', '`'.
Closes #1640, #2377.
2015-10-11 17:06:26 -07:00
|
|
|
|
(str "http://en.wikipedia.org/wiki/Sprite_{computer_graphics}"))
|
2013-01-15 12:28:31 -08:00
|
|
|
|
, ("http://example.com/Notification_Center-GitHub-20101108-140050.jpg",
|
|
|
|
|
autolink "http://example.com/Notification_Center-GitHub-20101108-140050.jpg")
|
|
|
|
|
, ("https://github.com/github/hubot/blob/master/scripts/cream.js#L20-20",
|
|
|
|
|
autolink "https://github.com/github/hubot/blob/master/scripts/cream.js#L20-20")
|
|
|
|
|
, ("http://www.rubyonrails.com",
|
|
|
|
|
autolink "http://www.rubyonrails.com")
|
|
|
|
|
, ("http://www.rubyonrails.com:80",
|
|
|
|
|
autolink "http://www.rubyonrails.com:80")
|
|
|
|
|
, ("http://www.rubyonrails.com/~minam",
|
|
|
|
|
autolink "http://www.rubyonrails.com/~minam")
|
|
|
|
|
, ("https://www.rubyonrails.com/~minam",
|
|
|
|
|
autolink "https://www.rubyonrails.com/~minam")
|
|
|
|
|
, ("http://www.rubyonrails.com/~minam/url%20with%20spaces",
|
|
|
|
|
autolink "http://www.rubyonrails.com/~minam/url%20with%20spaces")
|
|
|
|
|
, ("http://www.rubyonrails.com/foo.cgi?something=here",
|
|
|
|
|
autolink "http://www.rubyonrails.com/foo.cgi?something=here")
|
|
|
|
|
, ("http://www.rubyonrails.com/foo.cgi?something=here&and=here",
|
|
|
|
|
autolink "http://www.rubyonrails.com/foo.cgi?something=here&and=here")
|
|
|
|
|
, ("http://www.rubyonrails.com/contact;new",
|
|
|
|
|
autolink "http://www.rubyonrails.com/contact;new")
|
|
|
|
|
, ("http://www.rubyonrails.com/contact;new%20with%20spaces",
|
|
|
|
|
autolink "http://www.rubyonrails.com/contact;new%20with%20spaces")
|
|
|
|
|
, ("http://www.rubyonrails.com/contact;new?with=query&string=params",
|
|
|
|
|
autolink "http://www.rubyonrails.com/contact;new?with=query&string=params")
|
|
|
|
|
, ("http://www.rubyonrails.com/~minam/contact;new?with=query&string=params",
|
|
|
|
|
autolink "http://www.rubyonrails.com/~minam/contact;new?with=query&string=params")
|
|
|
|
|
, ("http://en.wikipedia.org/wiki/Wikipedia:Today%27s_featured_picture_%28animation%29/January_20%2C_2007",
|
|
|
|
|
autolink "http://en.wikipedia.org/wiki/Wikipedia:Today%27s_featured_picture_%28animation%29/January_20%2C_2007")
|
|
|
|
|
, ("http://www.mail-archive.com/rails@lists.rubyonrails.org/",
|
|
|
|
|
autolink "http://www.mail-archive.com/rails@lists.rubyonrails.org/")
|
|
|
|
|
, ("http://www.amazon.com/Testing-Equal-Sign-In-Path/ref=pd_bbs_sr_1?ie=UTF8&s=books&qid=1198861734&sr=8-1",
|
|
|
|
|
autolink "http://www.amazon.com/Testing-Equal-Sign-In-Path/ref=pd_bbs_sr_1?ie=UTF8&s=books&qid=1198861734&sr=8-1")
|
|
|
|
|
, ("http://en.wikipedia.org/wiki/Texas_hold%27em",
|
|
|
|
|
autolink "http://en.wikipedia.org/wiki/Texas_hold%27em")
|
|
|
|
|
, ("https://www.google.com/doku.php?id=gps:resource:scs:start",
|
|
|
|
|
autolink "https://www.google.com/doku.php?id=gps:resource:scs:start")
|
|
|
|
|
, ("http://www.rubyonrails.com",
|
|
|
|
|
autolink "http://www.rubyonrails.com")
|
|
|
|
|
, ("http://manuals.ruby-on-rails.com/read/chapter.need_a-period/103#page281",
|
|
|
|
|
autolink "http://manuals.ruby-on-rails.com/read/chapter.need_a-period/103#page281")
|
|
|
|
|
, ("http://foo.example.com/controller/action?parm=value&p2=v2#anchor123",
|
|
|
|
|
autolink "http://foo.example.com/controller/action?parm=value&p2=v2#anchor123")
|
|
|
|
|
, ("http://foo.example.com:3000/controller/action",
|
|
|
|
|
autolink "http://foo.example.com:3000/controller/action")
|
|
|
|
|
, ("http://foo.example.com:3000/controller/action+pack",
|
|
|
|
|
autolink "http://foo.example.com:3000/controller/action+pack")
|
|
|
|
|
, ("http://business.timesonline.co.uk/article/0,,9065-2473189,00.html",
|
|
|
|
|
autolink "http://business.timesonline.co.uk/article/0,,9065-2473189,00.html")
|
|
|
|
|
, ("http://www.mail-archive.com/ruby-talk@ruby-lang.org/",
|
|
|
|
|
autolink "http://www.mail-archive.com/ruby-talk@ruby-lang.org/")
|
2015-07-14 10:20:09 -07:00
|
|
|
|
, ("https://example.org/?anchor=lala-",
|
|
|
|
|
autolink "https://example.org/?anchor=lala-")
|
|
|
|
|
, ("https://example.org/?anchor=-lala",
|
|
|
|
|
autolink "https://example.org/?anchor=-lala")
|
2013-01-15 12:28:31 -08:00
|
|
|
|
]
|
|
|
|
|
|
2011-02-04 18:33:08 -08:00
|
|
|
|
{-
|
|
|
|
|
p_markdown_round_trip :: Block -> Bool
|
|
|
|
|
p_markdown_round_trip b = matches d' d''
|
|
|
|
|
where d' = normalize $ Pandoc (Meta [] [] []) [b]
|
|
|
|
|
d'' = normalize
|
2012-07-25 22:35:41 -07:00
|
|
|
|
$ readMarkdown def { readerSmart = True }
|
2012-07-26 22:59:56 -07:00
|
|
|
|
$ writeMarkdown def d'
|
2011-02-04 18:33:08 -08:00
|
|
|
|
matches (Pandoc _ [Plain []]) (Pandoc _ []) = True
|
|
|
|
|
matches (Pandoc _ [Para []]) (Pandoc _ []) = True
|
|
|
|
|
matches (Pandoc _ [Plain xs]) (Pandoc _ [Para xs']) = xs == xs'
|
|
|
|
|
matches x y = x == y
|
|
|
|
|
-}
|
|
|
|
|
|
2017-03-14 17:05:36 +01:00
|
|
|
|
tests :: [TestTree]
|
2011-01-26 22:09:09 -08:00
|
|
|
|
tests = [ testGroup "inline code"
|
|
|
|
|
[ "with attribute" =:
|
|
|
|
|
"`document.write(\"Hello\");`{.javascript}"
|
|
|
|
|
=?> para
|
|
|
|
|
(codeWith ("",["javascript"],[]) "document.write(\"Hello\");")
|
|
|
|
|
, "with attribute space" =:
|
|
|
|
|
"`*` {.haskell .special x=\"7\"}"
|
2016-12-24 15:34:07 +01:00
|
|
|
|
=?> para (code "*" <> space <> str "{.haskell" <> space <>
|
|
|
|
|
str ".special" <> space <> str "x=\"7\"}")
|
2011-01-26 22:09:09 -08:00
|
|
|
|
]
|
2013-11-22 19:41:08 -08:00
|
|
|
|
, testGroup "emph and strong"
|
|
|
|
|
[ "two strongs in emph" =:
|
|
|
|
|
"***a**b **c**d*" =?> para (emph (strong (str "a") <> str "b" <> space
|
|
|
|
|
<> strong (str "c") <> str "d"))
|
2014-07-07 22:21:04 -06:00
|
|
|
|
, "emph and strong emph alternating" =:
|
|
|
|
|
"*xxx* ***xxx*** xxx\n*xxx* ***xxx*** xxx"
|
|
|
|
|
=?> para (emph "xxx" <> space <> strong (emph "xxx") <>
|
2015-12-11 15:58:11 -08:00
|
|
|
|
space <> "xxx" <> softbreak <>
|
2014-07-07 22:21:04 -06:00
|
|
|
|
emph "xxx" <> space <> strong (emph "xxx") <>
|
|
|
|
|
space <> "xxx")
|
|
|
|
|
, "emph with spaced strong" =:
|
|
|
|
|
"*x **xx** x*"
|
|
|
|
|
=?> para (emph ("x" <> space <> strong "xx" <> space <> "x"))
|
2014-07-10 14:23:20 -07:00
|
|
|
|
, "intraword underscore with opening underscore (#1121)" =:
|
|
|
|
|
"_foot_ball_" =?> para (emph (text "foot_ball"))
|
2013-11-22 19:41:08 -08:00
|
|
|
|
]
|
2012-09-22 13:00:59 -07:00
|
|
|
|
, testGroup "raw LaTeX"
|
|
|
|
|
[ "in URL" =:
|
|
|
|
|
"\\begin\n" =?> para (text "\\begin")
|
|
|
|
|
]
|
2014-07-07 22:27:28 -06:00
|
|
|
|
, testGroup "raw HTML"
|
|
|
|
|
[ "nesting (issue #1330)" =:
|
|
|
|
|
"<del>test</del>" =?>
|
|
|
|
|
rawBlock "html" "<del>" <> plain (str "test") <>
|
|
|
|
|
rawBlock "html" "</del>"
|
2015-04-17 22:55:39 -07:00
|
|
|
|
, "invalid tag (issue #1820" =:
|
|
|
|
|
"</ div></.div>" =?>
|
|
|
|
|
para (text "</ div></.div>")
|
2015-07-21 22:44:18 -07:00
|
|
|
|
, "technically invalid comment" =:
|
|
|
|
|
"<!-- pandoc --help -->" =?>
|
|
|
|
|
rawBlock "html" "<!-- pandoc --help -->"
|
2015-10-22 21:18:06 -07:00
|
|
|
|
, test markdownGH "issue 2469" $
|
|
|
|
|
"<\n\na>" =?>
|
|
|
|
|
para (text "<") <> para (text "a>")
|
2014-07-07 22:27:28 -06:00
|
|
|
|
]
|
2016-10-23 23:12:36 +02:00
|
|
|
|
, testGroup "raw email addresses"
|
|
|
|
|
[ test markdownGH "issue 2940" $
|
|
|
|
|
"**@user**" =?>
|
|
|
|
|
para (strong (text "@user"))
|
|
|
|
|
]
|
2015-11-13 12:06:39 -08:00
|
|
|
|
, testGroup "emoji"
|
|
|
|
|
[ test markdownGH "emoji symbols" $
|
2018-07-16 00:14:40 +02:00
|
|
|
|
":smile: and :+1:" =?> para (spanWith ("", ["emoji"], [("data-emoji", "smile")]) "😄" <>
|
|
|
|
|
space <> str "and" <> space <>
|
|
|
|
|
spanWith ("", ["emoji"], [("data-emoji", "+1")]) "👍")
|
2015-11-13 12:06:39 -08:00
|
|
|
|
]
|
2012-09-22 13:59:30 -07:00
|
|
|
|
, "unbalanced brackets" =:
|
2018-01-14 12:23:16 -08:00
|
|
|
|
"[[[[[[[[[[[[hi" =?> para (text "[[[[[[[[[[[[hi")
|
2011-12-01 19:47:25 -08:00
|
|
|
|
, testGroup "backslash escapes"
|
|
|
|
|
[ "in URL" =:
|
|
|
|
|
"[hi](/there\\))"
|
2011-12-05 21:13:06 -08:00
|
|
|
|
=?> para (link "/there)" "" "hi")
|
2011-12-04 22:37:28 -08:00
|
|
|
|
, "in title" =:
|
2011-12-05 21:16:30 -08:00
|
|
|
|
"[hi](/there \"a\\\"a\")"
|
2011-12-05 21:13:06 -08:00
|
|
|
|
=?> para (link "/there" "a\"a" "hi")
|
2011-12-05 19:07:17 -08:00
|
|
|
|
, "in reference link title" =:
|
|
|
|
|
"[hi]\n\n[hi]: /there (a\\)a)"
|
2011-12-05 21:13:06 -08:00
|
|
|
|
=?> para (link "/there" "a)a" "hi")
|
2011-12-05 19:07:17 -08:00
|
|
|
|
, "in reference link URL" =:
|
2011-12-05 21:16:30 -08:00
|
|
|
|
"[hi]\n\n[hi]: /there\\.0"
|
|
|
|
|
=?> para (link "/there.0" "" "hi")
|
2011-12-01 19:47:25 -08:00
|
|
|
|
]
|
2013-01-15 12:28:31 -08:00
|
|
|
|
, testGroup "bare URIs"
|
2015-07-14 13:16:20 -07:00
|
|
|
|
(map testBareLink bareLinkTests)
|
2014-12-14 12:20:33 -08:00
|
|
|
|
, testGroup "autolinks"
|
|
|
|
|
[ "with unicode dash following" =:
|
|
|
|
|
"<http://foo.bar>\8212" =?> para (autolink "http://foo.bar" <>
|
|
|
|
|
str "\8212")
|
2015-07-10 10:28:39 -07:00
|
|
|
|
, "a partial URL (#2277)" =:
|
|
|
|
|
"<www.boe.es/buscar/act.php?id=BOE-A-1996-8930#a66>" =?>
|
|
|
|
|
para (text "<www.boe.es/buscar/act.php?id=BOE-A-1996-8930#a66>")
|
2016-10-26 12:18:58 +02:00
|
|
|
|
, "with some attributes" =:
|
|
|
|
|
"<http://foo.bar>{#i .j .z k=v}" =?>
|
|
|
|
|
para (autolinkWith ("i", ["j", "z"], [("k", "v")]) "http://foo.bar")
|
|
|
|
|
, "with some attributes and spaces" =:
|
|
|
|
|
"<http://foo.bar> {#i .j .z k=v}" =?>
|
|
|
|
|
para (autolink "http://foo.bar" <> space <> text "{#i .j .z k=v}")
|
2014-12-14 12:20:33 -08:00
|
|
|
|
]
|
2015-07-14 13:16:20 -07:00
|
|
|
|
, testGroup "links"
|
|
|
|
|
[ "no autolink inside link" =:
|
|
|
|
|
"[<https://example.org>](url)" =?>
|
|
|
|
|
para (link "url" "" (text "<https://example.org>"))
|
|
|
|
|
, "no inline link inside link" =:
|
|
|
|
|
"[[a](url2)](url)" =?>
|
|
|
|
|
para (link "url" "" (text "[a](url2)"))
|
|
|
|
|
, "no bare URI inside link" =:
|
|
|
|
|
"[https://example.org(](url)" =?>
|
|
|
|
|
para (link "url" "" (text "https://example.org("))
|
|
|
|
|
]
|
2013-06-19 09:27:11 -07:00
|
|
|
|
, testGroup "Headers"
|
|
|
|
|
[ "blank line before header" =:
|
|
|
|
|
"\n# Header\n"
|
|
|
|
|
=?> headerWith ("header",[],[]) 1 "Header"
|
2015-04-18 18:59:29 -07:00
|
|
|
|
, "bracketed text (#2062)" =:
|
|
|
|
|
"# [hi]\n"
|
|
|
|
|
=?> headerWith ("hi",[],[]) 1 "[hi]"
|
2015-07-23 02:31:24 -04:00
|
|
|
|
, "ATX header without trailing #s" =:
|
|
|
|
|
"# Foo bar\n\n" =?>
|
|
|
|
|
headerWith ("foo-bar",[],[]) 1 "Foo bar"
|
|
|
|
|
, "ATX header without trailing #s" =:
|
|
|
|
|
"# Foo bar with # #" =?>
|
|
|
|
|
headerWith ("foo-bar-with",[],[]) 1 "Foo bar with #"
|
|
|
|
|
, "setext header" =:
|
|
|
|
|
"Foo bar\n=\n\n Foo bar 2 \n=" =?>
|
|
|
|
|
headerWith ("foo-bar",[],[]) 1 "Foo bar"
|
|
|
|
|
<> headerWith ("foo-bar-2",[],[]) 1 "Foo bar 2"
|
2013-06-19 09:27:11 -07:00
|
|
|
|
]
|
2015-07-23 02:31:03 -04:00
|
|
|
|
, testGroup "Implicit header references"
|
|
|
|
|
[ "ATX header without trailing #s" =:
|
|
|
|
|
"# Header\n[header]\n\n[header ]\n\n[ header]" =?>
|
|
|
|
|
headerWith ("header",[],[]) 1 "Header"
|
|
|
|
|
<> para (link "#header" "" (text "header"))
|
2015-07-23 15:35:18 -07:00
|
|
|
|
<> para (link "#header" "" (text "header"))
|
|
|
|
|
<> para (link "#header" "" (text "header"))
|
2015-07-23 02:31:03 -04:00
|
|
|
|
, "ATX header with trailing #s" =:
|
|
|
|
|
"# Foo bar #\n[foo bar]\n\n[foo bar ]\n\n[ foo bar]" =?>
|
|
|
|
|
headerWith ("foo-bar",[],[]) 1 "Foo bar"
|
|
|
|
|
<> para (link "#foo-bar" "" (text "foo bar"))
|
2015-07-23 15:35:18 -07:00
|
|
|
|
<> para (link "#foo-bar" "" (text "foo bar"))
|
|
|
|
|
<> para (link "#foo-bar" "" (text "foo bar"))
|
2015-07-23 02:31:03 -04:00
|
|
|
|
, "setext header" =:
|
|
|
|
|
" Header \n=\n\n[header]\n\n[header ]\n\n[ header]" =?>
|
|
|
|
|
headerWith ("header",[],[]) 1 "Header"
|
|
|
|
|
<> para (link "#header" "" (text "header"))
|
2015-07-23 15:35:18 -07:00
|
|
|
|
<> para (link "#header" "" (text "header"))
|
|
|
|
|
<> para (link "#header" "" (text "header"))
|
2015-07-23 02:31:03 -04:00
|
|
|
|
]
|
2011-07-25 23:49:45 -07:00
|
|
|
|
, testGroup "smart punctuation"
|
|
|
|
|
[ test markdownSmart "quote before ellipses"
|
|
|
|
|
("'...hi'"
|
2013-12-19 20:43:25 -05:00
|
|
|
|
=?> para (singleQuoted "…hi"))
|
2011-12-26 23:03:20 -08:00
|
|
|
|
, test markdownSmart "apostrophe before emph"
|
|
|
|
|
("D'oh! A l'*aide*!"
|
|
|
|
|
=?> para ("D’oh! A l’" <> emph "aide" <> "!"))
|
|
|
|
|
, test markdownSmart "apostrophe in French"
|
|
|
|
|
("À l'arrivée de la guerre, le thème de l'«impossibilité du socialisme»"
|
2013-12-19 20:43:25 -05:00
|
|
|
|
=?> para "À l’arrivée de la guerre, le thème de l’«impossibilité du socialisme»")
|
2015-04-18 00:53:20 +03:00
|
|
|
|
, test markdownSmart "apostrophe after math" $ -- issue #1909
|
|
|
|
|
"The value of the $x$'s and the systems' condition." =?>
|
|
|
|
|
para (text "The value of the " <> math "x" <> text "\8217s and the systems\8217 condition.")
|
2018-04-17 11:55:37 +03:00
|
|
|
|
, test markdownSmart "unclosed double quote"
|
|
|
|
|
("**this should \"be bold**"
|
|
|
|
|
=?> para (strong "this should \"be bold"))
|
2011-07-25 23:49:45 -07:00
|
|
|
|
]
|
2011-01-31 20:05:11 -08:00
|
|
|
|
, testGroup "footnotes"
|
2011-02-01 07:37:22 -08:00
|
|
|
|
[ "indent followed by newline and flush-left text" =:
|
2011-01-31 20:42:49 -08:00
|
|
|
|
"[^1]\n\n[^1]: my note\n\n \nnot in note\n"
|
2011-12-13 14:29:07 -08:00
|
|
|
|
=?> para (note (para "my note")) <> para "not in note"
|
2011-02-01 07:37:22 -08:00
|
|
|
|
, "indent followed by newline and indented text" =:
|
2011-02-01 22:35:27 -08:00
|
|
|
|
"[^1]\n\n[^1]: my note\n \n in note\n"
|
2011-12-13 14:29:07 -08:00
|
|
|
|
=?> para (note (para "my note" <> para "in note"))
|
2011-04-20 11:42:27 -07:00
|
|
|
|
, "recursive note" =:
|
|
|
|
|
"[^1]\n\n[^1]: See [^1]\n"
|
|
|
|
|
=?> para (note (para "See [^1]"))
|
2011-01-31 20:05:11 -08:00
|
|
|
|
]
|
2011-03-02 11:18:38 -08:00
|
|
|
|
, testGroup "lhs"
|
2017-01-14 13:06:27 +01:00
|
|
|
|
[ test (purely $ readMarkdown def{ readerExtensions = enableExtension
|
2017-01-15 20:42:00 +01:00
|
|
|
|
Ext_literate_haskell pandocExtensions })
|
2011-03-02 11:18:38 -08:00
|
|
|
|
"inverse bird tracks and html" $
|
|
|
|
|
"> a\n\n< b\n\n<div>\n"
|
2019-01-08 11:36:33 -08:00
|
|
|
|
=?> codeBlockWith ("",["haskell","literate"],[]) "a"
|
2011-12-13 14:29:07 -08:00
|
|
|
|
<>
|
2019-01-08 11:36:33 -08:00
|
|
|
|
codeBlockWith ("",["haskell"],[]) "b"
|
2011-12-13 14:29:07 -08:00
|
|
|
|
<>
|
2011-03-02 11:18:38 -08:00
|
|
|
|
rawBlock "html" "<div>\n\n"
|
|
|
|
|
]
|
2011-02-04 18:33:08 -08:00
|
|
|
|
-- the round-trip properties frequently fail
|
|
|
|
|
-- , testGroup "round trip"
|
|
|
|
|
-- [ property "p_markdown_round_trip" p_markdown_round_trip
|
|
|
|
|
-- ]
|
2014-07-20 16:33:59 -07:00
|
|
|
|
, testGroup "definition lists"
|
|
|
|
|
[ "no blank space" =:
|
|
|
|
|
"foo1\n : bar\n\nfoo2\n : bar2\n : bar3\n" =?>
|
|
|
|
|
definitionList [ (text "foo1", [plain (text "bar")])
|
|
|
|
|
, (text "foo2", [plain (text "bar2"),
|
|
|
|
|
plain (text "bar3")])
|
|
|
|
|
]
|
|
|
|
|
, "blank space before first def" =:
|
|
|
|
|
"foo1\n\n : bar\n\nfoo2\n\n : bar2\n : bar3\n" =?>
|
|
|
|
|
definitionList [ (text "foo1", [para (text "bar")])
|
|
|
|
|
, (text "foo2", [para (text "bar2"),
|
|
|
|
|
plain (text "bar3")])
|
|
|
|
|
]
|
|
|
|
|
, "blank space before second def" =:
|
|
|
|
|
"foo1\n : bar\n\nfoo2\n : bar2\n\n : bar3\n" =?>
|
|
|
|
|
definitionList [ (text "foo1", [plain (text "bar")])
|
|
|
|
|
, (text "foo2", [plain (text "bar2"),
|
|
|
|
|
para (text "bar3")])
|
|
|
|
|
]
|
|
|
|
|
, "laziness" =:
|
|
|
|
|
"foo1\n : bar\nbaz\n : bar2\n" =?>
|
2015-12-11 15:58:11 -08:00
|
|
|
|
definitionList [ (text "foo1", [plain (text "bar" <>
|
|
|
|
|
softbreak <> text "baz"),
|
2014-07-20 16:33:59 -07:00
|
|
|
|
plain (text "bar2")])
|
|
|
|
|
]
|
|
|
|
|
, "no blank space before first of two paragraphs" =:
|
|
|
|
|
"foo1\n : bar\n\n baz\n" =?>
|
|
|
|
|
definitionList [ (text "foo1", [para (text "bar") <>
|
|
|
|
|
para (text "baz")])
|
|
|
|
|
]
|
2015-04-18 10:13:32 -07:00
|
|
|
|
, "first line not indented" =:
|
|
|
|
|
"foo\n: bar\n" =?>
|
|
|
|
|
definitionList [ (text "foo", [plain (text "bar")]) ]
|
2015-04-26 11:20:53 -07:00
|
|
|
|
, "list in definition" =:
|
|
|
|
|
"foo\n: - bar\n" =?>
|
|
|
|
|
definitionList [ (text "foo", [bulletList [plain (text "bar")]]) ]
|
2015-05-03 15:06:40 -07:00
|
|
|
|
, "in div" =:
|
|
|
|
|
"<div>foo\n: - bar\n</div>" =?>
|
|
|
|
|
divWith nullAttr (definitionList
|
|
|
|
|
[ (text "foo", [bulletList [plain (text "bar")]]) ])
|
2014-07-20 16:33:59 -07:00
|
|
|
|
]
|
|
|
|
|
, testGroup "+compact_definition_lists"
|
|
|
|
|
[ test markdownCDL "basic compact list" $
|
|
|
|
|
"foo1\n: bar\n baz\nfoo2\n: bar2\n" =?>
|
2015-12-11 15:58:11 -08:00
|
|
|
|
definitionList [ (text "foo1", [plain (text "bar" <> softbreak <>
|
|
|
|
|
text "baz")])
|
2014-07-20 16:33:59 -07:00
|
|
|
|
, (text "foo2", [plain (text "bar2")])
|
|
|
|
|
]
|
|
|
|
|
]
|
2014-05-04 08:19:48 -07:00
|
|
|
|
, testGroup "lists"
|
|
|
|
|
[ "issue #1154" =:
|
|
|
|
|
" - <div>\n first div breaks\n </div>\n\n <button>if this button exists</button>\n\n <div>\n with this div too.\n </div>\n"
|
2014-08-31 12:55:47 -07:00
|
|
|
|
=?> bulletList [divWith nullAttr (para $ text "first div breaks") <>
|
2014-05-04 08:19:48 -07:00
|
|
|
|
rawBlock "html" "<button>" <>
|
|
|
|
|
plain (text "if this button exists") <>
|
2014-07-07 15:47:51 -06:00
|
|
|
|
rawBlock "html" "</button>" <>
|
2014-08-31 12:55:47 -07:00
|
|
|
|
divWith nullAttr (para $ text "with this div too.")]
|
2014-09-26 13:32:08 +04:00
|
|
|
|
, test markdownGH "issue #1636" $
|
2017-06-10 18:26:44 +02:00
|
|
|
|
T.unlines [ "* a"
|
|
|
|
|
, "* b"
|
|
|
|
|
, "* c"
|
|
|
|
|
, " * d" ]
|
2014-09-26 13:32:08 +04:00
|
|
|
|
=?>
|
|
|
|
|
bulletList [ plain "a"
|
|
|
|
|
, plain "b"
|
|
|
|
|
, plain "c" <> bulletList [plain "d"] ]
|
2014-05-04 08:19:48 -07:00
|
|
|
|
]
|
2016-01-08 17:33:37 -08:00
|
|
|
|
, testGroup "entities"
|
|
|
|
|
[ "character references" =:
|
|
|
|
|
"⟨ ö" =?> para (text "\10216 ö")
|
|
|
|
|
, "numeric" =:
|
|
|
|
|
",DD" =?> para (text ",DD")
|
|
|
|
|
, "in link title" =:
|
|
|
|
|
"[link](/url \"title ⟨ ö ,\")" =?>
|
|
|
|
|
para (link "/url" "title \10216 ö ," (text "link"))
|
|
|
|
|
]
|
2015-02-18 13:54:25 +00:00
|
|
|
|
, testGroup "citations"
|
|
|
|
|
[ "simple" =:
|
|
|
|
|
"@item1" =?> para (cite [
|
|
|
|
|
Citation{ citationId = "item1"
|
|
|
|
|
, citationPrefix = []
|
|
|
|
|
, citationSuffix = []
|
|
|
|
|
, citationMode = AuthorInText
|
|
|
|
|
, citationNoteNum = 0
|
|
|
|
|
, citationHash = 0
|
|
|
|
|
}
|
|
|
|
|
] "@item1")
|
|
|
|
|
, "key starts with digit" =:
|
|
|
|
|
"@1657:huyghens" =?> para (cite [
|
|
|
|
|
Citation{ citationId = "1657:huyghens"
|
|
|
|
|
, citationPrefix = []
|
|
|
|
|
, citationSuffix = []
|
|
|
|
|
, citationMode = AuthorInText
|
|
|
|
|
, citationNoteNum = 0
|
|
|
|
|
, citationHash = 0
|
|
|
|
|
}
|
|
|
|
|
] "@1657:huyghens")
|
|
|
|
|
]
|
2015-04-17 17:30:55 +03:00
|
|
|
|
, let citation = cite [Citation "cita" [] [] AuthorInText 0 0] (str "@cita")
|
|
|
|
|
in testGroup "footnote/link following citation" -- issue #2083
|
|
|
|
|
[ "footnote" =:
|
2017-06-10 18:26:44 +02:00
|
|
|
|
T.unlines [ "@cita[^note]"
|
|
|
|
|
, ""
|
|
|
|
|
, "[^note]: note" ] =?>
|
2015-04-17 17:30:55 +03:00
|
|
|
|
para (
|
|
|
|
|
citation <> note (para $ str "note")
|
|
|
|
|
)
|
|
|
|
|
, "normal link" =:
|
|
|
|
|
"@cita [link](http://www.com)" =?>
|
|
|
|
|
para (
|
|
|
|
|
citation <> space <> link "http://www.com" "" (str "link")
|
|
|
|
|
)
|
|
|
|
|
, "reference link" =:
|
2017-06-10 18:26:44 +02:00
|
|
|
|
T.unlines [ "@cita [link][link]"
|
|
|
|
|
, ""
|
|
|
|
|
, "[link]: http://www.com" ] =?>
|
2015-04-17 17:30:55 +03:00
|
|
|
|
para (
|
|
|
|
|
citation <> space <> link "http://www.com" "" (str "link")
|
|
|
|
|
)
|
|
|
|
|
, "short reference link" =:
|
2017-06-10 18:26:44 +02:00
|
|
|
|
T.unlines [ "@cita [link]"
|
|
|
|
|
, ""
|
|
|
|
|
, "[link]: http://www.com" ] =?>
|
2015-04-17 17:30:55 +03:00
|
|
|
|
para (
|
|
|
|
|
citation <> space <> link "http://www.com" "" (str "link")
|
|
|
|
|
)
|
|
|
|
|
, "implicit header link" =:
|
2017-06-10 18:26:44 +02:00
|
|
|
|
T.unlines [ "# Header"
|
|
|
|
|
, "@cita [Header]" ] =?>
|
2015-04-17 17:30:55 +03:00
|
|
|
|
headerWith ("header",[],[]) 1 (str "Header") <> para (
|
|
|
|
|
citation <> space <> link "#header" "" (str "Header")
|
|
|
|
|
)
|
|
|
|
|
, "regular citation" =:
|
|
|
|
|
"@cita [foo]" =?>
|
|
|
|
|
para (
|
|
|
|
|
cite [Citation "cita" [] [Str "foo"] AuthorInText 0 0]
|
|
|
|
|
(str "@cita" <> space <> str "[foo]")
|
|
|
|
|
)
|
|
|
|
|
]
|
2011-01-26 22:09:09 -08:00
|
|
|
|
]
|