graphlog.py
588 lines
| 21.7 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 | ||
Matt Mackall
|
r10263 | # GNU General Public License version 2 or any later version. | ||
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
|
r9259 | commands. When this options is given, an ASCII representation of the | ||
revision graph is also shown. | ||||
Alpar Juttner
|
r7426 | ''' | ||
Joel Rosdahl
|
r4344 | |||
Matt Mackall
|
r14319 | from mercurial.cmdutil import show_changeset | ||
Joel Rosdahl
|
r4344 | from mercurial.i18n import _ | ||
Joel Rosdahl
|
r6212 | from mercurial.node import nullrev | ||
Matt Mackall
|
r14319 | from mercurial import cmdutil, commands, extensions, scmutil | ||
Patrick Mezard
|
r16412 | from mercurial import hg, util, graphmod, templatekw, revset | ||
Steve Borho
|
r5938 | |||
Adrian Buehlmann
|
r14311 | cmdtable = {} | ||
command = cmdutil.command(cmdtable) | ||||
Augie Fackler
|
r16743 | testedwith = 'internal' | ||
Adrian Buehlmann
|
r14311 | |||
Peter Arrenbrecht
|
r8840 | ASCIIDATA = 'ASC' | ||
Patrick Mezard
|
r14130 | def asciiedges(type, char, lines, seen, rev, parents): | ||
Peter Arrenbrecht
|
r8840 | """adds edge info to changelog DAG walk suitable for ascii()""" | ||
Dirkjan Ochtman
|
r9369 | if rev not in seen: | ||
seen.append(rev) | ||||
nodeidx = seen.index(rev) | ||||
Steve Borho
|
r5938 | |||
Dirkjan Ochtman
|
r9369 | knownparents = [] | ||
newparents = [] | ||||
for parent in parents: | ||||
if parent in seen: | ||||
knownparents.append(parent) | ||||
else: | ||||
newparents.append(parent) | ||||
Steve Borho
|
r5938 | |||
Dirkjan Ochtman
|
r9369 | ncols = len(seen) | ||
Patrick Mezard
|
r14130 | nextseen = seen[:] | ||
nextseen[nodeidx:nodeidx + 1] = newparents | ||||
edges = [(nodeidx, nextseen.index(p)) for p in knownparents] | ||||
while len(newparents) > 2: | ||||
# ascii() only knows how to add or remove a single column between two | ||||
# calls. Nodes with more than two parents break this constraint so we | ||||
# introduce intermediate expansion lines to grow the active node list | ||||
# slowly. | ||||
edges.append((nodeidx, nodeidx)) | ||||
edges.append((nodeidx, nodeidx + 1)) | ||||
nmorecols = 1 | ||||
yield (type, char, lines, (nodeidx, edges, ncols, nmorecols)) | ||||
char = '\\' | ||||
lines = [] | ||||
nodeidx += 1 | ||||
ncols += 1 | ||||
edges = [] | ||||
del newparents[0] | ||||
Peter Arrenbrecht
|
r7370 | |||
Dirkjan Ochtman
|
r9369 | if len(newparents) > 0: | ||
edges.append((nodeidx, nodeidx)) | ||||
if len(newparents) > 1: | ||||
edges.append((nodeidx, nodeidx + 1)) | ||||
Patrick Mezard
|
r14130 | nmorecols = len(nextseen) - ncols | ||
seen[:] = nextseen | ||||
yield (type, char, lines, (nodeidx, 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: | ||||
Matt Mackall
|
r15032 | if 2 * end >= len(nodeline): | ||
continue | ||||
Joel Rosdahl
|
r4344 | nodeline[2 * end] = "+" | ||
if start > end: | ||||
Martin Geisler
|
r9198 | (start, end) = (end, start) | ||
Joel Rosdahl
|
r4344 | 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
|
r9631 | def asciistate(): | ||
"""returns the initial value for the "state" argument to ascii()""" | ||||
return [0, 0] | ||||
def ascii(ui, state, type, char, text, coldata): | ||||
Peter Arrenbrecht
|
r8839 | """prints an ASCII graph of the DAG | ||
Joel Rosdahl
|
r4344 | |||
Dirkjan Ochtman
|
r9371 | takes the following arguments (one call per node in the graph): | ||
Joel Rosdahl
|
r4344 | |||
Dirkjan Ochtman
|
r9371 | - ui to write to | ||
Peter Arrenbrecht
|
r9631 | - Somewhere to keep the needed state in (init to asciistate()) | ||
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 | |||
Dirkjan Ochtman
|
r9371 | idx, edges, ncols, coldiff = coldata | ||
assert -2 < coldiff < 2 | ||||
if coldiff == -1: | ||||
# Transform | ||||
Joel Rosdahl
|
r4344 | # | ||
Dirkjan Ochtman
|
r9371 | # | | | | | | | ||
# o | | into o---+ | ||||
# |X / |/ / | ||||
# | | | | | ||||
fix_long_right_edges(edges) | ||||
# add_padding_line says whether to rewrite | ||||
# | ||||
# | | | | | | | | | ||||
# | o---+ into | o---+ | ||||
# | / / | | | # <--- padding line | ||||
# o | | | / / | ||||
# o | | | ||||
add_padding_line = (len(text) > 2 and coldiff == -1 and | ||||
[x for (x, y) in edges if x + 1 < y]) | ||||
Joel Rosdahl
|
r4344 | |||
Dirkjan Ochtman
|
r9371 | # fix_nodeline_tail says whether to rewrite | ||
# | ||||
# | | o | | | | o | | | ||||
# | | |/ / | | |/ / | ||||
# | o | | into | o / / # <--- fixed nodeline tail | ||||
# | |/ / | |/ / | ||||
# o | | o | | | ||||
fix_nodeline_tail = len(text) <= 2 and not add_padding_line | ||||
Joel Rosdahl
|
r4344 | |||
Dirkjan Ochtman
|
r9371 | # nodeline is the line containing the node character (typically o) | ||
nodeline = ["|", " "] * idx | ||||
nodeline.extend([char, " "]) | ||||
Joel Rosdahl
|
r4344 | |||
Dirkjan Ochtman
|
r9371 | nodeline.extend( | ||
Peter Arrenbrecht
|
r9631 | get_nodeline_edges_tail(idx, state[1], ncols, coldiff, | ||
state[0], fix_nodeline_tail)) | ||||
Joel Rosdahl
|
r4344 | |||
Dirkjan Ochtman
|
r9371 | # shift_interline is the line containing the non-vertical | ||
# edges between this entry and the next | ||||
shift_interline = ["|", " "] * idx | ||||
if coldiff == -1: | ||||
n_spaces = 1 | ||||
edge_ch = "/" | ||||
elif coldiff == 0: | ||||
n_spaces = 2 | ||||
edge_ch = "|" | ||||
else: | ||||
n_spaces = 3 | ||||
edge_ch = "\\" | ||||
shift_interline.extend(n_spaces * [" "]) | ||||
shift_interline.extend([edge_ch, " "] * (ncols - idx - 1)) | ||||
Joel Rosdahl
|
r4344 | |||
Dirkjan Ochtman
|
r9371 | # draw edges from the current node to its parents | ||
draw_edges(edges, nodeline, shift_interline) | ||||
Joel Rosdahl
|
r4344 | |||
Dirkjan Ochtman
|
r9371 | # lines is the list of all graph lines to print | ||
lines = [nodeline] | ||||
if add_padding_line: | ||||
lines.append(get_padding_line(idx, ncols, edges)) | ||||
lines.append(shift_interline) | ||||
Joel Rosdahl
|
r4344 | |||
Dirkjan Ochtman
|
r9371 | # make sure that there are as many graph lines as there are | ||
# log strings | ||||
while len(text) < len(lines): | ||||
text.append("") | ||||
if len(lines) < len(text): | ||||
extra_interline = ["|", " "] * (ncols + coldiff) | ||||
while len(lines) < len(text): | ||||
lines.append(extra_interline) | ||||
Joel Rosdahl
|
r4344 | |||
Dirkjan Ochtman
|
r9371 | # print lines | ||
indentation_level = max(ncols, ncols + coldiff) | ||||
for (line, logstr) in zip(lines, text): | ||||
ln = "%-*s %s" % (2 * indentation_level, "".join(line), logstr) | ||||
ui.write(ln.rstrip() + '\n') | ||||
Joel Rosdahl
|
r4344 | |||
Dirkjan Ochtman
|
r9371 | # ... and start over | ||
Peter Arrenbrecht
|
r9631 | state[0] = coldiff | ||
state[1] = idx | ||||
Joel Rosdahl
|
r4344 | |||
Dirkjan Ochtman
|
r7326 | def get_revs(repo, rev_opt): | ||
if rev_opt: | ||||
Matt Mackall
|
r14319 | revs = scmutil.revrange(repo, rev_opt) | ||
Eric Eisner
|
r11448 | if len(revs) == 0: | ||
return (nullrev, nullrev) | ||||
Dirkjan Ochtman
|
r7326 | return (max(revs), min(revs)) | ||
else: | ||||
return (len(repo) - 1, 0) | ||||
Patrick Mezard
|
r14086 | def check_unsupported_flags(pats, opts): | ||
Patrick Mezard
|
r16180 | for op in ["newest_first"]: | ||
Alpar Juttner
|
r7426 | if op in opts and opts[op]: | ||
Alexander Solovyov
|
r14043 | raise util.Abort(_("-G/--graph option is incompatible with --%s") | ||
Greg Ward
|
r10097 | % op.replace("_", "-")) | ||
Alpar Juttner
|
r7426 | |||
Patrick Mezard
|
r16412 | def _makefilematcher(repo, pats, followfirst): | ||
Patrick Mezard
|
r16186 | # When displaying a revision with --patch --follow FILE, we have | ||
# to know which file of the revision must be diffed. With | ||||
# --follow, we want the names of the ancestors of FILE in the | ||||
# revision, stored in "fcache". "fcache" is populated by | ||||
# reproducing the graph traversal already done by --follow revset | ||||
# and relating linkrevs to file names (which is not "correct" but | ||||
# good enough). | ||||
fcache = {} | ||||
fcacheready = [False] | ||||
pctx = repo['.'] | ||||
wctx = repo[None] | ||||
def populate(): | ||||
for fn in pats: | ||||
for i in ((pctx[fn],), pctx[fn].ancestors(followfirst=followfirst)): | ||||
for c in i: | ||||
fcache.setdefault(c.linkrev(), set()).add(c.path()) | ||||
def filematcher(rev): | ||||
if not fcacheready[0]: | ||||
# Lazy initialization | ||||
fcacheready[0] = True | ||||
populate() | ||||
return scmutil.match(wctx, fcache.get(rev, []), default='path') | ||||
return filematcher | ||||
Patrick Mezard
|
r16405 | def _makelogrevset(repo, pats, opts, revs): | ||
Patrick Mezard
|
r16186 | """Return (expr, filematcher) where expr is a revset string built | ||
Patrick Mezard
|
r16405 | from log options and file patterns or None. If --stat or --patch | ||
are not passed filematcher is None. Otherwise it is a callable | ||||
taking a revision number and returning a match objects filtering | ||||
the files to be detailed when displaying the revision. | ||||
Alexander Solovyov
|
r14043 | """ | ||
Patrick Mezard
|
r14085 | opt2revset = { | ||
Patrick Mezard
|
r16174 | 'no_merges': ('not merge()', None), | ||
'only_merges': ('merge()', None), | ||||
Patrick Mezard
|
r16408 | '_ancestors': ('ancestors(%(val)s)', None), | ||
Patrick Mezard
|
r16409 | '_fancestors': ('_firstancestors(%(val)s)', None), | ||
Patrick Mezard
|
r16408 | '_descendants': ('descendants(%(val)s)', None), | ||
Patrick Mezard
|
r16409 | '_fdescendants': ('_firstdescendants(%(val)s)', None), | ||
Patrick Mezard
|
r16316 | '_matchfiles': ('_matchfiles(%(val)s)', None), | ||
Patrick Mezard
|
r16174 | 'date': ('date(%(val)r)', None), | ||
'branch': ('branch(%(val)r)', ' or '), | ||||
'_patslog': ('filelog(%(val)r)', ' or '), | ||||
'_patsfollow': ('follow(%(val)r)', ' or '), | ||||
'_patsfollowfirst': ('_followfirst(%(val)r)', ' or '), | ||||
'keyword': ('keyword(%(val)r)', ' or '), | ||||
'prune': ('not (%(val)r or ancestors(%(val)r))', ' and '), | ||||
'user': ('user(%(val)r)', ' or '), | ||||
Patrick Mezard
|
r14085 | } | ||
Patrick Mezard
|
r16157 | |||
Patrick Mezard
|
r16159 | opts = dict(opts) | ||
Patrick Mezard
|
r16405 | # follow or not follow? | ||
Patrick Mezard
|
r16174 | follow = opts.get('follow') or opts.get('follow_first') | ||
Patrick Mezard
|
r16433 | followfirst = opts.get('follow_first') and 1 or 0 | ||
Patrick Mezard
|
r16408 | # --follow with FILE behaviour depends on revs... | ||
startrev = revs[0] | ||||
Patrick Mezard
|
r16433 | followdescendants = (len(revs) > 1 and revs[0] < revs[1]) and 1 or 0 | ||
Patrick Mezard
|
r16405 | |||
# branch and only_branch are really aliases and must be handled at | ||||
# the same time | ||||
opts['branch'] = opts.get('branch', []) + opts.get('only_branch', []) | ||||
Patrick Mezard
|
r16407 | opts['branch'] = [repo.lookupbranch(b) for b in opts['branch']] | ||
Patrick Mezard
|
r16171 | # pats/include/exclude are passed to match.match() directly in | ||
# _matchfile() revset but walkchangerevs() builds its matcher with | ||||
# scmutil.match(). The difference is input pats are globbed on | ||||
# platforms without shell expansion (windows). | ||||
Patrick Mezard
|
r16173 | pctx = repo[None] | ||
match, pats = scmutil.matchandpats(pctx, pats, opts) | ||||
Patrick Mezard
|
r16160 | slowpath = match.anypats() or (match.files() and opts.get('removed')) | ||
if not slowpath: | ||||
for f in match.files(): | ||||
Patrick Mezard
|
r16173 | if follow and f not in pctx: | ||
raise util.Abort(_('cannot follow file not in parent ' | ||||
'revision: "%s"') % f) | ||||
Patrick Mezard
|
r16160 | filelog = repo.file(f) | ||
if not len(filelog): | ||||
# A zero count may be a directory or deleted file, so | ||||
# try to find matching entries on the slow path. | ||||
Patrick Mezard
|
r16173 | if follow: | ||
raise util.Abort( | ||||
_('cannot follow nonexistent file: "%s"') % f) | ||||
Patrick Mezard
|
r16160 | slowpath = True | ||
if slowpath: | ||||
Patrick Mezard
|
r16161 | # See cmdutil.walkchangerevs() slow path. | ||
# | ||||
Patrick Mezard
|
r16173 | if follow: | ||
raise util.Abort(_('can only follow copies/renames for explicit ' | ||||
'filenames')) | ||||
Patrick Mezard
|
r16161 | # pats/include/exclude cannot be represented as separate | ||
# revset expressions as their filtering logic applies at file | ||||
# level. For instance "-I a -X a" matches a revision touching | ||||
Patrick Mezard
|
r16181 | # "a" and "b" while "file(a) and not file(b)" does | ||
# not. Besides, filesets are evaluated against the working | ||||
# directory. | ||||
Patrick Mezard
|
r16411 | matchargs = ['r:', 'd:relpath'] | ||
Patrick Mezard
|
r16161 | for p in pats: | ||
matchargs.append('p:' + p) | ||||
for p in opts.get('include', []): | ||||
matchargs.append('i:' + p) | ||||
for p in opts.get('exclude', []): | ||||
matchargs.append('x:' + p) | ||||
matchargs = ','.join(('%r' % p) for p in matchargs) | ||||
Patrick Mezard
|
r16316 | opts['_matchfiles'] = matchargs | ||
Patrick Mezard
|
r16160 | else: | ||
Patrick Mezard
|
r16173 | if follow: | ||
Patrick Mezard
|
r16433 | fpats = ('_patsfollow', '_patsfollowfirst') | ||
fnopats = (('_ancestors', '_fancestors'), | ||||
('_descendants', '_fdescendants')) | ||||
if pats: | ||||
Patrick Mezard
|
r16434 | # follow() revset inteprets its file argument as a | ||
# manifest entry, so use match.files(), not pats. | ||||
opts[fpats[followfirst]] = list(match.files()) | ||||
Patrick Mezard
|
r16173 | else: | ||
Patrick Mezard
|
r16433 | opts[fnopats[followdescendants][followfirst]] = str(startrev) | ||
Patrick Mezard
|
r16173 | else: | ||
opts['_patslog'] = list(pats) | ||||
Patrick Mezard
|
r16157 | |||
Patrick Mezard
|
r16186 | filematcher = None | ||
if opts.get('patch') or opts.get('stat'): | ||||
if follow: | ||||
Patrick Mezard
|
r16412 | filematcher = _makefilematcher(repo, pats, followfirst) | ||
Patrick Mezard
|
r16186 | else: | ||
filematcher = lambda rev: match | ||||
Patrick Mezard
|
r16412 | expr = [] | ||
Alexander Solovyov
|
r14043 | for op, val in opts.iteritems(): | ||
if not val: | ||||
continue | ||||
Patrick Mezard
|
r14085 | if op not in opt2revset: | ||
continue | ||||
Patrick Mezard
|
r16147 | revop, andor = opt2revset[op] | ||
Patrick Mezard
|
r16158 | if '%(val)' not in revop: | ||
Patrick Mezard
|
r16412 | expr.append(revop) | ||
Patrick Mezard
|
r14085 | else: | ||
Patrick Mezard
|
r16147 | if not isinstance(val, list): | ||
Patrick Mezard
|
r16412 | e = revop % {'val': val} | ||
Patrick Mezard
|
r16147 | else: | ||
Patrick Mezard
|
r16412 | e = '(' + andor.join((revop % {'val': v}) for v in val) + ')' | ||
expr.append(e) | ||||
Alexander Solovyov
|
r14043 | |||
Patrick Mezard
|
r16412 | if expr: | ||
expr = '(' + ' and '.join(expr) + ')' | ||||
Patrick Mezard
|
r14132 | else: | ||
Patrick Mezard
|
r16412 | expr = None | ||
return expr, filematcher | ||||
Alexander Solovyov
|
r14043 | |||
Patrick Mezard
|
r16405 | def getlogrevs(repo, pats, opts): | ||
Patrick Mezard
|
r16777 | """Return (revs, expr, filematcher) where revs is an iterable of | ||
Patrick Mezard
|
r16405 | revision numbers, expr is a revset string built from log options | ||
and file patterns or None, and used to filter 'revs'. If --stat or | ||||
--patch are not passed filematcher is None. Otherwise it is a | ||||
callable taking a revision number and returning a match objects | ||||
filtering the files to be detailed when displaying the revision. | ||||
""" | ||||
Patrick Mezard
|
r16777 | def increasingrevs(repo, revs, matcher): | ||
# The sorted input rev sequence is chopped in sub-sequences | ||||
# which are sorted in ascending order and passed to the | ||||
# matcher. The filtered revs are sorted again as they were in | ||||
# the original sub-sequence. This achieve several things: | ||||
# | ||||
# - getlogrevs() now returns a generator which behaviour is | ||||
# adapted to log need. First results come fast, last ones | ||||
# are batched for performances. | ||||
# | ||||
# - revset matchers often operate faster on revision in | ||||
# changelog order, because most filters deal with the | ||||
# changelog. | ||||
# | ||||
# - revset matchers can reorder revisions. "A or B" typically | ||||
# returns returns the revision matching A then the revision | ||||
# matching B. We want to hide this internal implementation | ||||
# detail from the caller, and sorting the filtered revision | ||||
# again achieves this. | ||||
for i, window in cmdutil.increasingwindows(0, len(revs), windowsize=1): | ||||
orevs = revs[i:i + window] | ||||
nrevs = set(matcher(repo, sorted(orevs))) | ||||
for rev in orevs: | ||||
if rev in nrevs: | ||||
yield rev | ||||
Patrick Mezard
|
r16405 | if not len(repo): | ||
Patrick Mezard
|
r16777 | return iter([]), None, None | ||
Patrick Mezard
|
r16408 | # Default --rev value depends on --follow but --follow behaviour | ||
# depends on revisions resolved from --rev... | ||||
follow = opts.get('follow') or opts.get('follow_first') | ||||
Patrick Mezard
|
r16405 | if opts.get('rev'): | ||
revs = scmutil.revrange(repo, opts['rev']) | ||||
else: | ||||
Patrick Mezard
|
r16408 | if follow and len(repo) > 0: | ||
revs = scmutil.revrange(repo, ['.:0']) | ||||
else: | ||||
revs = range(len(repo) - 1, -1, -1) | ||||
Patrick Mezard
|
r16405 | if not revs: | ||
Patrick Mezard
|
r16777 | return iter([]), None, None | ||
Patrick Mezard
|
r16405 | expr, filematcher = _makelogrevset(repo, pats, opts, revs) | ||
if expr: | ||||
Patrick Mezard
|
r16777 | matcher = revset.match(repo.ui, expr) | ||
revs = increasingrevs(repo, revs, matcher) | ||||
Patrick Mezard
|
r16431 | if not opts.get('hidden'): | ||
# --hidden is still experimental and not worth a dedicated revset | ||||
# yet. Fortunately, filtering revision number is fast. | ||||
Patrick Mezard
|
r16777 | revs = (r for r in revs if r not in repo.changelog.hiddenrevs) | ||
else: | ||||
revs = iter(revs) | ||||
Patrick Mezard
|
r16405 | return revs, expr, filematcher | ||
Patrick Mezard
|
r16186 | def generate(ui, dag, displayer, showparents, edgefn, getrenamed=None, | ||
filematcher=None): | ||||
Peter Arrenbrecht
|
r9631 | seen, state = [], asciistate() | ||
Dirkjan Ochtman
|
r9369 | for rev, type, ctx, parents in dag: | ||
char = ctx.node() in showparents and '@' or 'o' | ||||
Patrick Mezard
|
r16180 | copies = None | ||
if getrenamed and ctx.rev(): | ||||
copies = [] | ||||
for fn in ctx.files(): | ||||
rename = getrenamed(fn, ctx.rev()) | ||||
if rename: | ||||
copies.append((fn, rename[0])) | ||||
Patrick Mezard
|
r16186 | revmatchfn = None | ||
if filematcher is not None: | ||||
revmatchfn = filematcher(ctx.rev()) | ||||
displayer.show(ctx, copies=copies, matchfn=revmatchfn) | ||||
Dirkjan Ochtman
|
r9369 | lines = displayer.hunk.pop(rev).split('\n')[:-1] | ||
Mads Kiilerich
|
r12579 | displayer.flush(rev) | ||
Patrick Mezard
|
r14130 | edges = edgefn(type, char, lines, seen, rev, parents) | ||
for type, char, lines, coldata in edges: | ||||
ascii(ui, state, type, char, lines, coldata) | ||||
Mads Kiilerich
|
r12579 | displayer.close() | ||
Dirkjan Ochtman
|
r9369 | |||
Adrian Buehlmann
|
r14311 | @command('glog', | ||
Patrick Mezard
|
r16432 | [('f', 'follow', None, | ||
_('follow changeset history, or file history across copies and renames')), | ||||
('', 'follow-first', None, | ||||
_('only follow the first parent of merge changesets (DEPRECATED)')), | ||||
('d', 'date', '', _('show revisions matching date spec'), _('DATE')), | ||||
('C', 'copies', None, _('show copied files')), | ||||
('k', 'keyword', [], | ||||
_('do case-insensitive search for a given text'), _('TEXT')), | ||||
Adrian Buehlmann
|
r14311 | ('r', 'rev', [], _('show the specified revision or range'), _('REV')), | ||
Patrick Mezard
|
r16432 | ('', 'removed', None, _('include revisions where files were removed')), | ||
('m', 'only-merges', None, _('show only merges (DEPRECATED)')), | ||||
('u', 'user', [], _('revisions committed by user'), _('USER')), | ||||
('', 'only-branch', [], | ||||
_('show only changesets within the given named branch (DEPRECATED)'), | ||||
_('BRANCH')), | ||||
('b', 'branch', [], | ||||
_('show changesets within the given named branch'), _('BRANCH')), | ||||
('P', 'prune', [], | ||||
_('do not display revision or any of its ancestors'), _('REV')), | ||||
('', 'hidden', False, _('show hidden changesets (DEPRECATED)')), | ||||
] + commands.logopts + commands.walkopts, | ||||
_('[OPTION]... [FILE]')) | ||||
Alexander Solovyov
|
r14043 | def graphlog(ui, repo, *pats, **opts): | ||
Peter Arrenbrecht
|
r7325 | """show revision history alongside an ASCII revision graph | ||
Martin Geisler
|
r9259 | Print a revision history alongside a revision graph drawn with | ||
ASCII characters. | ||||
Peter Arrenbrecht
|
r7325 | |||
Martin Geisler
|
r9259 | Nodes printed as an @ character are parents of the working | ||
directory. | ||||
Peter Arrenbrecht
|
r7325 | """ | ||
Patrick Mezard
|
r16405 | revs, expr, filematcher = getlogrevs(repo, pats, opts) | ||
Patrick Mezard
|
r16316 | revs = sorted(revs, reverse=1) | ||
Patrick Mezard
|
r14133 | limit = cmdutil.loglimit(opts) | ||
if limit is not None: | ||||
revs = revs[:limit] | ||||
Alexander Solovyov
|
r14043 | revdag = graphmod.dagwalker(repo, revs) | ||
Peter Arrenbrecht
|
r7325 | |||
Patrick Mezard
|
r16180 | getrenamed = None | ||
if opts.get('copies'): | ||||
endrev = None | ||||
if opts.get('rev'): | ||||
endrev = max(scmutil.revrange(repo, opts.get('rev'))) + 1 | ||||
getrenamed = templatekw.getrenamedfn(repo, endrev=endrev) | ||||
Dirkjan Ochtman
|
r9368 | displayer = show_changeset(ui, repo, opts, buffered=True) | ||
showparents = [ctx.node() for ctx in repo[None].parents()] | ||||
Patrick Mezard
|
r16186 | generate(ui, revdag, displayer, showparents, asciiedges, getrenamed, | ||
filematcher) | ||||
Dirkjan Ochtman
|
r7716 | |||
def graphrevs(repo, nodes, opts): | ||||
limit = cmdutil.loglimit(opts) | ||||
Peter Arrenbrecht
|
r8837 | nodes.reverse() | ||
Nicolas Dumazet
|
r10111 | if limit is not None: | ||
Peter Arrenbrecht
|
r8837 | 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 | |||
Patrick Mezard
|
r14086 | check_unsupported_flags([], opts) | ||
Nicolas Dumazet
|
r12735 | o = hg._outgoing(ui, repo, dest, opts) | ||
if o is None: | ||||
Alpar Juttner
|
r7426 | return | ||
Dirkjan Ochtman
|
r7716 | |||
revdag = graphrevs(repo, o, opts) | ||||
Dirkjan Ochtman
|
r9368 | displayer = show_changeset(ui, repo, opts, buffered=True) | ||
showparents = [ctx.node() for ctx in repo[None].parents()] | ||||
Dirkjan Ochtman
|
r9371 | generate(ui, revdag, displayer, showparents, asciiedges) | ||
Alpar Juttner
|
r7426 | |||
def gincoming(ui, repo, source="default", **opts): | ||||
"""show the incoming changesets alongside an ASCII revision graph | ||||
Martin Geisler
|
r9259 | Print the incoming changesets alongside a revision graph drawn with | ||
ASCII characters. | ||||
Alpar Juttner
|
r7426 | |||
Martin Geisler
|
r9259 | Nodes printed as an @ character are parents of the working | ||
directory. | ||||
Alpar Juttner
|
r7426 | """ | ||
Nicolas Dumazet
|
r12730 | def subreporecurse(): | ||
return 1 | ||||
Alpar Juttner
|
r7426 | |||
Patrick Mezard
|
r14086 | check_unsupported_flags([], opts) | ||
Nicolas Dumazet
|
r12730 | def display(other, chlist, displayer): | ||
Dirkjan Ochtman
|
r7716 | revdag = graphrevs(other, chlist, opts) | ||
Dirkjan Ochtman
|
r9368 | showparents = [ctx.node() for ctx in repo[None].parents()] | ||
Dirkjan Ochtman
|
r9371 | generate(ui, revdag, displayer, showparents, asciiedges) | ||
Alpar Juttner
|
r7426 | |||
Nicolas Dumazet
|
r12730 | hg._incoming(display, subreporecurse, ui, repo, source, opts, buffered=True) | ||
Alpar Juttner
|
r7426 | |||
def uisetup(ui): | ||||
'''Initialize the extension.''' | ||||
Idan Kamara
|
r14416 | _wrapcmd('log', commands.table, graphlog) | ||
_wrapcmd('incoming', commands.table, gincoming) | ||||
_wrapcmd('outgoing', commands.table, goutgoing) | ||||
Alpar Juttner
|
r7426 | |||
Idan Kamara
|
r14416 | def _wrapcmd(cmd, table, wrapfn): | ||
Alpar Juttner
|
r7426 | '''wrap the command''' | ||
def graph(orig, *args, **kwargs): | ||||
if kwargs['graph']: | ||||
Alexander Solovyov
|
r14043 | return wrapfn(*args, **kwargs) | ||
Alpar Juttner
|
r7426 | return orig(*args, **kwargs) | ||
entry = extensions.wrapcommand(table, cmd, graph) | ||||
Jim Correia
|
r7763 | entry[1].append(('G', 'graph', None, _("show the revision DAG"))) | ||