#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 show_help(std::ostream& out) { std::cerr << "usage: ledger [options] COMMAND [options] [REGEXPS]" << std::endl << std::endl << "Basic options:" << std::endl << " -h display this help text" << std::endl << " -v display version information" << std::endl << " -f FILE specify pathname of ledger data file" << std::endl << std::endl << "Report filtering:" << std::endl << " -a REGEX specify an account regex for \"print\"" << std::endl << " -b DATE specify a beginning date" << std::endl << " -e DATE specify an ending date" << std::endl << " -c do not show future entries (same as -e TODAY)" << std::endl << " -d DATE specify a date mask ('-d mon', for all mondays)" << std::endl << " -C show only cleared transactions and balances" << std::endl << " -U show only uncleared transactions and balances" << std::endl << " -R do not consider virtual transactions: real only" << std::endl << " -l EXPR don't print entries for which EXPR yields 0" << std::endl << std::endl << "Customizing output:" << std::endl << " -n do not calculate parent account totals" << std::endl << " -s show sub-accounts in balance, and splits in register" << std::endl << " -M print register using monthly sub-totals" << std::endl << " -E show accounts that total to zero" << std::endl << " -S EXPR sort entry output based on EXPR" << std::endl << std::endl << "Commodity prices:" << std::endl << " -T report commodity totals, not their market value" << std::endl << " -B report cost basis of commodities" << std::endl << " -V report the market value of commodities" << std::endl << " -P FILE sets the price database, for reading/writing price info" << std::endl << " -Q download new price information (when needed) from the Internet" << std::endl << " (works by running \"getquote SYMBOL\")" << std::endl << " -L MINS with -Q, fetch quotes only if data is older than MINS" << std::endl << " -p STR specifies a direct commodity conversion: COMM=AMOUNT" << std::endl << std::endl << "commands:" << std::endl << " balance show balance totals" << std::endl << " register display a register for ACCOUNT" << std::endl << " print print all ledger entries" << std::endl << " entry output a newly formed entry, based on arguments" << std::endl << " equity output equity entries for specified accounts" << std::endl; } 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': std::cout << "Ledger " << version << ", the command-line accounting tool" << std::endl << " Copyright (c) 2003-2004, New Artisans LLC. All rights reserved." << std::endl << std::endl << "This program is made available under the terms of the BSD Public" << std::endl << "License. See the LICENSE file included with the distribution for" << std::endl << "details and disclaimer." << std::endl; 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::cout); 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 { // Treat the remaining command-line arguments as regular // expressions, used for refining report results. bool have_regexps = index < argc; bool first = true; for (; index < argc; index++) { if (std::strcmp(argv[index], "--") == 0) { index++; if (! first && index < argc) predicate_string += ")"; break; } if (! show_expanded && command == "b") show_expanded = true; if (first) { if (! predicate_string.empty()) predicate_string += "&("; else predicate_string += "("; first = false; } else if (argv[index][0] == '-') { predicate_string += "&"; } else { predicate_string += "|"; } if (argv[index][0] == '-') { predicate_string += "!/"; predicate_string += argv[index] + 1; } else { predicate_string += "/"; predicate_string += argv[index]; } predicate_string += "/"; } if (index < argc) { if (! predicate_string.empty()) predicate_string += "&("; else predicate_string += "("; } first = true; for (; index < argc; index++) { if (! show_expanded && command == "b") show_expanded = true; if (first) first = false; else if (argv[index][0] == '-') predicate_string += "&"; else predicate_string += "|"; if (argv[index][0] == '-') { predicate_string += "!//"; predicate_string += argv[index] + 1; } else { predicate_string += "//"; predicate_string += argv[index]; } predicate_string += "/"; } if (have_regexps) predicate_string += ")"; } // 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" && ! show_empty) display_predicate_string = "T"; else if (command == "E") display_predicate_string = "a"; } else { #ifdef DEBUG if (debug) std::cerr << "display-p = " << 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, show_expanded ? 0 : 1, sort_order.get()); if (! display_predicate.get() || item_predicate(display_predicate.get())(journal->master)) { std::string end_format = "--------------------\n"; format.reset(end_format + f); format_account(std::cout, format)(journal->master, true, display_predicate.get()); } } 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, 0, 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")); clear_transaction_display_flags(journal->entries.begin(), journal->entries.end()); write_binary_journal(outstr, journal.get(), std::getenv("LEDGER")); } return 0; } // main.cc ends here.