Source code for biweeklybudget.db_event_handlers

"""
The latest version of this package is available at:
<http://github.com/jantman/biweeklybudget>

################################################################################
Copyright 2016 Jason Antman <jason@jasonantman.com> <http://www.jasonantman.com>

    This file is part of biweeklybudget, also known as biweeklybudget.

    biweeklybudget is free software: you can redistribute it and/or modify
    it under the terms of the GNU Affero General Public License as published by
    the Free Software Foundation, either version 3 of the License, or
    (at your option) any later version.

    biweeklybudget is distributed in the hope that it will be useful,
    but WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    GNU Affero General Public License for more details.

    You should have received a copy of the GNU Affero General Public License
    along with biweeklybudget.  If not, see <http://www.gnu.org/licenses/>.

The Copyright and Authors attributions contained herein may not be removed or
otherwise altered, except to add the Author attribution of a contributor to
this work. (Additional Terms pursuant to Section 7b of the AGPL v3)
################################################################################
While not legally required, I sincerely request that anyone who finds
bugs please submit them at <https://github.com/jantman/biweeklybudget> or
to me via email, and that you send any contributions or improvements
either as a pull request on GitHub, or to me via email.
################################################################################

AUTHORS:
Jason Antman <jason@jasonantman.com> <http://www.jasonantman.com>
################################################################################

This is mostly based on http://flask.pocoo.org/docs/0.12/patterns/sqlalchemy/
"""

import logging
from sqlalchemy import event, inspect

from biweeklybudget.models.budget_model import Budget
from biweeklybudget.models.transaction import Transaction

logger = logging.getLogger(__name__)


[docs]def handle_trans_amount_change(**kwargs): """ Handle change of :py:attr:`.Transaction.actual_amount` for existing instances (``id`` is not None). For new instances, we rely on :py:func:`~.handle_new_transaction` called via :py:func:`~.handle_before_flush`. If the Transaction's :py:attr:`~.Transaction.budget` is a :py:class:`~.Budget` with :py:attr:`~.Budget.is_periodic` ``False`` (i.e. a standing budget), update the Budget's :py:attr:`~.Budget.current_balance` for this transaction. See: :py:meth:`sqlalchemy.orm.events.AttributeEvents.set` :param kwargs: keyword arguments :type kwargs: dict """ tgt = kwargs['target'] if tgt.id is None: logger.debug('got Transaction with id None; skipping') return if tgt.budget.is_periodic: logger.debug('got Transaction with periodic budget; skipping') return value = float(kwargs['value']) oldvalue = float(kwargs['oldvalue']) session = inspect(tgt).session diff = oldvalue - value old_budg_curr = float(tgt.budget.current_balance) new_budg = old_budg_curr + diff logger.info( 'Handle Transaction %d against standing budget %d UPDATE; ' 'actual_amount change from %s to %s; update budget current_balance ' 'from %s to %s', tgt.id, tgt.budget.id, oldvalue, value, old_budg_curr, new_budg ) tgt.budget.current_balance = new_budg session.add(tgt.budget)
[docs]def handle_new_transaction(session): """ ``before_flush`` event handler (:py:meth:`sqlalchemy.orm.events.SessionEvents.before_flush`) on the DB session, to handle creation of *new* Transactions. For updates to existing Transactions, we rely on :py:func:`~.handle_trans_amount_change`. If the Transaction's :py:attr:`~.Transaction.budget` is a :py:class:`~.Budget` with :py:attr:`~.Budget.is_periodic` ``False`` (i.e. a standing budget), update the Budget's :py:attr:`~.Budget.current_balance` for this transaction. :param session: current database session :type session: sqlalchemy.orm.session.Session """ updated = 0 for obj in session.new: if not isinstance(obj, Transaction): continue if obj.budget is not None: budg = obj.budget else: budg = session.query(Budget).get(obj.budget_id) if budg.is_periodic: continue logger.debug( 'Session has new Transaction referencing standing budget id=%s', budg.id ) old_amt = float(budg.current_balance) budg.current_balance = old_amt - float(obj.actual_amount) logger.info( 'New transaction (%s) for $%s against standing budget id=%s; ' 'update budget current_balance from %s to %s', obj.description, obj.actual_amount, budg.id, old_amt, budg.current_balance ) session.add(budg) updated += 1 logger.debug( 'Done handling new transactions; updated %d standing budgets', updated )
[docs]def handle_before_flush(session, flush_context, instances): """ Hook into ``before_flush`` (:py:meth:`sqlalchemy.orm.events.SessionEvents.before_flush`) on the DB session, to handle updates that need to be made before persisting data. Currently, this method just calls a number of other methods to handle specific cases: * :py:func:`~.handle_new_transaction` :param session: current database session :type session: sqlalchemy.orm.session.Session :param flush_context: internal SQLAlchemy object :type flush_context: sqlalchemy.orm.session.UOWTransaction :param instances: deprecated """ logger.debug('handle_before_flush handler') handle_new_transaction(session) logger.debug('handle_before_flush done')
[docs]def init_event_listeners(db_session): """ Initialize/register all SQLAlchemy event listeners. See http://docs.sqlalchemy.org/en/latest/orm/events.html :param db_session: the Database Session :type db_session: sqlalchemy.orm.session.Session """ logger.debug('Setting up DB model event listeners') event.listen( Transaction.actual_amount, 'set', handle_trans_amount_change, active_history=True, named=True ) event.listen( db_session, 'before_flush', handle_before_flush )