graphlog.py
376 lines
| 13.0 KiB
| text/x-python
|
PythonLexer
/ hgext / graphlog.py
Joel Rosdahl
|
r4344 | # ASCII graph log extension for Mercurial | ||
# | ||||
# Copyright 2007 Joel Rosdahl <joel@rosdahl.net> | ||||
Thomas Arendsen Hein
|
r4516 | # | ||
Martin Geisler
|
r8225 | # This software may be used and distributed according to the terms of the | ||
# GNU General Public License version 2, incorporated herein by reference. | ||||
Martin Geisler
|
r8228 | |||
Dirkjan Ochtman
|
r8934 | '''command to view revision graphs from a shell | ||
Alpar Juttner
|
r7426 | |||
This extension adds a --graph option to the incoming, outgoing and log | ||||
Martin Geisler
|
r8667 | commands. When this options is given, an ASCII representation of the | ||
Alpar Juttner
|
r7426 | revision graph is also shown. | ||
''' | ||||
Joel Rosdahl
|
r4344 | |||
Peter Arrenbrecht
|
r8837 | import os, sys | ||
Joel Rosdahl
|
r4344 | from mercurial.cmdutil import revrange, show_changeset | ||
Peter Arrenbrecht
|
r7873 | from mercurial.commands import templateopts | ||
Joel Rosdahl
|
r4344 | from mercurial.i18n import _ | ||
Joel Rosdahl
|
r6212 | from mercurial.node import nullrev | ||
Alpar Juttner
|
r7426 | from mercurial import bundlerepo, changegroup, cmdutil, commands, extensions | ||
Peter Arrenbrecht
|
r8836 | from mercurial import hg, url, util, graphmod | ||
Steve Borho
|
r5938 | |||
Peter Arrenbrecht
|
r8840 | ASCIIDATA = 'ASC' | ||
Peter Arrenbrecht
|
r8838 | def asciiformat(ui, repo, revdag, opts): | ||
"""formats a changelog DAG walk for ASCII output""" | ||||
showparents = [ctx.node() for ctx in repo[None].parents()] | ||||
displayer = show_changeset(ui, repo, opts, buffered=True) | ||||
Peter Arrenbrecht
|
r8840 | for (id, type, ctx, parentids) in revdag: | ||
if type != graphmod.CHANGESET: | ||||
continue | ||||
Peter Arrenbrecht
|
r8838 | displayer.show(ctx) | ||
lines = displayer.hunk.pop(ctx.rev()).split('\n')[:-1] | ||||
char = ctx.node() in showparents and '@' or 'o' | ||||
Peter Arrenbrecht
|
r8840 | yield (id, ASCIIDATA, (char, lines), parentids) | ||
Peter Arrenbrecht
|
r8838 | |||
Peter Arrenbrecht
|
r8839 | def asciiedges(nodes): | ||
Peter Arrenbrecht
|
r8840 | """adds edge info to changelog DAG walk suitable for ascii()""" | ||
Peter Arrenbrecht
|
r7370 | seen = [] | ||
Peter Arrenbrecht
|
r8840 | for node, type, data, parents in nodes: | ||
Peter Arrenbrecht
|
r7370 | if node not in seen: | ||
seen.append(node) | ||||
nodeidx = seen.index(node) | ||||
Steve Borho
|
r5938 | |||
Peter Arrenbrecht
|
r7370 | knownparents = [] | ||
newparents = [] | ||||
Steve Borho
|
r5938 | for parent in parents: | ||
Peter Arrenbrecht
|
r7370 | if parent in seen: | ||
knownparents.append(parent) | ||||
else: | ||||
newparents.append(parent) | ||||
Steve Borho
|
r5938 | |||
Peter Arrenbrecht
|
r7370 | ncols = len(seen) | ||
nextseen = seen[:] | ||||
nextseen[nodeidx:nodeidx + 1] = newparents | ||||
edges = [(nodeidx, nextseen.index(p)) for p in knownparents] | ||||
if len(newparents) > 0: | ||||
edges.append((nodeidx, nodeidx)) | ||||
if len(newparents) > 1: | ||||
edges.append((nodeidx, nodeidx + 1)) | ||||
nmorecols = len(nextseen) - ncols | ||||
seen = nextseen | ||||
Peter Arrenbrecht
|
r8840 | yield (nodeidx, type, data, edges, ncols, nmorecols) | ||
Steve Borho
|
r5938 | |||
Joel Rosdahl
|
r4344 | def fix_long_right_edges(edges): | ||
for (i, (start, end)) in enumerate(edges): | ||||
if end > start: | ||||
edges[i] = (start, end + 1) | ||||
Dirkjan Ochtman
|
r7326 | def get_nodeline_edges_tail( | ||
node_index, p_node_index, n_columns, n_columns_diff, p_diff, fix_tail): | ||||
if fix_tail and n_columns_diff == p_diff and n_columns_diff != 0: | ||||
# Still going in the same non-vertical direction. | ||||
if n_columns_diff == -1: | ||||
start = max(node_index + 1, p_node_index) | ||||
tail = ["|", " "] * (start - node_index - 1) | ||||
tail.extend(["/", " "] * (n_columns - start)) | ||||
return tail | ||||
else: | ||||
return ["\\", " "] * (n_columns - node_index - 1) | ||||
else: | ||||
return ["|", " "] * (n_columns - node_index - 1) | ||||
Joel Rosdahl
|
r4344 | def draw_edges(edges, nodeline, interline): | ||
for (start, end) in edges: | ||||
if start == end + 1: | ||||
interline[2 * end + 1] = "/" | ||||
elif start == end - 1: | ||||
interline[2 * start + 1] = "\\" | ||||
elif start == end: | ||||
interline[2 * start] = "|" | ||||
else: | ||||
nodeline[2 * end] = "+" | ||||
if start > end: | ||||
(start, end) = (end,start) | ||||
for i in range(2 * start + 1, 2 * end): | ||||
if nodeline[i] != "+": | ||||
nodeline[i] = "-" | ||||
def get_padding_line(ni, n_columns, edges): | ||||
line = [] | ||||
line.extend(["|", " "] * ni) | ||||
if (ni, ni - 1) in edges or (ni, ni) in edges: | ||||
# (ni, ni - 1) (ni, ni) | ||||
# | | | | | | | | | ||||
# +---o | | o---+ | ||||
# | | c | | c | | | ||||
# | |/ / | |/ / | ||||
# | | | | | | | ||||
c = "|" | ||||
else: | ||||
c = " " | ||||
line.extend([c, " "]) | ||||
line.extend(["|", " "] * (n_columns - ni - 1)) | ||||
return line | ||||
Peter Arrenbrecht
|
r8839 | def ascii(ui, dag): | ||
"""prints an ASCII graph of the DAG | ||||
Joel Rosdahl
|
r4344 | |||
Peter Arrenbrecht
|
r8839 | dag is a generator that emits tuples with the following elements: | ||
Joel Rosdahl
|
r4344 | |||
Peter Arrenbrecht
|
r7325 | - Column of the current node in the set of ongoing edges. | ||
Peter Arrenbrecht
|
r8840 | - Type indicator of node data == ASCIIDATA. | ||
- Payload: (char, lines): | ||||
- Character to use as node's symbol. | ||||
- List of lines to display as the node's text. | ||||
Peter Arrenbrecht
|
r7325 | - Edges; a list of (col, next_col) indicating the edges between | ||
the current node and its parents. | ||||
- Number of columns (ongoing edges) in the current revision. | ||||
- The difference between the number of columns (ongoing edges) | ||||
in the next revision and the number of columns (ongoing edges) | ||||
in the current revision. That is: -1 means one column removed; | ||||
0 means no columns added or removed; 1 means one column added. | ||||
""" | ||||
Joel Rosdahl
|
r4344 | prev_n_columns_diff = 0 | ||
prev_node_index = 0 | ||||
Peter Arrenbrecht
|
r8840 | for (node_index, type, (node_ch, node_lines), edges, n_columns, n_columns_diff) in dag: | ||
Joel Rosdahl
|
r4344 | |||
Peter Arrenbrecht
|
r7356 | assert -2 < n_columns_diff < 2 | ||
Joel Rosdahl
|
r4344 | if n_columns_diff == -1: | ||
# Transform | ||||
# | ||||
# | | | | | | | ||||
# o | | into o---+ | ||||
# |X / |/ / | ||||
# | | | | | ||||
fix_long_right_edges(edges) | ||||
# add_padding_line says whether to rewrite | ||||
# | ||||
# | | | | | | | | | ||||
# | o---+ into | o---+ | ||||
# | / / | | | # <--- padding line | ||||
# o | | | / / | ||||
# o | | | ||||
Peter Arrenbrecht
|
r7324 | add_padding_line = (len(node_lines) > 2 and | ||
Thomas Arendsen Hein
|
r4633 | n_columns_diff == -1 and | ||
[x for (x, y) in edges if x + 1 < y]) | ||||
Joel Rosdahl
|
r4344 | |||
# fix_nodeline_tail says whether to rewrite | ||||
# | ||||
# | | o | | | | o | | | ||||
# | | |/ / | | |/ / | ||||
# | o | | into | o / / # <--- fixed nodeline tail | ||||
# | |/ / | |/ / | ||||
# o | | o | | | ||||
Peter Arrenbrecht
|
r7324 | fix_nodeline_tail = len(node_lines) <= 2 and not add_padding_line | ||
Joel Rosdahl
|
r4344 | |||
Peter Arrenbrecht
|
r7323 | # nodeline is the line containing the node character (typically o) | ||
Joel Rosdahl
|
r4344 | nodeline = ["|", " "] * node_index | ||
nodeline.extend([node_ch, " "]) | ||||
nodeline.extend( | ||||
get_nodeline_edges_tail( | ||||
node_index, prev_node_index, n_columns, n_columns_diff, | ||||
prev_n_columns_diff, fix_nodeline_tail)) | ||||
# shift_interline is the line containing the non-vertical | ||||
Peter Arrenbrecht
|
r7323 | # edges between this entry and the next | ||
Joel Rosdahl
|
r4344 | shift_interline = ["|", " "] * node_index | ||
if n_columns_diff == -1: | ||||
n_spaces = 1 | ||||
edge_ch = "/" | ||||
elif n_columns_diff == 0: | ||||
n_spaces = 2 | ||||
edge_ch = "|" | ||||
else: | ||||
n_spaces = 3 | ||||
edge_ch = "\\" | ||||
shift_interline.extend(n_spaces * [" "]) | ||||
shift_interline.extend([edge_ch, " "] * (n_columns - node_index - 1)) | ||||
Peter Arrenbrecht
|
r7323 | # draw edges from the current node to its parents | ||
Joel Rosdahl
|
r4344 | draw_edges(edges, nodeline, shift_interline) | ||
Peter Arrenbrecht
|
r7323 | # lines is the list of all graph lines to print | ||
Joel Rosdahl
|
r4344 | lines = [nodeline] | ||
if add_padding_line: | ||||
lines.append(get_padding_line(node_index, n_columns, edges)) | ||||
lines.append(shift_interline) | ||||
Peter Arrenbrecht
|
r7323 | # make sure that there are as many graph lines as there are | ||
# log strings | ||||
Peter Arrenbrecht
|
r7324 | while len(node_lines) < len(lines): | ||
node_lines.append("") | ||||
if len(lines) < len(node_lines): | ||||
Joel Rosdahl
|
r4344 | extra_interline = ["|", " "] * (n_columns + n_columns_diff) | ||
Peter Arrenbrecht
|
r7324 | while len(lines) < len(node_lines): | ||
Joel Rosdahl
|
r4344 | lines.append(extra_interline) | ||
Peter Arrenbrecht
|
r7323 | # print lines | ||
Joel Rosdahl
|
r4344 | indentation_level = max(n_columns, n_columns + n_columns_diff) | ||
Peter Arrenbrecht
|
r7324 | for (line, logstr) in zip(lines, node_lines): | ||
Dirkjan Ochtman
|
r7326 | ln = "%-*s %s" % (2 * indentation_level, "".join(line), logstr) | ||
ui.write(ln.rstrip() + '\n') | ||||
Joel Rosdahl
|
r4344 | |||
Peter Arrenbrecht
|
r7323 | # ... and start over | ||
Joel Rosdahl
|
r4344 | prev_node_index = node_index | ||
prev_n_columns_diff = n_columns_diff | ||||
Dirkjan Ochtman
|
r7326 | def get_revs(repo, rev_opt): | ||
if rev_opt: | ||||
revs = revrange(repo, rev_opt) | ||||
return (max(revs), min(revs)) | ||||
else: | ||||
return (len(repo) - 1, 0) | ||||
Alpar Juttner
|
r7426 | def check_unsupported_flags(opts): | ||
for op in ["follow", "follow_first", "date", "copies", "keyword", "remove", | ||||
"only_merges", "user", "only_branch", "prune", "newest_first", | ||||
"no_merges", "include", "exclude"]: | ||||
if op in opts and opts[op]: | ||||
Dirkjan Ochtman
|
r7713 | raise util.Abort(_("--graph option is incompatible with --%s") % op) | ||
Alpar Juttner
|
r7426 | |||
Peter Arrenbrecht
|
r7325 | def graphlog(ui, repo, path=None, **opts): | ||
"""show revision history alongside an ASCII revision graph | ||||
Print a revision history alongside a revision graph drawn with | ||||
ASCII characters. | ||||
Nodes printed as an @ character are parents of the working | ||||
directory. | ||||
""" | ||||
Alpar Juttner
|
r7426 | check_unsupported_flags(opts) | ||
Dirkjan Ochtman
|
r7715 | limit = cmdutil.loglimit(opts) | ||
Peter Arrenbrecht
|
r7370 | start, stop = get_revs(repo, opts["rev"]) | ||
stop = max(stop, start - limit + 1) | ||||
if start == nullrev: | ||||
Peter Arrenbrecht
|
r7325 | return | ||
Peter Arrenbrecht
|
r7370 | |||
Peter Arrenbrecht
|
r7325 | if path: | ||
Dirkjan Ochtman
|
r7713 | path = util.canonpath(repo.root, os.getcwd(), path) | ||
Peter Arrenbrecht
|
r7370 | if path: # could be reset in canonpath | ||
Peter Arrenbrecht
|
r8836 | revdag = graphmod.filerevs(repo, path, start, stop) | ||
Peter Arrenbrecht
|
r7325 | else: | ||
Peter Arrenbrecht
|
r8836 | revdag = graphmod.revisions(repo, start, stop) | ||
Peter Arrenbrecht
|
r7325 | |||
Peter Arrenbrecht
|
r8839 | fmtdag = asciiformat(ui, repo, revdag, opts) | ||
ascii(ui, asciiedges(fmtdag)) | ||||
Dirkjan Ochtman
|
r7716 | |||
def graphrevs(repo, nodes, opts): | ||||
limit = cmdutil.loglimit(opts) | ||||
Peter Arrenbrecht
|
r8837 | nodes.reverse() | ||
if limit < sys.maxint: | ||||
nodes = nodes[:limit] | ||||
return graphmod.nodes(repo, nodes) | ||||
Dirkjan Ochtman
|
r7716 | |||
def goutgoing(ui, repo, dest=None, **opts): | ||||
"""show the outgoing changesets alongside an ASCII revision graph | ||||
Peter Arrenbrecht
|
r7325 | |||
Dirkjan Ochtman
|
r7716 | Print the outgoing changesets alongside a revision graph drawn with | ||
ASCII characters. | ||||
Alpar Juttner
|
r7426 | |||
Dirkjan Ochtman
|
r7716 | Nodes printed as an @ character are parents of the working | ||
directory. | ||||
Alpar Juttner
|
r7426 | """ | ||
Dirkjan Ochtman
|
r7716 | |||
check_unsupported_flags(opts) | ||||
Alpar Juttner
|
r7426 | dest, revs, checkout = hg.parseurl( | ||
ui.expandpath(dest or 'default-push', dest or 'default'), | ||||
opts.get('rev')) | ||||
if revs: | ||||
revs = [repo.lookup(rev) for rev in revs] | ||||
Matt Mackall
|
r8188 | other = hg.repository(cmdutil.remoteui(ui, opts), dest) | ||
Alpar Juttner
|
r7426 | ui.status(_('comparing with %s\n') % url.hidepassword(dest)) | ||
o = repo.findoutgoing(other, force=opts.get('force')) | ||||
if not o: | ||||
ui.status(_("no changes found\n")) | ||||
return | ||||
Dirkjan Ochtman
|
r7716 | |||
Alpar Juttner
|
r7426 | o = repo.changelog.nodesbetween(o, revs)[0] | ||
Dirkjan Ochtman
|
r7716 | revdag = graphrevs(repo, o, opts) | ||
Peter Arrenbrecht
|
r8839 | fmtdag = asciiformat(ui, repo, revdag, opts) | ||
ascii(ui, asciiedges(fmtdag)) | ||||
Alpar Juttner
|
r7426 | |||
def gincoming(ui, repo, source="default", **opts): | ||||
"""show the incoming changesets alongside an ASCII revision graph | ||||
Print the incoming changesets alongside a revision graph drawn with | ||||
ASCII characters. | ||||
Nodes printed as an @ character are parents of the working | ||||
directory. | ||||
""" | ||||
check_unsupported_flags(opts) | ||||
source, revs, checkout = hg.parseurl(ui.expandpath(source), opts.get('rev')) | ||||
Matt Mackall
|
r8188 | other = hg.repository(cmdutil.remoteui(repo, opts), source) | ||
Alpar Juttner
|
r7426 | ui.status(_('comparing with %s\n') % url.hidepassword(source)) | ||
if revs: | ||||
revs = [other.lookup(rev) for rev in revs] | ||||
incoming = repo.findincoming(other, heads=revs, force=opts["force"]) | ||||
if not incoming: | ||||
try: | ||||
os.unlink(opts["bundle"]) | ||||
except: | ||||
pass | ||||
ui.status(_("no changes found\n")) | ||||
return | ||||
cleanup = None | ||||
try: | ||||
Dirkjan Ochtman
|
r7716 | |||
Alpar Juttner
|
r7426 | fname = opts["bundle"] | ||
if fname or not other.local(): | ||||
# create a bundle (uncompressed if other repo is not local) | ||||
if revs is None: | ||||
cg = other.changegroup(incoming, "incoming") | ||||
else: | ||||
cg = other.changegroupsubset(incoming, revs, 'incoming') | ||||
bundletype = other.local() and "HG10BZ" or "HG10UN" | ||||
fname = cleanup = changegroup.writebundle(cg, fname, bundletype) | ||||
# keep written bundle? | ||||
if opts["bundle"]: | ||||
cleanup = None | ||||
if not other.local(): | ||||
# use the created uncompressed bundlerepo | ||||
other = bundlerepo.bundlerepository(ui, repo.root, fname) | ||||
chlist = other.changelog.nodesbetween(incoming, revs)[0] | ||||
Dirkjan Ochtman
|
r7716 | revdag = graphrevs(other, chlist, opts) | ||
Peter Arrenbrecht
|
r8839 | fmtdag = asciiformat(ui, repo, revdag, opts) | ||
ascii(ui, asciiedges(fmtdag)) | ||||
Alpar Juttner
|
r7426 | |||
finally: | ||||
if hasattr(other, 'close'): | ||||
other.close() | ||||
if cleanup: | ||||
os.unlink(cleanup) | ||||
def uisetup(ui): | ||||
'''Initialize the extension.''' | ||||
_wrapcmd(ui, 'log', commands.table, graphlog) | ||||
_wrapcmd(ui, 'incoming', commands.table, gincoming) | ||||
_wrapcmd(ui, 'outgoing', commands.table, goutgoing) | ||||
def _wrapcmd(ui, cmd, table, wrapfn): | ||||
'''wrap the command''' | ||||
def graph(orig, *args, **kwargs): | ||||
if kwargs['graph']: | ||||
return wrapfn(*args, **kwargs) | ||||
return orig(*args, **kwargs) | ||||
entry = extensions.wrapcommand(table, cmd, graph) | ||||
Jim Correia
|
r7763 | entry[1].append(('G', 'graph', None, _("show the revision DAG"))) | ||
Alpar Juttner
|
r7426 | |||
Joel Rosdahl
|
r4344 | cmdtable = { | ||
"glog": | ||||
Thomas Arendsen Hein
|
r4730 | (graphlog, | ||
[('l', 'limit', '', _('limit number of changes displayed')), | ||||
('p', 'patch', False, _('show patch')), | ||||
('r', 'rev', [], _('show the specified revision or range')), | ||||
Thomas Arendsen Hein
|
r6192 | ] + templateopts, | ||
Thomas Arendsen Hein
|
r5942 | _('hg glog [OPTION]... [FILE]')), | ||
Joel Rosdahl
|
r4344 | } | ||