2018-03-18 10:46:28 -07:00
|
|
|
{-# LANGUAGE NoImplicitPrelude #-}
|
2017-10-27 20:28:29 -07:00
|
|
|
{-# LANGUAGE OverloadedStrings #-}
|
2017-03-20 15:17:03 +01:00
|
|
|
module Tests.Lua ( tests ) where
|
|
|
|
|
2018-03-18 10:46:28 -07:00
|
|
|
import Prelude
|
2017-04-06 21:00:38 +02:00
|
|
|
import Control.Monad (when)
|
2018-01-07 14:06:34 +01:00
|
|
|
import Data.Version (Version (versionBranch))
|
2017-03-20 15:17:03 +01:00
|
|
|
import System.FilePath ((</>))
|
2017-05-07 11:45:06 +02:00
|
|
|
import Test.Tasty (TestTree, localOption)
|
2018-10-26 23:21:54 +02:00
|
|
|
import Test.Tasty.HUnit (Assertion, assertEqual, testCase)
|
2017-10-27 20:28:29 -07:00
|
|
|
import Test.Tasty.QuickCheck (QuickCheckTests (..), ioProperty, testProperty)
|
2017-04-06 21:00:38 +02:00
|
|
|
import Text.Pandoc.Arbitrary ()
|
2017-11-20 18:37:40 +01:00
|
|
|
import Text.Pandoc.Builder (bulletList, divWith, doc, doubleQuoted, emph,
|
|
|
|
header, linebreak, para, plain, rawBlock,
|
2018-04-29 16:20:38 +03:00
|
|
|
singleQuoted, space, str, strong,
|
|
|
|
math, displayMath)
|
2017-12-13 21:15:41 +01:00
|
|
|
import Text.Pandoc.Class (runIOorExplode, setUserDataDir)
|
2018-01-13 23:05:42 +01:00
|
|
|
import Text.Pandoc.Definition (Block (BlockQuote, Div, Para), Inline (Emph, Str),
|
|
|
|
Attr, Meta, Pandoc, pandocTypesVersion)
|
2018-10-26 23:21:54 +02:00
|
|
|
import Text.Pandoc.Filter (Filter (LuaFilter), applyFilters)
|
2018-10-26 23:29:50 +02:00
|
|
|
import Text.Pandoc.Lua (runLua)
|
2018-01-12 08:56:33 +01:00
|
|
|
import Text.Pandoc.Options (def)
|
2018-01-07 13:43:03 +01:00
|
|
|
import Text.Pandoc.Shared (pandocVersion)
|
2017-03-20 15:17:03 +01:00
|
|
|
|
2017-12-02 23:07:29 +01:00
|
|
|
import qualified Foreign.Lua as Lua
|
2017-04-06 21:00:38 +02:00
|
|
|
|
2017-03-20 15:17:03 +01:00
|
|
|
tests :: [TestTree]
|
2017-05-07 11:45:06 +02:00
|
|
|
tests = map (localOption (QuickCheckTests 20))
|
2017-04-14 23:24:52 +02:00
|
|
|
[ testProperty "inline elements can be round-tripped through the lua stack" $
|
|
|
|
\x -> ioProperty (roundtripEqual (x::Inline))
|
|
|
|
|
|
|
|
, testProperty "block elements can be round-tripped through the lua stack" $
|
|
|
|
\x -> ioProperty (roundtripEqual (x::Block))
|
|
|
|
|
|
|
|
, testProperty "meta blocks can be round-tripped through the lua stack" $
|
|
|
|
\x -> ioProperty (roundtripEqual (x::Meta))
|
|
|
|
|
|
|
|
, testProperty "documents can be round-tripped through the lua stack" $
|
|
|
|
\x -> ioProperty (roundtripEqual (x::Pandoc))
|
|
|
|
|
|
|
|
, testCase "macro expansion via filter" $
|
2017-03-20 15:17:03 +01:00
|
|
|
assertFilterConversion "a '{{helloworld}}' string is expanded"
|
|
|
|
"strmacro.lua"
|
|
|
|
(doc . para $ str "{{helloworld}}")
|
|
|
|
(doc . para . emph $ str "Hello, World")
|
|
|
|
|
|
|
|
, testCase "convert all plains to paras" $
|
|
|
|
assertFilterConversion "plains become para"
|
|
|
|
"plain-to-para.lua"
|
|
|
|
(doc $ bulletList [plain (str "alfa"), plain (str "bravo")])
|
|
|
|
(doc $ bulletList [para (str "alfa"), para (str "bravo")])
|
|
|
|
|
2018-04-29 16:20:38 +03:00
|
|
|
, testCase "convert display math to inline math" $
|
|
|
|
assertFilterConversion "display math becomes inline math"
|
|
|
|
"math.lua"
|
|
|
|
(doc $ para (displayMath "5+5"))
|
|
|
|
(doc $ para (math "5+5"))
|
|
|
|
|
2017-03-20 15:17:03 +01:00
|
|
|
, testCase "make hello world document" $
|
|
|
|
assertFilterConversion "Document contains 'Hello, World!'"
|
|
|
|
"hello-world-doc.lua"
|
|
|
|
(doc . para $ str "Hey!" <> linebreak <> str "What's up?")
|
|
|
|
(doc . para $ str "Hello," <> space <> str "World!")
|
2017-04-02 17:21:22 +02:00
|
|
|
|
2017-04-30 16:14:33 +02:00
|
|
|
, testCase "implicit doc filter" $
|
|
|
|
assertFilterConversion "Document contains 'Hello, World!'"
|
|
|
|
"implicit-doc-filter.lua"
|
|
|
|
(doc . plain $ linebreak)
|
|
|
|
(doc . para $ str "Hello," <> space <> str "World!")
|
|
|
|
|
2017-04-02 17:21:22 +02:00
|
|
|
, testCase "parse raw markdown blocks" $
|
|
|
|
assertFilterConversion "raw markdown block is converted"
|
|
|
|
"markdown-reader.lua"
|
|
|
|
(doc $ rawBlock "markdown" "*charly* **delta**")
|
|
|
|
(doc . para $ emph "charly" <> space <> strong "delta")
|
2017-04-06 21:00:38 +02:00
|
|
|
|
2017-04-14 23:24:52 +02:00
|
|
|
, testCase "allow shorthand functions for quote types" $
|
|
|
|
assertFilterConversion "single quoted becomes double quoted string"
|
|
|
|
"single-to-double-quoted.lua"
|
|
|
|
(doc . para . singleQuoted $ str "simple")
|
|
|
|
(doc . para . doubleQuoted $ str "simple")
|
2017-08-22 22:02:30 +02:00
|
|
|
|
|
|
|
, testCase "Count inlines via metatable catch-all" $
|
|
|
|
assertFilterConversion "filtering with metatable catch-all failed"
|
|
|
|
"metatable-catch-all.lua"
|
|
|
|
(doc . para $ "four words, three spaces")
|
|
|
|
(doc . para $ str "7")
|
2017-08-22 23:12:39 +02:00
|
|
|
|
|
|
|
, testCase "Count blocks via Block-specific catch-all" $
|
|
|
|
assertFilterConversion "filtering with Block catch-all failed"
|
|
|
|
"block-count.lua"
|
|
|
|
(doc $ para "one" <> para "two")
|
|
|
|
(doc $ para "2")
|
2017-11-18 22:24:06 +01:00
|
|
|
|
|
|
|
, testCase "Convert header upper case" $
|
|
|
|
assertFilterConversion "converting header to upper case failed"
|
|
|
|
"uppercase-header.lua"
|
|
|
|
(doc $ header 1 "les états-unis" <> para "text")
|
|
|
|
(doc $ header 1 "LES ÉTATS-UNIS" <> para "text")
|
2017-11-20 18:37:40 +01:00
|
|
|
|
|
|
|
, testCase "Attribute lists are convenient to use" $
|
|
|
|
let kv_before = [("one", "1"), ("two", "2"), ("three", "3")]
|
|
|
|
kv_after = [("one", "eins"), ("three", "3"), ("five", "5")]
|
|
|
|
in assertFilterConversion "Attr doesn't behave as expected"
|
|
|
|
"attr-test.lua"
|
|
|
|
(doc $ divWith ("", [], kv_before) (para "nil"))
|
|
|
|
(doc $ divWith ("", [], kv_after) (para "nil"))
|
2017-12-19 21:31:30 +01:00
|
|
|
|
|
|
|
, testCase "Test module pandoc.utils" $
|
|
|
|
assertFilterConversion "pandoc.utils doesn't work as expected."
|
|
|
|
"test-pandoc-utils.lua"
|
|
|
|
(doc $ para "doesn't matter")
|
2018-07-30 19:55:25 +02:00
|
|
|
(doc $ mconcat [ plain (str "blocks_to_inlines: OK")
|
|
|
|
, plain (str "hierarchicalize: OK")
|
2017-12-23 22:39:05 +01:00
|
|
|
, plain (str "normalize_date: OK")
|
2017-12-19 21:31:30 +01:00
|
|
|
, plain (str "pipe: OK")
|
|
|
|
, plain (str "failing pipe: OK")
|
|
|
|
, plain (str "read: OK")
|
|
|
|
, plain (str "failing read: OK")
|
2017-12-23 13:35:27 +01:00
|
|
|
, plain (str "sha1: OK")
|
2017-12-22 20:08:51 +01:00
|
|
|
, plain (str "stringify: OK")
|
2017-12-23 11:53:26 +01:00
|
|
|
, plain (str "to_roman_numeral: OK")
|
2017-12-19 21:31:30 +01:00
|
|
|
])
|
2018-01-07 13:43:03 +01:00
|
|
|
|
2018-02-24 21:59:50 +01:00
|
|
|
, testCase "Script filename is set" $
|
|
|
|
assertFilterConversion "unexpected script name"
|
|
|
|
"script-name.lua"
|
|
|
|
(doc $ para "ignored")
|
2018-02-25 08:06:52 +01:00
|
|
|
(doc $ para (str $ "lua" </> "script-name.lua"))
|
2018-02-24 21:59:50 +01:00
|
|
|
|
2018-10-26 23:29:50 +02:00
|
|
|
, testCase "Pandoc version is set" . runLua' $ do
|
2018-01-07 13:43:03 +01:00
|
|
|
Lua.getglobal' "table.concat"
|
|
|
|
Lua.getglobal "PANDOC_VERSION"
|
2018-07-02 18:51:51 +03:00
|
|
|
Lua.push ("." :: String) -- separator
|
2018-01-07 13:43:03 +01:00
|
|
|
Lua.call 2 1
|
|
|
|
Lua.liftIO . assertEqual "pandoc version is wrong" pandocVersion
|
|
|
|
=<< Lua.peek Lua.stackTop
|
2018-01-07 14:06:34 +01:00
|
|
|
|
2018-10-26 23:29:50 +02:00
|
|
|
, testCase "Pandoc types version is set" . runLua' $ do
|
2018-01-07 14:06:34 +01:00
|
|
|
let versionNums = versionBranch pandocTypesVersion
|
|
|
|
Lua.getglobal "PANDOC_API_VERSION"
|
2018-01-08 23:26:38 +01:00
|
|
|
Lua.liftIO . assertEqual "pandoc-types version is wrong" versionNums
|
2018-01-07 14:06:34 +01:00
|
|
|
=<< Lua.peek Lua.stackTop
|
2018-01-12 21:26:34 +01:00
|
|
|
|
2018-10-26 23:29:50 +02:00
|
|
|
, testCase "Allow singleton inline in constructors" . runLua' $ do
|
2018-01-13 22:29:16 +01:00
|
|
|
Lua.liftIO . assertEqual "Not the exptected Emph" (Emph [Str "test"])
|
|
|
|
=<< Lua.callFunc "pandoc.Emph" (Str "test")
|
|
|
|
Lua.liftIO . assertEqual "Unexpected element" (Para [Str "test"])
|
|
|
|
=<< Lua.callFunc "pandoc.Para" ("test" :: String)
|
|
|
|
Lua.liftIO . assertEqual "Unexptected element"
|
|
|
|
(BlockQuote [Para [Str "foo"]]) =<< (
|
|
|
|
do
|
|
|
|
Lua.getglobal' "pandoc.BlockQuote"
|
|
|
|
Lua.push (Para [Str "foo"])
|
|
|
|
_ <- Lua.call 1 1
|
|
|
|
Lua.peek Lua.stackTop
|
|
|
|
)
|
2018-01-13 18:52:17 +01:00
|
|
|
|
2018-10-26 23:29:50 +02:00
|
|
|
, testCase "Elements with Attr have `attr` accessor" . runLua' $ do
|
2018-01-13 23:05:42 +01:00
|
|
|
Lua.push (Div ("hi", ["moin"], [])
|
|
|
|
[Para [Str "ignored"]])
|
|
|
|
Lua.getfield Lua.stackTop "attr"
|
|
|
|
Lua.liftIO . assertEqual "no accessor" (("hi", ["moin"], []) :: Attr)
|
|
|
|
=<< Lua.peek Lua.stackTop
|
|
|
|
|
2018-10-26 23:29:50 +02:00
|
|
|
, testCase "informative error messages" . runLua' $ do
|
2018-01-12 21:26:34 +01:00
|
|
|
Lua.pushboolean True
|
2018-09-24 20:11:00 +02:00
|
|
|
err <- Lua.peekEither Lua.stackTop
|
|
|
|
case (err :: Either String Pandoc) of
|
2018-01-12 21:26:34 +01:00
|
|
|
Left msg -> do
|
|
|
|
let expectedMsg = "Could not get Pandoc value: "
|
2018-09-24 20:11:00 +02:00
|
|
|
<> "table expected, got boolean"
|
2018-01-12 21:26:34 +01:00
|
|
|
Lua.liftIO $ assertEqual "unexpected error message" expectedMsg msg
|
|
|
|
Right _ -> error "Getting a Pandoc element from a bool should fail."
|
2017-03-20 15:17:03 +01:00
|
|
|
]
|
|
|
|
|
|
|
|
assertFilterConversion :: String -> FilePath -> Pandoc -> Pandoc -> Assertion
|
2018-10-26 23:21:54 +02:00
|
|
|
assertFilterConversion msg filterPath docIn expectedDoc = do
|
|
|
|
actualDoc <- runIOorExplode $ do
|
2017-12-13 21:15:41 +01:00
|
|
|
setUserDataDir (Just "../data")
|
2018-10-26 23:21:54 +02:00
|
|
|
applyFilters def [LuaFilter ("lua" </> filterPath)] ["HTML"] docIn
|
|
|
|
assertEqual msg expectedDoc actualDoc
|
2017-04-06 21:00:38 +02:00
|
|
|
|
2018-09-24 20:11:00 +02:00
|
|
|
roundtripEqual :: (Eq a, Lua.Peekable a, Lua.Pushable a) => a -> IO Bool
|
2017-04-06 21:00:38 +02:00
|
|
|
roundtripEqual x = (x ==) <$> roundtripped
|
|
|
|
where
|
2018-09-24 20:11:00 +02:00
|
|
|
roundtripped :: (Lua.Peekable a, Lua.Pushable a) => IO a
|
2018-10-26 23:29:50 +02:00
|
|
|
roundtripped = runLua' $ do
|
2018-01-07 13:43:03 +01:00
|
|
|
oldSize <- Lua.gettop
|
|
|
|
Lua.push x
|
|
|
|
size <- Lua.gettop
|
|
|
|
when (size - oldSize /= 1) $
|
|
|
|
error ("not exactly one additional element on the stack: " ++ show size)
|
|
|
|
res <- Lua.peekEither (-1)
|
2017-08-13 12:37:10 +02:00
|
|
|
case res of
|
2018-01-08 23:26:38 +01:00
|
|
|
Left e -> error (show e)
|
2017-10-27 20:28:29 -07:00
|
|
|
Right y -> return y
|
2018-01-07 13:43:03 +01:00
|
|
|
|
2018-10-26 23:29:50 +02:00
|
|
|
runLua' :: Lua.Lua a -> IO a
|
|
|
|
runLua' op = runIOorExplode $ do
|
2018-01-07 13:43:03 +01:00
|
|
|
setUserDataDir (Just "../data")
|
2018-10-26 23:29:50 +02:00
|
|
|
res <- runLua op
|
2018-01-07 13:43:03 +01:00
|
|
|
case res of
|
|
|
|
Left e -> error (show e)
|
|
|
|
Right x -> return x
|