Moved files around
This commit is contained in:
parent
7380da43ab
commit
1482b7f080
11 changed files with 16 additions and 972 deletions
265
src/format.cc
265
src/format.cc
|
|
@ -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
|
|
||||||
139
src/format.h
139
src/format.h
|
|
@ -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
|
|
||||||
|
|
@ -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 {
|
||||||
|
|
@ -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
|
|
||||||
|
|
@ -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"
|
||||||
|
|
||||||
/**********************************************************************
|
/**********************************************************************
|
||||||
506
src/xmlparse.cc
506
src/xmlparse.cc
|
|
@ -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 << "<";
|
|
||||||
break;
|
|
||||||
case '>':
|
|
||||||
out << "&rt;";
|
|
||||||
break;
|
|
||||||
case '&':
|
|
||||||
out << "&";
|
|
||||||
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
|
|
||||||
Loading…
Add table
Reference in a new issue