diff options
author | John Wiegley <johnw@newartisans.com> | 2009-02-04 03:34:37 -0400 |
---|---|---|
committer | John Wiegley <johnw@newartisans.com> | 2009-02-04 03:34:37 -0400 |
commit | 1cc33531ea7cf82e41b1ed49bf8ffc5afea084e4 (patch) | |
tree | 0ecd593b7ca164743bb9a4addf70ec76072288ce /src | |
parent | 28da097fc245340b316539c5533d470d82f19965 (diff) | |
download | fork-ledger-1cc33531ea7cf82e41b1ed49bf8ffc5afea084e4.tar.gz fork-ledger-1cc33531ea7cf82e41b1ed49bf8ffc5afea084e4.tar.bz2 fork-ledger-1cc33531ea7cf82e41b1ed49bf8ffc5afea084e4.zip |
Simplified the textual parser, and improved metadata support.
Diffstat (limited to 'src')
-rw-r--r-- | src/error.cc | 14 | ||||
-rw-r--r-- | src/error.h | 6 | ||||
-rw-r--r-- | src/item.cc | 30 | ||||
-rw-r--r-- | src/item.h | 8 | ||||
-rw-r--r-- | src/journal.h | 14 | ||||
-rw-r--r-- | src/session.cc | 2 | ||||
-rw-r--r-- | src/textual.cc | 883 | ||||
-rw-r--r-- | src/textual.h | 39 | ||||
-rw-r--r-- | src/xact.h | 9 |
9 files changed, 454 insertions, 551 deletions
diff --git a/src/error.cc b/src/error.cc index 68a2d844..dbda8c30 100644 --- a/src/error.cc +++ b/src/error.cc @@ -50,21 +50,21 @@ string file_context(const path& file, std::size_t line) return buf.str(); } -string line_context(const string& line, - istream_pos_type pos, - istream_pos_type end_pos) +string line_context(const string& line, + std::size_t pos, + std::size_t end_pos) { std::ostringstream buf; buf << " " << line << "\n"; - if (pos != istream_pos_type(0)) { + if (pos != 0) { buf << " "; - if (end_pos == istream_pos_type(0)) { - for (istream_pos_type i = 0; i < pos; i += 1) + if (end_pos == 0) { + for (std::size_t i = 0; i < pos; i += 1) buf << " "; buf << "^"; } else { - for (istream_pos_type i = 0; i < end_pos; i += 1) { + for (std::size_t i = 0; i < end_pos; i += 1) { if (i >= pos) buf << "^"; else diff --git a/src/error.h b/src/error.h index 4ff5a04c..568e509a 100644 --- a/src/error.h +++ b/src/error.h @@ -68,9 +68,9 @@ extern std::ostringstream _ctxt_buffer; string error_context(); string file_context(const path& file, std::size_t line); -string line_context(const string& line, - istream_pos_type pos = istream_pos_type(0), - istream_pos_type end_pos = istream_pos_type(0)); +string line_context(const string& line, + std::size_t pos = 0, + std::size_t end_pos = 0); #define DECLARE_EXCEPTION(name, kind) \ class name : public kind { \ diff --git a/src/item.cc b/src/item.cc index 02983ba1..27d28782 100644 --- a/src/item.cc +++ b/src/item.cc @@ -71,6 +71,21 @@ void item_t::set_tag(const string& tag, void item_t::parse_tags(const char * p) { + if (char * b = std::strchr(p, '[')) { + if (char * e = std::strchr(p, ']')) { + char buf[256]; + std::strncpy(buf, b + 1, e - b - 1); + buf[e - b - 1] = '\0'; + + if (char * p = std::strchr(buf, '=')) { + *p++ = '\0'; + _date_eff = parse_date(p); + } + if (buf[0]) + _date = parse_date(buf); + } + } + if (! std::strchr(p, ':')) return; @@ -99,6 +114,17 @@ void item_t::parse_tags(const char * p) } } +void item_t::append_note(const char * p) +{ + if (note) + *note += p; + else + note = p; + *note += '\n'; + + parse_tags(p); +} + namespace { value_t get_status(item_t& item) { return long(item.state()); @@ -325,9 +351,9 @@ string item_context(const item_t& item) out << "While balancing item from \"" << item.pathname.string() << "\""; - if (item.beg_line != (item.end_line - 1)) + if (item.beg_line != item.end_line) out << ", lines " << item.beg_line << "-" - << (item.end_line - 1) << ":\n"; + << item.end_line << ":\n"; else out << ", line " << item.beg_line << ":\n"; @@ -127,13 +127,7 @@ public: virtual void set_tag(const string& tag, const optional<string>& value = none); virtual void parse_tags(const char * p); - - virtual void append_note(const char * p) { - if (note) - *note += p; - else - note = p; - } + virtual void append_note(const char * p); virtual optional<date_t> actual_date() const { return _date; diff --git a/src/journal.h b/src/journal.h index f8f53ec4..ff060d76 100644 --- a/src/journal.h +++ b/src/journal.h @@ -113,7 +113,9 @@ public: TRACE_DTOR(journal_t::parser_t); } +#if defined(TEST_FOR_PARSER) virtual bool test(std::istream& in) const = 0; +#endif virtual std::size_t parse(std::istream& in, session_t& session, @@ -122,18 +124,6 @@ public: const path * original_file = NULL) = 0; }; - class binary_parser_t : public parser_t - { - public: - virtual bool test(std::istream& in) const; - - virtual std::size_t parse(std::istream& in, - session_t& session, - journal_t& journal, - account_t * master = NULL, - const path * original_file = NULL); - }; - bool valid() const; }; diff --git a/src/session.cc b/src/session.cc index 42e02c75..075324ec 100644 --- a/src/session.cc +++ b/src/session.cc @@ -148,7 +148,9 @@ std::size_t session_t::read_journal(journal_t& journal, master = journal.master; foreach (journal_t::parser_t& parser, parsers) +#if defined(TEST_FOR_PARSER) if (parser.test(in)) +#endif return parser.parse(in, *this, journal, master, &pathname); return 0; diff --git a/src/textual.cc b/src/textual.cc index bb1cbfac..f7a036ab 100644 --- a/src/textual.cc +++ b/src/textual.cc @@ -45,6 +45,66 @@ namespace ledger { +namespace { + class ptristream : public std::istream + { + class ptrinbuf : public std::streambuf + { + protected: + char * ptr; + std::size_t len; + + public: + ptrinbuf(char * _ptr, std::size_t _len) : ptr(_ptr), len(_len) { + setg(ptr, // beginning of putback area + ptr, // read position + ptr+len); // end position + } + + protected: + virtual int_type underflow() { + // is read position before end of buffer? + if (gptr() < egptr()) + return traits_type::to_int_type(*gptr()); + else + return EOF; + } + + virtual pos_type seekoff(off_type off, ios_base::seekdir way, + ios_base::openmode mode = + ios_base::in | ios_base::out) + { + switch (way) { + case std::ios::cur: + setg(ptr, gptr()+off, ptr+len); + break; + case std::ios::beg: + setg(ptr, ptr+off, ptr+len); + break; + case std::ios::end: + setg(ptr, egptr()+off, ptr+len); + break; + + default: + assert(false); + break; + } + return pos_type(gptr() - ptr); + } + }; + + protected: + ptrinbuf buf; + public: + ptristream(char * ptr, std::size_t len) + : std::istream(0), buf(ptr, len) { + rdbuf(&buf); + } + }; +} + +#if defined(TEST_FOR_PARSER) + bool textual_parser_t::test(std::istream& in) const { char buf[12]; @@ -66,6 +126,8 @@ bool textual_parser_t::test(std::istream& in) const return true; } +#endif // TEST_FOR_PARSER + std::size_t textual_parser_t::parse(std::istream& in, session_t& session, journal_t& journal, @@ -160,15 +222,9 @@ textual_parser_t::instance_t::instance_t if (! master) master = journal.master; - account_stack.push_front(master); pathname = journal.sources.back(); - linenum = 1; - beg_pos = in.tellg(); - beg_line = linenum; - count = 0; - errors = 0; } textual_parser_t::instance_t::~instance_t() @@ -188,14 +244,17 @@ void textual_parser_t::instance_t::parse() TRACE_START(instance_parse, 1, "Done parsing file '" << pathname.string() << "'"); - errors = 0; + if (! in.good() || in.eof()) + return; + + linenum = 0; + errors = 0; + count = 0; + curr_pos = in.tellg(); while (in.good() && ! in.eof()) { try { read_next_directive(); - - beg_pos = end_pos; - beg_line = linenum; } catch (const std::exception& err) { string current_context = error_context(); @@ -211,10 +270,10 @@ void textual_parser_t::instance_t::parse() foreach (instance_t * instance, instances) add_error_context("In file included from " << file_context(instance->pathname, - instance->linenum - 1)); + instance->linenum)); } add_error_context("While parsing file " - << file_context(pathname, linenum - 1)); + << file_context(pathname, linenum)); string context = error_context(); if (! context.empty()) @@ -231,29 +290,46 @@ void textual_parser_t::instance_t::parse() TRACE_STOP(instance_parse, 1); } -void textual_parser_t::instance_t::read_next_directive() +std::streamsize textual_parser_t::instance_t::read_line(char *& line) { - char * line; + assert(in.good()); + assert(! in.eof()); // no one should call us in that case + line_beg_pos = curr_pos; + in.getline(linebuf, MAX_LINE); - if (in.eof()) - return; + std::streamsize len = in.gcount(); - if (linenum == 1 && utf8::is_bom(linebuf)) - line = &linebuf[3]; - else - line = linebuf; + if (len > 0) { + if (linenum == 0 && utf8::is_bom(linebuf)) + line = &linebuf[3]; + else + line = linebuf; + + if (line[len - 1] == '\r') // strip Windows CRLF down to LF + line[--len] = '\0'; + + linenum++; - int len = std::strlen(line); - if (line[len - 1] == '\r') - line[--len] = '\0'; + curr_pos = line_beg_pos; + curr_pos += len; - end_pos = beg_pos; - end_pos += len + 1; - linenum++; + return len - 1; // LF is being silently dropped + } + return 0; +} + +void textual_parser_t::instance_t::read_next_directive() +{ + char * line; + std::streamsize len = read_line(line); + + if (len == 0 || line == NULL) + return; switch (line[0]) { case '\0': + assert(false); // shouldn't ever reach here break; case ' ': @@ -282,7 +358,7 @@ void textual_parser_t::instance_t::read_next_directive() case '7': case '8': case '9': - entry_directive(line); + entry_directive(line, len); break; case '=': // automated entry automated_entry_directive(line); @@ -476,72 +552,70 @@ void textual_parser_t::instance_t::automated_entry_directive(char * line) journal.add_entry_finalizer(auto_entry_finalizer.get()); } - auto_entry_t * ae = new auto_entry_t(skip_ws(line + 1)); - if (parse_xacts(in, account_stack.front(), *ae, "automated", - end_pos)) { - journal.auto_entries.push_back(ae); + istream_pos_type pos = curr_pos; + std::size_t lnum = linenum; + + std::auto_ptr<auto_entry_t> ae(new auto_entry_t(skip_ws(line + 1))); + + if (parse_xacts(account_stack.front(), *ae.get(), "automated")) { + journal.auto_entries.push_back(ae.get()); + ae->pathname = pathname; - ae->beg_pos = beg_pos; - ae->beg_line = beg_line; - ae->end_pos = end_pos; + ae->beg_pos = pos; + ae->beg_line = lnum; + ae->end_pos = curr_pos; ae->end_line = linenum; + + ae.release(); } } void textual_parser_t::instance_t::period_entry_directive(char * line) { - period_entry_t * pe = new period_entry_t(skip_ws(line + 1)); + std::auto_ptr<period_entry_t> pe(new period_entry_t(skip_ws(line + 1))); if (! pe->period) throw_(parse_error, "Parsing time period '" << line << "'"); - if (parse_xacts(in, account_stack.front(), *pe, - "period", end_pos)) { + istream_pos_type pos = curr_pos; + std::size_t lnum = linenum; + + if (parse_xacts(account_stack.front(), *pe.get(), "period")) { if (pe->finalize()) { - extend_entry_base(&journal, *pe, true); - journal.period_entries.push_back(pe); + extend_entry_base(&journal, *pe.get(), true); + + journal.period_entries.push_back(pe.get()); + pe->pathname = pathname; - pe->beg_pos = beg_pos; - pe->beg_line = beg_line; - pe->end_pos = end_pos; + pe->beg_pos = pos; + pe->beg_line = lnum; + pe->end_pos = curr_pos; pe->end_line = linenum; + + pe.release(); } else { throw parse_error("Period entry failed to balance"); } } } -void textual_parser_t::instance_t::entry_directive(char * line) +void textual_parser_t::instance_t::entry_directive(char * line, std::streamsize len) { - istream_pos_type pos = beg_pos; - TRACE_START(entries, 1, "Time spent handling entries:"); - if (entry_t * entry = parse_entry(in, line, account_stack.front(), pos)) { - // The entry pointer is unowned at the minute, and there is a - // possibility that add_entry ma throw an exception, which - // would cause us to leak without this guard. - std::auto_ptr<entry_t> entry_ptr(entry); - - entry->pathname = pathname; - entry->beg_pos = beg_pos; - entry->beg_line = beg_line; - entry->end_pos = pos; - entry->end_line = linenum; + if (entry_t * entry = parse_entry(line, len, account_stack.front())) { + std::auto_ptr<entry_t> manager(entry); if (journal.add_entry(entry)) { - entry_ptr.release(); // it's owned by the journal now + manager.release(); // it's owned by the journal now count++; } - // It's perfectly valid for the journal to reject the entry, - // which it will do if the entry has no substantive effect - // (for example, a checking entry, all of whose transactions - // have null amounts). + // It's perfectly valid for the journal to reject the entry, which it will + // do if the entry has no substantive effect (for example, a checking + // entry, all of whose transactions have null amounts). } else { throw parse_error("Failed to parse entry"); } - end_pos = pos; - TRACE_STOP(entries, 1); } @@ -669,81 +743,79 @@ void textual_parser_t::instance_t::general_directive(char * line) } } -xact_t * textual_parser_t::instance_t::parse_xact(char * line, - account_t * account, - entry_t * entry) +xact_t * textual_parser_t::instance_t::parse_xact(char * line, + std::streamsize len, + account_t * account, + entry_t * entry) { - std::istringstream in(line); + TRACE_START(xact_details, 1, "Time spent parsing transactions:"); + + std::auto_ptr<xact_t> xact(new xact_t); + + xact->entry = entry; // this could be NULL + xact->pathname = pathname; + xact->beg_pos = line_beg_pos; + xact->beg_line = linenum; - istream_pos_type beg = in.tellg(); - istream_pos_type end = beg; + static char buf[MAX_LINE + 1]; + std::strcpy(buf, line); + std::size_t beg = 0; - string err_desc; try { // Parse the state flag - char p = peek_next_nonws(in); - if (in.eof()) - return NULL; - - // The account will be determined later... - std::auto_ptr<xact_t> xact(new xact_t); - if (entry) - xact->entry = entry; - switch (p) { + assert(line); + assert(*line); + + char * p = skip_ws(line); + + switch (*p) { case '*': xact->set_state(item_t::CLEARED); - in.get(p); - p = peek_next_nonws(in); - DEBUG("textual.parse", - "line " << linenum << ": " << "Parsed the CLEARED flag"); + p = skip_ws(p + 1); + DEBUG("textual.parse", "line " << linenum << ": " + << "Parsed the CLEARED flag"); break; + case '!': xact->set_state(item_t::PENDING); - in.get(p); - p = peek_next_nonws(in); - DEBUG("textual.parse", - "line " << linenum << ": " << "Parsed the PENDING flag"); + p = skip_ws(p + 1); + DEBUG("textual.parse", "line " << linenum << ": " + << "Parsed the PENDING flag"); break; } - // Parse the account name + if (entry && + ((entry->_state == item_t::CLEARED && xact->state() != item_t::CLEARED) || + (entry->_state == item_t::PENDING && xact->state() == item_t::UNCLEARED))) + xact->set_state(entry->_state); - beg = in.tellg(); - end = beg; - while (! in.eof()) { - in.get(p); - if (in.eof() || (std::isspace(p) && - (p == '\t' || in.peek() == EOF || - std::isspace(in.peek())))) - break; - end += 1; - } + // Parse the account name - if (beg == end) - throw parse_error("No account was specified"); + if (! *p) + throw parse_error("Transaction has no account"); - char * b = &line[long(beg)]; - char * e = &line[long(end)]; + char * next = next_element(p, true); + char * e = p + std::strlen(p); - if ((*b == '[' && *(e - 1) == ']') || - (*b == '(' && *(e - 1) == ')')) { + if ((*p == '[' && *(e - 1) == ']') || (*p == '(' && *(e - 1) == ')')) { xact->add_flags(XACT_VIRTUAL); - DEBUG("textual.parse", - "line " << linenum << ": " << "Parsed a virtual account name"); + DEBUG("textual.parse", "line " << linenum << ": " + << "Parsed a virtual account name"); - if (*b == '[') { + if (*p == '[') { xact->add_flags(XACT_MUST_BALANCE); - DEBUG("textual.parse", - "line " << linenum << ": " << "Transaction must balance"); + DEBUG("textual.parse", "line " << linenum << ": " + << "Transaction must balance"); } - b++; e--; + p++; e--; } - string name(b, e - b); - DEBUG("textual.parse", "line " << linenum << ": " << - "Parsed account name " << name); + string name(p, e - p); + DEBUG("textual.parse", "line " << linenum << ": " + << "Parsed account name " << name); + if (account_aliases.size() > 0) { accounts_map::const_iterator i = account_aliases.find(name); if (i != account_aliases.end()) @@ -756,302 +828,229 @@ xact_t * textual_parser_t::instance_t::parse_xact(char * line, bool saw_amount = false; - if (in.good() && ! in.eof()) { - p = peek_next_nonws(in); - if (in.eof()) - goto finished; - if (p == ';') - goto parse_note; - if (p == '=' && entry) - goto parse_assign; - - beg = in.tellg(); + if (next && *next && (*next != ';' && *next != '=')) { saw_amount = true; - if (p != '(') { // indicates a value expression - xact->amount.parse(in, amount_t::PARSE_NO_REDUCE); - } - else { -#if defined(STORE_XACT_EXPRS) - xact->amount_expr = -#endif - parse_amount_expr(in, xact->amount, xact.get(), - static_cast<uint_least8_t>(expr_t::PARSE_NO_REDUCE) | - static_cast<uint_least8_t>(expr_t::PARSE_NO_ASSIGN)); + beg = next - line; + ptristream stream(next, len - beg); -#if defined(STORE_XACT_EXPRS) - // We don't need to store the actual expression that resulted in the - // amount if it's constant - if (xact->amount_expr) { - if (xact->amount_expr->is_constant()) - xact->amount_expr = expr_t(); - - end = in.tellg(); - xact->amount_expr->set_text(string(line, long(beg), long(end - beg))); - } -#endif // STORE_XACT_EXPRS - } + if (*next != '(') // indicates a value expression + xact->amount.parse(stream, amount_t::PARSE_NO_REDUCE); + else + parse_amount_expr(stream, xact->amount, xact.get(), + static_cast<uint_least8_t>(expr_t::PARSE_NO_REDUCE) | + static_cast<uint_least8_t>(expr_t::PARSE_NO_ASSIGN)); - if (! xact->amount.is_null()) { + if (! xact->amount.is_null()) xact->amount.reduce(); - DEBUG("textual.parse", "line " << linenum << ": " << - "Reduced amount is " << xact->amount); - } - } - // Parse the optional cost (@ PER-UNIT-COST, @@ TOTAL-COST) - - if (in.good() && ! in.eof()) { - p = peek_next_nonws(in); - if (! in.eof() && p == '@') { - if (! saw_amount) - throw parse_error("Transaction cannot have a cost without an amount"); - - DEBUG("textual.parse", "line " << linenum << ": " << - "Found a price indicator"); - bool per_unit = true; - - in.get(p); - if (in.peek() == '@') { - in.get(p); - - per_unit = false; - DEBUG("textual.parse", "line " << linenum << ": " << - "And it's for a total price"); - } + DEBUG("textual.parse", "line " << linenum << ": " + << "xact amount = " << xact->amount); - p = peek_next_nonws(in); - if (in.good() && ! in.eof()) { - xact->cost = amount_t(); + if (stream.eof()) { + next = NULL; + } else { + next = skip_ws(next + stream.tellg()); - beg = in.tellg(); + // Parse the optional cost (@ PER-UNIT-COST, @@ TOTAL-COST) - if (p != '(') { // indicates a value expression - xact->cost->parse(in, amount_t::PARSE_NO_MIGRATE); - } else { -#if defined(STORE_XACT_EXPRS) - xact->cost_expr = -#endif - parse_amount_expr(in, *xact->cost, xact.get(), - static_cast<uint_least8_t>(expr_t::PARSE_NO_MIGRATE) | - static_cast<uint_least8_t>(expr_t::PARSE_NO_ASSIGN)); + if (*next == '@') { + DEBUG("textual.parse", "line " << linenum << ": " + << "Found a price indicator"); -#if defined(STORE_XACT_EXPRS) - if (xact->cost_expr) { - end = in.tellg(); - if (per_unit) - xact->cost_expr->set_text(string("@") + - string(line, long(beg), long(end - beg))); - else - xact->cost_expr->set_text(string("@@") + - string(line, long(beg), long(end - beg))); - } -#endif // STORE_XACT_EXPRS + bool per_unit = true; + + if (*++next == '@') { + per_unit = false; + DEBUG("textual.parse", "line " << linenum << ": " + << "And it's for a total price"); } - if (xact->cost->sign() < 0) - throw parse_error("A transaction's cost may not be negative"); + beg = ++next - line; - amount_t per_unit_cost(*xact->cost); - if (per_unit) - *xact->cost *= xact->amount; - else - per_unit_cost /= xact->amount; - - commodity_t::exchange(xact->amount.commodity(), - per_unit_cost, datetime_t(*xact->date())); - - DEBUG("textual.parse", "line " << linenum << ": " << - "Total cost is " << *xact->cost); - DEBUG("textual.parse", "line " << linenum << ": " << - "Per-unit cost is " << per_unit_cost); - DEBUG("textual.parse", "line " << linenum << ": " << - "Annotated amount is " << xact->amount); - } else { - throw_(parse_error, "Expected a cost amount"); + p = skip_ws(next); + if (*p) { + xact->cost = amount_t(); + + beg = p - line; + ptristream cstream(p, len - beg); + + if (*p != '(') // indicates a value expression + xact->cost->parse(cstream, amount_t::PARSE_NO_MIGRATE); + else + parse_amount_expr(cstream, *xact->cost, xact.get(), + static_cast<uint_least8_t>(expr_t::PARSE_NO_MIGRATE) | + static_cast<uint_least8_t>(expr_t::PARSE_NO_ASSIGN)); + + if (xact->cost->sign() < 0) + throw parse_error("A transaction's cost may not be negative"); + + amount_t per_unit_cost(*xact->cost); + if (per_unit) + *xact->cost *= xact->amount; + else + per_unit_cost /= xact->amount; + + commodity_t::exchange(xact->amount.commodity(), + per_unit_cost, datetime_t(*xact->date())); + + DEBUG("textual.parse", "line " << linenum << ": " + << "Total cost is " << *xact->cost); + DEBUG("textual.parse", "line " << linenum << ": " + << "Per-unit cost is " << per_unit_cost); + DEBUG("textual.parse", "line " << linenum << ": " + << "Annotated amount is " << xact->amount); + + if (cstream.eof()) + next = NULL; + else + next = skip_ws(p + cstream.tellg()); + } else { + throw parse_error("Expected a cost amount"); + } } } } - parse_assign: - if (entry != NULL) { - // Parse the optional assigned (= AMOUNT) + // Parse the optional balance assignment - if (in.good() && ! in.eof()) { - p = peek_next_nonws(in); - if (! in.eof() && p == '=') { - in.get(p); + if (entry && next && *next == '=') { + DEBUG("textual.parse", "line " << linenum << ": " + << "Found a balance assignment indicator"); - DEBUG("textual.parse", "line " << linenum << ": " << - "Found a balance assignment indicator"); + beg = ++next - line; - p = peek_next_nonws(in); - if (in.good() && ! in.eof()) { - xact->assigned_amount = amount_t(); + p = skip_ws(next); + if (*p) { + xact->assigned_amount = amount_t(); - beg = in.tellg(); + beg = p - line; + ptristream stream(p, len - beg); - if (p != '(') { // indicates a value expression - xact->assigned_amount->parse(in, amount_t::PARSE_NO_MIGRATE); - } else { -#if defined(STORE_XACT_EXPRS) - xact->assigned_amount_expr = -#endif - parse_amount_expr(in, *xact->assigned_amount, xact.get(), - static_cast<uint_least8_t>(expr_t::PARSE_NO_MIGRATE)); + if (*p != '(') // indicates a value expression + xact->assigned_amount->parse(stream, amount_t::PARSE_NO_MIGRATE); + else + parse_amount_expr(stream, *xact->assigned_amount, xact.get(), + static_cast<uint_least8_t>(expr_t::PARSE_NO_MIGRATE)); -#if defined(STORE_XACT_EXPRS) - if (xact->assigned_amount_expr) { - end = in.tellg(); - xact->assigned_amount_expr->set_text - (string("=") + string(line, long(beg), long(end - beg))); - } -#endif // STORE_XACT_EXPRS - } + if (xact->assigned_amount->is_null()) + throw parse_error("An assigned balance must evaluate to a constant value"); - if (xact->assigned_amount->is_null()) - throw parse_error - ("An assigned balance must evaluate to a constant value"); + DEBUG("textual.parse", "line " << linenum << ": " + << "XACT assign: parsed amt = " << *xact->assigned_amount); - DEBUG("textual.parse", "line " << linenum << ": " << - "XACT assign: parsed amt = " << *xact->assigned_amount); + account_t::xdata_t& xdata(xact->account->xdata()); + amount_t& amt(*xact->assigned_amount); - account_t::xdata_t& xdata(xact->account->xdata()); - amount_t& amt(*xact->assigned_amount); + DEBUG("xact.assign", "line " << linenum << ": " + "account balance = " << xdata.value.strip_annotations()); + DEBUG("xact.assign", "line " << linenum << ": " + "xact amount = " << amt.strip_annotations()); - DEBUG("xact.assign", - "account balance = " << xdata.value.strip_annotations()); - DEBUG("xact.assign", - "xact amount = " << amt.strip_annotations()); + amount_t diff; - amount_t diff; - if (xdata.value.is_amount()) { - diff = amt - xdata.value.as_amount(); - } - else if (xdata.value.is_balance()) { - if (optional<amount_t> comm_bal = - xdata.value.as_balance().commodity_amount(amt.commodity())) - diff = amt - *comm_bal; - else - diff = amt; - } - else if (xdata.value.is_balance_pair()) { - if (optional<amount_t> comm_bal = - xdata.value.as_balance_pair().commodity_amount(amt.commodity())) - diff = amt - *comm_bal; - else - diff = amt; - } - else { - diff = amt; - } + switch (xdata.value.type()) { + case value_t::AMOUNT: + diff = amt - xdata.value.as_amount(); + break; - DEBUG("xact.assign", "diff = " << diff.strip_annotations()); - DEBUG("textual.parse", "line " << linenum << ": " << - "XACT assign: diff = " << diff.strip_annotations()); + case value_t::BALANCE: + if (optional<amount_t> comm_bal = + xdata.value.as_balance().commodity_amount(amt.commodity())) + diff = amt - *comm_bal; + else + diff = amt; + break; + case value_t::BALANCE_PAIR: + if (optional<amount_t> comm_bal = + xdata.value.as_balance_pair().commodity_amount(amt.commodity())) + diff = amt - *comm_bal; + else + diff = amt; + break; + + default: + diff = amt; + break; + } + + DEBUG("xact.assign", "line " << linenum << ": " + << "diff = " << diff.strip_annotations()); + DEBUG("textual.parse", "line " << linenum << ": " + << "XACT assign: diff = " << diff.strip_annotations()); + + if (! diff.is_zero()) { + if (! xact->amount.is_null()) { + diff -= xact->amount; if (! diff.is_zero()) { - if (! xact->amount.is_null()) { - diff -= xact->amount; - if (! diff.is_zero()) { - xact_t * temp = new xact_t(xact->account, diff, - ITEM_GENERATED | XACT_CALCULATED); - entry->add_xact(temp); - - DEBUG("textual.parse", "line " << linenum << ": " << - "Created balancing transaction"); - } - } else { - xact->amount = diff; - DEBUG("textual.parse", "line " << linenum << ": " << - "Overwrite null transaction"); - } + xact_t * temp = new xact_t(xact->account, diff, + ITEM_GENERATED | XACT_CALCULATED); + entry->add_xact(temp); + + DEBUG("textual.parse", "line " << linenum << ": " + << "Created balancing transaction"); } } else { - throw_(parse_error, "Expected an assigned balance amount"); + xact->amount = diff; + DEBUG("textual.parse", "line " << linenum << ": " + << "Overwrite null transaction"); } } + + if (stream.eof()) + next = NULL; + else + next = skip_ws(p + stream.tellg()); + } else { + throw parse_error("Expected an assigned balance amount"); } } // Parse the optional note -parse_note: - if (in.good() && ! in.eof()) { - p = peek_next_nonws(in); - if (! in.eof()) { - if (p == ';') { - in.get(p); - if (! in.eof()) { - xact->note = &line[long(in.tellg())]; - DEBUG("textual.parse", "line " << linenum << ": " << - "Parsed a note '" << *xact->note << "'"); - - if (char * b = std::strchr(xact->note->c_str(), '[')) - if (char * e = std::strchr(xact->note->c_str(), ']')) { - char buf[256]; - std::strncpy(buf, b + 1, e - b - 1); - buf[e - b - 1] = '\0'; - - DEBUG("textual.parse", "line " << linenum << ": " << - "Parsed a transaction date " << buf); - - if (char * p = std::strchr(buf, '=')) { - *p++ = '\0'; - xact->_date_eff = parse_date(p); - } - if (buf[0]) - xact->_date = parse_date(buf); - } - } - } else { - beg = in.tellg(); - throw_(parse_error, "Unexpected char '" << p - << "' (Note: inline math requires parentheses)"); - } - } + if (next && *next == ';') { + xact->append_note(++next); + next = line + len; + DEBUG("textual.parse", "line " << linenum << ": " + << "Parsed a transaction note"); } - finished: + // There should be nothing more to read + + if (next && *next) + throw_(parse_error, "Unexpected char '" << *next + << "' (Note: inline math requires parentheses)"); + + xact->end_pos = curr_pos; + xact->end_line = linenum; + + TRACE_STOP(xact_details, 1); + return xact.release(); } catch (const std::exception& err) { add_error_context("While parsing transaction:"); - add_error_context(line_context(line, beg, in.tellg())); + add_error_context(line_context(buf, beg, len)); throw; } } -bool textual_parser_t::instance_t::parse_xacts(std::istream& in, - account_t * account, +bool textual_parser_t::instance_t::parse_xacts(account_t * account, entry_base_t& entry, - const string& kind, - istream_pos_type beg_pos) + const string& kind) { TRACE_START(entry_xacts, 1, "Time spent parsing transactions:"); - static char line[MAX_LINE + 1]; - bool added = false; + bool added = false; - while (! in.eof() && (in.peek() == ' ' || in.peek() == '\t')) { - in.getline(line, MAX_LINE); - if (in.eof()) - break; - - int len = std::strlen(line); - if (line[len - 1] == '\r') - line[--len] = '\0'; + while (peek_whitespace_line()) { + char * line; + std::streamsize len = read_line(line); + assert(len > 0); - beg_pos += len + 1; - linenum++; - - if (line[0] == ' ' || line[0] == '\t') { - char * p = skip_ws(line); - if (! *p) - break; - } - if (xact_t * xact = parse_xact(line, account, NULL)) { + if (xact_t * xact = parse_xact(line, len, account, NULL)) { entry.add_xact(xact); added = true; } @@ -1062,15 +1061,18 @@ bool textual_parser_t::instance_t::parse_xacts(std::istream& in, return added; } -entry_t * textual_parser_t::instance_t::parse_entry(std::istream& in, - char * line, - account_t * master, - istream_pos_type& pos) +entry_t * textual_parser_t::instance_t::parse_entry(char * line, + std::streamsize len, + account_t * account) { - TRACE_START(entry_text, 1, "Time spent preparing entry text:"); + TRACE_START(entry_text, 1, "Time spent parsing entry text:"); std::auto_ptr<entry_t> curr(new entry_t); + curr->pathname = pathname; + curr->beg_pos = line_beg_pos; + curr->beg_line = linenum; + // Parse the date char * next = next_element(line); @@ -1109,7 +1111,17 @@ entry_t * textual_parser_t::instance_t::parse_entry(std::istream& in, // Parse the description text - curr->payee = next ? next : "<Unspecified payee>"; + if (next && *next) { + curr->payee = next; + next = next_element(next, true); + } else { + curr->payee = "<Unspecified payee>"; + } + + // Parse the entry note + + if (next && *next == ';') + curr->append_note(next); TRACE_STOP(entry_text, 1); @@ -1117,62 +1129,37 @@ entry_t * textual_parser_t::instance_t::parse_entry(std::istream& in, TRACE_START(entry_details, 1, "Time spent parsing entry details:"); - istream_pos_type end_pos; - std::size_t beg_line = linenum; - xact_t * last_xact = NULL; + xact_t * last_xact = NULL; - while (! in.eof() && (in.peek() == ' ' || in.peek() == '\t')) { - istream_pos_type beg_pos = in.tellg(); + while (peek_whitespace_line()) { + len = read_line(line); - line[0] = '\0'; - in.getline(line, MAX_LINE); - if (in.eof() && line[0] == '\0') - break; - - int len = std::strlen(line); - if (line[len - 1] == '\r') - line[--len] = '\0'; - - end_pos = beg_pos; - end_pos += len + 1; - linenum++; - - const char * p = skip_ws(line); + char * p = skip_ws(line); if (! *p) - break; + throw parse_error("Line contains only whitespace"); if (*p == ';') { - // This is an entry note, and possibly a metadata info tag - if (last_xact) { - last_xact->append_note(p + 1); - last_xact->append_note("\n"); - last_xact->parse_tags(p + 1); - } else { - curr->append_note(p + 1); - curr->append_note("\n"); - curr->parse_tags(p + 1); - } - } - else if (xact_t * xact = parse_xact(line, master, curr.get())) { - if ((state == item_t::CLEARED && xact->state() != item_t::CLEARED) || - (state == item_t::PENDING && xact->state() == item_t::UNCLEARED)) - xact->set_state(state); - - xact->beg_pos = beg_pos; - xact->beg_line = beg_line; - xact->end_pos = end_pos; - xact->end_line = linenum; - - pos = end_pos; + item_t * item; + if (last_xact) + item = last_xact; + else + item = curr.get(); + // This is a trailing note, and possibly a metadata info tag + item->append_note(p + 1); + item->end_pos = curr_pos; + item->end_line++; + } + else if (xact_t * xact = parse_xact(p, len - (p - line), account, + curr.get())) { curr->add_xact(xact); last_xact = xact; } - - if (in.eof()) - break; } + curr->end_pos = curr_pos; + curr->end_line = linenum; + TRACE_STOP(entry_details, 1); return curr.release(); @@ -1183,94 +1170,4 @@ expr_t::ptr_op_t textual_parser_t::instance_t::lookup(const string& name) return session.lookup(name); } -void write_textual_journal(journal_t& journal, - const path& pathname, - xact_handler_ptr formatter, - const string& write_hdr_format, - std::ostream& out) -{ - std::size_t index = 0; - path found; - - // jww (2009-01-29): This function currently doesn't work - - if (pathname.empty()) { - if (! journal.sources.empty()) - found = *journal.sources.begin(); - } else { -#ifdef HAVE_REALPATH - char buf1[PATH_MAX]; - char buf2[PATH_MAX]; - - ::realpath(pathname.string().c_str(), buf1); - - foreach (const path& path, journal.sources) { - ::realpath(path.string().c_str(), buf2); - if (std::strcmp(buf1, buf2) == 0) { - found = path; - break; - } - index++; - } -#else - foreach (const path& path, journal.sources) { - if (pathname == path) { - found = path; - break; - } - index++; - } -#endif - } - - if (found.empty()) - throw_(std::runtime_error, - "Journal does not refer to file '" << pathname << "'"); - - entries_list::iterator el = journal.entries.begin(); - auto_entries_list::iterator al = journal.auto_entries.begin(); - period_entries_list::iterator pl = journal.period_entries.begin(); - - istream_pos_type pos = 0; - - format_t hdr_fmt(write_hdr_format); - boost::filesystem::ifstream in(found); - - while (! in.eof()) { - entry_base_t * base = NULL; - if (el != journal.entries.end() && pos == (*el)->beg_pos) { - hdr_fmt.format(out, **el); - base = *el++; - } - else if (al != journal.auto_entries.end() && pos == (*al)->beg_pos) { - out << "= " << (*al)->predicate.predicate.text() << '\n'; - base = *al++; - } - else if (pl != journal.period_entries.end() && pos == (*pl)->beg_pos) { - out << "~ " << (*pl)->period_string << '\n'; - base = *pl++; - } - - char c; - if (base) { - foreach (xact_t * xact, base->xacts) { - if (! xact->has_flags(XACT_AUTO)) { - xact->xdata().add_flags(XACT_EXT_TO_DISPLAY); - (*formatter)(*xact); - } - } - formatter->flush(); - - while (pos < base->end_pos) { - in.get(c); - pos = in.tellg(); // pos++; - } - } else { - in.get(c); - pos = in.tellg(); // pos++; - out.put(c); - } - } -} - } // namespace ledger diff --git a/src/textual.h b/src/textual.h index 244041fb..b6f67157 100644 --- a/src/textual.h +++ b/src/textual.h @@ -65,7 +65,9 @@ class time_log_t; class textual_parser_t : public journal_t::parser_t { public: +#if defined(TEST_FOR_PARSER) virtual bool test(std::istream& in) const; +#endif virtual std::size_t parse(std::istream& in, session_t& session, @@ -93,15 +95,13 @@ protected: accounts_map account_aliases; path pathname; + char linebuf[MAX_LINE + 1]; std::size_t linenum; - istream_pos_type beg_pos; - std::size_t beg_line; - istream_pos_type end_pos; + istream_pos_type line_beg_pos; + istream_pos_type curr_pos; std::size_t count; std::size_t errors; - char linebuf[MAX_LINE + 1]; - scoped_ptr<auto_entry_finalizer_t> auto_entry_finalizer; instance_t(std::list<account_t *>& _account_stack, @@ -118,6 +118,11 @@ protected: ~instance_t(); void parse(); + std::streamsize read_line(char *& line); + bool peek_whitespace_line() { + return (in.good() && ! in.eof() && + (in.peek() == ' ' || in.peek() == '\t')); + } void read_next_directive(); #if defined(TIMELOG_SUPPORT) @@ -134,7 +139,7 @@ protected: void option_directive(char * line); void automated_entry_directive(char * line); void period_entry_directive(char * line); - void entry_directive(char * line); + void entry_directive(char * line, std::streamsize len); void include_directive(char * line); void account_directive(char * line); void end_directive(char * line); @@ -142,20 +147,18 @@ protected: void define_directive(char * line); void general_directive(char * line); - xact_t * parse_xact(char * line, - account_t * account, - entry_t * entry); + xact_t * parse_xact(char * line, + std::streamsize len, + account_t * account, + entry_t * entry); - bool parse_xacts(std::istream& in, - account_t * account, - entry_base_t& entry, - const string& kind, - istream_pos_type beg_pos); + bool parse_xacts(account_t * account, + entry_base_t& entry, + const string& kind); - entry_t * parse_entry(std::istream& in, - char * line, - account_t * master, - istream_pos_type& pos); + entry_t * parse_entry(char * line, + std::streamsize len, + account_t * account); virtual expr_t::ptr_op_t lookup(const string& name); }; @@ -73,17 +73,8 @@ public: account_t * account; amount_t amount; // can be null until finalization -#if defined(STORE_XACT_EXPRS) - optional<expr_t> amount_expr; -#endif optional<amount_t> cost; -#if defined(STORE_XACT_EXPRS) - optional<expr_t> cost_expr; -#endif optional<amount_t> assigned_amount; -#if defined(STORE_XACT_EXPRS) - optional<expr_t> assigned_amount_expr; -#endif xact_t(account_t * _account = NULL, flags_t _flags = ITEM_NORMAL) |