summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rwxr-xr-xmain.py101
1 files changed, 92 insertions, 9 deletions
diff --git a/main.py b/main.py
index 3a5fe7bc..74e65c8a 100755
--- a/main.py
+++ b/main.py
@@ -8,22 +8,37 @@
# License. See the LICENSE file included with the distribution for
# details and disclaimer.
#
-# This script provides a Python front-end to the ledger library, which
-# replicates the functionality of the C++ front-end found in main.cc.
-# It is provided as an alternative to main.cc, or as a starting point
-# for creating custom front-ends based on the Ledger module. See the
-# documentation for API references, and how to use that module.
+# This script provides a Python front-end to the ledger library, and
+# replicates the functionality of the C++ front-end, main.cc. It is
+# provided as an example, and as a starting point for creating custom
+# front-ends based on the Ledger module. See the documentation for an
+# API reference, and how to use this module.
-import sys
import os
+import sys
import string
from ledger import *
+# Create the main journal object, into which all entries will be
+# recorded. Once done, the 'journal' may be iterated to yield those
+# entries, in the same order as which they appeared in the journal
+# file.
+
journal = Journal ()
+# This call registers all of the default command-line options that
+# Ledger supports into the option handling mechanism. Skip this call
+# if you wish to do all of your own processing -- in which case simply
+# modify the 'config' object however you like.
+
add_config_option_handlers ()
+# Process the command-line arguments, test whether caching should be
+# enabled, and then process any option settings from the execution
+# environment. Some historical environment variable names are also
+# supported.
+
args = process_arguments (sys.argv[1:])
config.use_cache = not config.data_file
process_environment (os.environ, "LEDGER_")
@@ -35,10 +50,17 @@ if os.environ.has_key ("PRICE_HIST"):
if os.environ.has_key ("PRICE_EXP"):
process_option ("price-exp", os.environ["PRICE_EXP"])
+# If no argument remain, then no command word was given. Report the
+# default help text and exit.
+
if len (args) == 0:
option_help ()
sys.exit (0)
+# The command word is in the first argument. Canonicalize it to a
+# unique, simple form that the remaining code can use to find out
+# which command was specified.
+
command = args.pop (0);
if command == "balance" or command == "bal" or command == "b":
@@ -55,25 +77,52 @@ else:
print "Unrecognized command:", command
sys.exit (1)
+# Create all the parser objects to be used. They are all registered,
+# so that Ledger will try each one in turn whenever it is presented
+# with a data file. They are attempted in reverse order to their
+# registry. Note that Gnucash parsing is only available if the Ledger
+# module was built with such support (which requires the xmlparse C
+# library).
+
text_parser = TextualParser ()
bin_parser = BinaryParser ()
qif_parser = QifParser ()
+try:
+ gnucush_parser = GnucashParser ()
+except:
+ gnucush_parser = None
register_parser (text_parser)
register_parser (bin_parser)
+if gnucash_parser:
+ register_parser (gnucash_parser)
register_parser (qif_parser)
+# Parse all entries from the user specified locations (found in
+# 'config') into the journal object we created. The two parsers given
+# as explicit arguments indicate: the parser to be used for standard
+# input, and the parser to be used for cache files.
+
parse_ledger_data (journal, text_parser, bin_parser)
+# Now that everything has been correctly parsed (parse_ledger_data
+# would have thrown an exception if not), we can take time to further
+# process the configuration options. This changes the configuration a
+# bit based on previous option settings, the command word, and the
+# remaining arguments.
+
config.process_options(command, args);
+# If the command is "e", use the method journal.derive_entry to create
+# a brand new entry based on the arguments given.
+
new_entry = None
if command == "e":
new_entry = journal.derive_entry (args)
if new_entry is None:
sys.exit (1)
-# Compile the format string
+# Determine the format string to used, based on the command.
if config.format_string:
format = config.format_string
@@ -86,6 +135,11 @@ elif command == "E":
else:
format = config.print_format
+# The following two classes are responsible for outputing transactions
+# and accounts to the user. There are corresponding C++ versions to
+# these, but they rely on I/O streams, which Boost.Python does not
+# provide a conversion layer for.
+
class FormatTransaction (TransactionHandler):
last_entry = None
output = None
@@ -155,11 +209,21 @@ class FormatAccount (AccountHandler):
self.output.write(self.formatter.format(account))
account_xdata (account).dflags |= ACCOUNT_DISPLAYED
+# Set the final transaction handler: for balances and equity reports,
+# it will simply add the value of the transaction to the account's
+# xdata, which is used a bit later to report those totals. For all
+# other reports, the transaction data is sent to the configured output
+# location (default is sys.stdout).
+
if command == "b" or command == "E":
handler = SetAccountValue()
else:
handler = FormatTransaction(format)
+# Chain transaction filters on top of the base handler. Most of these
+# filters customize the output for reporting. None of this is done
+# for balance or equity reports, which don't need it.
+
if not (command == "b" or command == "E"):
if config.display_predicate:
handler = FilterTransactions(handler, config.display_predicate)
@@ -179,25 +243,40 @@ if not (command == "b" or command == "E"):
handler = SubtotalTransactions(handler)
elif config.report_interval:
handler = IntervalTransactions(handler, config.report_interval)
+ handler = SortTransactions(handler, "d")
elif config.days_of_the_week:
handler = DowTransactions(handler)
+# The next two transaction filters are used by all reports.
+
if config.show_related:
handler = RelatedTransactions(handler, config.show_all_related)
if config.predicate:
handler = FilterTransactions(handler, config.predicate)
-if 0:
+# Walk the journal's entries, and pass each entry's transaction to the
+# handler chain established above. And although a journal's entries
+# can be walked using Python, it is significantly faster to do this
+# simple walk in C++, using `walk_entries'.
+
+if 1:
walk_entries (journal, handler)
else:
- # These for loops are equivalent to `walk_entries', but far slower
for entry in journal:
for xact in entry:
handler (xact)
+# Flush the handlers, causing them to output whatever data is still
+# pending.
+
handler.flush ()
+# For the balance and equity reports, the account totals now need to
+# be displayed. This is different from outputting transactions, in
+# that we are now outputting account totals to display a summary of
+# the transactions that were just walked.
+
if command == "b":
acct_formatter = FormatAccount (format, config.display_predicate)
sum_accounts (journal.master)
@@ -217,5 +296,9 @@ if command == "b":
# walk_accounts(*journal->master, acct_formatter, config.sort_string);
# acct_formatter.flush();
+# If the cache is being used, and is dirty, update it now.
+
if config.use_cache and config.cache_dirty and config.cache_file:
write_binary_journal(config.cache_file, journal);
+
+# We're done!