GitXplorerGitXplorer
m

tracker

public
35 stars
5 forks
3 issues

Commits

List of commits on branch master.
Unverified
c8c0df500a612f6d1097ad7730a9940bc1cb6bc4

ADD: RST file

mmadisonmay committed 9 years ago
Unverified
4191e7f2932c1a4702644d8b6382816370c5a142

UPDATE: README + setup.py

mmadisonmay committed 9 years ago
Unverified
a6a665917a42202bc943eeda892109c64506ed66

ADD: README

mmadisonmay committed 9 years ago
Unverified
acd9cd4d8796421673912230115a5f73adfa89fa

FIX: formatting and line numbers

mmadisonmay committed 9 years ago
Unverified
b928f576672d203b72278a8c4fdb5086a522e89f

FIX: user experience tweaks

mmadisonmay committed 9 years ago
Unverified
b495efbb96ed1ded00017e204343757c59c6dd36

ADD: CHANGES.txt

mmadisonmay committed 9 years ago

README

The README file for this repository.

tracker

A time machine for debugging pesky stateful errors.

Motivation

It's not always clear what lines of code introduce ugly bugs. By walking through modifications to an object's state step by step, tracker makes it simpler to debug where the flaws in your logic lie.

Installation

Tracker is conveniently available via pip:

pip install tracker

or installable via git clone and setup.py

git clone git@github.com:madisonmay/tracker.git
python setup.py install

To ensure Tracker is properly installed, you can run the unittest suite from the project root:

nosetests -v

Usage

Tracker uses class decorators to specify a list of attributes to monitor as a program executes. Let's walk through an example with a bit of code designed to implement a simple balance sheet and see how tracker might help hunt down a bug in some sample code. See the examples directory for runnable code snippets.

from tracker import tracked

@tracked('balance')
class BalanceSheet(object):

    def __init__(self, balance=0):
        self.balance = balance

    def add_expense(self, value):
        self.balance -= value

    def deposit(self, value):
        self.balance += value

Now let's add some simple erroneous code to demonstrate the usefulness of tracker. Let's imagine for a minute that a second developer working on the balance sheet project misunderstood how the BalanceSheet class was intended to work, added their own syntax for direct modification of the balance attribute, and also included calls to the BalanceSheet's methods. In other words, our second developer is doubling the amount of every expense and deposit.

def add_expense(sheet, value):
    sheet.balance -= value
    sheet.add_expense(value)

def deposit(sheet, value):
    sheet.balance += value
    sheet.deposit(value)

So when the second developer goes to run a routine deposit, they see that the balance is not what they expected.

from balance_sheet import BalanceSheet, add_expense, deposit
sheet = BalanceSheet()
deposit(sheet, 100)
print sheet.balance

Using tracker's replay functionality, we can rewind and see what lines of code changed the sheet's balance and help our second developer uncover their simple mistake.

for snapshot in sheet.replay():
    print snapshot

which outputs:

{'balance': 0}
--------------------------------------------------------------------------------
balance_sheet_debugging.py:3
  2       
  3       sheet = BalanceSheet()
  4       deposit(sheet, 100)

/home/m/Projects/Tracker/examples/balance_sheet.py:8
  7           def __init__(self):
  8               self.balance = 0
  9       


{'balance': 100}
--------------------------------------------------------------------------------
balance_sheet_debugging.py:4
  3       sheet = BalanceSheet()
  4       deposit(sheet, 100)
  5       print sheet.balance

balance_sheet.py:23
  22      def deposit(sheet, value):
  23          sheet.balance += value
  24          sheet.deposit(value)


{'balance': 200}
--------------------------------------------------------------------------------
balance_sheet_debugging.py:4
  3       sheet = BalanceSheet()
  4       deposit(sheet, 100)
  5       print sheet.balance

balance_sheet.py:24
  23          sheet.balance += value
  24          sheet.deposit(value)
  25      

balance_sheet.py:14
  13          def deposit(self, value):
  14              self.balance += value
  15      

Each Snapshot object in sheet.replay() has a state attribute (i.e. {'balance': 200}) and a stack attribute (an array of Frame objects). Each Frame object has a file attribute and a line attribute which point to the line of code which triggered the modification. For dead simple debugging, the __str__ method of the Snapshot object outputs a human readable represenations of those values.

Now we can easily walk through the code that caused the double deposit and follow how the balance attribute changed value over time.

How does it work?

Tracker patches __setattr__ and __setitem__ in order to record changes to variables that you've placed on a watch list. Whenever a change is caught, the inspect library is used to capture the lines of code that caused the modification. Tracker is barely over 100 lines of code long, so I encourage you to read the source for the gritty details.

Contributions

Tracker is a work in progress, so pull requests, critical feedback, github issues, and feedback of any kind is welcome.