Copied over all of the C++ tests to Python again.

This commit is contained in:
John Wiegley 2007-05-11 07:22:41 +00:00
parent 64219e948f
commit 1d065bf0dc
10 changed files with 564 additions and 223 deletions

20
.gitignore vendored
View file

@ -1,4 +1,22 @@
/acconf.h.in~ *.Plo
*.Po
*.a
*.elc
*.la
*.lai
*.lo
*.o
.gdb_history
Makefile
UnitTests
acconf.h
acconf.h.in~
autom4te.cache autom4te.cache
config.log
config.status
elc-stamp
libtool
pending pending
stamp-h1
utils utils
ledger

View file

@ -99,6 +99,7 @@ libpyledger_la_LDFLAGS = -release $(PACKAGE_VERSION)
libpyledger_la_SOURCES = \ libpyledger_la_SOURCES = \
src/py_utils.cc \ src/py_utils.cc \
src/py_times.cc \ src/py_times.cc \
src/py_commodity.cc \
src/py_amount.cc src/py_amount.cc
@ -179,6 +180,7 @@ ledger_so_SOURCES = \
src/pyledger.cc \ src/pyledger.cc \
src/py_utils.cc \ src/py_utils.cc \
src/py_times.cc \ src/py_times.cc \
src/py_commodity.cc \
src/py_amount.cc src/py_amount.cc
ledger_so_DEPENDENCIES = libledger.la gdtoa/libgdtoa.la libpyledger.la ledger_so_DEPENDENCIES = libledger.la gdtoa/libgdtoa.la libpyledger.la

View file

@ -118,7 +118,8 @@ libledger_la_LINK = $(LIBTOOL) --tag=CXX $(AM_LIBTOOLFLAGS) \
$(CXXFLAGS) $(libledger_la_LDFLAGS) $(LDFLAGS) -o $@ $(CXXFLAGS) $(libledger_la_LDFLAGS) $(LDFLAGS) -o $@
libpyledger_la_LIBADD = libpyledger_la_LIBADD =
am_libpyledger_la_OBJECTS = libpyledger_la-py_utils.lo \ am_libpyledger_la_OBJECTS = libpyledger_la-py_utils.lo \
libpyledger_la-py_times.lo libpyledger_la-py_amount.lo libpyledger_la-py_times.lo libpyledger_la-py_commodity.lo \
libpyledger_la-py_amount.lo
libpyledger_la_OBJECTS = $(am_libpyledger_la_OBJECTS) libpyledger_la_OBJECTS = $(am_libpyledger_la_OBJECTS)
libpyledger_la_LINK = $(LIBTOOL) --tag=CXX $(AM_LIBTOOLFLAGS) \ libpyledger_la_LINK = $(LIBTOOL) --tag=CXX $(AM_LIBTOOLFLAGS) \
$(LIBTOOLFLAGS) --mode=link $(CXXLD) $(AM_CXXFLAGS) \ $(LIBTOOLFLAGS) --mode=link $(CXXLD) $(AM_CXXFLAGS) \
@ -148,9 +149,10 @@ ledger_LINK = $(LIBTOOL) --tag=CXX $(AM_LIBTOOLFLAGS) $(LIBTOOLFLAGS) \
--mode=link $(CXXLD) $(AM_CXXFLAGS) $(CXXFLAGS) \ --mode=link $(CXXLD) $(AM_CXXFLAGS) $(CXXFLAGS) \
$(ledger_LDFLAGS) $(LDFLAGS) -o $@ $(ledger_LDFLAGS) $(LDFLAGS) -o $@
am__ledger_so_SOURCES_DIST = src/pyledger.cc src/py_utils.cc \ am__ledger_so_SOURCES_DIST = src/pyledger.cc src/py_utils.cc \
src/py_times.cc src/py_amount.cc src/py_times.cc src/py_commodity.cc src/py_amount.cc
@HAVE_BOOST_PYTHON_TRUE@am_ledger_so_OBJECTS = pyledger.$(OBJEXT) \ @HAVE_BOOST_PYTHON_TRUE@am_ledger_so_OBJECTS = pyledger.$(OBJEXT) \
@HAVE_BOOST_PYTHON_TRUE@ py_utils.$(OBJEXT) py_times.$(OBJEXT) \ @HAVE_BOOST_PYTHON_TRUE@ py_utils.$(OBJEXT) py_times.$(OBJEXT) \
@HAVE_BOOST_PYTHON_TRUE@ py_commodity.$(OBJEXT) \
@HAVE_BOOST_PYTHON_TRUE@ py_amount.$(OBJEXT) @HAVE_BOOST_PYTHON_TRUE@ py_amount.$(OBJEXT)
ledger_so_OBJECTS = $(am_ledger_so_OBJECTS) ledger_so_OBJECTS = $(am_ledger_so_OBJECTS)
ledger_so_LDADD = $(LDADD) ledger_so_LDADD = $(LDADD)
@ -351,7 +353,7 @@ CLEANFILES = $(am__append_12) $(am__append_14)
ESC_srcdir = `echo "$(srcdir)" | sed 's/\//\\\\\//g'` ESC_srcdir = `echo "$(srcdir)" | sed 's/\//\\\\\//g'`
ESC_builddir = `echo "$(top_builddir)" | sed 's/\//\\\\\//g'` ESC_builddir = `echo "$(top_builddir)" | sed 's/\//\\\\\//g'`
ESC_distdir = `echo "$(distdir)" | sed 's/\//\\\\\//g'` ESC_distdir = `echo "$(distdir)" | sed 's/\//\\\\\//g'`
EXTRA_DIST = docs tests ledger.pdf ledger.info EXTRA_DIST = docs tests
lib_LTLIBRARIES = libledger.la $(am__append_1) lib_LTLIBRARIES = libledger.la $(am__append_1)
AM_YFLAGS = -d AM_YFLAGS = -d
AM_LFLAGS = -o $(LEX_OUTPUT_ROOT).c AM_LFLAGS = -o $(LEX_OUTPUT_ROOT).c
@ -378,6 +380,7 @@ libpyledger_la_LDFLAGS = -release $(PACKAGE_VERSION)
libpyledger_la_SOURCES = \ libpyledger_la_SOURCES = \
src/py_utils.cc \ src/py_utils.cc \
src/py_times.cc \ src/py_times.cc \
src/py_commodity.cc \
src/py_amount.cc src/py_amount.cc
pkginclude_HEADERS = \ pkginclude_HEADERS = \
@ -390,6 +393,7 @@ pkginclude_HEADERS = \
src/csv.h \ src/csv.h \
src/derive.h \ src/derive.h \
src/emacs.h \ src/emacs.h \
src/fdstream.hpp \
src/flags.h \ src/flags.h \
src/format.h \ src/format.h \
src/gnucash.h \ src/gnucash.h \
@ -410,6 +414,7 @@ pkginclude_HEADERS = \
src/report.h \ src/report.h \
src/scoped_execute.h \ src/scoped_execute.h \
src/session.h \ src/session.h \
src/system.hh \
src/textual.h \ src/textual.h \
src/times.h \ src/times.h \
src/transform.h \ src/transform.h \
@ -434,6 +439,7 @@ dist_lisp_LISP = lisp/ledger.el lisp/timeclock.el
@HAVE_BOOST_PYTHON_TRUE@ src/pyledger.cc \ @HAVE_BOOST_PYTHON_TRUE@ src/pyledger.cc \
@HAVE_BOOST_PYTHON_TRUE@ src/py_utils.cc \ @HAVE_BOOST_PYTHON_TRUE@ src/py_utils.cc \
@HAVE_BOOST_PYTHON_TRUE@ src/py_times.cc \ @HAVE_BOOST_PYTHON_TRUE@ src/py_times.cc \
@HAVE_BOOST_PYTHON_TRUE@ src/py_commodity.cc \
@HAVE_BOOST_PYTHON_TRUE@ src/py_amount.cc @HAVE_BOOST_PYTHON_TRUE@ src/py_amount.cc
@HAVE_BOOST_PYTHON_TRUE@ledger_so_DEPENDENCIES = libledger.la gdtoa/libgdtoa.la libpyledger.la @HAVE_BOOST_PYTHON_TRUE@ledger_so_DEPENDENCIES = libledger.la gdtoa/libgdtoa.la libpyledger.la
@ -639,9 +645,11 @@ distclean-compile:
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/libledger_la-xmlparse.Plo@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/libledger_la-xmlparse.Plo@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/libledger_la-xpath.Plo@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/libledger_la-xpath.Plo@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/libpyledger_la-py_amount.Plo@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/libpyledger_la-py_amount.Plo@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/libpyledger_la-py_commodity.Plo@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/libpyledger_la-py_times.Plo@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/libpyledger_la-py_times.Plo@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/libpyledger_la-py_utils.Plo@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/libpyledger_la-py_utils.Plo@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/py_amount.Po@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/py_amount.Po@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/py_commodity.Po@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/py_times.Po@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/py_times.Po@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/py_utils.Po@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/py_utils.Po@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/pyledger.Po@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/pyledger.Po@am__quote@
@ -870,6 +878,13 @@ libpyledger_la-py_times.lo: src/py_times.cc
@AMDEP_TRUE@@am__fastdepCXX_FALSE@ DEPDIR=$(DEPDIR) $(CXXDEPMODE) $(depcomp) @AMDEPBACKSLASH@ @AMDEP_TRUE@@am__fastdepCXX_FALSE@ DEPDIR=$(DEPDIR) $(CXXDEPMODE) $(depcomp) @AMDEPBACKSLASH@
@am__fastdepCXX_FALSE@ $(LIBTOOL) --tag=CXX $(AM_LIBTOOLFLAGS) $(LIBTOOLFLAGS) --mode=compile $(CXX) $(DEFS) $(DEFAULT_INCLUDES) $(INCLUDES) $(libpyledger_la_CPPFLAGS) $(CPPFLAGS) $(AM_CXXFLAGS) $(CXXFLAGS) -c -o libpyledger_la-py_times.lo `test -f 'src/py_times.cc' || echo '$(srcdir)/'`src/py_times.cc @am__fastdepCXX_FALSE@ $(LIBTOOL) --tag=CXX $(AM_LIBTOOLFLAGS) $(LIBTOOLFLAGS) --mode=compile $(CXX) $(DEFS) $(DEFAULT_INCLUDES) $(INCLUDES) $(libpyledger_la_CPPFLAGS) $(CPPFLAGS) $(AM_CXXFLAGS) $(CXXFLAGS) -c -o libpyledger_la-py_times.lo `test -f 'src/py_times.cc' || echo '$(srcdir)/'`src/py_times.cc
libpyledger_la-py_commodity.lo: src/py_commodity.cc
@am__fastdepCXX_TRUE@ $(LIBTOOL) --tag=CXX $(AM_LIBTOOLFLAGS) $(LIBTOOLFLAGS) --mode=compile $(CXX) $(DEFS) $(DEFAULT_INCLUDES) $(INCLUDES) $(libpyledger_la_CPPFLAGS) $(CPPFLAGS) $(AM_CXXFLAGS) $(CXXFLAGS) -MT libpyledger_la-py_commodity.lo -MD -MP -MF $(DEPDIR)/libpyledger_la-py_commodity.Tpo -c -o libpyledger_la-py_commodity.lo `test -f 'src/py_commodity.cc' || echo '$(srcdir)/'`src/py_commodity.cc
@am__fastdepCXX_TRUE@ mv -f $(DEPDIR)/libpyledger_la-py_commodity.Tpo $(DEPDIR)/libpyledger_la-py_commodity.Plo
@AMDEP_TRUE@@am__fastdepCXX_FALSE@ source='src/py_commodity.cc' object='libpyledger_la-py_commodity.lo' libtool=yes @AMDEPBACKSLASH@
@AMDEP_TRUE@@am__fastdepCXX_FALSE@ DEPDIR=$(DEPDIR) $(CXXDEPMODE) $(depcomp) @AMDEPBACKSLASH@
@am__fastdepCXX_FALSE@ $(LIBTOOL) --tag=CXX $(AM_LIBTOOLFLAGS) $(LIBTOOLFLAGS) --mode=compile $(CXX) $(DEFS) $(DEFAULT_INCLUDES) $(INCLUDES) $(libpyledger_la_CPPFLAGS) $(CPPFLAGS) $(AM_CXXFLAGS) $(CXXFLAGS) -c -o libpyledger_la-py_commodity.lo `test -f 'src/py_commodity.cc' || echo '$(srcdir)/'`src/py_commodity.cc
libpyledger_la-py_amount.lo: src/py_amount.cc libpyledger_la-py_amount.lo: src/py_amount.cc
@am__fastdepCXX_TRUE@ $(LIBTOOL) --tag=CXX $(AM_LIBTOOLFLAGS) $(LIBTOOLFLAGS) --mode=compile $(CXX) $(DEFS) $(DEFAULT_INCLUDES) $(INCLUDES) $(libpyledger_la_CPPFLAGS) $(CPPFLAGS) $(AM_CXXFLAGS) $(CXXFLAGS) -MT libpyledger_la-py_amount.lo -MD -MP -MF $(DEPDIR)/libpyledger_la-py_amount.Tpo -c -o libpyledger_la-py_amount.lo `test -f 'src/py_amount.cc' || echo '$(srcdir)/'`src/py_amount.cc @am__fastdepCXX_TRUE@ $(LIBTOOL) --tag=CXX $(AM_LIBTOOLFLAGS) $(LIBTOOLFLAGS) --mode=compile $(CXX) $(DEFS) $(DEFAULT_INCLUDES) $(INCLUDES) $(libpyledger_la_CPPFLAGS) $(CPPFLAGS) $(AM_CXXFLAGS) $(CXXFLAGS) -MT libpyledger_la-py_amount.lo -MD -MP -MF $(DEPDIR)/libpyledger_la-py_amount.Tpo -c -o libpyledger_la-py_amount.lo `test -f 'src/py_amount.cc' || echo '$(srcdir)/'`src/py_amount.cc
@am__fastdepCXX_TRUE@ mv -f $(DEPDIR)/libpyledger_la-py_amount.Tpo $(DEPDIR)/libpyledger_la-py_amount.Plo @am__fastdepCXX_TRUE@ mv -f $(DEPDIR)/libpyledger_la-py_amount.Tpo $(DEPDIR)/libpyledger_la-py_amount.Plo
@ -1031,6 +1046,20 @@ py_times.obj: src/py_times.cc
@AMDEP_TRUE@@am__fastdepCXX_FALSE@ DEPDIR=$(DEPDIR) $(CXXDEPMODE) $(depcomp) @AMDEPBACKSLASH@ @AMDEP_TRUE@@am__fastdepCXX_FALSE@ DEPDIR=$(DEPDIR) $(CXXDEPMODE) $(depcomp) @AMDEPBACKSLASH@
@am__fastdepCXX_FALSE@ $(CXX) $(DEFS) $(DEFAULT_INCLUDES) $(INCLUDES) $(AM_CPPFLAGS) $(CPPFLAGS) $(AM_CXXFLAGS) $(CXXFLAGS) -c -o py_times.obj `if test -f 'src/py_times.cc'; then $(CYGPATH_W) 'src/py_times.cc'; else $(CYGPATH_W) '$(srcdir)/src/py_times.cc'; fi` @am__fastdepCXX_FALSE@ $(CXX) $(DEFS) $(DEFAULT_INCLUDES) $(INCLUDES) $(AM_CPPFLAGS) $(CPPFLAGS) $(AM_CXXFLAGS) $(CXXFLAGS) -c -o py_times.obj `if test -f 'src/py_times.cc'; then $(CYGPATH_W) 'src/py_times.cc'; else $(CYGPATH_W) '$(srcdir)/src/py_times.cc'; fi`
py_commodity.o: src/py_commodity.cc
@am__fastdepCXX_TRUE@ $(CXX) $(DEFS) $(DEFAULT_INCLUDES) $(INCLUDES) $(AM_CPPFLAGS) $(CPPFLAGS) $(AM_CXXFLAGS) $(CXXFLAGS) -MT py_commodity.o -MD -MP -MF $(DEPDIR)/py_commodity.Tpo -c -o py_commodity.o `test -f 'src/py_commodity.cc' || echo '$(srcdir)/'`src/py_commodity.cc
@am__fastdepCXX_TRUE@ mv -f $(DEPDIR)/py_commodity.Tpo $(DEPDIR)/py_commodity.Po
@AMDEP_TRUE@@am__fastdepCXX_FALSE@ source='src/py_commodity.cc' object='py_commodity.o' libtool=no @AMDEPBACKSLASH@
@AMDEP_TRUE@@am__fastdepCXX_FALSE@ DEPDIR=$(DEPDIR) $(CXXDEPMODE) $(depcomp) @AMDEPBACKSLASH@
@am__fastdepCXX_FALSE@ $(CXX) $(DEFS) $(DEFAULT_INCLUDES) $(INCLUDES) $(AM_CPPFLAGS) $(CPPFLAGS) $(AM_CXXFLAGS) $(CXXFLAGS) -c -o py_commodity.o `test -f 'src/py_commodity.cc' || echo '$(srcdir)/'`src/py_commodity.cc
py_commodity.obj: src/py_commodity.cc
@am__fastdepCXX_TRUE@ $(CXX) $(DEFS) $(DEFAULT_INCLUDES) $(INCLUDES) $(AM_CPPFLAGS) $(CPPFLAGS) $(AM_CXXFLAGS) $(CXXFLAGS) -MT py_commodity.obj -MD -MP -MF $(DEPDIR)/py_commodity.Tpo -c -o py_commodity.obj `if test -f 'src/py_commodity.cc'; then $(CYGPATH_W) 'src/py_commodity.cc'; else $(CYGPATH_W) '$(srcdir)/src/py_commodity.cc'; fi`
@am__fastdepCXX_TRUE@ mv -f $(DEPDIR)/py_commodity.Tpo $(DEPDIR)/py_commodity.Po
@AMDEP_TRUE@@am__fastdepCXX_FALSE@ source='src/py_commodity.cc' object='py_commodity.obj' libtool=no @AMDEPBACKSLASH@
@AMDEP_TRUE@@am__fastdepCXX_FALSE@ DEPDIR=$(DEPDIR) $(CXXDEPMODE) $(depcomp) @AMDEPBACKSLASH@
@am__fastdepCXX_FALSE@ $(CXX) $(DEFS) $(DEFAULT_INCLUDES) $(INCLUDES) $(AM_CPPFLAGS) $(CPPFLAGS) $(AM_CXXFLAGS) $(CXXFLAGS) -c -o py_commodity.obj `if test -f 'src/py_commodity.cc'; then $(CYGPATH_W) 'src/py_commodity.cc'; else $(CYGPATH_W) '$(srcdir)/src/py_commodity.cc'; fi`
py_amount.o: src/py_amount.cc py_amount.o: src/py_amount.cc
@am__fastdepCXX_TRUE@ $(CXX) $(DEFS) $(DEFAULT_INCLUDES) $(INCLUDES) $(AM_CPPFLAGS) $(CPPFLAGS) $(AM_CXXFLAGS) $(CXXFLAGS) -MT py_amount.o -MD -MP -MF $(DEPDIR)/py_amount.Tpo -c -o py_amount.o `test -f 'src/py_amount.cc' || echo '$(srcdir)/'`src/py_amount.cc @am__fastdepCXX_TRUE@ $(CXX) $(DEFS) $(DEFAULT_INCLUDES) $(INCLUDES) $(AM_CPPFLAGS) $(CPPFLAGS) $(AM_CXXFLAGS) $(CXXFLAGS) -MT py_amount.o -MD -MP -MF $(DEPDIR)/py_amount.Tpo -c -o py_amount.o `test -f 'src/py_amount.cc' || echo '$(srcdir)/'`src/py_amount.cc
@am__fastdepCXX_TRUE@ mv -f $(DEPDIR)/py_amount.Tpo $(DEPDIR)/py_amount.Po @am__fastdepCXX_TRUE@ mv -f $(DEPDIR)/py_amount.Tpo $(DEPDIR)/py_amount.Po

8
gdtoa/.gitignore vendored Normal file
View file

@ -0,0 +1,8 @@
Makefile
acconf.h
arith.h
config.log
config.status
gd_qnan.h
libtool
stamp-h1

View file

@ -49,7 +49,7 @@ class annotated_commodity_t;
class commodity_t class commodity_t
: public delegates_flags<>, : public delegates_flags<>,
equality_comparable1<commodity_t, noncopyable> public equality_comparable1<commodity_t, noncopyable>
{ {
friend class commodity_pool_t; friend class commodity_pool_t;
@ -255,7 +255,7 @@ inline std::ostream& operator<<(std::ostream& out, const annotation_t& details)
class annotated_commodity_t class annotated_commodity_t
: public commodity_t, : public commodity_t,
equality_comparable<annotated_commodity_t, public equality_comparable<annotated_commodity_t,
equality_comparable2<annotated_commodity_t, commodity_t, equality_comparable2<annotated_commodity_t, commodity_t,
noncopyable> > noncopyable> >
{ {

View file

@ -40,6 +40,13 @@ namespace ledger {
using namespace boost::python; using namespace boost::python;
amount_t py_round_0(const amount_t& amount) {
return amount.round();
}
amount_t py_round_1(const amount_t& amount, amount_t::precision_t prec) {
return amount.round(prec);
}
double py_to_double_0(amount_t& amount) { double py_to_double_0(amount_t& amount) {
return amount.to_double(); return amount.to_double();
} }
@ -54,20 +61,6 @@ long py_to_long_1(amount_t& amount, bool no_check) {
return amount.to_long(no_check); return amount.to_long(no_check);
} }
void py_parse_1(amount_t& amount, const string& str) {
amount.parse(str);
}
void py_parse_2(amount_t& amount, const string& str, unsigned char flags) {
amount.parse(str, flags);
}
amount_t py_round_0(const amount_t& amount) {
return amount.round();
}
amount_t py_round_1(const amount_t& amount, amount_t::precision_t prec) {
return amount.round(prec);
}
boost::optional<amount_t> py_value_0(const amount_t& amount) { boost::optional<amount_t> py_value_0(const amount_t& amount) {
return amount.value(); return amount.value();
} }
@ -76,6 +69,40 @@ boost::optional<amount_t> py_value_1(const amount_t& amount,
return amount.value(moment); return amount.value(moment);
} }
void py_parse_2(amount_t& amount, object in, unsigned char flags) {
if (PyFile_Check(in.ptr())) {
pyifstream instr(reinterpret_cast<PyFileObject *>(in.ptr()));
amount.parse(instr, flags);
} else {
PyErr_SetString(PyExc_IOError,
"Argument to amount.parse(file) is not a file object");
}
}
void py_parse_1(amount_t& amount, object in) {
py_parse_2(amount, in, 0);
}
void py_parse_str_1(amount_t& amount, const string& str) {
amount.parse(str);
}
void py_parse_str_2(amount_t& amount, const string& str, unsigned char flags) {
amount.parse(str, flags);
}
void py_read_1(amount_t& amount, object in) {
if (PyFile_Check(in.ptr())) {
pyifstream instr(reinterpret_cast<PyFileObject *>(in.ptr()));
amount.read(instr);
} else {
PyErr_SetString(PyExc_IOError,
"Argument to amount.parse(file) is not a file object");
}
}
void py_read_2(amount_t& amount, const std::string& str) {
const char * p = str.c_str();
amount.read(p);
}
#define EXC_TRANSLATOR(type) \ #define EXC_TRANSLATOR(type) \
void exc_translate_ ## type(const type& err) { \ void exc_translate_ ## type(const type& err) { \
PyErr_SetString(PyExc_ArithmeticError, err.what()); \ PyErr_SetString(PyExc_ArithmeticError, err.what()); \
@ -112,7 +139,9 @@ void export_amount()
.def(init<long>()) .def(init<long>())
.def(init<std::string>()) .def(init<std::string>())
.def("exact", &amount_t::exact) .def("exact", &amount_t::exact, args("value"),
"Construct an amount object whose display precision is always equal to its\n\
internal precision.")
.staticmethod("exact") .staticmethod("exact")
.def(init<amount_t>()) .def(init<amount_t>())
@ -197,7 +226,7 @@ void export_amount()
.def(self / double()) .def(self / double())
.def(double() / self) .def(double() / self)
.def("precision", &amount_t::precision) .add_property("precision", &amount_t::precision)
.def("negate", &amount_t::negate) .def("negate", &amount_t::negate)
.def("in_place_negate", &amount_t::in_place_negate, .def("in_place_negate", &amount_t::in_place_negate,
@ -243,7 +272,7 @@ void export_amount()
.def("fits_in_double", &amount_t::fits_in_double) .def("fits_in_double", &amount_t::fits_in_double)
.def("fits_in_long", &amount_t::fits_in_long) .def("fits_in_long", &amount_t::fits_in_long)
.def("quantity_string", &amount_t::quantity_string) .add_property("quantity_string", &amount_t::quantity_string)
.add_property("commodity", .add_property("commodity",
make_function(&amount_t::commodity, make_function(&amount_t::commodity,
@ -253,19 +282,25 @@ void export_amount()
.def("has_commodity", &amount_t::has_commodity) .def("has_commodity", &amount_t::has_commodity)
.def("clear_commodity", &amount_t::clear_commodity) .def("clear_commodity", &amount_t::clear_commodity)
.def("number", &amount_t::number) .add_property("number", &amount_t::number)
.def("annotate_commodity", &amount_t::annotate_commodity) .def("annotate_commodity", &amount_t::annotate_commodity)
.def("commodity_annotated", &amount_t::commodity_annotated) .def("commodity_annotated", &amount_t::commodity_annotated)
.def("annotation_details", &amount_t::annotation_details) .add_property("annotation_details", &amount_t::annotation_details)
.def("strip_annotations", &amount_t::strip_annotations) .def("strip_annotations", &amount_t::strip_annotations)
.def("parse", py_parse_1) .def("parse", py_parse_1)
.def("parse", py_parse_2) .def("parse", py_parse_2)
.def("parse", py_parse_str_1)
.def("parse", py_parse_str_2)
.def("parse_conversion", &amount_t::parse_conversion) .def("parse_conversion", &amount_t::parse_conversion)
.staticmethod("parse_conversion") .staticmethod("parse_conversion")
.def("read", py_read_1)
.def("read", py_read_2)
.def("write", &amount_t::write)
.def("valid", &amount_t::valid) .def("valid", &amount_t::valid)
; ;

View file

@ -30,47 +30,18 @@
*/ */
#include "pyinterp.h" #include "pyinterp.h"
#include "pyutils.h"
#include "amount.h" #include "amount.h"
#include <boost/python/exception_translator.hpp> #include <boost/python/exception_translator.hpp>
#include <boost/python/implicit.hpp>
namespace ledger { namespace ledger {
using namespace boost::python; using namespace boost::python;
struct commodity_updater_wrap : public commodity_base_t::updater_t
{
PyObject * self;
commodity_updater_wrap(PyObject * self_) : self(self_) {}
virtual void operator()(commodity_base_t& commodity,
const moment_t& moment,
const moment_t& date,
const moment_t& last,
amount_t& price) {
call_method<void>(self, "__call__", commodity, moment, date, last, price);
}
};
commodity_t * py_find_commodity(const string& symbol)
{
return commodity_t::find(symbol);
}
#define EXC_TRANSLATOR(type) \
void exc_translate_ ## type(const type& err) { \
PyErr_SetString(PyExc_ArithmeticError, err.what()); \
}
EXC_TRANSLATOR(commodity_error)
void export_commodity() void export_commodity()
{ {
class_< commodity_base_t::updater_t, commodity_updater_wrap,
boost::noncopyable >
("updater")
;
scope().attr("COMMODITY_STYLE_DEFAULTS") = COMMODITY_STYLE_DEFAULTS; scope().attr("COMMODITY_STYLE_DEFAULTS") = COMMODITY_STYLE_DEFAULTS;
scope().attr("COMMODITY_STYLE_SUFFIXED") = COMMODITY_STYLE_SUFFIXED; scope().attr("COMMODITY_STYLE_SUFFIXED") = COMMODITY_STYLE_SUFFIXED;
scope().attr("COMMODITY_STYLE_SEPARATED") = COMMODITY_STYLE_SEPARATED; scope().attr("COMMODITY_STYLE_SEPARATED") = COMMODITY_STYLE_SEPARATED;
@ -79,46 +50,14 @@ void export_commodity()
scope().attr("COMMODITY_STYLE_NOMARKET") = COMMODITY_STYLE_NOMARKET; scope().attr("COMMODITY_STYLE_NOMARKET") = COMMODITY_STYLE_NOMARKET;
scope().attr("COMMODITY_STYLE_BUILTIN") = COMMODITY_STYLE_BUILTIN; scope().attr("COMMODITY_STYLE_BUILTIN") = COMMODITY_STYLE_BUILTIN;
class_< commodity_t > ("commodity") class_< commodity_t, bases<>,
.add_property("symbol", &commodity_t::symbol) commodity_t, boost::noncopyable > ("commodity", no_init)
.def(self == self)
.add_property("name", &commodity_t::name, &commodity_t::set_name) .def("drop_flags", &commodity_t::drop_flags)
.add_property("note", &commodity_t::note, &commodity_t::set_note)
.add_property("precision", &commodity_t::precision,
&commodity_t::set_precision)
.add_property("flags", &commodity_t::flags, &commodity_t::set_flags)
.add_property("add_flags", &commodity_t::add_flags)
.add_property("drop_flags", &commodity_t::drop_flags)
//.add_property("updater", &commodity_t::updater)
.add_property("smaller", .add_property("precision", &commodity_t::precision)
make_getter(&commodity_t::smaller,
return_value_policy<reference_existing_object>()),
make_setter(&commodity_t::smaller,
return_value_policy<reference_existing_object>()))
.add_property("larger",
make_getter(&commodity_t::larger,
return_value_policy<reference_existing_object>()),
make_setter(&commodity_t::larger,
return_value_policy<reference_existing_object>()))
.def(self_ns::str(self))
.def("find", py_find_commodity,
return_value_policy<reference_existing_object>())
.staticmethod("find")
.def("add_price", &commodity_t::add_price)
.def("remove_price", &commodity_t::remove_price)
.def("value", &commodity_t::value)
.def("valid", &commodity_t::valid)
; ;
#define EXC_TRANSLATE(type) \
register_exception_translator<type>(&exc_translate_ ## type);
EXC_TRANSLATE(commodity_error);
} }
} // namespace ledger } // namespace ledger

View file

@ -31,6 +31,7 @@
#include "pyinterp.h" #include "pyinterp.h"
#include "pyutils.h" #include "pyutils.h"
#include "pyfstream.h"
#include <boost/python/module.hpp> #include <boost/python/module.hpp>
#include <boost/python/def.hpp> #include <boost/python/def.hpp>
@ -74,6 +75,7 @@ struct bool_from_python
typedef register_python_conversion<bool, bool_to_python, bool_from_python> typedef register_python_conversion<bool, bool_to_python, bool_from_python>
bool_python_conversion; bool_python_conversion;
struct string_to_python struct string_to_python
{ {
static PyObject* convert(const string& str) static PyObject* convert(const string& str)
@ -103,10 +105,69 @@ struct string_from_python
typedef register_python_conversion<string, string_to_python, string_from_python> typedef register_python_conversion<string, string_to_python, string_from_python>
string_python_conversion; string_python_conversion;
struct istream_to_python
{
static PyObject* convert(const std::istream& str)
{
return incref(boost::python::detail::none());
}
};
struct istream_from_python
{
static void* convertible(PyObject* obj_ptr)
{
if (!PyFile_Check(obj_ptr)) return 0;
return obj_ptr;
}
static void construct(PyObject* obj_ptr, converter::rvalue_from_python_stage1_data* data)
{
void* storage = ((converter::rvalue_from_python_storage<pyifstream>*) data)->storage.bytes;
new (storage) pyifstream(reinterpret_cast<PyFileObject *>(obj_ptr));
data->convertible = storage;
}
};
typedef register_python_conversion<std::istream, istream_to_python, istream_from_python>
istream_python_conversion;
struct ostream_to_python
{
static PyObject* convert(const std::ostream& str)
{
return incref(boost::python::detail::none());
}
};
struct ostream_from_python
{
static void* convertible(PyObject* obj_ptr)
{
if (!PyFile_Check(obj_ptr)) return 0;
return obj_ptr;
}
static void construct(PyObject* obj_ptr, converter::rvalue_from_python_stage1_data* data)
{
void* storage = ((converter::rvalue_from_python_storage<pyofstream>*) data)->storage.bytes;
new (storage) pyofstream(reinterpret_cast<PyFileObject *>(obj_ptr));
data->convertible = storage;
}
};
typedef register_python_conversion<std::ostream, ostream_to_python, ostream_from_python>
ostream_python_conversion;
void export_utils() void export_utils()
{ {
bool_python_conversion(); bool_python_conversion();
string_python_conversion(); string_python_conversion();
istream_python_conversion();
ostream_python_conversion();
} }
} // namespace ledger } // namespace ledger

View file

@ -38,6 +38,7 @@ namespace ledger {
void export_utils(); void export_utils();
void export_times(); void export_times();
void export_amount(); void export_amount();
void export_commodity();
#if 0 #if 0
void export_balance(); void export_balance();
void export_value(); void export_value();
@ -55,6 +56,7 @@ void initialize_for_python()
export_utils(); export_utils();
export_times(); export_times();
export_amount(); export_amount();
export_commodity();
#if 0 #if 0
export_balance(); export_balance();
export_value(); export_value();

View file

@ -1,28 +1,37 @@
# -*- coding: utf-8 -*- # -*- coding: utf-8 -*-
import sys
import unittest import unittest
import exceptions import exceptions
import operator import operator
from ledger import amount from ledger import *
from StringIO import *
internalAmount = amount.exact internalAmount = amount.exact
class AmountTestCase(unittest.TestCase): class t_amountTestCase(unittest.TestCase):
def setUp(self): testSession = None
# Cause the display precision for dollars to be initialized to 2.
x1 = amount("$1.00")
self.assertTrue(x1)
amount.full_strings = True # makes error reports from UnitTests accurate
def tearDown(self):
amount.full_strings = False
def assertValid(self, amt): def assertValid(self, amt):
self.assertTrue(amt.valid()) self.assertTrue(amt.valid())
def setUp(self):
#self.testSession = session()
#set_session_context(self.testSession)
# Cause the display precision for dollars to be initialized to 2.
x1 = amount("$1.00")
self.assertTrue(x1)
#amount.stream_fullstrings = True # make reports from UnitTests accurate
def tearDown(self):
pass
#amount.stream_fullstrings = False
#set_session_context()
#self.testSession = None
def testParser(self): def testParser(self):
x0 = amount() x0 = amount()
x1 = amount() x1 = amount()
@ -32,22 +41,57 @@ class AmountTestCase(unittest.TestCase):
x5 = amount(x4) x5 = amount(x4)
x6 = amount(x4) x6 = amount(x4)
x7 = amount(x4) x7 = amount(x4)
x8 = amount("$123.456") x8 = amount("$123.45")
x9 = amount(x8) x9 = amount(x8)
x10 = amount(x8) x10 = amount(x8)
x11 = amount(x8) x11 = amount(x8)
x12 = amount("$100") x12 = amount("$100")
self.assertEqual(3, x12.commodity().precision()) self.assertEqual(2, x12.commodity.precision)
#buf = "$100..."
#input = StringIO(buf)
#x13 = amount()
#x13.parse(input)
#self.assertEqual(x12, x13)
x14 = amount()
self.assertRaises(exceptions.ArithmeticError, lambda: x14.parse("DM"))
x15 = amount("$1.000.000,00")
x16 = amount("$2000")
self.assertEqual("$2.000,00", x16.to_string())
x16.parse("$2000,00")
self.assertEqual("$2.000,00", x16.to_string())
# Since European-ness is an additive quality, we must switch back
# to American-ness manually
x15.commodity.drop_flags(COMMODITY_STYLE_EUROPEAN)
x17 = amount("$1,000,000.00")
x18 = amount("$2000")
self.assertEqual("$2,000.00", x18.to_string())
x18.parse("$2,000")
self.assertEqual("$2,000.00", x18.to_string())
self.assertEqual(x15, x17)
x19 = amount("EUR 1000")
x20 = amount("EUR 1000")
self.assertEqual("EUR 1000", x19.to_string())
self.assertEqual("EUR 1000", x20.to_string())
x1.parse("$100.0000", AMOUNT_PARSE_NO_MIGRATE) x1.parse("$100.0000", AMOUNT_PARSE_NO_MIGRATE)
self.assertEqual(3, x12.commodity().precision()) self.assertEqual(2, x12.commodity.precision)
self.assertEqual(x1.commodity(), x12.commodity()) self.assertEqual(x1.commodity, x12.commodity)
self.assertEqual(x1, x12) self.assertEqual(x1, x12)
x0.parse("$100.0000") x0.parse("$100.0000")
self.assertEqual(4, x12.commodity().precision()) self.assertEqual(4, x12.commodity.precision)
self.assertEqual(x0.commodity(), x12.commodity()) self.assertEqual(x0.commodity, x12.commodity)
self.assertEqual(x0, x12) self.assertEqual(x0, x12)
x2.parse("$100.00", AMOUNT_PARSE_NO_REDUCE) x2.parse("$100.00", AMOUNT_PARSE_NO_REDUCE)
@ -93,16 +137,21 @@ class AmountTestCase(unittest.TestCase):
x3 = amount(123.456) x3 = amount(123.456)
x5 = amount("123456") x5 = amount("123456")
x6 = amount("123.456") x6 = amount("123.456")
x7 = amount("123456")
x8 = amount("123.456")
x9 = amount(x3) x9 = amount(x3)
x10 = amount(x6) x10 = amount(x6)
x11 = amount(x8)
self.assertRaises(exceptions.ArithmeticError, operator.eq, amount(0), x0) self.assertRaises(exceptions.ArithmeticError, lambda: amount(0) == x0)
self.assertRaises(exceptions.ArithmeticError, operator.eq, amount(), x0) self.assertRaises(exceptions.ArithmeticError, lambda: amount() == x0)
self.assertRaises(exceptions.ArithmeticError, operator.eq, amount("0"), x0) self.assertRaises(exceptions.ArithmeticError, lambda: amount("0") == x0)
self.assertRaises(exceptions.ArithmeticError, operator.eq, amount("0.0"), x0) self.assertRaises(exceptions.ArithmeticError, lambda: amount("0.0") == x0)
self.assertEqual(x2, x1) self.assertEqual(x2, x1)
self.assertEqual(x5, x1) self.assertEqual(x5, x1)
self.assertEqual(x7, x1)
self.assertEqual(x6, x3) self.assertEqual(x6, x3)
self.assertEqual(x8, x3)
self.assertEqual(x10, x3) self.assertEqual(x10, x3)
self.assertEqual(x10, x9) self.assertEqual(x10, x9)
@ -112,8 +161,11 @@ class AmountTestCase(unittest.TestCase):
self.assertValid(x3) self.assertValid(x3)
self.assertValid(x5) self.assertValid(x5)
self.assertValid(x6) self.assertValid(x6)
self.assertValid(x7)
self.assertValid(x8)
self.assertValid(x9) self.assertValid(x9)
self.assertValid(x10) self.assertValid(x10)
self.assertValid(x11)
def testCommodityConstructors(self): def testCommodityConstructors(self):
x1 = amount("$123.45") x1 = amount("$123.45")
@ -167,12 +219,16 @@ class AmountTestCase(unittest.TestCase):
x3 = amount(123.456) x3 = amount(123.456)
x5 = amount("123456") x5 = amount("123456")
x6 = amount("123.456") x6 = amount("123.456")
x9 = x3 x7 = "123456"
x8 = "123.456"
x9 = amount(x3)
x10 = amount(x6) x10 = amount(x6)
self.assertEqual(x2, x1) self.assertEqual(x2, x1)
self.assertEqual(x5, x1) self.assertEqual(x5, x1)
self.assertEqual(x7, x1)
self.assertEqual(x6, x3) self.assertEqual(x6, x3)
self.assertEqual(x8, x3)
self.assertEqual(x10, x3) self.assertEqual(x10, x3)
self.assertEqual(x10, x9) self.assertEqual(x10, x9)
@ -181,12 +237,16 @@ class AmountTestCase(unittest.TestCase):
x3 = amount(123.456) x3 = amount(123.456)
x5 = amount("123456") x5 = amount("123456")
x6 = amount("123.456") x6 = amount("123.456")
x7 = amount("123456")
x8 = amount("123.456")
x9 = x3 x9 = x3
x10 = amount(x6) x10 = amount(x6)
self.assertEqual(x2, x1) self.assertEqual(x2, x1)
self.assertEqual(x5, x1) self.assertEqual(x5, x1)
self.assertEqual(x7, x1)
self.assertEqual(x6, x3) self.assertEqual(x6, x3)
self.assertEqual(x8, x3)
self.assertEqual(x10, x3) self.assertEqual(x10, x3)
self.assertEqual(x10, x9) self.assertEqual(x10, x9)
@ -201,6 +261,8 @@ class AmountTestCase(unittest.TestCase):
self.assertValid(x3) self.assertValid(x3)
self.assertValid(x5) self.assertValid(x5)
self.assertValid(x6) self.assertValid(x6)
self.assertValid(x7)
self.assertValid(x8)
self.assertValid(x9) self.assertValid(x9)
self.assertValid(x10) self.assertValid(x10)
@ -255,18 +317,28 @@ class AmountTestCase(unittest.TestCase):
x3 = amount(333333) x3 = amount(333333)
x4 = amount(123456.0) x4 = amount(123456.0)
x5 = amount("123456.0") x5 = amount("123456.0")
x6 = amount(123456.0)
self.assertTrue(x1 == 123456) self.assertTrue(x1 == 123456)
self.assertTrue(x1 != x2) self.assertTrue(x1 != x2)
self.assertTrue(x1 == (x2 - x3)) self.assertTrue(x1 == (x2 - x3))
self.assertTrue(x1 == x4) self.assertTrue(x1 == x4)
self.assertTrue(x4 == x5) self.assertTrue(x4 == x5)
self.assertTrue(x4 == x6)
self.assertTrue(x1 == 123456)
self.assertTrue(123456 == x1)
self.assertTrue(x1 == 123456L)
self.assertTrue(123456L == x1)
self.assertTrue(x1 == 123456.0)
self.assertTrue(123456.0 == x1)
self.assertValid(x1) self.assertValid(x1)
self.assertValid(x2) self.assertValid(x2)
self.assertValid(x3) self.assertValid(x3)
self.assertValid(x4) self.assertValid(x4)
self.assertValid(x5) self.assertValid(x5)
self.assertValid(x6)
def testCommodityEquality(self): def testCommodityEquality(self):
x0 = amount() x0 = amount()
@ -282,10 +354,12 @@ class AmountTestCase(unittest.TestCase):
x10 = amount("-123.45€") x10 = amount("-123.45€")
self.assertTrue(x0.is_null()) self.assertTrue(x0.is_null())
self.assertRaises(exceptions.ArithmeticError, amount.is_zero, x0) self.assertRaises(exceptions.ArithmeticError, lambda: x0.is_zero())
self.assertRaises(exceptions.ArithmeticError, amount.is_realzero, x0) self.assertRaises(exceptions.ArithmeticError, lambda: x0.is_realzero())
self.assertRaises(exceptions.ArithmeticError, amount.sign, x0) self.assertRaises(exceptions.ArithmeticError, lambda: x0.sign() == 0)
self.assertRaises(exceptions.ArithmeticError, amount.compare, x0, 0) self.assertRaises(exceptions.ArithmeticError, lambda: x0.compare(x1) < 0)
self.assertRaises(exceptions.ArithmeticError, lambda: x0.compare(x2) > 0)
self.assertRaises(exceptions.ArithmeticError, lambda: x0.compare(x0) == 0)
self.assertTrue(x1 != x2) self.assertTrue(x1 != x2)
self.assertTrue(x1 != x4) self.assertTrue(x1 != x4)
@ -318,12 +392,12 @@ class AmountTestCase(unittest.TestCase):
x5 = amount("-123.45") x5 = amount("-123.45")
x6 = amount("123.45") x6 = amount("123.45")
self.assertRaises(exceptions.ArithmeticError, operator.gt, x0, x1) self.assertRaises(exceptions.ArithmeticError, lambda: x0 > x1)
self.assertRaises(exceptions.ArithmeticError, operator.lt, x0, x2) self.assertRaises(exceptions.ArithmeticError, lambda: x0 < x2)
self.assertRaises(exceptions.ArithmeticError, operator.gt, x0, x3) self.assertRaises(exceptions.ArithmeticError, lambda: x0 > x3)
self.assertRaises(exceptions.ArithmeticError, operator.lt, x0, x4) self.assertRaises(exceptions.ArithmeticError, lambda: x0 < x4)
self.assertRaises(exceptions.ArithmeticError, operator.gt, x0, x5) self.assertRaises(exceptions.ArithmeticError, lambda: x0 > x5)
self.assertRaises(exceptions.ArithmeticError, operator.lt, x0, x6) self.assertRaises(exceptions.ArithmeticError, lambda: x0 < x6)
self.assertTrue(x1 > x3) self.assertTrue(x1 > x3)
self.assertTrue(x3 <= x5) self.assertTrue(x3 <= x5)
@ -332,10 +406,10 @@ class AmountTestCase(unittest.TestCase):
self.assertTrue(x3 < x4) self.assertTrue(x3 < x4)
self.assertTrue(x1 < 100) self.assertTrue(x1 < 100)
self.assertTrue(x1 < 100L)
self.assertTrue(x1 < 100.0)
self.assertTrue(100 > x1) self.assertTrue(100 > x1)
self.assertTrue(x1 < 100L)
self.assertTrue(100L > x1) self.assertTrue(100L > x1)
self.assertTrue(x1 < 100.0)
self.assertTrue(100.0 > x1) self.assertTrue(100.0 > x1)
self.assertValid(x0) self.assertValid(x0)
@ -353,6 +427,7 @@ class AmountTestCase(unittest.TestCase):
x4 = amount(internalAmount("$123.4544")) x4 = amount(internalAmount("$123.4544"))
x5 = amount("$-123.45") x5 = amount("$-123.45")
x6 = amount("$123.45") x6 = amount("$123.45")
x7 = amount("DM 123.45")
self.assertTrue(x1 > x3) self.assertTrue(x1 > x3)
self.assertTrue(x3 <= x5) self.assertTrue(x3 <= x5)
@ -361,6 +436,8 @@ class AmountTestCase(unittest.TestCase):
self.assertFalse(x3 == x5) self.assertFalse(x3 == x5)
self.assertTrue(x3 < x1) self.assertTrue(x3 < x1)
self.assertTrue(x3 < x4) self.assertTrue(x3 < x4)
self.assertFalse(x6 == x7)
self.assertRaises(exceptions.ArithmeticError, lambda: x6 < x7)
self.assertValid(x1) self.assertValid(x1)
self.assertValid(x2) self.assertValid(x2)
@ -370,6 +447,7 @@ class AmountTestCase(unittest.TestCase):
self.assertValid(x6) self.assertValid(x6)
def testIntegerAddition(self): def testIntegerAddition(self):
x0 = amount()
x1 = amount(123) x1 = amount(123)
y1 = amount(456) y1 = amount(456)
@ -386,6 +464,7 @@ class AmountTestCase(unittest.TestCase):
self.assertEqual(amount("246913578246913578246913578"), x4 + x4) self.assertEqual(amount("246913578246913578246913578"), x4 + x4)
self.assertValid(x0)
self.assertValid(x1) self.assertValid(x1)
self.assertValid(y1) self.assertValid(y1)
self.assertValid(x4) self.assertValid(x4)
@ -430,13 +509,15 @@ class AmountTestCase(unittest.TestCase):
self.assertEqual("$246.90", (x1 + x1).to_string()) self.assertEqual("$246.90", (x1 + x1).to_string())
self.assertEqual("$246.91", (x1 + x2).to_string()) self.assertEqual("$246.91", (x1 + x2).to_string())
self.assertRaises(exceptions.ArithmeticError, operator.add, x1, x0) self.assertRaises(exceptions.ArithmeticError, lambda: x1 + x0)
self.assertRaises(exceptions.ArithmeticError, operator.add, x1, x3) self.assertRaises(exceptions.ArithmeticError, lambda: x0 + x1)
self.assertRaises(exceptions.ArithmeticError, operator.add, x1, x4) self.assertRaises(exceptions.ArithmeticError, lambda: x0 + x0)
self.assertRaises(exceptions.ArithmeticError, operator.add, x1, x5) self.assertRaises(exceptions.ArithmeticError, lambda: x1 + x3)
self.assertRaises(exceptions.ArithmeticError, operator.add, x1, x6) self.assertRaises(exceptions.ArithmeticError, lambda: x1 + x4)
self.assertRaises(exceptions.ArithmeticError, operator.add, x1, 123.45) self.assertRaises(exceptions.ArithmeticError, lambda: x1 + x5)
self.assertRaises(exceptions.ArithmeticError, operator.add, x1, 123) self.assertRaises(exceptions.ArithmeticError, lambda: x1 + x6)
self.assertRaises(exceptions.ArithmeticError, lambda: x1 + 123.45)
self.assertRaises(exceptions.ArithmeticError, lambda: x1 + 123)
self.assertEqual(amount("DM 246.90"), x3 + x3) self.assertEqual(amount("DM 246.90"), x3 + x3)
self.assertEqual(amount("246.90 euro"), x4 + x4) self.assertEqual(amount("246.90 euro"), x4 + x4)
@ -536,13 +617,15 @@ class AmountTestCase(unittest.TestCase):
self.assertEqual("$0.00", (x1 - x1).to_string()) self.assertEqual("$0.00", (x1 - x1).to_string())
self.assertEqual("$-0.01", (x1 - x2).to_string()) self.assertEqual("$-0.01", (x1 - x2).to_string())
self.assertRaises(exceptions.ArithmeticError, operator.sub, x1, x0) self.assertRaises(exceptions.ArithmeticError, lambda: x1 - x0)
self.assertRaises(exceptions.ArithmeticError, operator.sub, x1, x3) self.assertRaises(exceptions.ArithmeticError, lambda: x0 - x1)
self.assertRaises(exceptions.ArithmeticError, operator.sub, x1, x4) self.assertRaises(exceptions.ArithmeticError, lambda: x0 - x0)
self.assertRaises(exceptions.ArithmeticError, operator.sub, x1, x5) self.assertRaises(exceptions.ArithmeticError, lambda: x1 - x3)
self.assertRaises(exceptions.ArithmeticError, operator.sub, x1, x6) self.assertRaises(exceptions.ArithmeticError, lambda: x1 - x4)
self.assertRaises(exceptions.ArithmeticError, operator.sub, x1, 123.45) self.assertRaises(exceptions.ArithmeticError, lambda: x1 - x5)
self.assertRaises(exceptions.ArithmeticError, operator.sub, x1, 123) self.assertRaises(exceptions.ArithmeticError, lambda: x1 - x6)
self.assertRaises(exceptions.ArithmeticError, lambda: x1 - 123.45)
self.assertRaises(exceptions.ArithmeticError, lambda: x1 - 123)
self.assertEqual(amount("DM 0.00"), x3 - x3) self.assertEqual(amount("DM 0.00"), x3 - x3)
self.assertEqual(amount("DM 23.45"), x3 - amount("DM 100.00")) self.assertEqual(amount("DM 23.45"), x3 - amount("DM 100.00"))
@ -661,6 +744,7 @@ class AmountTestCase(unittest.TestCase):
self.assertValid(x2) self.assertValid(x2)
def testCommodityMultiplication(self): def testCommodityMultiplication(self):
x0 = amount()
x1 = amount("$123.12") x1 = amount("$123.12")
y1 = amount("$456.45") y1 = amount("$456.45")
x2 = amount(internalAmount("$123.456789")) x2 = amount(internalAmount("$123.456789"))
@ -686,9 +770,12 @@ class AmountTestCase(unittest.TestCase):
self.assertEqual("$15200.00", (x1 * x2).to_string()) self.assertEqual("$15200.00", (x1 * x2).to_string())
self.assertEqual("$15199.99986168", (x2 * x1).to_string()) self.assertEqual("$15199.99986168", (x2 * x1).to_string())
self.assertRaises(exceptions.ArithmeticError, operator.mul, x1, x3) self.assertRaises(exceptions.ArithmeticError, lambda: x1 * x0)
self.assertRaises(exceptions.ArithmeticError, operator.mul, x1, x4) self.assertRaises(exceptions.ArithmeticError, lambda: x0 * x1)
self.assertRaises(exceptions.ArithmeticError, operator.mul, x1, x5) self.assertRaises(exceptions.ArithmeticError, lambda: x0 * x0)
self.assertRaises(exceptions.ArithmeticError, lambda: x1 * x3)
self.assertRaises(exceptions.ArithmeticError, lambda: x1 * x4)
self.assertRaises(exceptions.ArithmeticError, lambda: x1 * x5)
x1 *= amount("123.12") x1 *= amount("123.12")
self.assertEqual(internalAmount("$15158.5344"), x1) self.assertEqual(internalAmount("$15158.5344"), x1)
@ -712,14 +799,11 @@ class AmountTestCase(unittest.TestCase):
self.assertValid(x5) self.assertValid(x5)
self.assertValid(x7) self.assertValid(x7)
def divideByZero(self, amt):
return amt / 0
def testIntegerDivision(self): def testIntegerDivision(self):
x1 = amount(123) x1 = amount(123)
y1 = amount(456) y1 = amount(456)
self.assertRaises(exceptions.ArithmeticError, self.divideByZero, x1) self.assertRaises(exceptions.ArithmeticError, lambda: x1 / 0)
self.assertEqual(amount(0), amount(0) / x1) self.assertEqual(amount(0), amount(0) / x1)
self.assertEqual(amount(0), 0 / x1) self.assertEqual(amount(0), 0 / x1)
self.assertEqual(x1, x1 / 1) self.assertEqual(x1, x1 / 1)
@ -754,7 +838,7 @@ class AmountTestCase(unittest.TestCase):
x1 = amount(123.123) x1 = amount(123.123)
y1 = amount(456.456) y1 = amount(456.456)
self.assertRaises(exceptions.ArithmeticError, self.divideByZero, x1) self.assertRaises(exceptions.ArithmeticError, lambda: x1 / 0)
self.assertEqual(amount("0.008121959"), amount(1.0) / x1) self.assertEqual(amount("0.008121959"), amount(1.0) / x1)
self.assertEqual(amount("0.008121959"), 1.0 / x1) self.assertEqual(amount("0.008121959"), 1.0 / x1)
self.assertEqual(x1, x1 / 1.0) self.assertEqual(x1, x1 / 1.0)
@ -780,8 +864,7 @@ class AmountTestCase(unittest.TestCase):
y4 = amount("56.789") y4 = amount("56.789")
self.assertEqual(amount(1.0), x4 / x4) self.assertEqual(amount(1.0), x4 / x4)
self.assertEqual(amount("21739560323910.7554497273748437197344556164046"), self.assertEqual(amount("21739560323910.7554497273748437197344556164046"), x4 / y4)
x4 / y4)
self.assertValid(x1) self.assertValid(x1)
self.assertValid(y1) self.assertValid(y1)
@ -789,6 +872,7 @@ class AmountTestCase(unittest.TestCase):
self.assertValid(y4) self.assertValid(y4)
def testCommodityDivision(self): def testCommodityDivision(self):
x0 = amount()
x1 = amount("$123.12") x1 = amount("$123.12")
y1 = amount("$456.45") y1 = amount("$456.45")
x2 = amount(internalAmount("$123.456789")) x2 = amount(internalAmount("$123.456789"))
@ -796,7 +880,7 @@ class AmountTestCase(unittest.TestCase):
x4 = amount("123.45 euro") x4 = amount("123.45 euro")
x5 = amount("123.45€") x5 = amount("123.45€")
self.assertRaises(exceptions.ArithmeticError, operator.div, x1, 0) self.assertRaises(exceptions.ArithmeticError, lambda: x1 / 0)
self.assertEqual(amount("$0.00"), 0 / x1) self.assertEqual(amount("$0.00"), 0 / x1)
self.assertEqual(x1, x1 / 1) self.assertEqual(x1, x1 / 1)
self.assertEqual(internalAmount("$0.00812216"), 1 / x1) self.assertEqual(internalAmount("$0.00812216"), 1 / x1)
@ -814,9 +898,12 @@ class AmountTestCase(unittest.TestCase):
self.assertEqual("$1.00", (x1 / x2).to_string()) self.assertEqual("$1.00", (x1 / x2).to_string())
self.assertEqual("$1.00273545321637426901", (x2 / x1).to_string()) self.assertEqual("$1.00273545321637426901", (x2 / x1).to_string())
self.assertRaises(exceptions.ArithmeticError, operator.div, x1, x3) self.assertRaises(exceptions.ArithmeticError, lambda: x1 / x0)
self.assertRaises(exceptions.ArithmeticError, operator.div, x1, x4) self.assertRaises(exceptions.ArithmeticError, lambda: x0 / x1)
self.assertRaises(exceptions.ArithmeticError, operator.div, x1, x5) self.assertRaises(exceptions.ArithmeticError, lambda: x0 / x0)
self.assertRaises(exceptions.ArithmeticError, lambda: x1 / x3)
self.assertRaises(exceptions.ArithmeticError, lambda: x1 / x4)
self.assertRaises(exceptions.ArithmeticError, lambda: x1 / x5)
x1 /= amount("123.12") x1 /= amount("123.12")
self.assertEqual(internalAmount("$1.00"), x1) self.assertEqual(internalAmount("$1.00"), x1)
@ -846,14 +933,20 @@ class AmountTestCase(unittest.TestCase):
self.assertValid(x7) self.assertValid(x7)
def testNegation(self): def testNegation(self):
x0 = amount()
x1 = amount(-123456) x1 = amount(-123456)
x3 = amount(-123.456) x3 = amount(-123.456)
x5 = amount("-123456") x5 = amount("-123456")
x6 = amount("-123.456") x6 = amount("-123.456")
x7 = amount("-123456")
x8 = amount("-123.456")
x9 = amount(- x3) x9 = amount(- x3)
self.assertRaises(exceptions.ArithmeticError, lambda: x0.negate())
self.assertEqual(x5, x1) self.assertEqual(x5, x1)
self.assertEqual(x7, x1)
self.assertEqual(x6, x3) self.assertEqual(x6, x3)
self.assertEqual(x8, x3)
self.assertEqual(- x6, x9) self.assertEqual(- x6, x9)
self.assertEqual(x3.negate(), x9) self.assertEqual(x3.negate(), x9)
@ -865,6 +958,8 @@ class AmountTestCase(unittest.TestCase):
self.assertValid(x3) self.assertValid(x3)
self.assertValid(x5) self.assertValid(x5)
self.assertValid(x6) self.assertValid(x6)
self.assertValid(x7)
self.assertValid(x8)
self.assertValid(x9) self.assertValid(x9)
self.assertValid(x10) self.assertValid(x10)
@ -926,9 +1021,9 @@ class AmountTestCase(unittest.TestCase):
x1 = amount(-1234) x1 = amount(-1234)
x2 = amount(1234) x2 = amount(1234)
self.assertRaises(exceptions.ArithmeticError, amount.abs, x0) self.assertRaises(exceptions.ArithmeticError, lambda: x0.abs())
self.assertEqual(amount(1234), abs(x1)) self.assertEqual(amount(1234), x1.abs())
self.assertEqual(amount(1234), abs(x2)) self.assertEqual(amount(1234), x2.abs())
self.assertValid(x0) self.assertValid(x0)
self.assertValid(x1) self.assertValid(x1)
@ -938,22 +1033,52 @@ class AmountTestCase(unittest.TestCase):
x1 = amount("$-1234.56") x1 = amount("$-1234.56")
x2 = amount("$1234.56") x2 = amount("$1234.56")
self.assertEqual(amount("$1234.56"), abs(x1)) self.assertEqual(amount("$1234.56"), x1.abs())
self.assertEqual(amount("$1234.56"), abs(x2)) self.assertEqual(amount("$1234.56"), x2.abs())
self.assertValid(x1) self.assertValid(x1)
self.assertValid(x2) self.assertValid(x2)
def testFractionalRound(self): def testFractionalRound(self):
x0 = amount()
x1 = amount("1234.567890") x1 = amount("1234.567890")
self.assertEqual(amount("1234.56789"), x1.round(6)) self.assertRaises(exceptions.ArithmeticError, lambda: x0.precision)
self.assertEqual(amount("1234.56789"), x1.round(5)) self.assertRaises(exceptions.ArithmeticError, lambda: x0.round())
self.assertEqual(amount("1234.5679"), x1.round(4)) self.assertRaises(exceptions.ArithmeticError, lambda: x0.round(2))
self.assertEqual(amount("1234.568"), x1.round(3)) self.assertRaises(exceptions.ArithmeticError, lambda: x0.unround())
self.assertEqual(amount("1234.57"), x1.round(2)) self.assertEqual(6, x1.precision)
self.assertEqual(amount("1234.6"), x1.round(1))
self.assertEqual(amount("1235"), x1.round(0)) x1b = amount(x1.unround())
self.assertEqual(x1b.precision, x1b.unround().precision)
y7 = amount(x1.round(7))
y6 = amount(x1.round(6))
y5 = amount(x1.round(5))
y4 = amount(x1.round(4))
y3 = amount(x1.round(3))
y2 = amount(x1.round(2))
y1 = amount(x1.round(1))
y0 = amount(x1.round(0))
self.assertEqual(6, y7.precision)
self.assertEqual(6, y6.precision)
self.assertEqual(5, y5.precision)
self.assertEqual(4, y4.precision)
self.assertEqual(3, y3.precision)
self.assertEqual(2, y2.precision)
self.assertEqual(1, y1.precision)
self.assertEqual(0, y0.precision)
self.assertEqual(amount("1234.56789"), y7)
self.assertEqual(amount("1234.56789"), y6)
self.assertEqual(amount("1234.56789"), y5)
self.assertEqual(amount("1234.5679"), y4)
self.assertEqual(amount("1234.568"), y3)
self.assertEqual(amount("1234.57"), y2)
self.assertEqual(amount("1234.6"), y1)
self.assertEqual(amount("1235"), y0)
x2 = amount("9876.543210") x2 = amount("9876.543210")
@ -1074,22 +1199,26 @@ class AmountTestCase(unittest.TestCase):
self.assertEqual("$1.13", x1.to_string()) self.assertEqual("$1.13", x1.to_string())
def testReduction(self): def testReduction(self):
x0 = amount()
x1 = amount("60s") x1 = amount("60s")
x2 = amount("600s") x2 = amount("600s")
x3 = amount("6000s") x3 = amount("6000s")
x4 = amount("360000s") x4 = amount("360000s")
x5 = amount("10m") # 600s x5 = amount("10m")
x6 = amount("100m") # 6000s x6 = amount("100m")
x7 = amount("1000m") # 60000s x7 = amount("1000m")
x8 = amount("10000m") # 600000s x8 = amount("10000m")
x9 = amount("10h") # 36000s x9 = amount("10h")
x10 = amount("100h") # 360000s x10 = amount("100h")
x11 = amount("1000h") # 3600000s x11 = amount("1000h")
x12 = amount("10000h") # 36000000s x12 = amount("10000h")
self.assertRaises(exceptions.ArithmeticError, lambda: x0.reduce())
self.assertRaises(exceptions.ArithmeticError, lambda: x0.unreduce())
self.assertEqual(x2, x5) self.assertEqual(x2, x5)
self.assertEqual(x3, x6) self.assertEqual(x3, x6)
self.assertEqual(x4, x10) self.assertEqual(x4, x10)
self.assertEqual("100.0h", x4.unreduce().to_string())
def testSign(self): def testSign(self):
x0 = amount() x0 = amount()
@ -1098,7 +1227,7 @@ class AmountTestCase(unittest.TestCase):
x3 = amount("1") x3 = amount("1")
x4 = amount("-1") x4 = amount("-1")
self.assertRaises(exceptions.ArithmeticError, amount.sign, x0) self.assertRaises(exceptions.ArithmeticError, lambda: x0.sign())
self.assertTrue(x1.sign() > 0) self.assertTrue(x1.sign() > 0)
self.assertTrue(x2.sign() < 0) self.assertTrue(x2.sign() < 0)
self.assertTrue(x3.sign() > 0) self.assertTrue(x3.sign() > 0)
@ -1131,7 +1260,8 @@ class AmountTestCase(unittest.TestCase):
x1 = amount("1234") x1 = amount("1234")
x2 = amount("1234.56") x2 = amount("1234.56")
self.assertRaises(exceptions.ArithmeticError, operator.truth, x0) self.assertRaises(exceptions.ArithmeticError, lambda: 1 if x0 else 0)
self.assertTrue(x1) self.assertTrue(x1)
self.assertTrue(x2) self.assertTrue(x2)
@ -1161,8 +1291,8 @@ class AmountTestCase(unittest.TestCase):
x1 = amount("0.000000000000000000001") x1 = amount("0.000000000000000000001")
self.assertTrue(x1) self.assertTrue(x1)
self.assertRaises(exceptions.ArithmeticError, amount.is_zero, x0) self.assertRaises(exceptions.ArithmeticError, lambda: x0.is_zero())
self.assertRaises(exceptions.ArithmeticError, amount.is_realzero, x0) self.assertRaises(exceptions.ArithmeticError, lambda: x0.is_realzero())
self.assertFalse(x1.is_zero()) self.assertFalse(x1.is_zero())
self.assertFalse(x1.is_realzero()) self.assertFalse(x1.is_realzero())
@ -1179,45 +1309,162 @@ class AmountTestCase(unittest.TestCase):
self.assertValid(x1) self.assertValid(x1)
def testIntegerConversion(self): def testIntegerConversion(self):
x0 = amount()
x1 = amount(123456) x1 = amount(123456)
x2 = amount("12345682348723487324")
self.assertEqual(123456, int(x1)) self.assertRaises(exceptions.ArithmeticError, lambda: x0.to_long())
self.assertEqual(123456.0, float(x1)) self.assertRaises(exceptions.ArithmeticError, lambda: x0.to_double())
self.assertFalse(x2.fits_in_long())
self.assertEqual(123456, x1.to_long())
self.assertEqual(123456.0, x1.to_double())
self.assertEqual("123456", x1.to_string()) self.assertEqual("123456", x1.to_string())
self.assertEqual("123456", x1.quantity_string()) self.assertEqual("123456", x1.quantity_string)
self.assertValid(x1) self.assertValid(x1)
def testFractionalConversion(self): def testFractionalConversion(self):
x1 = amount(1234.56) x1 = amount(1234.56)
x2 = amount("1234.5683787634678348734")
self.assertRaises(exceptions.ArithmeticError, amount.to_long, x1) self.assertRaises(exceptions.ArithmeticError, lambda: x1.to_long()) # loses precision
self.assertRaises(exceptions.ArithmeticError, lambda: x2.to_double()) # loses precision
self.assertFalse(x2.fits_in_double())
self.assertEqual(1234, x1.to_long(True)) self.assertEqual(1234, x1.to_long(True))
self.assertEqual(1234.56, float(x1)) self.assertEqual(1234.56, x1.to_double())
self.assertEqual("1234.56", x1.to_string()) self.assertEqual("1234.56", x1.to_string())
self.assertEqual("1234.56", x1.quantity_string()) self.assertEqual("1234.56", x1.quantity_string)
self.assertValid(x1) self.assertValid(x1)
def testCommodityConversion(self): def testCommodityConversion(self):
x1 = amount("$1234.56") x1 = amount("$1234.56")
self.assertRaises(exceptions.ArithmeticError, amount.to_long, x1) self.assertRaises(exceptions.ArithmeticError, lambda: x1.to_long()) # loses precision
self.assertEqual(1234, x1.to_long(True)) self.assertEqual(1234, x1.to_long(True))
self.assertEqual(1234.56, float(x1)) self.assertEqual(1234.56, x1.to_double())
self.assertEqual("$1234.56", x1.to_string()) self.assertEqual("$1234.56", x1.to_string())
self.assertEqual("1234.56", x1.quantity_string()) self.assertEqual("1234.56", x1.quantity_string)
self.assertValid(x1) self.assertValid(x1)
def testPrinting(self): def testPrinting(self):
pass pass
#x0 = amount()
#x1 = amount("982340823.380238098235098235098235098")
#
#bufstr = StringIO()
#self.assertRaises(exceptions.ArithmeticError, lambda: bufstr.write(x0))
#
#bufstr = StringIO()
#bufstr.write(x1)
#
#self.assertEqual("982340823.380238098235098235098235098",
# bufstr.getvalue())
#
#self.assertValid(x0)
#self.assertValid(x1)
def testCommodityPrinting(self): def testCommodityPrinting(self):
pass pass
#x1 = amount(internalAmount("$982340823.386238098235098235098235098"))
#x2 = amount("$982340823.38")
#
#bufstr = StringIO()
#bufstr.write(x1)
#
#self.assertEqual("$982340823.386238098235098235098235098",
# bufstr.getvalue())
#
#bufstr = StringIO()
#bufstr.write((x1 * x2).to_string())
#
#self.assertEqual("$964993493285024293.18099172508158508135413499124",
# bufstr.getvalue())
#
#bufstr = StringIO()
#bufstr.write((x2 * x1).to_string())
#
#self.assertEqual("$964993493285024293.18", bufstr.getvalue())
#
#self.assertValid(x1)
#self.assertValid(x2)
def testSerialization(self):
pass
#x0 = amount()
#x1 = amount("$8,192.34")
#x2 = amount("8192.34")
#x3 = amount("8192.34")
#x4 = amount("-8192.34")
#x5 = amount(x4)
#
## Force x3's pointer to actually be set to null_commodity
##x3.set_commodity(*x3.current_pool.null_commodity)
#
#buf = ""
#storage = StringIO()
#self.assertRaises(exceptions.ArithmeticError, lambda: x0.write(storage))
#x1.write(storage)
#x2.write(storage)
#x3.write(storage)
#x4.write(storage)
#x5.write(storage)
#buf = storage.getvalue()
#
#x1b = amount()
#x2b = amount()
#x3b = amount()
#x4b = amount()
#x5b = amount()
#
#storage = StringIO(buf)
#x1b.read(storage)
#x2b.read(storage)
#x3b.read(storage)
#x4b.read(storage)
#x5b.read(storage)
#
#self.assertEqual(x1, x1b)
#self.assertEqual(x2, x2b)
#self.assertEqual(x3, x3b)
#self.assertEqual(x4, x4b)
#
#ptr = buf.c_str()
#
#x1c = amount()
#x2c = amount()
#x3c = amount()
#x4c = amount()
#x5c = amount()
#
#x1c.read(ptr)
#x2c.read(ptr)
#x3c.read(ptr)
#x4c.read(ptr)
#x5c.read(ptr)
#
#self.assertEqual(x1, x1b)
#self.assertEqual(x2, x2b)
#self.assertEqual(x3, x3b)
#self.assertEqual(x4, x4b)
#
#self.assertValid(x1)
#self.assertValid(x2)
#self.assertValid(x3)
#self.assertValid(x4)
#self.assertValid(x1b)
#self.assertValid(x2b)
#self.assertValid(x3b)
#self.assertValid(x4b)
#self.assertValid(x1c)
#self.assertValid(x2c)
#self.assertValid(x3c)
#self.assertValid(x4c)
def suite(): def suite():
return unittest.TestLoader().loadTestsFromTestCase(AmountTestCase) return unittest.TestLoader().loadTestsFromTestCase(t_amountTestCase)
if __name__ == '__main__': if __name__ == '__main__':
unittest.main() unittest.main()