Update the PUNTED text
[iron-blogger] / render.py
1 #!/usr/bin/python
2 import yaml
3 from dateutil.parser import parse
4 import datetime
5 import dateutil.tz as tz
6 import sys
7 import os
8 import os.path
9 import subprocess
10 from mako.template import Template
11
12 START = datetime.datetime(2009, 12, 21, 6)
13 HERE  = os.path.dirname(__file__)
14
15 def get_balance(acct):
16     p = subprocess.Popen(['ledger', '-f', os.path.join(HERE,'ledger'),
17                           '-n', 'balance', acct],
18                          stdout=subprocess.PIPE)
19     (out, _) = p.communicate()
20     return float(out.split()[0][1:])
21
22 def get_debts():
23     p = subprocess.Popen(['ledger', '-f', os.path.join(HERE, 'ledger'),
24                           '-n', 'balance', 'Pool:Owed:'],
25                          stdout=subprocess.PIPE)
26     (out, _) = p.communicate()
27     debts = []
28     for line in out.split("\n"):
29         if not line: continue
30         (val, acct) = line.split()
31         user = acct[len("Pool:Owed:"):]
32         val  = float(val[len("$"):])
33         debts.append((user, val))
34     return debts
35
36 def to_week_num(date):
37     return (parse(date, default=START) - START).days / 7
38
39 def parse_skip(rec):
40     spec = rec.get('skip', [])
41     out = []
42     for s in spec:
43         if isinstance(s, list):
44             out.append(map(to_week_num, s))
45         else:
46             out.append(to_week_num(s))
47     return out
48
49 def should_skip(skips, week):
50     for e in skips:
51         if e == week:
52             return True
53         if isinstance(e, list) and e[0] <= week and e[1] > week:
54             return True
55     return False
56
57 def render_template(path, week=None, **kwargs):
58     with open('out/report.yml') as r:
59         report = yaml.safe_load(r)
60
61     with open('bloggers.yml') as f:
62         users = yaml.safe_load(f)
63     if week:
64         week = parse(week, default=START)
65     else:
66         week = START
67
68     week = (week - START).days / 7
69     week_start = START + (week * datetime.timedelta(7))
70     week_end   = START + ((week + 1) * datetime.timedelta(7))
71
72     good = []
73     lame = []
74     skip = []
75     userlist = []
76
77     class User(object):
78         pass
79
80     for (un, rec) in users.items():
81         u = User()
82         u.username = un
83         u.links = rec['links']
84         u.start = rec['start']
85         u.end   = rec.get('end')
86         u.skip  = parse_skip(rec)
87         u.weeks = report.get(un, [])
88
89         userlist.append(u)
90
91     def user_key(u):
92         return (u.start, u.username)
93
94     userlist.sort(key=user_key)
95
96     for u in userlist:
97         user_start = parse(u.start, default=START)
98         if u.end and parse(u.end, default=START) <= week_start:
99             continue
100
101         if should_skip(u.skip, week):
102             pass
103         elif user_start > week_start:
104             skip.append(u)
105         elif len(u.weeks) <= week or not u.weeks[week]:
106             lame.append(u)
107         else:
108             good.append(u)
109
110     debts = get_debts()
111
112     return Template(filename=path, output_encoding='utf-8').render(
113         week=week, week_start=week_start,week_end=week_end,
114         good=good, lame=lame, skip=skip, userlist=userlist,
115         pool=get_balance('Pool'), paid=get_balance('Pool:Paid'),
116         debts=debts, **kwargs)
117
118 if __name__ == '__main__':
119     if len(sys.argv) < 2:
120         print >>sys.stderr, "Usage: %s TEMPLATE [WEEK]"
121         sys.exit(1)
122
123     template = sys.argv[1]
124     week = None
125     if len(sys.argv) > 2: week = sys.argv[2]
126     print render_template(template, week)

Benjamin Mako Hill || Want to submit a patch?