Moved files around

This commit is contained in:
John Wiegley 2007-05-21 20:42:59 +00:00
parent 7380da43ab
commit 1482b7f080
11 changed files with 16 additions and 972 deletions

View file

@ -1,265 +0,0 @@
/*
* Copyright (c) 2003-2007, John Wiegley. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* - Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
*
* - Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* - Neither the name of New Artisans LLC nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "format.h"
namespace ledger {
void format_t::parse(const string& fmt)
{
element_t * current = NULL;
char buf[1024];
char * q = buf;
if (elements.size() > 0)
clear_elements();
format_string = fmt;
for (const char * p = fmt.c_str(); *p; p++) {
if (*p != '%' && *p != '\\') {
*q++ = *p;
continue;
}
else if (*p == '\\') {
p++;
switch (*p) {
case 'b': *q++ = '\b'; break;
case 'f': *q++ = '\f'; break;
case 'n': *q++ = '\n'; break;
case 'r': *q++ = '\r'; break;
case 't': *q++ = '\t'; break;
case 'v': *q++ = '\v'; break;
default:
*q++ = *p;
break;
}
continue;
}
else {
assert(*p == '%');
if (*(p + 1) == '%') {
p++; // %% is the same as \%
*q++ = *p;
continue;
}
}
current = new element_t;
elements.push_back(current);
if (q != buf) {
current->kind = element_t::TEXT;
current->chars = new string(buf, q);
q = buf;
current = new element_t;
elements.push_back(current);
}
++p;
if (*p == '-') {
current->align_left = true;
++p;
}
if (*p && std::isdigit(*p)) {
int num = *p++ - '0';
while (*p && std::isdigit(*p)) {
num *= 10;
num += *p++ - '0';
}
current->min_width = num;
}
if (*p == '.') {
++p;
int num = 0;
while (*p && std::isdigit(*p)) {
num *= 10;
num += *p++ - '0';
}
current->max_width = num;
if (current->min_width == -1)
current->min_width = current->max_width;
}
if (current->max_width != -1 && current->min_width != -1 &&
current->max_width < current->min_width)
throw_(format_error, "Maximum width is less than the minimum width");
switch (*p) {
case '|':
current->kind = element_t::COLUMN;
break;
case '{':
case '(': {
char open = *p;
char close = *p == '{' ? '}' : ')';
++p;
const char * b = p;
int depth = 1;
while (*p) {
if (*p == close && --depth == 0)
break;
else if (*p == open)
++depth;
p++;
}
if (*p != close)
throw_(format_error, "Missing '" << close << "'");
if (open == '{') {
assert(! current->xpath);
current->kind = element_t::XPATH;
current->xpath = new xml::xpath_t(string(b, p));
} else {
assert(! current->format);
current->kind = element_t::GROUP;
current->format = new format_t(string(b, p));
}
break;
}
default:
assert(! current->xpath);
current->kind = element_t::XPATH;
current->xpath = new xml::xpath_t(string(p, p + 1));
break;
}
}
if (q != buf) {
current = new element_t;
elements.push_back(current);
current->kind = element_t::TEXT;
current->chars = new string(buf, q);
}
}
void format_t::compile(xml::node_t * context)
{
for (std::list<element_t *>::iterator i = elements.begin();
i != elements.end();
i++)
switch ((*i)->kind) {
case element_t::XPATH:
assert((*i)->xpath);
(*i)->xpath->compile(context);
break;
case element_t::GROUP:
assert((*i)->format);
(*i)->format->compile(context);
break;
default:
break;
}
}
int format_t::element_formatter_t::operator()
(std::ostream& out_str, element_t * elem, xml::node_t * context,
int column) const
{
if (elem->kind == element_t::COLUMN) {
if (elem->max_width != -1 && elem->max_width < column) {
out_str << '\n';
column = 0;
}
if (elem->min_width != -1 && elem->min_width > column) {
out_str << string(elem->min_width - column, ' ');
column = elem->min_width;
}
return column;
}
std::ostringstream out;
if (elem->align_left)
out << std::left;
else
out << std::right;
if (elem->min_width > 0)
out.width(elem->min_width);
int start_column = column;
if (elem->kind == element_t::XPATH)
elem->xpath->calc(context).strip_annotations()
.print(out, elem->min_width, elem->max_width);
else if (elem->kind == element_t::GROUP)
column = elem->format->format(out, context, column);
else if (elem->kind == element_t::TEXT)
out << *elem->chars;
else
assert(false);
string temp = out.str();
for (string::const_iterator i = temp.begin();
i != temp.end();
i++)
if (*i == '\n' || *i == '\r')
column = 0;
else
column++;
int virtual_width = column - start_column;
if (elem->min_width != -1 && virtual_width < elem->min_width) {
out_str << temp << string(' ', elem->min_width - virtual_width);
}
else if (elem->max_width != -1 && virtual_width > elem->max_width) {
temp.erase(temp.length() - (virtual_width - elem->max_width));
out_str << temp;
}
else {
out_str << temp;
}
return column;
}
int format_t::format(std::ostream& out, xml::node_t * context,
int column, const element_formatter_t& formatter) const
{
for (std::list<element_t *>::const_iterator i = elements.begin();
i != elements.end();
i++)
column = formatter(out, *i, context, column);
return column;
}
} // namespace ledger

View file

@ -1,139 +0,0 @@
/*
* Copyright (c) 2003-2007, John Wiegley. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* - Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
*
* - Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* - Neither the name of New Artisans LLC nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#ifndef _FORMAT_H
#define _FORMAT_H
#include "xpath.h"
namespace ledger {
class format_t
{
public:
struct element_t
{
bool align_left;
short min_width;
short max_width;
enum kind_t { UNKNOWN, TEXT, COLUMN, XPATH, GROUP } kind;
union {
string * chars;
xml::xpath_t * xpath;
format_t * format;
};
element_t()
: align_left(false), min_width(-1), max_width(-1),
kind(UNKNOWN), chars(NULL) {
TRACE_CTOR(element_t, "");
}
~element_t() {
TRACE_DTOR(element_t);
switch (kind) {
case TEXT:
checked_delete(chars);
break;
case XPATH:
checked_delete(xpath);
break;
case GROUP:
checked_delete(format);
break;
default:
assert(! chars);
break;
}
}
private:
element_t(const element_t& other);
};
struct element_formatter_t {
virtual ~element_formatter_t() {}
virtual int operator()(std::ostream& out, element_t * element,
xml::node_t * context, int column) const;
};
string format_string;
std::list<element_t *> elements;
private:
format_t(const format_t&);
public:
format_t() {
TRACE_CTOR(format_t, "");
}
format_t(const string& fmt) {
TRACE_CTOR(format_t, "const string&");
parse(fmt);
}
void clear_elements() {
for (std::list<element_t *>::iterator i = elements.begin();
i != elements.end();
i++)
checked_delete(*i);
elements.clear();
}
virtual ~format_t() {
TRACE_DTOR(format_t);
clear_elements();
}
void parse(const string& fmt);
void compile(const string& fmt, xml::node_t * context = NULL) {
parse(fmt);
compile(context);
}
void compile(xml::node_t * context = NULL);
int format(std::ostream& out, xml::node_t * context = NULL,
int column = 0, const element_formatter_t& formatter =
element_formatter_t()) const;
operator bool() const {
return ! format_string.empty();
}
};
DECLARE_EXCEPTION(format_error);
} // namespace ledger
#endif // _FORMAT_H

View file

@ -29,7 +29,7 @@
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/ */
#include "binary.h" #include "utils.h"
namespace ledger { namespace ledger {
namespace binary { namespace binary {

View file

@ -30,7 +30,7 @@
*/ */
/** /**
* @file scoped_execute.h * @file scopevar.h
* @author John Wiegley * @author John Wiegley
* @date Sun May 6 20:10:52 2007 * @date Sun May 6 20:10:52 2007
* *
@ -105,7 +105,7 @@
* @endcode * @endcode
* *
* But what if it could be even easier? That is what this file is * But what if it could be even easier? That is what this file is
* for, to provide a scoped_execute<> class which guarantees execution * for, to provide a scopevar<> class which guarantees execution
* of arbtirary code after a scope has terminated, without having to * of arbtirary code after a scope has terminated, without having to
* resort to custom utility classes. It relies on boost::bind to * resort to custom utility classes. It relies on boost::bind to
* declare pending function calls. Here it what the above would look * declare pending function calls. Here it what the above would look
@ -114,7 +114,7 @@
* @code * @code
* void foo(pthread_mutex_t * mutex) { * void foo(pthread_mutex_t * mutex) {
* if (pthread_mutex_lock(mutex) == 0) { * if (pthread_mutex_lock(mutex) == 0) {
* scoped_execute<void> unlock_mutex * scopevar<void> unlock_mutex
* (boost::bind(pthread_mutex_unlock, mutex)); * (boost::bind(pthread_mutex_unlock, mutex));
* try { * try {
* // Do work that requires the mutex to be locked * // Do work that requires the mutex to be locked
@ -131,7 +131,7 @@
* The single call to boost::bind creates a closure binding that will * The single call to boost::bind creates a closure binding that will
* be invoked once the containing scope has terminated. * be invoked once the containing scope has terminated.
* *
* Another kind of scoped_execute is useful for setting the values of * Another kind of scopevar is useful for setting the values of
* variables to a predetermined value upon completion of a scope. * variables to a predetermined value upon completion of a scope.
* Consider this example: * Consider this example:
* *
@ -139,7 +139,7 @@
* bool foo_was_run; * bool foo_was_run;
* *
* void foo() { * void foo() {
* scoped_execute<bool&> set_success((_1 = true), foo_was_run); * scopevar<bool&> set_success((_1 = true), foo_was_run);
* // do some code, and make sure foo_was_run is set to true * // do some code, and make sure foo_was_run is set to true
* // once the scope is exited -- however this happens. * // once the scope is exited -- however this happens.
* } * }
@ -177,34 +177,31 @@
* } * }
* @endcode * @endcode
* *
* Finally, you can stop a scoped_execute or scoped_variable from * Finally, you can stop a scopevar or scoped_variable from
* invoking its completion code by calling the `clear' method on the * invoking its completion code by calling the `clear' method on the
* object instance. Once `clear' is called, the scoped execution * object instance. Once `clear' is called, the scoped execution
* becomes inert and will do nothing when the enclosing scope is * becomes inert and will do nothing when the enclosing scope is
* exited. * exited.
*/ */
#ifndef _SCOPED_EXECUTE_H #ifndef _SCOPEVAR_H
#define _SCOPED_EXECUTE_H #define _SCOPEVAR_H
#include <boost/noncopyable.hpp>
#include <boost/function.hpp>
template <typename T> template <typename T>
class scoped_variable : public boost::noncopyable class push_variable : public boost::noncopyable
{ {
T& var; T& var;
T prev; T prev;
bool enabled; bool enabled;
public: public:
explicit scoped_variable(T& _var) explicit push_variable(T& _var)
: var(_var), prev(var), enabled(true) {} : var(_var), prev(var), enabled(true) {}
explicit scoped_variable(T& _var, const T& value) explicit push_variable(T& _var, const T& value)
: var(_var), prev(var), enabled(true) { : var(_var), prev(var), enabled(true) {
var = value; var = value;
} }
~scoped_variable() { ~push_variable() {
if (enabled) if (enabled)
var = prev; var = prev;
} }
@ -214,49 +211,4 @@ public:
} }
}; };
template <typename T> #endif // _SCOPEVAR_H
class scoped_execute : public boost::noncopyable
{
typedef boost::function<void (T)> function_t;
function_t code;
T arg;
bool enabled;
public:
explicit scoped_execute(const function_t& _code, T _arg)
: code(_code), arg(_arg), enabled(true) {}
~scoped_execute() {
if (enabled)
code(arg);
}
void clear() {
enabled = false;
}
};
template <>
class scoped_execute<void> : public boost::noncopyable
{
typedef boost::function<void ()> function_t;
function_t code;
bool enabled;
public:
explicit scoped_execute(const function_t& _code)
: code(_code), enabled(true) {}
~scoped_execute() {
if (enabled)
code();
}
void clear() {
enabled = false;
}
};
#endif // _SCOPED_EXECUTE_H

View file

@ -507,11 +507,13 @@ inline void throw_unexpected_error(char, char) {
* *
* Date/time support classes * Date/time support classes
* General support for objects with "flags" * General support for objects with "flags"
* Support for object serialization (binary read/write)
* Support for scoped execution and variable restoration * Support for scoped execution and variable restoration
*/ */
#include "times.h" #include "times.h"
#include "flags.h" #include "flags.h"
#include "binary.h"
#include "pushvar.h" #include "pushvar.h"
/********************************************************************** /**********************************************************************

View file

@ -1,506 +0,0 @@
/*
* Copyright (c) 2003-2007, John Wiegley. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* - Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
*
* - Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* - Neither the name of New Artisans LLC nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "xml.h"
#include "journal.h"
namespace ledger {
namespace xml {
#if defined(HAVE_EXPAT) || defined(HAVE_XMLPARSE)
static XML_Parser current_parser;
static unsigned int count;
static journal_t * curr_journal;
static entry_t * curr_entry;
static commodity_t * curr_comm;
static string comm_flags;
static transaction_t::state_t curr_state;
static string data;
static bool ignore;
static string have_error;
static void startElement(void *userData, const char *name, const char **attrs)
{
if (ignore)
return;
if (std::strcmp(name, "entry") == 0) {
assert(! curr_entry);
curr_entry = new entry_t;
curr_state = transaction_t::UNCLEARED;
}
else if (std::strcmp(name, "transaction") == 0) {
assert(curr_entry);
curr_entry->add_transaction(new transaction_t);
if (curr_state != transaction_t::UNCLEARED)
curr_entry->transactions.back()->state = curr_state;
}
else if (std::strcmp(name, "commodity") == 0) {
if (string(attrs[0]) == "flags")
comm_flags = attrs[1];
}
else if (std::strcmp(name, "total") == 0) {
ignore = true;
}
}
static void endElement(void *userData, const char *name)
{
if (ignore) {
if (std::strcmp(name, "total") == 0)
ignore = false;
return;
}
if (std::strcmp(name, "entry") == 0) {
assert(curr_entry);
if (curr_journal->add_entry(curr_entry)) {
count++;
} else {
account_t * acct = curr_journal->find_account("<Unknown>");
curr_entry->add_transaction(new transaction_t(acct));
if (curr_journal->add_entry(curr_entry)) {
count++;
} else {
checked_delete(curr_entry);
have_error = "Entry cannot be balanced";
}
}
curr_entry = NULL;
}
else if (std::strcmp(name, "en:date") == 0) {
curr_entry->_date = parse_datetime(data);
}
else if (std::strcmp(name, "en:date_eff") == 0) {
curr_entry->_date_eff = parse_datetime(data);
}
else if (std::strcmp(name, "en:code") == 0) {
curr_entry->code = data;
}
else if (std::strcmp(name, "en:cleared") == 0) {
curr_state = transaction_t::CLEARED;
}
else if (std::strcmp(name, "en:pending") == 0) {
curr_state = transaction_t::PENDING;
}
else if (std::strcmp(name, "en:payee") == 0) {
curr_entry->payee = data;
}
else if (std::strcmp(name, "tr:account") == 0) {
curr_entry->transactions.back()->account = curr_journal->find_account(data);
}
else if (std::strcmp(name, "tr:cleared") == 0) {
curr_entry->transactions.back()->state = transaction_t::CLEARED;
}
else if (std::strcmp(name, "tr:pending") == 0) {
curr_entry->transactions.back()->state = transaction_t::PENDING;
}
else if (std::strcmp(name, "tr:virtual") == 0) {
curr_entry->transactions.back()->add_flags(TRANSACTION_VIRTUAL);
}
else if (std::strcmp(name, "tr:generated") == 0) {
curr_entry->transactions.back()->add_flags(TRANSACTION_AUTO);
}
else if (std::strcmp(name, "symbol") == 0) {
assert(! curr_comm);
curr_comm = amount_t::current_pool->find_or_create(data);
assert(curr_comm);
curr_comm->add_flags(COMMODITY_STYLE_SUFFIXED);
if (! comm_flags.empty()) {
for (string::size_type i = 0, l = comm_flags.length(); i < l; i++) {
switch (comm_flags[i]) {
case 'P': curr_comm->drop_flags(COMMODITY_STYLE_SUFFIXED); break;
case 'S': curr_comm->add_flags(COMMODITY_STYLE_SEPARATED); break;
case 'T': curr_comm->add_flags(COMMODITY_STYLE_THOUSANDS); break;
case 'E': curr_comm->add_flags(COMMODITY_STYLE_EUROPEAN); break;
}
}
}
}
#if 0
// jww (2006-03-02): !!!
else if (std::strcmp(name, "price") == 0) {
assert(curr_comm);
amount_t * price = new amount_t(data);
std::ostringstream symstr;
symstr << curr_comm->symbol << " {" << *price << "}";
commodity_t * priced_comm =
commodity_t::find_commodity(symstr.str(), true);
priced_comm->price = price;
priced_comm->base = curr_comm;
curr_comm = priced_comm;
}
#endif
else if (std::strcmp(name, "quantity") == 0) {
amount_t temp;
temp.parse(data);
curr_entry->transactions.back()->amount = temp;
if (curr_comm) {
string::size_type i = data.find('.');
if (i != string::npos) {
int precision = data.length() - i - 1;
if (precision > curr_comm->precision())
curr_comm->set_precision(precision);
}
curr_entry->transactions.back()->amount->set_commodity(*curr_comm);
curr_comm = NULL;
}
}
else if (std::strcmp(name, "tr:amount") == 0) {
curr_comm = NULL;
}
}
static void dataHandler(void *userData, const char *s, int len)
{
if (! ignore)
data = string(s, len);
}
bool xml_parser_t::test(std::istream& in) const
{
char buf[80];
in.getline(buf, 79);
if (std::strncmp(buf, "<?xml", 5) != 0) {
in.clear();
in.seekg(0, std::ios::beg);
return false;
}
in.getline(buf, 79);
if (! std::strstr(buf, "<ledger")) {
in.clear();
in.seekg(0, std::ios::beg);
return false;
}
in.clear();
in.seekg(0, std::ios::beg);
return true;
}
unsigned int xml_parser_t::parse(std::istream& in,
journal_t * journal,
account_t * master,
const optional<path>& original)
{
char buf[BUFSIZ];
count = 0;
curr_journal = journal;
curr_entry = NULL;
curr_comm = NULL;
ignore = false;
XML_Parser parser = XML_ParserCreate(NULL);
current_parser = parser;
XML_SetElementHandler(parser, startElement, endElement);
XML_SetCharacterDataHandler(parser, dataHandler);
while (! in.eof()) {
in.getline(buf, BUFSIZ - 1);
std::strcat(buf, "\n");
bool result;
try {
result = XML_Parse(parser, buf, std::strlen(buf), in.eof());
}
catch (const std::exception& err) {
//unsigned long line = XML_GetCurrentLineNumber(parser) - offset++;
XML_ParserFree(parser);
throw_(parse_error, err.what());
}
if (! have_error.empty()) {
//unsigned long line = XML_GetCurrentLineNumber(parser) - offset++;
#if 0
// jww (2007-04-26): What is this code doing?
parse_error err(have_error);
std::cerr << "Error: " << err.what() << std::endl;
#endif
have_error = "";
}
if (! result) {
//unsigned long line = XML_GetCurrentLineNumber(parser) - offset++;
const char * err = XML_ErrorString(XML_GetErrorCode(parser));
XML_ParserFree(parser);
throw_(parse_error, err);
}
}
XML_ParserFree(parser);
return count;
}
#endif // defined(HAVE_EXPAT) || defined(HAVE_XMLPARSE)
#if 0
void xml_write_amount(std::ostream& out, const amount_t& amount,
const int depth = 0)
{
for (int i = 0; i < depth; i++) out << ' ';
out << "<amount>\n";
commodity_t& c = amount.commodity();
for (int i = 0; i < depth + 2; i++) out << ' ';
out << "<commodity flags=\"";
if (! (c.flags() & COMMODITY_STYLE_SUFFIXED)) out << 'P';
if (c.flags() & COMMODITY_STYLE_SEPARATED) out << 'S';
if (c.flags() & COMMODITY_STYLE_THOUSANDS) out << 'T';
if (c.flags() & COMMODITY_STYLE_EUROPEAN) out << 'E';
out << "\">\n";
for (int i = 0; i < depth + 4; i++) out << ' ';
#if 0
// jww (2006-03-02): !!!
if (c.price) {
out << "<symbol>" << c.base->symbol << "</symbol>\n";
for (int i = 0; i < depth + 4; i++) out << ' ';
out << "<price>\n";
xml_write_amount(out, *c.price, depth + 6);
for (int i = 0; i < depth + 4; i++) out << ' ';
out << "</price>\n";
} else {
out << "<symbol>" << c.symbol << "</symbol>\n";
}
#endif
for (int i = 0; i < depth + 2; i++) out << ' ';
out << "</commodity>\n";
for (int i = 0; i < depth + 2; i++) out << ' ';
out << "<quantity>";
out << amount.quantity_string() << "</quantity>\n";
for (int i = 0; i < depth; i++) out << ' ';
out << "</amount>\n";
}
void xml_write_value(std::ostream& out, const value_t& value,
const int depth = 0)
{
balance_t * bal = NULL;
for (int i = 0; i < depth; i++) out << ' ';
out << "<value type=\"";
switch (value.type) {
case value_t::BOOLEAN: out << "boolean"; break;
case value_t::INTEGER: out << "integer"; break;
case value_t::AMOUNT: out << "amount"; break;
case value_t::BALANCE:
case value_t::BALANCE_PAIR: out << "balance"; break;
}
out << "\">\n";
switch (value.type) {
case value_t::BOOLEAN:
for (int i = 0; i < depth + 2; i++) out << ' ';
out << "<boolean>" << value.as_boolean() << "</boolean>\n";
break;
case value_t::INTEGER:
for (int i = 0; i < depth + 2; i++) out << ' ';
out << "<integer>" << value.as_long() << "</integer>\n";
break;
case value_t::AMOUNT:
xml_write_amount(out, value.as_amount(), depth + 2);
break;
case value_t::BALANCE:
bal = &value.as_balance();
// fall through...
case value_t::BALANCE_PAIR:
if (! bal)
bal = &value.as_balance_pair()->quantity;
for (int i = 0; i < depth + 2; i++) out << ' ';
out << "<balance>\n";
#if 0
// jww (2007-04-30): Change this so that types know how to stream
// themselves to XML on their own.
for (balance_t::amounts_map::const_iterator i = bal->amounts.begin();
i != bal->amounts.end();
i++)
xml_write_amount(out, (*i).second, depth + 4);
#endif
for (int i = 0; i < depth + 2; i++) out << ' ';
out << "</balance>\n";
break;
default:
assert(false);
break;
}
for (int i = 0; i < depth; i++) out << ' ';
out << "</value>\n";
}
void output_xml_string(std::ostream& out, const string& str)
{
for (const char * s = str.c_str(); *s; s++) {
switch (*s) {
case '<':
out << "&lt;";
break;
case '>':
out << "&rt;";
break;
case '&':
out << "&amp;";
break;
default:
out << *s;
break;
}
}
}
void format_xml_entries::format_last_entry()
{
output_stream << " <entry>\n"
<< " <en:date>" << last_entry->_date.to_string("%Y/%m/%d")
<< "</en:date>\n";
if (last_entry->_date_eff)
output_stream << " <en:date_eff>"
<< last_entry->_date_eff.to_string("%Y/%m/%d")
<< "</en:date_eff>\n";
if (! last_entry->code.empty()) {
output_stream << " <en:code>";
output_xml_string(output_stream, last_entry->code);
output_stream << "</en:code>\n";
}
if (! last_entry->payee.empty()) {
output_stream << " <en:payee>";
output_xml_string(output_stream, last_entry->payee);
output_stream << "</en:payee>\n";
}
bool first = true;
for (transactions_list::const_iterator i = last_entry->transactions.begin();
i != last_entry->transactions.end();
i++) {
if (transaction_has_xdata(**i) &&
transaction_xdata_(**i).dflags & TRANSACTION_TO_DISPLAY) {
if (first) {
output_stream << " <en:transactions>\n";
first = false;
}
output_stream << " <transaction>\n";
if ((*i)->_date)
output_stream << " <tr:date>"
<< (*i)->_date.to_string("%Y/%m/%d")
<< "</tr:date>\n";
if ((*i)->_date_eff)
output_stream << " <tr:date_eff>"
<< (*i)->_date_eff.to_string("%Y/%m/%d")
<< "</tr:date_eff>\n";
if ((*i)->state == transaction_t::CLEARED)
output_stream << " <tr:cleared/>\n";
else if ((*i)->state == transaction_t::PENDING)
output_stream << " <tr:pending/>\n";
if ((*i)->flags & TRANSACTION_VIRTUAL)
output_stream << " <tr:virtual/>\n";
if ((*i)->flags & TRANSACTION_AUTO)
output_stream << " <tr:generated/>\n";
if ((*i)->account) {
string name = (*i)->account->fullname();
if (name == "<Total>")
name = "[TOTAL]";
else if (name == "<Unknown>")
name = "[UNKNOWN]";
output_stream << " <tr:account>";
output_xml_string(output_stream, name);
output_stream << "</tr:account>\n";
}
output_stream << " <tr:amount>\n";
if (transaction_xdata_(**i).dflags & TRANSACTION_COMPOUND)
xml_write_value(output_stream,
transaction_xdata_(**i).value, 10);
else
xml_write_value(output_stream, value_t((*i)->amount), 10);
output_stream << " </tr:amount>\n";
if ((*i)->cost) {
output_stream << " <tr:cost>\n";
xml_write_value(output_stream, value_t(*(*i)->cost), 10);
output_stream << " </tr:cost>\n";
}
if (! (*i)->note.empty()) {
output_stream << " <tr:note>";
output_xml_string(output_stream, (*i)->note);
output_stream << "</tr:note>\n";
}
if (show_totals) {
output_stream << " <total>\n";
xml_write_value(output_stream, transaction_xdata_(**i).total, 10);
output_stream << " </total>\n";
}
output_stream << " </transaction>\n";
transaction_xdata_(**i).dflags |= TRANSACTION_DISPLAYED;
}
}
if (! first)
output_stream << " </en:transactions>\n";
output_stream << " </entry>\n";
}
#endif
} // namespace xml
} // namespace ledger