hgk.py
386 lines
| 11.8 KiB
| text/x-python
|
PythonLexer
/ hgext / hgk.py
Vadim Gelfer
|
r2432 | # Minimal support for git commands on an hg repository | ||
# | ||||
Vadim Gelfer
|
r2859 | # Copyright 2005, 2006 Chris Mason <mason@suse.com> | ||
Vadim Gelfer
|
r2432 | # | ||
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 | |||
Cédric Duval
|
r8894 | '''browse the repository in a graphical way | ||
Dirkjan Ochtman
|
r6666 | |||
Martin Geisler
|
r9261 | The hgk extension allows browsing the history of a repository in a | ||
graphical way. It requires Tcl/Tk version 8.4 or later. (Tcl/Tk is not | ||||
distributed with Mercurial.) | ||||
Dirkjan Ochtman
|
r6666 | |||
Martin Geisler
|
r9261 | hgk consists of two parts: a Tcl script that does the displaying and | ||
querying of information, and an extension to Mercurial named hgk.py, | ||||
which provides hooks for hgk to get information. hgk can be found in | ||||
the contrib directory, and the extension is shipped in the hgext | ||||
repository, and needs to be enabled. | ||||
Dirkjan Ochtman
|
r6666 | |||
Martin Geisler
|
r11193 | The :hg:`view` command will launch the hgk Tcl script. For this command | ||
Martin Geisler
|
r9261 | to work, hgk must be in your search path. Alternately, you can specify | ||
Brodie Rao
|
r12083 | the path to hgk in your configuration file:: | ||
Dirkjan Ochtman
|
r6666 | |||
[hgk] | ||||
Matt Mackall
|
r25793 | path = /location/of/hgk | ||
Dirkjan Ochtman
|
r6666 | |||
Martin Geisler
|
r9261 | hgk can make use of the extdiff extension to visualize revisions. | ||
Assuming you had already configured extdiff vdiff command, just add:: | ||||
Dirkjan Ochtman
|
r6666 | |||
[hgk] | ||||
vdiff=vdiff | ||||
Martin Geisler
|
r9261 | Revisions context menu will now display additional entries to fire | ||
vdiff on hovered and selected revisions. | ||||
Martin Geisler
|
r9063 | ''' | ||
Vadim Gelfer
|
r2432 | |||
Matt Harbison
|
r52756 | from __future__ import annotations | ||
Pulkit Goyal
|
r29125 | |||
Matt Mackall
|
r5878 | import os | ||
Yuya Nishihara
|
r29205 | |||
from mercurial.i18n import _ | ||||
from mercurial.node import ( | ||||
nullrev, | ||||
short, | ||||
) | ||||
Pulkit Goyal
|
r29125 | from mercurial import ( | ||
commands, | ||||
obsolete, | ||||
patch, | ||||
Pulkit Goyal
|
r34999 | pycompat, | ||
Yuya Nishihara
|
r32337 | registrar, | ||
Pulkit Goyal
|
r29125 | scmutil, | ||
) | ||||
Vadim Gelfer
|
r2432 | |||
Gregory Szorc
|
r21250 | cmdtable = {} | ||
Yuya Nishihara
|
r32337 | command = registrar.command(cmdtable) | ||
Augie Fackler
|
r29841 | # Note for extension authors: ONLY specify testedwith = 'ships-with-hg-core' for | ||
Augie Fackler
|
r25186 | # extensions which SHIP WITH MERCURIAL. Non-mainline extensions should | ||
# be specifying the version(s) of Mercurial they are tested with, or | ||||
# leave the attribute unspecified. | ||||
Augie Fackler
|
r43347 | testedwith = b'ships-with-hg-core' | ||
Augie Fackler
|
r16743 | |||
Boris Feld
|
r34500 | configtable = {} | ||
configitem = registrar.configitem(configtable) | ||||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'hgk', | ||
b'path', | ||||
default=b'hgk', | ||||
Boris Feld
|
r34500 | ) | ||
Augie Fackler
|
r43346 | |||
@command( | ||||
Augie Fackler
|
r43347 | b'debug-diff-tree', | ||
Augie Fackler
|
r43346 | [ | ||
Augie Fackler
|
r43347 | (b'p', b'patch', None, _(b'generate patch')), | ||
(b'r', b'recursive', None, _(b'recursive')), | ||||
(b'P', b'pretty', None, _(b'pretty')), | ||||
(b's', b'stdin', None, _(b'stdin')), | ||||
(b'C', b'copy', None, _(b'detect copies')), | ||||
(b'S', b'search', b"", _(b'search')), | ||||
Augie Fackler
|
r43346 | ], | ||
Augie Fackler
|
r43347 | b'[OPTION]... NODE1 NODE2 [FILE]...', | ||
Augie Fackler
|
r43346 | inferrepo=True, | ||
) | ||||
Benoit Boissinot
|
r3063 | def difftree(ui, repo, node1=None, node2=None, *files, **opts): | ||
Vadim Gelfer
|
r2432 | """diff trees from two commits""" | ||
Pulkit Goyal
|
r34999 | |||
Pierre-Yves David
|
r31409 | def __difftree(repo, node1, node2, files=None): | ||
Benoit Boissinot
|
r3978 | assert node2 is not None | ||
Pierre-Yves David
|
r31409 | if files is None: | ||
files = [] | ||||
Matt Mackall
|
r6747 | mmap = repo[node1].manifest() | ||
mmap2 = repo[node2].manifest() | ||||
Matt Mackall
|
r14671 | m = scmutil.match(repo[node1], files) | ||
Augie Fackler
|
r44038 | st = repo.status(node1, node2, m) | ||
Joerg Sonnenberger
|
r47771 | empty = short(repo.nullid) | ||
Vadim Gelfer
|
r2432 | |||
Augie Fackler
|
r44038 | for f in st.modified: | ||
Vadim Gelfer
|
r2432 | # TODO get file permissions | ||
Augie Fackler
|
r43350 | ui.writenoi18n( | ||
Augie Fackler
|
r43347 | b":100664 100664 %s %s M\t%s\t%s\n" | ||
Augie Fackler
|
r43346 | % (short(mmap[f]), short(mmap2[f]), f, f) | ||
) | ||||
Augie Fackler
|
r44038 | for f in st.added: | ||
Augie Fackler
|
r43350 | ui.writenoi18n( | ||
Augie Fackler
|
r43347 | b":000000 100664 %s %s N\t%s\t%s\n" | ||
Augie Fackler
|
r43346 | % (empty, short(mmap2[f]), f, f) | ||
) | ||||
Augie Fackler
|
r44038 | for f in st.removed: | ||
Augie Fackler
|
r43350 | ui.writenoi18n( | ||
Augie Fackler
|
r43347 | b":100664 000000 %s %s D\t%s\t%s\n" | ||
Augie Fackler
|
r43346 | % (short(mmap[f]), empty, f, f) | ||
) | ||||
Vadim Gelfer
|
r2432 | ## | ||
while True: | ||||
Augie Fackler
|
r43906 | if opts['stdin']: | ||
Yuya Nishihara
|
r36808 | line = ui.fin.readline() | ||
if not line: | ||||
Vadim Gelfer
|
r2432 | break | ||
Yuya Nishihara
|
r36808 | line = line.rstrip(pycompat.oslinesep).split(b' ') | ||
node1 = line[0] | ||||
if len(line) > 1: | ||||
node2 = line[1] | ||||
else: | ||||
node2 = None | ||||
Vadim Gelfer
|
r2432 | node1 = repo.lookup(node1) | ||
if node2: | ||||
node2 = repo.lookup(node2) | ||||
else: | ||||
node2 = node1 | ||||
node1 = repo.changelog.parents(node1)[0] | ||||
Augie Fackler
|
r43906 | if opts['patch']: | ||
if opts['pretty']: | ||||
Augie Fackler
|
r43347 | catcommit(ui, repo, node2, b"") | ||
Matt Mackall
|
r14671 | m = scmutil.match(repo[node1], files) | ||
Siddharth Agarwal
|
r23451 | diffopts = patch.difffeatureopts(ui) | ||
diffopts.git = True | ||||
Augie Fackler
|
r43346 | chunks = patch.diff(repo, node1, node2, match=m, opts=diffopts) | ||
Dirkjan Ochtman
|
r7308 | for chunk in chunks: | ||
Martin Geisler
|
r8615 | ui.write(chunk) | ||
Vadim Gelfer
|
r2432 | else: | ||
Benoit Boissinot
|
r3063 | __difftree(repo, node1, node2, files=files) | ||
Augie Fackler
|
r43906 | if not opts['stdin']: | ||
Vadim Gelfer
|
r2432 | break | ||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r5878 | def catcommit(ui, repo, n, prefix, ctx=None): | ||
Augie Fackler
|
r43347 | nlprefix = b'\n' + prefix | ||
Benoit Boissinot
|
r3979 | if ctx is None: | ||
Matt Mackall
|
r6747 | ctx = repo[n] | ||
Brodie Rao
|
r16683 | # use ctx.node() instead ?? | ||
Augie Fackler
|
r43347 | ui.write((b"tree %s\n" % short(ctx.changeset()[0]))) | ||
Matt Mackall
|
r6768 | for p in ctx.parents(): | ||
Augie Fackler
|
r43347 | ui.write((b"parent %s\n" % p)) | ||
Matt Mackall
|
r6768 | |||
Benoit Boissinot
|
r3979 | date = ctx.date() | ||
Augie Fackler
|
r43347 | description = ctx.description().replace(b"\0", b"") | ||
ui.write((b"author %s %d %d\n" % (ctx.user(), int(date[0]), date[1]))) | ||||
Vadim Gelfer
|
r2432 | |||
Augie Fackler
|
r43347 | if b'committer' in ctx.extra(): | ||
ui.write((b"committer %s\n" % ctx.extra()[b'committer'])) | ||||
Andrew Shadura
|
r24604 | |||
Augie Fackler
|
r43347 | ui.write((b"revision %d\n" % ctx.rev())) | ||
ui.write((b"branch %s\n" % ctx.branch())) | ||||
Andrew Shadura
|
r24513 | if obsolete.isenabled(repo, obsolete.createmarkersopt): | ||
if ctx.obsolete(): | ||||
Augie Fackler
|
r43350 | ui.writenoi18n(b"obsolete\n") | ||
Augie Fackler
|
r43347 | ui.write((b"phase %s\n\n" % ctx.phasestr())) | ||
Matt Mackall
|
r5878 | |||
Augie Fackler
|
r43347 | if prefix != b"": | ||
Augie Fackler
|
r43346 | ui.write( | ||
Augie Fackler
|
r43347 | b"%s%s\n" % (prefix, description.replace(b'\n', nlprefix).strip()) | ||
Augie Fackler
|
r43346 | ) | ||
Vadim Gelfer
|
r2432 | else: | ||
Augie Fackler
|
r43347 | ui.write(description + b"\n") | ||
Vadim Gelfer
|
r2432 | if prefix: | ||
Augie Fackler
|
r43347 | ui.write(b'\0') | ||
Vadim Gelfer
|
r2432 | |||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r43347 | @command(b'debug-merge-base', [], _(b'REV REV')) | ||
Vadim Gelfer
|
r2432 | def base(ui, repo, node1, node2): | ||
Martin Geisler
|
r7598 | """output common ancestor information""" | ||
Vadim Gelfer
|
r2432 | node1 = repo.lookup(node1) | ||
node2 = repo.lookup(node2) | ||||
n = repo.changelog.ancestor(node1, node2) | ||||
Augie Fackler
|
r43347 | ui.write(short(n) + b"\n") | ||
Vadim Gelfer
|
r2432 | |||
Augie Fackler
|
r43346 | |||
@command( | ||||
Augie Fackler
|
r43347 | b'debug-cat-file', | ||
[(b's', b'stdin', None, _(b'stdin'))], | ||||
_(b'[OPTION]... TYPE FILE'), | ||||
Augie Fackler
|
r43346 | inferrepo=True, | ||
) | ||||
Vadim Gelfer
|
r2432 | def catfile(ui, repo, type=None, r=None, **opts): | ||
"""cat a specific revision""" | ||||
# in stdin mode, every line except the commit is prefixed with two | ||||
# spaces. This way the our caller can find the commit without magic | ||||
# strings | ||||
# | ||||
Augie Fackler
|
r43347 | prefix = b"" | ||
Augie Fackler
|
r43906 | if opts['stdin']: | ||
Yuya Nishihara
|
r36808 | line = ui.fin.readline() | ||
if not line: | ||||
Vadim Gelfer
|
r2432 | return | ||
Yuya Nishihara
|
r36808 | (type, r) = line.rstrip(pycompat.oslinesep).split(b' ') | ||
Augie Fackler
|
r43347 | prefix = b" " | ||
Vadim Gelfer
|
r2432 | else: | ||
if not type or not r: | ||||
Augie Fackler
|
r43347 | ui.warn(_(b"cat-file: type or revision not supplied\n")) | ||
commands.help_(ui, b'cat-file') | ||||
Vadim Gelfer
|
r2432 | |||
while r: | ||||
Augie Fackler
|
r43347 | if type != b"commit": | ||
ui.warn(_(b"aborting hg cat-file only understands commits\n")) | ||||
Benoit Boissinot
|
r10394 | return 1 | ||
Vadim Gelfer
|
r2432 | n = repo.lookup(r) | ||
Matt Mackall
|
r5878 | catcommit(ui, repo, n, prefix) | ||
Augie Fackler
|
r43906 | if opts['stdin']: | ||
Yuya Nishihara
|
r36808 | line = ui.fin.readline() | ||
if not line: | ||||
Vadim Gelfer
|
r2432 | break | ||
Yuya Nishihara
|
r36808 | (type, r) = line.rstrip(pycompat.oslinesep).split(b' ') | ||
Vadim Gelfer
|
r2432 | else: | ||
break | ||||
Augie Fackler
|
r43346 | |||
Vadim Gelfer
|
r2432 | # git rev-tree is a confusing thing. You can supply a number of | ||
# commit sha1s on the command line, and it walks the commit history | ||||
# telling you which commits are reachable from the supplied ones via | ||||
# a bitmask based on arg position. | ||||
# you can specify a commit to stop at by starting the sha1 with ^ | ||||
Augie Fackler
|
r43347 | def revtree(ui, args, repo, full=b"tree", maxnr=0, parents=False): | ||
Vadim Gelfer
|
r2432 | def chlogwalk(): | ||
Matt Mackall
|
r6750 | count = len(repo) | ||
Vadim Gelfer
|
r2432 | i = count | ||
l = [0] * 100 | ||||
chunk = 100 | ||||
while True: | ||||
if chunk > i: | ||||
chunk = i | ||||
i = 0 | ||||
else: | ||||
i -= chunk | ||||
Manuel Jacob
|
r50179 | for x in range(chunk): | ||
Vadim Gelfer
|
r2432 | if i + x >= count: | ||
Augie Fackler
|
r43346 | l[chunk - x :] = [0] * (chunk - x) | ||
Vadim Gelfer
|
r2432 | break | ||
Martin Geisler
|
r13031 | if full is not None: | ||
Andrew Shadura
|
r22580 | if (i + x) in repo: | ||
l[x] = repo[i + x] | ||||
Augie Fackler
|
r43346 | l[x].changeset() # force reading | ||
Vadim Gelfer
|
r2432 | else: | ||
Andrew Shadura
|
r22580 | if (i + x) in repo: | ||
l[x] = 1 | ||||
Manuel Jacob
|
r50179 | for x in range(chunk - 1, -1, -1): | ||
Vadim Gelfer
|
r2432 | if l[x] != 0: | ||
Martin Geisler
|
r13031 | yield (i + x, full is not None and l[x] or None) | ||
Vadim Gelfer
|
r2432 | if i == 0: | ||
break | ||||
# calculate and return the reachability bitmask for sha | ||||
def is_reachable(ar, reachable, sha): | ||||
if len(ar) == 0: | ||||
return 1 | ||||
mask = 0 | ||||
Manuel Jacob
|
r50179 | for i in range(len(ar)): | ||
Vadim Gelfer
|
r2432 | if sha in reachable[i]: | ||
mask |= 1 << i | ||||
return mask | ||||
reachable = [] | ||||
stop_sha1 = [] | ||||
want_sha1 = [] | ||||
count = 0 | ||||
# figure out which commits they are asking for and which ones they | ||||
# want us to stop on | ||||
Martin Geisler
|
r8632 | for i, arg in enumerate(args): | ||
Augie Fackler
|
r43347 | if arg.startswith(b'^'): | ||
Martin Geisler
|
r8632 | s = repo.lookup(arg[1:]) | ||
Vadim Gelfer
|
r2432 | stop_sha1.append(s) | ||
want_sha1.append(s) | ||||
Augie Fackler
|
r43347 | elif arg != b'HEAD': | ||
Martin Geisler
|
r8632 | want_sha1.append(repo.lookup(arg)) | ||
Vadim Gelfer
|
r2432 | |||
# calculate the graph for the supplied commits | ||||
Martin Geisler
|
r8632 | for i, n in enumerate(want_sha1): | ||
Benoit Boissinot
|
r10394 | reachable.append(set()) | ||
visit = [n] | ||||
Benoit Boissinot
|
r8459 | reachable[i].add(n) | ||
Vadim Gelfer
|
r2432 | while visit: | ||
n = visit.pop(0) | ||||
if n in stop_sha1: | ||||
continue | ||||
for p in repo.changelog.parents(n): | ||||
if p not in reachable[i]: | ||||
Benoit Boissinot
|
r8459 | reachable[i].add(p) | ||
Vadim Gelfer
|
r2432 | visit.append(p) | ||
if p in stop_sha1: | ||||
continue | ||||
# walk the repository looking for commits that are in our | ||||
# reachability graph | ||||
Benoit Boissinot
|
r3979 | for i, ctx in chlogwalk(): | ||
Andrew Shadura
|
r22580 | if i not in repo: | ||
continue | ||||
Vadim Gelfer
|
r2432 | n = repo.changelog.node(i) | ||
mask = is_reachable(want_sha1, reachable, n) | ||||
if mask: | ||||
Augie Fackler
|
r43347 | parentstr = b"" | ||
Vadim Gelfer
|
r2432 | if parents: | ||
pp = repo.changelog.parents(n) | ||||
Joerg Sonnenberger
|
r47771 | if pp[0] != repo.nullid: | ||
Augie Fackler
|
r43347 | parentstr += b" " + short(pp[0]) | ||
Joerg Sonnenberger
|
r47771 | if pp[1] != repo.nullid: | ||
Augie Fackler
|
r43347 | parentstr += b" " + short(pp[1]) | ||
Vadim Gelfer
|
r2432 | if not full: | ||
Augie Fackler
|
r43347 | ui.write(b"%s%s\n" % (short(n), parentstr)) | ||
elif full == b"commit": | ||||
ui.write(b"%s%s\n" % (short(n), parentstr)) | ||||
catcommit(ui, repo, n, b' ', ctx) | ||||
Vadim Gelfer
|
r2432 | else: | ||
(p1, p2) = repo.changelog.parents(n) | ||||
Joel Rosdahl
|
r6217 | (h, h1, h2) = map(short, (n, p1, p2)) | ||
Vadim Gelfer
|
r2432 | (i1, i2) = map(repo.changelog.rev, (p1, p2)) | ||
Benoit Boissinot
|
r3979 | date = ctx.date()[0] | ||
Augie Fackler
|
r43347 | ui.write(b"%s %s:%s" % (date, h, mask)) | ||
Vadim Gelfer
|
r2432 | mask = is_reachable(want_sha1, reachable, p1) | ||
Joel Rosdahl
|
r6217 | if i1 != nullrev and mask > 0: | ||
Augie Fackler
|
r43347 | ui.write(b"%s:%s " % (h1, mask)), | ||
Vadim Gelfer
|
r2432 | mask = is_reachable(want_sha1, reachable, p2) | ||
Joel Rosdahl
|
r6217 | if i2 != nullrev and mask > 0: | ||
Augie Fackler
|
r43347 | ui.write(b"%s:%s " % (h2, mask)) | ||
ui.write(b"\n") | ||||
Vadim Gelfer
|
r2432 | if maxnr and count >= maxnr: | ||
break | ||||
count += 1 | ||||
Augie Fackler
|
r43346 | |||
Vadim Gelfer
|
r2432 | # git rev-list tries to order things by date, and has the ability to stop | ||
# at a given commit without walking the whole repo. TODO add the stop | ||||
# parameter | ||||
Augie Fackler
|
r43346 | @command( | ||
Augie Fackler
|
r43347 | b'debug-rev-list', | ||
Augie Fackler
|
r43346 | [ | ||
Augie Fackler
|
r43347 | (b'H', b'header', None, _(b'header')), | ||
(b't', b'topo-order', None, _(b'topo-order')), | ||||
(b'p', b'parents', None, _(b'parents')), | ||||
(b'n', b'max-count', 0, _(b'max-count')), | ||||
Augie Fackler
|
r43346 | ], | ||
Augie Fackler
|
r43347 | b'[OPTION]... REV...', | ||
Augie Fackler
|
r43346 | ) | ||
Vadim Gelfer
|
r2432 | def revlist(ui, repo, *revs, **opts): | ||
"""print revisions""" | ||||
Kyle Lippincott
|
r45167 | if opts['header']: | ||
Augie Fackler
|
r43347 | full = b"commit" | ||
Vadim Gelfer
|
r2432 | else: | ||
full = None | ||||
copy = [x for x in revs] | ||||
Augie Fackler
|
r43906 | revtree(ui, copy, repo, full, opts['max_count'], opts[r'parents']) | ||
Vadim Gelfer
|
r2432 | |||
Augie Fackler
|
r43346 | |||
@command( | ||||
Augie Fackler
|
r43347 | b'view', | ||
[(b'l', b'limit', b'', _(b'limit number of changes displayed'), _(b'NUM'))], | ||||
_(b'[-l LIMIT] [REVRANGE]'), | ||||
Augie Fackler
|
r43346 | helpcategory=command.CATEGORY_CHANGE_NAVIGATION, | ||
) | ||||
Brendan Cully
|
r3093 | def view(ui, repo, *etc, **opts): | ||
Matt Harbison
|
r44226 | """start interactive history viewer""" | ||
Pulkit Goyal
|
r34999 | opts = pycompat.byteskwargs(opts) | ||
Vadim Gelfer
|
r2432 | os.chdir(repo.root) | ||
Gregory Szorc
|
r49768 | optstr = b' '.join([b'--%s %s' % (k, v) for k, v in opts.items() if v]) | ||
Andrew Shadura
|
r24512 | if repo.filtername is None: | ||
Augie Fackler
|
r43347 | optstr += b'--hidden' | ||
Andrew Shadura
|
r24512 | |||
Augie Fackler
|
r43347 | cmd = ui.config(b"hgk", b"path") + b" %s %s" % (optstr, b" ".join(etc)) | ||
ui.debug(b"running %s\n" % cmd) | ||||
ui.system(cmd, blockedtag=b'hgk_view') | ||||