##// END OF EJS Templates
hook: redirect stdout to stderr for ssh and http servers
hook: redirect stdout to stderr for ssh and http servers

File last commit:

r5760:0145f9af default
r5833:323b9c55 default
Show More
hgk.py
358 lines | 11.8 KiB | text/x-python | PythonLexer
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 # Minimal support for git commands on an hg repository
#
Vadim Gelfer
update copyrights.
r2859 # Copyright 2005, 2006 Chris Mason <mason@suse.com>
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 #
# This software may be used and distributed according to the terms
# of the GNU General Public License, incorporated herein by reference.
Patrick Mezard
hgk: add basic usage and configuration documentation
r5395 #
# 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.)
#
# 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 hgk.py can be found in the hgext
Thomas Arendsen Hein
Removed tabs and trailing whitespace in python files
r5760 # directory.
Patrick Mezard
hgk: add basic usage and configuration documentation
r5395 #
# To load the hgext.py extension, add it to your .hgrc file (you have
# to use your global $HOME/.hgrc file, not one in a repository). You
# can specify an absolute path:
#
# [extensions]
# hgk=/usr/local/lib/hgk.py
#
# Mercurial can also scan the default python library path for a file
# named 'hgk.py' if you set hgk empty:
#
# [extensions]
# hgk=
#
# The hg view command will launch the hgk Tcl script. For this command
# to work, hgk must be in your search path. Alternately, you can
# specify the path to hgk in your .hgrc file:
#
# [hgk]
# path=/location/of/hgk
#
# hgk can make use of the extdiff extension to visualize
# revisions. Assuming you had already configured extdiff vdiff
# command, just add:
#
# [hgk]
# vdiff=vdiff
#
# Revisions context menu will now display additional entries to fire
# vdiff on hovered and selected revisions.
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432
Benoit Boissinot
remove various unused import
r3963 import sys, os
Matt Mackall
Replace demandload with new demandimport
r3877 from mercurial import hg, fancyopts, commands, ui, util, patch, revlog
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432
Benoit Boissinot
hgk.py: add an optional file list to debug-diff-tree...
r3063 def difftree(ui, repo, node1=None, node2=None, *files, **opts):
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 """diff trees from two commits"""
Benoit Boissinot
hgk.py: add an optional file list to debug-diff-tree...
r3063 def __difftree(repo, node1, node2, files=[]):
Benoit Boissinot
hgk: remove unused code, node2 is always set
r3978 assert node2 is not None
Benoit Boissinot
hgk: (re)optimize reading of changelog and manifest
r3981 mmap = repo.changectx(node1).manifest()
Benoit Boissinot
hgk: use contexts
r3979 mmap2 = repo.changectx(node2).manifest()
Benoit Boissinot
hgk: remove unused code, node2 is always set
r3978 status = repo.status(node1, node2, files=files)[:5]
modified, added, removed, deleted, unknown = status
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432
Benoit Boissinot
hgk.py: fix for a bug introduced in 32fd512ca7c0
r3069 empty = hg.short(hg.nullid)
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432
for f in modified:
# TODO get file permissions
TK Soh
hgk: use short changeset hashes...
r3059 print ":100664 100664 %s %s M\t%s\t%s" % (hg.short(mmap[f]),
hg.short(mmap2[f]),
f, f)
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 for f in added:
Benoit Boissinot
hgk.py: fix for a bug introduced in 32fd512ca7c0
r3069 print ":000000 100664 %s %s N\t%s\t%s" % (empty,
TK Soh
hgk: use short changeset hashes...
r3059 hg.short(mmap2[f]),
f, f)
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 for f in removed:
TK Soh
hgk: use short changeset hashes...
r3059 print ":100664 000000 %s %s D\t%s\t%s" % (hg.short(mmap[f]),
Benoit Boissinot
hgk.py: fix for a bug introduced in 32fd512ca7c0
r3069 empty,
TK Soh
hgk: use short changeset hashes...
r3059 f, f)
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 ##
while True:
if opts['stdin']:
try:
line = raw_input().split(' ')
node1 = line[0]
if len(line) > 1:
node2 = line[1]
else:
node2 = None
except EOFError:
break
node1 = repo.lookup(node1)
if node2:
node2 = repo.lookup(node2)
else:
node2 = node1
node1 = repo.changelog.parents(node1)[0]
if opts['patch']:
if opts['pretty']:
catcommit(repo, node2, "")
Benoit Boissinot
hgk.py: use mercurial.patch functions when possible
r3067 patch.diff(repo, node1, node2,
files=files,
opts=patch.diffopts(ui, {'git': True}))
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 else:
Benoit Boissinot
hgk.py: add an optional file list to debug-diff-tree...
r3063 __difftree(repo, node1, node2, files=files)
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 if not opts['stdin']:
break
Benoit Boissinot
hgk: use contexts
r3979 def catcommit(repo, n, prefix, ctx=None):
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 nlprefix = '\n' + prefix;
Benoit Boissinot
hgk: use contexts
r3979 if ctx is None:
ctx = repo.changectx(n)
(p1, p2) = ctx.parents()
print "tree %s" % (hg.short(ctx.changeset()[0])) # use ctx.node() instead ??
if p1: print "parent %s" % (hg.short(p1.node()))
if p2: print "parent %s" % (hg.short(p2.node()))
date = ctx.date()
Matt Mackall
hgk: remove embedded nulls in descriptions
r4039 description = ctx.description().replace("\0", "")
Benoit Boissinot
hgk: use contexts
r3979 lines = description.splitlines()
Brendan Cully
hgk: Don't choke on empty changelogs
r2525 if lines and lines[-1].startswith('committer:'):
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 committer = lines[-1].split(': ')[1].rstrip()
else:
Benoit Boissinot
hgk: use contexts
r3979 committer = ctx.user()
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432
Benoit Boissinot
hgk: use contexts
r3979 print "author %s %s %s" % (ctx.user(), int(date[0]), date[1])
print "committer %s %s %s" % (committer, int(date[0]), date[1])
print "revision %d" % ctx.rev()
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 print ""
if prefix != "":
Benoit Boissinot
hgk: use contexts
r3979 print "%s%s" % (prefix, description.replace('\n', nlprefix).strip())
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 else:
Benoit Boissinot
hgk: use contexts
r3979 print description
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 if prefix:
sys.stdout.write('\0')
def base(ui, repo, node1, node2):
"""Output common ancestor information"""
node1 = repo.lookup(node1)
node2 = repo.lookup(node2)
n = repo.changelog.ancestor(node1, node2)
TK Soh
hgk: use short changeset hashes...
r3059 print hg.short(n)
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
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
#
prefix = ""
if opts['stdin']:
try:
(type, r) = raw_input().split(' ');
prefix = " "
except EOFError:
return
else:
if not type or not r:
ui.warn("cat-file: type or revision not supplied\n")
commands.help_(ui, 'cat-file')
while r:
if type != "commit":
sys.stderr.write("aborting hg cat-file only understands commits\n")
sys.exit(1);
n = repo.lookup(r)
catcommit(repo, n, prefix)
if opts['stdin']:
try:
(type, r) = raw_input().split(' ');
except EOFError:
break
else:
break
# 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 ^
def revtree(args, repo, full="tree", maxnr=0, parents=False):
def chlogwalk():
Benoit Boissinot
hgk: use contexts
r3979 count = repo.changelog.count()
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 i = count
l = [0] * 100
chunk = 100
while True:
if chunk > i:
chunk = i
i = 0
else:
i -= chunk
for x in xrange(0, chunk):
if i + x >= count:
l[chunk - x:] = [0] * (chunk - x)
break
if full != None:
Benoit Boissinot
hgk: use contexts
r3979 l[x] = repo.changectx(i + x)
Benoit Boissinot
hgk: (re)optimize reading of changelog and manifest
r3981 l[x].changeset() # force reading
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 else:
l[x] = 1
for x in xrange(chunk-1, -1, -1):
if l[x] != 0:
yield (i + x, full != None and l[x] or None)
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
Benoit Boissinot
use xrange instead of range
r3473 for i in xrange(len(ar)):
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
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
Benoit Boissinot
use xrange instead of range
r3473 for i in xrange(len(args)):
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 if args[i].startswith('^'):
s = repo.lookup(args[i][1:])
stop_sha1.append(s)
want_sha1.append(s)
elif args[i] != 'HEAD':
want_sha1.append(repo.lookup(args[i]))
# calculate the graph for the supplied commits
Benoit Boissinot
use xrange instead of range
r3473 for i in xrange(len(want_sha1)):
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 reachable.append({});
n = want_sha1[i];
visit = [n];
reachable[i][n] = 1
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]:
reachable[i][p] = 1
visit.append(p)
if p in stop_sha1:
continue
# walk the repository looking for commits that are in our
# reachability graph
Benoit Boissinot
hgk: use contexts
r3979 for i, ctx in chlogwalk():
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 n = repo.changelog.node(i)
mask = is_reachable(want_sha1, reachable, n)
if mask:
parentstr = ""
if parents:
pp = repo.changelog.parents(n)
if pp[0] != hg.nullid:
TK Soh
hgk: use short changeset hashes...
r3059 parentstr += " " + hg.short(pp[0])
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 if pp[1] != hg.nullid:
TK Soh
hgk: use short changeset hashes...
r3059 parentstr += " " + hg.short(pp[1])
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 if not full:
TK Soh
hgk: use short changeset hashes...
r3059 print hg.short(n) + parentstr
Benoit Boissinot
hgk.py: fix warnings from pychecker...
r3064 elif full == "commit":
TK Soh
hgk: use short changeset hashes...
r3059 print hg.short(n) + parentstr
Benoit Boissinot
hgk: use contexts
r3979 catcommit(repo, n, ' ', ctx)
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 else:
(p1, p2) = repo.changelog.parents(n)
TK Soh
hgk: use short changeset hashes...
r3059 (h, h1, h2) = map(hg.short, (n, p1, p2))
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 (i1, i2) = map(repo.changelog.rev, (p1, p2))
Benoit Boissinot
hgk: use contexts
r3979 date = ctx.date()[0]
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 print "%s %s:%s" % (date, h, mask),
mask = is_reachable(want_sha1, reachable, p1)
Thomas Arendsen Hein
Define and use nullrev (revision of nullid) instead of -1.
r3578 if i1 != hg.nullrev and mask > 0:
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 print "%s:%s " % (h1, mask),
mask = is_reachable(want_sha1, reachable, p2)
Thomas Arendsen Hein
Define and use nullrev (revision of nullid) instead of -1.
r3578 if i2 != hg.nullrev and mask > 0:
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 print "%s:%s " % (h2, mask),
print ""
if maxnr and count >= maxnr:
break
count += 1
Brendan Cully
hgk: add --limit, and revranges
r3093 def revparse(ui, repo, *revs, **opts):
"""Parse given revisions"""
def revstr(rev):
if rev == 'HEAD':
rev = 'tip'
return revlog.hex(repo.lookup(rev))
for r in revs:
revrange = r.split(':', 1)
ui.write('%s\n' % revstr(revrange[0]))
if len(revrange) == 2:
ui.write('^%s\n' % revstr(revrange[1]))
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
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
def revlist(ui, repo, *revs, **opts):
"""print revisions"""
if opts['header']:
full = "commit"
else:
full = None
copy = [x for x in revs]
revtree(copy, repo, full, opts['max_count'], opts['parents'])
Patrick Mezard
hgk: add debug-config command to pass configuration options
r5393 def config(ui, repo, **opts):
"""print extension options"""
def writeopt(name, value):
Thomas Arendsen Hein
Removed tabs and trailing whitespace in python files
r5760 ui.write('k=%s\nv=%s\n' % (name, value))
Patrick Mezard
hgk: add debug-config command to pass configuration options
r5393
writeopt('vdiff', ui.config('hgk', 'vdiff', ''))
Thomas Arendsen Hein
Removed tabs and trailing whitespace in python files
r5760
Patrick Mezard
hgk: add debug-config command to pass configuration options
r5393
Brendan Cully
hgk: add --limit, and revranges
r3093 def view(ui, repo, *etc, **opts):
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 "start interactive history viewer"
os.chdir(repo.root)
TK Soh
hgk: fix mixup of --limit and REVRANGE in hgk call
r3180 optstr = ' '.join(['--%s %s' % (k, v) for k, v in opts.iteritems() if v])
cmd = ui.config("hgk", "path", "hgk") + " %s %s" % (optstr, " ".join(etc))
ui.debug("running %s\n" % cmd)
Thomas Arendsen Hein
hgk: Use $HG instead of hg (see 849f011dbf79)
r4688 util.system(cmd)
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432
cmdtable = {
Thomas Arendsen Hein
Updated command tables in commands.py and hgext extensions....
r4730 "^view":
(view,
[('l', 'limit', '', 'limit number of changes displayed')],
'hg view [-l LIMIT] [REVRANGE]'),
"debug-diff-tree":
(difftree,
[('p', 'patch', None, 'generate patch'),
('r', 'recursive', None, 'recursive'),
('P', 'pretty', None, 'pretty'),
('s', 'stdin', None, 'stdin'),
('C', 'copy', None, 'detect copies'),
('S', 'search', "", 'search')],
'hg git-diff-tree [OPTION]... NODE1 NODE2 [FILE]...'),
"debug-cat-file":
(catfile,
[('s', 'stdin', None, 'stdin')],
'hg debug-cat-file [OPTION]... TYPE FILE'),
Patrick Mezard
hgk: add debug-config command to pass configuration options
r5393 "debug-config":
(config, [], 'hg debug-config'),
Thomas Arendsen Hein
Updated command tables in commands.py and hgext extensions....
r4730 "debug-merge-base":
(base, [], 'hg debug-merge-base node node'),
"debug-rev-parse":
(revparse,
[('', 'default', '', 'ignored')],
'hg debug-rev-parse REV'),
"debug-rev-list":
(revlist,
[('H', 'header', None, 'header'),
('t', 'topo-order', None, 'topo-order'),
('p', 'parents', None, 'parents'),
('n', 'max-count', 0, 'max-count')],
'hg debug-rev-list [options] revs'),
Vadim Gelfer
move hgk.py into hgext. now to enable "hg view" is one less step....
r2432 }