#include "ledger.h" #include "error.h" #include "valexpr.h" #include "format.h" #include "walk.h" #include #include #include #include namespace ledger { static const std::string bal_fmt = "%20T %2_%-n\n"; static const std::string reg_fmt = "%10d %-.20p %-.22N %12.66t %12.80T\n\ %/ %-.22N %12.66t %12.80T\n"; static const std::string print_fmt = "\n%10d %X%C%p\n %-34N %12o\n%/ %-34N %12o\n"; static const std::string equity_fmt = "\n%10d %X%C%p\n%/ %-34N %12t\n"; void set_price_conversion(const std::string& setting) { char buf[128]; std::strcpy(buf, setting.c_str()); assert(setting.length() < 128); char * c = buf; char * p = std::strchr(buf, '='); if (! p) { std::cerr << "Warning: Invalid price setting: " << setting << std::endl; } else { *p++ = '\0'; amount_t price; price.parse(p); commodity_t * commodity = commodity_t::find_commodity(c, true); commodity->set_conversion(price); } } static long pricing_leeway = 24 * 3600; static std::string price_db; static bool cache_dirty = false; void download_price_quote(commodity_t * commodity, const std::time_t age, const amount_t& price, const std::time_t moment) { std::time_t now = std::time(NULL); // the time of the query if (! (commodity->flags & COMMODITY_STYLE_CONSULTED) && std::difftime(now, moment) < pricing_leeway && (! price || std::difftime(moment, age) > pricing_leeway)) { using namespace std; // Only consult the Internet once for any commodity commodity->flags |= COMMODITY_STYLE_CONSULTED; cache_dirty = true; char buf[256]; buf[0] = '\0'; if (FILE * fp = popen((string("getquote ") + commodity->symbol).c_str(), "r")) { if (feof(fp) || ! fgets(buf, 255, fp)) { fclose(fp); return; } fclose(fp); } if (buf[0]) { char * p = strchr(buf, '\n'); if (p) *p = '\0'; amount_t current; current.parse(buf); commodity->add_price(now, current); if (! price_db.empty()) { char buf[128]; strftime(buf, 127, "%Y/%m/%d %H:%M:%S", localtime(&now)); ofstream database(price_db.c_str(), ios_base::out | ios_base::app); database << "P " << buf << " " << commodity->symbol << " " << current << endl; } } } } } // namespace ledger static void assemble_regexp_predicate(std::string& predicate_string, const std::list& strings, const bool exclude = false, const bool payee = false) { if (strings.size() == 0) return; if (! predicate_string.empty()) predicate_string += "&"; if (exclude) predicate_string += "!"; if (payee) predicate_string += "/"; predicate_string += "/("; bool first = true; for (std::list::const_iterator i = strings.begin(); i != strings.end(); i++) { if (first) first = false; else predicate_string += "|"; predicate_string += *i; } predicate_string += ")/"; } static void show_version(std::ostream& out) { out << "Ledger " << ledger::version << ", the command-line accounting tool\n\n" << "Copyright (c) 2003-2004, New Artisans LLC. All rights reserved.\n\n" << "This program is made available under the terms of the BSD Public\n" << "License. See the LICENSE file included with the distribution for\n" << "details and disclaimer.\n"; } static void show_help(std::ostream& out) { out << "usage: ledger [options] COMMAND [options] [REGEXPS]\n\n" << "Basic options:\n" << " -h display this help text\n" << " -v display version information\n" << " -f FILE specify pathname of ledger data file\n\n" << "Report filtering:\n" << " -a REGEX specify an account regex for \"print\"\n" << " -b DATE specify a beginning date\n" << " -e DATE specify an ending date\n" << " -c do not show future entries (same as -e TODAY)\n" << " -d DATE specify a date mask ('-d mon', for all mondays)\n" << " -C show only cleared transactions and balances\n" << " -U show only uncleared transactions and balances\n" << " -R do not consider virtual transactions: real only\n" << " -l EXPR don't print entries for which EXPR yields 0\n\n" << "Customizing output:\n" << " -n do not calculate parent account totals\n" << " -s show sub-accounts in balance, and splits in register\n" << " -M print register using monthly sub-totals\n" << " -E show accounts that total to zero\n" << " -S EXPR sort entry output based on EXPR\n\n" << "Commodity prices:\n" << " -T report commodity totals, not their market value\n" << " -B report cost basis of commodities\n" << " -V report the market value of commodities\n" << " -P FILE sets the price database, for reading/writing price info\n" << " -Q download price information from the Internet\n" << " (works by running \"getquote SYMBOL\")\n" << " -L MINS with -Q, fetch quotes only if data is older than MINS\n" << " -p STR specifies a direct commodity conversion: COMM=AMOUNT\n\n" << "Commands:\n" << " balance show balance totals\n" << " register display a register for ACCOUNT\n" << " print print all ledger entries\n" << " entry output a newly formed entry, based on arguments\n" << " equity output equity entries for specified accounts\n"; } int main(int argc, char * argv[]) { using namespace ledger; std::auto_ptr journal(new journal_t); std::list files; std::auto_ptr predicate; std::auto_ptr display_predicate; std::auto_ptr sort_order; std::string predicate_string; std::string display_predicate_string; std::string format_string; std::string sort_string; std::string value_expr = "a"; std::string total_expr = "T"; bool show_subtotals = true; bool show_expanded = false; bool show_related = false; bool show_inverted = false; bool show_empty = false; bool show_commodities_revalued = false; bool show_commodities_revalued_only = false; #ifdef DEBUG bool debug = false; #endif // Initialize some variables based on environment variable settings if (char * p = std::getenv("PRICE_HIST")) price_db = p; if (char * p = std::getenv("PRICE_EXP")) pricing_leeway = std::atol(p) * 60; // A ledger data file must be specified bool use_cache = std::getenv("LEDGER") != NULL; if (use_cache) { for (int i = 0; i < argc; i++) if (std::strcmp(argv[i], "-f") == 0) { use_cache = false; break; } cache_dirty = true; if (use_cache) if (const char * p = std::getenv("LEDGER_CACHE")) if (access(p, R_OK) != -1) { std::ifstream instr(p); if (! read_binary_journal(instr, std::getenv("LEDGER"), journal.get())) { // Throw away what's been read, and create a new journal journal.reset(new journal_t); } else { cache_dirty = false; } } } // Parse the command-line options int c, index; while (-1 != (c = getopt(argc, argv, "+ABb:Ccd:DEe:F:f:Ghi:L:l:MnoOP:p:QRS:st:T:UVvWXZz"))) { switch (char(c)) { // Basic options case 'h': show_help(std::cout); break; #ifdef DEBUG case 'z': debug = 1; break; #endif case 'v': show_version(std::cout); return 0; case 'f': files.push_back(optarg); use_cache = false; break; case 'p': set_price_conversion(optarg); break; case 'b': if (! predicate_string.empty()) predicate_string += "&"; predicate_string += "(d>=["; predicate_string += optarg; predicate_string += "])"; break; case 'e': if (! predicate_string.empty()) predicate_string += "&"; predicate_string += "(d<["; predicate_string += optarg; predicate_string += "])"; break; case 'c': { if (! predicate_string.empty()) predicate_string += "&"; predicate_string += "(d<"; std::ostringstream now; now << std::time(NULL); predicate_string += now.str(); predicate_string += ")"; break; } case 'C': if (! predicate_string.empty()) predicate_string += "&"; predicate_string += "X"; break; case 'U': if (! predicate_string.empty()) predicate_string += "&"; predicate_string += "!X"; break; case 'R': if (! predicate_string.empty()) predicate_string += "&"; predicate_string += "R"; break; // Customizing output case 'F': format_string = optarg; break; case 'E': show_empty = true; break; case 'n': show_subtotals = false; break; case 's': show_expanded = true; break; case 'S': sort_string = optarg; break; case 'o': show_related = true; break; case 'l': if (! predicate_string.empty()) predicate_string += "&"; predicate_string += "("; predicate_string += optarg; predicate_string += ")"; break; case 'd': if (! display_predicate_string.empty()) display_predicate_string += "&"; display_predicate_string += "("; display_predicate_string += optarg; display_predicate_string += ")"; break; // Commodity reporting case 'P': price_db = optarg; break; case 'L': pricing_leeway = std::atol(optarg) * 60; break; case 'Q': commodity_t::updater = download_price_quote; break; case 't': value_expr = optarg; break; case 'T': total_expr = optarg; break; case 'O': value_expr = "a"; total_expr = "T"; break; case 'B': value_expr = "c"; total_expr = "C"; break; case 'V': show_commodities_revalued = true; value_expr = "v"; total_expr = "V"; break; case 'G': show_commodities_revalued = show_commodities_revalued_only = true; value_expr = "c"; total_expr = "G"; break; case 'A': value_expr = "a"; total_expr = "MT"; break; case 'D': value_expr = "a"; total_expr = "DMT"; break; case 'Z': value_expr = "a"; total_expr = "MDMT"; break; #if 0 case 'W': value_expr = "a"; total_expr = "MD(MT*(d-b/e-b))"; break; case 'X': value_expr = "a"; total_expr = "a+MD(MT*(d-b/e-b))"; break; #endif default: assert(0); break; } } if (optind == argc) { show_help(std::cerr); return 1; } index = optind; // Read the ledger file, unless we already read it from the cache if (! use_cache || cache_dirty) { int entry_count = 0; try { if (files.empty()) { if (char * p = std::getenv("LEDGER")) for (p = std::strtok(p, ":"); p; p = std::strtok(NULL, ":")) entry_count += parse_journal_file(p, journal.get()); } else { for (std::list::iterator i = files.begin(); i != files.end(); i++) { char buf[4096]; char * p = buf; std::strcpy(p, (*i).c_str()); entry_count += parse_journal_file(p, journal.get()); } } // Read prices from their own ledger file, after all others have // been read. if (! price_db.empty()) { const char * path = price_db.c_str(); std::ifstream db(path); journal->sources.push_back(path); entry_count += parse_textual_journal(db, 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; } } // Read the command word, and then check and simplify it std::string command = argv[index++]; if (command == "balance" || command == "bal" || command == "b") command = "b"; else if (command == "register" || command == "reg" || command == "r") command = "r"; else if (command == "print" || command == "p") command = "p"; else if (command == "entry") command = "e"; else if (command == "equity") command = "E"; else { std::cerr << "Error: Unrecognized command '" << command << "'." << std::endl; return 1; } // Process the remaining command-line arguments std::auto_ptr new_entry; if (command == "e") { new_entry.reset(journal->derive_entry(argc - index, &argv[index])); } else { std::list account_include_regexps; std::list account_exclude_regexps; std::list payee_include_regexps; std::list payee_exclude_regexps; // Treat the remaining command-line arguments as regular // expressions, used for refining report results. for (; index < argc; index++) { if (std::strcmp(argv[index], "--") == 0) { index++; break; } if (! show_expanded && command == "b") show_expanded = true; if (argv[index][0] == '-') account_exclude_regexps.push_back(argv[index] + 1); else account_include_regexps.push_back(argv[index]); } for (; index < argc; index++) { if (! show_expanded && command == "b") show_expanded = true; if (argv[index][0] == '-') payee_exclude_regexps.push_back(argv[index] + 1); else payee_include_regexps.push_back(argv[index]); } assemble_regexp_predicate(predicate_string, account_include_regexps); assemble_regexp_predicate(predicate_string, account_exclude_regexps, true); assemble_regexp_predicate(predicate_string, payee_include_regexps, false, true); assemble_regexp_predicate(predicate_string, payee_exclude_regexps, true, true); } // Compile the predicates if (! predicate_string.empty()) { #ifdef DEBUG if (debug) std::cerr << "predicate = " << predicate_string << std::endl; #endif predicate.reset(parse_expr(predicate_string)); } if (display_predicate_string.empty()) { if (command == "b") { if (! show_empty) display_predicate_string = "T"; if (! show_expanded) { if (! display_predicate_string.empty()) display_predicate_string += "&"; display_predicate_string += "!n"; } } else if (command == "E") { display_predicate_string = "a"; } } if (! display_predicate_string.empty()) { #ifdef DEBUG if (debug) std::cerr << "disp-pred = " << display_predicate_string << std::endl; #endif display_predicate.reset(parse_expr(display_predicate_string)); } // Compile the sorting string if (! sort_string.empty()) sort_order.reset(parse_expr(sort_string)); // Setup the meaning of %t and %T, used in format strings format_t::value_expr.reset(parse_expr(value_expr)); format_t::total_expr.reset(parse_expr(total_expr)); // Now handle the command that was identified above. unsigned int xact_display_flags = MATCHING_TRANSACTIONS; if (command == "p" || command == "e") { xact_display_flags |= OTHER_TRANSACTIONS; show_expanded = true; } else if (command == "E") { show_expanded = true; } else if (show_related) { if (command == "r") { xact_display_flags = OTHER_TRANSACTIONS; show_inverted = true; } else { xact_display_flags |= OTHER_TRANSACTIONS; } } 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; } if (command == "b") { format_t format(first_line_format); format_account formatter(std::cout, format, display_predicate.get()); walk_accounts(journal->master, formatter, predicate.get(), xact_display_flags, show_subtotals, sort_order.get()); if (format_account::disp_subaccounts_p(journal->master)) { std::string end_format = "--------------------\n"; format.reset(end_format + f); format.format_elements(std::cout, details_t(journal->master)); } } else if (command == "E") { format_t format(first_line_format); format_t nformat(next_lines_format); format_equity formatter(std::cout, format, nformat, display_predicate.get()); walk_accounts(journal->master, formatter, predicate.get(), xact_display_flags, true, sort_order.get()); } else if (command == "e") { format_t format(first_line_format); format_t nformat(next_lines_format); format_transaction formatter(std::cout, format, nformat); for (transactions_list::iterator i = new_entry->transactions.begin(); i != new_entry->transactions.end(); i++) handle_transaction(*i, formatter, xact_display_flags); } else { format_t format(first_line_format); format_t nformat(next_lines_format); format_transaction formatter(std::cout, format, nformat, display_predicate.get(), #ifdef COLLAPSED_REGISTER ! show_subtotals, #endif show_inverted); if (! sort_order.get()) { if (show_commodities_revalued) { changed_value_filter filtered_formatter(formatter); walk_entries(journal->entries.begin(), journal->entries.end(), filtered_formatter, predicate.get(), xact_display_flags); } else { walk_entries(journal->entries.begin(), journal->entries.end(), formatter, predicate.get(), xact_display_flags); } } else { transactions_deque transactions_pool; walk_entries(journal->entries.begin(), journal->entries.end(), collect_transactions(transactions_pool), predicate.get(), xact_display_flags); std::stable_sort(transactions_pool.begin(), transactions_pool.end(), compare_items(sort_order.get())); if (show_commodities_revalued) { changed_value_filter filtered_formatter(formatter); walk_transactions(transactions_pool.begin(), transactions_pool.end(), filtered_formatter); } else { walk_transactions(transactions_pool.begin(), transactions_pool.end(), formatter); } } } // Save the cache, if need be if (use_cache && cache_dirty) if (const char * p = std::getenv("LEDGER_CACHE")) { std::ofstream outstr(p); assert(std::getenv("LEDGER")); #if 0 clear_transaction_display_flags(journal->entries.begin(), journal->entries.end()); #endif write_binary_journal(outstr, journal.get(), std::getenv("LEDGER")); } return 0; } // main.cc ends here.