summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--report.cc414
-rw-r--r--report.h80
2 files changed, 494 insertions, 0 deletions
diff --git a/report.cc b/report.cc
new file mode 100644
index 00000000..bd6338d2
--- /dev/null
+++ b/report.cc
@@ -0,0 +1,414 @@
+#include "report.h"
+
+namespace ledger {
+
+report_t::report_t()
+{
+ ledger::amount_expr = "@a";
+ ledger::total_expr = "@O";
+
+ predicate = "";
+ secondary_predicate = "";
+ display_predicate = "";
+ descend_expr = "";
+
+ pricing_leeway = 24 * 3600;
+ budget_flags = BUDGET_NO_BUDGET;
+
+ head_entries = 0;
+ tail_entries = 0;
+
+ show_collapsed = false;
+ show_subtotal = false;
+ show_totals = false;
+ show_related = false;
+ show_all_related = false;
+ show_inverted = false;
+ show_empty = false;
+ days_of_the_week = false;
+ by_payee = false;
+ comm_as_payee = false;
+ code_as_payee = false;
+ show_revalued = false;
+ show_revalued_only = false;
+ keep_price = false;
+ keep_date = false;
+ keep_tag = false;
+ entry_sort = false;
+ sort_all = false;
+}
+
+void
+report_t::regexps_to_predicate(const std::string& command,
+ std::list<std::string>::const_iterator begin,
+ std::list<std::string>::const_iterator end,
+ const bool account_regexp,
+ const bool add_account_short_masks,
+ const bool logical_and)
+{
+ std::string regexps[2];
+
+ assert(begin != end);
+
+ // Treat the remaining command-line arguments as regular
+ // expressions, used for refining report results.
+
+ for (std::list<std::string>::const_iterator i = begin;
+ i != end;
+ i++)
+ if ((*i)[0] == '-') {
+ if (! regexps[1].empty())
+ regexps[1] += "|";
+ regexps[1] += (*i).substr(1);
+ }
+ else if ((*i)[0] == '+') {
+ if (! regexps[0].empty())
+ regexps[0] += "|";
+ regexps[0] += (*i).substr(1);
+ }
+ else {
+ if (! regexps[0].empty())
+ regexps[0] += "|";
+ regexps[0] += *i;
+ }
+
+ for (int i = 0; i < 2; i++) {
+ if (regexps[i].empty())
+ continue;
+
+ if (! predicate.empty())
+ predicate += logical_and ? "&" : "|";
+
+ int add_predicate = 0; // 1 adds /.../, 2 adds ///.../
+ if (i == 1) {
+ predicate += "!";
+ }
+ else if (add_account_short_masks) {
+ if (regexps[i].find(':') != std::string::npos ||
+ regexps[i].find('.') != std::string::npos ||
+ regexps[i].find('*') != std::string::npos ||
+ regexps[i].find('+') != std::string::npos ||
+ regexps[i].find('[') != std::string::npos ||
+ regexps[i].find('(') != std::string::npos) {
+ show_subtotal = true;
+ add_predicate = 1;
+ } else {
+ add_predicate = 2;
+ }
+ }
+ else {
+ add_predicate = 1;
+ }
+
+ if (i != 1 && command == "b" && account_regexp) {
+ if (! show_related && ! show_all_related) {
+ if (! display_predicate.empty())
+ display_predicate += "&";
+ if (! show_empty)
+ display_predicate += "T&";
+
+ if (add_predicate == 2)
+ display_predicate += "//";
+ display_predicate += "/(?:";
+ display_predicate += regexps[i];
+ display_predicate += ")/";
+ }
+ else if (! show_empty) {
+ if (! display_predicate.empty())
+ display_predicate += "&";
+ display_predicate += "T";
+ }
+ }
+
+ if (! account_regexp)
+ predicate += "/";
+ predicate += "/(?:";
+ predicate += regexps[i];
+ predicate += ")/";
+ }
+}
+
+void report_t::process_options(const std::string& command,
+ strings_list::iterator arg,
+ strings_list::iterator args_end)
+{
+ // Configure some other options depending on report type
+
+ if (command == "p" || command == "e" || command == "w") {
+ show_related =
+ show_all_related = true;
+ }
+ else if (command == "E") {
+ show_subtotal = true;
+ }
+ else if (show_related) {
+ if (command == "r") {
+ show_inverted = true;
+ } else {
+ show_subtotal = true;
+ show_all_related = true;
+ }
+ }
+
+ if (command != "b" && command != "r")
+ amount_t::keep_base = true;
+
+ // Process remaining command-line arguments
+
+ if (command != "e") {
+ // Treat the remaining command-line arguments as regular
+ // expressions, used for refining report results.
+
+ std::list<std::string>::iterator i = arg;
+ for (; i != args_end; i++)
+ if (*i == "--")
+ break;
+
+ if (i != arg)
+ regexps_to_predicate(command, arg, i, true,
+ (command == "b" && ! show_subtotal &&
+ display_predicate.empty()));
+ if (i != args_end && ++i != args_end)
+ regexps_to_predicate(command, i, args_end);
+ }
+
+ // Setup the default value for the display predicate
+
+ if (display_predicate.empty()) {
+ if (command == "b") {
+ if (! show_empty)
+ display_predicate = "T";
+ if (! show_subtotal) {
+ if (! display_predicate.empty())
+ display_predicate += "&";
+ display_predicate += "l<=1";
+ }
+ }
+ else if (command == "E") {
+ display_predicate = "t";
+ }
+ else if (command == "r" && ! show_empty) {
+ display_predicate = "a";
+ }
+ }
+
+ DEBUG_PRINT("ledger.config.predicates", "Predicate: " << predicate);
+ DEBUG_PRINT("ledger.config.predicates", "Display P: " << display_predicate);
+
+ // Setup the values of %t and %T, used in format strings
+
+ if (! amount_expr.empty())
+ ledger::amount_expr = amount_expr;
+ if (! total_expr.empty())
+ ledger::total_expr = total_expr;
+
+ // Now setup the various formatting strings
+
+ if (! date_format.empty())
+ datetime_t::date_format = date_format;
+
+ amount_t::keep_price = keep_price;
+ amount_t::keep_date = keep_date;
+ amount_t::keep_tag = keep_tag;
+
+ if (! report_period.empty() && ! sort_all)
+ entry_sort = true;
+}
+
+item_handler<transaction_t> *
+report_t::chain_xact_handlers(const std::string& command,
+ item_handler<transaction_t> * base_formatter,
+ journal_t * journal,
+ account_t * master,
+ std::list<item_handler<transaction_t> *>& ptrs)
+{
+ bool remember_components = false;
+
+ item_handler<transaction_t> * formatter = NULL;
+
+ ptrs.push_back(formatter = base_formatter);
+
+ // format_transactions write each transaction received to the
+ // output stream.
+ if (! (command == "b" || command == "E")) {
+ // truncate_entries cuts off a certain number of _entries_ from
+ // being displayed. It does not affect calculation.
+ if (head_entries || tail_entries)
+ ptrs.push_back(formatter =
+ new truncate_entries(formatter,
+ head_entries, tail_entries));
+
+ // filter_transactions will only pass through transactions
+ // matching the `display_predicate'.
+ if (! display_predicate.empty())
+ ptrs.push_back(formatter =
+ new filter_transactions(formatter,
+ display_predicate));
+
+ // calc_transactions computes the running total. When this
+ // appears will determine, for example, whether filtered
+ // transactions are included or excluded from the running total.
+ ptrs.push_back(formatter = new calc_transactions(formatter));
+
+ // component_transactions looks for reported transaction that
+ // match the given `descend_expr', and then reports the
+ // transactions which made up the total for that reported
+ // transaction.
+ if (! descend_expr.empty()) {
+ std::list<std::string> descend_exprs;
+
+ std::string::size_type beg = 0;
+ for (std::string::size_type pos = descend_expr.find(';');
+ pos != std::string::npos;
+ beg = pos + 1, pos = descend_expr.find(';', beg))
+ descend_exprs.push_back(std::string(descend_expr, beg, pos));
+ descend_exprs.push_back(std::string(descend_expr, beg));
+
+ for (std::list<std::string>::reverse_iterator i =
+ descend_exprs.rbegin();
+ i != descend_exprs.rend();
+ i++)
+ ptrs.push_back(formatter =
+ new component_transactions(formatter, *i));
+
+ remember_components = true;
+ }
+
+ // reconcile_transactions will pass through only those
+ // transactions which can be reconciled to a given balance
+ // (calculated against the transactions which it receives).
+ if (! reconcile_balance.empty()) {
+ std::time_t cutoff = now;
+ if (! reconcile_date.empty())
+ parse_date(reconcile_date.c_str(), &cutoff);
+ ptrs.push_back(formatter =
+ new reconcile_transactions
+ (formatter, value_t(reconcile_balance), cutoff));
+ }
+
+ // filter_transactions will only pass through transactions
+ // matching the `secondary_predicate'.
+ if (! secondary_predicate.empty())
+ ptrs.push_back(formatter =
+ new filter_transactions(formatter,
+ secondary_predicate));
+
+ // sort_transactions will sort all the transactions it sees, based
+ // on the `sort_order' value expression.
+ if (! sort_string.empty()) {
+ if (entry_sort)
+ ptrs.push_back(formatter =
+ new sort_entries(formatter, sort_string));
+ else
+ ptrs.push_back(formatter =
+ new sort_transactions(formatter, sort_string));
+ }
+
+ // changed_value_transactions adds virtual transactions to the
+ // list to account for changes in market value of commodities,
+ // which otherwise would affect the running total unpredictably.
+ if (show_revalued)
+ ptrs.push_back(formatter =
+ new changed_value_transactions(formatter,
+ show_revalued_only));
+
+ // collapse_transactions causes entries with multiple transactions
+ // to appear as entries with a subtotaled transaction for each
+ // commodity used.
+ if (show_collapsed)
+ ptrs.push_back(formatter = new collapse_transactions(formatter));
+
+ // subtotal_transactions combines all the transactions it receives
+ // into one subtotal entry, which has one transaction for each
+ // commodity in each account.
+ //
+ // period_transactions is like subtotal_transactions, but it
+ // subtotals according to time periods rather than totalling
+ // everything.
+ //
+ // dow_transactions is like period_transactions, except that it
+ // reports all the transactions that fall on each subsequent day
+ // of the week.
+ if (show_subtotal)
+ ptrs.push_back(formatter =
+ new subtotal_transactions(formatter, remember_components));
+
+ if (days_of_the_week)
+ ptrs.push_back(formatter =
+ new dow_transactions(formatter, remember_components));
+ else if (by_payee)
+ ptrs.push_back(formatter =
+ new by_payee_transactions(formatter, remember_components));
+
+ // interval_transactions groups transactions together based on a
+ // time period, such as weekly or monthly.
+ if (! report_period.empty()) {
+ ptrs.push_back(formatter =
+ new interval_transactions(formatter, report_period,
+ remember_components));
+ ptrs.push_back(formatter = new sort_transactions(formatter, "d"));
+ }
+ }
+
+ // invert_transactions inverts the value of the transactions it
+ // receives.
+ if (show_inverted)
+ ptrs.push_back(formatter = new invert_transactions(formatter));
+
+ // related_transactions will pass along all transactions related
+ // to the transaction received. If `show_all_related' is true,
+ // then all the entry's transactions are passed; meaning that if
+ // one transaction of an entry is to be printed, all the
+ // transaction for that entry will be printed.
+ if (show_related)
+ ptrs.push_back(formatter =
+ new related_transactions(formatter,
+ show_all_related));
+
+ // This filter_transactions will only pass through transactions
+ // matching the `predicate'.
+ if (! predicate.empty())
+ ptrs.push_back(formatter = new filter_transactions(formatter, predicate));
+
+ // budget_transactions takes a set of transactions from a data
+ // file and uses them to generate "budget transactions" which
+ // balance against the reported transactions.
+ //
+ // forecast_transactions is a lot like budget_transactions, except
+ // that it adds entries only for the future, and does not balance
+ // them against anything but the future balance.
+
+ if (budget_flags) {
+ budget_transactions * handler
+ = new budget_transactions(formatter, budget_flags);
+ handler->add_period_entries(journal->period_entries);
+ ptrs.push_back(formatter = handler);
+
+ // Apply this before the budget handler, so that only matching
+ // transactions are calculated toward the budget. The use of
+ // filter_transactions above will further clean the results so
+ // that no automated transactions that don't match the filter get
+ // reported.
+ if (! predicate.empty())
+ ptrs.push_back(formatter = new filter_transactions(formatter, predicate));
+ }
+ else if (! forecast_limit.empty()) {
+ forecast_transactions * handler
+ = new forecast_transactions(formatter, forecast_limit);
+ handler->add_period_entries(journal->period_entries);
+ ptrs.push_back(formatter = handler);
+
+ // See above, under budget_transactions.
+ if (! predicate.empty())
+ ptrs.push_back(formatter = new filter_transactions(formatter, predicate));
+ }
+
+ if (comm_as_payee)
+ ptrs.push_back(formatter = new set_comm_as_payee(formatter));
+ else if (code_as_payee)
+ ptrs.push_back(formatter = new set_code_as_payee(formatter));
+
+ return formatter;
+}
+
+} // namespace ledger
diff --git a/report.h b/report.h
new file mode 100644
index 00000000..14923ab5
--- /dev/null
+++ b/report.h
@@ -0,0 +1,80 @@
+#ifndef _REPORT_H
+#define _REPORT_H
+
+#include "ledger.h"
+#include "timing.h"
+
+#include <iostream>
+#include <memory>
+#include <list>
+
+namespace ledger {
+
+class report_t
+{
+ public:
+ std::string output_file;
+ std::string predicate;
+ std::string secondary_predicate;
+ std::string display_predicate;
+ std::string report_period;
+ std::string report_period_sort;
+ std::string format_string;
+ std::string sort_string;
+ std::string amount_expr;
+ std::string total_expr;
+ std::string descend_expr;
+ std::string forecast_limit;
+ std::string reconcile_balance;
+ std::string reconcile_date;
+ std::string date_format;
+
+ unsigned long budget_flags;
+ unsigned long pricing_leeway;
+
+ int head_entries;
+ int tail_entries;
+
+ bool show_collapsed;
+ bool show_subtotal;
+ bool show_totals;
+ bool show_related;
+ bool show_all_related;
+ bool show_inverted;
+ bool show_empty;
+ bool days_of_the_week;
+ bool by_payee;
+ bool comm_as_payee;
+ bool code_as_payee;
+ bool show_revalued;
+ bool show_revalued_only;
+ bool keep_price;
+ bool keep_date;
+ bool keep_tag;
+ bool entry_sort;
+ bool sort_all;
+
+ report_t();
+
+ void regexps_to_predicate(const std::string& command,
+ std::list<std::string>::const_iterator begin,
+ std::list<std::string>::const_iterator end,
+ const bool account_regexp = false,
+ const bool add_account_short_masks = false,
+ const bool logical_and = true);
+
+ void process_options(const std::string& command,
+ strings_list::iterator arg,
+ strings_list::iterator args_end);
+
+ item_handler<transaction_t> *
+ chain_xact_handlers(const std::string& command,
+ item_handler<transaction_t> * base_formatter,
+ journal_t * journal,
+ account_t * master,
+ std::list<item_handler<transaction_t> *>& ptrs);
+};
+
+} // namespace ledger
+
+#endif // _REPORT_H