{-# LANGUAGE FlexibleContexts #-}
{-# LANGUAGE NamedFieldPuns #-}
{-# LANGUAGE NoMonoLocalBinds #-}
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE PackageImports #-}
{-# LANGUAGE ScopedTypeVariables #-}
module Hledger.Read.JournalReader (
findReader,
splitReaderPrefix,
reader,
parseAndFinaliseJournal,
runJournalParser,
rjp,
runErroringJournalParser,
rejp,
getParentAccount,
journalp,
directivep,
defaultyeardirectivep,
marketpricedirectivep,
datetimep,
datep,
modifiedaccountnamep,
tmpostingrulep,
statusp,
emptyorcommentlinep,
followingcommentp,
accountaliasp
,tests_JournalReader
)
where
import qualified Control.Monad.Fail as Fail (fail)
import qualified Control.Exception as C
import Control.Monad (forM_, when, void, unless)
import Control.Monad.IO.Class (MonadIO, liftIO)
import Control.Monad.Except (ExceptT(..), runExceptT)
import Control.Monad.State.Strict (evalStateT,get,modify',put)
import Control.Monad.Trans.Class (lift)
import Data.Char (toLower)
import Data.Either (isRight)
import qualified Data.Map.Strict as M
import Data.Text (Text)
import Data.String
import Data.List
import Data.Maybe
import qualified Data.Text as T
import Data.Time.Calendar
import Data.Time.LocalTime
import Safe
import Text.Megaparsec hiding (parse)
import Text.Megaparsec.Char
import Text.Megaparsec.Custom
import Text.Printf
import System.FilePath
import "Glob" System.FilePath.Glob hiding (match)
import Hledger.Data
import Hledger.Read.Common
import Hledger.Utils
import qualified Hledger.Read.TimedotReader as TimedotReader (reader)
import qualified Hledger.Read.TimeclockReader as TimeclockReader (reader)
import qualified Hledger.Read.CsvReader as CsvReader (reader)
runJournalParser, rjp
:: Monad m
=> JournalParser m a -> Text -> m (Either HledgerParseErrors a)
runJournalParser :: JournalParser m a -> Text -> m (Either HledgerParseErrors a)
runJournalParser JournalParser m a
p = ParsecT HledgerParseErrorData Text m a
-> String -> Text -> m (Either HledgerParseErrors a)
forall (m :: * -> *) e s a.
Monad m =>
ParsecT e s m a
-> String -> s -> m (Either (ParseErrorBundle s e) a)
runParserT (JournalParser m a
-> Journal -> ParsecT HledgerParseErrorData Text m a
forall (m :: * -> *) s a. Monad m => StateT s m a -> s -> m a
evalStateT JournalParser m a
p Journal
nulljournal) String
""
rjp :: JournalParser m a -> Text -> m (Either HledgerParseErrors a)
rjp = JournalParser m a -> Text -> m (Either HledgerParseErrors a)
forall (m :: * -> *) a.
Monad m =>
JournalParser m a -> Text -> m (Either HledgerParseErrors a)
runJournalParser
runErroringJournalParser, rejp
:: Monad m
=> ErroringJournalParser m a
-> Text
-> m (Either FinalParseError (Either HledgerParseErrors a))
runErroringJournalParser :: ErroringJournalParser m a
-> Text -> m (Either FinalParseError (Either HledgerParseErrors a))
runErroringJournalParser ErroringJournalParser m a
p Text
t =
ExceptT FinalParseError m (Either HledgerParseErrors a)
-> m (Either FinalParseError (Either HledgerParseErrors a))
forall e (m :: * -> *) a. ExceptT e m a -> m (Either e a)
runExceptT (ExceptT FinalParseError m (Either HledgerParseErrors a)
-> m (Either FinalParseError (Either HledgerParseErrors a)))
-> ExceptT FinalParseError m (Either HledgerParseErrors a)
-> m (Either FinalParseError (Either HledgerParseErrors a))
forall a b. (a -> b) -> a -> b
$ ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m) a
-> String
-> Text
-> ExceptT FinalParseError m (Either HledgerParseErrors a)
forall (m :: * -> *) e s a.
Monad m =>
ParsecT e s m a
-> String -> s -> m (Either (ParseErrorBundle s e) a)
runParserT (ErroringJournalParser m a
-> Journal
-> ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m) a
forall (m :: * -> *) s a. Monad m => StateT s m a -> s -> m a
evalStateT ErroringJournalParser m a
p Journal
nulljournal) String
"" Text
t
rejp :: ErroringJournalParser m a
-> Text -> m (Either FinalParseError (Either HledgerParseErrors a))
rejp = ErroringJournalParser m a
-> Text -> m (Either FinalParseError (Either HledgerParseErrors a))
forall (m :: * -> *) a.
Monad m =>
ErroringJournalParser m a
-> Text -> m (Either FinalParseError (Either HledgerParseErrors a))
runErroringJournalParser
readers' :: MonadIO m => [Reader m]
readers' :: [Reader m]
readers' = [
Reader m
forall (m :: * -> *). MonadIO m => Reader m
reader
,Reader m
forall (m :: * -> *). MonadIO m => Reader m
TimeclockReader.reader
,Reader m
forall (m :: * -> *). MonadIO m => Reader m
TimedotReader.reader
,Reader m
forall (m :: * -> *). MonadIO m => Reader m
CsvReader.reader
]
readerNames :: [String]
readerNames :: [String]
readerNames = (Reader IO -> String) -> [Reader IO] -> [String]
forall a b. (a -> b) -> [a] -> [b]
map Reader IO -> String
forall (m :: * -> *). Reader m -> String
rFormat ([Reader IO]
forall (m :: * -> *). MonadIO m => [Reader m]
readers'::[Reader IO])
findReader :: MonadIO m => Maybe StorageFormat -> Maybe FilePath -> Maybe (Reader m)
findReader :: Maybe String -> Maybe String -> Maybe (Reader m)
findReader Maybe String
Nothing Maybe String
Nothing = Maybe (Reader m)
forall a. Maybe a
Nothing
findReader (Just String
fmt) Maybe String
_ = [Reader m] -> Maybe (Reader m)
forall a. [a] -> Maybe a
headMay [Reader m
r | Reader m
r <- [Reader m]
forall (m :: * -> *). MonadIO m => [Reader m]
readers', Reader m -> String
forall (m :: * -> *). Reader m -> String
rFormat Reader m
r String -> String -> Bool
forall a. Eq a => a -> a -> Bool
== String
fmt]
findReader Maybe String
Nothing (Just String
path) =
case Maybe String
prefix of
Just String
fmt -> [Reader m] -> Maybe (Reader m)
forall a. [a] -> Maybe a
headMay [Reader m
r | Reader m
r <- [Reader m]
forall (m :: * -> *). MonadIO m => [Reader m]
readers', Reader m -> String
forall (m :: * -> *). Reader m -> String
rFormat Reader m
r String -> String -> Bool
forall a. Eq a => a -> a -> Bool
== String
fmt]
Maybe String
Nothing -> [Reader m] -> Maybe (Reader m)
forall a. [a] -> Maybe a
headMay [Reader m
r | Reader m
r <- [Reader m]
forall (m :: * -> *). MonadIO m => [Reader m]
readers', String
ext String -> [String] -> Bool
forall (t :: * -> *) a. (Foldable t, Eq a) => a -> t a -> Bool
`elem` Reader m -> [String]
forall (m :: * -> *). Reader m -> [String]
rExtensions Reader m
r]
where
(Maybe String
prefix,String
path') = String -> (Maybe String, String)
splitReaderPrefix String
path
ext :: String
ext = (Char -> Char) -> String -> String
forall a b. (a -> b) -> [a] -> [b]
map Char -> Char
toLower (String -> String) -> String -> String
forall a b. (a -> b) -> a -> b
$ Int -> String -> String
forall a. Int -> [a] -> [a]
drop Int
1 (String -> String) -> String -> String
forall a b. (a -> b) -> a -> b
$ String -> String
takeExtension String
path'
type PrefixedFilePath = FilePath
splitReaderPrefix :: PrefixedFilePath -> (Maybe String, FilePath)
splitReaderPrefix :: String -> (Maybe String, String)
splitReaderPrefix String
f =
(Maybe String, String)
-> [(Maybe String, String)] -> (Maybe String, String)
forall a. a -> [a] -> a
headDef (Maybe String
forall a. Maybe a
Nothing, String
f)
[(String -> Maybe String
forall a. a -> Maybe a
Just String
r, Int -> String -> String
forall a. Int -> [a] -> [a]
drop (String -> Int
forall (t :: * -> *) a. Foldable t => t a -> Int
length String
r Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
1) String
f) | String
r <- [String]
readerNames, (String
rString -> String -> String
forall a. [a] -> [a] -> [a]
++String
":") String -> String -> Bool
forall a. Eq a => [a] -> [a] -> Bool
`isPrefixOf` String
f]
reader :: MonadIO m => Reader m
reader :: Reader m
reader = Reader :: forall (m :: * -> *).
String
-> [String]
-> (InputOpts -> String -> Text -> ExceptT String IO Journal)
-> (MonadIO m => ErroringJournalParser m Journal)
-> Reader m
Reader
{rFormat :: String
rFormat = String
"journal"
,rExtensions :: [String]
rExtensions = [String
"journal", String
"j", String
"hledger", String
"ledger"]
,rReadFn :: InputOpts -> String -> Text -> ExceptT String IO Journal
rReadFn = InputOpts -> String -> Text -> ExceptT String IO Journal
parse
,rParser :: MonadIO m => ErroringJournalParser m Journal
rParser = MonadIO m => ErroringJournalParser m Journal
forall (m :: * -> *). MonadIO m => ErroringJournalParser m Journal
journalp
}
parse :: InputOpts -> FilePath -> Text -> ExceptT String IO Journal
parse :: InputOpts -> String -> Text -> ExceptT String IO Journal
parse InputOpts
iopts = ErroringJournalParser IO Journal
-> InputOpts -> String -> Text -> ExceptT String IO Journal
parseAndFinaliseJournal ErroringJournalParser IO Journal
journalp' InputOpts
iopts
where
journalp' :: ErroringJournalParser IO Journal
journalp' = do
(AccountAlias
-> StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError IO))
())
-> [AccountAlias]
-> StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError IO))
()
forall (t :: * -> *) (m :: * -> *) a b.
(Foldable t, Monad m) =>
(a -> m b) -> t a -> m ()
mapM_ AccountAlias
-> StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError IO))
()
forall (m :: * -> *). MonadState Journal m => AccountAlias -> m ()
addAccountAlias ([AccountAlias] -> [AccountAlias]
forall a. [a] -> [a]
reverse ([AccountAlias] -> [AccountAlias])
-> [AccountAlias] -> [AccountAlias]
forall a b. (a -> b) -> a -> b
$ InputOpts -> [AccountAlias]
aliasesFromOpts InputOpts
iopts)
ErroringJournalParser IO Journal
forall (m :: * -> *). MonadIO m => ErroringJournalParser m Journal
journalp
journalp :: MonadIO m => ErroringJournalParser m ParsedJournal
journalp :: ErroringJournalParser m Journal
journalp = do
StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
()
-> StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
[()]
forall (m :: * -> *) a. MonadPlus m => m a -> m [a]
many StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
()
forall (m :: * -> *). MonadIO m => ErroringJournalParser m ()
addJournalItemP
StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
()
forall e s (m :: * -> *). MonadParsec e s m => m ()
eof
ErroringJournalParser m Journal
forall s (m :: * -> *). MonadState s m => m s
get
addJournalItemP :: MonadIO m => ErroringJournalParser m ()
addJournalItemP :: ErroringJournalParser m ()
addJournalItemP =
[ErroringJournalParser m ()] -> ErroringJournalParser m ()
forall (f :: * -> *) (m :: * -> *) a.
(Foldable f, Alternative m) =>
f (m a) -> m a
choice [
ErroringJournalParser m ()
forall (m :: * -> *). MonadIO m => ErroringJournalParser m ()
directivep
, JournalParser (ExceptT FinalParseError m) Transaction
forall (m :: * -> *). JournalParser m Transaction
transactionp JournalParser (ExceptT FinalParseError m) Transaction
-> (Transaction -> ErroringJournalParser m ())
-> ErroringJournalParser m ()
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= (Journal -> Journal) -> ErroringJournalParser m ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify' ((Journal -> Journal) -> ErroringJournalParser m ())
-> (Transaction -> Journal -> Journal)
-> Transaction
-> ErroringJournalParser m ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Transaction -> Journal -> Journal
addTransaction
, JournalParser (ExceptT FinalParseError m) TransactionModifier
forall (m :: * -> *). JournalParser m TransactionModifier
transactionmodifierp JournalParser (ExceptT FinalParseError m) TransactionModifier
-> (TransactionModifier -> ErroringJournalParser m ())
-> ErroringJournalParser m ()
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= (Journal -> Journal) -> ErroringJournalParser m ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify' ((Journal -> Journal) -> ErroringJournalParser m ())
-> (TransactionModifier -> Journal -> Journal)
-> TransactionModifier
-> ErroringJournalParser m ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. TransactionModifier -> Journal -> Journal
addTransactionModifier
, JournalParser (ExceptT FinalParseError m) PeriodicTransaction
forall (m :: * -> *).
MonadIO m =>
JournalParser m PeriodicTransaction
periodictransactionp JournalParser (ExceptT FinalParseError m) PeriodicTransaction
-> (PeriodicTransaction -> ErroringJournalParser m ())
-> ErroringJournalParser m ()
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= (Journal -> Journal) -> ErroringJournalParser m ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify' ((Journal -> Journal) -> ErroringJournalParser m ())
-> (PeriodicTransaction -> Journal -> Journal)
-> PeriodicTransaction
-> ErroringJournalParser m ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PeriodicTransaction -> Journal -> Journal
addPeriodicTransaction
, JournalParser (ExceptT FinalParseError m) PriceDirective
forall (m :: * -> *). JournalParser m PriceDirective
marketpricedirectivep JournalParser (ExceptT FinalParseError m) PriceDirective
-> (PriceDirective -> ErroringJournalParser m ())
-> ErroringJournalParser m ()
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= (Journal -> Journal) -> ErroringJournalParser m ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify' ((Journal -> Journal) -> ErroringJournalParser m ())
-> (PriceDirective -> Journal -> Journal)
-> PriceDirective
-> ErroringJournalParser m ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PriceDirective -> Journal -> Journal
addPriceDirective
, ErroringJournalParser m () -> ErroringJournalParser m ()
forall (f :: * -> *) a. Functor f => f a -> f ()
void (ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m) ()
-> ErroringJournalParser m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m) ()
forall (m :: * -> *). TextParser m ()
emptyorcommentlinep)
, ErroringJournalParser m () -> ErroringJournalParser m ()
forall (f :: * -> *) a. Functor f => f a -> f ()
void (ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m) ()
-> ErroringJournalParser m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m) ()
forall (m :: * -> *). TextParser m ()
multilinecommentp)
] ErroringJournalParser m () -> String -> ErroringJournalParser m ()
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"transaction or directive"
directivep :: MonadIO m => ErroringJournalParser m ()
directivep :: ErroringJournalParser m ()
directivep = (do
StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
Char
-> StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
(Maybe Char)
forall (f :: * -> *) a. Alternative f => f a -> f (Maybe a)
optional (StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
Char
-> StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
(Maybe Char))
-> StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
Char
-> StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
(Maybe Char)
forall a b. (a -> b) -> a -> b
$ Token Text
-> StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
(Token Text)
forall e s (m :: * -> *).
(MonadParsec e s m, Token s ~ Char) =>
Token s -> m (Token s)
char Char
Token Text
'!'
[ErroringJournalParser m ()] -> ErroringJournalParser m ()
forall (f :: * -> *) (m :: * -> *) a.
(Foldable f, Alternative m) =>
f (m a) -> m a
choice [
ErroringJournalParser m ()
forall (m :: * -> *). MonadIO m => ErroringJournalParser m ()
includedirectivep
,ErroringJournalParser m ()
forall (m :: * -> *). JournalParser m ()
aliasdirectivep
,ErroringJournalParser m ()
forall (m :: * -> *). JournalParser m ()
endaliasesdirectivep
,ErroringJournalParser m ()
forall (m :: * -> *). JournalParser m ()
accountdirectivep
,ErroringJournalParser m ()
forall (m :: * -> *). JournalParser m ()
applyaccountdirectivep
,ErroringJournalParser m ()
forall (m :: * -> *). JournalParser m ()
commoditydirectivep
,ErroringJournalParser m ()
forall (m :: * -> *). JournalParser m ()
endapplyaccountdirectivep
,ErroringJournalParser m ()
forall (m :: * -> *). JournalParser m ()
payeedirectivep
,ErroringJournalParser m ()
forall (m :: * -> *). JournalParser m ()
tagdirectivep
,ErroringJournalParser m ()
forall (m :: * -> *). JournalParser m ()
endtagdirectivep
,ErroringJournalParser m ()
forall (m :: * -> *). JournalParser m ()
defaultyeardirectivep
,ErroringJournalParser m ()
forall (m :: * -> *). JournalParser m ()
defaultcommoditydirectivep
,ErroringJournalParser m ()
forall (m :: * -> *). JournalParser m ()
commodityconversiondirectivep
,ErroringJournalParser m ()
forall (m :: * -> *). JournalParser m ()
ignoredpricecommoditydirectivep
,ErroringJournalParser m ()
forall (m :: * -> *). JournalParser m ()
decimalmarkdirectivep
]
) ErroringJournalParser m () -> String -> ErroringJournalParser m ()
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"directive"
includedirectivep :: MonadIO m => ErroringJournalParser m ()
includedirectivep :: ErroringJournalParser m ()
includedirectivep = do
Tokens Text
-> StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
(Tokens Text)
forall e s (m :: * -> *).
MonadParsec e s m =>
Tokens s -> m (Tokens s)
string Tokens Text
"include"
ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m) ()
-> ErroringJournalParser m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m) ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces1
String
prefixedglob <- Text -> String
T.unpack (Text -> String)
-> StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
Text
-> StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
String
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Maybe String
-> (Token Text -> Bool)
-> StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
(Tokens Text)
forall e s (m :: * -> *).
MonadParsec e s m =>
Maybe String -> (Token s -> Bool) -> m (Tokens s)
takeWhileP Maybe String
forall a. Maybe a
Nothing (Char -> Char -> Bool
forall a. Eq a => a -> a -> Bool
/= Char
'\n')
Int
parentoff <- StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
Int
forall e s (m :: * -> *). MonadParsec e s m => m Int
getOffset
SourcePos
parentpos <- StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
SourcePos
forall s e (m :: * -> *).
(TraversableStream s, MonadParsec e s m) =>
m SourcePos
getSourcePos
let (Maybe String
mprefix,String
glob) = String -> (Maybe String, String)
splitReaderPrefix String
prefixedglob
[String]
paths <- Int
-> SourcePos
-> String
-> JournalParser (ExceptT FinalParseError m) [String]
forall (m :: * -> *).
MonadIO m =>
Int -> SourcePos -> String -> JournalParser m [String]
getFilePaths Int
parentoff SourcePos
parentpos String
glob
let prefixedpaths :: [String]
prefixedpaths = case Maybe String
mprefix of
Maybe String
Nothing -> [String]
paths
Just String
fmt -> (String -> String) -> [String] -> [String]
forall a b. (a -> b) -> [a] -> [b]
map ((String
fmtString -> String -> String
forall a. [a] -> [a] -> [a]
++String
":")String -> String -> String
forall a. [a] -> [a] -> [a]
++) [String]
paths
[String]
-> (String -> ErroringJournalParser m ())
-> ErroringJournalParser m ()
forall (t :: * -> *) (m :: * -> *) a b.
(Foldable t, Monad m) =>
t a -> (a -> m b) -> m ()
forM_ [String]
prefixedpaths ((String -> ErroringJournalParser m ())
-> ErroringJournalParser m ())
-> (String -> ErroringJournalParser m ())
-> ErroringJournalParser m ()
forall a b. (a -> b) -> a -> b
$ SourcePos -> String -> ErroringJournalParser m ()
forall (m :: * -> *).
MonadIO m =>
SourcePos -> String -> ErroringJournalParser m ()
parseChild SourcePos
parentpos
StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
Char
-> ErroringJournalParser m ()
forall (f :: * -> *) a. Functor f => f a -> f ()
void StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
Char
forall e s (m :: * -> *).
(MonadParsec e s m, Token s ~ Char) =>
m (Token s)
newline
where
getFilePaths
:: MonadIO m => Int -> SourcePos -> FilePath -> JournalParser m [FilePath]
getFilePaths :: Int -> SourcePos -> String -> JournalParser m [String]
getFilePaths Int
parseroff SourcePos
parserpos String
filename = do
let curdir :: String
curdir = String -> String
takeDirectory (SourcePos -> String
sourceName SourcePos
parserpos)
String
filename' <- ParsecT HledgerParseErrorData Text m String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) String
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (ParsecT HledgerParseErrorData Text m String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) String)
-> ParsecT HledgerParseErrorData Text m String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) String
forall a b. (a -> b) -> a -> b
$ String -> IO String
expandHomePath String
filename
IO String -> String -> ParsecT HledgerParseErrorData Text m String
forall (m :: * -> *) a.
MonadIO m =>
IO a -> String -> TextParser m a
`orRethrowIOError` (SourcePos -> String
forall a. Show a => a -> String
show SourcePos
parserpos String -> String -> String
forall a. [a] -> [a] -> [a]
++ String
" locating " String -> String -> String
forall a. [a] -> [a] -> [a]
++ String
filename)
Pattern
fileglob <- case CompOptions -> String -> Either String Pattern
tryCompileWith CompOptions
compDefault{errorRecovery :: Bool
errorRecovery=Bool
False} String
filename' of
Right Pattern
x -> Pattern
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Pattern
forall (f :: * -> *) a. Applicative f => a -> f a
pure Pattern
x
Left String
e -> HledgerParseErrorData
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Pattern
forall e s (m :: * -> *) a. MonadParsec e s m => e -> m a
customFailure (HledgerParseErrorData
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Pattern)
-> HledgerParseErrorData
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Pattern
forall a b. (a -> b) -> a -> b
$
Int -> String -> HledgerParseErrorData
parseErrorAt Int
parseroff (String -> HledgerParseErrorData)
-> String -> HledgerParseErrorData
forall a b. (a -> b) -> a -> b
$ String
"Invalid glob pattern: " String -> String -> String
forall a. [a] -> [a] -> [a]
++ String
e
[String]
filepaths <- IO [String] -> JournalParser m [String]
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO (IO [String] -> JournalParser m [String])
-> IO [String] -> JournalParser m [String]
forall a b. (a -> b) -> a -> b
$ [String] -> [String]
forall a. Ord a => [a] -> [a]
sort ([String] -> [String]) -> IO [String] -> IO [String]
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Pattern -> String -> IO [String]
globDir1 Pattern
fileglob String
curdir
if (Bool -> Bool
not (Bool -> Bool) -> ([String] -> Bool) -> [String] -> Bool
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [String] -> Bool
forall (t :: * -> *) a. Foldable t => t a -> Bool
null) [String]
filepaths
then [String] -> JournalParser m [String]
forall (f :: * -> *) a. Applicative f => a -> f a
pure [String]
filepaths
else HledgerParseErrorData -> JournalParser m [String]
forall e s (m :: * -> *) a. MonadParsec e s m => e -> m a
customFailure (HledgerParseErrorData -> JournalParser m [String])
-> HledgerParseErrorData -> JournalParser m [String]
forall a b. (a -> b) -> a -> b
$ Int -> String -> HledgerParseErrorData
parseErrorAt Int
parseroff (String -> HledgerParseErrorData)
-> String -> HledgerParseErrorData
forall a b. (a -> b) -> a -> b
$
String
"No existing files match pattern: " String -> String -> String
forall a. [a] -> [a] -> [a]
++ String
filename
parseChild :: MonadIO m => SourcePos -> PrefixedFilePath -> ErroringJournalParser m ()
parseChild :: SourcePos -> String -> ErroringJournalParser m ()
parseChild SourcePos
parentpos String
prefixedpath = do
let (Maybe String
_mprefix,String
filepath) = String -> (Maybe String, String)
splitReaderPrefix String
prefixedpath
Journal
parentj <- StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
Journal
forall s (m :: * -> *). MonadState s m => m s
get
let parentfilestack :: [String]
parentfilestack = Journal -> [String]
jincludefilestack Journal
parentj
Bool -> ErroringJournalParser m () -> ErroringJournalParser m ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when (String
filepath String -> [String] -> Bool
forall (t :: * -> *) a. (Foldable t, Eq a) => a -> t a -> Bool
`elem` [String]
parentfilestack) (ErroringJournalParser m () -> ErroringJournalParser m ())
-> ErroringJournalParser m () -> ErroringJournalParser m ()
forall a b. (a -> b) -> a -> b
$
String -> ErroringJournalParser m ()
forall (m :: * -> *) a. MonadFail m => String -> m a
Fail.fail (String
"Cyclic include: " String -> String -> String
forall a. [a] -> [a] -> [a]
++ String
filepath)
Text
childInput <- ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m) Text
-> StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
Text
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (ParsecT
HledgerParseErrorData Text (ExceptT FinalParseError m) Text
-> StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
Text)
-> ParsecT
HledgerParseErrorData Text (ExceptT FinalParseError m) Text
-> StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
Text
forall a b. (a -> b) -> a -> b
$ String -> IO Text
readFilePortably String
filepath
IO Text
-> String
-> ParsecT
HledgerParseErrorData Text (ExceptT FinalParseError m) Text
forall (m :: * -> *) a.
MonadIO m =>
IO a -> String -> TextParser m a
`orRethrowIOError` (SourcePos -> String
forall a. Show a => a -> String
show SourcePos
parentpos String -> String -> String
forall a. [a] -> [a] -> [a]
++ String
" reading " String -> String -> String
forall a. [a] -> [a] -> [a]
++ String
filepath)
let initChildj :: Journal
initChildj = String -> Journal -> Journal
newJournalWithParseStateFrom String
filepath Journal
parentj
let r :: Reader m
r = Reader m -> Maybe (Reader m) -> Reader m
forall a. a -> Maybe a -> a
fromMaybe Reader m
forall (m :: * -> *). MonadIO m => Reader m
reader (Maybe (Reader m) -> Reader m) -> Maybe (Reader m) -> Reader m
forall a b. (a -> b) -> a -> b
$ Maybe String -> Maybe String -> Maybe (Reader m)
forall (m :: * -> *).
MonadIO m =>
Maybe String -> Maybe String -> Maybe (Reader m)
findReader Maybe String
forall a. Maybe a
Nothing (String -> Maybe String
forall a. a -> Maybe a
Just String
prefixedpath)
parser :: StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
Journal
parser = Reader m
-> MonadIO m =>
StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
Journal
forall (m :: * -> *).
Reader m -> MonadIO m => ErroringJournalParser m Journal
rParser Reader m
r
String -> String -> ErroringJournalParser m ()
forall (m :: * -> *) a. (MonadIO m, Show a) => String -> a -> m ()
dbg6IO String
"trying reader" (Reader m -> String
forall (m :: * -> *). Reader m -> String
rFormat Reader m
r)
Journal
updatedChildj <- (String, Text) -> Journal -> Journal
journalAddFile (String
filepath, Text
childInput) (Journal -> Journal)
-> StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
Journal
-> StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
Journal
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$>
StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
Journal
-> Journal
-> String
-> Text
-> StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
Journal
forall (m :: * -> *) st a.
Monad m =>
StateT
st
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
a
-> st
-> String
-> Text
-> StateT
st
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
a
parseIncludeFile StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError m))
Journal
parser Journal
initChildj String
filepath Text
childInput
Journal -> ErroringJournalParser m ()
forall s (m :: * -> *). MonadState s m => s -> m ()
put (Journal -> ErroringJournalParser m ())
-> Journal -> ErroringJournalParser m ()
forall a b. (a -> b) -> a -> b
$ Journal
updatedChildj Journal -> Journal -> Journal
forall a. Semigroup a => a -> a -> a
<> Journal
parentj
newJournalWithParseStateFrom :: FilePath -> Journal -> Journal
newJournalWithParseStateFrom :: String -> Journal -> Journal
newJournalWithParseStateFrom String
filepath Journal
j = Journal
nulljournal{
jparsedefaultyear :: Maybe Year
jparsedefaultyear = Journal -> Maybe Year
jparsedefaultyear Journal
j
,jparsedefaultcommodity :: Maybe (Text, AmountStyle)
jparsedefaultcommodity = Journal -> Maybe (Text, AmountStyle)
jparsedefaultcommodity Journal
j
,jparseparentaccounts :: [Text]
jparseparentaccounts = Journal -> [Text]
jparseparentaccounts Journal
j
,jparsedecimalmark :: Maybe Char
jparsedecimalmark = Journal -> Maybe Char
jparsedecimalmark Journal
j
,jparsealiases :: [AccountAlias]
jparsealiases = Journal -> [AccountAlias]
jparsealiases Journal
j
,jcommodities :: Map Text Commodity
jcommodities = Journal -> Map Text Commodity
jcommodities Journal
j
,jparsetimeclockentries :: [TimeclockEntry]
jparsetimeclockentries = Journal -> [TimeclockEntry]
jparsetimeclockentries Journal
j
,jincludefilestack :: [String]
jincludefilestack = String
filepath String -> [String] -> [String]
forall a. a -> [a] -> [a]
: Journal -> [String]
jincludefilestack Journal
j
}
orRethrowIOError :: MonadIO m => IO a -> String -> TextParser m a
orRethrowIOError :: IO a -> String -> TextParser m a
orRethrowIOError IO a
io String
msg = do
Either String a
eResult <- IO (Either String a)
-> ParsecT HledgerParseErrorData Text m (Either String a)
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO (IO (Either String a)
-> ParsecT HledgerParseErrorData Text m (Either String a))
-> IO (Either String a)
-> ParsecT HledgerParseErrorData Text m (Either String a)
forall a b. (a -> b) -> a -> b
$ (a -> Either String a
forall a b. b -> Either a b
Right (a -> Either String a) -> IO a -> IO (Either String a)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> IO a
io) IO (Either String a)
-> (IOException -> IO (Either String a)) -> IO (Either String a)
forall e a. Exception e => IO a -> (e -> IO a) -> IO a
`C.catch` \(IOException
e::C.IOException) -> Either String a -> IO (Either String a)
forall (f :: * -> *) a. Applicative f => a -> f a
pure (Either String a -> IO (Either String a))
-> Either String a -> IO (Either String a)
forall a b. (a -> b) -> a -> b
$ String -> Either String a
forall a b. a -> Either a b
Left (String -> Either String a) -> String -> Either String a
forall a b. (a -> b) -> a -> b
$ String -> String -> String -> String
forall r. PrintfType r => String -> r
printf String
"%s:\n%s" String
msg (IOException -> String
forall a. Show a => a -> String
show IOException
e)
case Either String a
eResult of
Right a
res -> a -> TextParser m a
forall (f :: * -> *) a. Applicative f => a -> f a
pure a
res
Left String
errMsg -> String -> TextParser m a
forall (m :: * -> *) a. MonadFail m => String -> m a
Fail.fail String
errMsg
accountdirectivep :: JournalParser m ()
accountdirectivep :: JournalParser m ()
accountdirectivep = do
Int
off <- StateT Journal (ParsecT HledgerParseErrorData Text m) Int
forall e s (m :: * -> *). MonadParsec e s m => m Int
getOffset
Tokens Text
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Tokens Text)
forall e s (m :: * -> *).
MonadParsec e s m =>
Tokens s -> m (Tokens s)
string Tokens Text
"account"
ParsecT HledgerParseErrorData Text m () -> JournalParser m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces1
Text
acct <- StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall (m :: * -> *). JournalParser m Text
modifiedaccountnamep
(Text
cmt, [Tag]
tags) <- ParsecT HledgerParseErrorData Text m (Text, [Tag])
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Text, [Tag])
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m (Text, [Tag])
forall (m :: * -> *). TextParser m (Text, [Tag])
transactioncommentp
StateT Journal (ParsecT HledgerParseErrorData Text m) String
-> JournalParser m ()
forall (m :: * -> *) a. MonadPlus m => m a -> m ()
skipMany StateT Journal (ParsecT HledgerParseErrorData Text m) String
forall (m :: * -> *). JournalParser m String
indentedlinep
let
metype :: Maybe (Either String AccountType)
metype = Text -> Either String AccountType
parseAccountTypeCode (Text -> Either String AccountType)
-> Maybe Text -> Maybe (Either String AccountType)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Text -> [Tag] -> Maybe Text
forall a b. Eq a => a -> [(a, b)] -> Maybe b
lookup Text
accountTypeTagName [Tag]
tags
(Text, Text, [Tag]) -> JournalParser m ()
forall (m :: * -> *). (Text, Text, [Tag]) -> JournalParser m ()
addAccountDeclaration (Text
acct, Text
cmt, [Tag]
tags)
Bool -> JournalParser m () -> JournalParser m ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
unless ([Tag] -> Bool
forall (t :: * -> *) a. Foldable t => t a -> Bool
null [Tag]
tags) (JournalParser m () -> JournalParser m ())
-> JournalParser m () -> JournalParser m ()
forall a b. (a -> b) -> a -> b
$ Text -> [Tag] -> JournalParser m ()
forall (m :: * -> *). Text -> [Tag] -> JournalParser m ()
addDeclaredAccountTags Text
acct [Tag]
tags
case Maybe (Either String AccountType)
metype of
Maybe (Either String AccountType)
Nothing -> () -> JournalParser m ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()
Just (Right AccountType
t) -> Text -> AccountType -> JournalParser m ()
forall (m :: * -> *). Text -> AccountType -> JournalParser m ()
addDeclaredAccountType Text
acct AccountType
t
Just (Left String
err) -> HledgerParseErrorData -> JournalParser m ()
forall e s (m :: * -> *) a. MonadParsec e s m => e -> m a
customFailure (HledgerParseErrorData -> JournalParser m ())
-> HledgerParseErrorData -> JournalParser m ()
forall a b. (a -> b) -> a -> b
$ Int -> String -> HledgerParseErrorData
parseErrorAt Int
off String
err
accountTypeTagName :: Text
accountTypeTagName = Text
"type"
parseAccountTypeCode :: Text -> Either String AccountType
parseAccountTypeCode :: Text -> Either String AccountType
parseAccountTypeCode Text
s =
case Text -> Text
T.toLower Text
s of
Text
"asset" -> AccountType -> Either String AccountType
forall a b. b -> Either a b
Right AccountType
Asset
Text
"a" -> AccountType -> Either String AccountType
forall a b. b -> Either a b
Right AccountType
Asset
Text
"liability" -> AccountType -> Either String AccountType
forall a b. b -> Either a b
Right AccountType
Liability
Text
"l" -> AccountType -> Either String AccountType
forall a b. b -> Either a b
Right AccountType
Liability
Text
"equity" -> AccountType -> Either String AccountType
forall a b. b -> Either a b
Right AccountType
Equity
Text
"e" -> AccountType -> Either String AccountType
forall a b. b -> Either a b
Right AccountType
Equity
Text
"revenue" -> AccountType -> Either String AccountType
forall a b. b -> Either a b
Right AccountType
Revenue
Text
"r" -> AccountType -> Either String AccountType
forall a b. b -> Either a b
Right AccountType
Revenue
Text
"expense" -> AccountType -> Either String AccountType
forall a b. b -> Either a b
Right AccountType
Expense
Text
"x" -> AccountType -> Either String AccountType
forall a b. b -> Either a b
Right AccountType
Expense
Text
"cash" -> AccountType -> Either String AccountType
forall a b. b -> Either a b
Right AccountType
Cash
Text
"c" -> AccountType -> Either String AccountType
forall a b. b -> Either a b
Right AccountType
Cash
Text
"conversion" -> AccountType -> Either String AccountType
forall a b. b -> Either a b
Right AccountType
Conversion
Text
"v" -> AccountType -> Either String AccountType
forall a b. b -> Either a b
Right AccountType
Conversion
Text
_ -> String -> Either String AccountType
forall a b. a -> Either a b
Left String
err
where
err :: String
err = Text -> String
T.unpack (Text -> String) -> Text -> String
forall a b. (a -> b) -> a -> b
$ Text
"invalid account type code "Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<>Text
sText -> Text -> Text
forall a. Semigroup a => a -> a -> a
<>Text
", should be one of " Text -> Text -> Text
forall a. Semigroup a => a -> a -> a
<>
Text -> [Text] -> Text
T.intercalate Text
", " [Text
"A",Text
"L",Text
"E",Text
"R",Text
"X",Text
"C",Text
"V",Text
"Asset",Text
"Liability",Text
"Equity",Text
"Revenue",Text
"Expense",Text
"Cash",Text
"Conversion"]
addAccountDeclaration :: (AccountName,Text,[Tag]) -> JournalParser m ()
addAccountDeclaration :: (Text, Text, [Tag]) -> JournalParser m ()
addAccountDeclaration (Text
a,Text
cmt,[Tag]
tags) =
(Journal -> Journal) -> JournalParser m ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify' (\Journal
j ->
let
decls :: [(Text, AccountDeclarationInfo)]
decls = Journal -> [(Text, AccountDeclarationInfo)]
jdeclaredaccounts Journal
j
d :: (Text, AccountDeclarationInfo)
d = (Text
a, AccountDeclarationInfo
nullaccountdeclarationinfo{
adicomment :: Text
adicomment = Text
cmt
,aditags :: [Tag]
aditags = [Tag]
tags
,adideclarationorder :: Int
adideclarationorder = [(Text, AccountDeclarationInfo)] -> Int
forall (t :: * -> *) a. Foldable t => t a -> Int
length [(Text, AccountDeclarationInfo)]
decls Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
1
})
in
Journal
j{jdeclaredaccounts :: [(Text, AccountDeclarationInfo)]
jdeclaredaccounts = (Text, AccountDeclarationInfo)
d(Text, AccountDeclarationInfo)
-> [(Text, AccountDeclarationInfo)]
-> [(Text, AccountDeclarationInfo)]
forall a. a -> [a] -> [a]
:[(Text, AccountDeclarationInfo)]
decls})
addPayeeDeclaration :: (Payee,Text,[Tag]) -> JournalParser m ()
addPayeeDeclaration :: (Text, Text, [Tag]) -> JournalParser m ()
addPayeeDeclaration (Text
p, Text
cmt, [Tag]
tags) =
(Journal -> Journal) -> JournalParser m ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify' (\j :: Journal
j@Journal{[(Text, PayeeDeclarationInfo)]
jdeclaredpayees :: Journal -> [(Text, PayeeDeclarationInfo)]
jdeclaredpayees :: [(Text, PayeeDeclarationInfo)]
jdeclaredpayees} -> Journal
j{jdeclaredpayees :: [(Text, PayeeDeclarationInfo)]
jdeclaredpayees=(Text, PayeeDeclarationInfo)
d(Text, PayeeDeclarationInfo)
-> [(Text, PayeeDeclarationInfo)] -> [(Text, PayeeDeclarationInfo)]
forall a. a -> [a] -> [a]
:[(Text, PayeeDeclarationInfo)]
jdeclaredpayees})
where
d :: (Text, PayeeDeclarationInfo)
d = (Text
p
,PayeeDeclarationInfo
nullpayeedeclarationinfo{
pdicomment :: Text
pdicomment = Text
cmt
,pditags :: [Tag]
pditags = [Tag]
tags
})
indentedlinep :: JournalParser m String
indentedlinep :: JournalParser m String
indentedlinep = ParsecT HledgerParseErrorData Text m ()
-> StateT Journal (ParsecT HledgerParseErrorData Text m) ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces1 StateT Journal (ParsecT HledgerParseErrorData Text m) ()
-> JournalParser m String -> JournalParser m String
forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>> (String -> String
rstrip (String -> String)
-> JournalParser m String -> JournalParser m String
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> ParsecT HledgerParseErrorData Text m String
-> JournalParser m String
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m String
forall (m :: * -> *). TextParser m String
restofline)
commoditydirectivep :: JournalParser m ()
commoditydirectivep :: JournalParser m ()
commoditydirectivep = JournalParser m ()
forall (m :: * -> *). JournalParser m ()
commoditydirectiveonelinep JournalParser m () -> JournalParser m () -> JournalParser m ()
forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|> JournalParser m ()
forall (m :: * -> *). JournalParser m ()
commoditydirectivemultilinep
commoditydirectiveonelinep :: JournalParser m ()
commoditydirectiveonelinep :: JournalParser m ()
commoditydirectiveonelinep = do
(Int
off, Amount{Text
acommodity :: Amount -> Text
acommodity :: Text
acommodity,AmountStyle
astyle :: Amount -> AmountStyle
astyle :: AmountStyle
astyle}) <- StateT Journal (ParsecT HledgerParseErrorData Text m) (Int, Amount)
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Int, Amount)
forall e s (m :: * -> *) a. MonadParsec e s m => m a -> m a
try (StateT
Journal (ParsecT HledgerParseErrorData Text m) (Int, Amount)
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Int, Amount))
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Int, Amount)
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Int, Amount)
forall a b. (a -> b) -> a -> b
$ do
Tokens Text
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Tokens Text)
forall e s (m :: * -> *).
MonadParsec e s m =>
Tokens s -> m (Tokens s)
string Tokens Text
"commodity"
ParsecT HledgerParseErrorData Text m () -> JournalParser m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces1
Int
off <- StateT Journal (ParsecT HledgerParseErrorData Text m) Int
forall e s (m :: * -> *). MonadParsec e s m => m Int
getOffset
Amount
amount <- JournalParser m Amount
forall (m :: * -> *). JournalParser m Amount
amountp
(Int, Amount)
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Int, Amount)
forall (f :: * -> *) a. Applicative f => a -> f a
pure ((Int, Amount)
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Int, Amount))
-> (Int, Amount)
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Int, Amount)
forall a b. (a -> b) -> a -> b
$ (Int
off, Amount
amount)
ParsecT HledgerParseErrorData Text m () -> JournalParser m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces
Text
_ <- ParsecT HledgerParseErrorData Text m Text
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m Text
forall (m :: * -> *). TextParser m Text
followingcommentp
let comm :: Commodity
comm = Commodity :: Text -> Maybe AmountStyle -> Commodity
Commodity{csymbol :: Text
csymbol=Text
acommodity, cformat :: Maybe AmountStyle
cformat=AmountStyle -> Maybe AmountStyle
forall a. a -> Maybe a
Just (AmountStyle -> Maybe AmountStyle)
-> AmountStyle -> Maybe AmountStyle
forall a b. (a -> b) -> a -> b
$ String -> AmountStyle -> AmountStyle
forall a. Show a => String -> a -> a
dbg6 String
"style from commodity directive" AmountStyle
astyle}
if Maybe Char -> Bool
forall a. Maybe a -> Bool
isNothing (Maybe Char -> Bool) -> Maybe Char -> Bool
forall a b. (a -> b) -> a -> b
$ AmountStyle -> Maybe Char
asdecimalpoint AmountStyle
astyle
then HledgerParseErrorData -> JournalParser m ()
forall e s (m :: * -> *) a. MonadParsec e s m => e -> m a
customFailure (HledgerParseErrorData -> JournalParser m ())
-> HledgerParseErrorData -> JournalParser m ()
forall a b. (a -> b) -> a -> b
$ Int -> String -> HledgerParseErrorData
parseErrorAt Int
off String
pleaseincludedecimalpoint
else (Journal -> Journal) -> JournalParser m ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify' (\Journal
j -> Journal
j{jcommodities :: Map Text Commodity
jcommodities=Text -> Commodity -> Map Text Commodity -> Map Text Commodity
forall k a. Ord k => k -> a -> Map k a -> Map k a
M.insert Text
acommodity Commodity
comm (Map Text Commodity -> Map Text Commodity)
-> Map Text Commodity -> Map Text Commodity
forall a b. (a -> b) -> a -> b
$ Journal -> Map Text Commodity
jcommodities Journal
j})
pleaseincludedecimalpoint :: String
pleaseincludedecimalpoint :: String
pleaseincludedecimalpoint = String -> String
chomp (String -> String) -> String -> String
forall a b. (a -> b) -> a -> b
$ [String] -> String
unlines [
String
"Please include a decimal point or decimal comma in commodity directives,"
,String
"to help us parse correctly. It may be followed by zero or more decimal digits."
,String
"Examples:"
,String
"commodity $1000. ; no thousands mark, decimal period, no decimals"
,String
"commodity 1.234,00 ARS ; period at thousands, decimal comma, 2 decimals"
,String
"commodity EUR 1 000,000 ; space at thousands, decimal comma, 3 decimals"
,String
"commodity INR1,23,45,678.0 ; comma at thousands/lakhs/crores, decimal period, 1 decimal"
]
commoditydirectivemultilinep :: JournalParser m ()
commoditydirectivemultilinep :: JournalParser m ()
commoditydirectivemultilinep = do
Tokens Text
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Tokens Text)
forall e s (m :: * -> *).
MonadParsec e s m =>
Tokens s -> m (Tokens s)
string Tokens Text
"commodity"
ParsecT HledgerParseErrorData Text m () -> JournalParser m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces1
Text
sym <- ParsecT HledgerParseErrorData Text m Text
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m Text
forall (m :: * -> *). TextParser m Text
commoditysymbolp
Text
_ <- ParsecT HledgerParseErrorData Text m Text
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m Text
forall (m :: * -> *). TextParser m Text
followingcommentp
Maybe AmountStyle
mformat <- [AmountStyle] -> Maybe AmountStyle
forall a. [a] -> Maybe a
lastMay ([AmountStyle] -> Maybe AmountStyle)
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) [AmountStyle]
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Maybe AmountStyle)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> StateT Journal (ParsecT HledgerParseErrorData Text m) AmountStyle
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) [AmountStyle]
forall (m :: * -> *) a. MonadPlus m => m a -> m [a]
many (StateT Journal (ParsecT HledgerParseErrorData Text m) AmountStyle
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) AmountStyle
forall b.
StateT Journal (ParsecT HledgerParseErrorData Text m) b
-> StateT Journal (ParsecT HledgerParseErrorData Text m) b
indented (StateT Journal (ParsecT HledgerParseErrorData Text m) AmountStyle
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) AmountStyle)
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) AmountStyle
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) AmountStyle
forall a b. (a -> b) -> a -> b
$ Text
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) AmountStyle
forall (m :: * -> *). Text -> JournalParser m AmountStyle
formatdirectivep Text
sym)
let comm :: Commodity
comm = Commodity :: Text -> Maybe AmountStyle -> Commodity
Commodity{csymbol :: Text
csymbol=Text
sym, cformat :: Maybe AmountStyle
cformat=Maybe AmountStyle
mformat}
(Journal -> Journal) -> JournalParser m ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify' (\Journal
j -> Journal
j{jcommodities :: Map Text Commodity
jcommodities=Text -> Commodity -> Map Text Commodity -> Map Text Commodity
forall k a. Ord k => k -> a -> Map k a -> Map k a
M.insert Text
sym Commodity
comm (Map Text Commodity -> Map Text Commodity)
-> Map Text Commodity -> Map Text Commodity
forall a b. (a -> b) -> a -> b
$ Journal -> Map Text Commodity
jcommodities Journal
j})
where
indented :: StateT Journal (ParsecT HledgerParseErrorData Text m) b
-> StateT Journal (ParsecT HledgerParseErrorData Text m) b
indented = (ParsecT HledgerParseErrorData Text m () -> JournalParser m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces1 JournalParser m ()
-> StateT Journal (ParsecT HledgerParseErrorData Text m) b
-> StateT Journal (ParsecT HledgerParseErrorData Text m) b
forall (m :: * -> *) a b. Monad m => m a -> m b -> m b
>>)
formatdirectivep :: CommoditySymbol -> JournalParser m AmountStyle
formatdirectivep :: Text -> JournalParser m AmountStyle
formatdirectivep Text
expectedsym = do
Tokens Text
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Tokens Text)
forall e s (m :: * -> *).
MonadParsec e s m =>
Tokens s -> m (Tokens s)
string Tokens Text
"format"
ParsecT HledgerParseErrorData Text m ()
-> StateT Journal (ParsecT HledgerParseErrorData Text m) ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces1
Int
off <- StateT Journal (ParsecT HledgerParseErrorData Text m) Int
forall e s (m :: * -> *). MonadParsec e s m => m Int
getOffset
Amount{Text
acommodity :: Text
acommodity :: Amount -> Text
acommodity,AmountStyle
astyle :: AmountStyle
astyle :: Amount -> AmountStyle
astyle} <- JournalParser m Amount
forall (m :: * -> *). JournalParser m Amount
amountp
Text
_ <- ParsecT HledgerParseErrorData Text m Text
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m Text
forall (m :: * -> *). TextParser m Text
followingcommentp
if Text
acommodityText -> Text -> Bool
forall a. Eq a => a -> a -> Bool
==Text
expectedsym
then
if Maybe Char -> Bool
forall a. Maybe a -> Bool
isNothing (Maybe Char -> Bool) -> Maybe Char -> Bool
forall a b. (a -> b) -> a -> b
$ AmountStyle -> Maybe Char
asdecimalpoint AmountStyle
astyle
then HledgerParseErrorData -> JournalParser m AmountStyle
forall e s (m :: * -> *) a. MonadParsec e s m => e -> m a
customFailure (HledgerParseErrorData -> JournalParser m AmountStyle)
-> HledgerParseErrorData -> JournalParser m AmountStyle
forall a b. (a -> b) -> a -> b
$ Int -> String -> HledgerParseErrorData
parseErrorAt Int
off String
pleaseincludedecimalpoint
else AmountStyle -> JournalParser m AmountStyle
forall (m :: * -> *) a. Monad m => a -> m a
return (AmountStyle -> JournalParser m AmountStyle)
-> AmountStyle -> JournalParser m AmountStyle
forall a b. (a -> b) -> a -> b
$ String -> AmountStyle -> AmountStyle
forall a. Show a => String -> a -> a
dbg6 String
"style from format subdirective" AmountStyle
astyle
else HledgerParseErrorData -> JournalParser m AmountStyle
forall e s (m :: * -> *) a. MonadParsec e s m => e -> m a
customFailure (HledgerParseErrorData -> JournalParser m AmountStyle)
-> HledgerParseErrorData -> JournalParser m AmountStyle
forall a b. (a -> b) -> a -> b
$ Int -> String -> HledgerParseErrorData
parseErrorAt Int
off (String -> HledgerParseErrorData)
-> String -> HledgerParseErrorData
forall a b. (a -> b) -> a -> b
$
String -> Text -> Text -> String
forall r. PrintfType r => String -> r
printf String
"commodity directive symbol \"%s\" and format directive symbol \"%s\" should be the same" Text
expectedsym Text
acommodity
keywordp :: String -> JournalParser m ()
keywordp :: String -> JournalParser m ()
keywordp = (() ()
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
-> JournalParser m ()
forall (f :: * -> *) a b. Functor f => a -> f b -> f a
<$) (StateT Journal (ParsecT HledgerParseErrorData Text m) Text
-> JournalParser m ())
-> (String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text)
-> String
-> JournalParser m ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Text -> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall e s (m :: * -> *).
MonadParsec e s m =>
Tokens s -> m (Tokens s)
string (Text
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text)
-> (String -> Text)
-> String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall b c a. (b -> c) -> (a -> b) -> a -> c
. String -> Text
forall a. IsString a => String -> a
fromString
spacesp :: JournalParser m ()
spacesp :: JournalParser m ()
spacesp = () () -> JournalParser m () -> JournalParser m ()
forall (f :: * -> *) a b. Functor f => a -> f b -> f a
<$ ParsecT HledgerParseErrorData Text m () -> JournalParser m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces1
keywordsp :: String -> JournalParser m ()
keywordsp :: String -> JournalParser m ()
keywordsp = JournalParser m () -> JournalParser m ()
forall e s (m :: * -> *) a. MonadParsec e s m => m a -> m a
try (JournalParser m () -> JournalParser m ())
-> (String -> JournalParser m ()) -> String -> JournalParser m ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [JournalParser m ()] -> JournalParser m ()
forall (t :: * -> *) (m :: * -> *) a.
(Foldable t, Monad m) =>
t (m a) -> m ()
sequence_ ([JournalParser m ()] -> JournalParser m ())
-> (String -> [JournalParser m ()]) -> String -> JournalParser m ()
forall b c a. (b -> c) -> (a -> b) -> a -> c
. JournalParser m () -> [JournalParser m ()] -> [JournalParser m ()]
forall a. a -> [a] -> [a]
intersperse JournalParser m ()
forall (m :: * -> *). JournalParser m ()
spacesp ([JournalParser m ()] -> [JournalParser m ()])
-> (String -> [JournalParser m ()])
-> String
-> [JournalParser m ()]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (String -> JournalParser m ()) -> [String] -> [JournalParser m ()]
forall a b. (a -> b) -> [a] -> [b]
map String -> JournalParser m ()
forall (m :: * -> *). String -> JournalParser m ()
keywordp ([String] -> [JournalParser m ()])
-> (String -> [String]) -> String -> [JournalParser m ()]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. String -> [String]
words
applyaccountdirectivep :: JournalParser m ()
applyaccountdirectivep :: JournalParser m ()
applyaccountdirectivep = do
String -> JournalParser m ()
forall (m :: * -> *). String -> JournalParser m ()
keywordsp String
"apply account" JournalParser m () -> String -> JournalParser m ()
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"apply account directive"
ParsecT HledgerParseErrorData Text m () -> JournalParser m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces1
Text
parent <- ParsecT HledgerParseErrorData Text m Text
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m Text
forall (m :: * -> *). TextParser m Text
accountnamep
StateT Journal (ParsecT HledgerParseErrorData Text m) Char
forall e s (m :: * -> *).
(MonadParsec e s m, Token s ~ Char) =>
m (Token s)
newline
Text -> JournalParser m ()
forall (m :: * -> *). Text -> JournalParser m ()
pushParentAccount Text
parent
endapplyaccountdirectivep :: JournalParser m ()
endapplyaccountdirectivep :: JournalParser m ()
endapplyaccountdirectivep = do
String -> JournalParser m ()
forall (m :: * -> *). String -> JournalParser m ()
keywordsp String
"end apply account" JournalParser m () -> String -> JournalParser m ()
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"end apply account directive"
JournalParser m ()
forall (m :: * -> *). JournalParser m ()
popParentAccount
aliasdirectivep :: JournalParser m ()
aliasdirectivep :: JournalParser m ()
aliasdirectivep = do
Tokens Text
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Tokens Text)
forall e s (m :: * -> *).
MonadParsec e s m =>
Tokens s -> m (Tokens s)
string Tokens Text
"alias"
ParsecT HledgerParseErrorData Text m () -> JournalParser m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces1
AccountAlias
alias <- ParsecT HledgerParseErrorData Text m AccountAlias
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) AccountAlias
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m AccountAlias
forall (m :: * -> *). TextParser m AccountAlias
accountaliasp
AccountAlias -> JournalParser m ()
forall (m :: * -> *). MonadState Journal m => AccountAlias -> m ()
addAccountAlias AccountAlias
alias
endaliasesdirectivep :: JournalParser m ()
endaliasesdirectivep :: JournalParser m ()
endaliasesdirectivep = do
String -> JournalParser m ()
forall (m :: * -> *). String -> JournalParser m ()
keywordsp String
"end aliases" JournalParser m () -> String -> JournalParser m ()
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"end aliases directive"
JournalParser m ()
forall (m :: * -> *). MonadState Journal m => m ()
clearAccountAliases
tagdirectivep :: JournalParser m ()
tagdirectivep :: JournalParser m ()
tagdirectivep = do
Tokens Text
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Tokens Text)
forall e s (m :: * -> *).
MonadParsec e s m =>
Tokens s -> m (Tokens s)
string Tokens Text
"tag" StateT Journal (ParsecT HledgerParseErrorData Text m) Text
-> String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"tag directive"
ParsecT HledgerParseErrorData Text m () -> JournalParser m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces1
String
_ <- ParsecT HledgerParseErrorData Text m String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) String
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (ParsecT HledgerParseErrorData Text m String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) String)
-> ParsecT HledgerParseErrorData Text m String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) String
forall a b. (a -> b) -> a -> b
$ ParsecT HledgerParseErrorData Text m Char
-> ParsecT HledgerParseErrorData Text m String
forall (m :: * -> *) a. MonadPlus m => m a -> m [a]
some ParsecT HledgerParseErrorData Text m Char
forall (m :: * -> *). TextParser m Char
nonspace
ParsecT HledgerParseErrorData Text m String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) String
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m String
forall (m :: * -> *). TextParser m String
restofline
() -> JournalParser m ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()
endtagdirectivep :: JournalParser m ()
endtagdirectivep :: JournalParser m ()
endtagdirectivep = do
(String -> JournalParser m ()
forall (m :: * -> *). String -> JournalParser m ()
keywordsp String
"end tag" JournalParser m () -> JournalParser m () -> JournalParser m ()
forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|> String -> JournalParser m ()
forall (m :: * -> *). String -> JournalParser m ()
keywordp String
"pop") JournalParser m () -> String -> JournalParser m ()
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"end tag or pop directive"
ParsecT HledgerParseErrorData Text m String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) String
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m String
forall (m :: * -> *). TextParser m String
restofline
() -> JournalParser m ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()
payeedirectivep :: JournalParser m ()
payeedirectivep :: JournalParser m ()
payeedirectivep = do
Tokens Text
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Tokens Text)
forall e s (m :: * -> *).
MonadParsec e s m =>
Tokens s -> m (Tokens s)
string Tokens Text
"payee" StateT Journal (ParsecT HledgerParseErrorData Text m) Text
-> String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"payee directive"
ParsecT HledgerParseErrorData Text m () -> JournalParser m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces1
Text
payee <- ParsecT HledgerParseErrorData Text m Text
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (ParsecT HledgerParseErrorData Text m Text
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text)
-> ParsecT HledgerParseErrorData Text m Text
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall a b. (a -> b) -> a -> b
$ Text -> Text
T.strip (Text -> Text)
-> ParsecT HledgerParseErrorData Text m Text
-> ParsecT HledgerParseErrorData Text m Text
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> ParsecT HledgerParseErrorData Text m Text
forall (m :: * -> *). TextParser m Text
noncommenttext1p
(Text
comment, [Tag]
tags) <- ParsecT HledgerParseErrorData Text m (Text, [Tag])
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Text, [Tag])
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m (Text, [Tag])
forall (m :: * -> *). TextParser m (Text, [Tag])
transactioncommentp
(Text, Text, [Tag]) -> JournalParser m ()
forall (m :: * -> *). (Text, Text, [Tag]) -> JournalParser m ()
addPayeeDeclaration (Text
payee, Text
comment, [Tag]
tags)
() -> JournalParser m ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()
defaultyeardirectivep :: JournalParser m ()
defaultyeardirectivep :: JournalParser m ()
defaultyeardirectivep = do
Token Text
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Token Text)
forall e s (m :: * -> *).
(MonadParsec e s m, Token s ~ Char) =>
Token s -> m (Token s)
char Char
Token Text
'Y' StateT Journal (ParsecT HledgerParseErrorData Text m) Char
-> String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Char
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"default year"
ParsecT HledgerParseErrorData Text m () -> JournalParser m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces
Year -> JournalParser m ()
forall (m :: * -> *). Year -> JournalParser m ()
setYear (Year -> JournalParser m ())
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Year
-> JournalParser m ()
forall (m :: * -> *) a b. Monad m => (a -> m b) -> m a -> m b
=<< ParsecT HledgerParseErrorData Text m Year
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Year
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m Year
forall (m :: * -> *). TextParser m Year
yearp
defaultcommoditydirectivep :: JournalParser m ()
defaultcommoditydirectivep :: JournalParser m ()
defaultcommoditydirectivep = do
Token Text
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Token Text)
forall e s (m :: * -> *).
(MonadParsec e s m, Token s ~ Char) =>
Token s -> m (Token s)
char Char
Token Text
'D' StateT Journal (ParsecT HledgerParseErrorData Text m) Char
-> String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Char
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"default commodity"
ParsecT HledgerParseErrorData Text m () -> JournalParser m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces1
Int
off <- StateT Journal (ParsecT HledgerParseErrorData Text m) Int
forall e s (m :: * -> *). MonadParsec e s m => m Int
getOffset
Amount{Text
acommodity :: Text
acommodity :: Amount -> Text
acommodity,AmountStyle
astyle :: AmountStyle
astyle :: Amount -> AmountStyle
astyle} <- JournalParser m Amount
forall (m :: * -> *). JournalParser m Amount
amountp
ParsecT HledgerParseErrorData Text m String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) String
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m String
forall (m :: * -> *). TextParser m String
restofline
if Maybe Char -> Bool
forall a. Maybe a -> Bool
isNothing (Maybe Char -> Bool) -> Maybe Char -> Bool
forall a b. (a -> b) -> a -> b
$ AmountStyle -> Maybe Char
asdecimalpoint AmountStyle
astyle
then HledgerParseErrorData -> JournalParser m ()
forall e s (m :: * -> *) a. MonadParsec e s m => e -> m a
customFailure (HledgerParseErrorData -> JournalParser m ())
-> HledgerParseErrorData -> JournalParser m ()
forall a b. (a -> b) -> a -> b
$ Int -> String -> HledgerParseErrorData
parseErrorAt Int
off String
pleaseincludedecimalpoint
else (Text, AmountStyle) -> JournalParser m ()
forall (m :: * -> *). (Text, AmountStyle) -> JournalParser m ()
setDefaultCommodityAndStyle (Text
acommodity, AmountStyle
astyle)
marketpricedirectivep :: JournalParser m PriceDirective
marketpricedirectivep :: JournalParser m PriceDirective
marketpricedirectivep = do
Token Text
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Token Text)
forall e s (m :: * -> *).
(MonadParsec e s m, Token s ~ Char) =>
Token s -> m (Token s)
char Char
Token Text
'P' StateT Journal (ParsecT HledgerParseErrorData Text m) Char
-> String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Char
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"market price"
ParsecT HledgerParseErrorData Text m ()
-> StateT Journal (ParsecT HledgerParseErrorData Text m) ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces
Day
date <- StateT Journal (ParsecT HledgerParseErrorData Text m) Day
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Day
forall e s (m :: * -> *) a. MonadParsec e s m => m a -> m a
try (do {LocalTime Day
d TimeOfDay
_ <- JournalParser m LocalTime
forall (m :: * -> *). JournalParser m LocalTime
datetimep; Day -> StateT Journal (ParsecT HledgerParseErrorData Text m) Day
forall (m :: * -> *) a. Monad m => a -> m a
return Day
d}) StateT Journal (ParsecT HledgerParseErrorData Text m) Day
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Day
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Day
forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|> StateT Journal (ParsecT HledgerParseErrorData Text m) Day
forall (m :: * -> *). JournalParser m Day
datep
ParsecT HledgerParseErrorData Text m ()
-> StateT Journal (ParsecT HledgerParseErrorData Text m) ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces1
Text
symbol <- ParsecT HledgerParseErrorData Text m Text
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m Text
forall (m :: * -> *). TextParser m Text
commoditysymbolp
ParsecT HledgerParseErrorData Text m ()
-> StateT Journal (ParsecT HledgerParseErrorData Text m) ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces
Amount
price <- JournalParser m Amount
forall (m :: * -> *). JournalParser m Amount
amountp
ParsecT HledgerParseErrorData Text m String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) String
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m String
forall (m :: * -> *). TextParser m String
restofline
PriceDirective -> JournalParser m PriceDirective
forall (m :: * -> *) a. Monad m => a -> m a
return (PriceDirective -> JournalParser m PriceDirective)
-> PriceDirective -> JournalParser m PriceDirective
forall a b. (a -> b) -> a -> b
$ Day -> Text -> Amount -> PriceDirective
PriceDirective Day
date Text
symbol Amount
price
ignoredpricecommoditydirectivep :: JournalParser m ()
ignoredpricecommoditydirectivep :: JournalParser m ()
ignoredpricecommoditydirectivep = do
Token Text
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Token Text)
forall e s (m :: * -> *).
(MonadParsec e s m, Token s ~ Char) =>
Token s -> m (Token s)
char Char
Token Text
'N' StateT Journal (ParsecT HledgerParseErrorData Text m) Char
-> String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Char
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"ignored-price commodity"
ParsecT HledgerParseErrorData Text m () -> JournalParser m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces1
ParsecT HledgerParseErrorData Text m Text
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m Text
forall (m :: * -> *). TextParser m Text
commoditysymbolp
ParsecT HledgerParseErrorData Text m String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) String
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m String
forall (m :: * -> *). TextParser m String
restofline
() -> JournalParser m ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()
commodityconversiondirectivep :: JournalParser m ()
commodityconversiondirectivep :: JournalParser m ()
commodityconversiondirectivep = do
Token Text
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Token Text)
forall e s (m :: * -> *).
(MonadParsec e s m, Token s ~ Char) =>
Token s -> m (Token s)
char Char
Token Text
'C' StateT Journal (ParsecT HledgerParseErrorData Text m) Char
-> String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Char
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"commodity conversion"
ParsecT HledgerParseErrorData Text m () -> JournalParser m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces1
JournalParser m Amount
forall (m :: * -> *). JournalParser m Amount
amountp
ParsecT HledgerParseErrorData Text m () -> JournalParser m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces
Token Text
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Token Text)
forall e s (m :: * -> *).
(MonadParsec e s m, Token s ~ Char) =>
Token s -> m (Token s)
char Char
Token Text
'='
ParsecT HledgerParseErrorData Text m () -> JournalParser m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces
JournalParser m Amount
forall (m :: * -> *). JournalParser m Amount
amountp
ParsecT HledgerParseErrorData Text m String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) String
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m String
forall (m :: * -> *). TextParser m String
restofline
() -> JournalParser m ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()
decimalmarkdirectivep :: JournalParser m ()
decimalmarkdirectivep :: JournalParser m ()
decimalmarkdirectivep = do
Tokens Text
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Tokens Text)
forall e s (m :: * -> *).
MonadParsec e s m =>
Tokens s -> m (Tokens s)
string Tokens Text
"decimal-mark" StateT Journal (ParsecT HledgerParseErrorData Text m) Text
-> String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"decimal mark"
ParsecT HledgerParseErrorData Text m () -> JournalParser m ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces1
Char
mark <- (Token Text -> Bool)
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Token Text)
forall e s (m :: * -> *).
MonadParsec e s m =>
(Token s -> Bool) -> m (Token s)
satisfy Char -> Bool
Token Text -> Bool
isDecimalMark
(Journal -> Journal) -> JournalParser m ()
forall s (m :: * -> *). MonadState s m => (s -> s) -> m ()
modify' ((Journal -> Journal) -> JournalParser m ())
-> (Journal -> Journal) -> JournalParser m ()
forall a b. (a -> b) -> a -> b
$ \Journal
j -> Journal
j{jparsedecimalmark :: Maybe Char
jparsedecimalmark=Char -> Maybe Char
forall a. a -> Maybe a
Just Char
mark}
ParsecT HledgerParseErrorData Text m String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) String
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m String
forall (m :: * -> *). TextParser m String
restofline
() -> JournalParser m ()
forall (m :: * -> *) a. Monad m => a -> m a
return ()
transactionmodifierp :: JournalParser m TransactionModifier
transactionmodifierp :: JournalParser m TransactionModifier
transactionmodifierp = do
Token Text
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Token Text)
forall e s (m :: * -> *).
(MonadParsec e s m, Token s ~ Char) =>
Token s -> m (Token s)
char Char
Token Text
'=' StateT Journal (ParsecT HledgerParseErrorData Text m) Char
-> String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Char
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"modifier transaction"
ParsecT HledgerParseErrorData Text m ()
-> StateT Journal (ParsecT HledgerParseErrorData Text m) ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces
Text
querytxt <- ParsecT HledgerParseErrorData Text m Text
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (ParsecT HledgerParseErrorData Text m Text
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text)
-> ParsecT HledgerParseErrorData Text m Text
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall a b. (a -> b) -> a -> b
$ Text -> Text
T.strip (Text -> Text)
-> ParsecT HledgerParseErrorData Text m Text
-> ParsecT HledgerParseErrorData Text m Text
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> ParsecT HledgerParseErrorData Text m Text
forall (m :: * -> *). TextParser m Text
descriptionp
(Text
_comment, [Tag]
_tags) <- ParsecT HledgerParseErrorData Text m (Text, [Tag])
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Text, [Tag])
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m (Text, [Tag])
forall (m :: * -> *). TextParser m (Text, [Tag])
transactioncommentp
[TMPostingRule]
postingrules <- Maybe Year -> JournalParser m [TMPostingRule]
forall (m :: * -> *). Maybe Year -> JournalParser m [TMPostingRule]
tmpostingrulesp Maybe Year
forall a. Maybe a
Nothing
TransactionModifier -> JournalParser m TransactionModifier
forall (m :: * -> *) a. Monad m => a -> m a
return (TransactionModifier -> JournalParser m TransactionModifier)
-> TransactionModifier -> JournalParser m TransactionModifier
forall a b. (a -> b) -> a -> b
$ Text -> [TMPostingRule] -> TransactionModifier
TransactionModifier Text
querytxt [TMPostingRule]
postingrules
periodictransactionp :: MonadIO m => JournalParser m PeriodicTransaction
periodictransactionp :: JournalParser m PeriodicTransaction
periodictransactionp = do
Token Text
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Token Text)
forall e s (m :: * -> *).
(MonadParsec e s m, Token s ~ Char) =>
Token s -> m (Token s)
char Char
Token Text
'~' StateT Journal (ParsecT HledgerParseErrorData Text m) Char
-> String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Char
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"periodic transaction"
ParsecT HledgerParseErrorData Text m ()
-> StateT Journal (ParsecT HledgerParseErrorData Text m) ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (ParsecT HledgerParseErrorData Text m ()
-> StateT Journal (ParsecT HledgerParseErrorData Text m) ())
-> ParsecT HledgerParseErrorData Text m ()
-> StateT Journal (ParsecT HledgerParseErrorData Text m) ()
forall a b. (a -> b) -> a -> b
$ ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces
Int
off <- StateT Journal (ParsecT HledgerParseErrorData Text m) Int
forall e s (m :: * -> *). MonadParsec e s m => m Int
getOffset
Day
today <- IO Day -> StateT Journal (ParsecT HledgerParseErrorData Text m) Day
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO IO Day
getCurrentDay
Maybe Year
mdefaultyear <- JournalParser m (Maybe Year)
forall (m :: * -> *). JournalParser m (Maybe Year)
getYear
let refdate :: Day
refdate = case Maybe Year
mdefaultyear of
Maybe Year
Nothing -> Day
today
Just Year
y -> Year -> Int -> Int -> Day
fromGregorian Year
y Int
1 Int
1
SourceExcerpt
periodExcerpt <- ParsecT HledgerParseErrorData Text m SourceExcerpt
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) SourceExcerpt
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (ParsecT HledgerParseErrorData Text m SourceExcerpt
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) SourceExcerpt)
-> ParsecT HledgerParseErrorData Text m SourceExcerpt
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) SourceExcerpt
forall a b. (a -> b) -> a -> b
$ ParsecT HledgerParseErrorData Text m Text
-> ParsecT HledgerParseErrorData Text m SourceExcerpt
forall (m :: * -> *) a.
MonadParsec HledgerParseErrorData Text m =>
m a -> m SourceExcerpt
excerpt_ (ParsecT HledgerParseErrorData Text m Text
-> ParsecT HledgerParseErrorData Text m SourceExcerpt)
-> ParsecT HledgerParseErrorData Text m Text
-> ParsecT HledgerParseErrorData Text m SourceExcerpt
forall a b. (a -> b) -> a -> b
$
(Char -> Bool) -> ParsecT HledgerParseErrorData Text m Text
forall (m :: * -> *). (Char -> Bool) -> TextParser m Text
singlespacedtextsatisfying1p (\Char
c -> Char
c Char -> Char -> Bool
forall a. Eq a => a -> a -> Bool
/= Char
';' Bool -> Bool -> Bool
&& Char
c Char -> Char -> Bool
forall a. Eq a => a -> a -> Bool
/= Char
'\n')
let periodtxt :: Text
periodtxt = Text -> Text
T.strip (Text -> Text) -> Text -> Text
forall a b. (a -> b) -> a -> b
$ SourceExcerpt -> Text
getExcerptText SourceExcerpt
periodExcerpt
(Interval
interval, DateSpan
span) <- ParsecT HledgerParseErrorData Text m (Interval, DateSpan)
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Interval, DateSpan)
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (ParsecT HledgerParseErrorData Text m (Interval, DateSpan)
-> StateT
Journal
(ParsecT HledgerParseErrorData Text m)
(Interval, DateSpan))
-> ParsecT HledgerParseErrorData Text m (Interval, DateSpan)
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Interval, DateSpan)
forall a b. (a -> b) -> a -> b
$ SourceExcerpt
-> ParsecT HledgerParseErrorData Text m (Interval, DateSpan)
-> ParsecT HledgerParseErrorData Text m (Interval, DateSpan)
forall (m :: * -> *) a.
Monad m =>
SourceExcerpt
-> ParsecT HledgerParseErrorData Text m a
-> ParsecT HledgerParseErrorData Text m a
reparseExcerpt SourceExcerpt
periodExcerpt (ParsecT HledgerParseErrorData Text m (Interval, DateSpan)
-> ParsecT HledgerParseErrorData Text m (Interval, DateSpan))
-> ParsecT HledgerParseErrorData Text m (Interval, DateSpan)
-> ParsecT HledgerParseErrorData Text m (Interval, DateSpan)
forall a b. (a -> b) -> a -> b
$ do
(Interval, DateSpan)
pexp <- Day -> ParsecT HledgerParseErrorData Text m (Interval, DateSpan)
forall (m :: * -> *). Day -> TextParser m (Interval, DateSpan)
periodexprp Day
refdate
ParsecT HledgerParseErrorData Text m ()
-> ParsecT HledgerParseErrorData Text m ()
-> ParsecT HledgerParseErrorData Text m ()
forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
(<|>) ParsecT HledgerParseErrorData Text m ()
forall e s (m :: * -> *). MonadParsec e s m => m ()
eof (ParsecT HledgerParseErrorData Text m ()
-> ParsecT HledgerParseErrorData Text m ())
-> ParsecT HledgerParseErrorData Text m ()
-> ParsecT HledgerParseErrorData Text m ()
forall a b. (a -> b) -> a -> b
$ do
Int
offset1 <- ParsecT HledgerParseErrorData Text m Int
forall e s (m :: * -> *). MonadParsec e s m => m Int
getOffset
ParsecT HledgerParseErrorData Text m Text
-> ParsecT HledgerParseErrorData Text m ()
forall (f :: * -> *) a. Functor f => f a -> f ()
void ParsecT HledgerParseErrorData Text m Text
forall e s (m :: * -> *). MonadParsec e s m => m (Tokens s)
takeRest
Int
offset2 <- ParsecT HledgerParseErrorData Text m Int
forall e s (m :: * -> *). MonadParsec e s m => m Int
getOffset
HledgerParseErrorData -> ParsecT HledgerParseErrorData Text m ()
forall e s (m :: * -> *) a. MonadParsec e s m => e -> m a
customFailure (HledgerParseErrorData -> ParsecT HledgerParseErrorData Text m ())
-> HledgerParseErrorData -> ParsecT HledgerParseErrorData Text m ()
forall a b. (a -> b) -> a -> b
$ Int -> Int -> String -> HledgerParseErrorData
parseErrorAtRegion Int
offset1 Int
offset2 (String -> HledgerParseErrorData)
-> String -> HledgerParseErrorData
forall a b. (a -> b) -> a -> b
$
String
"remainder of period expression cannot be parsed"
String -> String -> String
forall a. Semigroup a => a -> a -> a
<> String
"\nperhaps you need to terminate the period expression with a double space?"
String -> String -> String
forall a. Semigroup a => a -> a -> a
<> String
"\na double space is required between period expression and description/comment"
(Interval, DateSpan)
-> ParsecT HledgerParseErrorData Text m (Interval, DateSpan)
forall (f :: * -> *) a. Applicative f => a -> f a
pure (Interval, DateSpan)
pexp
case Interval -> DateSpan -> Text -> Maybe String
checkPeriodicTransactionStartDate Interval
interval DateSpan
span Text
periodtxt of
Just String
e -> HledgerParseErrorData
-> StateT Journal (ParsecT HledgerParseErrorData Text m) ()
forall e s (m :: * -> *) a. MonadParsec e s m => e -> m a
customFailure (HledgerParseErrorData
-> StateT Journal (ParsecT HledgerParseErrorData Text m) ())
-> HledgerParseErrorData
-> StateT Journal (ParsecT HledgerParseErrorData Text m) ()
forall a b. (a -> b) -> a -> b
$ Int -> String -> HledgerParseErrorData
parseErrorAt Int
off String
e
Maybe String
Nothing -> () -> StateT Journal (ParsecT HledgerParseErrorData Text m) ()
forall (f :: * -> *) a. Applicative f => a -> f a
pure ()
Status
status <- ParsecT HledgerParseErrorData Text m Status
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Status
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m Status
forall (m :: * -> *). TextParser m Status
statusp StateT Journal (ParsecT HledgerParseErrorData Text m) Status
-> String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Status
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"cleared status"
Text
code <- ParsecT HledgerParseErrorData Text m Text
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m Text
forall (m :: * -> *). TextParser m Text
codep StateT Journal (ParsecT HledgerParseErrorData Text m) Text
-> String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"transaction code"
Text
description <- ParsecT HledgerParseErrorData Text m Text
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (ParsecT HledgerParseErrorData Text m Text
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text)
-> ParsecT HledgerParseErrorData Text m Text
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall a b. (a -> b) -> a -> b
$ Text -> Text
T.strip (Text -> Text)
-> ParsecT HledgerParseErrorData Text m Text
-> ParsecT HledgerParseErrorData Text m Text
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> ParsecT HledgerParseErrorData Text m Text
forall (m :: * -> *). TextParser m Text
descriptionp
(Text
comment, [Tag]
tags) <- ParsecT HledgerParseErrorData Text m (Text, [Tag])
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Text, [Tag])
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m (Text, [Tag])
forall (m :: * -> *). TextParser m (Text, [Tag])
transactioncommentp
[Posting]
postings <- Maybe Year -> JournalParser m [Posting]
forall (m :: * -> *). Maybe Year -> JournalParser m [Posting]
postingsp (Year -> Maybe Year
forall a. a -> Maybe a
Just (Year -> Maybe Year) -> Year -> Maybe Year
forall a b. (a -> b) -> a -> b
$ (Year, Int, Int) -> Year
forall a b c. (a, b, c) -> a
first3 ((Year, Int, Int) -> Year) -> (Year, Int, Int) -> Year
forall a b. (a -> b) -> a -> b
$ Day -> (Year, Int, Int)
toGregorian Day
refdate)
PeriodicTransaction -> JournalParser m PeriodicTransaction
forall (m :: * -> *) a. Monad m => a -> m a
return (PeriodicTransaction -> JournalParser m PeriodicTransaction)
-> PeriodicTransaction -> JournalParser m PeriodicTransaction
forall a b. (a -> b) -> a -> b
$ PeriodicTransaction
nullperiodictransaction{
ptperiodexpr :: Text
ptperiodexpr=Text
periodtxt
,ptinterval :: Interval
ptinterval=Interval
interval
,ptspan :: DateSpan
ptspan=DateSpan
span
,ptstatus :: Status
ptstatus=Status
status
,ptcode :: Text
ptcode=Text
code
,ptdescription :: Text
ptdescription=Text
description
,ptcomment :: Text
ptcomment=Text
comment
,pttags :: [Tag]
pttags=[Tag]
tags
,ptpostings :: [Posting]
ptpostings=[Posting]
postings
}
transactionp :: JournalParser m Transaction
transactionp :: JournalParser m Transaction
transactionp = do
SourcePos
startpos <- StateT Journal (ParsecT HledgerParseErrorData Text m) SourcePos
forall s e (m :: * -> *).
(TraversableStream s, MonadParsec e s m) =>
m SourcePos
getSourcePos
Day
date <- JournalParser m Day
forall (m :: * -> *). JournalParser m Day
datep JournalParser m Day -> String -> JournalParser m Day
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"transaction"
Maybe Day
edate <- JournalParser m Day
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Maybe Day)
forall (f :: * -> *) a. Alternative f => f a -> f (Maybe a)
optional (ParsecT HledgerParseErrorData Text m Day -> JournalParser m Day
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (ParsecT HledgerParseErrorData Text m Day -> JournalParser m Day)
-> ParsecT HledgerParseErrorData Text m Day -> JournalParser m Day
forall a b. (a -> b) -> a -> b
$ Day -> ParsecT HledgerParseErrorData Text m Day
forall (m :: * -> *). Day -> TextParser m Day
secondarydatep Day
date) StateT Journal (ParsecT HledgerParseErrorData Text m) (Maybe Day)
-> String
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Maybe Day)
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"secondary date"
StateT Journal (ParsecT HledgerParseErrorData Text m) Char
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Char
forall e s (m :: * -> *) a. MonadParsec e s m => m a -> m a
lookAhead (ParsecT HledgerParseErrorData Text m Char
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Char
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m Char
forall s (m :: * -> *).
(Stream s, Char ~ Token s) =>
ParsecT HledgerParseErrorData s m Char
spacenonewline StateT Journal (ParsecT HledgerParseErrorData Text m) Char
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Char
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Char
forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|> StateT Journal (ParsecT HledgerParseErrorData Text m) Char
forall e s (m :: * -> *).
(MonadParsec e s m, Token s ~ Char) =>
m (Token s)
newline) StateT Journal (ParsecT HledgerParseErrorData Text m) Char
-> String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Char
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"whitespace or newline"
Status
status <- ParsecT HledgerParseErrorData Text m Status
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Status
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m Status
forall (m :: * -> *). TextParser m Status
statusp StateT Journal (ParsecT HledgerParseErrorData Text m) Status
-> String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Status
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"cleared status"
Text
code <- ParsecT HledgerParseErrorData Text m Text
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m Text
forall (m :: * -> *). TextParser m Text
codep StateT Journal (ParsecT HledgerParseErrorData Text m) Text
-> String
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"transaction code"
Text
description <- ParsecT HledgerParseErrorData Text m Text
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (ParsecT HledgerParseErrorData Text m Text
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text)
-> ParsecT HledgerParseErrorData Text m Text
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Text
forall a b. (a -> b) -> a -> b
$ Text -> Text
T.strip (Text -> Text)
-> ParsecT HledgerParseErrorData Text m Text
-> ParsecT HledgerParseErrorData Text m Text
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> ParsecT HledgerParseErrorData Text m Text
forall (m :: * -> *). TextParser m Text
descriptionp
(Text
comment, [Tag]
tags) <- ParsecT HledgerParseErrorData Text m (Text, [Tag])
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Text, [Tag])
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m (Text, [Tag])
forall (m :: * -> *). TextParser m (Text, [Tag])
transactioncommentp
let year :: Year
year = (Year, Int, Int) -> Year
forall a b c. (a, b, c) -> a
first3 ((Year, Int, Int) -> Year) -> (Year, Int, Int) -> Year
forall a b. (a -> b) -> a -> b
$ Day -> (Year, Int, Int)
toGregorian Day
date
[Posting]
postings <- Maybe Year -> JournalParser m [Posting]
forall (m :: * -> *). Maybe Year -> JournalParser m [Posting]
postingsp (Year -> Maybe Year
forall a. a -> Maybe a
Just Year
year)
SourcePos
endpos <- StateT Journal (ParsecT HledgerParseErrorData Text m) SourcePos
forall s e (m :: * -> *).
(TraversableStream s, MonadParsec e s m) =>
m SourcePos
getSourcePos
let sourcepos :: (SourcePos, SourcePos)
sourcepos = (SourcePos
startpos, SourcePos
endpos)
Transaction -> JournalParser m Transaction
forall (m :: * -> *) a. Monad m => a -> m a
return (Transaction -> JournalParser m Transaction)
-> Transaction -> JournalParser m Transaction
forall a b. (a -> b) -> a -> b
$ Transaction -> Transaction
txnTieKnot (Transaction -> Transaction) -> Transaction -> Transaction
forall a b. (a -> b) -> a -> b
$ Year
-> Text
-> (SourcePos, SourcePos)
-> Day
-> Maybe Day
-> Status
-> Text
-> Text
-> Text
-> [Tag]
-> [Posting]
-> Transaction
Transaction Year
0 Text
"" (SourcePos, SourcePos)
sourcepos Day
date Maybe Day
edate Status
status Text
code Text
description Text
comment [Tag]
tags [Posting]
postings
postingsp :: Maybe Year -> JournalParser m [Posting]
postingsp :: Maybe Year -> JournalParser m [Posting]
postingsp Maybe Year
mTransactionYear = StateT Journal (ParsecT HledgerParseErrorData Text m) Posting
-> JournalParser m [Posting]
forall (m :: * -> *) a. MonadPlus m => m a -> m [a]
many (Maybe Year
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Posting
forall (m :: * -> *). Maybe Year -> JournalParser m Posting
postingp Maybe Year
mTransactionYear) JournalParser m [Posting] -> String -> JournalParser m [Posting]
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"postings"
postingp :: Maybe Year -> JournalParser m Posting
postingp :: Maybe Year -> JournalParser m Posting
postingp = ((Posting, Bool) -> Posting)
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Posting, Bool)
-> JournalParser m Posting
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (Posting, Bool) -> Posting
forall a b. (a, b) -> a
fst (StateT
Journal (ParsecT HledgerParseErrorData Text m) (Posting, Bool)
-> JournalParser m Posting)
-> (Maybe Year
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Posting, Bool))
-> Maybe Year
-> JournalParser m Posting
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Bool
-> Maybe Year
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Posting, Bool)
forall (m :: * -> *).
Bool -> Maybe Year -> JournalParser m (Posting, Bool)
postingphelper Bool
False
tmpostingrulesp :: Maybe Year -> JournalParser m [TMPostingRule]
tmpostingrulesp :: Maybe Year -> JournalParser m [TMPostingRule]
tmpostingrulesp Maybe Year
mTransactionYear = StateT Journal (ParsecT HledgerParseErrorData Text m) TMPostingRule
-> JournalParser m [TMPostingRule]
forall (m :: * -> *) a. MonadPlus m => m a -> m [a]
many (Maybe Year
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) TMPostingRule
forall (m :: * -> *). Maybe Year -> JournalParser m TMPostingRule
tmpostingrulep Maybe Year
mTransactionYear) JournalParser m [TMPostingRule]
-> String -> JournalParser m [TMPostingRule]
forall e s (m :: * -> *) a.
MonadParsec e s m =>
m a -> String -> m a
<?> String
"posting rules"
tmpostingrulep :: Maybe Year -> JournalParser m TMPostingRule
tmpostingrulep :: Maybe Year -> JournalParser m TMPostingRule
tmpostingrulep = ((Posting, Bool) -> TMPostingRule)
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Posting, Bool)
-> JournalParser m TMPostingRule
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap ((Posting -> Bool -> TMPostingRule)
-> (Posting, Bool) -> TMPostingRule
forall a b c. (a -> b -> c) -> (a, b) -> c
uncurry Posting -> Bool -> TMPostingRule
TMPostingRule) (StateT
Journal (ParsecT HledgerParseErrorData Text m) (Posting, Bool)
-> JournalParser m TMPostingRule)
-> (Maybe Year
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Posting, Bool))
-> Maybe Year
-> JournalParser m TMPostingRule
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Bool
-> Maybe Year
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Posting, Bool)
forall (m :: * -> *).
Bool -> Maybe Year -> JournalParser m (Posting, Bool)
postingphelper Bool
True
postingphelper :: Bool -> Maybe Year -> JournalParser m (Posting, Bool)
postingphelper :: Bool -> Maybe Year -> JournalParser m (Posting, Bool)
postingphelper Bool
isPostingRule Maybe Year
mTransactionYear = do
(Status
status, Text
account) <- StateT
Journal (ParsecT HledgerParseErrorData Text m) (Status, Text)
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Status, Text)
forall e s (m :: * -> *) a. MonadParsec e s m => m a -> m a
try (StateT
Journal (ParsecT HledgerParseErrorData Text m) (Status, Text)
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Status, Text))
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Status, Text)
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Status, Text)
forall a b. (a -> b) -> a -> b
$ do
ParsecT HledgerParseErrorData Text m ()
-> StateT Journal (ParsecT HledgerParseErrorData Text m) ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces1
Status
status <- ParsecT HledgerParseErrorData Text m Status
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Status
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m Status
forall (m :: * -> *). TextParser m Status
statusp
ParsecT HledgerParseErrorData Text m ()
-> StateT Journal (ParsecT HledgerParseErrorData Text m) ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces
Text
account <- JournalParser m Text
forall (m :: * -> *). JournalParser m Text
modifiedaccountnamep
(Status, Text)
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Status, Text)
forall (m :: * -> *) a. Monad m => a -> m a
return (Status
status, Text
account)
let (PostingType
ptype, Text
account') = (Text -> PostingType
accountNamePostingType Text
account, Text -> Text
textUnbracket Text
account)
ParsecT HledgerParseErrorData Text m ()
-> StateT Journal (ParsecT HledgerParseErrorData Text m) ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces
Bool
mult <- if Bool
isPostingRule then StateT Journal (ParsecT HledgerParseErrorData Text m) Bool
multiplierp else Bool -> StateT Journal (ParsecT HledgerParseErrorData Text m) Bool
forall (f :: * -> *) a. Applicative f => a -> f a
pure Bool
False
Maybe Amount
amount <- StateT Journal (ParsecT HledgerParseErrorData Text m) Amount
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Maybe Amount)
forall (f :: * -> *) a. Alternative f => f a -> f (Maybe a)
optional (StateT Journal (ParsecT HledgerParseErrorData Text m) Amount
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Maybe Amount))
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Amount
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Maybe Amount)
forall a b. (a -> b) -> a -> b
$ Bool
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Amount
forall (m :: * -> *). Bool -> JournalParser m Amount
amountpwithmultiplier Bool
mult
ParsecT HledgerParseErrorData Text m ()
-> StateT Journal (ParsecT HledgerParseErrorData Text m) ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces
Maybe BalanceAssertion
massertion <- StateT
Journal (ParsecT HledgerParseErrorData Text m) BalanceAssertion
-> StateT
Journal
(ParsecT HledgerParseErrorData Text m)
(Maybe BalanceAssertion)
forall (f :: * -> *) a. Alternative f => f a -> f (Maybe a)
optional StateT
Journal (ParsecT HledgerParseErrorData Text m) BalanceAssertion
forall (m :: * -> *). JournalParser m BalanceAssertion
balanceassertionp
ParsecT HledgerParseErrorData Text m ()
-> StateT Journal (ParsecT HledgerParseErrorData Text m) ()
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text m ()
forall s (m :: * -> *).
(Stream s, Token s ~ Char) =>
ParsecT HledgerParseErrorData s m ()
skipNonNewlineSpaces
(Text
comment,[Tag]
tags,Maybe Day
mdate,Maybe Day
mdate2) <- ParsecT
HledgerParseErrorData Text m (Text, [Tag], Maybe Day, Maybe Day)
-> StateT
Journal
(ParsecT HledgerParseErrorData Text m)
(Text, [Tag], Maybe Day, Maybe Day)
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (ParsecT
HledgerParseErrorData Text m (Text, [Tag], Maybe Day, Maybe Day)
-> StateT
Journal
(ParsecT HledgerParseErrorData Text m)
(Text, [Tag], Maybe Day, Maybe Day))
-> ParsecT
HledgerParseErrorData Text m (Text, [Tag], Maybe Day, Maybe Day)
-> StateT
Journal
(ParsecT HledgerParseErrorData Text m)
(Text, [Tag], Maybe Day, Maybe Day)
forall a b. (a -> b) -> a -> b
$ Maybe Year
-> ParsecT
HledgerParseErrorData Text m (Text, [Tag], Maybe Day, Maybe Day)
forall (m :: * -> *).
Maybe Year -> TextParser m (Text, [Tag], Maybe Day, Maybe Day)
postingcommentp Maybe Year
mTransactionYear
let p :: Posting
p = Posting
posting
{ pdate :: Maybe Day
pdate=Maybe Day
mdate
, pdate2 :: Maybe Day
pdate2=Maybe Day
mdate2
, pstatus :: Status
pstatus=Status
status
, paccount :: Text
paccount=Text
account'
, pamount :: MixedAmount
pamount=MixedAmount
-> (Amount -> MixedAmount) -> Maybe Amount -> MixedAmount
forall b a. b -> (a -> b) -> Maybe a -> b
maybe MixedAmount
missingmixedamt Amount -> MixedAmount
mixedAmount Maybe Amount
amount
, pcomment :: Text
pcomment=Text
comment
, ptype :: PostingType
ptype=PostingType
ptype
, ptags :: [Tag]
ptags=[Tag]
tags
, pbalanceassertion :: Maybe BalanceAssertion
pbalanceassertion=Maybe BalanceAssertion
massertion
}
(Posting, Bool) -> JournalParser m (Posting, Bool)
forall (m :: * -> *) a. Monad m => a -> m a
return (Posting
p, Bool
mult)
where
multiplierp :: StateT Journal (ParsecT HledgerParseErrorData Text m) Bool
multiplierp = Bool
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Bool
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Bool
forall (m :: * -> *) a. Alternative m => a -> m a -> m a
option Bool
False (StateT Journal (ParsecT HledgerParseErrorData Text m) Bool
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Bool)
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Bool
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Bool
forall a b. (a -> b) -> a -> b
$ Bool
True Bool
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Char
-> StateT Journal (ParsecT HledgerParseErrorData Text m) Bool
forall (f :: * -> *) a b. Functor f => a -> f b -> f a
<$ Token Text
-> StateT
Journal (ParsecT HledgerParseErrorData Text m) (Token Text)
forall e s (m :: * -> *).
(MonadParsec e s m, Token s ~ Char) =>
Token s -> m (Token s)
char Char
Token Text
'*'
tests_JournalReader :: TestTree
tests_JournalReader = String -> [TestTree] -> TestTree
testGroup String
"JournalReader" [
let p :: JournalParser IO Text
p = ParsecT HledgerParseErrorData Text IO Text -> JournalParser IO Text
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift ParsecT HledgerParseErrorData Text IO Text
forall (m :: * -> *). TextParser m Text
accountnamep :: JournalParser IO AccountName in
String -> [TestTree] -> TestTree
testGroup String
"accountnamep" [
String -> Assertion -> TestTree
testCase String
"basic" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ JournalParser IO Text -> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse JournalParser IO Text
p Text
"a:b:c"
]
,String -> [TestTree] -> TestTree
testGroup String
"datep" [
String -> Assertion -> TestTree
testCase String
"YYYY/MM/DD" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) Day
-> Text -> Day -> Assertion
forall a st.
(HasCallStack, Eq a, Show a, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> a -> Assertion
assertParseEq StateT Journal (ParsecT HledgerParseErrorData Text IO) Day
forall (m :: * -> *). JournalParser m Day
datep Text
"2018/01/01" (Year -> Int -> Int -> Day
fromGregorian Year
2018 Int
1 Int
1)
,String -> Assertion -> TestTree
testCase String
"YYYY-MM-DD" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) Day
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse StateT Journal (ParsecT HledgerParseErrorData Text IO) Day
forall (m :: * -> *). JournalParser m Day
datep Text
"2018-01-01"
,String -> Assertion -> TestTree
testCase String
"YYYY.MM.DD" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) Day
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse StateT Journal (ParsecT HledgerParseErrorData Text IO) Day
forall (m :: * -> *). JournalParser m Day
datep Text
"2018.01.01"
,String -> Assertion -> TestTree
testCase String
"yearless date with no default year" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) Day
-> Text -> String -> Assertion
forall a st.
(HasCallStack, Eq a, Show a, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> String -> Assertion
assertParseError StateT Journal (ParsecT HledgerParseErrorData Text IO) Day
forall (m :: * -> *). JournalParser m Day
datep Text
"1/1" String
"current year is unknown"
,String -> Assertion -> TestTree
testCase String
"yearless date with default year" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ do
let s :: Text
s = Text
"1/1"
Either HledgerParseErrors Day
ep <- Journal
-> StateT Journal (ParsecT HledgerParseErrorData Text IO) Day
-> Text
-> IO (Either HledgerParseErrors Day)
forall (m :: * -> *) st a.
Monad m =>
st
-> StateT st (ParsecT HledgerParseErrorData Text m) a
-> Text
-> m (Either HledgerParseErrors a)
parseWithState Journal
nulljournal{jparsedefaultyear :: Maybe Year
jparsedefaultyear=Year -> Maybe Year
forall a. a -> Maybe a
Just Year
2018} StateT Journal (ParsecT HledgerParseErrorData Text IO) Day
forall (m :: * -> *). JournalParser m Day
datep Text
s
(HledgerParseErrors -> Assertion)
-> (Day -> Assertion) -> Either HledgerParseErrors Day -> Assertion
forall a c b. (a -> c) -> (b -> c) -> Either a b -> c
either (String -> Assertion
forall a. HasCallStack => String -> IO a
assertFailure (String -> Assertion)
-> (HledgerParseErrors -> String)
-> HledgerParseErrors
-> Assertion
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (String
"parse error at "String -> String -> String
forall a. [a] -> [a] -> [a]
++) (String -> String)
-> (HledgerParseErrors -> String) -> HledgerParseErrors -> String
forall b c a. (b -> c) -> (a -> b) -> a -> c
. HledgerParseErrors -> String
customErrorBundlePretty) (Assertion -> Day -> Assertion
forall a b. a -> b -> a
const (Assertion -> Day -> Assertion) -> Assertion -> Day -> Assertion
forall a b. (a -> b) -> a -> b
$ () -> Assertion
forall (m :: * -> *) a. Monad m => a -> m a
return ()) Either HledgerParseErrors Day
ep
,String -> Assertion -> TestTree
testCase String
"no leading zero" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) Day
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse StateT Journal (ParsecT HledgerParseErrorData Text IO) Day
forall (m :: * -> *). JournalParser m Day
datep Text
"2018/1/1"
]
,String -> Assertion -> TestTree
testCase String
"datetimep" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ do
let
good :: Text -> Assertion
good = StateT Journal (ParsecT HledgerParseErrorData Text IO) LocalTime
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse StateT Journal (ParsecT HledgerParseErrorData Text IO) LocalTime
forall (m :: * -> *). JournalParser m LocalTime
datetimep
bad :: Text -> Assertion
bad = (\Text
t -> StateT Journal (ParsecT HledgerParseErrorData Text IO) LocalTime
-> Text -> String -> Assertion
forall a st.
(HasCallStack, Eq a, Show a, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> String -> Assertion
assertParseError StateT Journal (ParsecT HledgerParseErrorData Text IO) LocalTime
forall (m :: * -> *). JournalParser m LocalTime
datetimep Text
t String
"")
Text -> Assertion
good Text
"2011/1/1 00:00"
Text -> Assertion
good Text
"2011/1/1 23:59:59"
Text -> Assertion
bad Text
"2011/1/1"
Text -> Assertion
bad Text
"2011/1/1 24:00:00"
Text -> Assertion
bad Text
"2011/1/1 00:60:00"
Text -> Assertion
bad Text
"2011/1/1 00:00:60"
Text -> Assertion
bad Text
"2011/1/1 3:5:7"
let t :: LocalTime
t = Day -> TimeOfDay -> LocalTime
LocalTime (Year -> Int -> Int -> Day
fromGregorian Year
2018 Int
1 Int
1) (Int -> Int -> Pico -> TimeOfDay
TimeOfDay Int
0 Int
0 Pico
0)
StateT Journal (ParsecT HledgerParseErrorData Text IO) LocalTime
-> Text -> LocalTime -> Assertion
forall a st.
(HasCallStack, Eq a, Show a, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> a -> Assertion
assertParseEq StateT Journal (ParsecT HledgerParseErrorData Text IO) LocalTime
forall (m :: * -> *). JournalParser m LocalTime
datetimep Text
"2018/1/1 00:00-0800" LocalTime
t
StateT Journal (ParsecT HledgerParseErrorData Text IO) LocalTime
-> Text -> LocalTime -> Assertion
forall a st.
(HasCallStack, Eq a, Show a, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> a -> Assertion
assertParseEq StateT Journal (ParsecT HledgerParseErrorData Text IO) LocalTime
forall (m :: * -> *). JournalParser m LocalTime
datetimep Text
"2018/1/1 00:00+1234" LocalTime
t
,String -> [TestTree] -> TestTree
testGroup String
"periodictransactionp" [
String -> Assertion -> TestTree
testCase String
"more period text in comment after one space" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT
Journal (ParsecT HledgerParseErrorData Text IO) PeriodicTransaction
-> Text -> PeriodicTransaction -> Assertion
forall a st.
(HasCallStack, Eq a, Show a, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> a -> Assertion
assertParseEq StateT
Journal (ParsecT HledgerParseErrorData Text IO) PeriodicTransaction
forall (m :: * -> *).
MonadIO m =>
JournalParser m PeriodicTransaction
periodictransactionp
Text
"~ monthly from 2018/6 ;In 2019 we will change this\n"
PeriodicTransaction
nullperiodictransaction {
ptperiodexpr :: Text
ptperiodexpr = Text
"monthly from 2018/6"
,ptinterval :: Interval
ptinterval = Int -> Interval
Months Int
1
,ptspan :: DateSpan
ptspan = Maybe Day -> Maybe Day -> DateSpan
DateSpan (Day -> Maybe Day
forall a. a -> Maybe a
Just (Day -> Maybe Day) -> Day -> Maybe Day
forall a b. (a -> b) -> a -> b
$ Year -> Int -> Int -> Day
fromGregorian Year
2018 Int
6 Int
1) Maybe Day
forall a. Maybe a
Nothing
,ptdescription :: Text
ptdescription = Text
""
,ptcomment :: Text
ptcomment = Text
"In 2019 we will change this\n"
}
,String -> Assertion -> TestTree
testCase String
"more period text in description after two spaces" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT
Journal (ParsecT HledgerParseErrorData Text IO) PeriodicTransaction
-> Text -> PeriodicTransaction -> Assertion
forall a st.
(HasCallStack, Eq a, Show a, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> a -> Assertion
assertParseEq StateT
Journal (ParsecT HledgerParseErrorData Text IO) PeriodicTransaction
forall (m :: * -> *).
MonadIO m =>
JournalParser m PeriodicTransaction
periodictransactionp
Text
"~ monthly from 2018/6 In 2019 we will change this\n"
PeriodicTransaction
nullperiodictransaction {
ptperiodexpr :: Text
ptperiodexpr = Text
"monthly from 2018/6"
,ptinterval :: Interval
ptinterval = Int -> Interval
Months Int
1
,ptspan :: DateSpan
ptspan = Maybe Day -> Maybe Day -> DateSpan
DateSpan (Day -> Maybe Day
forall a. a -> Maybe a
Just (Day -> Maybe Day) -> Day -> Maybe Day
forall a b. (a -> b) -> a -> b
$ Year -> Int -> Int -> Day
fromGregorian Year
2018 Int
6 Int
1) Maybe Day
forall a. Maybe a
Nothing
,ptdescription :: Text
ptdescription = Text
"In 2019 we will change this"
,ptcomment :: Text
ptcomment = Text
""
}
,String -> Assertion -> TestTree
testCase String
"Next year in description" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT
Journal (ParsecT HledgerParseErrorData Text IO) PeriodicTransaction
-> Text -> PeriodicTransaction -> Assertion
forall a st.
(HasCallStack, Eq a, Show a, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> a -> Assertion
assertParseEq StateT
Journal (ParsecT HledgerParseErrorData Text IO) PeriodicTransaction
forall (m :: * -> *).
MonadIO m =>
JournalParser m PeriodicTransaction
periodictransactionp
Text
"~ monthly Next year blah blah\n"
PeriodicTransaction
nullperiodictransaction {
ptperiodexpr :: Text
ptperiodexpr = Text
"monthly"
,ptinterval :: Interval
ptinterval = Int -> Interval
Months Int
1
,ptspan :: DateSpan
ptspan = Maybe Day -> Maybe Day -> DateSpan
DateSpan Maybe Day
forall a. Maybe a
Nothing Maybe Day
forall a. Maybe a
Nothing
,ptdescription :: Text
ptdescription = Text
"Next year blah blah"
,ptcomment :: Text
ptcomment = Text
""
}
,String -> Assertion -> TestTree
testCase String
"Just date, no description" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT
Journal (ParsecT HledgerParseErrorData Text IO) PeriodicTransaction
-> Text -> PeriodicTransaction -> Assertion
forall a st.
(HasCallStack, Eq a, Show a, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> a -> Assertion
assertParseEq StateT
Journal (ParsecT HledgerParseErrorData Text IO) PeriodicTransaction
forall (m :: * -> *).
MonadIO m =>
JournalParser m PeriodicTransaction
periodictransactionp
Text
"~ 2019-01-04\n"
PeriodicTransaction
nullperiodictransaction {
ptperiodexpr :: Text
ptperiodexpr = Text
"2019-01-04"
,ptinterval :: Interval
ptinterval = Interval
NoInterval
,ptspan :: DateSpan
ptspan = Maybe Day -> Maybe Day -> DateSpan
DateSpan (Day -> Maybe Day
forall a. a -> Maybe a
Just (Day -> Maybe Day) -> Day -> Maybe Day
forall a b. (a -> b) -> a -> b
$ Year -> Int -> Int -> Day
fromGregorian Year
2019 Int
1 Int
4) (Day -> Maybe Day
forall a. a -> Maybe a
Just (Day -> Maybe Day) -> Day -> Maybe Day
forall a b. (a -> b) -> a -> b
$ Year -> Int -> Int -> Day
fromGregorian Year
2019 Int
1 Int
5)
,ptdescription :: Text
ptdescription = Text
""
,ptcomment :: Text
ptcomment = Text
""
}
,String -> Assertion -> TestTree
testCase String
"Just date, no description + empty transaction comment" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT
Journal (ParsecT HledgerParseErrorData Text IO) PeriodicTransaction
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse StateT
Journal (ParsecT HledgerParseErrorData Text IO) PeriodicTransaction
forall (m :: * -> *).
MonadIO m =>
JournalParser m PeriodicTransaction
periodictransactionp
Text
"~ 2019-01-04\n ;\n a 1\n b\n"
]
,String -> [TestTree] -> TestTree
testGroup String
"postingp" [
String -> Assertion -> TestTree
testCase String
"basic" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
-> Text -> Posting -> Assertion
forall a st.
(HasCallStack, Eq a, Show a, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> a -> Assertion
assertParseEq (Maybe Year
-> StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
forall (m :: * -> *). Maybe Year -> JournalParser m Posting
postingp Maybe Year
forall a. Maybe a
Nothing)
Text
" expenses:food:dining $10.00 ; a: a a \n ; b: b b \n"
Posting
posting{
paccount :: Text
paccount=Text
"expenses:food:dining",
pamount :: MixedAmount
pamount=Amount -> MixedAmount
mixedAmount (DecimalRaw Year -> Amount
usd DecimalRaw Year
10),
pcomment :: Text
pcomment=Text
"a: a a\nb: b b\n",
ptags :: [Tag]
ptags=[(Text
"a",Text
"a a"), (Text
"b",Text
"b b")]
}
,String -> Assertion -> TestTree
testCase String
"posting dates" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
-> Text -> Posting -> Assertion
forall a st.
(HasCallStack, Eq a, Show a, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> a -> Assertion
assertParseEq (Maybe Year
-> StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
forall (m :: * -> *). Maybe Year -> JournalParser m Posting
postingp Maybe Year
forall a. Maybe a
Nothing)
Text
" a 1. ; date:2012/11/28, date2=2012/11/29,b:b\n"
Posting
nullposting{
paccount :: Text
paccount=Text
"a"
,pamount :: MixedAmount
pamount=Amount -> MixedAmount
mixedAmount (DecimalRaw Year -> Amount
num DecimalRaw Year
1)
,pcomment :: Text
pcomment=Text
"date:2012/11/28, date2=2012/11/29,b:b\n"
,ptags :: [Tag]
ptags=[(Text
"date", Text
"2012/11/28"), (Text
"date2=2012/11/29,b", Text
"b")]
,pdate :: Maybe Day
pdate=Day -> Maybe Day
forall a. a -> Maybe a
Just (Day -> Maybe Day) -> Day -> Maybe Day
forall a b. (a -> b) -> a -> b
$ Year -> Int -> Int -> Day
fromGregorian Year
2012 Int
11 Int
28
,pdate2 :: Maybe Day
pdate2=Maybe Day
forall a. Maybe a
Nothing
}
,String -> Assertion -> TestTree
testCase String
"posting dates bracket syntax" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
-> Text -> Posting -> Assertion
forall a st.
(HasCallStack, Eq a, Show a, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> a -> Assertion
assertParseEq (Maybe Year
-> StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
forall (m :: * -> *). Maybe Year -> JournalParser m Posting
postingp Maybe Year
forall a. Maybe a
Nothing)
Text
" a 1. ; [2012/11/28=2012/11/29]\n"
Posting
nullposting{
paccount :: Text
paccount=Text
"a"
,pamount :: MixedAmount
pamount=Amount -> MixedAmount
mixedAmount (DecimalRaw Year -> Amount
num DecimalRaw Year
1)
,pcomment :: Text
pcomment=Text
"[2012/11/28=2012/11/29]\n"
,ptags :: [Tag]
ptags=[]
,pdate :: Maybe Day
pdate= Day -> Maybe Day
forall a. a -> Maybe a
Just (Day -> Maybe Day) -> Day -> Maybe Day
forall a b. (a -> b) -> a -> b
$ Year -> Int -> Int -> Day
fromGregorian Year
2012 Int
11 Int
28
,pdate2 :: Maybe Day
pdate2=Day -> Maybe Day
forall a. a -> Maybe a
Just (Day -> Maybe Day) -> Day -> Maybe Day
forall a b. (a -> b) -> a -> b
$ Year -> Int -> Int -> Day
fromGregorian Year
2012 Int
11 Int
29
}
,String -> Assertion -> TestTree
testCase String
"quoted commodity symbol with digits" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse (Maybe Year
-> StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
forall (m :: * -> *). Maybe Year -> JournalParser m Posting
postingp Maybe Year
forall a. Maybe a
Nothing) Text
" a 1 \"DE123\"\n"
,String -> Assertion -> TestTree
testCase String
"only lot price" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse (Maybe Year
-> StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
forall (m :: * -> *). Maybe Year -> JournalParser m Posting
postingp Maybe Year
forall a. Maybe a
Nothing) Text
" a 1A {1B}\n"
,String -> Assertion -> TestTree
testCase String
"fixed lot price" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse (Maybe Year
-> StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
forall (m :: * -> *). Maybe Year -> JournalParser m Posting
postingp Maybe Year
forall a. Maybe a
Nothing) Text
" a 1A {=1B}\n"
,String -> Assertion -> TestTree
testCase String
"total lot price" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse (Maybe Year
-> StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
forall (m :: * -> *). Maybe Year -> JournalParser m Posting
postingp Maybe Year
forall a. Maybe a
Nothing) Text
" a 1A {{1B}}\n"
,String -> Assertion -> TestTree
testCase String
"fixed total lot price, and spaces" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse (Maybe Year
-> StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
forall (m :: * -> *). Maybe Year -> JournalParser m Posting
postingp Maybe Year
forall a. Maybe a
Nothing) Text
" a 1A {{ = 1B }}\n"
,String -> Assertion -> TestTree
testCase String
"lot price before transaction price" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse (Maybe Year
-> StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
forall (m :: * -> *). Maybe Year -> JournalParser m Posting
postingp Maybe Year
forall a. Maybe a
Nothing) Text
" a 1A {1B} @ 1B\n"
,String -> Assertion -> TestTree
testCase String
"lot price after transaction price" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse (Maybe Year
-> StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
forall (m :: * -> *). Maybe Year -> JournalParser m Posting
postingp Maybe Year
forall a. Maybe a
Nothing) Text
" a 1A @ 1B {1B}\n"
,String -> Assertion -> TestTree
testCase String
"lot price after balance assertion not allowed" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
-> Text -> String -> Assertion
forall a st.
(HasCallStack, Eq a, Show a, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> String -> Assertion
assertParseError (Maybe Year
-> StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
forall (m :: * -> *). Maybe Year -> JournalParser m Posting
postingp Maybe Year
forall a. Maybe a
Nothing) Text
" a 1A @ 1B = 1A {1B}\n" String
"unexpected '{'"
,String -> Assertion -> TestTree
testCase String
"only lot date" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse (Maybe Year
-> StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
forall (m :: * -> *). Maybe Year -> JournalParser m Posting
postingp Maybe Year
forall a. Maybe a
Nothing) Text
" a 1A [2000-01-01]\n"
,String -> Assertion -> TestTree
testCase String
"transaction price, lot price, lot date" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse (Maybe Year
-> StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
forall (m :: * -> *). Maybe Year -> JournalParser m Posting
postingp Maybe Year
forall a. Maybe a
Nothing) Text
" a 1A @ 1B {1B} [2000-01-01]\n"
,String -> Assertion -> TestTree
testCase String
"lot date, lot price, transaction price" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse (Maybe Year
-> StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
forall (m :: * -> *). Maybe Year -> JournalParser m Posting
postingp Maybe Year
forall a. Maybe a
Nothing) Text
" a 1A [2000-01-01] {1B} @ 1B\n"
,String -> Assertion -> TestTree
testCase String
"balance assertion over entire contents of account" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse (Maybe Year
-> StateT Journal (ParsecT HledgerParseErrorData Text IO) Posting
forall (m :: * -> *). Maybe Year -> JournalParser m Posting
postingp Maybe Year
forall a. Maybe a
Nothing) Text
" a $1 == $1\n"
]
,String -> [TestTree] -> TestTree
testGroup String
"transactionmodifierp" [
String -> Assertion -> TestTree
testCase String
"basic" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT
Journal (ParsecT HledgerParseErrorData Text IO) TransactionModifier
-> Text -> TransactionModifier -> Assertion
forall a st.
(HasCallStack, Eq a, Show a, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> a -> Assertion
assertParseEq StateT
Journal (ParsecT HledgerParseErrorData Text IO) TransactionModifier
forall (m :: * -> *). JournalParser m TransactionModifier
transactionmodifierp
Text
"= (some value expr)\n some:postings 1.\n"
TransactionModifier
nulltransactionmodifier {
tmquerytxt :: Text
tmquerytxt = Text
"(some value expr)"
,tmpostingrules :: [TMPostingRule]
tmpostingrules = [Posting -> Bool -> TMPostingRule
TMPostingRule Posting
nullposting{paccount :: Text
paccount=Text
"some:postings", pamount :: MixedAmount
pamount=Amount -> MixedAmount
mixedAmount (DecimalRaw Year -> Amount
num DecimalRaw Year
1)} Bool
False]
}
]
,String -> [TestTree] -> TestTree
testGroup String
"transactionp" [
String -> Assertion -> TestTree
testCase String
"just a date" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) Transaction
-> Text -> Transaction -> Assertion
forall a st.
(HasCallStack, Eq a, Show a, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> a -> Assertion
assertParseEq StateT Journal (ParsecT HledgerParseErrorData Text IO) Transaction
forall (m :: * -> *). JournalParser m Transaction
transactionp Text
"2015/1/1\n" Transaction
nulltransaction{tdate :: Day
tdate=Year -> Int -> Int -> Day
fromGregorian Year
2015 Int
1 Int
1}
,String -> Assertion -> TestTree
testCase String
"more complex" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) Transaction
-> Text -> Transaction -> Assertion
forall a st.
(HasCallStack, Eq a, Show a, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> a -> Assertion
assertParseEq StateT Journal (ParsecT HledgerParseErrorData Text IO) Transaction
forall (m :: * -> *). JournalParser m Transaction
transactionp
([Text] -> Text
T.unlines [
Text
"2012/05/14=2012/05/15 (code) desc ; tcomment1",
Text
" ; tcomment2",
Text
" ; ttag1: val1",
Text
" * a $1.00 ; pcomment1",
Text
" ; pcomment2",
Text
" ; ptag1: val1",
Text
" ; ptag2: val2"
])
Transaction
nulltransaction{
tsourcepos :: (SourcePos, SourcePos)
tsourcepos=(String -> Pos -> Pos -> SourcePos
SourcePos String
"" (Int -> Pos
mkPos Int
1) (Int -> Pos
mkPos Int
1), String -> Pos -> Pos -> SourcePos
SourcePos String
"" (Int -> Pos
mkPos Int
8) (Int -> Pos
mkPos Int
1)),
tprecedingcomment :: Text
tprecedingcomment=Text
"",
tdate :: Day
tdate=Year -> Int -> Int -> Day
fromGregorian Year
2012 Int
5 Int
14,
tdate2 :: Maybe Day
tdate2=Day -> Maybe Day
forall a. a -> Maybe a
Just (Day -> Maybe Day) -> Day -> Maybe Day
forall a b. (a -> b) -> a -> b
$ Year -> Int -> Int -> Day
fromGregorian Year
2012 Int
5 Int
15,
tstatus :: Status
tstatus=Status
Unmarked,
tcode :: Text
tcode=Text
"code",
tdescription :: Text
tdescription=Text
"desc",
tcomment :: Text
tcomment=Text
"tcomment1\ntcomment2\nttag1: val1\n",
ttags :: [Tag]
ttags=[(Text
"ttag1",Text
"val1")],
tpostings :: [Posting]
tpostings=[
Posting
nullposting{
pdate :: Maybe Day
pdate=Maybe Day
forall a. Maybe a
Nothing,
pstatus :: Status
pstatus=Status
Cleared,
paccount :: Text
paccount=Text
"a",
pamount :: MixedAmount
pamount=Amount -> MixedAmount
mixedAmount (DecimalRaw Year -> Amount
usd DecimalRaw Year
1),
pcomment :: Text
pcomment=Text
"pcomment1\npcomment2\nptag1: val1\nptag2: val2\n",
ptype :: PostingType
ptype=PostingType
RegularPosting,
ptags :: [Tag]
ptags=[(Text
"ptag1",Text
"val1"),(Text
"ptag2",Text
"val2")],
ptransaction :: Maybe Transaction
ptransaction=Maybe Transaction
forall a. Maybe a
Nothing
}
]
}
,String -> Assertion -> TestTree
testCase String
"parses a well-formed transaction" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$
HasCallStack => String -> Bool -> Assertion
String -> Bool -> Assertion
assertBool String
"" (Bool -> Assertion) -> Bool -> Assertion
forall a b. (a -> b) -> a -> b
$ Either Any (Either HledgerParseErrors Transaction) -> Bool
forall a b. Either a b -> Bool
isRight (Either Any (Either HledgerParseErrors Transaction) -> Bool)
-> Either Any (Either HledgerParseErrors Transaction) -> Bool
forall a b. (a -> b) -> a -> b
$ JournalParser (Either Any) Transaction
-> Text -> Either Any (Either HledgerParseErrors Transaction)
forall (m :: * -> *) a.
Monad m =>
JournalParser m a -> Text -> m (Either HledgerParseErrors a)
rjp JournalParser (Either Any) Transaction
forall (m :: * -> *). JournalParser m Transaction
transactionp (Text -> Either Any (Either HledgerParseErrors Transaction))
-> Text -> Either Any (Either HledgerParseErrors Transaction)
forall a b. (a -> b) -> a -> b
$ [Text] -> Text
T.unlines
[Text
"2007/01/28 coopportunity"
,Text
" expenses:food:groceries $47.18"
,Text
" assets:checking $-47.18"
,Text
""
]
,String -> Assertion -> TestTree
testCase String
"does not parse a following comment as part of the description" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$
StateT Journal (ParsecT HledgerParseErrorData Text IO) Transaction
-> Text -> (Transaction -> Text) -> Text -> Assertion
forall b st a.
(HasCallStack, Eq b, Show b, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> (a -> b) -> b -> Assertion
assertParseEqOn StateT Journal (ParsecT HledgerParseErrorData Text IO) Transaction
forall (m :: * -> *). JournalParser m Transaction
transactionp Text
"2009/1/1 a ;comment\n b 1\n" Transaction -> Text
tdescription Text
"a"
,String -> Assertion -> TestTree
testCase String
"parses a following whitespace line" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$
HasCallStack => String -> Bool -> Assertion
String -> Bool -> Assertion
assertBool String
"" (Bool -> Assertion) -> Bool -> Assertion
forall a b. (a -> b) -> a -> b
$ Either Any (Either HledgerParseErrors Transaction) -> Bool
forall a b. Either a b -> Bool
isRight (Either Any (Either HledgerParseErrors Transaction) -> Bool)
-> Either Any (Either HledgerParseErrors Transaction) -> Bool
forall a b. (a -> b) -> a -> b
$ JournalParser (Either Any) Transaction
-> Text -> Either Any (Either HledgerParseErrors Transaction)
forall (m :: * -> *) a.
Monad m =>
JournalParser m a -> Text -> m (Either HledgerParseErrors a)
rjp JournalParser (Either Any) Transaction
forall (m :: * -> *). JournalParser m Transaction
transactionp (Text -> Either Any (Either HledgerParseErrors Transaction))
-> Text -> Either Any (Either HledgerParseErrors Transaction)
forall a b. (a -> b) -> a -> b
$ [Text] -> Text
T.unlines
[Text
"2012/1/1"
,Text
" a 1"
,Text
" b"
,Text
" "
]
,String -> Assertion -> TestTree
testCase String
"parses an empty transaction comment following whitespace line" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$
HasCallStack => String -> Bool -> Assertion
String -> Bool -> Assertion
assertBool String
"" (Bool -> Assertion) -> Bool -> Assertion
forall a b. (a -> b) -> a -> b
$ Either Any (Either HledgerParseErrors Transaction) -> Bool
forall a b. Either a b -> Bool
isRight (Either Any (Either HledgerParseErrors Transaction) -> Bool)
-> Either Any (Either HledgerParseErrors Transaction) -> Bool
forall a b. (a -> b) -> a -> b
$ JournalParser (Either Any) Transaction
-> Text -> Either Any (Either HledgerParseErrors Transaction)
forall (m :: * -> *) a.
Monad m =>
JournalParser m a -> Text -> m (Either HledgerParseErrors a)
rjp JournalParser (Either Any) Transaction
forall (m :: * -> *). JournalParser m Transaction
transactionp (Text -> Either Any (Either HledgerParseErrors Transaction))
-> Text -> Either Any (Either HledgerParseErrors Transaction)
forall a b. (a -> b) -> a -> b
$ [Text] -> Text
T.unlines
[Text
"2012/1/1"
,Text
" ;"
,Text
" a 1"
,Text
" b"
,Text
" "
]
,String -> Assertion -> TestTree
testCase String
"comments everywhere, two postings parsed" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$
StateT Journal (ParsecT HledgerParseErrorData Text IO) Transaction
-> Text -> (Transaction -> Int) -> Int -> Assertion
forall b st a.
(HasCallStack, Eq b, Show b, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> (a -> b) -> b -> Assertion
assertParseEqOn StateT Journal (ParsecT HledgerParseErrorData Text IO) Transaction
forall (m :: * -> *). JournalParser m Transaction
transactionp
([Text] -> Text
T.unlines
[Text
"2009/1/1 x ; transaction comment"
,Text
" a 1 ; posting 1 comment"
,Text
" ; posting 1 comment 2"
,Text
" b"
,Text
" ; posting 2 comment"
])
([Posting] -> Int
forall (t :: * -> *) a. Foldable t => t a -> Int
length ([Posting] -> Int)
-> (Transaction -> [Posting]) -> Transaction -> Int
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Transaction -> [Posting]
tpostings)
Int
2
]
,String -> [TestTree] -> TestTree
testGroup String
"directivep" [
String -> Assertion -> TestTree
testCase String
"supports !" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ do
StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError IO))
()
-> Text -> Assertion
forall a st.
(HasCallStack, Eq a, Show a, Default st) =>
StateT
st
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError IO))
a
-> Text -> Assertion
assertParseE StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError IO))
()
forall (m :: * -> *). MonadIO m => ErroringJournalParser m ()
directivep Text
"!account a\n"
StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError IO))
()
-> Text -> Assertion
forall a st.
(HasCallStack, Eq a, Show a, Default st) =>
StateT
st
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError IO))
a
-> Text -> Assertion
assertParseE StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError IO))
()
forall (m :: * -> *). MonadIO m => ErroringJournalParser m ()
directivep Text
"!D 1.0\n"
]
,String -> [TestTree] -> TestTree
testGroup String
"accountdirectivep" [
String -> Assertion -> TestTree
testCase String
"with-comment" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
forall (m :: * -> *). JournalParser m ()
accountdirectivep Text
"account a:b ; a comment\n"
,String -> Assertion -> TestTree
testCase String
"does-not-support-!" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
-> Text -> String -> Assertion
forall a st.
(HasCallStack, Eq a, Show a, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> String -> Assertion
assertParseError StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
forall (m :: * -> *). JournalParser m ()
accountdirectivep Text
"!account a:b\n" String
""
,String -> Assertion -> TestTree
testCase String
"account-type-code" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
forall (m :: * -> *). JournalParser m ()
accountdirectivep Text
"account a:b ; type:A\n"
,String -> Assertion -> TestTree
testCase String
"account-type-tag" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
-> Text
-> (Journal -> [(Text, AccountDeclarationInfo)])
-> [(Text, AccountDeclarationInfo)]
-> Assertion
forall b st a.
(HasCallStack, Eq b, Show b, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> (st -> b) -> b -> Assertion
assertParseStateOn StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
forall (m :: * -> *). JournalParser m ()
accountdirectivep Text
"account a:b ; type:asset\n"
Journal -> [(Text, AccountDeclarationInfo)]
jdeclaredaccounts
[(Text
"a:b", AccountDeclarationInfo :: Text -> [Tag] -> Int -> AccountDeclarationInfo
AccountDeclarationInfo{adicomment :: Text
adicomment = Text
"type:asset\n"
,aditags :: [Tag]
aditags = [(Text
"type",Text
"asset")]
,adideclarationorder :: Int
adideclarationorder = Int
1
})
]
]
,String -> Assertion -> TestTree
testCase String
"commodityconversiondirectivep" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ do
StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
forall (m :: * -> *). JournalParser m ()
commodityconversiondirectivep Text
"C 1h = $50.00\n"
,String -> Assertion -> TestTree
testCase String
"defaultcommoditydirectivep" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ do
StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
forall (m :: * -> *). JournalParser m ()
defaultcommoditydirectivep Text
"D $1,000.0\n"
StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
-> Text -> String -> Assertion
forall a st.
(HasCallStack, Eq a, Show a, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> String -> Assertion
assertParseError StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
forall (m :: * -> *). JournalParser m ()
defaultcommoditydirectivep Text
"D $1000\n" String
"Please include a decimal point or decimal comma"
,String -> [TestTree] -> TestTree
testGroup String
"defaultyeardirectivep" [
String -> Assertion -> TestTree
testCase String
"1000" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
forall (m :: * -> *). JournalParser m ()
defaultyeardirectivep Text
"Y 1000"
,String -> Assertion -> TestTree
testCase String
"12345" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
forall (m :: * -> *). JournalParser m ()
defaultyeardirectivep Text
"Y 12345"
]
,String -> Assertion -> TestTree
testCase String
"ignoredpricecommoditydirectivep" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ do
StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
forall (m :: * -> *). JournalParser m ()
ignoredpricecommoditydirectivep Text
"N $\n"
,String -> [TestTree] -> TestTree
testGroup String
"includedirectivep" [
String -> Assertion -> TestTree
testCase String
"include" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError IO))
()
-> Text -> String -> Assertion
forall st a.
(Default st, Eq a, Show a, HasCallStack) =>
StateT
st
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError IO))
a
-> Text -> String -> Assertion
assertParseErrorE StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError IO))
()
forall (m :: * -> *). MonadIO m => ErroringJournalParser m ()
includedirectivep Text
"include nosuchfile\n" String
"No existing files match pattern: nosuchfile"
,String -> Assertion -> TestTree
testCase String
"glob" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError IO))
()
-> Text -> String -> Assertion
forall st a.
(Default st, Eq a, Show a, HasCallStack) =>
StateT
st
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError IO))
a
-> Text -> String -> Assertion
assertParseErrorE StateT
Journal
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError IO))
()
forall (m :: * -> *). MonadIO m => ErroringJournalParser m ()
includedirectivep Text
"include nosuchfile*\n" String
"No existing files match pattern: nosuchfile*"
]
,String -> Assertion -> TestTree
testCase String
"marketpricedirectivep" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT
Journal (ParsecT HledgerParseErrorData Text IO) PriceDirective
-> Text -> PriceDirective -> Assertion
forall a st.
(HasCallStack, Eq a, Show a, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> a -> Assertion
assertParseEq StateT
Journal (ParsecT HledgerParseErrorData Text IO) PriceDirective
forall (m :: * -> *). JournalParser m PriceDirective
marketpricedirectivep
Text
"P 2017/01/30 BTC $922.83\n"
PriceDirective :: Day -> Text -> Amount -> PriceDirective
PriceDirective{
pddate :: Day
pddate = Year -> Int -> Int -> Day
fromGregorian Year
2017 Int
1 Int
30,
pdcommodity :: Text
pdcommodity = Text
"BTC",
pdamount :: Amount
pdamount = DecimalRaw Year -> Amount
usd DecimalRaw Year
922.83
}
,String -> [TestTree] -> TestTree
testGroup String
"payeedirectivep" [
String -> Assertion -> TestTree
testCase String
"simple" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
forall (m :: * -> *). JournalParser m ()
payeedirectivep Text
"payee foo\n"
,String -> Assertion -> TestTree
testCase String
"with-comment" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
forall (m :: * -> *). JournalParser m ()
payeedirectivep Text
"payee foo ; comment\n"
]
,String -> Assertion -> TestTree
testCase String
"tagdirectivep" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ do
StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
forall (m :: * -> *). JournalParser m ()
tagdirectivep Text
"tag foo \n"
,String -> Assertion -> TestTree
testCase String
"endtagdirectivep" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ do
StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
forall (m :: * -> *). JournalParser m ()
endtagdirectivep Text
"end tag \n"
StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
-> Text -> Assertion
forall st a.
(HasCallStack, Default st) =>
StateT st (ParsecT HledgerParseErrorData Text IO) a
-> Text -> Assertion
assertParse StateT Journal (ParsecT HledgerParseErrorData Text IO) ()
forall (m :: * -> *). JournalParser m ()
endtagdirectivep Text
"pop \n"
,String -> [TestTree] -> TestTree
testGroup String
"journalp" [
String -> Assertion -> TestTree
testCase String
"empty file" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ ErroringJournalParser IO Journal -> Text -> Journal -> Assertion
forall st a.
(Default st, Eq a, Show a, HasCallStack) =>
StateT
st
(ParsecT HledgerParseErrorData Text (ExceptT FinalParseError IO))
a
-> Text -> a -> Assertion
assertParseEqE ErroringJournalParser IO Journal
forall (m :: * -> *). MonadIO m => ErroringJournalParser m Journal
journalp Text
"" Journal
nulljournal
]
,String -> Assertion -> TestTree
testCase String
"parseAndFinaliseJournal" (Assertion -> TestTree) -> Assertion -> TestTree
forall a b. (a -> b) -> a -> b
$ do
Either String Journal
ej <- ExceptT String IO Journal -> IO (Either String Journal)
forall e (m :: * -> *) a. ExceptT e m a -> m (Either e a)
runExceptT (ExceptT String IO Journal -> IO (Either String Journal))
-> ExceptT String IO Journal -> IO (Either String Journal)
forall a b. (a -> b) -> a -> b
$ ErroringJournalParser IO Journal
-> InputOpts -> String -> Text -> ExceptT String IO Journal
parseAndFinaliseJournal ErroringJournalParser IO Journal
forall (m :: * -> *). MonadIO m => ErroringJournalParser m Journal
journalp InputOpts
definputopts String
"" Text
"2019-1-1\n"
let Right Journal
j = Either String Journal
ej
String -> [String] -> [String] -> Assertion
forall a.
(Eq a, Show a, HasCallStack) =>
String -> a -> a -> Assertion
assertEqual String
"" [String
""] ([String] -> Assertion) -> [String] -> Assertion
forall a b. (a -> b) -> a -> b
$ Journal -> [String]
journalFilePaths Journal
j
]