diff options
-rw-r--r-- | constraint.h | 157 | ||||
-rw-r--r-- | expr.h | 50 | ||||
-rw-r--r-- | format.h | 1 | ||||
-rw-r--r-- | item.cc | 87 | ||||
-rw-r--r-- | item.h | 15 | ||||
-rw-r--r-- | main.cc | 259 | ||||
-rw-r--r-- | textual.cc | 2 |
7 files changed, 256 insertions, 315 deletions
diff --git a/constraint.h b/constraint.h deleted file mode 100644 index 7ee77581..00000000 --- a/constraint.h +++ /dev/null @@ -1,157 +0,0 @@ -#ifndef _CONSTRAINT_H -#define _CONSTRAINT_H - -#include "ledger.h" -#include "expr.h" -#include "item.h" - -template <typename ForwardIterator, typename ValueType, typename Constraint> -class constrained_iterator -{ - ForwardIterator iter, end; - const Constraint& constraint; - - constrained_iterator& operator=(const constrained_iterator&); - - public: - constrained_iterator(ForwardIterator begin, ForwardIterator _end, - const Constraint& _constraint) - : iter(begin), end(_end), constraint(_constraint) { - skip_nonmatching(); - } - - constrained_iterator(const constrained_iterator& other) - : iter(other.iter), end(other.end), constraint(other.constraint) {} - - constrained_iterator operator++() const { - ForwardIterator temp = iter; - temp++; - return constrained_iterator(temp, end); - } - - constrained_iterator& operator++(int) { - iter++; - skip_nonmatching(); - return *this; - } - - bool operator==(ForwardIterator other) const { - return iter == other; - } - bool operator!=(ForwardIterator other) const { - return ! (*this == other); - } - - ValueType operator*() const { - return *iter; - } - - void skip_nonmatching() { - bool failed; - do { - if (iter == end) return; - failed = false; - if (! constraint(*iter)) { - failed = true; - iter++; - } - } while (failed); - } -}; - -namespace ledger { - -class constraints_t -{ - public: - bool show_expanded; - bool show_related; - bool show_inverted; - bool show_subtotals; - bool show_empty; - - node_t * predicate; - - explicit constraints_t() { - show_expanded = false; - show_related = false; - show_inverted = false; - show_subtotals = true; - show_empty = false; - - predicate = NULL; - } - - ~constraints_t() { - if (predicate) delete predicate; - } - - bool operator ()(const transaction_t * xact) const { - if (! predicate) { - return true; - } else { - item_t temp; - temp.date = xact->entry->date; - temp.payee = xact->entry->payee; - temp.account = xact->account; - return predicate->compute(&temp); - } - } - - bool operator ()(const entry_t * entry) const { - if (! predicate) { - return true; - } else { - item_t temp; - temp.date = entry->date; - temp.payee = entry->payee; - - // Although there may be conflicting account masks for the whole - // set of transactions -- for example, /rent/&!/expenses/, which - // might match one by not another transactions -- we let the - // entry through if at least one of the transactions meets the - // criterion - - for (transactions_list::const_iterator i = entry->transactions.begin(); - i != entry->transactions.end(); - i++) { - temp.account = (*i)->account; - if (predicate->compute(&temp)) - return true; - } - return false; - } - } - - bool operator ()(const item_t * item) const { - return ! predicate || predicate->compute(item); - } -}; - -typedef constrained_iterator<transactions_list::const_iterator, transaction_t *, - constraints_t> - constrained_transactions_list_const_iterator; - -typedef constrained_iterator<transactions_list::iterator, transaction_t *, - constraints_t> - constrained_transactions_list_iterator; - -typedef constrained_iterator<entries_list::const_iterator, entry_t *, - constraints_t> - constrained_entries_list_const_iterator; - -typedef constrained_iterator<entries_list::iterator, entry_t *, - constraints_t> - constrained_entries_list_iterator; - -typedef constrained_iterator<items_deque::const_iterator, item_t *, - constraints_t> - constrained_items_deque_const_iterator; - -typedef constrained_iterator<items_deque::iterator, item_t *, - constraints_t> - constrained_items_deque_iterator; - -} // namespace ledger - -#endif // _CONSTRAINT_H @@ -122,6 +122,56 @@ inline node_t * find_node(node_t * node, node_t::kind_t type) { void dump_tree(std::ostream& out, node_t * node); +class value_predicate +{ + public: + const node_t * predicate; + + explicit value_predicate(const node_t * _predicate) + : predicate(_predicate) {} + + bool operator ()(const transaction_t * xact) const { + if (! predicate) { + return true; + } else { + item_t temp; + temp.date = xact->entry->date; + temp.payee = xact->entry->payee; + temp.account = xact->account; + return predicate->compute(&temp); + } + } + + bool operator ()(const entry_t * entry) const { + if (! predicate) { + return true; + } else { + item_t temp; + temp.date = entry->date; + temp.payee = entry->payee; + + // Although there may be conflicting account masks for the whole + // set of transactions -- for example, /rent/&!/expenses/, which + // might match one by not another transactions -- we let the + // entry through if at least one of the transactions meets the + // criterion + + for (transactions_list::const_iterator i = entry->transactions.begin(); + i != entry->transactions.end(); + i++) { + temp.account = (*i)->account; + if (predicate->compute(&temp)) + return true; + } + return false; + } + } + + bool operator ()(const item_t * item) const { + return ! predicate || predicate->compute(item); + } +}; + } // namespace report #endif // _REPORT_H @@ -3,7 +3,6 @@ #include "ledger.h" #include "balance.h" -#include "constraint.h" #include "expr.h" namespace ledger { @@ -1,5 +1,4 @@ #include "item.h" -#include "constraint.h" #include "expr.h" namespace ledger { @@ -7,73 +6,84 @@ namespace ledger { // jww (2004-07-21): If format.show_empty is set, then include all // subaccounts, empty, balanced or no -item_t * walk_accounts(const account_t * account, - const constraints_t& constraints) +item_t * walk_accounts(const account_t * account, + const node_t * predicate, + const bool show_subtotals) { item_t * item = new item_t; item->account = account; - for (constrained_transactions_list_const_iterator - i(account->transactions.begin(), - account->transactions.end(), constraints); + std::time_t latest = 0; + for (transactions_list::const_iterator i + = std::find_if(account->transactions.begin(), + account->transactions.end(), + value_predicate(predicate)); i != account->transactions.end(); - i++) { - item->value += *(*i); + i = std::find_if(++i, account->transactions.end(), + value_predicate(predicate))) { + if (std::difftime(latest, (*i)->entry->date) < 0) + latest = (*i)->entry->date; - if (constraints.show_subtotals) + item->value += *(*i); + if (show_subtotals) item->total += *(*i); } + item->date = latest; for (accounts_map::const_iterator i = account->accounts.begin(); i != account->accounts.end(); i++) { - item_t * subitem = walk_accounts((*i).second, constraints); + item_t * subitem = walk_accounts((*i).second, predicate, show_subtotals); subitem->parent = item; - if (constraints.show_subtotals) - item->total += subitem->total; + if (std::difftime(item->date, subitem->date) < 0) + item->date = subitem->date; - if (constraints.show_subtotals ? subitem->total : subitem->value) + if (show_subtotals) + item->total += subitem->total; + if (show_subtotals ? subitem->total : subitem->value) item->subitems.push_back(subitem); } return item; } -static inline void sum_items(const item_t * top, - const constraints_t& constraints, - item_t * item) +static inline void sum_items(const item_t * top, + const bool show_subtotals, + item_t * item) { if (top->account == item->account) { item->value += top->value; - if (constraints.show_subtotals) + if (show_subtotals) item->total += top->value; } for (items_deque::const_iterator i = top->subitems.begin(); i != top->subitems.end(); i++) - sum_items(*i, constraints, item); + sum_items(*i, show_subtotals, item); } -item_t * walk_items(const item_t * top, const account_t * account, - const constraints_t& constraints) +item_t * walk_items(const item_t * top, + const account_t * account, + const node_t * predicate, + const bool show_subtotals) { item_t * item = new item_t; item->account = account; - sum_items(top, constraints, item); + sum_items(top, show_subtotals, item); for (accounts_map::const_iterator i = account->accounts.begin(); i != account->accounts.end(); i++) { - item_t * subitem = walk_items(top, (*i).second, constraints); + item_t * subitem = walk_items(top, (*i).second, predicate, show_subtotals); subitem->parent = item; - if (constraints.show_subtotals) + if (show_subtotals) item->total += subitem->total; - if (constraints.show_subtotals ? subitem->total : subitem->value) + if (show_subtotals ? subitem->total : subitem->value) item->subitems.push_back(subitem); } @@ -82,21 +92,26 @@ item_t * walk_items(const item_t * top, const account_t * account, item_t * walk_entries(entries_list::const_iterator begin, entries_list::const_iterator end, - const constraints_t& constraints) + const node_t * predicate, + const bool show_related, + const bool show_inverted) { - unsigned int count = 0; - item_t * result = NULL; + unsigned int count = 0; + item_t * result = NULL; + value_predicate pred_obj(predicate); - for (constrained_entries_list_const_iterator i(begin, end, constraints); + for (entries_list::const_iterator i = std::find_if(begin, end, pred_obj); i != end; - i++) { + i = std::find_if(++i, end, pred_obj)) { item_t * item = NULL; - for (constrained_transactions_list_const_iterator - j((*i)->transactions.begin(), (*i)->transactions.end(), - constraints); + for (transactions_list::const_iterator j + = std::find_if((*i)->transactions.begin(), + (*i)->transactions.end(), pred_obj); j != (*i)->transactions.end(); - j++) { + j = std::find_if(++j, + transactions_list::const_iterator((*i)->transactions.end()), + pred_obj)) { assert(*i == (*j)->entry); if (! item) { @@ -107,7 +122,7 @@ item_t * walk_entries(entries_list::const_iterator begin, } // If show_inverted is true, it implies show_related. - if (! constraints.show_inverted) { + if (! show_inverted) { item_t * subitem = new item_t; subitem->parent = item; subitem->date = item->date; @@ -117,7 +132,7 @@ item_t * walk_entries(entries_list::const_iterator begin, item->subitems.push_back(subitem); } - if (constraints.show_related) + if (show_related) for (transactions_list::iterator k = (*i)->transactions.begin(); k != (*i)->transactions.end(); k++) @@ -128,7 +143,7 @@ item_t * walk_entries(entries_list::const_iterator begin, subitem->payee = item->payee; subitem->account = (*k)->account; subitem->value = *(*k); - if (constraints.show_inverted) + if (show_inverted) subitem->value.negate(); item->subitems.push_back(subitem); } @@ -36,17 +36,22 @@ struct item_t void sort(const node_t * sort_order); }; -class constraints_t; +struct node_t; item_t * walk_accounts(const account_t * account, - const constraints_t& constraints); + const node_t * predicate, + const bool show_subtotals); -item_t * walk_items(const item_t * top, const account_t * account, - const constraints_t& constraints); +item_t * walk_items(const item_t * top, + const account_t * account, + const node_t * predicate, + const bool show_subtotals); item_t * walk_entries(entries_list::const_iterator begin, entries_list::const_iterator end, - const constraints_t& constraints); + const node_t * predicate, + const bool show_related, + const bool show_inverted); } // namespace report @@ -3,7 +3,6 @@ #include "error.h" #include "textual.h" #include "binary.h" -#include "constraint.h" #include "item.h" #include "expr.h" #include "format.h" @@ -23,21 +22,23 @@ namespace ledger { static const std::string bal_fmt = "%20T%2_%-n\n"; -unsigned int show_balances(std::ostream& out, - items_deque& items, - const constraints_t& constraints, - const node_t * sort_order, - const format_t& format, - const item_t * displayed_parent) +unsigned int show_balances(std::ostream& out, + items_deque& items, + const node_t * predicate, + const node_t * sort_order, + const format_t& format, + const bool show_expanded, + const item_t * displayed_parent) { - unsigned int headlines = 0; + unsigned int headlines = 0; + value_predicate pred_obj(predicate); for (items_deque::const_iterator i = items.begin(); i != items.end(); i++) { const item_t * parent = displayed_parent; - if (constraints(*i) && + if (pred_obj(*i) && ((*i)->subitems.size() != 1 || (*i)->total != (*i)->subitems[0]->total)) { format.format_elements(out, *i, parent); @@ -50,9 +51,9 @@ unsigned int show_balances(std::ostream& out, if (sort_order) (*i)->sort(sort_order); - if (constraints.show_expanded) - headlines += show_balances(out, (*i)->subitems, constraints, - sort_order, format, parent); + if (show_expanded) + headlines += show_balances(out, (*i)->subitems, predicate, + sort_order, format, true, parent); } return headlines; @@ -60,17 +61,20 @@ unsigned int show_balances(std::ostream& out, void balance_report(std::ostream& out, item_t * top, - const constraints_t& constraints, + const node_t * predicate, const node_t * sort_order, - const format_t& format) + const format_t& format, + const bool show_expanded, + const bool show_subtotals) { if (sort_order) top->sort(sort_order); - unsigned int headlines = show_balances(out, top->subitems, constraints, - sort_order, format, top); + unsigned int headlines = show_balances(out, top->subitems, predicate, + sort_order, format, show_expanded, + top); - if (constraints.show_subtotals && headlines > 1 && top->total) { + if (show_subtotals && headlines > 1 && top->total) { std::cout << "--------------------\n"; format.format_elements(std::cout, top); } @@ -92,7 +96,7 @@ static void report_value_change(std::ostream& out, const std::time_t date, const balance_pair_t& balance, const balance_pair_t& prev_balance, - const constraints_t& constraints, + const node_t * predicate, const format_t& first_line_format, const format_t& next_lines_format) { @@ -116,7 +120,7 @@ static void report_value_change(std::ostream& out, temp.total = balance; temp.payee = "Commodities revalued"; - if (constraints(&temp)) { + if (value_predicate(predicate)(&temp)) { first_line_format.format_elements(out, &temp); next_lines_format.format_elements(out, &temp); } @@ -125,26 +129,28 @@ static void report_value_change(std::ostream& out, prev_date = date; } -void register_report(std::ostream& out, - item_t * top, - const constraints_t& constraints, - const node_t * sort_order, - const format_t& first_line_format, - const format_t& next_lines_format) +void register_report(std::ostream& out, + item_t * top, + const node_t * predicate, + const node_t * sort_order, + const format_t& first_line_format, + const format_t& next_lines_format, + const bool show_expanded) { if (sort_order) top->sort(sort_order); - balance_pair_t balance; - balance_pair_t last_reported; - account_t splits(NULL, "<Total>"); + balance_pair_t balance; + balance_pair_t last_reported; + account_t splits(NULL, "<Total>"); + value_predicate pred_obj(predicate); for (items_deque::const_iterator i = top->subitems.begin(); i != top->subitems.end(); i++) { bool first = true; - if ((*i)->subitems.size() > 1 && ! constraints.show_expanded) { + if ((*i)->subitems.size() > 1 && ! show_expanded) { item_t summary; summary.date = (*i)->date; summary.parent = *i; @@ -156,10 +162,10 @@ void register_report(std::ostream& out, summary.value += (*j)->value; summary.total = balance + summary.value; - bool show = constraints(&summary); + bool show = pred_obj(&summary); if (show && show_commodities_revalued) report_value_change(out, summary.date, balance, last_reported, - constraints, first_line_format, next_lines_format); + predicate, first_line_format, next_lines_format); balance += summary.value; @@ -176,11 +182,10 @@ void register_report(std::ostream& out, j++) { (*j)->total = balance + (*j)->value; - bool show = constraints(*j); + bool show = pred_obj(*j); if (show && first && show_commodities_revalued) { report_value_change(out, (*i)->date, balance, last_reported, - constraints, first_line_format, - next_lines_format); + predicate, first_line_format, next_lines_format); if (show_commodities_revalued_only) first = false; } @@ -205,7 +210,7 @@ void register_report(std::ostream& out, } if (show_commodities_revalued) - report_value_change(out, -1, balance, last_reported, constraints, + report_value_change(out, -1, balance, last_reported, predicate, first_line_format, next_lines_format); } @@ -471,14 +476,20 @@ static void show_help(std::ostream& out) int main(int argc, char * argv[]) { std::list<std::string> files; - ledger::ledger_t * journal = new ledger::ledger_t; - ledger::constraints_t constraints; - std::string predicate; - std::string format_string; - std::string sort_str; - ledger::node_t * sort_order = NULL; - std::string value_expr = "a"; - std::string total_expr = "T"; + + std::string predicate_string; + ledger::node_t * predicate = NULL; + std::string format_string; + std::string sort_string; + ledger::node_t * sort_order = NULL; + std::string value_expr = "a"; + std::string total_expr = "T"; + ledger::ledger_t * journal = new ledger::ledger_t; + + bool show_subtotals = true; + bool show_expanded = false; + bool show_related = false; + bool show_inverted = false; #ifdef DEBUG bool debug = false; @@ -562,48 +573,48 @@ int main(int argc, char * argv[]) break; case 'b': - if (! predicate.empty()) - predicate += "&"; - predicate += "(d>=["; - predicate += optarg; - predicate += "])"; + if (! predicate_string.empty()) + predicate_string += "&"; + predicate_string += "(d>=["; + predicate_string += optarg; + predicate_string += "])"; break; case 'e': - if (! predicate.empty()) - predicate += "&"; - predicate += "(d<["; - predicate += optarg; - predicate += "])"; + if (! predicate_string.empty()) + predicate_string += "&"; + predicate_string += "(d<["; + predicate_string += optarg; + predicate_string += "])"; break; case 'c': { - if (! predicate.empty()) - predicate += "&"; - predicate += "(d<"; + if (! predicate_string.empty()) + predicate_string += "&"; + predicate_string += "(d<"; std::ostringstream now; now << std::time(NULL); - predicate += now.str(); - predicate += ")"; + predicate_string += now.str(); + predicate_string += ")"; break; } case 'C': - if (! predicate.empty()) - predicate += "&"; - predicate += "X"; + if (! predicate_string.empty()) + predicate_string += "&"; + predicate_string += "X"; break; case 'U': - if (! predicate.empty()) - predicate += "&"; - predicate += "!X"; + if (! predicate_string.empty()) + predicate_string += "&"; + predicate_string += "!X"; break; case 'R': - if (! predicate.empty()) - predicate += "&"; - predicate += "R"; + if (! predicate_string.empty()) + predicate_string += "&"; + predicate_string += "R"; break; // Customizing output @@ -611,32 +622,34 @@ int main(int argc, char * argv[]) format_string = optarg; break; +#if 0 case 'E': - constraints.show_empty = true; + show_empty = true; break; +#endif case 'n': - constraints.show_subtotals = false; + show_subtotals = false; break; case 's': - constraints.show_expanded = true; + show_expanded = true; break; case 'S': - sort_str = optarg; + sort_string = optarg; break; case 'o': - constraints.show_related = true; + show_related = true; break; case 'l': - if (! predicate.empty()) - predicate += "&"; - predicate += "("; - predicate += optarg; - predicate += ")"; + if (! predicate_string.empty()) + predicate_string += "&"; + predicate_string += "("; + predicate_string += optarg; + predicate_string += ")"; break; // Commodity reporting @@ -700,6 +713,7 @@ int main(int argc, char * argv[]) total_expr = "MDMT"; break; +#if 0 case 'W': value_expr = "a"; total_expr = "MD(MT*(d-b/e-b))"; @@ -709,6 +723,11 @@ int main(int argc, char * argv[]) value_expr = "a"; total_expr = "a+MD(MT*(d-b/e-b))"; break; +#endif + + default: + assert(0); + break; } } @@ -778,50 +797,51 @@ int main(int argc, char * argv[]) break; } - constraints.show_expanded = true; + show_expanded = true; - if (! predicate.empty()) - predicate += "&"; + if (! predicate_string.empty()) + predicate_string += "&"; if (argv[index][0] == '-') { - predicate += "(!/"; - predicate += argv[index] + 1; + predicate_string += "(!/"; + predicate_string += argv[index] + 1; } else { - predicate += "(/"; - predicate += argv[index]; + predicate_string += "(/"; + predicate_string += argv[index]; } - predicate += "/)"; + predicate_string += "/)"; } for (; index < argc; index++) { - constraints.show_expanded = true; + show_expanded = true; - if (! predicate.empty()) - predicate += "&"; + if (! predicate_string.empty()) + predicate_string += "&"; if (argv[index][0] == '-') { - predicate += "(!//"; - predicate += argv[index] + 1; + predicate_string += "(!//"; + predicate_string += argv[index] + 1; } else { - predicate += "(//"; - predicate += argv[index]; + predicate_string += "(//"; + predicate_string += argv[index]; } - predicate += "/)"; + predicate_string += "/)"; } - // Copy the constraints to the format object, and compile the value - // and total style strings + // Compile the predicate - if (! predicate.empty()) { + if (! predicate_string.empty()) { #ifdef DEBUG if (debug) - std::cerr << "predicate = " << predicate << std::endl; + std::cerr << "predicate = " << predicate_string << std::endl; #endif - constraints.predicate = ledger::parse_expr(predicate); + predicate = ledger::parse_expr(predicate_string); } - if (! sort_str.empty()) - sort_order = ledger::parse_expr(sort_str); + // Compile the sorting string + + if (! sort_string.empty()) + sort_order = ledger::parse_expr(sort_string); // Setup the meaning of %t and %T encountered in format strings @@ -834,7 +854,8 @@ int main(int argc, char * argv[]) #if 0 if (ledger::item_t * top = ledger::walk_entries(journal->entries.begin(), - journal->entries.end(), constraints)) { + journal->entries.end(), predicate, + show_related, show_inverted)) { ledger::format_t * format = new ledger::format_t(format_string); ledger::entry_report(std::cout, top, *format); #ifdef DEBUG @@ -847,9 +868,9 @@ int main(int argc, char * argv[]) else if (command == "equity") { #if 0 if (ledger::item_t * top - = ledger::walk_accounts(journal->master, constraints)) { + = ledger::walk_accounts(journal->master, predicate, show_subtotals)) { ledger::format_t * format = new ledger::format_t(format_string); - ledger::entry_report(std::cout, top, constraints, *format); + ledger::entry_report(std::cout, top, predicate, *format); #ifdef DEBUG delete top; delete format; @@ -857,14 +878,15 @@ int main(int argc, char * argv[]) } #endif } - else if (! sort_order && ! constraints.show_related && + else if (! sort_order && ! show_related && (command == "balance" || command == "bal")) { if (ledger::item_t * top - = ledger::walk_accounts(journal->master, constraints)) { + = ledger::walk_accounts(journal->master, predicate, show_subtotals)) { ledger::format_t * format = new ledger::format_t(format_string.empty() ? ledger::bal_fmt : format_string); - ledger::balance_report(std::cout, top, constraints, sort_order, *format); + ledger::balance_report(std::cout, top, predicate, sort_order, *format, + show_expanded, show_subtotals); #ifdef DEBUG delete format; delete top; @@ -874,14 +896,16 @@ int main(int argc, char * argv[]) else if (command == "balance" || command == "bal") { if (ledger::item_t * list = ledger::walk_entries(journal->entries.begin(), - journal->entries.end(), constraints)) + journal->entries.end(), predicate, + show_related, show_inverted)) if (ledger::item_t * top - = ledger::walk_items(list, journal->master, constraints)) { + = ledger::walk_items(list, journal->master, predicate, + show_subtotals)) { ledger::format_t * format = new ledger::format_t(format_string.empty() ? ledger::bal_fmt : format_string); - ledger::balance_report(std::cout, top, constraints, sort_order, - *format); + ledger::balance_report(std::cout, top, predicate, sort_order, *format, + show_expanded, show_subtotals); #ifdef DEBUG delete format; delete top; @@ -890,12 +914,13 @@ int main(int argc, char * argv[]) } } else if (command == "register" || command == "reg") { - if (constraints.show_related) - constraints.show_inverted = true; + if (show_related) + show_inverted = true; if (ledger::item_t * top = ledger::walk_entries(journal->entries.begin(), - journal->entries.end(), constraints)) { + journal->entries.end(), predicate, + show_related, show_inverted)) { std::string first_line_format; std::string next_lines_format; @@ -911,9 +936,11 @@ int main(int argc, char * argv[]) ledger::format_t * format = new ledger::format_t(first_line_format); ledger::format_t * nformat = new ledger::format_t(next_lines_format); - ledger::register_report(std::cout, top, constraints, sort_order, - *format, *nformat); + + ledger::register_report(std::cout, top, predicate, sort_order, + *format, *nformat, show_expanded); #ifdef DEBUG + delete nformat; delete format; delete top; #endif @@ -937,6 +964,8 @@ int main(int argc, char * argv[]) #ifdef DEBUG delete journal; + if (predicate) + delete predicate; if (sort_order) delete sort_order; @@ -1,6 +1,6 @@ #include "textual.h" -#include "constraint.h" #include "error.h" +#include "expr.h" #include <vector> #include <fstream> |