##// END OF EJS Templates
churn: use genexps now that we dropped 2.3 compatibility
Nicolas Dumazet -
r9390:637f2726 default
parent child Browse files
Show More
@@ -1,177 +1,177
1 1 # churn.py - create a graph of revisions count grouped by template
2 2 #
3 3 # Copyright 2006 Josef "Jeff" Sipek <jeffpc@josefsipek.net>
4 4 # Copyright 2008 Alexander Solovyov <piranha@piranha.org.ua>
5 5 #
6 6 # This software may be used and distributed according to the terms of the
7 7 # GNU General Public License version 2, incorporated herein by reference.
8 8
9 9 '''command to display statistics about repository history'''
10 10
11 11 from mercurial.i18n import _
12 12 from mercurial import patch, cmdutil, util, templater
13 13 import sys, os
14 14 import time, datetime
15 15
16 16 def maketemplater(ui, repo, tmpl):
17 17 tmpl = templater.parsestring(tmpl, quoted=False)
18 18 try:
19 19 t = cmdutil.changeset_templater(ui, repo, False, None, None, False)
20 20 except SyntaxError, inst:
21 21 raise util.Abort(inst.args[0])
22 22 t.use_template(tmpl)
23 23 return t
24 24
25 25 def changedlines(ui, repo, ctx1, ctx2, fns):
26 26 lines = 0
27 27 fmatch = cmdutil.matchfiles(repo, fns)
28 28 diff = ''.join(patch.diff(repo, ctx1.node(), ctx2.node(), fmatch))
29 29 for l in diff.split('\n'):
30 30 if (l.startswith("+") and not l.startswith("+++ ") or
31 31 l.startswith("-") and not l.startswith("--- ")):
32 32 lines += 1
33 33 return lines
34 34
35 35 def countrate(ui, repo, amap, *pats, **opts):
36 36 """Calculate stats"""
37 37 if opts.get('dateformat'):
38 38 def getkey(ctx):
39 39 t, tz = ctx.date()
40 40 date = datetime.datetime(*time.gmtime(float(t) - tz)[:6])
41 41 return date.strftime(opts['dateformat'])
42 42 else:
43 43 tmpl = opts.get('template', '{author|email}')
44 44 tmpl = maketemplater(ui, repo, tmpl)
45 45 def getkey(ctx):
46 46 ui.pushbuffer()
47 47 tmpl.show(ctx)
48 48 return ui.popbuffer()
49 49
50 50 count = pct = 0
51 51 rate = {}
52 52 df = False
53 53 if opts.get('date'):
54 54 df = util.matchdate(opts['date'])
55 55
56 56 get = util.cachefunc(lambda r: repo[r])
57 57 changeiter, matchfn = cmdutil.walkchangerevs(ui, repo, pats, get, opts)
58 58 for st, rev, fns in changeiter:
59 59
60 60 if not st == 'add':
61 61 continue
62 62
63 63 ctx = get(rev)
64 64 if df and not df(ctx.date()[0]): # doesn't match date format
65 65 continue
66 66
67 67 key = getkey(ctx)
68 68 key = amap.get(key, key) # alias remap
69 69 if opts.get('changesets'):
70 70 rate[key] = rate.get(key, 0) + 1
71 71 else:
72 72 parents = ctx.parents()
73 73 if len(parents) > 1:
74 74 ui.note(_('Revision %d is a merge, ignoring...\n') % (rev,))
75 75 continue
76 76
77 77 ctx1 = parents[0]
78 78 lines = changedlines(ui, repo, ctx1, ctx, fns)
79 79 rate[key] = rate.get(key, 0) + lines
80 80
81 81 if opts.get('progress'):
82 82 count += 1
83 83 newpct = int(100.0 * count / max(len(repo), 1))
84 84 if pct < newpct:
85 85 pct = newpct
86 86 ui.write("\r" + _("generating stats: %d%%") % pct)
87 87 sys.stdout.flush()
88 88
89 89 if opts.get('progress'):
90 90 ui.write("\r")
91 91 sys.stdout.flush()
92 92
93 93 return rate
94 94
95 95
96 96 def churn(ui, repo, *pats, **opts):
97 97 '''histogram of changes to the repository
98 98
99 99 This command will display a histogram representing the number
100 100 of changed lines or revisions, grouped according to the given
101 101 template. The default template will group changes by author.
102 102 The --dateformat option may be used to group the results by
103 103 date instead.
104 104
105 105 Statistics are based on the number of changed lines, or
106 106 alternatively the number of matching revisions if the
107 107 --changesets option is specified.
108 108
109 109 Examples::
110 110
111 111 # display count of changed lines for every committer
112 112 hg churn -t '{author|email}'
113 113
114 114 # display daily activity graph
115 115 hg churn -f '%H' -s -c
116 116
117 117 # display activity of developers by month
118 118 hg churn -f '%Y-%m' -s -c
119 119
120 120 # display count of lines changed in every year
121 121 hg churn -f '%Y' -s
122 122
123 123 It is possible to map alternate email addresses to a main address
124 124 by providing a file using the following format::
125 125
126 126 <alias email> <actual email>
127 127
128 128 Such a file may be specified with the --aliases option, otherwise
129 129 a .hgchurn file will be looked for in the working directory root.
130 130 '''
131 131 def pad(s, l):
132 132 return (s + " " * l)[:l]
133 133
134 134 amap = {}
135 135 aliases = opts.get('aliases')
136 136 if not aliases and os.path.exists(repo.wjoin('.hgchurn')):
137 137 aliases = repo.wjoin('.hgchurn')
138 138 if aliases:
139 139 for l in open(aliases, "r"):
140 140 l = l.strip()
141 141 alias, actual = l.split()
142 142 amap[alias] = actual
143 143
144 144 rate = countrate(ui, repo, amap, *pats, **opts).items()
145 145 if not rate:
146 146 return
147 147
148 148 sortkey = ((not opts.get('sort')) and (lambda x: -x[1]) or None)
149 149 rate.sort(key=sortkey)
150 150
151 151 # Be careful not to have a zero maxcount (issue833)
152 maxcount = float(max([v for k, v in rate])) or 1.0
153 maxname = max([len(k) for k, v in rate])
152 maxcount = float(max(v for k, v in rate)) or 1.0
153 maxname = max(len(k) for k, v in rate)
154 154
155 155 ttywidth = util.termwidth()
156 156 ui.debug(_("assuming %i character terminal\n") % ttywidth)
157 157 width = ttywidth - maxname - 2 - 6 - 2 - 2
158 158
159 159 for date, count in rate:
160 160 print "%s %6d %s" % (pad(date, maxname), count,
161 161 "*" * int(count * width / maxcount))
162 162
163 163
164 164 cmdtable = {
165 165 "churn":
166 166 (churn,
167 167 [('r', 'rev', [], _('count rate for the specified revision or range')),
168 168 ('d', 'date', '', _('count rate for revisions matching date spec')),
169 169 ('t', 'template', '{author|email}', _('template to group changesets')),
170 170 ('f', 'dateformat', '',
171 171 _('strftime-compatible format for grouping by date')),
172 172 ('c', 'changesets', False, _('count rate by number of changesets')),
173 173 ('s', 'sort', False, _('sort by key (default: sort by count)')),
174 174 ('', 'aliases', '', _('file with email aliases')),
175 175 ('', 'progress', None, _('show progress'))],
176 176 _("hg churn [-d DATE] [-r REV] [--aliases FILE] [--progress] [FILE]")),
177 177 }
General Comments 0
You need to be logged in to leave comments. Login now