/* * Copyright (c) 2003-2009, 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 #include "pyinterp.h" #include "pyutils.h" #include "pyfstream.h" #include "commodity.h" #include "annotate.h" #include "pool.h" namespace ledger { using namespace boost::python; namespace { boost::optional py_value_0(const amount_t& amount) { return amount.value(); } boost::optional py_value_1(const amount_t& amount, const bool primary_only) { return amount.value(primary_only); } boost::optional py_value_2(const amount_t& amount, const bool primary_only, const boost::optional& moment) { return amount.value(primary_only, moment); } boost::optional py_value_3(const amount_t& amount, const bool primary_only, const boost::optional& moment, const boost::optional& in_terms_of) { return amount.value(primary_only, moment, in_terms_of); } void py_parse_2(amount_t& amount, object in, unsigned char flags) { if (PyFile_Check(in.ptr())) { pyifstream instr(reinterpret_cast(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_print(amount_t& amount, object out) { if (PyFile_Check(out.ptr())) { pyofstream outstr(reinterpret_cast(out.ptr())); amount.print(outstr); } else { PyErr_SetString(PyExc_IOError, _("Argument to amount.print_(file) is not a file object")); } } void py_amount_initialize() { amount_t::initialize(); } } // unnamed namespace #define EXC_TRANSLATOR(type) \ void exc_translate_ ## type(const type& err) { \ PyErr_SetString(PyExc_ArithmeticError, err.what()); \ } EXC_TRANSLATOR(amount_error) void export_amount() { class_< amount_t > ("Amount") .add_static_property("current_pool", make_getter(&amount_t::current_pool, return_value_policy())) .def("initialize", py_amount_initialize) // only for the PyUnitTests .staticmethod("initialize") .def("shutdown", &amount_t::shutdown) .staticmethod("shutdown") .add_static_property("is_initialized", make_getter(&amount_t::is_initialized), make_setter(&amount_t::is_initialized)) .add_static_property("stream_fullstrings", make_getter(&amount_t::stream_fullstrings), make_setter(&amount_t::stream_fullstrings)) .def(init()) .def(init()) .def("exact", &amount_t::exact, args("value"), _("Construct an amount object whose display precision is always equal to its\n\ internal precision.")) .staticmethod("exact") .def(init()) .def("compare", &amount_t::compare, args("amount"), _("Compare two amounts for equality, returning <0, 0 or >0.")) .def(self == self) .def(self == long()) .def(long() == self) .def(self != self) .def(self != long()) .def(long() != self) .def(! self) .def(self < self) .def(self < long()) .def(long() < self) .def(self <= self) .def(self <= long()) .def(long() <= self) .def(self > self) .def(self > long()) .def(long() > self) .def(self >= self) .def(self >= long()) .def(long() >= self) .def(self += self) .def(self += long()) .def(self + self) .def(self + long()) .def(long() + self) .def(self -= self) .def(self -= long()) .def(self - self) .def(self - long()) .def(long() - self) .def(self *= self) .def(self *= long()) .def(self * self) .def(self * long()) .def(long() * self) .def(self /= self) .def(self /= long()) .def(self / self) .def(self / long()) .def(long() / self) .def("precision", &amount_t::precision) .def("keep_precision", &amount_t::keep_precision) .def("set_keep_precision", &amount_t::set_keep_precision, args("keep"), _("Set whether an amount should always display at full precision.")) .def("display_precision", &amount_t::display_precision) .def("negated", &amount_t::negated) .def("in_place_negate", &amount_t::in_place_negate, return_value_policy()) .def(- self) .def("abs", &amount_t::abs) .def("__abs__", &amount_t::abs) .def("inverted", &amount_t::inverted) .def("rounded", &amount_t::rounded) .def("in_place_round", &amount_t::in_place_round, return_value_policy()) .def("truncated", &amount_t::truncated) .def("in_place_truncate", &amount_t::in_place_truncate, return_value_policy()) .def("unrounded", &amount_t::unrounded) .def("in_place_unround", &amount_t::in_place_unround, return_value_policy()) .def("reduced", &amount_t::reduced) .def("in_place_reduce", &amount_t::in_place_reduce, return_value_policy()) .def("unreduced", &amount_t::unreduced) .def("in_place_unreduce", &amount_t::in_place_unreduce, return_value_policy()) .def("value", py_value_0) .def("value", py_value_1, args("primary_only")) .def("value", py_value_2, args("primary_only", "moment")) .def("value", py_value_3, args("primary_only", "moment", "in_terms_of")) .def("price", &amount_t::price) .def("sign", &amount_t::sign) .def("__nonzero__", &amount_t::is_nonzero) .def("is_nonzero", &amount_t::is_nonzero) .def("is_zero", &amount_t::is_zero) .def("is_realzero", &amount_t::is_realzero) .def("is_null", &amount_t::is_null) .def("to_double", &amount_t::to_double) .def("__float__", &amount_t::to_double) .def("to_long", &amount_t::to_long) .def("__int__", &amount_t::to_long) .def("fits_in_long", &amount_t::fits_in_long) .def("to_string", &amount_t::to_string) .def("__str__", &amount_t::to_string) .def("to_fullstring", &amount_t::to_fullstring) .def("__repr__", &amount_t::to_fullstring) .def("quantity_string", &amount_t::quantity_string) .def("commodity", &amount_t::commodity, return_value_policy()) .def("has_commodity", &amount_t::has_commodity) .def("set_commodity", &amount_t::set_commodity, with_custodian_and_ward<1, 2>()) .def("clear_commodity", &amount_t::clear_commodity) .def("number", &amount_t::number) .def("annotate", &amount_t::annotate) .def("is_annotated", &amount_t::is_annotated) #if 0 .def("annotation", &amount_t::annotation) #endif .def("strip_annotations", &amount_t::strip_annotations) .def("parse", py_parse_1) .def("parse", py_parse_2) .def("parse", py_parse_str_1) .def("parse", py_parse_str_2) .def("parse_conversion", &amount_t::parse_conversion) .staticmethod("parse_conversion") .def("print_", py_print) .def("dump", &amount_t::dump) .def("valid", &amount_t::valid) ; enum_< amount_t::parse_flags_enum_t >("AmountParse") .value("DEFAULT", amount_t::PARSE_DEFAULT) .value("NO_MIGRATE", amount_t::PARSE_NO_MIGRATE) .value("NO_REDUCE", amount_t::PARSE_NO_REDUCE) .value("SOFT_FAIL", amount_t::PARSE_SOFT_FAIL) ; register_optional_to_python(); implicitly_convertible(); implicitly_convertible(); #define EXC_TRANSLATE(type) \ register_exception_translator(&exc_translate_ ## type); EXC_TRANSLATE(amount_error); } } // namespace ledger