summaryrefslogtreecommitdiff
path: root/amount.h
diff options
context:
space:
mode:
authorJohn Wiegley <johnw@newartisans.com>2008-04-13 03:35:00 -0400
committerJohn Wiegley <johnw@newartisans.com>2008-04-13 03:35:00 -0400
commit42f43b7686038e4cbca16d8d2118b139544e6de3 (patch)
tree52c5473401c57282242d66b8dd75f4c07bf41d07 /amount.h
parentc7b4370ff9c8ab5c96f15b1e712e6db6bdab6324 (diff)
downloadfork-ledger-42f43b7686038e4cbca16d8d2118b139544e6de3.tar.gz
fork-ledger-42f43b7686038e4cbca16d8d2118b139544e6de3.tar.bz2
fork-ledger-42f43b7686038e4cbca16d8d2118b139544e6de3.zip
Check in all changes made so far toward 3.0.
Diffstat (limited to 'amount.h')
-rw-r--r--amount.h140
1 files changed, 63 insertions, 77 deletions
diff --git a/amount.h b/amount.h
index dfe8d2df..7ab84177 100644
--- a/amount.h
+++ b/amount.h
@@ -2,6 +2,7 @@
#define _AMOUNT_H
#include <map>
+#include <deque>
#include <stack>
#include <string>
#include <cctype>
@@ -36,34 +37,29 @@ class amount_t
void _release();
void _dup();
void _resize(unsigned int prec);
-
- void _clear() {
- if (quantity) {
- assert(commodity_);
- _release();
- quantity = NULL;
- commodity_ = NULL;
- } else {
- assert(! commodity_);
- }
- }
+ void _clear();
bigint_t * quantity;
commodity_t * commodity_;
public:
// constructors
- amount_t() : quantity(NULL), commodity_(NULL) {}
+ amount_t() : quantity(NULL), commodity_(NULL) {
+ TRACE_CTOR("amount_t()");
+ }
amount_t(const amount_t& amt) : quantity(NULL) {
+ TRACE_CTOR("amount_t(copy)");
if (amt.quantity)
_copy(amt);
else
commodity_ = NULL;
}
amount_t(const std::string& value) : quantity(NULL) {
+ TRACE_CTOR("amount_t(const std::string&)");
parse(value);
}
amount_t(const char * value) : quantity(NULL) {
+ TRACE_CTOR("amount_t(const char *)");
parse(value);
}
amount_t(const bool value);
@@ -73,10 +69,12 @@ class amount_t
// destructor
~amount_t() {
+ TRACE_DTOR("amount_t");
if (quantity)
_release();
}
+ bool has_commodity() const;
commodity_t& commodity() const;
void set_commodity(commodity_t& comm) {
commodity_ = &comm;
@@ -260,35 +258,15 @@ class amount_t
negate();
}
-#define AMOUNT_PARSE_NO_MIGRATE 0x01
-#define AMOUNT_PARSE_NO_REDUCE 0x02
-
- void parse(std::istream& in, unsigned char flags = 0);
- void parse(const std::string& str, unsigned char flags = 0);
void reduce();
-
amount_t reduced() const {
amount_t temp(*this);
temp.reduce();
return temp;
}
- void read_quantity(char *& data);
- void read_quantity(std::istream& in);
- void write_quantity(std::ostream& out) const;
-
bool valid() const;
- // Classes that are friends, and help to implement this class
-
- friend std::ostream& operator<<(std::ostream& out, const amount_t& amt);
- friend std::istream& operator>>(std::istream& in, amount_t& amt);
-
- friend unsigned int sizeof_bigint_t();
-
- friend void read_binary_amount(char *& data, amount_t& amt);
- friend void write_binary_amount(std::ostream& out, const amount_t& amt);
-
// This function is special, and exists only to support a custom
// optimization in binary.cc (which offers a significant enough gain
// to be worth the trouble).
@@ -298,52 +276,44 @@ class amount_t
friend void parse_annotations(std::istream& in, amount_t& price,
datetime_t& date, std::string& tag);
-};
-unsigned int sizeof_bigint_t();
+ // Streaming interface
-void parse_quantity(std::istream& in, std::string& value);
-void parse_commodity(std::istream& in, std::string& symbol);
-void parse_annotations(std::istream& in, const std::string& symbol,
- std::string& name, std::string& price,
- std::string& date, std::string& tag);
-void parse_conversion(const std::string& larger,
- const std::string& smaller);
+ void dump(std::ostream& out) const {
+ out << "AMOUNT(";
+ print(out);
+ out << ")";
+ }
-inline bool is_quote_or_paren(char * p) {
- return *p == '"' || *p == '{' || *p == '[' || *p == '(';
-}
+#define AMOUNT_PARSE_NO_MIGRATE 0x01
+#define AMOUNT_PARSE_NO_REDUCE 0x02
-inline char * scan_past_quotes_and_parens(char * expr)
-{
- std::stack<char> paren_stack;
-
- char * p;
- for (p = expr; *p; p++) {
- if (*p == '"' ||
- ((*p == '(' || ((*p == '{' || *p == '[') &&
- paren_stack.top() != '(')) &&
- paren_stack.top() != '"')) {
- paren_stack.push(*p);
- }
- else if ((*p == ')' && paren_stack.top() == '(') ||
- (*p == '}' && paren_stack.top() == '{') ||
- (*p == ']' && paren_stack.top() == '[') ||
- (*p == '"' && paren_stack.top() == '"')) {
- paren_stack.pop();
- if (paren_stack.size() == 0)
- break;
- }
- }
- return p;
-}
+ void print(std::ostream& out) const;
+ void parse(std::istream& in, unsigned char flags = 0);
+ void parse(const std::string& str, unsigned char flags = 0) {
+ std::istringstream stream(str);
+ parse(stream, flags);
+ }
+
+ void print_quantity(std::ostream& out) const;
+
+ void write(std::ostream& out) const;
+ void read(std::istream& in);
+ void read(char *& data);
+
+ void write_quantity(std::ostream& out) const;
+ void read_quantity(std::istream& in);
+ void read_quantity(char *& data);
+};
inline amount_t abs(const amount_t& amt) {
return amt < 0 ? amt.negated() : amt;
}
-std::ostream& operator<<(std::ostream& out, const amount_t& amt);
-
+inline std::ostream& operator<<(std::ostream& out, const amount_t& amt) {
+ amt.print(out);
+ return out;
+}
inline std::istream& operator>>(std::istream& in, amount_t& amt) {
amt.parse(in);
return in;
@@ -432,6 +402,8 @@ class commodity_base_t
typedef std::map<const std::string, commodity_t *> commodities_map;
typedef std::pair<const std::string, commodity_t *> commodities_pair;
+typedef std::deque<commodity_t *> commodities_array;
+
class commodity_t
{
friend class annotated_commodity_t;
@@ -439,10 +411,11 @@ class commodity_t
public:
// This map remembers all commodities that have been defined.
- static commodities_map commodities;
- static bool commodities_sorted;
- static commodity_t * null_commodity;
- static commodity_t * default_commodity;
+ static commodities_map commodities;
+ static commodities_array commodities_by_ident;
+ static bool commodities_sorted;
+ static commodity_t * null_commodity;
+ static commodity_t * default_commodity;
static commodity_t * create(const std::string& symbol);
static commodity_t * find(const std::string& name);
@@ -463,8 +436,12 @@ class commodity_t
bool annotated;
public:
- explicit commodity_t() : base(NULL), annotated(false) {}
- virtual ~commodity_t() {}
+ explicit commodity_t() : base(NULL), annotated(false) {
+ TRACE_CTOR("commodity_t()");
+ }
+ virtual ~commodity_t() {
+ TRACE_DTOR("commodity_t");
+ }
operator bool() const {
return this != null_commodity;
@@ -568,6 +545,7 @@ class annotated_commodity_t : public commodity_t
std::string tag;
explicit annotated_commodity_t() {
+ TRACE_CTOR("annotated_commodity_t()");
annotated = true;
}
@@ -597,8 +575,7 @@ class annotated_commodity_t : public commodity_t
friend class amount_t;
};
-inline std::ostream& operator<<(std::ostream& out,
- const commodity_t& comm) {
+inline std::ostream& operator<<(std::ostream& out, const commodity_t& comm) {
out << comm.symbol();
return out;
}
@@ -607,6 +584,10 @@ inline amount_t amount_t::round() const {
return round(commodity().precision());
}
+inline bool amount_t::has_commodity() const {
+ return commodity_ && commodity_ != commodity_t::null_commodity;
+}
+
inline commodity_t& amount_t::commodity() const {
if (! commodity_)
return *commodity_t::null_commodity;
@@ -614,6 +595,11 @@ inline commodity_t& amount_t::commodity() const {
return *commodity_;
}
+
+void parse_conversion(const std::string& larger_str,
+ const std::string& smaller_str);
+
+
class amount_error : public error {
public:
amount_error(const std::string& reason) throw() : error(reason) {}