cln: Reduce duplication in test utilities.
This commit is contained in:
parent
8968733630
commit
1aff74f702
@ -20,7 +20,7 @@ module Hledger.Utils.Test (
|
|||||||
)
|
)
|
||||||
where
|
where
|
||||||
|
|
||||||
import Control.Monad.Except (ExceptT, runExceptT)
|
import Control.Monad.Except (ExceptT(..), liftEither, runExceptT, withExceptT, unless)
|
||||||
import Control.Monad.State.Strict (StateT, evalStateT, execStateT)
|
import Control.Monad.State.Strict (StateT, evalStateT, execStateT)
|
||||||
import Data.Default (Default(..))
|
import Data.Default (Default(..))
|
||||||
import Data.List (isInfixOf)
|
import Data.List (isInfixOf)
|
||||||
@ -54,16 +54,28 @@ assertRight :: (HasCallStack, Eq a, Show a) => Either a b -> Assertion
|
|||||||
assertRight (Right _) = return ()
|
assertRight (Right _) = return ()
|
||||||
assertRight (Left a) = assertFailure $ "expected Right, got (Left " ++ show a ++ ")"
|
assertRight (Left a) = assertFailure $ "expected Right, got (Left " ++ show a ++ ")"
|
||||||
|
|
||||||
|
-- | Run a parser on the given text and display a helpful error.
|
||||||
|
parseHelper :: (HasCallStack, Default st, Monad m) =>
|
||||||
|
StateT st (ParsecT CustomErr T.Text m) a -> T.Text -> ExceptT String m a
|
||||||
|
parseHelper parser input =
|
||||||
|
withExceptT (\e -> "\nparse error at " ++ customErrorBundlePretty e ++ "\n") . ExceptT
|
||||||
|
$ runParserT (evalStateT (parser <* eof) def) "" input
|
||||||
|
|
||||||
|
-- | Run a stateful parser in IO and process either a failure or success to
|
||||||
|
-- produce an 'Assertion'. Suitable for hledger's JournalParser parsers.
|
||||||
|
assertParseHelper :: (HasCallStack, Default st) =>
|
||||||
|
(String -> Assertion) -> (a -> Assertion)
|
||||||
|
-> StateT st (ParsecT CustomErr T.Text IO) a -> T.Text
|
||||||
|
-> Assertion
|
||||||
|
assertParseHelper onFailure onSuccess parser input =
|
||||||
|
either onFailure onSuccess =<< runExceptT (parseHelper parser input)
|
||||||
|
|
||||||
-- | Assert that this stateful parser runnable in IO successfully parses
|
-- | Assert that this stateful parser runnable in IO successfully parses
|
||||||
-- all of the given input text, showing the parse error if it fails.
|
-- all of the given input text, showing the parse error if it fails.
|
||||||
-- Suitable for hledger's JournalParser parsers.
|
-- Suitable for hledger's JournalParser parsers.
|
||||||
assertParse :: (HasCallStack, Eq a, Show a, Default st) =>
|
assertParse :: (HasCallStack, Default st) =>
|
||||||
StateT st (ParsecT CustomErr T.Text IO) a -> T.Text -> Assertion
|
StateT st (ParsecT CustomErr T.Text IO) a -> T.Text -> Assertion
|
||||||
assertParse parser input = do
|
assertParse = assertParseHelper assertFailure (const $ return ())
|
||||||
ep <- runParserT (evalStateT (parser <* eof) def) "" input
|
|
||||||
either (assertFailure.(++"\n").("\nparse error at "++).customErrorBundlePretty)
|
|
||||||
(const $ return ())
|
|
||||||
ep
|
|
||||||
|
|
||||||
-- | Assert a parser produces an expected value.
|
-- | Assert a parser produces an expected value.
|
||||||
assertParseEq :: (HasCallStack, Eq a, Show a, Default st) =>
|
assertParseEq :: (HasCallStack, Eq a, Show a, Default st) =>
|
||||||
@ -74,35 +86,23 @@ assertParseEq parser input = assertParseEqOn parser input id
|
|||||||
-- before comparing it.
|
-- before comparing it.
|
||||||
assertParseEqOn :: (HasCallStack, Eq b, Show b, Default st) =>
|
assertParseEqOn :: (HasCallStack, Eq b, Show b, Default st) =>
|
||||||
StateT st (ParsecT CustomErr T.Text IO) a -> T.Text -> (a -> b) -> b -> Assertion
|
StateT st (ParsecT CustomErr T.Text IO) a -> T.Text -> (a -> b) -> b -> Assertion
|
||||||
assertParseEqOn parser input f expected = do
|
assertParseEqOn parser input f expected =
|
||||||
ep <- runParserT (evalStateT (parser <* eof) def) "" input
|
assertParseHelper assertFailure (assertEqual "" expected . f) parser input
|
||||||
either (assertFailure . (++"\n") . ("\nparse error at "++) . customErrorBundlePretty)
|
|
||||||
(assertEqual "" expected . f)
|
|
||||||
ep
|
|
||||||
|
|
||||||
-- | Assert that this stateful parser runnable in IO fails to parse
|
-- | Assert that this stateful parser runnable in IO fails to parse
|
||||||
-- the given input text, with a parse error containing the given string.
|
-- the given input text, with a parse error containing the given string.
|
||||||
assertParseError :: (HasCallStack, Eq a, Show a, Default st) =>
|
assertParseError :: (HasCallStack, Eq a, Show a, Default st) =>
|
||||||
StateT st (ParsecT CustomErr T.Text IO) a -> String -> String -> Assertion
|
StateT st (ParsecT CustomErr T.Text IO) a -> T.Text -> String -> Assertion
|
||||||
assertParseError parser input errstr = do
|
assertParseError parser input errstr = assertParseHelper
|
||||||
ep <- runParserT (evalStateT parser def) "" (T.pack input)
|
(\e -> unless (errstr `isInfixOf` e) $ assertFailure $ "\nparse error is not as expected:" ++ e)
|
||||||
case ep of
|
(\v -> assertFailure $ "\nparse succeeded unexpectedly, producing:\n" ++ pshow v ++ "\n")
|
||||||
Right v -> assertFailure $ "\nparse succeeded unexpectedly, producing:\n" ++ pshow v ++ "\n"
|
parser input
|
||||||
Left e -> do
|
|
||||||
let e' = customErrorBundlePretty e
|
|
||||||
if errstr `isInfixOf` e'
|
|
||||||
then return ()
|
|
||||||
else assertFailure $ "\nparse error is not as expected:\n" ++ e' ++ "\n"
|
|
||||||
|
|
||||||
-- | Run a stateful parser in IO like assertParse, then assert that the
|
-- | Run a stateful parser in IO like assertParse, then assert that the
|
||||||
-- final state (the wrapped state, not megaparsec's internal state),
|
-- final state (the wrapped state, not megaparsec's internal state),
|
||||||
-- transformed by the given function, matches the given expected value.
|
-- transformed by the given function, matches the given expected value.
|
||||||
assertParseStateOn :: (HasCallStack, Eq b, Show b, Default st) =>
|
assertParseStateOn :: (HasCallStack, Eq b, Show b, Default st) =>
|
||||||
StateT st (ParsecT CustomErr T.Text IO) a
|
StateT st (ParsecT CustomErr T.Text IO) a -> T.Text -> (st -> b) -> b -> Assertion
|
||||||
-> T.Text
|
|
||||||
-> (st -> b)
|
|
||||||
-> b
|
|
||||||
-> Assertion
|
|
||||||
assertParseStateOn parser input f expected = do
|
assertParseStateOn parser input f expected = do
|
||||||
es <- runParserT (execStateT (parser <* eof) def) "" input
|
es <- runParserT (execStateT (parser <* eof) def) "" input
|
||||||
case es of
|
case es of
|
||||||
@ -110,70 +110,40 @@ assertParseStateOn parser input f expected = do
|
|||||||
Right s -> assertEqual "" expected $ f s
|
Right s -> assertEqual "" expected $ f s
|
||||||
|
|
||||||
-- | These "E" variants of the above are suitable for hledger's ErroringJournalParser parsers.
|
-- | These "E" variants of the above are suitable for hledger's ErroringJournalParser parsers.
|
||||||
|
parseHelperE :: (HasCallStack, Default st, Monad m) =>
|
||||||
|
StateT st (ParsecT CustomErr T.Text (ExceptT FinalParseError m)) a -> T.Text -> ExceptT String m a
|
||||||
|
parseHelperE parser input = do
|
||||||
|
withExceptT (\e -> "\nparse error at " ++ customErrorBundlePretty e ++ "\n") . liftEither
|
||||||
|
=<< withExceptT (\e -> "parse error at " ++ finalErrorBundlePretty (attachSource "" input e))
|
||||||
|
(runParserT (evalStateT (parser <* eof) def) "" input)
|
||||||
|
|
||||||
|
assertParseHelperE :: (HasCallStack, Default st) =>
|
||||||
|
(String -> Assertion) -> (a -> Assertion)
|
||||||
|
-> StateT st (ParsecT CustomErr T.Text (ExceptT FinalParseError IO)) a -> T.Text
|
||||||
|
-> Assertion
|
||||||
|
assertParseHelperE onFailure onSuccess parser input =
|
||||||
|
either onFailure onSuccess =<< runExceptT (parseHelperE parser input)
|
||||||
|
|
||||||
assertParseE
|
assertParseE
|
||||||
:: (HasCallStack, Eq a, Show a, Default st)
|
:: (HasCallStack, Eq a, Show a, Default st)
|
||||||
=> StateT st (ParsecT CustomErr T.Text (ExceptT FinalParseError IO)) a
|
=> StateT st (ParsecT CustomErr T.Text (ExceptT FinalParseError IO)) a -> T.Text -> Assertion
|
||||||
-> T.Text
|
assertParseE = assertParseHelperE assertFailure (const $ return ())
|
||||||
-> Assertion
|
|
||||||
assertParseE parser input = do
|
|
||||||
let filepath = ""
|
|
||||||
eep <- runExceptT $
|
|
||||||
runParserT (evalStateT (parser <* eof) def) filepath input
|
|
||||||
case eep of
|
|
||||||
Left finalErr ->
|
|
||||||
let prettyErr = finalErrorBundlePretty $ attachSource filepath input finalErr
|
|
||||||
in assertFailure $ "parse error at " <> prettyErr
|
|
||||||
Right ep ->
|
|
||||||
either (assertFailure.(++"\n").("\nparse error at "++).customErrorBundlePretty)
|
|
||||||
(const $ return ())
|
|
||||||
ep
|
|
||||||
|
|
||||||
assertParseEqE
|
assertParseEqE
|
||||||
:: (Default st, Eq a, Show a, HasCallStack)
|
:: (Default st, Eq a, Show a, HasCallStack)
|
||||||
=> StateT st (ParsecT CustomErr T.Text (ExceptT FinalParseError IO)) a
|
=> StateT st (ParsecT CustomErr T.Text (ExceptT FinalParseError IO)) a -> T.Text -> a -> Assertion
|
||||||
-> T.Text
|
|
||||||
-> a
|
|
||||||
-> Assertion
|
|
||||||
assertParseEqE parser input = assertParseEqOnE parser input id
|
assertParseEqE parser input = assertParseEqOnE parser input id
|
||||||
|
|
||||||
assertParseEqOnE
|
assertParseEqOnE
|
||||||
:: (HasCallStack, Eq b, Show b, Default st)
|
:: (HasCallStack, Eq b, Show b, Default st)
|
||||||
=> StateT st (ParsecT CustomErr T.Text (ExceptT FinalParseError IO)) a
|
=> StateT st (ParsecT CustomErr T.Text (ExceptT FinalParseError IO)) a -> T.Text -> (a -> b) -> b -> Assertion
|
||||||
-> T.Text
|
assertParseEqOnE parser input f expected =
|
||||||
-> (a -> b)
|
assertParseHelperE assertFailure (assertEqual "" expected . f) parser input
|
||||||
-> b
|
|
||||||
-> Assertion
|
|
||||||
assertParseEqOnE parser input f expected = do
|
|
||||||
let filepath = ""
|
|
||||||
eep <- runExceptT $ runParserT (evalStateT (parser <* eof) def) filepath input
|
|
||||||
case eep of
|
|
||||||
Left finalErr ->
|
|
||||||
let prettyErr = finalErrorBundlePretty $ attachSource filepath input finalErr
|
|
||||||
in assertFailure $ "parse error at " <> prettyErr
|
|
||||||
Right ep ->
|
|
||||||
either (assertFailure . (++"\n") . ("\nparse error at "++) . customErrorBundlePretty)
|
|
||||||
(assertEqual "" expected . f)
|
|
||||||
ep
|
|
||||||
|
|
||||||
assertParseErrorE
|
assertParseErrorE
|
||||||
:: (Default st, Eq a, Show a, HasCallStack)
|
:: (Default st, Eq a, Show a, HasCallStack)
|
||||||
=> StateT st (ParsecT CustomErr T.Text (ExceptT FinalParseError IO)) a
|
=> StateT st (ParsecT CustomErr T.Text (ExceptT FinalParseError IO)) a -> T.Text -> String -> Assertion
|
||||||
-> T.Text
|
assertParseErrorE parser input errstr = assertParseHelperE
|
||||||
-> String
|
(\e -> unless (errstr `isInfixOf` e) $ assertFailure $ "\nparse error is not as expected:" ++ e)
|
||||||
-> Assertion
|
(\v -> assertFailure $ "\nparse succeeded unexpectedly, producing:\n" ++ pshow v ++ "\n")
|
||||||
assertParseErrorE parser input errstr = do
|
parser input
|
||||||
let filepath = ""
|
|
||||||
eep <- runExceptT $ runParserT (evalStateT parser def) filepath input
|
|
||||||
case eep of
|
|
||||||
Left finalErr -> do
|
|
||||||
let prettyErr = finalErrorBundlePretty $ attachSource filepath input finalErr
|
|
||||||
if errstr `isInfixOf` prettyErr
|
|
||||||
then return ()
|
|
||||||
else assertFailure $ "\nparse error is not as expected:\n" ++ prettyErr ++ "\n"
|
|
||||||
Right ep -> case ep of
|
|
||||||
Right v -> assertFailure $ "\nparse succeeded unexpectedly, producing:\n" ++ pshow v ++ "\n"
|
|
||||||
Left e -> do
|
|
||||||
let e' = customErrorBundlePretty e
|
|
||||||
if errstr `isInfixOf` e'
|
|
||||||
then return ()
|
|
||||||
else assertFailure $ "\nparse error is not as expected:\n" ++ e' ++ "\n"
|
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user