summaryrefslogtreecommitdiff
path: root/gnucash.cc
diff options
context:
space:
mode:
Diffstat (limited to 'gnucash.cc')
-rw-r--r--gnucash.cc341
1 files changed, 145 insertions, 196 deletions
diff --git a/gnucash.cc b/gnucash.cc
index 970fba3b..b988e0d7 100644
--- a/gnucash.cc
+++ b/gnucash.cc
@@ -1,217 +1,160 @@
-#include "gnucash.h"
-#include "journal.h"
-#include "format.h"
-#include "error.h"
-#include "acconf.h"
-
-#include <iostream>
-#include <sstream>
-#include <cstring>
-
-extern "C" {
-#if defined(HAVE_EXPAT)
-#include <expat.h> // expat XML parser
-#elif defined(HAVE_XMLPARSE)
-#include <xmlparse.h> // expat XML parser
+#ifdef USE_PCH
+#include "pch.h"
#else
-#error "No XML parser library defined."
+#include "gnucash.h"
#endif
-}
namespace ledger {
-typedef std::map<const std::string, account_t *> accounts_map;
-typedef std::pair<const std::string, account_t *> accounts_pair;
-
-typedef std::map<account_t *, commodity_t *> account_comm_map;
-typedef std::pair<account_t *, commodity_t *> account_comm_pair;
-
-static journal_t * curr_journal;
-static account_t * master_account;
-static account_t * curr_account;
-static std::string curr_account_id;
-static entry_t * curr_entry;
-static commodity_t * entry_comm;
-static commodity_t * curr_comm;
-static amount_t curr_value;
-static amount_t curr_quant;
-static XML_Parser current_parser;
-static accounts_map accounts_by_id;
-static account_comm_map account_comms;
-static unsigned int count;
-static std::string have_error;
-
-static std::istream * instreamp;
-static unsigned int offset;
-static XML_Parser parser;
-static std::string path;
-static unsigned int src_idx;
-static istream_pos_type beg_pos;
-static unsigned long beg_line;
-
-static transaction_t::state_t curr_state;
-
-static enum action_t {
- NO_ACTION,
- ACCOUNT_NAME,
- ACCOUNT_ID,
- ACCOUNT_PARENT,
- COMM_SYM,
- COMM_NAME,
- COMM_PREC,
- ENTRY_NUM,
- ALMOST_ENTRY_DATE,
- ENTRY_DATE,
- ENTRY_DESC,
- XACT_STATE,
- XACT_AMOUNT,
- XACT_VALUE,
- XACT_QUANTITY,
- XACT_ACCOUNT,
- XACT_NOTE
-} action;
-
-static void startElement(void *userData, const char *name, const char **atts)
+void startElement(void *userData, const char *name, const char **atts)
{
+ gnucash_parser_t * parser = static_cast<gnucash_parser_t *>(userData);
+
if (std::strcmp(name, "gnc:account") == 0) {
- curr_account = new account_t(master_account);
+ parser->curr_account = new account_t(parser->master_account);
}
else if (std::strcmp(name, "act:name") == 0)
- action = ACCOUNT_NAME;
+ parser->action = gnucash_parser_t::ACCOUNT_NAME;
else if (std::strcmp(name, "act:id") == 0)
- action = ACCOUNT_ID;
+ parser->action = gnucash_parser_t::ACCOUNT_ID;
else if (std::strcmp(name, "act:parent") == 0)
- action = ACCOUNT_PARENT;
+ parser->action = gnucash_parser_t::ACCOUNT_PARENT;
else if (std::strcmp(name, "gnc:commodity") == 0) {
- assert(! curr_comm);
+ assert(! parser->curr_comm);
#if 0
// jww (2006-03-02): !!!
- curr_comm = new commodity_t("");
+ parser->curr_comm = new commodity_t("");
#endif
}
else if (std::strcmp(name, "cmdty:id") == 0)
- action = COMM_SYM;
+ parser->action = gnucash_parser_t::COMM_SYM;
else if (std::strcmp(name, "cmdty:name") == 0)
- action = COMM_NAME;
+ parser->action = gnucash_parser_t::COMM_NAME;
else if (std::strcmp(name, "cmdty:fraction") == 0)
- action = COMM_PREC;
+ parser->action = gnucash_parser_t::COMM_PREC;
else if (std::strcmp(name, "gnc:transaction") == 0) {
- assert(! curr_entry);
- curr_entry = new entry_t;
+ assert(! parser->curr_entry);
+ parser->curr_entry = new entry_t;
}
else if (std::strcmp(name, "trn:num") == 0)
- action = ENTRY_NUM;
+ parser->action = gnucash_parser_t::ENTRY_NUM;
else if (std::strcmp(name, "trn:date-posted") == 0)
- action = ALMOST_ENTRY_DATE;
- else if (action == ALMOST_ENTRY_DATE && std::strcmp(name, "ts:date") == 0)
- action = ENTRY_DATE;
+ parser->action = gnucash_parser_t::ALMOST_ENTRY_DATE;
+ else if (parser->action == gnucash_parser_t::ALMOST_ENTRY_DATE &&
+ std::strcmp(name, "ts:date") == 0)
+ parser->action = gnucash_parser_t::ENTRY_DATE;
else if (std::strcmp(name, "trn:description") == 0)
- action = ENTRY_DESC;
+ parser->action = gnucash_parser_t::ENTRY_DESC;
else if (std::strcmp(name, "trn:split") == 0) {
- assert(curr_entry);
- curr_entry->add_transaction(new transaction_t(curr_account));
+ assert(parser->curr_entry);
+ parser->curr_entry->add_transaction(new transaction_t(parser->curr_account));
}
else if (std::strcmp(name, "split:reconciled-state") == 0)
- action = XACT_STATE;
+ parser->action = gnucash_parser_t::XACT_STATE;
else if (std::strcmp(name, "split:amount") == 0)
- action = XACT_AMOUNT;
+ parser->action = gnucash_parser_t::XACT_AMOUNT;
else if (std::strcmp(name, "split:value") == 0)
- action = XACT_VALUE;
+ parser->action = gnucash_parser_t::XACT_VALUE;
else if (std::strcmp(name, "split:quantity") == 0)
- action = XACT_QUANTITY;
+ parser->action = gnucash_parser_t::XACT_QUANTITY;
else if (std::strcmp(name, "split:account") == 0)
- action = XACT_ACCOUNT;
+ parser->action = gnucash_parser_t::XACT_ACCOUNT;
else if (std::strcmp(name, "split:memo") == 0)
- action = XACT_NOTE;
+ parser->action = gnucash_parser_t::XACT_NOTE;
}
-static void endElement(void *userData, const char *name)
+void endElement(void *userData, const char *name)
{
+ gnucash_parser_t * parser = static_cast<gnucash_parser_t *>(userData);
+
if (std::strcmp(name, "gnc:account") == 0) {
- assert(curr_account);
- if (curr_account->parent == master_account)
- curr_journal->add_account(curr_account);
- accounts_by_id.insert(accounts_pair(curr_account_id, curr_account));
- curr_account = NULL;
+ assert(parser->curr_account);
+ if (parser->curr_account->parent == parser->master_account)
+ parser->curr_journal->add_account(parser->curr_account);
+ parser->accounts_by_id.insert(accounts_pair(parser->curr_account_id,
+ parser->curr_account));
+ parser->curr_account = NULL;
}
else if (std::strcmp(name, "gnc:commodity") == 0) {
- assert(curr_comm);
+ assert(parser->curr_comm);
#if 0
// jww (2006-03-02): !!!
- commodity_t::add_commodity(curr_comm);
+ commodity_t::add_commodity(parser->curr_comm);
#endif
- curr_comm = NULL;
+ parser->curr_comm = NULL;
}
else if (std::strcmp(name, "gnc:transaction") == 0) {
- assert(curr_entry);
+ assert(parser->curr_entry);
// Add the new entry (what gnucash calls a 'transaction') to the
// journal
- if (! curr_journal->add_entry(curr_entry)) {
- print_entry(std::cerr, *curr_entry);
- have_error = "The above entry does not balance";
- delete curr_entry;
+ if (! parser->curr_journal->add_entry(parser->curr_entry)) {
+ print_entry(std::cerr, *parser->curr_entry);
+ parser->have_error = "The above entry does not balance";
+ delete parser->curr_entry;
} else {
- curr_entry->src_idx = src_idx;
- curr_entry->beg_pos = beg_pos;
- curr_entry->beg_line = beg_line;
- curr_entry->end_pos = instreamp->tellg();
- curr_entry->end_line = XML_GetCurrentLineNumber(parser) - offset;
- count++;
+ parser->curr_entry->src_idx = parser->src_idx;
+ parser->curr_entry->beg_pos = parser->beg_pos;
+ parser->curr_entry->beg_line = parser->beg_line;
+ parser->curr_entry->end_pos = parser->instreamp->tellg();
+ parser->curr_entry->end_line =
+ XML_GetCurrentLineNumber(parser->expat_parser) - parser->offset;
+ parser->count++;
}
// Clear the relevant variables for the next run
- curr_entry = NULL;
- entry_comm = NULL;
+ parser->curr_entry = NULL;
+ parser->entry_comm = NULL;
}
else if (std::strcmp(name, "trn:split") == 0) {
- transaction_t * xact = curr_entry->transactions.back();
+ transaction_t * xact = parser->curr_entry->transactions.back();
// Identify the commodity to use for the value of this
// transaction. The quantity indicates how many times that value
// the transaction is worth.
amount_t value;
commodity_t * default_commodity = NULL;
- if (entry_comm) {
- default_commodity = entry_comm;
+ if (parser->entry_comm) {
+ default_commodity = parser->entry_comm;
} else {
- account_comm_map::iterator ac = account_comms.find(xact->account);
- if (ac != account_comms.end())
+ gnucash_parser_t::account_comm_map::iterator ac =
+ parser->account_comms.find(xact->account);
+ if (ac != parser->account_comms.end())
default_commodity = (*ac).second;
}
if (default_commodity) {
- curr_quant.set_commodity(*default_commodity);
- value = curr_quant.round();
+ parser->curr_quant.set_commodity(*default_commodity);
+ value = parser->curr_quant.round();
- if (curr_value.commodity() == *default_commodity)
- curr_value = value;
+ if (parser->curr_value.commodity() == *default_commodity)
+ parser->curr_value = value;
} else {
- value = curr_quant;
+ value = parser->curr_quant;
}
- xact->state = curr_state;
+ xact->state = parser->curr_state;
xact->amount = value;
- if (value != curr_value)
- xact->cost = new amount_t(curr_value);
+ if (value != parser->curr_value)
+ xact->cost = new amount_t(parser->curr_value);
- xact->beg_pos = beg_pos;
- xact->beg_line = beg_line;
- xact->end_pos = instreamp->tellg();
- xact->end_line = XML_GetCurrentLineNumber(parser) - offset;
+ xact->beg_pos = parser->beg_pos;
+ xact->beg_line = parser->beg_line;
+ xact->end_pos = parser->instreamp->tellg();
+ xact->end_line =
+ XML_GetCurrentLineNumber(parser->expat_parser) - parser->offset;
// Clear the relevant variables for the next run
- curr_state = transaction_t::UNCLEARED;
- curr_value = amount_t();
- curr_quant = amount_t();
+ parser->curr_state = transaction_t::UNCLEARED;
+ parser->curr_value = amount_t();
+ parser->curr_quant = amount_t();
}
- action = NO_ACTION;
+ parser->action = gnucash_parser_t::NO_ACTION;
}
-
-static amount_t convert_number(const std::string& number,
- int * precision = NULL)
+amount_t gnucash_parser_t::convert_number(const std::string& number,
+ int * precision)
{
const char * num = number.c_str();
@@ -236,116 +179,120 @@ static amount_t convert_number(const std::string& number,
}
}
-static void dataHandler(void *userData, const char *s, int len)
+void dataHandler(void *userData, const char *s, int len)
{
- switch (action) {
- case ACCOUNT_NAME:
- curr_account->name = std::string(s, len);
+ gnucash_parser_t * parser = static_cast<gnucash_parser_t *>(userData);
+
+ switch (parser->action) {
+ case gnucash_parser_t::ACCOUNT_NAME:
+ parser->curr_account->name = std::string(s, len);
break;
- case ACCOUNT_ID:
- curr_account_id = std::string(s, len);
+ case gnucash_parser_t::ACCOUNT_ID:
+ parser->curr_account_id = std::string(s, len);
break;
- case ACCOUNT_PARENT: {
- accounts_map::iterator i = accounts_by_id.find(std::string(s, len));
- assert(i != accounts_by_id.end());
- curr_account->parent = (*i).second;
- curr_account->depth = curr_account->parent->depth + 1;
- (*i).second->add_account(curr_account);
+ case gnucash_parser_t::ACCOUNT_PARENT: {
+ accounts_map::iterator i = parser->accounts_by_id.find(std::string(s, len));
+ assert(i != parser->accounts_by_id.end());
+ parser->curr_account->parent = (*i).second;
+ parser->curr_account->depth = parser->curr_account->parent->depth + 1;
+ (*i).second->add_account(parser->curr_account);
break;
}
- case COMM_SYM:
- if (curr_comm) {
+ case gnucash_parser_t::COMM_SYM:
+ if (parser->curr_comm) {
#if 0
// jww (2006-03-02): !!!
- curr_comm->set_symbol(std::string(s, len));
+ parser->curr_comm->set_symbol(std::string(s, len));
#endif
}
- else if (curr_account) {
+ else if (parser->curr_account) {
std::string symbol(s, len);
commodity_t * comm = commodity_t::find_or_create(symbol);
assert(comm);
if (symbol != "$" && symbol != "USD")
comm->add_flags(COMMODITY_STYLE_SEPARATED);
- account_comms.insert(account_comm_pair(curr_account, comm));
+ parser->account_comms.insert
+ (gnucash_parser_t::account_comm_pair(parser->curr_account, comm));
}
- else if (curr_entry) {
+ else if (parser->curr_entry) {
std::string symbol(s, len);
- entry_comm = commodity_t::find_or_create(symbol);
- assert(entry_comm);
+ parser->entry_comm = commodity_t::find_or_create(symbol);
+ assert(parser->entry_comm);
if (symbol != "$" && symbol != "USD")
- entry_comm->add_flags(COMMODITY_STYLE_SEPARATED);
+ parser->entry_comm->add_flags(COMMODITY_STYLE_SEPARATED);
}
break;
- case COMM_NAME:
- curr_comm->name() = std::string(s, len);
+ case gnucash_parser_t::COMM_NAME:
+ parser->curr_comm->name() = std::string(s, len);
break;
- case COMM_PREC:
- curr_comm->set_precision(len - 1);
+ case gnucash_parser_t::COMM_PREC:
+ parser->curr_comm->set_precision(len - 1);
break;
- case ENTRY_NUM:
- curr_entry->code = std::string(s, len);
+ case gnucash_parser_t::ENTRY_NUM:
+ parser->curr_entry->code = std::string(s, len);
break;
- case ENTRY_DATE:
- curr_entry->_date = std::string(s, len);
+ case gnucash_parser_t::ENTRY_DATE:
+ parser->curr_entry->_date = std::string(s, len);
break;
- case ENTRY_DESC:
- curr_entry->payee = std::string(s, len);
+ case gnucash_parser_t::ENTRY_DESC:
+ parser->curr_entry->payee = std::string(s, len);
break;
- case XACT_STATE:
+ case gnucash_parser_t::XACT_STATE:
if (*s == 'y')
- curr_state = transaction_t::CLEARED;
+ parser->curr_state = transaction_t::CLEARED;
else if (*s == 'n')
- curr_state = transaction_t::UNCLEARED;
+ parser->curr_state = transaction_t::UNCLEARED;
else
- curr_state = transaction_t::PENDING;
+ parser->curr_state = transaction_t::PENDING;
break;
- case XACT_VALUE: {
+ case gnucash_parser_t::XACT_VALUE: {
int precision;
- assert(entry_comm);
- curr_value = convert_number(std::string(s, len), &precision);
- curr_value.set_commodity(*entry_comm);
+ assert(parser->entry_comm);
+ parser->curr_value = parser->convert_number(std::string(s, len), &precision);
+ parser->curr_value.set_commodity(*parser->entry_comm);
- if (precision > entry_comm->precision())
- entry_comm->set_precision(precision);
+ if (precision > parser->entry_comm->precision())
+ parser->entry_comm->set_precision(precision);
break;
}
- case XACT_QUANTITY:
- curr_quant = convert_number(std::string(s, len));
+ case gnucash_parser_t::XACT_QUANTITY:
+ parser->curr_quant = parser->convert_number(std::string(s, len));
break;
- case XACT_ACCOUNT: {
- transaction_t * xact = curr_entry->transactions.back();
+ case gnucash_parser_t::XACT_ACCOUNT: {
+ transaction_t * xact = parser->curr_entry->transactions.back();
- accounts_map::iterator i = accounts_by_id.find(std::string(s, len));
- if (i != accounts_by_id.end()) {
+ accounts_map::iterator i =
+ parser->accounts_by_id.find(std::string(s, len));
+ if (i != parser->accounts_by_id.end()) {
xact->account = (*i).second;
} else {
- xact->account = curr_journal->find_account("<Unknown>");
+ xact->account = parser->curr_journal->find_account("<Unknown>");
- have_error = (std::string("Could not find account ") +
- std::string(s, len));
+ parser->have_error = (std::string("Could not find account ") +
+ std::string(s, len));
}
break;
}
- case XACT_NOTE:
- curr_entry->transactions.back()->note = std::string(s, len);
+ case gnucash_parser_t::XACT_NOTE:
+ parser->curr_entry->transactions.back()->note = std::string(s, len);
break;
- case NO_ACTION:
- case ALMOST_ENTRY_DATE:
- case XACT_AMOUNT:
+ case gnucash_parser_t::NO_ACTION:
+ case gnucash_parser_t::ALMOST_ENTRY_DATE:
+ case gnucash_parser_t::XACT_AMOUNT:
break;
default:
@@ -365,7 +312,6 @@ bool gnucash_parser_t::test(std::istream& in) const
}
unsigned int gnucash_parser_t::parse(std::istream& in,
- config_t& config,
journal_t * journal,
account_t * master,
const std::string * original_file)
@@ -374,6 +320,8 @@ unsigned int gnucash_parser_t::parse(std::istream& in,
// This is the date format used by Gnucash, so override whatever the
// user specified.
+ //
+ // jww (2006-09-13): Make this parser local somehow.
date_t::input_format = "%Y-%m-%d %H:%M:%S %z";
count = 0;
@@ -401,10 +349,11 @@ unsigned int gnucash_parser_t::parse(std::istream& in,
#endif
offset = 2;
- parser = current_parser = XML_ParserCreate(NULL);
+ expat_parser = XML_ParserCreate(NULL);
XML_SetElementHandler(parser, startElement, endElement);
XML_SetCharacterDataHandler(parser, dataHandler);
+ XML_SetUserData(parser, this);
while (in.good() && ! in.eof()) {
beg_pos = in.tellg();