2014-03-20 04:11:48 +04:00
|
|
|
{-|
|
|
|
|
|
|
|
|
Balance report, used by the balance command.
|
|
|
|
|
|
|
|
-}
|
|
|
|
|
2019-04-26 21:45:51 +03:00
|
|
|
{-# LANGUAGE FlexibleInstances, RecordWildCards, ScopedTypeVariables, OverloadedStrings #-}
|
2016-08-09 01:56:50 +03:00
|
|
|
|
2014-03-20 04:11:48 +04:00
|
|
|
module Hledger.Reports.BalanceReport (
|
|
|
|
BalanceReport,
|
|
|
|
BalanceReportItem,
|
|
|
|
balanceReport,
|
2014-03-26 06:27:18 +04:00
|
|
|
flatShowsExclusiveBalance,
|
journal: a new account sorting mechanism, and a bunch of sorting fixes
A bunch of account sorting changes that got intermingled.
First, account codes have been dropped. They can still be parsed and
will be ignored, for now. I don't know if anyone used them.
Instead, account display order is now controlled by the order of account
directives, if any. From the mail list:
I'd like to drop account codes, introduced in hledger 1.9 to control
the display order of accounts. In my experience,
- they are tedious to maintain
- they duplicate/compete with the natural tendency to arrange account
directives to match your mental chart of accounts
- they duplicate/compete with the tree structure created by account
names
and it gets worse if you think about using them more extensively,
eg to classify accounts by type.
Instead, I plan to just let the position (parse order) of account
directives determine the display order of those declared accounts.
Undeclared accounts will be displayed after declared accounts,
sorted alphabetically as usual.
Second, the various account sorting modes have been implemented more
widely and more correctly. All sorting modes (alphabetically, by account
declaration, by amount) should now work correctly in almost all commands
and modes (non-tabular and tabular balance reports, tree and flat modes,
the accounts command). Sorting bugs have been fixed, eg #875.
Only the budget report (balance --budget) does not yet support sorting.
Comprehensive functional tests for sorting in the accounts and balance
commands have been added. If you are confused by some sorting behaviour,
studying these tests is recommended, as sorting gets tricky.
2018-09-23 10:45:07 +03:00
|
|
|
sortAccountItemsLike,
|
2014-03-20 04:11:48 +04:00
|
|
|
|
|
|
|
-- * Tests
|
2018-09-06 23:08:26 +03:00
|
|
|
tests_BalanceReport
|
2014-03-20 04:11:48 +04:00
|
|
|
)
|
|
|
|
where
|
|
|
|
|
2016-12-30 22:41:12 +03:00
|
|
|
import Data.List
|
|
|
|
import Data.Ord
|
2014-03-20 04:11:48 +04:00
|
|
|
import Data.Maybe
|
2015-08-26 20:38:45 +03:00
|
|
|
import Data.Time.Calendar
|
2014-03-20 04:11:48 +04:00
|
|
|
|
|
|
|
import Hledger.Data
|
|
|
|
import Hledger.Read (mamountp')
|
|
|
|
import Hledger.Query
|
2018-09-04 23:52:36 +03:00
|
|
|
import Hledger.Utils
|
2014-03-20 04:11:48 +04:00
|
|
|
import Hledger.Reports.ReportOptions
|
|
|
|
|
|
|
|
|
2018-01-23 22:32:24 +03:00
|
|
|
-- | A simple balance report. It has:
|
2014-03-20 04:11:48 +04:00
|
|
|
--
|
2016-08-09 01:56:50 +03:00
|
|
|
-- 1. a list of items, one per account, each containing:
|
2014-03-20 04:11:48 +04:00
|
|
|
--
|
2016-08-09 01:56:50 +03:00
|
|
|
-- * the full account name
|
|
|
|
--
|
|
|
|
-- * the Ledger-style elided short account name
|
|
|
|
-- (the leaf account name, prefixed by any boring parents immediately above);
|
|
|
|
-- or with --flat, the full account name again
|
2014-03-20 04:11:48 +04:00
|
|
|
--
|
2016-08-09 01:56:50 +03:00
|
|
|
-- * the number of indentation steps for rendering a Ledger-style account tree,
|
|
|
|
-- taking into account elided boring parents, --no-elide and --flat
|
2014-09-11 00:07:53 +04:00
|
|
|
--
|
2016-08-09 01:56:50 +03:00
|
|
|
-- * an amount
|
2014-09-11 00:07:53 +04:00
|
|
|
--
|
2016-08-09 01:56:50 +03:00
|
|
|
-- 2. the total of all amounts
|
|
|
|
--
|
|
|
|
type BalanceReport = ([BalanceReportItem], MixedAmount)
|
|
|
|
type BalanceReportItem = (AccountName, AccountName, Int, MixedAmount)
|
2014-03-20 04:11:48 +04:00
|
|
|
|
2014-03-26 06:27:18 +04:00
|
|
|
-- | When true (the default), this makes balance --flat reports and their implementation clearer.
|
|
|
|
-- Single/multi-col balance reports currently aren't all correct if this is false.
|
|
|
|
flatShowsExclusiveBalance = True
|
|
|
|
|
|
|
|
-- | Enabling this makes balance --flat --empty also show parent accounts without postings,
|
|
|
|
-- in addition to those with postings and a zero balance. Disabling it shows only the latter.
|
|
|
|
-- No longer supported, but leave this here for a bit.
|
|
|
|
-- flatShowsPostinglessAccounts = True
|
|
|
|
|
2014-03-20 04:11:48 +04:00
|
|
|
-- | Generate a simple balance report, containing the matched accounts and
|
|
|
|
-- their balances (change of balance) during the specified period.
|
2016-08-12 19:44:31 +03:00
|
|
|
-- This is like PeriodChangeReport with a single column (but more mature,
|
2014-03-20 04:11:48 +04:00
|
|
|
-- eg this can do hierarchical display).
|
|
|
|
balanceReport :: ReportOpts -> Query -> Journal -> BalanceReport
|
2019-05-05 19:57:03 +03:00
|
|
|
balanceReport ropts@ReportOpts{..} q j =
|
|
|
|
(if invert_ then brNegate else id) $
|
journal: a new account sorting mechanism, and a bunch of sorting fixes
A bunch of account sorting changes that got intermingled.
First, account codes have been dropped. They can still be parsed and
will be ignored, for now. I don't know if anyone used them.
Instead, account display order is now controlled by the order of account
directives, if any. From the mail list:
I'd like to drop account codes, introduced in hledger 1.9 to control
the display order of accounts. In my experience,
- they are tedious to maintain
- they duplicate/compete with the natural tendency to arrange account
directives to match your mental chart of accounts
- they duplicate/compete with the tree structure created by account
names
and it gets worse if you think about using them more extensively,
eg to classify accounts by type.
Instead, I plan to just let the position (parse order) of account
directives determine the display order of those declared accounts.
Undeclared accounts will be displayed after declared accounts,
sorted alphabetically as usual.
Second, the various account sorting modes have been implemented more
widely and more correctly. All sorting modes (alphabetically, by account
declaration, by amount) should now work correctly in almost all commands
and modes (non-tabular and tabular balance reports, tree and flat modes,
the accounts command). Sorting bugs have been fixed, eg #875.
Only the budget report (balance --budget) does not yet support sorting.
Comprehensive functional tests for sorting in the accounts and balance
commands have been added. If you are confused by some sorting behaviour,
studying these tests is recommended, as sorting gets tricky.
2018-09-23 10:45:07 +03:00
|
|
|
(sorteditems, total)
|
2014-03-20 04:11:48 +04:00
|
|
|
where
|
2015-05-14 22:49:17 +03:00
|
|
|
-- dbg1 = const id -- exclude from debug output
|
|
|
|
dbg1 s = let p = "balanceReport" in Hledger.Utils.dbg1 (p++" "++s) -- add prefix in debug output
|
2014-03-26 06:27:18 +04:00
|
|
|
|
2019-05-05 20:49:44 +03:00
|
|
|
-- We may be converting amounts to value, according to --value-at:
|
2019-05-08 02:19:49 +03:00
|
|
|
-- transaction: value each posting at posting date before summing
|
|
|
|
-- period: value totals at period end
|
|
|
|
-- date: value totals at date
|
2019-05-05 20:49:44 +03:00
|
|
|
today = fromMaybe (error' "balanceReport: ReportOpts today_ is unset so could not satisfy --value-at=now") today_
|
|
|
|
|
|
|
|
-- For --value-at=transaction, convert all postings to value before summing them.
|
|
|
|
-- The report might not use them all but laziness probably helps here.
|
2019-05-23 10:36:16 +03:00
|
|
|
j' -- | mvalueat==Just AtTransaction = mapJournalPostings (\p -> postingValueAtDate j (postingDate p) p) j
|
2019-05-05 20:49:44 +03:00
|
|
|
| otherwise = j
|
|
|
|
|
2019-05-05 19:49:45 +03:00
|
|
|
-- Get all the summed accounts & balances, according to the query, as an account tree.
|
2019-05-05 20:49:44 +03:00
|
|
|
accttree = ledgerRootAccount $ ledgerFromJournal q $ journalSelectingAmountFromOpts ropts j'
|
|
|
|
|
|
|
|
-- For --value-at=(all except transaction, done above), convert the summed amounts to value.
|
|
|
|
valuedaccttree = mapAccounts valueaccount accttree
|
|
|
|
where
|
|
|
|
valueaccount a@Account{..} = a{aebalance=val aebalance, aibalance=val aibalance}
|
|
|
|
where
|
2019-05-23 10:36:16 +03:00
|
|
|
val = case value_ of
|
|
|
|
Just (AtEnd _mc) -> mixedAmountValue prices periodlastday
|
|
|
|
Just (AtNow _mc) -> mixedAmountValue prices today
|
|
|
|
Just (AtDate d _mc) -> mixedAmountValue prices d
|
|
|
|
_ -> id
|
2019-05-05 20:49:44 +03:00
|
|
|
where
|
|
|
|
-- prices are in parse order - sort into date then parse order,
|
|
|
|
-- & reversed for quick lookup of the latest price.
|
|
|
|
prices = reverse $ sortOn mpdate $ jmarketprices j'
|
|
|
|
periodlastday =
|
|
|
|
fromMaybe (error' "balanceReport: expected a non-empty journal") $ -- XXX shouldn't happen
|
|
|
|
reportPeriodOrJournalLastDay ropts j'
|
2019-05-05 19:49:45 +03:00
|
|
|
|
|
|
|
-- Modify this tree for display - depth limit, boring parents, zeroes - and convert to a list.
|
|
|
|
displayaccts :: [Account]
|
2014-10-20 04:53:20 +04:00
|
|
|
| queryDepth q == 0 =
|
2019-05-05 20:49:44 +03:00
|
|
|
dbg1 "displayaccts" $
|
|
|
|
take 1 $ clipAccountsAndAggregate (queryDepth q) $ flattenAccounts valuedaccttree
|
|
|
|
| flat_ ropts = dbg1 "displayaccts" $
|
2014-03-26 06:27:18 +04:00
|
|
|
filterzeros $
|
|
|
|
filterempty $
|
2019-05-05 20:49:44 +03:00
|
|
|
drop 1 $ clipAccountsAndAggregate (queryDepth q) $ flattenAccounts valuedaccttree
|
|
|
|
| otherwise = dbg1 "displayaccts" $
|
2014-03-26 06:27:18 +04:00
|
|
|
filter (not.aboring) $
|
|
|
|
drop 1 $ flattenAccounts $
|
2014-09-11 00:07:53 +04:00
|
|
|
markboring $
|
2014-10-20 04:53:20 +04:00
|
|
|
prunezeros $
|
2019-05-05 19:57:03 +03:00
|
|
|
sortAccountTreeByAmount (fromMaybe NormallyPositive normalbalance_) $
|
2019-05-05 20:49:44 +03:00
|
|
|
clipAccounts (queryDepth q) valuedaccttree
|
2014-03-20 04:11:48 +04:00
|
|
|
where
|
2019-05-05 20:49:44 +03:00
|
|
|
balance = if flat_ ropts then aebalance else aibalance
|
2019-05-05 19:57:03 +03:00
|
|
|
filterzeros = if empty_ then id else filter (not . isZeroMixedAmount . balance)
|
2014-03-26 06:27:18 +04:00
|
|
|
filterempty = filter (\a -> anumpostings a > 0 || not (isZeroMixedAmount (balance a)))
|
2019-05-05 19:57:03 +03:00
|
|
|
prunezeros = if empty_ then id else fromMaybe nullacct . pruneAccounts (isZeroMixedAmount . balance)
|
|
|
|
markboring = if no_elide_ then id else markBoringParentAccounts
|
journal: a new account sorting mechanism, and a bunch of sorting fixes
A bunch of account sorting changes that got intermingled.
First, account codes have been dropped. They can still be parsed and
will be ignored, for now. I don't know if anyone used them.
Instead, account display order is now controlled by the order of account
directives, if any. From the mail list:
I'd like to drop account codes, introduced in hledger 1.9 to control
the display order of accounts. In my experience,
- they are tedious to maintain
- they duplicate/compete with the natural tendency to arrange account
directives to match your mental chart of accounts
- they duplicate/compete with the tree structure created by account
names
and it gets worse if you think about using them more extensively,
eg to classify accounts by type.
Instead, I plan to just let the position (parse order) of account
directives determine the display order of those declared accounts.
Undeclared accounts will be displayed after declared accounts,
sorted alphabetically as usual.
Second, the various account sorting modes have been implemented more
widely and more correctly. All sorting modes (alphabetically, by account
declaration, by amount) should now work correctly in almost all commands
and modes (non-tabular and tabular balance reports, tree and flat modes,
the accounts command). Sorting bugs have been fixed, eg #875.
Only the budget report (balance --budget) does not yet support sorting.
Comprehensive functional tests for sorting in the accounts and balance
commands have been added. If you are confused by some sorting behaviour,
studying these tests is recommended, as sorting gets tricky.
2018-09-23 10:45:07 +03:00
|
|
|
|
2019-05-05 19:49:45 +03:00
|
|
|
-- Make a report row for each account.
|
2019-05-05 19:57:03 +03:00
|
|
|
items = dbg1 "items" $ map (balanceReportItem ropts q) displayaccts
|
journal: a new account sorting mechanism, and a bunch of sorting fixes
A bunch of account sorting changes that got intermingled.
First, account codes have been dropped. They can still be parsed and
will be ignored, for now. I don't know if anyone used them.
Instead, account display order is now controlled by the order of account
directives, if any. From the mail list:
I'd like to drop account codes, introduced in hledger 1.9 to control
the display order of accounts. In my experience,
- they are tedious to maintain
- they duplicate/compete with the natural tendency to arrange account
directives to match your mental chart of accounts
- they duplicate/compete with the tree structure created by account
names
and it gets worse if you think about using them more extensively,
eg to classify accounts by type.
Instead, I plan to just let the position (parse order) of account
directives determine the display order of those declared accounts.
Undeclared accounts will be displayed after declared accounts,
sorted alphabetically as usual.
Second, the various account sorting modes have been implemented more
widely and more correctly. All sorting modes (alphabetically, by account
declaration, by amount) should now work correctly in almost all commands
and modes (non-tabular and tabular balance reports, tree and flat modes,
the accounts command). Sorting bugs have been fixed, eg #875.
Only the budget report (balance --budget) does not yet support sorting.
Comprehensive functional tests for sorting in the accounts and balance
commands have been added. If you are confused by some sorting behaviour,
studying these tests is recommended, as sorting gets tricky.
2018-09-23 10:45:07 +03:00
|
|
|
|
2019-05-05 19:49:45 +03:00
|
|
|
-- Sort report rows (except sorting by amount in tree mode, which was done above).
|
journal: a new account sorting mechanism, and a bunch of sorting fixes
A bunch of account sorting changes that got intermingled.
First, account codes have been dropped. They can still be parsed and
will be ignored, for now. I don't know if anyone used them.
Instead, account display order is now controlled by the order of account
directives, if any. From the mail list:
I'd like to drop account codes, introduced in hledger 1.9 to control
the display order of accounts. In my experience,
- they are tedious to maintain
- they duplicate/compete with the natural tendency to arrange account
directives to match your mental chart of accounts
- they duplicate/compete with the tree structure created by account
names
and it gets worse if you think about using them more extensively,
eg to classify accounts by type.
Instead, I plan to just let the position (parse order) of account
directives determine the display order of those declared accounts.
Undeclared accounts will be displayed after declared accounts,
sorted alphabetically as usual.
Second, the various account sorting modes have been implemented more
widely and more correctly. All sorting modes (alphabetically, by account
declaration, by amount) should now work correctly in almost all commands
and modes (non-tabular and tabular balance reports, tree and flat modes,
the accounts command). Sorting bugs have been fixed, eg #875.
Only the budget report (balance --budget) does not yet support sorting.
Comprehensive functional tests for sorting in the accounts and balance
commands have been added. If you are confused by some sorting behaviour,
studying these tests is recommended, as sorting gets tricky.
2018-09-23 10:45:07 +03:00
|
|
|
sorteditems
|
2019-05-05 19:57:03 +03:00
|
|
|
| sort_amount_ && tree_ ropts = items
|
|
|
|
| sort_amount_ = sortFlatBRByAmount items
|
|
|
|
| otherwise = sortBRByAccountDeclaration items
|
journal: a new account sorting mechanism, and a bunch of sorting fixes
A bunch of account sorting changes that got intermingled.
First, account codes have been dropped. They can still be parsed and
will be ignored, for now. I don't know if anyone used them.
Instead, account display order is now controlled by the order of account
directives, if any. From the mail list:
I'd like to drop account codes, introduced in hledger 1.9 to control
the display order of accounts. In my experience,
- they are tedious to maintain
- they duplicate/compete with the natural tendency to arrange account
directives to match your mental chart of accounts
- they duplicate/compete with the tree structure created by account
names
and it gets worse if you think about using them more extensively,
eg to classify accounts by type.
Instead, I plan to just let the position (parse order) of account
directives determine the display order of those declared accounts.
Undeclared accounts will be displayed after declared accounts,
sorted alphabetically as usual.
Second, the various account sorting modes have been implemented more
widely and more correctly. All sorting modes (alphabetically, by account
declaration, by amount) should now work correctly in almost all commands
and modes (non-tabular and tabular balance reports, tree and flat modes,
the accounts command). Sorting bugs have been fixed, eg #875.
Only the budget report (balance --budget) does not yet support sorting.
Comprehensive functional tests for sorting in the accounts and balance
commands have been added. If you are confused by some sorting behaviour,
studying these tests is recommended, as sorting gets tricky.
2018-09-23 10:45:07 +03:00
|
|
|
where
|
|
|
|
-- Sort the report rows, representing a flat account list, by row total.
|
|
|
|
sortFlatBRByAmount :: [BalanceReportItem] -> [BalanceReportItem]
|
|
|
|
sortFlatBRByAmount = sortBy (maybeflip $ comparing (normaliseMixedAmountSquashPricesForDisplay . fourth4))
|
|
|
|
where
|
2019-05-05 19:57:03 +03:00
|
|
|
maybeflip = if normalbalance_ == Just NormallyNegative then id else flip
|
journal: a new account sorting mechanism, and a bunch of sorting fixes
A bunch of account sorting changes that got intermingled.
First, account codes have been dropped. They can still be parsed and
will be ignored, for now. I don't know if anyone used them.
Instead, account display order is now controlled by the order of account
directives, if any. From the mail list:
I'd like to drop account codes, introduced in hledger 1.9 to control
the display order of accounts. In my experience,
- they are tedious to maintain
- they duplicate/compete with the natural tendency to arrange account
directives to match your mental chart of accounts
- they duplicate/compete with the tree structure created by account
names
and it gets worse if you think about using them more extensively,
eg to classify accounts by type.
Instead, I plan to just let the position (parse order) of account
directives determine the display order of those declared accounts.
Undeclared accounts will be displayed after declared accounts,
sorted alphabetically as usual.
Second, the various account sorting modes have been implemented more
widely and more correctly. All sorting modes (alphabetically, by account
declaration, by amount) should now work correctly in almost all commands
and modes (non-tabular and tabular balance reports, tree and flat modes,
the accounts command). Sorting bugs have been fixed, eg #875.
Only the budget report (balance --budget) does not yet support sorting.
Comprehensive functional tests for sorting in the accounts and balance
commands have been added. If you are confused by some sorting behaviour,
studying these tests is recommended, as sorting gets tricky.
2018-09-23 10:45:07 +03:00
|
|
|
-- Sort the report rows by account declaration order then account name.
|
|
|
|
sortBRByAccountDeclaration :: [BalanceReportItem] -> [BalanceReportItem]
|
|
|
|
sortBRByAccountDeclaration rows = sortedrows
|
|
|
|
where
|
|
|
|
anamesandrows = [(first4 r, r) | r <- rows]
|
|
|
|
anames = map fst anamesandrows
|
2019-05-05 20:49:44 +03:00
|
|
|
sortedanames = sortAccountNamesByDeclaration j' (tree_ ropts) anames
|
journal: a new account sorting mechanism, and a bunch of sorting fixes
A bunch of account sorting changes that got intermingled.
First, account codes have been dropped. They can still be parsed and
will be ignored, for now. I don't know if anyone used them.
Instead, account display order is now controlled by the order of account
directives, if any. From the mail list:
I'd like to drop account codes, introduced in hledger 1.9 to control
the display order of accounts. In my experience,
- they are tedious to maintain
- they duplicate/compete with the natural tendency to arrange account
directives to match your mental chart of accounts
- they duplicate/compete with the tree structure created by account
names
and it gets worse if you think about using them more extensively,
eg to classify accounts by type.
Instead, I plan to just let the position (parse order) of account
directives determine the display order of those declared accounts.
Undeclared accounts will be displayed after declared accounts,
sorted alphabetically as usual.
Second, the various account sorting modes have been implemented more
widely and more correctly. All sorting modes (alphabetically, by account
declaration, by amount) should now work correctly in almost all commands
and modes (non-tabular and tabular balance reports, tree and flat modes,
the accounts command). Sorting bugs have been fixed, eg #875.
Only the budget report (balance --budget) does not yet support sorting.
Comprehensive functional tests for sorting in the accounts and balance
commands have been added. If you are confused by some sorting behaviour,
studying these tests is recommended, as sorting gets tricky.
2018-09-23 10:45:07 +03:00
|
|
|
sortedrows = sortAccountItemsLike sortedanames anamesandrows
|
|
|
|
|
2019-05-05 19:49:45 +03:00
|
|
|
-- Calculate the grand total.
|
2019-05-05 19:57:03 +03:00
|
|
|
total | not (flat_ ropts) = dbg1 "total" $ sum [amt | (_,_,indent,amt) <- items, indent == 0]
|
|
|
|
| otherwise = dbg1 "total" $
|
|
|
|
if flatShowsExclusiveBalance
|
|
|
|
then sum $ map fourth4 items
|
|
|
|
else sum $ map aebalance $ clipAccountsAndAggregate 1 displayaccts
|
2014-03-20 04:11:48 +04:00
|
|
|
|
journal: a new account sorting mechanism, and a bunch of sorting fixes
A bunch of account sorting changes that got intermingled.
First, account codes have been dropped. They can still be parsed and
will be ignored, for now. I don't know if anyone used them.
Instead, account display order is now controlled by the order of account
directives, if any. From the mail list:
I'd like to drop account codes, introduced in hledger 1.9 to control
the display order of accounts. In my experience,
- they are tedious to maintain
- they duplicate/compete with the natural tendency to arrange account
directives to match your mental chart of accounts
- they duplicate/compete with the tree structure created by account
names
and it gets worse if you think about using them more extensively,
eg to classify accounts by type.
Instead, I plan to just let the position (parse order) of account
directives determine the display order of those declared accounts.
Undeclared accounts will be displayed after declared accounts,
sorted alphabetically as usual.
Second, the various account sorting modes have been implemented more
widely and more correctly. All sorting modes (alphabetically, by account
declaration, by amount) should now work correctly in almost all commands
and modes (non-tabular and tabular balance reports, tree and flat modes,
the accounts command). Sorting bugs have been fixed, eg #875.
Only the budget report (balance --budget) does not yet support sorting.
Comprehensive functional tests for sorting in the accounts and balance
commands have been added. If you are confused by some sorting behaviour,
studying these tests is recommended, as sorting gets tricky.
2018-09-23 10:45:07 +03:00
|
|
|
-- | A sorting helper: sort a list of things (eg report rows) keyed by account name
|
|
|
|
-- to match the provided ordering of those same account names.
|
|
|
|
sortAccountItemsLike :: [AccountName] -> [(AccountName, b)] -> [b]
|
|
|
|
sortAccountItemsLike sortedas items =
|
|
|
|
concatMap (\a -> maybe [] (:[]) $ lookup a items) sortedas
|
|
|
|
|
2014-03-20 04:11:48 +04:00
|
|
|
-- | In an account tree with zero-balance leaves removed, mark the
|
|
|
|
-- elidable parent accounts (those with one subaccount and no balance
|
|
|
|
-- of their own).
|
|
|
|
markBoringParentAccounts :: Account -> Account
|
|
|
|
markBoringParentAccounts = tieAccountParents . mapAccounts mark
|
|
|
|
where
|
|
|
|
mark a | length (asubs a) == 1 && isZeroMixedAmount (aebalance a) = a{aboring=True}
|
|
|
|
| otherwise = a
|
|
|
|
|
2014-03-26 06:27:18 +04:00
|
|
|
balanceReportItem :: ReportOpts -> Query -> Account -> BalanceReportItem
|
2014-10-20 04:53:20 +04:00
|
|
|
balanceReportItem opts q a
|
2016-08-09 01:56:50 +03:00
|
|
|
| flat_ opts = (name, name, 0, (if flatShowsExclusiveBalance then aebalance else aibalance) a)
|
|
|
|
| otherwise = (name, elidedname, indent, aibalance a)
|
2014-03-20 04:11:48 +04:00
|
|
|
where
|
2014-10-20 04:53:20 +04:00
|
|
|
name | queryDepth q > 0 = aname a
|
|
|
|
| otherwise = "..."
|
2014-03-20 04:11:48 +04:00
|
|
|
elidedname = accountNameFromComponents (adjacentboringparentnames ++ [accountLeafName name])
|
2016-08-08 18:31:01 +03:00
|
|
|
adjacentboringparentnames = reverse $ map (accountLeafName.aname) $ takeWhile aboring parents
|
2014-03-20 04:11:48 +04:00
|
|
|
indent = length $ filter (not.aboring) parents
|
2014-10-20 04:53:20 +04:00
|
|
|
-- parents exclude the tree's root node
|
|
|
|
parents = case parentAccounts a of [] -> []
|
|
|
|
as -> init as
|
2014-03-20 04:11:48 +04:00
|
|
|
|
|
|
|
-- -- the above using the newer multi balance report code:
|
|
|
|
-- balanceReport' opts q j = (items, total)
|
|
|
|
-- where
|
2016-08-12 19:44:31 +03:00
|
|
|
-- MultiBalanceReport (_,mbrrows,mbrtotals) = PeriodChangeReport opts q j
|
2014-03-20 04:11:48 +04:00
|
|
|
-- items = [(a,a',n, headDef 0 bs) | ((a,a',n), bs) <- mbrrows]
|
|
|
|
-- total = headDef 0 mbrtotals
|
|
|
|
|
2018-01-30 01:52:03 +03:00
|
|
|
-- | Flip the sign of all amounts in a BalanceReport.
|
|
|
|
brNegate :: BalanceReport -> BalanceReport
|
|
|
|
brNegate (is, tot) = (map brItemNegate is, -tot)
|
|
|
|
where
|
|
|
|
brItemNegate (a, a', d, amt) = (a, a', d, -amt)
|
2016-08-08 17:10:36 +03:00
|
|
|
|
speed up -V/--value by converting reports, not the journal (#999)
Instead of converting all journal amounts to value early on, we now
convert just the report amounts to value, before rendering.
This was basically how it originally worked (for the balance command),
but now it's built in to the four basic reports used by print,
register, balance and their variants - Entries, Postings, Balance,
MultiBalance - each of which now has its own xxValue helper.
This should mostly fix -V's performance when there are many
transactions and prices (the price lookups could still be optimised),
and allow more flexibility for report-specific value calculations.
+------------------------------------------++-----------------+-------------------+--------------------------+
| || hledger.999.pre | hledger.999.1sort | hledger.999.after-report |
+==========================================++=================+===================+==========================+
| -f examples/1000x1000x10.journal bal -V || 1.08 | 0.96 | 0.76 |
| -f examples/2000x1000x10.journal bal -V || 1.65 | 1.05 | 0.73 |
| -f examples/3000x1000x10.journal bal -V || 2.43 | 1.58 | 0.84 |
| -f examples/4000x1000x10.journal bal -V || 4.39 | 1.96 | 0.93 |
| -f examples/5000x1000x10.journal bal -V || 7.75 | 2.99 | 1.07 |
| -f examples/6000x1000x10.journal bal -V || 11.21 | 3.72 | 1.16 |
| -f examples/7000x1000x10.journal bal -V || 16.91 | 4.72 | 1.19 |
| -f examples/8000x1000x10.journal bal -V || 27.10 | 9.83 | 1.40 |
| -f examples/9000x1000x10.journal bal -V || 39.73 | 15.00 | 1.51 |
| -f examples/10000x1000x10.journal bal -V || 50.72 | 25.61 | 2.15 |
+------------------------------------------++-----------------+-------------------+--------------------------+
There's one new limitation, not yet resolved: -V once again can pick a
valuation date in the future, if no report end date is specified and
the journal has future-dated transactions. We prefer to avoid that,
but reports currently are pure and don't have access to today's date.
2019-04-24 03:39:01 +03:00
|
|
|
|
|
|
|
-- tests
|
|
|
|
|
2016-08-08 18:31:01 +03:00
|
|
|
Right samplejournal2 =
|
2016-12-10 18:04:48 +03:00
|
|
|
journalBalanceTransactions False
|
2016-08-08 18:31:01 +03:00
|
|
|
nulljournal{
|
|
|
|
jtxns = [
|
|
|
|
txnTieKnot Transaction{
|
|
|
|
tindex=0,
|
|
|
|
tsourcepos=nullsourcepos,
|
|
|
|
tdate=parsedate "2008/01/01",
|
|
|
|
tdate2=Just $ parsedate "2009/01/01",
|
2017-06-16 02:25:37 +03:00
|
|
|
tstatus=Unmarked,
|
2016-08-08 18:31:01 +03:00
|
|
|
tcode="",
|
|
|
|
tdescription="income",
|
|
|
|
tcomment="",
|
|
|
|
ttags=[],
|
|
|
|
tpostings=
|
|
|
|
[posting {paccount="assets:bank:checking", pamount=Mixed [usd 1]}
|
|
|
|
,posting {paccount="income:salary", pamount=missingmixedamt}
|
|
|
|
],
|
2019-01-04 23:01:45 +03:00
|
|
|
tprecedingcomment=""
|
2016-08-08 18:31:01 +03:00
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
2014-09-11 00:07:53 +04:00
|
|
|
|
2018-09-06 23:08:26 +03:00
|
|
|
tests_BalanceReport = tests "BalanceReport" [
|
2018-09-04 22:23:07 +03:00
|
|
|
tests "balanceReport" $
|
|
|
|
let
|
|
|
|
(opts,journal) `gives` r = do
|
|
|
|
let (eitems, etotal) = r
|
|
|
|
(aitems, atotal) = balanceReport opts (queryFromOpts nulldate opts) journal
|
|
|
|
showw (acct,acct',indent,amt) = (acct, acct', indent, showMixedAmountDebug amt)
|
|
|
|
(map showw eitems) `is` (map showw aitems)
|
|
|
|
(showMixedAmountDebug etotal) `is` (showMixedAmountDebug atotal)
|
|
|
|
usd0 = usd 0
|
|
|
|
in [
|
|
|
|
|
|
|
|
test "balanceReport with no args on null journal" $
|
|
|
|
(defreportopts, nulljournal) `gives` ([], Mixed [nullamt])
|
|
|
|
|
|
|
|
,test "balanceReport with no args on sample journal" $
|
|
|
|
(defreportopts, samplejournal) `gives`
|
|
|
|
([
|
|
|
|
("assets","assets",0, mamountp' "$0.00")
|
|
|
|
,("assets:bank","bank",1, mamountp' "$2.00")
|
|
|
|
,("assets:bank:checking","checking",2, mamountp' "$1.00")
|
|
|
|
,("assets:bank:saving","saving",2, mamountp' "$1.00")
|
|
|
|
,("assets:cash","cash",1, mamountp' "$-2.00")
|
|
|
|
,("expenses","expenses",0, mamountp' "$2.00")
|
|
|
|
,("expenses:food","food",1, mamountp' "$1.00")
|
|
|
|
,("expenses:supplies","supplies",1, mamountp' "$1.00")
|
|
|
|
,("income","income",0, mamountp' "$-2.00")
|
|
|
|
,("income:gifts","gifts",1, mamountp' "$-1.00")
|
|
|
|
,("income:salary","salary",1, mamountp' "$-1.00")
|
|
|
|
],
|
|
|
|
Mixed [usd0])
|
|
|
|
|
|
|
|
,test "balanceReport with --depth=N" $
|
|
|
|
(defreportopts{depth_=Just 1}, samplejournal) `gives`
|
|
|
|
([
|
|
|
|
("expenses", "expenses", 0, mamountp' "$2.00")
|
|
|
|
,("income", "income", 0, mamountp' "$-2.00")
|
|
|
|
],
|
|
|
|
Mixed [usd0])
|
|
|
|
|
|
|
|
,test "balanceReport with depth:N" $
|
|
|
|
(defreportopts{query_="depth:1"}, samplejournal) `gives`
|
|
|
|
([
|
|
|
|
("expenses", "expenses", 0, mamountp' "$2.00")
|
|
|
|
,("income", "income", 0, mamountp' "$-2.00")
|
|
|
|
],
|
|
|
|
Mixed [usd0])
|
|
|
|
|
|
|
|
,tests "balanceReport with a date or secondary date span" [
|
|
|
|
(defreportopts{query_="date:'in 2009'"}, samplejournal2) `gives`
|
|
|
|
([],
|
|
|
|
Mixed [nullamt])
|
|
|
|
,(defreportopts{query_="date2:'in 2009'"}, samplejournal2) `gives`
|
|
|
|
([
|
|
|
|
("assets:bank:checking","assets:bank:checking",0,mamountp' "$1.00")
|
|
|
|
,("income:salary","income:salary",0,mamountp' "$-1.00")
|
|
|
|
],
|
|
|
|
Mixed [usd0])
|
|
|
|
]
|
|
|
|
|
|
|
|
,test "balanceReport with desc:" $
|
|
|
|
(defreportopts{query_="desc:income"}, samplejournal) `gives`
|
|
|
|
([
|
|
|
|
("assets:bank:checking","assets:bank:checking",0,mamountp' "$1.00")
|
|
|
|
,("income:salary","income:salary",0, mamountp' "$-1.00")
|
|
|
|
],
|
|
|
|
Mixed [usd0])
|
|
|
|
|
|
|
|
,test "balanceReport with not:desc:" $
|
|
|
|
(defreportopts{query_="not:desc:income"}, samplejournal) `gives`
|
|
|
|
([
|
|
|
|
("assets","assets",0, mamountp' "$-1.00")
|
|
|
|
,("assets:bank:saving","bank:saving",1, mamountp' "$1.00")
|
|
|
|
,("assets:cash","cash",1, mamountp' "$-2.00")
|
|
|
|
,("expenses","expenses",0, mamountp' "$2.00")
|
|
|
|
,("expenses:food","food",1, mamountp' "$1.00")
|
|
|
|
,("expenses:supplies","supplies",1, mamountp' "$1.00")
|
|
|
|
,("income:gifts","income:gifts",0, mamountp' "$-1.00")
|
|
|
|
],
|
|
|
|
Mixed [usd0])
|
|
|
|
|
|
|
|
,test "balanceReport with period on a populated period" $
|
|
|
|
(defreportopts{period_= PeriodBetween (fromGregorian 2008 1 1) (fromGregorian 2008 1 2)}, samplejournal) `gives`
|
|
|
|
(
|
|
|
|
[
|
|
|
|
("assets:bank:checking","assets:bank:checking",0, mamountp' "$1.00")
|
|
|
|
,("income:salary","income:salary",0, mamountp' "$-1.00")
|
|
|
|
],
|
|
|
|
Mixed [usd0])
|
|
|
|
|
|
|
|
,test "balanceReport with period on an unpopulated period" $
|
|
|
|
(defreportopts{period_= PeriodBetween (fromGregorian 2008 1 2) (fromGregorian 2008 1 3)}, samplejournal) `gives`
|
|
|
|
([],Mixed [nullamt])
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{-
|
|
|
|
,test "accounts report with account pattern o" ~:
|
|
|
|
defreportopts{patterns_=["o"]} `gives`
|
|
|
|
[" $1 expenses:food"
|
|
|
|
," $-2 income"
|
|
|
|
," $-1 gifts"
|
|
|
|
," $-1 salary"
|
|
|
|
,"--------------------"
|
|
|
|
," $-1"
|
|
|
|
]
|
|
|
|
|
|
|
|
,test "accounts report with account pattern o and --depth 1" ~:
|
|
|
|
defreportopts{patterns_=["o"],depth_=Just 1} `gives`
|
|
|
|
[" $1 expenses"
|
|
|
|
," $-2 income"
|
|
|
|
,"--------------------"
|
|
|
|
," $-1"
|
|
|
|
]
|
|
|
|
|
|
|
|
,test "accounts report with account pattern a" ~:
|
|
|
|
defreportopts{patterns_=["a"]} `gives`
|
|
|
|
[" $-1 assets"
|
|
|
|
," $1 bank:saving"
|
|
|
|
," $-2 cash"
|
|
|
|
," $-1 income:salary"
|
|
|
|
," $1 liabilities:debts"
|
|
|
|
,"--------------------"
|
|
|
|
," $-1"
|
|
|
|
]
|
|
|
|
|
|
|
|
,test "accounts report with account pattern e" ~:
|
|
|
|
defreportopts{patterns_=["e"]} `gives`
|
|
|
|
[" $-1 assets"
|
|
|
|
," $1 bank:saving"
|
|
|
|
," $-2 cash"
|
|
|
|
," $2 expenses"
|
|
|
|
," $1 food"
|
|
|
|
," $1 supplies"
|
|
|
|
," $-2 income"
|
|
|
|
," $-1 gifts"
|
|
|
|
," $-1 salary"
|
|
|
|
," $1 liabilities:debts"
|
|
|
|
,"--------------------"
|
|
|
|
," 0"
|
|
|
|
]
|
|
|
|
|
|
|
|
,test "accounts report with unmatched parent of two matched subaccounts" ~:
|
|
|
|
defreportopts{patterns_=["cash","saving"]} `gives`
|
|
|
|
[" $-1 assets"
|
|
|
|
," $1 bank:saving"
|
|
|
|
," $-2 cash"
|
|
|
|
,"--------------------"
|
|
|
|
," $-1"
|
|
|
|
]
|
|
|
|
|
|
|
|
,test "accounts report with multi-part account name" ~:
|
|
|
|
defreportopts{patterns_=["expenses:food"]} `gives`
|
|
|
|
[" $1 expenses:food"
|
|
|
|
,"--------------------"
|
|
|
|
," $1"
|
|
|
|
]
|
|
|
|
|
|
|
|
,test "accounts report with negative account pattern" ~:
|
|
|
|
defreportopts{patterns_=["not:assets"]} `gives`
|
|
|
|
[" $2 expenses"
|
|
|
|
," $1 food"
|
|
|
|
," $1 supplies"
|
|
|
|
," $-2 income"
|
|
|
|
," $-1 gifts"
|
|
|
|
," $-1 salary"
|
|
|
|
," $1 liabilities:debts"
|
|
|
|
,"--------------------"
|
|
|
|
," $1"
|
|
|
|
]
|
|
|
|
|
|
|
|
,test "accounts report negative account pattern always matches full name" ~:
|
|
|
|
defreportopts{patterns_=["not:e"]} `gives`
|
|
|
|
["--------------------"
|
|
|
|
," 0"
|
|
|
|
]
|
|
|
|
|
|
|
|
,test "accounts report negative patterns affect totals" ~:
|
|
|
|
defreportopts{patterns_=["expenses","not:food"]} `gives`
|
|
|
|
[" $1 expenses:supplies"
|
|
|
|
,"--------------------"
|
|
|
|
," $1"
|
|
|
|
]
|
|
|
|
|
|
|
|
,test "accounts report with -E shows zero-balance accounts" ~:
|
|
|
|
defreportopts{patterns_=["assets"],empty_=True} `gives`
|
|
|
|
[" $-1 assets"
|
|
|
|
," $1 bank"
|
|
|
|
," 0 checking"
|
|
|
|
," $1 saving"
|
|
|
|
," $-2 cash"
|
|
|
|
,"--------------------"
|
|
|
|
," $-1"
|
|
|
|
]
|
|
|
|
|
|
|
|
,test "accounts report with cost basis" $
|
|
|
|
j <- (readJournal def Nothing $ unlines
|
|
|
|
[""
|
|
|
|
,"2008/1/1 test "
|
|
|
|
," a:b 10h @ $50"
|
|
|
|
," c:d "
|
|
|
|
]) >>= either error' return
|
|
|
|
let j' = journalCanonicaliseAmounts $ journalConvertAmountsToCost j -- enable cost basis adjustment
|
|
|
|
balanceReportAsText defreportopts (balanceReport defreportopts Any j') `is`
|
|
|
|
[" $500 a:b"
|
|
|
|
," $-500 c:d"
|
|
|
|
,"--------------------"
|
|
|
|
," 0"
|
|
|
|
]
|
|
|
|
-}
|
|
|
|
]
|
2014-09-11 00:07:53 +04:00
|
|
|
|
2018-09-04 22:23:07 +03:00
|
|
|
]
|
2014-03-20 04:11:48 +04:00
|
|
|
|