diff options
-rwxr-xr-x | acprep | 2 | ||||
-rw-r--r-- | amount.cc | 2 | ||||
-rw-r--r-- | config.cc | 406 | ||||
-rw-r--r-- | config.h | 40 | ||||
-rw-r--r-- | format.h | 1 | ||||
-rw-r--r-- | journal.cc | 6 | ||||
-rw-r--r-- | main.cc | 615 | ||||
-rw-r--r-- | main.py | 13 | ||||
-rw-r--r-- | option.cc | 133 | ||||
-rw-r--r-- | option.h | 37 | ||||
-rw-r--r-- | python.cc | 2 |
11 files changed, 685 insertions, 572 deletions
@@ -1,6 +1,6 @@ #!/bin/sh -libtoolize -f +libtoolize --automake -fc aclocal autoheader touch AUTHORS ChangeLog @@ -1048,7 +1048,7 @@ using namespace ledger; void (amount_t::*parse_1)(std::istream& in) = &amount_t::parse; void (amount_t::*parse_2)(const std::string& str) = &amount_t::parse; -struct commodity_updater_wrap : commodity_t::updater_t +struct commodity_updater_wrap : public commodity_t::updater_t { PyObject * self; commodity_updater_wrap(PyObject * self_) : self(self_) {} @@ -1,9 +1,12 @@ #include "config.h" #include "option.h" +#include "quotes.h" + +#include <fstream> namespace ledger { -config_t * config = NULL; +config_t config; std::string bal_fmt = "%20T %2_%-n\n"; std::string reg_fmt @@ -30,12 +33,255 @@ config_t::config_t() show_collapsed = false; show_subtotal = false; show_related = false; + show_all_related = false; show_inverted = false; show_empty = false; days_of_the_week = false; show_revalued = false; show_revalued_only = false; download_quotes = false; + use_cache = false; + cache_dirty = false; + interval_begin = 0; +} + +static void +regexps_to_predicate(config_t& config, + std::deque<std::string>::const_iterator begin, + std::deque<std::string>::const_iterator end, + const bool account_regexp = false, + const bool add_account_short_masks = false) +{ + std::string regexps[2]; + + // Treat the remaining command-line arguments as regular + // expressions, used for refining report results. + + for (std::deque<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 (! config.predicate.empty()) + config.predicate += "&"; + + if (i == 1) { + config.predicate += "!"; + } + else if (add_account_short_masks) { + if (regexps[i].find(':') != std::string::npos) { + config.show_subtotal = true; + } else { + if (! config.display_predicate.empty()) + config.display_predicate += "&"; + else if (! config.show_empty) + config.display_predicate += "T&"; + + config.display_predicate += "///(?:"; + config.display_predicate += regexps[i]; + config.display_predicate += ")/"; + } + } + + if (! account_regexp) + config.predicate += "/"; + config.predicate += "/(?:"; + config.predicate += regexps[i]; + config.predicate += ")/"; + } +} + +void config_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") { + 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; + } + } + + // Process remaining command-line arguments + + if (command != "e") { + // Treat the remaining command-line arguments as regular + // expressions, used for refining report results. + + std::deque<std::string>::iterator i = arg; + for (; i != args_end; i++) + if (*i == "--") + break; + + regexps_to_predicate(*this, arg, i, true, + (command == "b" && ! show_subtotal && + display_predicate.empty())); + if (i != args_end) + regexps_to_predicate(*this, 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"; + } + } + + // Parse the sort_string + + if (! sort_order.get() && ! sort_string.empty()) { + try { + std::istringstream stream(sort_string); + sort_order.reset(parse_value_expr(stream)); + if (stream.peek() != -1) { + throw value_expr_error(std::string("Unexpected character '") + + char(stream.peek()) + "'"); + } + else if (! sort_order.get()) { + throw error("Failed to parse sort criteria!"); + } + } + catch (const value_expr_error& err) { + throw error(std::string("In sort criteria: ") + err.what()); + } + } + + // Setup the values of %t and %T, used in format strings + + try { + if (! format_t::value_expr) + format_t::value_expr = parse_value_expr(value_expr); + if (! format_t::value_expr) + throw value_expr_error(std::string("Failed to parse '") + + value_expr + "'"); + } + catch (const value_expr_error& err) { + throw error(std::string("In value expression to -t: ") + err.what()); + } + + try { + if (! format_t::total_expr) + format_t::total_expr = parse_value_expr(total_expr); + if (! format_t::total_expr) + throw value_expr_error(std::string("Failed to parse '") + + total_expr + "'"); + } + catch (const value_expr_error& err) { + throw error(std::string("In value expression to -T: ") + err.what()); + } + + // If downloading is to be supported, configure the updater + + if (! commodity_t::updater && download_quotes) + commodity_t::updater = new quotes_by_script(price_db, + pricing_leeway, + cache_dirty); + + // Configure the output stream + + if (! output_stream.get() && ! output_file.empty()) + output_stream.reset(new std::ofstream(output_file.c_str())); + + // Parse the interval specifier, if provided + + if (! report_interval && ! interval_text.empty()) { + try { + std::istringstream stream(interval_text); + std::time_t begin = -1, end = -1; + + report_interval = interval_t::parse(stream, &begin, &end); + + if (begin != -1) { + interval_begin = begin; + + if (! predicate.empty()) + predicate += "&"; + char buf[32]; + std::sprintf(buf, "d>=%lu", begin); + predicate += buf; + } + + if (end != -1) { + if (! predicate.empty()) + predicate += "&"; + char buf[32]; + std::sprintf(buf, "d<%lu", end); + predicate += buf; + } + } + catch (const interval_expr_error& err) { + throw error(std::string("In interval (-z) specifier: ") + err.what()); + } + } + + if (format_t::date_format.empty() && ! date_format.empty()) + format_t::date_format = date_format; + + // Compile the format strings + + const char * f; + if (! format_string.empty()) + f = format_string.c_str(); + else if (command == "b") + f = bal_fmt.c_str(); + else if (command == "r") + f = reg_fmt.c_str(); + else if (command == "E") + f = equity_fmt.c_str(); + else + f = print_fmt.c_str(); + + std::string first_line_format; + std::string next_lines_format; + + if (const char * p = std::strstr(f, "%/")) { + first_line_format = std::string(f, 0, p - f); + next_lines_format = std::string(p + 2); + } else { + first_line_format = next_lines_format = f; + } + + format.reset(first_line_format); + nformat.reset(next_lines_format); } static void show_version(std::ostream& out) @@ -115,8 +361,6 @@ Commands:\n\ // // Basic options -DEF_OPT_HANDLERS(); - OPT_BEGIN(help, "h") { option_help(std::cout); throw 0; @@ -128,31 +372,31 @@ OPT_BEGIN(version, "v") { } OPT_END(version); OPT_BEGIN(init, "i:") { - config->init_file = optarg; + config.init_file = optarg; } OPT_END(init); OPT_BEGIN(file, "f:") { - config->data_file = optarg; + config.data_file = optarg; } OPT_END(file); OPT_BEGIN(cache, ":") { - config->cache_file = optarg; + config.cache_file = optarg; } OPT_END(cache); OPT_BEGIN(output, "o:") { if (std::string(optarg) != "-") - config->output_file = optarg; + config.output_file = optarg; } OPT_END(output); OPT_BEGIN(set_price, "z:") { if (std::strchr(optarg, '=')) - config->price_settings.push_back(optarg); + config.price_settings.push_back(optarg); else std::cerr << "Error: Invalid price setting: " << optarg << std::endl; } OPT_END(set_price); OPT_BEGIN(account, "a:") { - config->account = optarg; + config.account = optarg; } OPT_END(account); ////////////////////////////////////////////////////////////////////// @@ -160,43 +404,43 @@ OPT_BEGIN(account, "a:") { // Report filtering OPT_BEGIN(begin_date, "b:") { - if (! config->predicate.empty()) - config->predicate += "&"; - config->predicate += "d>=["; - config->predicate += optarg; - config->predicate += "]"; + if (! config.predicate.empty()) + config.predicate += "&"; + config.predicate += "d>=["; + config.predicate += optarg; + config.predicate += "]"; } OPT_END(begin_date); OPT_BEGIN(end_date, "e:") { - if (! config->predicate.empty()) - config->predicate += "&"; - config->predicate += "d<["; - config->predicate += optarg; - config->predicate += "]"; + if (! config.predicate.empty()) + config.predicate += "&"; + config.predicate += "d<["; + config.predicate += optarg; + config.predicate += "]"; } OPT_END(end_date); OPT_BEGIN(current, "c") { - if (! config->predicate.empty()) - config->predicate += "&"; - config->predicate += "d<=m"; + if (! config.predicate.empty()) + config.predicate += "&"; + config.predicate += "d<=m"; } OPT_END(current); OPT_BEGIN(cleared, "C") { - if (! config->predicate.empty()) - config->predicate += "&"; - config->predicate += "X"; + if (! config.predicate.empty()) + config.predicate += "&"; + config.predicate += "X"; } OPT_END(cleared); OPT_BEGIN(uncleared, "U") { - if (! config->predicate.empty()) - config->predicate += "&"; - config->predicate += "!X"; + if (! config.predicate.empty()) + config.predicate += "&"; + config.predicate += "!X"; } OPT_END(uncleared); OPT_BEGIN(real, "R") { - if (! config->predicate.empty()) - config->predicate += "&"; - config->predicate += "R"; + if (! config.predicate.empty()) + config.predicate += "&"; + config.predicate += "R"; } OPT_END(real); ////////////////////////////////////////////////////////////////////// @@ -204,11 +448,11 @@ OPT_BEGIN(real, "R") { // Output customization OPT_BEGIN(format, "F:") { - config->format_string = optarg; + config.format_string = optarg; } OPT_END(format); OPT_BEGIN(date_format, "y:") { - config->date_format = optarg; + config.date_format = optarg; } OPT_END(date_format); OPT_BEGIN(balance_format, ":") { @@ -236,77 +480,77 @@ OPT_BEGIN(equity_format, ":") { } OPT_END(equity_format); OPT_BEGIN(empty, "E") { - config->show_empty = true; + config.show_empty = true; } OPT_END(empty); OPT_BEGIN(collapse, "n") { - config->show_collapsed = true; + config.show_collapsed = true; } OPT_END(collapse); OPT_BEGIN(subtotal, "s") { - config->show_subtotal = true; + config.show_subtotal = true; } OPT_END(subtotal); OPT_BEGIN(sort, "S:") { - config->sort_string = optarg; + config.sort_string = optarg; } OPT_END(sort); OPT_BEGIN(related, "r") { - config->show_related = true; + config.show_related = true; } OPT_END(related); OPT_BEGIN(interval, "p:") { - config->interval_text = optarg; + config.interval_text = optarg; } OPT_END(interval); OPT_BEGIN(weekly, "W") { - config->interval_text = "weekly"; + config.interval_text = "weekly"; } OPT_END(weekly); OPT_BEGIN(dow, "") { - config->days_of_the_week = true; + config.days_of_the_week = true; } OPT_END(dow); OPT_BEGIN(monthly, "M") { - config->interval_text = "monthly"; + config.interval_text = "monthly"; } OPT_END(monthly); OPT_BEGIN(yearly, "Y") { - config->interval_text = "yearly"; + config.interval_text = "yearly"; } OPT_END(yearly); OPT_BEGIN(limit, "l:") { - if (! config->predicate.empty()) - config->predicate += "&"; - config->predicate += "("; - config->predicate += optarg; - config->predicate += ")"; + if (! config.predicate.empty()) + config.predicate += "&"; + config.predicate += "("; + config.predicate += optarg; + config.predicate += ")"; } OPT_END(limit); OPT_BEGIN(display, "d:") { - if (! config->display_predicate.empty()) - config->display_predicate += "&"; - config->display_predicate += "("; - config->display_predicate += optarg; - config->display_predicate += ")"; + if (! config.display_predicate.empty()) + config.display_predicate += "&"; + config.display_predicate += "("; + config.display_predicate += optarg; + config.display_predicate += ")"; } OPT_END(display); OPT_BEGIN(value, "t:") { - config->value_expr = optarg; + config.value_expr = optarg; } OPT_END(value); OPT_BEGIN(total, "T:") { - config->total_expr = optarg; + config.total_expr = optarg; } OPT_END(total); OPT_BEGIN(value_data, "j") { - config->value_expr = "S" + config->value_expr; - config->format_string = plot_value_fmt; + config.value_expr = "S" + config.value_expr; + config.format_string = plot_value_fmt; } OPT_END(value_data); OPT_BEGIN(total_data, "J") { - config->total_expr = "S" + config->total_expr; - config->format_string = plot_total_fmt; + config.total_expr = "S" + config.total_expr; + config.format_string = plot_total_fmt; } OPT_END(total_data); ////////////////////////////////////////////////////////////////////// @@ -314,60 +558,60 @@ OPT_BEGIN(total_data, "J") { // Commodity reporting OPT_BEGIN(price_db, "P:") { - config->price_db = optarg; + config.price_db = optarg; } OPT_END(price_db); OPT_BEGIN(price_exp, "L:") { - config->pricing_leeway = std::atol(optarg) * 60; + config.pricing_leeway = std::atol(optarg) * 60; } OPT_END(price_exp); OPT_BEGIN(download, "Q") { - config->download_quotes = true; + config.download_quotes = true; } OPT_END(download); OPT_BEGIN(quantity, "O") { - config->value_expr = "a"; - config->total_expr = "O"; + config.value_expr = "a"; + config.total_expr = "O"; } OPT_END(quantity); OPT_BEGIN(basis, "B") { - config->value_expr = "c"; - config->total_expr = "C"; + config.value_expr = "c"; + config.total_expr = "C"; } OPT_END(basis); OPT_BEGIN(market, "V") { - config->show_revalued = true; + config.show_revalued = true; - config->value_expr = "v"; - config->total_expr = "V"; + config.value_expr = "v"; + config.total_expr = "V"; } OPT_END(market); OPT_BEGIN(gain, "G") { - config->show_revalued = - config->show_revalued_only = true; + config.show_revalued = + config.show_revalued_only = true; - config->value_expr = "a"; - config->total_expr = "G"; + config.value_expr = "a"; + config.total_expr = "G"; } OPT_END(gain); OPT_BEGIN(average, "A") { - config->value_expr = "a"; - config->total_expr = "MO"; + config.value_expr = "a"; + config.total_expr = "MO"; } OPT_END(average); OPT_BEGIN(deviation, "D") { - config->value_expr = "a"; - config->total_expr = "DMO"; + config.value_expr = "a"; + config.total_expr = "DMO"; } OPT_END(deviation); OPT_BEGIN(trend, "X") { - config->value_expr = "a"; - config->total_expr = "MDMO"; + config.value_expr = "a"; + config.total_expr = "MDMO"; } OPT_END(trend); OPT_BEGIN(weighted_trend, "Z") { - config->value_expr = "a"; - config->total_expr + config.value_expr = "a"; + config.total_expr = "MD(MO/(1+(((m-d)/(30*86400))<0?0:((m-d)/(30*86400)))))"; } OPT_END(weighted_trend); @@ -2,6 +2,10 @@ #define _CONFIG_H #include "ledger.h" +#include "option.h" +#include "valexpr.h" +#include "datetime.h" +#include "format.h" #include <iostream> #include <memory> @@ -17,6 +21,8 @@ extern std::string equity_fmt; struct config_t { + // These options can all be set used text fields. + strings_list price_settings; std::string init_file; std::string data_file; @@ -36,6 +42,7 @@ struct config_t bool show_collapsed; bool show_subtotal; bool show_related; + bool show_all_related; bool show_inverted; bool show_empty; bool days_of_the_week; @@ -43,13 +50,44 @@ struct config_t bool show_revalued_only; bool download_quotes; + // These settings require processing of the above. + + bool use_cache; + bool cache_dirty; + interval_t report_interval; + std::time_t interval_begin; + format_t format; + format_t nformat; + + std::auto_ptr<value_expr_t> sort_order; + std::auto_ptr<std::ostream> output_stream; + config_t(); + + void process_options(const std::string& command, + strings_list::iterator arg, + strings_list::iterator args_end); }; -extern config_t * config; +extern config_t config; void option_help(std::ostream& out); +struct declared_option_handler : public option_handler { + declared_option_handler(const std::string& label, + const std::string& opt_chars) { + register_option(label, opt_chars, *this); + } +}; + +#define OPT_BEGIN(tag, chars) \ + static struct opt_ ## tag ## _handler \ + : public declared_option_handler { \ + opt_ ## tag ## _handler() : declared_option_handler(#tag, chars) {} \ + virtual void operator()(const char * optarg) + +#define OPT_END(tag) } opt_ ## tag ## _handler_obj + } // namespace ledger #endif // _CONFIG_H @@ -57,6 +57,7 @@ struct format_t static value_expr_t * value_expr; static value_expr_t * total_expr; + format_t() : elements(NULL) {} format_t(const std::string& _format) : elements(NULL) { reset(_format); } @@ -139,8 +139,12 @@ account_t * account_t::find_account(const std::string& name, if (i == accounts.end()) { if (! auto_create) return NULL; + account = new account_t(this, first); - accounts.insert(accounts_pair(first, account)); + + std::pair<accounts_map::iterator, bool> result + = accounts.insert(accounts_pair(first, account)); + assert(result.second); } else { account = (*i).second; } @@ -31,14 +31,10 @@ using namespace ledger; #include <ctime> namespace { - bool cache_dirty = false; - TIMER_DEF(write_cache, "writing cache file"); TIMER_DEF(report_gen, "generation of final report"); - TIMER_DEF(handle_options, "configuring based on options"); TIMER_DEF(parse_files, "parsing ledger files"); - TIMER_DEF(process_env, "processing environment"); - TIMER_DEF(process_args, "processing command-line arguments"); + TIMER_DEF(process_opts, "processing args and environment"); TIMER_DEF(read_cache, "reading cache file"); } @@ -76,85 +72,166 @@ namespace std { #endif -static void -regexps_to_predicate(std::deque<std::string>::const_iterator begin, - std::deque<std::string>::const_iterator end, - config_t * config, - const bool account_regexp = false, - const bool add_account_short_masks = false) +void parse_ledger_data(journal_t * journal, + parser_t * text_parser, + parser_t * cache_parser) { - std::string regexps[2]; - - // Treat the remaining command-line arguments as regular - // expressions, used for refining report results. - - for (std::deque<std::string>::const_iterator i = begin; - i != end; - i++) - if ((*i)[0] == '-') { - if (! regexps[1].empty()) - regexps[1] += "|"; - regexps[1] += (*i).substr(1); + TIMER_START(parse_files); + + int entry_count = 0; + + if (! config.init_file.empty()) { + if (parse_journal_file(config.init_file, journal)) + throw error("Entries not allowed in initialization file"); + journal->sources.pop_front(); // remove init file + } + + if (config.use_cache && ! config.cache_file.empty() && + ! config.data_file.empty()) { + config.cache_dirty = true; + if (access(config.cache_file.c_str(), R_OK) != -1) { + std::ifstream stream(config.cache_file.c_str()); + if (cache_parser->test(stream)) { + entry_count += cache_parser->parse(stream, journal, NULL, + &config.data_file); + if (entry_count > 0) + config.cache_dirty = false; + } } - else if ((*i)[0] == '+') { - if (! regexps[0].empty()) - regexps[0] += "|"; - regexps[0] += (*i).substr(1); + } + + if (entry_count == 0 && ! config.data_file.empty()) { + account_t * account = NULL; + if (! config.account.empty()) + account = journal->find_account(config.account); + + if (config.data_file == "-") { + config.use_cache = false; + entry_count += parse_journal(std::cin, journal, account); + } else { + entry_count += parse_journal_file(config.data_file, journal, account); } - else { - if (! regexps[0].empty()) - regexps[0] += "|"; - regexps[0] += *i; + + if (! config.price_db.empty()) + if (parse_journal_file(config.price_db, journal)) + throw error("Entries not allowed in price history file"); + } + + for (strings_list::iterator i = config.price_settings.begin(); + i != config.price_settings.end(); + i++) { + std::string conversion = "C "; + conversion += *i; + int i = conversion.find('='); + if (i != -1) { + conversion[i] = ' '; + std::istringstream stream(conversion); + text_parser->parse(stream, journal, journal->master); } + } - for (int i = 0; i < 2; i++) { - if (regexps[i].empty()) - continue; + if (entry_count == 0) + throw error("Please specify ledger file using -f," + " or LEDGER_FILE environment variable."); - if (! config->predicate.empty()) - config->predicate += "&"; + VALIDATE(journal->valid()); - if (i == 1) { - config->predicate += "!"; - } - else if (add_account_short_masks) { - if (regexps[i].find(':') != std::string::npos) { - config->show_subtotal = true; - } else { - if (! config->display_predicate.empty()) - config->display_predicate += "&"; - else if (! config->show_empty) - config->display_predicate += "T&"; - - config->display_predicate += "///(?:"; - config->display_predicate += regexps[i]; - config->display_predicate += ")/"; - } - } + TIMER_STOP(parse_files); +} + +item_handler<transaction_t> * +chain_formatters(const std::string& command, + item_handler<transaction_t> * base_formatter) +{ + std::auto_ptr<item_handler<transaction_t> > formatter; + + // format_transactions write each transaction received to the + // output stream. + if (command == "b" || command == "E") { + formatter.reset(base_formatter); + } else { + formatter.reset(base_formatter); - if (! account_regexp) - config->predicate += "/"; - config->predicate += "/(?:"; - config->predicate += regexps[i]; - config->predicate += ")/"; + // filter_transactions will only pass through transactions + // matching the `display_predicate'. + if (! config.display_predicate.empty()) + formatter.reset(new filter_transactions(formatter.release(), + config.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. + formatter.reset(new calc_transactions(formatter.release(), + config.show_inverted)); + + // sort_transactions will sort all the transactions it sees, based + // on the `sort_order' value expression. + if (config.sort_order.get()) + formatter.reset(new sort_transactions(formatter.release(), + config.sort_order.get())); + + // 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 (config.show_revalued) + formatter.reset(new changed_value_transactions(formatter.release(), + config.show_revalued_only)); + + // collapse_transactions causes entries with multiple transactions + // to appear as entries with a subtotaled transaction for each + // commodity used. + if (config.show_collapsed) + formatter.reset(new collapse_transactions(formatter.release())); + + // subtotal_transactions combines all the transactions it receives + // into one subtotal entry, which has one transaction for each + // commodity in each account. + // + // interval_transactions is like subtotal_transactions, but it + // subtotals according to time intervals rather than totalling + // everything. + // + // dow_transactions is like interval_transactions, except that it + // reports all the transactions that fall on each subsequent day + // of the week. + if (config.show_subtotal) + formatter.reset(new subtotal_transactions(formatter.release())); + else if (config.report_interval) + formatter.reset(new interval_transactions(formatter.release(), + config.report_interval, + config.interval_begin)); + else if (config.days_of_the_week) + formatter.reset(new dow_transactions(formatter.release())); } + + // 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 (config.show_related) + formatter.reset(new related_transactions(formatter.release(), + config.show_all_related)); + + // This filter_transactions will only pass through transactions + // matching the `predicate'. + if (! config.predicate.empty()) + formatter.reset(new filter_transactions(formatter.release(), + config.predicate)); + + return formatter.release(); } int parse_and_report(int argc, char * argv[], char * envp[]) { std::auto_ptr<journal_t> journal(new journal_t); - // Initialize the global configuration object for this run - - std::auto_ptr<config_t> global_config(new config_t); - config = global_config.get(); + // Parse command-line arguments, and those set in the environment - // Parse command-line arguments - - TIMER_START(process_args); + TIMER_START(process_opts); std::deque<std::string> args; - process_arguments(argc, argv, false, args); + process_arguments(argc - 1, argv + 1, false, args); if (args.empty()) { option_help(std::cerr); @@ -162,13 +239,7 @@ int parse_and_report(int argc, char * argv[], char * envp[]) } strings_list::iterator arg = args.begin(); - TIMER_STOP(process_args); - - bool use_cache = config->data_file.empty(); - - // Process options from the environment - - TIMER_START(process_env); + config.use_cache = config.data_file.empty(); process_environment(envp, "LEDGER_"); @@ -183,13 +254,10 @@ int parse_and_report(int argc, char * argv[], char * envp[]) process_option("price-exp", p); #endif - TIMER_STOP(process_env); + TIMER_STOP(process_opts); - // Parse ledger files - - TIMER_START(parse_files); + // Parse initialization files, ledger data, price database, etc. - // Setup the parsers std::auto_ptr<binary_parser_t> bin_parser(new binary_parser_t); #ifdef READ_GNUCASH std::auto_ptr<gnucash_parser_t> gnucash_parser(new gnucash_parser_t); @@ -204,81 +272,14 @@ int parse_and_report(int argc, char * argv[], char * envp[]) register_parser(qif_parser.get()); register_parser(text_parser.get()); - int entry_count = 0; + parse_ledger_data(journal.get(), text_parser.get(), bin_parser.get()); - try { - if (! config->init_file.empty()) { - if (parse_journal_file(config->init_file, journal.get())) - throw error("Entries not allowed in initialization file"); - journal->sources.pop_front(); // remove init file - } - - if (use_cache && ! config->cache_file.empty() && - ! config->data_file.empty()) { - cache_dirty = true; - if (access(config->cache_file.c_str(), R_OK) != -1) { - std::ifstream stream(config->cache_file.c_str()); - if (bin_parser->test(stream)) { - entry_count += bin_parser->parse(stream, journal.get(), NULL, - &config->data_file); - if (entry_count > 0) - cache_dirty = false; - } - } - } - - if (entry_count == 0 && ! config->data_file.empty()) { - account_t * account = NULL; - if (! config->account.empty()) - account = journal->find_account(config->account); - - if (config->data_file == "-") { - use_cache = false; - entry_count += text_parser->parse(std::cin, journal.get(), account); - } else { - entry_count += parse_journal_file(config->data_file, journal.get(), - account); - } - - if (! config->price_db.empty()) - if (parse_journal_file(config->price_db, journal.get())) - throw error("Entries not allowed in price history file"); - } - - for (strings_list::iterator i = config->price_settings.begin(); - i != config->price_settings.end(); - i++) { - std::string conversion = "C "; - conversion += *i; - int i = conversion.find('='); - if (i != -1) { - conversion[i] = ' '; - std::istringstream stream(conversion); - text_parser->parse(stream, journal.get(), journal->master); - } - } - } - catch (error& err) { - std::cerr << "Fatal: " << err.what() << std::endl; - return 1; - } - - if (entry_count == 0) { - std::cerr << "Please specify ledger file(s) using -f option " - << "or LEDGER environment variable." << std::endl; - return 1; - } - - VALIDATE(journal->valid()); - - TIMER_STOP(parse_files); - - // Read the command word, and then check and simplify it + // Read the command word, canonicalize it to its one letter form, + // then configure the system based on the kind of report to be + // generated std::string command = *arg++; - TIMER_START(handle_options); - if (command == "balance" || command == "bal" || command == "b") command = "b"; else if (command == "register" || command == "reg" || command == "r") @@ -290,294 +291,35 @@ int parse_and_report(int argc, char * argv[], char * envp[]) else if (command == "equity") command = "E"; else { - std::cerr << "Error: Unrecognized command '" << command << "'." - << std::endl; - return 1; + std::ostringstream msg; + msg << "Unrecognized command '" << command << "'"; + throw error(msg.str()); } - // Configure some other options depending on report type - - bool show_all_related = false; - - if (command == "p" || command == "e") { - config->show_related = - show_all_related = true; - } - else if (command == "E") { - config->show_subtotal = true; - } - else if (config->show_related) { - if (command == "r") { - config->show_inverted = true; - } else { - config->show_subtotal = true; - show_all_related = true; - } - } - - // Process remaining command-line arguments + config.process_options(command, arg, args.end()); std::auto_ptr<entry_t> new_entry; if (command == "e") { new_entry.reset(journal->derive_entry(arg, args.end())); if (! new_entry.get()) return 1; - } else { - // Treat the remaining command-line arguments as regular - // expressions, used for refining report results. - - std::deque<std::string>::iterator i = args.begin(); - for (; i != args.end(); i++) - if (*i == "--") - break; - - regexps_to_predicate(arg, i, config, true, - command == "b" && ! config->show_subtotal && - config->display_predicate.empty()); - if (i != args.end()) - regexps_to_predicate(i, args.end(), config); - } - - // Setup default value for the display predicate - - if (config->display_predicate.empty()) { - if (command == "b") { - if (! config->show_empty) - config->display_predicate = "T"; - if (! config->show_subtotal) { - if (! config->display_predicate.empty()) - config->display_predicate += "&"; - config->display_predicate += "l<=1"; - } - } - else if (command == "E") { - config->display_predicate = "t"; - } - } - - // Compile sorting criteria - - std::auto_ptr<value_expr_t> sort_order; - - if (! config->sort_string.empty()) { - try { - std::istringstream stream(config->sort_string); - sort_order.reset(parse_value_expr(stream)); - if (stream.peek() != -1) { - std::ostringstream err; - err << "Unexpected character '" << char(stream.peek()) << "'"; - throw value_expr_error(err.str()); - } - else if (! sort_order.get()) { - std::cerr << "Failed to parse sort criteria!" << std::endl; - return 1; - } - } - catch (const value_expr_error& err) { - std::cerr << "Error in sort criteria: " << err.what() << std::endl; - return 1; - } } - // Setup the values of %t and %T, used in format strings - - try { - format_t::value_expr = parse_value_expr(config->value_expr); - } - catch (const value_expr_error& err) { - std::cerr << "Error in amount (-t) specifier: " << err.what() - << std::endl; - return 1; - } - - try { - format_t::total_expr = parse_value_expr(config->total_expr); - } - catch (const value_expr_error& err) { - std::cerr << "Error in total (-T) specifier: " << err.what() - << std::endl; - return 1; - } - - // Setup local and global variables, depending on config settings. - - std::auto_ptr<std::ostream> output_stream; - - interval_t report_interval; - std::time_t interval_begin = 0; - - if (config->download_quotes) - commodity_t::updater = new quotes_by_script(config->price_db, - config->pricing_leeway, - cache_dirty); - - if (! config->output_file.empty()) - output_stream.reset(new std::ofstream(config->output_file.c_str())); - -#define OUT() (output_stream.get() ? *output_stream : std::cout) - - if (! config->interval_text.empty()) { - try { - std::istringstream stream(config->interval_text); - std::time_t begin = -1, end = -1; - - report_interval = interval_t::parse(stream, &begin, &end); - - if (begin != -1) { - interval_begin = begin; - - if (! config->predicate.empty()) - config->predicate += "&"; - char buf[32]; - std::sprintf(buf, "d>=%lu", begin); - config->predicate += buf; - } - - if (end != -1) { - if (! config->predicate.empty()) - config->predicate += "&"; - char buf[32]; - std::sprintf(buf, "d<%lu", end); - config->predicate += buf; - } - } - catch (const interval_expr_error& err) { - std::cerr << "Error in interval (-z) specifier: " << err.what() - << std::endl; - return 1; - } - } - - if (! config->date_format.empty()) - format_t::date_format = config->date_format; - -#ifdef DEBUG_ENABLED - DEBUG_PRINT("ledger.main.predicates", "predicate: " << config->predicate); - DEBUG_PRINT("ledger.main.predicates", - "disp-pred: " << config->display_predicate); -#endif - - // Compile the format strings - - const char * f; - if (! config->format_string.empty()) - f = config->format_string.c_str(); - else if (command == "b") - f = bal_fmt.c_str(); - else if (command == "r") - f = reg_fmt.c_str(); - else if (command == "E") - f = equity_fmt.c_str(); - else - f = print_fmt.c_str(); - - std::string first_line_format; - std::string next_lines_format; - - if (const char * p = std::strstr(f, "%/")) { - first_line_format = std::string(f, 0, p - f); - next_lines_format = std::string(p + 2); - } else { - first_line_format = next_lines_format = f; - } - - format_t format(first_line_format); - format_t nformat(next_lines_format); - - TIMER_STOP(handle_options); - // Walk the entries based on the report type and the options TIMER_START(report_gen); - // Stack up all the formatter needed to fulfills the user's - // requests. Some of these are order dependent, in terms of - // whether calc_transactions occurs before or after them. - std::auto_ptr<item_handler<transaction_t> > formatter; - // format_transactions write each transaction received to the - // output stream. if (command == "b" || command == "E") { -#ifdef DEBUG_ENABLED - if (DEBUG("ledger.balance.items")) { - formatter.reset(new format_transactions(OUT(), format, nformat)); - formatter.reset(new set_account_value(formatter.release())); - } else -#endif - formatter.reset(new set_account_value); + formatter.reset(chain_formatters(command, new set_account_value)); } else { - formatter.reset(new format_transactions(OUT(), format, nformat)); - - // filter_transactions will only pass through transactions - // matching the `display_predicate'. - if (! config->display_predicate.empty()) - formatter.reset(new filter_transactions(formatter.release(), - config->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. - formatter.reset(new calc_transactions(formatter.release(), - config->show_inverted)); - - // sort_transactions will sort all the transactions it sees, based - // on the `sort_order' value expression. - if (sort_order.get()) - formatter.reset(new sort_transactions(formatter.release(), - sort_order.get())); - - // 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 (config->show_revalued) - formatter.reset(new changed_value_transactions(formatter.release(), - config->show_revalued_only)); - - // collapse_transactions causes entries with multiple transactions - // to appear as entries with a subtotaled transaction for each - // commodity used. - if (config->show_collapsed) - formatter.reset(new collapse_transactions(formatter.release())); - - // subtotal_transactions combines all the transactions it receives - // into one subtotal entry, which has one transaction for each - // commodity in each account. - // - // interval_transactions is like subtotal_transactions, but it - // subtotals according to time intervals rather than totalling - // everything. - // - // dow_transactions is like interval_transactions, except that it - // reports all the transactions that fall on each subsequent day - // of the week. - if (config->show_subtotal) - formatter.reset(new subtotal_transactions(formatter.release())); - else if (report_interval) - formatter.reset(new interval_transactions(formatter.release(), - report_interval, - interval_begin)); - else if (config->days_of_the_week) - formatter.reset(new dow_transactions(formatter.release())); + std::ostream& out(config.output_stream.get() ? + *config.output_stream : std::cout); + formatter.reset(chain_formatters(command, + new format_transactions(out, config.format, config.nformat))); } - // 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 (config->show_related) - formatter.reset(new related_transactions(formatter.release(), - show_all_related)); - - // This filter_transactions will only pass through transactions - // matching the `predicate'. - if (! config->predicate.empty()) - formatter.reset(new filter_transactions(formatter.release(), - config->predicate)); - - // Once the filters are chained, walk `journal's entries and start - // feeding each transaction that matches `predicate' to the chain. if (command == "e") walk_transactions(new_entry->transactions, *formatter); else @@ -585,32 +327,32 @@ int parse_and_report(int argc, char * argv[], char * envp[]) formatter->flush(); - // At this point all printing is finished if doing a register - // report; but if it's a balance or equity report, we've only - // finished calculating the totals and there is still reporting to - // be done. + // For the balance and equity reports, output the sum totals. + + std::ostream& out(config.output_stream.get() ? + *config.output_stream : std::cout); if (command == "b") { - format_account acct_formatter(OUT(), format, config->display_predicate); + format_account acct_formatter(out, config.format, + config.display_predicate); sum_accounts(journal->master); - walk_accounts(journal->master, acct_formatter, sort_order.get()); + walk_accounts(journal->master, acct_formatter, config.sort_order.get()); acct_formatter.flush(); if (journal->master->data) { ACCT_DATA(journal->master)->value = ACCT_DATA(journal->master)->total; if (ACCT_DATA(journal->master)->dflags & ACCOUNT_TO_DISPLAY) { - std::string end_format = "--------------------\n"; - format.reset(end_format + f); - format.format_elements(OUT(), details_t(journal->master)); + out << "--------------------\n"; + config.format.format_elements(out, details_t(journal->master)); } } } else if (command == "E") { - format_equity acct_formatter(OUT(), format, nformat, - config->display_predicate); + format_equity acct_formatter(out, config.format, config.nformat, + config.display_predicate); sum_accounts(journal->master); - walk_accounts(journal->master, acct_formatter, sort_order.get()); + walk_accounts(journal->master, acct_formatter, config.sort_order.get()); acct_formatter.flush(); } @@ -626,12 +368,12 @@ int parse_and_report(int argc, char * argv[], char * envp[]) TIMER_STOP(report_gen); - // Save the cache, if need be + // Write out the binary cache, if need be TIMER_START(write_cache); - if (use_cache && cache_dirty && ! config->cache_file.empty()) { - std::ofstream stream(config->cache_file.c_str()); + if (config.use_cache && config.cache_dirty && ! config.cache_file.empty()) { + std::ofstream stream(config.cache_file.c_str()); write_binary_journal(stream, journal.get(), &journal->sources); } @@ -649,11 +391,12 @@ int main(int argc, char * argv[], char * envp[]) try { status = parse_and_report(argc, argv, envp); } + catch (error& err) { + std::cerr << "Error: " << err.what() << std::endl; + status = 1; + } catch (int& val) { -#if DEBUG_LEVEL >= BETA - shutdown(); -#endif - return val; + status = val; } #if DEBUG_LEVEL >= BETA @@ -1,7 +1,20 @@ import sys +import os from ledger import * +def foo (str): + print "Hello:", str +def bar (str): + print "Goodbye:", str + +register_option ("hello", "h:", foo) +register_option ("goodbye", "g:", bar) +print process_arguments (sys.argv[1:]) +process_environment (os.environ, "TEST_") + +sys.exit(0) + parser = TextualParser () register_parser (parser) @@ -6,9 +6,11 @@ #include "util.h" -option_handler::option_handler(const std::string& label, - const std::string& opt_chars) - : handled(false) +static std::deque<option_t> options; + +void register_option(const std::string& label, + const std::string& opt_chars, + option_handler& option) { DEBUG_PRINT("ledger.memory.ctors", "ctor option_handler"); @@ -25,8 +27,6 @@ option_handler::option_handler(const std::string& label, *p = '\0'; opt.long_opt = buf; - handlers.insert(option_handler_pair(opt.long_opt, this)); - if (! opt_chars.empty()) { if (opt_chars[0] != ':') opt.short_opt = opt_chars[0]; @@ -35,7 +35,7 @@ option_handler::option_handler(const std::string& label, opt.wants_arg = true; } - opt.handler = this; + opt.handler = &option; options.push_back(opt); } @@ -43,29 +43,33 @@ option_handler::option_handler(const std::string& label, static inline void process_option(const option_t& opt, const char * arg = NULL) { if (! opt.handler->handled) { - opt.handler->handle_option(arg); + (*opt.handler)(arg); opt.handler->handled = true; } } bool process_option(const std::string& opt, const char * arg) { - option_handler_map::iterator handler = option_handler::handlers.find(opt); - if (handler != option_handler::handlers.end()) { - if (! (*handler).second->handled) { - (*handler).second->handle_option(arg); - (*handler).second->handled = true; + for (std::deque<option_t>::iterator i = options.begin(); + i != options.end(); + i++) + if ((*i).long_opt == opt) { + if (! (*i).handler->handled) { + (*(*i).handler)(arg); + (*i).handler->handled = true; + return true; + } + break; } - return true; - } + return false; } void process_arguments(int argc, char ** argv, const bool anywhere, std::deque<std::string>& args) { - int index = 1; - for (char ** i = argv + 1; index < argc; i++, index++) { + int index = 0; + for (char ** i = argv; index < argc; i++, index++) { if ((*i)[0] != '-') { if (anywhere) { args.push_back(*i); @@ -83,8 +87,8 @@ void process_arguments(int argc, char ** argv, const bool anywhere, if ((*i)[2] == '\0') break; - for (std::deque<option_t>::iterator j = option_handler::options.begin(); - j != option_handler::options.end(); + for (std::deque<option_t>::iterator j = options.begin(); + j != options.end(); j++) if ((*j).wants_arg) { if (const char * p = std::strchr(*i + 2, '=')) { @@ -107,8 +111,8 @@ void process_arguments(int argc, char ** argv, const bool anywhere, std::cerr << "Error: illegal option " << *i << std::endl; std::exit(1); } else { - for (std::deque<option_t>::iterator j = option_handler::options.begin(); - j != option_handler::options.end(); + for (std::deque<option_t>::iterator j = options.begin(); + j != options.end(); j++) if ((*i)[1] == (*j).short_opt) { if ((*j).wants_arg) { @@ -136,12 +140,15 @@ void process_arguments(int argc, char ** argv, const bool anywhere, void process_environment(char ** envp, const std::string& tag) { + const char * tag_p = tag.c_str(); + int tag_len = tag.length(); + for (char ** p = envp; *p; p++) - if (std::strncmp(*p, tag.c_str(), 7) == 0) { + if (std::strncmp(*p, tag_p, tag_len) == 0) { char * q; static char buf[128]; char * r = buf; - for (q = *p + 7; *q && *q != '='; q++) + for (q = *p + tag_len; *q && *q != '='; q++) if (*q == '_') *r++ += '-'; else @@ -153,3 +160,85 @@ void process_environment(char ** envp, const std::string& tag) process_option(buf, q + 1); } } + +#ifdef USE_BOOST_PYTHON + +#include <boost/python.hpp> +#include <boost/python/detail/api_placeholder.hpp> +#include <Python.h> +#include <vector> + +using namespace boost::python; + +struct func_option_wrapper : public option_handler +{ + object self; + func_option_wrapper(object _self) : self(_self) {} + + virtual void operator()(const char * arg) { + call<void>(self.ptr(), arg); + } +}; + +static std::deque<func_option_wrapper> wrappers; + +void py_register_option(const std::string& long_opt, + const std::string& short_opt, object func) +{ + wrappers.push_back(func_option_wrapper(func)); + register_option(long_opt, short_opt, wrappers.back()); +} + +bool (*process_option_1)(const std::string& opt, const char * arg) + = process_option; + +list py_process_arguments(list args, bool anywhere = false) +{ + std::vector<char *> strs; + + int l = len(args); + for (int i = 0; i < l; i++) + strs.push_back(extract<char *>(args[i])); + + std::deque<std::string> newargs; + process_arguments(strs.size(), &strs.front(), anywhere, newargs); + + list py_newargs; + for (std::deque<std::string>::iterator i = newargs.begin(); + i != newargs.end(); + i++) + py_newargs.append(*i); + return py_newargs; +} + +void py_process_environment(object env, const std::string& tag) +{ + std::vector<char *> strs; + std::vector<std::string> storage; + + list items = call_method<list>(env.ptr(), "items"); + int l = len(items); + for (int i = 0; i < l; i++) { + tuple pair = extract<tuple>(items[i]); + std::string s = extract<std::string>(pair[0]); + s += "="; + s += extract<std::string>(pair[1]); + storage.push_back(s); + strs.push_back(const_cast<char *>(storage.back().c_str())); + } + + process_environment(&strs.front(), tag); +} + +BOOST_PYTHON_FUNCTION_OVERLOADS(py_proc_args_overloads, + py_process_arguments, 1, 2) + +void export_option() +{ + def("register_option", py_register_option); + def("process_option", process_option_1); + def("process_arguments", py_process_arguments, py_proc_args_overloads()); + def("process_environment", py_process_environment); +} + +#endif // USE_BOOST_PYTHON @@ -1,11 +1,14 @@ #ifndef _OPTION_H #define _OPTION_H -#include <map> #include <deque> #include <string> -struct option_handler; +struct option_handler { + bool handled; + option_handler() : handled(false) {} + virtual void operator()(const char * arg = NULL) = 0; +}; struct option_t { char short_opt; @@ -13,38 +16,14 @@ struct option_t { bool wants_arg; option_handler * handler; - option_t() : short_opt(0), wants_arg(false) {} -}; - -typedef std::map<const std::string, option_handler *> option_handler_map; -typedef std::pair<const std::string, option_handler *> option_handler_pair; - -struct option_handler { - bool handled; - - static std::deque<option_t> options; - static option_handler_map handlers; - - option_handler(const std::string& label, - const std::string& opt_chars); - - virtual void handle_option(const char * arg = NULL) = 0; + option_t() : short_opt(0), wants_arg(false), handler(NULL) {} }; +void register_option(const std::string& label, + const std::string& opt_chars, option_handler& option); bool process_option(const std::string& opt, const char * arg = NULL); void process_arguments(int argc, char ** argv, const bool anywhere, std::deque<std::string>& args); void process_environment(char ** envp, const std::string& tag); -#define DEF_OPT_HANDLERS() \ - std::deque<option_t> option_handler::options; \ - option_handler_map option_handler::handlers - -#define OPT_BEGIN(tag, chars) \ - static struct opt_ ## tag ## _handler : public option_handler { \ - opt_ ## tag ## _handler() : option_handler(#tag, chars) {} \ - virtual void handle_option(const char * optarg) - -#define OPT_END(tag) } opt_ ## tag ## _handler_obj - #endif // _OPTION_H @@ -22,6 +22,7 @@ void export_qif(); #ifdef READ_GNUCASH void export_gnucash(); #endif +void export_option(); BOOST_PYTHON_MODULE(ledger) { export_amount(); @@ -35,5 +36,6 @@ BOOST_PYTHON_MODULE(ledger) { #ifdef READ_GNUCASH export_gnucash(); #endif + export_option(); ledger::initialize(); } |