From 38dd1b86554ded5982857324873027cdb3be33ee Mon Sep 17 00:00:00 2001 From: John Wiegley Date: Tue, 24 Feb 2009 13:19:00 -0400 Subject: Fixed an issue involving costs and reduced values --- src/textual.cc | 3 +- src/xact.cc | 138 ++++++++++++++++++++++++-------------------- test/baseline/opt-base.test | 18 +++++- 3 files changed, 93 insertions(+), 66 deletions(-) diff --git a/src/textual.cc b/src/textual.cc index 8012dedf..56d88d9f 100644 --- a/src/textual.cc +++ b/src/textual.cc @@ -828,9 +828,10 @@ post_t * instance_t::parse_post(char * line, ptristream stream(next, len - beg); if (*next != '(') // indicates a value expression - post->amount.parse(stream); + post->amount.parse(stream, amount_t::PARSE_NO_REDUCE); else parse_amount_expr(session_scope, stream, post->amount, post.get(), + static_cast(expr_t::PARSE_NO_REDUCE) | static_cast(expr_t::PARSE_NO_ASSIGN)); if (! post->amount.is_null() && honor_strict && strict && diff --git a/src/xact.cc b/src/xact.cc index 8a7cbb46..af1538e0 100644 --- a/src/xact.cc +++ b/src/xact.cc @@ -89,37 +89,40 @@ bool xact_base_t::finalize() post_t * null_post = NULL; foreach (post_t * post, posts) { - if (post->must_balance()) { - amount_t& p(post->cost ? *post->cost : post->amount); - DEBUG("xact.finalize", "post must balance = " << p); - if (! p.is_null()) { - if (! post->cost && post->amount.is_annotated() && - post->amount.annotation().price) { - // If the amount has no cost, but is annotated with a per-unit - // price, use the price times the amount as the cost - post->cost = *post->amount.annotation().price * post->amount; - DEBUG("xact.finalize", - "annotation price = " << *post->amount.annotation().price); - DEBUG("xact.finalize", "amount = " << post->amount); - DEBUG("xact.finalize", "priced cost = " << *post->cost); - post->add_flags(POST_PRICED); - add_or_set_value(balance, post->cost->rounded()); - } else { - // If the amount was a cost, it very likely has the "keep_precision" - // flag set, meaning commodity display precision is ignored when - // displaying the amount. We never want this set for the balance, - // so we must clear the flag in a temporary to avoid it propagating - // into the balance. - add_or_set_value(balance, p.keep_precision() ? p.rounded() : p); - } + if (! post->must_balance()) + continue; + + amount_t& p(post->cost ? *post->cost : post->amount); + DEBUG("xact.finalize", "post must balance = " << p.reduced()); + if (! p.is_null()) { + if (! post->cost && post->amount.is_annotated() && + post->amount.annotation().price) { + // If the amount has no cost, but is annotated with a per-unit + // price, use the price times the amount as the cost + post->cost = *post->amount.annotation().price * post->amount; + DEBUG("xact.finalize", + "annotation price = " << *post->amount.annotation().price); + DEBUG("xact.finalize", "amount = " << post->amount); + DEBUG("xact.finalize", "priced cost = " << *post->cost); + post->add_flags(POST_PRICED); + add_or_set_value(balance, post->cost->rounded().reduced()); } else { - if (null_post) - throw_(std::logic_error, - "Only one posting with null amount allowed per transaction"); - else - null_post = post; + // If the amount was a cost, it very likely has the "keep_precision" + // flag set, meaning commodity display precision is ignored when + // displaying the amount. We never want this set for the balance, + // so we must clear the flag in a temporary to avoid it propagating + // into the balance. + add_or_set_value(balance, p.keep_precision() ? + p.rounded().reduced() : p.reduced()); } } + else if (null_post) { + throw_(std::logic_error, + _("Only one posting with null amount allowed per transaction")); + } + else { + null_post = post; + } } assert(balance.valid()); @@ -213,42 +216,44 @@ bool xact_base_t::finalize() y = t; } - DEBUG("xact.finalize", "primary amount = " << *x); - DEBUG("xact.finalize", "secondary amount = " << *y); - - commodity_t& comm(x->commodity()); - amount_t per_unit_cost; - amount_t total_cost; - - foreach (post_t * post, posts) { - if (post != top_post && post->must_balance() && - ! post->amount.is_null() && - post->amount.is_annotated() && - post->amount.annotation().price) { - amount_t temp = *post->amount.annotation().price * post->amount; - if (total_cost.is_null()) { - total_cost = temp; - y = &total_cost; - } else { - total_cost += temp; + if (*x && *y) { + DEBUG("xact.finalize", "primary amount = " << *x); + DEBUG("xact.finalize", "secondary amount = " << *y); + + commodity_t& comm(x->commodity()); + amount_t per_unit_cost; + amount_t total_cost; + + foreach (post_t * post, posts) { + if (post != top_post && post->must_balance() && + ! post->amount.is_null() && + post->amount.is_annotated() && + post->amount.annotation().price) { + amount_t temp = *post->amount.annotation().price * post->amount; + if (total_cost.is_null()) { + total_cost = temp; + y = &total_cost; + } else { + total_cost += temp; + } + DEBUG("xact.finalize", "total_cost = " << total_cost); } - DEBUG("xact.finalize", "total_cost = " << total_cost); } - } - per_unit_cost = (*y / *x).abs(); + per_unit_cost = (*y / *x).abs(); - DEBUG("xact.finalize", "per_unit_cost = " << per_unit_cost); + DEBUG("xact.finalize", "per_unit_cost = " << per_unit_cost); - foreach (post_t * post, posts) { - const amount_t& amt(post->amount); + foreach (post_t * post, posts) { + const amount_t& amt(post->amount); - if (post->must_balance() && amt.commodity() == comm) { - balance -= amt; - post->cost = per_unit_cost * amt; - post->add_flags(POST_PRICED); - balance += *post->cost; + if (post->must_balance() && amt.commodity() == comm) { + balance -= amt; + post->cost = per_unit_cost * amt; + post->add_flags(POST_PRICED); + balance += *post->cost; - DEBUG("xact.finalize", "set post->cost to = " << *post->cost); + DEBUG("xact.finalize", "set post->cost to = " << *post->cost); + } } } } @@ -279,7 +284,7 @@ bool xact_base_t::finalize() breakdown.final_cost).rounded()) { DEBUG("xact.finalize", "gain_loss = " << gain_loss); - add_or_set_value(balance, gain_loss); + add_or_set_value(balance, gain_loss.reduced()); account_t * account; if (gain_loss.sign() > 0) @@ -308,22 +313,28 @@ bool xact_base_t::finalize() // Add the final calculated totals each to their related account if (dynamic_cast(this)) { - bool all_null = true; + bool all_null = true; + bool some_null = false; foreach (post_t * post, posts) { if (! post->amount.is_null()) { all_null = false; - // jww (2008-08-09): For now, this feature only works for non-specific - // commodities. + post->amount.in_place_reduce(); + add_or_set_value(post->account->xdata().value, post->amount); DEBUG("xact.finalize.totals", "Total for " << post->account->fullname() << " + " << post->amount << ": " << post->account->xdata().value); + } else { + some_null = true; } } if (all_null) return false; // ignore this xact completely + else if (some_null) + throw_(balance_error, + "There cannot be null amounts after balancing a transaction"); } return true; @@ -403,9 +414,8 @@ void auto_xact_t::extend_xact(xact_base_t& xact, bool post_handler) amount_t amt; assert(post->amount); if (! post->amount.commodity()) { - if (post_handler) + if (post_handler || initial_post->amount.is_null()) continue; - assert(initial_post->amount); amt = initial_post->amount * post->amount; } else { if (post_handler) diff --git a/test/baseline/opt-base.test b/test/baseline/opt-base.test index f82fef16..3c193b1a 100644 --- a/test/baseline/opt-base.test +++ b/test/baseline/opt-base.test @@ -4,10 +4,26 @@ i 2007/03/01 23:00:00 A o 2007/03/02 01:00:00 i 2007/03/11 23:00:00 B o 2007/03/12 01:00:00 + +2006/05/22 * Company + Assets:Receivable $4,000.00 + Income:Contracts -40h @ $100.00 + +2006/05/22 * Company + Assets:Receivable $4,000.00 + Income:Contracts -40h {$20} @ $100.00 + Income:Gains $-3,200.00 >>>1 7200s A + $8,000.00 Assets:Receivable 7200s B + $3,200.00 Equity:Capital Gains + $-3,200.00 + -288000s Income + -288000s Contracts + $-3,200.00 Gains -------------------- - 14400s + $8,000.00 + -273600s >>>2 === 0 -- cgit v1.2.3