##// END OF EJS Templates
add blank line after copyright notices and after header
Martin Geisler -
r8228:eee2319c default
parent child Browse files
Show More
@@ -1,51 +1,52
1 #!/usr/bin/env python
1 #!/usr/bin/env python
2 #
2 #
3 # Copyright 2005-2007 by Intevation GmbH <intevation@intevation.de>
3 # Copyright 2005-2007 by Intevation GmbH <intevation@intevation.de>
4 #
4 # Author(s):
5 # Author(s):
5 # Thomas Arendsen Hein <thomas@intevation.de>
6 # Thomas Arendsen Hein <thomas@intevation.de>
6 #
7 #
7 # This software may be used and distributed according to the terms of the
8 # This software may be used and distributed according to the terms of the
8 # GNU General Public License version 2, incorporated herein by reference.
9 # GNU General Public License version 2, incorporated herein by reference.
9
10
10 """
11 """
11 hg-ssh - a wrapper for ssh access to a limited set of mercurial repos
12 hg-ssh - a wrapper for ssh access to a limited set of mercurial repos
12
13
13 To be used in ~/.ssh/authorized_keys with the "command" option, see sshd(8):
14 To be used in ~/.ssh/authorized_keys with the "command" option, see sshd(8):
14 command="hg-ssh path/to/repo1 /path/to/repo2 ~/repo3 ~user/repo4" ssh-dss ...
15 command="hg-ssh path/to/repo1 /path/to/repo2 ~/repo3 ~user/repo4" ssh-dss ...
15 (probably together with these other useful options:
16 (probably together with these other useful options:
16 no-port-forwarding,no-X11-forwarding,no-agent-forwarding)
17 no-port-forwarding,no-X11-forwarding,no-agent-forwarding)
17
18
18 This allows pull/push over ssh to to the repositories given as arguments.
19 This allows pull/push over ssh to to the repositories given as arguments.
19
20
20 If all your repositories are subdirectories of a common directory, you can
21 If all your repositories are subdirectories of a common directory, you can
21 allow shorter paths with:
22 allow shorter paths with:
22 command="cd path/to/my/repositories && hg-ssh repo1 subdir/repo2"
23 command="cd path/to/my/repositories && hg-ssh repo1 subdir/repo2"
23
24
24 You can use pattern matching of your normal shell, e.g.:
25 You can use pattern matching of your normal shell, e.g.:
25 command="cd repos && hg-ssh user/thomas/* projects/{mercurial,foo}"
26 command="cd repos && hg-ssh user/thomas/* projects/{mercurial,foo}"
26 """
27 """
27
28
28 # enable importing on demand to reduce startup time
29 # enable importing on demand to reduce startup time
29 from mercurial import demandimport; demandimport.enable()
30 from mercurial import demandimport; demandimport.enable()
30
31
31 from mercurial import dispatch
32 from mercurial import dispatch
32
33
33 import sys, os
34 import sys, os
34
35
35 cwd = os.getcwd()
36 cwd = os.getcwd()
36 allowed_paths = [os.path.normpath(os.path.join(cwd, os.path.expanduser(path)))
37 allowed_paths = [os.path.normpath(os.path.join(cwd, os.path.expanduser(path)))
37 for path in sys.argv[1:]]
38 for path in sys.argv[1:]]
38 orig_cmd = os.getenv('SSH_ORIGINAL_COMMAND', '?')
39 orig_cmd = os.getenv('SSH_ORIGINAL_COMMAND', '?')
39
40
40 if orig_cmd.startswith('hg -R ') and orig_cmd.endswith(' serve --stdio'):
41 if orig_cmd.startswith('hg -R ') and orig_cmd.endswith(' serve --stdio'):
41 path = orig_cmd[6:-14]
42 path = orig_cmd[6:-14]
42 repo = os.path.normpath(os.path.join(cwd, os.path.expanduser(path)))
43 repo = os.path.normpath(os.path.join(cwd, os.path.expanduser(path)))
43 if repo in allowed_paths:
44 if repo in allowed_paths:
44 dispatch.dispatch(['-R', repo, 'serve', '--stdio'])
45 dispatch.dispatch(['-R', repo, 'serve', '--stdio'])
45 else:
46 else:
46 sys.stderr.write("Illegal repository %r\n" % repo)
47 sys.stderr.write("Illegal repository %r\n" % repo)
47 sys.exit(-1)
48 sys.exit(-1)
48 else:
49 else:
49 sys.stderr.write("Illegal command %r\n" % orig_cmd)
50 sys.stderr.write("Illegal command %r\n" % orig_cmd)
50 sys.exit(-1)
51 sys.exit(-1)
51
52
@@ -1,41 +1,42
1 # Mercurial extension to provide the 'hg children' command
1 # Mercurial extension to provide the 'hg children' command
2 #
2 #
3 # Copyright 2007 by Intevation GmbH <intevation@intevation.de>
3 # Copyright 2007 by Intevation GmbH <intevation@intevation.de>
4 #
4 # Author(s):
5 # Author(s):
5 # Thomas Arendsen Hein <thomas@intevation.de>
6 # Thomas Arendsen Hein <thomas@intevation.de>
6 #
7 #
7 # This software may be used and distributed according to the terms of the
8 # This software may be used and distributed according to the terms of the
8 # GNU General Public License version 2, incorporated herein by reference.
9 # GNU General Public License version 2, incorporated herein by reference.
9
10
10 from mercurial import cmdutil
11 from mercurial import cmdutil
11 from mercurial.commands import templateopts
12 from mercurial.commands import templateopts
12 from mercurial.i18n import _
13 from mercurial.i18n import _
13
14
14
15
15 def children(ui, repo, file_=None, **opts):
16 def children(ui, repo, file_=None, **opts):
16 """show the children of the given or working directory revision
17 """show the children of the given or working directory revision
17
18
18 Print the children of the working directory's revisions. If a
19 Print the children of the working directory's revisions. If a
19 revision is given via --rev/-r, the children of that revision will
20 revision is given via --rev/-r, the children of that revision will
20 be printed. If a file argument is given, revision in which the
21 be printed. If a file argument is given, revision in which the
21 file was last changed (after the working directory revision or the
22 file was last changed (after the working directory revision or the
22 argument to --rev if given) is printed.
23 argument to --rev if given) is printed.
23 """
24 """
24 rev = opts.get('rev')
25 rev = opts.get('rev')
25 if file_:
26 if file_:
26 ctx = repo.filectx(file_, changeid=rev)
27 ctx = repo.filectx(file_, changeid=rev)
27 else:
28 else:
28 ctx = repo[rev]
29 ctx = repo[rev]
29
30
30 displayer = cmdutil.show_changeset(ui, repo, opts)
31 displayer = cmdutil.show_changeset(ui, repo, opts)
31 for cctx in ctx.children():
32 for cctx in ctx.children():
32 displayer.show(cctx)
33 displayer.show(cctx)
33
34
34
35
35 cmdtable = {
36 cmdtable = {
36 "children":
37 "children":
37 (children,
38 (children,
38 [('r', 'rev', '', _('show children of the specified revision')),
39 [('r', 'rev', '', _('show children of the specified revision')),
39 ] + templateopts,
40 ] + templateopts,
40 _('hg children [-r REV] [FILE]')),
41 _('hg children [-r REV] [FILE]')),
41 }
42 }
@@ -1,162 +1,163
1 # churn.py - create a graph of revisions count grouped by template
1 # churn.py - create a graph of revisions count grouped by template
2 #
2 #
3 # Copyright 2006 Josef "Jeff" Sipek <jeffpc@josefsipek.net>
3 # Copyright 2006 Josef "Jeff" Sipek <jeffpc@josefsipek.net>
4 # Copyright 2008 Alexander Solovyov <piranha@piranha.org.ua>
4 # Copyright 2008 Alexander Solovyov <piranha@piranha.org.ua>
5 #
5 #
6 # This software may be used and distributed according to the terms of the
6 # This software may be used and distributed according to the terms of the
7 # GNU General Public License version 2, incorporated herein by reference.
7 # GNU General Public License version 2, incorporated herein by reference.
8
8 '''command to show certain statistics about revision history'''
9 '''command to show certain statistics about revision history'''
9
10
10 from mercurial.i18n import _
11 from mercurial.i18n import _
11 from mercurial import patch, cmdutil, util, templater
12 from mercurial import patch, cmdutil, util, templater
12 import sys
13 import sys
13 import time, datetime
14 import time, datetime
14
15
15 def maketemplater(ui, repo, tmpl):
16 def maketemplater(ui, repo, tmpl):
16 tmpl = templater.parsestring(tmpl, quoted=False)
17 tmpl = templater.parsestring(tmpl, quoted=False)
17 try:
18 try:
18 t = cmdutil.changeset_templater(ui, repo, False, None, None, False)
19 t = cmdutil.changeset_templater(ui, repo, False, None, None, False)
19 except SyntaxError, inst:
20 except SyntaxError, inst:
20 raise util.Abort(inst.args[0])
21 raise util.Abort(inst.args[0])
21 t.use_template(tmpl)
22 t.use_template(tmpl)
22 return t
23 return t
23
24
24 def changedlines(ui, repo, ctx1, ctx2, fns):
25 def changedlines(ui, repo, ctx1, ctx2, fns):
25 lines = 0
26 lines = 0
26 fmatch = cmdutil.match(repo, pats=fns)
27 fmatch = cmdutil.match(repo, pats=fns)
27 diff = ''.join(patch.diff(repo, ctx1.node(), ctx2.node(), fmatch))
28 diff = ''.join(patch.diff(repo, ctx1.node(), ctx2.node(), fmatch))
28 for l in diff.split('\n'):
29 for l in diff.split('\n'):
29 if (l.startswith("+") and not l.startswith("+++ ") or
30 if (l.startswith("+") and not l.startswith("+++ ") or
30 l.startswith("-") and not l.startswith("--- ")):
31 l.startswith("-") and not l.startswith("--- ")):
31 lines += 1
32 lines += 1
32 return lines
33 return lines
33
34
34 def countrate(ui, repo, amap, *pats, **opts):
35 def countrate(ui, repo, amap, *pats, **opts):
35 """Calculate stats"""
36 """Calculate stats"""
36 if opts.get('dateformat'):
37 if opts.get('dateformat'):
37 def getkey(ctx):
38 def getkey(ctx):
38 t, tz = ctx.date()
39 t, tz = ctx.date()
39 date = datetime.datetime(*time.gmtime(float(t) - tz)[:6])
40 date = datetime.datetime(*time.gmtime(float(t) - tz)[:6])
40 return date.strftime(opts['dateformat'])
41 return date.strftime(opts['dateformat'])
41 else:
42 else:
42 tmpl = opts.get('template', '{author|email}')
43 tmpl = opts.get('template', '{author|email}')
43 tmpl = maketemplater(ui, repo, tmpl)
44 tmpl = maketemplater(ui, repo, tmpl)
44 def getkey(ctx):
45 def getkey(ctx):
45 ui.pushbuffer()
46 ui.pushbuffer()
46 tmpl.show(ctx)
47 tmpl.show(ctx)
47 return ui.popbuffer()
48 return ui.popbuffer()
48
49
49 count = pct = 0
50 count = pct = 0
50 rate = {}
51 rate = {}
51 df = False
52 df = False
52 if opts.get('date'):
53 if opts.get('date'):
53 df = util.matchdate(opts['date'])
54 df = util.matchdate(opts['date'])
54
55
55 get = util.cachefunc(lambda r: repo[r].changeset())
56 get = util.cachefunc(lambda r: repo[r].changeset())
56 changeiter, matchfn = cmdutil.walkchangerevs(ui, repo, pats, get, opts)
57 changeiter, matchfn = cmdutil.walkchangerevs(ui, repo, pats, get, opts)
57 for st, rev, fns in changeiter:
58 for st, rev, fns in changeiter:
58 if not st == 'add':
59 if not st == 'add':
59 continue
60 continue
60 if df and not df(get(rev)[2][0]): # doesn't match date format
61 if df and not df(get(rev)[2][0]): # doesn't match date format
61 continue
62 continue
62
63
63 ctx = repo[rev]
64 ctx = repo[rev]
64 key = getkey(ctx)
65 key = getkey(ctx)
65 key = amap.get(key, key) # alias remap
66 key = amap.get(key, key) # alias remap
66 if opts.get('changesets'):
67 if opts.get('changesets'):
67 rate[key] = rate.get(key, 0) + 1
68 rate[key] = rate.get(key, 0) + 1
68 else:
69 else:
69 parents = ctx.parents()
70 parents = ctx.parents()
70 if len(parents) > 1:
71 if len(parents) > 1:
71 ui.note(_('Revision %d is a merge, ignoring...\n') % (rev,))
72 ui.note(_('Revision %d is a merge, ignoring...\n') % (rev,))
72 continue
73 continue
73
74
74 ctx1 = parents[0]
75 ctx1 = parents[0]
75 lines = changedlines(ui, repo, ctx1, ctx, fns)
76 lines = changedlines(ui, repo, ctx1, ctx, fns)
76 rate[key] = rate.get(key, 0) + lines
77 rate[key] = rate.get(key, 0) + lines
77
78
78 if opts.get('progress'):
79 if opts.get('progress'):
79 count += 1
80 count += 1
80 newpct = int(100.0 * count / max(len(repo), 1))
81 newpct = int(100.0 * count / max(len(repo), 1))
81 if pct < newpct:
82 if pct < newpct:
82 pct = newpct
83 pct = newpct
83 ui.write("\r" + _("generating stats: %d%%") % pct)
84 ui.write("\r" + _("generating stats: %d%%") % pct)
84 sys.stdout.flush()
85 sys.stdout.flush()
85
86
86 if opts.get('progress'):
87 if opts.get('progress'):
87 ui.write("\r")
88 ui.write("\r")
88 sys.stdout.flush()
89 sys.stdout.flush()
89
90
90 return rate
91 return rate
91
92
92
93
93 def churn(ui, repo, *pats, **opts):
94 def churn(ui, repo, *pats, **opts):
94 '''graph count of revisions grouped by template
95 '''graph count of revisions grouped by template
95
96
96 Will graph count of changed lines or revisions grouped by template
97 Will graph count of changed lines or revisions grouped by template
97 or alternatively by date, if dateformat is used. In this case it
98 or alternatively by date, if dateformat is used. In this case it
98 will override template.
99 will override template.
99
100
100 By default statistics are counted for number of changed lines.
101 By default statistics are counted for number of changed lines.
101
102
102 Examples:
103 Examples:
103
104
104 # display count of changed lines for every committer
105 # display count of changed lines for every committer
105 hg churn -t '{author|email}'
106 hg churn -t '{author|email}'
106
107
107 # display daily activity graph
108 # display daily activity graph
108 hg churn -f '%H' -s -c
109 hg churn -f '%H' -s -c
109
110
110 # display activity of developers by month
111 # display activity of developers by month
111 hg churn -f '%Y-%m' -s -c
112 hg churn -f '%Y-%m' -s -c
112
113
113 # display count of lines changed in every year
114 # display count of lines changed in every year
114 hg churn -f '%Y' -s
115 hg churn -f '%Y' -s
115
116
116 The map file format used to specify aliases is fairly simple:
117 The map file format used to specify aliases is fairly simple:
117
118
118 <alias email> <actual email>'''
119 <alias email> <actual email>'''
119 def pad(s, l):
120 def pad(s, l):
120 return (s + " " * l)[:l]
121 return (s + " " * l)[:l]
121
122
122 amap = {}
123 amap = {}
123 aliases = opts.get('aliases')
124 aliases = opts.get('aliases')
124 if aliases:
125 if aliases:
125 for l in open(aliases, "r"):
126 for l in open(aliases, "r"):
126 l = l.strip()
127 l = l.strip()
127 alias, actual = l.split()
128 alias, actual = l.split()
128 amap[alias] = actual
129 amap[alias] = actual
129
130
130 rate = countrate(ui, repo, amap, *pats, **opts).items()
131 rate = countrate(ui, repo, amap, *pats, **opts).items()
131 if not rate:
132 if not rate:
132 return
133 return
133
134
134 sortfn = ((not opts.get('sort')) and (lambda a, b: cmp(b[1], a[1])) or None)
135 sortfn = ((not opts.get('sort')) and (lambda a, b: cmp(b[1], a[1])) or None)
135 rate.sort(sortfn)
136 rate.sort(sortfn)
136
137
137 maxcount = float(max([v for k, v in rate]))
138 maxcount = float(max([v for k, v in rate]))
138 maxname = max([len(k) for k, v in rate])
139 maxname = max([len(k) for k, v in rate])
139
140
140 ttywidth = util.termwidth()
141 ttywidth = util.termwidth()
141 ui.debug(_("assuming %i character terminal\n") % ttywidth)
142 ui.debug(_("assuming %i character terminal\n") % ttywidth)
142 width = ttywidth - maxname - 2 - 6 - 2 - 2
143 width = ttywidth - maxname - 2 - 6 - 2 - 2
143
144
144 for date, count in rate:
145 for date, count in rate:
145 print "%s %6d %s" % (pad(date, maxname), count,
146 print "%s %6d %s" % (pad(date, maxname), count,
146 "*" * int(count * width / maxcount))
147 "*" * int(count * width / maxcount))
147
148
148
149
149 cmdtable = {
150 cmdtable = {
150 "churn":
151 "churn":
151 (churn,
152 (churn,
152 [('r', 'rev', [], _('count rate for the specified revision or range')),
153 [('r', 'rev', [], _('count rate for the specified revision or range')),
153 ('d', 'date', '', _('count rate for revisions matching date spec')),
154 ('d', 'date', '', _('count rate for revisions matching date spec')),
154 ('t', 'template', '{author|email}', _('template to group changesets')),
155 ('t', 'template', '{author|email}', _('template to group changesets')),
155 ('f', 'dateformat', '',
156 ('f', 'dateformat', '',
156 _('strftime-compatible format for grouping by date')),
157 _('strftime-compatible format for grouping by date')),
157 ('c', 'changesets', False, _('count rate by number of changesets')),
158 ('c', 'changesets', False, _('count rate by number of changesets')),
158 ('s', 'sort', False, _('sort by key (default: sort by count)')),
159 ('s', 'sort', False, _('sort by key (default: sort by count)')),
159 ('', 'aliases', '', _('file with email aliases')),
160 ('', 'aliases', '', _('file with email aliases')),
160 ('', 'progress', None, _('show progress'))],
161 ('', 'progress', None, _('show progress'))],
161 _("hg churn [-d DATE] [-r REV] [--aliases FILE] [--progress] [FILE]")),
162 _("hg churn [-d DATE] [-r REV] [--aliases FILE] [--progress] [FILE]")),
162 }
163 }
@@ -1,261 +1,262
1 # convert.py Foreign SCM converter
1 # convert.py Foreign SCM converter
2 #
2 #
3 # Copyright 2005-2007 Matt Mackall <mpm@selenic.com>
3 # Copyright 2005-2007 Matt Mackall <mpm@selenic.com>
4 #
4 #
5 # This software may be used and distributed according to the terms of the
5 # This software may be used and distributed according to the terms of the
6 # GNU General Public License version 2, incorporated herein by reference.
6 # GNU General Public License version 2, incorporated herein by reference.
7
7 '''converting foreign VCS repositories to Mercurial'''
8 '''converting foreign VCS repositories to Mercurial'''
8
9
9 import convcmd
10 import convcmd
10 import cvsps
11 import cvsps
11 import subversion
12 import subversion
12 from mercurial import commands
13 from mercurial import commands
13 from mercurial.i18n import _
14 from mercurial.i18n import _
14
15
15 # Commands definition was moved elsewhere to ease demandload job.
16 # Commands definition was moved elsewhere to ease demandload job.
16
17
17 def convert(ui, src, dest=None, revmapfile=None, **opts):
18 def convert(ui, src, dest=None, revmapfile=None, **opts):
18 """convert a foreign SCM repository to a Mercurial one.
19 """convert a foreign SCM repository to a Mercurial one.
19
20
20 Accepted source formats [identifiers]:
21 Accepted source formats [identifiers]:
21 - Mercurial [hg]
22 - Mercurial [hg]
22 - CVS [cvs]
23 - CVS [cvs]
23 - Darcs [darcs]
24 - Darcs [darcs]
24 - git [git]
25 - git [git]
25 - Subversion [svn]
26 - Subversion [svn]
26 - Monotone [mtn]
27 - Monotone [mtn]
27 - GNU Arch [gnuarch]
28 - GNU Arch [gnuarch]
28 - Bazaar [bzr]
29 - Bazaar [bzr]
29 - Perforce [p4]
30 - Perforce [p4]
30
31
31 Accepted destination formats [identifiers]:
32 Accepted destination formats [identifiers]:
32 - Mercurial [hg]
33 - Mercurial [hg]
33 - Subversion [svn] (history on branches is not preserved)
34 - Subversion [svn] (history on branches is not preserved)
34
35
35 If no revision is given, all revisions will be converted.
36 If no revision is given, all revisions will be converted.
36 Otherwise, convert will only import up to the named revision
37 Otherwise, convert will only import up to the named revision
37 (given in a format understood by the source).
38 (given in a format understood by the source).
38
39
39 If no destination directory name is specified, it defaults to the
40 If no destination directory name is specified, it defaults to the
40 basename of the source with '-hg' appended. If the destination
41 basename of the source with '-hg' appended. If the destination
41 repository doesn't exist, it will be created.
42 repository doesn't exist, it will be created.
42
43
43 If <REVMAP> isn't given, it will be put in a default location
44 If <REVMAP> isn't given, it will be put in a default location
44 (<dest>/.hg/shamap by default). The <REVMAP> is a simple text file
45 (<dest>/.hg/shamap by default). The <REVMAP> is a simple text file
45 that maps each source commit ID to the destination ID for that
46 that maps each source commit ID to the destination ID for that
46 revision, like so:
47 revision, like so:
47 <source ID> <destination ID>
48 <source ID> <destination ID>
48
49
49 If the file doesn't exist, it's automatically created. It's
50 If the file doesn't exist, it's automatically created. It's
50 updated on each commit copied, so convert-repo can be interrupted
51 updated on each commit copied, so convert-repo can be interrupted
51 and can be run repeatedly to copy new commits.
52 and can be run repeatedly to copy new commits.
52
53
53 The [username mapping] file is a simple text file that maps each
54 The [username mapping] file is a simple text file that maps each
54 source commit author to a destination commit author. It is handy
55 source commit author to a destination commit author. It is handy
55 for source SCMs that use unix logins to identify authors (eg:
56 for source SCMs that use unix logins to identify authors (eg:
56 CVS). One line per author mapping and the line format is:
57 CVS). One line per author mapping and the line format is:
57 srcauthor=whatever string you want
58 srcauthor=whatever string you want
58
59
59 The filemap is a file that allows filtering and remapping of files
60 The filemap is a file that allows filtering and remapping of files
60 and directories. Comment lines start with '#'. Each line can
61 and directories. Comment lines start with '#'. Each line can
61 contain one of the following directives:
62 contain one of the following directives:
62
63
63 include path/to/file
64 include path/to/file
64
65
65 exclude path/to/file
66 exclude path/to/file
66
67
67 rename from/file to/file
68 rename from/file to/file
68
69
69 The 'include' directive causes a file, or all files under a
70 The 'include' directive causes a file, or all files under a
70 directory, to be included in the destination repository, and the
71 directory, to be included in the destination repository, and the
71 exclusion of all other files and directories not explicitely included.
72 exclusion of all other files and directories not explicitely included.
72 The 'exclude' directive causes files or directories to be omitted.
73 The 'exclude' directive causes files or directories to be omitted.
73 The 'rename' directive renames a file or directory. To rename from
74 The 'rename' directive renames a file or directory. To rename from
74 a subdirectory into the root of the repository, use '.' as the
75 a subdirectory into the root of the repository, use '.' as the
75 path to rename to.
76 path to rename to.
76
77
77 The splicemap is a file that allows insertion of synthetic
78 The splicemap is a file that allows insertion of synthetic
78 history, letting you specify the parents of a revision. This is
79 history, letting you specify the parents of a revision. This is
79 useful if you want to e.g. give a Subversion merge two parents, or
80 useful if you want to e.g. give a Subversion merge two parents, or
80 graft two disconnected series of history together. Each entry
81 graft two disconnected series of history together. Each entry
81 contains a key, followed by a space, followed by one or two
82 contains a key, followed by a space, followed by one or two
82 comma-separated values. The key is the revision ID in the source
83 comma-separated values. The key is the revision ID in the source
83 revision control system whose parents should be modified (same
84 revision control system whose parents should be modified (same
84 format as a key in .hg/shamap). The values are the revision IDs
85 format as a key in .hg/shamap). The values are the revision IDs
85 (in either the source or destination revision control system) that
86 (in either the source or destination revision control system) that
86 should be used as the new parents for that node.
87 should be used as the new parents for that node.
87
88
88 Mercurial Source
89 Mercurial Source
89 -----------------
90 -----------------
90
91
91 --config convert.hg.ignoreerrors=False (boolean)
92 --config convert.hg.ignoreerrors=False (boolean)
92 ignore integrity errors when reading. Use it to fix Mercurial
93 ignore integrity errors when reading. Use it to fix Mercurial
93 repositories with missing revlogs, by converting from and to
94 repositories with missing revlogs, by converting from and to
94 Mercurial.
95 Mercurial.
95 --config convert.hg.saverev=False (boolean)
96 --config convert.hg.saverev=False (boolean)
96 store original revision ID in changeset (forces target IDs to
97 store original revision ID in changeset (forces target IDs to
97 change)
98 change)
98 --config convert.hg.startrev=0 (hg revision identifier)
99 --config convert.hg.startrev=0 (hg revision identifier)
99 convert start revision and its descendants
100 convert start revision and its descendants
100
101
101 CVS Source
102 CVS Source
102 ----------
103 ----------
103
104
104 CVS source will use a sandbox (i.e. a checked-out copy) from CVS
105 CVS source will use a sandbox (i.e. a checked-out copy) from CVS
105 to indicate the starting point of what will be converted. Direct
106 to indicate the starting point of what will be converted. Direct
106 access to the repository files is not needed, unless of course the
107 access to the repository files is not needed, unless of course the
107 repository is :local:. The conversion uses the top level directory
108 repository is :local:. The conversion uses the top level directory
108 in the sandbox to find the CVS repository, and then uses CVS rlog
109 in the sandbox to find the CVS repository, and then uses CVS rlog
109 commands to find files to convert. This means that unless a
110 commands to find files to convert. This means that unless a
110 filemap is given, all files under the starting directory will be
111 filemap is given, all files under the starting directory will be
111 converted, and that any directory reorganisation in the CVS
112 converted, and that any directory reorganisation in the CVS
112 sandbox is ignored.
113 sandbox is ignored.
113
114
114 Because CVS does not have changesets, it is necessary to collect
115 Because CVS does not have changesets, it is necessary to collect
115 individual commits to CVS and merge them into changesets. CVS
116 individual commits to CVS and merge them into changesets. CVS
116 source uses its internal changeset merging code by default but can
117 source uses its internal changeset merging code by default but can
117 be configured to call the external 'cvsps' program by setting:
118 be configured to call the external 'cvsps' program by setting:
118 --config convert.cvsps='cvsps -A -u --cvs-direct -q'
119 --config convert.cvsps='cvsps -A -u --cvs-direct -q'
119 This is a legacy option and may be removed in future.
120 This is a legacy option and may be removed in future.
120
121
121 The options shown are the defaults.
122 The options shown are the defaults.
122
123
123 Internal cvsps is selected by setting
124 Internal cvsps is selected by setting
124 --config convert.cvsps=builtin
125 --config convert.cvsps=builtin
125 and has a few more configurable options:
126 and has a few more configurable options:
126 --config convert.cvsps.cache=True (boolean)
127 --config convert.cvsps.cache=True (boolean)
127 Set to False to disable remote log caching, for testing and
128 Set to False to disable remote log caching, for testing and
128 debugging purposes.
129 debugging purposes.
129 --config convert.cvsps.fuzz=60 (integer)
130 --config convert.cvsps.fuzz=60 (integer)
130 Specify the maximum time (in seconds) that is allowed
131 Specify the maximum time (in seconds) that is allowed
131 between commits with identical user and log message in a
132 between commits with identical user and log message in a
132 single changeset. When very large files were checked in as
133 single changeset. When very large files were checked in as
133 part of a changeset then the default may not be long
134 part of a changeset then the default may not be long
134 enough.
135 enough.
135 --config convert.cvsps.mergeto='{{mergetobranch ([-\w]+)}}'
136 --config convert.cvsps.mergeto='{{mergetobranch ([-\w]+)}}'
136 Specify a regular expression to which commit log messages
137 Specify a regular expression to which commit log messages
137 are matched. If a match occurs, then the conversion
138 are matched. If a match occurs, then the conversion
138 process will insert a dummy revision merging the branch on
139 process will insert a dummy revision merging the branch on
139 which this log message occurs to the branch indicated in
140 which this log message occurs to the branch indicated in
140 the regex.
141 the regex.
141 --config convert.cvsps.mergefrom='{{mergefrombranch ([-\w]+)}}'
142 --config convert.cvsps.mergefrom='{{mergefrombranch ([-\w]+)}}'
142 Specify a regular expression to which commit log messages
143 Specify a regular expression to which commit log messages
143 are matched. If a match occurs, then the conversion
144 are matched. If a match occurs, then the conversion
144 process will add the most recent revision on the branch
145 process will add the most recent revision on the branch
145 indicated in the regex as the second parent of the
146 indicated in the regex as the second parent of the
146 changeset.
147 changeset.
147
148
148 The hgext/convert/cvsps wrapper script allows the builtin
149 The hgext/convert/cvsps wrapper script allows the builtin
149 changeset merging code to be run without doing a conversion. Its
150 changeset merging code to be run without doing a conversion. Its
150 parameters and output are similar to that of cvsps 2.1.
151 parameters and output are similar to that of cvsps 2.1.
151
152
152 Subversion Source
153 Subversion Source
153 -----------------
154 -----------------
154
155
155 Subversion source detects classical trunk/branches/tags layouts.
156 Subversion source detects classical trunk/branches/tags layouts.
156 By default, the supplied "svn://repo/path/" source URL is
157 By default, the supplied "svn://repo/path/" source URL is
157 converted as a single branch. If "svn://repo/path/trunk" exists it
158 converted as a single branch. If "svn://repo/path/trunk" exists it
158 replaces the default branch. If "svn://repo/path/branches" exists,
159 replaces the default branch. If "svn://repo/path/branches" exists,
159 its subdirectories are listed as possible branches. If
160 its subdirectories are listed as possible branches. If
160 "svn://repo/path/tags" exists, it is looked for tags referencing
161 "svn://repo/path/tags" exists, it is looked for tags referencing
161 converted branches. Default "trunk", "branches" and "tags" values
162 converted branches. Default "trunk", "branches" and "tags" values
162 can be overriden with following options. Set them to paths
163 can be overriden with following options. Set them to paths
163 relative to the source URL, or leave them blank to disable
164 relative to the source URL, or leave them blank to disable
164 autodetection.
165 autodetection.
165
166
166 --config convert.svn.branches=branches (directory name)
167 --config convert.svn.branches=branches (directory name)
167 specify the directory containing branches
168 specify the directory containing branches
168 --config convert.svn.tags=tags (directory name)
169 --config convert.svn.tags=tags (directory name)
169 specify the directory containing tags
170 specify the directory containing tags
170 --config convert.svn.trunk=trunk (directory name)
171 --config convert.svn.trunk=trunk (directory name)
171 specify the name of the trunk branch
172 specify the name of the trunk branch
172
173
173 Source history can be retrieved starting at a specific revision,
174 Source history can be retrieved starting at a specific revision,
174 instead of being integrally converted. Only single branch
175 instead of being integrally converted. Only single branch
175 conversions are supported.
176 conversions are supported.
176
177
177 --config convert.svn.startrev=0 (svn revision number)
178 --config convert.svn.startrev=0 (svn revision number)
178 specify start Subversion revision.
179 specify start Subversion revision.
179
180
180 Perforce Source
181 Perforce Source
181 ---------------
182 ---------------
182
183
183 The Perforce (P4) importer can be given a p4 depot path or a
184 The Perforce (P4) importer can be given a p4 depot path or a
184 client specification as source. It will convert all files in the
185 client specification as source. It will convert all files in the
185 source to a flat Mercurial repository, ignoring labels, branches
186 source to a flat Mercurial repository, ignoring labels, branches
186 and integrations. Note that when a depot path is given you then
187 and integrations. Note that when a depot path is given you then
187 usually should specify a target directory, because otherwise the
188 usually should specify a target directory, because otherwise the
188 target may be named ...-hg.
189 target may be named ...-hg.
189
190
190 It is possible to limit the amount of source history to be
191 It is possible to limit the amount of source history to be
191 converted by specifying an initial Perforce revision.
192 converted by specifying an initial Perforce revision.
192
193
193 --config convert.p4.startrev=0 (perforce changelist number)
194 --config convert.p4.startrev=0 (perforce changelist number)
194 specify initial Perforce revision.
195 specify initial Perforce revision.
195
196
196
197
197 Mercurial Destination
198 Mercurial Destination
198 ---------------------
199 ---------------------
199
200
200 --config convert.hg.clonebranches=False (boolean)
201 --config convert.hg.clonebranches=False (boolean)
201 dispatch source branches in separate clones.
202 dispatch source branches in separate clones.
202 --config convert.hg.tagsbranch=default (branch name)
203 --config convert.hg.tagsbranch=default (branch name)
203 tag revisions branch name
204 tag revisions branch name
204 --config convert.hg.usebranchnames=True (boolean)
205 --config convert.hg.usebranchnames=True (boolean)
205 preserve branch names
206 preserve branch names
206
207
207 """
208 """
208 return convcmd.convert(ui, src, dest, revmapfile, **opts)
209 return convcmd.convert(ui, src, dest, revmapfile, **opts)
209
210
210 def debugsvnlog(ui, **opts):
211 def debugsvnlog(ui, **opts):
211 return subversion.debugsvnlog(ui, **opts)
212 return subversion.debugsvnlog(ui, **opts)
212
213
213 def debugcvsps(ui, *args, **opts):
214 def debugcvsps(ui, *args, **opts):
214 '''create changeset information from CVS
215 '''create changeset information from CVS
215
216
216 This command is intended as a debugging tool for the CVS to
217 This command is intended as a debugging tool for the CVS to
217 Mercurial converter, and can be used as a direct replacement for
218 Mercurial converter, and can be used as a direct replacement for
218 cvsps.
219 cvsps.
219
220
220 Hg debugcvsps reads the CVS rlog for current directory (or any
221 Hg debugcvsps reads the CVS rlog for current directory (or any
221 named directory) in the CVS repository, and converts the log to a
222 named directory) in the CVS repository, and converts the log to a
222 series of changesets based on matching commit log entries and
223 series of changesets based on matching commit log entries and
223 dates.'''
224 dates.'''
224 return cvsps.debugcvsps(ui, *args, **opts)
225 return cvsps.debugcvsps(ui, *args, **opts)
225
226
226 commands.norepo += " convert debugsvnlog debugcvsps"
227 commands.norepo += " convert debugsvnlog debugcvsps"
227
228
228 cmdtable = {
229 cmdtable = {
229 "convert":
230 "convert":
230 (convert,
231 (convert,
231 [('A', 'authors', '', _('username mapping filename')),
232 [('A', 'authors', '', _('username mapping filename')),
232 ('d', 'dest-type', '', _('destination repository type')),
233 ('d', 'dest-type', '', _('destination repository type')),
233 ('', 'filemap', '', _('remap file names using contents of file')),
234 ('', 'filemap', '', _('remap file names using contents of file')),
234 ('r', 'rev', '', _('import up to target revision REV')),
235 ('r', 'rev', '', _('import up to target revision REV')),
235 ('s', 'source-type', '', _('source repository type')),
236 ('s', 'source-type', '', _('source repository type')),
236 ('', 'splicemap', '', _('splice synthesized history into place')),
237 ('', 'splicemap', '', _('splice synthesized history into place')),
237 ('', 'datesort', None, _('try to sort changesets by date'))],
238 ('', 'datesort', None, _('try to sort changesets by date'))],
238 _('hg convert [OPTION]... SOURCE [DEST [REVMAP]]')),
239 _('hg convert [OPTION]... SOURCE [DEST [REVMAP]]')),
239 "debugsvnlog":
240 "debugsvnlog":
240 (debugsvnlog,
241 (debugsvnlog,
241 [],
242 [],
242 'hg debugsvnlog'),
243 'hg debugsvnlog'),
243 "debugcvsps":
244 "debugcvsps":
244 (debugcvsps,
245 (debugcvsps,
245 [
246 [
246 # Main options shared with cvsps-2.1
247 # Main options shared with cvsps-2.1
247 ('b', 'branches', [], _('only return changes on specified branches')),
248 ('b', 'branches', [], _('only return changes on specified branches')),
248 ('p', 'prefix', '', _('prefix to remove from file names')),
249 ('p', 'prefix', '', _('prefix to remove from file names')),
249 ('r', 'revisions', [], _('only return changes after or between specified tags')),
250 ('r', 'revisions', [], _('only return changes after or between specified tags')),
250 ('u', 'update-cache', None, _("update cvs log cache")),
251 ('u', 'update-cache', None, _("update cvs log cache")),
251 ('x', 'new-cache', None, _("create new cvs log cache")),
252 ('x', 'new-cache', None, _("create new cvs log cache")),
252 ('z', 'fuzz', 60, _('set commit time fuzz in seconds')),
253 ('z', 'fuzz', 60, _('set commit time fuzz in seconds')),
253 ('', 'root', '', _('specify cvsroot')),
254 ('', 'root', '', _('specify cvsroot')),
254 # Options specific to builtin cvsps
255 # Options specific to builtin cvsps
255 ('', 'parents', '', _('show parent changesets')),
256 ('', 'parents', '', _('show parent changesets')),
256 ('', 'ancestors', '', _('show current changeset in ancestor branches')),
257 ('', 'ancestors', '', _('show current changeset in ancestor branches')),
257 # Options that are ignored for compatibility with cvsps-2.1
258 # Options that are ignored for compatibility with cvsps-2.1
258 ('A', 'cvs-direct', None, _('ignored for compatibility')),
259 ('A', 'cvs-direct', None, _('ignored for compatibility')),
259 ],
260 ],
260 _('hg debugcvsps [OPTION]... [PATH]...')),
261 _('hg debugcvsps [OPTION]... [PATH]...')),
261 }
262 }
@@ -1,146 +1,147
1 # fetch.py - pull and merge remote changes
1 # fetch.py - pull and merge remote changes
2 #
2 #
3 # Copyright 2006 Vadim Gelfer <vadim.gelfer@gmail.com>
3 # Copyright 2006 Vadim Gelfer <vadim.gelfer@gmail.com>
4 #
4 #
5 # This software may be used and distributed according to the terms of the
5 # This software may be used and distributed according to the terms of the
6 # GNU General Public License version 2, incorporated herein by reference.
6 # GNU General Public License version 2, incorporated herein by reference.
7
7 '''pulling, updating and merging in one command'''
8 '''pulling, updating and merging in one command'''
8
9
9 from mercurial.i18n import _
10 from mercurial.i18n import _
10 from mercurial.node import nullid, short
11 from mercurial.node import nullid, short
11 from mercurial import commands, cmdutil, hg, util, url
12 from mercurial import commands, cmdutil, hg, util, url
12 from mercurial.lock import release
13 from mercurial.lock import release
13
14
14 def fetch(ui, repo, source='default', **opts):
15 def fetch(ui, repo, source='default', **opts):
15 '''pull changes from a remote repository, merge new changes if needed.
16 '''pull changes from a remote repository, merge new changes if needed.
16
17
17 This finds all changes from the repository at the specified path
18 This finds all changes from the repository at the specified path
18 or URL and adds them to the local repository.
19 or URL and adds them to the local repository.
19
20
20 If the pulled changes add a new branch head, the head is
21 If the pulled changes add a new branch head, the head is
21 automatically merged, and the result of the merge is committed.
22 automatically merged, and the result of the merge is committed.
22 Otherwise, the working directory is updated to include the new
23 Otherwise, the working directory is updated to include the new
23 changes.
24 changes.
24
25
25 When a merge occurs, the newly pulled changes are assumed to be
26 When a merge occurs, the newly pulled changes are assumed to be
26 "authoritative". The head of the new changes is used as the first
27 "authoritative". The head of the new changes is used as the first
27 parent, with local changes as the second. To switch the merge
28 parent, with local changes as the second. To switch the merge
28 order, use --switch-parent.
29 order, use --switch-parent.
29
30
30 See 'hg help dates' for a list of formats valid for -d/--date.
31 See 'hg help dates' for a list of formats valid for -d/--date.
31 '''
32 '''
32
33
33 date = opts.get('date')
34 date = opts.get('date')
34 if date:
35 if date:
35 opts['date'] = util.parsedate(date)
36 opts['date'] = util.parsedate(date)
36
37
37 parent, p2 = repo.dirstate.parents()
38 parent, p2 = repo.dirstate.parents()
38 branch = repo.dirstate.branch()
39 branch = repo.dirstate.branch()
39 branchnode = repo.branchtags().get(branch)
40 branchnode = repo.branchtags().get(branch)
40 if parent != branchnode:
41 if parent != branchnode:
41 raise util.Abort(_('working dir not at branch tip '
42 raise util.Abort(_('working dir not at branch tip '
42 '(use "hg update" to check out branch tip)'))
43 '(use "hg update" to check out branch tip)'))
43
44
44 if p2 != nullid:
45 if p2 != nullid:
45 raise util.Abort(_('outstanding uncommitted merge'))
46 raise util.Abort(_('outstanding uncommitted merge'))
46
47
47 wlock = lock = None
48 wlock = lock = None
48 try:
49 try:
49 wlock = repo.wlock()
50 wlock = repo.wlock()
50 lock = repo.lock()
51 lock = repo.lock()
51 mod, add, rem, del_ = repo.status()[:4]
52 mod, add, rem, del_ = repo.status()[:4]
52
53
53 if mod or add or rem:
54 if mod or add or rem:
54 raise util.Abort(_('outstanding uncommitted changes'))
55 raise util.Abort(_('outstanding uncommitted changes'))
55 if del_:
56 if del_:
56 raise util.Abort(_('working directory is missing some files'))
57 raise util.Abort(_('working directory is missing some files'))
57 bheads = repo.branchheads(branch)
58 bheads = repo.branchheads(branch)
58 bheads = [head for head in bheads if len(repo[head].children()) == 0]
59 bheads = [head for head in bheads if len(repo[head].children()) == 0]
59 if len(bheads) > 1:
60 if len(bheads) > 1:
60 raise util.Abort(_('multiple heads in this branch '
61 raise util.Abort(_('multiple heads in this branch '
61 '(use "hg heads ." and "hg merge" to merge)'))
62 '(use "hg heads ." and "hg merge" to merge)'))
62
63
63 other = hg.repository(cmdutil.remoteui(repo, opts),
64 other = hg.repository(cmdutil.remoteui(repo, opts),
64 ui.expandpath(source))
65 ui.expandpath(source))
65 ui.status(_('pulling from %s\n') %
66 ui.status(_('pulling from %s\n') %
66 url.hidepassword(ui.expandpath(source)))
67 url.hidepassword(ui.expandpath(source)))
67 revs = None
68 revs = None
68 if opts['rev']:
69 if opts['rev']:
69 if not other.local():
70 if not other.local():
70 raise util.Abort(_("fetch -r doesn't work for remote "
71 raise util.Abort(_("fetch -r doesn't work for remote "
71 "repositories yet"))
72 "repositories yet"))
72 else:
73 else:
73 revs = [other.lookup(rev) for rev in opts['rev']]
74 revs = [other.lookup(rev) for rev in opts['rev']]
74
75
75 # Are there any changes at all?
76 # Are there any changes at all?
76 modheads = repo.pull(other, heads=revs)
77 modheads = repo.pull(other, heads=revs)
77 if modheads == 0:
78 if modheads == 0:
78 return 0
79 return 0
79
80
80 # Is this a simple fast-forward along the current branch?
81 # Is this a simple fast-forward along the current branch?
81 newheads = repo.branchheads(branch)
82 newheads = repo.branchheads(branch)
82 newheads = [head for head in newheads if len(repo[head].children()) == 0]
83 newheads = [head for head in newheads if len(repo[head].children()) == 0]
83 newchildren = repo.changelog.nodesbetween([parent], newheads)[2]
84 newchildren = repo.changelog.nodesbetween([parent], newheads)[2]
84 if len(newheads) == 1:
85 if len(newheads) == 1:
85 if newchildren[0] != parent:
86 if newchildren[0] != parent:
86 return hg.clean(repo, newchildren[0])
87 return hg.clean(repo, newchildren[0])
87 else:
88 else:
88 return
89 return
89
90
90 # Are there more than one additional branch heads?
91 # Are there more than one additional branch heads?
91 newchildren = [n for n in newchildren if n != parent]
92 newchildren = [n for n in newchildren if n != parent]
92 newparent = parent
93 newparent = parent
93 if newchildren:
94 if newchildren:
94 newparent = newchildren[0]
95 newparent = newchildren[0]
95 hg.clean(repo, newparent)
96 hg.clean(repo, newparent)
96 newheads = [n for n in newheads if n != newparent]
97 newheads = [n for n in newheads if n != newparent]
97 if len(newheads) > 1:
98 if len(newheads) > 1:
98 ui.status(_('not merging with %d other new branch heads '
99 ui.status(_('not merging with %d other new branch heads '
99 '(use "hg heads ." and "hg merge" to merge them)\n') %
100 '(use "hg heads ." and "hg merge" to merge them)\n') %
100 (len(newheads) - 1))
101 (len(newheads) - 1))
101 return
102 return
102
103
103 # Otherwise, let's merge.
104 # Otherwise, let's merge.
104 err = False
105 err = False
105 if newheads:
106 if newheads:
106 # By default, we consider the repository we're pulling
107 # By default, we consider the repository we're pulling
107 # *from* as authoritative, so we merge our changes into
108 # *from* as authoritative, so we merge our changes into
108 # theirs.
109 # theirs.
109 if opts['switch_parent']:
110 if opts['switch_parent']:
110 firstparent, secondparent = newparent, newheads[0]
111 firstparent, secondparent = newparent, newheads[0]
111 else:
112 else:
112 firstparent, secondparent = newheads[0], newparent
113 firstparent, secondparent = newheads[0], newparent
113 ui.status(_('updating to %d:%s\n') %
114 ui.status(_('updating to %d:%s\n') %
114 (repo.changelog.rev(firstparent),
115 (repo.changelog.rev(firstparent),
115 short(firstparent)))
116 short(firstparent)))
116 hg.clean(repo, firstparent)
117 hg.clean(repo, firstparent)
117 ui.status(_('merging with %d:%s\n') %
118 ui.status(_('merging with %d:%s\n') %
118 (repo.changelog.rev(secondparent), short(secondparent)))
119 (repo.changelog.rev(secondparent), short(secondparent)))
119 err = hg.merge(repo, secondparent, remind=False)
120 err = hg.merge(repo, secondparent, remind=False)
120
121
121 if not err:
122 if not err:
122 mod, add, rem = repo.status()[:3]
123 mod, add, rem = repo.status()[:3]
123 message = (cmdutil.logmessage(opts) or
124 message = (cmdutil.logmessage(opts) or
124 (_('Automated merge with %s') %
125 (_('Automated merge with %s') %
125 url.removeauth(other.url())))
126 url.removeauth(other.url())))
126 force_editor = opts.get('force_editor') or opts.get('edit')
127 force_editor = opts.get('force_editor') or opts.get('edit')
127 n = repo.commit(mod + add + rem, message,
128 n = repo.commit(mod + add + rem, message,
128 opts['user'], opts['date'], force=True,
129 opts['user'], opts['date'], force=True,
129 force_editor=force_editor)
130 force_editor=force_editor)
130 ui.status(_('new changeset %d:%s merges remote changes '
131 ui.status(_('new changeset %d:%s merges remote changes '
131 'with local\n') % (repo.changelog.rev(n),
132 'with local\n') % (repo.changelog.rev(n),
132 short(n)))
133 short(n)))
133
134
134 finally:
135 finally:
135 release(lock, wlock)
136 release(lock, wlock)
136
137
137 cmdtable = {
138 cmdtable = {
138 'fetch':
139 'fetch':
139 (fetch,
140 (fetch,
140 [('r', 'rev', [], _('a specific revision you would like to pull')),
141 [('r', 'rev', [], _('a specific revision you would like to pull')),
141 ('e', 'edit', None, _('edit commit message')),
142 ('e', 'edit', None, _('edit commit message')),
142 ('', 'force-editor', None, _('edit commit message (DEPRECATED)')),
143 ('', 'force-editor', None, _('edit commit message (DEPRECATED)')),
143 ('', 'switch-parent', None, _('switch parents when merging')),
144 ('', 'switch-parent', None, _('switch parents when merging')),
144 ] + commands.commitopts + commands.commitopts2 + commands.remoteopts,
145 ] + commands.commitopts + commands.commitopts2 + commands.remoteopts,
145 _('hg fetch [SOURCE]')),
146 _('hg fetch [SOURCE]')),
146 }
147 }
@@ -1,415 +1,416
1 # ASCII graph log extension for Mercurial
1 # ASCII graph log extension for Mercurial
2 #
2 #
3 # Copyright 2007 Joel Rosdahl <joel@rosdahl.net>
3 # Copyright 2007 Joel Rosdahl <joel@rosdahl.net>
4 #
4 #
5 # This software may be used and distributed according to the terms of the
5 # This software may be used and distributed according to the terms of the
6 # GNU General Public License version 2, incorporated herein by reference.
6 # GNU General Public License version 2, incorporated herein by reference.
7
7 '''show revision graphs in terminal windows
8 '''show revision graphs in terminal windows
8
9
9 This extension adds a --graph option to the incoming, outgoing and log
10 This extension adds a --graph option to the incoming, outgoing and log
10 commands. When this options is given, an ascii representation of the
11 commands. When this options is given, an ascii representation of the
11 revision graph is also shown.
12 revision graph is also shown.
12 '''
13 '''
13
14
14 import os
15 import os
15 from mercurial.cmdutil import revrange, show_changeset
16 from mercurial.cmdutil import revrange, show_changeset
16 from mercurial.commands import templateopts
17 from mercurial.commands import templateopts
17 from mercurial.i18n import _
18 from mercurial.i18n import _
18 from mercurial.node import nullrev
19 from mercurial.node import nullrev
19 from mercurial import bundlerepo, changegroup, cmdutil, commands, extensions
20 from mercurial import bundlerepo, changegroup, cmdutil, commands, extensions
20 from mercurial import hg, url, util
21 from mercurial import hg, url, util
21
22
22 def revisions(repo, start, stop):
23 def revisions(repo, start, stop):
23 """cset DAG generator yielding (rev, node, [parents]) tuples
24 """cset DAG generator yielding (rev, node, [parents]) tuples
24
25
25 This generator function walks through the revision history from revision
26 This generator function walks through the revision history from revision
26 start to revision stop (which must be less than or equal to start).
27 start to revision stop (which must be less than or equal to start).
27 """
28 """
28 assert start >= stop
29 assert start >= stop
29 cur = start
30 cur = start
30 while cur >= stop:
31 while cur >= stop:
31 ctx = repo[cur]
32 ctx = repo[cur]
32 parents = [p.rev() for p in ctx.parents() if p.rev() != nullrev]
33 parents = [p.rev() for p in ctx.parents() if p.rev() != nullrev]
33 parents.sort()
34 parents.sort()
34 yield (ctx, parents)
35 yield (ctx, parents)
35 cur -= 1
36 cur -= 1
36
37
37 def filerevs(repo, path, start, stop):
38 def filerevs(repo, path, start, stop):
38 """file cset DAG generator yielding (rev, node, [parents]) tuples
39 """file cset DAG generator yielding (rev, node, [parents]) tuples
39
40
40 This generator function walks through the revision history of a single
41 This generator function walks through the revision history of a single
41 file from revision start to revision stop (which must be less than or
42 file from revision start to revision stop (which must be less than or
42 equal to start).
43 equal to start).
43 """
44 """
44 assert start >= stop
45 assert start >= stop
45 filerev = len(repo.file(path)) - 1
46 filerev = len(repo.file(path)) - 1
46 while filerev >= 0:
47 while filerev >= 0:
47 fctx = repo.filectx(path, fileid=filerev)
48 fctx = repo.filectx(path, fileid=filerev)
48 parents = [f.linkrev() for f in fctx.parents() if f.path() == path]
49 parents = [f.linkrev() for f in fctx.parents() if f.path() == path]
49 parents.sort()
50 parents.sort()
50 if fctx.rev() <= start:
51 if fctx.rev() <= start:
51 yield (fctx, parents)
52 yield (fctx, parents)
52 if fctx.rev() <= stop:
53 if fctx.rev() <= stop:
53 break
54 break
54 filerev -= 1
55 filerev -= 1
55
56
56 def grapher(nodes):
57 def grapher(nodes):
57 """grapher for asciigraph on a list of nodes and their parents
58 """grapher for asciigraph on a list of nodes and their parents
58
59
59 nodes must generate tuples (node, parents, char, lines) where
60 nodes must generate tuples (node, parents, char, lines) where
60 - parents must generate the parents of node, in sorted order,
61 - parents must generate the parents of node, in sorted order,
61 and max length 2,
62 and max length 2,
62 - char is the char to print as the node symbol, and
63 - char is the char to print as the node symbol, and
63 - lines are the lines to display next to the node.
64 - lines are the lines to display next to the node.
64 """
65 """
65 seen = []
66 seen = []
66 for node, parents, char, lines in nodes:
67 for node, parents, char, lines in nodes:
67 if node not in seen:
68 if node not in seen:
68 seen.append(node)
69 seen.append(node)
69 nodeidx = seen.index(node)
70 nodeidx = seen.index(node)
70
71
71 knownparents = []
72 knownparents = []
72 newparents = []
73 newparents = []
73 for parent in parents:
74 for parent in parents:
74 if parent in seen:
75 if parent in seen:
75 knownparents.append(parent)
76 knownparents.append(parent)
76 else:
77 else:
77 newparents.append(parent)
78 newparents.append(parent)
78
79
79 ncols = len(seen)
80 ncols = len(seen)
80 nextseen = seen[:]
81 nextseen = seen[:]
81 nextseen[nodeidx:nodeidx + 1] = newparents
82 nextseen[nodeidx:nodeidx + 1] = newparents
82 edges = [(nodeidx, nextseen.index(p)) for p in knownparents]
83 edges = [(nodeidx, nextseen.index(p)) for p in knownparents]
83
84
84 if len(newparents) > 0:
85 if len(newparents) > 0:
85 edges.append((nodeidx, nodeidx))
86 edges.append((nodeidx, nodeidx))
86 if len(newparents) > 1:
87 if len(newparents) > 1:
87 edges.append((nodeidx, nodeidx + 1))
88 edges.append((nodeidx, nodeidx + 1))
88 nmorecols = len(nextseen) - ncols
89 nmorecols = len(nextseen) - ncols
89 seen = nextseen
90 seen = nextseen
90 yield (char, lines, nodeidx, edges, ncols, nmorecols)
91 yield (char, lines, nodeidx, edges, ncols, nmorecols)
91
92
92 def fix_long_right_edges(edges):
93 def fix_long_right_edges(edges):
93 for (i, (start, end)) in enumerate(edges):
94 for (i, (start, end)) in enumerate(edges):
94 if end > start:
95 if end > start:
95 edges[i] = (start, end + 1)
96 edges[i] = (start, end + 1)
96
97
97 def get_nodeline_edges_tail(
98 def get_nodeline_edges_tail(
98 node_index, p_node_index, n_columns, n_columns_diff, p_diff, fix_tail):
99 node_index, p_node_index, n_columns, n_columns_diff, p_diff, fix_tail):
99 if fix_tail and n_columns_diff == p_diff and n_columns_diff != 0:
100 if fix_tail and n_columns_diff == p_diff and n_columns_diff != 0:
100 # Still going in the same non-vertical direction.
101 # Still going in the same non-vertical direction.
101 if n_columns_diff == -1:
102 if n_columns_diff == -1:
102 start = max(node_index + 1, p_node_index)
103 start = max(node_index + 1, p_node_index)
103 tail = ["|", " "] * (start - node_index - 1)
104 tail = ["|", " "] * (start - node_index - 1)
104 tail.extend(["/", " "] * (n_columns - start))
105 tail.extend(["/", " "] * (n_columns - start))
105 return tail
106 return tail
106 else:
107 else:
107 return ["\\", " "] * (n_columns - node_index - 1)
108 return ["\\", " "] * (n_columns - node_index - 1)
108 else:
109 else:
109 return ["|", " "] * (n_columns - node_index - 1)
110 return ["|", " "] * (n_columns - node_index - 1)
110
111
111 def draw_edges(edges, nodeline, interline):
112 def draw_edges(edges, nodeline, interline):
112 for (start, end) in edges:
113 for (start, end) in edges:
113 if start == end + 1:
114 if start == end + 1:
114 interline[2 * end + 1] = "/"
115 interline[2 * end + 1] = "/"
115 elif start == end - 1:
116 elif start == end - 1:
116 interline[2 * start + 1] = "\\"
117 interline[2 * start + 1] = "\\"
117 elif start == end:
118 elif start == end:
118 interline[2 * start] = "|"
119 interline[2 * start] = "|"
119 else:
120 else:
120 nodeline[2 * end] = "+"
121 nodeline[2 * end] = "+"
121 if start > end:
122 if start > end:
122 (start, end) = (end,start)
123 (start, end) = (end,start)
123 for i in range(2 * start + 1, 2 * end):
124 for i in range(2 * start + 1, 2 * end):
124 if nodeline[i] != "+":
125 if nodeline[i] != "+":
125 nodeline[i] = "-"
126 nodeline[i] = "-"
126
127
127 def get_padding_line(ni, n_columns, edges):
128 def get_padding_line(ni, n_columns, edges):
128 line = []
129 line = []
129 line.extend(["|", " "] * ni)
130 line.extend(["|", " "] * ni)
130 if (ni, ni - 1) in edges or (ni, ni) in edges:
131 if (ni, ni - 1) in edges or (ni, ni) in edges:
131 # (ni, ni - 1) (ni, ni)
132 # (ni, ni - 1) (ni, ni)
132 # | | | | | | | |
133 # | | | | | | | |
133 # +---o | | o---+
134 # +---o | | o---+
134 # | | c | | c | |
135 # | | c | | c | |
135 # | |/ / | |/ /
136 # | |/ / | |/ /
136 # | | | | | |
137 # | | | | | |
137 c = "|"
138 c = "|"
138 else:
139 else:
139 c = " "
140 c = " "
140 line.extend([c, " "])
141 line.extend([c, " "])
141 line.extend(["|", " "] * (n_columns - ni - 1))
142 line.extend(["|", " "] * (n_columns - ni - 1))
142 return line
143 return line
143
144
144 def ascii(ui, grapher):
145 def ascii(ui, grapher):
145 """prints an ASCII graph of the DAG returned by the grapher
146 """prints an ASCII graph of the DAG returned by the grapher
146
147
147 grapher is a generator that emits tuples with the following elements:
148 grapher is a generator that emits tuples with the following elements:
148
149
149 - Character to use as node's symbol.
150 - Character to use as node's symbol.
150 - List of lines to display as the node's text.
151 - List of lines to display as the node's text.
151 - Column of the current node in the set of ongoing edges.
152 - Column of the current node in the set of ongoing edges.
152 - Edges; a list of (col, next_col) indicating the edges between
153 - Edges; a list of (col, next_col) indicating the edges between
153 the current node and its parents.
154 the current node and its parents.
154 - Number of columns (ongoing edges) in the current revision.
155 - Number of columns (ongoing edges) in the current revision.
155 - The difference between the number of columns (ongoing edges)
156 - The difference between the number of columns (ongoing edges)
156 in the next revision and the number of columns (ongoing edges)
157 in the next revision and the number of columns (ongoing edges)
157 in the current revision. That is: -1 means one column removed;
158 in the current revision. That is: -1 means one column removed;
158 0 means no columns added or removed; 1 means one column added.
159 0 means no columns added or removed; 1 means one column added.
159 """
160 """
160 prev_n_columns_diff = 0
161 prev_n_columns_diff = 0
161 prev_node_index = 0
162 prev_node_index = 0
162 for (node_ch, node_lines, node_index, edges, n_columns, n_columns_diff) in grapher:
163 for (node_ch, node_lines, node_index, edges, n_columns, n_columns_diff) in grapher:
163
164
164 assert -2 < n_columns_diff < 2
165 assert -2 < n_columns_diff < 2
165 if n_columns_diff == -1:
166 if n_columns_diff == -1:
166 # Transform
167 # Transform
167 #
168 #
168 # | | | | | |
169 # | | | | | |
169 # o | | into o---+
170 # o | | into o---+
170 # |X / |/ /
171 # |X / |/ /
171 # | | | |
172 # | | | |
172 fix_long_right_edges(edges)
173 fix_long_right_edges(edges)
173
174
174 # add_padding_line says whether to rewrite
175 # add_padding_line says whether to rewrite
175 #
176 #
176 # | | | | | | | |
177 # | | | | | | | |
177 # | o---+ into | o---+
178 # | o---+ into | o---+
178 # | / / | | | # <--- padding line
179 # | / / | | | # <--- padding line
179 # o | | | / /
180 # o | | | / /
180 # o | |
181 # o | |
181 add_padding_line = (len(node_lines) > 2 and
182 add_padding_line = (len(node_lines) > 2 and
182 n_columns_diff == -1 and
183 n_columns_diff == -1 and
183 [x for (x, y) in edges if x + 1 < y])
184 [x for (x, y) in edges if x + 1 < y])
184
185
185 # fix_nodeline_tail says whether to rewrite
186 # fix_nodeline_tail says whether to rewrite
186 #
187 #
187 # | | o | | | | o | |
188 # | | o | | | | o | |
188 # | | |/ / | | |/ /
189 # | | |/ / | | |/ /
189 # | o | | into | o / / # <--- fixed nodeline tail
190 # | o | | into | o / / # <--- fixed nodeline tail
190 # | |/ / | |/ /
191 # | |/ / | |/ /
191 # o | | o | |
192 # o | | o | |
192 fix_nodeline_tail = len(node_lines) <= 2 and not add_padding_line
193 fix_nodeline_tail = len(node_lines) <= 2 and not add_padding_line
193
194
194 # nodeline is the line containing the node character (typically o)
195 # nodeline is the line containing the node character (typically o)
195 nodeline = ["|", " "] * node_index
196 nodeline = ["|", " "] * node_index
196 nodeline.extend([node_ch, " "])
197 nodeline.extend([node_ch, " "])
197
198
198 nodeline.extend(
199 nodeline.extend(
199 get_nodeline_edges_tail(
200 get_nodeline_edges_tail(
200 node_index, prev_node_index, n_columns, n_columns_diff,
201 node_index, prev_node_index, n_columns, n_columns_diff,
201 prev_n_columns_diff, fix_nodeline_tail))
202 prev_n_columns_diff, fix_nodeline_tail))
202
203
203 # shift_interline is the line containing the non-vertical
204 # shift_interline is the line containing the non-vertical
204 # edges between this entry and the next
205 # edges between this entry and the next
205 shift_interline = ["|", " "] * node_index
206 shift_interline = ["|", " "] * node_index
206 if n_columns_diff == -1:
207 if n_columns_diff == -1:
207 n_spaces = 1
208 n_spaces = 1
208 edge_ch = "/"
209 edge_ch = "/"
209 elif n_columns_diff == 0:
210 elif n_columns_diff == 0:
210 n_spaces = 2
211 n_spaces = 2
211 edge_ch = "|"
212 edge_ch = "|"
212 else:
213 else:
213 n_spaces = 3
214 n_spaces = 3
214 edge_ch = "\\"
215 edge_ch = "\\"
215 shift_interline.extend(n_spaces * [" "])
216 shift_interline.extend(n_spaces * [" "])
216 shift_interline.extend([edge_ch, " "] * (n_columns - node_index - 1))
217 shift_interline.extend([edge_ch, " "] * (n_columns - node_index - 1))
217
218
218 # draw edges from the current node to its parents
219 # draw edges from the current node to its parents
219 draw_edges(edges, nodeline, shift_interline)
220 draw_edges(edges, nodeline, shift_interline)
220
221
221 # lines is the list of all graph lines to print
222 # lines is the list of all graph lines to print
222 lines = [nodeline]
223 lines = [nodeline]
223 if add_padding_line:
224 if add_padding_line:
224 lines.append(get_padding_line(node_index, n_columns, edges))
225 lines.append(get_padding_line(node_index, n_columns, edges))
225 lines.append(shift_interline)
226 lines.append(shift_interline)
226
227
227 # make sure that there are as many graph lines as there are
228 # make sure that there are as many graph lines as there are
228 # log strings
229 # log strings
229 while len(node_lines) < len(lines):
230 while len(node_lines) < len(lines):
230 node_lines.append("")
231 node_lines.append("")
231 if len(lines) < len(node_lines):
232 if len(lines) < len(node_lines):
232 extra_interline = ["|", " "] * (n_columns + n_columns_diff)
233 extra_interline = ["|", " "] * (n_columns + n_columns_diff)
233 while len(lines) < len(node_lines):
234 while len(lines) < len(node_lines):
234 lines.append(extra_interline)
235 lines.append(extra_interline)
235
236
236 # print lines
237 # print lines
237 indentation_level = max(n_columns, n_columns + n_columns_diff)
238 indentation_level = max(n_columns, n_columns + n_columns_diff)
238 for (line, logstr) in zip(lines, node_lines):
239 for (line, logstr) in zip(lines, node_lines):
239 ln = "%-*s %s" % (2 * indentation_level, "".join(line), logstr)
240 ln = "%-*s %s" % (2 * indentation_level, "".join(line), logstr)
240 ui.write(ln.rstrip() + '\n')
241 ui.write(ln.rstrip() + '\n')
241
242
242 # ... and start over
243 # ... and start over
243 prev_node_index = node_index
244 prev_node_index = node_index
244 prev_n_columns_diff = n_columns_diff
245 prev_n_columns_diff = n_columns_diff
245
246
246 def get_revs(repo, rev_opt):
247 def get_revs(repo, rev_opt):
247 if rev_opt:
248 if rev_opt:
248 revs = revrange(repo, rev_opt)
249 revs = revrange(repo, rev_opt)
249 return (max(revs), min(revs))
250 return (max(revs), min(revs))
250 else:
251 else:
251 return (len(repo) - 1, 0)
252 return (len(repo) - 1, 0)
252
253
253 def check_unsupported_flags(opts):
254 def check_unsupported_flags(opts):
254 for op in ["follow", "follow_first", "date", "copies", "keyword", "remove",
255 for op in ["follow", "follow_first", "date", "copies", "keyword", "remove",
255 "only_merges", "user", "only_branch", "prune", "newest_first",
256 "only_merges", "user", "only_branch", "prune", "newest_first",
256 "no_merges", "include", "exclude"]:
257 "no_merges", "include", "exclude"]:
257 if op in opts and opts[op]:
258 if op in opts and opts[op]:
258 raise util.Abort(_("--graph option is incompatible with --%s") % op)
259 raise util.Abort(_("--graph option is incompatible with --%s") % op)
259
260
260 def graphlog(ui, repo, path=None, **opts):
261 def graphlog(ui, repo, path=None, **opts):
261 """show revision history alongside an ASCII revision graph
262 """show revision history alongside an ASCII revision graph
262
263
263 Print a revision history alongside a revision graph drawn with
264 Print a revision history alongside a revision graph drawn with
264 ASCII characters.
265 ASCII characters.
265
266
266 Nodes printed as an @ character are parents of the working
267 Nodes printed as an @ character are parents of the working
267 directory.
268 directory.
268 """
269 """
269
270
270 check_unsupported_flags(opts)
271 check_unsupported_flags(opts)
271 limit = cmdutil.loglimit(opts)
272 limit = cmdutil.loglimit(opts)
272 start, stop = get_revs(repo, opts["rev"])
273 start, stop = get_revs(repo, opts["rev"])
273 stop = max(stop, start - limit + 1)
274 stop = max(stop, start - limit + 1)
274 if start == nullrev:
275 if start == nullrev:
275 return
276 return
276
277
277 if path:
278 if path:
278 path = util.canonpath(repo.root, os.getcwd(), path)
279 path = util.canonpath(repo.root, os.getcwd(), path)
279 if path: # could be reset in canonpath
280 if path: # could be reset in canonpath
280 revdag = filerevs(repo, path, start, stop)
281 revdag = filerevs(repo, path, start, stop)
281 else:
282 else:
282 revdag = revisions(repo, start, stop)
283 revdag = revisions(repo, start, stop)
283
284
284 graphdag = graphabledag(ui, repo, revdag, opts)
285 graphdag = graphabledag(ui, repo, revdag, opts)
285 ascii(ui, grapher(graphdag))
286 ascii(ui, grapher(graphdag))
286
287
287 def graphrevs(repo, nodes, opts):
288 def graphrevs(repo, nodes, opts):
288 include = set(nodes)
289 include = set(nodes)
289 limit = cmdutil.loglimit(opts)
290 limit = cmdutil.loglimit(opts)
290 count = 0
291 count = 0
291 for node in reversed(nodes):
292 for node in reversed(nodes):
292 if count >= limit:
293 if count >= limit:
293 break
294 break
294 ctx = repo[node]
295 ctx = repo[node]
295 parents = [p.rev() for p in ctx.parents() if p.node() in include]
296 parents = [p.rev() for p in ctx.parents() if p.node() in include]
296 parents.sort()
297 parents.sort()
297 yield (ctx, parents)
298 yield (ctx, parents)
298 count += 1
299 count += 1
299
300
300 def graphabledag(ui, repo, revdag, opts):
301 def graphabledag(ui, repo, revdag, opts):
301 showparents = [ctx.node() for ctx in repo[None].parents()]
302 showparents = [ctx.node() for ctx in repo[None].parents()]
302 displayer = show_changeset(ui, repo, opts, buffered=True)
303 displayer = show_changeset(ui, repo, opts, buffered=True)
303 for (ctx, parents) in revdag:
304 for (ctx, parents) in revdag:
304 displayer.show(ctx)
305 displayer.show(ctx)
305 lines = displayer.hunk.pop(ctx.rev()).split('\n')[:-1]
306 lines = displayer.hunk.pop(ctx.rev()).split('\n')[:-1]
306 char = ctx.node() in showparents and '@' or 'o'
307 char = ctx.node() in showparents and '@' or 'o'
307 yield (ctx.rev(), parents, char, lines)
308 yield (ctx.rev(), parents, char, lines)
308
309
309 def goutgoing(ui, repo, dest=None, **opts):
310 def goutgoing(ui, repo, dest=None, **opts):
310 """show the outgoing changesets alongside an ASCII revision graph
311 """show the outgoing changesets alongside an ASCII revision graph
311
312
312 Print the outgoing changesets alongside a revision graph drawn with
313 Print the outgoing changesets alongside a revision graph drawn with
313 ASCII characters.
314 ASCII characters.
314
315
315 Nodes printed as an @ character are parents of the working
316 Nodes printed as an @ character are parents of the working
316 directory.
317 directory.
317 """
318 """
318
319
319 check_unsupported_flags(opts)
320 check_unsupported_flags(opts)
320 dest, revs, checkout = hg.parseurl(
321 dest, revs, checkout = hg.parseurl(
321 ui.expandpath(dest or 'default-push', dest or 'default'),
322 ui.expandpath(dest or 'default-push', dest or 'default'),
322 opts.get('rev'))
323 opts.get('rev'))
323 if revs:
324 if revs:
324 revs = [repo.lookup(rev) for rev in revs]
325 revs = [repo.lookup(rev) for rev in revs]
325 other = hg.repository(cmdutil.remoteui(ui, opts), dest)
326 other = hg.repository(cmdutil.remoteui(ui, opts), dest)
326 ui.status(_('comparing with %s\n') % url.hidepassword(dest))
327 ui.status(_('comparing with %s\n') % url.hidepassword(dest))
327 o = repo.findoutgoing(other, force=opts.get('force'))
328 o = repo.findoutgoing(other, force=opts.get('force'))
328 if not o:
329 if not o:
329 ui.status(_("no changes found\n"))
330 ui.status(_("no changes found\n"))
330 return
331 return
331
332
332 o = repo.changelog.nodesbetween(o, revs)[0]
333 o = repo.changelog.nodesbetween(o, revs)[0]
333 revdag = graphrevs(repo, o, opts)
334 revdag = graphrevs(repo, o, opts)
334 graphdag = graphabledag(ui, repo, revdag, opts)
335 graphdag = graphabledag(ui, repo, revdag, opts)
335 ascii(ui, grapher(graphdag))
336 ascii(ui, grapher(graphdag))
336
337
337 def gincoming(ui, repo, source="default", **opts):
338 def gincoming(ui, repo, source="default", **opts):
338 """show the incoming changesets alongside an ASCII revision graph
339 """show the incoming changesets alongside an ASCII revision graph
339
340
340 Print the incoming changesets alongside a revision graph drawn with
341 Print the incoming changesets alongside a revision graph drawn with
341 ASCII characters.
342 ASCII characters.
342
343
343 Nodes printed as an @ character are parents of the working
344 Nodes printed as an @ character are parents of the working
344 directory.
345 directory.
345 """
346 """
346
347
347 check_unsupported_flags(opts)
348 check_unsupported_flags(opts)
348 source, revs, checkout = hg.parseurl(ui.expandpath(source), opts.get('rev'))
349 source, revs, checkout = hg.parseurl(ui.expandpath(source), opts.get('rev'))
349 other = hg.repository(cmdutil.remoteui(repo, opts), source)
350 other = hg.repository(cmdutil.remoteui(repo, opts), source)
350 ui.status(_('comparing with %s\n') % url.hidepassword(source))
351 ui.status(_('comparing with %s\n') % url.hidepassword(source))
351 if revs:
352 if revs:
352 revs = [other.lookup(rev) for rev in revs]
353 revs = [other.lookup(rev) for rev in revs]
353 incoming = repo.findincoming(other, heads=revs, force=opts["force"])
354 incoming = repo.findincoming(other, heads=revs, force=opts["force"])
354 if not incoming:
355 if not incoming:
355 try:
356 try:
356 os.unlink(opts["bundle"])
357 os.unlink(opts["bundle"])
357 except:
358 except:
358 pass
359 pass
359 ui.status(_("no changes found\n"))
360 ui.status(_("no changes found\n"))
360 return
361 return
361
362
362 cleanup = None
363 cleanup = None
363 try:
364 try:
364
365
365 fname = opts["bundle"]
366 fname = opts["bundle"]
366 if fname or not other.local():
367 if fname or not other.local():
367 # create a bundle (uncompressed if other repo is not local)
368 # create a bundle (uncompressed if other repo is not local)
368 if revs is None:
369 if revs is None:
369 cg = other.changegroup(incoming, "incoming")
370 cg = other.changegroup(incoming, "incoming")
370 else:
371 else:
371 cg = other.changegroupsubset(incoming, revs, 'incoming')
372 cg = other.changegroupsubset(incoming, revs, 'incoming')
372 bundletype = other.local() and "HG10BZ" or "HG10UN"
373 bundletype = other.local() and "HG10BZ" or "HG10UN"
373 fname = cleanup = changegroup.writebundle(cg, fname, bundletype)
374 fname = cleanup = changegroup.writebundle(cg, fname, bundletype)
374 # keep written bundle?
375 # keep written bundle?
375 if opts["bundle"]:
376 if opts["bundle"]:
376 cleanup = None
377 cleanup = None
377 if not other.local():
378 if not other.local():
378 # use the created uncompressed bundlerepo
379 # use the created uncompressed bundlerepo
379 other = bundlerepo.bundlerepository(ui, repo.root, fname)
380 other = bundlerepo.bundlerepository(ui, repo.root, fname)
380
381
381 chlist = other.changelog.nodesbetween(incoming, revs)[0]
382 chlist = other.changelog.nodesbetween(incoming, revs)[0]
382 revdag = graphrevs(other, chlist, opts)
383 revdag = graphrevs(other, chlist, opts)
383 graphdag = graphabledag(ui, repo, revdag, opts)
384 graphdag = graphabledag(ui, repo, revdag, opts)
384 ascii(ui, grapher(graphdag))
385 ascii(ui, grapher(graphdag))
385
386
386 finally:
387 finally:
387 if hasattr(other, 'close'):
388 if hasattr(other, 'close'):
388 other.close()
389 other.close()
389 if cleanup:
390 if cleanup:
390 os.unlink(cleanup)
391 os.unlink(cleanup)
391
392
392 def uisetup(ui):
393 def uisetup(ui):
393 '''Initialize the extension.'''
394 '''Initialize the extension.'''
394 _wrapcmd(ui, 'log', commands.table, graphlog)
395 _wrapcmd(ui, 'log', commands.table, graphlog)
395 _wrapcmd(ui, 'incoming', commands.table, gincoming)
396 _wrapcmd(ui, 'incoming', commands.table, gincoming)
396 _wrapcmd(ui, 'outgoing', commands.table, goutgoing)
397 _wrapcmd(ui, 'outgoing', commands.table, goutgoing)
397
398
398 def _wrapcmd(ui, cmd, table, wrapfn):
399 def _wrapcmd(ui, cmd, table, wrapfn):
399 '''wrap the command'''
400 '''wrap the command'''
400 def graph(orig, *args, **kwargs):
401 def graph(orig, *args, **kwargs):
401 if kwargs['graph']:
402 if kwargs['graph']:
402 return wrapfn(*args, **kwargs)
403 return wrapfn(*args, **kwargs)
403 return orig(*args, **kwargs)
404 return orig(*args, **kwargs)
404 entry = extensions.wrapcommand(table, cmd, graph)
405 entry = extensions.wrapcommand(table, cmd, graph)
405 entry[1].append(('G', 'graph', None, _("show the revision DAG")))
406 entry[1].append(('G', 'graph', None, _("show the revision DAG")))
406
407
407 cmdtable = {
408 cmdtable = {
408 "glog":
409 "glog":
409 (graphlog,
410 (graphlog,
410 [('l', 'limit', '', _('limit number of changes displayed')),
411 [('l', 'limit', '', _('limit number of changes displayed')),
411 ('p', 'patch', False, _('show patch')),
412 ('p', 'patch', False, _('show patch')),
412 ('r', 'rev', [], _('show the specified revision or range')),
413 ('r', 'rev', [], _('show the specified revision or range')),
413 ] + templateopts,
414 ] + templateopts,
414 _('hg glog [OPTION]... [FILE]')),
415 _('hg glog [OPTION]... [FILE]')),
415 }
416 }
@@ -1,358 +1,359
1 # Minimal support for git commands on an hg repository
1 # Minimal support for git commands on an hg repository
2 #
2 #
3 # Copyright 2005, 2006 Chris Mason <mason@suse.com>
3 # Copyright 2005, 2006 Chris Mason <mason@suse.com>
4 #
4 #
5 # This software may be used and distributed according to the terms of the
5 # This software may be used and distributed according to the terms of the
6 # GNU General Public License version 2, incorporated herein by reference.
6 # GNU General Public License version 2, incorporated herein by reference.
7
7 '''browsing the repository in a graphical way
8 '''browsing the repository in a graphical way
8
9
9 The hgk extension allows browsing the history of a repository in a
10 The hgk extension allows browsing the history of a repository in a
10 graphical way. It requires Tcl/Tk version 8.4 or later. (Tcl/Tk is not
11 graphical way. It requires Tcl/Tk version 8.4 or later. (Tcl/Tk is not
11 distributed with Mercurial.)
12 distributed with Mercurial.)
12
13
13 hgk consists of two parts: a Tcl script that does the displaying and
14 hgk consists of two parts: a Tcl script that does the displaying and
14 querying of information, and an extension to mercurial named hgk.py,
15 querying of information, and an extension to mercurial named hgk.py,
15 which provides hooks for hgk to get information. hgk can be found in
16 which provides hooks for hgk to get information. hgk can be found in
16 the contrib directory, and hgk.py can be found in the hgext directory.
17 the contrib directory, and hgk.py can be found in the hgext directory.
17
18
18 To load the hgext.py extension, add it to your .hgrc file (you have to
19 To load the hgext.py extension, add it to your .hgrc file (you have to
19 use your global $HOME/.hgrc file, not one in a repository). You can
20 use your global $HOME/.hgrc file, not one in a repository). You can
20 specify an absolute path:
21 specify an absolute path:
21
22
22 [extensions]
23 [extensions]
23 hgk=/usr/local/lib/hgk.py
24 hgk=/usr/local/lib/hgk.py
24
25
25 Mercurial can also scan the default python library path for a file
26 Mercurial can also scan the default python library path for a file
26 named 'hgk.py' if you set hgk empty:
27 named 'hgk.py' if you set hgk empty:
27
28
28 [extensions]
29 [extensions]
29 hgk=
30 hgk=
30
31
31 The hg view command will launch the hgk Tcl script. For this command
32 The hg view command will launch the hgk Tcl script. For this command
32 to work, hgk must be in your search path. Alternately, you can specify
33 to work, hgk must be in your search path. Alternately, you can specify
33 the path to hgk in your .hgrc file:
34 the path to hgk in your .hgrc file:
34
35
35 [hgk]
36 [hgk]
36 path=/location/of/hgk
37 path=/location/of/hgk
37
38
38 hgk can make use of the extdiff extension to visualize revisions.
39 hgk can make use of the extdiff extension to visualize revisions.
39 Assuming you had already configured extdiff vdiff command, just add:
40 Assuming you had already configured extdiff vdiff command, just add:
40
41
41 [hgk]
42 [hgk]
42 vdiff=vdiff
43 vdiff=vdiff
43
44
44 Revisions context menu will now display additional entries to fire
45 Revisions context menu will now display additional entries to fire
45 vdiff on hovered and selected revisions.'''
46 vdiff on hovered and selected revisions.'''
46
47
47 import os
48 import os
48 from mercurial import commands, util, patch, revlog, cmdutil
49 from mercurial import commands, util, patch, revlog, cmdutil
49 from mercurial.node import nullid, nullrev, short
50 from mercurial.node import nullid, nullrev, short
50 from mercurial.i18n import _
51 from mercurial.i18n import _
51
52
52 def difftree(ui, repo, node1=None, node2=None, *files, **opts):
53 def difftree(ui, repo, node1=None, node2=None, *files, **opts):
53 """diff trees from two commits"""
54 """diff trees from two commits"""
54 def __difftree(repo, node1, node2, files=[]):
55 def __difftree(repo, node1, node2, files=[]):
55 assert node2 is not None
56 assert node2 is not None
56 mmap = repo[node1].manifest()
57 mmap = repo[node1].manifest()
57 mmap2 = repo[node2].manifest()
58 mmap2 = repo[node2].manifest()
58 m = cmdutil.match(repo, files)
59 m = cmdutil.match(repo, files)
59 modified, added, removed = repo.status(node1, node2, m)[:3]
60 modified, added, removed = repo.status(node1, node2, m)[:3]
60 empty = short(nullid)
61 empty = short(nullid)
61
62
62 for f in modified:
63 for f in modified:
63 # TODO get file permissions
64 # TODO get file permissions
64 ui.write(":100664 100664 %s %s M\t%s\t%s\n" %
65 ui.write(":100664 100664 %s %s M\t%s\t%s\n" %
65 (short(mmap[f]), short(mmap2[f]), f, f))
66 (short(mmap[f]), short(mmap2[f]), f, f))
66 for f in added:
67 for f in added:
67 ui.write(":000000 100664 %s %s N\t%s\t%s\n" %
68 ui.write(":000000 100664 %s %s N\t%s\t%s\n" %
68 (empty, short(mmap2[f]), f, f))
69 (empty, short(mmap2[f]), f, f))
69 for f in removed:
70 for f in removed:
70 ui.write(":100664 000000 %s %s D\t%s\t%s\n" %
71 ui.write(":100664 000000 %s %s D\t%s\t%s\n" %
71 (short(mmap[f]), empty, f, f))
72 (short(mmap[f]), empty, f, f))
72 ##
73 ##
73
74
74 while True:
75 while True:
75 if opts['stdin']:
76 if opts['stdin']:
76 try:
77 try:
77 line = raw_input().split(' ')
78 line = raw_input().split(' ')
78 node1 = line[0]
79 node1 = line[0]
79 if len(line) > 1:
80 if len(line) > 1:
80 node2 = line[1]
81 node2 = line[1]
81 else:
82 else:
82 node2 = None
83 node2 = None
83 except EOFError:
84 except EOFError:
84 break
85 break
85 node1 = repo.lookup(node1)
86 node1 = repo.lookup(node1)
86 if node2:
87 if node2:
87 node2 = repo.lookup(node2)
88 node2 = repo.lookup(node2)
88 else:
89 else:
89 node2 = node1
90 node2 = node1
90 node1 = repo.changelog.parents(node1)[0]
91 node1 = repo.changelog.parents(node1)[0]
91 if opts['patch']:
92 if opts['patch']:
92 if opts['pretty']:
93 if opts['pretty']:
93 catcommit(ui, repo, node2, "")
94 catcommit(ui, repo, node2, "")
94 m = cmdutil.match(repo, files)
95 m = cmdutil.match(repo, files)
95 chunks = patch.diff(repo, node1, node2, match=m,
96 chunks = patch.diff(repo, node1, node2, match=m,
96 opts=patch.diffopts(ui, {'git': True}))
97 opts=patch.diffopts(ui, {'git': True}))
97 for chunk in chunks:
98 for chunk in chunks:
98 repo.ui.write(chunk)
99 repo.ui.write(chunk)
99 else:
100 else:
100 __difftree(repo, node1, node2, files=files)
101 __difftree(repo, node1, node2, files=files)
101 if not opts['stdin']:
102 if not opts['stdin']:
102 break
103 break
103
104
104 def catcommit(ui, repo, n, prefix, ctx=None):
105 def catcommit(ui, repo, n, prefix, ctx=None):
105 nlprefix = '\n' + prefix;
106 nlprefix = '\n' + prefix;
106 if ctx is None:
107 if ctx is None:
107 ctx = repo[n]
108 ctx = repo[n]
108 ui.write("tree %s\n" % short(ctx.changeset()[0])) # use ctx.node() instead ??
109 ui.write("tree %s\n" % short(ctx.changeset()[0])) # use ctx.node() instead ??
109 for p in ctx.parents():
110 for p in ctx.parents():
110 ui.write("parent %s\n" % p)
111 ui.write("parent %s\n" % p)
111
112
112 date = ctx.date()
113 date = ctx.date()
113 description = ctx.description().replace("\0", "")
114 description = ctx.description().replace("\0", "")
114 lines = description.splitlines()
115 lines = description.splitlines()
115 if lines and lines[-1].startswith('committer:'):
116 if lines and lines[-1].startswith('committer:'):
116 committer = lines[-1].split(': ')[1].rstrip()
117 committer = lines[-1].split(': ')[1].rstrip()
117 else:
118 else:
118 committer = ctx.user()
119 committer = ctx.user()
119
120
120 ui.write("author %s %s %s\n" % (ctx.user(), int(date[0]), date[1]))
121 ui.write("author %s %s %s\n" % (ctx.user(), int(date[0]), date[1]))
121 ui.write("committer %s %s %s\n" % (committer, int(date[0]), date[1]))
122 ui.write("committer %s %s %s\n" % (committer, int(date[0]), date[1]))
122 ui.write("revision %d\n" % ctx.rev())
123 ui.write("revision %d\n" % ctx.rev())
123 ui.write("branch %s\n\n" % ctx.branch())
124 ui.write("branch %s\n\n" % ctx.branch())
124
125
125 if prefix != "":
126 if prefix != "":
126 ui.write("%s%s\n" % (prefix, description.replace('\n', nlprefix).strip()))
127 ui.write("%s%s\n" % (prefix, description.replace('\n', nlprefix).strip()))
127 else:
128 else:
128 ui.write(description + "\n")
129 ui.write(description + "\n")
129 if prefix:
130 if prefix:
130 ui.write('\0')
131 ui.write('\0')
131
132
132 def base(ui, repo, node1, node2):
133 def base(ui, repo, node1, node2):
133 """output common ancestor information"""
134 """output common ancestor information"""
134 node1 = repo.lookup(node1)
135 node1 = repo.lookup(node1)
135 node2 = repo.lookup(node2)
136 node2 = repo.lookup(node2)
136 n = repo.changelog.ancestor(node1, node2)
137 n = repo.changelog.ancestor(node1, node2)
137 ui.write(short(n) + "\n")
138 ui.write(short(n) + "\n")
138
139
139 def catfile(ui, repo, type=None, r=None, **opts):
140 def catfile(ui, repo, type=None, r=None, **opts):
140 """cat a specific revision"""
141 """cat a specific revision"""
141 # in stdin mode, every line except the commit is prefixed with two
142 # in stdin mode, every line except the commit is prefixed with two
142 # spaces. This way the our caller can find the commit without magic
143 # spaces. This way the our caller can find the commit without magic
143 # strings
144 # strings
144 #
145 #
145 prefix = ""
146 prefix = ""
146 if opts['stdin']:
147 if opts['stdin']:
147 try:
148 try:
148 (type, r) = raw_input().split(' ');
149 (type, r) = raw_input().split(' ');
149 prefix = " "
150 prefix = " "
150 except EOFError:
151 except EOFError:
151 return
152 return
152
153
153 else:
154 else:
154 if not type or not r:
155 if not type or not r:
155 ui.warn(_("cat-file: type or revision not supplied\n"))
156 ui.warn(_("cat-file: type or revision not supplied\n"))
156 commands.help_(ui, 'cat-file')
157 commands.help_(ui, 'cat-file')
157
158
158 while r:
159 while r:
159 if type != "commit":
160 if type != "commit":
160 ui.warn(_("aborting hg cat-file only understands commits\n"))
161 ui.warn(_("aborting hg cat-file only understands commits\n"))
161 return 1;
162 return 1;
162 n = repo.lookup(r)
163 n = repo.lookup(r)
163 catcommit(ui, repo, n, prefix)
164 catcommit(ui, repo, n, prefix)
164 if opts['stdin']:
165 if opts['stdin']:
165 try:
166 try:
166 (type, r) = raw_input().split(' ');
167 (type, r) = raw_input().split(' ');
167 except EOFError:
168 except EOFError:
168 break
169 break
169 else:
170 else:
170 break
171 break
171
172
172 # git rev-tree is a confusing thing. You can supply a number of
173 # git rev-tree is a confusing thing. You can supply a number of
173 # commit sha1s on the command line, and it walks the commit history
174 # commit sha1s on the command line, and it walks the commit history
174 # telling you which commits are reachable from the supplied ones via
175 # telling you which commits are reachable from the supplied ones via
175 # a bitmask based on arg position.
176 # a bitmask based on arg position.
176 # you can specify a commit to stop at by starting the sha1 with ^
177 # you can specify a commit to stop at by starting the sha1 with ^
177 def revtree(ui, args, repo, full="tree", maxnr=0, parents=False):
178 def revtree(ui, args, repo, full="tree", maxnr=0, parents=False):
178 def chlogwalk():
179 def chlogwalk():
179 count = len(repo)
180 count = len(repo)
180 i = count
181 i = count
181 l = [0] * 100
182 l = [0] * 100
182 chunk = 100
183 chunk = 100
183 while True:
184 while True:
184 if chunk > i:
185 if chunk > i:
185 chunk = i
186 chunk = i
186 i = 0
187 i = 0
187 else:
188 else:
188 i -= chunk
189 i -= chunk
189
190
190 for x in xrange(0, chunk):
191 for x in xrange(0, chunk):
191 if i + x >= count:
192 if i + x >= count:
192 l[chunk - x:] = [0] * (chunk - x)
193 l[chunk - x:] = [0] * (chunk - x)
193 break
194 break
194 if full != None:
195 if full != None:
195 l[x] = repo[i + x]
196 l[x] = repo[i + x]
196 l[x].changeset() # force reading
197 l[x].changeset() # force reading
197 else:
198 else:
198 l[x] = 1
199 l[x] = 1
199 for x in xrange(chunk-1, -1, -1):
200 for x in xrange(chunk-1, -1, -1):
200 if l[x] != 0:
201 if l[x] != 0:
201 yield (i + x, full != None and l[x] or None)
202 yield (i + x, full != None and l[x] or None)
202 if i == 0:
203 if i == 0:
203 break
204 break
204
205
205 # calculate and return the reachability bitmask for sha
206 # calculate and return the reachability bitmask for sha
206 def is_reachable(ar, reachable, sha):
207 def is_reachable(ar, reachable, sha):
207 if len(ar) == 0:
208 if len(ar) == 0:
208 return 1
209 return 1
209 mask = 0
210 mask = 0
210 for i in xrange(len(ar)):
211 for i in xrange(len(ar)):
211 if sha in reachable[i]:
212 if sha in reachable[i]:
212 mask |= 1 << i
213 mask |= 1 << i
213
214
214 return mask
215 return mask
215
216
216 reachable = []
217 reachable = []
217 stop_sha1 = []
218 stop_sha1 = []
218 want_sha1 = []
219 want_sha1 = []
219 count = 0
220 count = 0
220
221
221 # figure out which commits they are asking for and which ones they
222 # figure out which commits they are asking for and which ones they
222 # want us to stop on
223 # want us to stop on
223 for i in xrange(len(args)):
224 for i in xrange(len(args)):
224 if args[i].startswith('^'):
225 if args[i].startswith('^'):
225 s = repo.lookup(args[i][1:])
226 s = repo.lookup(args[i][1:])
226 stop_sha1.append(s)
227 stop_sha1.append(s)
227 want_sha1.append(s)
228 want_sha1.append(s)
228 elif args[i] != 'HEAD':
229 elif args[i] != 'HEAD':
229 want_sha1.append(repo.lookup(args[i]))
230 want_sha1.append(repo.lookup(args[i]))
230
231
231 # calculate the graph for the supplied commits
232 # calculate the graph for the supplied commits
232 for i in xrange(len(want_sha1)):
233 for i in xrange(len(want_sha1)):
233 reachable.append({});
234 reachable.append({});
234 n = want_sha1[i];
235 n = want_sha1[i];
235 visit = [n];
236 visit = [n];
236 reachable[i][n] = 1
237 reachable[i][n] = 1
237 while visit:
238 while visit:
238 n = visit.pop(0)
239 n = visit.pop(0)
239 if n in stop_sha1:
240 if n in stop_sha1:
240 continue
241 continue
241 for p in repo.changelog.parents(n):
242 for p in repo.changelog.parents(n):
242 if p not in reachable[i]:
243 if p not in reachable[i]:
243 reachable[i][p] = 1
244 reachable[i][p] = 1
244 visit.append(p)
245 visit.append(p)
245 if p in stop_sha1:
246 if p in stop_sha1:
246 continue
247 continue
247
248
248 # walk the repository looking for commits that are in our
249 # walk the repository looking for commits that are in our
249 # reachability graph
250 # reachability graph
250 for i, ctx in chlogwalk():
251 for i, ctx in chlogwalk():
251 n = repo.changelog.node(i)
252 n = repo.changelog.node(i)
252 mask = is_reachable(want_sha1, reachable, n)
253 mask = is_reachable(want_sha1, reachable, n)
253 if mask:
254 if mask:
254 parentstr = ""
255 parentstr = ""
255 if parents:
256 if parents:
256 pp = repo.changelog.parents(n)
257 pp = repo.changelog.parents(n)
257 if pp[0] != nullid:
258 if pp[0] != nullid:
258 parentstr += " " + short(pp[0])
259 parentstr += " " + short(pp[0])
259 if pp[1] != nullid:
260 if pp[1] != nullid:
260 parentstr += " " + short(pp[1])
261 parentstr += " " + short(pp[1])
261 if not full:
262 if not full:
262 ui.write("%s%s\n" % (short(n), parentstr))
263 ui.write("%s%s\n" % (short(n), parentstr))
263 elif full == "commit":
264 elif full == "commit":
264 ui.write("%s%s\n" % (short(n), parentstr))
265 ui.write("%s%s\n" % (short(n), parentstr))
265 catcommit(ui, repo, n, ' ', ctx)
266 catcommit(ui, repo, n, ' ', ctx)
266 else:
267 else:
267 (p1, p2) = repo.changelog.parents(n)
268 (p1, p2) = repo.changelog.parents(n)
268 (h, h1, h2) = map(short, (n, p1, p2))
269 (h, h1, h2) = map(short, (n, p1, p2))
269 (i1, i2) = map(repo.changelog.rev, (p1, p2))
270 (i1, i2) = map(repo.changelog.rev, (p1, p2))
270
271
271 date = ctx.date()[0]
272 date = ctx.date()[0]
272 ui.write("%s %s:%s" % (date, h, mask))
273 ui.write("%s %s:%s" % (date, h, mask))
273 mask = is_reachable(want_sha1, reachable, p1)
274 mask = is_reachable(want_sha1, reachable, p1)
274 if i1 != nullrev and mask > 0:
275 if i1 != nullrev and mask > 0:
275 ui.write("%s:%s " % (h1, mask)),
276 ui.write("%s:%s " % (h1, mask)),
276 mask = is_reachable(want_sha1, reachable, p2)
277 mask = is_reachable(want_sha1, reachable, p2)
277 if i2 != nullrev and mask > 0:
278 if i2 != nullrev and mask > 0:
278 ui.write("%s:%s " % (h2, mask))
279 ui.write("%s:%s " % (h2, mask))
279 ui.write("\n")
280 ui.write("\n")
280 if maxnr and count >= maxnr:
281 if maxnr and count >= maxnr:
281 break
282 break
282 count += 1
283 count += 1
283
284
284 def revparse(ui, repo, *revs, **opts):
285 def revparse(ui, repo, *revs, **opts):
285 """parse given revisions"""
286 """parse given revisions"""
286 def revstr(rev):
287 def revstr(rev):
287 if rev == 'HEAD':
288 if rev == 'HEAD':
288 rev = 'tip'
289 rev = 'tip'
289 return revlog.hex(repo.lookup(rev))
290 return revlog.hex(repo.lookup(rev))
290
291
291 for r in revs:
292 for r in revs:
292 revrange = r.split(':', 1)
293 revrange = r.split(':', 1)
293 ui.write('%s\n' % revstr(revrange[0]))
294 ui.write('%s\n' % revstr(revrange[0]))
294 if len(revrange) == 2:
295 if len(revrange) == 2:
295 ui.write('^%s\n' % revstr(revrange[1]))
296 ui.write('^%s\n' % revstr(revrange[1]))
296
297
297 # git rev-list tries to order things by date, and has the ability to stop
298 # git rev-list tries to order things by date, and has the ability to stop
298 # at a given commit without walking the whole repo. TODO add the stop
299 # at a given commit without walking the whole repo. TODO add the stop
299 # parameter
300 # parameter
300 def revlist(ui, repo, *revs, **opts):
301 def revlist(ui, repo, *revs, **opts):
301 """print revisions"""
302 """print revisions"""
302 if opts['header']:
303 if opts['header']:
303 full = "commit"
304 full = "commit"
304 else:
305 else:
305 full = None
306 full = None
306 copy = [x for x in revs]
307 copy = [x for x in revs]
307 revtree(ui, copy, repo, full, opts['max_count'], opts['parents'])
308 revtree(ui, copy, repo, full, opts['max_count'], opts['parents'])
308
309
309 def config(ui, repo, **opts):
310 def config(ui, repo, **opts):
310 """print extension options"""
311 """print extension options"""
311 def writeopt(name, value):
312 def writeopt(name, value):
312 ui.write('k=%s\nv=%s\n' % (name, value))
313 ui.write('k=%s\nv=%s\n' % (name, value))
313
314
314 writeopt('vdiff', ui.config('hgk', 'vdiff', ''))
315 writeopt('vdiff', ui.config('hgk', 'vdiff', ''))
315
316
316
317
317 def view(ui, repo, *etc, **opts):
318 def view(ui, repo, *etc, **opts):
318 "start interactive history viewer"
319 "start interactive history viewer"
319 os.chdir(repo.root)
320 os.chdir(repo.root)
320 optstr = ' '.join(['--%s %s' % (k, v) for k, v in opts.iteritems() if v])
321 optstr = ' '.join(['--%s %s' % (k, v) for k, v in opts.iteritems() if v])
321 cmd = ui.config("hgk", "path", "hgk") + " %s %s" % (optstr, " ".join(etc))
322 cmd = ui.config("hgk", "path", "hgk") + " %s %s" % (optstr, " ".join(etc))
322 ui.debug(_("running %s\n") % cmd)
323 ui.debug(_("running %s\n") % cmd)
323 util.system(cmd)
324 util.system(cmd)
324
325
325 cmdtable = {
326 cmdtable = {
326 "^view":
327 "^view":
327 (view,
328 (view,
328 [('l', 'limit', '', _('limit number of changes displayed'))],
329 [('l', 'limit', '', _('limit number of changes displayed'))],
329 _('hg view [-l LIMIT] [REVRANGE]')),
330 _('hg view [-l LIMIT] [REVRANGE]')),
330 "debug-diff-tree":
331 "debug-diff-tree":
331 (difftree,
332 (difftree,
332 [('p', 'patch', None, _('generate patch')),
333 [('p', 'patch', None, _('generate patch')),
333 ('r', 'recursive', None, _('recursive')),
334 ('r', 'recursive', None, _('recursive')),
334 ('P', 'pretty', None, _('pretty')),
335 ('P', 'pretty', None, _('pretty')),
335 ('s', 'stdin', None, _('stdin')),
336 ('s', 'stdin', None, _('stdin')),
336 ('C', 'copy', None, _('detect copies')),
337 ('C', 'copy', None, _('detect copies')),
337 ('S', 'search', "", _('search'))],
338 ('S', 'search', "", _('search'))],
338 _('hg git-diff-tree [OPTION]... NODE1 NODE2 [FILE]...')),
339 _('hg git-diff-tree [OPTION]... NODE1 NODE2 [FILE]...')),
339 "debug-cat-file":
340 "debug-cat-file":
340 (catfile,
341 (catfile,
341 [('s', 'stdin', None, _('stdin'))],
342 [('s', 'stdin', None, _('stdin'))],
342 _('hg debug-cat-file [OPTION]... TYPE FILE')),
343 _('hg debug-cat-file [OPTION]... TYPE FILE')),
343 "debug-config":
344 "debug-config":
344 (config, [], _('hg debug-config')),
345 (config, [], _('hg debug-config')),
345 "debug-merge-base":
346 "debug-merge-base":
346 (base, [], _('hg debug-merge-base node node')),
347 (base, [], _('hg debug-merge-base node node')),
347 "debug-rev-parse":
348 "debug-rev-parse":
348 (revparse,
349 (revparse,
349 [('', 'default', '', _('ignored'))],
350 [('', 'default', '', _('ignored'))],
350 _('hg debug-rev-parse REV')),
351 _('hg debug-rev-parse REV')),
351 "debug-rev-list":
352 "debug-rev-list":
352 (revlist,
353 (revlist,
353 [('H', 'header', None, _('header')),
354 [('H', 'header', None, _('header')),
354 ('t', 'topo-order', None, _('topo-order')),
355 ('t', 'topo-order', None, _('topo-order')),
355 ('p', 'parents', None, _('parents')),
356 ('p', 'parents', None, _('parents')),
356 ('n', 'max-count', 0, _('max-count'))],
357 ('n', 'max-count', 0, _('max-count'))],
357 _('hg debug-rev-list [options] revs')),
358 _('hg debug-rev-list [options] revs')),
358 }
359 }
@@ -1,96 +1,97
1 # Mercurial extension to make it easy to refer to the parent of a revision
1 # Mercurial extension to make it easy to refer to the parent of a revision
2 #
2 #
3 # Copyright (C) 2007 Alexis S. L. Carvalho <alexis@cecm.usp.br>
3 # Copyright (C) 2007 Alexis S. L. Carvalho <alexis@cecm.usp.br>
4 #
4 #
5 # This software may be used and distributed according to the terms of the
5 # This software may be used and distributed according to the terms of the
6 # GNU General Public License version 2, incorporated herein by reference.
6 # GNU General Public License version 2, incorporated herein by reference.
7
7 '''\
8 '''\
8 use suffixes to refer to ancestor revisions
9 use suffixes to refer to ancestor revisions
9
10
10 This extension allows you to use git-style suffixes to refer to the
11 This extension allows you to use git-style suffixes to refer to the
11 ancestors of a specific revision.
12 ancestors of a specific revision.
12
13
13 For example, if you can refer to a revision as "foo", then:
14 For example, if you can refer to a revision as "foo", then:
14
15
15 - foo^N = Nth parent of foo:
16 - foo^N = Nth parent of foo:
16 foo^0 = foo
17 foo^0 = foo
17 foo^1 = first parent of foo
18 foo^1 = first parent of foo
18 foo^2 = second parent of foo
19 foo^2 = second parent of foo
19 foo^ = foo^1
20 foo^ = foo^1
20
21
21 - foo~N = Nth first grandparent of foo
22 - foo~N = Nth first grandparent of foo
22 foo~0 = foo
23 foo~0 = foo
23 foo~1 = foo^1 = foo^ = first parent of foo
24 foo~1 = foo^1 = foo^ = first parent of foo
24 foo~2 = foo^1^1 = foo^^ = first parent of first parent of foo
25 foo~2 = foo^1^1 = foo^^ = first parent of first parent of foo
25 '''
26 '''
26 from mercurial import error
27 from mercurial import error
27
28
28 def reposetup(ui, repo):
29 def reposetup(ui, repo):
29 if not repo.local():
30 if not repo.local():
30 return
31 return
31
32
32 class parentrevspecrepo(repo.__class__):
33 class parentrevspecrepo(repo.__class__):
33 def lookup(self, key):
34 def lookup(self, key):
34 try:
35 try:
35 _super = super(parentrevspecrepo, self)
36 _super = super(parentrevspecrepo, self)
36 return _super.lookup(key)
37 return _super.lookup(key)
37 except error.RepoError:
38 except error.RepoError:
38 pass
39 pass
39
40
40 circ = key.find('^')
41 circ = key.find('^')
41 tilde = key.find('~')
42 tilde = key.find('~')
42 if circ < 0 and tilde < 0:
43 if circ < 0 and tilde < 0:
43 raise
44 raise
44 elif circ >= 0 and tilde >= 0:
45 elif circ >= 0 and tilde >= 0:
45 end = min(circ, tilde)
46 end = min(circ, tilde)
46 else:
47 else:
47 end = max(circ, tilde)
48 end = max(circ, tilde)
48
49
49 cl = self.changelog
50 cl = self.changelog
50 base = key[:end]
51 base = key[:end]
51 try:
52 try:
52 node = _super.lookup(base)
53 node = _super.lookup(base)
53 except error.RepoError:
54 except error.RepoError:
54 # eek - reraise the first error
55 # eek - reraise the first error
55 return _super.lookup(key)
56 return _super.lookup(key)
56
57
57 rev = cl.rev(node)
58 rev = cl.rev(node)
58 suffix = key[end:]
59 suffix = key[end:]
59 i = 0
60 i = 0
60 while i < len(suffix):
61 while i < len(suffix):
61 # foo^N => Nth parent of foo
62 # foo^N => Nth parent of foo
62 # foo^0 == foo
63 # foo^0 == foo
63 # foo^1 == foo^ == 1st parent of foo
64 # foo^1 == foo^ == 1st parent of foo
64 # foo^2 == 2nd parent of foo
65 # foo^2 == 2nd parent of foo
65 if suffix[i] == '^':
66 if suffix[i] == '^':
66 j = i + 1
67 j = i + 1
67 p = cl.parentrevs(rev)
68 p = cl.parentrevs(rev)
68 if j < len(suffix) and suffix[j].isdigit():
69 if j < len(suffix) and suffix[j].isdigit():
69 j += 1
70 j += 1
70 n = int(suffix[i+1:j])
71 n = int(suffix[i+1:j])
71 if n > 2 or n == 2 and p[1] == -1:
72 if n > 2 or n == 2 and p[1] == -1:
72 raise
73 raise
73 else:
74 else:
74 n = 1
75 n = 1
75 if n:
76 if n:
76 rev = p[n - 1]
77 rev = p[n - 1]
77 i = j
78 i = j
78 # foo~N => Nth first grandparent of foo
79 # foo~N => Nth first grandparent of foo
79 # foo~0 = foo
80 # foo~0 = foo
80 # foo~1 = foo^1 == foo^ == 1st parent of foo
81 # foo~1 = foo^1 == foo^ == 1st parent of foo
81 # foo~2 = foo^1^1 == foo^^ == 1st parent of 1st parent of foo
82 # foo~2 = foo^1^1 == foo^^ == 1st parent of 1st parent of foo
82 elif suffix[i] == '~':
83 elif suffix[i] == '~':
83 j = i + 1
84 j = i + 1
84 while j < len(suffix) and suffix[j].isdigit():
85 while j < len(suffix) and suffix[j].isdigit():
85 j += 1
86 j += 1
86 if j == i + 1:
87 if j == i + 1:
87 raise
88 raise
88 n = int(suffix[i+1:j])
89 n = int(suffix[i+1:j])
89 for k in xrange(n):
90 for k in xrange(n):
90 rev = cl.parentrevs(rev)[0]
91 rev = cl.parentrevs(rev)[0]
91 i = j
92 i = j
92 else:
93 else:
93 raise
94 raise
94 return cl.node(rev)
95 return cl.node(rev)
95
96
96 repo.__class__ = parentrevspecrepo
97 repo.__class__ = parentrevspecrepo
@@ -1,125 +1,126
1 # win32mbcs.py -- MBCS filename support for Mercurial
1 # win32mbcs.py -- MBCS filename support for Mercurial
2 #
2 #
3 # Copyright (c) 2008 Shun-ichi Goto <shunichi.goto@gmail.com>
3 # Copyright (c) 2008 Shun-ichi Goto <shunichi.goto@gmail.com>
4 #
4 #
5 # Version: 0.2
5 # Version: 0.2
6 # Author: Shun-ichi Goto <shunichi.goto@gmail.com>
6 # Author: Shun-ichi Goto <shunichi.goto@gmail.com>
7 #
7 #
8 # This software may be used and distributed according to the terms of the
8 # This software may be used and distributed according to the terms of the
9 # GNU General Public License version 2, incorporated herein by reference.
9 # GNU General Public License version 2, incorporated herein by reference.
10 #
10 #
11
11 """allow to use MBCS path with problematic encoding.
12 """allow to use MBCS path with problematic encoding.
12
13
13 Some MBCS encodings are not good for some path operations (i.e.
14 Some MBCS encodings are not good for some path operations (i.e.
14 splitting path, case conversion, etc.) with its encoded bytes. We call
15 splitting path, case conversion, etc.) with its encoded bytes. We call
15 such a encoding (i.e. shift_jis and big5) as "problematic encoding".
16 such a encoding (i.e. shift_jis and big5) as "problematic encoding".
16 This extension can be used to fix the issue with those encodings by
17 This extension can be used to fix the issue with those encodings by
17 wrapping some functions to convert to unicode string before path
18 wrapping some functions to convert to unicode string before path
18 operation.
19 operation.
19
20
20 This extension is usefull for:
21 This extension is usefull for:
21 * Japanese Windows users using shift_jis encoding.
22 * Japanese Windows users using shift_jis encoding.
22 * Chinese Windows users using big5 encoding.
23 * Chinese Windows users using big5 encoding.
23 * All users who use a repository with one of problematic encodings on
24 * All users who use a repository with one of problematic encodings on
24 case-insensitive file system.
25 case-insensitive file system.
25
26
26 This extension is not needed for:
27 This extension is not needed for:
27 * Any user who use only ascii chars in path.
28 * Any user who use only ascii chars in path.
28 * Any user who do not use any of problematic encodings.
29 * Any user who do not use any of problematic encodings.
29
30
30 Note that there are some limitations on using this extension:
31 Note that there are some limitations on using this extension:
31 * You should use single encoding in one repository.
32 * You should use single encoding in one repository.
32 * You should set same encoding for the repository by locale or
33 * You should set same encoding for the repository by locale or
33 HGENCODING.
34 HGENCODING.
34
35
35 To use this extension, enable the extension in .hg/hgrc or ~/.hgrc:
36 To use this extension, enable the extension in .hg/hgrc or ~/.hgrc:
36
37
37 [extensions]
38 [extensions]
38 hgext.win32mbcs =
39 hgext.win32mbcs =
39
40
40 Path encoding conversion are done between unicode and
41 Path encoding conversion are done between unicode and
41 encoding.encoding which is decided by mercurial from current locale
42 encoding.encoding which is decided by mercurial from current locale
42 setting or HGENCODING.
43 setting or HGENCODING.
43
44
44 """
45 """
45
46
46 import os
47 import os
47 from mercurial.i18n import _
48 from mercurial.i18n import _
48 from mercurial import util, encoding
49 from mercurial import util, encoding
49
50
50 def decode(arg):
51 def decode(arg):
51 if isinstance(arg, str):
52 if isinstance(arg, str):
52 uarg = arg.decode(encoding.encoding)
53 uarg = arg.decode(encoding.encoding)
53 if arg == uarg.encode(encoding.encoding):
54 if arg == uarg.encode(encoding.encoding):
54 return uarg
55 return uarg
55 raise UnicodeError("Not local encoding")
56 raise UnicodeError("Not local encoding")
56 elif isinstance(arg, tuple):
57 elif isinstance(arg, tuple):
57 return tuple(map(decode, arg))
58 return tuple(map(decode, arg))
58 elif isinstance(arg, list):
59 elif isinstance(arg, list):
59 return map(decode, arg)
60 return map(decode, arg)
60 return arg
61 return arg
61
62
62 def encode(arg):
63 def encode(arg):
63 if isinstance(arg, unicode):
64 if isinstance(arg, unicode):
64 return arg.encode(encoding.encoding)
65 return arg.encode(encoding.encoding)
65 elif isinstance(arg, tuple):
66 elif isinstance(arg, tuple):
66 return tuple(map(encode, arg))
67 return tuple(map(encode, arg))
67 elif isinstance(arg, list):
68 elif isinstance(arg, list):
68 return map(encode, arg)
69 return map(encode, arg)
69 return arg
70 return arg
70
71
71 def wrapper(func, args):
72 def wrapper(func, args):
72 # check argument is unicode, then call original
73 # check argument is unicode, then call original
73 for arg in args:
74 for arg in args:
74 if isinstance(arg, unicode):
75 if isinstance(arg, unicode):
75 return func(*args)
76 return func(*args)
76
77
77 try:
78 try:
78 # convert arguments to unicode, call func, then convert back
79 # convert arguments to unicode, call func, then convert back
79 return encode(func(*decode(args)))
80 return encode(func(*decode(args)))
80 except UnicodeError:
81 except UnicodeError:
81 # If not encoded with encoding.encoding, report it then
82 # If not encoded with encoding.encoding, report it then
82 # continue with calling original function.
83 # continue with calling original function.
83 raise util.Abort(_("[win32mbcs] filename conversion fail with"
84 raise util.Abort(_("[win32mbcs] filename conversion fail with"
84 " %s encoding\n") % (encoding.encoding))
85 " %s encoding\n") % (encoding.encoding))
85
86
86 def wrapname(name):
87 def wrapname(name):
87 idx = name.rfind('.')
88 idx = name.rfind('.')
88 module = name[:idx]
89 module = name[:idx]
89 name = name[idx+1:]
90 name = name[idx+1:]
90 module = eval(module)
91 module = eval(module)
91 func = getattr(module, name)
92 func = getattr(module, name)
92 def f(*args):
93 def f(*args):
93 return wrapper(func, args)
94 return wrapper(func, args)
94 try:
95 try:
95 f.__name__ = func.__name__ # fail with python23
96 f.__name__ = func.__name__ # fail with python23
96 except Exception:
97 except Exception:
97 pass
98 pass
98 setattr(module, name, f)
99 setattr(module, name, f)
99
100
100 # List of functions to be wrapped.
101 # List of functions to be wrapped.
101 # NOTE: os.path.dirname() and os.path.basename() are safe because
102 # NOTE: os.path.dirname() and os.path.basename() are safe because
102 # they use result of os.path.split()
103 # they use result of os.path.split()
103 funcs = '''os.path.join os.path.split os.path.splitext
104 funcs = '''os.path.join os.path.split os.path.splitext
104 os.path.splitunc os.path.normpath os.path.normcase os.makedirs
105 os.path.splitunc os.path.normpath os.path.normcase os.makedirs
105 util.endswithsep util.splitpath util.checkcase util.fspath'''
106 util.endswithsep util.splitpath util.checkcase util.fspath'''
106
107
107 # codec and alias names of sjis and big5 to be faked.
108 # codec and alias names of sjis and big5 to be faked.
108 problematic_encodings = '''big5 big5-tw csbig5 big5hkscs big5-hkscs
109 problematic_encodings = '''big5 big5-tw csbig5 big5hkscs big5-hkscs
109 hkscs cp932 932 ms932 mskanji ms-kanji shift_jis csshiftjis shiftjis
110 hkscs cp932 932 ms932 mskanji ms-kanji shift_jis csshiftjis shiftjis
110 sjis s_jis shift_jis_2004 shiftjis2004 sjis_2004 sjis2004
111 sjis s_jis shift_jis_2004 shiftjis2004 sjis_2004 sjis2004
111 shift_jisx0213 shiftjisx0213 sjisx0213 s_jisx0213'''
112 shift_jisx0213 shiftjisx0213 sjisx0213 s_jisx0213'''
112
113
113 def reposetup(ui, repo):
114 def reposetup(ui, repo):
114 # TODO: decide use of config section for this extension
115 # TODO: decide use of config section for this extension
115 if not os.path.supports_unicode_filenames:
116 if not os.path.supports_unicode_filenames:
116 ui.warn(_("[win32mbcs] cannot activate on this platform.\n"))
117 ui.warn(_("[win32mbcs] cannot activate on this platform.\n"))
117 return
118 return
118
119
119 # fake is only for relevant environment.
120 # fake is only for relevant environment.
120 if encoding.encoding.lower() in problematic_encodings.split():
121 if encoding.encoding.lower() in problematic_encodings.split():
121 for f in funcs.split():
122 for f in funcs.split():
122 wrapname(f)
123 wrapname(f)
123 ui.debug(_("[win32mbcs] activated with encoding: %s\n")
124 ui.debug(_("[win32mbcs] activated with encoding: %s\n")
124 % encoding.encoding)
125 % encoding.encoding)
125
126
@@ -1,144 +1,145
1 # changelog bisection for mercurial
1 # changelog bisection for mercurial
2 #
2 #
3 # Copyright 2007 Matt Mackall
3 # Copyright 2007 Matt Mackall
4 # Copyright 2005, 2006 Benoit Boissinot <benoit.boissinot@ens-lyon.org>
4 # Copyright 2005, 2006 Benoit Boissinot <benoit.boissinot@ens-lyon.org>
5 #
5 # Inspired by git bisect, extension skeleton taken from mq.py.
6 # Inspired by git bisect, extension skeleton taken from mq.py.
6 #
7 #
7 # This software may be used and distributed according to the terms of the
8 # This software may be used and distributed according to the terms of the
8 # GNU General Public License version 2, incorporated herein by reference.
9 # GNU General Public License version 2, incorporated herein by reference.
9
10
10 import os
11 import os
11 from i18n import _
12 from i18n import _
12 from node import short, hex
13 from node import short, hex
13 import util
14 import util
14
15
15 def bisect(changelog, state):
16 def bisect(changelog, state):
16 """find the next node (if any) for testing during a bisect search.
17 """find the next node (if any) for testing during a bisect search.
17 returns a (nodes, number, good) tuple.
18 returns a (nodes, number, good) tuple.
18
19
19 'nodes' is the final result of the bisect if 'number' is 0.
20 'nodes' is the final result of the bisect if 'number' is 0.
20 Otherwise 'number' indicates the remaining possible candidates for
21 Otherwise 'number' indicates the remaining possible candidates for
21 the search and 'nodes' contains the next bisect target.
22 the search and 'nodes' contains the next bisect target.
22 'good' is True if bisect is searching for a first good changeset, False
23 'good' is True if bisect is searching for a first good changeset, False
23 if searching for a first bad one.
24 if searching for a first bad one.
24 """
25 """
25
26
26 clparents = changelog.parentrevs
27 clparents = changelog.parentrevs
27 skip = set([changelog.rev(n) for n in state['skip']])
28 skip = set([changelog.rev(n) for n in state['skip']])
28
29
29 def buildancestors(bad, good):
30 def buildancestors(bad, good):
30 # only the earliest bad revision matters
31 # only the earliest bad revision matters
31 badrev = min([changelog.rev(n) for n in bad])
32 badrev = min([changelog.rev(n) for n in bad])
32 goodrevs = [changelog.rev(n) for n in good]
33 goodrevs = [changelog.rev(n) for n in good]
33 # build ancestors array
34 # build ancestors array
34 ancestors = [[]] * (len(changelog) + 1) # an extra for [-1]
35 ancestors = [[]] * (len(changelog) + 1) # an extra for [-1]
35
36
36 # clear good revs from array
37 # clear good revs from array
37 for node in goodrevs:
38 for node in goodrevs:
38 ancestors[node] = None
39 ancestors[node] = None
39 for rev in xrange(len(changelog), -1, -1):
40 for rev in xrange(len(changelog), -1, -1):
40 if ancestors[rev] is None:
41 if ancestors[rev] is None:
41 for prev in clparents(rev):
42 for prev in clparents(rev):
42 ancestors[prev] = None
43 ancestors[prev] = None
43
44
44 if ancestors[badrev] is None:
45 if ancestors[badrev] is None:
45 return badrev, None
46 return badrev, None
46 return badrev, ancestors
47 return badrev, ancestors
47
48
48 good = 0
49 good = 0
49 badrev, ancestors = buildancestors(state['bad'], state['good'])
50 badrev, ancestors = buildancestors(state['bad'], state['good'])
50 if not ancestors: # looking for bad to good transition?
51 if not ancestors: # looking for bad to good transition?
51 good = 1
52 good = 1
52 badrev, ancestors = buildancestors(state['good'], state['bad'])
53 badrev, ancestors = buildancestors(state['good'], state['bad'])
53 bad = changelog.node(badrev)
54 bad = changelog.node(badrev)
54 if not ancestors: # now we're confused
55 if not ancestors: # now we're confused
55 raise util.Abort(_("Inconsistent state, %s:%s is good and bad")
56 raise util.Abort(_("Inconsistent state, %s:%s is good and bad")
56 % (badrev, short(bad)))
57 % (badrev, short(bad)))
57
58
58 # build children dict
59 # build children dict
59 children = {}
60 children = {}
60 visit = [badrev]
61 visit = [badrev]
61 candidates = []
62 candidates = []
62 while visit:
63 while visit:
63 rev = visit.pop(0)
64 rev = visit.pop(0)
64 if ancestors[rev] == []:
65 if ancestors[rev] == []:
65 candidates.append(rev)
66 candidates.append(rev)
66 for prev in clparents(rev):
67 for prev in clparents(rev):
67 if prev != -1:
68 if prev != -1:
68 if prev in children:
69 if prev in children:
69 children[prev].append(rev)
70 children[prev].append(rev)
70 else:
71 else:
71 children[prev] = [rev]
72 children[prev] = [rev]
72 visit.append(prev)
73 visit.append(prev)
73
74
74 candidates.sort()
75 candidates.sort()
75 # have we narrowed it down to one entry?
76 # have we narrowed it down to one entry?
76 # or have all other possible candidates besides 'bad' have been skipped?
77 # or have all other possible candidates besides 'bad' have been skipped?
77 tot = len(candidates)
78 tot = len(candidates)
78 unskipped = [c for c in candidates if (c not in skip) and (c != badrev)]
79 unskipped = [c for c in candidates if (c not in skip) and (c != badrev)]
79 if tot == 1 or not unskipped:
80 if tot == 1 or not unskipped:
80 return ([changelog.node(rev) for rev in candidates], 0, good)
81 return ([changelog.node(rev) for rev in candidates], 0, good)
81 perfect = tot // 2
82 perfect = tot // 2
82
83
83 # find the best node to test
84 # find the best node to test
84 best_rev = None
85 best_rev = None
85 best_len = -1
86 best_len = -1
86 poison = {}
87 poison = {}
87 for rev in candidates:
88 for rev in candidates:
88 if rev in poison:
89 if rev in poison:
89 for c in children.get(rev, []):
90 for c in children.get(rev, []):
90 poison[c] = True # poison children
91 poison[c] = True # poison children
91 continue
92 continue
92
93
93 a = ancestors[rev] or [rev]
94 a = ancestors[rev] or [rev]
94 ancestors[rev] = None
95 ancestors[rev] = None
95
96
96 x = len(a) # number of ancestors
97 x = len(a) # number of ancestors
97 y = tot - x # number of non-ancestors
98 y = tot - x # number of non-ancestors
98 value = min(x, y) # how good is this test?
99 value = min(x, y) # how good is this test?
99 if value > best_len and rev not in skip:
100 if value > best_len and rev not in skip:
100 best_len = value
101 best_len = value
101 best_rev = rev
102 best_rev = rev
102 if value == perfect: # found a perfect candidate? quit early
103 if value == perfect: # found a perfect candidate? quit early
103 break
104 break
104
105
105 if y < perfect and rev not in skip: # all downhill from here?
106 if y < perfect and rev not in skip: # all downhill from here?
106 for c in children.get(rev, []):
107 for c in children.get(rev, []):
107 poison[c] = True # poison children
108 poison[c] = True # poison children
108 continue
109 continue
109
110
110 for c in children.get(rev, []):
111 for c in children.get(rev, []):
111 if ancestors[c]:
112 if ancestors[c]:
112 ancestors[c] = list(set(ancestors[c] + a))
113 ancestors[c] = list(set(ancestors[c] + a))
113 else:
114 else:
114 ancestors[c] = a + [c]
115 ancestors[c] = a + [c]
115
116
116 assert best_rev is not None
117 assert best_rev is not None
117 best_node = changelog.node(best_rev)
118 best_node = changelog.node(best_rev)
118
119
119 return ([best_node], tot, good)
120 return ([best_node], tot, good)
120
121
121
122
122 def load_state(repo):
123 def load_state(repo):
123 state = {'good': [], 'bad': [], 'skip': []}
124 state = {'good': [], 'bad': [], 'skip': []}
124 if os.path.exists(repo.join("bisect.state")):
125 if os.path.exists(repo.join("bisect.state")):
125 for l in repo.opener("bisect.state"):
126 for l in repo.opener("bisect.state"):
126 kind, node = l[:-1].split()
127 kind, node = l[:-1].split()
127 node = repo.lookup(node)
128 node = repo.lookup(node)
128 if kind not in state:
129 if kind not in state:
129 raise util.Abort(_("unknown bisect kind %s") % kind)
130 raise util.Abort(_("unknown bisect kind %s") % kind)
130 state[kind].append(node)
131 state[kind].append(node)
131 return state
132 return state
132
133
133
134
134 def save_state(repo, state):
135 def save_state(repo, state):
135 f = repo.opener("bisect.state", "w", atomictemp=True)
136 f = repo.opener("bisect.state", "w", atomictemp=True)
136 wlock = repo.wlock()
137 wlock = repo.wlock()
137 try:
138 try:
138 for kind in state:
139 for kind in state:
139 for node in state[kind]:
140 for node in state[kind]:
140 f.write("%s %s\n" % (kind, hex(node)))
141 f.write("%s %s\n" % (kind, hex(node)))
141 f.rename()
142 f.rename()
142 finally:
143 finally:
143 wlock.release()
144 wlock.release()
144
145
General Comments 0
You need to be logged in to leave comments. Login now