##// END OF EJS Templates
debugmergestate: print out record type for files...
Siddharth Agarwal -
r26976:c48fee95 default
parent child Browse files
Show More
@@ -1,6772 +1,6772
1 # commands.py - command processing for mercurial
1 # commands.py - command processing for mercurial
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 or any later version.
6 # GNU General Public License version 2 or any later version.
7
7
8 from node import hex, bin, nullid, nullrev, short
8 from node import hex, bin, nullid, nullrev, short
9 from lock import release
9 from lock import release
10 from i18n import _
10 from i18n import _
11 import os, re, difflib, time, tempfile, errno, shlex
11 import os, re, difflib, time, tempfile, errno, shlex
12 import sys, socket
12 import sys, socket
13 import hg, scmutil, util, revlog, copies, error, bookmarks
13 import hg, scmutil, util, revlog, copies, error, bookmarks
14 import patch, help, encoding, templatekw, discovery
14 import patch, help, encoding, templatekw, discovery
15 import archival, changegroup, cmdutil, hbisect
15 import archival, changegroup, cmdutil, hbisect
16 import sshserver, hgweb
16 import sshserver, hgweb
17 import extensions
17 import extensions
18 from hgweb import server as hgweb_server
18 from hgweb import server as hgweb_server
19 import merge as mergemod
19 import merge as mergemod
20 import minirst, revset, fileset
20 import minirst, revset, fileset
21 import dagparser, context, simplemerge, graphmod, copies
21 import dagparser, context, simplemerge, graphmod, copies
22 import random, operator
22 import random, operator
23 import setdiscovery, treediscovery, dagutil, pvec, localrepo, destutil
23 import setdiscovery, treediscovery, dagutil, pvec, localrepo, destutil
24 import phases, obsolete, exchange, bundle2, repair, lock as lockmod
24 import phases, obsolete, exchange, bundle2, repair, lock as lockmod
25 import ui as uimod
25 import ui as uimod
26 import streamclone
26 import streamclone
27
27
28 table = {}
28 table = {}
29
29
30 command = cmdutil.command(table)
30 command = cmdutil.command(table)
31
31
32 # Space delimited list of commands that don't require local repositories.
32 # Space delimited list of commands that don't require local repositories.
33 # This should be populated by passing norepo=True into the @command decorator.
33 # This should be populated by passing norepo=True into the @command decorator.
34 norepo = ''
34 norepo = ''
35 # Space delimited list of commands that optionally require local repositories.
35 # Space delimited list of commands that optionally require local repositories.
36 # This should be populated by passing optionalrepo=True into the @command
36 # This should be populated by passing optionalrepo=True into the @command
37 # decorator.
37 # decorator.
38 optionalrepo = ''
38 optionalrepo = ''
39 # Space delimited list of commands that will examine arguments looking for
39 # Space delimited list of commands that will examine arguments looking for
40 # a repository. This should be populated by passing inferrepo=True into the
40 # a repository. This should be populated by passing inferrepo=True into the
41 # @command decorator.
41 # @command decorator.
42 inferrepo = ''
42 inferrepo = ''
43
43
44 # label constants
44 # label constants
45 # until 3.5, bookmarks.current was the advertised name, not
45 # until 3.5, bookmarks.current was the advertised name, not
46 # bookmarks.active, so we must use both to avoid breaking old
46 # bookmarks.active, so we must use both to avoid breaking old
47 # custom styles
47 # custom styles
48 activebookmarklabel = 'bookmarks.active bookmarks.current'
48 activebookmarklabel = 'bookmarks.active bookmarks.current'
49
49
50 # common command options
50 # common command options
51
51
52 globalopts = [
52 globalopts = [
53 ('R', 'repository', '',
53 ('R', 'repository', '',
54 _('repository root directory or name of overlay bundle file'),
54 _('repository root directory or name of overlay bundle file'),
55 _('REPO')),
55 _('REPO')),
56 ('', 'cwd', '',
56 ('', 'cwd', '',
57 _('change working directory'), _('DIR')),
57 _('change working directory'), _('DIR')),
58 ('y', 'noninteractive', None,
58 ('y', 'noninteractive', None,
59 _('do not prompt, automatically pick the first choice for all prompts')),
59 _('do not prompt, automatically pick the first choice for all prompts')),
60 ('q', 'quiet', None, _('suppress output')),
60 ('q', 'quiet', None, _('suppress output')),
61 ('v', 'verbose', None, _('enable additional output')),
61 ('v', 'verbose', None, _('enable additional output')),
62 ('', 'config', [],
62 ('', 'config', [],
63 _('set/override config option (use \'section.name=value\')'),
63 _('set/override config option (use \'section.name=value\')'),
64 _('CONFIG')),
64 _('CONFIG')),
65 ('', 'debug', None, _('enable debugging output')),
65 ('', 'debug', None, _('enable debugging output')),
66 ('', 'debugger', None, _('start debugger')),
66 ('', 'debugger', None, _('start debugger')),
67 ('', 'encoding', encoding.encoding, _('set the charset encoding'),
67 ('', 'encoding', encoding.encoding, _('set the charset encoding'),
68 _('ENCODE')),
68 _('ENCODE')),
69 ('', 'encodingmode', encoding.encodingmode,
69 ('', 'encodingmode', encoding.encodingmode,
70 _('set the charset encoding mode'), _('MODE')),
70 _('set the charset encoding mode'), _('MODE')),
71 ('', 'traceback', None, _('always print a traceback on exception')),
71 ('', 'traceback', None, _('always print a traceback on exception')),
72 ('', 'time', None, _('time how long the command takes')),
72 ('', 'time', None, _('time how long the command takes')),
73 ('', 'profile', None, _('print command execution profile')),
73 ('', 'profile', None, _('print command execution profile')),
74 ('', 'version', None, _('output version information and exit')),
74 ('', 'version', None, _('output version information and exit')),
75 ('h', 'help', None, _('display help and exit')),
75 ('h', 'help', None, _('display help and exit')),
76 ('', 'hidden', False, _('consider hidden changesets')),
76 ('', 'hidden', False, _('consider hidden changesets')),
77 ]
77 ]
78
78
79 dryrunopts = [('n', 'dry-run', None,
79 dryrunopts = [('n', 'dry-run', None,
80 _('do not perform actions, just print output'))]
80 _('do not perform actions, just print output'))]
81
81
82 remoteopts = [
82 remoteopts = [
83 ('e', 'ssh', '',
83 ('e', 'ssh', '',
84 _('specify ssh command to use'), _('CMD')),
84 _('specify ssh command to use'), _('CMD')),
85 ('', 'remotecmd', '',
85 ('', 'remotecmd', '',
86 _('specify hg command to run on the remote side'), _('CMD')),
86 _('specify hg command to run on the remote side'), _('CMD')),
87 ('', 'insecure', None,
87 ('', 'insecure', None,
88 _('do not verify server certificate (ignoring web.cacerts config)')),
88 _('do not verify server certificate (ignoring web.cacerts config)')),
89 ]
89 ]
90
90
91 walkopts = [
91 walkopts = [
92 ('I', 'include', [],
92 ('I', 'include', [],
93 _('include names matching the given patterns'), _('PATTERN')),
93 _('include names matching the given patterns'), _('PATTERN')),
94 ('X', 'exclude', [],
94 ('X', 'exclude', [],
95 _('exclude names matching the given patterns'), _('PATTERN')),
95 _('exclude names matching the given patterns'), _('PATTERN')),
96 ]
96 ]
97
97
98 commitopts = [
98 commitopts = [
99 ('m', 'message', '',
99 ('m', 'message', '',
100 _('use text as commit message'), _('TEXT')),
100 _('use text as commit message'), _('TEXT')),
101 ('l', 'logfile', '',
101 ('l', 'logfile', '',
102 _('read commit message from file'), _('FILE')),
102 _('read commit message from file'), _('FILE')),
103 ]
103 ]
104
104
105 commitopts2 = [
105 commitopts2 = [
106 ('d', 'date', '',
106 ('d', 'date', '',
107 _('record the specified date as commit date'), _('DATE')),
107 _('record the specified date as commit date'), _('DATE')),
108 ('u', 'user', '',
108 ('u', 'user', '',
109 _('record the specified user as committer'), _('USER')),
109 _('record the specified user as committer'), _('USER')),
110 ]
110 ]
111
111
112 # hidden for now
112 # hidden for now
113 formatteropts = [
113 formatteropts = [
114 ('T', 'template', '',
114 ('T', 'template', '',
115 _('display with template (EXPERIMENTAL)'), _('TEMPLATE')),
115 _('display with template (EXPERIMENTAL)'), _('TEMPLATE')),
116 ]
116 ]
117
117
118 templateopts = [
118 templateopts = [
119 ('', 'style', '',
119 ('', 'style', '',
120 _('display using template map file (DEPRECATED)'), _('STYLE')),
120 _('display using template map file (DEPRECATED)'), _('STYLE')),
121 ('T', 'template', '',
121 ('T', 'template', '',
122 _('display with template'), _('TEMPLATE')),
122 _('display with template'), _('TEMPLATE')),
123 ]
123 ]
124
124
125 logopts = [
125 logopts = [
126 ('p', 'patch', None, _('show patch')),
126 ('p', 'patch', None, _('show patch')),
127 ('g', 'git', None, _('use git extended diff format')),
127 ('g', 'git', None, _('use git extended diff format')),
128 ('l', 'limit', '',
128 ('l', 'limit', '',
129 _('limit number of changes displayed'), _('NUM')),
129 _('limit number of changes displayed'), _('NUM')),
130 ('M', 'no-merges', None, _('do not show merges')),
130 ('M', 'no-merges', None, _('do not show merges')),
131 ('', 'stat', None, _('output diffstat-style summary of changes')),
131 ('', 'stat', None, _('output diffstat-style summary of changes')),
132 ('G', 'graph', None, _("show the revision DAG")),
132 ('G', 'graph', None, _("show the revision DAG")),
133 ] + templateopts
133 ] + templateopts
134
134
135 diffopts = [
135 diffopts = [
136 ('a', 'text', None, _('treat all files as text')),
136 ('a', 'text', None, _('treat all files as text')),
137 ('g', 'git', None, _('use git extended diff format')),
137 ('g', 'git', None, _('use git extended diff format')),
138 ('', 'nodates', None, _('omit dates from diff headers'))
138 ('', 'nodates', None, _('omit dates from diff headers'))
139 ]
139 ]
140
140
141 diffwsopts = [
141 diffwsopts = [
142 ('w', 'ignore-all-space', None,
142 ('w', 'ignore-all-space', None,
143 _('ignore white space when comparing lines')),
143 _('ignore white space when comparing lines')),
144 ('b', 'ignore-space-change', None,
144 ('b', 'ignore-space-change', None,
145 _('ignore changes in the amount of white space')),
145 _('ignore changes in the amount of white space')),
146 ('B', 'ignore-blank-lines', None,
146 ('B', 'ignore-blank-lines', None,
147 _('ignore changes whose lines are all blank')),
147 _('ignore changes whose lines are all blank')),
148 ]
148 ]
149
149
150 diffopts2 = [
150 diffopts2 = [
151 ('', 'noprefix', None, _('omit a/ and b/ prefixes from filenames')),
151 ('', 'noprefix', None, _('omit a/ and b/ prefixes from filenames')),
152 ('p', 'show-function', None, _('show which function each change is in')),
152 ('p', 'show-function', None, _('show which function each change is in')),
153 ('', 'reverse', None, _('produce a diff that undoes the changes')),
153 ('', 'reverse', None, _('produce a diff that undoes the changes')),
154 ] + diffwsopts + [
154 ] + diffwsopts + [
155 ('U', 'unified', '',
155 ('U', 'unified', '',
156 _('number of lines of context to show'), _('NUM')),
156 _('number of lines of context to show'), _('NUM')),
157 ('', 'stat', None, _('output diffstat-style summary of changes')),
157 ('', 'stat', None, _('output diffstat-style summary of changes')),
158 ('', 'root', '', _('produce diffs relative to subdirectory'), _('DIR')),
158 ('', 'root', '', _('produce diffs relative to subdirectory'), _('DIR')),
159 ]
159 ]
160
160
161 mergetoolopts = [
161 mergetoolopts = [
162 ('t', 'tool', '', _('specify merge tool')),
162 ('t', 'tool', '', _('specify merge tool')),
163 ]
163 ]
164
164
165 similarityopts = [
165 similarityopts = [
166 ('s', 'similarity', '',
166 ('s', 'similarity', '',
167 _('guess renamed files by similarity (0<=s<=100)'), _('SIMILARITY'))
167 _('guess renamed files by similarity (0<=s<=100)'), _('SIMILARITY'))
168 ]
168 ]
169
169
170 subrepoopts = [
170 subrepoopts = [
171 ('S', 'subrepos', None,
171 ('S', 'subrepos', None,
172 _('recurse into subrepositories'))
172 _('recurse into subrepositories'))
173 ]
173 ]
174
174
175 # Commands start here, listed alphabetically
175 # Commands start here, listed alphabetically
176
176
177 @command('^add',
177 @command('^add',
178 walkopts + subrepoopts + dryrunopts,
178 walkopts + subrepoopts + dryrunopts,
179 _('[OPTION]... [FILE]...'),
179 _('[OPTION]... [FILE]...'),
180 inferrepo=True)
180 inferrepo=True)
181 def add(ui, repo, *pats, **opts):
181 def add(ui, repo, *pats, **opts):
182 """add the specified files on the next commit
182 """add the specified files on the next commit
183
183
184 Schedule files to be version controlled and added to the
184 Schedule files to be version controlled and added to the
185 repository.
185 repository.
186
186
187 The files will be added to the repository at the next commit. To
187 The files will be added to the repository at the next commit. To
188 undo an add before that, see :hg:`forget`.
188 undo an add before that, see :hg:`forget`.
189
189
190 If no names are given, add all files to the repository.
190 If no names are given, add all files to the repository.
191
191
192 .. container:: verbose
192 .. container:: verbose
193
193
194 An example showing how new (unknown) files are added
194 An example showing how new (unknown) files are added
195 automatically by :hg:`add`::
195 automatically by :hg:`add`::
196
196
197 $ ls
197 $ ls
198 foo.c
198 foo.c
199 $ hg status
199 $ hg status
200 ? foo.c
200 ? foo.c
201 $ hg add
201 $ hg add
202 adding foo.c
202 adding foo.c
203 $ hg status
203 $ hg status
204 A foo.c
204 A foo.c
205
205
206 Returns 0 if all files are successfully added.
206 Returns 0 if all files are successfully added.
207 """
207 """
208
208
209 m = scmutil.match(repo[None], pats, opts)
209 m = scmutil.match(repo[None], pats, opts)
210 rejected = cmdutil.add(ui, repo, m, "", False, **opts)
210 rejected = cmdutil.add(ui, repo, m, "", False, **opts)
211 return rejected and 1 or 0
211 return rejected and 1 or 0
212
212
213 @command('addremove',
213 @command('addremove',
214 similarityopts + subrepoopts + walkopts + dryrunopts,
214 similarityopts + subrepoopts + walkopts + dryrunopts,
215 _('[OPTION]... [FILE]...'),
215 _('[OPTION]... [FILE]...'),
216 inferrepo=True)
216 inferrepo=True)
217 def addremove(ui, repo, *pats, **opts):
217 def addremove(ui, repo, *pats, **opts):
218 """add all new files, delete all missing files
218 """add all new files, delete all missing files
219
219
220 Add all new files and remove all missing files from the
220 Add all new files and remove all missing files from the
221 repository.
221 repository.
222
222
223 New files are ignored if they match any of the patterns in
223 New files are ignored if they match any of the patterns in
224 ``.hgignore``. As with add, these changes take effect at the next
224 ``.hgignore``. As with add, these changes take effect at the next
225 commit.
225 commit.
226
226
227 Use the -s/--similarity option to detect renamed files. This
227 Use the -s/--similarity option to detect renamed files. This
228 option takes a percentage between 0 (disabled) and 100 (files must
228 option takes a percentage between 0 (disabled) and 100 (files must
229 be identical) as its parameter. With a parameter greater than 0,
229 be identical) as its parameter. With a parameter greater than 0,
230 this compares every removed file with every added file and records
230 this compares every removed file with every added file and records
231 those similar enough as renames. Detecting renamed files this way
231 those similar enough as renames. Detecting renamed files this way
232 can be expensive. After using this option, :hg:`status -C` can be
232 can be expensive. After using this option, :hg:`status -C` can be
233 used to check which files were identified as moved or renamed. If
233 used to check which files were identified as moved or renamed. If
234 not specified, -s/--similarity defaults to 100 and only renames of
234 not specified, -s/--similarity defaults to 100 and only renames of
235 identical files are detected.
235 identical files are detected.
236
236
237 Returns 0 if all files are successfully added.
237 Returns 0 if all files are successfully added.
238 """
238 """
239 try:
239 try:
240 sim = float(opts.get('similarity') or 100)
240 sim = float(opts.get('similarity') or 100)
241 except ValueError:
241 except ValueError:
242 raise error.Abort(_('similarity must be a number'))
242 raise error.Abort(_('similarity must be a number'))
243 if sim < 0 or sim > 100:
243 if sim < 0 or sim > 100:
244 raise error.Abort(_('similarity must be between 0 and 100'))
244 raise error.Abort(_('similarity must be between 0 and 100'))
245 matcher = scmutil.match(repo[None], pats, opts)
245 matcher = scmutil.match(repo[None], pats, opts)
246 return scmutil.addremove(repo, matcher, "", opts, similarity=sim / 100.0)
246 return scmutil.addremove(repo, matcher, "", opts, similarity=sim / 100.0)
247
247
248 @command('^annotate|blame',
248 @command('^annotate|blame',
249 [('r', 'rev', '', _('annotate the specified revision'), _('REV')),
249 [('r', 'rev', '', _('annotate the specified revision'), _('REV')),
250 ('', 'follow', None,
250 ('', 'follow', None,
251 _('follow copies/renames and list the filename (DEPRECATED)')),
251 _('follow copies/renames and list the filename (DEPRECATED)')),
252 ('', 'no-follow', None, _("don't follow copies and renames")),
252 ('', 'no-follow', None, _("don't follow copies and renames")),
253 ('a', 'text', None, _('treat all files as text')),
253 ('a', 'text', None, _('treat all files as text')),
254 ('u', 'user', None, _('list the author (long with -v)')),
254 ('u', 'user', None, _('list the author (long with -v)')),
255 ('f', 'file', None, _('list the filename')),
255 ('f', 'file', None, _('list the filename')),
256 ('d', 'date', None, _('list the date (short with -q)')),
256 ('d', 'date', None, _('list the date (short with -q)')),
257 ('n', 'number', None, _('list the revision number (default)')),
257 ('n', 'number', None, _('list the revision number (default)')),
258 ('c', 'changeset', None, _('list the changeset')),
258 ('c', 'changeset', None, _('list the changeset')),
259 ('l', 'line-number', None, _('show line number at the first appearance'))
259 ('l', 'line-number', None, _('show line number at the first appearance'))
260 ] + diffwsopts + walkopts + formatteropts,
260 ] + diffwsopts + walkopts + formatteropts,
261 _('[-r REV] [-f] [-a] [-u] [-d] [-n] [-c] [-l] FILE...'),
261 _('[-r REV] [-f] [-a] [-u] [-d] [-n] [-c] [-l] FILE...'),
262 inferrepo=True)
262 inferrepo=True)
263 def annotate(ui, repo, *pats, **opts):
263 def annotate(ui, repo, *pats, **opts):
264 """show changeset information by line for each file
264 """show changeset information by line for each file
265
265
266 List changes in files, showing the revision id responsible for
266 List changes in files, showing the revision id responsible for
267 each line
267 each line
268
268
269 This command is useful for discovering when a change was made and
269 This command is useful for discovering when a change was made and
270 by whom.
270 by whom.
271
271
272 Without the -a/--text option, annotate will avoid processing files
272 Without the -a/--text option, annotate will avoid processing files
273 it detects as binary. With -a, annotate will annotate the file
273 it detects as binary. With -a, annotate will annotate the file
274 anyway, although the results will probably be neither useful
274 anyway, although the results will probably be neither useful
275 nor desirable.
275 nor desirable.
276
276
277 Returns 0 on success.
277 Returns 0 on success.
278 """
278 """
279 if not pats:
279 if not pats:
280 raise error.Abort(_('at least one filename or pattern is required'))
280 raise error.Abort(_('at least one filename or pattern is required'))
281
281
282 if opts.get('follow'):
282 if opts.get('follow'):
283 # --follow is deprecated and now just an alias for -f/--file
283 # --follow is deprecated and now just an alias for -f/--file
284 # to mimic the behavior of Mercurial before version 1.5
284 # to mimic the behavior of Mercurial before version 1.5
285 opts['file'] = True
285 opts['file'] = True
286
286
287 ctx = scmutil.revsingle(repo, opts.get('rev'))
287 ctx = scmutil.revsingle(repo, opts.get('rev'))
288
288
289 fm = ui.formatter('annotate', opts)
289 fm = ui.formatter('annotate', opts)
290 if ui.quiet:
290 if ui.quiet:
291 datefunc = util.shortdate
291 datefunc = util.shortdate
292 else:
292 else:
293 datefunc = util.datestr
293 datefunc = util.datestr
294 if ctx.rev() is None:
294 if ctx.rev() is None:
295 def hexfn(node):
295 def hexfn(node):
296 if node is None:
296 if node is None:
297 return None
297 return None
298 else:
298 else:
299 return fm.hexfunc(node)
299 return fm.hexfunc(node)
300 if opts.get('changeset'):
300 if opts.get('changeset'):
301 # omit "+" suffix which is appended to node hex
301 # omit "+" suffix which is appended to node hex
302 def formatrev(rev):
302 def formatrev(rev):
303 if rev is None:
303 if rev is None:
304 return '%d' % ctx.p1().rev()
304 return '%d' % ctx.p1().rev()
305 else:
305 else:
306 return '%d' % rev
306 return '%d' % rev
307 else:
307 else:
308 def formatrev(rev):
308 def formatrev(rev):
309 if rev is None:
309 if rev is None:
310 return '%d+' % ctx.p1().rev()
310 return '%d+' % ctx.p1().rev()
311 else:
311 else:
312 return '%d ' % rev
312 return '%d ' % rev
313 def formathex(hex):
313 def formathex(hex):
314 if hex is None:
314 if hex is None:
315 return '%s+' % fm.hexfunc(ctx.p1().node())
315 return '%s+' % fm.hexfunc(ctx.p1().node())
316 else:
316 else:
317 return '%s ' % hex
317 return '%s ' % hex
318 else:
318 else:
319 hexfn = fm.hexfunc
319 hexfn = fm.hexfunc
320 formatrev = formathex = str
320 formatrev = formathex = str
321
321
322 opmap = [('user', ' ', lambda x: x[0].user(), ui.shortuser),
322 opmap = [('user', ' ', lambda x: x[0].user(), ui.shortuser),
323 ('number', ' ', lambda x: x[0].rev(), formatrev),
323 ('number', ' ', lambda x: x[0].rev(), formatrev),
324 ('changeset', ' ', lambda x: hexfn(x[0].node()), formathex),
324 ('changeset', ' ', lambda x: hexfn(x[0].node()), formathex),
325 ('date', ' ', lambda x: x[0].date(), util.cachefunc(datefunc)),
325 ('date', ' ', lambda x: x[0].date(), util.cachefunc(datefunc)),
326 ('file', ' ', lambda x: x[0].path(), str),
326 ('file', ' ', lambda x: x[0].path(), str),
327 ('line_number', ':', lambda x: x[1], str),
327 ('line_number', ':', lambda x: x[1], str),
328 ]
328 ]
329 fieldnamemap = {'number': 'rev', 'changeset': 'node'}
329 fieldnamemap = {'number': 'rev', 'changeset': 'node'}
330
330
331 if (not opts.get('user') and not opts.get('changeset')
331 if (not opts.get('user') and not opts.get('changeset')
332 and not opts.get('date') and not opts.get('file')):
332 and not opts.get('date') and not opts.get('file')):
333 opts['number'] = True
333 opts['number'] = True
334
334
335 linenumber = opts.get('line_number') is not None
335 linenumber = opts.get('line_number') is not None
336 if linenumber and (not opts.get('changeset')) and (not opts.get('number')):
336 if linenumber and (not opts.get('changeset')) and (not opts.get('number')):
337 raise error.Abort(_('at least one of -n/-c is required for -l'))
337 raise error.Abort(_('at least one of -n/-c is required for -l'))
338
338
339 if fm:
339 if fm:
340 def makefunc(get, fmt):
340 def makefunc(get, fmt):
341 return get
341 return get
342 else:
342 else:
343 def makefunc(get, fmt):
343 def makefunc(get, fmt):
344 return lambda x: fmt(get(x))
344 return lambda x: fmt(get(x))
345 funcmap = [(makefunc(get, fmt), sep) for op, sep, get, fmt in opmap
345 funcmap = [(makefunc(get, fmt), sep) for op, sep, get, fmt in opmap
346 if opts.get(op)]
346 if opts.get(op)]
347 funcmap[0] = (funcmap[0][0], '') # no separator in front of first column
347 funcmap[0] = (funcmap[0][0], '') # no separator in front of first column
348 fields = ' '.join(fieldnamemap.get(op, op) for op, sep, get, fmt in opmap
348 fields = ' '.join(fieldnamemap.get(op, op) for op, sep, get, fmt in opmap
349 if opts.get(op))
349 if opts.get(op))
350
350
351 def bad(x, y):
351 def bad(x, y):
352 raise error.Abort("%s: %s" % (x, y))
352 raise error.Abort("%s: %s" % (x, y))
353
353
354 m = scmutil.match(ctx, pats, opts, badfn=bad)
354 m = scmutil.match(ctx, pats, opts, badfn=bad)
355
355
356 follow = not opts.get('no_follow')
356 follow = not opts.get('no_follow')
357 diffopts = patch.difffeatureopts(ui, opts, section='annotate',
357 diffopts = patch.difffeatureopts(ui, opts, section='annotate',
358 whitespace=True)
358 whitespace=True)
359 for abs in ctx.walk(m):
359 for abs in ctx.walk(m):
360 fctx = ctx[abs]
360 fctx = ctx[abs]
361 if not opts.get('text') and util.binary(fctx.data()):
361 if not opts.get('text') and util.binary(fctx.data()):
362 fm.plain(_("%s: binary file\n") % ((pats and m.rel(abs)) or abs))
362 fm.plain(_("%s: binary file\n") % ((pats and m.rel(abs)) or abs))
363 continue
363 continue
364
364
365 lines = fctx.annotate(follow=follow, linenumber=linenumber,
365 lines = fctx.annotate(follow=follow, linenumber=linenumber,
366 diffopts=diffopts)
366 diffopts=diffopts)
367 formats = []
367 formats = []
368 pieces = []
368 pieces = []
369
369
370 for f, sep in funcmap:
370 for f, sep in funcmap:
371 l = [f(n) for n, dummy in lines]
371 l = [f(n) for n, dummy in lines]
372 if l:
372 if l:
373 if fm:
373 if fm:
374 formats.append(['%s' for x in l])
374 formats.append(['%s' for x in l])
375 else:
375 else:
376 sizes = [encoding.colwidth(x) for x in l]
376 sizes = [encoding.colwidth(x) for x in l]
377 ml = max(sizes)
377 ml = max(sizes)
378 formats.append([sep + ' ' * (ml - w) + '%s' for w in sizes])
378 formats.append([sep + ' ' * (ml - w) + '%s' for w in sizes])
379 pieces.append(l)
379 pieces.append(l)
380
380
381 for f, p, l in zip(zip(*formats), zip(*pieces), lines):
381 for f, p, l in zip(zip(*formats), zip(*pieces), lines):
382 fm.startitem()
382 fm.startitem()
383 fm.write(fields, "".join(f), *p)
383 fm.write(fields, "".join(f), *p)
384 fm.write('line', ": %s", l[1])
384 fm.write('line', ": %s", l[1])
385
385
386 if lines and not lines[-1][1].endswith('\n'):
386 if lines and not lines[-1][1].endswith('\n'):
387 fm.plain('\n')
387 fm.plain('\n')
388
388
389 fm.end()
389 fm.end()
390
390
391 @command('archive',
391 @command('archive',
392 [('', 'no-decode', None, _('do not pass files through decoders')),
392 [('', 'no-decode', None, _('do not pass files through decoders')),
393 ('p', 'prefix', '', _('directory prefix for files in archive'),
393 ('p', 'prefix', '', _('directory prefix for files in archive'),
394 _('PREFIX')),
394 _('PREFIX')),
395 ('r', 'rev', '', _('revision to distribute'), _('REV')),
395 ('r', 'rev', '', _('revision to distribute'), _('REV')),
396 ('t', 'type', '', _('type of distribution to create'), _('TYPE')),
396 ('t', 'type', '', _('type of distribution to create'), _('TYPE')),
397 ] + subrepoopts + walkopts,
397 ] + subrepoopts + walkopts,
398 _('[OPTION]... DEST'))
398 _('[OPTION]... DEST'))
399 def archive(ui, repo, dest, **opts):
399 def archive(ui, repo, dest, **opts):
400 '''create an unversioned archive of a repository revision
400 '''create an unversioned archive of a repository revision
401
401
402 By default, the revision used is the parent of the working
402 By default, the revision used is the parent of the working
403 directory; use -r/--rev to specify a different revision.
403 directory; use -r/--rev to specify a different revision.
404
404
405 The archive type is automatically detected based on file
405 The archive type is automatically detected based on file
406 extension (or override using -t/--type).
406 extension (or override using -t/--type).
407
407
408 .. container:: verbose
408 .. container:: verbose
409
409
410 Examples:
410 Examples:
411
411
412 - create a zip file containing the 1.0 release::
412 - create a zip file containing the 1.0 release::
413
413
414 hg archive -r 1.0 project-1.0.zip
414 hg archive -r 1.0 project-1.0.zip
415
415
416 - create a tarball excluding .hg files::
416 - create a tarball excluding .hg files::
417
417
418 hg archive project.tar.gz -X ".hg*"
418 hg archive project.tar.gz -X ".hg*"
419
419
420 Valid types are:
420 Valid types are:
421
421
422 :``files``: a directory full of files (default)
422 :``files``: a directory full of files (default)
423 :``tar``: tar archive, uncompressed
423 :``tar``: tar archive, uncompressed
424 :``tbz2``: tar archive, compressed using bzip2
424 :``tbz2``: tar archive, compressed using bzip2
425 :``tgz``: tar archive, compressed using gzip
425 :``tgz``: tar archive, compressed using gzip
426 :``uzip``: zip archive, uncompressed
426 :``uzip``: zip archive, uncompressed
427 :``zip``: zip archive, compressed using deflate
427 :``zip``: zip archive, compressed using deflate
428
428
429 The exact name of the destination archive or directory is given
429 The exact name of the destination archive or directory is given
430 using a format string; see :hg:`help export` for details.
430 using a format string; see :hg:`help export` for details.
431
431
432 Each member added to an archive file has a directory prefix
432 Each member added to an archive file has a directory prefix
433 prepended. Use -p/--prefix to specify a format string for the
433 prepended. Use -p/--prefix to specify a format string for the
434 prefix. The default is the basename of the archive, with suffixes
434 prefix. The default is the basename of the archive, with suffixes
435 removed.
435 removed.
436
436
437 Returns 0 on success.
437 Returns 0 on success.
438 '''
438 '''
439
439
440 ctx = scmutil.revsingle(repo, opts.get('rev'))
440 ctx = scmutil.revsingle(repo, opts.get('rev'))
441 if not ctx:
441 if not ctx:
442 raise error.Abort(_('no working directory: please specify a revision'))
442 raise error.Abort(_('no working directory: please specify a revision'))
443 node = ctx.node()
443 node = ctx.node()
444 dest = cmdutil.makefilename(repo, dest, node)
444 dest = cmdutil.makefilename(repo, dest, node)
445 if os.path.realpath(dest) == repo.root:
445 if os.path.realpath(dest) == repo.root:
446 raise error.Abort(_('repository root cannot be destination'))
446 raise error.Abort(_('repository root cannot be destination'))
447
447
448 kind = opts.get('type') or archival.guesskind(dest) or 'files'
448 kind = opts.get('type') or archival.guesskind(dest) or 'files'
449 prefix = opts.get('prefix')
449 prefix = opts.get('prefix')
450
450
451 if dest == '-':
451 if dest == '-':
452 if kind == 'files':
452 if kind == 'files':
453 raise error.Abort(_('cannot archive plain files to stdout'))
453 raise error.Abort(_('cannot archive plain files to stdout'))
454 dest = cmdutil.makefileobj(repo, dest)
454 dest = cmdutil.makefileobj(repo, dest)
455 if not prefix:
455 if not prefix:
456 prefix = os.path.basename(repo.root) + '-%h'
456 prefix = os.path.basename(repo.root) + '-%h'
457
457
458 prefix = cmdutil.makefilename(repo, prefix, node)
458 prefix = cmdutil.makefilename(repo, prefix, node)
459 matchfn = scmutil.match(ctx, [], opts)
459 matchfn = scmutil.match(ctx, [], opts)
460 archival.archive(repo, dest, node, kind, not opts.get('no_decode'),
460 archival.archive(repo, dest, node, kind, not opts.get('no_decode'),
461 matchfn, prefix, subrepos=opts.get('subrepos'))
461 matchfn, prefix, subrepos=opts.get('subrepos'))
462
462
463 @command('backout',
463 @command('backout',
464 [('', 'merge', None, _('merge with old dirstate parent after backout')),
464 [('', 'merge', None, _('merge with old dirstate parent after backout')),
465 ('', 'commit', None, _('commit if no conflicts were encountered')),
465 ('', 'commit', None, _('commit if no conflicts were encountered')),
466 ('', 'parent', '',
466 ('', 'parent', '',
467 _('parent to choose when backing out merge (DEPRECATED)'), _('REV')),
467 _('parent to choose when backing out merge (DEPRECATED)'), _('REV')),
468 ('r', 'rev', '', _('revision to backout'), _('REV')),
468 ('r', 'rev', '', _('revision to backout'), _('REV')),
469 ('e', 'edit', False, _('invoke editor on commit messages')),
469 ('e', 'edit', False, _('invoke editor on commit messages')),
470 ] + mergetoolopts + walkopts + commitopts + commitopts2,
470 ] + mergetoolopts + walkopts + commitopts + commitopts2,
471 _('[OPTION]... [-r] REV'))
471 _('[OPTION]... [-r] REV'))
472 def backout(ui, repo, node=None, rev=None, commit=False, **opts):
472 def backout(ui, repo, node=None, rev=None, commit=False, **opts):
473 '''reverse effect of earlier changeset
473 '''reverse effect of earlier changeset
474
474
475 Prepare a new changeset with the effect of REV undone in the
475 Prepare a new changeset with the effect of REV undone in the
476 current working directory.
476 current working directory.
477
477
478 If REV is the parent of the working directory, then this new changeset
478 If REV is the parent of the working directory, then this new changeset
479 is committed automatically. Otherwise, hg needs to merge the
479 is committed automatically. Otherwise, hg needs to merge the
480 changes and the merged result is left uncommitted.
480 changes and the merged result is left uncommitted.
481
481
482 .. note::
482 .. note::
483
483
484 backout cannot be used to fix either an unwanted or
484 backout cannot be used to fix either an unwanted or
485 incorrect merge.
485 incorrect merge.
486
486
487 .. container:: verbose
487 .. container:: verbose
488
488
489 By default, the pending changeset will have one parent,
489 By default, the pending changeset will have one parent,
490 maintaining a linear history. With --merge, the pending
490 maintaining a linear history. With --merge, the pending
491 changeset will instead have two parents: the old parent of the
491 changeset will instead have two parents: the old parent of the
492 working directory and a new child of REV that simply undoes REV.
492 working directory and a new child of REV that simply undoes REV.
493
493
494 Before version 1.7, the behavior without --merge was equivalent
494 Before version 1.7, the behavior without --merge was equivalent
495 to specifying --merge followed by :hg:`update --clean .` to
495 to specifying --merge followed by :hg:`update --clean .` to
496 cancel the merge and leave the child of REV as a head to be
496 cancel the merge and leave the child of REV as a head to be
497 merged separately.
497 merged separately.
498
498
499 See :hg:`help dates` for a list of formats valid for -d/--date.
499 See :hg:`help dates` for a list of formats valid for -d/--date.
500
500
501 See :hg:`help revert` for a way to restore files to the state
501 See :hg:`help revert` for a way to restore files to the state
502 of another revision.
502 of another revision.
503
503
504 Returns 0 on success, 1 if nothing to backout or there are unresolved
504 Returns 0 on success, 1 if nothing to backout or there are unresolved
505 files.
505 files.
506 '''
506 '''
507 if rev and node:
507 if rev and node:
508 raise error.Abort(_("please specify just one revision"))
508 raise error.Abort(_("please specify just one revision"))
509
509
510 if not rev:
510 if not rev:
511 rev = node
511 rev = node
512
512
513 if not rev:
513 if not rev:
514 raise error.Abort(_("please specify a revision to backout"))
514 raise error.Abort(_("please specify a revision to backout"))
515
515
516 date = opts.get('date')
516 date = opts.get('date')
517 if date:
517 if date:
518 opts['date'] = util.parsedate(date)
518 opts['date'] = util.parsedate(date)
519
519
520 cmdutil.checkunfinished(repo)
520 cmdutil.checkunfinished(repo)
521 cmdutil.bailifchanged(repo)
521 cmdutil.bailifchanged(repo)
522 node = scmutil.revsingle(repo, rev).node()
522 node = scmutil.revsingle(repo, rev).node()
523
523
524 op1, op2 = repo.dirstate.parents()
524 op1, op2 = repo.dirstate.parents()
525 if not repo.changelog.isancestor(node, op1):
525 if not repo.changelog.isancestor(node, op1):
526 raise error.Abort(_('cannot backout change that is not an ancestor'))
526 raise error.Abort(_('cannot backout change that is not an ancestor'))
527
527
528 p1, p2 = repo.changelog.parents(node)
528 p1, p2 = repo.changelog.parents(node)
529 if p1 == nullid:
529 if p1 == nullid:
530 raise error.Abort(_('cannot backout a change with no parents'))
530 raise error.Abort(_('cannot backout a change with no parents'))
531 if p2 != nullid:
531 if p2 != nullid:
532 if not opts.get('parent'):
532 if not opts.get('parent'):
533 raise error.Abort(_('cannot backout a merge changeset'))
533 raise error.Abort(_('cannot backout a merge changeset'))
534 p = repo.lookup(opts['parent'])
534 p = repo.lookup(opts['parent'])
535 if p not in (p1, p2):
535 if p not in (p1, p2):
536 raise error.Abort(_('%s is not a parent of %s') %
536 raise error.Abort(_('%s is not a parent of %s') %
537 (short(p), short(node)))
537 (short(p), short(node)))
538 parent = p
538 parent = p
539 else:
539 else:
540 if opts.get('parent'):
540 if opts.get('parent'):
541 raise error.Abort(_('cannot use --parent on non-merge changeset'))
541 raise error.Abort(_('cannot use --parent on non-merge changeset'))
542 parent = p1
542 parent = p1
543
543
544 # the backout should appear on the same branch
544 # the backout should appear on the same branch
545 wlock = repo.wlock()
545 wlock = repo.wlock()
546 try:
546 try:
547 branch = repo.dirstate.branch()
547 branch = repo.dirstate.branch()
548 bheads = repo.branchheads(branch)
548 bheads = repo.branchheads(branch)
549 rctx = scmutil.revsingle(repo, hex(parent))
549 rctx = scmutil.revsingle(repo, hex(parent))
550 if not opts.get('merge') and op1 != node:
550 if not opts.get('merge') and op1 != node:
551 dsguard = cmdutil.dirstateguard(repo, 'backout')
551 dsguard = cmdutil.dirstateguard(repo, 'backout')
552 try:
552 try:
553 ui.setconfig('ui', 'forcemerge', opts.get('tool', ''),
553 ui.setconfig('ui', 'forcemerge', opts.get('tool', ''),
554 'backout')
554 'backout')
555 stats = mergemod.update(repo, parent, True, True, False,
555 stats = mergemod.update(repo, parent, True, True, False,
556 node, False)
556 node, False)
557 repo.setparents(op1, op2)
557 repo.setparents(op1, op2)
558 dsguard.close()
558 dsguard.close()
559 hg._showstats(repo, stats)
559 hg._showstats(repo, stats)
560 if stats[3]:
560 if stats[3]:
561 repo.ui.status(_("use 'hg resolve' to retry unresolved "
561 repo.ui.status(_("use 'hg resolve' to retry unresolved "
562 "file merges\n"))
562 "file merges\n"))
563 return 1
563 return 1
564 elif not commit:
564 elif not commit:
565 msg = _("changeset %s backed out, "
565 msg = _("changeset %s backed out, "
566 "don't forget to commit.\n")
566 "don't forget to commit.\n")
567 ui.status(msg % short(node))
567 ui.status(msg % short(node))
568 return 0
568 return 0
569 finally:
569 finally:
570 ui.setconfig('ui', 'forcemerge', '', '')
570 ui.setconfig('ui', 'forcemerge', '', '')
571 lockmod.release(dsguard)
571 lockmod.release(dsguard)
572 else:
572 else:
573 hg.clean(repo, node, show_stats=False)
573 hg.clean(repo, node, show_stats=False)
574 repo.dirstate.setbranch(branch)
574 repo.dirstate.setbranch(branch)
575 cmdutil.revert(ui, repo, rctx, repo.dirstate.parents())
575 cmdutil.revert(ui, repo, rctx, repo.dirstate.parents())
576
576
577
577
578 def commitfunc(ui, repo, message, match, opts):
578 def commitfunc(ui, repo, message, match, opts):
579 editform = 'backout'
579 editform = 'backout'
580 e = cmdutil.getcommiteditor(editform=editform, **opts)
580 e = cmdutil.getcommiteditor(editform=editform, **opts)
581 if not message:
581 if not message:
582 # we don't translate commit messages
582 # we don't translate commit messages
583 message = "Backed out changeset %s" % short(node)
583 message = "Backed out changeset %s" % short(node)
584 e = cmdutil.getcommiteditor(edit=True, editform=editform)
584 e = cmdutil.getcommiteditor(edit=True, editform=editform)
585 return repo.commit(message, opts.get('user'), opts.get('date'),
585 return repo.commit(message, opts.get('user'), opts.get('date'),
586 match, editor=e)
586 match, editor=e)
587 newnode = cmdutil.commit(ui, repo, commitfunc, [], opts)
587 newnode = cmdutil.commit(ui, repo, commitfunc, [], opts)
588 if not newnode:
588 if not newnode:
589 ui.status(_("nothing changed\n"))
589 ui.status(_("nothing changed\n"))
590 return 1
590 return 1
591 cmdutil.commitstatus(repo, newnode, branch, bheads)
591 cmdutil.commitstatus(repo, newnode, branch, bheads)
592
592
593 def nice(node):
593 def nice(node):
594 return '%d:%s' % (repo.changelog.rev(node), short(node))
594 return '%d:%s' % (repo.changelog.rev(node), short(node))
595 ui.status(_('changeset %s backs out changeset %s\n') %
595 ui.status(_('changeset %s backs out changeset %s\n') %
596 (nice(repo.changelog.tip()), nice(node)))
596 (nice(repo.changelog.tip()), nice(node)))
597 if opts.get('merge') and op1 != node:
597 if opts.get('merge') and op1 != node:
598 hg.clean(repo, op1, show_stats=False)
598 hg.clean(repo, op1, show_stats=False)
599 ui.status(_('merging with changeset %s\n')
599 ui.status(_('merging with changeset %s\n')
600 % nice(repo.changelog.tip()))
600 % nice(repo.changelog.tip()))
601 try:
601 try:
602 ui.setconfig('ui', 'forcemerge', opts.get('tool', ''),
602 ui.setconfig('ui', 'forcemerge', opts.get('tool', ''),
603 'backout')
603 'backout')
604 return hg.merge(repo, hex(repo.changelog.tip()))
604 return hg.merge(repo, hex(repo.changelog.tip()))
605 finally:
605 finally:
606 ui.setconfig('ui', 'forcemerge', '', '')
606 ui.setconfig('ui', 'forcemerge', '', '')
607 finally:
607 finally:
608 wlock.release()
608 wlock.release()
609 return 0
609 return 0
610
610
611 @command('bisect',
611 @command('bisect',
612 [('r', 'reset', False, _('reset bisect state')),
612 [('r', 'reset', False, _('reset bisect state')),
613 ('g', 'good', False, _('mark changeset good')),
613 ('g', 'good', False, _('mark changeset good')),
614 ('b', 'bad', False, _('mark changeset bad')),
614 ('b', 'bad', False, _('mark changeset bad')),
615 ('s', 'skip', False, _('skip testing changeset')),
615 ('s', 'skip', False, _('skip testing changeset')),
616 ('e', 'extend', False, _('extend the bisect range')),
616 ('e', 'extend', False, _('extend the bisect range')),
617 ('c', 'command', '', _('use command to check changeset state'), _('CMD')),
617 ('c', 'command', '', _('use command to check changeset state'), _('CMD')),
618 ('U', 'noupdate', False, _('do not update to target'))],
618 ('U', 'noupdate', False, _('do not update to target'))],
619 _("[-gbsr] [-U] [-c CMD] [REV]"))
619 _("[-gbsr] [-U] [-c CMD] [REV]"))
620 def bisect(ui, repo, rev=None, extra=None, command=None,
620 def bisect(ui, repo, rev=None, extra=None, command=None,
621 reset=None, good=None, bad=None, skip=None, extend=None,
621 reset=None, good=None, bad=None, skip=None, extend=None,
622 noupdate=None):
622 noupdate=None):
623 """subdivision search of changesets
623 """subdivision search of changesets
624
624
625 This command helps to find changesets which introduce problems. To
625 This command helps to find changesets which introduce problems. To
626 use, mark the earliest changeset you know exhibits the problem as
626 use, mark the earliest changeset you know exhibits the problem as
627 bad, then mark the latest changeset which is free from the problem
627 bad, then mark the latest changeset which is free from the problem
628 as good. Bisect will update your working directory to a revision
628 as good. Bisect will update your working directory to a revision
629 for testing (unless the -U/--noupdate option is specified). Once
629 for testing (unless the -U/--noupdate option is specified). Once
630 you have performed tests, mark the working directory as good or
630 you have performed tests, mark the working directory as good or
631 bad, and bisect will either update to another candidate changeset
631 bad, and bisect will either update to another candidate changeset
632 or announce that it has found the bad revision.
632 or announce that it has found the bad revision.
633
633
634 As a shortcut, you can also use the revision argument to mark a
634 As a shortcut, you can also use the revision argument to mark a
635 revision as good or bad without checking it out first.
635 revision as good or bad without checking it out first.
636
636
637 If you supply a command, it will be used for automatic bisection.
637 If you supply a command, it will be used for automatic bisection.
638 The environment variable HG_NODE will contain the ID of the
638 The environment variable HG_NODE will contain the ID of the
639 changeset being tested. The exit status of the command will be
639 changeset being tested. The exit status of the command will be
640 used to mark revisions as good or bad: status 0 means good, 125
640 used to mark revisions as good or bad: status 0 means good, 125
641 means to skip the revision, 127 (command not found) will abort the
641 means to skip the revision, 127 (command not found) will abort the
642 bisection, and any other non-zero exit status means the revision
642 bisection, and any other non-zero exit status means the revision
643 is bad.
643 is bad.
644
644
645 .. container:: verbose
645 .. container:: verbose
646
646
647 Some examples:
647 Some examples:
648
648
649 - start a bisection with known bad revision 34, and good revision 12::
649 - start a bisection with known bad revision 34, and good revision 12::
650
650
651 hg bisect --bad 34
651 hg bisect --bad 34
652 hg bisect --good 12
652 hg bisect --good 12
653
653
654 - advance the current bisection by marking current revision as good or
654 - advance the current bisection by marking current revision as good or
655 bad::
655 bad::
656
656
657 hg bisect --good
657 hg bisect --good
658 hg bisect --bad
658 hg bisect --bad
659
659
660 - mark the current revision, or a known revision, to be skipped (e.g. if
660 - mark the current revision, or a known revision, to be skipped (e.g. if
661 that revision is not usable because of another issue)::
661 that revision is not usable because of another issue)::
662
662
663 hg bisect --skip
663 hg bisect --skip
664 hg bisect --skip 23
664 hg bisect --skip 23
665
665
666 - skip all revisions that do not touch directories ``foo`` or ``bar``::
666 - skip all revisions that do not touch directories ``foo`` or ``bar``::
667
667
668 hg bisect --skip "!( file('path:foo') & file('path:bar') )"
668 hg bisect --skip "!( file('path:foo') & file('path:bar') )"
669
669
670 - forget the current bisection::
670 - forget the current bisection::
671
671
672 hg bisect --reset
672 hg bisect --reset
673
673
674 - use 'make && make tests' to automatically find the first broken
674 - use 'make && make tests' to automatically find the first broken
675 revision::
675 revision::
676
676
677 hg bisect --reset
677 hg bisect --reset
678 hg bisect --bad 34
678 hg bisect --bad 34
679 hg bisect --good 12
679 hg bisect --good 12
680 hg bisect --command "make && make tests"
680 hg bisect --command "make && make tests"
681
681
682 - see all changesets whose states are already known in the current
682 - see all changesets whose states are already known in the current
683 bisection::
683 bisection::
684
684
685 hg log -r "bisect(pruned)"
685 hg log -r "bisect(pruned)"
686
686
687 - see the changeset currently being bisected (especially useful
687 - see the changeset currently being bisected (especially useful
688 if running with -U/--noupdate)::
688 if running with -U/--noupdate)::
689
689
690 hg log -r "bisect(current)"
690 hg log -r "bisect(current)"
691
691
692 - see all changesets that took part in the current bisection::
692 - see all changesets that took part in the current bisection::
693
693
694 hg log -r "bisect(range)"
694 hg log -r "bisect(range)"
695
695
696 - you can even get a nice graph::
696 - you can even get a nice graph::
697
697
698 hg log --graph -r "bisect(range)"
698 hg log --graph -r "bisect(range)"
699
699
700 See :hg:`help revsets` for more about the `bisect()` keyword.
700 See :hg:`help revsets` for more about the `bisect()` keyword.
701
701
702 Returns 0 on success.
702 Returns 0 on success.
703 """
703 """
704 def extendbisectrange(nodes, good):
704 def extendbisectrange(nodes, good):
705 # bisect is incomplete when it ends on a merge node and
705 # bisect is incomplete when it ends on a merge node and
706 # one of the parent was not checked.
706 # one of the parent was not checked.
707 parents = repo[nodes[0]].parents()
707 parents = repo[nodes[0]].parents()
708 if len(parents) > 1:
708 if len(parents) > 1:
709 if good:
709 if good:
710 side = state['bad']
710 side = state['bad']
711 else:
711 else:
712 side = state['good']
712 side = state['good']
713 num = len(set(i.node() for i in parents) & set(side))
713 num = len(set(i.node() for i in parents) & set(side))
714 if num == 1:
714 if num == 1:
715 return parents[0].ancestor(parents[1])
715 return parents[0].ancestor(parents[1])
716 return None
716 return None
717
717
718 def print_result(nodes, good):
718 def print_result(nodes, good):
719 displayer = cmdutil.show_changeset(ui, repo, {})
719 displayer = cmdutil.show_changeset(ui, repo, {})
720 if len(nodes) == 1:
720 if len(nodes) == 1:
721 # narrowed it down to a single revision
721 # narrowed it down to a single revision
722 if good:
722 if good:
723 ui.write(_("The first good revision is:\n"))
723 ui.write(_("The first good revision is:\n"))
724 else:
724 else:
725 ui.write(_("The first bad revision is:\n"))
725 ui.write(_("The first bad revision is:\n"))
726 displayer.show(repo[nodes[0]])
726 displayer.show(repo[nodes[0]])
727 extendnode = extendbisectrange(nodes, good)
727 extendnode = extendbisectrange(nodes, good)
728 if extendnode is not None:
728 if extendnode is not None:
729 ui.write(_('Not all ancestors of this changeset have been'
729 ui.write(_('Not all ancestors of this changeset have been'
730 ' checked.\nUse bisect --extend to continue the '
730 ' checked.\nUse bisect --extend to continue the '
731 'bisection from\nthe common ancestor, %s.\n')
731 'bisection from\nthe common ancestor, %s.\n')
732 % extendnode)
732 % extendnode)
733 else:
733 else:
734 # multiple possible revisions
734 # multiple possible revisions
735 if good:
735 if good:
736 ui.write(_("Due to skipped revisions, the first "
736 ui.write(_("Due to skipped revisions, the first "
737 "good revision could be any of:\n"))
737 "good revision could be any of:\n"))
738 else:
738 else:
739 ui.write(_("Due to skipped revisions, the first "
739 ui.write(_("Due to skipped revisions, the first "
740 "bad revision could be any of:\n"))
740 "bad revision could be any of:\n"))
741 for n in nodes:
741 for n in nodes:
742 displayer.show(repo[n])
742 displayer.show(repo[n])
743 displayer.close()
743 displayer.close()
744
744
745 def check_state(state, interactive=True):
745 def check_state(state, interactive=True):
746 if not state['good'] or not state['bad']:
746 if not state['good'] or not state['bad']:
747 if (good or bad or skip or reset) and interactive:
747 if (good or bad or skip or reset) and interactive:
748 return
748 return
749 if not state['good']:
749 if not state['good']:
750 raise error.Abort(_('cannot bisect (no known good revisions)'))
750 raise error.Abort(_('cannot bisect (no known good revisions)'))
751 else:
751 else:
752 raise error.Abort(_('cannot bisect (no known bad revisions)'))
752 raise error.Abort(_('cannot bisect (no known bad revisions)'))
753 return True
753 return True
754
754
755 # backward compatibility
755 # backward compatibility
756 if rev in "good bad reset init".split():
756 if rev in "good bad reset init".split():
757 ui.warn(_("(use of 'hg bisect <cmd>' is deprecated)\n"))
757 ui.warn(_("(use of 'hg bisect <cmd>' is deprecated)\n"))
758 cmd, rev, extra = rev, extra, None
758 cmd, rev, extra = rev, extra, None
759 if cmd == "good":
759 if cmd == "good":
760 good = True
760 good = True
761 elif cmd == "bad":
761 elif cmd == "bad":
762 bad = True
762 bad = True
763 else:
763 else:
764 reset = True
764 reset = True
765 elif extra or good + bad + skip + reset + extend + bool(command) > 1:
765 elif extra or good + bad + skip + reset + extend + bool(command) > 1:
766 raise error.Abort(_('incompatible arguments'))
766 raise error.Abort(_('incompatible arguments'))
767
767
768 cmdutil.checkunfinished(repo)
768 cmdutil.checkunfinished(repo)
769
769
770 if reset:
770 if reset:
771 p = repo.join("bisect.state")
771 p = repo.join("bisect.state")
772 if os.path.exists(p):
772 if os.path.exists(p):
773 os.unlink(p)
773 os.unlink(p)
774 return
774 return
775
775
776 state = hbisect.load_state(repo)
776 state = hbisect.load_state(repo)
777
777
778 if command:
778 if command:
779 changesets = 1
779 changesets = 1
780 if noupdate:
780 if noupdate:
781 try:
781 try:
782 node = state['current'][0]
782 node = state['current'][0]
783 except LookupError:
783 except LookupError:
784 raise error.Abort(_('current bisect revision is unknown - '
784 raise error.Abort(_('current bisect revision is unknown - '
785 'start a new bisect to fix'))
785 'start a new bisect to fix'))
786 else:
786 else:
787 node, p2 = repo.dirstate.parents()
787 node, p2 = repo.dirstate.parents()
788 if p2 != nullid:
788 if p2 != nullid:
789 raise error.Abort(_('current bisect revision is a merge'))
789 raise error.Abort(_('current bisect revision is a merge'))
790 try:
790 try:
791 while changesets:
791 while changesets:
792 # update state
792 # update state
793 state['current'] = [node]
793 state['current'] = [node]
794 hbisect.save_state(repo, state)
794 hbisect.save_state(repo, state)
795 status = ui.system(command, environ={'HG_NODE': hex(node)})
795 status = ui.system(command, environ={'HG_NODE': hex(node)})
796 if status == 125:
796 if status == 125:
797 transition = "skip"
797 transition = "skip"
798 elif status == 0:
798 elif status == 0:
799 transition = "good"
799 transition = "good"
800 # status < 0 means process was killed
800 # status < 0 means process was killed
801 elif status == 127:
801 elif status == 127:
802 raise error.Abort(_("failed to execute %s") % command)
802 raise error.Abort(_("failed to execute %s") % command)
803 elif status < 0:
803 elif status < 0:
804 raise error.Abort(_("%s killed") % command)
804 raise error.Abort(_("%s killed") % command)
805 else:
805 else:
806 transition = "bad"
806 transition = "bad"
807 ctx = scmutil.revsingle(repo, rev, node)
807 ctx = scmutil.revsingle(repo, rev, node)
808 rev = None # clear for future iterations
808 rev = None # clear for future iterations
809 state[transition].append(ctx.node())
809 state[transition].append(ctx.node())
810 ui.status(_('changeset %d:%s: %s\n') % (ctx, ctx, transition))
810 ui.status(_('changeset %d:%s: %s\n') % (ctx, ctx, transition))
811 check_state(state, interactive=False)
811 check_state(state, interactive=False)
812 # bisect
812 # bisect
813 nodes, changesets, bgood = hbisect.bisect(repo.changelog, state)
813 nodes, changesets, bgood = hbisect.bisect(repo.changelog, state)
814 # update to next check
814 # update to next check
815 node = nodes[0]
815 node = nodes[0]
816 if not noupdate:
816 if not noupdate:
817 cmdutil.bailifchanged(repo)
817 cmdutil.bailifchanged(repo)
818 hg.clean(repo, node, show_stats=False)
818 hg.clean(repo, node, show_stats=False)
819 finally:
819 finally:
820 state['current'] = [node]
820 state['current'] = [node]
821 hbisect.save_state(repo, state)
821 hbisect.save_state(repo, state)
822 print_result(nodes, bgood)
822 print_result(nodes, bgood)
823 return
823 return
824
824
825 # update state
825 # update state
826
826
827 if rev:
827 if rev:
828 nodes = [repo.lookup(i) for i in scmutil.revrange(repo, [rev])]
828 nodes = [repo.lookup(i) for i in scmutil.revrange(repo, [rev])]
829 else:
829 else:
830 nodes = [repo.lookup('.')]
830 nodes = [repo.lookup('.')]
831
831
832 if good or bad or skip:
832 if good or bad or skip:
833 if good:
833 if good:
834 state['good'] += nodes
834 state['good'] += nodes
835 elif bad:
835 elif bad:
836 state['bad'] += nodes
836 state['bad'] += nodes
837 elif skip:
837 elif skip:
838 state['skip'] += nodes
838 state['skip'] += nodes
839 hbisect.save_state(repo, state)
839 hbisect.save_state(repo, state)
840
840
841 if not check_state(state):
841 if not check_state(state):
842 return
842 return
843
843
844 # actually bisect
844 # actually bisect
845 nodes, changesets, good = hbisect.bisect(repo.changelog, state)
845 nodes, changesets, good = hbisect.bisect(repo.changelog, state)
846 if extend:
846 if extend:
847 if not changesets:
847 if not changesets:
848 extendnode = extendbisectrange(nodes, good)
848 extendnode = extendbisectrange(nodes, good)
849 if extendnode is not None:
849 if extendnode is not None:
850 ui.write(_("Extending search to changeset %d:%s\n")
850 ui.write(_("Extending search to changeset %d:%s\n")
851 % (extendnode.rev(), extendnode))
851 % (extendnode.rev(), extendnode))
852 state['current'] = [extendnode.node()]
852 state['current'] = [extendnode.node()]
853 hbisect.save_state(repo, state)
853 hbisect.save_state(repo, state)
854 if noupdate:
854 if noupdate:
855 return
855 return
856 cmdutil.bailifchanged(repo)
856 cmdutil.bailifchanged(repo)
857 return hg.clean(repo, extendnode.node())
857 return hg.clean(repo, extendnode.node())
858 raise error.Abort(_("nothing to extend"))
858 raise error.Abort(_("nothing to extend"))
859
859
860 if changesets == 0:
860 if changesets == 0:
861 print_result(nodes, good)
861 print_result(nodes, good)
862 else:
862 else:
863 assert len(nodes) == 1 # only a single node can be tested next
863 assert len(nodes) == 1 # only a single node can be tested next
864 node = nodes[0]
864 node = nodes[0]
865 # compute the approximate number of remaining tests
865 # compute the approximate number of remaining tests
866 tests, size = 0, 2
866 tests, size = 0, 2
867 while size <= changesets:
867 while size <= changesets:
868 tests, size = tests + 1, size * 2
868 tests, size = tests + 1, size * 2
869 rev = repo.changelog.rev(node)
869 rev = repo.changelog.rev(node)
870 ui.write(_("Testing changeset %d:%s "
870 ui.write(_("Testing changeset %d:%s "
871 "(%d changesets remaining, ~%d tests)\n")
871 "(%d changesets remaining, ~%d tests)\n")
872 % (rev, short(node), changesets, tests))
872 % (rev, short(node), changesets, tests))
873 state['current'] = [node]
873 state['current'] = [node]
874 hbisect.save_state(repo, state)
874 hbisect.save_state(repo, state)
875 if not noupdate:
875 if not noupdate:
876 cmdutil.bailifchanged(repo)
876 cmdutil.bailifchanged(repo)
877 return hg.clean(repo, node)
877 return hg.clean(repo, node)
878
878
879 @command('bookmarks|bookmark',
879 @command('bookmarks|bookmark',
880 [('f', 'force', False, _('force')),
880 [('f', 'force', False, _('force')),
881 ('r', 'rev', '', _('revision'), _('REV')),
881 ('r', 'rev', '', _('revision'), _('REV')),
882 ('d', 'delete', False, _('delete a given bookmark')),
882 ('d', 'delete', False, _('delete a given bookmark')),
883 ('m', 'rename', '', _('rename a given bookmark'), _('OLD')),
883 ('m', 'rename', '', _('rename a given bookmark'), _('OLD')),
884 ('i', 'inactive', False, _('mark a bookmark inactive')),
884 ('i', 'inactive', False, _('mark a bookmark inactive')),
885 ] + formatteropts,
885 ] + formatteropts,
886 _('hg bookmarks [OPTIONS]... [NAME]...'))
886 _('hg bookmarks [OPTIONS]... [NAME]...'))
887 def bookmark(ui, repo, *names, **opts):
887 def bookmark(ui, repo, *names, **opts):
888 '''create a new bookmark or list existing bookmarks
888 '''create a new bookmark or list existing bookmarks
889
889
890 Bookmarks are labels on changesets to help track lines of development.
890 Bookmarks are labels on changesets to help track lines of development.
891 Bookmarks are unversioned and can be moved, renamed and deleted.
891 Bookmarks are unversioned and can be moved, renamed and deleted.
892 Deleting or moving a bookmark has no effect on the associated changesets.
892 Deleting or moving a bookmark has no effect on the associated changesets.
893
893
894 Creating or updating to a bookmark causes it to be marked as 'active'.
894 Creating or updating to a bookmark causes it to be marked as 'active'.
895 The active bookmark is indicated with a '*'.
895 The active bookmark is indicated with a '*'.
896 When a commit is made, the active bookmark will advance to the new commit.
896 When a commit is made, the active bookmark will advance to the new commit.
897 A plain :hg:`update` will also advance an active bookmark, if possible.
897 A plain :hg:`update` will also advance an active bookmark, if possible.
898 Updating away from a bookmark will cause it to be deactivated.
898 Updating away from a bookmark will cause it to be deactivated.
899
899
900 Bookmarks can be pushed and pulled between repositories (see
900 Bookmarks can be pushed and pulled between repositories (see
901 :hg:`help push` and :hg:`help pull`). If a shared bookmark has
901 :hg:`help push` and :hg:`help pull`). If a shared bookmark has
902 diverged, a new 'divergent bookmark' of the form 'name@path' will
902 diverged, a new 'divergent bookmark' of the form 'name@path' will
903 be created. Using :hg:`merge` will resolve the divergence.
903 be created. Using :hg:`merge` will resolve the divergence.
904
904
905 A bookmark named '@' has the special property that :hg:`clone` will
905 A bookmark named '@' has the special property that :hg:`clone` will
906 check it out by default if it exists.
906 check it out by default if it exists.
907
907
908 .. container:: verbose
908 .. container:: verbose
909
909
910 Examples:
910 Examples:
911
911
912 - create an active bookmark for a new line of development::
912 - create an active bookmark for a new line of development::
913
913
914 hg book new-feature
914 hg book new-feature
915
915
916 - create an inactive bookmark as a place marker::
916 - create an inactive bookmark as a place marker::
917
917
918 hg book -i reviewed
918 hg book -i reviewed
919
919
920 - create an inactive bookmark on another changeset::
920 - create an inactive bookmark on another changeset::
921
921
922 hg book -r .^ tested
922 hg book -r .^ tested
923
923
924 - rename bookmark turkey to dinner::
924 - rename bookmark turkey to dinner::
925
925
926 hg book -m turkey dinner
926 hg book -m turkey dinner
927
927
928 - move the '@' bookmark from another branch::
928 - move the '@' bookmark from another branch::
929
929
930 hg book -f @
930 hg book -f @
931 '''
931 '''
932 force = opts.get('force')
932 force = opts.get('force')
933 rev = opts.get('rev')
933 rev = opts.get('rev')
934 delete = opts.get('delete')
934 delete = opts.get('delete')
935 rename = opts.get('rename')
935 rename = opts.get('rename')
936 inactive = opts.get('inactive')
936 inactive = opts.get('inactive')
937
937
938 def checkformat(mark):
938 def checkformat(mark):
939 mark = mark.strip()
939 mark = mark.strip()
940 if not mark:
940 if not mark:
941 raise error.Abort(_("bookmark names cannot consist entirely of "
941 raise error.Abort(_("bookmark names cannot consist entirely of "
942 "whitespace"))
942 "whitespace"))
943 scmutil.checknewlabel(repo, mark, 'bookmark')
943 scmutil.checknewlabel(repo, mark, 'bookmark')
944 return mark
944 return mark
945
945
946 def checkconflict(repo, mark, cur, force=False, target=None):
946 def checkconflict(repo, mark, cur, force=False, target=None):
947 if mark in marks and not force:
947 if mark in marks and not force:
948 if target:
948 if target:
949 if marks[mark] == target and target == cur:
949 if marks[mark] == target and target == cur:
950 # re-activating a bookmark
950 # re-activating a bookmark
951 return
951 return
952 anc = repo.changelog.ancestors([repo[target].rev()])
952 anc = repo.changelog.ancestors([repo[target].rev()])
953 bmctx = repo[marks[mark]]
953 bmctx = repo[marks[mark]]
954 divs = [repo[b].node() for b in marks
954 divs = [repo[b].node() for b in marks
955 if b.split('@', 1)[0] == mark.split('@', 1)[0]]
955 if b.split('@', 1)[0] == mark.split('@', 1)[0]]
956
956
957 # allow resolving a single divergent bookmark even if moving
957 # allow resolving a single divergent bookmark even if moving
958 # the bookmark across branches when a revision is specified
958 # the bookmark across branches when a revision is specified
959 # that contains a divergent bookmark
959 # that contains a divergent bookmark
960 if bmctx.rev() not in anc and target in divs:
960 if bmctx.rev() not in anc and target in divs:
961 bookmarks.deletedivergent(repo, [target], mark)
961 bookmarks.deletedivergent(repo, [target], mark)
962 return
962 return
963
963
964 deletefrom = [b for b in divs
964 deletefrom = [b for b in divs
965 if repo[b].rev() in anc or b == target]
965 if repo[b].rev() in anc or b == target]
966 bookmarks.deletedivergent(repo, deletefrom, mark)
966 bookmarks.deletedivergent(repo, deletefrom, mark)
967 if bookmarks.validdest(repo, bmctx, repo[target]):
967 if bookmarks.validdest(repo, bmctx, repo[target]):
968 ui.status(_("moving bookmark '%s' forward from %s\n") %
968 ui.status(_("moving bookmark '%s' forward from %s\n") %
969 (mark, short(bmctx.node())))
969 (mark, short(bmctx.node())))
970 return
970 return
971 raise error.Abort(_("bookmark '%s' already exists "
971 raise error.Abort(_("bookmark '%s' already exists "
972 "(use -f to force)") % mark)
972 "(use -f to force)") % mark)
973 if ((mark in repo.branchmap() or mark == repo.dirstate.branch())
973 if ((mark in repo.branchmap() or mark == repo.dirstate.branch())
974 and not force):
974 and not force):
975 raise error.Abort(
975 raise error.Abort(
976 _("a bookmark cannot have the name of an existing branch"))
976 _("a bookmark cannot have the name of an existing branch"))
977
977
978 if delete and rename:
978 if delete and rename:
979 raise error.Abort(_("--delete and --rename are incompatible"))
979 raise error.Abort(_("--delete and --rename are incompatible"))
980 if delete and rev:
980 if delete and rev:
981 raise error.Abort(_("--rev is incompatible with --delete"))
981 raise error.Abort(_("--rev is incompatible with --delete"))
982 if rename and rev:
982 if rename and rev:
983 raise error.Abort(_("--rev is incompatible with --rename"))
983 raise error.Abort(_("--rev is incompatible with --rename"))
984 if not names and (delete or rev):
984 if not names and (delete or rev):
985 raise error.Abort(_("bookmark name required"))
985 raise error.Abort(_("bookmark name required"))
986
986
987 if delete or rename or names or inactive:
987 if delete or rename or names or inactive:
988 wlock = lock = tr = None
988 wlock = lock = tr = None
989 try:
989 try:
990 wlock = repo.wlock()
990 wlock = repo.wlock()
991 lock = repo.lock()
991 lock = repo.lock()
992 cur = repo.changectx('.').node()
992 cur = repo.changectx('.').node()
993 marks = repo._bookmarks
993 marks = repo._bookmarks
994 if delete:
994 if delete:
995 tr = repo.transaction('bookmark')
995 tr = repo.transaction('bookmark')
996 for mark in names:
996 for mark in names:
997 if mark not in marks:
997 if mark not in marks:
998 raise error.Abort(_("bookmark '%s' does not exist") %
998 raise error.Abort(_("bookmark '%s' does not exist") %
999 mark)
999 mark)
1000 if mark == repo._activebookmark:
1000 if mark == repo._activebookmark:
1001 bookmarks.deactivate(repo)
1001 bookmarks.deactivate(repo)
1002 del marks[mark]
1002 del marks[mark]
1003
1003
1004 elif rename:
1004 elif rename:
1005 tr = repo.transaction('bookmark')
1005 tr = repo.transaction('bookmark')
1006 if not names:
1006 if not names:
1007 raise error.Abort(_("new bookmark name required"))
1007 raise error.Abort(_("new bookmark name required"))
1008 elif len(names) > 1:
1008 elif len(names) > 1:
1009 raise error.Abort(_("only one new bookmark name allowed"))
1009 raise error.Abort(_("only one new bookmark name allowed"))
1010 mark = checkformat(names[0])
1010 mark = checkformat(names[0])
1011 if rename not in marks:
1011 if rename not in marks:
1012 raise error.Abort(_("bookmark '%s' does not exist")
1012 raise error.Abort(_("bookmark '%s' does not exist")
1013 % rename)
1013 % rename)
1014 checkconflict(repo, mark, cur, force)
1014 checkconflict(repo, mark, cur, force)
1015 marks[mark] = marks[rename]
1015 marks[mark] = marks[rename]
1016 if repo._activebookmark == rename and not inactive:
1016 if repo._activebookmark == rename and not inactive:
1017 bookmarks.activate(repo, mark)
1017 bookmarks.activate(repo, mark)
1018 del marks[rename]
1018 del marks[rename]
1019 elif names:
1019 elif names:
1020 tr = repo.transaction('bookmark')
1020 tr = repo.transaction('bookmark')
1021 newact = None
1021 newact = None
1022 for mark in names:
1022 for mark in names:
1023 mark = checkformat(mark)
1023 mark = checkformat(mark)
1024 if newact is None:
1024 if newact is None:
1025 newact = mark
1025 newact = mark
1026 if inactive and mark == repo._activebookmark:
1026 if inactive and mark == repo._activebookmark:
1027 bookmarks.deactivate(repo)
1027 bookmarks.deactivate(repo)
1028 return
1028 return
1029 tgt = cur
1029 tgt = cur
1030 if rev:
1030 if rev:
1031 tgt = scmutil.revsingle(repo, rev).node()
1031 tgt = scmutil.revsingle(repo, rev).node()
1032 checkconflict(repo, mark, cur, force, tgt)
1032 checkconflict(repo, mark, cur, force, tgt)
1033 marks[mark] = tgt
1033 marks[mark] = tgt
1034 if not inactive and cur == marks[newact] and not rev:
1034 if not inactive and cur == marks[newact] and not rev:
1035 bookmarks.activate(repo, newact)
1035 bookmarks.activate(repo, newact)
1036 elif cur != tgt and newact == repo._activebookmark:
1036 elif cur != tgt and newact == repo._activebookmark:
1037 bookmarks.deactivate(repo)
1037 bookmarks.deactivate(repo)
1038 elif inactive:
1038 elif inactive:
1039 if len(marks) == 0:
1039 if len(marks) == 0:
1040 ui.status(_("no bookmarks set\n"))
1040 ui.status(_("no bookmarks set\n"))
1041 elif not repo._activebookmark:
1041 elif not repo._activebookmark:
1042 ui.status(_("no active bookmark\n"))
1042 ui.status(_("no active bookmark\n"))
1043 else:
1043 else:
1044 bookmarks.deactivate(repo)
1044 bookmarks.deactivate(repo)
1045 if tr is not None:
1045 if tr is not None:
1046 marks.recordchange(tr)
1046 marks.recordchange(tr)
1047 tr.close()
1047 tr.close()
1048 finally:
1048 finally:
1049 lockmod.release(tr, lock, wlock)
1049 lockmod.release(tr, lock, wlock)
1050 else: # show bookmarks
1050 else: # show bookmarks
1051 fm = ui.formatter('bookmarks', opts)
1051 fm = ui.formatter('bookmarks', opts)
1052 hexfn = fm.hexfunc
1052 hexfn = fm.hexfunc
1053 marks = repo._bookmarks
1053 marks = repo._bookmarks
1054 if len(marks) == 0 and not fm:
1054 if len(marks) == 0 and not fm:
1055 ui.status(_("no bookmarks set\n"))
1055 ui.status(_("no bookmarks set\n"))
1056 for bmark, n in sorted(marks.iteritems()):
1056 for bmark, n in sorted(marks.iteritems()):
1057 active = repo._activebookmark
1057 active = repo._activebookmark
1058 if bmark == active:
1058 if bmark == active:
1059 prefix, label = '*', activebookmarklabel
1059 prefix, label = '*', activebookmarklabel
1060 else:
1060 else:
1061 prefix, label = ' ', ''
1061 prefix, label = ' ', ''
1062
1062
1063 fm.startitem()
1063 fm.startitem()
1064 if not ui.quiet:
1064 if not ui.quiet:
1065 fm.plain(' %s ' % prefix, label=label)
1065 fm.plain(' %s ' % prefix, label=label)
1066 fm.write('bookmark', '%s', bmark, label=label)
1066 fm.write('bookmark', '%s', bmark, label=label)
1067 pad = " " * (25 - encoding.colwidth(bmark))
1067 pad = " " * (25 - encoding.colwidth(bmark))
1068 fm.condwrite(not ui.quiet, 'rev node', pad + ' %d:%s',
1068 fm.condwrite(not ui.quiet, 'rev node', pad + ' %d:%s',
1069 repo.changelog.rev(n), hexfn(n), label=label)
1069 repo.changelog.rev(n), hexfn(n), label=label)
1070 fm.data(active=(bmark == active))
1070 fm.data(active=(bmark == active))
1071 fm.plain('\n')
1071 fm.plain('\n')
1072 fm.end()
1072 fm.end()
1073
1073
1074 @command('branch',
1074 @command('branch',
1075 [('f', 'force', None,
1075 [('f', 'force', None,
1076 _('set branch name even if it shadows an existing branch')),
1076 _('set branch name even if it shadows an existing branch')),
1077 ('C', 'clean', None, _('reset branch name to parent branch name'))],
1077 ('C', 'clean', None, _('reset branch name to parent branch name'))],
1078 _('[-fC] [NAME]'))
1078 _('[-fC] [NAME]'))
1079 def branch(ui, repo, label=None, **opts):
1079 def branch(ui, repo, label=None, **opts):
1080 """set or show the current branch name
1080 """set or show the current branch name
1081
1081
1082 .. note::
1082 .. note::
1083
1083
1084 Branch names are permanent and global. Use :hg:`bookmark` to create a
1084 Branch names are permanent and global. Use :hg:`bookmark` to create a
1085 light-weight bookmark instead. See :hg:`help glossary` for more
1085 light-weight bookmark instead. See :hg:`help glossary` for more
1086 information about named branches and bookmarks.
1086 information about named branches and bookmarks.
1087
1087
1088 With no argument, show the current branch name. With one argument,
1088 With no argument, show the current branch name. With one argument,
1089 set the working directory branch name (the branch will not exist
1089 set the working directory branch name (the branch will not exist
1090 in the repository until the next commit). Standard practice
1090 in the repository until the next commit). Standard practice
1091 recommends that primary development take place on the 'default'
1091 recommends that primary development take place on the 'default'
1092 branch.
1092 branch.
1093
1093
1094 Unless -f/--force is specified, branch will not let you set a
1094 Unless -f/--force is specified, branch will not let you set a
1095 branch name that already exists.
1095 branch name that already exists.
1096
1096
1097 Use -C/--clean to reset the working directory branch to that of
1097 Use -C/--clean to reset the working directory branch to that of
1098 the parent of the working directory, negating a previous branch
1098 the parent of the working directory, negating a previous branch
1099 change.
1099 change.
1100
1100
1101 Use the command :hg:`update` to switch to an existing branch. Use
1101 Use the command :hg:`update` to switch to an existing branch. Use
1102 :hg:`commit --close-branch` to mark this branch head as closed.
1102 :hg:`commit --close-branch` to mark this branch head as closed.
1103 When all heads of the branch are closed, the branch will be
1103 When all heads of the branch are closed, the branch will be
1104 considered closed.
1104 considered closed.
1105
1105
1106 Returns 0 on success.
1106 Returns 0 on success.
1107 """
1107 """
1108 if label:
1108 if label:
1109 label = label.strip()
1109 label = label.strip()
1110
1110
1111 if not opts.get('clean') and not label:
1111 if not opts.get('clean') and not label:
1112 ui.write("%s\n" % repo.dirstate.branch())
1112 ui.write("%s\n" % repo.dirstate.branch())
1113 return
1113 return
1114
1114
1115 wlock = repo.wlock()
1115 wlock = repo.wlock()
1116 try:
1116 try:
1117 if opts.get('clean'):
1117 if opts.get('clean'):
1118 label = repo[None].p1().branch()
1118 label = repo[None].p1().branch()
1119 repo.dirstate.setbranch(label)
1119 repo.dirstate.setbranch(label)
1120 ui.status(_('reset working directory to branch %s\n') % label)
1120 ui.status(_('reset working directory to branch %s\n') % label)
1121 elif label:
1121 elif label:
1122 if not opts.get('force') and label in repo.branchmap():
1122 if not opts.get('force') and label in repo.branchmap():
1123 if label not in [p.branch() for p in repo.parents()]:
1123 if label not in [p.branch() for p in repo.parents()]:
1124 raise error.Abort(_('a branch of the same name already'
1124 raise error.Abort(_('a branch of the same name already'
1125 ' exists'),
1125 ' exists'),
1126 # i18n: "it" refers to an existing branch
1126 # i18n: "it" refers to an existing branch
1127 hint=_("use 'hg update' to switch to it"))
1127 hint=_("use 'hg update' to switch to it"))
1128 scmutil.checknewlabel(repo, label, 'branch')
1128 scmutil.checknewlabel(repo, label, 'branch')
1129 repo.dirstate.setbranch(label)
1129 repo.dirstate.setbranch(label)
1130 ui.status(_('marked working directory as branch %s\n') % label)
1130 ui.status(_('marked working directory as branch %s\n') % label)
1131
1131
1132 # find any open named branches aside from default
1132 # find any open named branches aside from default
1133 others = [n for n, h, t, c in repo.branchmap().iterbranches()
1133 others = [n for n, h, t, c in repo.branchmap().iterbranches()
1134 if n != "default" and not c]
1134 if n != "default" and not c]
1135 if not others:
1135 if not others:
1136 ui.status(_('(branches are permanent and global, '
1136 ui.status(_('(branches are permanent and global, '
1137 'did you want a bookmark?)\n'))
1137 'did you want a bookmark?)\n'))
1138 finally:
1138 finally:
1139 wlock.release()
1139 wlock.release()
1140
1140
1141 @command('branches',
1141 @command('branches',
1142 [('a', 'active', False,
1142 [('a', 'active', False,
1143 _('show only branches that have unmerged heads (DEPRECATED)')),
1143 _('show only branches that have unmerged heads (DEPRECATED)')),
1144 ('c', 'closed', False, _('show normal and closed branches')),
1144 ('c', 'closed', False, _('show normal and closed branches')),
1145 ] + formatteropts,
1145 ] + formatteropts,
1146 _('[-ac]'))
1146 _('[-ac]'))
1147 def branches(ui, repo, active=False, closed=False, **opts):
1147 def branches(ui, repo, active=False, closed=False, **opts):
1148 """list repository named branches
1148 """list repository named branches
1149
1149
1150 List the repository's named branches, indicating which ones are
1150 List the repository's named branches, indicating which ones are
1151 inactive. If -c/--closed is specified, also list branches which have
1151 inactive. If -c/--closed is specified, also list branches which have
1152 been marked closed (see :hg:`commit --close-branch`).
1152 been marked closed (see :hg:`commit --close-branch`).
1153
1153
1154 Use the command :hg:`update` to switch to an existing branch.
1154 Use the command :hg:`update` to switch to an existing branch.
1155
1155
1156 Returns 0.
1156 Returns 0.
1157 """
1157 """
1158
1158
1159 fm = ui.formatter('branches', opts)
1159 fm = ui.formatter('branches', opts)
1160 hexfunc = fm.hexfunc
1160 hexfunc = fm.hexfunc
1161
1161
1162 allheads = set(repo.heads())
1162 allheads = set(repo.heads())
1163 branches = []
1163 branches = []
1164 for tag, heads, tip, isclosed in repo.branchmap().iterbranches():
1164 for tag, heads, tip, isclosed in repo.branchmap().iterbranches():
1165 isactive = not isclosed and bool(set(heads) & allheads)
1165 isactive = not isclosed and bool(set(heads) & allheads)
1166 branches.append((tag, repo[tip], isactive, not isclosed))
1166 branches.append((tag, repo[tip], isactive, not isclosed))
1167 branches.sort(key=lambda i: (i[2], i[1].rev(), i[0], i[3]),
1167 branches.sort(key=lambda i: (i[2], i[1].rev(), i[0], i[3]),
1168 reverse=True)
1168 reverse=True)
1169
1169
1170 for tag, ctx, isactive, isopen in branches:
1170 for tag, ctx, isactive, isopen in branches:
1171 if active and not isactive:
1171 if active and not isactive:
1172 continue
1172 continue
1173 if isactive:
1173 if isactive:
1174 label = 'branches.active'
1174 label = 'branches.active'
1175 notice = ''
1175 notice = ''
1176 elif not isopen:
1176 elif not isopen:
1177 if not closed:
1177 if not closed:
1178 continue
1178 continue
1179 label = 'branches.closed'
1179 label = 'branches.closed'
1180 notice = _(' (closed)')
1180 notice = _(' (closed)')
1181 else:
1181 else:
1182 label = 'branches.inactive'
1182 label = 'branches.inactive'
1183 notice = _(' (inactive)')
1183 notice = _(' (inactive)')
1184 current = (tag == repo.dirstate.branch())
1184 current = (tag == repo.dirstate.branch())
1185 if current:
1185 if current:
1186 label = 'branches.current'
1186 label = 'branches.current'
1187
1187
1188 fm.startitem()
1188 fm.startitem()
1189 fm.write('branch', '%s', tag, label=label)
1189 fm.write('branch', '%s', tag, label=label)
1190 rev = ctx.rev()
1190 rev = ctx.rev()
1191 padsize = max(31 - len(str(rev)) - encoding.colwidth(tag), 0)
1191 padsize = max(31 - len(str(rev)) - encoding.colwidth(tag), 0)
1192 fmt = ' ' * padsize + ' %d:%s'
1192 fmt = ' ' * padsize + ' %d:%s'
1193 fm.condwrite(not ui.quiet, 'rev node', fmt, rev, hexfunc(ctx.node()),
1193 fm.condwrite(not ui.quiet, 'rev node', fmt, rev, hexfunc(ctx.node()),
1194 label='log.changeset changeset.%s' % ctx.phasestr())
1194 label='log.changeset changeset.%s' % ctx.phasestr())
1195 fm.data(active=isactive, closed=not isopen, current=current)
1195 fm.data(active=isactive, closed=not isopen, current=current)
1196 if not ui.quiet:
1196 if not ui.quiet:
1197 fm.plain(notice)
1197 fm.plain(notice)
1198 fm.plain('\n')
1198 fm.plain('\n')
1199 fm.end()
1199 fm.end()
1200
1200
1201 @command('bundle',
1201 @command('bundle',
1202 [('f', 'force', None, _('run even when the destination is unrelated')),
1202 [('f', 'force', None, _('run even when the destination is unrelated')),
1203 ('r', 'rev', [], _('a changeset intended to be added to the destination'),
1203 ('r', 'rev', [], _('a changeset intended to be added to the destination'),
1204 _('REV')),
1204 _('REV')),
1205 ('b', 'branch', [], _('a specific branch you would like to bundle'),
1205 ('b', 'branch', [], _('a specific branch you would like to bundle'),
1206 _('BRANCH')),
1206 _('BRANCH')),
1207 ('', 'base', [],
1207 ('', 'base', [],
1208 _('a base changeset assumed to be available at the destination'),
1208 _('a base changeset assumed to be available at the destination'),
1209 _('REV')),
1209 _('REV')),
1210 ('a', 'all', None, _('bundle all changesets in the repository')),
1210 ('a', 'all', None, _('bundle all changesets in the repository')),
1211 ('t', 'type', 'bzip2', _('bundle compression type to use'), _('TYPE')),
1211 ('t', 'type', 'bzip2', _('bundle compression type to use'), _('TYPE')),
1212 ] + remoteopts,
1212 ] + remoteopts,
1213 _('[-f] [-t TYPE] [-a] [-r REV]... [--base REV]... FILE [DEST]'))
1213 _('[-f] [-t TYPE] [-a] [-r REV]... [--base REV]... FILE [DEST]'))
1214 def bundle(ui, repo, fname, dest=None, **opts):
1214 def bundle(ui, repo, fname, dest=None, **opts):
1215 """create a changegroup file
1215 """create a changegroup file
1216
1216
1217 Generate a compressed changegroup file collecting changesets not
1217 Generate a compressed changegroup file collecting changesets not
1218 known to be in another repository.
1218 known to be in another repository.
1219
1219
1220 If you omit the destination repository, then hg assumes the
1220 If you omit the destination repository, then hg assumes the
1221 destination will have all the nodes you specify with --base
1221 destination will have all the nodes you specify with --base
1222 parameters. To create a bundle containing all changesets, use
1222 parameters. To create a bundle containing all changesets, use
1223 -a/--all (or --base null).
1223 -a/--all (or --base null).
1224
1224
1225 You can change bundle format with the -t/--type option. You can
1225 You can change bundle format with the -t/--type option. You can
1226 specify a compression, a bundle version or both using a dash
1226 specify a compression, a bundle version or both using a dash
1227 (comp-version). The available compression methods are: none, bzip2,
1227 (comp-version). The available compression methods are: none, bzip2,
1228 and gzip (by default, bundles are compressed using bzip2). The
1228 and gzip (by default, bundles are compressed using bzip2). The
1229 available format are: v1, v2 (default to most suitable).
1229 available format are: v1, v2 (default to most suitable).
1230
1230
1231 The bundle file can then be transferred using conventional means
1231 The bundle file can then be transferred using conventional means
1232 and applied to another repository with the unbundle or pull
1232 and applied to another repository with the unbundle or pull
1233 command. This is useful when direct push and pull are not
1233 command. This is useful when direct push and pull are not
1234 available or when exporting an entire repository is undesirable.
1234 available or when exporting an entire repository is undesirable.
1235
1235
1236 Applying bundles preserves all changeset contents including
1236 Applying bundles preserves all changeset contents including
1237 permissions, copy/rename information, and revision history.
1237 permissions, copy/rename information, and revision history.
1238
1238
1239 Returns 0 on success, 1 if no changes found.
1239 Returns 0 on success, 1 if no changes found.
1240 """
1240 """
1241 revs = None
1241 revs = None
1242 if 'rev' in opts:
1242 if 'rev' in opts:
1243 revs = scmutil.revrange(repo, opts['rev'])
1243 revs = scmutil.revrange(repo, opts['rev'])
1244
1244
1245 bundletype = opts.get('type', 'bzip2').lower()
1245 bundletype = opts.get('type', 'bzip2').lower()
1246 try:
1246 try:
1247 bcompression, cgversion, params = exchange.parsebundlespec(
1247 bcompression, cgversion, params = exchange.parsebundlespec(
1248 repo, bundletype, strict=False)
1248 repo, bundletype, strict=False)
1249 except error.UnsupportedBundleSpecification as e:
1249 except error.UnsupportedBundleSpecification as e:
1250 raise error.Abort(str(e),
1250 raise error.Abort(str(e),
1251 hint=_('see "hg help bundle" for supported '
1251 hint=_('see "hg help bundle" for supported '
1252 'values for --type'))
1252 'values for --type'))
1253
1253
1254 # Packed bundles are a pseudo bundle format for now.
1254 # Packed bundles are a pseudo bundle format for now.
1255 if cgversion == 's1':
1255 if cgversion == 's1':
1256 raise error.Abort(_('packed bundles cannot be produced by "hg bundle"'),
1256 raise error.Abort(_('packed bundles cannot be produced by "hg bundle"'),
1257 hint=_('use "hg debugcreatestreamclonebundle"'))
1257 hint=_('use "hg debugcreatestreamclonebundle"'))
1258
1258
1259 if opts.get('all'):
1259 if opts.get('all'):
1260 base = ['null']
1260 base = ['null']
1261 else:
1261 else:
1262 base = scmutil.revrange(repo, opts.get('base'))
1262 base = scmutil.revrange(repo, opts.get('base'))
1263 # TODO: get desired bundlecaps from command line.
1263 # TODO: get desired bundlecaps from command line.
1264 bundlecaps = None
1264 bundlecaps = None
1265 if base:
1265 if base:
1266 if dest:
1266 if dest:
1267 raise error.Abort(_("--base is incompatible with specifying "
1267 raise error.Abort(_("--base is incompatible with specifying "
1268 "a destination"))
1268 "a destination"))
1269 common = [repo.lookup(rev) for rev in base]
1269 common = [repo.lookup(rev) for rev in base]
1270 heads = revs and map(repo.lookup, revs) or revs
1270 heads = revs and map(repo.lookup, revs) or revs
1271 cg = changegroup.getchangegroup(repo, 'bundle', heads=heads,
1271 cg = changegroup.getchangegroup(repo, 'bundle', heads=heads,
1272 common=common, bundlecaps=bundlecaps,
1272 common=common, bundlecaps=bundlecaps,
1273 version=cgversion)
1273 version=cgversion)
1274 outgoing = None
1274 outgoing = None
1275 else:
1275 else:
1276 dest = ui.expandpath(dest or 'default-push', dest or 'default')
1276 dest = ui.expandpath(dest or 'default-push', dest or 'default')
1277 dest, branches = hg.parseurl(dest, opts.get('branch'))
1277 dest, branches = hg.parseurl(dest, opts.get('branch'))
1278 other = hg.peer(repo, opts, dest)
1278 other = hg.peer(repo, opts, dest)
1279 revs, checkout = hg.addbranchrevs(repo, repo, branches, revs)
1279 revs, checkout = hg.addbranchrevs(repo, repo, branches, revs)
1280 heads = revs and map(repo.lookup, revs) or revs
1280 heads = revs and map(repo.lookup, revs) or revs
1281 outgoing = discovery.findcommonoutgoing(repo, other,
1281 outgoing = discovery.findcommonoutgoing(repo, other,
1282 onlyheads=heads,
1282 onlyheads=heads,
1283 force=opts.get('force'),
1283 force=opts.get('force'),
1284 portable=True)
1284 portable=True)
1285 cg = changegroup.getlocalchangegroup(repo, 'bundle', outgoing,
1285 cg = changegroup.getlocalchangegroup(repo, 'bundle', outgoing,
1286 bundlecaps, version=cgversion)
1286 bundlecaps, version=cgversion)
1287 if not cg:
1287 if not cg:
1288 scmutil.nochangesfound(ui, repo, outgoing and outgoing.excluded)
1288 scmutil.nochangesfound(ui, repo, outgoing and outgoing.excluded)
1289 return 1
1289 return 1
1290
1290
1291 if cgversion == '01': #bundle1
1291 if cgversion == '01': #bundle1
1292 if bcompression is None:
1292 if bcompression is None:
1293 bcompression = 'UN'
1293 bcompression = 'UN'
1294 bversion = 'HG10' + bcompression
1294 bversion = 'HG10' + bcompression
1295 bcompression = None
1295 bcompression = None
1296 else:
1296 else:
1297 assert cgversion == '02'
1297 assert cgversion == '02'
1298 bversion = 'HG20'
1298 bversion = 'HG20'
1299
1299
1300
1300
1301 changegroup.writebundle(ui, cg, fname, bversion, compression=bcompression)
1301 changegroup.writebundle(ui, cg, fname, bversion, compression=bcompression)
1302
1302
1303 @command('cat',
1303 @command('cat',
1304 [('o', 'output', '',
1304 [('o', 'output', '',
1305 _('print output to file with formatted name'), _('FORMAT')),
1305 _('print output to file with formatted name'), _('FORMAT')),
1306 ('r', 'rev', '', _('print the given revision'), _('REV')),
1306 ('r', 'rev', '', _('print the given revision'), _('REV')),
1307 ('', 'decode', None, _('apply any matching decode filter')),
1307 ('', 'decode', None, _('apply any matching decode filter')),
1308 ] + walkopts,
1308 ] + walkopts,
1309 _('[OPTION]... FILE...'),
1309 _('[OPTION]... FILE...'),
1310 inferrepo=True)
1310 inferrepo=True)
1311 def cat(ui, repo, file1, *pats, **opts):
1311 def cat(ui, repo, file1, *pats, **opts):
1312 """output the current or given revision of files
1312 """output the current or given revision of files
1313
1313
1314 Print the specified files as they were at the given revision. If
1314 Print the specified files as they were at the given revision. If
1315 no revision is given, the parent of the working directory is used.
1315 no revision is given, the parent of the working directory is used.
1316
1316
1317 Output may be to a file, in which case the name of the file is
1317 Output may be to a file, in which case the name of the file is
1318 given using a format string. The formatting rules as follows:
1318 given using a format string. The formatting rules as follows:
1319
1319
1320 :``%%``: literal "%" character
1320 :``%%``: literal "%" character
1321 :``%s``: basename of file being printed
1321 :``%s``: basename of file being printed
1322 :``%d``: dirname of file being printed, or '.' if in repository root
1322 :``%d``: dirname of file being printed, or '.' if in repository root
1323 :``%p``: root-relative path name of file being printed
1323 :``%p``: root-relative path name of file being printed
1324 :``%H``: changeset hash (40 hexadecimal digits)
1324 :``%H``: changeset hash (40 hexadecimal digits)
1325 :``%R``: changeset revision number
1325 :``%R``: changeset revision number
1326 :``%h``: short-form changeset hash (12 hexadecimal digits)
1326 :``%h``: short-form changeset hash (12 hexadecimal digits)
1327 :``%r``: zero-padded changeset revision number
1327 :``%r``: zero-padded changeset revision number
1328 :``%b``: basename of the exporting repository
1328 :``%b``: basename of the exporting repository
1329
1329
1330 Returns 0 on success.
1330 Returns 0 on success.
1331 """
1331 """
1332 ctx = scmutil.revsingle(repo, opts.get('rev'))
1332 ctx = scmutil.revsingle(repo, opts.get('rev'))
1333 m = scmutil.match(ctx, (file1,) + pats, opts)
1333 m = scmutil.match(ctx, (file1,) + pats, opts)
1334
1334
1335 return cmdutil.cat(ui, repo, ctx, m, '', **opts)
1335 return cmdutil.cat(ui, repo, ctx, m, '', **opts)
1336
1336
1337 @command('^clone',
1337 @command('^clone',
1338 [('U', 'noupdate', None, _('the clone will include an empty working '
1338 [('U', 'noupdate', None, _('the clone will include an empty working '
1339 'directory (only a repository)')),
1339 'directory (only a repository)')),
1340 ('u', 'updaterev', '', _('revision, tag or branch to check out'), _('REV')),
1340 ('u', 'updaterev', '', _('revision, tag or branch to check out'), _('REV')),
1341 ('r', 'rev', [], _('include the specified changeset'), _('REV')),
1341 ('r', 'rev', [], _('include the specified changeset'), _('REV')),
1342 ('b', 'branch', [], _('clone only the specified branch'), _('BRANCH')),
1342 ('b', 'branch', [], _('clone only the specified branch'), _('BRANCH')),
1343 ('', 'pull', None, _('use pull protocol to copy metadata')),
1343 ('', 'pull', None, _('use pull protocol to copy metadata')),
1344 ('', 'uncompressed', None, _('use uncompressed transfer (fast over LAN)')),
1344 ('', 'uncompressed', None, _('use uncompressed transfer (fast over LAN)')),
1345 ] + remoteopts,
1345 ] + remoteopts,
1346 _('[OPTION]... SOURCE [DEST]'),
1346 _('[OPTION]... SOURCE [DEST]'),
1347 norepo=True)
1347 norepo=True)
1348 def clone(ui, source, dest=None, **opts):
1348 def clone(ui, source, dest=None, **opts):
1349 """make a copy of an existing repository
1349 """make a copy of an existing repository
1350
1350
1351 Create a copy of an existing repository in a new directory.
1351 Create a copy of an existing repository in a new directory.
1352
1352
1353 If no destination directory name is specified, it defaults to the
1353 If no destination directory name is specified, it defaults to the
1354 basename of the source.
1354 basename of the source.
1355
1355
1356 The location of the source is added to the new repository's
1356 The location of the source is added to the new repository's
1357 ``.hg/hgrc`` file, as the default to be used for future pulls.
1357 ``.hg/hgrc`` file, as the default to be used for future pulls.
1358
1358
1359 Only local paths and ``ssh://`` URLs are supported as
1359 Only local paths and ``ssh://`` URLs are supported as
1360 destinations. For ``ssh://`` destinations, no working directory or
1360 destinations. For ``ssh://`` destinations, no working directory or
1361 ``.hg/hgrc`` will be created on the remote side.
1361 ``.hg/hgrc`` will be created on the remote side.
1362
1362
1363 To pull only a subset of changesets, specify one or more revisions
1363 To pull only a subset of changesets, specify one or more revisions
1364 identifiers with -r/--rev or branches with -b/--branch. The
1364 identifiers with -r/--rev or branches with -b/--branch. The
1365 resulting clone will contain only the specified changesets and
1365 resulting clone will contain only the specified changesets and
1366 their ancestors. These options (or 'clone src#rev dest') imply
1366 their ancestors. These options (or 'clone src#rev dest') imply
1367 --pull, even for local source repositories. Note that specifying a
1367 --pull, even for local source repositories. Note that specifying a
1368 tag will include the tagged changeset but not the changeset
1368 tag will include the tagged changeset but not the changeset
1369 containing the tag.
1369 containing the tag.
1370
1370
1371 If the source repository has a bookmark called '@' set, that
1371 If the source repository has a bookmark called '@' set, that
1372 revision will be checked out in the new repository by default.
1372 revision will be checked out in the new repository by default.
1373
1373
1374 To check out a particular version, use -u/--update, or
1374 To check out a particular version, use -u/--update, or
1375 -U/--noupdate to create a clone with no working directory.
1375 -U/--noupdate to create a clone with no working directory.
1376
1376
1377 .. container:: verbose
1377 .. container:: verbose
1378
1378
1379 For efficiency, hardlinks are used for cloning whenever the
1379 For efficiency, hardlinks are used for cloning whenever the
1380 source and destination are on the same filesystem (note this
1380 source and destination are on the same filesystem (note this
1381 applies only to the repository data, not to the working
1381 applies only to the repository data, not to the working
1382 directory). Some filesystems, such as AFS, implement hardlinking
1382 directory). Some filesystems, such as AFS, implement hardlinking
1383 incorrectly, but do not report errors. In these cases, use the
1383 incorrectly, but do not report errors. In these cases, use the
1384 --pull option to avoid hardlinking.
1384 --pull option to avoid hardlinking.
1385
1385
1386 In some cases, you can clone repositories and the working
1386 In some cases, you can clone repositories and the working
1387 directory using full hardlinks with ::
1387 directory using full hardlinks with ::
1388
1388
1389 $ cp -al REPO REPOCLONE
1389 $ cp -al REPO REPOCLONE
1390
1390
1391 This is the fastest way to clone, but it is not always safe. The
1391 This is the fastest way to clone, but it is not always safe. The
1392 operation is not atomic (making sure REPO is not modified during
1392 operation is not atomic (making sure REPO is not modified during
1393 the operation is up to you) and you have to make sure your
1393 the operation is up to you) and you have to make sure your
1394 editor breaks hardlinks (Emacs and most Linux Kernel tools do
1394 editor breaks hardlinks (Emacs and most Linux Kernel tools do
1395 so). Also, this is not compatible with certain extensions that
1395 so). Also, this is not compatible with certain extensions that
1396 place their metadata under the .hg directory, such as mq.
1396 place their metadata under the .hg directory, such as mq.
1397
1397
1398 Mercurial will update the working directory to the first applicable
1398 Mercurial will update the working directory to the first applicable
1399 revision from this list:
1399 revision from this list:
1400
1400
1401 a) null if -U or the source repository has no changesets
1401 a) null if -U or the source repository has no changesets
1402 b) if -u . and the source repository is local, the first parent of
1402 b) if -u . and the source repository is local, the first parent of
1403 the source repository's working directory
1403 the source repository's working directory
1404 c) the changeset specified with -u (if a branch name, this means the
1404 c) the changeset specified with -u (if a branch name, this means the
1405 latest head of that branch)
1405 latest head of that branch)
1406 d) the changeset specified with -r
1406 d) the changeset specified with -r
1407 e) the tipmost head specified with -b
1407 e) the tipmost head specified with -b
1408 f) the tipmost head specified with the url#branch source syntax
1408 f) the tipmost head specified with the url#branch source syntax
1409 g) the revision marked with the '@' bookmark, if present
1409 g) the revision marked with the '@' bookmark, if present
1410 h) the tipmost head of the default branch
1410 h) the tipmost head of the default branch
1411 i) tip
1411 i) tip
1412
1412
1413 Examples:
1413 Examples:
1414
1414
1415 - clone a remote repository to a new directory named hg/::
1415 - clone a remote repository to a new directory named hg/::
1416
1416
1417 hg clone http://selenic.com/hg
1417 hg clone http://selenic.com/hg
1418
1418
1419 - create a lightweight local clone::
1419 - create a lightweight local clone::
1420
1420
1421 hg clone project/ project-feature/
1421 hg clone project/ project-feature/
1422
1422
1423 - clone from an absolute path on an ssh server (note double-slash)::
1423 - clone from an absolute path on an ssh server (note double-slash)::
1424
1424
1425 hg clone ssh://user@server//home/projects/alpha/
1425 hg clone ssh://user@server//home/projects/alpha/
1426
1426
1427 - do a high-speed clone over a LAN while checking out a
1427 - do a high-speed clone over a LAN while checking out a
1428 specified version::
1428 specified version::
1429
1429
1430 hg clone --uncompressed http://server/repo -u 1.5
1430 hg clone --uncompressed http://server/repo -u 1.5
1431
1431
1432 - create a repository without changesets after a particular revision::
1432 - create a repository without changesets after a particular revision::
1433
1433
1434 hg clone -r 04e544 experimental/ good/
1434 hg clone -r 04e544 experimental/ good/
1435
1435
1436 - clone (and track) a particular named branch::
1436 - clone (and track) a particular named branch::
1437
1437
1438 hg clone http://selenic.com/hg#stable
1438 hg clone http://selenic.com/hg#stable
1439
1439
1440 See :hg:`help urls` for details on specifying URLs.
1440 See :hg:`help urls` for details on specifying URLs.
1441
1441
1442 Returns 0 on success.
1442 Returns 0 on success.
1443 """
1443 """
1444 if opts.get('noupdate') and opts.get('updaterev'):
1444 if opts.get('noupdate') and opts.get('updaterev'):
1445 raise error.Abort(_("cannot specify both --noupdate and --updaterev"))
1445 raise error.Abort(_("cannot specify both --noupdate and --updaterev"))
1446
1446
1447 r = hg.clone(ui, opts, source, dest,
1447 r = hg.clone(ui, opts, source, dest,
1448 pull=opts.get('pull'),
1448 pull=opts.get('pull'),
1449 stream=opts.get('uncompressed'),
1449 stream=opts.get('uncompressed'),
1450 rev=opts.get('rev'),
1450 rev=opts.get('rev'),
1451 update=opts.get('updaterev') or not opts.get('noupdate'),
1451 update=opts.get('updaterev') or not opts.get('noupdate'),
1452 branch=opts.get('branch'),
1452 branch=opts.get('branch'),
1453 shareopts=opts.get('shareopts'))
1453 shareopts=opts.get('shareopts'))
1454
1454
1455 return r is None
1455 return r is None
1456
1456
1457 @command('^commit|ci',
1457 @command('^commit|ci',
1458 [('A', 'addremove', None,
1458 [('A', 'addremove', None,
1459 _('mark new/missing files as added/removed before committing')),
1459 _('mark new/missing files as added/removed before committing')),
1460 ('', 'close-branch', None,
1460 ('', 'close-branch', None,
1461 _('mark a branch head as closed')),
1461 _('mark a branch head as closed')),
1462 ('', 'amend', None, _('amend the parent of the working directory')),
1462 ('', 'amend', None, _('amend the parent of the working directory')),
1463 ('s', 'secret', None, _('use the secret phase for committing')),
1463 ('s', 'secret', None, _('use the secret phase for committing')),
1464 ('e', 'edit', None, _('invoke editor on commit messages')),
1464 ('e', 'edit', None, _('invoke editor on commit messages')),
1465 ('i', 'interactive', None, _('use interactive mode')),
1465 ('i', 'interactive', None, _('use interactive mode')),
1466 ] + walkopts + commitopts + commitopts2 + subrepoopts,
1466 ] + walkopts + commitopts + commitopts2 + subrepoopts,
1467 _('[OPTION]... [FILE]...'),
1467 _('[OPTION]... [FILE]...'),
1468 inferrepo=True)
1468 inferrepo=True)
1469 def commit(ui, repo, *pats, **opts):
1469 def commit(ui, repo, *pats, **opts):
1470 """commit the specified files or all outstanding changes
1470 """commit the specified files or all outstanding changes
1471
1471
1472 Commit changes to the given files into the repository. Unlike a
1472 Commit changes to the given files into the repository. Unlike a
1473 centralized SCM, this operation is a local operation. See
1473 centralized SCM, this operation is a local operation. See
1474 :hg:`push` for a way to actively distribute your changes.
1474 :hg:`push` for a way to actively distribute your changes.
1475
1475
1476 If a list of files is omitted, all changes reported by :hg:`status`
1476 If a list of files is omitted, all changes reported by :hg:`status`
1477 will be committed.
1477 will be committed.
1478
1478
1479 If you are committing the result of a merge, do not provide any
1479 If you are committing the result of a merge, do not provide any
1480 filenames or -I/-X filters.
1480 filenames or -I/-X filters.
1481
1481
1482 If no commit message is specified, Mercurial starts your
1482 If no commit message is specified, Mercurial starts your
1483 configured editor where you can enter a message. In case your
1483 configured editor where you can enter a message. In case your
1484 commit fails, you will find a backup of your message in
1484 commit fails, you will find a backup of your message in
1485 ``.hg/last-message.txt``.
1485 ``.hg/last-message.txt``.
1486
1486
1487 The --close-branch flag can be used to mark the current branch
1487 The --close-branch flag can be used to mark the current branch
1488 head closed. When all heads of a branch are closed, the branch
1488 head closed. When all heads of a branch are closed, the branch
1489 will be considered closed and no longer listed.
1489 will be considered closed and no longer listed.
1490
1490
1491 The --amend flag can be used to amend the parent of the
1491 The --amend flag can be used to amend the parent of the
1492 working directory with a new commit that contains the changes
1492 working directory with a new commit that contains the changes
1493 in the parent in addition to those currently reported by :hg:`status`,
1493 in the parent in addition to those currently reported by :hg:`status`,
1494 if there are any. The old commit is stored in a backup bundle in
1494 if there are any. The old commit is stored in a backup bundle in
1495 ``.hg/strip-backup`` (see :hg:`help bundle` and :hg:`help unbundle`
1495 ``.hg/strip-backup`` (see :hg:`help bundle` and :hg:`help unbundle`
1496 on how to restore it).
1496 on how to restore it).
1497
1497
1498 Message, user and date are taken from the amended commit unless
1498 Message, user and date are taken from the amended commit unless
1499 specified. When a message isn't specified on the command line,
1499 specified. When a message isn't specified on the command line,
1500 the editor will open with the message of the amended commit.
1500 the editor will open with the message of the amended commit.
1501
1501
1502 It is not possible to amend public changesets (see :hg:`help phases`)
1502 It is not possible to amend public changesets (see :hg:`help phases`)
1503 or changesets that have children.
1503 or changesets that have children.
1504
1504
1505 See :hg:`help dates` for a list of formats valid for -d/--date.
1505 See :hg:`help dates` for a list of formats valid for -d/--date.
1506
1506
1507 Returns 0 on success, 1 if nothing changed.
1507 Returns 0 on success, 1 if nothing changed.
1508 """
1508 """
1509 if opts.get('interactive'):
1509 if opts.get('interactive'):
1510 opts.pop('interactive')
1510 opts.pop('interactive')
1511 cmdutil.dorecord(ui, repo, commit, None, False,
1511 cmdutil.dorecord(ui, repo, commit, None, False,
1512 cmdutil.recordfilter, *pats, **opts)
1512 cmdutil.recordfilter, *pats, **opts)
1513 return
1513 return
1514
1514
1515 if opts.get('subrepos'):
1515 if opts.get('subrepos'):
1516 if opts.get('amend'):
1516 if opts.get('amend'):
1517 raise error.Abort(_('cannot amend with --subrepos'))
1517 raise error.Abort(_('cannot amend with --subrepos'))
1518 # Let --subrepos on the command line override config setting.
1518 # Let --subrepos on the command line override config setting.
1519 ui.setconfig('ui', 'commitsubrepos', True, 'commit')
1519 ui.setconfig('ui', 'commitsubrepos', True, 'commit')
1520
1520
1521 cmdutil.checkunfinished(repo, commit=True)
1521 cmdutil.checkunfinished(repo, commit=True)
1522
1522
1523 branch = repo[None].branch()
1523 branch = repo[None].branch()
1524 bheads = repo.branchheads(branch)
1524 bheads = repo.branchheads(branch)
1525
1525
1526 extra = {}
1526 extra = {}
1527 if opts.get('close_branch'):
1527 if opts.get('close_branch'):
1528 extra['close'] = 1
1528 extra['close'] = 1
1529
1529
1530 if not bheads:
1530 if not bheads:
1531 raise error.Abort(_('can only close branch heads'))
1531 raise error.Abort(_('can only close branch heads'))
1532 elif opts.get('amend'):
1532 elif opts.get('amend'):
1533 if repo.parents()[0].p1().branch() != branch and \
1533 if repo.parents()[0].p1().branch() != branch and \
1534 repo.parents()[0].p2().branch() != branch:
1534 repo.parents()[0].p2().branch() != branch:
1535 raise error.Abort(_('can only close branch heads'))
1535 raise error.Abort(_('can only close branch heads'))
1536
1536
1537 if opts.get('amend'):
1537 if opts.get('amend'):
1538 if ui.configbool('ui', 'commitsubrepos'):
1538 if ui.configbool('ui', 'commitsubrepos'):
1539 raise error.Abort(_('cannot amend with ui.commitsubrepos enabled'))
1539 raise error.Abort(_('cannot amend with ui.commitsubrepos enabled'))
1540
1540
1541 old = repo['.']
1541 old = repo['.']
1542 if not old.mutable():
1542 if not old.mutable():
1543 raise error.Abort(_('cannot amend public changesets'))
1543 raise error.Abort(_('cannot amend public changesets'))
1544 if len(repo[None].parents()) > 1:
1544 if len(repo[None].parents()) > 1:
1545 raise error.Abort(_('cannot amend while merging'))
1545 raise error.Abort(_('cannot amend while merging'))
1546 allowunstable = obsolete.isenabled(repo, obsolete.allowunstableopt)
1546 allowunstable = obsolete.isenabled(repo, obsolete.allowunstableopt)
1547 if not allowunstable and old.children():
1547 if not allowunstable and old.children():
1548 raise error.Abort(_('cannot amend changeset with children'))
1548 raise error.Abort(_('cannot amend changeset with children'))
1549
1549
1550 # commitfunc is used only for temporary amend commit by cmdutil.amend
1550 # commitfunc is used only for temporary amend commit by cmdutil.amend
1551 def commitfunc(ui, repo, message, match, opts):
1551 def commitfunc(ui, repo, message, match, opts):
1552 return repo.commit(message,
1552 return repo.commit(message,
1553 opts.get('user') or old.user(),
1553 opts.get('user') or old.user(),
1554 opts.get('date') or old.date(),
1554 opts.get('date') or old.date(),
1555 match,
1555 match,
1556 extra=extra)
1556 extra=extra)
1557
1557
1558 node = cmdutil.amend(ui, repo, commitfunc, old, extra, pats, opts)
1558 node = cmdutil.amend(ui, repo, commitfunc, old, extra, pats, opts)
1559 if node == old.node():
1559 if node == old.node():
1560 ui.status(_("nothing changed\n"))
1560 ui.status(_("nothing changed\n"))
1561 return 1
1561 return 1
1562 else:
1562 else:
1563 def commitfunc(ui, repo, message, match, opts):
1563 def commitfunc(ui, repo, message, match, opts):
1564 backup = ui.backupconfig('phases', 'new-commit')
1564 backup = ui.backupconfig('phases', 'new-commit')
1565 baseui = repo.baseui
1565 baseui = repo.baseui
1566 basebackup = baseui.backupconfig('phases', 'new-commit')
1566 basebackup = baseui.backupconfig('phases', 'new-commit')
1567 try:
1567 try:
1568 if opts.get('secret'):
1568 if opts.get('secret'):
1569 ui.setconfig('phases', 'new-commit', 'secret', 'commit')
1569 ui.setconfig('phases', 'new-commit', 'secret', 'commit')
1570 # Propagate to subrepos
1570 # Propagate to subrepos
1571 baseui.setconfig('phases', 'new-commit', 'secret', 'commit')
1571 baseui.setconfig('phases', 'new-commit', 'secret', 'commit')
1572
1572
1573 editform = cmdutil.mergeeditform(repo[None], 'commit.normal')
1573 editform = cmdutil.mergeeditform(repo[None], 'commit.normal')
1574 editor = cmdutil.getcommiteditor(editform=editform, **opts)
1574 editor = cmdutil.getcommiteditor(editform=editform, **opts)
1575 return repo.commit(message, opts.get('user'), opts.get('date'),
1575 return repo.commit(message, opts.get('user'), opts.get('date'),
1576 match,
1576 match,
1577 editor=editor,
1577 editor=editor,
1578 extra=extra)
1578 extra=extra)
1579 finally:
1579 finally:
1580 ui.restoreconfig(backup)
1580 ui.restoreconfig(backup)
1581 repo.baseui.restoreconfig(basebackup)
1581 repo.baseui.restoreconfig(basebackup)
1582
1582
1583
1583
1584 node = cmdutil.commit(ui, repo, commitfunc, pats, opts)
1584 node = cmdutil.commit(ui, repo, commitfunc, pats, opts)
1585
1585
1586 if not node:
1586 if not node:
1587 stat = repo.status(match=scmutil.match(repo[None], pats, opts))
1587 stat = repo.status(match=scmutil.match(repo[None], pats, opts))
1588 if stat[3]:
1588 if stat[3]:
1589 ui.status(_("nothing changed (%d missing files, see "
1589 ui.status(_("nothing changed (%d missing files, see "
1590 "'hg status')\n") % len(stat[3]))
1590 "'hg status')\n") % len(stat[3]))
1591 else:
1591 else:
1592 ui.status(_("nothing changed\n"))
1592 ui.status(_("nothing changed\n"))
1593 return 1
1593 return 1
1594
1594
1595 cmdutil.commitstatus(repo, node, branch, bheads, opts)
1595 cmdutil.commitstatus(repo, node, branch, bheads, opts)
1596
1596
1597 @command('config|showconfig|debugconfig',
1597 @command('config|showconfig|debugconfig',
1598 [('u', 'untrusted', None, _('show untrusted configuration options')),
1598 [('u', 'untrusted', None, _('show untrusted configuration options')),
1599 ('e', 'edit', None, _('edit user config')),
1599 ('e', 'edit', None, _('edit user config')),
1600 ('l', 'local', None, _('edit repository config')),
1600 ('l', 'local', None, _('edit repository config')),
1601 ('g', 'global', None, _('edit global config'))],
1601 ('g', 'global', None, _('edit global config'))],
1602 _('[-u] [NAME]...'),
1602 _('[-u] [NAME]...'),
1603 optionalrepo=True)
1603 optionalrepo=True)
1604 def config(ui, repo, *values, **opts):
1604 def config(ui, repo, *values, **opts):
1605 """show combined config settings from all hgrc files
1605 """show combined config settings from all hgrc files
1606
1606
1607 With no arguments, print names and values of all config items.
1607 With no arguments, print names and values of all config items.
1608
1608
1609 With one argument of the form section.name, print just the value
1609 With one argument of the form section.name, print just the value
1610 of that config item.
1610 of that config item.
1611
1611
1612 With multiple arguments, print names and values of all config
1612 With multiple arguments, print names and values of all config
1613 items with matching section names.
1613 items with matching section names.
1614
1614
1615 With --edit, start an editor on the user-level config file. With
1615 With --edit, start an editor on the user-level config file. With
1616 --global, edit the system-wide config file. With --local, edit the
1616 --global, edit the system-wide config file. With --local, edit the
1617 repository-level config file.
1617 repository-level config file.
1618
1618
1619 With --debug, the source (filename and line number) is printed
1619 With --debug, the source (filename and line number) is printed
1620 for each config item.
1620 for each config item.
1621
1621
1622 See :hg:`help config` for more information about config files.
1622 See :hg:`help config` for more information about config files.
1623
1623
1624 Returns 0 on success, 1 if NAME does not exist.
1624 Returns 0 on success, 1 if NAME does not exist.
1625
1625
1626 """
1626 """
1627
1627
1628 if opts.get('edit') or opts.get('local') or opts.get('global'):
1628 if opts.get('edit') or opts.get('local') or opts.get('global'):
1629 if opts.get('local') and opts.get('global'):
1629 if opts.get('local') and opts.get('global'):
1630 raise error.Abort(_("can't use --local and --global together"))
1630 raise error.Abort(_("can't use --local and --global together"))
1631
1631
1632 if opts.get('local'):
1632 if opts.get('local'):
1633 if not repo:
1633 if not repo:
1634 raise error.Abort(_("can't use --local outside a repository"))
1634 raise error.Abort(_("can't use --local outside a repository"))
1635 paths = [repo.join('hgrc')]
1635 paths = [repo.join('hgrc')]
1636 elif opts.get('global'):
1636 elif opts.get('global'):
1637 paths = scmutil.systemrcpath()
1637 paths = scmutil.systemrcpath()
1638 else:
1638 else:
1639 paths = scmutil.userrcpath()
1639 paths = scmutil.userrcpath()
1640
1640
1641 for f in paths:
1641 for f in paths:
1642 if os.path.exists(f):
1642 if os.path.exists(f):
1643 break
1643 break
1644 else:
1644 else:
1645 if opts.get('global'):
1645 if opts.get('global'):
1646 samplehgrc = uimod.samplehgrcs['global']
1646 samplehgrc = uimod.samplehgrcs['global']
1647 elif opts.get('local'):
1647 elif opts.get('local'):
1648 samplehgrc = uimod.samplehgrcs['local']
1648 samplehgrc = uimod.samplehgrcs['local']
1649 else:
1649 else:
1650 samplehgrc = uimod.samplehgrcs['user']
1650 samplehgrc = uimod.samplehgrcs['user']
1651
1651
1652 f = paths[0]
1652 f = paths[0]
1653 fp = open(f, "w")
1653 fp = open(f, "w")
1654 fp.write(samplehgrc)
1654 fp.write(samplehgrc)
1655 fp.close()
1655 fp.close()
1656
1656
1657 editor = ui.geteditor()
1657 editor = ui.geteditor()
1658 ui.system("%s \"%s\"" % (editor, f),
1658 ui.system("%s \"%s\"" % (editor, f),
1659 onerr=error.Abort, errprefix=_("edit failed"))
1659 onerr=error.Abort, errprefix=_("edit failed"))
1660 return
1660 return
1661
1661
1662 for f in scmutil.rcpath():
1662 for f in scmutil.rcpath():
1663 ui.debug('read config from: %s\n' % f)
1663 ui.debug('read config from: %s\n' % f)
1664 untrusted = bool(opts.get('untrusted'))
1664 untrusted = bool(opts.get('untrusted'))
1665 if values:
1665 if values:
1666 sections = [v for v in values if '.' not in v]
1666 sections = [v for v in values if '.' not in v]
1667 items = [v for v in values if '.' in v]
1667 items = [v for v in values if '.' in v]
1668 if len(items) > 1 or items and sections:
1668 if len(items) > 1 or items and sections:
1669 raise error.Abort(_('only one config item permitted'))
1669 raise error.Abort(_('only one config item permitted'))
1670 matched = False
1670 matched = False
1671 for section, name, value in ui.walkconfig(untrusted=untrusted):
1671 for section, name, value in ui.walkconfig(untrusted=untrusted):
1672 value = str(value).replace('\n', '\\n')
1672 value = str(value).replace('\n', '\\n')
1673 sectname = section + '.' + name
1673 sectname = section + '.' + name
1674 if values:
1674 if values:
1675 for v in values:
1675 for v in values:
1676 if v == section:
1676 if v == section:
1677 ui.debug('%s: ' %
1677 ui.debug('%s: ' %
1678 ui.configsource(section, name, untrusted))
1678 ui.configsource(section, name, untrusted))
1679 ui.write('%s=%s\n' % (sectname, value))
1679 ui.write('%s=%s\n' % (sectname, value))
1680 matched = True
1680 matched = True
1681 elif v == sectname:
1681 elif v == sectname:
1682 ui.debug('%s: ' %
1682 ui.debug('%s: ' %
1683 ui.configsource(section, name, untrusted))
1683 ui.configsource(section, name, untrusted))
1684 ui.write(value, '\n')
1684 ui.write(value, '\n')
1685 matched = True
1685 matched = True
1686 else:
1686 else:
1687 ui.debug('%s: ' %
1687 ui.debug('%s: ' %
1688 ui.configsource(section, name, untrusted))
1688 ui.configsource(section, name, untrusted))
1689 ui.write('%s=%s\n' % (sectname, value))
1689 ui.write('%s=%s\n' % (sectname, value))
1690 matched = True
1690 matched = True
1691 if matched:
1691 if matched:
1692 return 0
1692 return 0
1693 return 1
1693 return 1
1694
1694
1695 @command('copy|cp',
1695 @command('copy|cp',
1696 [('A', 'after', None, _('record a copy that has already occurred')),
1696 [('A', 'after', None, _('record a copy that has already occurred')),
1697 ('f', 'force', None, _('forcibly copy over an existing managed file')),
1697 ('f', 'force', None, _('forcibly copy over an existing managed file')),
1698 ] + walkopts + dryrunopts,
1698 ] + walkopts + dryrunopts,
1699 _('[OPTION]... [SOURCE]... DEST'))
1699 _('[OPTION]... [SOURCE]... DEST'))
1700 def copy(ui, repo, *pats, **opts):
1700 def copy(ui, repo, *pats, **opts):
1701 """mark files as copied for the next commit
1701 """mark files as copied for the next commit
1702
1702
1703 Mark dest as having copies of source files. If dest is a
1703 Mark dest as having copies of source files. If dest is a
1704 directory, copies are put in that directory. If dest is a file,
1704 directory, copies are put in that directory. If dest is a file,
1705 the source must be a single file.
1705 the source must be a single file.
1706
1706
1707 By default, this command copies the contents of files as they
1707 By default, this command copies the contents of files as they
1708 exist in the working directory. If invoked with -A/--after, the
1708 exist in the working directory. If invoked with -A/--after, the
1709 operation is recorded, but no copying is performed.
1709 operation is recorded, but no copying is performed.
1710
1710
1711 This command takes effect with the next commit. To undo a copy
1711 This command takes effect with the next commit. To undo a copy
1712 before that, see :hg:`revert`.
1712 before that, see :hg:`revert`.
1713
1713
1714 Returns 0 on success, 1 if errors are encountered.
1714 Returns 0 on success, 1 if errors are encountered.
1715 """
1715 """
1716 wlock = repo.wlock(False)
1716 wlock = repo.wlock(False)
1717 try:
1717 try:
1718 return cmdutil.copy(ui, repo, pats, opts)
1718 return cmdutil.copy(ui, repo, pats, opts)
1719 finally:
1719 finally:
1720 wlock.release()
1720 wlock.release()
1721
1721
1722 @command('debugancestor', [], _('[INDEX] REV1 REV2'), optionalrepo=True)
1722 @command('debugancestor', [], _('[INDEX] REV1 REV2'), optionalrepo=True)
1723 def debugancestor(ui, repo, *args):
1723 def debugancestor(ui, repo, *args):
1724 """find the ancestor revision of two revisions in a given index"""
1724 """find the ancestor revision of two revisions in a given index"""
1725 if len(args) == 3:
1725 if len(args) == 3:
1726 index, rev1, rev2 = args
1726 index, rev1, rev2 = args
1727 r = revlog.revlog(scmutil.opener(os.getcwd(), audit=False), index)
1727 r = revlog.revlog(scmutil.opener(os.getcwd(), audit=False), index)
1728 lookup = r.lookup
1728 lookup = r.lookup
1729 elif len(args) == 2:
1729 elif len(args) == 2:
1730 if not repo:
1730 if not repo:
1731 raise error.Abort(_("there is no Mercurial repository here "
1731 raise error.Abort(_("there is no Mercurial repository here "
1732 "(.hg not found)"))
1732 "(.hg not found)"))
1733 rev1, rev2 = args
1733 rev1, rev2 = args
1734 r = repo.changelog
1734 r = repo.changelog
1735 lookup = repo.lookup
1735 lookup = repo.lookup
1736 else:
1736 else:
1737 raise error.Abort(_('either two or three arguments required'))
1737 raise error.Abort(_('either two or three arguments required'))
1738 a = r.ancestor(lookup(rev1), lookup(rev2))
1738 a = r.ancestor(lookup(rev1), lookup(rev2))
1739 ui.write("%d:%s\n" % (r.rev(a), hex(a)))
1739 ui.write("%d:%s\n" % (r.rev(a), hex(a)))
1740
1740
1741 @command('debugbuilddag',
1741 @command('debugbuilddag',
1742 [('m', 'mergeable-file', None, _('add single file mergeable changes')),
1742 [('m', 'mergeable-file', None, _('add single file mergeable changes')),
1743 ('o', 'overwritten-file', None, _('add single file all revs overwrite')),
1743 ('o', 'overwritten-file', None, _('add single file all revs overwrite')),
1744 ('n', 'new-file', None, _('add new file at each rev'))],
1744 ('n', 'new-file', None, _('add new file at each rev'))],
1745 _('[OPTION]... [TEXT]'))
1745 _('[OPTION]... [TEXT]'))
1746 def debugbuilddag(ui, repo, text=None,
1746 def debugbuilddag(ui, repo, text=None,
1747 mergeable_file=False,
1747 mergeable_file=False,
1748 overwritten_file=False,
1748 overwritten_file=False,
1749 new_file=False):
1749 new_file=False):
1750 """builds a repo with a given DAG from scratch in the current empty repo
1750 """builds a repo with a given DAG from scratch in the current empty repo
1751
1751
1752 The description of the DAG is read from stdin if not given on the
1752 The description of the DAG is read from stdin if not given on the
1753 command line.
1753 command line.
1754
1754
1755 Elements:
1755 Elements:
1756
1756
1757 - "+n" is a linear run of n nodes based on the current default parent
1757 - "+n" is a linear run of n nodes based on the current default parent
1758 - "." is a single node based on the current default parent
1758 - "." is a single node based on the current default parent
1759 - "$" resets the default parent to null (implied at the start);
1759 - "$" resets the default parent to null (implied at the start);
1760 otherwise the default parent is always the last node created
1760 otherwise the default parent is always the last node created
1761 - "<p" sets the default parent to the backref p
1761 - "<p" sets the default parent to the backref p
1762 - "*p" is a fork at parent p, which is a backref
1762 - "*p" is a fork at parent p, which is a backref
1763 - "*p1/p2" is a merge of parents p1 and p2, which are backrefs
1763 - "*p1/p2" is a merge of parents p1 and p2, which are backrefs
1764 - "/p2" is a merge of the preceding node and p2
1764 - "/p2" is a merge of the preceding node and p2
1765 - ":tag" defines a local tag for the preceding node
1765 - ":tag" defines a local tag for the preceding node
1766 - "@branch" sets the named branch for subsequent nodes
1766 - "@branch" sets the named branch for subsequent nodes
1767 - "#...\\n" is a comment up to the end of the line
1767 - "#...\\n" is a comment up to the end of the line
1768
1768
1769 Whitespace between the above elements is ignored.
1769 Whitespace between the above elements is ignored.
1770
1770
1771 A backref is either
1771 A backref is either
1772
1772
1773 - a number n, which references the node curr-n, where curr is the current
1773 - a number n, which references the node curr-n, where curr is the current
1774 node, or
1774 node, or
1775 - the name of a local tag you placed earlier using ":tag", or
1775 - the name of a local tag you placed earlier using ":tag", or
1776 - empty to denote the default parent.
1776 - empty to denote the default parent.
1777
1777
1778 All string valued-elements are either strictly alphanumeric, or must
1778 All string valued-elements are either strictly alphanumeric, or must
1779 be enclosed in double quotes ("..."), with "\\" as escape character.
1779 be enclosed in double quotes ("..."), with "\\" as escape character.
1780 """
1780 """
1781
1781
1782 if text is None:
1782 if text is None:
1783 ui.status(_("reading DAG from stdin\n"))
1783 ui.status(_("reading DAG from stdin\n"))
1784 text = ui.fin.read()
1784 text = ui.fin.read()
1785
1785
1786 cl = repo.changelog
1786 cl = repo.changelog
1787 if len(cl) > 0:
1787 if len(cl) > 0:
1788 raise error.Abort(_('repository is not empty'))
1788 raise error.Abort(_('repository is not empty'))
1789
1789
1790 # determine number of revs in DAG
1790 # determine number of revs in DAG
1791 total = 0
1791 total = 0
1792 for type, data in dagparser.parsedag(text):
1792 for type, data in dagparser.parsedag(text):
1793 if type == 'n':
1793 if type == 'n':
1794 total += 1
1794 total += 1
1795
1795
1796 if mergeable_file:
1796 if mergeable_file:
1797 linesperrev = 2
1797 linesperrev = 2
1798 # make a file with k lines per rev
1798 # make a file with k lines per rev
1799 initialmergedlines = [str(i) for i in xrange(0, total * linesperrev)]
1799 initialmergedlines = [str(i) for i in xrange(0, total * linesperrev)]
1800 initialmergedlines.append("")
1800 initialmergedlines.append("")
1801
1801
1802 tags = []
1802 tags = []
1803
1803
1804 lock = tr = None
1804 lock = tr = None
1805 try:
1805 try:
1806 lock = repo.lock()
1806 lock = repo.lock()
1807 tr = repo.transaction("builddag")
1807 tr = repo.transaction("builddag")
1808
1808
1809 at = -1
1809 at = -1
1810 atbranch = 'default'
1810 atbranch = 'default'
1811 nodeids = []
1811 nodeids = []
1812 id = 0
1812 id = 0
1813 ui.progress(_('building'), id, unit=_('revisions'), total=total)
1813 ui.progress(_('building'), id, unit=_('revisions'), total=total)
1814 for type, data in dagparser.parsedag(text):
1814 for type, data in dagparser.parsedag(text):
1815 if type == 'n':
1815 if type == 'n':
1816 ui.note(('node %s\n' % str(data)))
1816 ui.note(('node %s\n' % str(data)))
1817 id, ps = data
1817 id, ps = data
1818
1818
1819 files = []
1819 files = []
1820 fctxs = {}
1820 fctxs = {}
1821
1821
1822 p2 = None
1822 p2 = None
1823 if mergeable_file:
1823 if mergeable_file:
1824 fn = "mf"
1824 fn = "mf"
1825 p1 = repo[ps[0]]
1825 p1 = repo[ps[0]]
1826 if len(ps) > 1:
1826 if len(ps) > 1:
1827 p2 = repo[ps[1]]
1827 p2 = repo[ps[1]]
1828 pa = p1.ancestor(p2)
1828 pa = p1.ancestor(p2)
1829 base, local, other = [x[fn].data() for x in (pa, p1,
1829 base, local, other = [x[fn].data() for x in (pa, p1,
1830 p2)]
1830 p2)]
1831 m3 = simplemerge.Merge3Text(base, local, other)
1831 m3 = simplemerge.Merge3Text(base, local, other)
1832 ml = [l.strip() for l in m3.merge_lines()]
1832 ml = [l.strip() for l in m3.merge_lines()]
1833 ml.append("")
1833 ml.append("")
1834 elif at > 0:
1834 elif at > 0:
1835 ml = p1[fn].data().split("\n")
1835 ml = p1[fn].data().split("\n")
1836 else:
1836 else:
1837 ml = initialmergedlines
1837 ml = initialmergedlines
1838 ml[id * linesperrev] += " r%i" % id
1838 ml[id * linesperrev] += " r%i" % id
1839 mergedtext = "\n".join(ml)
1839 mergedtext = "\n".join(ml)
1840 files.append(fn)
1840 files.append(fn)
1841 fctxs[fn] = context.memfilectx(repo, fn, mergedtext)
1841 fctxs[fn] = context.memfilectx(repo, fn, mergedtext)
1842
1842
1843 if overwritten_file:
1843 if overwritten_file:
1844 fn = "of"
1844 fn = "of"
1845 files.append(fn)
1845 files.append(fn)
1846 fctxs[fn] = context.memfilectx(repo, fn, "r%i\n" % id)
1846 fctxs[fn] = context.memfilectx(repo, fn, "r%i\n" % id)
1847
1847
1848 if new_file:
1848 if new_file:
1849 fn = "nf%i" % id
1849 fn = "nf%i" % id
1850 files.append(fn)
1850 files.append(fn)
1851 fctxs[fn] = context.memfilectx(repo, fn, "r%i\n" % id)
1851 fctxs[fn] = context.memfilectx(repo, fn, "r%i\n" % id)
1852 if len(ps) > 1:
1852 if len(ps) > 1:
1853 if not p2:
1853 if not p2:
1854 p2 = repo[ps[1]]
1854 p2 = repo[ps[1]]
1855 for fn in p2:
1855 for fn in p2:
1856 if fn.startswith("nf"):
1856 if fn.startswith("nf"):
1857 files.append(fn)
1857 files.append(fn)
1858 fctxs[fn] = p2[fn]
1858 fctxs[fn] = p2[fn]
1859
1859
1860 def fctxfn(repo, cx, path):
1860 def fctxfn(repo, cx, path):
1861 return fctxs.get(path)
1861 return fctxs.get(path)
1862
1862
1863 if len(ps) == 0 or ps[0] < 0:
1863 if len(ps) == 0 or ps[0] < 0:
1864 pars = [None, None]
1864 pars = [None, None]
1865 elif len(ps) == 1:
1865 elif len(ps) == 1:
1866 pars = [nodeids[ps[0]], None]
1866 pars = [nodeids[ps[0]], None]
1867 else:
1867 else:
1868 pars = [nodeids[p] for p in ps]
1868 pars = [nodeids[p] for p in ps]
1869 cx = context.memctx(repo, pars, "r%i" % id, files, fctxfn,
1869 cx = context.memctx(repo, pars, "r%i" % id, files, fctxfn,
1870 date=(id, 0),
1870 date=(id, 0),
1871 user="debugbuilddag",
1871 user="debugbuilddag",
1872 extra={'branch': atbranch})
1872 extra={'branch': atbranch})
1873 nodeid = repo.commitctx(cx)
1873 nodeid = repo.commitctx(cx)
1874 nodeids.append(nodeid)
1874 nodeids.append(nodeid)
1875 at = id
1875 at = id
1876 elif type == 'l':
1876 elif type == 'l':
1877 id, name = data
1877 id, name = data
1878 ui.note(('tag %s\n' % name))
1878 ui.note(('tag %s\n' % name))
1879 tags.append("%s %s\n" % (hex(repo.changelog.node(id)), name))
1879 tags.append("%s %s\n" % (hex(repo.changelog.node(id)), name))
1880 elif type == 'a':
1880 elif type == 'a':
1881 ui.note(('branch %s\n' % data))
1881 ui.note(('branch %s\n' % data))
1882 atbranch = data
1882 atbranch = data
1883 ui.progress(_('building'), id, unit=_('revisions'), total=total)
1883 ui.progress(_('building'), id, unit=_('revisions'), total=total)
1884 tr.close()
1884 tr.close()
1885
1885
1886 if tags:
1886 if tags:
1887 repo.vfs.write("localtags", "".join(tags))
1887 repo.vfs.write("localtags", "".join(tags))
1888 finally:
1888 finally:
1889 ui.progress(_('building'), None)
1889 ui.progress(_('building'), None)
1890 release(tr, lock)
1890 release(tr, lock)
1891
1891
1892 @command('debugbundle',
1892 @command('debugbundle',
1893 [('a', 'all', None, _('show all details'))],
1893 [('a', 'all', None, _('show all details'))],
1894 _('FILE'),
1894 _('FILE'),
1895 norepo=True)
1895 norepo=True)
1896 def debugbundle(ui, bundlepath, all=None, **opts):
1896 def debugbundle(ui, bundlepath, all=None, **opts):
1897 """lists the contents of a bundle"""
1897 """lists the contents of a bundle"""
1898 f = hg.openpath(ui, bundlepath)
1898 f = hg.openpath(ui, bundlepath)
1899 try:
1899 try:
1900 gen = exchange.readbundle(ui, f, bundlepath)
1900 gen = exchange.readbundle(ui, f, bundlepath)
1901 if isinstance(gen, bundle2.unbundle20):
1901 if isinstance(gen, bundle2.unbundle20):
1902 return _debugbundle2(ui, gen, all=all, **opts)
1902 return _debugbundle2(ui, gen, all=all, **opts)
1903 if all:
1903 if all:
1904 ui.write(("format: id, p1, p2, cset, delta base, len(delta)\n"))
1904 ui.write(("format: id, p1, p2, cset, delta base, len(delta)\n"))
1905
1905
1906 def showchunks(named):
1906 def showchunks(named):
1907 ui.write("\n%s\n" % named)
1907 ui.write("\n%s\n" % named)
1908 chain = None
1908 chain = None
1909 while True:
1909 while True:
1910 chunkdata = gen.deltachunk(chain)
1910 chunkdata = gen.deltachunk(chain)
1911 if not chunkdata:
1911 if not chunkdata:
1912 break
1912 break
1913 node = chunkdata['node']
1913 node = chunkdata['node']
1914 p1 = chunkdata['p1']
1914 p1 = chunkdata['p1']
1915 p2 = chunkdata['p2']
1915 p2 = chunkdata['p2']
1916 cs = chunkdata['cs']
1916 cs = chunkdata['cs']
1917 deltabase = chunkdata['deltabase']
1917 deltabase = chunkdata['deltabase']
1918 delta = chunkdata['delta']
1918 delta = chunkdata['delta']
1919 ui.write("%s %s %s %s %s %s\n" %
1919 ui.write("%s %s %s %s %s %s\n" %
1920 (hex(node), hex(p1), hex(p2),
1920 (hex(node), hex(p1), hex(p2),
1921 hex(cs), hex(deltabase), len(delta)))
1921 hex(cs), hex(deltabase), len(delta)))
1922 chain = node
1922 chain = node
1923
1923
1924 chunkdata = gen.changelogheader()
1924 chunkdata = gen.changelogheader()
1925 showchunks("changelog")
1925 showchunks("changelog")
1926 chunkdata = gen.manifestheader()
1926 chunkdata = gen.manifestheader()
1927 showchunks("manifest")
1927 showchunks("manifest")
1928 while True:
1928 while True:
1929 chunkdata = gen.filelogheader()
1929 chunkdata = gen.filelogheader()
1930 if not chunkdata:
1930 if not chunkdata:
1931 break
1931 break
1932 fname = chunkdata['filename']
1932 fname = chunkdata['filename']
1933 showchunks(fname)
1933 showchunks(fname)
1934 else:
1934 else:
1935 if isinstance(gen, bundle2.unbundle20):
1935 if isinstance(gen, bundle2.unbundle20):
1936 raise error.Abort(_('use debugbundle2 for this file'))
1936 raise error.Abort(_('use debugbundle2 for this file'))
1937 chunkdata = gen.changelogheader()
1937 chunkdata = gen.changelogheader()
1938 chain = None
1938 chain = None
1939 while True:
1939 while True:
1940 chunkdata = gen.deltachunk(chain)
1940 chunkdata = gen.deltachunk(chain)
1941 if not chunkdata:
1941 if not chunkdata:
1942 break
1942 break
1943 node = chunkdata['node']
1943 node = chunkdata['node']
1944 ui.write("%s\n" % hex(node))
1944 ui.write("%s\n" % hex(node))
1945 chain = node
1945 chain = node
1946 finally:
1946 finally:
1947 f.close()
1947 f.close()
1948
1948
1949 def _debugbundle2(ui, gen, **opts):
1949 def _debugbundle2(ui, gen, **opts):
1950 """lists the contents of a bundle2"""
1950 """lists the contents of a bundle2"""
1951 if not isinstance(gen, bundle2.unbundle20):
1951 if not isinstance(gen, bundle2.unbundle20):
1952 raise error.Abort(_('not a bundle2 file'))
1952 raise error.Abort(_('not a bundle2 file'))
1953 ui.write(('Stream params: %s\n' % repr(gen.params)))
1953 ui.write(('Stream params: %s\n' % repr(gen.params)))
1954 for part in gen.iterparts():
1954 for part in gen.iterparts():
1955 ui.write('%s -- %r\n' % (part.type, repr(part.params)))
1955 ui.write('%s -- %r\n' % (part.type, repr(part.params)))
1956 if part.type == 'changegroup':
1956 if part.type == 'changegroup':
1957 version = part.params.get('version', '01')
1957 version = part.params.get('version', '01')
1958 cg = changegroup.packermap[version][1](part, 'UN')
1958 cg = changegroup.packermap[version][1](part, 'UN')
1959 chunkdata = cg.changelogheader()
1959 chunkdata = cg.changelogheader()
1960 chain = None
1960 chain = None
1961 while True:
1961 while True:
1962 chunkdata = cg.deltachunk(chain)
1962 chunkdata = cg.deltachunk(chain)
1963 if not chunkdata:
1963 if not chunkdata:
1964 break
1964 break
1965 node = chunkdata['node']
1965 node = chunkdata['node']
1966 ui.write(" %s\n" % hex(node))
1966 ui.write(" %s\n" % hex(node))
1967 chain = node
1967 chain = node
1968
1968
1969 @command('debugcreatestreamclonebundle', [], 'FILE')
1969 @command('debugcreatestreamclonebundle', [], 'FILE')
1970 def debugcreatestreamclonebundle(ui, repo, fname):
1970 def debugcreatestreamclonebundle(ui, repo, fname):
1971 """create a stream clone bundle file
1971 """create a stream clone bundle file
1972
1972
1973 Stream bundles are special bundles that are essentially archives of
1973 Stream bundles are special bundles that are essentially archives of
1974 revlog files. They are commonly used for cloning very quickly.
1974 revlog files. They are commonly used for cloning very quickly.
1975 """
1975 """
1976 requirements, gen = streamclone.generatebundlev1(repo)
1976 requirements, gen = streamclone.generatebundlev1(repo)
1977 changegroup.writechunks(ui, gen, fname)
1977 changegroup.writechunks(ui, gen, fname)
1978
1978
1979 ui.write(_('bundle requirements: %s\n') % ', '.join(sorted(requirements)))
1979 ui.write(_('bundle requirements: %s\n') % ', '.join(sorted(requirements)))
1980
1980
1981 @command('debugapplystreamclonebundle', [], 'FILE')
1981 @command('debugapplystreamclonebundle', [], 'FILE')
1982 def debugapplystreamclonebundle(ui, repo, fname):
1982 def debugapplystreamclonebundle(ui, repo, fname):
1983 """apply a stream clone bundle file"""
1983 """apply a stream clone bundle file"""
1984 f = hg.openpath(ui, fname)
1984 f = hg.openpath(ui, fname)
1985 gen = exchange.readbundle(ui, f, fname)
1985 gen = exchange.readbundle(ui, f, fname)
1986 gen.apply(repo)
1986 gen.apply(repo)
1987
1987
1988 @command('debugcheckstate', [], '')
1988 @command('debugcheckstate', [], '')
1989 def debugcheckstate(ui, repo):
1989 def debugcheckstate(ui, repo):
1990 """validate the correctness of the current dirstate"""
1990 """validate the correctness of the current dirstate"""
1991 parent1, parent2 = repo.dirstate.parents()
1991 parent1, parent2 = repo.dirstate.parents()
1992 m1 = repo[parent1].manifest()
1992 m1 = repo[parent1].manifest()
1993 m2 = repo[parent2].manifest()
1993 m2 = repo[parent2].manifest()
1994 errors = 0
1994 errors = 0
1995 for f in repo.dirstate:
1995 for f in repo.dirstate:
1996 state = repo.dirstate[f]
1996 state = repo.dirstate[f]
1997 if state in "nr" and f not in m1:
1997 if state in "nr" and f not in m1:
1998 ui.warn(_("%s in state %s, but not in manifest1\n") % (f, state))
1998 ui.warn(_("%s in state %s, but not in manifest1\n") % (f, state))
1999 errors += 1
1999 errors += 1
2000 if state in "a" and f in m1:
2000 if state in "a" and f in m1:
2001 ui.warn(_("%s in state %s, but also in manifest1\n") % (f, state))
2001 ui.warn(_("%s in state %s, but also in manifest1\n") % (f, state))
2002 errors += 1
2002 errors += 1
2003 if state in "m" and f not in m1 and f not in m2:
2003 if state in "m" and f not in m1 and f not in m2:
2004 ui.warn(_("%s in state %s, but not in either manifest\n") %
2004 ui.warn(_("%s in state %s, but not in either manifest\n") %
2005 (f, state))
2005 (f, state))
2006 errors += 1
2006 errors += 1
2007 for f in m1:
2007 for f in m1:
2008 state = repo.dirstate[f]
2008 state = repo.dirstate[f]
2009 if state not in "nrm":
2009 if state not in "nrm":
2010 ui.warn(_("%s in manifest1, but listed as state %s") % (f, state))
2010 ui.warn(_("%s in manifest1, but listed as state %s") % (f, state))
2011 errors += 1
2011 errors += 1
2012 if errors:
2012 if errors:
2013 error = _(".hg/dirstate inconsistent with current parent's manifest")
2013 error = _(".hg/dirstate inconsistent with current parent's manifest")
2014 raise error.Abort(error)
2014 raise error.Abort(error)
2015
2015
2016 @command('debugcommands', [], _('[COMMAND]'), norepo=True)
2016 @command('debugcommands', [], _('[COMMAND]'), norepo=True)
2017 def debugcommands(ui, cmd='', *args):
2017 def debugcommands(ui, cmd='', *args):
2018 """list all available commands and options"""
2018 """list all available commands and options"""
2019 for cmd, vals in sorted(table.iteritems()):
2019 for cmd, vals in sorted(table.iteritems()):
2020 cmd = cmd.split('|')[0].strip('^')
2020 cmd = cmd.split('|')[0].strip('^')
2021 opts = ', '.join([i[1] for i in vals[1]])
2021 opts = ', '.join([i[1] for i in vals[1]])
2022 ui.write('%s: %s\n' % (cmd, opts))
2022 ui.write('%s: %s\n' % (cmd, opts))
2023
2023
2024 @command('debugcomplete',
2024 @command('debugcomplete',
2025 [('o', 'options', None, _('show the command options'))],
2025 [('o', 'options', None, _('show the command options'))],
2026 _('[-o] CMD'),
2026 _('[-o] CMD'),
2027 norepo=True)
2027 norepo=True)
2028 def debugcomplete(ui, cmd='', **opts):
2028 def debugcomplete(ui, cmd='', **opts):
2029 """returns the completion list associated with the given command"""
2029 """returns the completion list associated with the given command"""
2030
2030
2031 if opts.get('options'):
2031 if opts.get('options'):
2032 options = []
2032 options = []
2033 otables = [globalopts]
2033 otables = [globalopts]
2034 if cmd:
2034 if cmd:
2035 aliases, entry = cmdutil.findcmd(cmd, table, False)
2035 aliases, entry = cmdutil.findcmd(cmd, table, False)
2036 otables.append(entry[1])
2036 otables.append(entry[1])
2037 for t in otables:
2037 for t in otables:
2038 for o in t:
2038 for o in t:
2039 if "(DEPRECATED)" in o[3]:
2039 if "(DEPRECATED)" in o[3]:
2040 continue
2040 continue
2041 if o[0]:
2041 if o[0]:
2042 options.append('-%s' % o[0])
2042 options.append('-%s' % o[0])
2043 options.append('--%s' % o[1])
2043 options.append('--%s' % o[1])
2044 ui.write("%s\n" % "\n".join(options))
2044 ui.write("%s\n" % "\n".join(options))
2045 return
2045 return
2046
2046
2047 cmdlist, unused_allcmds = cmdutil.findpossible(cmd, table)
2047 cmdlist, unused_allcmds = cmdutil.findpossible(cmd, table)
2048 if ui.verbose:
2048 if ui.verbose:
2049 cmdlist = [' '.join(c[0]) for c in cmdlist.values()]
2049 cmdlist = [' '.join(c[0]) for c in cmdlist.values()]
2050 ui.write("%s\n" % "\n".join(sorted(cmdlist)))
2050 ui.write("%s\n" % "\n".join(sorted(cmdlist)))
2051
2051
2052 @command('debugdag',
2052 @command('debugdag',
2053 [('t', 'tags', None, _('use tags as labels')),
2053 [('t', 'tags', None, _('use tags as labels')),
2054 ('b', 'branches', None, _('annotate with branch names')),
2054 ('b', 'branches', None, _('annotate with branch names')),
2055 ('', 'dots', None, _('use dots for runs')),
2055 ('', 'dots', None, _('use dots for runs')),
2056 ('s', 'spaces', None, _('separate elements by spaces'))],
2056 ('s', 'spaces', None, _('separate elements by spaces'))],
2057 _('[OPTION]... [FILE [REV]...]'),
2057 _('[OPTION]... [FILE [REV]...]'),
2058 optionalrepo=True)
2058 optionalrepo=True)
2059 def debugdag(ui, repo, file_=None, *revs, **opts):
2059 def debugdag(ui, repo, file_=None, *revs, **opts):
2060 """format the changelog or an index DAG as a concise textual description
2060 """format the changelog or an index DAG as a concise textual description
2061
2061
2062 If you pass a revlog index, the revlog's DAG is emitted. If you list
2062 If you pass a revlog index, the revlog's DAG is emitted. If you list
2063 revision numbers, they get labeled in the output as rN.
2063 revision numbers, they get labeled in the output as rN.
2064
2064
2065 Otherwise, the changelog DAG of the current repo is emitted.
2065 Otherwise, the changelog DAG of the current repo is emitted.
2066 """
2066 """
2067 spaces = opts.get('spaces')
2067 spaces = opts.get('spaces')
2068 dots = opts.get('dots')
2068 dots = opts.get('dots')
2069 if file_:
2069 if file_:
2070 rlog = revlog.revlog(scmutil.opener(os.getcwd(), audit=False), file_)
2070 rlog = revlog.revlog(scmutil.opener(os.getcwd(), audit=False), file_)
2071 revs = set((int(r) for r in revs))
2071 revs = set((int(r) for r in revs))
2072 def events():
2072 def events():
2073 for r in rlog:
2073 for r in rlog:
2074 yield 'n', (r, list(p for p in rlog.parentrevs(r)
2074 yield 'n', (r, list(p for p in rlog.parentrevs(r)
2075 if p != -1))
2075 if p != -1))
2076 if r in revs:
2076 if r in revs:
2077 yield 'l', (r, "r%i" % r)
2077 yield 'l', (r, "r%i" % r)
2078 elif repo:
2078 elif repo:
2079 cl = repo.changelog
2079 cl = repo.changelog
2080 tags = opts.get('tags')
2080 tags = opts.get('tags')
2081 branches = opts.get('branches')
2081 branches = opts.get('branches')
2082 if tags:
2082 if tags:
2083 labels = {}
2083 labels = {}
2084 for l, n in repo.tags().items():
2084 for l, n in repo.tags().items():
2085 labels.setdefault(cl.rev(n), []).append(l)
2085 labels.setdefault(cl.rev(n), []).append(l)
2086 def events():
2086 def events():
2087 b = "default"
2087 b = "default"
2088 for r in cl:
2088 for r in cl:
2089 if branches:
2089 if branches:
2090 newb = cl.read(cl.node(r))[5]['branch']
2090 newb = cl.read(cl.node(r))[5]['branch']
2091 if newb != b:
2091 if newb != b:
2092 yield 'a', newb
2092 yield 'a', newb
2093 b = newb
2093 b = newb
2094 yield 'n', (r, list(p for p in cl.parentrevs(r)
2094 yield 'n', (r, list(p for p in cl.parentrevs(r)
2095 if p != -1))
2095 if p != -1))
2096 if tags:
2096 if tags:
2097 ls = labels.get(r)
2097 ls = labels.get(r)
2098 if ls:
2098 if ls:
2099 for l in ls:
2099 for l in ls:
2100 yield 'l', (r, l)
2100 yield 'l', (r, l)
2101 else:
2101 else:
2102 raise error.Abort(_('need repo for changelog dag'))
2102 raise error.Abort(_('need repo for changelog dag'))
2103
2103
2104 for line in dagparser.dagtextlines(events(),
2104 for line in dagparser.dagtextlines(events(),
2105 addspaces=spaces,
2105 addspaces=spaces,
2106 wraplabels=True,
2106 wraplabels=True,
2107 wrapannotations=True,
2107 wrapannotations=True,
2108 wrapnonlinear=dots,
2108 wrapnonlinear=dots,
2109 usedots=dots,
2109 usedots=dots,
2110 maxlinewidth=70):
2110 maxlinewidth=70):
2111 ui.write(line)
2111 ui.write(line)
2112 ui.write("\n")
2112 ui.write("\n")
2113
2113
2114 @command('debugdata',
2114 @command('debugdata',
2115 [('c', 'changelog', False, _('open changelog')),
2115 [('c', 'changelog', False, _('open changelog')),
2116 ('m', 'manifest', False, _('open manifest')),
2116 ('m', 'manifest', False, _('open manifest')),
2117 ('', 'dir', False, _('open directory manifest'))],
2117 ('', 'dir', False, _('open directory manifest'))],
2118 _('-c|-m|FILE REV'))
2118 _('-c|-m|FILE REV'))
2119 def debugdata(ui, repo, file_, rev=None, **opts):
2119 def debugdata(ui, repo, file_, rev=None, **opts):
2120 """dump the contents of a data file revision"""
2120 """dump the contents of a data file revision"""
2121 if opts.get('changelog') or opts.get('manifest'):
2121 if opts.get('changelog') or opts.get('manifest'):
2122 file_, rev = None, file_
2122 file_, rev = None, file_
2123 elif rev is None:
2123 elif rev is None:
2124 raise error.CommandError('debugdata', _('invalid arguments'))
2124 raise error.CommandError('debugdata', _('invalid arguments'))
2125 r = cmdutil.openrevlog(repo, 'debugdata', file_, opts)
2125 r = cmdutil.openrevlog(repo, 'debugdata', file_, opts)
2126 try:
2126 try:
2127 ui.write(r.revision(r.lookup(rev)))
2127 ui.write(r.revision(r.lookup(rev)))
2128 except KeyError:
2128 except KeyError:
2129 raise error.Abort(_('invalid revision identifier %s') % rev)
2129 raise error.Abort(_('invalid revision identifier %s') % rev)
2130
2130
2131 @command('debugdate',
2131 @command('debugdate',
2132 [('e', 'extended', None, _('try extended date formats'))],
2132 [('e', 'extended', None, _('try extended date formats'))],
2133 _('[-e] DATE [RANGE]'),
2133 _('[-e] DATE [RANGE]'),
2134 norepo=True, optionalrepo=True)
2134 norepo=True, optionalrepo=True)
2135 def debugdate(ui, date, range=None, **opts):
2135 def debugdate(ui, date, range=None, **opts):
2136 """parse and display a date"""
2136 """parse and display a date"""
2137 if opts["extended"]:
2137 if opts["extended"]:
2138 d = util.parsedate(date, util.extendeddateformats)
2138 d = util.parsedate(date, util.extendeddateformats)
2139 else:
2139 else:
2140 d = util.parsedate(date)
2140 d = util.parsedate(date)
2141 ui.write(("internal: %s %s\n") % d)
2141 ui.write(("internal: %s %s\n") % d)
2142 ui.write(("standard: %s\n") % util.datestr(d))
2142 ui.write(("standard: %s\n") % util.datestr(d))
2143 if range:
2143 if range:
2144 m = util.matchdate(range)
2144 m = util.matchdate(range)
2145 ui.write(("match: %s\n") % m(d[0]))
2145 ui.write(("match: %s\n") % m(d[0]))
2146
2146
2147 @command('debugdiscovery',
2147 @command('debugdiscovery',
2148 [('', 'old', None, _('use old-style discovery')),
2148 [('', 'old', None, _('use old-style discovery')),
2149 ('', 'nonheads', None,
2149 ('', 'nonheads', None,
2150 _('use old-style discovery with non-heads included')),
2150 _('use old-style discovery with non-heads included')),
2151 ] + remoteopts,
2151 ] + remoteopts,
2152 _('[-l REV] [-r REV] [-b BRANCH]... [OTHER]'))
2152 _('[-l REV] [-r REV] [-b BRANCH]... [OTHER]'))
2153 def debugdiscovery(ui, repo, remoteurl="default", **opts):
2153 def debugdiscovery(ui, repo, remoteurl="default", **opts):
2154 """runs the changeset discovery protocol in isolation"""
2154 """runs the changeset discovery protocol in isolation"""
2155 remoteurl, branches = hg.parseurl(ui.expandpath(remoteurl),
2155 remoteurl, branches = hg.parseurl(ui.expandpath(remoteurl),
2156 opts.get('branch'))
2156 opts.get('branch'))
2157 remote = hg.peer(repo, opts, remoteurl)
2157 remote = hg.peer(repo, opts, remoteurl)
2158 ui.status(_('comparing with %s\n') % util.hidepassword(remoteurl))
2158 ui.status(_('comparing with %s\n') % util.hidepassword(remoteurl))
2159
2159
2160 # make sure tests are repeatable
2160 # make sure tests are repeatable
2161 random.seed(12323)
2161 random.seed(12323)
2162
2162
2163 def doit(localheads, remoteheads, remote=remote):
2163 def doit(localheads, remoteheads, remote=remote):
2164 if opts.get('old'):
2164 if opts.get('old'):
2165 if localheads:
2165 if localheads:
2166 raise error.Abort('cannot use localheads with old style '
2166 raise error.Abort('cannot use localheads with old style '
2167 'discovery')
2167 'discovery')
2168 if not util.safehasattr(remote, 'branches'):
2168 if not util.safehasattr(remote, 'branches'):
2169 # enable in-client legacy support
2169 # enable in-client legacy support
2170 remote = localrepo.locallegacypeer(remote.local())
2170 remote = localrepo.locallegacypeer(remote.local())
2171 common, _in, hds = treediscovery.findcommonincoming(repo, remote,
2171 common, _in, hds = treediscovery.findcommonincoming(repo, remote,
2172 force=True)
2172 force=True)
2173 common = set(common)
2173 common = set(common)
2174 if not opts.get('nonheads'):
2174 if not opts.get('nonheads'):
2175 ui.write(("unpruned common: %s\n") %
2175 ui.write(("unpruned common: %s\n") %
2176 " ".join(sorted(short(n) for n in common)))
2176 " ".join(sorted(short(n) for n in common)))
2177 dag = dagutil.revlogdag(repo.changelog)
2177 dag = dagutil.revlogdag(repo.changelog)
2178 all = dag.ancestorset(dag.internalizeall(common))
2178 all = dag.ancestorset(dag.internalizeall(common))
2179 common = dag.externalizeall(dag.headsetofconnecteds(all))
2179 common = dag.externalizeall(dag.headsetofconnecteds(all))
2180 else:
2180 else:
2181 common, any, hds = setdiscovery.findcommonheads(ui, repo, remote)
2181 common, any, hds = setdiscovery.findcommonheads(ui, repo, remote)
2182 common = set(common)
2182 common = set(common)
2183 rheads = set(hds)
2183 rheads = set(hds)
2184 lheads = set(repo.heads())
2184 lheads = set(repo.heads())
2185 ui.write(("common heads: %s\n") %
2185 ui.write(("common heads: %s\n") %
2186 " ".join(sorted(short(n) for n in common)))
2186 " ".join(sorted(short(n) for n in common)))
2187 if lheads <= common:
2187 if lheads <= common:
2188 ui.write(("local is subset\n"))
2188 ui.write(("local is subset\n"))
2189 elif rheads <= common:
2189 elif rheads <= common:
2190 ui.write(("remote is subset\n"))
2190 ui.write(("remote is subset\n"))
2191
2191
2192 serverlogs = opts.get('serverlog')
2192 serverlogs = opts.get('serverlog')
2193 if serverlogs:
2193 if serverlogs:
2194 for filename in serverlogs:
2194 for filename in serverlogs:
2195 logfile = open(filename, 'r')
2195 logfile = open(filename, 'r')
2196 try:
2196 try:
2197 line = logfile.readline()
2197 line = logfile.readline()
2198 while line:
2198 while line:
2199 parts = line.strip().split(';')
2199 parts = line.strip().split(';')
2200 op = parts[1]
2200 op = parts[1]
2201 if op == 'cg':
2201 if op == 'cg':
2202 pass
2202 pass
2203 elif op == 'cgss':
2203 elif op == 'cgss':
2204 doit(parts[2].split(' '), parts[3].split(' '))
2204 doit(parts[2].split(' '), parts[3].split(' '))
2205 elif op == 'unb':
2205 elif op == 'unb':
2206 doit(parts[3].split(' '), parts[2].split(' '))
2206 doit(parts[3].split(' '), parts[2].split(' '))
2207 line = logfile.readline()
2207 line = logfile.readline()
2208 finally:
2208 finally:
2209 logfile.close()
2209 logfile.close()
2210
2210
2211 else:
2211 else:
2212 remoterevs, _checkout = hg.addbranchrevs(repo, remote, branches,
2212 remoterevs, _checkout = hg.addbranchrevs(repo, remote, branches,
2213 opts.get('remote_head'))
2213 opts.get('remote_head'))
2214 localrevs = opts.get('local_head')
2214 localrevs = opts.get('local_head')
2215 doit(localrevs, remoterevs)
2215 doit(localrevs, remoterevs)
2216
2216
2217 @command('debugextensions', formatteropts, [], norepo=True)
2217 @command('debugextensions', formatteropts, [], norepo=True)
2218 def debugextensions(ui, **opts):
2218 def debugextensions(ui, **opts):
2219 '''show information about active extensions'''
2219 '''show information about active extensions'''
2220 exts = extensions.extensions(ui)
2220 exts = extensions.extensions(ui)
2221 fm = ui.formatter('debugextensions', opts)
2221 fm = ui.formatter('debugextensions', opts)
2222 for extname, extmod in sorted(exts, key=operator.itemgetter(0)):
2222 for extname, extmod in sorted(exts, key=operator.itemgetter(0)):
2223 extsource = extmod.__file__
2223 extsource = extmod.__file__
2224 exttestedwith = getattr(extmod, 'testedwith', None)
2224 exttestedwith = getattr(extmod, 'testedwith', None)
2225 if exttestedwith is not None:
2225 if exttestedwith is not None:
2226 exttestedwith = exttestedwith.split()
2226 exttestedwith = exttestedwith.split()
2227 extbuglink = getattr(extmod, 'buglink', None)
2227 extbuglink = getattr(extmod, 'buglink', None)
2228
2228
2229 fm.startitem()
2229 fm.startitem()
2230
2230
2231 if ui.quiet or ui.verbose:
2231 if ui.quiet or ui.verbose:
2232 fm.write('name', '%s\n', extname)
2232 fm.write('name', '%s\n', extname)
2233 else:
2233 else:
2234 fm.write('name', '%s', extname)
2234 fm.write('name', '%s', extname)
2235 if not exttestedwith:
2235 if not exttestedwith:
2236 fm.plain(_(' (untested!)\n'))
2236 fm.plain(_(' (untested!)\n'))
2237 else:
2237 else:
2238 if exttestedwith == ['internal'] or \
2238 if exttestedwith == ['internal'] or \
2239 util.version() in exttestedwith:
2239 util.version() in exttestedwith:
2240 fm.plain('\n')
2240 fm.plain('\n')
2241 else:
2241 else:
2242 lasttestedversion = exttestedwith[-1]
2242 lasttestedversion = exttestedwith[-1]
2243 fm.plain(' (%s!)\n' % lasttestedversion)
2243 fm.plain(' (%s!)\n' % lasttestedversion)
2244
2244
2245 fm.condwrite(ui.verbose and extsource, 'source',
2245 fm.condwrite(ui.verbose and extsource, 'source',
2246 _(' location: %s\n'), extsource or "")
2246 _(' location: %s\n'), extsource or "")
2247
2247
2248 fm.condwrite(ui.verbose and exttestedwith, 'testedwith',
2248 fm.condwrite(ui.verbose and exttestedwith, 'testedwith',
2249 _(' tested with: %s\n'), ' '.join(exttestedwith or []))
2249 _(' tested with: %s\n'), ' '.join(exttestedwith or []))
2250
2250
2251 fm.condwrite(ui.verbose and extbuglink, 'buglink',
2251 fm.condwrite(ui.verbose and extbuglink, 'buglink',
2252 _(' bug reporting: %s\n'), extbuglink or "")
2252 _(' bug reporting: %s\n'), extbuglink or "")
2253
2253
2254 fm.end()
2254 fm.end()
2255
2255
2256 @command('debugfileset',
2256 @command('debugfileset',
2257 [('r', 'rev', '', _('apply the filespec on this revision'), _('REV'))],
2257 [('r', 'rev', '', _('apply the filespec on this revision'), _('REV'))],
2258 _('[-r REV] FILESPEC'))
2258 _('[-r REV] FILESPEC'))
2259 def debugfileset(ui, repo, expr, **opts):
2259 def debugfileset(ui, repo, expr, **opts):
2260 '''parse and apply a fileset specification'''
2260 '''parse and apply a fileset specification'''
2261 ctx = scmutil.revsingle(repo, opts.get('rev'), None)
2261 ctx = scmutil.revsingle(repo, opts.get('rev'), None)
2262 if ui.verbose:
2262 if ui.verbose:
2263 tree = fileset.parse(expr)
2263 tree = fileset.parse(expr)
2264 ui.note(fileset.prettyformat(tree), "\n")
2264 ui.note(fileset.prettyformat(tree), "\n")
2265
2265
2266 for f in ctx.getfileset(expr):
2266 for f in ctx.getfileset(expr):
2267 ui.write("%s\n" % f)
2267 ui.write("%s\n" % f)
2268
2268
2269 @command('debugfsinfo', [], _('[PATH]'), norepo=True)
2269 @command('debugfsinfo', [], _('[PATH]'), norepo=True)
2270 def debugfsinfo(ui, path="."):
2270 def debugfsinfo(ui, path="."):
2271 """show information detected about current filesystem"""
2271 """show information detected about current filesystem"""
2272 util.writefile('.debugfsinfo', '')
2272 util.writefile('.debugfsinfo', '')
2273 ui.write(('exec: %s\n') % (util.checkexec(path) and 'yes' or 'no'))
2273 ui.write(('exec: %s\n') % (util.checkexec(path) and 'yes' or 'no'))
2274 ui.write(('symlink: %s\n') % (util.checklink(path) and 'yes' or 'no'))
2274 ui.write(('symlink: %s\n') % (util.checklink(path) and 'yes' or 'no'))
2275 ui.write(('hardlink: %s\n') % (util.checknlink(path) and 'yes' or 'no'))
2275 ui.write(('hardlink: %s\n') % (util.checknlink(path) and 'yes' or 'no'))
2276 ui.write(('case-sensitive: %s\n') % (util.checkcase('.debugfsinfo')
2276 ui.write(('case-sensitive: %s\n') % (util.checkcase('.debugfsinfo')
2277 and 'yes' or 'no'))
2277 and 'yes' or 'no'))
2278 os.unlink('.debugfsinfo')
2278 os.unlink('.debugfsinfo')
2279
2279
2280 @command('debuggetbundle',
2280 @command('debuggetbundle',
2281 [('H', 'head', [], _('id of head node'), _('ID')),
2281 [('H', 'head', [], _('id of head node'), _('ID')),
2282 ('C', 'common', [], _('id of common node'), _('ID')),
2282 ('C', 'common', [], _('id of common node'), _('ID')),
2283 ('t', 'type', 'bzip2', _('bundle compression type to use'), _('TYPE'))],
2283 ('t', 'type', 'bzip2', _('bundle compression type to use'), _('TYPE'))],
2284 _('REPO FILE [-H|-C ID]...'),
2284 _('REPO FILE [-H|-C ID]...'),
2285 norepo=True)
2285 norepo=True)
2286 def debuggetbundle(ui, repopath, bundlepath, head=None, common=None, **opts):
2286 def debuggetbundle(ui, repopath, bundlepath, head=None, common=None, **opts):
2287 """retrieves a bundle from a repo
2287 """retrieves a bundle from a repo
2288
2288
2289 Every ID must be a full-length hex node id string. Saves the bundle to the
2289 Every ID must be a full-length hex node id string. Saves the bundle to the
2290 given file.
2290 given file.
2291 """
2291 """
2292 repo = hg.peer(ui, opts, repopath)
2292 repo = hg.peer(ui, opts, repopath)
2293 if not repo.capable('getbundle'):
2293 if not repo.capable('getbundle'):
2294 raise error.Abort("getbundle() not supported by target repository")
2294 raise error.Abort("getbundle() not supported by target repository")
2295 args = {}
2295 args = {}
2296 if common:
2296 if common:
2297 args['common'] = [bin(s) for s in common]
2297 args['common'] = [bin(s) for s in common]
2298 if head:
2298 if head:
2299 args['heads'] = [bin(s) for s in head]
2299 args['heads'] = [bin(s) for s in head]
2300 # TODO: get desired bundlecaps from command line.
2300 # TODO: get desired bundlecaps from command line.
2301 args['bundlecaps'] = None
2301 args['bundlecaps'] = None
2302 bundle = repo.getbundle('debug', **args)
2302 bundle = repo.getbundle('debug', **args)
2303
2303
2304 bundletype = opts.get('type', 'bzip2').lower()
2304 bundletype = opts.get('type', 'bzip2').lower()
2305 btypes = {'none': 'HG10UN',
2305 btypes = {'none': 'HG10UN',
2306 'bzip2': 'HG10BZ',
2306 'bzip2': 'HG10BZ',
2307 'gzip': 'HG10GZ',
2307 'gzip': 'HG10GZ',
2308 'bundle2': 'HG20'}
2308 'bundle2': 'HG20'}
2309 bundletype = btypes.get(bundletype)
2309 bundletype = btypes.get(bundletype)
2310 if bundletype not in changegroup.bundletypes:
2310 if bundletype not in changegroup.bundletypes:
2311 raise error.Abort(_('unknown bundle type specified with --type'))
2311 raise error.Abort(_('unknown bundle type specified with --type'))
2312 changegroup.writebundle(ui, bundle, bundlepath, bundletype)
2312 changegroup.writebundle(ui, bundle, bundlepath, bundletype)
2313
2313
2314 @command('debugignore', [], '')
2314 @command('debugignore', [], '')
2315 def debugignore(ui, repo, *values, **opts):
2315 def debugignore(ui, repo, *values, **opts):
2316 """display the combined ignore pattern"""
2316 """display the combined ignore pattern"""
2317 ignore = repo.dirstate._ignore
2317 ignore = repo.dirstate._ignore
2318 includepat = getattr(ignore, 'includepat', None)
2318 includepat = getattr(ignore, 'includepat', None)
2319 if includepat is not None:
2319 if includepat is not None:
2320 ui.write("%s\n" % includepat)
2320 ui.write("%s\n" % includepat)
2321 else:
2321 else:
2322 raise error.Abort(_("no ignore patterns found"))
2322 raise error.Abort(_("no ignore patterns found"))
2323
2323
2324 @command('debugindex',
2324 @command('debugindex',
2325 [('c', 'changelog', False, _('open changelog')),
2325 [('c', 'changelog', False, _('open changelog')),
2326 ('m', 'manifest', False, _('open manifest')),
2326 ('m', 'manifest', False, _('open manifest')),
2327 ('', 'dir', False, _('open directory manifest')),
2327 ('', 'dir', False, _('open directory manifest')),
2328 ('f', 'format', 0, _('revlog format'), _('FORMAT'))],
2328 ('f', 'format', 0, _('revlog format'), _('FORMAT'))],
2329 _('[-f FORMAT] -c|-m|FILE'),
2329 _('[-f FORMAT] -c|-m|FILE'),
2330 optionalrepo=True)
2330 optionalrepo=True)
2331 def debugindex(ui, repo, file_=None, **opts):
2331 def debugindex(ui, repo, file_=None, **opts):
2332 """dump the contents of an index file"""
2332 """dump the contents of an index file"""
2333 r = cmdutil.openrevlog(repo, 'debugindex', file_, opts)
2333 r = cmdutil.openrevlog(repo, 'debugindex', file_, opts)
2334 format = opts.get('format', 0)
2334 format = opts.get('format', 0)
2335 if format not in (0, 1):
2335 if format not in (0, 1):
2336 raise error.Abort(_("unknown format %d") % format)
2336 raise error.Abort(_("unknown format %d") % format)
2337
2337
2338 generaldelta = r.version & revlog.REVLOGGENERALDELTA
2338 generaldelta = r.version & revlog.REVLOGGENERALDELTA
2339 if generaldelta:
2339 if generaldelta:
2340 basehdr = ' delta'
2340 basehdr = ' delta'
2341 else:
2341 else:
2342 basehdr = ' base'
2342 basehdr = ' base'
2343
2343
2344 if ui.debugflag:
2344 if ui.debugflag:
2345 shortfn = hex
2345 shortfn = hex
2346 else:
2346 else:
2347 shortfn = short
2347 shortfn = short
2348
2348
2349 # There might not be anything in r, so have a sane default
2349 # There might not be anything in r, so have a sane default
2350 idlen = 12
2350 idlen = 12
2351 for i in r:
2351 for i in r:
2352 idlen = len(shortfn(r.node(i)))
2352 idlen = len(shortfn(r.node(i)))
2353 break
2353 break
2354
2354
2355 if format == 0:
2355 if format == 0:
2356 ui.write(" rev offset length " + basehdr + " linkrev"
2356 ui.write(" rev offset length " + basehdr + " linkrev"
2357 " %s %s p2\n" % ("nodeid".ljust(idlen), "p1".ljust(idlen)))
2357 " %s %s p2\n" % ("nodeid".ljust(idlen), "p1".ljust(idlen)))
2358 elif format == 1:
2358 elif format == 1:
2359 ui.write(" rev flag offset length"
2359 ui.write(" rev flag offset length"
2360 " size " + basehdr + " link p1 p2"
2360 " size " + basehdr + " link p1 p2"
2361 " %s\n" % "nodeid".rjust(idlen))
2361 " %s\n" % "nodeid".rjust(idlen))
2362
2362
2363 for i in r:
2363 for i in r:
2364 node = r.node(i)
2364 node = r.node(i)
2365 if generaldelta:
2365 if generaldelta:
2366 base = r.deltaparent(i)
2366 base = r.deltaparent(i)
2367 else:
2367 else:
2368 base = r.chainbase(i)
2368 base = r.chainbase(i)
2369 if format == 0:
2369 if format == 0:
2370 try:
2370 try:
2371 pp = r.parents(node)
2371 pp = r.parents(node)
2372 except Exception:
2372 except Exception:
2373 pp = [nullid, nullid]
2373 pp = [nullid, nullid]
2374 ui.write("% 6d % 9d % 7d % 6d % 7d %s %s %s\n" % (
2374 ui.write("% 6d % 9d % 7d % 6d % 7d %s %s %s\n" % (
2375 i, r.start(i), r.length(i), base, r.linkrev(i),
2375 i, r.start(i), r.length(i), base, r.linkrev(i),
2376 shortfn(node), shortfn(pp[0]), shortfn(pp[1])))
2376 shortfn(node), shortfn(pp[0]), shortfn(pp[1])))
2377 elif format == 1:
2377 elif format == 1:
2378 pr = r.parentrevs(i)
2378 pr = r.parentrevs(i)
2379 ui.write("% 6d %04x % 8d % 8d % 8d % 6d % 6d % 6d % 6d %s\n" % (
2379 ui.write("% 6d %04x % 8d % 8d % 8d % 6d % 6d % 6d % 6d %s\n" % (
2380 i, r.flags(i), r.start(i), r.length(i), r.rawsize(i),
2380 i, r.flags(i), r.start(i), r.length(i), r.rawsize(i),
2381 base, r.linkrev(i), pr[0], pr[1], shortfn(node)))
2381 base, r.linkrev(i), pr[0], pr[1], shortfn(node)))
2382
2382
2383 @command('debugindexdot', [], _('FILE'), optionalrepo=True)
2383 @command('debugindexdot', [], _('FILE'), optionalrepo=True)
2384 def debugindexdot(ui, repo, file_):
2384 def debugindexdot(ui, repo, file_):
2385 """dump an index DAG as a graphviz dot file"""
2385 """dump an index DAG as a graphviz dot file"""
2386 r = None
2386 r = None
2387 if repo:
2387 if repo:
2388 filelog = repo.file(file_)
2388 filelog = repo.file(file_)
2389 if len(filelog):
2389 if len(filelog):
2390 r = filelog
2390 r = filelog
2391 if not r:
2391 if not r:
2392 r = revlog.revlog(scmutil.opener(os.getcwd(), audit=False), file_)
2392 r = revlog.revlog(scmutil.opener(os.getcwd(), audit=False), file_)
2393 ui.write(("digraph G {\n"))
2393 ui.write(("digraph G {\n"))
2394 for i in r:
2394 for i in r:
2395 node = r.node(i)
2395 node = r.node(i)
2396 pp = r.parents(node)
2396 pp = r.parents(node)
2397 ui.write("\t%d -> %d\n" % (r.rev(pp[0]), i))
2397 ui.write("\t%d -> %d\n" % (r.rev(pp[0]), i))
2398 if pp[1] != nullid:
2398 if pp[1] != nullid:
2399 ui.write("\t%d -> %d\n" % (r.rev(pp[1]), i))
2399 ui.write("\t%d -> %d\n" % (r.rev(pp[1]), i))
2400 ui.write("}\n")
2400 ui.write("}\n")
2401
2401
2402 @command('debuginstall', [], '', norepo=True)
2402 @command('debuginstall', [], '', norepo=True)
2403 def debuginstall(ui):
2403 def debuginstall(ui):
2404 '''test Mercurial installation
2404 '''test Mercurial installation
2405
2405
2406 Returns 0 on success.
2406 Returns 0 on success.
2407 '''
2407 '''
2408
2408
2409 def writetemp(contents):
2409 def writetemp(contents):
2410 (fd, name) = tempfile.mkstemp(prefix="hg-debuginstall-")
2410 (fd, name) = tempfile.mkstemp(prefix="hg-debuginstall-")
2411 f = os.fdopen(fd, "wb")
2411 f = os.fdopen(fd, "wb")
2412 f.write(contents)
2412 f.write(contents)
2413 f.close()
2413 f.close()
2414 return name
2414 return name
2415
2415
2416 problems = 0
2416 problems = 0
2417
2417
2418 # encoding
2418 # encoding
2419 ui.status(_("checking encoding (%s)...\n") % encoding.encoding)
2419 ui.status(_("checking encoding (%s)...\n") % encoding.encoding)
2420 try:
2420 try:
2421 encoding.fromlocal("test")
2421 encoding.fromlocal("test")
2422 except error.Abort as inst:
2422 except error.Abort as inst:
2423 ui.write(" %s\n" % inst)
2423 ui.write(" %s\n" % inst)
2424 ui.write(_(" (check that your locale is properly set)\n"))
2424 ui.write(_(" (check that your locale is properly set)\n"))
2425 problems += 1
2425 problems += 1
2426
2426
2427 # Python
2427 # Python
2428 ui.status(_("checking Python executable (%s)\n") % sys.executable)
2428 ui.status(_("checking Python executable (%s)\n") % sys.executable)
2429 ui.status(_("checking Python version (%s)\n")
2429 ui.status(_("checking Python version (%s)\n")
2430 % ("%s.%s.%s" % sys.version_info[:3]))
2430 % ("%s.%s.%s" % sys.version_info[:3]))
2431 ui.status(_("checking Python lib (%s)...\n")
2431 ui.status(_("checking Python lib (%s)...\n")
2432 % os.path.dirname(os.__file__))
2432 % os.path.dirname(os.__file__))
2433
2433
2434 # compiled modules
2434 # compiled modules
2435 ui.status(_("checking installed modules (%s)...\n")
2435 ui.status(_("checking installed modules (%s)...\n")
2436 % os.path.dirname(__file__))
2436 % os.path.dirname(__file__))
2437 try:
2437 try:
2438 import bdiff, mpatch, base85, osutil
2438 import bdiff, mpatch, base85, osutil
2439 dir(bdiff), dir(mpatch), dir(base85), dir(osutil) # quiet pyflakes
2439 dir(bdiff), dir(mpatch), dir(base85), dir(osutil) # quiet pyflakes
2440 except Exception as inst:
2440 except Exception as inst:
2441 ui.write(" %s\n" % inst)
2441 ui.write(" %s\n" % inst)
2442 ui.write(_(" One or more extensions could not be found"))
2442 ui.write(_(" One or more extensions could not be found"))
2443 ui.write(_(" (check that you compiled the extensions)\n"))
2443 ui.write(_(" (check that you compiled the extensions)\n"))
2444 problems += 1
2444 problems += 1
2445
2445
2446 # templates
2446 # templates
2447 import templater
2447 import templater
2448 p = templater.templatepaths()
2448 p = templater.templatepaths()
2449 ui.status(_("checking templates (%s)...\n") % ' '.join(p))
2449 ui.status(_("checking templates (%s)...\n") % ' '.join(p))
2450 if p:
2450 if p:
2451 m = templater.templatepath("map-cmdline.default")
2451 m = templater.templatepath("map-cmdline.default")
2452 if m:
2452 if m:
2453 # template found, check if it is working
2453 # template found, check if it is working
2454 try:
2454 try:
2455 templater.templater(m)
2455 templater.templater(m)
2456 except Exception as inst:
2456 except Exception as inst:
2457 ui.write(" %s\n" % inst)
2457 ui.write(" %s\n" % inst)
2458 p = None
2458 p = None
2459 else:
2459 else:
2460 ui.write(_(" template 'default' not found\n"))
2460 ui.write(_(" template 'default' not found\n"))
2461 p = None
2461 p = None
2462 else:
2462 else:
2463 ui.write(_(" no template directories found\n"))
2463 ui.write(_(" no template directories found\n"))
2464 if not p:
2464 if not p:
2465 ui.write(_(" (templates seem to have been installed incorrectly)\n"))
2465 ui.write(_(" (templates seem to have been installed incorrectly)\n"))
2466 problems += 1
2466 problems += 1
2467
2467
2468 # editor
2468 # editor
2469 ui.status(_("checking commit editor...\n"))
2469 ui.status(_("checking commit editor...\n"))
2470 editor = ui.geteditor()
2470 editor = ui.geteditor()
2471 editor = util.expandpath(editor)
2471 editor = util.expandpath(editor)
2472 cmdpath = util.findexe(shlex.split(editor)[0])
2472 cmdpath = util.findexe(shlex.split(editor)[0])
2473 if not cmdpath:
2473 if not cmdpath:
2474 if editor == 'vi':
2474 if editor == 'vi':
2475 ui.write(_(" No commit editor set and can't find vi in PATH\n"))
2475 ui.write(_(" No commit editor set and can't find vi in PATH\n"))
2476 ui.write(_(" (specify a commit editor in your configuration"
2476 ui.write(_(" (specify a commit editor in your configuration"
2477 " file)\n"))
2477 " file)\n"))
2478 else:
2478 else:
2479 ui.write(_(" Can't find editor '%s' in PATH\n") % editor)
2479 ui.write(_(" Can't find editor '%s' in PATH\n") % editor)
2480 ui.write(_(" (specify a commit editor in your configuration"
2480 ui.write(_(" (specify a commit editor in your configuration"
2481 " file)\n"))
2481 " file)\n"))
2482 problems += 1
2482 problems += 1
2483
2483
2484 # check username
2484 # check username
2485 ui.status(_("checking username...\n"))
2485 ui.status(_("checking username...\n"))
2486 try:
2486 try:
2487 ui.username()
2487 ui.username()
2488 except error.Abort as e:
2488 except error.Abort as e:
2489 ui.write(" %s\n" % e)
2489 ui.write(" %s\n" % e)
2490 ui.write(_(" (specify a username in your configuration file)\n"))
2490 ui.write(_(" (specify a username in your configuration file)\n"))
2491 problems += 1
2491 problems += 1
2492
2492
2493 if not problems:
2493 if not problems:
2494 ui.status(_("no problems detected\n"))
2494 ui.status(_("no problems detected\n"))
2495 else:
2495 else:
2496 ui.write(_("%s problems detected,"
2496 ui.write(_("%s problems detected,"
2497 " please check your install!\n") % problems)
2497 " please check your install!\n") % problems)
2498
2498
2499 return problems
2499 return problems
2500
2500
2501 @command('debugknown', [], _('REPO ID...'), norepo=True)
2501 @command('debugknown', [], _('REPO ID...'), norepo=True)
2502 def debugknown(ui, repopath, *ids, **opts):
2502 def debugknown(ui, repopath, *ids, **opts):
2503 """test whether node ids are known to a repo
2503 """test whether node ids are known to a repo
2504
2504
2505 Every ID must be a full-length hex node id string. Returns a list of 0s
2505 Every ID must be a full-length hex node id string. Returns a list of 0s
2506 and 1s indicating unknown/known.
2506 and 1s indicating unknown/known.
2507 """
2507 """
2508 repo = hg.peer(ui, opts, repopath)
2508 repo = hg.peer(ui, opts, repopath)
2509 if not repo.capable('known'):
2509 if not repo.capable('known'):
2510 raise error.Abort("known() not supported by target repository")
2510 raise error.Abort("known() not supported by target repository")
2511 flags = repo.known([bin(s) for s in ids])
2511 flags = repo.known([bin(s) for s in ids])
2512 ui.write("%s\n" % ("".join([f and "1" or "0" for f in flags])))
2512 ui.write("%s\n" % ("".join([f and "1" or "0" for f in flags])))
2513
2513
2514 @command('debuglabelcomplete', [], _('LABEL...'))
2514 @command('debuglabelcomplete', [], _('LABEL...'))
2515 def debuglabelcomplete(ui, repo, *args):
2515 def debuglabelcomplete(ui, repo, *args):
2516 '''backwards compatibility with old bash completion scripts (DEPRECATED)'''
2516 '''backwards compatibility with old bash completion scripts (DEPRECATED)'''
2517 debugnamecomplete(ui, repo, *args)
2517 debugnamecomplete(ui, repo, *args)
2518
2518
2519 @command('debugmergestate', [], '')
2519 @command('debugmergestate', [], '')
2520 def debugmergestate(ui, repo, *args):
2520 def debugmergestate(ui, repo, *args):
2521 """print merge state
2521 """print merge state
2522
2522
2523 Use --verbose to print out information about whether v1 or v2 merge state
2523 Use --verbose to print out information about whether v1 or v2 merge state
2524 was chosen."""
2524 was chosen."""
2525 def printrecords(version):
2525 def printrecords(version):
2526 ui.write(('* version %s records\n') % version)
2526 ui.write(('* version %s records\n') % version)
2527 if version == 1:
2527 if version == 1:
2528 records = v1records
2528 records = v1records
2529 else:
2529 else:
2530 records = v2records
2530 records = v2records
2531
2531
2532 for rtype, record in records:
2532 for rtype, record in records:
2533 # pretty print some record types
2533 # pretty print some record types
2534 if rtype == 'L':
2534 if rtype == 'L':
2535 ui.write(('local: %s\n') % record)
2535 ui.write(('local: %s\n') % record)
2536 elif rtype == 'O':
2536 elif rtype == 'O':
2537 ui.write(('other: %s\n') % record)
2537 ui.write(('other: %s\n') % record)
2538 elif rtype == 'm':
2538 elif rtype == 'm':
2539 driver, mdstate = record.split('\0', 1)
2539 driver, mdstate = record.split('\0', 1)
2540 ui.write(('merge driver: %s (state "%s")\n')
2540 ui.write(('merge driver: %s (state "%s")\n')
2541 % (driver, mdstate))
2541 % (driver, mdstate))
2542 elif rtype in 'FD':
2542 elif rtype in 'FD':
2543 r = record.split('\0')
2543 r = record.split('\0')
2544 f, state, hash, lfile, afile, anode, ofile = r[0:7]
2544 f, state, hash, lfile, afile, anode, ofile = r[0:7]
2545 if version == 1:
2545 if version == 1:
2546 onode = 'not stored in v1 format'
2546 onode = 'not stored in v1 format'
2547 flags = r[7]
2547 flags = r[7]
2548 else:
2548 else:
2549 onode, flags = r[7:9]
2549 onode, flags = r[7:9]
2550 ui.write(('file: %s (state "%s", hash %s)\n')
2550 ui.write(('file: %s (record type "%s", state "%s", hash %s)\n')
2551 % (f, state, hash))
2551 % (f, rtype, state, hash))
2552 ui.write((' local path: %s (flags "%s")\n') % (lfile, flags))
2552 ui.write((' local path: %s (flags "%s")\n') % (lfile, flags))
2553 ui.write((' ancestor path: %s (node %s)\n') % (afile, anode))
2553 ui.write((' ancestor path: %s (node %s)\n') % (afile, anode))
2554 ui.write((' other path: %s (node %s)\n') % (ofile, onode))
2554 ui.write((' other path: %s (node %s)\n') % (ofile, onode))
2555 else:
2555 else:
2556 ui.write(('unrecognized entry: %s\t%s\n')
2556 ui.write(('unrecognized entry: %s\t%s\n')
2557 % (rtype, record.replace('\0', '\t')))
2557 % (rtype, record.replace('\0', '\t')))
2558
2558
2559 ms = mergemod.mergestate(repo)
2559 ms = mergemod.mergestate(repo)
2560
2560
2561 # sort so that reasonable information is on top
2561 # sort so that reasonable information is on top
2562 v1records = ms._readrecordsv1()
2562 v1records = ms._readrecordsv1()
2563 v2records = ms._readrecordsv2()
2563 v2records = ms._readrecordsv2()
2564 order = 'LOm'
2564 order = 'LOm'
2565 def key(r):
2565 def key(r):
2566 idx = order.find(r[0])
2566 idx = order.find(r[0])
2567 if idx == -1:
2567 if idx == -1:
2568 return (1, r[1])
2568 return (1, r[1])
2569 else:
2569 else:
2570 return (0, idx)
2570 return (0, idx)
2571 v1records.sort(key=key)
2571 v1records.sort(key=key)
2572 v2records.sort(key=key)
2572 v2records.sort(key=key)
2573
2573
2574 if not v1records and not v2records:
2574 if not v1records and not v2records:
2575 ui.write(('no merge state found\n'))
2575 ui.write(('no merge state found\n'))
2576 elif not v2records:
2576 elif not v2records:
2577 ui.note(('no version 2 merge state\n'))
2577 ui.note(('no version 2 merge state\n'))
2578 printrecords(1)
2578 printrecords(1)
2579 elif ms._v1v2match(v1records, v2records):
2579 elif ms._v1v2match(v1records, v2records):
2580 ui.note(('v1 and v2 states match: using v2\n'))
2580 ui.note(('v1 and v2 states match: using v2\n'))
2581 printrecords(2)
2581 printrecords(2)
2582 else:
2582 else:
2583 ui.note(('v1 and v2 states mismatch: using v1\n'))
2583 ui.note(('v1 and v2 states mismatch: using v1\n'))
2584 printrecords(1)
2584 printrecords(1)
2585 if ui.verbose:
2585 if ui.verbose:
2586 printrecords(2)
2586 printrecords(2)
2587
2587
2588 @command('debugnamecomplete', [], _('NAME...'))
2588 @command('debugnamecomplete', [], _('NAME...'))
2589 def debugnamecomplete(ui, repo, *args):
2589 def debugnamecomplete(ui, repo, *args):
2590 '''complete "names" - tags, open branch names, bookmark names'''
2590 '''complete "names" - tags, open branch names, bookmark names'''
2591
2591
2592 names = set()
2592 names = set()
2593 # since we previously only listed open branches, we will handle that
2593 # since we previously only listed open branches, we will handle that
2594 # specially (after this for loop)
2594 # specially (after this for loop)
2595 for name, ns in repo.names.iteritems():
2595 for name, ns in repo.names.iteritems():
2596 if name != 'branches':
2596 if name != 'branches':
2597 names.update(ns.listnames(repo))
2597 names.update(ns.listnames(repo))
2598 names.update(tag for (tag, heads, tip, closed)
2598 names.update(tag for (tag, heads, tip, closed)
2599 in repo.branchmap().iterbranches() if not closed)
2599 in repo.branchmap().iterbranches() if not closed)
2600 completions = set()
2600 completions = set()
2601 if not args:
2601 if not args:
2602 args = ['']
2602 args = ['']
2603 for a in args:
2603 for a in args:
2604 completions.update(n for n in names if n.startswith(a))
2604 completions.update(n for n in names if n.startswith(a))
2605 ui.write('\n'.join(sorted(completions)))
2605 ui.write('\n'.join(sorted(completions)))
2606 ui.write('\n')
2606 ui.write('\n')
2607
2607
2608 @command('debuglocks',
2608 @command('debuglocks',
2609 [('L', 'force-lock', None, _('free the store lock (DANGEROUS)')),
2609 [('L', 'force-lock', None, _('free the store lock (DANGEROUS)')),
2610 ('W', 'force-wlock', None,
2610 ('W', 'force-wlock', None,
2611 _('free the working state lock (DANGEROUS)'))],
2611 _('free the working state lock (DANGEROUS)'))],
2612 _('[OPTION]...'))
2612 _('[OPTION]...'))
2613 def debuglocks(ui, repo, **opts):
2613 def debuglocks(ui, repo, **opts):
2614 """show or modify state of locks
2614 """show or modify state of locks
2615
2615
2616 By default, this command will show which locks are held. This
2616 By default, this command will show which locks are held. This
2617 includes the user and process holding the lock, the amount of time
2617 includes the user and process holding the lock, the amount of time
2618 the lock has been held, and the machine name where the process is
2618 the lock has been held, and the machine name where the process is
2619 running if it's not local.
2619 running if it's not local.
2620
2620
2621 Locks protect the integrity of Mercurial's data, so should be
2621 Locks protect the integrity of Mercurial's data, so should be
2622 treated with care. System crashes or other interruptions may cause
2622 treated with care. System crashes or other interruptions may cause
2623 locks to not be properly released, though Mercurial will usually
2623 locks to not be properly released, though Mercurial will usually
2624 detect and remove such stale locks automatically.
2624 detect and remove such stale locks automatically.
2625
2625
2626 However, detecting stale locks may not always be possible (for
2626 However, detecting stale locks may not always be possible (for
2627 instance, on a shared filesystem). Removing locks may also be
2627 instance, on a shared filesystem). Removing locks may also be
2628 blocked by filesystem permissions.
2628 blocked by filesystem permissions.
2629
2629
2630 Returns 0 if no locks are held.
2630 Returns 0 if no locks are held.
2631
2631
2632 """
2632 """
2633
2633
2634 if opts.get('force_lock'):
2634 if opts.get('force_lock'):
2635 repo.svfs.unlink('lock')
2635 repo.svfs.unlink('lock')
2636 if opts.get('force_wlock'):
2636 if opts.get('force_wlock'):
2637 repo.vfs.unlink('wlock')
2637 repo.vfs.unlink('wlock')
2638 if opts.get('force_lock') or opts.get('force_lock'):
2638 if opts.get('force_lock') or opts.get('force_lock'):
2639 return 0
2639 return 0
2640
2640
2641 now = time.time()
2641 now = time.time()
2642 held = 0
2642 held = 0
2643
2643
2644 def report(vfs, name, method):
2644 def report(vfs, name, method):
2645 # this causes stale locks to get reaped for more accurate reporting
2645 # this causes stale locks to get reaped for more accurate reporting
2646 try:
2646 try:
2647 l = method(False)
2647 l = method(False)
2648 except error.LockHeld:
2648 except error.LockHeld:
2649 l = None
2649 l = None
2650
2650
2651 if l:
2651 if l:
2652 l.release()
2652 l.release()
2653 else:
2653 else:
2654 try:
2654 try:
2655 stat = vfs.lstat(name)
2655 stat = vfs.lstat(name)
2656 age = now - stat.st_mtime
2656 age = now - stat.st_mtime
2657 user = util.username(stat.st_uid)
2657 user = util.username(stat.st_uid)
2658 locker = vfs.readlock(name)
2658 locker = vfs.readlock(name)
2659 if ":" in locker:
2659 if ":" in locker:
2660 host, pid = locker.split(':')
2660 host, pid = locker.split(':')
2661 if host == socket.gethostname():
2661 if host == socket.gethostname():
2662 locker = 'user %s, process %s' % (user, pid)
2662 locker = 'user %s, process %s' % (user, pid)
2663 else:
2663 else:
2664 locker = 'user %s, process %s, host %s' \
2664 locker = 'user %s, process %s, host %s' \
2665 % (user, pid, host)
2665 % (user, pid, host)
2666 ui.write("%-6s %s (%ds)\n" % (name + ":", locker, age))
2666 ui.write("%-6s %s (%ds)\n" % (name + ":", locker, age))
2667 return 1
2667 return 1
2668 except OSError as e:
2668 except OSError as e:
2669 if e.errno != errno.ENOENT:
2669 if e.errno != errno.ENOENT:
2670 raise
2670 raise
2671
2671
2672 ui.write("%-6s free\n" % (name + ":"))
2672 ui.write("%-6s free\n" % (name + ":"))
2673 return 0
2673 return 0
2674
2674
2675 held += report(repo.svfs, "lock", repo.lock)
2675 held += report(repo.svfs, "lock", repo.lock)
2676 held += report(repo.vfs, "wlock", repo.wlock)
2676 held += report(repo.vfs, "wlock", repo.wlock)
2677
2677
2678 return held
2678 return held
2679
2679
2680 @command('debugobsolete',
2680 @command('debugobsolete',
2681 [('', 'flags', 0, _('markers flag')),
2681 [('', 'flags', 0, _('markers flag')),
2682 ('', 'record-parents', False,
2682 ('', 'record-parents', False,
2683 _('record parent information for the precursor')),
2683 _('record parent information for the precursor')),
2684 ('r', 'rev', [], _('display markers relevant to REV')),
2684 ('r', 'rev', [], _('display markers relevant to REV')),
2685 ] + commitopts2,
2685 ] + commitopts2,
2686 _('[OBSOLETED [REPLACEMENT ...]]'))
2686 _('[OBSOLETED [REPLACEMENT ...]]'))
2687 def debugobsolete(ui, repo, precursor=None, *successors, **opts):
2687 def debugobsolete(ui, repo, precursor=None, *successors, **opts):
2688 """create arbitrary obsolete marker
2688 """create arbitrary obsolete marker
2689
2689
2690 With no arguments, displays the list of obsolescence markers."""
2690 With no arguments, displays the list of obsolescence markers."""
2691
2691
2692 def parsenodeid(s):
2692 def parsenodeid(s):
2693 try:
2693 try:
2694 # We do not use revsingle/revrange functions here to accept
2694 # We do not use revsingle/revrange functions here to accept
2695 # arbitrary node identifiers, possibly not present in the
2695 # arbitrary node identifiers, possibly not present in the
2696 # local repository.
2696 # local repository.
2697 n = bin(s)
2697 n = bin(s)
2698 if len(n) != len(nullid):
2698 if len(n) != len(nullid):
2699 raise TypeError()
2699 raise TypeError()
2700 return n
2700 return n
2701 except TypeError:
2701 except TypeError:
2702 raise error.Abort('changeset references must be full hexadecimal '
2702 raise error.Abort('changeset references must be full hexadecimal '
2703 'node identifiers')
2703 'node identifiers')
2704
2704
2705 if precursor is not None:
2705 if precursor is not None:
2706 if opts['rev']:
2706 if opts['rev']:
2707 raise error.Abort('cannot select revision when creating marker')
2707 raise error.Abort('cannot select revision when creating marker')
2708 metadata = {}
2708 metadata = {}
2709 metadata['user'] = opts['user'] or ui.username()
2709 metadata['user'] = opts['user'] or ui.username()
2710 succs = tuple(parsenodeid(succ) for succ in successors)
2710 succs = tuple(parsenodeid(succ) for succ in successors)
2711 l = repo.lock()
2711 l = repo.lock()
2712 try:
2712 try:
2713 tr = repo.transaction('debugobsolete')
2713 tr = repo.transaction('debugobsolete')
2714 try:
2714 try:
2715 date = opts.get('date')
2715 date = opts.get('date')
2716 if date:
2716 if date:
2717 date = util.parsedate(date)
2717 date = util.parsedate(date)
2718 else:
2718 else:
2719 date = None
2719 date = None
2720 prec = parsenodeid(precursor)
2720 prec = parsenodeid(precursor)
2721 parents = None
2721 parents = None
2722 if opts['record_parents']:
2722 if opts['record_parents']:
2723 if prec not in repo.unfiltered():
2723 if prec not in repo.unfiltered():
2724 raise error.Abort('cannot used --record-parents on '
2724 raise error.Abort('cannot used --record-parents on '
2725 'unknown changesets')
2725 'unknown changesets')
2726 parents = repo.unfiltered()[prec].parents()
2726 parents = repo.unfiltered()[prec].parents()
2727 parents = tuple(p.node() for p in parents)
2727 parents = tuple(p.node() for p in parents)
2728 repo.obsstore.create(tr, prec, succs, opts['flags'],
2728 repo.obsstore.create(tr, prec, succs, opts['flags'],
2729 parents=parents, date=date,
2729 parents=parents, date=date,
2730 metadata=metadata)
2730 metadata=metadata)
2731 tr.close()
2731 tr.close()
2732 except ValueError as exc:
2732 except ValueError as exc:
2733 raise error.Abort(_('bad obsmarker input: %s') % exc)
2733 raise error.Abort(_('bad obsmarker input: %s') % exc)
2734 finally:
2734 finally:
2735 tr.release()
2735 tr.release()
2736 finally:
2736 finally:
2737 l.release()
2737 l.release()
2738 else:
2738 else:
2739 if opts['rev']:
2739 if opts['rev']:
2740 revs = scmutil.revrange(repo, opts['rev'])
2740 revs = scmutil.revrange(repo, opts['rev'])
2741 nodes = [repo[r].node() for r in revs]
2741 nodes = [repo[r].node() for r in revs]
2742 markers = list(obsolete.getmarkers(repo, nodes=nodes))
2742 markers = list(obsolete.getmarkers(repo, nodes=nodes))
2743 markers.sort(key=lambda x: x._data)
2743 markers.sort(key=lambda x: x._data)
2744 else:
2744 else:
2745 markers = obsolete.getmarkers(repo)
2745 markers = obsolete.getmarkers(repo)
2746
2746
2747 for m in markers:
2747 for m in markers:
2748 cmdutil.showmarker(ui, m)
2748 cmdutil.showmarker(ui, m)
2749
2749
2750 @command('debugpathcomplete',
2750 @command('debugpathcomplete',
2751 [('f', 'full', None, _('complete an entire path')),
2751 [('f', 'full', None, _('complete an entire path')),
2752 ('n', 'normal', None, _('show only normal files')),
2752 ('n', 'normal', None, _('show only normal files')),
2753 ('a', 'added', None, _('show only added files')),
2753 ('a', 'added', None, _('show only added files')),
2754 ('r', 'removed', None, _('show only removed files'))],
2754 ('r', 'removed', None, _('show only removed files'))],
2755 _('FILESPEC...'))
2755 _('FILESPEC...'))
2756 def debugpathcomplete(ui, repo, *specs, **opts):
2756 def debugpathcomplete(ui, repo, *specs, **opts):
2757 '''complete part or all of a tracked path
2757 '''complete part or all of a tracked path
2758
2758
2759 This command supports shells that offer path name completion. It
2759 This command supports shells that offer path name completion. It
2760 currently completes only files already known to the dirstate.
2760 currently completes only files already known to the dirstate.
2761
2761
2762 Completion extends only to the next path segment unless
2762 Completion extends only to the next path segment unless
2763 --full is specified, in which case entire paths are used.'''
2763 --full is specified, in which case entire paths are used.'''
2764
2764
2765 def complete(path, acceptable):
2765 def complete(path, acceptable):
2766 dirstate = repo.dirstate
2766 dirstate = repo.dirstate
2767 spec = os.path.normpath(os.path.join(os.getcwd(), path))
2767 spec = os.path.normpath(os.path.join(os.getcwd(), path))
2768 rootdir = repo.root + os.sep
2768 rootdir = repo.root + os.sep
2769 if spec != repo.root and not spec.startswith(rootdir):
2769 if spec != repo.root and not spec.startswith(rootdir):
2770 return [], []
2770 return [], []
2771 if os.path.isdir(spec):
2771 if os.path.isdir(spec):
2772 spec += '/'
2772 spec += '/'
2773 spec = spec[len(rootdir):]
2773 spec = spec[len(rootdir):]
2774 fixpaths = os.sep != '/'
2774 fixpaths = os.sep != '/'
2775 if fixpaths:
2775 if fixpaths:
2776 spec = spec.replace(os.sep, '/')
2776 spec = spec.replace(os.sep, '/')
2777 speclen = len(spec)
2777 speclen = len(spec)
2778 fullpaths = opts['full']
2778 fullpaths = opts['full']
2779 files, dirs = set(), set()
2779 files, dirs = set(), set()
2780 adddir, addfile = dirs.add, files.add
2780 adddir, addfile = dirs.add, files.add
2781 for f, st in dirstate.iteritems():
2781 for f, st in dirstate.iteritems():
2782 if f.startswith(spec) and st[0] in acceptable:
2782 if f.startswith(spec) and st[0] in acceptable:
2783 if fixpaths:
2783 if fixpaths:
2784 f = f.replace('/', os.sep)
2784 f = f.replace('/', os.sep)
2785 if fullpaths:
2785 if fullpaths:
2786 addfile(f)
2786 addfile(f)
2787 continue
2787 continue
2788 s = f.find(os.sep, speclen)
2788 s = f.find(os.sep, speclen)
2789 if s >= 0:
2789 if s >= 0:
2790 adddir(f[:s])
2790 adddir(f[:s])
2791 else:
2791 else:
2792 addfile(f)
2792 addfile(f)
2793 return files, dirs
2793 return files, dirs
2794
2794
2795 acceptable = ''
2795 acceptable = ''
2796 if opts['normal']:
2796 if opts['normal']:
2797 acceptable += 'nm'
2797 acceptable += 'nm'
2798 if opts['added']:
2798 if opts['added']:
2799 acceptable += 'a'
2799 acceptable += 'a'
2800 if opts['removed']:
2800 if opts['removed']:
2801 acceptable += 'r'
2801 acceptable += 'r'
2802 cwd = repo.getcwd()
2802 cwd = repo.getcwd()
2803 if not specs:
2803 if not specs:
2804 specs = ['.']
2804 specs = ['.']
2805
2805
2806 files, dirs = set(), set()
2806 files, dirs = set(), set()
2807 for spec in specs:
2807 for spec in specs:
2808 f, d = complete(spec, acceptable or 'nmar')
2808 f, d = complete(spec, acceptable or 'nmar')
2809 files.update(f)
2809 files.update(f)
2810 dirs.update(d)
2810 dirs.update(d)
2811 files.update(dirs)
2811 files.update(dirs)
2812 ui.write('\n'.join(repo.pathto(p, cwd) for p in sorted(files)))
2812 ui.write('\n'.join(repo.pathto(p, cwd) for p in sorted(files)))
2813 ui.write('\n')
2813 ui.write('\n')
2814
2814
2815 @command('debugpushkey', [], _('REPO NAMESPACE [KEY OLD NEW]'), norepo=True)
2815 @command('debugpushkey', [], _('REPO NAMESPACE [KEY OLD NEW]'), norepo=True)
2816 def debugpushkey(ui, repopath, namespace, *keyinfo, **opts):
2816 def debugpushkey(ui, repopath, namespace, *keyinfo, **opts):
2817 '''access the pushkey key/value protocol
2817 '''access the pushkey key/value protocol
2818
2818
2819 With two args, list the keys in the given namespace.
2819 With two args, list the keys in the given namespace.
2820
2820
2821 With five args, set a key to new if it currently is set to old.
2821 With five args, set a key to new if it currently is set to old.
2822 Reports success or failure.
2822 Reports success or failure.
2823 '''
2823 '''
2824
2824
2825 target = hg.peer(ui, {}, repopath)
2825 target = hg.peer(ui, {}, repopath)
2826 if keyinfo:
2826 if keyinfo:
2827 key, old, new = keyinfo
2827 key, old, new = keyinfo
2828 r = target.pushkey(namespace, key, old, new)
2828 r = target.pushkey(namespace, key, old, new)
2829 ui.status(str(r) + '\n')
2829 ui.status(str(r) + '\n')
2830 return not r
2830 return not r
2831 else:
2831 else:
2832 for k, v in sorted(target.listkeys(namespace).iteritems()):
2832 for k, v in sorted(target.listkeys(namespace).iteritems()):
2833 ui.write("%s\t%s\n" % (k.encode('string-escape'),
2833 ui.write("%s\t%s\n" % (k.encode('string-escape'),
2834 v.encode('string-escape')))
2834 v.encode('string-escape')))
2835
2835
2836 @command('debugpvec', [], _('A B'))
2836 @command('debugpvec', [], _('A B'))
2837 def debugpvec(ui, repo, a, b=None):
2837 def debugpvec(ui, repo, a, b=None):
2838 ca = scmutil.revsingle(repo, a)
2838 ca = scmutil.revsingle(repo, a)
2839 cb = scmutil.revsingle(repo, b)
2839 cb = scmutil.revsingle(repo, b)
2840 pa = pvec.ctxpvec(ca)
2840 pa = pvec.ctxpvec(ca)
2841 pb = pvec.ctxpvec(cb)
2841 pb = pvec.ctxpvec(cb)
2842 if pa == pb:
2842 if pa == pb:
2843 rel = "="
2843 rel = "="
2844 elif pa > pb:
2844 elif pa > pb:
2845 rel = ">"
2845 rel = ">"
2846 elif pa < pb:
2846 elif pa < pb:
2847 rel = "<"
2847 rel = "<"
2848 elif pa | pb:
2848 elif pa | pb:
2849 rel = "|"
2849 rel = "|"
2850 ui.write(_("a: %s\n") % pa)
2850 ui.write(_("a: %s\n") % pa)
2851 ui.write(_("b: %s\n") % pb)
2851 ui.write(_("b: %s\n") % pb)
2852 ui.write(_("depth(a): %d depth(b): %d\n") % (pa._depth, pb._depth))
2852 ui.write(_("depth(a): %d depth(b): %d\n") % (pa._depth, pb._depth))
2853 ui.write(_("delta: %d hdist: %d distance: %d relation: %s\n") %
2853 ui.write(_("delta: %d hdist: %d distance: %d relation: %s\n") %
2854 (abs(pa._depth - pb._depth), pvec._hamming(pa._vec, pb._vec),
2854 (abs(pa._depth - pb._depth), pvec._hamming(pa._vec, pb._vec),
2855 pa.distance(pb), rel))
2855 pa.distance(pb), rel))
2856
2856
2857 @command('debugrebuilddirstate|debugrebuildstate',
2857 @command('debugrebuilddirstate|debugrebuildstate',
2858 [('r', 'rev', '', _('revision to rebuild to'), _('REV')),
2858 [('r', 'rev', '', _('revision to rebuild to'), _('REV')),
2859 ('', 'minimal', None, _('only rebuild files that are inconsistent with '
2859 ('', 'minimal', None, _('only rebuild files that are inconsistent with '
2860 'the working copy parent')),
2860 'the working copy parent')),
2861 ],
2861 ],
2862 _('[-r REV]'))
2862 _('[-r REV]'))
2863 def debugrebuilddirstate(ui, repo, rev, **opts):
2863 def debugrebuilddirstate(ui, repo, rev, **opts):
2864 """rebuild the dirstate as it would look like for the given revision
2864 """rebuild the dirstate as it would look like for the given revision
2865
2865
2866 If no revision is specified the first current parent will be used.
2866 If no revision is specified the first current parent will be used.
2867
2867
2868 The dirstate will be set to the files of the given revision.
2868 The dirstate will be set to the files of the given revision.
2869 The actual working directory content or existing dirstate
2869 The actual working directory content or existing dirstate
2870 information such as adds or removes is not considered.
2870 information such as adds or removes is not considered.
2871
2871
2872 ``minimal`` will only rebuild the dirstate status for files that claim to be
2872 ``minimal`` will only rebuild the dirstate status for files that claim to be
2873 tracked but are not in the parent manifest, or that exist in the parent
2873 tracked but are not in the parent manifest, or that exist in the parent
2874 manifest but are not in the dirstate. It will not change adds, removes, or
2874 manifest but are not in the dirstate. It will not change adds, removes, or
2875 modified files that are in the working copy parent.
2875 modified files that are in the working copy parent.
2876
2876
2877 One use of this command is to make the next :hg:`status` invocation
2877 One use of this command is to make the next :hg:`status` invocation
2878 check the actual file content.
2878 check the actual file content.
2879 """
2879 """
2880 ctx = scmutil.revsingle(repo, rev)
2880 ctx = scmutil.revsingle(repo, rev)
2881 wlock = repo.wlock()
2881 wlock = repo.wlock()
2882 try:
2882 try:
2883 dirstate = repo.dirstate
2883 dirstate = repo.dirstate
2884
2884
2885 # See command doc for what minimal does.
2885 # See command doc for what minimal does.
2886 if opts.get('minimal'):
2886 if opts.get('minimal'):
2887 dirstatefiles = set(dirstate)
2887 dirstatefiles = set(dirstate)
2888 ctxfiles = set(ctx.manifest().keys())
2888 ctxfiles = set(ctx.manifest().keys())
2889 for file in (dirstatefiles | ctxfiles):
2889 for file in (dirstatefiles | ctxfiles):
2890 indirstate = file in dirstatefiles
2890 indirstate = file in dirstatefiles
2891 inctx = file in ctxfiles
2891 inctx = file in ctxfiles
2892
2892
2893 if indirstate and not inctx and dirstate[file] != 'a':
2893 if indirstate and not inctx and dirstate[file] != 'a':
2894 dirstate.drop(file)
2894 dirstate.drop(file)
2895 elif inctx and not indirstate:
2895 elif inctx and not indirstate:
2896 dirstate.normallookup(file)
2896 dirstate.normallookup(file)
2897 else:
2897 else:
2898 dirstate.rebuild(ctx.node(), ctx.manifest())
2898 dirstate.rebuild(ctx.node(), ctx.manifest())
2899 finally:
2899 finally:
2900 wlock.release()
2900 wlock.release()
2901
2901
2902 @command('debugrebuildfncache', [], '')
2902 @command('debugrebuildfncache', [], '')
2903 def debugrebuildfncache(ui, repo):
2903 def debugrebuildfncache(ui, repo):
2904 """rebuild the fncache file"""
2904 """rebuild the fncache file"""
2905 repair.rebuildfncache(ui, repo)
2905 repair.rebuildfncache(ui, repo)
2906
2906
2907 @command('debugrename',
2907 @command('debugrename',
2908 [('r', 'rev', '', _('revision to debug'), _('REV'))],
2908 [('r', 'rev', '', _('revision to debug'), _('REV'))],
2909 _('[-r REV] FILE'))
2909 _('[-r REV] FILE'))
2910 def debugrename(ui, repo, file1, *pats, **opts):
2910 def debugrename(ui, repo, file1, *pats, **opts):
2911 """dump rename information"""
2911 """dump rename information"""
2912
2912
2913 ctx = scmutil.revsingle(repo, opts.get('rev'))
2913 ctx = scmutil.revsingle(repo, opts.get('rev'))
2914 m = scmutil.match(ctx, (file1,) + pats, opts)
2914 m = scmutil.match(ctx, (file1,) + pats, opts)
2915 for abs in ctx.walk(m):
2915 for abs in ctx.walk(m):
2916 fctx = ctx[abs]
2916 fctx = ctx[abs]
2917 o = fctx.filelog().renamed(fctx.filenode())
2917 o = fctx.filelog().renamed(fctx.filenode())
2918 rel = m.rel(abs)
2918 rel = m.rel(abs)
2919 if o:
2919 if o:
2920 ui.write(_("%s renamed from %s:%s\n") % (rel, o[0], hex(o[1])))
2920 ui.write(_("%s renamed from %s:%s\n") % (rel, o[0], hex(o[1])))
2921 else:
2921 else:
2922 ui.write(_("%s not renamed\n") % rel)
2922 ui.write(_("%s not renamed\n") % rel)
2923
2923
2924 @command('debugrevlog',
2924 @command('debugrevlog',
2925 [('c', 'changelog', False, _('open changelog')),
2925 [('c', 'changelog', False, _('open changelog')),
2926 ('m', 'manifest', False, _('open manifest')),
2926 ('m', 'manifest', False, _('open manifest')),
2927 ('', 'dir', False, _('open directory manifest')),
2927 ('', 'dir', False, _('open directory manifest')),
2928 ('d', 'dump', False, _('dump index data'))],
2928 ('d', 'dump', False, _('dump index data'))],
2929 _('-c|-m|FILE'),
2929 _('-c|-m|FILE'),
2930 optionalrepo=True)
2930 optionalrepo=True)
2931 def debugrevlog(ui, repo, file_=None, **opts):
2931 def debugrevlog(ui, repo, file_=None, **opts):
2932 """show data and statistics about a revlog"""
2932 """show data and statistics about a revlog"""
2933 r = cmdutil.openrevlog(repo, 'debugrevlog', file_, opts)
2933 r = cmdutil.openrevlog(repo, 'debugrevlog', file_, opts)
2934
2934
2935 if opts.get("dump"):
2935 if opts.get("dump"):
2936 numrevs = len(r)
2936 numrevs = len(r)
2937 ui.write("# rev p1rev p2rev start end deltastart base p1 p2"
2937 ui.write("# rev p1rev p2rev start end deltastart base p1 p2"
2938 " rawsize totalsize compression heads chainlen\n")
2938 " rawsize totalsize compression heads chainlen\n")
2939 ts = 0
2939 ts = 0
2940 heads = set()
2940 heads = set()
2941
2941
2942 for rev in xrange(numrevs):
2942 for rev in xrange(numrevs):
2943 dbase = r.deltaparent(rev)
2943 dbase = r.deltaparent(rev)
2944 if dbase == -1:
2944 if dbase == -1:
2945 dbase = rev
2945 dbase = rev
2946 cbase = r.chainbase(rev)
2946 cbase = r.chainbase(rev)
2947 clen = r.chainlen(rev)
2947 clen = r.chainlen(rev)
2948 p1, p2 = r.parentrevs(rev)
2948 p1, p2 = r.parentrevs(rev)
2949 rs = r.rawsize(rev)
2949 rs = r.rawsize(rev)
2950 ts = ts + rs
2950 ts = ts + rs
2951 heads -= set(r.parentrevs(rev))
2951 heads -= set(r.parentrevs(rev))
2952 heads.add(rev)
2952 heads.add(rev)
2953 ui.write("%5d %5d %5d %5d %5d %10d %4d %4d %4d %7d %9d "
2953 ui.write("%5d %5d %5d %5d %5d %10d %4d %4d %4d %7d %9d "
2954 "%11d %5d %8d\n" %
2954 "%11d %5d %8d\n" %
2955 (rev, p1, p2, r.start(rev), r.end(rev),
2955 (rev, p1, p2, r.start(rev), r.end(rev),
2956 r.start(dbase), r.start(cbase),
2956 r.start(dbase), r.start(cbase),
2957 r.start(p1), r.start(p2),
2957 r.start(p1), r.start(p2),
2958 rs, ts, ts / r.end(rev), len(heads), clen))
2958 rs, ts, ts / r.end(rev), len(heads), clen))
2959 return 0
2959 return 0
2960
2960
2961 v = r.version
2961 v = r.version
2962 format = v & 0xFFFF
2962 format = v & 0xFFFF
2963 flags = []
2963 flags = []
2964 gdelta = False
2964 gdelta = False
2965 if v & revlog.REVLOGNGINLINEDATA:
2965 if v & revlog.REVLOGNGINLINEDATA:
2966 flags.append('inline')
2966 flags.append('inline')
2967 if v & revlog.REVLOGGENERALDELTA:
2967 if v & revlog.REVLOGGENERALDELTA:
2968 gdelta = True
2968 gdelta = True
2969 flags.append('generaldelta')
2969 flags.append('generaldelta')
2970 if not flags:
2970 if not flags:
2971 flags = ['(none)']
2971 flags = ['(none)']
2972
2972
2973 nummerges = 0
2973 nummerges = 0
2974 numfull = 0
2974 numfull = 0
2975 numprev = 0
2975 numprev = 0
2976 nump1 = 0
2976 nump1 = 0
2977 nump2 = 0
2977 nump2 = 0
2978 numother = 0
2978 numother = 0
2979 nump1prev = 0
2979 nump1prev = 0
2980 nump2prev = 0
2980 nump2prev = 0
2981 chainlengths = []
2981 chainlengths = []
2982
2982
2983 datasize = [None, 0, 0L]
2983 datasize = [None, 0, 0L]
2984 fullsize = [None, 0, 0L]
2984 fullsize = [None, 0, 0L]
2985 deltasize = [None, 0, 0L]
2985 deltasize = [None, 0, 0L]
2986
2986
2987 def addsize(size, l):
2987 def addsize(size, l):
2988 if l[0] is None or size < l[0]:
2988 if l[0] is None or size < l[0]:
2989 l[0] = size
2989 l[0] = size
2990 if size > l[1]:
2990 if size > l[1]:
2991 l[1] = size
2991 l[1] = size
2992 l[2] += size
2992 l[2] += size
2993
2993
2994 numrevs = len(r)
2994 numrevs = len(r)
2995 for rev in xrange(numrevs):
2995 for rev in xrange(numrevs):
2996 p1, p2 = r.parentrevs(rev)
2996 p1, p2 = r.parentrevs(rev)
2997 delta = r.deltaparent(rev)
2997 delta = r.deltaparent(rev)
2998 if format > 0:
2998 if format > 0:
2999 addsize(r.rawsize(rev), datasize)
2999 addsize(r.rawsize(rev), datasize)
3000 if p2 != nullrev:
3000 if p2 != nullrev:
3001 nummerges += 1
3001 nummerges += 1
3002 size = r.length(rev)
3002 size = r.length(rev)
3003 if delta == nullrev:
3003 if delta == nullrev:
3004 chainlengths.append(0)
3004 chainlengths.append(0)
3005 numfull += 1
3005 numfull += 1
3006 addsize(size, fullsize)
3006 addsize(size, fullsize)
3007 else:
3007 else:
3008 chainlengths.append(chainlengths[delta] + 1)
3008 chainlengths.append(chainlengths[delta] + 1)
3009 addsize(size, deltasize)
3009 addsize(size, deltasize)
3010 if delta == rev - 1:
3010 if delta == rev - 1:
3011 numprev += 1
3011 numprev += 1
3012 if delta == p1:
3012 if delta == p1:
3013 nump1prev += 1
3013 nump1prev += 1
3014 elif delta == p2:
3014 elif delta == p2:
3015 nump2prev += 1
3015 nump2prev += 1
3016 elif delta == p1:
3016 elif delta == p1:
3017 nump1 += 1
3017 nump1 += 1
3018 elif delta == p2:
3018 elif delta == p2:
3019 nump2 += 1
3019 nump2 += 1
3020 elif delta != nullrev:
3020 elif delta != nullrev:
3021 numother += 1
3021 numother += 1
3022
3022
3023 # Adjust size min value for empty cases
3023 # Adjust size min value for empty cases
3024 for size in (datasize, fullsize, deltasize):
3024 for size in (datasize, fullsize, deltasize):
3025 if size[0] is None:
3025 if size[0] is None:
3026 size[0] = 0
3026 size[0] = 0
3027
3027
3028 numdeltas = numrevs - numfull
3028 numdeltas = numrevs - numfull
3029 numoprev = numprev - nump1prev - nump2prev
3029 numoprev = numprev - nump1prev - nump2prev
3030 totalrawsize = datasize[2]
3030 totalrawsize = datasize[2]
3031 datasize[2] /= numrevs
3031 datasize[2] /= numrevs
3032 fulltotal = fullsize[2]
3032 fulltotal = fullsize[2]
3033 fullsize[2] /= numfull
3033 fullsize[2] /= numfull
3034 deltatotal = deltasize[2]
3034 deltatotal = deltasize[2]
3035 if numrevs - numfull > 0:
3035 if numrevs - numfull > 0:
3036 deltasize[2] /= numrevs - numfull
3036 deltasize[2] /= numrevs - numfull
3037 totalsize = fulltotal + deltatotal
3037 totalsize = fulltotal + deltatotal
3038 avgchainlen = sum(chainlengths) / numrevs
3038 avgchainlen = sum(chainlengths) / numrevs
3039 maxchainlen = max(chainlengths)
3039 maxchainlen = max(chainlengths)
3040 compratio = 1
3040 compratio = 1
3041 if totalsize:
3041 if totalsize:
3042 compratio = totalrawsize / totalsize
3042 compratio = totalrawsize / totalsize
3043
3043
3044 basedfmtstr = '%%%dd\n'
3044 basedfmtstr = '%%%dd\n'
3045 basepcfmtstr = '%%%dd %s(%%5.2f%%%%)\n'
3045 basepcfmtstr = '%%%dd %s(%%5.2f%%%%)\n'
3046
3046
3047 def dfmtstr(max):
3047 def dfmtstr(max):
3048 return basedfmtstr % len(str(max))
3048 return basedfmtstr % len(str(max))
3049 def pcfmtstr(max, padding=0):
3049 def pcfmtstr(max, padding=0):
3050 return basepcfmtstr % (len(str(max)), ' ' * padding)
3050 return basepcfmtstr % (len(str(max)), ' ' * padding)
3051
3051
3052 def pcfmt(value, total):
3052 def pcfmt(value, total):
3053 if total:
3053 if total:
3054 return (value, 100 * float(value) / total)
3054 return (value, 100 * float(value) / total)
3055 else:
3055 else:
3056 return value, 100.0
3056 return value, 100.0
3057
3057
3058 ui.write(('format : %d\n') % format)
3058 ui.write(('format : %d\n') % format)
3059 ui.write(('flags : %s\n') % ', '.join(flags))
3059 ui.write(('flags : %s\n') % ', '.join(flags))
3060
3060
3061 ui.write('\n')
3061 ui.write('\n')
3062 fmt = pcfmtstr(totalsize)
3062 fmt = pcfmtstr(totalsize)
3063 fmt2 = dfmtstr(totalsize)
3063 fmt2 = dfmtstr(totalsize)
3064 ui.write(('revisions : ') + fmt2 % numrevs)
3064 ui.write(('revisions : ') + fmt2 % numrevs)
3065 ui.write((' merges : ') + fmt % pcfmt(nummerges, numrevs))
3065 ui.write((' merges : ') + fmt % pcfmt(nummerges, numrevs))
3066 ui.write((' normal : ') + fmt % pcfmt(numrevs - nummerges, numrevs))
3066 ui.write((' normal : ') + fmt % pcfmt(numrevs - nummerges, numrevs))
3067 ui.write(('revisions : ') + fmt2 % numrevs)
3067 ui.write(('revisions : ') + fmt2 % numrevs)
3068 ui.write((' full : ') + fmt % pcfmt(numfull, numrevs))
3068 ui.write((' full : ') + fmt % pcfmt(numfull, numrevs))
3069 ui.write((' deltas : ') + fmt % pcfmt(numdeltas, numrevs))
3069 ui.write((' deltas : ') + fmt % pcfmt(numdeltas, numrevs))
3070 ui.write(('revision size : ') + fmt2 % totalsize)
3070 ui.write(('revision size : ') + fmt2 % totalsize)
3071 ui.write((' full : ') + fmt % pcfmt(fulltotal, totalsize))
3071 ui.write((' full : ') + fmt % pcfmt(fulltotal, totalsize))
3072 ui.write((' deltas : ') + fmt % pcfmt(deltatotal, totalsize))
3072 ui.write((' deltas : ') + fmt % pcfmt(deltatotal, totalsize))
3073
3073
3074 ui.write('\n')
3074 ui.write('\n')
3075 fmt = dfmtstr(max(avgchainlen, compratio))
3075 fmt = dfmtstr(max(avgchainlen, compratio))
3076 ui.write(('avg chain length : ') + fmt % avgchainlen)
3076 ui.write(('avg chain length : ') + fmt % avgchainlen)
3077 ui.write(('max chain length : ') + fmt % maxchainlen)
3077 ui.write(('max chain length : ') + fmt % maxchainlen)
3078 ui.write(('compression ratio : ') + fmt % compratio)
3078 ui.write(('compression ratio : ') + fmt % compratio)
3079
3079
3080 if format > 0:
3080 if format > 0:
3081 ui.write('\n')
3081 ui.write('\n')
3082 ui.write(('uncompressed data size (min/max/avg) : %d / %d / %d\n')
3082 ui.write(('uncompressed data size (min/max/avg) : %d / %d / %d\n')
3083 % tuple(datasize))
3083 % tuple(datasize))
3084 ui.write(('full revision size (min/max/avg) : %d / %d / %d\n')
3084 ui.write(('full revision size (min/max/avg) : %d / %d / %d\n')
3085 % tuple(fullsize))
3085 % tuple(fullsize))
3086 ui.write(('delta size (min/max/avg) : %d / %d / %d\n')
3086 ui.write(('delta size (min/max/avg) : %d / %d / %d\n')
3087 % tuple(deltasize))
3087 % tuple(deltasize))
3088
3088
3089 if numdeltas > 0:
3089 if numdeltas > 0:
3090 ui.write('\n')
3090 ui.write('\n')
3091 fmt = pcfmtstr(numdeltas)
3091 fmt = pcfmtstr(numdeltas)
3092 fmt2 = pcfmtstr(numdeltas, 4)
3092 fmt2 = pcfmtstr(numdeltas, 4)
3093 ui.write(('deltas against prev : ') + fmt % pcfmt(numprev, numdeltas))
3093 ui.write(('deltas against prev : ') + fmt % pcfmt(numprev, numdeltas))
3094 if numprev > 0:
3094 if numprev > 0:
3095 ui.write((' where prev = p1 : ') + fmt2 % pcfmt(nump1prev,
3095 ui.write((' where prev = p1 : ') + fmt2 % pcfmt(nump1prev,
3096 numprev))
3096 numprev))
3097 ui.write((' where prev = p2 : ') + fmt2 % pcfmt(nump2prev,
3097 ui.write((' where prev = p2 : ') + fmt2 % pcfmt(nump2prev,
3098 numprev))
3098 numprev))
3099 ui.write((' other : ') + fmt2 % pcfmt(numoprev,
3099 ui.write((' other : ') + fmt2 % pcfmt(numoprev,
3100 numprev))
3100 numprev))
3101 if gdelta:
3101 if gdelta:
3102 ui.write(('deltas against p1 : ')
3102 ui.write(('deltas against p1 : ')
3103 + fmt % pcfmt(nump1, numdeltas))
3103 + fmt % pcfmt(nump1, numdeltas))
3104 ui.write(('deltas against p2 : ')
3104 ui.write(('deltas against p2 : ')
3105 + fmt % pcfmt(nump2, numdeltas))
3105 + fmt % pcfmt(nump2, numdeltas))
3106 ui.write(('deltas against other : ') + fmt % pcfmt(numother,
3106 ui.write(('deltas against other : ') + fmt % pcfmt(numother,
3107 numdeltas))
3107 numdeltas))
3108
3108
3109 @command('debugrevspec',
3109 @command('debugrevspec',
3110 [('', 'optimize', None, _('print parsed tree after optimizing'))],
3110 [('', 'optimize', None, _('print parsed tree after optimizing'))],
3111 ('REVSPEC'))
3111 ('REVSPEC'))
3112 def debugrevspec(ui, repo, expr, **opts):
3112 def debugrevspec(ui, repo, expr, **opts):
3113 """parse and apply a revision specification
3113 """parse and apply a revision specification
3114
3114
3115 Use --verbose to print the parsed tree before and after aliases
3115 Use --verbose to print the parsed tree before and after aliases
3116 expansion.
3116 expansion.
3117 """
3117 """
3118 if ui.verbose:
3118 if ui.verbose:
3119 tree = revset.parse(expr, lookup=repo.__contains__)
3119 tree = revset.parse(expr, lookup=repo.__contains__)
3120 ui.note(revset.prettyformat(tree), "\n")
3120 ui.note(revset.prettyformat(tree), "\n")
3121 newtree = revset.findaliases(ui, tree)
3121 newtree = revset.findaliases(ui, tree)
3122 if newtree != tree:
3122 if newtree != tree:
3123 ui.note(revset.prettyformat(newtree), "\n")
3123 ui.note(revset.prettyformat(newtree), "\n")
3124 tree = newtree
3124 tree = newtree
3125 newtree = revset.foldconcat(tree)
3125 newtree = revset.foldconcat(tree)
3126 if newtree != tree:
3126 if newtree != tree:
3127 ui.note(revset.prettyformat(newtree), "\n")
3127 ui.note(revset.prettyformat(newtree), "\n")
3128 if opts["optimize"]:
3128 if opts["optimize"]:
3129 weight, optimizedtree = revset.optimize(newtree, True)
3129 weight, optimizedtree = revset.optimize(newtree, True)
3130 ui.note("* optimized:\n", revset.prettyformat(optimizedtree), "\n")
3130 ui.note("* optimized:\n", revset.prettyformat(optimizedtree), "\n")
3131 func = revset.match(ui, expr, repo)
3131 func = revset.match(ui, expr, repo)
3132 revs = func(repo)
3132 revs = func(repo)
3133 if ui.verbose:
3133 if ui.verbose:
3134 ui.note("* set:\n", revset.prettyformatset(revs), "\n")
3134 ui.note("* set:\n", revset.prettyformatset(revs), "\n")
3135 for c in revs:
3135 for c in revs:
3136 ui.write("%s\n" % c)
3136 ui.write("%s\n" % c)
3137
3137
3138 @command('debugsetparents', [], _('REV1 [REV2]'))
3138 @command('debugsetparents', [], _('REV1 [REV2]'))
3139 def debugsetparents(ui, repo, rev1, rev2=None):
3139 def debugsetparents(ui, repo, rev1, rev2=None):
3140 """manually set the parents of the current working directory
3140 """manually set the parents of the current working directory
3141
3141
3142 This is useful for writing repository conversion tools, but should
3142 This is useful for writing repository conversion tools, but should
3143 be used with care. For example, neither the working directory nor the
3143 be used with care. For example, neither the working directory nor the
3144 dirstate is updated, so file status may be incorrect after running this
3144 dirstate is updated, so file status may be incorrect after running this
3145 command.
3145 command.
3146
3146
3147 Returns 0 on success.
3147 Returns 0 on success.
3148 """
3148 """
3149
3149
3150 r1 = scmutil.revsingle(repo, rev1).node()
3150 r1 = scmutil.revsingle(repo, rev1).node()
3151 r2 = scmutil.revsingle(repo, rev2, 'null').node()
3151 r2 = scmutil.revsingle(repo, rev2, 'null').node()
3152
3152
3153 wlock = repo.wlock()
3153 wlock = repo.wlock()
3154 try:
3154 try:
3155 repo.dirstate.beginparentchange()
3155 repo.dirstate.beginparentchange()
3156 repo.setparents(r1, r2)
3156 repo.setparents(r1, r2)
3157 repo.dirstate.endparentchange()
3157 repo.dirstate.endparentchange()
3158 finally:
3158 finally:
3159 wlock.release()
3159 wlock.release()
3160
3160
3161 @command('debugdirstate|debugstate',
3161 @command('debugdirstate|debugstate',
3162 [('', 'nodates', None, _('do not display the saved mtime')),
3162 [('', 'nodates', None, _('do not display the saved mtime')),
3163 ('', 'datesort', None, _('sort by saved mtime'))],
3163 ('', 'datesort', None, _('sort by saved mtime'))],
3164 _('[OPTION]...'))
3164 _('[OPTION]...'))
3165 def debugstate(ui, repo, nodates=None, datesort=None):
3165 def debugstate(ui, repo, nodates=None, datesort=None):
3166 """show the contents of the current dirstate"""
3166 """show the contents of the current dirstate"""
3167 timestr = ""
3167 timestr = ""
3168 if datesort:
3168 if datesort:
3169 keyfunc = lambda x: (x[1][3], x[0]) # sort by mtime, then by filename
3169 keyfunc = lambda x: (x[1][3], x[0]) # sort by mtime, then by filename
3170 else:
3170 else:
3171 keyfunc = None # sort by filename
3171 keyfunc = None # sort by filename
3172 for file_, ent in sorted(repo.dirstate._map.iteritems(), key=keyfunc):
3172 for file_, ent in sorted(repo.dirstate._map.iteritems(), key=keyfunc):
3173 if ent[3] == -1:
3173 if ent[3] == -1:
3174 timestr = 'unset '
3174 timestr = 'unset '
3175 elif nodates:
3175 elif nodates:
3176 timestr = 'set '
3176 timestr = 'set '
3177 else:
3177 else:
3178 timestr = time.strftime("%Y-%m-%d %H:%M:%S ",
3178 timestr = time.strftime("%Y-%m-%d %H:%M:%S ",
3179 time.localtime(ent[3]))
3179 time.localtime(ent[3]))
3180 if ent[1] & 0o20000:
3180 if ent[1] & 0o20000:
3181 mode = 'lnk'
3181 mode = 'lnk'
3182 else:
3182 else:
3183 mode = '%3o' % (ent[1] & 0o777 & ~util.umask)
3183 mode = '%3o' % (ent[1] & 0o777 & ~util.umask)
3184 ui.write("%c %s %10d %s%s\n" % (ent[0], mode, ent[2], timestr, file_))
3184 ui.write("%c %s %10d %s%s\n" % (ent[0], mode, ent[2], timestr, file_))
3185 for f in repo.dirstate.copies():
3185 for f in repo.dirstate.copies():
3186 ui.write(_("copy: %s -> %s\n") % (repo.dirstate.copied(f), f))
3186 ui.write(_("copy: %s -> %s\n") % (repo.dirstate.copied(f), f))
3187
3187
3188 @command('debugsub',
3188 @command('debugsub',
3189 [('r', 'rev', '',
3189 [('r', 'rev', '',
3190 _('revision to check'), _('REV'))],
3190 _('revision to check'), _('REV'))],
3191 _('[-r REV] [REV]'))
3191 _('[-r REV] [REV]'))
3192 def debugsub(ui, repo, rev=None):
3192 def debugsub(ui, repo, rev=None):
3193 ctx = scmutil.revsingle(repo, rev, None)
3193 ctx = scmutil.revsingle(repo, rev, None)
3194 for k, v in sorted(ctx.substate.items()):
3194 for k, v in sorted(ctx.substate.items()):
3195 ui.write(('path %s\n') % k)
3195 ui.write(('path %s\n') % k)
3196 ui.write((' source %s\n') % v[0])
3196 ui.write((' source %s\n') % v[0])
3197 ui.write((' revision %s\n') % v[1])
3197 ui.write((' revision %s\n') % v[1])
3198
3198
3199 @command('debugsuccessorssets',
3199 @command('debugsuccessorssets',
3200 [],
3200 [],
3201 _('[REV]'))
3201 _('[REV]'))
3202 def debugsuccessorssets(ui, repo, *revs):
3202 def debugsuccessorssets(ui, repo, *revs):
3203 """show set of successors for revision
3203 """show set of successors for revision
3204
3204
3205 A successors set of changeset A is a consistent group of revisions that
3205 A successors set of changeset A is a consistent group of revisions that
3206 succeed A. It contains non-obsolete changesets only.
3206 succeed A. It contains non-obsolete changesets only.
3207
3207
3208 In most cases a changeset A has a single successors set containing a single
3208 In most cases a changeset A has a single successors set containing a single
3209 successor (changeset A replaced by A').
3209 successor (changeset A replaced by A').
3210
3210
3211 A changeset that is made obsolete with no successors are called "pruned".
3211 A changeset that is made obsolete with no successors are called "pruned".
3212 Such changesets have no successors sets at all.
3212 Such changesets have no successors sets at all.
3213
3213
3214 A changeset that has been "split" will have a successors set containing
3214 A changeset that has been "split" will have a successors set containing
3215 more than one successor.
3215 more than one successor.
3216
3216
3217 A changeset that has been rewritten in multiple different ways is called
3217 A changeset that has been rewritten in multiple different ways is called
3218 "divergent". Such changesets have multiple successor sets (each of which
3218 "divergent". Such changesets have multiple successor sets (each of which
3219 may also be split, i.e. have multiple successors).
3219 may also be split, i.e. have multiple successors).
3220
3220
3221 Results are displayed as follows::
3221 Results are displayed as follows::
3222
3222
3223 <rev1>
3223 <rev1>
3224 <successors-1A>
3224 <successors-1A>
3225 <rev2>
3225 <rev2>
3226 <successors-2A>
3226 <successors-2A>
3227 <successors-2B1> <successors-2B2> <successors-2B3>
3227 <successors-2B1> <successors-2B2> <successors-2B3>
3228
3228
3229 Here rev2 has two possible (i.e. divergent) successors sets. The first
3229 Here rev2 has two possible (i.e. divergent) successors sets. The first
3230 holds one element, whereas the second holds three (i.e. the changeset has
3230 holds one element, whereas the second holds three (i.e. the changeset has
3231 been split).
3231 been split).
3232 """
3232 """
3233 # passed to successorssets caching computation from one call to another
3233 # passed to successorssets caching computation from one call to another
3234 cache = {}
3234 cache = {}
3235 ctx2str = str
3235 ctx2str = str
3236 node2str = short
3236 node2str = short
3237 if ui.debug():
3237 if ui.debug():
3238 def ctx2str(ctx):
3238 def ctx2str(ctx):
3239 return ctx.hex()
3239 return ctx.hex()
3240 node2str = hex
3240 node2str = hex
3241 for rev in scmutil.revrange(repo, revs):
3241 for rev in scmutil.revrange(repo, revs):
3242 ctx = repo[rev]
3242 ctx = repo[rev]
3243 ui.write('%s\n'% ctx2str(ctx))
3243 ui.write('%s\n'% ctx2str(ctx))
3244 for succsset in obsolete.successorssets(repo, ctx.node(), cache):
3244 for succsset in obsolete.successorssets(repo, ctx.node(), cache):
3245 if succsset:
3245 if succsset:
3246 ui.write(' ')
3246 ui.write(' ')
3247 ui.write(node2str(succsset[0]))
3247 ui.write(node2str(succsset[0]))
3248 for node in succsset[1:]:
3248 for node in succsset[1:]:
3249 ui.write(' ')
3249 ui.write(' ')
3250 ui.write(node2str(node))
3250 ui.write(node2str(node))
3251 ui.write('\n')
3251 ui.write('\n')
3252
3252
3253 @command('debugwalk', walkopts, _('[OPTION]... [FILE]...'), inferrepo=True)
3253 @command('debugwalk', walkopts, _('[OPTION]... [FILE]...'), inferrepo=True)
3254 def debugwalk(ui, repo, *pats, **opts):
3254 def debugwalk(ui, repo, *pats, **opts):
3255 """show how files match on given patterns"""
3255 """show how files match on given patterns"""
3256 m = scmutil.match(repo[None], pats, opts)
3256 m = scmutil.match(repo[None], pats, opts)
3257 items = list(repo.walk(m))
3257 items = list(repo.walk(m))
3258 if not items:
3258 if not items:
3259 return
3259 return
3260 f = lambda fn: fn
3260 f = lambda fn: fn
3261 if ui.configbool('ui', 'slash') and os.sep != '/':
3261 if ui.configbool('ui', 'slash') and os.sep != '/':
3262 f = lambda fn: util.normpath(fn)
3262 f = lambda fn: util.normpath(fn)
3263 fmt = 'f %%-%ds %%-%ds %%s' % (
3263 fmt = 'f %%-%ds %%-%ds %%s' % (
3264 max([len(abs) for abs in items]),
3264 max([len(abs) for abs in items]),
3265 max([len(m.rel(abs)) for abs in items]))
3265 max([len(m.rel(abs)) for abs in items]))
3266 for abs in items:
3266 for abs in items:
3267 line = fmt % (abs, f(m.rel(abs)), m.exact(abs) and 'exact' or '')
3267 line = fmt % (abs, f(m.rel(abs)), m.exact(abs) and 'exact' or '')
3268 ui.write("%s\n" % line.rstrip())
3268 ui.write("%s\n" % line.rstrip())
3269
3269
3270 @command('debugwireargs',
3270 @command('debugwireargs',
3271 [('', 'three', '', 'three'),
3271 [('', 'three', '', 'three'),
3272 ('', 'four', '', 'four'),
3272 ('', 'four', '', 'four'),
3273 ('', 'five', '', 'five'),
3273 ('', 'five', '', 'five'),
3274 ] + remoteopts,
3274 ] + remoteopts,
3275 _('REPO [OPTIONS]... [ONE [TWO]]'),
3275 _('REPO [OPTIONS]... [ONE [TWO]]'),
3276 norepo=True)
3276 norepo=True)
3277 def debugwireargs(ui, repopath, *vals, **opts):
3277 def debugwireargs(ui, repopath, *vals, **opts):
3278 repo = hg.peer(ui, opts, repopath)
3278 repo = hg.peer(ui, opts, repopath)
3279 for opt in remoteopts:
3279 for opt in remoteopts:
3280 del opts[opt[1]]
3280 del opts[opt[1]]
3281 args = {}
3281 args = {}
3282 for k, v in opts.iteritems():
3282 for k, v in opts.iteritems():
3283 if v:
3283 if v:
3284 args[k] = v
3284 args[k] = v
3285 # run twice to check that we don't mess up the stream for the next command
3285 # run twice to check that we don't mess up the stream for the next command
3286 res1 = repo.debugwireargs(*vals, **args)
3286 res1 = repo.debugwireargs(*vals, **args)
3287 res2 = repo.debugwireargs(*vals, **args)
3287 res2 = repo.debugwireargs(*vals, **args)
3288 ui.write("%s\n" % res1)
3288 ui.write("%s\n" % res1)
3289 if res1 != res2:
3289 if res1 != res2:
3290 ui.warn("%s\n" % res2)
3290 ui.warn("%s\n" % res2)
3291
3291
3292 @command('^diff',
3292 @command('^diff',
3293 [('r', 'rev', [], _('revision'), _('REV')),
3293 [('r', 'rev', [], _('revision'), _('REV')),
3294 ('c', 'change', '', _('change made by revision'), _('REV'))
3294 ('c', 'change', '', _('change made by revision'), _('REV'))
3295 ] + diffopts + diffopts2 + walkopts + subrepoopts,
3295 ] + diffopts + diffopts2 + walkopts + subrepoopts,
3296 _('[OPTION]... ([-c REV] | [-r REV1 [-r REV2]]) [FILE]...'),
3296 _('[OPTION]... ([-c REV] | [-r REV1 [-r REV2]]) [FILE]...'),
3297 inferrepo=True)
3297 inferrepo=True)
3298 def diff(ui, repo, *pats, **opts):
3298 def diff(ui, repo, *pats, **opts):
3299 """diff repository (or selected files)
3299 """diff repository (or selected files)
3300
3300
3301 Show differences between revisions for the specified files.
3301 Show differences between revisions for the specified files.
3302
3302
3303 Differences between files are shown using the unified diff format.
3303 Differences between files are shown using the unified diff format.
3304
3304
3305 .. note::
3305 .. note::
3306
3306
3307 diff may generate unexpected results for merges, as it will
3307 diff may generate unexpected results for merges, as it will
3308 default to comparing against the working directory's first
3308 default to comparing against the working directory's first
3309 parent changeset if no revisions are specified.
3309 parent changeset if no revisions are specified.
3310
3310
3311 When two revision arguments are given, then changes are shown
3311 When two revision arguments are given, then changes are shown
3312 between those revisions. If only one revision is specified then
3312 between those revisions. If only one revision is specified then
3313 that revision is compared to the working directory, and, when no
3313 that revision is compared to the working directory, and, when no
3314 revisions are specified, the working directory files are compared
3314 revisions are specified, the working directory files are compared
3315 to its parent.
3315 to its parent.
3316
3316
3317 Alternatively you can specify -c/--change with a revision to see
3317 Alternatively you can specify -c/--change with a revision to see
3318 the changes in that changeset relative to its first parent.
3318 the changes in that changeset relative to its first parent.
3319
3319
3320 Without the -a/--text option, diff will avoid generating diffs of
3320 Without the -a/--text option, diff will avoid generating diffs of
3321 files it detects as binary. With -a, diff will generate a diff
3321 files it detects as binary. With -a, diff will generate a diff
3322 anyway, probably with undesirable results.
3322 anyway, probably with undesirable results.
3323
3323
3324 Use the -g/--git option to generate diffs in the git extended diff
3324 Use the -g/--git option to generate diffs in the git extended diff
3325 format. For more information, read :hg:`help diffs`.
3325 format. For more information, read :hg:`help diffs`.
3326
3326
3327 .. container:: verbose
3327 .. container:: verbose
3328
3328
3329 Examples:
3329 Examples:
3330
3330
3331 - compare a file in the current working directory to its parent::
3331 - compare a file in the current working directory to its parent::
3332
3332
3333 hg diff foo.c
3333 hg diff foo.c
3334
3334
3335 - compare two historical versions of a directory, with rename info::
3335 - compare two historical versions of a directory, with rename info::
3336
3336
3337 hg diff --git -r 1.0:1.2 lib/
3337 hg diff --git -r 1.0:1.2 lib/
3338
3338
3339 - get change stats relative to the last change on some date::
3339 - get change stats relative to the last change on some date::
3340
3340
3341 hg diff --stat -r "date('may 2')"
3341 hg diff --stat -r "date('may 2')"
3342
3342
3343 - diff all newly-added files that contain a keyword::
3343 - diff all newly-added files that contain a keyword::
3344
3344
3345 hg diff "set:added() and grep(GNU)"
3345 hg diff "set:added() and grep(GNU)"
3346
3346
3347 - compare a revision and its parents::
3347 - compare a revision and its parents::
3348
3348
3349 hg diff -c 9353 # compare against first parent
3349 hg diff -c 9353 # compare against first parent
3350 hg diff -r 9353^:9353 # same using revset syntax
3350 hg diff -r 9353^:9353 # same using revset syntax
3351 hg diff -r 9353^2:9353 # compare against the second parent
3351 hg diff -r 9353^2:9353 # compare against the second parent
3352
3352
3353 Returns 0 on success.
3353 Returns 0 on success.
3354 """
3354 """
3355
3355
3356 revs = opts.get('rev')
3356 revs = opts.get('rev')
3357 change = opts.get('change')
3357 change = opts.get('change')
3358 stat = opts.get('stat')
3358 stat = opts.get('stat')
3359 reverse = opts.get('reverse')
3359 reverse = opts.get('reverse')
3360
3360
3361 if revs and change:
3361 if revs and change:
3362 msg = _('cannot specify --rev and --change at the same time')
3362 msg = _('cannot specify --rev and --change at the same time')
3363 raise error.Abort(msg)
3363 raise error.Abort(msg)
3364 elif change:
3364 elif change:
3365 node2 = scmutil.revsingle(repo, change, None).node()
3365 node2 = scmutil.revsingle(repo, change, None).node()
3366 node1 = repo[node2].p1().node()
3366 node1 = repo[node2].p1().node()
3367 else:
3367 else:
3368 node1, node2 = scmutil.revpair(repo, revs)
3368 node1, node2 = scmutil.revpair(repo, revs)
3369
3369
3370 if reverse:
3370 if reverse:
3371 node1, node2 = node2, node1
3371 node1, node2 = node2, node1
3372
3372
3373 diffopts = patch.diffallopts(ui, opts)
3373 diffopts = patch.diffallopts(ui, opts)
3374 m = scmutil.match(repo[node2], pats, opts)
3374 m = scmutil.match(repo[node2], pats, opts)
3375 cmdutil.diffordiffstat(ui, repo, diffopts, node1, node2, m, stat=stat,
3375 cmdutil.diffordiffstat(ui, repo, diffopts, node1, node2, m, stat=stat,
3376 listsubrepos=opts.get('subrepos'),
3376 listsubrepos=opts.get('subrepos'),
3377 root=opts.get('root'))
3377 root=opts.get('root'))
3378
3378
3379 @command('^export',
3379 @command('^export',
3380 [('o', 'output', '',
3380 [('o', 'output', '',
3381 _('print output to file with formatted name'), _('FORMAT')),
3381 _('print output to file with formatted name'), _('FORMAT')),
3382 ('', 'switch-parent', None, _('diff against the second parent')),
3382 ('', 'switch-parent', None, _('diff against the second parent')),
3383 ('r', 'rev', [], _('revisions to export'), _('REV')),
3383 ('r', 'rev', [], _('revisions to export'), _('REV')),
3384 ] + diffopts,
3384 ] + diffopts,
3385 _('[OPTION]... [-o OUTFILESPEC] [-r] [REV]...'))
3385 _('[OPTION]... [-o OUTFILESPEC] [-r] [REV]...'))
3386 def export(ui, repo, *changesets, **opts):
3386 def export(ui, repo, *changesets, **opts):
3387 """dump the header and diffs for one or more changesets
3387 """dump the header and diffs for one or more changesets
3388
3388
3389 Print the changeset header and diffs for one or more revisions.
3389 Print the changeset header and diffs for one or more revisions.
3390 If no revision is given, the parent of the working directory is used.
3390 If no revision is given, the parent of the working directory is used.
3391
3391
3392 The information shown in the changeset header is: author, date,
3392 The information shown in the changeset header is: author, date,
3393 branch name (if non-default), changeset hash, parent(s) and commit
3393 branch name (if non-default), changeset hash, parent(s) and commit
3394 comment.
3394 comment.
3395
3395
3396 .. note::
3396 .. note::
3397
3397
3398 export may generate unexpected diff output for merge
3398 export may generate unexpected diff output for merge
3399 changesets, as it will compare the merge changeset against its
3399 changesets, as it will compare the merge changeset against its
3400 first parent only.
3400 first parent only.
3401
3401
3402 Output may be to a file, in which case the name of the file is
3402 Output may be to a file, in which case the name of the file is
3403 given using a format string. The formatting rules are as follows:
3403 given using a format string. The formatting rules are as follows:
3404
3404
3405 :``%%``: literal "%" character
3405 :``%%``: literal "%" character
3406 :``%H``: changeset hash (40 hexadecimal digits)
3406 :``%H``: changeset hash (40 hexadecimal digits)
3407 :``%N``: number of patches being generated
3407 :``%N``: number of patches being generated
3408 :``%R``: changeset revision number
3408 :``%R``: changeset revision number
3409 :``%b``: basename of the exporting repository
3409 :``%b``: basename of the exporting repository
3410 :``%h``: short-form changeset hash (12 hexadecimal digits)
3410 :``%h``: short-form changeset hash (12 hexadecimal digits)
3411 :``%m``: first line of the commit message (only alphanumeric characters)
3411 :``%m``: first line of the commit message (only alphanumeric characters)
3412 :``%n``: zero-padded sequence number, starting at 1
3412 :``%n``: zero-padded sequence number, starting at 1
3413 :``%r``: zero-padded changeset revision number
3413 :``%r``: zero-padded changeset revision number
3414
3414
3415 Without the -a/--text option, export will avoid generating diffs
3415 Without the -a/--text option, export will avoid generating diffs
3416 of files it detects as binary. With -a, export will generate a
3416 of files it detects as binary. With -a, export will generate a
3417 diff anyway, probably with undesirable results.
3417 diff anyway, probably with undesirable results.
3418
3418
3419 Use the -g/--git option to generate diffs in the git extended diff
3419 Use the -g/--git option to generate diffs in the git extended diff
3420 format. See :hg:`help diffs` for more information.
3420 format. See :hg:`help diffs` for more information.
3421
3421
3422 With the --switch-parent option, the diff will be against the
3422 With the --switch-parent option, the diff will be against the
3423 second parent. It can be useful to review a merge.
3423 second parent. It can be useful to review a merge.
3424
3424
3425 .. container:: verbose
3425 .. container:: verbose
3426
3426
3427 Examples:
3427 Examples:
3428
3428
3429 - use export and import to transplant a bugfix to the current
3429 - use export and import to transplant a bugfix to the current
3430 branch::
3430 branch::
3431
3431
3432 hg export -r 9353 | hg import -
3432 hg export -r 9353 | hg import -
3433
3433
3434 - export all the changesets between two revisions to a file with
3434 - export all the changesets between two revisions to a file with
3435 rename information::
3435 rename information::
3436
3436
3437 hg export --git -r 123:150 > changes.txt
3437 hg export --git -r 123:150 > changes.txt
3438
3438
3439 - split outgoing changes into a series of patches with
3439 - split outgoing changes into a series of patches with
3440 descriptive names::
3440 descriptive names::
3441
3441
3442 hg export -r "outgoing()" -o "%n-%m.patch"
3442 hg export -r "outgoing()" -o "%n-%m.patch"
3443
3443
3444 Returns 0 on success.
3444 Returns 0 on success.
3445 """
3445 """
3446 changesets += tuple(opts.get('rev', []))
3446 changesets += tuple(opts.get('rev', []))
3447 if not changesets:
3447 if not changesets:
3448 changesets = ['.']
3448 changesets = ['.']
3449 revs = scmutil.revrange(repo, changesets)
3449 revs = scmutil.revrange(repo, changesets)
3450 if not revs:
3450 if not revs:
3451 raise error.Abort(_("export requires at least one changeset"))
3451 raise error.Abort(_("export requires at least one changeset"))
3452 if len(revs) > 1:
3452 if len(revs) > 1:
3453 ui.note(_('exporting patches:\n'))
3453 ui.note(_('exporting patches:\n'))
3454 else:
3454 else:
3455 ui.note(_('exporting patch:\n'))
3455 ui.note(_('exporting patch:\n'))
3456 cmdutil.export(repo, revs, template=opts.get('output'),
3456 cmdutil.export(repo, revs, template=opts.get('output'),
3457 switch_parent=opts.get('switch_parent'),
3457 switch_parent=opts.get('switch_parent'),
3458 opts=patch.diffallopts(ui, opts))
3458 opts=patch.diffallopts(ui, opts))
3459
3459
3460 @command('files',
3460 @command('files',
3461 [('r', 'rev', '', _('search the repository as it is in REV'), _('REV')),
3461 [('r', 'rev', '', _('search the repository as it is in REV'), _('REV')),
3462 ('0', 'print0', None, _('end filenames with NUL, for use with xargs')),
3462 ('0', 'print0', None, _('end filenames with NUL, for use with xargs')),
3463 ] + walkopts + formatteropts + subrepoopts,
3463 ] + walkopts + formatteropts + subrepoopts,
3464 _('[OPTION]... [PATTERN]...'))
3464 _('[OPTION]... [PATTERN]...'))
3465 def files(ui, repo, *pats, **opts):
3465 def files(ui, repo, *pats, **opts):
3466 """list tracked files
3466 """list tracked files
3467
3467
3468 Print files under Mercurial control in the working directory or
3468 Print files under Mercurial control in the working directory or
3469 specified revision whose names match the given patterns (excluding
3469 specified revision whose names match the given patterns (excluding
3470 removed files).
3470 removed files).
3471
3471
3472 If no patterns are given to match, this command prints the names
3472 If no patterns are given to match, this command prints the names
3473 of all files under Mercurial control in the working directory.
3473 of all files under Mercurial control in the working directory.
3474
3474
3475 .. container:: verbose
3475 .. container:: verbose
3476
3476
3477 Examples:
3477 Examples:
3478
3478
3479 - list all files under the current directory::
3479 - list all files under the current directory::
3480
3480
3481 hg files .
3481 hg files .
3482
3482
3483 - shows sizes and flags for current revision::
3483 - shows sizes and flags for current revision::
3484
3484
3485 hg files -vr .
3485 hg files -vr .
3486
3486
3487 - list all files named README::
3487 - list all files named README::
3488
3488
3489 hg files -I "**/README"
3489 hg files -I "**/README"
3490
3490
3491 - list all binary files::
3491 - list all binary files::
3492
3492
3493 hg files "set:binary()"
3493 hg files "set:binary()"
3494
3494
3495 - find files containing a regular expression::
3495 - find files containing a regular expression::
3496
3496
3497 hg files "set:grep('bob')"
3497 hg files "set:grep('bob')"
3498
3498
3499 - search tracked file contents with xargs and grep::
3499 - search tracked file contents with xargs and grep::
3500
3500
3501 hg files -0 | xargs -0 grep foo
3501 hg files -0 | xargs -0 grep foo
3502
3502
3503 See :hg:`help patterns` and :hg:`help filesets` for more information
3503 See :hg:`help patterns` and :hg:`help filesets` for more information
3504 on specifying file patterns.
3504 on specifying file patterns.
3505
3505
3506 Returns 0 if a match is found, 1 otherwise.
3506 Returns 0 if a match is found, 1 otherwise.
3507
3507
3508 """
3508 """
3509 ctx = scmutil.revsingle(repo, opts.get('rev'), None)
3509 ctx = scmutil.revsingle(repo, opts.get('rev'), None)
3510
3510
3511 end = '\n'
3511 end = '\n'
3512 if opts.get('print0'):
3512 if opts.get('print0'):
3513 end = '\0'
3513 end = '\0'
3514 fm = ui.formatter('files', opts)
3514 fm = ui.formatter('files', opts)
3515 fmt = '%s' + end
3515 fmt = '%s' + end
3516
3516
3517 m = scmutil.match(ctx, pats, opts)
3517 m = scmutil.match(ctx, pats, opts)
3518 ret = cmdutil.files(ui, ctx, m, fm, fmt, opts.get('subrepos'))
3518 ret = cmdutil.files(ui, ctx, m, fm, fmt, opts.get('subrepos'))
3519
3519
3520 fm.end()
3520 fm.end()
3521
3521
3522 return ret
3522 return ret
3523
3523
3524 @command('^forget', walkopts, _('[OPTION]... FILE...'), inferrepo=True)
3524 @command('^forget', walkopts, _('[OPTION]... FILE...'), inferrepo=True)
3525 def forget(ui, repo, *pats, **opts):
3525 def forget(ui, repo, *pats, **opts):
3526 """forget the specified files on the next commit
3526 """forget the specified files on the next commit
3527
3527
3528 Mark the specified files so they will no longer be tracked
3528 Mark the specified files so they will no longer be tracked
3529 after the next commit.
3529 after the next commit.
3530
3530
3531 This only removes files from the current branch, not from the
3531 This only removes files from the current branch, not from the
3532 entire project history, and it does not delete them from the
3532 entire project history, and it does not delete them from the
3533 working directory.
3533 working directory.
3534
3534
3535 To delete the file from the working directory, see :hg:`remove`.
3535 To delete the file from the working directory, see :hg:`remove`.
3536
3536
3537 To undo a forget before the next commit, see :hg:`add`.
3537 To undo a forget before the next commit, see :hg:`add`.
3538
3538
3539 .. container:: verbose
3539 .. container:: verbose
3540
3540
3541 Examples:
3541 Examples:
3542
3542
3543 - forget newly-added binary files::
3543 - forget newly-added binary files::
3544
3544
3545 hg forget "set:added() and binary()"
3545 hg forget "set:added() and binary()"
3546
3546
3547 - forget files that would be excluded by .hgignore::
3547 - forget files that would be excluded by .hgignore::
3548
3548
3549 hg forget "set:hgignore()"
3549 hg forget "set:hgignore()"
3550
3550
3551 Returns 0 on success.
3551 Returns 0 on success.
3552 """
3552 """
3553
3553
3554 if not pats:
3554 if not pats:
3555 raise error.Abort(_('no files specified'))
3555 raise error.Abort(_('no files specified'))
3556
3556
3557 m = scmutil.match(repo[None], pats, opts)
3557 m = scmutil.match(repo[None], pats, opts)
3558 rejected = cmdutil.forget(ui, repo, m, prefix="", explicitonly=False)[0]
3558 rejected = cmdutil.forget(ui, repo, m, prefix="", explicitonly=False)[0]
3559 return rejected and 1 or 0
3559 return rejected and 1 or 0
3560
3560
3561 @command(
3561 @command(
3562 'graft',
3562 'graft',
3563 [('r', 'rev', [], _('revisions to graft'), _('REV')),
3563 [('r', 'rev', [], _('revisions to graft'), _('REV')),
3564 ('c', 'continue', False, _('resume interrupted graft')),
3564 ('c', 'continue', False, _('resume interrupted graft')),
3565 ('e', 'edit', False, _('invoke editor on commit messages')),
3565 ('e', 'edit', False, _('invoke editor on commit messages')),
3566 ('', 'log', None, _('append graft info to log message')),
3566 ('', 'log', None, _('append graft info to log message')),
3567 ('f', 'force', False, _('force graft')),
3567 ('f', 'force', False, _('force graft')),
3568 ('D', 'currentdate', False,
3568 ('D', 'currentdate', False,
3569 _('record the current date as commit date')),
3569 _('record the current date as commit date')),
3570 ('U', 'currentuser', False,
3570 ('U', 'currentuser', False,
3571 _('record the current user as committer'), _('DATE'))]
3571 _('record the current user as committer'), _('DATE'))]
3572 + commitopts2 + mergetoolopts + dryrunopts,
3572 + commitopts2 + mergetoolopts + dryrunopts,
3573 _('[OPTION]... [-r] REV...'))
3573 _('[OPTION]... [-r] REV...'))
3574 def graft(ui, repo, *revs, **opts):
3574 def graft(ui, repo, *revs, **opts):
3575 '''copy changes from other branches onto the current branch
3575 '''copy changes from other branches onto the current branch
3576
3576
3577 This command uses Mercurial's merge logic to copy individual
3577 This command uses Mercurial's merge logic to copy individual
3578 changes from other branches without merging branches in the
3578 changes from other branches without merging branches in the
3579 history graph. This is sometimes known as 'backporting' or
3579 history graph. This is sometimes known as 'backporting' or
3580 'cherry-picking'. By default, graft will copy user, date, and
3580 'cherry-picking'. By default, graft will copy user, date, and
3581 description from the source changesets.
3581 description from the source changesets.
3582
3582
3583 Changesets that are ancestors of the current revision, that have
3583 Changesets that are ancestors of the current revision, that have
3584 already been grafted, or that are merges will be skipped.
3584 already been grafted, or that are merges will be skipped.
3585
3585
3586 If --log is specified, log messages will have a comment appended
3586 If --log is specified, log messages will have a comment appended
3587 of the form::
3587 of the form::
3588
3588
3589 (grafted from CHANGESETHASH)
3589 (grafted from CHANGESETHASH)
3590
3590
3591 If --force is specified, revisions will be grafted even if they
3591 If --force is specified, revisions will be grafted even if they
3592 are already ancestors of or have been grafted to the destination.
3592 are already ancestors of or have been grafted to the destination.
3593 This is useful when the revisions have since been backed out.
3593 This is useful when the revisions have since been backed out.
3594
3594
3595 If a graft merge results in conflicts, the graft process is
3595 If a graft merge results in conflicts, the graft process is
3596 interrupted so that the current merge can be manually resolved.
3596 interrupted so that the current merge can be manually resolved.
3597 Once all conflicts are addressed, the graft process can be
3597 Once all conflicts are addressed, the graft process can be
3598 continued with the -c/--continue option.
3598 continued with the -c/--continue option.
3599
3599
3600 .. note::
3600 .. note::
3601
3601
3602 The -c/--continue option does not reapply earlier options, except
3602 The -c/--continue option does not reapply earlier options, except
3603 for --force.
3603 for --force.
3604
3604
3605 .. container:: verbose
3605 .. container:: verbose
3606
3606
3607 Examples:
3607 Examples:
3608
3608
3609 - copy a single change to the stable branch and edit its description::
3609 - copy a single change to the stable branch and edit its description::
3610
3610
3611 hg update stable
3611 hg update stable
3612 hg graft --edit 9393
3612 hg graft --edit 9393
3613
3613
3614 - graft a range of changesets with one exception, updating dates::
3614 - graft a range of changesets with one exception, updating dates::
3615
3615
3616 hg graft -D "2085::2093 and not 2091"
3616 hg graft -D "2085::2093 and not 2091"
3617
3617
3618 - continue a graft after resolving conflicts::
3618 - continue a graft after resolving conflicts::
3619
3619
3620 hg graft -c
3620 hg graft -c
3621
3621
3622 - show the source of a grafted changeset::
3622 - show the source of a grafted changeset::
3623
3623
3624 hg log --debug -r .
3624 hg log --debug -r .
3625
3625
3626 See :hg:`help revisions` and :hg:`help revsets` for more about
3626 See :hg:`help revisions` and :hg:`help revsets` for more about
3627 specifying revisions.
3627 specifying revisions.
3628
3628
3629 Returns 0 on successful completion.
3629 Returns 0 on successful completion.
3630 '''
3630 '''
3631
3631
3632 revs = list(revs)
3632 revs = list(revs)
3633 revs.extend(opts['rev'])
3633 revs.extend(opts['rev'])
3634
3634
3635 if not opts.get('user') and opts.get('currentuser'):
3635 if not opts.get('user') and opts.get('currentuser'):
3636 opts['user'] = ui.username()
3636 opts['user'] = ui.username()
3637 if not opts.get('date') and opts.get('currentdate'):
3637 if not opts.get('date') and opts.get('currentdate'):
3638 opts['date'] = "%d %d" % util.makedate()
3638 opts['date'] = "%d %d" % util.makedate()
3639
3639
3640 editor = cmdutil.getcommiteditor(editform='graft', **opts)
3640 editor = cmdutil.getcommiteditor(editform='graft', **opts)
3641
3641
3642 cont = False
3642 cont = False
3643 if opts['continue']:
3643 if opts['continue']:
3644 cont = True
3644 cont = True
3645 if revs:
3645 if revs:
3646 raise error.Abort(_("can't specify --continue and revisions"))
3646 raise error.Abort(_("can't specify --continue and revisions"))
3647 # read in unfinished revisions
3647 # read in unfinished revisions
3648 try:
3648 try:
3649 nodes = repo.vfs.read('graftstate').splitlines()
3649 nodes = repo.vfs.read('graftstate').splitlines()
3650 revs = [repo[node].rev() for node in nodes]
3650 revs = [repo[node].rev() for node in nodes]
3651 except IOError as inst:
3651 except IOError as inst:
3652 if inst.errno != errno.ENOENT:
3652 if inst.errno != errno.ENOENT:
3653 raise
3653 raise
3654 raise error.Abort(_("no graft state found, can't continue"))
3654 raise error.Abort(_("no graft state found, can't continue"))
3655 else:
3655 else:
3656 cmdutil.checkunfinished(repo)
3656 cmdutil.checkunfinished(repo)
3657 cmdutil.bailifchanged(repo)
3657 cmdutil.bailifchanged(repo)
3658 if not revs:
3658 if not revs:
3659 raise error.Abort(_('no revisions specified'))
3659 raise error.Abort(_('no revisions specified'))
3660 revs = scmutil.revrange(repo, revs)
3660 revs = scmutil.revrange(repo, revs)
3661
3661
3662 skipped = set()
3662 skipped = set()
3663 # check for merges
3663 # check for merges
3664 for rev in repo.revs('%ld and merge()', revs):
3664 for rev in repo.revs('%ld and merge()', revs):
3665 ui.warn(_('skipping ungraftable merge revision %s\n') % rev)
3665 ui.warn(_('skipping ungraftable merge revision %s\n') % rev)
3666 skipped.add(rev)
3666 skipped.add(rev)
3667 revs = [r for r in revs if r not in skipped]
3667 revs = [r for r in revs if r not in skipped]
3668 if not revs:
3668 if not revs:
3669 return -1
3669 return -1
3670
3670
3671 # Don't check in the --continue case, in effect retaining --force across
3671 # Don't check in the --continue case, in effect retaining --force across
3672 # --continues. That's because without --force, any revisions we decided to
3672 # --continues. That's because without --force, any revisions we decided to
3673 # skip would have been filtered out here, so they wouldn't have made their
3673 # skip would have been filtered out here, so they wouldn't have made their
3674 # way to the graftstate. With --force, any revisions we would have otherwise
3674 # way to the graftstate. With --force, any revisions we would have otherwise
3675 # skipped would not have been filtered out, and if they hadn't been applied
3675 # skipped would not have been filtered out, and if they hadn't been applied
3676 # already, they'd have been in the graftstate.
3676 # already, they'd have been in the graftstate.
3677 if not (cont or opts.get('force')):
3677 if not (cont or opts.get('force')):
3678 # check for ancestors of dest branch
3678 # check for ancestors of dest branch
3679 crev = repo['.'].rev()
3679 crev = repo['.'].rev()
3680 ancestors = repo.changelog.ancestors([crev], inclusive=True)
3680 ancestors = repo.changelog.ancestors([crev], inclusive=True)
3681 # Cannot use x.remove(y) on smart set, this has to be a list.
3681 # Cannot use x.remove(y) on smart set, this has to be a list.
3682 # XXX make this lazy in the future
3682 # XXX make this lazy in the future
3683 revs = list(revs)
3683 revs = list(revs)
3684 # don't mutate while iterating, create a copy
3684 # don't mutate while iterating, create a copy
3685 for rev in list(revs):
3685 for rev in list(revs):
3686 if rev in ancestors:
3686 if rev in ancestors:
3687 ui.warn(_('skipping ancestor revision %d:%s\n') %
3687 ui.warn(_('skipping ancestor revision %d:%s\n') %
3688 (rev, repo[rev]))
3688 (rev, repo[rev]))
3689 # XXX remove on list is slow
3689 # XXX remove on list is slow
3690 revs.remove(rev)
3690 revs.remove(rev)
3691 if not revs:
3691 if not revs:
3692 return -1
3692 return -1
3693
3693
3694 # analyze revs for earlier grafts
3694 # analyze revs for earlier grafts
3695 ids = {}
3695 ids = {}
3696 for ctx in repo.set("%ld", revs):
3696 for ctx in repo.set("%ld", revs):
3697 ids[ctx.hex()] = ctx.rev()
3697 ids[ctx.hex()] = ctx.rev()
3698 n = ctx.extra().get('source')
3698 n = ctx.extra().get('source')
3699 if n:
3699 if n:
3700 ids[n] = ctx.rev()
3700 ids[n] = ctx.rev()
3701
3701
3702 # check ancestors for earlier grafts
3702 # check ancestors for earlier grafts
3703 ui.debug('scanning for duplicate grafts\n')
3703 ui.debug('scanning for duplicate grafts\n')
3704
3704
3705 for rev in repo.changelog.findmissingrevs(revs, [crev]):
3705 for rev in repo.changelog.findmissingrevs(revs, [crev]):
3706 ctx = repo[rev]
3706 ctx = repo[rev]
3707 n = ctx.extra().get('source')
3707 n = ctx.extra().get('source')
3708 if n in ids:
3708 if n in ids:
3709 try:
3709 try:
3710 r = repo[n].rev()
3710 r = repo[n].rev()
3711 except error.RepoLookupError:
3711 except error.RepoLookupError:
3712 r = None
3712 r = None
3713 if r in revs:
3713 if r in revs:
3714 ui.warn(_('skipping revision %d:%s '
3714 ui.warn(_('skipping revision %d:%s '
3715 '(already grafted to %d:%s)\n')
3715 '(already grafted to %d:%s)\n')
3716 % (r, repo[r], rev, ctx))
3716 % (r, repo[r], rev, ctx))
3717 revs.remove(r)
3717 revs.remove(r)
3718 elif ids[n] in revs:
3718 elif ids[n] in revs:
3719 if r is None:
3719 if r is None:
3720 ui.warn(_('skipping already grafted revision %d:%s '
3720 ui.warn(_('skipping already grafted revision %d:%s '
3721 '(%d:%s also has unknown origin %s)\n')
3721 '(%d:%s also has unknown origin %s)\n')
3722 % (ids[n], repo[ids[n]], rev, ctx, n[:12]))
3722 % (ids[n], repo[ids[n]], rev, ctx, n[:12]))
3723 else:
3723 else:
3724 ui.warn(_('skipping already grafted revision %d:%s '
3724 ui.warn(_('skipping already grafted revision %d:%s '
3725 '(%d:%s also has origin %d:%s)\n')
3725 '(%d:%s also has origin %d:%s)\n')
3726 % (ids[n], repo[ids[n]], rev, ctx, r, n[:12]))
3726 % (ids[n], repo[ids[n]], rev, ctx, r, n[:12]))
3727 revs.remove(ids[n])
3727 revs.remove(ids[n])
3728 elif ctx.hex() in ids:
3728 elif ctx.hex() in ids:
3729 r = ids[ctx.hex()]
3729 r = ids[ctx.hex()]
3730 ui.warn(_('skipping already grafted revision %d:%s '
3730 ui.warn(_('skipping already grafted revision %d:%s '
3731 '(was grafted from %d:%s)\n') %
3731 '(was grafted from %d:%s)\n') %
3732 (r, repo[r], rev, ctx))
3732 (r, repo[r], rev, ctx))
3733 revs.remove(r)
3733 revs.remove(r)
3734 if not revs:
3734 if not revs:
3735 return -1
3735 return -1
3736
3736
3737 wlock = repo.wlock()
3737 wlock = repo.wlock()
3738 try:
3738 try:
3739 for pos, ctx in enumerate(repo.set("%ld", revs)):
3739 for pos, ctx in enumerate(repo.set("%ld", revs)):
3740 desc = '%d:%s "%s"' % (ctx.rev(), ctx,
3740 desc = '%d:%s "%s"' % (ctx.rev(), ctx,
3741 ctx.description().split('\n', 1)[0])
3741 ctx.description().split('\n', 1)[0])
3742 names = repo.nodetags(ctx.node()) + repo.nodebookmarks(ctx.node())
3742 names = repo.nodetags(ctx.node()) + repo.nodebookmarks(ctx.node())
3743 if names:
3743 if names:
3744 desc += ' (%s)' % ' '.join(names)
3744 desc += ' (%s)' % ' '.join(names)
3745 ui.status(_('grafting %s\n') % desc)
3745 ui.status(_('grafting %s\n') % desc)
3746 if opts.get('dry_run'):
3746 if opts.get('dry_run'):
3747 continue
3747 continue
3748
3748
3749 source = ctx.extra().get('source')
3749 source = ctx.extra().get('source')
3750 extra = {}
3750 extra = {}
3751 if source:
3751 if source:
3752 extra['source'] = source
3752 extra['source'] = source
3753 extra['intermediate-source'] = ctx.hex()
3753 extra['intermediate-source'] = ctx.hex()
3754 else:
3754 else:
3755 extra['source'] = ctx.hex()
3755 extra['source'] = ctx.hex()
3756 user = ctx.user()
3756 user = ctx.user()
3757 if opts.get('user'):
3757 if opts.get('user'):
3758 user = opts['user']
3758 user = opts['user']
3759 date = ctx.date()
3759 date = ctx.date()
3760 if opts.get('date'):
3760 if opts.get('date'):
3761 date = opts['date']
3761 date = opts['date']
3762 message = ctx.description()
3762 message = ctx.description()
3763 if opts.get('log'):
3763 if opts.get('log'):
3764 message += '\n(grafted from %s)' % ctx.hex()
3764 message += '\n(grafted from %s)' % ctx.hex()
3765
3765
3766 # we don't merge the first commit when continuing
3766 # we don't merge the first commit when continuing
3767 if not cont:
3767 if not cont:
3768 # perform the graft merge with p1(rev) as 'ancestor'
3768 # perform the graft merge with p1(rev) as 'ancestor'
3769 try:
3769 try:
3770 # ui.forcemerge is an internal variable, do not document
3770 # ui.forcemerge is an internal variable, do not document
3771 repo.ui.setconfig('ui', 'forcemerge', opts.get('tool', ''),
3771 repo.ui.setconfig('ui', 'forcemerge', opts.get('tool', ''),
3772 'graft')
3772 'graft')
3773 stats = mergemod.graft(repo, ctx, ctx.p1(),
3773 stats = mergemod.graft(repo, ctx, ctx.p1(),
3774 ['local', 'graft'])
3774 ['local', 'graft'])
3775 finally:
3775 finally:
3776 repo.ui.setconfig('ui', 'forcemerge', '', 'graft')
3776 repo.ui.setconfig('ui', 'forcemerge', '', 'graft')
3777 # report any conflicts
3777 # report any conflicts
3778 if stats and stats[3] > 0:
3778 if stats and stats[3] > 0:
3779 # write out state for --continue
3779 # write out state for --continue
3780 nodelines = [repo[rev].hex() + "\n" for rev in revs[pos:]]
3780 nodelines = [repo[rev].hex() + "\n" for rev in revs[pos:]]
3781 repo.vfs.write('graftstate', ''.join(nodelines))
3781 repo.vfs.write('graftstate', ''.join(nodelines))
3782 raise error.Abort(
3782 raise error.Abort(
3783 _("unresolved conflicts, can't continue"),
3783 _("unresolved conflicts, can't continue"),
3784 hint=_('use hg resolve and hg graft --continue'))
3784 hint=_('use hg resolve and hg graft --continue'))
3785 else:
3785 else:
3786 cont = False
3786 cont = False
3787
3787
3788 # commit
3788 # commit
3789 node = repo.commit(text=message, user=user,
3789 node = repo.commit(text=message, user=user,
3790 date=date, extra=extra, editor=editor)
3790 date=date, extra=extra, editor=editor)
3791 if node is None:
3791 if node is None:
3792 ui.warn(
3792 ui.warn(
3793 _('note: graft of %d:%s created no changes to commit\n') %
3793 _('note: graft of %d:%s created no changes to commit\n') %
3794 (ctx.rev(), ctx))
3794 (ctx.rev(), ctx))
3795 finally:
3795 finally:
3796 wlock.release()
3796 wlock.release()
3797
3797
3798 # remove state when we complete successfully
3798 # remove state when we complete successfully
3799 if not opts.get('dry_run'):
3799 if not opts.get('dry_run'):
3800 util.unlinkpath(repo.join('graftstate'), ignoremissing=True)
3800 util.unlinkpath(repo.join('graftstate'), ignoremissing=True)
3801
3801
3802 return 0
3802 return 0
3803
3803
3804 @command('grep',
3804 @command('grep',
3805 [('0', 'print0', None, _('end fields with NUL')),
3805 [('0', 'print0', None, _('end fields with NUL')),
3806 ('', 'all', None, _('print all revisions that match')),
3806 ('', 'all', None, _('print all revisions that match')),
3807 ('a', 'text', None, _('treat all files as text')),
3807 ('a', 'text', None, _('treat all files as text')),
3808 ('f', 'follow', None,
3808 ('f', 'follow', None,
3809 _('follow changeset history,'
3809 _('follow changeset history,'
3810 ' or file history across copies and renames')),
3810 ' or file history across copies and renames')),
3811 ('i', 'ignore-case', None, _('ignore case when matching')),
3811 ('i', 'ignore-case', None, _('ignore case when matching')),
3812 ('l', 'files-with-matches', None,
3812 ('l', 'files-with-matches', None,
3813 _('print only filenames and revisions that match')),
3813 _('print only filenames and revisions that match')),
3814 ('n', 'line-number', None, _('print matching line numbers')),
3814 ('n', 'line-number', None, _('print matching line numbers')),
3815 ('r', 'rev', [],
3815 ('r', 'rev', [],
3816 _('only search files changed within revision range'), _('REV')),
3816 _('only search files changed within revision range'), _('REV')),
3817 ('u', 'user', None, _('list the author (long with -v)')),
3817 ('u', 'user', None, _('list the author (long with -v)')),
3818 ('d', 'date', None, _('list the date (short with -q)')),
3818 ('d', 'date', None, _('list the date (short with -q)')),
3819 ] + walkopts,
3819 ] + walkopts,
3820 _('[OPTION]... PATTERN [FILE]...'),
3820 _('[OPTION]... PATTERN [FILE]...'),
3821 inferrepo=True)
3821 inferrepo=True)
3822 def grep(ui, repo, pattern, *pats, **opts):
3822 def grep(ui, repo, pattern, *pats, **opts):
3823 """search for a pattern in specified files and revisions
3823 """search for a pattern in specified files and revisions
3824
3824
3825 Search revisions of files for a regular expression.
3825 Search revisions of files for a regular expression.
3826
3826
3827 This command behaves differently than Unix grep. It only accepts
3827 This command behaves differently than Unix grep. It only accepts
3828 Python/Perl regexps. It searches repository history, not the
3828 Python/Perl regexps. It searches repository history, not the
3829 working directory. It always prints the revision number in which a
3829 working directory. It always prints the revision number in which a
3830 match appears.
3830 match appears.
3831
3831
3832 By default, grep only prints output for the first revision of a
3832 By default, grep only prints output for the first revision of a
3833 file in which it finds a match. To get it to print every revision
3833 file in which it finds a match. To get it to print every revision
3834 that contains a change in match status ("-" for a match that
3834 that contains a change in match status ("-" for a match that
3835 becomes a non-match, or "+" for a non-match that becomes a match),
3835 becomes a non-match, or "+" for a non-match that becomes a match),
3836 use the --all flag.
3836 use the --all flag.
3837
3837
3838 Returns 0 if a match is found, 1 otherwise.
3838 Returns 0 if a match is found, 1 otherwise.
3839 """
3839 """
3840 reflags = re.M
3840 reflags = re.M
3841 if opts.get('ignore_case'):
3841 if opts.get('ignore_case'):
3842 reflags |= re.I
3842 reflags |= re.I
3843 try:
3843 try:
3844 regexp = util.re.compile(pattern, reflags)
3844 regexp = util.re.compile(pattern, reflags)
3845 except re.error as inst:
3845 except re.error as inst:
3846 ui.warn(_("grep: invalid match pattern: %s\n") % inst)
3846 ui.warn(_("grep: invalid match pattern: %s\n") % inst)
3847 return 1
3847 return 1
3848 sep, eol = ':', '\n'
3848 sep, eol = ':', '\n'
3849 if opts.get('print0'):
3849 if opts.get('print0'):
3850 sep = eol = '\0'
3850 sep = eol = '\0'
3851
3851
3852 getfile = util.lrucachefunc(repo.file)
3852 getfile = util.lrucachefunc(repo.file)
3853
3853
3854 def matchlines(body):
3854 def matchlines(body):
3855 begin = 0
3855 begin = 0
3856 linenum = 0
3856 linenum = 0
3857 while begin < len(body):
3857 while begin < len(body):
3858 match = regexp.search(body, begin)
3858 match = regexp.search(body, begin)
3859 if not match:
3859 if not match:
3860 break
3860 break
3861 mstart, mend = match.span()
3861 mstart, mend = match.span()
3862 linenum += body.count('\n', begin, mstart) + 1
3862 linenum += body.count('\n', begin, mstart) + 1
3863 lstart = body.rfind('\n', begin, mstart) + 1 or begin
3863 lstart = body.rfind('\n', begin, mstart) + 1 or begin
3864 begin = body.find('\n', mend) + 1 or len(body) + 1
3864 begin = body.find('\n', mend) + 1 or len(body) + 1
3865 lend = begin - 1
3865 lend = begin - 1
3866 yield linenum, mstart - lstart, mend - lstart, body[lstart:lend]
3866 yield linenum, mstart - lstart, mend - lstart, body[lstart:lend]
3867
3867
3868 class linestate(object):
3868 class linestate(object):
3869 def __init__(self, line, linenum, colstart, colend):
3869 def __init__(self, line, linenum, colstart, colend):
3870 self.line = line
3870 self.line = line
3871 self.linenum = linenum
3871 self.linenum = linenum
3872 self.colstart = colstart
3872 self.colstart = colstart
3873 self.colend = colend
3873 self.colend = colend
3874
3874
3875 def __hash__(self):
3875 def __hash__(self):
3876 return hash((self.linenum, self.line))
3876 return hash((self.linenum, self.line))
3877
3877
3878 def __eq__(self, other):
3878 def __eq__(self, other):
3879 return self.line == other.line
3879 return self.line == other.line
3880
3880
3881 def __iter__(self):
3881 def __iter__(self):
3882 yield (self.line[:self.colstart], '')
3882 yield (self.line[:self.colstart], '')
3883 yield (self.line[self.colstart:self.colend], 'grep.match')
3883 yield (self.line[self.colstart:self.colend], 'grep.match')
3884 rest = self.line[self.colend:]
3884 rest = self.line[self.colend:]
3885 while rest != '':
3885 while rest != '':
3886 match = regexp.search(rest)
3886 match = regexp.search(rest)
3887 if not match:
3887 if not match:
3888 yield (rest, '')
3888 yield (rest, '')
3889 break
3889 break
3890 mstart, mend = match.span()
3890 mstart, mend = match.span()
3891 yield (rest[:mstart], '')
3891 yield (rest[:mstart], '')
3892 yield (rest[mstart:mend], 'grep.match')
3892 yield (rest[mstart:mend], 'grep.match')
3893 rest = rest[mend:]
3893 rest = rest[mend:]
3894
3894
3895 matches = {}
3895 matches = {}
3896 copies = {}
3896 copies = {}
3897 def grepbody(fn, rev, body):
3897 def grepbody(fn, rev, body):
3898 matches[rev].setdefault(fn, [])
3898 matches[rev].setdefault(fn, [])
3899 m = matches[rev][fn]
3899 m = matches[rev][fn]
3900 for lnum, cstart, cend, line in matchlines(body):
3900 for lnum, cstart, cend, line in matchlines(body):
3901 s = linestate(line, lnum, cstart, cend)
3901 s = linestate(line, lnum, cstart, cend)
3902 m.append(s)
3902 m.append(s)
3903
3903
3904 def difflinestates(a, b):
3904 def difflinestates(a, b):
3905 sm = difflib.SequenceMatcher(None, a, b)
3905 sm = difflib.SequenceMatcher(None, a, b)
3906 for tag, alo, ahi, blo, bhi in sm.get_opcodes():
3906 for tag, alo, ahi, blo, bhi in sm.get_opcodes():
3907 if tag == 'insert':
3907 if tag == 'insert':
3908 for i in xrange(blo, bhi):
3908 for i in xrange(blo, bhi):
3909 yield ('+', b[i])
3909 yield ('+', b[i])
3910 elif tag == 'delete':
3910 elif tag == 'delete':
3911 for i in xrange(alo, ahi):
3911 for i in xrange(alo, ahi):
3912 yield ('-', a[i])
3912 yield ('-', a[i])
3913 elif tag == 'replace':
3913 elif tag == 'replace':
3914 for i in xrange(alo, ahi):
3914 for i in xrange(alo, ahi):
3915 yield ('-', a[i])
3915 yield ('-', a[i])
3916 for i in xrange(blo, bhi):
3916 for i in xrange(blo, bhi):
3917 yield ('+', b[i])
3917 yield ('+', b[i])
3918
3918
3919 def display(fn, ctx, pstates, states):
3919 def display(fn, ctx, pstates, states):
3920 rev = ctx.rev()
3920 rev = ctx.rev()
3921 if ui.quiet:
3921 if ui.quiet:
3922 datefunc = util.shortdate
3922 datefunc = util.shortdate
3923 else:
3923 else:
3924 datefunc = util.datestr
3924 datefunc = util.datestr
3925 found = False
3925 found = False
3926 @util.cachefunc
3926 @util.cachefunc
3927 def binary():
3927 def binary():
3928 flog = getfile(fn)
3928 flog = getfile(fn)
3929 return util.binary(flog.read(ctx.filenode(fn)))
3929 return util.binary(flog.read(ctx.filenode(fn)))
3930
3930
3931 if opts.get('all'):
3931 if opts.get('all'):
3932 iter = difflinestates(pstates, states)
3932 iter = difflinestates(pstates, states)
3933 else:
3933 else:
3934 iter = [('', l) for l in states]
3934 iter = [('', l) for l in states]
3935 for change, l in iter:
3935 for change, l in iter:
3936 cols = [(fn, 'grep.filename'), (str(rev), 'grep.rev')]
3936 cols = [(fn, 'grep.filename'), (str(rev), 'grep.rev')]
3937
3937
3938 if opts.get('line_number'):
3938 if opts.get('line_number'):
3939 cols.append((str(l.linenum), 'grep.linenumber'))
3939 cols.append((str(l.linenum), 'grep.linenumber'))
3940 if opts.get('all'):
3940 if opts.get('all'):
3941 cols.append((change, 'grep.change'))
3941 cols.append((change, 'grep.change'))
3942 if opts.get('user'):
3942 if opts.get('user'):
3943 cols.append((ui.shortuser(ctx.user()), 'grep.user'))
3943 cols.append((ui.shortuser(ctx.user()), 'grep.user'))
3944 if opts.get('date'):
3944 if opts.get('date'):
3945 cols.append((datefunc(ctx.date()), 'grep.date'))
3945 cols.append((datefunc(ctx.date()), 'grep.date'))
3946 for col, label in cols[:-1]:
3946 for col, label in cols[:-1]:
3947 ui.write(col, label=label)
3947 ui.write(col, label=label)
3948 ui.write(sep, label='grep.sep')
3948 ui.write(sep, label='grep.sep')
3949 ui.write(cols[-1][0], label=cols[-1][1])
3949 ui.write(cols[-1][0], label=cols[-1][1])
3950 if not opts.get('files_with_matches'):
3950 if not opts.get('files_with_matches'):
3951 ui.write(sep, label='grep.sep')
3951 ui.write(sep, label='grep.sep')
3952 if not opts.get('text') and binary():
3952 if not opts.get('text') and binary():
3953 ui.write(" Binary file matches")
3953 ui.write(" Binary file matches")
3954 else:
3954 else:
3955 for s, label in l:
3955 for s, label in l:
3956 ui.write(s, label=label)
3956 ui.write(s, label=label)
3957 ui.write(eol)
3957 ui.write(eol)
3958 found = True
3958 found = True
3959 if opts.get('files_with_matches'):
3959 if opts.get('files_with_matches'):
3960 break
3960 break
3961 return found
3961 return found
3962
3962
3963 skip = {}
3963 skip = {}
3964 revfiles = {}
3964 revfiles = {}
3965 matchfn = scmutil.match(repo[None], pats, opts)
3965 matchfn = scmutil.match(repo[None], pats, opts)
3966 found = False
3966 found = False
3967 follow = opts.get('follow')
3967 follow = opts.get('follow')
3968
3968
3969 def prep(ctx, fns):
3969 def prep(ctx, fns):
3970 rev = ctx.rev()
3970 rev = ctx.rev()
3971 pctx = ctx.p1()
3971 pctx = ctx.p1()
3972 parent = pctx.rev()
3972 parent = pctx.rev()
3973 matches.setdefault(rev, {})
3973 matches.setdefault(rev, {})
3974 matches.setdefault(parent, {})
3974 matches.setdefault(parent, {})
3975 files = revfiles.setdefault(rev, [])
3975 files = revfiles.setdefault(rev, [])
3976 for fn in fns:
3976 for fn in fns:
3977 flog = getfile(fn)
3977 flog = getfile(fn)
3978 try:
3978 try:
3979 fnode = ctx.filenode(fn)
3979 fnode = ctx.filenode(fn)
3980 except error.LookupError:
3980 except error.LookupError:
3981 continue
3981 continue
3982
3982
3983 copied = flog.renamed(fnode)
3983 copied = flog.renamed(fnode)
3984 copy = follow and copied and copied[0]
3984 copy = follow and copied and copied[0]
3985 if copy:
3985 if copy:
3986 copies.setdefault(rev, {})[fn] = copy
3986 copies.setdefault(rev, {})[fn] = copy
3987 if fn in skip:
3987 if fn in skip:
3988 if copy:
3988 if copy:
3989 skip[copy] = True
3989 skip[copy] = True
3990 continue
3990 continue
3991 files.append(fn)
3991 files.append(fn)
3992
3992
3993 if fn not in matches[rev]:
3993 if fn not in matches[rev]:
3994 grepbody(fn, rev, flog.read(fnode))
3994 grepbody(fn, rev, flog.read(fnode))
3995
3995
3996 pfn = copy or fn
3996 pfn = copy or fn
3997 if pfn not in matches[parent]:
3997 if pfn not in matches[parent]:
3998 try:
3998 try:
3999 fnode = pctx.filenode(pfn)
3999 fnode = pctx.filenode(pfn)
4000 grepbody(pfn, parent, flog.read(fnode))
4000 grepbody(pfn, parent, flog.read(fnode))
4001 except error.LookupError:
4001 except error.LookupError:
4002 pass
4002 pass
4003
4003
4004 for ctx in cmdutil.walkchangerevs(repo, matchfn, opts, prep):
4004 for ctx in cmdutil.walkchangerevs(repo, matchfn, opts, prep):
4005 rev = ctx.rev()
4005 rev = ctx.rev()
4006 parent = ctx.p1().rev()
4006 parent = ctx.p1().rev()
4007 for fn in sorted(revfiles.get(rev, [])):
4007 for fn in sorted(revfiles.get(rev, [])):
4008 states = matches[rev][fn]
4008 states = matches[rev][fn]
4009 copy = copies.get(rev, {}).get(fn)
4009 copy = copies.get(rev, {}).get(fn)
4010 if fn in skip:
4010 if fn in skip:
4011 if copy:
4011 if copy:
4012 skip[copy] = True
4012 skip[copy] = True
4013 continue
4013 continue
4014 pstates = matches.get(parent, {}).get(copy or fn, [])
4014 pstates = matches.get(parent, {}).get(copy or fn, [])
4015 if pstates or states:
4015 if pstates or states:
4016 r = display(fn, ctx, pstates, states)
4016 r = display(fn, ctx, pstates, states)
4017 found = found or r
4017 found = found or r
4018 if r and not opts.get('all'):
4018 if r and not opts.get('all'):
4019 skip[fn] = True
4019 skip[fn] = True
4020 if copy:
4020 if copy:
4021 skip[copy] = True
4021 skip[copy] = True
4022 del matches[rev]
4022 del matches[rev]
4023 del revfiles[rev]
4023 del revfiles[rev]
4024
4024
4025 return not found
4025 return not found
4026
4026
4027 @command('heads',
4027 @command('heads',
4028 [('r', 'rev', '',
4028 [('r', 'rev', '',
4029 _('show only heads which are descendants of STARTREV'), _('STARTREV')),
4029 _('show only heads which are descendants of STARTREV'), _('STARTREV')),
4030 ('t', 'topo', False, _('show topological heads only')),
4030 ('t', 'topo', False, _('show topological heads only')),
4031 ('a', 'active', False, _('show active branchheads only (DEPRECATED)')),
4031 ('a', 'active', False, _('show active branchheads only (DEPRECATED)')),
4032 ('c', 'closed', False, _('show normal and closed branch heads')),
4032 ('c', 'closed', False, _('show normal and closed branch heads')),
4033 ] + templateopts,
4033 ] + templateopts,
4034 _('[-ct] [-r STARTREV] [REV]...'))
4034 _('[-ct] [-r STARTREV] [REV]...'))
4035 def heads(ui, repo, *branchrevs, **opts):
4035 def heads(ui, repo, *branchrevs, **opts):
4036 """show branch heads
4036 """show branch heads
4037
4037
4038 With no arguments, show all open branch heads in the repository.
4038 With no arguments, show all open branch heads in the repository.
4039 Branch heads are changesets that have no descendants on the
4039 Branch heads are changesets that have no descendants on the
4040 same branch. They are where development generally takes place and
4040 same branch. They are where development generally takes place and
4041 are the usual targets for update and merge operations.
4041 are the usual targets for update and merge operations.
4042
4042
4043 If one or more REVs are given, only open branch heads on the
4043 If one or more REVs are given, only open branch heads on the
4044 branches associated with the specified changesets are shown. This
4044 branches associated with the specified changesets are shown. This
4045 means that you can use :hg:`heads .` to see the heads on the
4045 means that you can use :hg:`heads .` to see the heads on the
4046 currently checked-out branch.
4046 currently checked-out branch.
4047
4047
4048 If -c/--closed is specified, also show branch heads marked closed
4048 If -c/--closed is specified, also show branch heads marked closed
4049 (see :hg:`commit --close-branch`).
4049 (see :hg:`commit --close-branch`).
4050
4050
4051 If STARTREV is specified, only those heads that are descendants of
4051 If STARTREV is specified, only those heads that are descendants of
4052 STARTREV will be displayed.
4052 STARTREV will be displayed.
4053
4053
4054 If -t/--topo is specified, named branch mechanics will be ignored and only
4054 If -t/--topo is specified, named branch mechanics will be ignored and only
4055 topological heads (changesets with no children) will be shown.
4055 topological heads (changesets with no children) will be shown.
4056
4056
4057 Returns 0 if matching heads are found, 1 if not.
4057 Returns 0 if matching heads are found, 1 if not.
4058 """
4058 """
4059
4059
4060 start = None
4060 start = None
4061 if 'rev' in opts:
4061 if 'rev' in opts:
4062 start = scmutil.revsingle(repo, opts['rev'], None).node()
4062 start = scmutil.revsingle(repo, opts['rev'], None).node()
4063
4063
4064 if opts.get('topo'):
4064 if opts.get('topo'):
4065 heads = [repo[h] for h in repo.heads(start)]
4065 heads = [repo[h] for h in repo.heads(start)]
4066 else:
4066 else:
4067 heads = []
4067 heads = []
4068 for branch in repo.branchmap():
4068 for branch in repo.branchmap():
4069 heads += repo.branchheads(branch, start, opts.get('closed'))
4069 heads += repo.branchheads(branch, start, opts.get('closed'))
4070 heads = [repo[h] for h in heads]
4070 heads = [repo[h] for h in heads]
4071
4071
4072 if branchrevs:
4072 if branchrevs:
4073 branches = set(repo[br].branch() for br in branchrevs)
4073 branches = set(repo[br].branch() for br in branchrevs)
4074 heads = [h for h in heads if h.branch() in branches]
4074 heads = [h for h in heads if h.branch() in branches]
4075
4075
4076 if opts.get('active') and branchrevs:
4076 if opts.get('active') and branchrevs:
4077 dagheads = repo.heads(start)
4077 dagheads = repo.heads(start)
4078 heads = [h for h in heads if h.node() in dagheads]
4078 heads = [h for h in heads if h.node() in dagheads]
4079
4079
4080 if branchrevs:
4080 if branchrevs:
4081 haveheads = set(h.branch() for h in heads)
4081 haveheads = set(h.branch() for h in heads)
4082 if branches - haveheads:
4082 if branches - haveheads:
4083 headless = ', '.join(b for b in branches - haveheads)
4083 headless = ', '.join(b for b in branches - haveheads)
4084 msg = _('no open branch heads found on branches %s')
4084 msg = _('no open branch heads found on branches %s')
4085 if opts.get('rev'):
4085 if opts.get('rev'):
4086 msg += _(' (started at %s)') % opts['rev']
4086 msg += _(' (started at %s)') % opts['rev']
4087 ui.warn((msg + '\n') % headless)
4087 ui.warn((msg + '\n') % headless)
4088
4088
4089 if not heads:
4089 if not heads:
4090 return 1
4090 return 1
4091
4091
4092 heads = sorted(heads, key=lambda x: -x.rev())
4092 heads = sorted(heads, key=lambda x: -x.rev())
4093 displayer = cmdutil.show_changeset(ui, repo, opts)
4093 displayer = cmdutil.show_changeset(ui, repo, opts)
4094 for ctx in heads:
4094 for ctx in heads:
4095 displayer.show(ctx)
4095 displayer.show(ctx)
4096 displayer.close()
4096 displayer.close()
4097
4097
4098 @command('help',
4098 @command('help',
4099 [('e', 'extension', None, _('show only help for extensions')),
4099 [('e', 'extension', None, _('show only help for extensions')),
4100 ('c', 'command', None, _('show only help for commands')),
4100 ('c', 'command', None, _('show only help for commands')),
4101 ('k', 'keyword', None, _('show topics matching keyword')),
4101 ('k', 'keyword', None, _('show topics matching keyword')),
4102 ],
4102 ],
4103 _('[-eck] [TOPIC]'),
4103 _('[-eck] [TOPIC]'),
4104 norepo=True)
4104 norepo=True)
4105 def help_(ui, name=None, **opts):
4105 def help_(ui, name=None, **opts):
4106 """show help for a given topic or a help overview
4106 """show help for a given topic or a help overview
4107
4107
4108 With no arguments, print a list of commands with short help messages.
4108 With no arguments, print a list of commands with short help messages.
4109
4109
4110 Given a topic, extension, or command name, print help for that
4110 Given a topic, extension, or command name, print help for that
4111 topic.
4111 topic.
4112
4112
4113 Returns 0 if successful.
4113 Returns 0 if successful.
4114 """
4114 """
4115
4115
4116 textwidth = min(ui.termwidth(), 80) - 2
4116 textwidth = min(ui.termwidth(), 80) - 2
4117
4117
4118 keep = []
4118 keep = []
4119 if ui.verbose:
4119 if ui.verbose:
4120 keep.append('verbose')
4120 keep.append('verbose')
4121 if sys.platform.startswith('win'):
4121 if sys.platform.startswith('win'):
4122 keep.append('windows')
4122 keep.append('windows')
4123 elif sys.platform == 'OpenVMS':
4123 elif sys.platform == 'OpenVMS':
4124 keep.append('vms')
4124 keep.append('vms')
4125 elif sys.platform == 'plan9':
4125 elif sys.platform == 'plan9':
4126 keep.append('plan9')
4126 keep.append('plan9')
4127 else:
4127 else:
4128 keep.append('unix')
4128 keep.append('unix')
4129 keep.append(sys.platform.lower())
4129 keep.append(sys.platform.lower())
4130
4130
4131 section = None
4131 section = None
4132 if name and '.' in name:
4132 if name and '.' in name:
4133 name, section = name.split('.', 1)
4133 name, section = name.split('.', 1)
4134 section = section.lower()
4134 section = section.lower()
4135
4135
4136 text = help.help_(ui, name, **opts)
4136 text = help.help_(ui, name, **opts)
4137
4137
4138 formatted, pruned = minirst.format(text, textwidth, keep=keep,
4138 formatted, pruned = minirst.format(text, textwidth, keep=keep,
4139 section=section)
4139 section=section)
4140
4140
4141 # We could have been given a weird ".foo" section without a name
4141 # We could have been given a weird ".foo" section without a name
4142 # to look for, or we could have simply failed to found "foo.bar"
4142 # to look for, or we could have simply failed to found "foo.bar"
4143 # because bar isn't a section of foo
4143 # because bar isn't a section of foo
4144 if section and not (formatted and name):
4144 if section and not (formatted and name):
4145 raise error.Abort(_("help section not found"))
4145 raise error.Abort(_("help section not found"))
4146
4146
4147 if 'verbose' in pruned:
4147 if 'verbose' in pruned:
4148 keep.append('omitted')
4148 keep.append('omitted')
4149 else:
4149 else:
4150 keep.append('notomitted')
4150 keep.append('notomitted')
4151 formatted, pruned = minirst.format(text, textwidth, keep=keep,
4151 formatted, pruned = minirst.format(text, textwidth, keep=keep,
4152 section=section)
4152 section=section)
4153 ui.write(formatted)
4153 ui.write(formatted)
4154
4154
4155
4155
4156 @command('identify|id',
4156 @command('identify|id',
4157 [('r', 'rev', '',
4157 [('r', 'rev', '',
4158 _('identify the specified revision'), _('REV')),
4158 _('identify the specified revision'), _('REV')),
4159 ('n', 'num', None, _('show local revision number')),
4159 ('n', 'num', None, _('show local revision number')),
4160 ('i', 'id', None, _('show global revision id')),
4160 ('i', 'id', None, _('show global revision id')),
4161 ('b', 'branch', None, _('show branch')),
4161 ('b', 'branch', None, _('show branch')),
4162 ('t', 'tags', None, _('show tags')),
4162 ('t', 'tags', None, _('show tags')),
4163 ('B', 'bookmarks', None, _('show bookmarks')),
4163 ('B', 'bookmarks', None, _('show bookmarks')),
4164 ] + remoteopts,
4164 ] + remoteopts,
4165 _('[-nibtB] [-r REV] [SOURCE]'),
4165 _('[-nibtB] [-r REV] [SOURCE]'),
4166 optionalrepo=True)
4166 optionalrepo=True)
4167 def identify(ui, repo, source=None, rev=None,
4167 def identify(ui, repo, source=None, rev=None,
4168 num=None, id=None, branch=None, tags=None, bookmarks=None, **opts):
4168 num=None, id=None, branch=None, tags=None, bookmarks=None, **opts):
4169 """identify the working directory or specified revision
4169 """identify the working directory or specified revision
4170
4170
4171 Print a summary identifying the repository state at REV using one or
4171 Print a summary identifying the repository state at REV using one or
4172 two parent hash identifiers, followed by a "+" if the working
4172 two parent hash identifiers, followed by a "+" if the working
4173 directory has uncommitted changes, the branch name (if not default),
4173 directory has uncommitted changes, the branch name (if not default),
4174 a list of tags, and a list of bookmarks.
4174 a list of tags, and a list of bookmarks.
4175
4175
4176 When REV is not given, print a summary of the current state of the
4176 When REV is not given, print a summary of the current state of the
4177 repository.
4177 repository.
4178
4178
4179 Specifying a path to a repository root or Mercurial bundle will
4179 Specifying a path to a repository root or Mercurial bundle will
4180 cause lookup to operate on that repository/bundle.
4180 cause lookup to operate on that repository/bundle.
4181
4181
4182 .. container:: verbose
4182 .. container:: verbose
4183
4183
4184 Examples:
4184 Examples:
4185
4185
4186 - generate a build identifier for the working directory::
4186 - generate a build identifier for the working directory::
4187
4187
4188 hg id --id > build-id.dat
4188 hg id --id > build-id.dat
4189
4189
4190 - find the revision corresponding to a tag::
4190 - find the revision corresponding to a tag::
4191
4191
4192 hg id -n -r 1.3
4192 hg id -n -r 1.3
4193
4193
4194 - check the most recent revision of a remote repository::
4194 - check the most recent revision of a remote repository::
4195
4195
4196 hg id -r tip http://selenic.com/hg/
4196 hg id -r tip http://selenic.com/hg/
4197
4197
4198 Returns 0 if successful.
4198 Returns 0 if successful.
4199 """
4199 """
4200
4200
4201 if not repo and not source:
4201 if not repo and not source:
4202 raise error.Abort(_("there is no Mercurial repository here "
4202 raise error.Abort(_("there is no Mercurial repository here "
4203 "(.hg not found)"))
4203 "(.hg not found)"))
4204
4204
4205 if ui.debugflag:
4205 if ui.debugflag:
4206 hexfunc = hex
4206 hexfunc = hex
4207 else:
4207 else:
4208 hexfunc = short
4208 hexfunc = short
4209 default = not (num or id or branch or tags or bookmarks)
4209 default = not (num or id or branch or tags or bookmarks)
4210 output = []
4210 output = []
4211 revs = []
4211 revs = []
4212
4212
4213 if source:
4213 if source:
4214 source, branches = hg.parseurl(ui.expandpath(source))
4214 source, branches = hg.parseurl(ui.expandpath(source))
4215 peer = hg.peer(repo or ui, opts, source) # only pass ui when no repo
4215 peer = hg.peer(repo or ui, opts, source) # only pass ui when no repo
4216 repo = peer.local()
4216 repo = peer.local()
4217 revs, checkout = hg.addbranchrevs(repo, peer, branches, None)
4217 revs, checkout = hg.addbranchrevs(repo, peer, branches, None)
4218
4218
4219 if not repo:
4219 if not repo:
4220 if num or branch or tags:
4220 if num or branch or tags:
4221 raise error.Abort(
4221 raise error.Abort(
4222 _("can't query remote revision number, branch, or tags"))
4222 _("can't query remote revision number, branch, or tags"))
4223 if not rev and revs:
4223 if not rev and revs:
4224 rev = revs[0]
4224 rev = revs[0]
4225 if not rev:
4225 if not rev:
4226 rev = "tip"
4226 rev = "tip"
4227
4227
4228 remoterev = peer.lookup(rev)
4228 remoterev = peer.lookup(rev)
4229 if default or id:
4229 if default or id:
4230 output = [hexfunc(remoterev)]
4230 output = [hexfunc(remoterev)]
4231
4231
4232 def getbms():
4232 def getbms():
4233 bms = []
4233 bms = []
4234
4234
4235 if 'bookmarks' in peer.listkeys('namespaces'):
4235 if 'bookmarks' in peer.listkeys('namespaces'):
4236 hexremoterev = hex(remoterev)
4236 hexremoterev = hex(remoterev)
4237 bms = [bm for bm, bmr in peer.listkeys('bookmarks').iteritems()
4237 bms = [bm for bm, bmr in peer.listkeys('bookmarks').iteritems()
4238 if bmr == hexremoterev]
4238 if bmr == hexremoterev]
4239
4239
4240 return sorted(bms)
4240 return sorted(bms)
4241
4241
4242 if bookmarks:
4242 if bookmarks:
4243 output.extend(getbms())
4243 output.extend(getbms())
4244 elif default and not ui.quiet:
4244 elif default and not ui.quiet:
4245 # multiple bookmarks for a single parent separated by '/'
4245 # multiple bookmarks for a single parent separated by '/'
4246 bm = '/'.join(getbms())
4246 bm = '/'.join(getbms())
4247 if bm:
4247 if bm:
4248 output.append(bm)
4248 output.append(bm)
4249 else:
4249 else:
4250 ctx = scmutil.revsingle(repo, rev, None)
4250 ctx = scmutil.revsingle(repo, rev, None)
4251
4251
4252 if ctx.rev() is None:
4252 if ctx.rev() is None:
4253 ctx = repo[None]
4253 ctx = repo[None]
4254 parents = ctx.parents()
4254 parents = ctx.parents()
4255 taglist = []
4255 taglist = []
4256 for p in parents:
4256 for p in parents:
4257 taglist.extend(p.tags())
4257 taglist.extend(p.tags())
4258
4258
4259 changed = ""
4259 changed = ""
4260 if default or id or num:
4260 if default or id or num:
4261 if (any(repo.status())
4261 if (any(repo.status())
4262 or any(ctx.sub(s).dirty() for s in ctx.substate)):
4262 or any(ctx.sub(s).dirty() for s in ctx.substate)):
4263 changed = '+'
4263 changed = '+'
4264 if default or id:
4264 if default or id:
4265 output = ["%s%s" %
4265 output = ["%s%s" %
4266 ('+'.join([hexfunc(p.node()) for p in parents]), changed)]
4266 ('+'.join([hexfunc(p.node()) for p in parents]), changed)]
4267 if num:
4267 if num:
4268 output.append("%s%s" %
4268 output.append("%s%s" %
4269 ('+'.join([str(p.rev()) for p in parents]), changed))
4269 ('+'.join([str(p.rev()) for p in parents]), changed))
4270 else:
4270 else:
4271 if default or id:
4271 if default or id:
4272 output = [hexfunc(ctx.node())]
4272 output = [hexfunc(ctx.node())]
4273 if num:
4273 if num:
4274 output.append(str(ctx.rev()))
4274 output.append(str(ctx.rev()))
4275 taglist = ctx.tags()
4275 taglist = ctx.tags()
4276
4276
4277 if default and not ui.quiet:
4277 if default and not ui.quiet:
4278 b = ctx.branch()
4278 b = ctx.branch()
4279 if b != 'default':
4279 if b != 'default':
4280 output.append("(%s)" % b)
4280 output.append("(%s)" % b)
4281
4281
4282 # multiple tags for a single parent separated by '/'
4282 # multiple tags for a single parent separated by '/'
4283 t = '/'.join(taglist)
4283 t = '/'.join(taglist)
4284 if t:
4284 if t:
4285 output.append(t)
4285 output.append(t)
4286
4286
4287 # multiple bookmarks for a single parent separated by '/'
4287 # multiple bookmarks for a single parent separated by '/'
4288 bm = '/'.join(ctx.bookmarks())
4288 bm = '/'.join(ctx.bookmarks())
4289 if bm:
4289 if bm:
4290 output.append(bm)
4290 output.append(bm)
4291 else:
4291 else:
4292 if branch:
4292 if branch:
4293 output.append(ctx.branch())
4293 output.append(ctx.branch())
4294
4294
4295 if tags:
4295 if tags:
4296 output.extend(taglist)
4296 output.extend(taglist)
4297
4297
4298 if bookmarks:
4298 if bookmarks:
4299 output.extend(ctx.bookmarks())
4299 output.extend(ctx.bookmarks())
4300
4300
4301 ui.write("%s\n" % ' '.join(output))
4301 ui.write("%s\n" % ' '.join(output))
4302
4302
4303 @command('import|patch',
4303 @command('import|patch',
4304 [('p', 'strip', 1,
4304 [('p', 'strip', 1,
4305 _('directory strip option for patch. This has the same '
4305 _('directory strip option for patch. This has the same '
4306 'meaning as the corresponding patch option'), _('NUM')),
4306 'meaning as the corresponding patch option'), _('NUM')),
4307 ('b', 'base', '', _('base path (DEPRECATED)'), _('PATH')),
4307 ('b', 'base', '', _('base path (DEPRECATED)'), _('PATH')),
4308 ('e', 'edit', False, _('invoke editor on commit messages')),
4308 ('e', 'edit', False, _('invoke editor on commit messages')),
4309 ('f', 'force', None,
4309 ('f', 'force', None,
4310 _('skip check for outstanding uncommitted changes (DEPRECATED)')),
4310 _('skip check for outstanding uncommitted changes (DEPRECATED)')),
4311 ('', 'no-commit', None,
4311 ('', 'no-commit', None,
4312 _("don't commit, just update the working directory")),
4312 _("don't commit, just update the working directory")),
4313 ('', 'bypass', None,
4313 ('', 'bypass', None,
4314 _("apply patch without touching the working directory")),
4314 _("apply patch without touching the working directory")),
4315 ('', 'partial', None,
4315 ('', 'partial', None,
4316 _('commit even if some hunks fail')),
4316 _('commit even if some hunks fail')),
4317 ('', 'exact', None,
4317 ('', 'exact', None,
4318 _('apply patch to the nodes from which it was generated')),
4318 _('apply patch to the nodes from which it was generated')),
4319 ('', 'prefix', '',
4319 ('', 'prefix', '',
4320 _('apply patch to subdirectory'), _('DIR')),
4320 _('apply patch to subdirectory'), _('DIR')),
4321 ('', 'import-branch', None,
4321 ('', 'import-branch', None,
4322 _('use any branch information in patch (implied by --exact)'))] +
4322 _('use any branch information in patch (implied by --exact)'))] +
4323 commitopts + commitopts2 + similarityopts,
4323 commitopts + commitopts2 + similarityopts,
4324 _('[OPTION]... PATCH...'))
4324 _('[OPTION]... PATCH...'))
4325 def import_(ui, repo, patch1=None, *patches, **opts):
4325 def import_(ui, repo, patch1=None, *patches, **opts):
4326 """import an ordered set of patches
4326 """import an ordered set of patches
4327
4327
4328 Import a list of patches and commit them individually (unless
4328 Import a list of patches and commit them individually (unless
4329 --no-commit is specified).
4329 --no-commit is specified).
4330
4330
4331 Because import first applies changes to the working directory,
4331 Because import first applies changes to the working directory,
4332 import will abort if there are outstanding changes.
4332 import will abort if there are outstanding changes.
4333
4333
4334 You can import a patch straight from a mail message. Even patches
4334 You can import a patch straight from a mail message. Even patches
4335 as attachments work (to use the body part, it must have type
4335 as attachments work (to use the body part, it must have type
4336 text/plain or text/x-patch). From and Subject headers of email
4336 text/plain or text/x-patch). From and Subject headers of email
4337 message are used as default committer and commit message. All
4337 message are used as default committer and commit message. All
4338 text/plain body parts before first diff are added to commit
4338 text/plain body parts before first diff are added to commit
4339 message.
4339 message.
4340
4340
4341 If the imported patch was generated by :hg:`export`, user and
4341 If the imported patch was generated by :hg:`export`, user and
4342 description from patch override values from message headers and
4342 description from patch override values from message headers and
4343 body. Values given on command line with -m/--message and -u/--user
4343 body. Values given on command line with -m/--message and -u/--user
4344 override these.
4344 override these.
4345
4345
4346 If --exact is specified, import will set the working directory to
4346 If --exact is specified, import will set the working directory to
4347 the parent of each patch before applying it, and will abort if the
4347 the parent of each patch before applying it, and will abort if the
4348 resulting changeset has a different ID than the one recorded in
4348 resulting changeset has a different ID than the one recorded in
4349 the patch. This may happen due to character set problems or other
4349 the patch. This may happen due to character set problems or other
4350 deficiencies in the text patch format.
4350 deficiencies in the text patch format.
4351
4351
4352 Use --bypass to apply and commit patches directly to the
4352 Use --bypass to apply and commit patches directly to the
4353 repository, not touching the working directory. Without --exact,
4353 repository, not touching the working directory. Without --exact,
4354 patches will be applied on top of the working directory parent
4354 patches will be applied on top of the working directory parent
4355 revision.
4355 revision.
4356
4356
4357 With -s/--similarity, hg will attempt to discover renames and
4357 With -s/--similarity, hg will attempt to discover renames and
4358 copies in the patch in the same way as :hg:`addremove`.
4358 copies in the patch in the same way as :hg:`addremove`.
4359
4359
4360 Use --partial to ensure a changeset will be created from the patch
4360 Use --partial to ensure a changeset will be created from the patch
4361 even if some hunks fail to apply. Hunks that fail to apply will be
4361 even if some hunks fail to apply. Hunks that fail to apply will be
4362 written to a <target-file>.rej file. Conflicts can then be resolved
4362 written to a <target-file>.rej file. Conflicts can then be resolved
4363 by hand before :hg:`commit --amend` is run to update the created
4363 by hand before :hg:`commit --amend` is run to update the created
4364 changeset. This flag exists to let people import patches that
4364 changeset. This flag exists to let people import patches that
4365 partially apply without losing the associated metadata (author,
4365 partially apply without losing the associated metadata (author,
4366 date, description, ...). Note that when none of the hunk applies
4366 date, description, ...). Note that when none of the hunk applies
4367 cleanly, :hg:`import --partial` will create an empty changeset,
4367 cleanly, :hg:`import --partial` will create an empty changeset,
4368 importing only the patch metadata.
4368 importing only the patch metadata.
4369
4369
4370 It is possible to use external patch programs to perform the patch
4370 It is possible to use external patch programs to perform the patch
4371 by setting the ``ui.patch`` configuration option. For the default
4371 by setting the ``ui.patch`` configuration option. For the default
4372 internal tool, the fuzz can also be configured via ``patch.fuzz``.
4372 internal tool, the fuzz can also be configured via ``patch.fuzz``.
4373 See :hg:`help config` for more information about configuration
4373 See :hg:`help config` for more information about configuration
4374 files and how to use these options.
4374 files and how to use these options.
4375
4375
4376 To read a patch from standard input, use "-" as the patch name. If
4376 To read a patch from standard input, use "-" as the patch name. If
4377 a URL is specified, the patch will be downloaded from it.
4377 a URL is specified, the patch will be downloaded from it.
4378 See :hg:`help dates` for a list of formats valid for -d/--date.
4378 See :hg:`help dates` for a list of formats valid for -d/--date.
4379
4379
4380 .. container:: verbose
4380 .. container:: verbose
4381
4381
4382 Examples:
4382 Examples:
4383
4383
4384 - import a traditional patch from a website and detect renames::
4384 - import a traditional patch from a website and detect renames::
4385
4385
4386 hg import -s 80 http://example.com/bugfix.patch
4386 hg import -s 80 http://example.com/bugfix.patch
4387
4387
4388 - import a changeset from an hgweb server::
4388 - import a changeset from an hgweb server::
4389
4389
4390 hg import http://www.selenic.com/hg/rev/5ca8c111e9aa
4390 hg import http://www.selenic.com/hg/rev/5ca8c111e9aa
4391
4391
4392 - import all the patches in an Unix-style mbox::
4392 - import all the patches in an Unix-style mbox::
4393
4393
4394 hg import incoming-patches.mbox
4394 hg import incoming-patches.mbox
4395
4395
4396 - attempt to exactly restore an exported changeset (not always
4396 - attempt to exactly restore an exported changeset (not always
4397 possible)::
4397 possible)::
4398
4398
4399 hg import --exact proposed-fix.patch
4399 hg import --exact proposed-fix.patch
4400
4400
4401 - use an external tool to apply a patch which is too fuzzy for
4401 - use an external tool to apply a patch which is too fuzzy for
4402 the default internal tool.
4402 the default internal tool.
4403
4403
4404 hg import --config ui.patch="patch --merge" fuzzy.patch
4404 hg import --config ui.patch="patch --merge" fuzzy.patch
4405
4405
4406 - change the default fuzzing from 2 to a less strict 7
4406 - change the default fuzzing from 2 to a less strict 7
4407
4407
4408 hg import --config ui.fuzz=7 fuzz.patch
4408 hg import --config ui.fuzz=7 fuzz.patch
4409
4409
4410 Returns 0 on success, 1 on partial success (see --partial).
4410 Returns 0 on success, 1 on partial success (see --partial).
4411 """
4411 """
4412
4412
4413 if not patch1:
4413 if not patch1:
4414 raise error.Abort(_('need at least one patch to import'))
4414 raise error.Abort(_('need at least one patch to import'))
4415
4415
4416 patches = (patch1,) + patches
4416 patches = (patch1,) + patches
4417
4417
4418 date = opts.get('date')
4418 date = opts.get('date')
4419 if date:
4419 if date:
4420 opts['date'] = util.parsedate(date)
4420 opts['date'] = util.parsedate(date)
4421
4421
4422 update = not opts.get('bypass')
4422 update = not opts.get('bypass')
4423 if not update and opts.get('no_commit'):
4423 if not update and opts.get('no_commit'):
4424 raise error.Abort(_('cannot use --no-commit with --bypass'))
4424 raise error.Abort(_('cannot use --no-commit with --bypass'))
4425 try:
4425 try:
4426 sim = float(opts.get('similarity') or 0)
4426 sim = float(opts.get('similarity') or 0)
4427 except ValueError:
4427 except ValueError:
4428 raise error.Abort(_('similarity must be a number'))
4428 raise error.Abort(_('similarity must be a number'))
4429 if sim < 0 or sim > 100:
4429 if sim < 0 or sim > 100:
4430 raise error.Abort(_('similarity must be between 0 and 100'))
4430 raise error.Abort(_('similarity must be between 0 and 100'))
4431 if sim and not update:
4431 if sim and not update:
4432 raise error.Abort(_('cannot use --similarity with --bypass'))
4432 raise error.Abort(_('cannot use --similarity with --bypass'))
4433 if opts.get('exact') and opts.get('edit'):
4433 if opts.get('exact') and opts.get('edit'):
4434 raise error.Abort(_('cannot use --exact with --edit'))
4434 raise error.Abort(_('cannot use --exact with --edit'))
4435 if opts.get('exact') and opts.get('prefix'):
4435 if opts.get('exact') and opts.get('prefix'):
4436 raise error.Abort(_('cannot use --exact with --prefix'))
4436 raise error.Abort(_('cannot use --exact with --prefix'))
4437
4437
4438 if update:
4438 if update:
4439 cmdutil.checkunfinished(repo)
4439 cmdutil.checkunfinished(repo)
4440 if (opts.get('exact') or not opts.get('force')) and update:
4440 if (opts.get('exact') or not opts.get('force')) and update:
4441 cmdutil.bailifchanged(repo)
4441 cmdutil.bailifchanged(repo)
4442
4442
4443 base = opts["base"]
4443 base = opts["base"]
4444 wlock = dsguard = lock = tr = None
4444 wlock = dsguard = lock = tr = None
4445 msgs = []
4445 msgs = []
4446 ret = 0
4446 ret = 0
4447
4447
4448
4448
4449 try:
4449 try:
4450 try:
4450 try:
4451 wlock = repo.wlock()
4451 wlock = repo.wlock()
4452 if not opts.get('no_commit'):
4452 if not opts.get('no_commit'):
4453 lock = repo.lock()
4453 lock = repo.lock()
4454 tr = repo.transaction('import')
4454 tr = repo.transaction('import')
4455 else:
4455 else:
4456 dsguard = cmdutil.dirstateguard(repo, 'import')
4456 dsguard = cmdutil.dirstateguard(repo, 'import')
4457 parents = repo.parents()
4457 parents = repo.parents()
4458 for patchurl in patches:
4458 for patchurl in patches:
4459 if patchurl == '-':
4459 if patchurl == '-':
4460 ui.status(_('applying patch from stdin\n'))
4460 ui.status(_('applying patch from stdin\n'))
4461 patchfile = ui.fin
4461 patchfile = ui.fin
4462 patchurl = 'stdin' # for error message
4462 patchurl = 'stdin' # for error message
4463 else:
4463 else:
4464 patchurl = os.path.join(base, patchurl)
4464 patchurl = os.path.join(base, patchurl)
4465 ui.status(_('applying %s\n') % patchurl)
4465 ui.status(_('applying %s\n') % patchurl)
4466 patchfile = hg.openpath(ui, patchurl)
4466 patchfile = hg.openpath(ui, patchurl)
4467
4467
4468 haspatch = False
4468 haspatch = False
4469 for hunk in patch.split(patchfile):
4469 for hunk in patch.split(patchfile):
4470 (msg, node, rej) = cmdutil.tryimportone(ui, repo, hunk,
4470 (msg, node, rej) = cmdutil.tryimportone(ui, repo, hunk,
4471 parents, opts,
4471 parents, opts,
4472 msgs, hg.clean)
4472 msgs, hg.clean)
4473 if msg:
4473 if msg:
4474 haspatch = True
4474 haspatch = True
4475 ui.note(msg + '\n')
4475 ui.note(msg + '\n')
4476 if update or opts.get('exact'):
4476 if update or opts.get('exact'):
4477 parents = repo.parents()
4477 parents = repo.parents()
4478 else:
4478 else:
4479 parents = [repo[node]]
4479 parents = [repo[node]]
4480 if rej:
4480 if rej:
4481 ui.write_err(_("patch applied partially\n"))
4481 ui.write_err(_("patch applied partially\n"))
4482 ui.write_err(_("(fix the .rej files and run "
4482 ui.write_err(_("(fix the .rej files and run "
4483 "`hg commit --amend`)\n"))
4483 "`hg commit --amend`)\n"))
4484 ret = 1
4484 ret = 1
4485 break
4485 break
4486
4486
4487 if not haspatch:
4487 if not haspatch:
4488 raise error.Abort(_('%s: no diffs found') % patchurl)
4488 raise error.Abort(_('%s: no diffs found') % patchurl)
4489
4489
4490 if tr:
4490 if tr:
4491 tr.close()
4491 tr.close()
4492 if msgs:
4492 if msgs:
4493 repo.savecommitmessage('\n* * *\n'.join(msgs))
4493 repo.savecommitmessage('\n* * *\n'.join(msgs))
4494 if dsguard:
4494 if dsguard:
4495 dsguard.close()
4495 dsguard.close()
4496 return ret
4496 return ret
4497 finally:
4497 finally:
4498 # TODO: get rid of this meaningless try/finally enclosing.
4498 # TODO: get rid of this meaningless try/finally enclosing.
4499 # this is kept only to reduce changes in a patch.
4499 # this is kept only to reduce changes in a patch.
4500 pass
4500 pass
4501 finally:
4501 finally:
4502 if tr:
4502 if tr:
4503 tr.release()
4503 tr.release()
4504 release(lock, dsguard, wlock)
4504 release(lock, dsguard, wlock)
4505
4505
4506 @command('incoming|in',
4506 @command('incoming|in',
4507 [('f', 'force', None,
4507 [('f', 'force', None,
4508 _('run even if remote repository is unrelated')),
4508 _('run even if remote repository is unrelated')),
4509 ('n', 'newest-first', None, _('show newest record first')),
4509 ('n', 'newest-first', None, _('show newest record first')),
4510 ('', 'bundle', '',
4510 ('', 'bundle', '',
4511 _('file to store the bundles into'), _('FILE')),
4511 _('file to store the bundles into'), _('FILE')),
4512 ('r', 'rev', [], _('a remote changeset intended to be added'), _('REV')),
4512 ('r', 'rev', [], _('a remote changeset intended to be added'), _('REV')),
4513 ('B', 'bookmarks', False, _("compare bookmarks")),
4513 ('B', 'bookmarks', False, _("compare bookmarks")),
4514 ('b', 'branch', [],
4514 ('b', 'branch', [],
4515 _('a specific branch you would like to pull'), _('BRANCH')),
4515 _('a specific branch you would like to pull'), _('BRANCH')),
4516 ] + logopts + remoteopts + subrepoopts,
4516 ] + logopts + remoteopts + subrepoopts,
4517 _('[-p] [-n] [-M] [-f] [-r REV]... [--bundle FILENAME] [SOURCE]'))
4517 _('[-p] [-n] [-M] [-f] [-r REV]... [--bundle FILENAME] [SOURCE]'))
4518 def incoming(ui, repo, source="default", **opts):
4518 def incoming(ui, repo, source="default", **opts):
4519 """show new changesets found in source
4519 """show new changesets found in source
4520
4520
4521 Show new changesets found in the specified path/URL or the default
4521 Show new changesets found in the specified path/URL or the default
4522 pull location. These are the changesets that would have been pulled
4522 pull location. These are the changesets that would have been pulled
4523 if a pull at the time you issued this command.
4523 if a pull at the time you issued this command.
4524
4524
4525 See pull for valid source format details.
4525 See pull for valid source format details.
4526
4526
4527 .. container:: verbose
4527 .. container:: verbose
4528
4528
4529 With -B/--bookmarks, the result of bookmark comparison between
4529 With -B/--bookmarks, the result of bookmark comparison between
4530 local and remote repositories is displayed. With -v/--verbose,
4530 local and remote repositories is displayed. With -v/--verbose,
4531 status is also displayed for each bookmark like below::
4531 status is also displayed for each bookmark like below::
4532
4532
4533 BM1 01234567890a added
4533 BM1 01234567890a added
4534 BM2 1234567890ab advanced
4534 BM2 1234567890ab advanced
4535 BM3 234567890abc diverged
4535 BM3 234567890abc diverged
4536 BM4 34567890abcd changed
4536 BM4 34567890abcd changed
4537
4537
4538 The action taken locally when pulling depends on the
4538 The action taken locally when pulling depends on the
4539 status of each bookmark:
4539 status of each bookmark:
4540
4540
4541 :``added``: pull will create it
4541 :``added``: pull will create it
4542 :``advanced``: pull will update it
4542 :``advanced``: pull will update it
4543 :``diverged``: pull will create a divergent bookmark
4543 :``diverged``: pull will create a divergent bookmark
4544 :``changed``: result depends on remote changesets
4544 :``changed``: result depends on remote changesets
4545
4545
4546 From the point of view of pulling behavior, bookmark
4546 From the point of view of pulling behavior, bookmark
4547 existing only in the remote repository are treated as ``added``,
4547 existing only in the remote repository are treated as ``added``,
4548 even if it is in fact locally deleted.
4548 even if it is in fact locally deleted.
4549
4549
4550 .. container:: verbose
4550 .. container:: verbose
4551
4551
4552 For remote repository, using --bundle avoids downloading the
4552 For remote repository, using --bundle avoids downloading the
4553 changesets twice if the incoming is followed by a pull.
4553 changesets twice if the incoming is followed by a pull.
4554
4554
4555 Examples:
4555 Examples:
4556
4556
4557 - show incoming changes with patches and full description::
4557 - show incoming changes with patches and full description::
4558
4558
4559 hg incoming -vp
4559 hg incoming -vp
4560
4560
4561 - show incoming changes excluding merges, store a bundle::
4561 - show incoming changes excluding merges, store a bundle::
4562
4562
4563 hg in -vpM --bundle incoming.hg
4563 hg in -vpM --bundle incoming.hg
4564 hg pull incoming.hg
4564 hg pull incoming.hg
4565
4565
4566 - briefly list changes inside a bundle::
4566 - briefly list changes inside a bundle::
4567
4567
4568 hg in changes.hg -T "{desc|firstline}\\n"
4568 hg in changes.hg -T "{desc|firstline}\\n"
4569
4569
4570 Returns 0 if there are incoming changes, 1 otherwise.
4570 Returns 0 if there are incoming changes, 1 otherwise.
4571 """
4571 """
4572 if opts.get('graph'):
4572 if opts.get('graph'):
4573 cmdutil.checkunsupportedgraphflags([], opts)
4573 cmdutil.checkunsupportedgraphflags([], opts)
4574 def display(other, chlist, displayer):
4574 def display(other, chlist, displayer):
4575 revdag = cmdutil.graphrevs(other, chlist, opts)
4575 revdag = cmdutil.graphrevs(other, chlist, opts)
4576 showparents = [ctx.node() for ctx in repo[None].parents()]
4576 showparents = [ctx.node() for ctx in repo[None].parents()]
4577 cmdutil.displaygraph(ui, revdag, displayer, showparents,
4577 cmdutil.displaygraph(ui, revdag, displayer, showparents,
4578 graphmod.asciiedges)
4578 graphmod.asciiedges)
4579
4579
4580 hg._incoming(display, lambda: 1, ui, repo, source, opts, buffered=True)
4580 hg._incoming(display, lambda: 1, ui, repo, source, opts, buffered=True)
4581 return 0
4581 return 0
4582
4582
4583 if opts.get('bundle') and opts.get('subrepos'):
4583 if opts.get('bundle') and opts.get('subrepos'):
4584 raise error.Abort(_('cannot combine --bundle and --subrepos'))
4584 raise error.Abort(_('cannot combine --bundle and --subrepos'))
4585
4585
4586 if opts.get('bookmarks'):
4586 if opts.get('bookmarks'):
4587 source, branches = hg.parseurl(ui.expandpath(source),
4587 source, branches = hg.parseurl(ui.expandpath(source),
4588 opts.get('branch'))
4588 opts.get('branch'))
4589 other = hg.peer(repo, opts, source)
4589 other = hg.peer(repo, opts, source)
4590 if 'bookmarks' not in other.listkeys('namespaces'):
4590 if 'bookmarks' not in other.listkeys('namespaces'):
4591 ui.warn(_("remote doesn't support bookmarks\n"))
4591 ui.warn(_("remote doesn't support bookmarks\n"))
4592 return 0
4592 return 0
4593 ui.status(_('comparing with %s\n') % util.hidepassword(source))
4593 ui.status(_('comparing with %s\n') % util.hidepassword(source))
4594 return bookmarks.incoming(ui, repo, other)
4594 return bookmarks.incoming(ui, repo, other)
4595
4595
4596 repo._subtoppath = ui.expandpath(source)
4596 repo._subtoppath = ui.expandpath(source)
4597 try:
4597 try:
4598 return hg.incoming(ui, repo, source, opts)
4598 return hg.incoming(ui, repo, source, opts)
4599 finally:
4599 finally:
4600 del repo._subtoppath
4600 del repo._subtoppath
4601
4601
4602
4602
4603 @command('^init', remoteopts, _('[-e CMD] [--remotecmd CMD] [DEST]'),
4603 @command('^init', remoteopts, _('[-e CMD] [--remotecmd CMD] [DEST]'),
4604 norepo=True)
4604 norepo=True)
4605 def init(ui, dest=".", **opts):
4605 def init(ui, dest=".", **opts):
4606 """create a new repository in the given directory
4606 """create a new repository in the given directory
4607
4607
4608 Initialize a new repository in the given directory. If the given
4608 Initialize a new repository in the given directory. If the given
4609 directory does not exist, it will be created.
4609 directory does not exist, it will be created.
4610
4610
4611 If no directory is given, the current directory is used.
4611 If no directory is given, the current directory is used.
4612
4612
4613 It is possible to specify an ``ssh://`` URL as the destination.
4613 It is possible to specify an ``ssh://`` URL as the destination.
4614 See :hg:`help urls` for more information.
4614 See :hg:`help urls` for more information.
4615
4615
4616 Returns 0 on success.
4616 Returns 0 on success.
4617 """
4617 """
4618 hg.peer(ui, opts, ui.expandpath(dest), create=True)
4618 hg.peer(ui, opts, ui.expandpath(dest), create=True)
4619
4619
4620 @command('locate',
4620 @command('locate',
4621 [('r', 'rev', '', _('search the repository as it is in REV'), _('REV')),
4621 [('r', 'rev', '', _('search the repository as it is in REV'), _('REV')),
4622 ('0', 'print0', None, _('end filenames with NUL, for use with xargs')),
4622 ('0', 'print0', None, _('end filenames with NUL, for use with xargs')),
4623 ('f', 'fullpath', None, _('print complete paths from the filesystem root')),
4623 ('f', 'fullpath', None, _('print complete paths from the filesystem root')),
4624 ] + walkopts,
4624 ] + walkopts,
4625 _('[OPTION]... [PATTERN]...'))
4625 _('[OPTION]... [PATTERN]...'))
4626 def locate(ui, repo, *pats, **opts):
4626 def locate(ui, repo, *pats, **opts):
4627 """locate files matching specific patterns (DEPRECATED)
4627 """locate files matching specific patterns (DEPRECATED)
4628
4628
4629 Print files under Mercurial control in the working directory whose
4629 Print files under Mercurial control in the working directory whose
4630 names match the given patterns.
4630 names match the given patterns.
4631
4631
4632 By default, this command searches all directories in the working
4632 By default, this command searches all directories in the working
4633 directory. To search just the current directory and its
4633 directory. To search just the current directory and its
4634 subdirectories, use "--include .".
4634 subdirectories, use "--include .".
4635
4635
4636 If no patterns are given to match, this command prints the names
4636 If no patterns are given to match, this command prints the names
4637 of all files under Mercurial control in the working directory.
4637 of all files under Mercurial control in the working directory.
4638
4638
4639 If you want to feed the output of this command into the "xargs"
4639 If you want to feed the output of this command into the "xargs"
4640 command, use the -0 option to both this command and "xargs". This
4640 command, use the -0 option to both this command and "xargs". This
4641 will avoid the problem of "xargs" treating single filenames that
4641 will avoid the problem of "xargs" treating single filenames that
4642 contain whitespace as multiple filenames.
4642 contain whitespace as multiple filenames.
4643
4643
4644 See :hg:`help files` for a more versatile command.
4644 See :hg:`help files` for a more versatile command.
4645
4645
4646 Returns 0 if a match is found, 1 otherwise.
4646 Returns 0 if a match is found, 1 otherwise.
4647 """
4647 """
4648 if opts.get('print0'):
4648 if opts.get('print0'):
4649 end = '\0'
4649 end = '\0'
4650 else:
4650 else:
4651 end = '\n'
4651 end = '\n'
4652 rev = scmutil.revsingle(repo, opts.get('rev'), None).node()
4652 rev = scmutil.revsingle(repo, opts.get('rev'), None).node()
4653
4653
4654 ret = 1
4654 ret = 1
4655 ctx = repo[rev]
4655 ctx = repo[rev]
4656 m = scmutil.match(ctx, pats, opts, default='relglob',
4656 m = scmutil.match(ctx, pats, opts, default='relglob',
4657 badfn=lambda x, y: False)
4657 badfn=lambda x, y: False)
4658
4658
4659 for abs in ctx.matches(m):
4659 for abs in ctx.matches(m):
4660 if opts.get('fullpath'):
4660 if opts.get('fullpath'):
4661 ui.write(repo.wjoin(abs), end)
4661 ui.write(repo.wjoin(abs), end)
4662 else:
4662 else:
4663 ui.write(((pats and m.rel(abs)) or abs), end)
4663 ui.write(((pats and m.rel(abs)) or abs), end)
4664 ret = 0
4664 ret = 0
4665
4665
4666 return ret
4666 return ret
4667
4667
4668 @command('^log|history',
4668 @command('^log|history',
4669 [('f', 'follow', None,
4669 [('f', 'follow', None,
4670 _('follow changeset history, or file history across copies and renames')),
4670 _('follow changeset history, or file history across copies and renames')),
4671 ('', 'follow-first', None,
4671 ('', 'follow-first', None,
4672 _('only follow the first parent of merge changesets (DEPRECATED)')),
4672 _('only follow the first parent of merge changesets (DEPRECATED)')),
4673 ('d', 'date', '', _('show revisions matching date spec'), _('DATE')),
4673 ('d', 'date', '', _('show revisions matching date spec'), _('DATE')),
4674 ('C', 'copies', None, _('show copied files')),
4674 ('C', 'copies', None, _('show copied files')),
4675 ('k', 'keyword', [],
4675 ('k', 'keyword', [],
4676 _('do case-insensitive search for a given text'), _('TEXT')),
4676 _('do case-insensitive search for a given text'), _('TEXT')),
4677 ('r', 'rev', [], _('show the specified revision or revset'), _('REV')),
4677 ('r', 'rev', [], _('show the specified revision or revset'), _('REV')),
4678 ('', 'removed', None, _('include revisions where files were removed')),
4678 ('', 'removed', None, _('include revisions where files were removed')),
4679 ('m', 'only-merges', None, _('show only merges (DEPRECATED)')),
4679 ('m', 'only-merges', None, _('show only merges (DEPRECATED)')),
4680 ('u', 'user', [], _('revisions committed by user'), _('USER')),
4680 ('u', 'user', [], _('revisions committed by user'), _('USER')),
4681 ('', 'only-branch', [],
4681 ('', 'only-branch', [],
4682 _('show only changesets within the given named branch (DEPRECATED)'),
4682 _('show only changesets within the given named branch (DEPRECATED)'),
4683 _('BRANCH')),
4683 _('BRANCH')),
4684 ('b', 'branch', [],
4684 ('b', 'branch', [],
4685 _('show changesets within the given named branch'), _('BRANCH')),
4685 _('show changesets within the given named branch'), _('BRANCH')),
4686 ('P', 'prune', [],
4686 ('P', 'prune', [],
4687 _('do not display revision or any of its ancestors'), _('REV')),
4687 _('do not display revision or any of its ancestors'), _('REV')),
4688 ] + logopts + walkopts,
4688 ] + logopts + walkopts,
4689 _('[OPTION]... [FILE]'),
4689 _('[OPTION]... [FILE]'),
4690 inferrepo=True)
4690 inferrepo=True)
4691 def log(ui, repo, *pats, **opts):
4691 def log(ui, repo, *pats, **opts):
4692 """show revision history of entire repository or files
4692 """show revision history of entire repository or files
4693
4693
4694 Print the revision history of the specified files or the entire
4694 Print the revision history of the specified files or the entire
4695 project.
4695 project.
4696
4696
4697 If no revision range is specified, the default is ``tip:0`` unless
4697 If no revision range is specified, the default is ``tip:0`` unless
4698 --follow is set, in which case the working directory parent is
4698 --follow is set, in which case the working directory parent is
4699 used as the starting revision.
4699 used as the starting revision.
4700
4700
4701 File history is shown without following rename or copy history of
4701 File history is shown without following rename or copy history of
4702 files. Use -f/--follow with a filename to follow history across
4702 files. Use -f/--follow with a filename to follow history across
4703 renames and copies. --follow without a filename will only show
4703 renames and copies. --follow without a filename will only show
4704 ancestors or descendants of the starting revision.
4704 ancestors or descendants of the starting revision.
4705
4705
4706 By default this command prints revision number and changeset id,
4706 By default this command prints revision number and changeset id,
4707 tags, non-trivial parents, user, date and time, and a summary for
4707 tags, non-trivial parents, user, date and time, and a summary for
4708 each commit. When the -v/--verbose switch is used, the list of
4708 each commit. When the -v/--verbose switch is used, the list of
4709 changed files and full commit message are shown.
4709 changed files and full commit message are shown.
4710
4710
4711 With --graph the revisions are shown as an ASCII art DAG with the most
4711 With --graph the revisions are shown as an ASCII art DAG with the most
4712 recent changeset at the top.
4712 recent changeset at the top.
4713 'o' is a changeset, '@' is a working directory parent, 'x' is obsolete,
4713 'o' is a changeset, '@' is a working directory parent, 'x' is obsolete,
4714 and '+' represents a fork where the changeset from the lines below is a
4714 and '+' represents a fork where the changeset from the lines below is a
4715 parent of the 'o' merge on the same line.
4715 parent of the 'o' merge on the same line.
4716
4716
4717 .. note::
4717 .. note::
4718
4718
4719 log -p/--patch may generate unexpected diff output for merge
4719 log -p/--patch may generate unexpected diff output for merge
4720 changesets, as it will only compare the merge changeset against
4720 changesets, as it will only compare the merge changeset against
4721 its first parent. Also, only files different from BOTH parents
4721 its first parent. Also, only files different from BOTH parents
4722 will appear in files:.
4722 will appear in files:.
4723
4723
4724 .. note::
4724 .. note::
4725
4725
4726 for performance reasons, log FILE may omit duplicate changes
4726 for performance reasons, log FILE may omit duplicate changes
4727 made on branches and will not show removals or mode changes. To
4727 made on branches and will not show removals or mode changes. To
4728 see all such changes, use the --removed switch.
4728 see all such changes, use the --removed switch.
4729
4729
4730 .. container:: verbose
4730 .. container:: verbose
4731
4731
4732 Some examples:
4732 Some examples:
4733
4733
4734 - changesets with full descriptions and file lists::
4734 - changesets with full descriptions and file lists::
4735
4735
4736 hg log -v
4736 hg log -v
4737
4737
4738 - changesets ancestral to the working directory::
4738 - changesets ancestral to the working directory::
4739
4739
4740 hg log -f
4740 hg log -f
4741
4741
4742 - last 10 commits on the current branch::
4742 - last 10 commits on the current branch::
4743
4743
4744 hg log -l 10 -b .
4744 hg log -l 10 -b .
4745
4745
4746 - changesets showing all modifications of a file, including removals::
4746 - changesets showing all modifications of a file, including removals::
4747
4747
4748 hg log --removed file.c
4748 hg log --removed file.c
4749
4749
4750 - all changesets that touch a directory, with diffs, excluding merges::
4750 - all changesets that touch a directory, with diffs, excluding merges::
4751
4751
4752 hg log -Mp lib/
4752 hg log -Mp lib/
4753
4753
4754 - all revision numbers that match a keyword::
4754 - all revision numbers that match a keyword::
4755
4755
4756 hg log -k bug --template "{rev}\\n"
4756 hg log -k bug --template "{rev}\\n"
4757
4757
4758 - list available log templates::
4758 - list available log templates::
4759
4759
4760 hg log -T list
4760 hg log -T list
4761
4761
4762 - check if a given changeset is included in a tagged release::
4762 - check if a given changeset is included in a tagged release::
4763
4763
4764 hg log -r "a21ccf and ancestor(1.9)"
4764 hg log -r "a21ccf and ancestor(1.9)"
4765
4765
4766 - find all changesets by some user in a date range::
4766 - find all changesets by some user in a date range::
4767
4767
4768 hg log -k alice -d "may 2008 to jul 2008"
4768 hg log -k alice -d "may 2008 to jul 2008"
4769
4769
4770 - summary of all changesets after the last tag::
4770 - summary of all changesets after the last tag::
4771
4771
4772 hg log -r "last(tagged())::" --template "{desc|firstline}\\n"
4772 hg log -r "last(tagged())::" --template "{desc|firstline}\\n"
4773
4773
4774 See :hg:`help dates` for a list of formats valid for -d/--date.
4774 See :hg:`help dates` for a list of formats valid for -d/--date.
4775
4775
4776 See :hg:`help revisions` and :hg:`help revsets` for more about
4776 See :hg:`help revisions` and :hg:`help revsets` for more about
4777 specifying revisions.
4777 specifying revisions.
4778
4778
4779 See :hg:`help templates` for more about pre-packaged styles and
4779 See :hg:`help templates` for more about pre-packaged styles and
4780 specifying custom templates.
4780 specifying custom templates.
4781
4781
4782 Returns 0 on success.
4782 Returns 0 on success.
4783
4783
4784 """
4784 """
4785 if opts.get('follow') and opts.get('rev'):
4785 if opts.get('follow') and opts.get('rev'):
4786 opts['rev'] = [revset.formatspec('reverse(::%lr)', opts.get('rev'))]
4786 opts['rev'] = [revset.formatspec('reverse(::%lr)', opts.get('rev'))]
4787 del opts['follow']
4787 del opts['follow']
4788
4788
4789 if opts.get('graph'):
4789 if opts.get('graph'):
4790 return cmdutil.graphlog(ui, repo, *pats, **opts)
4790 return cmdutil.graphlog(ui, repo, *pats, **opts)
4791
4791
4792 revs, expr, filematcher = cmdutil.getlogrevs(repo, pats, opts)
4792 revs, expr, filematcher = cmdutil.getlogrevs(repo, pats, opts)
4793 limit = cmdutil.loglimit(opts)
4793 limit = cmdutil.loglimit(opts)
4794 count = 0
4794 count = 0
4795
4795
4796 getrenamed = None
4796 getrenamed = None
4797 if opts.get('copies'):
4797 if opts.get('copies'):
4798 endrev = None
4798 endrev = None
4799 if opts.get('rev'):
4799 if opts.get('rev'):
4800 endrev = scmutil.revrange(repo, opts.get('rev')).max() + 1
4800 endrev = scmutil.revrange(repo, opts.get('rev')).max() + 1
4801 getrenamed = templatekw.getrenamedfn(repo, endrev=endrev)
4801 getrenamed = templatekw.getrenamedfn(repo, endrev=endrev)
4802
4802
4803 displayer = cmdutil.show_changeset(ui, repo, opts, buffered=True)
4803 displayer = cmdutil.show_changeset(ui, repo, opts, buffered=True)
4804 for rev in revs:
4804 for rev in revs:
4805 if count == limit:
4805 if count == limit:
4806 break
4806 break
4807 ctx = repo[rev]
4807 ctx = repo[rev]
4808 copies = None
4808 copies = None
4809 if getrenamed is not None and rev:
4809 if getrenamed is not None and rev:
4810 copies = []
4810 copies = []
4811 for fn in ctx.files():
4811 for fn in ctx.files():
4812 rename = getrenamed(fn, rev)
4812 rename = getrenamed(fn, rev)
4813 if rename:
4813 if rename:
4814 copies.append((fn, rename[0]))
4814 copies.append((fn, rename[0]))
4815 if filematcher:
4815 if filematcher:
4816 revmatchfn = filematcher(ctx.rev())
4816 revmatchfn = filematcher(ctx.rev())
4817 else:
4817 else:
4818 revmatchfn = None
4818 revmatchfn = None
4819 displayer.show(ctx, copies=copies, matchfn=revmatchfn)
4819 displayer.show(ctx, copies=copies, matchfn=revmatchfn)
4820 if displayer.flush(ctx):
4820 if displayer.flush(ctx):
4821 count += 1
4821 count += 1
4822
4822
4823 displayer.close()
4823 displayer.close()
4824
4824
4825 @command('manifest',
4825 @command('manifest',
4826 [('r', 'rev', '', _('revision to display'), _('REV')),
4826 [('r', 'rev', '', _('revision to display'), _('REV')),
4827 ('', 'all', False, _("list files from all revisions"))]
4827 ('', 'all', False, _("list files from all revisions"))]
4828 + formatteropts,
4828 + formatteropts,
4829 _('[-r REV]'))
4829 _('[-r REV]'))
4830 def manifest(ui, repo, node=None, rev=None, **opts):
4830 def manifest(ui, repo, node=None, rev=None, **opts):
4831 """output the current or given revision of the project manifest
4831 """output the current or given revision of the project manifest
4832
4832
4833 Print a list of version controlled files for the given revision.
4833 Print a list of version controlled files for the given revision.
4834 If no revision is given, the first parent of the working directory
4834 If no revision is given, the first parent of the working directory
4835 is used, or the null revision if no revision is checked out.
4835 is used, or the null revision if no revision is checked out.
4836
4836
4837 With -v, print file permissions, symlink and executable bits.
4837 With -v, print file permissions, symlink and executable bits.
4838 With --debug, print file revision hashes.
4838 With --debug, print file revision hashes.
4839
4839
4840 If option --all is specified, the list of all files from all revisions
4840 If option --all is specified, the list of all files from all revisions
4841 is printed. This includes deleted and renamed files.
4841 is printed. This includes deleted and renamed files.
4842
4842
4843 Returns 0 on success.
4843 Returns 0 on success.
4844 """
4844 """
4845
4845
4846 fm = ui.formatter('manifest', opts)
4846 fm = ui.formatter('manifest', opts)
4847
4847
4848 if opts.get('all'):
4848 if opts.get('all'):
4849 if rev or node:
4849 if rev or node:
4850 raise error.Abort(_("can't specify a revision with --all"))
4850 raise error.Abort(_("can't specify a revision with --all"))
4851
4851
4852 res = []
4852 res = []
4853 prefix = "data/"
4853 prefix = "data/"
4854 suffix = ".i"
4854 suffix = ".i"
4855 plen = len(prefix)
4855 plen = len(prefix)
4856 slen = len(suffix)
4856 slen = len(suffix)
4857 lock = repo.lock()
4857 lock = repo.lock()
4858 try:
4858 try:
4859 for fn, b, size in repo.store.datafiles():
4859 for fn, b, size in repo.store.datafiles():
4860 if size != 0 and fn[-slen:] == suffix and fn[:plen] == prefix:
4860 if size != 0 and fn[-slen:] == suffix and fn[:plen] == prefix:
4861 res.append(fn[plen:-slen])
4861 res.append(fn[plen:-slen])
4862 finally:
4862 finally:
4863 lock.release()
4863 lock.release()
4864 for f in res:
4864 for f in res:
4865 fm.startitem()
4865 fm.startitem()
4866 fm.write("path", '%s\n', f)
4866 fm.write("path", '%s\n', f)
4867 fm.end()
4867 fm.end()
4868 return
4868 return
4869
4869
4870 if rev and node:
4870 if rev and node:
4871 raise error.Abort(_("please specify just one revision"))
4871 raise error.Abort(_("please specify just one revision"))
4872
4872
4873 if not node:
4873 if not node:
4874 node = rev
4874 node = rev
4875
4875
4876 char = {'l': '@', 'x': '*', '': ''}
4876 char = {'l': '@', 'x': '*', '': ''}
4877 mode = {'l': '644', 'x': '755', '': '644'}
4877 mode = {'l': '644', 'x': '755', '': '644'}
4878 ctx = scmutil.revsingle(repo, node)
4878 ctx = scmutil.revsingle(repo, node)
4879 mf = ctx.manifest()
4879 mf = ctx.manifest()
4880 for f in ctx:
4880 for f in ctx:
4881 fm.startitem()
4881 fm.startitem()
4882 fl = ctx[f].flags()
4882 fl = ctx[f].flags()
4883 fm.condwrite(ui.debugflag, 'hash', '%s ', hex(mf[f]))
4883 fm.condwrite(ui.debugflag, 'hash', '%s ', hex(mf[f]))
4884 fm.condwrite(ui.verbose, 'mode type', '%s %1s ', mode[fl], char[fl])
4884 fm.condwrite(ui.verbose, 'mode type', '%s %1s ', mode[fl], char[fl])
4885 fm.write('path', '%s\n', f)
4885 fm.write('path', '%s\n', f)
4886 fm.end()
4886 fm.end()
4887
4887
4888 @command('^merge',
4888 @command('^merge',
4889 [('f', 'force', None,
4889 [('f', 'force', None,
4890 _('force a merge including outstanding changes (DEPRECATED)')),
4890 _('force a merge including outstanding changes (DEPRECATED)')),
4891 ('r', 'rev', '', _('revision to merge'), _('REV')),
4891 ('r', 'rev', '', _('revision to merge'), _('REV')),
4892 ('P', 'preview', None,
4892 ('P', 'preview', None,
4893 _('review revisions to merge (no merge is performed)'))
4893 _('review revisions to merge (no merge is performed)'))
4894 ] + mergetoolopts,
4894 ] + mergetoolopts,
4895 _('[-P] [-f] [[-r] REV]'))
4895 _('[-P] [-f] [[-r] REV]'))
4896 def merge(ui, repo, node=None, **opts):
4896 def merge(ui, repo, node=None, **opts):
4897 """merge another revision into working directory
4897 """merge another revision into working directory
4898
4898
4899 The current working directory is updated with all changes made in
4899 The current working directory is updated with all changes made in
4900 the requested revision since the last common predecessor revision.
4900 the requested revision since the last common predecessor revision.
4901
4901
4902 Files that changed between either parent are marked as changed for
4902 Files that changed between either parent are marked as changed for
4903 the next commit and a commit must be performed before any further
4903 the next commit and a commit must be performed before any further
4904 updates to the repository are allowed. The next commit will have
4904 updates to the repository are allowed. The next commit will have
4905 two parents.
4905 two parents.
4906
4906
4907 ``--tool`` can be used to specify the merge tool used for file
4907 ``--tool`` can be used to specify the merge tool used for file
4908 merges. It overrides the HGMERGE environment variable and your
4908 merges. It overrides the HGMERGE environment variable and your
4909 configuration files. See :hg:`help merge-tools` for options.
4909 configuration files. See :hg:`help merge-tools` for options.
4910
4910
4911 If no revision is specified, the working directory's parent is a
4911 If no revision is specified, the working directory's parent is a
4912 head revision, and the current branch contains exactly one other
4912 head revision, and the current branch contains exactly one other
4913 head, the other head is merged with by default. Otherwise, an
4913 head, the other head is merged with by default. Otherwise, an
4914 explicit revision with which to merge with must be provided.
4914 explicit revision with which to merge with must be provided.
4915
4915
4916 :hg:`resolve` must be used to resolve unresolved files.
4916 :hg:`resolve` must be used to resolve unresolved files.
4917
4917
4918 To undo an uncommitted merge, use :hg:`update --clean .` which
4918 To undo an uncommitted merge, use :hg:`update --clean .` which
4919 will check out a clean copy of the original merge parent, losing
4919 will check out a clean copy of the original merge parent, losing
4920 all changes.
4920 all changes.
4921
4921
4922 Returns 0 on success, 1 if there are unresolved files.
4922 Returns 0 on success, 1 if there are unresolved files.
4923 """
4923 """
4924
4924
4925 if opts.get('rev') and node:
4925 if opts.get('rev') and node:
4926 raise error.Abort(_("please specify just one revision"))
4926 raise error.Abort(_("please specify just one revision"))
4927 if not node:
4927 if not node:
4928 node = opts.get('rev')
4928 node = opts.get('rev')
4929
4929
4930 if node:
4930 if node:
4931 node = scmutil.revsingle(repo, node).node()
4931 node = scmutil.revsingle(repo, node).node()
4932
4932
4933 if not node:
4933 if not node:
4934 node = repo[destutil.destmerge(repo)].node()
4934 node = repo[destutil.destmerge(repo)].node()
4935
4935
4936 if opts.get('preview'):
4936 if opts.get('preview'):
4937 # find nodes that are ancestors of p2 but not of p1
4937 # find nodes that are ancestors of p2 but not of p1
4938 p1 = repo.lookup('.')
4938 p1 = repo.lookup('.')
4939 p2 = repo.lookup(node)
4939 p2 = repo.lookup(node)
4940 nodes = repo.changelog.findmissing(common=[p1], heads=[p2])
4940 nodes = repo.changelog.findmissing(common=[p1], heads=[p2])
4941
4941
4942 displayer = cmdutil.show_changeset(ui, repo, opts)
4942 displayer = cmdutil.show_changeset(ui, repo, opts)
4943 for node in nodes:
4943 for node in nodes:
4944 displayer.show(repo[node])
4944 displayer.show(repo[node])
4945 displayer.close()
4945 displayer.close()
4946 return 0
4946 return 0
4947
4947
4948 try:
4948 try:
4949 # ui.forcemerge is an internal variable, do not document
4949 # ui.forcemerge is an internal variable, do not document
4950 repo.ui.setconfig('ui', 'forcemerge', opts.get('tool', ''), 'merge')
4950 repo.ui.setconfig('ui', 'forcemerge', opts.get('tool', ''), 'merge')
4951 return hg.merge(repo, node, force=opts.get('force'))
4951 return hg.merge(repo, node, force=opts.get('force'))
4952 finally:
4952 finally:
4953 ui.setconfig('ui', 'forcemerge', '', 'merge')
4953 ui.setconfig('ui', 'forcemerge', '', 'merge')
4954
4954
4955 @command('outgoing|out',
4955 @command('outgoing|out',
4956 [('f', 'force', None, _('run even when the destination is unrelated')),
4956 [('f', 'force', None, _('run even when the destination is unrelated')),
4957 ('r', 'rev', [],
4957 ('r', 'rev', [],
4958 _('a changeset intended to be included in the destination'), _('REV')),
4958 _('a changeset intended to be included in the destination'), _('REV')),
4959 ('n', 'newest-first', None, _('show newest record first')),
4959 ('n', 'newest-first', None, _('show newest record first')),
4960 ('B', 'bookmarks', False, _('compare bookmarks')),
4960 ('B', 'bookmarks', False, _('compare bookmarks')),
4961 ('b', 'branch', [], _('a specific branch you would like to push'),
4961 ('b', 'branch', [], _('a specific branch you would like to push'),
4962 _('BRANCH')),
4962 _('BRANCH')),
4963 ] + logopts + remoteopts + subrepoopts,
4963 ] + logopts + remoteopts + subrepoopts,
4964 _('[-M] [-p] [-n] [-f] [-r REV]... [DEST]'))
4964 _('[-M] [-p] [-n] [-f] [-r REV]... [DEST]'))
4965 def outgoing(ui, repo, dest=None, **opts):
4965 def outgoing(ui, repo, dest=None, **opts):
4966 """show changesets not found in the destination
4966 """show changesets not found in the destination
4967
4967
4968 Show changesets not found in the specified destination repository
4968 Show changesets not found in the specified destination repository
4969 or the default push location. These are the changesets that would
4969 or the default push location. These are the changesets that would
4970 be pushed if a push was requested.
4970 be pushed if a push was requested.
4971
4971
4972 See pull for details of valid destination formats.
4972 See pull for details of valid destination formats.
4973
4973
4974 .. container:: verbose
4974 .. container:: verbose
4975
4975
4976 With -B/--bookmarks, the result of bookmark comparison between
4976 With -B/--bookmarks, the result of bookmark comparison between
4977 local and remote repositories is displayed. With -v/--verbose,
4977 local and remote repositories is displayed. With -v/--verbose,
4978 status is also displayed for each bookmark like below::
4978 status is also displayed for each bookmark like below::
4979
4979
4980 BM1 01234567890a added
4980 BM1 01234567890a added
4981 BM2 deleted
4981 BM2 deleted
4982 BM3 234567890abc advanced
4982 BM3 234567890abc advanced
4983 BM4 34567890abcd diverged
4983 BM4 34567890abcd diverged
4984 BM5 4567890abcde changed
4984 BM5 4567890abcde changed
4985
4985
4986 The action taken when pushing depends on the
4986 The action taken when pushing depends on the
4987 status of each bookmark:
4987 status of each bookmark:
4988
4988
4989 :``added``: push with ``-B`` will create it
4989 :``added``: push with ``-B`` will create it
4990 :``deleted``: push with ``-B`` will delete it
4990 :``deleted``: push with ``-B`` will delete it
4991 :``advanced``: push will update it
4991 :``advanced``: push will update it
4992 :``diverged``: push with ``-B`` will update it
4992 :``diverged``: push with ``-B`` will update it
4993 :``changed``: push with ``-B`` will update it
4993 :``changed``: push with ``-B`` will update it
4994
4994
4995 From the point of view of pushing behavior, bookmarks
4995 From the point of view of pushing behavior, bookmarks
4996 existing only in the remote repository are treated as
4996 existing only in the remote repository are treated as
4997 ``deleted``, even if it is in fact added remotely.
4997 ``deleted``, even if it is in fact added remotely.
4998
4998
4999 Returns 0 if there are outgoing changes, 1 otherwise.
4999 Returns 0 if there are outgoing changes, 1 otherwise.
5000 """
5000 """
5001 if opts.get('graph'):
5001 if opts.get('graph'):
5002 cmdutil.checkunsupportedgraphflags([], opts)
5002 cmdutil.checkunsupportedgraphflags([], opts)
5003 o, other = hg._outgoing(ui, repo, dest, opts)
5003 o, other = hg._outgoing(ui, repo, dest, opts)
5004 if not o:
5004 if not o:
5005 cmdutil.outgoinghooks(ui, repo, other, opts, o)
5005 cmdutil.outgoinghooks(ui, repo, other, opts, o)
5006 return
5006 return
5007
5007
5008 revdag = cmdutil.graphrevs(repo, o, opts)
5008 revdag = cmdutil.graphrevs(repo, o, opts)
5009 displayer = cmdutil.show_changeset(ui, repo, opts, buffered=True)
5009 displayer = cmdutil.show_changeset(ui, repo, opts, buffered=True)
5010 showparents = [ctx.node() for ctx in repo[None].parents()]
5010 showparents = [ctx.node() for ctx in repo[None].parents()]
5011 cmdutil.displaygraph(ui, revdag, displayer, showparents,
5011 cmdutil.displaygraph(ui, revdag, displayer, showparents,
5012 graphmod.asciiedges)
5012 graphmod.asciiedges)
5013 cmdutil.outgoinghooks(ui, repo, other, opts, o)
5013 cmdutil.outgoinghooks(ui, repo, other, opts, o)
5014 return 0
5014 return 0
5015
5015
5016 if opts.get('bookmarks'):
5016 if opts.get('bookmarks'):
5017 dest = ui.expandpath(dest or 'default-push', dest or 'default')
5017 dest = ui.expandpath(dest or 'default-push', dest or 'default')
5018 dest, branches = hg.parseurl(dest, opts.get('branch'))
5018 dest, branches = hg.parseurl(dest, opts.get('branch'))
5019 other = hg.peer(repo, opts, dest)
5019 other = hg.peer(repo, opts, dest)
5020 if 'bookmarks' not in other.listkeys('namespaces'):
5020 if 'bookmarks' not in other.listkeys('namespaces'):
5021 ui.warn(_("remote doesn't support bookmarks\n"))
5021 ui.warn(_("remote doesn't support bookmarks\n"))
5022 return 0
5022 return 0
5023 ui.status(_('comparing with %s\n') % util.hidepassword(dest))
5023 ui.status(_('comparing with %s\n') % util.hidepassword(dest))
5024 return bookmarks.outgoing(ui, repo, other)
5024 return bookmarks.outgoing(ui, repo, other)
5025
5025
5026 repo._subtoppath = ui.expandpath(dest or 'default-push', dest or 'default')
5026 repo._subtoppath = ui.expandpath(dest or 'default-push', dest or 'default')
5027 try:
5027 try:
5028 return hg.outgoing(ui, repo, dest, opts)
5028 return hg.outgoing(ui, repo, dest, opts)
5029 finally:
5029 finally:
5030 del repo._subtoppath
5030 del repo._subtoppath
5031
5031
5032 @command('parents',
5032 @command('parents',
5033 [('r', 'rev', '', _('show parents of the specified revision'), _('REV')),
5033 [('r', 'rev', '', _('show parents of the specified revision'), _('REV')),
5034 ] + templateopts,
5034 ] + templateopts,
5035 _('[-r REV] [FILE]'),
5035 _('[-r REV] [FILE]'),
5036 inferrepo=True)
5036 inferrepo=True)
5037 def parents(ui, repo, file_=None, **opts):
5037 def parents(ui, repo, file_=None, **opts):
5038 """show the parents of the working directory or revision (DEPRECATED)
5038 """show the parents of the working directory or revision (DEPRECATED)
5039
5039
5040 Print the working directory's parent revisions. If a revision is
5040 Print the working directory's parent revisions. If a revision is
5041 given via -r/--rev, the parent of that revision will be printed.
5041 given via -r/--rev, the parent of that revision will be printed.
5042 If a file argument is given, the revision in which the file was
5042 If a file argument is given, the revision in which the file was
5043 last changed (before the working directory revision or the
5043 last changed (before the working directory revision or the
5044 argument to --rev if given) is printed.
5044 argument to --rev if given) is printed.
5045
5045
5046 See :hg:`summary` and :hg:`help revsets` for related information.
5046 See :hg:`summary` and :hg:`help revsets` for related information.
5047
5047
5048 Returns 0 on success.
5048 Returns 0 on success.
5049 """
5049 """
5050
5050
5051 ctx = scmutil.revsingle(repo, opts.get('rev'), None)
5051 ctx = scmutil.revsingle(repo, opts.get('rev'), None)
5052
5052
5053 if file_:
5053 if file_:
5054 m = scmutil.match(ctx, (file_,), opts)
5054 m = scmutil.match(ctx, (file_,), opts)
5055 if m.anypats() or len(m.files()) != 1:
5055 if m.anypats() or len(m.files()) != 1:
5056 raise error.Abort(_('can only specify an explicit filename'))
5056 raise error.Abort(_('can only specify an explicit filename'))
5057 file_ = m.files()[0]
5057 file_ = m.files()[0]
5058 filenodes = []
5058 filenodes = []
5059 for cp in ctx.parents():
5059 for cp in ctx.parents():
5060 if not cp:
5060 if not cp:
5061 continue
5061 continue
5062 try:
5062 try:
5063 filenodes.append(cp.filenode(file_))
5063 filenodes.append(cp.filenode(file_))
5064 except error.LookupError:
5064 except error.LookupError:
5065 pass
5065 pass
5066 if not filenodes:
5066 if not filenodes:
5067 raise error.Abort(_("'%s' not found in manifest!") % file_)
5067 raise error.Abort(_("'%s' not found in manifest!") % file_)
5068 p = []
5068 p = []
5069 for fn in filenodes:
5069 for fn in filenodes:
5070 fctx = repo.filectx(file_, fileid=fn)
5070 fctx = repo.filectx(file_, fileid=fn)
5071 p.append(fctx.node())
5071 p.append(fctx.node())
5072 else:
5072 else:
5073 p = [cp.node() for cp in ctx.parents()]
5073 p = [cp.node() for cp in ctx.parents()]
5074
5074
5075 displayer = cmdutil.show_changeset(ui, repo, opts)
5075 displayer = cmdutil.show_changeset(ui, repo, opts)
5076 for n in p:
5076 for n in p:
5077 if n != nullid:
5077 if n != nullid:
5078 displayer.show(repo[n])
5078 displayer.show(repo[n])
5079 displayer.close()
5079 displayer.close()
5080
5080
5081 @command('paths', [], _('[NAME]'), optionalrepo=True)
5081 @command('paths', [], _('[NAME]'), optionalrepo=True)
5082 def paths(ui, repo, search=None):
5082 def paths(ui, repo, search=None):
5083 """show aliases for remote repositories
5083 """show aliases for remote repositories
5084
5084
5085 Show definition of symbolic path name NAME. If no name is given,
5085 Show definition of symbolic path name NAME. If no name is given,
5086 show definition of all available names.
5086 show definition of all available names.
5087
5087
5088 Option -q/--quiet suppresses all output when searching for NAME
5088 Option -q/--quiet suppresses all output when searching for NAME
5089 and shows only the path names when listing all definitions.
5089 and shows only the path names when listing all definitions.
5090
5090
5091 Path names are defined in the [paths] section of your
5091 Path names are defined in the [paths] section of your
5092 configuration file and in ``/etc/mercurial/hgrc``. If run inside a
5092 configuration file and in ``/etc/mercurial/hgrc``. If run inside a
5093 repository, ``.hg/hgrc`` is used, too.
5093 repository, ``.hg/hgrc`` is used, too.
5094
5094
5095 The path names ``default`` and ``default-push`` have a special
5095 The path names ``default`` and ``default-push`` have a special
5096 meaning. When performing a push or pull operation, they are used
5096 meaning. When performing a push or pull operation, they are used
5097 as fallbacks if no location is specified on the command-line.
5097 as fallbacks if no location is specified on the command-line.
5098 When ``default-push`` is set, it will be used for push and
5098 When ``default-push`` is set, it will be used for push and
5099 ``default`` will be used for pull; otherwise ``default`` is used
5099 ``default`` will be used for pull; otherwise ``default`` is used
5100 as the fallback for both. When cloning a repository, the clone
5100 as the fallback for both. When cloning a repository, the clone
5101 source is written as ``default`` in ``.hg/hgrc``. Note that
5101 source is written as ``default`` in ``.hg/hgrc``. Note that
5102 ``default`` and ``default-push`` apply to all inbound (e.g.
5102 ``default`` and ``default-push`` apply to all inbound (e.g.
5103 :hg:`incoming`) and outbound (e.g. :hg:`outgoing`, :hg:`email` and
5103 :hg:`incoming`) and outbound (e.g. :hg:`outgoing`, :hg:`email` and
5104 :hg:`bundle`) operations.
5104 :hg:`bundle`) operations.
5105
5105
5106 See :hg:`help urls` for more information.
5106 See :hg:`help urls` for more information.
5107
5107
5108 Returns 0 on success.
5108 Returns 0 on success.
5109 """
5109 """
5110 if search:
5110 if search:
5111 for name, path in sorted(ui.paths.iteritems()):
5111 for name, path in sorted(ui.paths.iteritems()):
5112 if name == search:
5112 if name == search:
5113 ui.status("%s\n" % util.hidepassword(path.loc))
5113 ui.status("%s\n" % util.hidepassword(path.loc))
5114 return
5114 return
5115 if not ui.quiet:
5115 if not ui.quiet:
5116 ui.warn(_("not found!\n"))
5116 ui.warn(_("not found!\n"))
5117 return 1
5117 return 1
5118 else:
5118 else:
5119 for name, path in sorted(ui.paths.iteritems()):
5119 for name, path in sorted(ui.paths.iteritems()):
5120 if ui.quiet:
5120 if ui.quiet:
5121 ui.write("%s\n" % name)
5121 ui.write("%s\n" % name)
5122 else:
5122 else:
5123 ui.write("%s = %s\n" % (name,
5123 ui.write("%s = %s\n" % (name,
5124 util.hidepassword(path.loc)))
5124 util.hidepassword(path.loc)))
5125
5125
5126 @command('phase',
5126 @command('phase',
5127 [('p', 'public', False, _('set changeset phase to public')),
5127 [('p', 'public', False, _('set changeset phase to public')),
5128 ('d', 'draft', False, _('set changeset phase to draft')),
5128 ('d', 'draft', False, _('set changeset phase to draft')),
5129 ('s', 'secret', False, _('set changeset phase to secret')),
5129 ('s', 'secret', False, _('set changeset phase to secret')),
5130 ('f', 'force', False, _('allow to move boundary backward')),
5130 ('f', 'force', False, _('allow to move boundary backward')),
5131 ('r', 'rev', [], _('target revision'), _('REV')),
5131 ('r', 'rev', [], _('target revision'), _('REV')),
5132 ],
5132 ],
5133 _('[-p|-d|-s] [-f] [-r] [REV...]'))
5133 _('[-p|-d|-s] [-f] [-r] [REV...]'))
5134 def phase(ui, repo, *revs, **opts):
5134 def phase(ui, repo, *revs, **opts):
5135 """set or show the current phase name
5135 """set or show the current phase name
5136
5136
5137 With no argument, show the phase name of the current revision(s).
5137 With no argument, show the phase name of the current revision(s).
5138
5138
5139 With one of -p/--public, -d/--draft or -s/--secret, change the
5139 With one of -p/--public, -d/--draft or -s/--secret, change the
5140 phase value of the specified revisions.
5140 phase value of the specified revisions.
5141
5141
5142 Unless -f/--force is specified, :hg:`phase` won't move changeset from a
5142 Unless -f/--force is specified, :hg:`phase` won't move changeset from a
5143 lower phase to an higher phase. Phases are ordered as follows::
5143 lower phase to an higher phase. Phases are ordered as follows::
5144
5144
5145 public < draft < secret
5145 public < draft < secret
5146
5146
5147 Returns 0 on success, 1 if some phases could not be changed.
5147 Returns 0 on success, 1 if some phases could not be changed.
5148
5148
5149 (For more information about the phases concept, see :hg:`help phases`.)
5149 (For more information about the phases concept, see :hg:`help phases`.)
5150 """
5150 """
5151 # search for a unique phase argument
5151 # search for a unique phase argument
5152 targetphase = None
5152 targetphase = None
5153 for idx, name in enumerate(phases.phasenames):
5153 for idx, name in enumerate(phases.phasenames):
5154 if opts[name]:
5154 if opts[name]:
5155 if targetphase is not None:
5155 if targetphase is not None:
5156 raise error.Abort(_('only one phase can be specified'))
5156 raise error.Abort(_('only one phase can be specified'))
5157 targetphase = idx
5157 targetphase = idx
5158
5158
5159 # look for specified revision
5159 # look for specified revision
5160 revs = list(revs)
5160 revs = list(revs)
5161 revs.extend(opts['rev'])
5161 revs.extend(opts['rev'])
5162 if not revs:
5162 if not revs:
5163 # display both parents as the second parent phase can influence
5163 # display both parents as the second parent phase can influence
5164 # the phase of a merge commit
5164 # the phase of a merge commit
5165 revs = [c.rev() for c in repo[None].parents()]
5165 revs = [c.rev() for c in repo[None].parents()]
5166
5166
5167 revs = scmutil.revrange(repo, revs)
5167 revs = scmutil.revrange(repo, revs)
5168
5168
5169 lock = None
5169 lock = None
5170 ret = 0
5170 ret = 0
5171 if targetphase is None:
5171 if targetphase is None:
5172 # display
5172 # display
5173 for r in revs:
5173 for r in revs:
5174 ctx = repo[r]
5174 ctx = repo[r]
5175 ui.write('%i: %s\n' % (ctx.rev(), ctx.phasestr()))
5175 ui.write('%i: %s\n' % (ctx.rev(), ctx.phasestr()))
5176 else:
5176 else:
5177 tr = None
5177 tr = None
5178 lock = repo.lock()
5178 lock = repo.lock()
5179 try:
5179 try:
5180 tr = repo.transaction("phase")
5180 tr = repo.transaction("phase")
5181 # set phase
5181 # set phase
5182 if not revs:
5182 if not revs:
5183 raise error.Abort(_('empty revision set'))
5183 raise error.Abort(_('empty revision set'))
5184 nodes = [repo[r].node() for r in revs]
5184 nodes = [repo[r].node() for r in revs]
5185 # moving revision from public to draft may hide them
5185 # moving revision from public to draft may hide them
5186 # We have to check result on an unfiltered repository
5186 # We have to check result on an unfiltered repository
5187 unfi = repo.unfiltered()
5187 unfi = repo.unfiltered()
5188 getphase = unfi._phasecache.phase
5188 getphase = unfi._phasecache.phase
5189 olddata = [getphase(unfi, r) for r in unfi]
5189 olddata = [getphase(unfi, r) for r in unfi]
5190 phases.advanceboundary(repo, tr, targetphase, nodes)
5190 phases.advanceboundary(repo, tr, targetphase, nodes)
5191 if opts['force']:
5191 if opts['force']:
5192 phases.retractboundary(repo, tr, targetphase, nodes)
5192 phases.retractboundary(repo, tr, targetphase, nodes)
5193 tr.close()
5193 tr.close()
5194 finally:
5194 finally:
5195 if tr is not None:
5195 if tr is not None:
5196 tr.release()
5196 tr.release()
5197 lock.release()
5197 lock.release()
5198 getphase = unfi._phasecache.phase
5198 getphase = unfi._phasecache.phase
5199 newdata = [getphase(unfi, r) for r in unfi]
5199 newdata = [getphase(unfi, r) for r in unfi]
5200 changes = sum(newdata[r] != olddata[r] for r in unfi)
5200 changes = sum(newdata[r] != olddata[r] for r in unfi)
5201 cl = unfi.changelog
5201 cl = unfi.changelog
5202 rejected = [n for n in nodes
5202 rejected = [n for n in nodes
5203 if newdata[cl.rev(n)] < targetphase]
5203 if newdata[cl.rev(n)] < targetphase]
5204 if rejected:
5204 if rejected:
5205 ui.warn(_('cannot move %i changesets to a higher '
5205 ui.warn(_('cannot move %i changesets to a higher '
5206 'phase, use --force\n') % len(rejected))
5206 'phase, use --force\n') % len(rejected))
5207 ret = 1
5207 ret = 1
5208 if changes:
5208 if changes:
5209 msg = _('phase changed for %i changesets\n') % changes
5209 msg = _('phase changed for %i changesets\n') % changes
5210 if ret:
5210 if ret:
5211 ui.status(msg)
5211 ui.status(msg)
5212 else:
5212 else:
5213 ui.note(msg)
5213 ui.note(msg)
5214 else:
5214 else:
5215 ui.warn(_('no phases changed\n'))
5215 ui.warn(_('no phases changed\n'))
5216 return ret
5216 return ret
5217
5217
5218 def postincoming(ui, repo, modheads, optupdate, checkout):
5218 def postincoming(ui, repo, modheads, optupdate, checkout):
5219 if modheads == 0:
5219 if modheads == 0:
5220 return
5220 return
5221 if optupdate:
5221 if optupdate:
5222 try:
5222 try:
5223 brev = checkout
5223 brev = checkout
5224 movemarkfrom = None
5224 movemarkfrom = None
5225 if not checkout:
5225 if not checkout:
5226 updata = destutil.destupdate(repo)
5226 updata = destutil.destupdate(repo)
5227 checkout, movemarkfrom, brev = updata
5227 checkout, movemarkfrom, brev = updata
5228 ret = hg.update(repo, checkout)
5228 ret = hg.update(repo, checkout)
5229 except error.UpdateAbort as inst:
5229 except error.UpdateAbort as inst:
5230 msg = _("not updating: %s") % str(inst)
5230 msg = _("not updating: %s") % str(inst)
5231 hint = inst.hint
5231 hint = inst.hint
5232 raise error.UpdateAbort(msg, hint=hint)
5232 raise error.UpdateAbort(msg, hint=hint)
5233 if not ret and not checkout:
5233 if not ret and not checkout:
5234 if bookmarks.update(repo, [movemarkfrom], repo['.'].node()):
5234 if bookmarks.update(repo, [movemarkfrom], repo['.'].node()):
5235 ui.status(_("updating bookmark %s\n") % repo._activebookmark)
5235 ui.status(_("updating bookmark %s\n") % repo._activebookmark)
5236 return ret
5236 return ret
5237 if modheads > 1:
5237 if modheads > 1:
5238 currentbranchheads = len(repo.branchheads())
5238 currentbranchheads = len(repo.branchheads())
5239 if currentbranchheads == modheads:
5239 if currentbranchheads == modheads:
5240 ui.status(_("(run 'hg heads' to see heads, 'hg merge' to merge)\n"))
5240 ui.status(_("(run 'hg heads' to see heads, 'hg merge' to merge)\n"))
5241 elif currentbranchheads > 1:
5241 elif currentbranchheads > 1:
5242 ui.status(_("(run 'hg heads .' to see heads, 'hg merge' to "
5242 ui.status(_("(run 'hg heads .' to see heads, 'hg merge' to "
5243 "merge)\n"))
5243 "merge)\n"))
5244 else:
5244 else:
5245 ui.status(_("(run 'hg heads' to see heads)\n"))
5245 ui.status(_("(run 'hg heads' to see heads)\n"))
5246 else:
5246 else:
5247 ui.status(_("(run 'hg update' to get a working copy)\n"))
5247 ui.status(_("(run 'hg update' to get a working copy)\n"))
5248
5248
5249 @command('^pull',
5249 @command('^pull',
5250 [('u', 'update', None,
5250 [('u', 'update', None,
5251 _('update to new branch head if changesets were pulled')),
5251 _('update to new branch head if changesets were pulled')),
5252 ('f', 'force', None, _('run even when remote repository is unrelated')),
5252 ('f', 'force', None, _('run even when remote repository is unrelated')),
5253 ('r', 'rev', [], _('a remote changeset intended to be added'), _('REV')),
5253 ('r', 'rev', [], _('a remote changeset intended to be added'), _('REV')),
5254 ('B', 'bookmark', [], _("bookmark to pull"), _('BOOKMARK')),
5254 ('B', 'bookmark', [], _("bookmark to pull"), _('BOOKMARK')),
5255 ('b', 'branch', [], _('a specific branch you would like to pull'),
5255 ('b', 'branch', [], _('a specific branch you would like to pull'),
5256 _('BRANCH')),
5256 _('BRANCH')),
5257 ] + remoteopts,
5257 ] + remoteopts,
5258 _('[-u] [-f] [-r REV]... [-e CMD] [--remotecmd CMD] [SOURCE]'))
5258 _('[-u] [-f] [-r REV]... [-e CMD] [--remotecmd CMD] [SOURCE]'))
5259 def pull(ui, repo, source="default", **opts):
5259 def pull(ui, repo, source="default", **opts):
5260 """pull changes from the specified source
5260 """pull changes from the specified source
5261
5261
5262 Pull changes from a remote repository to a local one.
5262 Pull changes from a remote repository to a local one.
5263
5263
5264 This finds all changes from the repository at the specified path
5264 This finds all changes from the repository at the specified path
5265 or URL and adds them to a local repository (the current one unless
5265 or URL and adds them to a local repository (the current one unless
5266 -R is specified). By default, this does not update the copy of the
5266 -R is specified). By default, this does not update the copy of the
5267 project in the working directory.
5267 project in the working directory.
5268
5268
5269 Use :hg:`incoming` if you want to see what would have been added
5269 Use :hg:`incoming` if you want to see what would have been added
5270 by a pull at the time you issued this command. If you then decide
5270 by a pull at the time you issued this command. If you then decide
5271 to add those changes to the repository, you should use :hg:`pull
5271 to add those changes to the repository, you should use :hg:`pull
5272 -r X` where ``X`` is the last changeset listed by :hg:`incoming`.
5272 -r X` where ``X`` is the last changeset listed by :hg:`incoming`.
5273
5273
5274 If SOURCE is omitted, the 'default' path will be used.
5274 If SOURCE is omitted, the 'default' path will be used.
5275 See :hg:`help urls` for more information.
5275 See :hg:`help urls` for more information.
5276
5276
5277 Returns 0 on success, 1 if an update had unresolved files.
5277 Returns 0 on success, 1 if an update had unresolved files.
5278 """
5278 """
5279 source, branches = hg.parseurl(ui.expandpath(source), opts.get('branch'))
5279 source, branches = hg.parseurl(ui.expandpath(source), opts.get('branch'))
5280 ui.status(_('pulling from %s\n') % util.hidepassword(source))
5280 ui.status(_('pulling from %s\n') % util.hidepassword(source))
5281 other = hg.peer(repo, opts, source)
5281 other = hg.peer(repo, opts, source)
5282 try:
5282 try:
5283 revs, checkout = hg.addbranchrevs(repo, other, branches,
5283 revs, checkout = hg.addbranchrevs(repo, other, branches,
5284 opts.get('rev'))
5284 opts.get('rev'))
5285
5285
5286
5286
5287 pullopargs = {}
5287 pullopargs = {}
5288 if opts.get('bookmark'):
5288 if opts.get('bookmark'):
5289 if not revs:
5289 if not revs:
5290 revs = []
5290 revs = []
5291 # The list of bookmark used here is not the one used to actually
5291 # The list of bookmark used here is not the one used to actually
5292 # update the bookmark name. This can result in the revision pulled
5292 # update the bookmark name. This can result in the revision pulled
5293 # not ending up with the name of the bookmark because of a race
5293 # not ending up with the name of the bookmark because of a race
5294 # condition on the server. (See issue 4689 for details)
5294 # condition on the server. (See issue 4689 for details)
5295 remotebookmarks = other.listkeys('bookmarks')
5295 remotebookmarks = other.listkeys('bookmarks')
5296 pullopargs['remotebookmarks'] = remotebookmarks
5296 pullopargs['remotebookmarks'] = remotebookmarks
5297 for b in opts['bookmark']:
5297 for b in opts['bookmark']:
5298 if b not in remotebookmarks:
5298 if b not in remotebookmarks:
5299 raise error.Abort(_('remote bookmark %s not found!') % b)
5299 raise error.Abort(_('remote bookmark %s not found!') % b)
5300 revs.append(remotebookmarks[b])
5300 revs.append(remotebookmarks[b])
5301
5301
5302 if revs:
5302 if revs:
5303 try:
5303 try:
5304 # When 'rev' is a bookmark name, we cannot guarantee that it
5304 # When 'rev' is a bookmark name, we cannot guarantee that it
5305 # will be updated with that name because of a race condition
5305 # will be updated with that name because of a race condition
5306 # server side. (See issue 4689 for details)
5306 # server side. (See issue 4689 for details)
5307 oldrevs = revs
5307 oldrevs = revs
5308 revs = [] # actually, nodes
5308 revs = [] # actually, nodes
5309 for r in oldrevs:
5309 for r in oldrevs:
5310 node = other.lookup(r)
5310 node = other.lookup(r)
5311 revs.append(node)
5311 revs.append(node)
5312 if r == checkout:
5312 if r == checkout:
5313 checkout = node
5313 checkout = node
5314 except error.CapabilityError:
5314 except error.CapabilityError:
5315 err = _("other repository doesn't support revision lookup, "
5315 err = _("other repository doesn't support revision lookup, "
5316 "so a rev cannot be specified.")
5316 "so a rev cannot be specified.")
5317 raise error.Abort(err)
5317 raise error.Abort(err)
5318
5318
5319 pullopargs.update(opts.get('opargs', {}))
5319 pullopargs.update(opts.get('opargs', {}))
5320 modheads = exchange.pull(repo, other, heads=revs,
5320 modheads = exchange.pull(repo, other, heads=revs,
5321 force=opts.get('force'),
5321 force=opts.get('force'),
5322 bookmarks=opts.get('bookmark', ()),
5322 bookmarks=opts.get('bookmark', ()),
5323 opargs=pullopargs).cgresult
5323 opargs=pullopargs).cgresult
5324 if checkout:
5324 if checkout:
5325 checkout = str(repo.changelog.rev(checkout))
5325 checkout = str(repo.changelog.rev(checkout))
5326 repo._subtoppath = source
5326 repo._subtoppath = source
5327 try:
5327 try:
5328 ret = postincoming(ui, repo, modheads, opts.get('update'), checkout)
5328 ret = postincoming(ui, repo, modheads, opts.get('update'), checkout)
5329
5329
5330 finally:
5330 finally:
5331 del repo._subtoppath
5331 del repo._subtoppath
5332
5332
5333 finally:
5333 finally:
5334 other.close()
5334 other.close()
5335 return ret
5335 return ret
5336
5336
5337 @command('^push',
5337 @command('^push',
5338 [('f', 'force', None, _('force push')),
5338 [('f', 'force', None, _('force push')),
5339 ('r', 'rev', [],
5339 ('r', 'rev', [],
5340 _('a changeset intended to be included in the destination'),
5340 _('a changeset intended to be included in the destination'),
5341 _('REV')),
5341 _('REV')),
5342 ('B', 'bookmark', [], _("bookmark to push"), _('BOOKMARK')),
5342 ('B', 'bookmark', [], _("bookmark to push"), _('BOOKMARK')),
5343 ('b', 'branch', [],
5343 ('b', 'branch', [],
5344 _('a specific branch you would like to push'), _('BRANCH')),
5344 _('a specific branch you would like to push'), _('BRANCH')),
5345 ('', 'new-branch', False, _('allow pushing a new branch')),
5345 ('', 'new-branch', False, _('allow pushing a new branch')),
5346 ] + remoteopts,
5346 ] + remoteopts,
5347 _('[-f] [-r REV]... [-e CMD] [--remotecmd CMD] [DEST]'))
5347 _('[-f] [-r REV]... [-e CMD] [--remotecmd CMD] [DEST]'))
5348 def push(ui, repo, dest=None, **opts):
5348 def push(ui, repo, dest=None, **opts):
5349 """push changes to the specified destination
5349 """push changes to the specified destination
5350
5350
5351 Push changesets from the local repository to the specified
5351 Push changesets from the local repository to the specified
5352 destination.
5352 destination.
5353
5353
5354 This operation is symmetrical to pull: it is identical to a pull
5354 This operation is symmetrical to pull: it is identical to a pull
5355 in the destination repository from the current one.
5355 in the destination repository from the current one.
5356
5356
5357 By default, push will not allow creation of new heads at the
5357 By default, push will not allow creation of new heads at the
5358 destination, since multiple heads would make it unclear which head
5358 destination, since multiple heads would make it unclear which head
5359 to use. In this situation, it is recommended to pull and merge
5359 to use. In this situation, it is recommended to pull and merge
5360 before pushing.
5360 before pushing.
5361
5361
5362 Use --new-branch if you want to allow push to create a new named
5362 Use --new-branch if you want to allow push to create a new named
5363 branch that is not present at the destination. This allows you to
5363 branch that is not present at the destination. This allows you to
5364 only create a new branch without forcing other changes.
5364 only create a new branch without forcing other changes.
5365
5365
5366 .. note::
5366 .. note::
5367
5367
5368 Extra care should be taken with the -f/--force option,
5368 Extra care should be taken with the -f/--force option,
5369 which will push all new heads on all branches, an action which will
5369 which will push all new heads on all branches, an action which will
5370 almost always cause confusion for collaborators.
5370 almost always cause confusion for collaborators.
5371
5371
5372 If -r/--rev is used, the specified revision and all its ancestors
5372 If -r/--rev is used, the specified revision and all its ancestors
5373 will be pushed to the remote repository.
5373 will be pushed to the remote repository.
5374
5374
5375 If -B/--bookmark is used, the specified bookmarked revision, its
5375 If -B/--bookmark is used, the specified bookmarked revision, its
5376 ancestors, and the bookmark will be pushed to the remote
5376 ancestors, and the bookmark will be pushed to the remote
5377 repository.
5377 repository.
5378
5378
5379 Please see :hg:`help urls` for important details about ``ssh://``
5379 Please see :hg:`help urls` for important details about ``ssh://``
5380 URLs. If DESTINATION is omitted, a default path will be used.
5380 URLs. If DESTINATION is omitted, a default path will be used.
5381
5381
5382 Returns 0 if push was successful, 1 if nothing to push.
5382 Returns 0 if push was successful, 1 if nothing to push.
5383 """
5383 """
5384
5384
5385 if opts.get('bookmark'):
5385 if opts.get('bookmark'):
5386 ui.setconfig('bookmarks', 'pushing', opts['bookmark'], 'push')
5386 ui.setconfig('bookmarks', 'pushing', opts['bookmark'], 'push')
5387 for b in opts['bookmark']:
5387 for b in opts['bookmark']:
5388 # translate -B options to -r so changesets get pushed
5388 # translate -B options to -r so changesets get pushed
5389 if b in repo._bookmarks:
5389 if b in repo._bookmarks:
5390 opts.setdefault('rev', []).append(b)
5390 opts.setdefault('rev', []).append(b)
5391 else:
5391 else:
5392 # if we try to push a deleted bookmark, translate it to null
5392 # if we try to push a deleted bookmark, translate it to null
5393 # this lets simultaneous -r, -b options continue working
5393 # this lets simultaneous -r, -b options continue working
5394 opts.setdefault('rev', []).append("null")
5394 opts.setdefault('rev', []).append("null")
5395
5395
5396 path = ui.paths.getpath(dest, default='default')
5396 path = ui.paths.getpath(dest, default='default')
5397 if not path:
5397 if not path:
5398 raise error.Abort(_('default repository not configured!'),
5398 raise error.Abort(_('default repository not configured!'),
5399 hint=_('see the "path" section in "hg help config"'))
5399 hint=_('see the "path" section in "hg help config"'))
5400 dest, branches = path.pushloc, (path.branch, opts.get('branch') or [])
5400 dest, branches = path.pushloc, (path.branch, opts.get('branch') or [])
5401 ui.status(_('pushing to %s\n') % util.hidepassword(dest))
5401 ui.status(_('pushing to %s\n') % util.hidepassword(dest))
5402 revs, checkout = hg.addbranchrevs(repo, repo, branches, opts.get('rev'))
5402 revs, checkout = hg.addbranchrevs(repo, repo, branches, opts.get('rev'))
5403 other = hg.peer(repo, opts, dest)
5403 other = hg.peer(repo, opts, dest)
5404
5404
5405 if revs:
5405 if revs:
5406 revs = [repo.lookup(r) for r in scmutil.revrange(repo, revs)]
5406 revs = [repo.lookup(r) for r in scmutil.revrange(repo, revs)]
5407 if not revs:
5407 if not revs:
5408 raise error.Abort(_("specified revisions evaluate to an empty set"),
5408 raise error.Abort(_("specified revisions evaluate to an empty set"),
5409 hint=_("use different revision arguments"))
5409 hint=_("use different revision arguments"))
5410
5410
5411 repo._subtoppath = dest
5411 repo._subtoppath = dest
5412 try:
5412 try:
5413 # push subrepos depth-first for coherent ordering
5413 # push subrepos depth-first for coherent ordering
5414 c = repo['']
5414 c = repo['']
5415 subs = c.substate # only repos that are committed
5415 subs = c.substate # only repos that are committed
5416 for s in sorted(subs):
5416 for s in sorted(subs):
5417 result = c.sub(s).push(opts)
5417 result = c.sub(s).push(opts)
5418 if result == 0:
5418 if result == 0:
5419 return not result
5419 return not result
5420 finally:
5420 finally:
5421 del repo._subtoppath
5421 del repo._subtoppath
5422 pushop = exchange.push(repo, other, opts.get('force'), revs=revs,
5422 pushop = exchange.push(repo, other, opts.get('force'), revs=revs,
5423 newbranch=opts.get('new_branch'),
5423 newbranch=opts.get('new_branch'),
5424 bookmarks=opts.get('bookmark', ()),
5424 bookmarks=opts.get('bookmark', ()),
5425 opargs=opts.get('opargs'))
5425 opargs=opts.get('opargs'))
5426
5426
5427 result = not pushop.cgresult
5427 result = not pushop.cgresult
5428
5428
5429 if pushop.bkresult is not None:
5429 if pushop.bkresult is not None:
5430 if pushop.bkresult == 2:
5430 if pushop.bkresult == 2:
5431 result = 2
5431 result = 2
5432 elif not result and pushop.bkresult:
5432 elif not result and pushop.bkresult:
5433 result = 2
5433 result = 2
5434
5434
5435 return result
5435 return result
5436
5436
5437 @command('recover', [])
5437 @command('recover', [])
5438 def recover(ui, repo):
5438 def recover(ui, repo):
5439 """roll back an interrupted transaction
5439 """roll back an interrupted transaction
5440
5440
5441 Recover from an interrupted commit or pull.
5441 Recover from an interrupted commit or pull.
5442
5442
5443 This command tries to fix the repository status after an
5443 This command tries to fix the repository status after an
5444 interrupted operation. It should only be necessary when Mercurial
5444 interrupted operation. It should only be necessary when Mercurial
5445 suggests it.
5445 suggests it.
5446
5446
5447 Returns 0 if successful, 1 if nothing to recover or verify fails.
5447 Returns 0 if successful, 1 if nothing to recover or verify fails.
5448 """
5448 """
5449 if repo.recover():
5449 if repo.recover():
5450 return hg.verify(repo)
5450 return hg.verify(repo)
5451 return 1
5451 return 1
5452
5452
5453 @command('^remove|rm',
5453 @command('^remove|rm',
5454 [('A', 'after', None, _('record delete for missing files')),
5454 [('A', 'after', None, _('record delete for missing files')),
5455 ('f', 'force', None,
5455 ('f', 'force', None,
5456 _('remove (and delete) file even if added or modified')),
5456 _('remove (and delete) file even if added or modified')),
5457 ] + subrepoopts + walkopts,
5457 ] + subrepoopts + walkopts,
5458 _('[OPTION]... FILE...'),
5458 _('[OPTION]... FILE...'),
5459 inferrepo=True)
5459 inferrepo=True)
5460 def remove(ui, repo, *pats, **opts):
5460 def remove(ui, repo, *pats, **opts):
5461 """remove the specified files on the next commit
5461 """remove the specified files on the next commit
5462
5462
5463 Schedule the indicated files for removal from the current branch.
5463 Schedule the indicated files for removal from the current branch.
5464
5464
5465 This command schedules the files to be removed at the next commit.
5465 This command schedules the files to be removed at the next commit.
5466 To undo a remove before that, see :hg:`revert`. To undo added
5466 To undo a remove before that, see :hg:`revert`. To undo added
5467 files, see :hg:`forget`.
5467 files, see :hg:`forget`.
5468
5468
5469 .. container:: verbose
5469 .. container:: verbose
5470
5470
5471 -A/--after can be used to remove only files that have already
5471 -A/--after can be used to remove only files that have already
5472 been deleted, -f/--force can be used to force deletion, and -Af
5472 been deleted, -f/--force can be used to force deletion, and -Af
5473 can be used to remove files from the next revision without
5473 can be used to remove files from the next revision without
5474 deleting them from the working directory.
5474 deleting them from the working directory.
5475
5475
5476 The following table details the behavior of remove for different
5476 The following table details the behavior of remove for different
5477 file states (columns) and option combinations (rows). The file
5477 file states (columns) and option combinations (rows). The file
5478 states are Added [A], Clean [C], Modified [M] and Missing [!]
5478 states are Added [A], Clean [C], Modified [M] and Missing [!]
5479 (as reported by :hg:`status`). The actions are Warn, Remove
5479 (as reported by :hg:`status`). The actions are Warn, Remove
5480 (from branch) and Delete (from disk):
5480 (from branch) and Delete (from disk):
5481
5481
5482 ========= == == == ==
5482 ========= == == == ==
5483 opt/state A C M !
5483 opt/state A C M !
5484 ========= == == == ==
5484 ========= == == == ==
5485 none W RD W R
5485 none W RD W R
5486 -f R RD RD R
5486 -f R RD RD R
5487 -A W W W R
5487 -A W W W R
5488 -Af R R R R
5488 -Af R R R R
5489 ========= == == == ==
5489 ========= == == == ==
5490
5490
5491 Note that remove never deletes files in Added [A] state from the
5491 Note that remove never deletes files in Added [A] state from the
5492 working directory, not even if option --force is specified.
5492 working directory, not even if option --force is specified.
5493
5493
5494 Returns 0 on success, 1 if any warnings encountered.
5494 Returns 0 on success, 1 if any warnings encountered.
5495 """
5495 """
5496
5496
5497 after, force = opts.get('after'), opts.get('force')
5497 after, force = opts.get('after'), opts.get('force')
5498 if not pats and not after:
5498 if not pats and not after:
5499 raise error.Abort(_('no files specified'))
5499 raise error.Abort(_('no files specified'))
5500
5500
5501 m = scmutil.match(repo[None], pats, opts)
5501 m = scmutil.match(repo[None], pats, opts)
5502 subrepos = opts.get('subrepos')
5502 subrepos = opts.get('subrepos')
5503 return cmdutil.remove(ui, repo, m, "", after, force, subrepos)
5503 return cmdutil.remove(ui, repo, m, "", after, force, subrepos)
5504
5504
5505 @command('rename|move|mv',
5505 @command('rename|move|mv',
5506 [('A', 'after', None, _('record a rename that has already occurred')),
5506 [('A', 'after', None, _('record a rename that has already occurred')),
5507 ('f', 'force', None, _('forcibly copy over an existing managed file')),
5507 ('f', 'force', None, _('forcibly copy over an existing managed file')),
5508 ] + walkopts + dryrunopts,
5508 ] + walkopts + dryrunopts,
5509 _('[OPTION]... SOURCE... DEST'))
5509 _('[OPTION]... SOURCE... DEST'))
5510 def rename(ui, repo, *pats, **opts):
5510 def rename(ui, repo, *pats, **opts):
5511 """rename files; equivalent of copy + remove
5511 """rename files; equivalent of copy + remove
5512
5512
5513 Mark dest as copies of sources; mark sources for deletion. If dest
5513 Mark dest as copies of sources; mark sources for deletion. If dest
5514 is a directory, copies are put in that directory. If dest is a
5514 is a directory, copies are put in that directory. If dest is a
5515 file, there can only be one source.
5515 file, there can only be one source.
5516
5516
5517 By default, this command copies the contents of files as they
5517 By default, this command copies the contents of files as they
5518 exist in the working directory. If invoked with -A/--after, the
5518 exist in the working directory. If invoked with -A/--after, the
5519 operation is recorded, but no copying is performed.
5519 operation is recorded, but no copying is performed.
5520
5520
5521 This command takes effect at the next commit. To undo a rename
5521 This command takes effect at the next commit. To undo a rename
5522 before that, see :hg:`revert`.
5522 before that, see :hg:`revert`.
5523
5523
5524 Returns 0 on success, 1 if errors are encountered.
5524 Returns 0 on success, 1 if errors are encountered.
5525 """
5525 """
5526 wlock = repo.wlock(False)
5526 wlock = repo.wlock(False)
5527 try:
5527 try:
5528 return cmdutil.copy(ui, repo, pats, opts, rename=True)
5528 return cmdutil.copy(ui, repo, pats, opts, rename=True)
5529 finally:
5529 finally:
5530 wlock.release()
5530 wlock.release()
5531
5531
5532 @command('resolve',
5532 @command('resolve',
5533 [('a', 'all', None, _('select all unresolved files')),
5533 [('a', 'all', None, _('select all unresolved files')),
5534 ('l', 'list', None, _('list state of files needing merge')),
5534 ('l', 'list', None, _('list state of files needing merge')),
5535 ('m', 'mark', None, _('mark files as resolved')),
5535 ('m', 'mark', None, _('mark files as resolved')),
5536 ('u', 'unmark', None, _('mark files as unresolved')),
5536 ('u', 'unmark', None, _('mark files as unresolved')),
5537 ('n', 'no-status', None, _('hide status prefix'))]
5537 ('n', 'no-status', None, _('hide status prefix'))]
5538 + mergetoolopts + walkopts + formatteropts,
5538 + mergetoolopts + walkopts + formatteropts,
5539 _('[OPTION]... [FILE]...'),
5539 _('[OPTION]... [FILE]...'),
5540 inferrepo=True)
5540 inferrepo=True)
5541 def resolve(ui, repo, *pats, **opts):
5541 def resolve(ui, repo, *pats, **opts):
5542 """redo merges or set/view the merge status of files
5542 """redo merges or set/view the merge status of files
5543
5543
5544 Merges with unresolved conflicts are often the result of
5544 Merges with unresolved conflicts are often the result of
5545 non-interactive merging using the ``internal:merge`` configuration
5545 non-interactive merging using the ``internal:merge`` configuration
5546 setting, or a command-line merge tool like ``diff3``. The resolve
5546 setting, or a command-line merge tool like ``diff3``. The resolve
5547 command is used to manage the files involved in a merge, after
5547 command is used to manage the files involved in a merge, after
5548 :hg:`merge` has been run, and before :hg:`commit` is run (i.e. the
5548 :hg:`merge` has been run, and before :hg:`commit` is run (i.e. the
5549 working directory must have two parents). See :hg:`help
5549 working directory must have two parents). See :hg:`help
5550 merge-tools` for information on configuring merge tools.
5550 merge-tools` for information on configuring merge tools.
5551
5551
5552 The resolve command can be used in the following ways:
5552 The resolve command can be used in the following ways:
5553
5553
5554 - :hg:`resolve [--tool TOOL] FILE...`: attempt to re-merge the specified
5554 - :hg:`resolve [--tool TOOL] FILE...`: attempt to re-merge the specified
5555 files, discarding any previous merge attempts. Re-merging is not
5555 files, discarding any previous merge attempts. Re-merging is not
5556 performed for files already marked as resolved. Use ``--all/-a``
5556 performed for files already marked as resolved. Use ``--all/-a``
5557 to select all unresolved files. ``--tool`` can be used to specify
5557 to select all unresolved files. ``--tool`` can be used to specify
5558 the merge tool used for the given files. It overrides the HGMERGE
5558 the merge tool used for the given files. It overrides the HGMERGE
5559 environment variable and your configuration files. Previous file
5559 environment variable and your configuration files. Previous file
5560 contents are saved with a ``.orig`` suffix.
5560 contents are saved with a ``.orig`` suffix.
5561
5561
5562 - :hg:`resolve -m [FILE]`: mark a file as having been resolved
5562 - :hg:`resolve -m [FILE]`: mark a file as having been resolved
5563 (e.g. after having manually fixed-up the files). The default is
5563 (e.g. after having manually fixed-up the files). The default is
5564 to mark all unresolved files.
5564 to mark all unresolved files.
5565
5565
5566 - :hg:`resolve -u [FILE]...`: mark a file as unresolved. The
5566 - :hg:`resolve -u [FILE]...`: mark a file as unresolved. The
5567 default is to mark all resolved files.
5567 default is to mark all resolved files.
5568
5568
5569 - :hg:`resolve -l`: list files which had or still have conflicts.
5569 - :hg:`resolve -l`: list files which had or still have conflicts.
5570 In the printed list, ``U`` = unresolved and ``R`` = resolved.
5570 In the printed list, ``U`` = unresolved and ``R`` = resolved.
5571
5571
5572 Note that Mercurial will not let you commit files with unresolved
5572 Note that Mercurial will not let you commit files with unresolved
5573 merge conflicts. You must use :hg:`resolve -m ...` before you can
5573 merge conflicts. You must use :hg:`resolve -m ...` before you can
5574 commit after a conflicting merge.
5574 commit after a conflicting merge.
5575
5575
5576 Returns 0 on success, 1 if any files fail a resolve attempt.
5576 Returns 0 on success, 1 if any files fail a resolve attempt.
5577 """
5577 """
5578
5578
5579 all, mark, unmark, show, nostatus = \
5579 all, mark, unmark, show, nostatus = \
5580 [opts.get(o) for o in 'all mark unmark list no_status'.split()]
5580 [opts.get(o) for o in 'all mark unmark list no_status'.split()]
5581
5581
5582 if (show and (mark or unmark)) or (mark and unmark):
5582 if (show and (mark or unmark)) or (mark and unmark):
5583 raise error.Abort(_("too many options specified"))
5583 raise error.Abort(_("too many options specified"))
5584 if pats and all:
5584 if pats and all:
5585 raise error.Abort(_("can't specify --all and patterns"))
5585 raise error.Abort(_("can't specify --all and patterns"))
5586 if not (all or pats or show or mark or unmark):
5586 if not (all or pats or show or mark or unmark):
5587 raise error.Abort(_('no files or directories specified'),
5587 raise error.Abort(_('no files or directories specified'),
5588 hint=('use --all to re-merge all unresolved files'))
5588 hint=('use --all to re-merge all unresolved files'))
5589
5589
5590 if show:
5590 if show:
5591 fm = ui.formatter('resolve', opts)
5591 fm = ui.formatter('resolve', opts)
5592 ms = mergemod.mergestate(repo)
5592 ms = mergemod.mergestate(repo)
5593 m = scmutil.match(repo[None], pats, opts)
5593 m = scmutil.match(repo[None], pats, opts)
5594 for f in ms:
5594 for f in ms:
5595 if not m(f):
5595 if not m(f):
5596 continue
5596 continue
5597 l = 'resolve.' + {'u': 'unresolved', 'r': 'resolved',
5597 l = 'resolve.' + {'u': 'unresolved', 'r': 'resolved',
5598 'd': 'driverresolved'}[ms[f]]
5598 'd': 'driverresolved'}[ms[f]]
5599 fm.startitem()
5599 fm.startitem()
5600 fm.condwrite(not nostatus, 'status', '%s ', ms[f].upper(), label=l)
5600 fm.condwrite(not nostatus, 'status', '%s ', ms[f].upper(), label=l)
5601 fm.write('path', '%s\n', f, label=l)
5601 fm.write('path', '%s\n', f, label=l)
5602 fm.end()
5602 fm.end()
5603 return 0
5603 return 0
5604
5604
5605 wlock = repo.wlock()
5605 wlock = repo.wlock()
5606 try:
5606 try:
5607 ms = mergemod.mergestate(repo)
5607 ms = mergemod.mergestate(repo)
5608
5608
5609 if not (ms.active() or repo.dirstate.p2() != nullid):
5609 if not (ms.active() or repo.dirstate.p2() != nullid):
5610 raise error.Abort(
5610 raise error.Abort(
5611 _('resolve command not applicable when not merging'))
5611 _('resolve command not applicable when not merging'))
5612
5612
5613 wctx = repo[None]
5613 wctx = repo[None]
5614
5614
5615 if ms.mergedriver and ms.mdstate() == 'u':
5615 if ms.mergedriver and ms.mdstate() == 'u':
5616 proceed = mergemod.driverpreprocess(repo, ms, wctx)
5616 proceed = mergemod.driverpreprocess(repo, ms, wctx)
5617 ms.commit()
5617 ms.commit()
5618 # allow mark and unmark to go through
5618 # allow mark and unmark to go through
5619 if not mark and not unmark and not proceed:
5619 if not mark and not unmark and not proceed:
5620 return 1
5620 return 1
5621
5621
5622 m = scmutil.match(wctx, pats, opts)
5622 m = scmutil.match(wctx, pats, opts)
5623 ret = 0
5623 ret = 0
5624 didwork = False
5624 didwork = False
5625 runconclude = False
5625 runconclude = False
5626
5626
5627 tocomplete = []
5627 tocomplete = []
5628 for f in ms:
5628 for f in ms:
5629 if not m(f):
5629 if not m(f):
5630 continue
5630 continue
5631
5631
5632 didwork = True
5632 didwork = True
5633
5633
5634 # don't let driver-resolved files be marked, and run the conclude
5634 # don't let driver-resolved files be marked, and run the conclude
5635 # step if asked to resolve
5635 # step if asked to resolve
5636 if ms[f] == "d":
5636 if ms[f] == "d":
5637 exact = m.exact(f)
5637 exact = m.exact(f)
5638 if mark:
5638 if mark:
5639 if exact:
5639 if exact:
5640 ui.warn(_('not marking %s as it is driver-resolved\n')
5640 ui.warn(_('not marking %s as it is driver-resolved\n')
5641 % f)
5641 % f)
5642 elif unmark:
5642 elif unmark:
5643 if exact:
5643 if exact:
5644 ui.warn(_('not unmarking %s as it is driver-resolved\n')
5644 ui.warn(_('not unmarking %s as it is driver-resolved\n')
5645 % f)
5645 % f)
5646 else:
5646 else:
5647 runconclude = True
5647 runconclude = True
5648 continue
5648 continue
5649
5649
5650 if mark:
5650 if mark:
5651 ms.mark(f, "r")
5651 ms.mark(f, "r")
5652 elif unmark:
5652 elif unmark:
5653 ms.mark(f, "u")
5653 ms.mark(f, "u")
5654 else:
5654 else:
5655 # backup pre-resolve (merge uses .orig for its own purposes)
5655 # backup pre-resolve (merge uses .orig for its own purposes)
5656 a = repo.wjoin(f)
5656 a = repo.wjoin(f)
5657 try:
5657 try:
5658 util.copyfile(a, a + ".resolve")
5658 util.copyfile(a, a + ".resolve")
5659 except (IOError, OSError) as inst:
5659 except (IOError, OSError) as inst:
5660 if inst.errno != errno.ENOENT:
5660 if inst.errno != errno.ENOENT:
5661 raise
5661 raise
5662
5662
5663 try:
5663 try:
5664 # preresolve file
5664 # preresolve file
5665 ui.setconfig('ui', 'forcemerge', opts.get('tool', ''),
5665 ui.setconfig('ui', 'forcemerge', opts.get('tool', ''),
5666 'resolve')
5666 'resolve')
5667 complete, r = ms.preresolve(f, wctx)
5667 complete, r = ms.preresolve(f, wctx)
5668 if not complete:
5668 if not complete:
5669 tocomplete.append(f)
5669 tocomplete.append(f)
5670 elif r:
5670 elif r:
5671 ret = 1
5671 ret = 1
5672 finally:
5672 finally:
5673 ui.setconfig('ui', 'forcemerge', '', 'resolve')
5673 ui.setconfig('ui', 'forcemerge', '', 'resolve')
5674 ms.commit()
5674 ms.commit()
5675
5675
5676 # replace filemerge's .orig file with our resolve file
5676 # replace filemerge's .orig file with our resolve file
5677 # for files in tocomplete, ms.resolve will not overwrite
5677 # for files in tocomplete, ms.resolve will not overwrite
5678 # .orig -- only preresolve does
5678 # .orig -- only preresolve does
5679 try:
5679 try:
5680 util.rename(a + ".resolve", cmdutil.origpath(ui, repo, a))
5680 util.rename(a + ".resolve", cmdutil.origpath(ui, repo, a))
5681 except OSError as inst:
5681 except OSError as inst:
5682 if inst.errno != errno.ENOENT:
5682 if inst.errno != errno.ENOENT:
5683 raise
5683 raise
5684
5684
5685 for f in tocomplete:
5685 for f in tocomplete:
5686 try:
5686 try:
5687 # resolve file
5687 # resolve file
5688 ui.setconfig('ui', 'forcemerge', opts.get('tool', ''),
5688 ui.setconfig('ui', 'forcemerge', opts.get('tool', ''),
5689 'resolve')
5689 'resolve')
5690 r = ms.resolve(f, wctx)
5690 r = ms.resolve(f, wctx)
5691 if r:
5691 if r:
5692 ret = 1
5692 ret = 1
5693 finally:
5693 finally:
5694 ui.setconfig('ui', 'forcemerge', '', 'resolve')
5694 ui.setconfig('ui', 'forcemerge', '', 'resolve')
5695 ms.commit()
5695 ms.commit()
5696
5696
5697 ms.commit()
5697 ms.commit()
5698
5698
5699 if not didwork and pats:
5699 if not didwork and pats:
5700 ui.warn(_("arguments do not match paths that need resolving\n"))
5700 ui.warn(_("arguments do not match paths that need resolving\n"))
5701 elif ms.mergedriver and ms.mdstate() != 's':
5701 elif ms.mergedriver and ms.mdstate() != 's':
5702 # run conclude step when either a driver-resolved file is requested
5702 # run conclude step when either a driver-resolved file is requested
5703 # or there are no driver-resolved files
5703 # or there are no driver-resolved files
5704 # we can't use 'ret' to determine whether any files are unresolved
5704 # we can't use 'ret' to determine whether any files are unresolved
5705 # because we might not have tried to resolve some
5705 # because we might not have tried to resolve some
5706 if ((runconclude or not list(ms.driverresolved()))
5706 if ((runconclude or not list(ms.driverresolved()))
5707 and not list(ms.unresolved())):
5707 and not list(ms.unresolved())):
5708 proceed = mergemod.driverconclude(repo, ms, wctx)
5708 proceed = mergemod.driverconclude(repo, ms, wctx)
5709 ms.commit()
5709 ms.commit()
5710 if not proceed:
5710 if not proceed:
5711 return 1
5711 return 1
5712
5712
5713 finally:
5713 finally:
5714 wlock.release()
5714 wlock.release()
5715
5715
5716 # Nudge users into finishing an unfinished operation
5716 # Nudge users into finishing an unfinished operation
5717 unresolvedf = list(ms.unresolved())
5717 unresolvedf = list(ms.unresolved())
5718 driverresolvedf = list(ms.driverresolved())
5718 driverresolvedf = list(ms.driverresolved())
5719 if not unresolvedf and not driverresolvedf:
5719 if not unresolvedf and not driverresolvedf:
5720 ui.status(_('(no more unresolved files)\n'))
5720 ui.status(_('(no more unresolved files)\n'))
5721 elif not unresolvedf:
5721 elif not unresolvedf:
5722 ui.status(_('(no more unresolved files -- '
5722 ui.status(_('(no more unresolved files -- '
5723 'run "hg resolve --all" to conclude)\n'))
5723 'run "hg resolve --all" to conclude)\n'))
5724
5724
5725 return ret
5725 return ret
5726
5726
5727 @command('revert',
5727 @command('revert',
5728 [('a', 'all', None, _('revert all changes when no arguments given')),
5728 [('a', 'all', None, _('revert all changes when no arguments given')),
5729 ('d', 'date', '', _('tipmost revision matching date'), _('DATE')),
5729 ('d', 'date', '', _('tipmost revision matching date'), _('DATE')),
5730 ('r', 'rev', '', _('revert to the specified revision'), _('REV')),
5730 ('r', 'rev', '', _('revert to the specified revision'), _('REV')),
5731 ('C', 'no-backup', None, _('do not save backup copies of files')),
5731 ('C', 'no-backup', None, _('do not save backup copies of files')),
5732 ('i', 'interactive', None,
5732 ('i', 'interactive', None,
5733 _('interactively select the changes (EXPERIMENTAL)')),
5733 _('interactively select the changes (EXPERIMENTAL)')),
5734 ] + walkopts + dryrunopts,
5734 ] + walkopts + dryrunopts,
5735 _('[OPTION]... [-r REV] [NAME]...'))
5735 _('[OPTION]... [-r REV] [NAME]...'))
5736 def revert(ui, repo, *pats, **opts):
5736 def revert(ui, repo, *pats, **opts):
5737 """restore files to their checkout state
5737 """restore files to their checkout state
5738
5738
5739 .. note::
5739 .. note::
5740
5740
5741 To check out earlier revisions, you should use :hg:`update REV`.
5741 To check out earlier revisions, you should use :hg:`update REV`.
5742 To cancel an uncommitted merge (and lose your changes),
5742 To cancel an uncommitted merge (and lose your changes),
5743 use :hg:`update --clean .`.
5743 use :hg:`update --clean .`.
5744
5744
5745 With no revision specified, revert the specified files or directories
5745 With no revision specified, revert the specified files or directories
5746 to the contents they had in the parent of the working directory.
5746 to the contents they had in the parent of the working directory.
5747 This restores the contents of files to an unmodified
5747 This restores the contents of files to an unmodified
5748 state and unschedules adds, removes, copies, and renames. If the
5748 state and unschedules adds, removes, copies, and renames. If the
5749 working directory has two parents, you must explicitly specify a
5749 working directory has two parents, you must explicitly specify a
5750 revision.
5750 revision.
5751
5751
5752 Using the -r/--rev or -d/--date options, revert the given files or
5752 Using the -r/--rev or -d/--date options, revert the given files or
5753 directories to their states as of a specific revision. Because
5753 directories to their states as of a specific revision. Because
5754 revert does not change the working directory parents, this will
5754 revert does not change the working directory parents, this will
5755 cause these files to appear modified. This can be helpful to "back
5755 cause these files to appear modified. This can be helpful to "back
5756 out" some or all of an earlier change. See :hg:`backout` for a
5756 out" some or all of an earlier change. See :hg:`backout` for a
5757 related method.
5757 related method.
5758
5758
5759 Modified files are saved with a .orig suffix before reverting.
5759 Modified files are saved with a .orig suffix before reverting.
5760 To disable these backups, use --no-backup.
5760 To disable these backups, use --no-backup.
5761
5761
5762 See :hg:`help dates` for a list of formats valid for -d/--date.
5762 See :hg:`help dates` for a list of formats valid for -d/--date.
5763
5763
5764 See :hg:`help backout` for a way to reverse the effect of an
5764 See :hg:`help backout` for a way to reverse the effect of an
5765 earlier changeset.
5765 earlier changeset.
5766
5766
5767 Returns 0 on success.
5767 Returns 0 on success.
5768 """
5768 """
5769
5769
5770 if opts.get("date"):
5770 if opts.get("date"):
5771 if opts.get("rev"):
5771 if opts.get("rev"):
5772 raise error.Abort(_("you can't specify a revision and a date"))
5772 raise error.Abort(_("you can't specify a revision and a date"))
5773 opts["rev"] = cmdutil.finddate(ui, repo, opts["date"])
5773 opts["rev"] = cmdutil.finddate(ui, repo, opts["date"])
5774
5774
5775 parent, p2 = repo.dirstate.parents()
5775 parent, p2 = repo.dirstate.parents()
5776 if not opts.get('rev') and p2 != nullid:
5776 if not opts.get('rev') and p2 != nullid:
5777 # revert after merge is a trap for new users (issue2915)
5777 # revert after merge is a trap for new users (issue2915)
5778 raise error.Abort(_('uncommitted merge with no revision specified'),
5778 raise error.Abort(_('uncommitted merge with no revision specified'),
5779 hint=_('use "hg update" or see "hg help revert"'))
5779 hint=_('use "hg update" or see "hg help revert"'))
5780
5780
5781 ctx = scmutil.revsingle(repo, opts.get('rev'))
5781 ctx = scmutil.revsingle(repo, opts.get('rev'))
5782
5782
5783 if (not (pats or opts.get('include') or opts.get('exclude') or
5783 if (not (pats or opts.get('include') or opts.get('exclude') or
5784 opts.get('all') or opts.get('interactive'))):
5784 opts.get('all') or opts.get('interactive'))):
5785 msg = _("no files or directories specified")
5785 msg = _("no files or directories specified")
5786 if p2 != nullid:
5786 if p2 != nullid:
5787 hint = _("uncommitted merge, use --all to discard all changes,"
5787 hint = _("uncommitted merge, use --all to discard all changes,"
5788 " or 'hg update -C .' to abort the merge")
5788 " or 'hg update -C .' to abort the merge")
5789 raise error.Abort(msg, hint=hint)
5789 raise error.Abort(msg, hint=hint)
5790 dirty = any(repo.status())
5790 dirty = any(repo.status())
5791 node = ctx.node()
5791 node = ctx.node()
5792 if node != parent:
5792 if node != parent:
5793 if dirty:
5793 if dirty:
5794 hint = _("uncommitted changes, use --all to discard all"
5794 hint = _("uncommitted changes, use --all to discard all"
5795 " changes, or 'hg update %s' to update") % ctx.rev()
5795 " changes, or 'hg update %s' to update") % ctx.rev()
5796 else:
5796 else:
5797 hint = _("use --all to revert all files,"
5797 hint = _("use --all to revert all files,"
5798 " or 'hg update %s' to update") % ctx.rev()
5798 " or 'hg update %s' to update") % ctx.rev()
5799 elif dirty:
5799 elif dirty:
5800 hint = _("uncommitted changes, use --all to discard all changes")
5800 hint = _("uncommitted changes, use --all to discard all changes")
5801 else:
5801 else:
5802 hint = _("use --all to revert all files")
5802 hint = _("use --all to revert all files")
5803 raise error.Abort(msg, hint=hint)
5803 raise error.Abort(msg, hint=hint)
5804
5804
5805 return cmdutil.revert(ui, repo, ctx, (parent, p2), *pats, **opts)
5805 return cmdutil.revert(ui, repo, ctx, (parent, p2), *pats, **opts)
5806
5806
5807 @command('rollback', dryrunopts +
5807 @command('rollback', dryrunopts +
5808 [('f', 'force', False, _('ignore safety measures'))])
5808 [('f', 'force', False, _('ignore safety measures'))])
5809 def rollback(ui, repo, **opts):
5809 def rollback(ui, repo, **opts):
5810 """roll back the last transaction (DANGEROUS) (DEPRECATED)
5810 """roll back the last transaction (DANGEROUS) (DEPRECATED)
5811
5811
5812 Please use :hg:`commit --amend` instead of rollback to correct
5812 Please use :hg:`commit --amend` instead of rollback to correct
5813 mistakes in the last commit.
5813 mistakes in the last commit.
5814
5814
5815 This command should be used with care. There is only one level of
5815 This command should be used with care. There is only one level of
5816 rollback, and there is no way to undo a rollback. It will also
5816 rollback, and there is no way to undo a rollback. It will also
5817 restore the dirstate at the time of the last transaction, losing
5817 restore the dirstate at the time of the last transaction, losing
5818 any dirstate changes since that time. This command does not alter
5818 any dirstate changes since that time. This command does not alter
5819 the working directory.
5819 the working directory.
5820
5820
5821 Transactions are used to encapsulate the effects of all commands
5821 Transactions are used to encapsulate the effects of all commands
5822 that create new changesets or propagate existing changesets into a
5822 that create new changesets or propagate existing changesets into a
5823 repository.
5823 repository.
5824
5824
5825 .. container:: verbose
5825 .. container:: verbose
5826
5826
5827 For example, the following commands are transactional, and their
5827 For example, the following commands are transactional, and their
5828 effects can be rolled back:
5828 effects can be rolled back:
5829
5829
5830 - commit
5830 - commit
5831 - import
5831 - import
5832 - pull
5832 - pull
5833 - push (with this repository as the destination)
5833 - push (with this repository as the destination)
5834 - unbundle
5834 - unbundle
5835
5835
5836 To avoid permanent data loss, rollback will refuse to rollback a
5836 To avoid permanent data loss, rollback will refuse to rollback a
5837 commit transaction if it isn't checked out. Use --force to
5837 commit transaction if it isn't checked out. Use --force to
5838 override this protection.
5838 override this protection.
5839
5839
5840 This command is not intended for use on public repositories. Once
5840 This command is not intended for use on public repositories. Once
5841 changes are visible for pull by other users, rolling a transaction
5841 changes are visible for pull by other users, rolling a transaction
5842 back locally is ineffective (someone else may already have pulled
5842 back locally is ineffective (someone else may already have pulled
5843 the changes). Furthermore, a race is possible with readers of the
5843 the changes). Furthermore, a race is possible with readers of the
5844 repository; for example an in-progress pull from the repository
5844 repository; for example an in-progress pull from the repository
5845 may fail if a rollback is performed.
5845 may fail if a rollback is performed.
5846
5846
5847 Returns 0 on success, 1 if no rollback data is available.
5847 Returns 0 on success, 1 if no rollback data is available.
5848 """
5848 """
5849 return repo.rollback(dryrun=opts.get('dry_run'),
5849 return repo.rollback(dryrun=opts.get('dry_run'),
5850 force=opts.get('force'))
5850 force=opts.get('force'))
5851
5851
5852 @command('root', [])
5852 @command('root', [])
5853 def root(ui, repo):
5853 def root(ui, repo):
5854 """print the root (top) of the current working directory
5854 """print the root (top) of the current working directory
5855
5855
5856 Print the root directory of the current repository.
5856 Print the root directory of the current repository.
5857
5857
5858 Returns 0 on success.
5858 Returns 0 on success.
5859 """
5859 """
5860 ui.write(repo.root + "\n")
5860 ui.write(repo.root + "\n")
5861
5861
5862 @command('^serve',
5862 @command('^serve',
5863 [('A', 'accesslog', '', _('name of access log file to write to'),
5863 [('A', 'accesslog', '', _('name of access log file to write to'),
5864 _('FILE')),
5864 _('FILE')),
5865 ('d', 'daemon', None, _('run server in background')),
5865 ('d', 'daemon', None, _('run server in background')),
5866 ('', 'daemon-pipefds', '', _('used internally by daemon mode'), _('FILE')),
5866 ('', 'daemon-pipefds', '', _('used internally by daemon mode'), _('FILE')),
5867 ('E', 'errorlog', '', _('name of error log file to write to'), _('FILE')),
5867 ('E', 'errorlog', '', _('name of error log file to write to'), _('FILE')),
5868 # use string type, then we can check if something was passed
5868 # use string type, then we can check if something was passed
5869 ('p', 'port', '', _('port to listen on (default: 8000)'), _('PORT')),
5869 ('p', 'port', '', _('port to listen on (default: 8000)'), _('PORT')),
5870 ('a', 'address', '', _('address to listen on (default: all interfaces)'),
5870 ('a', 'address', '', _('address to listen on (default: all interfaces)'),
5871 _('ADDR')),
5871 _('ADDR')),
5872 ('', 'prefix', '', _('prefix path to serve from (default: server root)'),
5872 ('', 'prefix', '', _('prefix path to serve from (default: server root)'),
5873 _('PREFIX')),
5873 _('PREFIX')),
5874 ('n', 'name', '',
5874 ('n', 'name', '',
5875 _('name to show in web pages (default: working directory)'), _('NAME')),
5875 _('name to show in web pages (default: working directory)'), _('NAME')),
5876 ('', 'web-conf', '',
5876 ('', 'web-conf', '',
5877 _('name of the hgweb config file (see "hg help hgweb")'), _('FILE')),
5877 _('name of the hgweb config file (see "hg help hgweb")'), _('FILE')),
5878 ('', 'webdir-conf', '', _('name of the hgweb config file (DEPRECATED)'),
5878 ('', 'webdir-conf', '', _('name of the hgweb config file (DEPRECATED)'),
5879 _('FILE')),
5879 _('FILE')),
5880 ('', 'pid-file', '', _('name of file to write process ID to'), _('FILE')),
5880 ('', 'pid-file', '', _('name of file to write process ID to'), _('FILE')),
5881 ('', 'stdio', None, _('for remote clients')),
5881 ('', 'stdio', None, _('for remote clients')),
5882 ('', 'cmdserver', '', _('for remote clients'), _('MODE')),
5882 ('', 'cmdserver', '', _('for remote clients'), _('MODE')),
5883 ('t', 'templates', '', _('web templates to use'), _('TEMPLATE')),
5883 ('t', 'templates', '', _('web templates to use'), _('TEMPLATE')),
5884 ('', 'style', '', _('template style to use'), _('STYLE')),
5884 ('', 'style', '', _('template style to use'), _('STYLE')),
5885 ('6', 'ipv6', None, _('use IPv6 in addition to IPv4')),
5885 ('6', 'ipv6', None, _('use IPv6 in addition to IPv4')),
5886 ('', 'certificate', '', _('SSL certificate file'), _('FILE'))],
5886 ('', 'certificate', '', _('SSL certificate file'), _('FILE'))],
5887 _('[OPTION]...'),
5887 _('[OPTION]...'),
5888 optionalrepo=True)
5888 optionalrepo=True)
5889 def serve(ui, repo, **opts):
5889 def serve(ui, repo, **opts):
5890 """start stand-alone webserver
5890 """start stand-alone webserver
5891
5891
5892 Start a local HTTP repository browser and pull server. You can use
5892 Start a local HTTP repository browser and pull server. You can use
5893 this for ad-hoc sharing and browsing of repositories. It is
5893 this for ad-hoc sharing and browsing of repositories. It is
5894 recommended to use a real web server to serve a repository for
5894 recommended to use a real web server to serve a repository for
5895 longer periods of time.
5895 longer periods of time.
5896
5896
5897 Please note that the server does not implement access control.
5897 Please note that the server does not implement access control.
5898 This means that, by default, anybody can read from the server and
5898 This means that, by default, anybody can read from the server and
5899 nobody can write to it by default. Set the ``web.allow_push``
5899 nobody can write to it by default. Set the ``web.allow_push``
5900 option to ``*`` to allow everybody to push to the server. You
5900 option to ``*`` to allow everybody to push to the server. You
5901 should use a real web server if you need to authenticate users.
5901 should use a real web server if you need to authenticate users.
5902
5902
5903 By default, the server logs accesses to stdout and errors to
5903 By default, the server logs accesses to stdout and errors to
5904 stderr. Use the -A/--accesslog and -E/--errorlog options to log to
5904 stderr. Use the -A/--accesslog and -E/--errorlog options to log to
5905 files.
5905 files.
5906
5906
5907 To have the server choose a free port number to listen on, specify
5907 To have the server choose a free port number to listen on, specify
5908 a port number of 0; in this case, the server will print the port
5908 a port number of 0; in this case, the server will print the port
5909 number it uses.
5909 number it uses.
5910
5910
5911 Returns 0 on success.
5911 Returns 0 on success.
5912 """
5912 """
5913
5913
5914 if opts["stdio"] and opts["cmdserver"]:
5914 if opts["stdio"] and opts["cmdserver"]:
5915 raise error.Abort(_("cannot use --stdio with --cmdserver"))
5915 raise error.Abort(_("cannot use --stdio with --cmdserver"))
5916
5916
5917 if opts["stdio"]:
5917 if opts["stdio"]:
5918 if repo is None:
5918 if repo is None:
5919 raise error.RepoError(_("there is no Mercurial repository here"
5919 raise error.RepoError(_("there is no Mercurial repository here"
5920 " (.hg not found)"))
5920 " (.hg not found)"))
5921 s = sshserver.sshserver(ui, repo)
5921 s = sshserver.sshserver(ui, repo)
5922 s.serve_forever()
5922 s.serve_forever()
5923
5923
5924 if opts["cmdserver"]:
5924 if opts["cmdserver"]:
5925 import commandserver
5925 import commandserver
5926 service = commandserver.createservice(ui, repo, opts)
5926 service = commandserver.createservice(ui, repo, opts)
5927 return cmdutil.service(opts, initfn=service.init, runfn=service.run)
5927 return cmdutil.service(opts, initfn=service.init, runfn=service.run)
5928
5928
5929 # this way we can check if something was given in the command-line
5929 # this way we can check if something was given in the command-line
5930 if opts.get('port'):
5930 if opts.get('port'):
5931 opts['port'] = util.getport(opts.get('port'))
5931 opts['port'] = util.getport(opts.get('port'))
5932
5932
5933 if repo:
5933 if repo:
5934 baseui = repo.baseui
5934 baseui = repo.baseui
5935 else:
5935 else:
5936 baseui = ui
5936 baseui = ui
5937 optlist = ("name templates style address port prefix ipv6"
5937 optlist = ("name templates style address port prefix ipv6"
5938 " accesslog errorlog certificate encoding")
5938 " accesslog errorlog certificate encoding")
5939 for o in optlist.split():
5939 for o in optlist.split():
5940 val = opts.get(o, '')
5940 val = opts.get(o, '')
5941 if val in (None, ''): # should check against default options instead
5941 if val in (None, ''): # should check against default options instead
5942 continue
5942 continue
5943 baseui.setconfig("web", o, val, 'serve')
5943 baseui.setconfig("web", o, val, 'serve')
5944 if repo and repo.ui != baseui:
5944 if repo and repo.ui != baseui:
5945 repo.ui.setconfig("web", o, val, 'serve')
5945 repo.ui.setconfig("web", o, val, 'serve')
5946
5946
5947 o = opts.get('web_conf') or opts.get('webdir_conf')
5947 o = opts.get('web_conf') or opts.get('webdir_conf')
5948 if not o:
5948 if not o:
5949 if not repo:
5949 if not repo:
5950 raise error.RepoError(_("there is no Mercurial repository"
5950 raise error.RepoError(_("there is no Mercurial repository"
5951 " here (.hg not found)"))
5951 " here (.hg not found)"))
5952 o = repo
5952 o = repo
5953
5953
5954 app = hgweb.hgweb(o, baseui=baseui)
5954 app = hgweb.hgweb(o, baseui=baseui)
5955 service = httpservice(ui, app, opts)
5955 service = httpservice(ui, app, opts)
5956 cmdutil.service(opts, initfn=service.init, runfn=service.run)
5956 cmdutil.service(opts, initfn=service.init, runfn=service.run)
5957
5957
5958 class httpservice(object):
5958 class httpservice(object):
5959 def __init__(self, ui, app, opts):
5959 def __init__(self, ui, app, opts):
5960 self.ui = ui
5960 self.ui = ui
5961 self.app = app
5961 self.app = app
5962 self.opts = opts
5962 self.opts = opts
5963
5963
5964 def init(self):
5964 def init(self):
5965 util.setsignalhandler()
5965 util.setsignalhandler()
5966 self.httpd = hgweb_server.create_server(self.ui, self.app)
5966 self.httpd = hgweb_server.create_server(self.ui, self.app)
5967
5967
5968 if self.opts['port'] and not self.ui.verbose:
5968 if self.opts['port'] and not self.ui.verbose:
5969 return
5969 return
5970
5970
5971 if self.httpd.prefix:
5971 if self.httpd.prefix:
5972 prefix = self.httpd.prefix.strip('/') + '/'
5972 prefix = self.httpd.prefix.strip('/') + '/'
5973 else:
5973 else:
5974 prefix = ''
5974 prefix = ''
5975
5975
5976 port = ':%d' % self.httpd.port
5976 port = ':%d' % self.httpd.port
5977 if port == ':80':
5977 if port == ':80':
5978 port = ''
5978 port = ''
5979
5979
5980 bindaddr = self.httpd.addr
5980 bindaddr = self.httpd.addr
5981 if bindaddr == '0.0.0.0':
5981 if bindaddr == '0.0.0.0':
5982 bindaddr = '*'
5982 bindaddr = '*'
5983 elif ':' in bindaddr: # IPv6
5983 elif ':' in bindaddr: # IPv6
5984 bindaddr = '[%s]' % bindaddr
5984 bindaddr = '[%s]' % bindaddr
5985
5985
5986 fqaddr = self.httpd.fqaddr
5986 fqaddr = self.httpd.fqaddr
5987 if ':' in fqaddr:
5987 if ':' in fqaddr:
5988 fqaddr = '[%s]' % fqaddr
5988 fqaddr = '[%s]' % fqaddr
5989 if self.opts['port']:
5989 if self.opts['port']:
5990 write = self.ui.status
5990 write = self.ui.status
5991 else:
5991 else:
5992 write = self.ui.write
5992 write = self.ui.write
5993 write(_('listening at http://%s%s/%s (bound to %s:%d)\n') %
5993 write(_('listening at http://%s%s/%s (bound to %s:%d)\n') %
5994 (fqaddr, port, prefix, bindaddr, self.httpd.port))
5994 (fqaddr, port, prefix, bindaddr, self.httpd.port))
5995 self.ui.flush() # avoid buffering of status message
5995 self.ui.flush() # avoid buffering of status message
5996
5996
5997 def run(self):
5997 def run(self):
5998 self.httpd.serve_forever()
5998 self.httpd.serve_forever()
5999
5999
6000
6000
6001 @command('^status|st',
6001 @command('^status|st',
6002 [('A', 'all', None, _('show status of all files')),
6002 [('A', 'all', None, _('show status of all files')),
6003 ('m', 'modified', None, _('show only modified files')),
6003 ('m', 'modified', None, _('show only modified files')),
6004 ('a', 'added', None, _('show only added files')),
6004 ('a', 'added', None, _('show only added files')),
6005 ('r', 'removed', None, _('show only removed files')),
6005 ('r', 'removed', None, _('show only removed files')),
6006 ('d', 'deleted', None, _('show only deleted (but tracked) files')),
6006 ('d', 'deleted', None, _('show only deleted (but tracked) files')),
6007 ('c', 'clean', None, _('show only files without changes')),
6007 ('c', 'clean', None, _('show only files without changes')),
6008 ('u', 'unknown', None, _('show only unknown (not tracked) files')),
6008 ('u', 'unknown', None, _('show only unknown (not tracked) files')),
6009 ('i', 'ignored', None, _('show only ignored files')),
6009 ('i', 'ignored', None, _('show only ignored files')),
6010 ('n', 'no-status', None, _('hide status prefix')),
6010 ('n', 'no-status', None, _('hide status prefix')),
6011 ('C', 'copies', None, _('show source of copied files')),
6011 ('C', 'copies', None, _('show source of copied files')),
6012 ('0', 'print0', None, _('end filenames with NUL, for use with xargs')),
6012 ('0', 'print0', None, _('end filenames with NUL, for use with xargs')),
6013 ('', 'rev', [], _('show difference from revision'), _('REV')),
6013 ('', 'rev', [], _('show difference from revision'), _('REV')),
6014 ('', 'change', '', _('list the changed files of a revision'), _('REV')),
6014 ('', 'change', '', _('list the changed files of a revision'), _('REV')),
6015 ] + walkopts + subrepoopts + formatteropts,
6015 ] + walkopts + subrepoopts + formatteropts,
6016 _('[OPTION]... [FILE]...'),
6016 _('[OPTION]... [FILE]...'),
6017 inferrepo=True)
6017 inferrepo=True)
6018 def status(ui, repo, *pats, **opts):
6018 def status(ui, repo, *pats, **opts):
6019 """show changed files in the working directory
6019 """show changed files in the working directory
6020
6020
6021 Show status of files in the repository. If names are given, only
6021 Show status of files in the repository. If names are given, only
6022 files that match are shown. Files that are clean or ignored or
6022 files that match are shown. Files that are clean or ignored or
6023 the source of a copy/move operation, are not listed unless
6023 the source of a copy/move operation, are not listed unless
6024 -c/--clean, -i/--ignored, -C/--copies or -A/--all are given.
6024 -c/--clean, -i/--ignored, -C/--copies or -A/--all are given.
6025 Unless options described with "show only ..." are given, the
6025 Unless options described with "show only ..." are given, the
6026 options -mardu are used.
6026 options -mardu are used.
6027
6027
6028 Option -q/--quiet hides untracked (unknown and ignored) files
6028 Option -q/--quiet hides untracked (unknown and ignored) files
6029 unless explicitly requested with -u/--unknown or -i/--ignored.
6029 unless explicitly requested with -u/--unknown or -i/--ignored.
6030
6030
6031 .. note::
6031 .. note::
6032
6032
6033 status may appear to disagree with diff if permissions have
6033 status may appear to disagree with diff if permissions have
6034 changed or a merge has occurred. The standard diff format does
6034 changed or a merge has occurred. The standard diff format does
6035 not report permission changes and diff only reports changes
6035 not report permission changes and diff only reports changes
6036 relative to one merge parent.
6036 relative to one merge parent.
6037
6037
6038 If one revision is given, it is used as the base revision.
6038 If one revision is given, it is used as the base revision.
6039 If two revisions are given, the differences between them are
6039 If two revisions are given, the differences between them are
6040 shown. The --change option can also be used as a shortcut to list
6040 shown. The --change option can also be used as a shortcut to list
6041 the changed files of a revision from its first parent.
6041 the changed files of a revision from its first parent.
6042
6042
6043 The codes used to show the status of files are::
6043 The codes used to show the status of files are::
6044
6044
6045 M = modified
6045 M = modified
6046 A = added
6046 A = added
6047 R = removed
6047 R = removed
6048 C = clean
6048 C = clean
6049 ! = missing (deleted by non-hg command, but still tracked)
6049 ! = missing (deleted by non-hg command, but still tracked)
6050 ? = not tracked
6050 ? = not tracked
6051 I = ignored
6051 I = ignored
6052 = origin of the previous file (with --copies)
6052 = origin of the previous file (with --copies)
6053
6053
6054 .. container:: verbose
6054 .. container:: verbose
6055
6055
6056 Examples:
6056 Examples:
6057
6057
6058 - show changes in the working directory relative to a
6058 - show changes in the working directory relative to a
6059 changeset::
6059 changeset::
6060
6060
6061 hg status --rev 9353
6061 hg status --rev 9353
6062
6062
6063 - show changes in the working directory relative to the
6063 - show changes in the working directory relative to the
6064 current directory (see :hg:`help patterns` for more information)::
6064 current directory (see :hg:`help patterns` for more information)::
6065
6065
6066 hg status re:
6066 hg status re:
6067
6067
6068 - show all changes including copies in an existing changeset::
6068 - show all changes including copies in an existing changeset::
6069
6069
6070 hg status --copies --change 9353
6070 hg status --copies --change 9353
6071
6071
6072 - get a NUL separated list of added files, suitable for xargs::
6072 - get a NUL separated list of added files, suitable for xargs::
6073
6073
6074 hg status -an0
6074 hg status -an0
6075
6075
6076 Returns 0 on success.
6076 Returns 0 on success.
6077 """
6077 """
6078
6078
6079 revs = opts.get('rev')
6079 revs = opts.get('rev')
6080 change = opts.get('change')
6080 change = opts.get('change')
6081
6081
6082 if revs and change:
6082 if revs and change:
6083 msg = _('cannot specify --rev and --change at the same time')
6083 msg = _('cannot specify --rev and --change at the same time')
6084 raise error.Abort(msg)
6084 raise error.Abort(msg)
6085 elif change:
6085 elif change:
6086 node2 = scmutil.revsingle(repo, change, None).node()
6086 node2 = scmutil.revsingle(repo, change, None).node()
6087 node1 = repo[node2].p1().node()
6087 node1 = repo[node2].p1().node()
6088 else:
6088 else:
6089 node1, node2 = scmutil.revpair(repo, revs)
6089 node1, node2 = scmutil.revpair(repo, revs)
6090
6090
6091 if pats:
6091 if pats:
6092 cwd = repo.getcwd()
6092 cwd = repo.getcwd()
6093 else:
6093 else:
6094 cwd = ''
6094 cwd = ''
6095
6095
6096 if opts.get('print0'):
6096 if opts.get('print0'):
6097 end = '\0'
6097 end = '\0'
6098 else:
6098 else:
6099 end = '\n'
6099 end = '\n'
6100 copy = {}
6100 copy = {}
6101 states = 'modified added removed deleted unknown ignored clean'.split()
6101 states = 'modified added removed deleted unknown ignored clean'.split()
6102 show = [k for k in states if opts.get(k)]
6102 show = [k for k in states if opts.get(k)]
6103 if opts.get('all'):
6103 if opts.get('all'):
6104 show += ui.quiet and (states[:4] + ['clean']) or states
6104 show += ui.quiet and (states[:4] + ['clean']) or states
6105 if not show:
6105 if not show:
6106 if ui.quiet:
6106 if ui.quiet:
6107 show = states[:4]
6107 show = states[:4]
6108 else:
6108 else:
6109 show = states[:5]
6109 show = states[:5]
6110
6110
6111 m = scmutil.match(repo[node2], pats, opts)
6111 m = scmutil.match(repo[node2], pats, opts)
6112 stat = repo.status(node1, node2, m,
6112 stat = repo.status(node1, node2, m,
6113 'ignored' in show, 'clean' in show, 'unknown' in show,
6113 'ignored' in show, 'clean' in show, 'unknown' in show,
6114 opts.get('subrepos'))
6114 opts.get('subrepos'))
6115 changestates = zip(states, 'MAR!?IC', stat)
6115 changestates = zip(states, 'MAR!?IC', stat)
6116
6116
6117 if (opts.get('all') or opts.get('copies')
6117 if (opts.get('all') or opts.get('copies')
6118 or ui.configbool('ui', 'statuscopies')) and not opts.get('no_status'):
6118 or ui.configbool('ui', 'statuscopies')) and not opts.get('no_status'):
6119 copy = copies.pathcopies(repo[node1], repo[node2], m)
6119 copy = copies.pathcopies(repo[node1], repo[node2], m)
6120
6120
6121 fm = ui.formatter('status', opts)
6121 fm = ui.formatter('status', opts)
6122 fmt = '%s' + end
6122 fmt = '%s' + end
6123 showchar = not opts.get('no_status')
6123 showchar = not opts.get('no_status')
6124
6124
6125 for state, char, files in changestates:
6125 for state, char, files in changestates:
6126 if state in show:
6126 if state in show:
6127 label = 'status.' + state
6127 label = 'status.' + state
6128 for f in files:
6128 for f in files:
6129 fm.startitem()
6129 fm.startitem()
6130 fm.condwrite(showchar, 'status', '%s ', char, label=label)
6130 fm.condwrite(showchar, 'status', '%s ', char, label=label)
6131 fm.write('path', fmt, repo.pathto(f, cwd), label=label)
6131 fm.write('path', fmt, repo.pathto(f, cwd), label=label)
6132 if f in copy:
6132 if f in copy:
6133 fm.write("copy", ' %s' + end, repo.pathto(copy[f], cwd),
6133 fm.write("copy", ' %s' + end, repo.pathto(copy[f], cwd),
6134 label='status.copied')
6134 label='status.copied')
6135 fm.end()
6135 fm.end()
6136
6136
6137 @command('^summary|sum',
6137 @command('^summary|sum',
6138 [('', 'remote', None, _('check for push and pull'))], '[--remote]')
6138 [('', 'remote', None, _('check for push and pull'))], '[--remote]')
6139 def summary(ui, repo, **opts):
6139 def summary(ui, repo, **opts):
6140 """summarize working directory state
6140 """summarize working directory state
6141
6141
6142 This generates a brief summary of the working directory state,
6142 This generates a brief summary of the working directory state,
6143 including parents, branch, commit status, phase and available updates.
6143 including parents, branch, commit status, phase and available updates.
6144
6144
6145 With the --remote option, this will check the default paths for
6145 With the --remote option, this will check the default paths for
6146 incoming and outgoing changes. This can be time-consuming.
6146 incoming and outgoing changes. This can be time-consuming.
6147
6147
6148 Returns 0 on success.
6148 Returns 0 on success.
6149 """
6149 """
6150
6150
6151 ctx = repo[None]
6151 ctx = repo[None]
6152 parents = ctx.parents()
6152 parents = ctx.parents()
6153 pnode = parents[0].node()
6153 pnode = parents[0].node()
6154 marks = []
6154 marks = []
6155
6155
6156 for p in parents:
6156 for p in parents:
6157 # label with log.changeset (instead of log.parent) since this
6157 # label with log.changeset (instead of log.parent) since this
6158 # shows a working directory parent *changeset*:
6158 # shows a working directory parent *changeset*:
6159 # i18n: column positioning for "hg summary"
6159 # i18n: column positioning for "hg summary"
6160 ui.write(_('parent: %d:%s ') % (p.rev(), str(p)),
6160 ui.write(_('parent: %d:%s ') % (p.rev(), str(p)),
6161 label='log.changeset changeset.%s' % p.phasestr())
6161 label='log.changeset changeset.%s' % p.phasestr())
6162 ui.write(' '.join(p.tags()), label='log.tag')
6162 ui.write(' '.join(p.tags()), label='log.tag')
6163 if p.bookmarks():
6163 if p.bookmarks():
6164 marks.extend(p.bookmarks())
6164 marks.extend(p.bookmarks())
6165 if p.rev() == -1:
6165 if p.rev() == -1:
6166 if not len(repo):
6166 if not len(repo):
6167 ui.write(_(' (empty repository)'))
6167 ui.write(_(' (empty repository)'))
6168 else:
6168 else:
6169 ui.write(_(' (no revision checked out)'))
6169 ui.write(_(' (no revision checked out)'))
6170 ui.write('\n')
6170 ui.write('\n')
6171 if p.description():
6171 if p.description():
6172 ui.status(' ' + p.description().splitlines()[0].strip() + '\n',
6172 ui.status(' ' + p.description().splitlines()[0].strip() + '\n',
6173 label='log.summary')
6173 label='log.summary')
6174
6174
6175 branch = ctx.branch()
6175 branch = ctx.branch()
6176 bheads = repo.branchheads(branch)
6176 bheads = repo.branchheads(branch)
6177 # i18n: column positioning for "hg summary"
6177 # i18n: column positioning for "hg summary"
6178 m = _('branch: %s\n') % branch
6178 m = _('branch: %s\n') % branch
6179 if branch != 'default':
6179 if branch != 'default':
6180 ui.write(m, label='log.branch')
6180 ui.write(m, label='log.branch')
6181 else:
6181 else:
6182 ui.status(m, label='log.branch')
6182 ui.status(m, label='log.branch')
6183
6183
6184 if marks:
6184 if marks:
6185 active = repo._activebookmark
6185 active = repo._activebookmark
6186 # i18n: column positioning for "hg summary"
6186 # i18n: column positioning for "hg summary"
6187 ui.write(_('bookmarks:'), label='log.bookmark')
6187 ui.write(_('bookmarks:'), label='log.bookmark')
6188 if active is not None:
6188 if active is not None:
6189 if active in marks:
6189 if active in marks:
6190 ui.write(' *' + active, label=activebookmarklabel)
6190 ui.write(' *' + active, label=activebookmarklabel)
6191 marks.remove(active)
6191 marks.remove(active)
6192 else:
6192 else:
6193 ui.write(' [%s]' % active, label=activebookmarklabel)
6193 ui.write(' [%s]' % active, label=activebookmarklabel)
6194 for m in marks:
6194 for m in marks:
6195 ui.write(' ' + m, label='log.bookmark')
6195 ui.write(' ' + m, label='log.bookmark')
6196 ui.write('\n', label='log.bookmark')
6196 ui.write('\n', label='log.bookmark')
6197
6197
6198 status = repo.status(unknown=True)
6198 status = repo.status(unknown=True)
6199
6199
6200 c = repo.dirstate.copies()
6200 c = repo.dirstate.copies()
6201 copied, renamed = [], []
6201 copied, renamed = [], []
6202 for d, s in c.iteritems():
6202 for d, s in c.iteritems():
6203 if s in status.removed:
6203 if s in status.removed:
6204 status.removed.remove(s)
6204 status.removed.remove(s)
6205 renamed.append(d)
6205 renamed.append(d)
6206 else:
6206 else:
6207 copied.append(d)
6207 copied.append(d)
6208 if d in status.added:
6208 if d in status.added:
6209 status.added.remove(d)
6209 status.added.remove(d)
6210
6210
6211 ms = mergemod.mergestate(repo)
6211 ms = mergemod.mergestate(repo)
6212 unresolved = [f for f in ms if ms[f] == 'u']
6212 unresolved = [f for f in ms if ms[f] == 'u']
6213
6213
6214 subs = [s for s in ctx.substate if ctx.sub(s).dirty()]
6214 subs = [s for s in ctx.substate if ctx.sub(s).dirty()]
6215
6215
6216 labels = [(ui.label(_('%d modified'), 'status.modified'), status.modified),
6216 labels = [(ui.label(_('%d modified'), 'status.modified'), status.modified),
6217 (ui.label(_('%d added'), 'status.added'), status.added),
6217 (ui.label(_('%d added'), 'status.added'), status.added),
6218 (ui.label(_('%d removed'), 'status.removed'), status.removed),
6218 (ui.label(_('%d removed'), 'status.removed'), status.removed),
6219 (ui.label(_('%d renamed'), 'status.copied'), renamed),
6219 (ui.label(_('%d renamed'), 'status.copied'), renamed),
6220 (ui.label(_('%d copied'), 'status.copied'), copied),
6220 (ui.label(_('%d copied'), 'status.copied'), copied),
6221 (ui.label(_('%d deleted'), 'status.deleted'), status.deleted),
6221 (ui.label(_('%d deleted'), 'status.deleted'), status.deleted),
6222 (ui.label(_('%d unknown'), 'status.unknown'), status.unknown),
6222 (ui.label(_('%d unknown'), 'status.unknown'), status.unknown),
6223 (ui.label(_('%d unresolved'), 'resolve.unresolved'), unresolved),
6223 (ui.label(_('%d unresolved'), 'resolve.unresolved'), unresolved),
6224 (ui.label(_('%d subrepos'), 'status.modified'), subs)]
6224 (ui.label(_('%d subrepos'), 'status.modified'), subs)]
6225 t = []
6225 t = []
6226 for l, s in labels:
6226 for l, s in labels:
6227 if s:
6227 if s:
6228 t.append(l % len(s))
6228 t.append(l % len(s))
6229
6229
6230 t = ', '.join(t)
6230 t = ', '.join(t)
6231 cleanworkdir = False
6231 cleanworkdir = False
6232
6232
6233 if repo.vfs.exists('updatestate'):
6233 if repo.vfs.exists('updatestate'):
6234 t += _(' (interrupted update)')
6234 t += _(' (interrupted update)')
6235 elif len(parents) > 1:
6235 elif len(parents) > 1:
6236 t += _(' (merge)')
6236 t += _(' (merge)')
6237 elif branch != parents[0].branch():
6237 elif branch != parents[0].branch():
6238 t += _(' (new branch)')
6238 t += _(' (new branch)')
6239 elif (parents[0].closesbranch() and
6239 elif (parents[0].closesbranch() and
6240 pnode in repo.branchheads(branch, closed=True)):
6240 pnode in repo.branchheads(branch, closed=True)):
6241 t += _(' (head closed)')
6241 t += _(' (head closed)')
6242 elif not (status.modified or status.added or status.removed or renamed or
6242 elif not (status.modified or status.added or status.removed or renamed or
6243 copied or subs):
6243 copied or subs):
6244 t += _(' (clean)')
6244 t += _(' (clean)')
6245 cleanworkdir = True
6245 cleanworkdir = True
6246 elif pnode not in bheads:
6246 elif pnode not in bheads:
6247 t += _(' (new branch head)')
6247 t += _(' (new branch head)')
6248
6248
6249 if parents:
6249 if parents:
6250 pendingphase = max(p.phase() for p in parents)
6250 pendingphase = max(p.phase() for p in parents)
6251 else:
6251 else:
6252 pendingphase = phases.public
6252 pendingphase = phases.public
6253
6253
6254 if pendingphase > phases.newcommitphase(ui):
6254 if pendingphase > phases.newcommitphase(ui):
6255 t += ' (%s)' % phases.phasenames[pendingphase]
6255 t += ' (%s)' % phases.phasenames[pendingphase]
6256
6256
6257 if cleanworkdir:
6257 if cleanworkdir:
6258 # i18n: column positioning for "hg summary"
6258 # i18n: column positioning for "hg summary"
6259 ui.status(_('commit: %s\n') % t.strip())
6259 ui.status(_('commit: %s\n') % t.strip())
6260 else:
6260 else:
6261 # i18n: column positioning for "hg summary"
6261 # i18n: column positioning for "hg summary"
6262 ui.write(_('commit: %s\n') % t.strip())
6262 ui.write(_('commit: %s\n') % t.strip())
6263
6263
6264 # all ancestors of branch heads - all ancestors of parent = new csets
6264 # all ancestors of branch heads - all ancestors of parent = new csets
6265 new = len(repo.changelog.findmissing([pctx.node() for pctx in parents],
6265 new = len(repo.changelog.findmissing([pctx.node() for pctx in parents],
6266 bheads))
6266 bheads))
6267
6267
6268 if new == 0:
6268 if new == 0:
6269 # i18n: column positioning for "hg summary"
6269 # i18n: column positioning for "hg summary"
6270 ui.status(_('update: (current)\n'))
6270 ui.status(_('update: (current)\n'))
6271 elif pnode not in bheads:
6271 elif pnode not in bheads:
6272 # i18n: column positioning for "hg summary"
6272 # i18n: column positioning for "hg summary"
6273 ui.write(_('update: %d new changesets (update)\n') % new)
6273 ui.write(_('update: %d new changesets (update)\n') % new)
6274 else:
6274 else:
6275 # i18n: column positioning for "hg summary"
6275 # i18n: column positioning for "hg summary"
6276 ui.write(_('update: %d new changesets, %d branch heads (merge)\n') %
6276 ui.write(_('update: %d new changesets, %d branch heads (merge)\n') %
6277 (new, len(bheads)))
6277 (new, len(bheads)))
6278
6278
6279 t = []
6279 t = []
6280 draft = len(repo.revs('draft()'))
6280 draft = len(repo.revs('draft()'))
6281 if draft:
6281 if draft:
6282 t.append(_('%d draft') % draft)
6282 t.append(_('%d draft') % draft)
6283 secret = len(repo.revs('secret()'))
6283 secret = len(repo.revs('secret()'))
6284 if secret:
6284 if secret:
6285 t.append(_('%d secret') % secret)
6285 t.append(_('%d secret') % secret)
6286
6286
6287 if draft or secret:
6287 if draft or secret:
6288 ui.status(_('phases: %s\n') % ', '.join(t))
6288 ui.status(_('phases: %s\n') % ', '.join(t))
6289
6289
6290 cmdutil.summaryhooks(ui, repo)
6290 cmdutil.summaryhooks(ui, repo)
6291
6291
6292 if opts.get('remote'):
6292 if opts.get('remote'):
6293 needsincoming, needsoutgoing = True, True
6293 needsincoming, needsoutgoing = True, True
6294 else:
6294 else:
6295 needsincoming, needsoutgoing = False, False
6295 needsincoming, needsoutgoing = False, False
6296 for i, o in cmdutil.summaryremotehooks(ui, repo, opts, None):
6296 for i, o in cmdutil.summaryremotehooks(ui, repo, opts, None):
6297 if i:
6297 if i:
6298 needsincoming = True
6298 needsincoming = True
6299 if o:
6299 if o:
6300 needsoutgoing = True
6300 needsoutgoing = True
6301 if not needsincoming and not needsoutgoing:
6301 if not needsincoming and not needsoutgoing:
6302 return
6302 return
6303
6303
6304 def getincoming():
6304 def getincoming():
6305 source, branches = hg.parseurl(ui.expandpath('default'))
6305 source, branches = hg.parseurl(ui.expandpath('default'))
6306 sbranch = branches[0]
6306 sbranch = branches[0]
6307 try:
6307 try:
6308 other = hg.peer(repo, {}, source)
6308 other = hg.peer(repo, {}, source)
6309 except error.RepoError:
6309 except error.RepoError:
6310 if opts.get('remote'):
6310 if opts.get('remote'):
6311 raise
6311 raise
6312 return source, sbranch, None, None, None
6312 return source, sbranch, None, None, None
6313 revs, checkout = hg.addbranchrevs(repo, other, branches, None)
6313 revs, checkout = hg.addbranchrevs(repo, other, branches, None)
6314 if revs:
6314 if revs:
6315 revs = [other.lookup(rev) for rev in revs]
6315 revs = [other.lookup(rev) for rev in revs]
6316 ui.debug('comparing with %s\n' % util.hidepassword(source))
6316 ui.debug('comparing with %s\n' % util.hidepassword(source))
6317 repo.ui.pushbuffer()
6317 repo.ui.pushbuffer()
6318 commoninc = discovery.findcommonincoming(repo, other, heads=revs)
6318 commoninc = discovery.findcommonincoming(repo, other, heads=revs)
6319 repo.ui.popbuffer()
6319 repo.ui.popbuffer()
6320 return source, sbranch, other, commoninc, commoninc[1]
6320 return source, sbranch, other, commoninc, commoninc[1]
6321
6321
6322 if needsincoming:
6322 if needsincoming:
6323 source, sbranch, sother, commoninc, incoming = getincoming()
6323 source, sbranch, sother, commoninc, incoming = getincoming()
6324 else:
6324 else:
6325 source = sbranch = sother = commoninc = incoming = None
6325 source = sbranch = sother = commoninc = incoming = None
6326
6326
6327 def getoutgoing():
6327 def getoutgoing():
6328 dest, branches = hg.parseurl(ui.expandpath('default-push', 'default'))
6328 dest, branches = hg.parseurl(ui.expandpath('default-push', 'default'))
6329 dbranch = branches[0]
6329 dbranch = branches[0]
6330 revs, checkout = hg.addbranchrevs(repo, repo, branches, None)
6330 revs, checkout = hg.addbranchrevs(repo, repo, branches, None)
6331 if source != dest:
6331 if source != dest:
6332 try:
6332 try:
6333 dother = hg.peer(repo, {}, dest)
6333 dother = hg.peer(repo, {}, dest)
6334 except error.RepoError:
6334 except error.RepoError:
6335 if opts.get('remote'):
6335 if opts.get('remote'):
6336 raise
6336 raise
6337 return dest, dbranch, None, None
6337 return dest, dbranch, None, None
6338 ui.debug('comparing with %s\n' % util.hidepassword(dest))
6338 ui.debug('comparing with %s\n' % util.hidepassword(dest))
6339 elif sother is None:
6339 elif sother is None:
6340 # there is no explicit destination peer, but source one is invalid
6340 # there is no explicit destination peer, but source one is invalid
6341 return dest, dbranch, None, None
6341 return dest, dbranch, None, None
6342 else:
6342 else:
6343 dother = sother
6343 dother = sother
6344 if (source != dest or (sbranch is not None and sbranch != dbranch)):
6344 if (source != dest or (sbranch is not None and sbranch != dbranch)):
6345 common = None
6345 common = None
6346 else:
6346 else:
6347 common = commoninc
6347 common = commoninc
6348 if revs:
6348 if revs:
6349 revs = [repo.lookup(rev) for rev in revs]
6349 revs = [repo.lookup(rev) for rev in revs]
6350 repo.ui.pushbuffer()
6350 repo.ui.pushbuffer()
6351 outgoing = discovery.findcommonoutgoing(repo, dother, onlyheads=revs,
6351 outgoing = discovery.findcommonoutgoing(repo, dother, onlyheads=revs,
6352 commoninc=common)
6352 commoninc=common)
6353 repo.ui.popbuffer()
6353 repo.ui.popbuffer()
6354 return dest, dbranch, dother, outgoing
6354 return dest, dbranch, dother, outgoing
6355
6355
6356 if needsoutgoing:
6356 if needsoutgoing:
6357 dest, dbranch, dother, outgoing = getoutgoing()
6357 dest, dbranch, dother, outgoing = getoutgoing()
6358 else:
6358 else:
6359 dest = dbranch = dother = outgoing = None
6359 dest = dbranch = dother = outgoing = None
6360
6360
6361 if opts.get('remote'):
6361 if opts.get('remote'):
6362 t = []
6362 t = []
6363 if incoming:
6363 if incoming:
6364 t.append(_('1 or more incoming'))
6364 t.append(_('1 or more incoming'))
6365 o = outgoing.missing
6365 o = outgoing.missing
6366 if o:
6366 if o:
6367 t.append(_('%d outgoing') % len(o))
6367 t.append(_('%d outgoing') % len(o))
6368 other = dother or sother
6368 other = dother or sother
6369 if 'bookmarks' in other.listkeys('namespaces'):
6369 if 'bookmarks' in other.listkeys('namespaces'):
6370 counts = bookmarks.summary(repo, other)
6370 counts = bookmarks.summary(repo, other)
6371 if counts[0] > 0:
6371 if counts[0] > 0:
6372 t.append(_('%d incoming bookmarks') % counts[0])
6372 t.append(_('%d incoming bookmarks') % counts[0])
6373 if counts[1] > 0:
6373 if counts[1] > 0:
6374 t.append(_('%d outgoing bookmarks') % counts[1])
6374 t.append(_('%d outgoing bookmarks') % counts[1])
6375
6375
6376 if t:
6376 if t:
6377 # i18n: column positioning for "hg summary"
6377 # i18n: column positioning for "hg summary"
6378 ui.write(_('remote: %s\n') % (', '.join(t)))
6378 ui.write(_('remote: %s\n') % (', '.join(t)))
6379 else:
6379 else:
6380 # i18n: column positioning for "hg summary"
6380 # i18n: column positioning for "hg summary"
6381 ui.status(_('remote: (synced)\n'))
6381 ui.status(_('remote: (synced)\n'))
6382
6382
6383 cmdutil.summaryremotehooks(ui, repo, opts,
6383 cmdutil.summaryremotehooks(ui, repo, opts,
6384 ((source, sbranch, sother, commoninc),
6384 ((source, sbranch, sother, commoninc),
6385 (dest, dbranch, dother, outgoing)))
6385 (dest, dbranch, dother, outgoing)))
6386
6386
6387 @command('tag',
6387 @command('tag',
6388 [('f', 'force', None, _('force tag')),
6388 [('f', 'force', None, _('force tag')),
6389 ('l', 'local', None, _('make the tag local')),
6389 ('l', 'local', None, _('make the tag local')),
6390 ('r', 'rev', '', _('revision to tag'), _('REV')),
6390 ('r', 'rev', '', _('revision to tag'), _('REV')),
6391 ('', 'remove', None, _('remove a tag')),
6391 ('', 'remove', None, _('remove a tag')),
6392 # -l/--local is already there, commitopts cannot be used
6392 # -l/--local is already there, commitopts cannot be used
6393 ('e', 'edit', None, _('invoke editor on commit messages')),
6393 ('e', 'edit', None, _('invoke editor on commit messages')),
6394 ('m', 'message', '', _('use text as commit message'), _('TEXT')),
6394 ('m', 'message', '', _('use text as commit message'), _('TEXT')),
6395 ] + commitopts2,
6395 ] + commitopts2,
6396 _('[-f] [-l] [-m TEXT] [-d DATE] [-u USER] [-r REV] NAME...'))
6396 _('[-f] [-l] [-m TEXT] [-d DATE] [-u USER] [-r REV] NAME...'))
6397 def tag(ui, repo, name1, *names, **opts):
6397 def tag(ui, repo, name1, *names, **opts):
6398 """add one or more tags for the current or given revision
6398 """add one or more tags for the current or given revision
6399
6399
6400 Name a particular revision using <name>.
6400 Name a particular revision using <name>.
6401
6401
6402 Tags are used to name particular revisions of the repository and are
6402 Tags are used to name particular revisions of the repository and are
6403 very useful to compare different revisions, to go back to significant
6403 very useful to compare different revisions, to go back to significant
6404 earlier versions or to mark branch points as releases, etc. Changing
6404 earlier versions or to mark branch points as releases, etc. Changing
6405 an existing tag is normally disallowed; use -f/--force to override.
6405 an existing tag is normally disallowed; use -f/--force to override.
6406
6406
6407 If no revision is given, the parent of the working directory is
6407 If no revision is given, the parent of the working directory is
6408 used.
6408 used.
6409
6409
6410 To facilitate version control, distribution, and merging of tags,
6410 To facilitate version control, distribution, and merging of tags,
6411 they are stored as a file named ".hgtags" which is managed similarly
6411 they are stored as a file named ".hgtags" which is managed similarly
6412 to other project files and can be hand-edited if necessary. This
6412 to other project files and can be hand-edited if necessary. This
6413 also means that tagging creates a new commit. The file
6413 also means that tagging creates a new commit. The file
6414 ".hg/localtags" is used for local tags (not shared among
6414 ".hg/localtags" is used for local tags (not shared among
6415 repositories).
6415 repositories).
6416
6416
6417 Tag commits are usually made at the head of a branch. If the parent
6417 Tag commits are usually made at the head of a branch. If the parent
6418 of the working directory is not a branch head, :hg:`tag` aborts; use
6418 of the working directory is not a branch head, :hg:`tag` aborts; use
6419 -f/--force to force the tag commit to be based on a non-head
6419 -f/--force to force the tag commit to be based on a non-head
6420 changeset.
6420 changeset.
6421
6421
6422 See :hg:`help dates` for a list of formats valid for -d/--date.
6422 See :hg:`help dates` for a list of formats valid for -d/--date.
6423
6423
6424 Since tag names have priority over branch names during revision
6424 Since tag names have priority over branch names during revision
6425 lookup, using an existing branch name as a tag name is discouraged.
6425 lookup, using an existing branch name as a tag name is discouraged.
6426
6426
6427 Returns 0 on success.
6427 Returns 0 on success.
6428 """
6428 """
6429 wlock = lock = None
6429 wlock = lock = None
6430 try:
6430 try:
6431 wlock = repo.wlock()
6431 wlock = repo.wlock()
6432 lock = repo.lock()
6432 lock = repo.lock()
6433 rev_ = "."
6433 rev_ = "."
6434 names = [t.strip() for t in (name1,) + names]
6434 names = [t.strip() for t in (name1,) + names]
6435 if len(names) != len(set(names)):
6435 if len(names) != len(set(names)):
6436 raise error.Abort(_('tag names must be unique'))
6436 raise error.Abort(_('tag names must be unique'))
6437 for n in names:
6437 for n in names:
6438 scmutil.checknewlabel(repo, n, 'tag')
6438 scmutil.checknewlabel(repo, n, 'tag')
6439 if not n:
6439 if not n:
6440 raise error.Abort(_('tag names cannot consist entirely of '
6440 raise error.Abort(_('tag names cannot consist entirely of '
6441 'whitespace'))
6441 'whitespace'))
6442 if opts.get('rev') and opts.get('remove'):
6442 if opts.get('rev') and opts.get('remove'):
6443 raise error.Abort(_("--rev and --remove are incompatible"))
6443 raise error.Abort(_("--rev and --remove are incompatible"))
6444 if opts.get('rev'):
6444 if opts.get('rev'):
6445 rev_ = opts['rev']
6445 rev_ = opts['rev']
6446 message = opts.get('message')
6446 message = opts.get('message')
6447 if opts.get('remove'):
6447 if opts.get('remove'):
6448 if opts.get('local'):
6448 if opts.get('local'):
6449 expectedtype = 'local'
6449 expectedtype = 'local'
6450 else:
6450 else:
6451 expectedtype = 'global'
6451 expectedtype = 'global'
6452
6452
6453 for n in names:
6453 for n in names:
6454 if not repo.tagtype(n):
6454 if not repo.tagtype(n):
6455 raise error.Abort(_("tag '%s' does not exist") % n)
6455 raise error.Abort(_("tag '%s' does not exist") % n)
6456 if repo.tagtype(n) != expectedtype:
6456 if repo.tagtype(n) != expectedtype:
6457 if expectedtype == 'global':
6457 if expectedtype == 'global':
6458 raise error.Abort(_("tag '%s' is not a global tag") % n)
6458 raise error.Abort(_("tag '%s' is not a global tag") % n)
6459 else:
6459 else:
6460 raise error.Abort(_("tag '%s' is not a local tag") % n)
6460 raise error.Abort(_("tag '%s' is not a local tag") % n)
6461 rev_ = 'null'
6461 rev_ = 'null'
6462 if not message:
6462 if not message:
6463 # we don't translate commit messages
6463 # we don't translate commit messages
6464 message = 'Removed tag %s' % ', '.join(names)
6464 message = 'Removed tag %s' % ', '.join(names)
6465 elif not opts.get('force'):
6465 elif not opts.get('force'):
6466 for n in names:
6466 for n in names:
6467 if n in repo.tags():
6467 if n in repo.tags():
6468 raise error.Abort(_("tag '%s' already exists "
6468 raise error.Abort(_("tag '%s' already exists "
6469 "(use -f to force)") % n)
6469 "(use -f to force)") % n)
6470 if not opts.get('local'):
6470 if not opts.get('local'):
6471 p1, p2 = repo.dirstate.parents()
6471 p1, p2 = repo.dirstate.parents()
6472 if p2 != nullid:
6472 if p2 != nullid:
6473 raise error.Abort(_('uncommitted merge'))
6473 raise error.Abort(_('uncommitted merge'))
6474 bheads = repo.branchheads()
6474 bheads = repo.branchheads()
6475 if not opts.get('force') and bheads and p1 not in bheads:
6475 if not opts.get('force') and bheads and p1 not in bheads:
6476 raise error.Abort(_('not at a branch head (use -f to force)'))
6476 raise error.Abort(_('not at a branch head (use -f to force)'))
6477 r = scmutil.revsingle(repo, rev_).node()
6477 r = scmutil.revsingle(repo, rev_).node()
6478
6478
6479 if not message:
6479 if not message:
6480 # we don't translate commit messages
6480 # we don't translate commit messages
6481 message = ('Added tag %s for changeset %s' %
6481 message = ('Added tag %s for changeset %s' %
6482 (', '.join(names), short(r)))
6482 (', '.join(names), short(r)))
6483
6483
6484 date = opts.get('date')
6484 date = opts.get('date')
6485 if date:
6485 if date:
6486 date = util.parsedate(date)
6486 date = util.parsedate(date)
6487
6487
6488 if opts.get('remove'):
6488 if opts.get('remove'):
6489 editform = 'tag.remove'
6489 editform = 'tag.remove'
6490 else:
6490 else:
6491 editform = 'tag.add'
6491 editform = 'tag.add'
6492 editor = cmdutil.getcommiteditor(editform=editform, **opts)
6492 editor = cmdutil.getcommiteditor(editform=editform, **opts)
6493
6493
6494 # don't allow tagging the null rev
6494 # don't allow tagging the null rev
6495 if (not opts.get('remove') and
6495 if (not opts.get('remove') and
6496 scmutil.revsingle(repo, rev_).rev() == nullrev):
6496 scmutil.revsingle(repo, rev_).rev() == nullrev):
6497 raise error.Abort(_("cannot tag null revision"))
6497 raise error.Abort(_("cannot tag null revision"))
6498
6498
6499 repo.tag(names, r, message, opts.get('local'), opts.get('user'), date,
6499 repo.tag(names, r, message, opts.get('local'), opts.get('user'), date,
6500 editor=editor)
6500 editor=editor)
6501 finally:
6501 finally:
6502 release(lock, wlock)
6502 release(lock, wlock)
6503
6503
6504 @command('tags', formatteropts, '')
6504 @command('tags', formatteropts, '')
6505 def tags(ui, repo, **opts):
6505 def tags(ui, repo, **opts):
6506 """list repository tags
6506 """list repository tags
6507
6507
6508 This lists both regular and local tags. When the -v/--verbose
6508 This lists both regular and local tags. When the -v/--verbose
6509 switch is used, a third column "local" is printed for local tags.
6509 switch is used, a third column "local" is printed for local tags.
6510
6510
6511 Returns 0 on success.
6511 Returns 0 on success.
6512 """
6512 """
6513
6513
6514 fm = ui.formatter('tags', opts)
6514 fm = ui.formatter('tags', opts)
6515 hexfunc = fm.hexfunc
6515 hexfunc = fm.hexfunc
6516 tagtype = ""
6516 tagtype = ""
6517
6517
6518 for t, n in reversed(repo.tagslist()):
6518 for t, n in reversed(repo.tagslist()):
6519 hn = hexfunc(n)
6519 hn = hexfunc(n)
6520 label = 'tags.normal'
6520 label = 'tags.normal'
6521 tagtype = ''
6521 tagtype = ''
6522 if repo.tagtype(t) == 'local':
6522 if repo.tagtype(t) == 'local':
6523 label = 'tags.local'
6523 label = 'tags.local'
6524 tagtype = 'local'
6524 tagtype = 'local'
6525
6525
6526 fm.startitem()
6526 fm.startitem()
6527 fm.write('tag', '%s', t, label=label)
6527 fm.write('tag', '%s', t, label=label)
6528 fmt = " " * (30 - encoding.colwidth(t)) + ' %5d:%s'
6528 fmt = " " * (30 - encoding.colwidth(t)) + ' %5d:%s'
6529 fm.condwrite(not ui.quiet, 'rev node', fmt,
6529 fm.condwrite(not ui.quiet, 'rev node', fmt,
6530 repo.changelog.rev(n), hn, label=label)
6530 repo.changelog.rev(n), hn, label=label)
6531 fm.condwrite(ui.verbose and tagtype, 'type', ' %s',
6531 fm.condwrite(ui.verbose and tagtype, 'type', ' %s',
6532 tagtype, label=label)
6532 tagtype, label=label)
6533 fm.plain('\n')
6533 fm.plain('\n')
6534 fm.end()
6534 fm.end()
6535
6535
6536 @command('tip',
6536 @command('tip',
6537 [('p', 'patch', None, _('show patch')),
6537 [('p', 'patch', None, _('show patch')),
6538 ('g', 'git', None, _('use git extended diff format')),
6538 ('g', 'git', None, _('use git extended diff format')),
6539 ] + templateopts,
6539 ] + templateopts,
6540 _('[-p] [-g]'))
6540 _('[-p] [-g]'))
6541 def tip(ui, repo, **opts):
6541 def tip(ui, repo, **opts):
6542 """show the tip revision (DEPRECATED)
6542 """show the tip revision (DEPRECATED)
6543
6543
6544 The tip revision (usually just called the tip) is the changeset
6544 The tip revision (usually just called the tip) is the changeset
6545 most recently added to the repository (and therefore the most
6545 most recently added to the repository (and therefore the most
6546 recently changed head).
6546 recently changed head).
6547
6547
6548 If you have just made a commit, that commit will be the tip. If
6548 If you have just made a commit, that commit will be the tip. If
6549 you have just pulled changes from another repository, the tip of
6549 you have just pulled changes from another repository, the tip of
6550 that repository becomes the current tip. The "tip" tag is special
6550 that repository becomes the current tip. The "tip" tag is special
6551 and cannot be renamed or assigned to a different changeset.
6551 and cannot be renamed or assigned to a different changeset.
6552
6552
6553 This command is deprecated, please use :hg:`heads` instead.
6553 This command is deprecated, please use :hg:`heads` instead.
6554
6554
6555 Returns 0 on success.
6555 Returns 0 on success.
6556 """
6556 """
6557 displayer = cmdutil.show_changeset(ui, repo, opts)
6557 displayer = cmdutil.show_changeset(ui, repo, opts)
6558 displayer.show(repo['tip'])
6558 displayer.show(repo['tip'])
6559 displayer.close()
6559 displayer.close()
6560
6560
6561 @command('unbundle',
6561 @command('unbundle',
6562 [('u', 'update', None,
6562 [('u', 'update', None,
6563 _('update to new branch head if changesets were unbundled'))],
6563 _('update to new branch head if changesets were unbundled'))],
6564 _('[-u] FILE...'))
6564 _('[-u] FILE...'))
6565 def unbundle(ui, repo, fname1, *fnames, **opts):
6565 def unbundle(ui, repo, fname1, *fnames, **opts):
6566 """apply one or more changegroup files
6566 """apply one or more changegroup files
6567
6567
6568 Apply one or more compressed changegroup files generated by the
6568 Apply one or more compressed changegroup files generated by the
6569 bundle command.
6569 bundle command.
6570
6570
6571 Returns 0 on success, 1 if an update has unresolved files.
6571 Returns 0 on success, 1 if an update has unresolved files.
6572 """
6572 """
6573 fnames = (fname1,) + fnames
6573 fnames = (fname1,) + fnames
6574
6574
6575 lock = repo.lock()
6575 lock = repo.lock()
6576 try:
6576 try:
6577 for fname in fnames:
6577 for fname in fnames:
6578 f = hg.openpath(ui, fname)
6578 f = hg.openpath(ui, fname)
6579 gen = exchange.readbundle(ui, f, fname)
6579 gen = exchange.readbundle(ui, f, fname)
6580 if isinstance(gen, bundle2.unbundle20):
6580 if isinstance(gen, bundle2.unbundle20):
6581 tr = repo.transaction('unbundle')
6581 tr = repo.transaction('unbundle')
6582 try:
6582 try:
6583 op = bundle2.applybundle(repo, gen, tr, source='unbundle',
6583 op = bundle2.applybundle(repo, gen, tr, source='unbundle',
6584 url='bundle:' + fname)
6584 url='bundle:' + fname)
6585 tr.close()
6585 tr.close()
6586 except error.BundleUnknownFeatureError as exc:
6586 except error.BundleUnknownFeatureError as exc:
6587 raise error.Abort(_('%s: unknown bundle feature, %s')
6587 raise error.Abort(_('%s: unknown bundle feature, %s')
6588 % (fname, exc),
6588 % (fname, exc),
6589 hint=_("see https://mercurial-scm.org/"
6589 hint=_("see https://mercurial-scm.org/"
6590 "wiki/BundleFeature for more "
6590 "wiki/BundleFeature for more "
6591 "information"))
6591 "information"))
6592 finally:
6592 finally:
6593 if tr:
6593 if tr:
6594 tr.release()
6594 tr.release()
6595 changes = [r.get('return', 0)
6595 changes = [r.get('return', 0)
6596 for r in op.records['changegroup']]
6596 for r in op.records['changegroup']]
6597 modheads = changegroup.combineresults(changes)
6597 modheads = changegroup.combineresults(changes)
6598 elif isinstance(gen, streamclone.streamcloneapplier):
6598 elif isinstance(gen, streamclone.streamcloneapplier):
6599 raise error.Abort(
6599 raise error.Abort(
6600 _('packed bundles cannot be applied with '
6600 _('packed bundles cannot be applied with '
6601 '"hg unbundle"'),
6601 '"hg unbundle"'),
6602 hint=_('use "hg debugapplystreamclonebundle"'))
6602 hint=_('use "hg debugapplystreamclonebundle"'))
6603 else:
6603 else:
6604 modheads = gen.apply(repo, 'unbundle', 'bundle:' + fname)
6604 modheads = gen.apply(repo, 'unbundle', 'bundle:' + fname)
6605 finally:
6605 finally:
6606 lock.release()
6606 lock.release()
6607
6607
6608 return postincoming(ui, repo, modheads, opts.get('update'), None)
6608 return postincoming(ui, repo, modheads, opts.get('update'), None)
6609
6609
6610 @command('^update|up|checkout|co',
6610 @command('^update|up|checkout|co',
6611 [('C', 'clean', None, _('discard uncommitted changes (no backup)')),
6611 [('C', 'clean', None, _('discard uncommitted changes (no backup)')),
6612 ('c', 'check', None,
6612 ('c', 'check', None,
6613 _('update across branches if no uncommitted changes')),
6613 _('update across branches if no uncommitted changes')),
6614 ('d', 'date', '', _('tipmost revision matching date'), _('DATE')),
6614 ('d', 'date', '', _('tipmost revision matching date'), _('DATE')),
6615 ('r', 'rev', '', _('revision'), _('REV'))
6615 ('r', 'rev', '', _('revision'), _('REV'))
6616 ] + mergetoolopts,
6616 ] + mergetoolopts,
6617 _('[-c] [-C] [-d DATE] [[-r] REV]'))
6617 _('[-c] [-C] [-d DATE] [[-r] REV]'))
6618 def update(ui, repo, node=None, rev=None, clean=False, date=None, check=False,
6618 def update(ui, repo, node=None, rev=None, clean=False, date=None, check=False,
6619 tool=None):
6619 tool=None):
6620 """update working directory (or switch revisions)
6620 """update working directory (or switch revisions)
6621
6621
6622 Update the repository's working directory to the specified
6622 Update the repository's working directory to the specified
6623 changeset. If no changeset is specified, update to the tip of the
6623 changeset. If no changeset is specified, update to the tip of the
6624 current named branch and move the active bookmark (see :hg:`help
6624 current named branch and move the active bookmark (see :hg:`help
6625 bookmarks`).
6625 bookmarks`).
6626
6626
6627 Update sets the working directory's parent revision to the specified
6627 Update sets the working directory's parent revision to the specified
6628 changeset (see :hg:`help parents`).
6628 changeset (see :hg:`help parents`).
6629
6629
6630 If the changeset is not a descendant or ancestor of the working
6630 If the changeset is not a descendant or ancestor of the working
6631 directory's parent, the update is aborted. With the -c/--check
6631 directory's parent, the update is aborted. With the -c/--check
6632 option, the working directory is checked for uncommitted changes; if
6632 option, the working directory is checked for uncommitted changes; if
6633 none are found, the working directory is updated to the specified
6633 none are found, the working directory is updated to the specified
6634 changeset.
6634 changeset.
6635
6635
6636 .. container:: verbose
6636 .. container:: verbose
6637
6637
6638 The following rules apply when the working directory contains
6638 The following rules apply when the working directory contains
6639 uncommitted changes:
6639 uncommitted changes:
6640
6640
6641 1. If neither -c/--check nor -C/--clean is specified, and if
6641 1. If neither -c/--check nor -C/--clean is specified, and if
6642 the requested changeset is an ancestor or descendant of
6642 the requested changeset is an ancestor or descendant of
6643 the working directory's parent, the uncommitted changes
6643 the working directory's parent, the uncommitted changes
6644 are merged into the requested changeset and the merged
6644 are merged into the requested changeset and the merged
6645 result is left uncommitted. If the requested changeset is
6645 result is left uncommitted. If the requested changeset is
6646 not an ancestor or descendant (that is, it is on another
6646 not an ancestor or descendant (that is, it is on another
6647 branch), the update is aborted and the uncommitted changes
6647 branch), the update is aborted and the uncommitted changes
6648 are preserved.
6648 are preserved.
6649
6649
6650 2. With the -c/--check option, the update is aborted and the
6650 2. With the -c/--check option, the update is aborted and the
6651 uncommitted changes are preserved.
6651 uncommitted changes are preserved.
6652
6652
6653 3. With the -C/--clean option, uncommitted changes are discarded and
6653 3. With the -C/--clean option, uncommitted changes are discarded and
6654 the working directory is updated to the requested changeset.
6654 the working directory is updated to the requested changeset.
6655
6655
6656 To cancel an uncommitted merge (and lose your changes), use
6656 To cancel an uncommitted merge (and lose your changes), use
6657 :hg:`update --clean .`.
6657 :hg:`update --clean .`.
6658
6658
6659 Use null as the changeset to remove the working directory (like
6659 Use null as the changeset to remove the working directory (like
6660 :hg:`clone -U`).
6660 :hg:`clone -U`).
6661
6661
6662 If you want to revert just one file to an older revision, use
6662 If you want to revert just one file to an older revision, use
6663 :hg:`revert [-r REV] NAME`.
6663 :hg:`revert [-r REV] NAME`.
6664
6664
6665 See :hg:`help dates` for a list of formats valid for -d/--date.
6665 See :hg:`help dates` for a list of formats valid for -d/--date.
6666
6666
6667 Returns 0 on success, 1 if there are unresolved files.
6667 Returns 0 on success, 1 if there are unresolved files.
6668 """
6668 """
6669 movemarkfrom = None
6669 movemarkfrom = None
6670 if rev and node:
6670 if rev and node:
6671 raise error.Abort(_("please specify just one revision"))
6671 raise error.Abort(_("please specify just one revision"))
6672
6672
6673 if rev is None or rev == '':
6673 if rev is None or rev == '':
6674 rev = node
6674 rev = node
6675
6675
6676 wlock = repo.wlock()
6676 wlock = repo.wlock()
6677 try:
6677 try:
6678 cmdutil.clearunfinished(repo)
6678 cmdutil.clearunfinished(repo)
6679
6679
6680 if date:
6680 if date:
6681 if rev is not None:
6681 if rev is not None:
6682 raise error.Abort(_("you can't specify a revision and a date"))
6682 raise error.Abort(_("you can't specify a revision and a date"))
6683 rev = cmdutil.finddate(ui, repo, date)
6683 rev = cmdutil.finddate(ui, repo, date)
6684
6684
6685 # if we defined a bookmark, we have to remember the original name
6685 # if we defined a bookmark, we have to remember the original name
6686 brev = rev
6686 brev = rev
6687 rev = scmutil.revsingle(repo, rev, rev).rev()
6687 rev = scmutil.revsingle(repo, rev, rev).rev()
6688
6688
6689 if check and clean:
6689 if check and clean:
6690 raise error.Abort(_("cannot specify both -c/--check and -C/--clean")
6690 raise error.Abort(_("cannot specify both -c/--check and -C/--clean")
6691 )
6691 )
6692
6692
6693 if check:
6693 if check:
6694 cmdutil.bailifchanged(repo, merge=False)
6694 cmdutil.bailifchanged(repo, merge=False)
6695 if rev is None:
6695 if rev is None:
6696 updata = destutil.destupdate(repo, clean=clean, check=check)
6696 updata = destutil.destupdate(repo, clean=clean, check=check)
6697 rev, movemarkfrom, brev = updata
6697 rev, movemarkfrom, brev = updata
6698
6698
6699 repo.ui.setconfig('ui', 'forcemerge', tool, 'update')
6699 repo.ui.setconfig('ui', 'forcemerge', tool, 'update')
6700
6700
6701 if clean:
6701 if clean:
6702 ret = hg.clean(repo, rev)
6702 ret = hg.clean(repo, rev)
6703 else:
6703 else:
6704 ret = hg.update(repo, rev)
6704 ret = hg.update(repo, rev)
6705
6705
6706 if not ret and movemarkfrom:
6706 if not ret and movemarkfrom:
6707 if movemarkfrom == repo['.'].node():
6707 if movemarkfrom == repo['.'].node():
6708 pass # no-op update
6708 pass # no-op update
6709 elif bookmarks.update(repo, [movemarkfrom], repo['.'].node()):
6709 elif bookmarks.update(repo, [movemarkfrom], repo['.'].node()):
6710 ui.status(_("updating bookmark %s\n") % repo._activebookmark)
6710 ui.status(_("updating bookmark %s\n") % repo._activebookmark)
6711 else:
6711 else:
6712 # this can happen with a non-linear update
6712 # this can happen with a non-linear update
6713 ui.status(_("(leaving bookmark %s)\n") %
6713 ui.status(_("(leaving bookmark %s)\n") %
6714 repo._activebookmark)
6714 repo._activebookmark)
6715 bookmarks.deactivate(repo)
6715 bookmarks.deactivate(repo)
6716 elif brev in repo._bookmarks:
6716 elif brev in repo._bookmarks:
6717 bookmarks.activate(repo, brev)
6717 bookmarks.activate(repo, brev)
6718 ui.status(_("(activating bookmark %s)\n") % brev)
6718 ui.status(_("(activating bookmark %s)\n") % brev)
6719 elif brev:
6719 elif brev:
6720 if repo._activebookmark:
6720 if repo._activebookmark:
6721 ui.status(_("(leaving bookmark %s)\n") %
6721 ui.status(_("(leaving bookmark %s)\n") %
6722 repo._activebookmark)
6722 repo._activebookmark)
6723 bookmarks.deactivate(repo)
6723 bookmarks.deactivate(repo)
6724 finally:
6724 finally:
6725 wlock.release()
6725 wlock.release()
6726
6726
6727 return ret
6727 return ret
6728
6728
6729 @command('verify', [])
6729 @command('verify', [])
6730 def verify(ui, repo):
6730 def verify(ui, repo):
6731 """verify the integrity of the repository
6731 """verify the integrity of the repository
6732
6732
6733 Verify the integrity of the current repository.
6733 Verify the integrity of the current repository.
6734
6734
6735 This will perform an extensive check of the repository's
6735 This will perform an extensive check of the repository's
6736 integrity, validating the hashes and checksums of each entry in
6736 integrity, validating the hashes and checksums of each entry in
6737 the changelog, manifest, and tracked files, as well as the
6737 the changelog, manifest, and tracked files, as well as the
6738 integrity of their crosslinks and indices.
6738 integrity of their crosslinks and indices.
6739
6739
6740 Please see https://mercurial-scm.org/wiki/RepositoryCorruption
6740 Please see https://mercurial-scm.org/wiki/RepositoryCorruption
6741 for more information about recovery from corruption of the
6741 for more information about recovery from corruption of the
6742 repository.
6742 repository.
6743
6743
6744 Returns 0 on success, 1 if errors are encountered.
6744 Returns 0 on success, 1 if errors are encountered.
6745 """
6745 """
6746 return hg.verify(repo)
6746 return hg.verify(repo)
6747
6747
6748 @command('version', [], norepo=True)
6748 @command('version', [], norepo=True)
6749 def version_(ui):
6749 def version_(ui):
6750 """output version and copyright information"""
6750 """output version and copyright information"""
6751 ui.write(_("Mercurial Distributed SCM (version %s)\n")
6751 ui.write(_("Mercurial Distributed SCM (version %s)\n")
6752 % util.version())
6752 % util.version())
6753 ui.status(_(
6753 ui.status(_(
6754 "(see https://mercurial-scm.org for more information)\n"
6754 "(see https://mercurial-scm.org for more information)\n"
6755 "\nCopyright (C) 2005-2015 Matt Mackall and others\n"
6755 "\nCopyright (C) 2005-2015 Matt Mackall and others\n"
6756 "This is free software; see the source for copying conditions. "
6756 "This is free software; see the source for copying conditions. "
6757 "There is NO\nwarranty; "
6757 "There is NO\nwarranty; "
6758 "not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.\n"
6758 "not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.\n"
6759 ))
6759 ))
6760
6760
6761 ui.note(_("\nEnabled extensions:\n\n"))
6761 ui.note(_("\nEnabled extensions:\n\n"))
6762 if ui.verbose:
6762 if ui.verbose:
6763 # format names and versions into columns
6763 # format names and versions into columns
6764 names = []
6764 names = []
6765 vers = []
6765 vers = []
6766 for name, module in extensions.extensions():
6766 for name, module in extensions.extensions():
6767 names.append(name)
6767 names.append(name)
6768 vers.append(extensions.moduleversion(module))
6768 vers.append(extensions.moduleversion(module))
6769 if names:
6769 if names:
6770 maxnamelen = max(len(n) for n in names)
6770 maxnamelen = max(len(n) for n in names)
6771 for i, name in enumerate(names):
6771 for i, name in enumerate(names):
6772 ui.write(" %-*s %s\n" % (maxnamelen, name, vers[i]))
6772 ui.write(" %-*s %s\n" % (maxnamelen, name, vers[i]))
@@ -1,741 +1,741
1 $ hg init basic
1 $ hg init basic
2 $ cd basic
2 $ cd basic
3
3
4 should complain
4 should complain
5
5
6 $ hg backout
6 $ hg backout
7 abort: please specify a revision to backout
7 abort: please specify a revision to backout
8 [255]
8 [255]
9 $ hg backout -r 0 0
9 $ hg backout -r 0 0
10 abort: please specify just one revision
10 abort: please specify just one revision
11 [255]
11 [255]
12
12
13 basic operation
13 basic operation
14 (this also tests that editor is invoked if the commit message is not
14 (this also tests that editor is invoked if the commit message is not
15 specified explicitly)
15 specified explicitly)
16
16
17 $ echo a > a
17 $ echo a > a
18 $ hg commit -d '0 0' -A -m a
18 $ hg commit -d '0 0' -A -m a
19 adding a
19 adding a
20 $ echo b >> a
20 $ echo b >> a
21 $ hg commit -d '1 0' -m b
21 $ hg commit -d '1 0' -m b
22
22
23 $ hg status --rev tip --rev "tip^1"
23 $ hg status --rev tip --rev "tip^1"
24 M a
24 M a
25 $ HGEDITOR=cat hg backout -d '2 0' tip --tool=true
25 $ HGEDITOR=cat hg backout -d '2 0' tip --tool=true
26 reverting a
26 reverting a
27 Backed out changeset a820f4f40a57
27 Backed out changeset a820f4f40a57
28
28
29
29
30 HG: Enter commit message. Lines beginning with 'HG:' are removed.
30 HG: Enter commit message. Lines beginning with 'HG:' are removed.
31 HG: Leave message empty to abort commit.
31 HG: Leave message empty to abort commit.
32 HG: --
32 HG: --
33 HG: user: test
33 HG: user: test
34 HG: branch 'default'
34 HG: branch 'default'
35 HG: changed a
35 HG: changed a
36 changeset 2:2929462c3dff backs out changeset 1:a820f4f40a57
36 changeset 2:2929462c3dff backs out changeset 1:a820f4f40a57
37 $ cat a
37 $ cat a
38 a
38 a
39 $ hg summary
39 $ hg summary
40 parent: 2:2929462c3dff tip
40 parent: 2:2929462c3dff tip
41 Backed out changeset a820f4f40a57
41 Backed out changeset a820f4f40a57
42 branch: default
42 branch: default
43 commit: (clean)
43 commit: (clean)
44 update: (current)
44 update: (current)
45 phases: 3 draft
45 phases: 3 draft
46
46
47 commit option
47 commit option
48
48
49 $ cd ..
49 $ cd ..
50 $ hg init commit
50 $ hg init commit
51 $ cd commit
51 $ cd commit
52
52
53 $ echo tomatoes > a
53 $ echo tomatoes > a
54 $ hg add a
54 $ hg add a
55 $ hg commit -d '0 0' -m tomatoes
55 $ hg commit -d '0 0' -m tomatoes
56
56
57 $ echo chair > b
57 $ echo chair > b
58 $ hg add b
58 $ hg add b
59 $ hg commit -d '1 0' -m chair
59 $ hg commit -d '1 0' -m chair
60
60
61 $ echo grapes >> a
61 $ echo grapes >> a
62 $ hg commit -d '2 0' -m grapes
62 $ hg commit -d '2 0' -m grapes
63
63
64 $ hg backout --commit -d '4 0' 1 --tool=:fail
64 $ hg backout --commit -d '4 0' 1 --tool=:fail
65 0 files updated, 0 files merged, 1 files removed, 0 files unresolved
65 0 files updated, 0 files merged, 1 files removed, 0 files unresolved
66 changeset 3:1c2161e97c0a backs out changeset 1:22cb4f70d813
66 changeset 3:1c2161e97c0a backs out changeset 1:22cb4f70d813
67 $ hg summary
67 $ hg summary
68 parent: 3:1c2161e97c0a tip
68 parent: 3:1c2161e97c0a tip
69 Backed out changeset 22cb4f70d813
69 Backed out changeset 22cb4f70d813
70 branch: default
70 branch: default
71 commit: (clean)
71 commit: (clean)
72 update: (current)
72 update: (current)
73 phases: 4 draft
73 phases: 4 draft
74
74
75 $ echo ypples > a
75 $ echo ypples > a
76 $ hg commit -d '5 0' -m ypples
76 $ hg commit -d '5 0' -m ypples
77
77
78 $ hg backout --commit -d '6 0' 2 --tool=:fail
78 $ hg backout --commit -d '6 0' 2 --tool=:fail
79 0 files updated, 0 files merged, 0 files removed, 1 files unresolved
79 0 files updated, 0 files merged, 0 files removed, 1 files unresolved
80 use 'hg resolve' to retry unresolved file merges
80 use 'hg resolve' to retry unresolved file merges
81 [1]
81 [1]
82 $ hg summary
82 $ hg summary
83 parent: 4:ed99997b793d tip
83 parent: 4:ed99997b793d tip
84 ypples
84 ypples
85 branch: default
85 branch: default
86 commit: 1 unresolved (clean)
86 commit: 1 unresolved (clean)
87 update: (current)
87 update: (current)
88 phases: 5 draft
88 phases: 5 draft
89
89
90 file that was removed is recreated
90 file that was removed is recreated
91 (this also tests that editor is not invoked if the commit message is
91 (this also tests that editor is not invoked if the commit message is
92 specified explicitly)
92 specified explicitly)
93
93
94 $ cd ..
94 $ cd ..
95 $ hg init remove
95 $ hg init remove
96 $ cd remove
96 $ cd remove
97
97
98 $ echo content > a
98 $ echo content > a
99 $ hg commit -d '0 0' -A -m a
99 $ hg commit -d '0 0' -A -m a
100 adding a
100 adding a
101
101
102 $ hg rm a
102 $ hg rm a
103 $ hg commit -d '1 0' -m b
103 $ hg commit -d '1 0' -m b
104
104
105 $ HGEDITOR=cat hg backout -d '2 0' tip --tool=true -m "Backed out changeset 76862dcce372"
105 $ HGEDITOR=cat hg backout -d '2 0' tip --tool=true -m "Backed out changeset 76862dcce372"
106 adding a
106 adding a
107 changeset 2:de31bdc76c0d backs out changeset 1:76862dcce372
107 changeset 2:de31bdc76c0d backs out changeset 1:76862dcce372
108 $ cat a
108 $ cat a
109 content
109 content
110 $ hg summary
110 $ hg summary
111 parent: 2:de31bdc76c0d tip
111 parent: 2:de31bdc76c0d tip
112 Backed out changeset 76862dcce372
112 Backed out changeset 76862dcce372
113 branch: default
113 branch: default
114 commit: (clean)
114 commit: (clean)
115 update: (current)
115 update: (current)
116 phases: 3 draft
116 phases: 3 draft
117
117
118 backout of backout is as if nothing happened
118 backout of backout is as if nothing happened
119
119
120 $ hg backout -d '3 0' --merge tip --tool=true
120 $ hg backout -d '3 0' --merge tip --tool=true
121 removing a
121 removing a
122 changeset 3:7f6d0f120113 backs out changeset 2:de31bdc76c0d
122 changeset 3:7f6d0f120113 backs out changeset 2:de31bdc76c0d
123 $ test -f a
123 $ test -f a
124 [1]
124 [1]
125 $ hg summary
125 $ hg summary
126 parent: 3:7f6d0f120113 tip
126 parent: 3:7f6d0f120113 tip
127 Backed out changeset de31bdc76c0d
127 Backed out changeset de31bdc76c0d
128 branch: default
128 branch: default
129 commit: (clean)
129 commit: (clean)
130 update: (current)
130 update: (current)
131 phases: 4 draft
131 phases: 4 draft
132
132
133 Test that 'hg rollback' restores dirstate just before opening
133 Test that 'hg rollback' restores dirstate just before opening
134 transaction: in-memory dirstate changes should be written into
134 transaction: in-memory dirstate changes should be written into
135 '.hg/journal.dirstate' as expected.
135 '.hg/journal.dirstate' as expected.
136
136
137 $ echo 'removed soon' > b
137 $ echo 'removed soon' > b
138 $ hg commit -A -d '4 0' -m 'prepare for subsequent removing'
138 $ hg commit -A -d '4 0' -m 'prepare for subsequent removing'
139 adding b
139 adding b
140 $ echo 'newly added' > c
140 $ echo 'newly added' > c
141 $ hg add c
141 $ hg add c
142 $ hg remove b
142 $ hg remove b
143 $ hg commit -d '5 0' -m 'prepare for subsequent backout'
143 $ hg commit -d '5 0' -m 'prepare for subsequent backout'
144 $ touch -t 200001010000 c
144 $ touch -t 200001010000 c
145 $ hg status -A
145 $ hg status -A
146 C c
146 C c
147 $ hg debugstate --nodates
147 $ hg debugstate --nodates
148 n 644 12 set c
148 n 644 12 set c
149 $ hg backout -d '6 0' -m 'to be rollback-ed soon' -r .
149 $ hg backout -d '6 0' -m 'to be rollback-ed soon' -r .
150 adding b
150 adding b
151 removing c
151 removing c
152 changeset 6:4bfec048029d backs out changeset 5:fac0b729a654
152 changeset 6:4bfec048029d backs out changeset 5:fac0b729a654
153 $ hg rollback -q
153 $ hg rollback -q
154 $ hg status -A
154 $ hg status -A
155 A b
155 A b
156 R c
156 R c
157 $ hg debugstate --nodates
157 $ hg debugstate --nodates
158 a 0 -1 unset b
158 a 0 -1 unset b
159 r 0 0 set c
159 r 0 0 set c
160
160
161 across branch
161 across branch
162
162
163 $ cd ..
163 $ cd ..
164 $ hg init branch
164 $ hg init branch
165 $ cd branch
165 $ cd branch
166 $ echo a > a
166 $ echo a > a
167 $ hg ci -Am0
167 $ hg ci -Am0
168 adding a
168 adding a
169 $ echo b > b
169 $ echo b > b
170 $ hg ci -Am1
170 $ hg ci -Am1
171 adding b
171 adding b
172 $ hg co -C 0
172 $ hg co -C 0
173 0 files updated, 0 files merged, 1 files removed, 0 files unresolved
173 0 files updated, 0 files merged, 1 files removed, 0 files unresolved
174 $ hg summary
174 $ hg summary
175 parent: 0:f7b1eb17ad24
175 parent: 0:f7b1eb17ad24
176 0
176 0
177 branch: default
177 branch: default
178 commit: (clean)
178 commit: (clean)
179 update: 1 new changesets (update)
179 update: 1 new changesets (update)
180 phases: 2 draft
180 phases: 2 draft
181
181
182 should fail
182 should fail
183
183
184 $ hg backout 1
184 $ hg backout 1
185 abort: cannot backout change that is not an ancestor
185 abort: cannot backout change that is not an ancestor
186 [255]
186 [255]
187 $ echo c > c
187 $ echo c > c
188 $ hg ci -Am2
188 $ hg ci -Am2
189 adding c
189 adding c
190 created new head
190 created new head
191 $ hg summary
191 $ hg summary
192 parent: 2:db815d6d32e6 tip
192 parent: 2:db815d6d32e6 tip
193 2
193 2
194 branch: default
194 branch: default
195 commit: (clean)
195 commit: (clean)
196 update: 1 new changesets, 2 branch heads (merge)
196 update: 1 new changesets, 2 branch heads (merge)
197 phases: 3 draft
197 phases: 3 draft
198
198
199 should fail
199 should fail
200
200
201 $ hg backout 1
201 $ hg backout 1
202 abort: cannot backout change that is not an ancestor
202 abort: cannot backout change that is not an ancestor
203 [255]
203 [255]
204 $ hg summary
204 $ hg summary
205 parent: 2:db815d6d32e6 tip
205 parent: 2:db815d6d32e6 tip
206 2
206 2
207 branch: default
207 branch: default
208 commit: (clean)
208 commit: (clean)
209 update: 1 new changesets, 2 branch heads (merge)
209 update: 1 new changesets, 2 branch heads (merge)
210 phases: 3 draft
210 phases: 3 draft
211
211
212 backout with merge
212 backout with merge
213
213
214 $ cd ..
214 $ cd ..
215 $ hg init merge
215 $ hg init merge
216 $ cd merge
216 $ cd merge
217
217
218 $ echo line 1 > a
218 $ echo line 1 > a
219 $ echo line 2 >> a
219 $ echo line 2 >> a
220 $ hg commit -d '0 0' -A -m a
220 $ hg commit -d '0 0' -A -m a
221 adding a
221 adding a
222 $ hg summary
222 $ hg summary
223 parent: 0:59395513a13a tip
223 parent: 0:59395513a13a tip
224 a
224 a
225 branch: default
225 branch: default
226 commit: (clean)
226 commit: (clean)
227 update: (current)
227 update: (current)
228 phases: 1 draft
228 phases: 1 draft
229
229
230 remove line 1
230 remove line 1
231
231
232 $ echo line 2 > a
232 $ echo line 2 > a
233 $ hg commit -d '1 0' -m b
233 $ hg commit -d '1 0' -m b
234
234
235 $ echo line 3 >> a
235 $ echo line 3 >> a
236 $ hg commit -d '2 0' -m c
236 $ hg commit -d '2 0' -m c
237
237
238 $ hg backout --merge -d '3 0' 1 --tool=true
238 $ hg backout --merge -d '3 0' 1 --tool=true
239 reverting a
239 reverting a
240 created new head
240 created new head
241 changeset 3:26b8ccb9ad91 backs out changeset 1:5a50a024c182
241 changeset 3:26b8ccb9ad91 backs out changeset 1:5a50a024c182
242 merging with changeset 3:26b8ccb9ad91
242 merging with changeset 3:26b8ccb9ad91
243 merging a
243 merging a
244 0 files updated, 1 files merged, 0 files removed, 0 files unresolved
244 0 files updated, 1 files merged, 0 files removed, 0 files unresolved
245 (branch merge, don't forget to commit)
245 (branch merge, don't forget to commit)
246 $ hg commit -d '4 0' -m d
246 $ hg commit -d '4 0' -m d
247 $ hg summary
247 $ hg summary
248 parent: 4:c7df5e0b9c09 tip
248 parent: 4:c7df5e0b9c09 tip
249 d
249 d
250 branch: default
250 branch: default
251 commit: (clean)
251 commit: (clean)
252 update: (current)
252 update: (current)
253 phases: 5 draft
253 phases: 5 draft
254
254
255 check line 1 is back
255 check line 1 is back
256
256
257 $ cat a
257 $ cat a
258 line 1
258 line 1
259 line 2
259 line 2
260 line 3
260 line 3
261
261
262 Test visibility of in-memory dirstate changes outside transaction to
262 Test visibility of in-memory dirstate changes outside transaction to
263 external hook process
263 external hook process
264
264
265 $ cat > $TESTTMP/checkvisibility.sh <<EOF
265 $ cat > $TESTTMP/checkvisibility.sh <<EOF
266 > echo "==== \$1:"
266 > echo "==== \$1:"
267 > hg parents --template "{rev}:{node|short}\n"
267 > hg parents --template "{rev}:{node|short}\n"
268 > echo "===="
268 > echo "===="
269 > EOF
269 > EOF
270
270
271 "hg backout --merge REV1" at REV2 below implies steps below:
271 "hg backout --merge REV1" at REV2 below implies steps below:
272
272
273 (1) update to REV1 (REV2 => REV1)
273 (1) update to REV1 (REV2 => REV1)
274 (2) revert by REV1^1
274 (2) revert by REV1^1
275 (3) commit backnig out revision (REV3)
275 (3) commit backnig out revision (REV3)
276 (4) update to REV2 (REV3 => REV2)
276 (4) update to REV2 (REV3 => REV2)
277 (5) merge with REV3 (REV2 => REV2, REV3)
277 (5) merge with REV3 (REV2 => REV2, REV3)
278
278
279 == test visibility to external preupdate hook
279 == test visibility to external preupdate hook
280
280
281 $ hg update -q -C 2
281 $ hg update -q -C 2
282 $ hg --config extensions.strip= strip 3
282 $ hg --config extensions.strip= strip 3
283 saved backup bundle to * (glob)
283 saved backup bundle to * (glob)
284
284
285 $ cat >> .hg/hgrc <<EOF
285 $ cat >> .hg/hgrc <<EOF
286 > [hooks]
286 > [hooks]
287 > preupdate.visibility = sh $TESTTMP/checkvisibility.sh preupdate
287 > preupdate.visibility = sh $TESTTMP/checkvisibility.sh preupdate
288 > EOF
288 > EOF
289
289
290 ("-m" is needed to avoid writing dirstte changes out at other than
290 ("-m" is needed to avoid writing dirstte changes out at other than
291 invocation of the hook to be examined)
291 invocation of the hook to be examined)
292
292
293 $ hg backout --merge -d '3 0' 1 --tool=true -m 'fixed comment'
293 $ hg backout --merge -d '3 0' 1 --tool=true -m 'fixed comment'
294 ==== preupdate:
294 ==== preupdate:
295 2:6ea3f2a197a2
295 2:6ea3f2a197a2
296 ====
296 ====
297 reverting a
297 reverting a
298 created new head
298 created new head
299 changeset 3:d92a3f57f067 backs out changeset 1:5a50a024c182
299 changeset 3:d92a3f57f067 backs out changeset 1:5a50a024c182
300 ==== preupdate:
300 ==== preupdate:
301 3:d92a3f57f067
301 3:d92a3f57f067
302 ====
302 ====
303 merging with changeset 3:d92a3f57f067
303 merging with changeset 3:d92a3f57f067
304 ==== preupdate:
304 ==== preupdate:
305 2:6ea3f2a197a2
305 2:6ea3f2a197a2
306 ====
306 ====
307 merging a
307 merging a
308 0 files updated, 1 files merged, 0 files removed, 0 files unresolved
308 0 files updated, 1 files merged, 0 files removed, 0 files unresolved
309 (branch merge, don't forget to commit)
309 (branch merge, don't forget to commit)
310
310
311 $ cat >> .hg/hgrc <<EOF
311 $ cat >> .hg/hgrc <<EOF
312 > [hooks]
312 > [hooks]
313 > preupdate.visibility =
313 > preupdate.visibility =
314 > EOF
314 > EOF
315
315
316 == test visibility to external update hook
316 == test visibility to external update hook
317
317
318 $ hg update -q -C 2
318 $ hg update -q -C 2
319 $ hg --config extensions.strip= strip 3
319 $ hg --config extensions.strip= strip 3
320 saved backup bundle to * (glob)
320 saved backup bundle to * (glob)
321
321
322 $ cat >> .hg/hgrc <<EOF
322 $ cat >> .hg/hgrc <<EOF
323 > [hooks]
323 > [hooks]
324 > update.visibility = sh $TESTTMP/checkvisibility.sh update
324 > update.visibility = sh $TESTTMP/checkvisibility.sh update
325 > EOF
325 > EOF
326
326
327 $ hg backout --merge -d '3 0' 1 --tool=true -m 'fixed comment'
327 $ hg backout --merge -d '3 0' 1 --tool=true -m 'fixed comment'
328 ==== update:
328 ==== update:
329 1:5a50a024c182
329 1:5a50a024c182
330 ====
330 ====
331 reverting a
331 reverting a
332 created new head
332 created new head
333 changeset 3:d92a3f57f067 backs out changeset 1:5a50a024c182
333 changeset 3:d92a3f57f067 backs out changeset 1:5a50a024c182
334 ==== update:
334 ==== update:
335 2:6ea3f2a197a2
335 2:6ea3f2a197a2
336 ====
336 ====
337 merging with changeset 3:d92a3f57f067
337 merging with changeset 3:d92a3f57f067
338 merging a
338 merging a
339 ==== update:
339 ==== update:
340 2:6ea3f2a197a2
340 2:6ea3f2a197a2
341 3:d92a3f57f067
341 3:d92a3f57f067
342 ====
342 ====
343 0 files updated, 1 files merged, 0 files removed, 0 files unresolved
343 0 files updated, 1 files merged, 0 files removed, 0 files unresolved
344 (branch merge, don't forget to commit)
344 (branch merge, don't forget to commit)
345
345
346 $ cat >> .hg/hgrc <<EOF
346 $ cat >> .hg/hgrc <<EOF
347 > [hooks]
347 > [hooks]
348 > update.visibility =
348 > update.visibility =
349 > EOF
349 > EOF
350
350
351 $ cd ..
351 $ cd ..
352
352
353 backout should not back out subsequent changesets
353 backout should not back out subsequent changesets
354
354
355 $ hg init onecs
355 $ hg init onecs
356 $ cd onecs
356 $ cd onecs
357 $ echo 1 > a
357 $ echo 1 > a
358 $ hg commit -d '0 0' -A -m a
358 $ hg commit -d '0 0' -A -m a
359 adding a
359 adding a
360 $ echo 2 >> a
360 $ echo 2 >> a
361 $ hg commit -d '1 0' -m b
361 $ hg commit -d '1 0' -m b
362 $ echo 1 > b
362 $ echo 1 > b
363 $ hg commit -d '2 0' -A -m c
363 $ hg commit -d '2 0' -A -m c
364 adding b
364 adding b
365 $ hg summary
365 $ hg summary
366 parent: 2:882396649954 tip
366 parent: 2:882396649954 tip
367 c
367 c
368 branch: default
368 branch: default
369 commit: (clean)
369 commit: (clean)
370 update: (current)
370 update: (current)
371 phases: 3 draft
371 phases: 3 draft
372
372
373 without --merge
373 without --merge
374 $ hg backout -d '3 0' 1 --tool=true
374 $ hg backout -d '3 0' 1 --tool=true
375 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
375 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
376 changeset 22bca4c721e5 backed out, don't forget to commit.
376 changeset 22bca4c721e5 backed out, don't forget to commit.
377 $ hg locate b
377 $ hg locate b
378 b
378 b
379 $ hg update -C tip
379 $ hg update -C tip
380 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
380 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
381 $ hg locate b
381 $ hg locate b
382 b
382 b
383 $ hg summary
383 $ hg summary
384 parent: 2:882396649954 tip
384 parent: 2:882396649954 tip
385 c
385 c
386 branch: default
386 branch: default
387 commit: (clean)
387 commit: (clean)
388 update: (current)
388 update: (current)
389 phases: 3 draft
389 phases: 3 draft
390
390
391 with --merge
391 with --merge
392 $ hg backout --merge -d '3 0' 1 --tool=true
392 $ hg backout --merge -d '3 0' 1 --tool=true
393 reverting a
393 reverting a
394 created new head
394 created new head
395 changeset 3:3202beb76721 backs out changeset 1:22bca4c721e5
395 changeset 3:3202beb76721 backs out changeset 1:22bca4c721e5
396 merging with changeset 3:3202beb76721
396 merging with changeset 3:3202beb76721
397 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
397 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
398 (branch merge, don't forget to commit)
398 (branch merge, don't forget to commit)
399 $ hg locate b
399 $ hg locate b
400 b
400 b
401 $ hg update -C tip
401 $ hg update -C tip
402 1 files updated, 0 files merged, 1 files removed, 0 files unresolved
402 1 files updated, 0 files merged, 1 files removed, 0 files unresolved
403 $ hg locate b
403 $ hg locate b
404 [1]
404 [1]
405
405
406 $ cd ..
406 $ cd ..
407 $ hg init m
407 $ hg init m
408 $ cd m
408 $ cd m
409 $ echo a > a
409 $ echo a > a
410 $ hg commit -d '0 0' -A -m a
410 $ hg commit -d '0 0' -A -m a
411 adding a
411 adding a
412 $ echo b > b
412 $ echo b > b
413 $ hg commit -d '1 0' -A -m b
413 $ hg commit -d '1 0' -A -m b
414 adding b
414 adding b
415 $ echo c > c
415 $ echo c > c
416 $ hg commit -d '2 0' -A -m b
416 $ hg commit -d '2 0' -A -m b
417 adding c
417 adding c
418 $ hg update 1
418 $ hg update 1
419 0 files updated, 0 files merged, 1 files removed, 0 files unresolved
419 0 files updated, 0 files merged, 1 files removed, 0 files unresolved
420 $ echo d > d
420 $ echo d > d
421 $ hg commit -d '3 0' -A -m c
421 $ hg commit -d '3 0' -A -m c
422 adding d
422 adding d
423 created new head
423 created new head
424 $ hg merge 2
424 $ hg merge 2
425 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
425 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
426 (branch merge, don't forget to commit)
426 (branch merge, don't forget to commit)
427 $ hg commit -d '4 0' -A -m d
427 $ hg commit -d '4 0' -A -m d
428 $ hg summary
428 $ hg summary
429 parent: 4:b2f3bb92043e tip
429 parent: 4:b2f3bb92043e tip
430 d
430 d
431 branch: default
431 branch: default
432 commit: (clean)
432 commit: (clean)
433 update: (current)
433 update: (current)
434 phases: 5 draft
434 phases: 5 draft
435
435
436 backout of merge should fail
436 backout of merge should fail
437
437
438 $ hg backout 4
438 $ hg backout 4
439 abort: cannot backout a merge changeset
439 abort: cannot backout a merge changeset
440 [255]
440 [255]
441
441
442 backout of merge with bad parent should fail
442 backout of merge with bad parent should fail
443
443
444 $ hg backout --parent 0 4
444 $ hg backout --parent 0 4
445 abort: cb9a9f314b8b is not a parent of b2f3bb92043e
445 abort: cb9a9f314b8b is not a parent of b2f3bb92043e
446 [255]
446 [255]
447
447
448 backout of non-merge with parent should fail
448 backout of non-merge with parent should fail
449
449
450 $ hg backout --parent 0 3
450 $ hg backout --parent 0 3
451 abort: cannot use --parent on non-merge changeset
451 abort: cannot use --parent on non-merge changeset
452 [255]
452 [255]
453
453
454 backout with valid parent should be ok
454 backout with valid parent should be ok
455
455
456 $ hg backout -d '5 0' --parent 2 4 --tool=true
456 $ hg backout -d '5 0' --parent 2 4 --tool=true
457 removing d
457 removing d
458 changeset 5:10e5328c8435 backs out changeset 4:b2f3bb92043e
458 changeset 5:10e5328c8435 backs out changeset 4:b2f3bb92043e
459 $ hg summary
459 $ hg summary
460 parent: 5:10e5328c8435 tip
460 parent: 5:10e5328c8435 tip
461 Backed out changeset b2f3bb92043e
461 Backed out changeset b2f3bb92043e
462 branch: default
462 branch: default
463 commit: (clean)
463 commit: (clean)
464 update: (current)
464 update: (current)
465 phases: 6 draft
465 phases: 6 draft
466
466
467 $ hg rollback
467 $ hg rollback
468 repository tip rolled back to revision 4 (undo commit)
468 repository tip rolled back to revision 4 (undo commit)
469 working directory now based on revision 4
469 working directory now based on revision 4
470 $ hg update -C
470 $ hg update -C
471 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
471 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
472 $ hg summary
472 $ hg summary
473 parent: 4:b2f3bb92043e tip
473 parent: 4:b2f3bb92043e tip
474 d
474 d
475 branch: default
475 branch: default
476 commit: (clean)
476 commit: (clean)
477 update: (current)
477 update: (current)
478 phases: 5 draft
478 phases: 5 draft
479
479
480 $ hg backout -d '6 0' --parent 3 4 --tool=true
480 $ hg backout -d '6 0' --parent 3 4 --tool=true
481 removing c
481 removing c
482 changeset 5:033590168430 backs out changeset 4:b2f3bb92043e
482 changeset 5:033590168430 backs out changeset 4:b2f3bb92043e
483 $ hg summary
483 $ hg summary
484 parent: 5:033590168430 tip
484 parent: 5:033590168430 tip
485 Backed out changeset b2f3bb92043e
485 Backed out changeset b2f3bb92043e
486 branch: default
486 branch: default
487 commit: (clean)
487 commit: (clean)
488 update: (current)
488 update: (current)
489 phases: 6 draft
489 phases: 6 draft
490
490
491 $ cd ..
491 $ cd ..
492
492
493 named branches
493 named branches
494
494
495 $ hg init named_branches
495 $ hg init named_branches
496 $ cd named_branches
496 $ cd named_branches
497
497
498 $ echo default > default
498 $ echo default > default
499 $ hg ci -d '0 0' -Am default
499 $ hg ci -d '0 0' -Am default
500 adding default
500 adding default
501 $ hg branch branch1
501 $ hg branch branch1
502 marked working directory as branch branch1
502 marked working directory as branch branch1
503 (branches are permanent and global, did you want a bookmark?)
503 (branches are permanent and global, did you want a bookmark?)
504 $ echo branch1 > file1
504 $ echo branch1 > file1
505 $ hg ci -d '1 0' -Am file1
505 $ hg ci -d '1 0' -Am file1
506 adding file1
506 adding file1
507 $ hg branch branch2
507 $ hg branch branch2
508 marked working directory as branch branch2
508 marked working directory as branch branch2
509 $ echo branch2 > file2
509 $ echo branch2 > file2
510 $ hg ci -d '2 0' -Am file2
510 $ hg ci -d '2 0' -Am file2
511 adding file2
511 adding file2
512
512
513 without --merge
513 without --merge
514 $ hg backout -r 1 --tool=true
514 $ hg backout -r 1 --tool=true
515 0 files updated, 0 files merged, 1 files removed, 0 files unresolved
515 0 files updated, 0 files merged, 1 files removed, 0 files unresolved
516 changeset bf1602f437f3 backed out, don't forget to commit.
516 changeset bf1602f437f3 backed out, don't forget to commit.
517 $ hg branch
517 $ hg branch
518 branch2
518 branch2
519 $ hg status -A
519 $ hg status -A
520 R file1
520 R file1
521 C default
521 C default
522 C file2
522 C file2
523 $ hg summary
523 $ hg summary
524 parent: 2:45bbcd363bf0 tip
524 parent: 2:45bbcd363bf0 tip
525 file2
525 file2
526 branch: branch2
526 branch: branch2
527 commit: 1 removed
527 commit: 1 removed
528 update: (current)
528 update: (current)
529 phases: 3 draft
529 phases: 3 draft
530
530
531 with --merge
531 with --merge
532 (this also tests that editor is invoked if '--edit' is specified
532 (this also tests that editor is invoked if '--edit' is specified
533 explicitly regardless of '--message')
533 explicitly regardless of '--message')
534
534
535 $ hg update -qC
535 $ hg update -qC
536 $ HGEDITOR=cat hg backout --merge -d '3 0' -r 1 -m 'backout on branch1' --tool=true --edit
536 $ HGEDITOR=cat hg backout --merge -d '3 0' -r 1 -m 'backout on branch1' --tool=true --edit
537 removing file1
537 removing file1
538 backout on branch1
538 backout on branch1
539
539
540
540
541 HG: Enter commit message. Lines beginning with 'HG:' are removed.
541 HG: Enter commit message. Lines beginning with 'HG:' are removed.
542 HG: Leave message empty to abort commit.
542 HG: Leave message empty to abort commit.
543 HG: --
543 HG: --
544 HG: user: test
544 HG: user: test
545 HG: branch 'branch2'
545 HG: branch 'branch2'
546 HG: removed file1
546 HG: removed file1
547 created new head
547 created new head
548 changeset 3:d4e8f6db59fb backs out changeset 1:bf1602f437f3
548 changeset 3:d4e8f6db59fb backs out changeset 1:bf1602f437f3
549 merging with changeset 3:d4e8f6db59fb
549 merging with changeset 3:d4e8f6db59fb
550 0 files updated, 0 files merged, 1 files removed, 0 files unresolved
550 0 files updated, 0 files merged, 1 files removed, 0 files unresolved
551 (branch merge, don't forget to commit)
551 (branch merge, don't forget to commit)
552 $ hg summary
552 $ hg summary
553 parent: 2:45bbcd363bf0
553 parent: 2:45bbcd363bf0
554 file2
554 file2
555 parent: 3:d4e8f6db59fb tip
555 parent: 3:d4e8f6db59fb tip
556 backout on branch1
556 backout on branch1
557 branch: branch2
557 branch: branch2
558 commit: 1 removed (merge)
558 commit: 1 removed (merge)
559 update: (current)
559 update: (current)
560 phases: 4 draft
560 phases: 4 draft
561 $ hg update -q -C 2
561 $ hg update -q -C 2
562
562
563 on branch2 with branch1 not merged, so file1 should still exist:
563 on branch2 with branch1 not merged, so file1 should still exist:
564
564
565 $ hg id
565 $ hg id
566 45bbcd363bf0 (branch2)
566 45bbcd363bf0 (branch2)
567 $ hg st -A
567 $ hg st -A
568 C default
568 C default
569 C file1
569 C file1
570 C file2
570 C file2
571 $ hg summary
571 $ hg summary
572 parent: 2:45bbcd363bf0
572 parent: 2:45bbcd363bf0
573 file2
573 file2
574 branch: branch2
574 branch: branch2
575 commit: (clean)
575 commit: (clean)
576 update: 1 new changesets, 2 branch heads (merge)
576 update: 1 new changesets, 2 branch heads (merge)
577 phases: 4 draft
577 phases: 4 draft
578
578
579 on branch2 with branch1 merged, so file1 should be gone:
579 on branch2 with branch1 merged, so file1 should be gone:
580
580
581 $ hg merge
581 $ hg merge
582 0 files updated, 0 files merged, 1 files removed, 0 files unresolved
582 0 files updated, 0 files merged, 1 files removed, 0 files unresolved
583 (branch merge, don't forget to commit)
583 (branch merge, don't forget to commit)
584 $ hg ci -d '4 0' -m 'merge backout of branch1'
584 $ hg ci -d '4 0' -m 'merge backout of branch1'
585 $ hg id
585 $ hg id
586 22149cdde76d (branch2) tip
586 22149cdde76d (branch2) tip
587 $ hg st -A
587 $ hg st -A
588 C default
588 C default
589 C file2
589 C file2
590 $ hg summary
590 $ hg summary
591 parent: 4:22149cdde76d tip
591 parent: 4:22149cdde76d tip
592 merge backout of branch1
592 merge backout of branch1
593 branch: branch2
593 branch: branch2
594 commit: (clean)
594 commit: (clean)
595 update: (current)
595 update: (current)
596 phases: 5 draft
596 phases: 5 draft
597
597
598 on branch1, so no file1 and file2:
598 on branch1, so no file1 and file2:
599
599
600 $ hg co -C branch1
600 $ hg co -C branch1
601 1 files updated, 0 files merged, 1 files removed, 0 files unresolved
601 1 files updated, 0 files merged, 1 files removed, 0 files unresolved
602 $ hg id
602 $ hg id
603 bf1602f437f3 (branch1)
603 bf1602f437f3 (branch1)
604 $ hg st -A
604 $ hg st -A
605 C default
605 C default
606 C file1
606 C file1
607 $ hg summary
607 $ hg summary
608 parent: 1:bf1602f437f3
608 parent: 1:bf1602f437f3
609 file1
609 file1
610 branch: branch1
610 branch: branch1
611 commit: (clean)
611 commit: (clean)
612 update: (current)
612 update: (current)
613 phases: 5 draft
613 phases: 5 draft
614
614
615 $ cd ..
615 $ cd ..
616
616
617 backout of empty changeset (issue4190)
617 backout of empty changeset (issue4190)
618
618
619 $ hg init emptycommit
619 $ hg init emptycommit
620 $ cd emptycommit
620 $ cd emptycommit
621
621
622 $ touch file1
622 $ touch file1
623 $ hg ci -Aqm file1
623 $ hg ci -Aqm file1
624 $ hg branch -q branch1
624 $ hg branch -q branch1
625 $ hg ci -qm branch1
625 $ hg ci -qm branch1
626 $ hg backout -v 1
626 $ hg backout -v 1
627 resolving manifests
627 resolving manifests
628 nothing changed
628 nothing changed
629 [1]
629 [1]
630
630
631 $ cd ..
631 $ cd ..
632
632
633
633
634 Test usage of `hg resolve` in case of conflict
634 Test usage of `hg resolve` in case of conflict
635 (issue4163)
635 (issue4163)
636
636
637 $ hg init issue4163
637 $ hg init issue4163
638 $ cd issue4163
638 $ cd issue4163
639 $ touch foo
639 $ touch foo
640 $ hg add foo
640 $ hg add foo
641 $ cat > foo << EOF
641 $ cat > foo << EOF
642 > one
642 > one
643 > two
643 > two
644 > three
644 > three
645 > four
645 > four
646 > five
646 > five
647 > six
647 > six
648 > seven
648 > seven
649 > height
649 > height
650 > nine
650 > nine
651 > ten
651 > ten
652 > EOF
652 > EOF
653 $ hg ci -m 'initial'
653 $ hg ci -m 'initial'
654 $ cat > foo << EOF
654 $ cat > foo << EOF
655 > one
655 > one
656 > two
656 > two
657 > THREE
657 > THREE
658 > four
658 > four
659 > five
659 > five
660 > six
660 > six
661 > seven
661 > seven
662 > height
662 > height
663 > nine
663 > nine
664 > ten
664 > ten
665 > EOF
665 > EOF
666 $ hg ci -m 'capital three'
666 $ hg ci -m 'capital three'
667 $ cat > foo << EOF
667 $ cat > foo << EOF
668 > one
668 > one
669 > two
669 > two
670 > THREE
670 > THREE
671 > four
671 > four
672 > five
672 > five
673 > six
673 > six
674 > seven
674 > seven
675 > height
675 > height
676 > nine
676 > nine
677 > TEN
677 > TEN
678 > EOF
678 > EOF
679 $ hg ci -m 'capital ten'
679 $ hg ci -m 'capital ten'
680 $ hg backout -r 'desc("capital three")' --tool internal:fail
680 $ hg backout -r 'desc("capital three")' --tool internal:fail
681 0 files updated, 0 files merged, 0 files removed, 1 files unresolved
681 0 files updated, 0 files merged, 0 files removed, 1 files unresolved
682 use 'hg resolve' to retry unresolved file merges
682 use 'hg resolve' to retry unresolved file merges
683 [1]
683 [1]
684 $ hg status
684 $ hg status
685 $ hg debugmergestate
685 $ hg debugmergestate
686 * version 2 records
686 * version 2 records
687 local: b71750c4b0fdf719734971e3ef90dbeab5919a2d
687 local: b71750c4b0fdf719734971e3ef90dbeab5919a2d
688 other: a30dd8addae3ce71b8667868478542bc417439e6
688 other: a30dd8addae3ce71b8667868478542bc417439e6
689 file: foo (state "u", hash 0beec7b5ea3f0fdbc95d0dd47f3c5bc275da8a33)
689 file: foo (record type "F", state "u", hash 0beec7b5ea3f0fdbc95d0dd47f3c5bc275da8a33)
690 local path: foo (flags "")
690 local path: foo (flags "")
691 ancestor path: foo (node f89532f44c247a0e993d63e3a734dd781ab04708)
691 ancestor path: foo (node f89532f44c247a0e993d63e3a734dd781ab04708)
692 other path: foo (node f50039b486d6fa1a90ae51778388cad161f425ee)
692 other path: foo (node f50039b486d6fa1a90ae51778388cad161f425ee)
693 $ mv .hg/merge/state2 .hg/merge/state2-moved
693 $ mv .hg/merge/state2 .hg/merge/state2-moved
694 $ hg debugmergestate
694 $ hg debugmergestate
695 * version 1 records
695 * version 1 records
696 local: b71750c4b0fdf719734971e3ef90dbeab5919a2d
696 local: b71750c4b0fdf719734971e3ef90dbeab5919a2d
697 file: foo (state "u", hash 0beec7b5ea3f0fdbc95d0dd47f3c5bc275da8a33)
697 file: foo (record type "F", state "u", hash 0beec7b5ea3f0fdbc95d0dd47f3c5bc275da8a33)
698 local path: foo (flags "")
698 local path: foo (flags "")
699 ancestor path: foo (node f89532f44c247a0e993d63e3a734dd781ab04708)
699 ancestor path: foo (node f89532f44c247a0e993d63e3a734dd781ab04708)
700 other path: foo (node not stored in v1 format)
700 other path: foo (node not stored in v1 format)
701 $ mv .hg/merge/state2-moved .hg/merge/state2
701 $ mv .hg/merge/state2-moved .hg/merge/state2
702 $ hg resolve -l # still unresolved
702 $ hg resolve -l # still unresolved
703 U foo
703 U foo
704 $ hg summary
704 $ hg summary
705 parent: 2:b71750c4b0fd tip
705 parent: 2:b71750c4b0fd tip
706 capital ten
706 capital ten
707 branch: default
707 branch: default
708 commit: 1 unresolved (clean)
708 commit: 1 unresolved (clean)
709 update: (current)
709 update: (current)
710 phases: 3 draft
710 phases: 3 draft
711 $ hg resolve --all --debug
711 $ hg resolve --all --debug
712 picked tool ':merge' for foo (binary False symlink False)
712 picked tool ':merge' for foo (binary False symlink False)
713 merging foo
713 merging foo
714 my foo@b71750c4b0fd+ other foo@a30dd8addae3 ancestor foo@913609522437
714 my foo@b71750c4b0fd+ other foo@a30dd8addae3 ancestor foo@913609522437
715 premerge successful
715 premerge successful
716 (no more unresolved files)
716 (no more unresolved files)
717 $ hg status
717 $ hg status
718 M foo
718 M foo
719 ? foo.orig
719 ? foo.orig
720 $ hg resolve -l
720 $ hg resolve -l
721 R foo
721 R foo
722 $ hg summary
722 $ hg summary
723 parent: 2:b71750c4b0fd tip
723 parent: 2:b71750c4b0fd tip
724 capital ten
724 capital ten
725 branch: default
725 branch: default
726 commit: 1 modified, 1 unknown
726 commit: 1 modified, 1 unknown
727 update: (current)
727 update: (current)
728 phases: 3 draft
728 phases: 3 draft
729 $ cat foo
729 $ cat foo
730 one
730 one
731 two
731 two
732 three
732 three
733 four
733 four
734 five
734 five
735 six
735 six
736 seven
736 seven
737 height
737 height
738 nine
738 nine
739 TEN
739 TEN
740
740
741
741
General Comments 0
You need to be logged in to leave comments. Login now