##// END OF EJS Templates
filemerge: switch the default name for internal tools from internal:x to :x
Mads Kiilerich -
r22707:38e0363d default
parent child Browse files
Show More
@@ -1,471 +1,471
1 1 # filemerge.py - file-level merge handling for Mercurial
2 2 #
3 3 # Copyright 2006, 2007, 2008 Matt Mackall <mpm@selenic.com>
4 4 #
5 5 # This software may be used and distributed according to the terms of the
6 6 # GNU General Public License version 2 or any later version.
7 7
8 8 from node import short
9 9 from i18n import _
10 10 import util, simplemerge, match, error, templater, templatekw
11 11 import os, tempfile, re, filecmp
12 12 import tagmerge
13 13
14 14 def _toolstr(ui, tool, part, default=""):
15 15 return ui.config("merge-tools", tool + "." + part, default)
16 16
17 17 def _toolbool(ui, tool, part, default=False):
18 18 return ui.configbool("merge-tools", tool + "." + part, default)
19 19
20 20 def _toollist(ui, tool, part, default=[]):
21 21 return ui.configlist("merge-tools", tool + "." + part, default)
22 22
23 23 internals = {}
24 24
25 25 def internaltool(name, trymerge, onfailure=None):
26 26 '''return a decorator for populating internal merge tool table'''
27 27 def decorator(func):
28 fullname = 'internal:' + name
28 fullname = ':' + name
29 29 func.__doc__ = "``%s``\n" % fullname + func.__doc__.strip()
30 30 internals[fullname] = func
31 internals[':' + name] = func
31 internals['internal:' + name] = func
32 32 func.trymerge = trymerge
33 33 func.onfailure = onfailure
34 34 return func
35 35 return decorator
36 36
37 37 def _findtool(ui, tool):
38 38 if tool in internals:
39 39 return tool
40 40 for kn in ("regkey", "regkeyalt"):
41 41 k = _toolstr(ui, tool, kn)
42 42 if not k:
43 43 continue
44 44 p = util.lookupreg(k, _toolstr(ui, tool, "regname"))
45 45 if p:
46 46 p = util.findexe(p + _toolstr(ui, tool, "regappend"))
47 47 if p:
48 48 return p
49 49 exe = _toolstr(ui, tool, "executable", tool)
50 50 return util.findexe(util.expandpath(exe))
51 51
52 52 def _picktool(repo, ui, path, binary, symlink):
53 53 def check(tool, pat, symlink, binary):
54 54 tmsg = tool
55 55 if pat:
56 56 tmsg += " specified for " + pat
57 57 if not _findtool(ui, tool):
58 58 if pat: # explicitly requested tool deserves a warning
59 59 ui.warn(_("couldn't find merge tool %s\n") % tmsg)
60 60 else: # configured but non-existing tools are more silent
61 61 ui.note(_("couldn't find merge tool %s\n") % tmsg)
62 62 elif symlink and not _toolbool(ui, tool, "symlink"):
63 63 ui.warn(_("tool %s can't handle symlinks\n") % tmsg)
64 64 elif binary and not _toolbool(ui, tool, "binary"):
65 65 ui.warn(_("tool %s can't handle binary\n") % tmsg)
66 66 elif not util.gui() and _toolbool(ui, tool, "gui"):
67 67 ui.warn(_("tool %s requires a GUI\n") % tmsg)
68 68 else:
69 69 return True
70 70 return False
71 71
72 72 # forcemerge comes from command line arguments, highest priority
73 73 force = ui.config('ui', 'forcemerge')
74 74 if force:
75 75 toolpath = _findtool(ui, force)
76 76 if toolpath:
77 77 return (force, util.shellquote(toolpath))
78 78 else:
79 79 # mimic HGMERGE if given tool not found
80 80 return (force, force)
81 81
82 82 # HGMERGE takes next precedence
83 83 hgmerge = os.environ.get("HGMERGE")
84 84 if hgmerge:
85 85 return (hgmerge, hgmerge)
86 86
87 87 # then patterns
88 88 for pat, tool in ui.configitems("merge-patterns"):
89 89 mf = match.match(repo.root, '', [pat])
90 90 if mf(path) and check(tool, pat, symlink, False):
91 91 toolpath = _findtool(ui, tool)
92 92 return (tool, util.shellquote(toolpath))
93 93
94 94 # then merge tools
95 95 tools = {}
96 96 for k, v in ui.configitems("merge-tools"):
97 97 t = k.split('.')[0]
98 98 if t not in tools:
99 99 tools[t] = int(_toolstr(ui, t, "priority", "0"))
100 100 names = tools.keys()
101 101 tools = sorted([(-p, t) for t, p in tools.items()])
102 102 uimerge = ui.config("ui", "merge")
103 103 if uimerge:
104 104 if uimerge not in names:
105 105 return (uimerge, uimerge)
106 106 tools.insert(0, (None, uimerge)) # highest priority
107 107 tools.append((None, "hgmerge")) # the old default, if found
108 108 for p, t in tools:
109 109 if check(t, None, symlink, binary):
110 110 toolpath = _findtool(ui, t)
111 111 return (t, util.shellquote(toolpath))
112 112
113 113 # internal merge or prompt as last resort
114 114 if symlink or binary:
115 return "internal:prompt", None
116 return "internal:merge", None
115 return ":prompt", None
116 return ":merge", None
117 117
118 118 def _eoltype(data):
119 119 "Guess the EOL type of a file"
120 120 if '\0' in data: # binary
121 121 return None
122 122 if '\r\n' in data: # Windows
123 123 return '\r\n'
124 124 if '\r' in data: # Old Mac
125 125 return '\r'
126 126 if '\n' in data: # UNIX
127 127 return '\n'
128 128 return None # unknown
129 129
130 130 def _matcheol(file, origfile):
131 131 "Convert EOL markers in a file to match origfile"
132 132 tostyle = _eoltype(util.readfile(origfile))
133 133 if tostyle:
134 134 data = util.readfile(file)
135 135 style = _eoltype(data)
136 136 if style:
137 137 newdata = data.replace(style, tostyle)
138 138 if newdata != data:
139 139 util.writefile(file, newdata)
140 140
141 141 @internaltool('prompt', False)
142 142 def _iprompt(repo, mynode, orig, fcd, fco, fca, toolconf):
143 143 """Asks the user which of the local or the other version to keep as
144 144 the merged version."""
145 145 ui = repo.ui
146 146 fd = fcd.path()
147 147
148 148 if ui.promptchoice(_(" no tool found to merge %s\n"
149 149 "keep (l)ocal or take (o)ther?"
150 150 "$$ &Local $$ &Other") % fd, 0):
151 151 return _iother(repo, mynode, orig, fcd, fco, fca, toolconf)
152 152 else:
153 153 return _ilocal(repo, mynode, orig, fcd, fco, fca, toolconf)
154 154
155 155 @internaltool('local', False)
156 156 def _ilocal(repo, mynode, orig, fcd, fco, fca, toolconf):
157 157 """Uses the local version of files as the merged version."""
158 158 return 0
159 159
160 160 @internaltool('other', False)
161 161 def _iother(repo, mynode, orig, fcd, fco, fca, toolconf):
162 162 """Uses the other version of files as the merged version."""
163 163 repo.wwrite(fcd.path(), fco.data(), fco.flags())
164 164 return 0
165 165
166 166 @internaltool('fail', False)
167 167 def _ifail(repo, mynode, orig, fcd, fco, fca, toolconf):
168 168 """
169 169 Rather than attempting to merge files that were modified on both
170 170 branches, it marks them as unresolved. The resolve command must be
171 171 used to resolve these conflicts."""
172 172 return 1
173 173
174 174 def _premerge(repo, toolconf, files, labels=None):
175 175 tool, toolpath, binary, symlink = toolconf
176 176 if symlink:
177 177 return 1
178 178 a, b, c, back = files
179 179
180 180 ui = repo.ui
181 181
182 182 validkeep = ['keep', 'keep-merge3']
183 183
184 184 # do we attempt to simplemerge first?
185 185 try:
186 186 premerge = _toolbool(ui, tool, "premerge", not binary)
187 187 except error.ConfigError:
188 188 premerge = _toolstr(ui, tool, "premerge").lower()
189 189 if premerge not in validkeep:
190 190 _valid = ', '.join(["'" + v + "'" for v in validkeep])
191 191 raise error.ConfigError(_("%s.premerge not valid "
192 192 "('%s' is neither boolean nor %s)") %
193 193 (tool, premerge, _valid))
194 194
195 195 if premerge:
196 196 if premerge == 'keep-merge3':
197 197 if not labels:
198 198 labels = _defaultconflictlabels
199 199 if len(labels) < 3:
200 200 labels.append('base')
201 201 r = simplemerge.simplemerge(ui, a, b, c, quiet=True, label=labels)
202 202 if not r:
203 203 ui.debug(" premerge successful\n")
204 204 return 0
205 205 if premerge not in validkeep:
206 206 util.copyfile(back, a) # restore from backup and try again
207 207 return 1 # continue merging
208 208
209 209 @internaltool('merge', True,
210 210 _("merging %s incomplete! "
211 211 "(edit conflicts, then use 'hg resolve --mark')\n"))
212 212 def _imerge(repo, mynode, orig, fcd, fco, fca, toolconf, files, labels=None):
213 213 """
214 214 Uses the internal non-interactive simple merge algorithm for merging
215 215 files. It will fail if there are any conflicts and leave markers in
216 216 the partially merged file. Markers will have two sections, one for each side
217 217 of merge."""
218 218 tool, toolpath, binary, symlink = toolconf
219 219 if symlink:
220 220 repo.ui.warn(_('warning: internal:merge cannot merge symlinks '
221 221 'for %s\n') % fcd.path())
222 222 return False, 1
223 223 r = _premerge(repo, toolconf, files, labels=labels)
224 224 if r:
225 225 a, b, c, back = files
226 226
227 227 ui = repo.ui
228 228
229 229 r = simplemerge.simplemerge(ui, a, b, c, label=labels)
230 230 return True, r
231 231 return False, 0
232 232
233 233 @internaltool('merge3', True,
234 234 _("merging %s incomplete! "
235 235 "(edit conflicts, then use 'hg resolve --mark')\n"))
236 236 def _imerge3(repo, mynode, orig, fcd, fco, fca, toolconf, files, labels=None):
237 237 """
238 238 Uses the internal non-interactive simple merge algorithm for merging
239 239 files. It will fail if there are any conflicts and leave markers in
240 240 the partially merged file. Marker will have three sections, one from each
241 241 side of the merge and one for the base content."""
242 242 if not labels:
243 243 labels = _defaultconflictlabels
244 244 if len(labels) < 3:
245 245 labels.append('base')
246 246 return _imerge(repo, mynode, orig, fcd, fco, fca, toolconf, files, labels)
247 247
248 248 @internaltool('tagmerge', True,
249 249 _("automatic tag merging of %s failed! "
250 "(use 'hg resolve --tool internal:merge' or another merge "
250 "(use 'hg resolve --tool :merge' or another merge "
251 251 "tool of your choice)\n"))
252 252 def _itagmerge(repo, mynode, orig, fcd, fco, fca, toolconf, files, labels=None):
253 253 """
254 254 Uses the internal tag merge algorithm (experimental).
255 255 """
256 256 return tagmerge.merge(repo, fcd, fco, fca)
257 257
258 258 @internaltool('dump', True)
259 259 def _idump(repo, mynode, orig, fcd, fco, fca, toolconf, files, labels=None):
260 260 """
261 261 Creates three versions of the files to merge, containing the
262 262 contents of local, other and base. These files can then be used to
263 263 perform a merge manually. If the file to be merged is named
264 264 ``a.txt``, these files will accordingly be named ``a.txt.local``,
265 265 ``a.txt.other`` and ``a.txt.base`` and they will be placed in the
266 266 same directory as ``a.txt``."""
267 267 r = _premerge(repo, toolconf, files, labels=labels)
268 268 if r:
269 269 a, b, c, back = files
270 270
271 271 fd = fcd.path()
272 272
273 273 util.copyfile(a, a + ".local")
274 274 repo.wwrite(fd + ".other", fco.data(), fco.flags())
275 275 repo.wwrite(fd + ".base", fca.data(), fca.flags())
276 276 return False, r
277 277
278 278 def _xmerge(repo, mynode, orig, fcd, fco, fca, toolconf, files, labels=None):
279 279 r = _premerge(repo, toolconf, files, labels=labels)
280 280 if r:
281 281 tool, toolpath, binary, symlink = toolconf
282 282 a, b, c, back = files
283 283 out = ""
284 284 env = {'HG_FILE': fcd.path(),
285 285 'HG_MY_NODE': short(mynode),
286 286 'HG_OTHER_NODE': str(fco.changectx()),
287 287 'HG_BASE_NODE': str(fca.changectx()),
288 288 'HG_MY_ISLINK': 'l' in fcd.flags(),
289 289 'HG_OTHER_ISLINK': 'l' in fco.flags(),
290 290 'HG_BASE_ISLINK': 'l' in fca.flags(),
291 291 }
292 292
293 293 ui = repo.ui
294 294
295 295 args = _toolstr(ui, tool, "args", '$local $base $other')
296 296 if "$output" in args:
297 297 out, a = a, back # read input from backup, write to original
298 298 replace = {'local': a, 'base': b, 'other': c, 'output': out}
299 299 args = util.interpolate(r'\$', replace, args,
300 300 lambda s: util.shellquote(util.localpath(s)))
301 301 r = util.system(toolpath + ' ' + args, cwd=repo.root, environ=env,
302 302 out=ui.fout)
303 303 return True, r
304 304 return False, 0
305 305
306 306 def _formatconflictmarker(repo, ctx, template, label, pad):
307 307 """Applies the given template to the ctx, prefixed by the label.
308 308
309 309 Pad is the minimum width of the label prefix, so that multiple markers
310 310 can have aligned templated parts.
311 311 """
312 312 if ctx.node() is None:
313 313 ctx = ctx.p1()
314 314
315 315 props = templatekw.keywords.copy()
316 316 props['templ'] = template
317 317 props['ctx'] = ctx
318 318 props['repo'] = repo
319 319 templateresult = template('conflictmarker', **props)
320 320
321 321 label = ('%s:' % label).ljust(pad + 1)
322 322 mark = '%s %s' % (label, templater.stringify(templateresult))
323 323
324 324 if mark:
325 325 mark = mark.splitlines()[0] # split for safety
326 326
327 327 # 8 for the prefix of conflict marker lines (e.g. '<<<<<<< ')
328 328 return util.ellipsis(mark, 80 - 8)
329 329
330 330 _defaultconflictmarker = ('{node|short} ' +
331 331 '{ifeq(tags, "tip", "", "{tags} ")}' +
332 332 '{if(bookmarks, "{bookmarks} ")}' +
333 333 '{ifeq(branch, "default", "", "{branch} ")}' +
334 334 '- {author|user}: {desc|firstline}')
335 335
336 336 _defaultconflictlabels = ['local', 'other']
337 337
338 338 def _formatlabels(repo, fcd, fco, fca, labels):
339 339 """Formats the given labels using the conflict marker template.
340 340
341 341 Returns a list of formatted labels.
342 342 """
343 343 cd = fcd.changectx()
344 344 co = fco.changectx()
345 345 ca = fca.changectx()
346 346
347 347 ui = repo.ui
348 348 template = ui.config('ui', 'mergemarkertemplate', _defaultconflictmarker)
349 349 template = templater.parsestring(template, quoted=False)
350 350 tmpl = templater.templater(None, cache={'conflictmarker': template})
351 351
352 352 pad = max(len(l) for l in labels)
353 353
354 354 newlabels = [_formatconflictmarker(repo, cd, tmpl, labels[0], pad),
355 355 _formatconflictmarker(repo, co, tmpl, labels[1], pad)]
356 356 if len(labels) > 2:
357 357 newlabels.append(_formatconflictmarker(repo, ca, tmpl, labels[2], pad))
358 358 return newlabels
359 359
360 360 def filemerge(repo, mynode, orig, fcd, fco, fca, labels=None):
361 361 """perform a 3-way merge in the working directory
362 362
363 363 mynode = parent node before merge
364 364 orig = original local filename before merge
365 365 fco = other file context
366 366 fca = ancestor file context
367 367 fcd = local file context for current/destination file
368 368 """
369 369
370 370 def temp(prefix, ctx):
371 371 pre = "%s~%s." % (os.path.basename(ctx.path()), prefix)
372 372 (fd, name) = tempfile.mkstemp(prefix=pre)
373 373 data = repo.wwritedata(ctx.path(), ctx.data())
374 374 f = os.fdopen(fd, "wb")
375 375 f.write(data)
376 376 f.close()
377 377 return name
378 378
379 379 if not fco.cmp(fcd): # files identical?
380 380 return None
381 381
382 382 ui = repo.ui
383 383 fd = fcd.path()
384 384 binary = fcd.isbinary() or fco.isbinary() or fca.isbinary()
385 385 symlink = 'l' in fcd.flags() + fco.flags()
386 386 tool, toolpath = _picktool(repo, ui, fd, binary, symlink)
387 387 ui.debug("picked tool '%s' for %s (binary %s symlink %s)\n" %
388 388 (tool, fd, binary, symlink))
389 389
390 390 if tool in internals:
391 391 func = internals[tool]
392 392 trymerge = func.trymerge
393 393 onfailure = func.onfailure
394 394 else:
395 395 func = _xmerge
396 396 trymerge = True
397 397 onfailure = _("merging %s failed!\n")
398 398
399 399 toolconf = tool, toolpath, binary, symlink
400 400
401 401 if not trymerge:
402 402 return func(repo, mynode, orig, fcd, fco, fca, toolconf)
403 403
404 404 a = repo.wjoin(fd)
405 405 b = temp("base", fca)
406 406 c = temp("other", fco)
407 407 back = a + ".orig"
408 408 util.copyfile(a, back)
409 409
410 410 if orig != fco.path():
411 411 ui.status(_("merging %s and %s to %s\n") % (orig, fco.path(), fd))
412 412 else:
413 413 ui.status(_("merging %s\n") % fd)
414 414
415 415 ui.debug("my %s other %s ancestor %s\n" % (fcd, fco, fca))
416 416
417 417 markerstyle = ui.config('ui', 'mergemarkers', 'basic')
418 418 if not labels:
419 419 labels = _defaultconflictlabels
420 420 if markerstyle != 'basic':
421 421 labels = _formatlabels(repo, fcd, fco, fca, labels)
422 422
423 423 needcheck, r = func(repo, mynode, orig, fcd, fco, fca, toolconf,
424 424 (a, b, c, back), labels=labels)
425 425 if not needcheck:
426 426 if r:
427 427 if onfailure:
428 428 ui.warn(onfailure % fd)
429 429 else:
430 430 util.unlink(back)
431 431
432 432 util.unlink(b)
433 433 util.unlink(c)
434 434 return r
435 435
436 436 if not r and (_toolbool(ui, tool, "checkconflicts") or
437 437 'conflicts' in _toollist(ui, tool, "check")):
438 438 if re.search("^(<<<<<<< .*|=======|>>>>>>> .*)$", fcd.data(),
439 439 re.MULTILINE):
440 440 r = 1
441 441
442 442 checked = False
443 443 if 'prompt' in _toollist(ui, tool, "check"):
444 444 checked = True
445 445 if ui.promptchoice(_("was merge of '%s' successful (yn)?"
446 446 "$$ &Yes $$ &No") % fd, 1):
447 447 r = 1
448 448
449 449 if not r and not checked and (_toolbool(ui, tool, "checkchanged") or
450 450 'changed' in _toollist(ui, tool, "check")):
451 451 if filecmp.cmp(a, back):
452 452 if ui.promptchoice(_(" output file %s appears unchanged\n"
453 453 "was merge successful (yn)?"
454 454 "$$ &Yes $$ &No") % fd, 1):
455 455 r = 1
456 456
457 457 if _toolbool(ui, tool, "fixeol"):
458 458 _matcheol(a, back)
459 459
460 460 if r:
461 461 if onfailure:
462 462 ui.warn(onfailure % fd)
463 463 else:
464 464 util.unlink(back)
465 465
466 466 util.unlink(b)
467 467 util.unlink(c)
468 468 return r
469 469
470 470 # tell hggettext to extract docstrings from these functions:
471 471 i18nfunctions = internals.values()
@@ -1,85 +1,85
1 1 To merge files Mercurial uses merge tools.
2 2
3 3 A merge tool combines two different versions of a file into a merged
4 4 file. Merge tools are given the two files and the greatest common
5 5 ancestor of the two file versions, so they can determine the changes
6 6 made on both branches.
7 7
8 8 Merge tools are used both for :hg:`resolve`, :hg:`merge`, :hg:`update`,
9 9 :hg:`backout` and in several extensions.
10 10
11 11 Usually, the merge tool tries to automatically reconcile the files by
12 12 combining all non-overlapping changes that occurred separately in
13 13 the two different evolutions of the same initial base file. Furthermore, some
14 14 interactive merge programs make it easier to manually resolve
15 15 conflicting merges, either in a graphical way, or by inserting some
16 16 conflict markers. Mercurial does not include any interactive merge
17 17 programs but relies on external tools for that.
18 18
19 19 Available merge tools
20 20 =====================
21 21
22 22 External merge tools and their properties are configured in the
23 23 merge-tools configuration section - see hgrc(5) - but they can often just
24 24 be named by their executable.
25 25
26 26 A merge tool is generally usable if its executable can be found on the
27 27 system and if it can handle the merge. The executable is found if it
28 28 is an absolute or relative executable path or the name of an
29 29 application in the executable search path. The tool is assumed to be
30 30 able to handle the merge if it can handle symlinks if the file is a
31 31 symlink, if it can handle binary files if the file is binary, and if a
32 32 GUI is available if the tool requires a GUI.
33 33
34 34 There are some internal merge tools which can be used. The internal
35 35 merge tools are:
36 36
37 37 .. internaltoolsmarker
38 38
39 39 Internal tools are always available and do not require a GUI but will by default
40 40 not handle symlinks or binary files.
41 41
42 42 Choosing a merge tool
43 43 =====================
44 44
45 45 Mercurial uses these rules when deciding which merge tool to use:
46 46
47 47 1. If a tool has been specified with the --tool option to merge or resolve, it
48 48 is used. If it is the name of a tool in the merge-tools configuration, its
49 49 configuration is used. Otherwise the specified tool must be executable by
50 50 the shell.
51 51
52 52 2. If the ``HGMERGE`` environment variable is present, its value is used and
53 53 must be executable by the shell.
54 54
55 55 3. If the filename of the file to be merged matches any of the patterns in the
56 56 merge-patterns configuration section, the first usable merge tool
57 57 corresponding to a matching pattern is used. Here, binary capabilities of the
58 58 merge tool are not considered.
59 59
60 60 4. If ui.merge is set it will be considered next. If the value is not the name
61 61 of a configured tool, the specified value is used and must be executable by
62 62 the shell. Otherwise the named tool is used if it is usable.
63 63
64 64 5. If any usable merge tools are present in the merge-tools configuration
65 65 section, the one with the highest priority is used.
66 66
67 67 6. If a program named ``hgmerge`` can be found on the system, it is used - but
68 68 it will by default not be used for symlinks and binary files.
69 69
70 70 7. If the file to be merged is not binary and is not a symlink, then
71 ``internal:merge`` is used.
71 internal ``:merge`` is used.
72 72
73 73 8. The merge of the file fails and must be resolved before commit.
74 74
75 75 .. note::
76 76
77 77 After selecting a merge program, Mercurial will by default attempt
78 78 to merge the files using a simple merge algorithm first. Only if it doesn't
79 79 succeed because of conflicting changes Mercurial will actually execute the
80 80 merge program. Whether to use the simple merge algorithm first can be
81 81 controlled by the premerge setting of the merge tool. Premerge is enabled by
82 82 default unless the file is binary or a symlink.
83 83
84 84 See the merge-tools and ui sections of hgrc(5) for details on the
85 85 configuration of merge tools.
@@ -1,379 +1,379
1 1 #require symlink execbit
2 2
3 3 $ tellmeabout() {
4 4 > if [ -h $1 ]; then
5 5 > echo $1 is a symlink:
6 6 > $TESTDIR/readlink.py $1
7 7 > elif [ -x $1 ]; then
8 8 > echo $1 is an executable file with content:
9 9 > cat $1
10 10 > else
11 11 > echo $1 is a plain file with content:
12 12 > cat $1
13 13 > fi
14 14 > }
15 15
16 16 $ hg init test1
17 17 $ cd test1
18 18
19 19 $ echo a > a
20 20 $ hg ci -Aqmadd
21 21 $ chmod +x a
22 22 $ hg ci -mexecutable
23 23
24 24 $ hg up -q 0
25 25 $ rm a
26 26 $ ln -s symlink a
27 27 $ hg ci -msymlink
28 28 created new head
29 29
30 30 Symlink is local parent, executable is other:
31 31
32 32 $ hg merge --debug
33 33 searching for copies back to rev 1
34 34 resolving manifests
35 35 branchmerge: True, force: False, partial: False
36 36 ancestor: c334dc3be0da, local: 521a1e40188f+, remote: 3574f3e69b1c
37 37 preserving a for resolve of a
38 38 a: versions differ -> m
39 39 updating: a 1/1 files (100.00%)
40 40 picked tool 'internal:merge' for a (binary False symlink True)
41 41 merging a
42 42 my a@521a1e40188f+ other a@3574f3e69b1c ancestor a@c334dc3be0da
43 43 warning: internal:merge cannot merge symlinks for a
44 44 merging a incomplete! (edit conflicts, then use 'hg resolve --mark')
45 45 0 files updated, 0 files merged, 0 files removed, 1 files unresolved
46 46 use 'hg resolve' to retry unresolved file merges or 'hg update -C .' to abandon
47 47 [1]
48 48
49 49 $ tellmeabout a
50 50 a is a symlink:
51 51 a -> symlink
52 52 $ hg resolve a --tool internal:other
53 53 (no more unresolved files)
54 54 $ tellmeabout a
55 55 a is an executable file with content:
56 56 a
57 57 $ hg st
58 58 M a
59 59 ? a.orig
60 60
61 61 Symlink is other parent, executable is local:
62 62
63 63 $ hg update -C 1
64 64 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
65 65
66 66 $ hg merge --debug
67 67 searching for copies back to rev 1
68 68 resolving manifests
69 69 branchmerge: True, force: False, partial: False
70 70 ancestor: c334dc3be0da, local: 3574f3e69b1c+, remote: 521a1e40188f
71 71 preserving a for resolve of a
72 72 a: versions differ -> m
73 73 updating: a 1/1 files (100.00%)
74 74 picked tool 'internal:merge' for a (binary False symlink True)
75 75 merging a
76 76 my a@3574f3e69b1c+ other a@521a1e40188f ancestor a@c334dc3be0da
77 77 warning: internal:merge cannot merge symlinks for a
78 78 merging a incomplete! (edit conflicts, then use 'hg resolve --mark')
79 79 0 files updated, 0 files merged, 0 files removed, 1 files unresolved
80 80 use 'hg resolve' to retry unresolved file merges or 'hg update -C .' to abandon
81 81 [1]
82 82
83 83 $ tellmeabout a
84 84 a is an executable file with content:
85 85 a
86 86
87 87 Update to link without local change should get us a symlink (issue3316):
88 88
89 89 $ hg up -C 0
90 90 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
91 91 $ hg up
92 92 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
93 93 $ hg st
94 94 ? a.orig
95 95
96 96 Update to link with local change should cause a merge prompt (issue3200):
97 97
98 98 $ hg up -Cq 0
99 99 $ echo data > a
100 100 $ HGMERGE= hg up -y --debug
101 101 searching for copies back to rev 2
102 102 resolving manifests
103 103 branchmerge: False, force: False, partial: False
104 104 ancestor: c334dc3be0da, local: c334dc3be0da+, remote: 521a1e40188f
105 105 preserving a for resolve of a
106 106 a: versions differ -> m
107 107 updating: a 1/1 files (100.00%)
108 108 (couldn't find merge tool hgmerge|tool hgmerge can't handle symlinks) (re)
109 picked tool 'internal:prompt' for a (binary False symlink True)
109 picked tool ':prompt' for a (binary False symlink True)
110 110 no tool found to merge a
111 111 keep (l)ocal or take (o)ther? l
112 112 0 files updated, 1 files merged, 0 files removed, 0 files unresolved
113 113 $ hg diff --git
114 114 diff --git a/a b/a
115 115 old mode 120000
116 116 new mode 100644
117 117 --- a/a
118 118 +++ b/a
119 119 @@ -1,1 +1,1 @@
120 120 -symlink
121 121 \ No newline at end of file
122 122 +data
123 123
124 124
125 125 Test only 'l' change - happens rarely, except when recovering from situations
126 126 where that was what happened.
127 127
128 128 $ hg init test2
129 129 $ cd test2
130 130 $ printf base > f
131 131 $ hg ci -Aqm0
132 132 $ echo file > f
133 133 $ echo content >> f
134 134 $ hg ci -qm1
135 135 $ hg up -qr0
136 136 $ rm f
137 137 $ ln -s base f
138 138 $ hg ci -qm2
139 139 $ hg merge
140 140 merging f
141 141 warning: internal:merge cannot merge symlinks for f
142 142 merging f incomplete! (edit conflicts, then use 'hg resolve --mark')
143 143 0 files updated, 0 files merged, 0 files removed, 1 files unresolved
144 144 use 'hg resolve' to retry unresolved file merges or 'hg update -C .' to abandon
145 145 [1]
146 146 $ tellmeabout f
147 147 f is a symlink:
148 148 f -> base
149 149
150 150 $ hg up -Cqr1
151 151 $ hg merge
152 152 merging f
153 153 warning: internal:merge cannot merge symlinks for f
154 154 merging f incomplete! (edit conflicts, then use 'hg resolve --mark')
155 155 0 files updated, 0 files merged, 0 files removed, 1 files unresolved
156 156 use 'hg resolve' to retry unresolved file merges or 'hg update -C .' to abandon
157 157 [1]
158 158 $ tellmeabout f
159 159 f is a plain file with content:
160 160 file
161 161 content
162 162
163 163 $ cd ..
164 164
165 165 Test removed 'x' flag merged with change to symlink
166 166
167 167 $ hg init test3
168 168 $ cd test3
169 169 $ echo f > f
170 170 $ chmod +x f
171 171 $ hg ci -Aqm0
172 172 $ chmod -x f
173 173 $ hg ci -qm1
174 174 $ hg up -qr0
175 175 $ rm f
176 176 $ ln -s dangling f
177 177 $ hg ci -qm2
178 178 $ hg merge
179 179 merging f
180 180 warning: internal:merge cannot merge symlinks for f
181 181 merging f incomplete! (edit conflicts, then use 'hg resolve --mark')
182 182 0 files updated, 0 files merged, 0 files removed, 1 files unresolved
183 183 use 'hg resolve' to retry unresolved file merges or 'hg update -C .' to abandon
184 184 [1]
185 185 $ tellmeabout f
186 186 f is a symlink:
187 187 f -> dangling
188 188
189 189 $ hg up -Cqr1
190 190 $ hg merge
191 191 merging f
192 192 warning: internal:merge cannot merge symlinks for f
193 193 merging f incomplete! (edit conflicts, then use 'hg resolve --mark')
194 194 0 files updated, 0 files merged, 0 files removed, 1 files unresolved
195 195 use 'hg resolve' to retry unresolved file merges or 'hg update -C .' to abandon
196 196 [1]
197 197 $ tellmeabout f
198 198 f is a plain file with content:
199 199 f
200 200
201 201 Test removed 'x' flag merged with content change - both ways
202 202
203 203 $ hg up -Cqr0
204 204 $ echo change > f
205 205 $ hg ci -qm3
206 206 $ hg merge -r1
207 207 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
208 208 (branch merge, don't forget to commit)
209 209 $ tellmeabout f
210 210 f is a plain file with content:
211 211 change
212 212
213 213 $ hg up -qCr1
214 214 $ hg merge -r3
215 215 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
216 216 (branch merge, don't forget to commit)
217 217 $ tellmeabout f
218 218 f is a plain file with content:
219 219 change
220 220
221 221 $ cd ..
222 222
223 223 Test merge with no common ancestor:
224 224 a: just different
225 225 b: x vs -, different (cannot calculate x, cannot ask merge tool)
226 226 c: x vs -, same (cannot calculate x, merge tool is no good)
227 227 d: x vs l, different
228 228 e: x vs l, same
229 229 f: - vs l, different
230 230 g: - vs l, same
231 231 h: l vs l, different
232 232 (where same means the filelog entry is shared and there thus is an ancestor!)
233 233
234 234 $ hg init test4
235 235 $ cd test4
236 236 $ echo 0 > 0
237 237 $ hg ci -Aqm0
238 238
239 239 $ echo 1 > a
240 240 $ echo 1 > b
241 241 $ chmod +x b
242 242 $ echo x > c
243 243 $ chmod +x c
244 244 $ echo 1 > d
245 245 $ chmod +x d
246 246 $ printf x > e
247 247 $ chmod +x e
248 248 $ echo 1 > f
249 249 $ printf x > g
250 250 $ ln -s 1 h
251 251 $ hg ci -qAm1
252 252
253 253 $ hg up -qr0
254 254 $ echo 2 > a
255 255 $ echo 2 > b
256 256 $ echo x > c
257 257 $ ln -s 2 d
258 258 $ ln -s x e
259 259 $ ln -s 2 f
260 260 $ ln -s x g
261 261 $ ln -s 2 h
262 262 $ hg ci -Aqm2
263 263
264 264 $ hg merge
265 265 merging a
266 266 warning: conflicts during merge.
267 267 merging a incomplete! (edit conflicts, then use 'hg resolve --mark')
268 268 warning: cannot merge flags for b
269 269 merging b
270 270 warning: conflicts during merge.
271 271 merging b incomplete! (edit conflicts, then use 'hg resolve --mark')
272 272 warning: cannot merge flags for c
273 273 merging d
274 274 warning: internal:merge cannot merge symlinks for d
275 275 merging d incomplete! (edit conflicts, then use 'hg resolve --mark')
276 276 merging f
277 277 warning: internal:merge cannot merge symlinks for f
278 278 merging f incomplete! (edit conflicts, then use 'hg resolve --mark')
279 279 merging h
280 280 warning: internal:merge cannot merge symlinks for h
281 281 merging h incomplete! (edit conflicts, then use 'hg resolve --mark')
282 282 3 files updated, 0 files merged, 0 files removed, 5 files unresolved
283 283 use 'hg resolve' to retry unresolved file merges or 'hg update -C .' to abandon
284 284 [1]
285 285 $ hg resolve -l
286 286 U a
287 287 U b
288 288 U d
289 289 U f
290 290 U h
291 291 $ tellmeabout a
292 292 a is a plain file with content:
293 293 <<<<<<< local: 0139c5610547 - test: 2
294 294 2
295 295 =======
296 296 1
297 297 >>>>>>> other: 97e29675e796 - test: 1
298 298 $ tellmeabout b
299 299 b is a plain file with content:
300 300 <<<<<<< local: 0139c5610547 - test: 2
301 301 2
302 302 =======
303 303 1
304 304 >>>>>>> other: 97e29675e796 - test: 1
305 305 $ tellmeabout c
306 306 c is a plain file with content:
307 307 x
308 308 $ tellmeabout d
309 309 d is a symlink:
310 310 d -> 2
311 311 $ tellmeabout e
312 312 e is a symlink:
313 313 e -> x
314 314 $ tellmeabout f
315 315 f is a symlink:
316 316 f -> 2
317 317 $ tellmeabout g
318 318 g is a symlink:
319 319 g -> x
320 320 $ tellmeabout h
321 321 h is a symlink:
322 322 h -> 2
323 323
324 324 $ hg up -Cqr1
325 325 $ hg merge
326 326 merging a
327 327 warning: conflicts during merge.
328 328 merging a incomplete! (edit conflicts, then use 'hg resolve --mark')
329 329 warning: cannot merge flags for b
330 330 merging b
331 331 warning: conflicts during merge.
332 332 merging b incomplete! (edit conflicts, then use 'hg resolve --mark')
333 333 warning: cannot merge flags for c
334 334 merging d
335 335 warning: internal:merge cannot merge symlinks for d
336 336 merging d incomplete! (edit conflicts, then use 'hg resolve --mark')
337 337 merging f
338 338 warning: internal:merge cannot merge symlinks for f
339 339 merging f incomplete! (edit conflicts, then use 'hg resolve --mark')
340 340 merging h
341 341 warning: internal:merge cannot merge symlinks for h
342 342 merging h incomplete! (edit conflicts, then use 'hg resolve --mark')
343 343 3 files updated, 0 files merged, 0 files removed, 5 files unresolved
344 344 use 'hg resolve' to retry unresolved file merges or 'hg update -C .' to abandon
345 345 [1]
346 346 $ tellmeabout a
347 347 a is a plain file with content:
348 348 <<<<<<< local: 97e29675e796 - test: 1
349 349 1
350 350 =======
351 351 2
352 352 >>>>>>> other: 0139c5610547 - test: 2
353 353 $ tellmeabout b
354 354 b is an executable file with content:
355 355 <<<<<<< local: 97e29675e796 - test: 1
356 356 1
357 357 =======
358 358 2
359 359 >>>>>>> other: 0139c5610547 - test: 2
360 360 $ tellmeabout c
361 361 c is an executable file with content:
362 362 x
363 363 $ tellmeabout d
364 364 d is an executable file with content:
365 365 1
366 366 $ tellmeabout e
367 367 e is an executable file with content:
368 368 x (no-eol)
369 369 $ tellmeabout f
370 370 f is a plain file with content:
371 371 1
372 372 $ tellmeabout g
373 373 g is a plain file with content:
374 374 x (no-eol)
375 375 $ tellmeabout h
376 376 h is a symlink:
377 377 h -> 1
378 378
379 379 $ cd ..
@@ -1,606 +1,606
1 1 $ hg init test
2 2 $ cd test
3 3
4 4 $ echo a > a
5 5 $ hg add a
6 6 $ hg commit -m "test"
7 7 $ hg history
8 8 changeset: 0:acb14030fe0a
9 9 tag: tip
10 10 user: test
11 11 date: Thu Jan 01 00:00:00 1970 +0000
12 12 summary: test
13 13
14 14
15 15 $ hg tag ' '
16 16 abort: tag names cannot consist entirely of whitespace
17 17 [255]
18 18
19 19 (this tests also that editor is not invoked, if '--edit' is not
20 20 specified)
21 21
22 22 $ HGEDITOR=cat hg tag "bleah"
23 23 $ hg history
24 24 changeset: 1:d4f0d2909abc
25 25 tag: tip
26 26 user: test
27 27 date: Thu Jan 01 00:00:00 1970 +0000
28 28 summary: Added tag bleah for changeset acb14030fe0a
29 29
30 30 changeset: 0:acb14030fe0a
31 31 tag: bleah
32 32 user: test
33 33 date: Thu Jan 01 00:00:00 1970 +0000
34 34 summary: test
35 35
36 36
37 37 $ echo foo >> .hgtags
38 38 $ hg tag "bleah2"
39 39 abort: working copy of .hgtags is changed
40 40 (please commit .hgtags manually)
41 41 [255]
42 42
43 43 $ hg revert .hgtags
44 44 $ hg tag -r 0 x y z y y z
45 45 abort: tag names must be unique
46 46 [255]
47 47 $ hg tag tap nada dot tip
48 48 abort: the name 'tip' is reserved
49 49 [255]
50 50 $ hg tag .
51 51 abort: the name '.' is reserved
52 52 [255]
53 53 $ hg tag null
54 54 abort: the name 'null' is reserved
55 55 [255]
56 56 $ hg tag "bleah"
57 57 abort: tag 'bleah' already exists (use -f to force)
58 58 [255]
59 59 $ hg tag "blecch" "bleah"
60 60 abort: tag 'bleah' already exists (use -f to force)
61 61 [255]
62 62
63 63 $ hg tag --remove "blecch"
64 64 abort: tag 'blecch' does not exist
65 65 [255]
66 66 $ hg tag --remove "bleah" "blecch" "blough"
67 67 abort: tag 'blecch' does not exist
68 68 [255]
69 69
70 70 $ hg tag -r 0 "bleah0"
71 71 $ hg tag -l -r 1 "bleah1"
72 72 $ hg tag gack gawk gorp
73 73 $ hg tag -f gack
74 74 $ hg tag --remove gack gorp
75 75
76 76 $ hg tag "bleah "
77 77 abort: tag 'bleah' already exists (use -f to force)
78 78 [255]
79 79 $ hg tag " bleah"
80 80 abort: tag 'bleah' already exists (use -f to force)
81 81 [255]
82 82 $ hg tag " bleah"
83 83 abort: tag 'bleah' already exists (use -f to force)
84 84 [255]
85 85 $ hg tag -r 0 " bleahbleah "
86 86 $ hg tag -r 0 " bleah bleah "
87 87
88 88 $ cat .hgtags
89 89 acb14030fe0a21b60322c440ad2d20cf7685a376 bleah
90 90 acb14030fe0a21b60322c440ad2d20cf7685a376 bleah0
91 91 336fccc858a4eb69609a291105009e484a6b6b8d gack
92 92 336fccc858a4eb69609a291105009e484a6b6b8d gawk
93 93 336fccc858a4eb69609a291105009e484a6b6b8d gorp
94 94 336fccc858a4eb69609a291105009e484a6b6b8d gack
95 95 799667b6f2d9b957f73fa644a918c2df22bab58f gack
96 96 799667b6f2d9b957f73fa644a918c2df22bab58f gack
97 97 0000000000000000000000000000000000000000 gack
98 98 336fccc858a4eb69609a291105009e484a6b6b8d gorp
99 99 0000000000000000000000000000000000000000 gorp
100 100 acb14030fe0a21b60322c440ad2d20cf7685a376 bleahbleah
101 101 acb14030fe0a21b60322c440ad2d20cf7685a376 bleah bleah
102 102
103 103 $ cat .hg/localtags
104 104 d4f0d2909abc9290e2773c08837d70c1794e3f5a bleah1
105 105
106 106 tagging on a non-head revision
107 107
108 108 $ hg update 0
109 109 0 files updated, 0 files merged, 1 files removed, 0 files unresolved
110 110 $ hg tag -l localblah
111 111 $ hg tag "foobar"
112 112 abort: not at a branch head (use -f to force)
113 113 [255]
114 114 $ hg tag -f "foobar"
115 115 $ cat .hgtags
116 116 acb14030fe0a21b60322c440ad2d20cf7685a376 foobar
117 117 $ cat .hg/localtags
118 118 d4f0d2909abc9290e2773c08837d70c1794e3f5a bleah1
119 119 acb14030fe0a21b60322c440ad2d20cf7685a376 localblah
120 120
121 121 $ hg tag -l 'xx
122 122 > newline'
123 123 abort: '\n' cannot be used in a name
124 124 [255]
125 125 $ hg tag -l 'xx:xx'
126 126 abort: ':' cannot be used in a name
127 127 [255]
128 128
129 129 cloning local tags
130 130
131 131 $ cd ..
132 132 $ hg -R test log -r0:5
133 133 changeset: 0:acb14030fe0a
134 134 tag: bleah
135 135 tag: bleah bleah
136 136 tag: bleah0
137 137 tag: bleahbleah
138 138 tag: foobar
139 139 tag: localblah
140 140 user: test
141 141 date: Thu Jan 01 00:00:00 1970 +0000
142 142 summary: test
143 143
144 144 changeset: 1:d4f0d2909abc
145 145 tag: bleah1
146 146 user: test
147 147 date: Thu Jan 01 00:00:00 1970 +0000
148 148 summary: Added tag bleah for changeset acb14030fe0a
149 149
150 150 changeset: 2:336fccc858a4
151 151 tag: gawk
152 152 user: test
153 153 date: Thu Jan 01 00:00:00 1970 +0000
154 154 summary: Added tag bleah0 for changeset acb14030fe0a
155 155
156 156 changeset: 3:799667b6f2d9
157 157 user: test
158 158 date: Thu Jan 01 00:00:00 1970 +0000
159 159 summary: Added tag gack, gawk, gorp for changeset 336fccc858a4
160 160
161 161 changeset: 4:154eeb7c0138
162 162 user: test
163 163 date: Thu Jan 01 00:00:00 1970 +0000
164 164 summary: Added tag gack for changeset 799667b6f2d9
165 165
166 166 changeset: 5:b4bb47aaff09
167 167 user: test
168 168 date: Thu Jan 01 00:00:00 1970 +0000
169 169 summary: Removed tag gack, gorp
170 170
171 171 $ hg clone -q -rbleah1 test test1
172 172 $ hg -R test1 parents --style=compact
173 173 1[tip] d4f0d2909abc 1970-01-01 00:00 +0000 test
174 174 Added tag bleah for changeset acb14030fe0a
175 175
176 176 $ hg clone -q -r5 test#bleah1 test2
177 177 $ hg -R test2 parents --style=compact
178 178 5[tip] b4bb47aaff09 1970-01-01 00:00 +0000 test
179 179 Removed tag gack, gorp
180 180
181 181 $ hg clone -q -U test#bleah1 test3
182 182 $ hg -R test3 parents --style=compact
183 183
184 184 $ cd test
185 185
186 186 Issue601: hg tag doesn't do the right thing if .hgtags or localtags
187 187 doesn't end with EOL
188 188
189 189 $ python << EOF
190 190 > f = file('.hg/localtags'); last = f.readlines()[-1][:-1]; f.close()
191 191 > f = file('.hg/localtags', 'w'); f.write(last); f.close()
192 192 > EOF
193 193 $ cat .hg/localtags; echo
194 194 acb14030fe0a21b60322c440ad2d20cf7685a376 localblah
195 195 $ hg tag -l localnewline
196 196 $ cat .hg/localtags; echo
197 197 acb14030fe0a21b60322c440ad2d20cf7685a376 localblah
198 198 c2899151f4e76890c602a2597a650a72666681bf localnewline
199 199
200 200
201 201 $ python << EOF
202 202 > f = file('.hgtags'); last = f.readlines()[-1][:-1]; f.close()
203 203 > f = file('.hgtags', 'w'); f.write(last); f.close()
204 204 > EOF
205 205 $ hg ci -m'broken manual edit of .hgtags'
206 206 $ cat .hgtags; echo
207 207 acb14030fe0a21b60322c440ad2d20cf7685a376 foobar
208 208 $ hg tag newline
209 209 $ cat .hgtags; echo
210 210 acb14030fe0a21b60322c440ad2d20cf7685a376 foobar
211 211 a0eea09de1eeec777b46f2085260a373b2fbc293 newline
212 212
213 213
214 214 tag and branch using same name
215 215
216 216 $ hg branch tag-and-branch-same-name
217 217 marked working directory as branch tag-and-branch-same-name
218 218 (branches are permanent and global, did you want a bookmark?)
219 219 $ hg ci -m"discouraged"
220 220 $ hg tag tag-and-branch-same-name
221 221 warning: tag tag-and-branch-same-name conflicts with existing branch name
222 222
223 223 test custom commit messages
224 224
225 225 $ cat > editor.sh << '__EOF__'
226 226 > echo "==== before editing"
227 227 > cat "$1"
228 228 > echo "===="
229 229 > echo "custom tag message" > "$1"
230 230 > echo "second line" >> "$1"
231 231 > __EOF__
232 232
233 233 at first, test saving last-message.txt
234 234
235 235 (test that editor is not invoked before transaction starting)
236 236
237 237 $ cat > .hg/hgrc << '__EOF__'
238 238 > [hooks]
239 239 > # this failure occurs before editor invocation
240 240 > pretag.test-saving-lastmessage = false
241 241 > __EOF__
242 242 $ rm -f .hg/last-message.txt
243 243 $ HGEDITOR="\"sh\" \"`pwd`/editor.sh\"" hg tag custom-tag -e
244 244 abort: pretag.test-saving-lastmessage hook exited with status 1
245 245 [255]
246 246 $ test -f .hg/last-message.txt
247 247 [1]
248 248
249 249 (test that editor is invoked and commit message is saved into
250 250 "last-message.txt")
251 251
252 252 $ cat >> .hg/hgrc << '__EOF__'
253 253 > [hooks]
254 254 > pretag.test-saving-lastmessage =
255 255 > # this failure occurs after editor invocation
256 256 > pretxncommit.unexpectedabort = false
257 257 > __EOF__
258 258
259 259 (this tests also that editor is invoked, if '--edit' is specified,
260 260 regardless of '--message')
261 261
262 262 $ rm -f .hg/last-message.txt
263 263 $ HGEDITOR="\"sh\" \"`pwd`/editor.sh\"" hg tag custom-tag -e -m "foo bar"
264 264 ==== before editing
265 265 foo bar
266 266
267 267
268 268 HG: Enter commit message. Lines beginning with 'HG:' are removed.
269 269 HG: Leave message empty to abort commit.
270 270 HG: --
271 271 HG: user: test
272 272 HG: branch 'tag-and-branch-same-name'
273 273 HG: changed .hgtags
274 274 ====
275 275 transaction abort!
276 276 rollback completed
277 277 note: commit message saved in .hg/last-message.txt
278 278 abort: pretxncommit.unexpectedabort hook exited with status 1
279 279 [255]
280 280 $ cat .hg/last-message.txt
281 281 custom tag message
282 282 second line
283 283
284 284 $ cat >> .hg/hgrc << '__EOF__'
285 285 > [hooks]
286 286 > pretxncommit.unexpectedabort =
287 287 > __EOF__
288 288 $ hg status .hgtags
289 289 M .hgtags
290 290 $ hg revert --no-backup -q .hgtags
291 291
292 292 then, test custom commit message itself
293 293
294 294 $ HGEDITOR="\"sh\" \"`pwd`/editor.sh\"" hg tag custom-tag -e
295 295 ==== before editing
296 296 Added tag custom-tag for changeset 75a534207be6
297 297
298 298
299 299 HG: Enter commit message. Lines beginning with 'HG:' are removed.
300 300 HG: Leave message empty to abort commit.
301 301 HG: --
302 302 HG: user: test
303 303 HG: branch 'tag-and-branch-same-name'
304 304 HG: changed .hgtags
305 305 ====
306 306 $ hg log -l1 --template "{desc}\n"
307 307 custom tag message
308 308 second line
309 309
310 310
311 311 local tag with .hgtags modified
312 312
313 313 $ hg tag hgtags-modified
314 314 $ hg rollback
315 315 repository tip rolled back to revision 13 (undo commit)
316 316 working directory now based on revision 13
317 317 $ hg st
318 318 M .hgtags
319 319 ? .hgtags.orig
320 320 ? editor.sh
321 321 $ hg tag --local baz
322 322 $ hg revert --no-backup .hgtags
323 323
324 324
325 325 tagging when at named-branch-head that's not a topo-head
326 326
327 327 $ hg up default
328 328 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
329 329 $ hg merge -t internal:local
330 330 0 files updated, 1 files merged, 0 files removed, 0 files unresolved
331 331 (branch merge, don't forget to commit)
332 332 $ hg ci -m 'merge named branch'
333 333 $ hg up 13
334 334 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
335 335 $ hg tag new-topo-head
336 336
337 337 tagging on null rev
338 338
339 339 $ hg up null
340 340 0 files updated, 0 files merged, 2 files removed, 0 files unresolved
341 341 $ hg tag nullrev
342 342 abort: not at a branch head (use -f to force)
343 343 [255]
344 344
345 345 $ hg init empty
346 346 $ hg tag -R empty nullrev
347 347 abort: cannot tag null revision
348 348 [255]
349 349
350 350 $ hg tag -R empty -r 00000000000 -f nulltag
351 351 abort: cannot tag null revision
352 352 [255]
353 353
354 354 $ cd ..
355 355
356 356 tagging on an uncommitted merge (issue2542)
357 357
358 358 $ hg init repo-tag-uncommitted-merge
359 359 $ cd repo-tag-uncommitted-merge
360 360 $ echo c1 > f1
361 361 $ hg ci -Am0
362 362 adding f1
363 363 $ echo c2 > f2
364 364 $ hg ci -Am1
365 365 adding f2
366 366 $ hg co -q 0
367 367 $ hg branch b1
368 368 marked working directory as branch b1
369 369 (branches are permanent and global, did you want a bookmark?)
370 370 $ hg ci -m2
371 371 $ hg up default
372 372 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
373 373 $ hg merge b1
374 374 0 files updated, 0 files merged, 0 files removed, 0 files unresolved
375 375 (branch merge, don't forget to commit)
376 376
377 377 $ hg tag t1
378 378 abort: uncommitted merge
379 379 [255]
380 380 $ hg status
381 381 $ hg tag --rev 1 t2
382 382 abort: uncommitted merge
383 383 [255]
384 384 $ hg tag --rev 1 --local t3
385 385 $ hg tags -v
386 386 tip 2:2a156e8887cc
387 387 t3 1:c3adabd1a5f4 local
388 388
389 389 $ cd ..
390 390
391 391 commit hook on tag used to be run without write lock - issue3344
392 392
393 393 $ hg init repo-tag
394 394 $ touch repo-tag/test
395 395 $ hg -R repo-tag commit -A -m "test"
396 396 adding test
397 397 $ hg init repo-tag-target
398 398 $ hg -R repo-tag --config hooks.commit="\"hg\" push \"`pwd`/repo-tag-target\"" tag tag
399 399 pushing to $TESTTMP/repo-tag-target (glob)
400 400 searching for changes
401 401 adding changesets
402 402 adding manifests
403 403 adding file changes
404 404 added 2 changesets with 2 changes to 2 files
405 405
406 406 automatically merge resolvable tag conflicts (i.e. tags that differ in rank)
407 407 create two clones with some different tags as well as some common tags
408 408 check that we can merge tags that differ in rank
409 409
410 410 $ hg init repo-automatic-tag-merge
411 411 $ cd repo-automatic-tag-merge
412 412 $ echo c0 > f0
413 413 $ hg ci -A -m0
414 414 adding f0
415 415 $ hg tag tbase
416 416 $ cd ..
417 417 $ hg clone repo-automatic-tag-merge repo-automatic-tag-merge-clone
418 418 updating to branch default
419 419 2 files updated, 0 files merged, 0 files removed, 0 files unresolved
420 420 $ cd repo-automatic-tag-merge-clone
421 421 $ echo c1 > f1
422 422 $ hg ci -A -m1
423 423 adding f1
424 424 $ hg tag t1 t2 t3
425 425 $ hg tag --remove t2
426 426 $ hg tag t5
427 427 $ echo c2 > f2
428 428 $ hg ci -A -m2
429 429 adding f2
430 430 $ hg tag -f t3
431 431
432 432 $ cd ../repo-automatic-tag-merge
433 433 $ echo c3 > f3
434 434 $ hg ci -A -m3
435 435 adding f3
436 436 $ hg tag -f t4 t5 t6
437 437 $ hg tag --remove t5
438 438 $ echo c4 > f4
439 439 $ hg ci -A -m4
440 440 adding f4
441 441 $ hg tag t2
442 442 $ hg tag -f t6
443 443
444 444 $ cd ../repo-automatic-tag-merge-clone
445 445 $ hg pull
446 446 pulling from $TESTTMP/repo-automatic-tag-merge (glob)
447 447 searching for changes
448 448 adding changesets
449 449 adding manifests
450 450 adding file changes
451 451 added 6 changesets with 6 changes to 3 files (+1 heads)
452 452 (run 'hg heads' to see heads, 'hg merge' to merge)
453 453 $ hg merge --tool internal:tagmerge
454 454 merging .hgtags
455 455 2 files updated, 1 files merged, 0 files removed, 0 files unresolved
456 456 (branch merge, don't forget to commit)
457 457 $ hg status
458 458 M .hgtags
459 459 M f3
460 460 M f4
461 461 $ hg resolve -l
462 462 R .hgtags
463 463 $ cat .hgtags
464 464 9aa4e1292a27a248f8d07339bed9931d54907be7 t4
465 465 9aa4e1292a27a248f8d07339bed9931d54907be7 t6
466 466 9aa4e1292a27a248f8d07339bed9931d54907be7 t6
467 467 09af2ce14077a94effef208b49a718f4836d4338 t6
468 468 6cee5c8f3e5b4ae1a3996d2f6489c3e08eb5aea7 tbase
469 469 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t1
470 470 929bca7b18d067cbf3844c3896319a940059d748 t2
471 471 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t2
472 472 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t3
473 473 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t2
474 474 0000000000000000000000000000000000000000 t2
475 475 875517b4806a848f942811a315a5bce30804ae85 t5
476 476 9aa4e1292a27a248f8d07339bed9931d54907be7 t5
477 477 9aa4e1292a27a248f8d07339bed9931d54907be7 t5
478 478 0000000000000000000000000000000000000000 t5
479 479 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t3
480 480 79505d5360b07e3e79d1052e347e73c02b8afa5b t3
481 481
482 482 check that the merge tried to minimize the diff witht he first merge parent
483 483
484 484 $ hg diff --git -r 'p1()' .hgtags
485 485 diff --git a/.hgtags b/.hgtags
486 486 --- a/.hgtags
487 487 +++ b/.hgtags
488 488 @@ -1,9 +1,17 @@
489 489 +9aa4e1292a27a248f8d07339bed9931d54907be7 t4
490 490 +9aa4e1292a27a248f8d07339bed9931d54907be7 t6
491 491 +9aa4e1292a27a248f8d07339bed9931d54907be7 t6
492 492 +09af2ce14077a94effef208b49a718f4836d4338 t6
493 493 6cee5c8f3e5b4ae1a3996d2f6489c3e08eb5aea7 tbase
494 494 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t1
495 495 +929bca7b18d067cbf3844c3896319a940059d748 t2
496 496 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t2
497 497 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t3
498 498 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t2
499 499 0000000000000000000000000000000000000000 t2
500 500 875517b4806a848f942811a315a5bce30804ae85 t5
501 501 +9aa4e1292a27a248f8d07339bed9931d54907be7 t5
502 502 +9aa4e1292a27a248f8d07339bed9931d54907be7 t5
503 503 +0000000000000000000000000000000000000000 t5
504 504 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t3
505 505 79505d5360b07e3e79d1052e347e73c02b8afa5b t3
506 506
507 507 detect merge tag conflicts
508 508
509 509 $ hg update -C -r tip
510 510 3 files updated, 0 files merged, 2 files removed, 0 files unresolved
511 511 $ hg tag t7
512 512 $ hg update -C -r 'first(sort(head()))'
513 513 3 files updated, 0 files merged, 2 files removed, 0 files unresolved
514 514 $ printf "%s %s\n" `hg log -r . --template "{node} t7"` >> .hgtags
515 515 $ hg commit -m "manually add conflicting t7 tag"
516 516 $ hg merge --tool internal:tagmerge
517 517 merging .hgtags
518 518 automatic .hgtags merge failed
519 519 the following 1 tags are in conflict: t7
520 automatic tag merging of .hgtags failed! (use 'hg resolve --tool internal:merge' or another merge tool of your choice)
520 automatic tag merging of .hgtags failed! (use 'hg resolve --tool :merge' or another merge tool of your choice)
521 521 2 files updated, 0 files merged, 0 files removed, 1 files unresolved
522 522 use 'hg resolve' to retry unresolved file merges or 'hg update -C .' to abandon
523 523 [1]
524 524 $ hg resolve -l
525 525 U .hgtags
526 526 $ cat .hgtags
527 527 6cee5c8f3e5b4ae1a3996d2f6489c3e08eb5aea7 tbase
528 528 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t1
529 529 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t2
530 530 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t3
531 531 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t2
532 532 0000000000000000000000000000000000000000 t2
533 533 875517b4806a848f942811a315a5bce30804ae85 t5
534 534 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t3
535 535 79505d5360b07e3e79d1052e347e73c02b8afa5b t3
536 536 ea918d56be86a4afc5a95312e8b6750e1428d9d2 t7
537 537
538 538 $ cd ..
539 539
540 540 handle the loss of tags
541 541
542 542 $ hg clone repo-automatic-tag-merge-clone repo-merge-lost-tags
543 543 updating to branch default
544 544 4 files updated, 0 files merged, 0 files removed, 0 files unresolved
545 545 $ cd repo-merge-lost-tags
546 546 $ echo c5 > f5
547 547 $ hg ci -A -m5
548 548 adding f5
549 549 $ hg tag -f t7
550 550 $ hg update -r 'p1(t7)'
551 551 1 files updated, 0 files merged, 1 files removed, 0 files unresolved
552 552 $ printf '' > .hgtags
553 553 $ hg commit -m 'delete all tags'
554 554 created new head
555 555 $ hg update -r 'max(t7::)'
556 556 2 files updated, 0 files merged, 0 files removed, 0 files unresolved
557 557 $ hg merge -r tip --tool internal:tagmerge
558 558 merging .hgtags
559 559 0 files updated, 1 files merged, 0 files removed, 0 files unresolved
560 560 (branch merge, don't forget to commit)
561 561 $ hg resolve -l
562 562 R .hgtags
563 563 $ cat .hgtags
564 564 6cee5c8f3e5b4ae1a3996d2f6489c3e08eb5aea7 tbase
565 565 0000000000000000000000000000000000000000 tbase
566 566 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t1
567 567 0000000000000000000000000000000000000000 t1
568 568 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t2
569 569 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t3
570 570 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t2
571 571 0000000000000000000000000000000000000000 t2
572 572 875517b4806a848f942811a315a5bce30804ae85 t5
573 573 0000000000000000000000000000000000000000 t5
574 574 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t3
575 575 79505d5360b07e3e79d1052e347e73c02b8afa5b t3
576 576 0000000000000000000000000000000000000000 t3
577 577 ea918d56be86a4afc5a95312e8b6750e1428d9d2 t7
578 578 0000000000000000000000000000000000000000 t7
579 579 ea918d56be86a4afc5a95312e8b6750e1428d9d2 t7
580 580 fd3a9e394ce3afb354a496323bf68ac1755a30de t7
581 581
582 582 also check that we minimize the diff with the 1st merge parent
583 583
584 584 $ hg diff --git -r 'p1()' .hgtags
585 585 diff --git a/.hgtags b/.hgtags
586 586 --- a/.hgtags
587 587 +++ b/.hgtags
588 588 @@ -1,12 +1,17 @@
589 589 6cee5c8f3e5b4ae1a3996d2f6489c3e08eb5aea7 tbase
590 590 +0000000000000000000000000000000000000000 tbase
591 591 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t1
592 592 +0000000000000000000000000000000000000000 t1
593 593 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t2
594 594 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t3
595 595 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t2
596 596 0000000000000000000000000000000000000000 t2
597 597 875517b4806a848f942811a315a5bce30804ae85 t5
598 598 +0000000000000000000000000000000000000000 t5
599 599 4f3e9b90005b68b4d8a3f4355cedc302a8364f5c t3
600 600 79505d5360b07e3e79d1052e347e73c02b8afa5b t3
601 601 +0000000000000000000000000000000000000000 t3
602 602 ea918d56be86a4afc5a95312e8b6750e1428d9d2 t7
603 603 +0000000000000000000000000000000000000000 t7
604 604 ea918d56be86a4afc5a95312e8b6750e1428d9d2 t7
605 605 fd3a9e394ce3afb354a496323bf68ac1755a30de t7
606 606
General Comments 0
You need to be logged in to leave comments. Login now