##// END OF EJS Templates
tiny tab cleanup
tiny tab cleanup

File last commit:

r6223:bab6c8f2 default
r6347:3b42f7ac default
Show More
churn.py
206 lines | 5.4 KiB | text/x-python | PythonLexer
Josef "Jeff" Sipek
[churn] Moved churn extension from hgext to contrib
r3052 # churn.py - create a graph showing who changed the most lines
#
# Copyright 2006 Josef "Jeff" Sipek <jeffpc@josefsipek.net>
#
# This software may be used and distributed according to the terms
# of the GNU General Public License, incorporated herein by reference.
#
#
# Aliases map file format is simple one alias per line in the following
# format:
#
# <alias email> <actual email>
from mercurial.i18n import gettext as _
Joel Rosdahl
Remove unused imports
r6212 from mercurial import mdiff, cmdutil, util, node
Christian Ebert
churn: get current terminal width if possible
r4955 import os, sys
def get_tty_width():
if 'COLUMNS' in os.environ:
try:
return int(os.environ['COLUMNS'])
except ValueError:
pass
try:
Christian Ebert
churn: simplify code to get terminal width
r5419 import termios, array, fcntl
Christian Ebert
churn: get current terminal width if possible
r4955 for dev in (sys.stdout, sys.stdin):
try:
fd = dev.fileno()
if not os.isatty(fd):
continue
Christian Ebert
churn: simplify code to get terminal width
r5419 arri = fcntl.ioctl(fd, termios.TIOCGWINSZ, '\0' * 8)
return array.array('h', arri)[1]
Christian Ebert
churn: get current terminal width if possible
r4955 except ValueError:
pass
except ImportError:
pass
return 80
Josef "Jeff" Sipek
[churn] Moved churn extension from hgext to contrib
r3052
def __gather(ui, repo, node1, node2):
def dirtywork(f, mmap1, mmap2):
lines = 0
to = mmap1 and repo.file(f).read(mmap1[f]) or None
tn = mmap2 and repo.file(f).read(mmap2[f]) or None
Dustin Sallings
Use both the from and to name in mdiff.unidiff....
r5482 diff = mdiff.unidiff(to, "", tn, "", f, f).split("\n")
Josef "Jeff" Sipek
[churn] Moved churn extension from hgext to contrib
r3052
for line in diff:
if not line:
continue # skip EOF
if line.startswith(" "):
continue # context line
if line.startswith("--- ") or line.startswith("+++ "):
continue # begining of diff
if line.startswith("@@ "):
continue # info line
# changed lines
lines += 1
return lines
##
lines = 0
changes = repo.status(node1, node2, None, util.always)[:5]
modified, added, removed, deleted, unknown = changes
who = repo.changelog.read(node2)[1]
Matt Mackall
templater: move email function to util
r5975 who = util.email(who) # get the email of the person
Josef "Jeff" Sipek
[churn] Moved churn extension from hgext to contrib
r3052
mmap1 = repo.manifest.read(repo.changelog.read(node1)[0])
mmap2 = repo.manifest.read(repo.changelog.read(node2)[0])
for f in modified:
lines += dirtywork(f, mmap1, mmap2)
for f in added:
lines += dirtywork(f, None, mmap2)
Thomas Arendsen Hein
Whitespace/Tab cleanup
r3223
Josef "Jeff" Sipek
[churn] Moved churn extension from hgext to contrib
r3052 for f in removed:
lines += dirtywork(f, mmap1, None)
for f in deleted:
lines += dirtywork(f, mmap1, mmap2)
for f in unknown:
lines += dirtywork(f, mmap1, mmap2)
return (who, lines)
def gather_stats(ui, repo, amap, revs=None, progress=False):
stats = {}
Thomas Arendsen Hein
Whitespace/Tab cleanup
r3223
Josef "Jeff" Sipek
[churn] Moved churn extension from hgext to contrib
r3052 cl = repo.changelog
if not revs:
revs = range(0, cl.count())
nr_revs = len(revs)
cur_rev = 0
for rev in revs:
cur_rev += 1 # next revision
node2 = cl.node(rev)
node1 = cl.parents(node2)[0]
if cl.parents(node2)[1] != node.nullid:
ui.note(_('Revision %d is a merge, ignoring...\n') % (rev,))
continue
who, lines = __gather(ui, repo, node1, node2)
# remap the owner if possible
Christian Ebert
Prefer i in d over d.has_key(i)
r5915 if who in amap:
Josef "Jeff" Sipek
[churn] Moved churn extension from hgext to contrib
r3052 ui.note("using '%s' alias for '%s'\n" % (amap[who], who))
who = amap[who]
Christian Ebert
Prefer i in d over d.has_key(i)
r5915 if not who in stats:
Josef "Jeff" Sipek
[churn] Moved churn extension from hgext to contrib
r3052 stats[who] = 0
stats[who] += lines
ui.note("rev %d: %d lines by %s\n" % (rev, lines, who))
if progress:
Matt Mackall
churn: avoid division by zero
r5588 nr_revs = max(nr_revs, 1)
Josef "Jeff" Sipek
[churn] Moved churn extension from hgext to contrib
r3052 if int(100.0*(cur_rev - 1)/nr_revs) < int(100.0*cur_rev/nr_revs):
Thomas Arendsen Hein
Fixed typo: Gnerating -> Generating
r5990 ui.write("\rGenerating stats: %d%%" % (int(100.0*cur_rev/nr_revs),))
Josef "Jeff" Sipek
[churn] Moved churn extension from hgext to contrib
r3052 sys.stdout.flush()
if progress:
Armin Ronacher
added \r for progress counting in churn extension
r5989 ui.write("\r")
Josef "Jeff" Sipek
[churn] Moved churn extension from hgext to contrib
r3052 sys.stdout.flush()
return stats
def churn(ui, repo, **opts):
"Graphs the number of lines changed"
Thomas Arendsen Hein
Whitespace/Tab cleanup
r3223
Josef "Jeff" Sipek
[churn] Moved churn extension from hgext to contrib
r3052 def pad(s, l):
if len(s) < l:
return s + " " * (l-len(s))
return s[0:l]
def graph(n, maximum, width, char):
Matt Mackall
churn: avoid division by zero
r5588 maximum = max(1, maximum)
Josef "Jeff" Sipek
[churn] Moved churn extension from hgext to contrib
r3052 n = int(n * width / float(maximum))
Thomas Arendsen Hein
Whitespace/Tab cleanup
r3223
Josef "Jeff" Sipek
[churn] Moved churn extension from hgext to contrib
r3052 return char * (n)
def get_aliases(f):
aliases = {}
for l in f.readlines():
l = l.strip()
alias, actual = l.split(" ")
aliases[alias] = actual
return aliases
Thomas Arendsen Hein
Whitespace/Tab cleanup
r3223
Josef "Jeff" Sipek
[churn] Moved churn extension from hgext to contrib
r3052 amap = {}
aliases = opts.get('aliases')
if aliases:
try:
f = open(aliases,"r")
except OSError, e:
print "Error: " + e
return
amap = get_aliases(f)
f.close()
Edouard Gomez
Fix revrange() call in the churn contrib
r3792 revs = [int(r) for r in cmdutil.revrange(repo, opts['rev'])]
Josef "Jeff" Sipek
[churn] Moved churn extension from hgext to contrib
r3052 revs.sort()
stats = gather_stats(ui, repo, amap, revs, opts.get('progress'))
# make a list of tuples (name, lines) and sort it in descending order
ordered = stats.items()
Matt Mackall
churn: avoid division by zero
r5588 if not ordered:
return
Stephen Deasey
churn: show comitter email addresses unclipped (bug 1023)...
r6223 ordered.sort(lambda x, y: cmp(y[1], x[1]))
max_churn = ordered[0][1]
Josef "Jeff" Sipek
[churn] Moved churn extension from hgext to contrib
r3052
Stephen Deasey
churn: show comitter email addresses unclipped (bug 1023)...
r6223 tty_width = get_tty_width()
ui.note(_("assuming %i character terminal\n") % tty_width)
tty_width -= 1
Josef "Jeff" Sipek
[churn] Moved churn extension from hgext to contrib
r3052
Stephen Deasey
churn: show comitter email addresses unclipped (bug 1023)...
r6223 max_user_width = max([len(user) for user, churn in ordered])
graph_width = tty_width - max_user_width - 1 - 6 - 2 - 2
for user, churn in ordered:
print "%s %6d %s" % (pad(user, max_user_width),
churn,
graph(churn, max_churn, graph_width, '*'))
Josef "Jeff" Sipek
[churn] Moved churn extension from hgext to contrib
r3052
cmdtable = {
"churn":
(churn,
[('r', 'rev', [], _('limit statistics to the specified revisions')),
('', 'aliases', '', _('file with email aliases')),
('', 'progress', None, _('show progress'))],
'hg churn [-r revision range] [-a file] [--progress]'),
}