From 4e30fcdf4094a0c450cbe1918c2e12dd19eb58f2 Mon Sep 17 00:00:00 2001 From: John Wiegley Date: Thu, 19 Nov 2009 21:53:02 -0500 Subject: Many improvements to Ledger's Python bindings --- src/py_commodity.cc | 199 +++++++++++++++++++++++++++++++++++++++++----------- 1 file changed, 159 insertions(+), 40 deletions(-) (limited to 'src/py_commodity.cc') diff --git a/src/py_commodity.cc b/src/py_commodity.cc index 08af8f62..dfaf7f5b 100644 --- a/src/py_commodity.cc +++ b/src/py_commodity.cc @@ -32,6 +32,7 @@ #include #include "pyinterp.h" +#include "pyutils.h" #include "commodity.h" #include "annotate.h" #include "pool.h" @@ -81,6 +82,12 @@ namespace { // Exchange one commodity for another, while recording the factored price. + void py_exchange_2(commodity_pool_t& pool, + commodity_t& commodity, + const amount_t& per_unit_cost) + { + pool.exchange(commodity, per_unit_cost, CURRENT_TIME()); + } void py_exchange_3(commodity_pool_t& pool, commodity_t& commodity, const amount_t& per_unit_cost, @@ -99,6 +106,77 @@ namespace { return pool.exchange(amount, cost, is_per_unit, moment, tag); } + commodity_t * py_pool_getitem(commodity_pool_t& pool, const string& symbol) + { + commodity_pool_t::commodities_map::iterator i = + pool.commodities.find(symbol); + if (i == pool.commodities.end()) { + PyErr_SetString(PyExc_ValueError, + (string("Could not find commodity ") + symbol).c_str()); + throw boost::python::error_already_set(); + } + return (*i).second; + } + + python::list py_pool_keys(commodity_pool_t& pool) { + python::list keys; + BOOST_REVERSE_FOREACH + (const commodity_pool_t::commodities_map::value_type& pair, + pool.commodities) { + keys.insert(0, pair.first); + } + return keys; + } + + bool py_pool_contains(commodity_pool_t& pool, const string& symbol) { + return pool.commodities.find(symbol) != pool.commodities.end(); + } + + commodity_pool_t::commodities_map::iterator + py_pool_commodities_begin(commodity_pool_t& pool) { + return pool.commodities.begin(); + } + commodity_pool_t::commodities_map::iterator + py_pool_commodities_end(commodity_pool_t& pool) { + return pool.commodities.end(); + } + + typedef transform_iterator + , + commodity_pool_t::commodities_map::iterator> + commodities_map_firsts_iterator; + commodities_map_firsts_iterator + + py_pool_commodities_keys_begin(commodity_pool_t& pool) { + return make_transform_iterator + (pool.commodities.begin(), + bind(&commodity_pool_t::commodities_map::value_type::first, _1)); + } + commodities_map_firsts_iterator + py_pool_commodities_keys_end(commodity_pool_t& pool) { + return make_transform_iterator + (pool.commodities.end(), + bind(&commodity_pool_t::commodities_map::value_type::first, _1)); + } + + typedef transform_iterator + , + commodity_pool_t::commodities_map::iterator> + commodities_map_seconds_iterator; + + commodities_map_seconds_iterator + py_pool_commodities_values_begin(commodity_pool_t& pool) { + return make_transform_iterator + (pool.commodities.begin(), + bind(&commodity_pool_t::commodities_map::value_type::second, _1)); + } + commodities_map_seconds_iterator + py_pool_commodities_values_end(commodity_pool_t& pool) { + return make_transform_iterator + (pool.commodities.end(), + bind(&commodity_pool_t::commodities_map::value_type::second, _1)); + } + void py_add_price_2(commodity_t& commodity, const datetime_t& date, const amount_t& price) { commodity.add_price(date, price); @@ -123,16 +201,38 @@ namespace { return details.keep_any(comm); } + commodity_t& py_commodity_referent(commodity_t& comm) { + return comm.referent(); + } + commodity_t& py_annotated_commodity_referent(annotated_commodity_t& comm) { + return comm.referent(); + } + + commodity_t& py_strip_annotations_0(commodity_t& comm) { + return comm.strip_annotations(keep_details_t()); + } + commodity_t& py_strip_annotations_1(commodity_t& comm, + const keep_details_t& keep) { + return comm.strip_annotations(keep); + } + + commodity_t& py_strip_ann_annotations_0(annotated_commodity_t& comm) { + return comm.strip_annotations(keep_details_t()); + } + commodity_t& py_strip_ann_annotations_1(annotated_commodity_t& comm, + const keep_details_t& keep) { + return comm.strip_annotations(keep); + } + } // unnamed namespace void export_commodity() { - class_< commodity_pool_t, boost::noncopyable > ("CommodityPool", no_init) + class_< commodity_pool_t, shared_ptr, + boost::noncopyable > ("CommodityPool", no_init) .add_property("null_commodity", make_getter(&commodity_pool_t::null_commodity, - return_internal_reference<>()), - make_setter(&commodity_pool_t::null_commodity, - with_custodian_and_ward<1, 2>())) + return_internal_reference<>())) .add_property("default_commodity", make_getter(&commodity_pool_t::default_commodity, return_internal_reference<>()), @@ -157,25 +257,46 @@ void export_commodity() .def("make_qualified_name", &commodity_pool_t::make_qualified_name) - .def("create", py_create_1, return_internal_reference<>()) - .def("create", py_create_2, return_internal_reference<>()) + .def("create", py_create_1, + return_value_policy()) + .def("create", py_create_2, + return_value_policy()) .def("find_or_create", py_find_or_create_1, - return_internal_reference<>()) + return_value_policy()) .def("find_or_create", py_find_or_create_2, - return_internal_reference<>()) + return_value_policy()) - .def("find", py_find_1, return_internal_reference<>()) - .def("find", py_find_2, return_internal_reference<>()) + .def("find", py_find_1, return_value_policy()) + .def("find", py_find_2, return_value_policy()) + .def("exchange", py_exchange_2, with_custodian_and_ward<1, 2>()) .def("exchange", py_exchange_3, with_custodian_and_ward<1, 2>()) .def("exchange", py_exchange_5) .def("parse_price_directive", &commodity_pool_t::parse_price_directive) .def("parse_price_expression", &commodity_pool_t::parse_price_expression, - return_internal_reference<>()) + return_value_policy()) + + .def("__getitem__", py_pool_getitem, + return_value_policy()) + .def("keys", py_pool_keys) + .def("has_key", py_pool_contains) + .def("__contains__", py_pool_contains) + .def("__iter__", range > + (py_pool_commodities_begin, py_pool_commodities_end)) + .def("iteritems", range > + (py_pool_commodities_begin, py_pool_commodities_end)) + .def("iterkeys", range<>(py_pool_commodities_keys_begin, + py_pool_commodities_keys_end)) + .def("itervalues", range > + (py_pool_commodities_values_begin, py_pool_commodities_values_end)) ; + map_value_type_converter(); + + scope().attr("commodity_pool") = commodity_pool_t::current_pool; + scope().attr("COMMODITY_STYLE_DEFAULTS") = COMMODITY_STYLE_DEFAULTS; scope().attr("COMMODITY_STYLE_SUFFIXED") = COMMODITY_STYLE_SUFFIXED; scope().attr("COMMODITY_STYLE_SEPARATED") = COMMODITY_STYLE_SEPARATED; @@ -209,33 +330,30 @@ void export_commodity() .def("symbol_needs_quotes", &commodity_t::symbol_needs_quotes) .staticmethod("symbol_needs_quotes") -#if 0 - .def("referent", &commodity_t::referent, - return_internal_reference<>()) -#endif + .add_property("referent", + make_function(py_commodity_referent, + return_value_policy())) - .def("is_annotated", &commodity_t::is_annotated) - .def("strip_annotations", &commodity_t::strip_annotations, - return_internal_reference<>()) + .def("has_annotation", &commodity_t::has_annotation) + .def("strip_annotations", py_strip_annotations_0, + return_value_policy()) + .def("strip_annotations", py_strip_annotations_1, + return_value_policy()) .def("write_annotations", &commodity_t::write_annotations) .def("pool", &commodity_t::pool, - return_internal_reference<>()) - - .def("base_symbol", &commodity_t::base_symbol) - .def("symbol", &commodity_t::symbol) - .def("mapping_key", &commodity_t::mapping_key) - - .def("name", &commodity_t::name) - .def("set_name", &commodity_t::set_name) - .def("note", &commodity_t::note) - .def("set_note", &commodity_t::set_note) - .def("precision", &commodity_t::precision) - .def("set_precision", &commodity_t::set_precision) - .def("smaller", &commodity_t::smaller) - .def("set_smaller", &commodity_t::set_smaller) - .def("larger", &commodity_t::larger) - .def("set_larger", &commodity_t::set_larger) + return_value_policy()) + + .add_property("base_symbol", &commodity_t::base_symbol) + .add_property("symbol", &commodity_t::symbol) + .add_property("mapping_key", &commodity_t::mapping_key) + + .add_property("name", &commodity_t::name, &commodity_t::set_name) + .add_property("note", &commodity_t::note, &commodity_t::set_note) + .add_property("precision", &commodity_t::precision, + &commodity_t::set_precision) + .add_property("smaller", &commodity_t::smaller, &commodity_t::set_smaller) + .add_property("larger", &commodity_t::larger, &commodity_t::set_larger) .def("add_price", py_add_price_2) .def("add_price", py_add_price_3) @@ -306,13 +424,14 @@ void export_commodity() .def(self == self) .def(self == other()) -#if 0 - .def("referent", &annotated_commodity_t::referent, - return_internal_reference<>()) -#endif + .add_property("referent", + make_function(py_annotated_commodity_referent, + return_value_policy())) - .def("strip_annotations", &annotated_commodity_t::strip_annotations, - return_internal_reference<>()) + .def("strip_annotations", py_strip_ann_annotations_0, + return_value_policy()) + .def("strip_annotations", py_strip_ann_annotations_1, + return_value_policy()) .def("write_annotations", &annotated_commodity_t::write_annotations) ; } -- cgit v1.2.3 From b00e7ac19a096a7b736863dced616d552843ed6e Mon Sep 17 00:00:00 2001 From: John Wiegley Date: Fri, 20 Nov 2009 05:56:24 -0500 Subject: Added more documentation to python/demo.py --- python/demo.py | 285 +++++++++++++++++++++++++++++++++++++++++----------- src/py_commodity.cc | 14 ++- 2 files changed, 239 insertions(+), 60 deletions(-) mode change 100644 => 100755 python/demo.py (limited to 'src/py_commodity.cc') diff --git a/python/demo.py b/python/demo.py old mode 100644 new mode 100755 index 788610d7..7b4003f3 --- a/python/demo.py +++ b/python/demo.py @@ -1,91 +1,264 @@ +#!/usr/bin/env python + import sys +from datetime import datetime + +# The following literate program will demonstrate, by example, how to use the +# Ledger Python module to access your data and build custom reports using the +# magic of Python. import ledger print "Welcome to the Ledger.Python demo!" +# Some quick helper functions to help us assert various types of truth +# throughout the script. + def assertEqual(pat, candidate): if pat != candidate: - print "FAILED: %s != %s" % (pat, candidate) + raise Exception("FAILED: %s != %s" % (pat, candidate)) sys.exit(1) +############################################################################### +# # COMMODITIES - -pool = ledger.commodity_pool - -usd = pool.find_or_create('$') -eur = pool.find_or_create('EUR') -xcd = pool.find_or_create('XCD') +# +# Every amount in Ledger has a commodity, even if it is the "null commodity". +# What's special about commodities are not just their symbol, but how they +# alter the way amounts are displayed. +# +# For example, internally Ledger uses infinite precision rational numbers, +# which have no decimal point. So how does it know that $1.00 / $0.75 should +# be displayed as $1.33, and not with an infinitely repeating decimal? It +# does it by consulting the commodity. +# +# Whenever an amount is encountered in your data file, Ledger observes how you +# specified it: +# - How many digits of precision did you use? +# - Was the commodity name before or after the amount? +# - Was the commodity separated from the amount by a space? +# - Did you use thousands markers (1,000)? +# - Did you use European-style numbers (1.000,00)? +# +# By tracking this information for each commodity, Ledger knows how you want +# to see the amount in your reports. This way, dollars can be output as +# $123.56, while stock options could be output as 10.113 AAPL. +# +# Your program can access the known set of commodities using the global +# `ledger.commodities'. This object behaves like a dict, and support all of +# the non-modifying dict protocol methods. If you wish to create a new +# commodity without parsing an amount, you can use the method +# `find_or_create': + +comms = ledger.commodities + +usd = comms.find_or_create('$') +eur = comms.find_or_create('EUR') +xcd = comms.find_or_create('XCD') + +assert not comms.find('CAD') +assert not comms.has_key('CAD') +assert not 'CAD' in comms + +# The above mentioned commodity display attributes can be set using commodity +# display flags. This is not something you will usually be doing, however, as +# these flags can be inferred correctly from a large enough set of sample +# amounts, such as those found in your data file. If you live in Europe and +# want all amounts to default to the European-style, set the static variable +# `european_by_default'. + +eur.add_flags(ledger.COMMODITY_STYLE_EUROPEAN) +assert eur.has_flags(ledger.COMMODITY_STYLE_EUROPEAN) +assert not eur.has_flags(ledger.COMMODITY_STYLE_THOUSANDS) + +comms.european_by_default = True + +# There are a few built-in commodities: null, %, h, m and s. Normally you +# don't need to worry about them, but they'll show up if you examine all the +# keys in the commodities dict. + +assertEqual([u'', u'$', u'%', u'EUR', u'XCD', u'h', u'm', u's'], + sorted(comms.keys())) + +# All the styles of dict iteration are supported: + +for symbol in comms.iterkeys(): + pass +for commodity in comms.itervalues(): + pass +#for symbol, commodity in comms.iteritems(): +# pass +#for symbol, commodity in comms: +# pass + +# Another important thing about commodities is that they remember if they've +# been exchanged for another commodity, and what the conversion rate was on +# that date. You can record specific conversion rates for any date using the +# `exchange' method. + +comms.exchange(eur, ledger.Amount('$0.77')) # Trade 1 EUR for $0.77 +comms.exchange(eur, ledger.Amount('$0.66'), datetime.now()) + +# For the most part, however, you won't be interacting with commodities +# directly, except maybe to look at their `symbol'. assertEqual('$', usd.symbol) -assertEqual('$', pool['$'].symbol) +assertEqual('$', comms['$'].symbol) -assert not pool.find('CAD') -assert not pool.has_key('CAD') -assert not 'CAD' in pool +############################################################################### +# +# AMOUNTS & BALANCES +# +# Ledger deals with two basic numerical values: Amount and Balance objects. +# An Amount is an infinite-precision rational with an associated commodity +# (even if it is the null commodity, which is called an "uncommoditized +# amount"). A Balance is a collection of Amounts of differing commodities. +# +# Amounts support all the math operations you might expect of an integer, +# except it carries a commodity. Let's take dollars for example: + +zero = ledger.Amount("$0") +one = ledger.Amount("$1") +oneb = ledger.Amount("$1") +two = ledger.Amount("$2") +three = ledger.Amount("3") # uncommoditized + +assert one == oneb # numeric equality, not identity +assert one != two +assert not zero # tests if it would *display* as a zero +assert one < two +assert one > zero + +# For addition and subtraction, only amounts of the same commodity may be +# used, unless one of the amounts has no commodity at all -- in which case the +# result uses the commodity of the other value. Adding $10 to 10 EUR, for +# example, causes an ArithmeticError exception, but adding 10 to $10 gives +# $20. + +four = ledger.Amount(two) # make a copy +four += two +assertEqual(four, two + two) +assertEqual(zero, one - one) + +try: + two += ledger.Amount("20 EUR") + assert False +except ArithmeticError: + pass + +# Use `number' to get the uncommoditized version of an Amount + +assertEqual(three, (two + one).number()) + +# Multiplication and division does supports Amounts of different commodities, +# however: +# - If either amount is uncommoditized, the result carries the commodity of +# the other amount. +# - Otherwise, the result always carries the commodity of the first amount. + +five = ledger.Amount("5 CAD") + +assertEqual(one, two / two) +assertEqual(five, (five * ledger.Amount("$2")) - ledger.Amount("5")) + +# An amount's commodity determines the decimal precision it's displayed with. +# However, this "precision" is a notional thing only. You can tell an amount +# to ignore its display precision by setting `keep_precision' to True. +# (Uncommoditized amounts ignore the value of `keep_precision', and assume it +# is always True). In this case, Ledger does its best to maintain maximal +# precision by watching how the Amount is used. That is, 1.01 * 1.01 yields a +# precision of 4. This tracking is just a best estimate, however, since +# internally Ledger never uses floating-point values. + +amt = ledger.Amount('$100.12') +mini = ledger.Amount('0.00045') -# There are a few built-in commodities: null, %, h, m and s -assertEqual([u'', u'$', u'%', u'EUR', u'XCD', - u'h', u'm', u's'], sorted(pool.keys())) +assert not amt.keep_precision -for symbol in pool.iterkeys(): pass -for commodity in pool.itervalues(): pass +assertEqual(5, mini.precision) +assertEqual(5, mini.display_precision) # display_precision == precision +assertEqual(2, amt.precision) +assertEqual(2, amt.display_precision) -# jww (2009-11-19): Not working: missing conversion from std::pair -#for symbol, commodity in pool.iteritems(): pass -#for symbol, commodity in pool: pass +mini *= mini +amt *= amt -# This creates a price exchange entry, trading EUR for $0.77 each at the -# current time. -pool.exchange(eur, ledger.Amount('$0.77')) +assertEqual(10, mini.precision) +assertEqual(10, mini.display_precision) +assertEqual(4, amt.precision) +assertEqual(2, amt.display_precision) -# AMOUNTS & BALANCES +# There are several other supported math operations: -# When two amounts are multipied or divided, the result carries the commodity -# of the first term. So, 1 EUR / $0.77 == roughly 1.2987 EUR amt = ledger.Amount('$100.12') market = ((ledger.Amount('1 EUR') / ledger.Amount('$0.77')) * amt) -# An amount's "precision" is a notional thing only. Since Ledger uses -# rational numbers throughout, and only renders to decimal form for printing -# to the user, the meaning of amt.precision should not be relied on as -# meaningful. It only controls how much precision unrounded numbers (those -# for which keep_precision is True, and thus that ignore display_precision) -# are rendered into strings. This is the case, btw, for all uncommoditized -# amounts. -assert not amt.keep_precision -assertEqual(2, amt.precision) -assertEqual(2, amt.display_precision) - -assertEqual('$-100.12', str(amt.negated())) # negate the amount -assertEqual('$0.01', str(amt.inverted())) # reverse NUM/DEM -assertEqual('$100.12', str(amt.rounded())) # round it to display precision -assertEqual('$100.12', str(amt.truncated())) # truncate to display precision -assertEqual('$100.00', str(amt.floored())) # floor it to nearest integral -assertEqual(market, amt.value(eur)) # find present market value -assertEqual('$100.12', str(abs(amt))) # absolute value -assertEqual('$100.12', str(amt)) # render to a string -assertEqual('100.12', amt.quantity_string()) # render quantity to a string -assertEqual('100.12', str(amt.number())) # strip away commodity -assertEqual(1, amt.sign()) # -1, 0 or 1 -assert amt.is_nonzero() # True if display amount nonzero -assert not amt.is_zero() # True if display amount is zero -assert not amt.is_realzero() # True only if value is 0/0 -assert not amt.is_null() # True if uninitialized +assertEqual(market, amt.value(eur)) # find present market value + +assertEqual('$-100.12', str(amt.negated())) # negate the amount +assertEqual('$-100.12', str(- amt)) # negate it more simply +assertEqual('$0.01', str(amt.inverted())) # reverse NUM/DEM +assertEqual('$100.12', str(amt.rounded())) # round it to display precision +assertEqual('$100.12', str(amt.truncated())) # truncate to display precision +assertEqual('$100.00', str(amt.floored())) # floor it to nearest integral +assertEqual('$100.12', str(abs(amt))) # absolute value +assertEqual('$100.12', str(amt)) # render to a string +assertEqual('100.12', amt.quantity_string()) # render quantity to a string +assertEqual('100.12', str(amt.number())) # strip away commodity +assertEqual(1, amt.sign()) # -1, 0 or 1 +assert amt.is_nonzero() # True if display amount nonzero +assert not amt.is_zero() # True if display amount is zero +assert not amt.is_realzero() # True only if value is 0/0 +assert not amt.is_null() # True if uninitialized + +# Amounts can also be converted the standard floats and integers, although +# this is not recommend since it can lose precision. assertEqual(100.12, amt.to_double()) -assert amt.fits_in_long() +assert amt.fits_in_long() # there is no `fits_in_double' assertEqual(100, amt.to_long()) -amt2 = ledger.Amount('$100.12 {140 EUR}') +# Finally, amounts can be annotated to provide additional information about +# "lots" of a given commodity. This example shows $100.12 that was purchased +# on 2009/10/01 for 140 EUR. Lot information can be accessed through via the +# Amount's `annotation' property. You can also strip away lot details to get +# the underlying amount. If you want the total price of any Amount, by +# multiplying by its per-unit lot price, call the `Amount.price' method +# instead of the `Annotation.price' property. + +amt2 = ledger.Amount('$100.12 {140 EUR} [2009/10/01]') assert amt2.has_annotation() -assertEqual(amt, amt2.strip_annotations(ledger.KeepDetails())) +assertEqual(amt, amt2.strip_annotations()) -# jww (2009-11-19): Not working: missing conversion from optional -#assertEqual(ledger.Amount('20 EUR'), amt.annotation.price) +assertEqual(ledger.Amount('140 EUR'), amt2.annotation.price) +assertEqual(ledger.Amount('14016,8 EUR'), amt2.price()) # european amount! +############################################################################### +# # VALUES +# +# As common as Amounts and Balances are, there is a more prevalent numeric +# type you will encounter when generating reports: Value objects. A Value is +# a variadic type that can be any of the following types: +# - Amount +# - Balance +# - boolean +# - integer +# - datetime +# - date +# - string +# - regex +# - sequence +# +# The reason for the variadic type is that it supports dynamic self-promotion. +# For example, it is illegal to add two Amounts of different commodities, but +# it is not illegal to add two Value amounts of different commodities. In the +# former case an exception in raised, but in the latter the Value simply +# promotes itself to a Balance object to make the addition valid. +# +# Values are not used by any of Ledger's data objects (Journal, Transaction, +# Posting or Account), but they are used extensively by value expressions. val = ledger.Value('$100.00') diff --git a/src/py_commodity.cc b/src/py_commodity.cc index dfaf7f5b..c201d370 100644 --- a/src/py_commodity.cc +++ b/src/py_commodity.cc @@ -224,6 +224,14 @@ namespace { return comm.strip_annotations(keep); } + boost::optional py_price(annotation_t& ann) { + return ann.price; + } + boost::optional py_set_price(annotation_t& ann, + const boost::optional& price) { + return ann.price = price; + } + } // unnamed namespace void export_commodity() @@ -295,7 +303,7 @@ void export_commodity() map_value_type_converter(); - scope().attr("commodity_pool") = commodity_pool_t::current_pool; + scope().attr("commodities") = commodity_pool_t::current_pool; scope().attr("COMMODITY_STYLE_DEFAULTS") = COMMODITY_STYLE_DEFAULTS; scope().attr("COMMODITY_STYLE_SUFFIXED") = COMMODITY_STYLE_SUFFIXED; @@ -375,9 +383,7 @@ void export_commodity() .def("drop_flags", &supports_flags<>::drop_flags) #endif - .add_property("price", - make_getter(&annotation_t::price), - make_setter(&annotation_t::price)) + .add_property("price", py_price, py_set_price) .add_property("date", make_getter(&annotation_t::date), make_setter(&annotation_t::date)) -- cgit v1.2.3