diff options
Diffstat (limited to 'src/textual.cc')
-rw-r--r-- | src/textual.cc | 579 |
1 files changed, 296 insertions, 283 deletions
diff --git a/src/textual.cc b/src/textual.cc index 85b1a14b..9a4fee8e 100644 --- a/src/textual.cc +++ b/src/textual.cc @@ -49,26 +49,26 @@ namespace ledger { namespace { - typedef std::pair<commodity_t *, amount_t> fixed_rate_t; + typedef std::pair<commodity_t *, amount_t> fixed_rate_t; typedef variant<account_t *, string, fixed_rate_t> state_t; class parse_context_t : public noncopyable { public: - journal_t& journal; - scope_t& scope; + journal_t& journal; + scope_t& scope; std::list<state_t> state_stack; #if defined(TIMELOG_SUPPORT) - time_log_t timelog; + time_log_t timelog; #endif - bool strict; - std::size_t count; - std::size_t errors; - std::size_t sequence; + bool strict; + std::size_t count; + std::size_t errors; + std::size_t sequence; parse_context_t(journal_t& _journal, scope_t& _scope) - : journal(_journal), scope(_scope), timelog(journal), - strict(false), count(0), errors(0), sequence(1) {} + : journal(_journal), scope(_scope), timelog(journal, scope), + strict(false), count(0), errors(0), sequence(1) {} bool front_is_account() { return state_stack.front().type() == typeid(account_t *); @@ -82,8 +82,8 @@ namespace { account_t * top_account() { foreach (state_t& state, state_stack) - if (state.type() == typeid(account_t *)) - return boost::get<account_t *>(state); + if (state.type() == typeid(account_t *)) + return boost::get<account_t *>(state); return NULL; } }; @@ -97,9 +97,9 @@ namespace { instance_t * parent; accounts_map account_aliases; const path * original_file; - path pathname; + path pathname; std::istream& in; - char linebuf[MAX_LINE + 1]; + char linebuf[MAX_LINE + 1]; std::size_t linenum; istream_pos_type line_beg_pos; istream_pos_type curr_pos; @@ -107,9 +107,9 @@ namespace { optional<date_t::year_type> current_year; instance_t(parse_context_t& _context, - std::istream& _in, - const path * _original_file = NULL, - instance_t * _parent = NULL); + std::istream& _in, + const path * _original_file = NULL, + instance_t * _parent = NULL); ~instance_t(); @@ -117,7 +117,7 @@ namespace { std::streamsize read_line(char *& line); bool peek_whitespace_line() { return (in.good() && ! in.eof() && - (in.peek() == ' ' || in.peek() == '\t')); + (in.peek() == ' ' || in.peek() == '\t')); } void read_next_directive(); @@ -148,31 +148,31 @@ namespace { void define_directive(char * line); bool general_directive(char * line); - post_t * parse_post(char * line, - std::streamsize len, - account_t * account, - xact_t * xact, - bool defer_expr = false); + post_t * parse_post(char * line, + std::streamsize len, + account_t * account, + xact_t * xact, + bool defer_expr = false); bool parse_posts(account_t * account, - xact_base_t& xact, - const bool defer_expr = false); + xact_base_t& xact, + const bool defer_expr = false); - xact_t * parse_xact(char * line, - std::streamsize len, - account_t * account); + xact_t * parse_xact(char * line, + std::streamsize len, + account_t * account); virtual expr_t::ptr_op_t lookup(const symbol_t::kind_t kind, - const string& name); + const string& name); }; - void parse_amount_expr(std::istream& in, - scope_t& scope, - post_t& post, - amount_t& amount, - const parse_flags_t& flags = PARSE_DEFAULT, - const bool defer_expr = false, - optional<expr_t> * amount_expr = NULL) + void parse_amount_expr(std::istream& in, + scope_t& scope, + post_t& post, + amount_t& amount, + const parse_flags_t& flags = PARSE_DEFAULT, + const bool defer_expr = false, + optional<expr_t> * amount_expr = NULL) { expr_t expr(in, flags.plus_flags(PARSE_PARTIAL)); @@ -181,25 +181,25 @@ namespace { #if defined(DEBUG_ENABLED) DEBUG_IF("textual.parse") { if (_debug_stream) { - ledger::dump_value_expr(*_debug_stream, expr); - *_debug_stream << std::endl; + ledger::dump_value_expr(*_debug_stream, expr); + *_debug_stream << std::endl; } } #endif if (expr) { if (amount_expr) - *amount_expr = expr; + *amount_expr = expr; if (! defer_expr) - amount = post.resolve_expr(scope, expr); + amount = post.resolve_expr(scope, expr); } } } instance_t::instance_t(parse_context_t& _context, - std::istream& _in, - const path * _original_file, - instance_t * _parent) + std::istream& _in, + const path * _original_file, + instance_t * _parent) : context(_context), parent(_parent), original_file(_original_file), pathname(original_file ? *original_file : "/dev/stdin"), in(_in) { @@ -216,7 +216,7 @@ void instance_t::parse() INFO("Parsing file '" << pathname.string() << "'"); TRACE_START(instance_parse, 1, - "Done parsing file '" << pathname.string() << "'"); + "Done parsing file '" << pathname.string() << "'"); if (! in.good() || in.eof()) return; @@ -232,30 +232,30 @@ void instance_t::parse() string current_context = error_context(); if (parent) { - std::list<instance_t *> instances; + std::list<instance_t *> instances; - for (instance_t * instance = parent; - instance; - instance = instance->parent) - instances.push_front(instance); + for (instance_t * instance = parent; + instance; + instance = instance->parent) + instances.push_front(instance); - foreach (instance_t * instance, instances) - add_error_context(_("In file included from %1") - << file_context(instance->pathname, - instance->linenum)); + foreach (instance_t * instance, instances) + add_error_context(_("In file included from %1") + << file_context(instance->pathname, + instance->linenum)); } add_error_context(_("While parsing file %1") - << file_context(pathname, linenum)); + << file_context(pathname, linenum)); if (caught_signal != NONE_CAUGHT) - throw; + throw; string err_context = error_context(); if (! err_context.empty()) - std::cerr << err_context << std::endl; + std::cerr << err_context << std::endl; if (! current_context.empty()) - std::cerr << current_context << std::endl; + std::cerr << current_context << std::endl; std::cerr << _("Error: ") << err.what() << std::endl; context.errors++; @@ -268,7 +268,7 @@ void instance_t::parse() std::streamsize instance_t::read_line(char *& line) { assert(in.good()); - assert(! in.eof()); // no one should call us in that case + assert(! in.eof()); // no one should call us in that case line_beg_pos = curr_pos; @@ -283,7 +283,7 @@ std::streamsize instance_t::read_line(char *& line) else line = linebuf; - if (line[len - 1] == '\r') // strip Windows CRLF down to LF + if (line[len - 1] == '\r') // strip Windows CRLF down to LF line[--len] = '\0'; linenum++; @@ -291,7 +291,7 @@ std::streamsize instance_t::read_line(char *& line) curr_pos = line_beg_pos; curr_pos += len; - return len - 1; // LF is being silently dropped + return len - 1; // LF is being silently dropped } return 0; } @@ -306,7 +306,7 @@ void instance_t::read_next_directive() switch (line[0]) { case '\0': - assert(false); // shouldn't ever reach here + assert(false); // shouldn't ever reach here break; case ' ': @@ -314,13 +314,13 @@ void instance_t::read_next_directive() break; } - case ';': // comments + case ';': // comments case '#': case '*': case '|': break; - case '-': // option setting + case '-': // option setting option_directive(line); break; @@ -336,10 +336,10 @@ void instance_t::read_next_directive() case '9': xact_directive(line, len); break; - case '=': // automated xact + case '=': // automated xact automated_xact_directive(line); break; - case '~': // period xact + case '~': // period xact period_xact_directive(line); break; @@ -347,47 +347,47 @@ void instance_t::read_next_directive() case '!': line++; // fall through... - default: // some other directive + default: // some other directive if (! general_directive(line)) { switch (line[0]) { #if defined(TIMELOG_SUPPORT) case 'i': - clock_in_directive(line, false); - break; + clock_in_directive(line, false); + break; case 'I': - clock_in_directive(line, true); - break; + clock_in_directive(line, true); + break; case 'o': - clock_out_directive(line, false); - break; + clock_out_directive(line, false); + break; case 'O': - clock_out_directive(line, true); - break; + clock_out_directive(line, true); + break; case 'h': case 'b': - break; + break; #endif // TIMELOG_SUPPORT - case 'A': // a default account for unbalanced posts - default_account_directive(line); - break; - case 'C': // a set of conversions - price_conversion_directive(line); - break; - case 'D': // a default commodity for "xact" - default_commodity_directive(line); - break; - case 'N': // don't download prices - nomarket_directive(line); - break; - case 'P': // a pricing xact - price_xact_directive(line); - break; - case 'Y': // set the current year - year_directive(line); - break; + case 'A': // a default account for unbalanced posts + default_account_directive(line); + break; + case 'C': // a set of conversions + price_conversion_directive(line); + break; + case 'D': // a default commodity for "xact" + default_commodity_directive(line); + break; + case 'N': // don't download prices + nomarket_directive(line); + break; + case 'P': // a pricing xact + price_xact_directive(line); + break; + case 'Y': // set the current year + year_directive(line); + break; } } break; @@ -418,9 +418,9 @@ void instance_t::clock_in_directive(char * line, bool /*capitalized*/) position.sequence = context.sequence++; time_xact_t event(position, parse_datetime(datetime, current_year), - p ? context.top_account()->find_account(p) : NULL, - n ? n : "", - end ? end : ""); + p ? context.top_account()->find_account(p) : NULL, + n ? n : "", + end ? end : ""); context.timelog.clock_in(event); } @@ -447,9 +447,9 @@ void instance_t::clock_out_directive(char * line, bool /*capitalized*/) position.sequence = context.sequence++; time_xact_t event(position, parse_datetime(datetime, current_year), - p ? context.top_account()->find_account(p) : NULL, - n ? n : "", - end ? end : ""); + p ? context.top_account()->find_account(p) : NULL, + n ? n : "", + end ? end : ""); context.timelog.clock_out(event); context.count++; @@ -526,8 +526,8 @@ void instance_t::automated_xact_directive(char * line) std::auto_ptr<auto_xact_t> ae (new auto_xact_t(query_t(string(skip_ws(line + 1)), - keep_details_t(true, true, true), false))); - ae->pos = position_t(); + keep_details_t(true, true, true), false))); + ae->pos = position_t(); ae->pos->pathname = pathname; ae->pos->beg_pos = line_beg_pos; ae->pos->beg_line = linenum; @@ -540,7 +540,7 @@ void instance_t::automated_xact_directive(char * line) context.journal.auto_xacts.push_back(ae.get()); - ae->journal = &context.journal; + ae->journal = &context.journal; ae->pos->end_pos = curr_pos; ae->pos->end_line = linenum; @@ -610,7 +610,7 @@ void instance_t::xact_directive(char * line, std::streamsize len) std::auto_ptr<xact_t> manager(xact); if (context.journal.add_xact(xact)) { - manager.release(); // it's owned by the journal now + manager.release(); // it's owned by the journal now context.count++; } // It's perfectly valid for the journal to reject the xact, which it will @@ -650,10 +650,10 @@ void instance_t::include_directive(char * line) mask_t glob; #if BOOST_VERSION >= 103700 - path parent_path = filename.parent_path(); + path parent_path = filename.parent_path(); glob.assign_glob(filename.filename()); #else // BOOST_VERSION >= 103700 - path parent_path = filename.branch_path(); + path parent_path = filename.branch_path(); glob.assign_glob(filename.leaf()); #endif // BOOST_VERSION >= 103700 @@ -661,33 +661,33 @@ void instance_t::include_directive(char * line) if (exists(parent_path)) { filesystem::directory_iterator end; for (filesystem::directory_iterator iter(parent_path); - iter != end; - ++iter) { + iter != end; + ++iter) { #if BOOST_VERSION <= 103500 if (is_regular(*iter)) #else if (is_regular_file(*iter)) #endif - { + { #if BOOST_VERSION >= 103700 - string base = (*iter).filename(); + string base = (*iter).filename(); #else // BOOST_VERSION >= 103700 - string base = (*iter).leaf(); + string base = (*iter).leaf(); #endif // BOOST_VERSION >= 103700 - if (glob.match(base)) { - path inner_file(*iter); - ifstream stream(inner_file); - instance_t instance(context, stream, &inner_file, this); - instance.parse(); - files_found = true; - } + if (glob.match(base)) { + path inner_file(*iter); + ifstream stream(inner_file); + instance_t instance(context, stream, &inner_file, this); + instance.parse(); + files_found = true; + } } } } if (! files_found) throw_(std::runtime_error, - _("File to include was not found: '%1'") << filename); + _("File to include was not found: '%1'") << filename); } @@ -707,17 +707,17 @@ void instance_t::end_directive(char * kind) if ((name.empty() || name == "account") && ! context.front_is_account()) throw_(std::runtime_error, - _("'end account' directive does not match open directive")); + _("'end account' directive does not match open directive")); else if (name == "tag" && ! context.front_is_string()) throw_(std::runtime_error, - _("'end tag' directive does not match open directive")); + _("'end tag' directive does not match open directive")); else if (name == "fixed" && ! context.front_is_fixed_rate()) throw_(std::runtime_error, - _("'end fixed' directive does not match open directive")); + _("'end fixed' directive does not match open directive")); if (context.state_stack.size() <= 1) throw_(std::runtime_error, - _("'end' found, but no enclosing tag or account directive")); + _("'end' found, but no enclosing tag or account directive")); else context.state_stack.pop_front(); } @@ -747,9 +747,9 @@ void instance_t::fixed_directive(char * line) { if (optional<std::pair<commodity_t *, price_point_t> > price_point = commodity_pool_t::current_pool->parse_price_directive(trim_ws(line), - true)) { + true)) { context.state_stack.push_front(fixed_rate_t(price_point->first, - price_point->second.price)); + price_point->second.price)); } else { throw_(std::runtime_error, _("Error in fixed directive")); } @@ -789,7 +789,7 @@ void instance_t::account_mapping_directive(char * line) if (payee_regex) context.journal.account_mappings.push_back (account_mapping_t(mask_t(payee_regex), - context.top_account()->find_account(account_name))); + context.top_account()->find_account(account_name))); while (peek_whitespace_line()) { #if defined(NO_ASSERTS) @@ -805,7 +805,7 @@ void instance_t::account_mapping_directive(char * line) context.journal.account_mappings.push_back (account_mapping_t(mask_t(payee_regex), - context.top_account()->find_account(account_name))); + context.top_account()->find_account(account_name))); } } @@ -822,7 +822,7 @@ void instance_t::tag_directive(char * line) void instance_t::define_directive(char * line) { expr_t def(skip_ws(line)); - def.compile(context.scope); // causes definitions to be established + def.compile(context.scope); // causes definitions to be established } bool instance_t::general_directive(char * line) @@ -923,18 +923,18 @@ bool instance_t::general_directive(char * line) return false; } -post_t * instance_t::parse_post(char * line, - std::streamsize len, - account_t * account, - xact_t * xact, - bool defer_expr) +post_t * instance_t::parse_post(char * line, + std::streamsize len, + account_t * account, + xact_t * xact, + bool defer_expr) { TRACE_START(post_details, 1, "Time spent parsing postings:"); std::auto_ptr<post_t> post(new post_t); - post->xact = xact; // this could be NULL - post->pos = position_t(); + post->xact = xact; // this could be NULL + post->pos = position_t(); post->pos->pathname = pathname; post->pos->beg_pos = line_beg_pos; post->pos->beg_line = linenum; @@ -958,14 +958,14 @@ post_t * instance_t::parse_post(char * line, post->set_state(item_t::CLEARED); p = skip_ws(p + 1); DEBUG("textual.parse", "line " << linenum << ": " - << "Parsed the CLEARED flag"); + << "Parsed the CLEARED flag"); break; case '!': post->set_state(item_t::PENDING); p = skip_ws(p + 1); DEBUG("textual.parse", "line " << linenum << ": " - << "Parsed the PENDING flag"); + << "Parsed the PENDING flag"); break; } @@ -988,19 +988,19 @@ post_t * instance_t::parse_post(char * line, if ((*p == '[' && *(e - 1) == ']') || (*p == '(' && *(e - 1) == ')')) { post->add_flags(POST_VIRTUAL); DEBUG("textual.parse", "line " << linenum << ": " - << "Parsed a virtual account name"); + << "Parsed a virtual account name"); if (*p == '[') { post->add_flags(POST_MUST_BALANCE); DEBUG("textual.parse", "line " << linenum << ": " - << "Posting must balance"); + << "Posting must balance"); } p++; e--; } string name(p, e - p); DEBUG("textual.parse", "line " << linenum << ": " - << "Parsed account name " << name); + << "Parsed account name " << name); if (account_aliases.size() > 0) { accounts_map::const_iterator i = account_aliases.find(name); @@ -1013,15 +1013,15 @@ post_t * instance_t::parse_post(char * line, if (context.strict && ! post->account->has_flags(ACCOUNT_KNOWN)) { if (post->_state == item_t::UNCLEARED) warning_(_("\"%1\", line %2: Unknown account '%3'") - << pathname << linenum << post->account->fullname()); + << pathname << linenum << post->account->fullname()); post->account->add_flags(ACCOUNT_KNOWN); } if (post->account->name == _("Unknown")) { foreach (account_mapping_t& value, context.journal.account_mappings) { if (value.first.match(xact->payee)) { - post->account = value.second; - break; + post->account = value.second; + break; } } } @@ -1036,39 +1036,39 @@ post_t * instance_t::parse_post(char * line, beg = next - line; ptristream stream(next, len - beg); - if (*next != '(') // indicates a value expression + if (*next != '(') // indicates a value expression post->amount.parse(stream, PARSE_NO_REDUCE); else parse_amount_expr(stream, context.scope, *post.get(), post->amount, - PARSE_NO_REDUCE | PARSE_SINGLE | PARSE_NO_ASSIGN, - defer_expr, &post->amount_expr); + PARSE_NO_REDUCE | PARSE_SINGLE | PARSE_NO_ASSIGN, + defer_expr, &post->amount_expr); if (! post->amount.is_null() && post->amount.has_commodity()) { if (context.strict && - ! post->amount.commodity().has_flags(COMMODITY_KNOWN)) { - if (post->_state == item_t::UNCLEARED) - warning_(_("\"%1\", line %2: Unknown commodity '%3'") - << pathname << linenum << post->amount.commodity()); - post->amount.commodity().add_flags(COMMODITY_KNOWN); + ! post->amount.commodity().has_flags(COMMODITY_KNOWN)) { + if (post->_state == item_t::UNCLEARED) + warning_(_("\"%1\", line %2: Unknown commodity '%3'") + << pathname << linenum << post->amount.commodity()); + post->amount.commodity().add_flags(COMMODITY_KNOWN); } if (! post->amount.has_annotation()) { - foreach (state_t& state, context.state_stack) { - if (state.type() == typeid(fixed_rate_t)) { - fixed_rate_t& rate(boost::get<fixed_rate_t>(state)); - if (*rate.first == post->amount.commodity()) { - annotation_t details(rate.second); - details.add_flags(ANNOTATION_PRICE_FIXATED); - post->amount.annotate(details); - break; - } - } - } + foreach (state_t& state, context.state_stack) { + if (state.type() == typeid(fixed_rate_t)) { + fixed_rate_t& rate(boost::get<fixed_rate_t>(state)); + if (*rate.first == post->amount.commodity()) { + annotation_t details(rate.second); + details.add_flags(ANNOTATION_PRICE_FIXATED); + post->amount.annotate(details); + break; + } + } + } } } DEBUG("textual.parse", "line " << linenum << ": " - << "post amount = " << post->amount); + << "post amount = " << post->amount); if (stream.eof()) { next = NULL; @@ -1078,62 +1078,73 @@ post_t * instance_t::parse_post(char * line, // Parse the optional cost (@ PER-UNIT-COST, @@ TOTAL-COST) if (*next == '@') { - DEBUG("textual.parse", "line " << linenum << ": " - << "Found a price indicator"); - - bool per_unit = true; - - if (*++next == '@') { - per_unit = false; - post->add_flags(POST_COST_IN_FULL); - DEBUG("textual.parse", "line " << linenum << ": " - << "And it's for a total price"); - } - - beg = ++next - line; - - p = skip_ws(next); - if (*p) { - post->cost = amount_t(); - - beg = p - line; - ptristream cstream(p, len - beg); - - if (*p != '(') // indicates a value expression - post->cost->parse(cstream, PARSE_NO_MIGRATE); - else - parse_amount_expr(cstream, context.scope, *post.get(), *post->cost, - PARSE_NO_MIGRATE | PARSE_SINGLE | PARSE_NO_ASSIGN); - - if (post->cost->sign() < 0) - throw parse_error(_("A posting's cost may not be negative")); - - post->cost->in_place_unround(); - - if (per_unit) { - // For the sole case where the cost might be uncommoditized, - // guarantee that the commodity of the cost after multiplication - // is the same as it was before. - commodity_t& cost_commodity(post->cost->commodity()); - *post->cost *= post->amount; - post->cost->set_commodity(cost_commodity); - } - else if (post->amount.sign() < 0) { - post->cost->in_place_negate(); - } - - DEBUG("textual.parse", "line " << linenum << ": " - << "Total cost is " << *post->cost); - DEBUG("textual.parse", "line " << linenum << ": " - << "Annotated amount is " << post->amount); - - if (cstream.eof()) - next = NULL; - else - next = skip_ws(p + static_cast<std::ptrdiff_t>(cstream.tellg())); - } else { - throw parse_error(_("Expected a cost amount")); - } + DEBUG("textual.parse", "line " << linenum << ": " + << "Found a price indicator"); + + bool per_unit = true; + + if (*++next == '@') { + per_unit = false; + post->add_flags(POST_COST_IN_FULL); + DEBUG("textual.parse", "line " << linenum << ": " + << "And it's for a total price"); + } + + beg = ++next - line; + + p = skip_ws(next); + if (*p) { + post->cost = amount_t(); + + bool fixed_cost = false; + if (*p == '=') { + p++; + fixed_cost = true; + if (*p == '\0') + throw parse_error(_("Posting is missing a cost amount")); + } + + beg = p - line; + ptristream cstream(p, len - beg); + + if (*p != '(') // indicates a value expression + post->cost->parse(cstream, PARSE_NO_MIGRATE); + else + parse_amount_expr(cstream, context.scope, *post.get(), *post->cost, + PARSE_NO_MIGRATE | PARSE_SINGLE | PARSE_NO_ASSIGN); + + if (post->cost->sign() < 0) + throw parse_error(_("A posting's cost may not be negative")); + + post->cost->in_place_unround(); + + if (per_unit) { + // For the sole case where the cost might be uncommoditized, + // guarantee that the commodity of the cost after multiplication + // is the same as it was before. + commodity_t& cost_commodity(post->cost->commodity()); + *post->cost *= post->amount; + post->cost->set_commodity(cost_commodity); + } + else if (post->amount.sign() < 0) { + post->cost->in_place_negate(); + } + + if (fixed_cost) + post->add_flags(POST_COST_FIXATED); + + DEBUG("textual.parse", "line " << linenum << ": " + << "Total cost is " << *post->cost); + DEBUG("textual.parse", "line " << linenum << ": " + << "Annotated amount is " << post->amount); + + if (cstream.eof()) + next = NULL; + else + next = skip_ws(p + static_cast<std::ptrdiff_t>(cstream.tellg())); + } else { + throw parse_error(_("Expected a cost amount")); + } } } } @@ -1142,7 +1153,7 @@ post_t * instance_t::parse_post(char * line, if (xact && next && *next == '=') { DEBUG("textual.parse", "line " << linenum << ": " - << "Found a balance assignment indicator"); + << "Found a balance assignment indicator"); beg = ++next - line; @@ -1153,70 +1164,70 @@ post_t * instance_t::parse_post(char * line, beg = p - line; ptristream stream(p, len - beg); - if (*p != '(') // indicates a value expression - post->assigned_amount->parse(stream, PARSE_NO_MIGRATE); + if (*p != '(') // indicates a value expression + post->assigned_amount->parse(stream, PARSE_NO_MIGRATE); else - parse_amount_expr(stream, context.scope, *post.get(), - *post->assigned_amount, - PARSE_SINGLE | PARSE_NO_MIGRATE); + parse_amount_expr(stream, context.scope, *post.get(), + *post->assigned_amount, + PARSE_SINGLE | PARSE_NO_MIGRATE); if (post->assigned_amount->is_null()) { - if (post->amount.is_null()) - throw parse_error(_("Balance assignment must evaluate to a constant")); - else - throw parse_error(_("Balance assertion must evaluate to a constant")); + if (post->amount.is_null()) + throw parse_error(_("Balance assignment must evaluate to a constant")); + else + throw parse_error(_("Balance assertion must evaluate to a constant")); } DEBUG("textual.parse", "line " << linenum << ": " - << "POST assign: parsed amt = " << *post->assigned_amount); + << "POST assign: parsed amt = " << *post->assigned_amount); - amount_t& amt(*post->assigned_amount); + amount_t& amt(*post->assigned_amount); value_t account_total - (post->account->amount(false).strip_annotations(keep_details_t())); + (post->account->amount(false).strip_annotations(keep_details_t())); DEBUG("post.assign", - "line " << linenum << ": " "account balance = " << account_total); + "line " << linenum << ": " "account balance = " << account_total); DEBUG("post.assign", - "line " << linenum << ": " "post amount = " << amt); + "line " << linenum << ": " "post amount = " << amt); amount_t diff = amt; switch (account_total.type()) { case value_t::AMOUNT: - diff -= account_total.as_amount(); - break; + diff -= account_total.as_amount(); + break; case value_t::BALANCE: - if (optional<amount_t> comm_bal = - account_total.as_balance().commodity_amount(amt.commodity())) - diff -= *comm_bal; - break; + if (optional<amount_t> comm_bal = + account_total.as_balance().commodity_amount(amt.commodity())) + diff -= *comm_bal; + break; default: - break; + break; } DEBUG("post.assign", - "line " << linenum << ": " << "diff = " << diff); + "line " << linenum << ": " << "diff = " << diff); DEBUG("textual.parse", - "line " << linenum << ": " << "POST assign: diff = " << diff); + "line " << linenum << ": " << "POST assign: diff = " << diff); if (! diff.is_zero()) { - if (! post->amount.is_null()) { - diff -= post->amount; - if (! diff.is_zero()) - throw_(parse_error, _("Balance assertion off by %1") << diff); - } else { - post->amount = diff; - DEBUG("textual.parse", "line " << linenum << ": " - << "Overwrite null posting"); - } + if (! post->amount.is_null()) { + diff -= post->amount; + if (! diff.is_zero()) + throw_(parse_error, _("Balance assertion off by %1") << diff); + } else { + post->amount = diff; + DEBUG("textual.parse", "line " << linenum << ": " + << "Overwrite null posting"); + } } if (stream.eof()) - next = NULL; + next = NULL; else - next = skip_ws(p + static_cast<std::ptrdiff_t>(stream.tellg())); + next = skip_ws(p + static_cast<std::ptrdiff_t>(stream.tellg())); } else { throw parse_error(_("Expected an balance assignment/assertion amount")); } @@ -1225,18 +1236,18 @@ post_t * instance_t::parse_post(char * line, // Parse the optional note if (next && *next == ';') { - post->append_note(++next, true, current_year); + post->append_note(++next, context.scope, true, current_year); next = line + len; DEBUG("textual.parse", "line " << linenum << ": " - << "Parsed a posting note"); + << "Parsed a posting note"); } // There should be nothing more to read if (next && *next) throw_(parse_error, - _("Unexpected char '%1' (Note: inline math requires parentheses)") - << *next); + _("Unexpected char '%1' (Note: inline math requires parentheses)") + << *next); post->pos->end_pos = curr_pos; post->pos->end_line = linenum; @@ -1244,7 +1255,8 @@ post_t * instance_t::parse_post(char * line, if (! context.state_stack.empty()) { foreach (const state_t& state, context.state_stack) if (state.type() == typeid(string)) - post->parse_tags(boost::get<string>(state).c_str(), true); + post->parse_tags(boost::get<string>(state).c_str(), context.scope, + true); } TRACE_STOP(post_details, 1); @@ -1260,8 +1272,8 @@ post_t * instance_t::parse_post(char * line, } bool instance_t::parse_posts(account_t * account, - xact_base_t& xact, - const bool defer_expr) + xact_base_t& xact, + const bool defer_expr) { TRACE_START(xact_posts, 1, "Time spent parsing postings:"); @@ -1283,15 +1295,15 @@ bool instance_t::parse_posts(account_t * account, return added; } -xact_t * instance_t::parse_xact(char * line, - std::streamsize len, - account_t * account) +xact_t * instance_t::parse_xact(char * line, + std::streamsize len, + account_t * account) { TRACE_START(xact_text, 1, "Time spent parsing transaction text:"); std::auto_ptr<xact_t> xact(new xact_t); - xact->pos = position_t(); + xact->pos = position_t(); xact->pos->pathname = pathname; xact->pos->beg_pos = line_beg_pos; xact->pos->beg_line = linenum; @@ -1342,8 +1354,8 @@ xact_t * instance_t::parse_xact(char * line, char * p = next_element(next, true); foreach (payee_mapping_t& value, context.journal.payee_mappings) { if (value.first.match(next)) { - xact->payee = value.second; - break; + xact->payee = value.second; + break; } } if (xact->payee.empty()) @@ -1356,7 +1368,7 @@ xact_t * instance_t::parse_xact(char * line, // Parse the xact note if (next && *next == ';') - xact->append_note(++next, current_year); + xact->append_note(++next, context.scope, false, current_year); TRACE_STOP(xact_text, 1); @@ -1376,21 +1388,21 @@ xact_t * instance_t::parse_xact(char * line, if (*p == ';') { item_t * item; if (last_post) - item = last_post; + item = last_post; else - item = xact.get(); + item = xact.get(); // This is a trailing note, and possibly a metadata info tag - item->append_note(p + 1, true, current_year); + item->append_note(p + 1, context.scope, true, current_year); item->pos->end_pos = curr_pos; item->pos->end_line++; } else { reveal_context = false; if (post_t * post = - parse_post(p, len - (p - line), account, xact.get())) { - xact->add_post(post); - last_post = post; + parse_post(p, len - (p - line), account, xact.get())) { + xact->add_post(post); + last_post = post; } } } @@ -1400,11 +1412,11 @@ xact_t * instance_t::parse_xact(char * line, foreach (post_t * post, xact->posts) { if (post->_state == item_t::UNCLEARED) { - result = item_t::UNCLEARED; - break; + result = item_t::UNCLEARED; + break; } else if (post->_state == item_t::PENDING) { - result = item_t::PENDING; + result = item_t::PENDING; } } } @@ -1415,7 +1427,8 @@ xact_t * instance_t::parse_xact(char * line, if (! context.state_stack.empty()) { foreach (const state_t& state, context.state_stack) if (state.type() == typeid(string)) - xact->parse_tags(boost::get<string>(state).c_str(), false); + xact->parse_tags(boost::get<string>(state).c_str(), context.scope, + false); } TRACE_STOP(xact_details, 1); @@ -1427,23 +1440,23 @@ xact_t * instance_t::parse_xact(char * line, if (reveal_context) { add_error_context(_("While parsing transaction:")); add_error_context(source_context(xact->pos->pathname, - xact->pos->beg_pos, curr_pos, "> ")); + xact->pos->beg_pos, curr_pos, "> ")); } throw; } } expr_t::ptr_op_t instance_t::lookup(const symbol_t::kind_t kind, - const string& name) + const string& name) { return context.scope.lookup(kind, name); } std::size_t journal_t::parse(std::istream& in, - scope_t& scope, - account_t * master, - const path * original_file, - bool strict) + scope_t& scope, + account_t * master, + const path * original_file, + bool strict) { TRACE_START(parsing_total, 1, "Total time spent parsing text:"); |