From 1fcf7b5fc1eb107178b009fcc9208faa5f73cdbf Mon Sep 17 00:00:00 2001 From: John Wiegley Date: Thu, 18 Sep 2008 11:42:17 -0400 Subject: Move handling of all commands to the value expression subsystem, rather than handling them explicitly in main.cc. --- src/main.cc | 200 ++++------------------------------------------------------ src/report.cc | 184 +++++++++++++++++++++++++++++++++++++++++++++++++++-- 2 files changed, 189 insertions(+), 195 deletions(-) diff --git a/src/main.cc b/src/main.cc index 9b03afd2..f669a82d 100644 --- a/src/main.cc +++ b/src/main.cc @@ -32,7 +32,6 @@ #include "session.h" #include "report.h" #include "option.h" -#include "output.h" #include "help.h" #include "textual.h" @@ -52,131 +51,6 @@ #endif namespace ledger { - string args_to_predicate(value_t::sequence_t::const_iterator begin, - value_t::sequence_t::const_iterator end) - { - string acct_value_expr; - string payee_value_expr; - string note_value_expr; - - string * value_expr; - - enum regexp_kind_t { - ACCOUNT_REGEXP, - PAYEE_REGEXP, - NOTE_REGEXP - } - kind = ACCOUNT_REGEXP; - - value_expr = &acct_value_expr; - - for ( ; begin != end; begin++) { - const string& arg((*begin).as_string()); - - if (arg == "--") { - kind = PAYEE_REGEXP; - value_expr = &payee_value_expr; - } - else if (arg == "/") { - kind = NOTE_REGEXP; - value_expr = ¬e_value_expr; - } - else { - if (! value_expr->empty()) - *value_expr += "|"; - - switch (kind) { - case ACCOUNT_REGEXP: - *value_expr += "account =~ "; - break; - case PAYEE_REGEXP: - *value_expr += "payee =~ "; - break; - case NOTE_REGEXP: - *value_expr += "note =~ "; - break; - } - - const char * p = arg.c_str(); - if (*p == '-') { - *value_expr += "!"; - p++; - } - - *value_expr += "/"; - if (kind == NOTE_REGEXP) *value_expr += ":"; - while (*p) { - if (*p == '/') - *value_expr += "\\"; - *value_expr += *p; - p++; - } - if (kind == NOTE_REGEXP) *value_expr += ":"; - *value_expr += "/"; - } - } - - string final_value_expr; - - if (! acct_value_expr.empty()) { - if (! payee_value_expr.empty() || - ! note_value_expr.empty()) - final_value_expr = string("(") + acct_value_expr + ")"; - else - final_value_expr = acct_value_expr; - } - - if (! payee_value_expr.empty()) { - if (! acct_value_expr.empty()) - final_value_expr += string("&(") + payee_value_expr + ")"; - else if (! note_value_expr.empty()) - final_value_expr = string("(") + payee_value_expr + ")"; - else - final_value_expr = payee_value_expr; - } - - if (! note_value_expr.empty()) { - if (! acct_value_expr.empty() || - ! payee_value_expr.empty()) - final_value_expr += string("&(") + note_value_expr + ")"; - else if (acct_value_expr.empty() && - payee_value_expr.empty()) - final_value_expr = note_value_expr; - } - - DEBUG("report.predicate", - "Regexp predicate expression = " << final_value_expr); - - return final_value_expr; - } - - template - class reporter - { - shared_ptr > handler; - - public: - reporter(item_handler * _handler) - : handler(_handler) {} - - value_t operator()(call_scope_t& args) - { - report_t& report(find_scope(args)); - - if (args.value().size() > 0) - report.append_predicate - (args_to_predicate(args.value().as_sequence().begin(), - args.value().as_sequence().end())); - - (report.*report_method)(handler_ptr(handler)); - - return true; - } - }; - int read_and_report(ledger::report_t& report, int argc, char * argv[], char * envp[]) { @@ -393,66 +267,14 @@ namespace ledger { TRACE_FINISH(entries, 1); TRACE_FINISH(parsing_total, 1); - // Create a command object based on the command verb that was seen - // above. + // Lookup the command object corresponding to the command verb. function_t command; + if (expr_t::ptr_op_t def = report.lookup(string("cmd_") + verb)) + command = def->as_function(); - if (verb == "register" || verb == "reg" || verb == "r") { - verb = "register"; - command = reporter<>(new format_xacts(report, - report.format_string.empty() ? - session.register_format : - report.format_string)); - } - else if (verb == "print" || verb == "p") { - verb = "print"; - command = reporter<>(new format_xacts(report, - report.format_string.empty() ? - session.print_format : - report.format_string)); - } - else if (verb == "balance" || verb == "bal" || verb == "b") { - verb = "balance"; - command = reporter - (new format_accounts(report, session.balance_format)); - } - else if (verb == "equity") { - verb = "equity"; - command = reporter - (new format_equity(report, session.print_format)); - } -#if 0 - else if (verb == "entry") - command = entry_command(); - else if (verb == "dump") - command = dump_command(); - else if (verb == "output") - command = output_command(); - else if (verb == "prices") - command = prices_command(); - else if (verb == "pricesdb") - command = pricesdb_command(); - else if (verb == "csv") - command = csv_command(); - else if (verb == "emacs" || verb == "lisp") - command = emacs_command(); - else if (verb == "xml") - command = xml_command(); -#endif - else { - char buf[128]; - std::strcpy(buf, "cmd_"); - std::strcat(buf, verb.c_str()); - - if (expr_t::ptr_op_t def = report.lookup(buf)) - command = def->as_function(); - - if (! command) - throw_(std::logic_error, string("Unrecognized command '") + verb + "'"); - } + if (! command) + throw_(std::logic_error, string("Unrecognized command '") + verb + "'"); // Patch up some of the reporting options based on what kind of // command it was. @@ -460,7 +282,7 @@ namespace ledger { // jww (2008-08-14): This code really needs to be rationalized away // for 3.0. - if (verb == "print" || verb == "entry" || verb == "dump") { + if (verb[0] == 'p' || verb == "entry" || verb == "dump") { report.show_related = true; report.show_all_related = true; } @@ -468,7 +290,7 @@ namespace ledger { report.show_subtotal = true; } else if (report.show_related) { - if (verb == "register") { + if (verb[0] == 'r') { report.show_inverted = true; } else { report.show_subtotal = true; @@ -476,13 +298,13 @@ namespace ledger { } } - if (verb != "balance" && verb != "register") + if (verb[0] != 'b' && verb[0] != 'r') amount_t::keep_base = true; // Setup the default value for the display predicate if (report.display_predicate.empty()) { - if (verb == "balance") { + if (verb[0] == 'b') { if (! report.show_empty) report.display_predicate = "total"; if (! report.show_subtotal) { @@ -492,9 +314,9 @@ namespace ledger { } } else if (verb == "equity") { - report.display_predicate = "fmt_t"; // jww (2008-08-14): ??? + report.display_predicate = "amount_expr"; // jww (2008-08-14): ??? } - else if (verb == "register" && ! report.show_empty) { + else if (verb[0] == 'r' && ! report.show_empty) { report.display_predicate = "amount"; } } diff --git a/src/report.cc b/src/report.cc index f706ea78..743ddf71 100644 --- a/src/report.cc +++ b/src/report.cc @@ -30,6 +30,7 @@ */ #include "report.h" +#include "output.h" #include "reconcile.h" namespace ledger { @@ -284,20 +285,191 @@ namespace { args[0].strip_annotations().dump(out, *first_width, *latter_width); return string_value(out.str()); } + + string args_to_predicate(value_t::sequence_t::const_iterator begin, + value_t::sequence_t::const_iterator end) + { + string acct_value_expr; + string payee_value_expr; + string note_value_expr; + + string * value_expr; + + enum regexp_kind_t { + ACCOUNT_REGEXP, + PAYEE_REGEXP, + NOTE_REGEXP + } + kind = ACCOUNT_REGEXP; + + value_expr = &acct_value_expr; + + for ( ; begin != end; begin++) { + const string& arg((*begin).as_string()); + + if (arg == "--") { + kind = PAYEE_REGEXP; + value_expr = &payee_value_expr; + } + else if (arg == "/") { + kind = NOTE_REGEXP; + value_expr = ¬e_value_expr; + } + else { + if (! value_expr->empty()) + *value_expr += "|"; + + switch (kind) { + case ACCOUNT_REGEXP: + *value_expr += "account =~ "; + break; + case PAYEE_REGEXP: + *value_expr += "payee =~ "; + break; + case NOTE_REGEXP: + *value_expr += "note =~ "; + break; + } + + const char * p = arg.c_str(); + if (*p == '-') { + *value_expr += "!"; + p++; + } + + *value_expr += "/"; + if (kind == NOTE_REGEXP) *value_expr += ":"; + while (*p) { + if (*p == '/') + *value_expr += "\\"; + *value_expr += *p; + p++; + } + if (kind == NOTE_REGEXP) *value_expr += ":"; + *value_expr += "/"; + } + } + + string final_value_expr; + + if (! acct_value_expr.empty()) { + if (! payee_value_expr.empty() || + ! note_value_expr.empty()) + final_value_expr = string("(") + acct_value_expr + ")"; + else + final_value_expr = acct_value_expr; + } + + if (! payee_value_expr.empty()) { + if (! acct_value_expr.empty()) + final_value_expr += string("&(") + payee_value_expr + ")"; + else if (! note_value_expr.empty()) + final_value_expr = string("(") + payee_value_expr + ")"; + else + final_value_expr = payee_value_expr; + } + + if (! note_value_expr.empty()) { + if (! acct_value_expr.empty() || + ! payee_value_expr.empty()) + final_value_expr += string("&(") + note_value_expr + ")"; + else if (acct_value_expr.empty() && + payee_value_expr.empty()) + final_value_expr = note_value_expr; + } + + DEBUG("report.predicate", + "Regexp predicate expression = " << final_value_expr); + + return final_value_expr; + } + + template + class reporter + { + shared_ptr > handler; + + public: + reporter(item_handler * _handler) + : handler(_handler) {} + + value_t operator()(call_scope_t& args) + { + report_t& report(find_scope(args)); + + if (args.value().size() > 0) + report.append_predicate + (args_to_predicate(args.value().as_sequence().begin(), + args.value().as_sequence().end())); + + (report.*report_method)(handler_ptr(handler)); + + return true; + } + }; } expr_t::ptr_op_t report_t::lookup(const string& name) { const char * p = name.c_str(); switch (*p) { - case 'f': - if (std::strncmp(p, "fmt_", 4) == 0) { + case 'a': + if (std::strcmp(p, "amount_expr") == 0) + return MAKE_FUNCTOR(report_t::get_amount_expr); + break; + + case 'c': + if (std::strncmp(p, "cmd_", 4) == 0) { + +#define FORMAT(str) \ + (format_string.empty() ? session. str : format_string) + +#if 0 + // Commands yet to implement: + // + // entry + // dump + // output + // prices + // pricesdb + // csv + // emacs | lisp + // xml +#endif + p = p + 4; switch (*p) { - case 't': - return MAKE_FUNCTOR(report_t::get_amount_expr); - case 'T': - return MAKE_FUNCTOR(report_t::get_total_expr); + case 'b': + if (*(p + 1) == '\0' || + std::strcmp(p, "bal") == 0 || + std::strcmp(p, "balance") == 0) + return expr_t::op_t::wrap_functor + (reporter + (new format_accounts(*this, FORMAT(balance_format)))); + + case 'e': + if (std::strcmp(p, "equity") == 0) + return expr_t::op_t::wrap_functor + (reporter + (new format_equity(*this, FORMAT(print_format)))); + + case 'p': + if (*(p + 1) == '\0' || + std::strcmp(p, "print") == 0) + return WRAP_FUNCTOR + (reporter<>(new format_xacts(*this, FORMAT(print_format)))); + break; + + case 'r': + if (*(p + 1) == '\0' || + std::strcmp(p, "reg") == 0 || + std::strcmp(p, "register") == 0) + return WRAP_FUNCTOR + (reporter<>(new format_xacts(*this, FORMAT(register_format)))); + break; } } break; -- cgit v1.2.3