summaryrefslogtreecommitdiff
path: root/src/draft.cc
diff options
context:
space:
mode:
authorJohn Wiegley <johnw@newartisans.com>2010-06-13 00:42:25 -0400
committerJohn Wiegley <johnw@newartisans.com>2010-06-13 00:42:25 -0400
commit40f553228f5a28034c6635fdcb4c86af28a385ed (patch)
tree2c40305c9f9841a4c3d453a4a5c49ec69056b4b2 /src/draft.cc
parent556211e623cad88213e5087b5c9c36e754d9aa02 (diff)
parentb1b4e2aadff5983d443d70c09ea86a41b015873f (diff)
downloadfork-ledger-40f553228f5a28034c6635fdcb4c86af28a385ed.tar.gz
fork-ledger-40f553228f5a28034c6635fdcb4c86af28a385ed.tar.bz2
fork-ledger-40f553228f5a28034c6635fdcb4c86af28a385ed.zip
Merge branch 'next'
Diffstat (limited to 'src/draft.cc')
-rw-r--r--src/draft.cc469
1 files changed, 234 insertions, 235 deletions
diff --git a/src/draft.cc b/src/draft.cc
index 69dc7025..ba78fc42 100644
--- a/src/draft.cc
+++ b/src/draft.cc
@@ -57,45 +57,45 @@ void draft_t::xact_template_t::dump(std::ostream& out) const
if (payee_mask.empty())
out << _("Payee mask: INVALID (template expression will cause an error)")
- << std::endl;
+ << std::endl;
else
out << _("Payee mask: ") << payee_mask << std::endl;
if (posts.empty()) {
out << std::endl
- << _("<Posting copied from last related transaction>")
- << std::endl;
+ << _("<Posting copied from last related transaction>")
+ << std::endl;
} else {
bool has_only_from = true;
bool has_only_to = true;
foreach (const post_template_t& post, posts) {
if (post.from)
- has_only_to = false;
+ has_only_to = false;
else
- has_only_from = false;
+ has_only_from = false;
}
foreach (const post_template_t& post, posts) {
straccstream accum;
out << std::endl
- << ACCUM(accum << _("[Posting \"%1\"]")
- << (post.from ? _("from") : _("to")))
- << std::endl;
+ << ACCUM(accum << _("[Posting \"%1\"]")
+ << (post.from ? _("from") : _("to")))
+ << std::endl;
if (post.account_mask)
- out << _(" Account mask: ") << *post.account_mask << std::endl;
+ out << _(" Account mask: ") << *post.account_mask << std::endl;
else if (post.from)
- out << _(" Account mask: <use last of last related accounts>") << std::endl;
+ out << _(" Account mask: <use last of last related accounts>") << std::endl;
else
- out << _(" Account mask: <use first of last related accounts>") << std::endl;
+ out << _(" Account mask: <use first of last related accounts>") << std::endl;
if (post.amount)
- out << _(" Amount: ") << *post.amount << std::endl;
+ out << _(" Amount: ") << *post.amount << std::endl;
if (post.cost)
- out << _(" Cost: ") << *post.cost_operator
- << " " << *post.cost << std::endl;
+ out << _(" Cost: ") << *post.cost_operator
+ << " " << *post.cost << std::endl;
}
}
}
@@ -104,7 +104,7 @@ void draft_t::parse_args(const value_t& args)
{
regex date_mask(_("([0-9]+(?:[-/.][0-9]+)?(?:[-/.][0-9]+))?"));
smatch what;
- bool check_for_date = true;
+ bool check_for_date = true;
tmpl = xact_template_t();
@@ -116,16 +116,16 @@ void draft_t::parse_args(const value_t& args)
for (; begin != end; begin++) {
if (check_for_date &&
- regex_match((*begin).to_string(), what, date_mask)) {
+ regex_match((*begin).to_string(), what, date_mask)) {
tmpl->date = parse_date(what[0]);
check_for_date = false;
}
else if (check_for_date &&
- bool(weekday = string_to_day_of_week(what[0]))) {
+ bool(weekday = string_to_day_of_week(what[0]))) {
short dow = static_cast<short>(*weekday);
date_t date = CURRENT_DATE() - date_duration(1);
while (date.day_of_week() != dow)
- date -= date_duration(1);
+ date -= date_duration(1);
tmpl->date = date;
check_for_date = false;
}
@@ -133,81 +133,80 @@ void draft_t::parse_args(const value_t& args)
string arg = (*begin).to_string();
if (arg == "at") {
- if (begin == end)
- throw std::runtime_error(_("Invalid xact command arguments"));
- tmpl->payee_mask = (*++begin).to_string();
+ if (begin == end)
+ throw std::runtime_error(_("Invalid xact command arguments"));
+ tmpl->payee_mask = (*++begin).to_string();
}
else if (arg == "to" || arg == "from") {
- if (! post || post->account_mask) {
- tmpl->posts.push_back(xact_template_t::post_template_t());
- post = &tmpl->posts.back();
- }
- if (begin == end)
- throw std::runtime_error(_("Invalid xact command arguments"));
- post->account_mask = mask_t((*++begin).to_string());
- post->from = arg == "from";
+ if (! post || post->account_mask) {
+ tmpl->posts.push_back(xact_template_t::post_template_t());
+ post = &tmpl->posts.back();
+ }
+ if (begin == end)
+ throw std::runtime_error(_("Invalid xact command arguments"));
+ post->account_mask = mask_t((*++begin).to_string());
+ post->from = arg == "from";
}
else if (arg == "on") {
- if (begin == end)
- throw std::runtime_error(_("Invalid xact command arguments"));
- tmpl->date = parse_date((*++begin).to_string());
- check_for_date = false;
+ if (begin == end)
+ throw std::runtime_error(_("Invalid xact command arguments"));
+ tmpl->date = parse_date((*++begin).to_string());
+ check_for_date = false;
}
else if (arg == "code") {
- if (begin == end)
- throw std::runtime_error(_("Invalid xact command arguments"));
- tmpl->code = (*++begin).to_string();
+ if (begin == end)
+ throw std::runtime_error(_("Invalid xact command arguments"));
+ tmpl->code = (*++begin).to_string();
}
else if (arg == "note") {
- if (begin == end)
- throw std::runtime_error(_("Invalid xact command arguments"));
- tmpl->note = (*++begin).to_string();
+ if (begin == end)
+ throw std::runtime_error(_("Invalid xact command arguments"));
+ tmpl->note = (*++begin).to_string();
}
else if (arg == "rest") {
- ; // just ignore this argument
+ ; // just ignore this argument
}
else if (arg == "@" || arg == "@@") {
- amount_t cost;
- post->cost_operator = arg;
- if (begin == end)
- throw std::runtime_error(_("Invalid xact command arguments"));
- arg = (*++begin).to_string();
- if (! cost.parse(arg, PARSE_SOFT_FAIL | PARSE_NO_MIGRATE))
- throw std::runtime_error(_("Invalid xact command arguments"));
- post->cost = cost;
+ amount_t cost;
+ post->cost_operator = arg;
+ if (begin == end)
+ throw std::runtime_error(_("Invalid xact command arguments"));
+ arg = (*++begin).to_string();
+ if (! cost.parse(arg, PARSE_SOFT_FAIL | PARSE_NO_MIGRATE))
+ throw std::runtime_error(_("Invalid xact command arguments"));
+ post->cost = cost;
}
else {
- // Without a preposition, it is either:
- //
- // A payee, if we have not seen one
- // An account or an amount, if we have
- // An account if an amount has just been seen
- // An amount if an account has just been seen
-
- if (tmpl->payee_mask.empty()) {
- tmpl->payee_mask = arg;
- }
- else {
- amount_t amt;
- optional<mask_t> account;
-
- if (! amt.parse(arg, PARSE_SOFT_FAIL | PARSE_NO_MIGRATE))
- account = mask_t(arg);
-
- if (! post ||
- (account && post->account_mask) ||
- (! account && post->amount)) {
- tmpl->posts.push_back(xact_template_t::post_template_t());
- post = &tmpl->posts.back();
- }
-
- if (account) {
- post->from = false;
- post->account_mask = account;
- } else {
- post->amount = amt;
- }
- }
+ // Without a preposition, it is either:
+ //
+ // A payee, if we have not seen one
+ // An account or an amount, if we have
+ // An account if an amount has just been seen
+ // An amount if an account has just been seen
+
+ if (tmpl->payee_mask.empty()) {
+ tmpl->payee_mask = arg;
+ } else {
+ amount_t amt;
+ optional<mask_t> account;
+
+ if (! amt.parse(arg, PARSE_SOFT_FAIL | PARSE_NO_MIGRATE))
+ account = mask_t(arg);
+
+ if (! post ||
+ (account && post->account_mask) ||
+ (! account && post->amount)) {
+ tmpl->posts.push_back(xact_template_t::post_template_t());
+ post = &tmpl->posts.back();
+ }
+
+ if (account) {
+ post->account_mask = account;
+ } else {
+ post->amount = amt;
+ post = NULL; // an amount concludes this posting
+ }
+ }
}
}
}
@@ -218,14 +217,14 @@ void draft_t::parse_args(const value_t& args)
// A single account at the end of the line is the "from" account
if (tmpl->posts.size() > 1 &&
- tmpl->posts.back().account_mask && ! tmpl->posts.back().amount)
+ tmpl->posts.back().account_mask && ! tmpl->posts.back().amount)
tmpl->posts.back().from = true;
foreach (xact_template_t::post_template_t& post, tmpl->posts) {
if (post.from)
- has_only_to = false;
+ has_only_to = false;
else
- has_only_from = false;
+ has_only_from = false;
}
if (has_only_from) {
@@ -252,28 +251,28 @@ xact_t * draft_t::insert(journal_t& journal)
xacts_iterator xi(journal);
if (xact_t * xact = lookup_probable_account(tmpl->payee_mask.str(), xi).first) {
- DEBUG("derive.xact", "Found payee by lookup: transaction on line "
- << xact->pos->beg_line);
+ DEBUG("draft.xact", "Found payee by lookup: transaction on line "
+ << xact->pos->beg_line);
matching = xact;
} else {
for (xacts_list::reverse_iterator j = journal.xacts.rbegin();
- j != journal.xacts.rend();
- j++) {
+ j != journal.xacts.rend();
+ j++) {
if (tmpl->payee_mask.match((*j)->payee)) {
- matching = *j;
- DEBUG("derive.xact",
- "Found payee match: transaction on line " << (*j)->pos->beg_line);
- break;
+ matching = *j;
+ DEBUG("draft.xact",
+ "Found payee match: transaction on line " << (*j)->pos->beg_line);
+ break;
}
}
}
if (! tmpl->date) {
added->_date = CURRENT_DATE();
- DEBUG("derive.xact", "Setting date to current date");
+ DEBUG("draft.xact", "Setting date to current date");
} else {
added->_date = tmpl->date;
- DEBUG("derive.xact", "Setting date to template date: " << *tmpl->date);
+ DEBUG("draft.xact", "Setting date to template date: " << *tmpl->date);
}
added->set_state(item_t::UNCLEARED);
@@ -284,48 +283,48 @@ xact_t * draft_t::insert(journal_t& journal)
//added->note = matching->note;
#if defined(DEBUG_ON)
- DEBUG("derive.xact", "Setting payee from match: " << added->payee);
+ DEBUG("draft.xact", "Setting payee from match: " << added->payee);
//if (added->code)
- // DEBUG("derive.xact", "Setting code from match: " << *added->code);
+ // DEBUG("draft.xact", "Setting code from match: " << *added->code);
//if (added->note)
- // DEBUG("derive.xact", "Setting note from match: " << *added->note);
+ // DEBUG("draft.xact", "Setting note from match: " << *added->note);
#endif
} else {
added->payee = tmpl->payee_mask.str();
- DEBUG("derive.xact", "Setting payee from template: " << added->payee);
+ DEBUG("draft.xact", "Setting payee from template: " << added->payee);
}
if (tmpl->code) {
added->code = tmpl->code;
- DEBUG("derive.xact", "Now setting code from template: " << *added->code);
+ DEBUG("draft.xact", "Now setting code from template: " << *added->code);
}
if (tmpl->note) {
added->note = tmpl->note;
- DEBUG("derive.xact", "Now setting note from template: " << *added->note);
+ DEBUG("draft.xact", "Now setting note from template: " << *added->note);
}
if (tmpl->posts.empty()) {
if (matching) {
- DEBUG("derive.xact", "Template had no postings, copying from match");
+ DEBUG("draft.xact", "Template had no postings, copying from match");
foreach (post_t * post, matching->posts) {
- added->add_post(new post_t(*post));
- added->posts.back()->set_state(item_t::UNCLEARED);
+ added->add_post(new post_t(*post));
+ added->posts.back()->set_state(item_t::UNCLEARED);
}
} else {
throw_(std::runtime_error,
- _("No accounts, and no past transaction matching '%1'")
- << tmpl->payee_mask);
+ _("No accounts, and no past transaction matching '%1'")
+ << tmpl->payee_mask);
}
} else {
- DEBUG("derive.xact", "Template had postings");
+ DEBUG("draft.xact", "Template had postings");
bool any_post_has_amount = false;
foreach (xact_template_t::post_template_t& post, tmpl->posts) {
if (post.amount) {
- DEBUG("derive.xact", " and at least one has an amount specified");
- any_post_has_amount = true;
- break;
+ DEBUG("draft.xact", " and at least one has an amount specified");
+ any_post_has_amount = true;
+ break;
}
}
@@ -335,171 +334,171 @@ xact_t * draft_t::insert(journal_t& journal)
commodity_t * found_commodity = NULL;
if (matching) {
- if (post.account_mask) {
- DEBUG("derive.xact",
- "Looking for matching posting based on account mask");
-
- foreach (post_t * x, matching->posts) {
- if (post.account_mask->match(x->account->fullname())) {
- new_post.reset(new post_t(*x));
- DEBUG("derive.xact",
- "Founding posting from line " << x->pos->beg_line);
- break;
- }
- }
- } else {
- if (post.from) {
- for (posts_list::reverse_iterator j = matching->posts.rbegin();
- j != matching->posts.rend();
- j++) {
- if ((*j)->must_balance()) {
- new_post.reset(new post_t(**j));
- DEBUG("derive.xact",
- "Copied last real posting from matching");
- break;
- }
- }
- } else {
- for (posts_list::iterator j = matching->posts.begin();
- j != matching->posts.end();
- j++) {
- if ((*j)->must_balance()) {
- new_post.reset(new post_t(**j));
- DEBUG("derive.xact",
- "Copied first real posting from matching");
- break;
- }
- }
- }
- }
+ if (post.account_mask) {
+ DEBUG("draft.xact",
+ "Looking for matching posting based on account mask");
+
+ foreach (post_t * x, matching->posts) {
+ if (post.account_mask->match(x->account->fullname())) {
+ new_post.reset(new post_t(*x));
+ DEBUG("draft.xact",
+ "Founding posting from line " << x->pos->beg_line);
+ break;
+ }
+ }
+ } else {
+ if (post.from) {
+ for (posts_list::reverse_iterator j = matching->posts.rbegin();
+ j != matching->posts.rend();
+ j++) {
+ if ((*j)->must_balance()) {
+ new_post.reset(new post_t(**j));
+ DEBUG("draft.xact",
+ "Copied last real posting from matching");
+ break;
+ }
+ }
+ } else {
+ for (posts_list::iterator j = matching->posts.begin();
+ j != matching->posts.end();
+ j++) {
+ if ((*j)->must_balance()) {
+ new_post.reset(new post_t(**j));
+ DEBUG("draft.xact",
+ "Copied first real posting from matching");
+ break;
+ }
+ }
+ }
+ }
}
if (! new_post.get()) {
- new_post.reset(new post_t);
- DEBUG("derive.xact", "New posting was NULL, creating a blank one");
+ new_post.reset(new post_t);
+ DEBUG("draft.xact", "New posting was NULL, creating a blank one");
}
if (! new_post->account) {
- DEBUG("derive.xact", "New posting still needs an account");
+ DEBUG("draft.xact", "New posting still needs an account");
- if (post.account_mask) {
- DEBUG("derive.xact", "The template has an account mask");
+ if (post.account_mask) {
+ DEBUG("draft.xact", "The template has an account mask");
- account_t * acct = NULL;
- if (! acct) {
- acct = journal.find_account_re(post.account_mask->str());
+ account_t * acct = NULL;
+ if (! acct) {
+ acct = journal.find_account_re(post.account_mask->str());
#if defined(DEBUG_ON)
- if (acct)
- DEBUG("derive.xact", "Found account as a regular expression");
+ if (acct)
+ DEBUG("draft.xact", "Found account as a regular expression");
#endif
- }
- if (! acct) {
- acct = journal.find_account(post.account_mask->str());
+ }
+ if (! acct) {
+ acct = journal.find_account(post.account_mask->str());
#if defined(DEBUG_ON)
- if (acct)
- DEBUG("derive.xact", "Found (or created) account by name");
+ if (acct)
+ DEBUG("draft.xact", "Found (or created) account by name");
#endif
- }
-
- // Find out the default commodity to use by looking at the last
- // commodity used in that account
- for (xacts_list::reverse_iterator j = journal.xacts.rbegin();
- j != journal.xacts.rend();
- j++) {
- foreach (post_t * x, (*j)->posts) {
- if (x->account == acct && ! x->amount.is_null()) {
- new_post.reset(new post_t(*x));
- DEBUG("derive.xact",
- "Found account in journal postings, setting new posting");
- break;
- }
- }
- }
-
- new_post->account = acct;
- DEBUG("derive.xact",
- "Set new posting's account to: " << acct->fullname());
- } else {
- if (post.from) {
- new_post->account = journal.find_account(_("Liabilities:Unknown"));
- DEBUG("derive.xact",
- "Set new posting's account to: Liabilities:Unknown");
- } else {
- new_post->account = journal.find_account(_("Expenses:Unknown"));
- DEBUG("derive.xact",
- "Set new posting's account to: Expenses:Unknown");
- }
- }
+ }
+
+ // Find out the default commodity to use by looking at the last
+ // commodity used in that account
+ for (xacts_list::reverse_iterator j = journal.xacts.rbegin();
+ j != journal.xacts.rend();
+ j++) {
+ foreach (post_t * x, (*j)->posts) {
+ if (x->account == acct && ! x->amount.is_null()) {
+ new_post.reset(new post_t(*x));
+ DEBUG("draft.xact",
+ "Found account in journal postings, setting new posting");
+ break;
+ }
+ }
+ }
+
+ new_post->account = acct;
+ DEBUG("draft.xact",
+ "Set new posting's account to: " << acct->fullname());
+ } else {
+ if (post.from) {
+ new_post->account = journal.find_account(_("Liabilities:Unknown"));
+ DEBUG("draft.xact",
+ "Set new posting's account to: Liabilities:Unknown");
+ } else {
+ new_post->account = journal.find_account(_("Expenses:Unknown"));
+ DEBUG("draft.xact",
+ "Set new posting's account to: Expenses:Unknown");
+ }
+ }
}
assert(new_post->account);
if (new_post.get() && ! new_post->amount.is_null()) {
- found_commodity = &new_post->amount.commodity();
-
- if (any_post_has_amount) {
- new_post->amount = amount_t();
- DEBUG("derive.xact", "New posting has an amount, but we cleared it");
- } else {
- any_post_has_amount = true;
- DEBUG("derive.xact", "New posting has an amount, and we're using it");
- }
+ found_commodity = &new_post->amount.commodity();
+
+ if (any_post_has_amount) {
+ new_post->amount = amount_t();
+ DEBUG("draft.xact", "New posting has an amount, but we cleared it");
+ } else {
+ any_post_has_amount = true;
+ DEBUG("draft.xact", "New posting has an amount, and we're using it");
+ }
}
if (post.amount) {
- new_post->amount = *post.amount;
- DEBUG("derive.xact", "Copied over posting amount");
+ new_post->amount = *post.amount;
+ DEBUG("draft.xact", "Copied over posting amount");
- if (post.from) {
- new_post->amount.in_place_negate();
- DEBUG("derive.xact", "Negated new posting amount");
- }
+ if (post.from) {
+ new_post->amount.in_place_negate();
+ DEBUG("draft.xact", "Negated new posting amount");
+ }
}
if (post.cost) {
- if (post.cost->sign() < 0)
- throw parse_error(_("A posting's cost may not be negative"));
-
- post.cost->in_place_unround();
-
- if (*post.cost_operator == "@") {
- // 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 *= new_post->amount;
- post.cost->set_commodity(cost_commodity);
- }
- else if (new_post->amount.sign() < 0) {
- new_post->cost->in_place_negate();
- }
-
- new_post->cost = *post.cost;
- DEBUG("derive.xact", "Copied over posting cost");
+ if (post.cost->sign() < 0)
+ throw parse_error(_("A posting's cost may not be negative"));
+
+ post.cost->in_place_unround();
+
+ if (*post.cost_operator == "@") {
+ // 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 *= new_post->amount;
+ post.cost->set_commodity(cost_commodity);
+ }
+ else if (new_post->amount.sign() < 0) {
+ new_post->cost->in_place_negate();
+ }
+
+ new_post->cost = *post.cost;
+ DEBUG("draft.xact", "Copied over posting cost");
}
if (found_commodity &&
- ! new_post->amount.is_null() &&
- ! new_post->amount.has_commodity()) {
- new_post->amount.set_commodity(*found_commodity);
- DEBUG("derive.xact", "Set posting amount commodity to: "
- << new_post->amount.commodity());
-
- new_post->amount = new_post->amount.rounded();
- DEBUG("derive.xact",
- "Rounded posting amount to: " << new_post->amount);
+ ! new_post->amount.is_null() &&
+ ! new_post->amount.has_commodity()) {
+ new_post->amount.set_commodity(*found_commodity);
+ DEBUG("draft.xact", "Set posting amount commodity to: "
+ << new_post->amount.commodity());
+
+ new_post->amount = new_post->amount.rounded();
+ DEBUG("draft.xact",
+ "Rounded posting amount to: " << new_post->amount);
}
added->add_post(new_post.release());
added->posts.back()->account->add_post(added->posts.back());
added->posts.back()->set_state(item_t::UNCLEARED);
- DEBUG("derive.xact", "Added new posting to derived entry");
+ DEBUG("draft.xact", "Added new posting to derived entry");
}
}
if (! journal.add_xact(added.get()))
throw_(std::runtime_error,
- _("Failed to finalize derived transaction (check commodities)"));
+ _("Failed to finalize derived transaction (check commodities)"));
return added.release();
}