##// END OF EJS Templates
strip: ignore -n (issue3326) (BC)...
Matt Mackall -
r16278:900eee07 stable
parent child Browse files
Show More
@@ -1,3406 +1,3407
1 1 # mq.py - patch queues for mercurial
2 2 #
3 3 # Copyright 2005, 2006 Chris Mason <mason@suse.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 '''manage a stack of patches
9 9
10 10 This extension lets you work with a stack of patches in a Mercurial
11 11 repository. It manages two stacks of patches - all known patches, and
12 12 applied patches (subset of known patches).
13 13
14 14 Known patches are represented as patch files in the .hg/patches
15 15 directory. Applied patches are both patch files and changesets.
16 16
17 17 Common tasks (use :hg:`help command` for more details)::
18 18
19 19 create new patch qnew
20 20 import existing patch qimport
21 21
22 22 print patch series qseries
23 23 print applied patches qapplied
24 24
25 25 add known patch to applied stack qpush
26 26 remove patch from applied stack qpop
27 27 refresh contents of top applied patch qrefresh
28 28
29 29 By default, mq will automatically use git patches when required to
30 30 avoid losing file mode changes, copy records, binary files or empty
31 31 files creations or deletions. This behaviour can be configured with::
32 32
33 33 [mq]
34 34 git = auto/keep/yes/no
35 35
36 36 If set to 'keep', mq will obey the [diff] section configuration while
37 37 preserving existing git patches upon qrefresh. If set to 'yes' or
38 38 'no', mq will override the [diff] section and always generate git or
39 39 regular patches, possibly losing data in the second case.
40 40
41 41 It may be desirable for mq changesets to be kept in the secret phase (see
42 42 :hg:`help phases`), which can be enabled with the following setting::
43 43
44 44 [mq]
45 45 secret = True
46 46
47 47 You will by default be managing a patch queue named "patches". You can
48 48 create other, independent patch queues with the :hg:`qqueue` command.
49 49 '''
50 50
51 51 from mercurial.i18n import _
52 52 from mercurial.node import bin, hex, short, nullid, nullrev
53 53 from mercurial.lock import release
54 54 from mercurial import commands, cmdutil, hg, scmutil, util, revset
55 55 from mercurial import repair, extensions, url, error, phases
56 56 from mercurial import patch as patchmod
57 57 import os, re, errno, shutil
58 58
59 59 commands.norepo += " qclone"
60 60
61 61 seriesopts = [('s', 'summary', None, _('print first line of patch header'))]
62 62
63 63 cmdtable = {}
64 64 command = cmdutil.command(cmdtable)
65 65
66 66 # Patch names looks like unix-file names.
67 67 # They must be joinable with queue directory and result in the patch path.
68 68 normname = util.normpath
69 69
70 70 class statusentry(object):
71 71 def __init__(self, node, name):
72 72 self.node, self.name = node, name
73 73 def __repr__(self):
74 74 return hex(self.node) + ':' + self.name
75 75
76 76 class patchheader(object):
77 77 def __init__(self, pf, plainmode=False):
78 78 def eatdiff(lines):
79 79 while lines:
80 80 l = lines[-1]
81 81 if (l.startswith("diff -") or
82 82 l.startswith("Index:") or
83 83 l.startswith("===========")):
84 84 del lines[-1]
85 85 else:
86 86 break
87 87 def eatempty(lines):
88 88 while lines:
89 89 if not lines[-1].strip():
90 90 del lines[-1]
91 91 else:
92 92 break
93 93
94 94 message = []
95 95 comments = []
96 96 user = None
97 97 date = None
98 98 parent = None
99 99 format = None
100 100 subject = None
101 101 branch = None
102 102 nodeid = None
103 103 diffstart = 0
104 104
105 105 for line in file(pf):
106 106 line = line.rstrip()
107 107 if (line.startswith('diff --git')
108 108 or (diffstart and line.startswith('+++ '))):
109 109 diffstart = 2
110 110 break
111 111 diffstart = 0 # reset
112 112 if line.startswith("--- "):
113 113 diffstart = 1
114 114 continue
115 115 elif format == "hgpatch":
116 116 # parse values when importing the result of an hg export
117 117 if line.startswith("# User "):
118 118 user = line[7:]
119 119 elif line.startswith("# Date "):
120 120 date = line[7:]
121 121 elif line.startswith("# Parent "):
122 122 parent = line[9:].lstrip()
123 123 elif line.startswith("# Branch "):
124 124 branch = line[9:]
125 125 elif line.startswith("# Node ID "):
126 126 nodeid = line[10:]
127 127 elif not line.startswith("# ") and line:
128 128 message.append(line)
129 129 format = None
130 130 elif line == '# HG changeset patch':
131 131 message = []
132 132 format = "hgpatch"
133 133 elif (format != "tagdone" and (line.startswith("Subject: ") or
134 134 line.startswith("subject: "))):
135 135 subject = line[9:]
136 136 format = "tag"
137 137 elif (format != "tagdone" and (line.startswith("From: ") or
138 138 line.startswith("from: "))):
139 139 user = line[6:]
140 140 format = "tag"
141 141 elif (format != "tagdone" and (line.startswith("Date: ") or
142 142 line.startswith("date: "))):
143 143 date = line[6:]
144 144 format = "tag"
145 145 elif format == "tag" and line == "":
146 146 # when looking for tags (subject: from: etc) they
147 147 # end once you find a blank line in the source
148 148 format = "tagdone"
149 149 elif message or line:
150 150 message.append(line)
151 151 comments.append(line)
152 152
153 153 eatdiff(message)
154 154 eatdiff(comments)
155 155 # Remember the exact starting line of the patch diffs before consuming
156 156 # empty lines, for external use by TortoiseHg and others
157 157 self.diffstartline = len(comments)
158 158 eatempty(message)
159 159 eatempty(comments)
160 160
161 161 # make sure message isn't empty
162 162 if format and format.startswith("tag") and subject:
163 163 message.insert(0, "")
164 164 message.insert(0, subject)
165 165
166 166 self.message = message
167 167 self.comments = comments
168 168 self.user = user
169 169 self.date = date
170 170 self.parent = parent
171 171 # nodeid and branch are for external use by TortoiseHg and others
172 172 self.nodeid = nodeid
173 173 self.branch = branch
174 174 self.haspatch = diffstart > 1
175 175 self.plainmode = plainmode
176 176
177 177 def setuser(self, user):
178 178 if not self.updateheader(['From: ', '# User '], user):
179 179 try:
180 180 patchheaderat = self.comments.index('# HG changeset patch')
181 181 self.comments.insert(patchheaderat + 1, '# User ' + user)
182 182 except ValueError:
183 183 if self.plainmode or self._hasheader(['Date: ']):
184 184 self.comments = ['From: ' + user] + self.comments
185 185 else:
186 186 tmp = ['# HG changeset patch', '# User ' + user, '']
187 187 self.comments = tmp + self.comments
188 188 self.user = user
189 189
190 190 def setdate(self, date):
191 191 if not self.updateheader(['Date: ', '# Date '], date):
192 192 try:
193 193 patchheaderat = self.comments.index('# HG changeset patch')
194 194 self.comments.insert(patchheaderat + 1, '# Date ' + date)
195 195 except ValueError:
196 196 if self.plainmode or self._hasheader(['From: ']):
197 197 self.comments = ['Date: ' + date] + self.comments
198 198 else:
199 199 tmp = ['# HG changeset patch', '# Date ' + date, '']
200 200 self.comments = tmp + self.comments
201 201 self.date = date
202 202
203 203 def setparent(self, parent):
204 204 if not self.updateheader(['# Parent '], parent):
205 205 try:
206 206 patchheaderat = self.comments.index('# HG changeset patch')
207 207 self.comments.insert(patchheaderat + 1, '# Parent ' + parent)
208 208 except ValueError:
209 209 pass
210 210 self.parent = parent
211 211
212 212 def setmessage(self, message):
213 213 if self.comments:
214 214 self._delmsg()
215 215 self.message = [message]
216 216 self.comments += self.message
217 217
218 218 def updateheader(self, prefixes, new):
219 219 '''Update all references to a field in the patch header.
220 220 Return whether the field is present.'''
221 221 res = False
222 222 for prefix in prefixes:
223 223 for i in xrange(len(self.comments)):
224 224 if self.comments[i].startswith(prefix):
225 225 self.comments[i] = prefix + new
226 226 res = True
227 227 break
228 228 return res
229 229
230 230 def _hasheader(self, prefixes):
231 231 '''Check if a header starts with any of the given prefixes.'''
232 232 for prefix in prefixes:
233 233 for comment in self.comments:
234 234 if comment.startswith(prefix):
235 235 return True
236 236 return False
237 237
238 238 def __str__(self):
239 239 if not self.comments:
240 240 return ''
241 241 return '\n'.join(self.comments) + '\n\n'
242 242
243 243 def _delmsg(self):
244 244 '''Remove existing message, keeping the rest of the comments fields.
245 245 If comments contains 'subject: ', message will prepend
246 246 the field and a blank line.'''
247 247 if self.message:
248 248 subj = 'subject: ' + self.message[0].lower()
249 249 for i in xrange(len(self.comments)):
250 250 if subj == self.comments[i].lower():
251 251 del self.comments[i]
252 252 self.message = self.message[2:]
253 253 break
254 254 ci = 0
255 255 for mi in self.message:
256 256 while mi != self.comments[ci]:
257 257 ci += 1
258 258 del self.comments[ci]
259 259
260 260 def secretcommit(repo, phase, *args, **kwargs):
261 261 """helper dedicated to ensure a commit are secret
262 262
263 263 It should be used instead of repo.commit inside the mq source
264 264 """
265 265 if phase is None:
266 266 if repo.ui.configbool('mq', 'secret', False):
267 267 phase = phases.secret
268 268 if phase is not None:
269 269 backup = repo.ui.backupconfig('phases', 'new-commit')
270 270 # Marking the repository as committing an mq patch can be used
271 271 # to optimize operations like _branchtags().
272 272 repo._committingpatch = True
273 273 try:
274 274 if phase is not None:
275 275 repo.ui.setconfig('phases', 'new-commit', phase)
276 276 return repo.commit(*args, **kwargs)
277 277 finally:
278 278 repo._committingpatch = False
279 279 if phase is not None:
280 280 repo.ui.restoreconfig(backup)
281 281
282 282 class queue(object):
283 283 def __init__(self, ui, path, patchdir=None):
284 284 self.basepath = path
285 285 try:
286 286 fh = open(os.path.join(path, 'patches.queue'))
287 287 cur = fh.read().rstrip()
288 288 fh.close()
289 289 if not cur:
290 290 curpath = os.path.join(path, 'patches')
291 291 else:
292 292 curpath = os.path.join(path, 'patches-' + cur)
293 293 except IOError:
294 294 curpath = os.path.join(path, 'patches')
295 295 self.path = patchdir or curpath
296 296 self.opener = scmutil.opener(self.path)
297 297 self.ui = ui
298 298 self.applieddirty = False
299 299 self.seriesdirty = False
300 300 self.added = []
301 301 self.seriespath = "series"
302 302 self.statuspath = "status"
303 303 self.guardspath = "guards"
304 304 self.activeguards = None
305 305 self.guardsdirty = False
306 306 # Handle mq.git as a bool with extended values
307 307 try:
308 308 gitmode = ui.configbool('mq', 'git', None)
309 309 if gitmode is None:
310 310 raise error.ConfigError()
311 311 self.gitmode = gitmode and 'yes' or 'no'
312 312 except error.ConfigError:
313 313 self.gitmode = ui.config('mq', 'git', 'auto').lower()
314 314 self.plainmode = ui.configbool('mq', 'plain', False)
315 315
316 316 @util.propertycache
317 317 def applied(self):
318 318 def parselines(lines):
319 319 for l in lines:
320 320 entry = l.split(':', 1)
321 321 if len(entry) > 1:
322 322 n, name = entry
323 323 yield statusentry(bin(n), name)
324 324 elif l.strip():
325 325 self.ui.warn(_('malformated mq status line: %s\n') % entry)
326 326 # else we ignore empty lines
327 327 try:
328 328 lines = self.opener.read(self.statuspath).splitlines()
329 329 return list(parselines(lines))
330 330 except IOError, e:
331 331 if e.errno == errno.ENOENT:
332 332 return []
333 333 raise
334 334
335 335 @util.propertycache
336 336 def fullseries(self):
337 337 try:
338 338 return self.opener.read(self.seriespath).splitlines()
339 339 except IOError, e:
340 340 if e.errno == errno.ENOENT:
341 341 return []
342 342 raise
343 343
344 344 @util.propertycache
345 345 def series(self):
346 346 self.parseseries()
347 347 return self.series
348 348
349 349 @util.propertycache
350 350 def seriesguards(self):
351 351 self.parseseries()
352 352 return self.seriesguards
353 353
354 354 def invalidate(self):
355 355 for a in 'applied fullseries series seriesguards'.split():
356 356 if a in self.__dict__:
357 357 delattr(self, a)
358 358 self.applieddirty = False
359 359 self.seriesdirty = False
360 360 self.guardsdirty = False
361 361 self.activeguards = None
362 362
363 363 def diffopts(self, opts={}, patchfn=None):
364 364 diffopts = patchmod.diffopts(self.ui, opts)
365 365 if self.gitmode == 'auto':
366 366 diffopts.upgrade = True
367 367 elif self.gitmode == 'keep':
368 368 pass
369 369 elif self.gitmode in ('yes', 'no'):
370 370 diffopts.git = self.gitmode == 'yes'
371 371 else:
372 372 raise util.Abort(_('mq.git option can be auto/keep/yes/no'
373 373 ' got %s') % self.gitmode)
374 374 if patchfn:
375 375 diffopts = self.patchopts(diffopts, patchfn)
376 376 return diffopts
377 377
378 378 def patchopts(self, diffopts, *patches):
379 379 """Return a copy of input diff options with git set to true if
380 380 referenced patch is a git patch and should be preserved as such.
381 381 """
382 382 diffopts = diffopts.copy()
383 383 if not diffopts.git and self.gitmode == 'keep':
384 384 for patchfn in patches:
385 385 patchf = self.opener(patchfn, 'r')
386 386 # if the patch was a git patch, refresh it as a git patch
387 387 for line in patchf:
388 388 if line.startswith('diff --git'):
389 389 diffopts.git = True
390 390 break
391 391 patchf.close()
392 392 return diffopts
393 393
394 394 def join(self, *p):
395 395 return os.path.join(self.path, *p)
396 396
397 397 def findseries(self, patch):
398 398 def matchpatch(l):
399 399 l = l.split('#', 1)[0]
400 400 return l.strip() == patch
401 401 for index, l in enumerate(self.fullseries):
402 402 if matchpatch(l):
403 403 return index
404 404 return None
405 405
406 406 guard_re = re.compile(r'\s?#([-+][^-+# \t\r\n\f][^# \t\r\n\f]*)')
407 407
408 408 def parseseries(self):
409 409 self.series = []
410 410 self.seriesguards = []
411 411 for l in self.fullseries:
412 412 h = l.find('#')
413 413 if h == -1:
414 414 patch = l
415 415 comment = ''
416 416 elif h == 0:
417 417 continue
418 418 else:
419 419 patch = l[:h]
420 420 comment = l[h:]
421 421 patch = patch.strip()
422 422 if patch:
423 423 if patch in self.series:
424 424 raise util.Abort(_('%s appears more than once in %s') %
425 425 (patch, self.join(self.seriespath)))
426 426 self.series.append(patch)
427 427 self.seriesguards.append(self.guard_re.findall(comment))
428 428
429 429 def checkguard(self, guard):
430 430 if not guard:
431 431 return _('guard cannot be an empty string')
432 432 bad_chars = '# \t\r\n\f'
433 433 first = guard[0]
434 434 if first in '-+':
435 435 return (_('guard %r starts with invalid character: %r') %
436 436 (guard, first))
437 437 for c in bad_chars:
438 438 if c in guard:
439 439 return _('invalid character in guard %r: %r') % (guard, c)
440 440
441 441 def setactive(self, guards):
442 442 for guard in guards:
443 443 bad = self.checkguard(guard)
444 444 if bad:
445 445 raise util.Abort(bad)
446 446 guards = sorted(set(guards))
447 447 self.ui.debug('active guards: %s\n' % ' '.join(guards))
448 448 self.activeguards = guards
449 449 self.guardsdirty = True
450 450
451 451 def active(self):
452 452 if self.activeguards is None:
453 453 self.activeguards = []
454 454 try:
455 455 guards = self.opener.read(self.guardspath).split()
456 456 except IOError, err:
457 457 if err.errno != errno.ENOENT:
458 458 raise
459 459 guards = []
460 460 for i, guard in enumerate(guards):
461 461 bad = self.checkguard(guard)
462 462 if bad:
463 463 self.ui.warn('%s:%d: %s\n' %
464 464 (self.join(self.guardspath), i + 1, bad))
465 465 else:
466 466 self.activeguards.append(guard)
467 467 return self.activeguards
468 468
469 469 def setguards(self, idx, guards):
470 470 for g in guards:
471 471 if len(g) < 2:
472 472 raise util.Abort(_('guard %r too short') % g)
473 473 if g[0] not in '-+':
474 474 raise util.Abort(_('guard %r starts with invalid char') % g)
475 475 bad = self.checkguard(g[1:])
476 476 if bad:
477 477 raise util.Abort(bad)
478 478 drop = self.guard_re.sub('', self.fullseries[idx])
479 479 self.fullseries[idx] = drop + ''.join([' #' + g for g in guards])
480 480 self.parseseries()
481 481 self.seriesdirty = True
482 482
483 483 def pushable(self, idx):
484 484 if isinstance(idx, str):
485 485 idx = self.series.index(idx)
486 486 patchguards = self.seriesguards[idx]
487 487 if not patchguards:
488 488 return True, None
489 489 guards = self.active()
490 490 exactneg = [g for g in patchguards if g[0] == '-' and g[1:] in guards]
491 491 if exactneg:
492 492 return False, repr(exactneg[0])
493 493 pos = [g for g in patchguards if g[0] == '+']
494 494 exactpos = [g for g in pos if g[1:] in guards]
495 495 if pos:
496 496 if exactpos:
497 497 return True, repr(exactpos[0])
498 498 return False, ' '.join(map(repr, pos))
499 499 return True, ''
500 500
501 501 def explainpushable(self, idx, all_patches=False):
502 502 write = all_patches and self.ui.write or self.ui.warn
503 503 if all_patches or self.ui.verbose:
504 504 if isinstance(idx, str):
505 505 idx = self.series.index(idx)
506 506 pushable, why = self.pushable(idx)
507 507 if all_patches and pushable:
508 508 if why is None:
509 509 write(_('allowing %s - no guards in effect\n') %
510 510 self.series[idx])
511 511 else:
512 512 if not why:
513 513 write(_('allowing %s - no matching negative guards\n') %
514 514 self.series[idx])
515 515 else:
516 516 write(_('allowing %s - guarded by %s\n') %
517 517 (self.series[idx], why))
518 518 if not pushable:
519 519 if why:
520 520 write(_('skipping %s - guarded by %s\n') %
521 521 (self.series[idx], why))
522 522 else:
523 523 write(_('skipping %s - no matching guards\n') %
524 524 self.series[idx])
525 525
526 526 def savedirty(self):
527 527 def writelist(items, path):
528 528 fp = self.opener(path, 'w')
529 529 for i in items:
530 530 fp.write("%s\n" % i)
531 531 fp.close()
532 532 if self.applieddirty:
533 533 writelist(map(str, self.applied), self.statuspath)
534 534 self.applieddirty = False
535 535 if self.seriesdirty:
536 536 writelist(self.fullseries, self.seriespath)
537 537 self.seriesdirty = False
538 538 if self.guardsdirty:
539 539 writelist(self.activeguards, self.guardspath)
540 540 self.guardsdirty = False
541 541 if self.added:
542 542 qrepo = self.qrepo()
543 543 if qrepo:
544 544 qrepo[None].add(f for f in self.added if f not in qrepo[None])
545 545 self.added = []
546 546
547 547 def removeundo(self, repo):
548 548 undo = repo.sjoin('undo')
549 549 if not os.path.exists(undo):
550 550 return
551 551 try:
552 552 os.unlink(undo)
553 553 except OSError, inst:
554 554 self.ui.warn(_('error removing undo: %s\n') % str(inst))
555 555
556 556 def printdiff(self, repo, diffopts, node1, node2=None, files=None,
557 557 fp=None, changes=None, opts={}):
558 558 stat = opts.get('stat')
559 559 m = scmutil.match(repo[node1], files, opts)
560 560 cmdutil.diffordiffstat(self.ui, repo, diffopts, node1, node2, m,
561 561 changes, stat, fp)
562 562
563 563 def mergeone(self, repo, mergeq, head, patch, rev, diffopts):
564 564 # first try just applying the patch
565 565 (err, n) = self.apply(repo, [patch], update_status=False,
566 566 strict=True, merge=rev)
567 567
568 568 if err == 0:
569 569 return (err, n)
570 570
571 571 if n is None:
572 572 raise util.Abort(_("apply failed for patch %s") % patch)
573 573
574 574 self.ui.warn(_("patch didn't work out, merging %s\n") % patch)
575 575
576 576 # apply failed, strip away that rev and merge.
577 577 hg.clean(repo, head)
578 578 self.strip(repo, [n], update=False, backup='strip')
579 579
580 580 ctx = repo[rev]
581 581 ret = hg.merge(repo, rev)
582 582 if ret:
583 583 raise util.Abort(_("update returned %d") % ret)
584 584 n = secretcommit(repo, None, ctx.description(), ctx.user(), force=True)
585 585 if n is None:
586 586 raise util.Abort(_("repo commit failed"))
587 587 try:
588 588 ph = patchheader(mergeq.join(patch), self.plainmode)
589 589 except:
590 590 raise util.Abort(_("unable to read %s") % patch)
591 591
592 592 diffopts = self.patchopts(diffopts, patch)
593 593 patchf = self.opener(patch, "w")
594 594 comments = str(ph)
595 595 if comments:
596 596 patchf.write(comments)
597 597 self.printdiff(repo, diffopts, head, n, fp=patchf)
598 598 patchf.close()
599 599 self.removeundo(repo)
600 600 return (0, n)
601 601
602 602 def qparents(self, repo, rev=None):
603 603 if rev is None:
604 604 (p1, p2) = repo.dirstate.parents()
605 605 if p2 == nullid:
606 606 return p1
607 607 if not self.applied:
608 608 return None
609 609 return self.applied[-1].node
610 610 p1, p2 = repo.changelog.parents(rev)
611 611 if p2 != nullid and p2 in [x.node for x in self.applied]:
612 612 return p2
613 613 return p1
614 614
615 615 def mergepatch(self, repo, mergeq, series, diffopts):
616 616 if not self.applied:
617 617 # each of the patches merged in will have two parents. This
618 618 # can confuse the qrefresh, qdiff, and strip code because it
619 619 # needs to know which parent is actually in the patch queue.
620 620 # so, we insert a merge marker with only one parent. This way
621 621 # the first patch in the queue is never a merge patch
622 622 #
623 623 pname = ".hg.patches.merge.marker"
624 624 n = secretcommit(repo, None, '[mq]: merge marker', force=True)
625 625 self.removeundo(repo)
626 626 self.applied.append(statusentry(n, pname))
627 627 self.applieddirty = True
628 628
629 629 head = self.qparents(repo)
630 630
631 631 for patch in series:
632 632 patch = mergeq.lookup(patch, strict=True)
633 633 if not patch:
634 634 self.ui.warn(_("patch %s does not exist\n") % patch)
635 635 return (1, None)
636 636 pushable, reason = self.pushable(patch)
637 637 if not pushable:
638 638 self.explainpushable(patch, all_patches=True)
639 639 continue
640 640 info = mergeq.isapplied(patch)
641 641 if not info:
642 642 self.ui.warn(_("patch %s is not applied\n") % patch)
643 643 return (1, None)
644 644 rev = info[1]
645 645 err, head = self.mergeone(repo, mergeq, head, patch, rev, diffopts)
646 646 if head:
647 647 self.applied.append(statusentry(head, patch))
648 648 self.applieddirty = True
649 649 if err:
650 650 return (err, head)
651 651 self.savedirty()
652 652 return (0, head)
653 653
654 654 def patch(self, repo, patchfile):
655 655 '''Apply patchfile to the working directory.
656 656 patchfile: name of patch file'''
657 657 files = set()
658 658 try:
659 659 fuzz = patchmod.patch(self.ui, repo, patchfile, strip=1,
660 660 files=files, eolmode=None)
661 661 return (True, list(files), fuzz)
662 662 except Exception, inst:
663 663 self.ui.note(str(inst) + '\n')
664 664 if not self.ui.verbose:
665 665 self.ui.warn(_("patch failed, unable to continue (try -v)\n"))
666 666 self.ui.traceback()
667 667 return (False, list(files), False)
668 668
669 669 def apply(self, repo, series, list=False, update_status=True,
670 670 strict=False, patchdir=None, merge=None, all_files=None):
671 671 wlock = lock = tr = None
672 672 try:
673 673 wlock = repo.wlock()
674 674 lock = repo.lock()
675 675 tr = repo.transaction("qpush")
676 676 try:
677 677 ret = self._apply(repo, series, list, update_status,
678 678 strict, patchdir, merge, all_files=all_files)
679 679 tr.close()
680 680 self.savedirty()
681 681 return ret
682 682 except:
683 683 try:
684 684 tr.abort()
685 685 finally:
686 686 repo.invalidate()
687 687 repo.dirstate.invalidate()
688 688 self.invalidate()
689 689 raise
690 690 finally:
691 691 release(tr, lock, wlock)
692 692 self.removeundo(repo)
693 693
694 694 def _apply(self, repo, series, list=False, update_status=True,
695 695 strict=False, patchdir=None, merge=None, all_files=None):
696 696 '''returns (error, hash)
697 697 error = 1 for unable to read, 2 for patch failed, 3 for patch fuzz'''
698 698 # TODO unify with commands.py
699 699 if not patchdir:
700 700 patchdir = self.path
701 701 err = 0
702 702 n = None
703 703 for patchname in series:
704 704 pushable, reason = self.pushable(patchname)
705 705 if not pushable:
706 706 self.explainpushable(patchname, all_patches=True)
707 707 continue
708 708 self.ui.status(_("applying %s\n") % patchname)
709 709 pf = os.path.join(patchdir, patchname)
710 710
711 711 try:
712 712 ph = patchheader(self.join(patchname), self.plainmode)
713 713 except IOError:
714 714 self.ui.warn(_("unable to read %s\n") % patchname)
715 715 err = 1
716 716 break
717 717
718 718 message = ph.message
719 719 if not message:
720 720 # The commit message should not be translated
721 721 message = "imported patch %s\n" % patchname
722 722 else:
723 723 if list:
724 724 # The commit message should not be translated
725 725 message.append("\nimported patch %s" % patchname)
726 726 message = '\n'.join(message)
727 727
728 728 if ph.haspatch:
729 729 (patcherr, files, fuzz) = self.patch(repo, pf)
730 730 if all_files is not None:
731 731 all_files.update(files)
732 732 patcherr = not patcherr
733 733 else:
734 734 self.ui.warn(_("patch %s is empty\n") % patchname)
735 735 patcherr, files, fuzz = 0, [], 0
736 736
737 737 if merge and files:
738 738 # Mark as removed/merged and update dirstate parent info
739 739 removed = []
740 740 merged = []
741 741 for f in files:
742 742 if os.path.lexists(repo.wjoin(f)):
743 743 merged.append(f)
744 744 else:
745 745 removed.append(f)
746 746 for f in removed:
747 747 repo.dirstate.remove(f)
748 748 for f in merged:
749 749 repo.dirstate.merge(f)
750 750 p1, p2 = repo.dirstate.parents()
751 751 repo.dirstate.setparents(p1, merge)
752 752
753 753 match = scmutil.matchfiles(repo, files or [])
754 754 oldtip = repo['tip']
755 755 n = secretcommit(repo, None, message, ph.user, ph.date, match=match,
756 756 force=True)
757 757 if repo['tip'] == oldtip:
758 758 raise util.Abort(_("qpush exactly duplicates child changeset"))
759 759 if n is None:
760 760 raise util.Abort(_("repository commit failed"))
761 761
762 762 if update_status:
763 763 self.applied.append(statusentry(n, patchname))
764 764
765 765 if patcherr:
766 766 self.ui.warn(_("patch failed, rejects left in working dir\n"))
767 767 err = 2
768 768 break
769 769
770 770 if fuzz and strict:
771 771 self.ui.warn(_("fuzz found when applying patch, stopping\n"))
772 772 err = 3
773 773 break
774 774 return (err, n)
775 775
776 776 def _cleanup(self, patches, numrevs, keep=False):
777 777 if not keep:
778 778 r = self.qrepo()
779 779 if r:
780 780 r[None].forget(patches)
781 781 for p in patches:
782 782 os.unlink(self.join(p))
783 783
784 784 qfinished = []
785 785 if numrevs:
786 786 qfinished = self.applied[:numrevs]
787 787 del self.applied[:numrevs]
788 788 self.applieddirty = True
789 789
790 790 unknown = []
791 791
792 792 for (i, p) in sorted([(self.findseries(p), p) for p in patches],
793 793 reverse=True):
794 794 if i is not None:
795 795 del self.fullseries[i]
796 796 else:
797 797 unknown.append(p)
798 798
799 799 if unknown:
800 800 if numrevs:
801 801 rev = dict((entry.name, entry.node) for entry in qfinished)
802 802 for p in unknown:
803 803 msg = _('revision %s refers to unknown patches: %s\n')
804 804 self.ui.warn(msg % (short(rev[p]), p))
805 805 else:
806 806 msg = _('unknown patches: %s\n')
807 807 raise util.Abort(''.join(msg % p for p in unknown))
808 808
809 809 self.parseseries()
810 810 self.seriesdirty = True
811 811 return [entry.node for entry in qfinished]
812 812
813 813 def _revpatches(self, repo, revs):
814 814 firstrev = repo[self.applied[0].node].rev()
815 815 patches = []
816 816 for i, rev in enumerate(revs):
817 817
818 818 if rev < firstrev:
819 819 raise util.Abort(_('revision %d is not managed') % rev)
820 820
821 821 ctx = repo[rev]
822 822 base = self.applied[i].node
823 823 if ctx.node() != base:
824 824 msg = _('cannot delete revision %d above applied patches')
825 825 raise util.Abort(msg % rev)
826 826
827 827 patch = self.applied[i].name
828 828 for fmt in ('[mq]: %s', 'imported patch %s'):
829 829 if ctx.description() == fmt % patch:
830 830 msg = _('patch %s finalized without changeset message\n')
831 831 repo.ui.status(msg % patch)
832 832 break
833 833
834 834 patches.append(patch)
835 835 return patches
836 836
837 837 def finish(self, repo, revs):
838 838 # Manually trigger phase computation to ensure phasedefaults is
839 839 # executed before we remove the patches.
840 840 repo._phaserev
841 841 patches = self._revpatches(repo, sorted(revs))
842 842 qfinished = self._cleanup(patches, len(patches))
843 843 if qfinished and repo.ui.configbool('mq', 'secret', False):
844 844 # only use this logic when the secret option is added
845 845 oldqbase = repo[qfinished[0]]
846 846 if oldqbase.p1().phase() < phases.secret:
847 847 phases.advanceboundary(repo, phases.draft, qfinished)
848 848
849 849 def delete(self, repo, patches, opts):
850 850 if not patches and not opts.get('rev'):
851 851 raise util.Abort(_('qdelete requires at least one revision or '
852 852 'patch name'))
853 853
854 854 realpatches = []
855 855 for patch in patches:
856 856 patch = self.lookup(patch, strict=True)
857 857 info = self.isapplied(patch)
858 858 if info:
859 859 raise util.Abort(_("cannot delete applied patch %s") % patch)
860 860 if patch not in self.series:
861 861 raise util.Abort(_("patch %s not in series file") % patch)
862 862 if patch not in realpatches:
863 863 realpatches.append(patch)
864 864
865 865 numrevs = 0
866 866 if opts.get('rev'):
867 867 if not self.applied:
868 868 raise util.Abort(_('no patches applied'))
869 869 revs = scmutil.revrange(repo, opts.get('rev'))
870 870 if len(revs) > 1 and revs[0] > revs[1]:
871 871 revs.reverse()
872 872 revpatches = self._revpatches(repo, revs)
873 873 realpatches += revpatches
874 874 numrevs = len(revpatches)
875 875
876 876 self._cleanup(realpatches, numrevs, opts.get('keep'))
877 877
878 878 def checktoppatch(self, repo):
879 879 if self.applied:
880 880 top = self.applied[-1].node
881 881 patch = self.applied[-1].name
882 882 pp = repo.dirstate.parents()
883 883 if top not in pp:
884 884 raise util.Abort(_("working directory revision is not qtip"))
885 885 return top, patch
886 886 return None, None
887 887
888 888 def checksubstate(self, repo):
889 889 '''return list of subrepos at a different revision than substate.
890 890 Abort if any subrepos have uncommitted changes.'''
891 891 inclsubs = []
892 892 wctx = repo[None]
893 893 for s in wctx.substate:
894 894 if wctx.sub(s).dirty(True):
895 895 raise util.Abort(
896 896 _("uncommitted changes in subrepository %s") % s)
897 897 elif wctx.sub(s).dirty():
898 898 inclsubs.append(s)
899 899 return inclsubs
900 900
901 901 def localchangesfound(self, refresh=True):
902 902 if refresh:
903 903 raise util.Abort(_("local changes found, refresh first"))
904 904 else:
905 905 raise util.Abort(_("local changes found"))
906 906
907 907 def checklocalchanges(self, repo, force=False, refresh=True):
908 908 m, a, r, d = repo.status()[:4]
909 909 if (m or a or r or d) and not force:
910 910 self.localchangesfound(refresh)
911 911 return m, a, r, d
912 912
913 913 _reserved = ('series', 'status', 'guards', '.', '..')
914 914 def checkreservedname(self, name):
915 915 if name in self._reserved:
916 916 raise util.Abort(_('"%s" cannot be used as the name of a patch')
917 917 % name)
918 918 for prefix in ('.hg', '.mq'):
919 919 if name.startswith(prefix):
920 920 raise util.Abort(_('patch name cannot begin with "%s"')
921 921 % prefix)
922 922 for c in ('#', ':'):
923 923 if c in name:
924 924 raise util.Abort(_('"%s" cannot be used in the name of a patch')
925 925 % c)
926 926
927 927 def checkpatchname(self, name, force=False):
928 928 self.checkreservedname(name)
929 929 if not force and os.path.exists(self.join(name)):
930 930 if os.path.isdir(self.join(name)):
931 931 raise util.Abort(_('"%s" already exists as a directory')
932 932 % name)
933 933 else:
934 934 raise util.Abort(_('patch "%s" already exists') % name)
935 935
936 936 def new(self, repo, patchfn, *pats, **opts):
937 937 """options:
938 938 msg: a string or a no-argument function returning a string
939 939 """
940 940 msg = opts.get('msg')
941 941 user = opts.get('user')
942 942 date = opts.get('date')
943 943 if date:
944 944 date = util.parsedate(date)
945 945 diffopts = self.diffopts({'git': opts.get('git')})
946 946 if opts.get('checkname', True):
947 947 self.checkpatchname(patchfn)
948 948 inclsubs = self.checksubstate(repo)
949 949 if inclsubs:
950 950 inclsubs.append('.hgsubstate')
951 951 if opts.get('include') or opts.get('exclude') or pats:
952 952 if inclsubs:
953 953 pats = list(pats or []) + inclsubs
954 954 match = scmutil.match(repo[None], pats, opts)
955 955 # detect missing files in pats
956 956 def badfn(f, msg):
957 957 if f != '.hgsubstate': # .hgsubstate is auto-created
958 958 raise util.Abort('%s: %s' % (f, msg))
959 959 match.bad = badfn
960 960 m, a, r, d = repo.status(match=match)[:4]
961 961 else:
962 962 m, a, r, d = self.checklocalchanges(repo, force=True)
963 963 match = scmutil.matchfiles(repo, m + a + r + inclsubs)
964 964 if len(repo[None].parents()) > 1:
965 965 raise util.Abort(_('cannot manage merge changesets'))
966 966 commitfiles = m + a + r
967 967 self.checktoppatch(repo)
968 968 insert = self.fullseriesend()
969 969 wlock = repo.wlock()
970 970 try:
971 971 try:
972 972 # if patch file write fails, abort early
973 973 p = self.opener(patchfn, "w")
974 974 except IOError, e:
975 975 raise util.Abort(_('cannot write patch "%s": %s')
976 976 % (patchfn, e.strerror))
977 977 try:
978 978 if self.plainmode:
979 979 if user:
980 980 p.write("From: " + user + "\n")
981 981 if not date:
982 982 p.write("\n")
983 983 if date:
984 984 p.write("Date: %d %d\n\n" % date)
985 985 else:
986 986 p.write("# HG changeset patch\n")
987 987 p.write("# Parent "
988 988 + hex(repo[None].p1().node()) + "\n")
989 989 if user:
990 990 p.write("# User " + user + "\n")
991 991 if date:
992 992 p.write("# Date %s %s\n\n" % date)
993 993 if util.safehasattr(msg, '__call__'):
994 994 msg = msg()
995 995 commitmsg = msg and msg or ("[mq]: %s" % patchfn)
996 996 n = secretcommit(repo, None, commitmsg, user, date, match=match,
997 997 force=True)
998 998 if n is None:
999 999 raise util.Abort(_("repo commit failed"))
1000 1000 try:
1001 1001 self.fullseries[insert:insert] = [patchfn]
1002 1002 self.applied.append(statusentry(n, patchfn))
1003 1003 self.parseseries()
1004 1004 self.seriesdirty = True
1005 1005 self.applieddirty = True
1006 1006 if msg:
1007 1007 msg = msg + "\n\n"
1008 1008 p.write(msg)
1009 1009 if commitfiles:
1010 1010 parent = self.qparents(repo, n)
1011 1011 chunks = patchmod.diff(repo, node1=parent, node2=n,
1012 1012 match=match, opts=diffopts)
1013 1013 for chunk in chunks:
1014 1014 p.write(chunk)
1015 1015 p.close()
1016 1016 r = self.qrepo()
1017 1017 if r:
1018 1018 r[None].add([patchfn])
1019 1019 except:
1020 1020 repo.rollback()
1021 1021 raise
1022 1022 except Exception:
1023 1023 patchpath = self.join(patchfn)
1024 1024 try:
1025 1025 os.unlink(patchpath)
1026 1026 except:
1027 1027 self.ui.warn(_('error unlinking %s\n') % patchpath)
1028 1028 raise
1029 1029 self.removeundo(repo)
1030 1030 finally:
1031 1031 release(wlock)
1032 1032
1033 1033 def strip(self, repo, revs, update=True, backup="all", force=None):
1034 1034 wlock = lock = None
1035 1035 try:
1036 1036 wlock = repo.wlock()
1037 1037 lock = repo.lock()
1038 1038
1039 1039 if update:
1040 1040 self.checklocalchanges(repo, force=force, refresh=False)
1041 1041 urev = self.qparents(repo, revs[0])
1042 1042 hg.clean(repo, urev)
1043 1043 repo.dirstate.write()
1044 1044
1045 1045 self.removeundo(repo)
1046 1046 repair.strip(self.ui, repo, revs, backup)
1047 1047 # strip may have unbundled a set of backed up revisions after
1048 1048 # the actual strip
1049 1049 self.removeundo(repo)
1050 1050 finally:
1051 1051 release(lock, wlock)
1052 1052
1053 1053 def isapplied(self, patch):
1054 1054 """returns (index, rev, patch)"""
1055 1055 for i, a in enumerate(self.applied):
1056 1056 if a.name == patch:
1057 1057 return (i, a.node, a.name)
1058 1058 return None
1059 1059
1060 1060 # if the exact patch name does not exist, we try a few
1061 1061 # variations. If strict is passed, we try only #1
1062 1062 #
1063 1063 # 1) a number (as string) to indicate an offset in the series file
1064 1064 # 2) a unique substring of the patch name was given
1065 1065 # 3) patchname[-+]num to indicate an offset in the series file
1066 1066 def lookup(self, patch, strict=False):
1067 1067 def partialname(s):
1068 1068 if s in self.series:
1069 1069 return s
1070 1070 matches = [x for x in self.series if s in x]
1071 1071 if len(matches) > 1:
1072 1072 self.ui.warn(_('patch name "%s" is ambiguous:\n') % s)
1073 1073 for m in matches:
1074 1074 self.ui.warn(' %s\n' % m)
1075 1075 return None
1076 1076 if matches:
1077 1077 return matches[0]
1078 1078 if self.series and self.applied:
1079 1079 if s == 'qtip':
1080 1080 return self.series[self.seriesend(True)-1]
1081 1081 if s == 'qbase':
1082 1082 return self.series[0]
1083 1083 return None
1084 1084
1085 1085 if patch in self.series:
1086 1086 return patch
1087 1087
1088 1088 if not os.path.isfile(self.join(patch)):
1089 1089 try:
1090 1090 sno = int(patch)
1091 1091 except (ValueError, OverflowError):
1092 1092 pass
1093 1093 else:
1094 1094 if -len(self.series) <= sno < len(self.series):
1095 1095 return self.series[sno]
1096 1096
1097 1097 if not strict:
1098 1098 res = partialname(patch)
1099 1099 if res:
1100 1100 return res
1101 1101 minus = patch.rfind('-')
1102 1102 if minus >= 0:
1103 1103 res = partialname(patch[:minus])
1104 1104 if res:
1105 1105 i = self.series.index(res)
1106 1106 try:
1107 1107 off = int(patch[minus + 1:] or 1)
1108 1108 except (ValueError, OverflowError):
1109 1109 pass
1110 1110 else:
1111 1111 if i - off >= 0:
1112 1112 return self.series[i - off]
1113 1113 plus = patch.rfind('+')
1114 1114 if plus >= 0:
1115 1115 res = partialname(patch[:plus])
1116 1116 if res:
1117 1117 i = self.series.index(res)
1118 1118 try:
1119 1119 off = int(patch[plus + 1:] or 1)
1120 1120 except (ValueError, OverflowError):
1121 1121 pass
1122 1122 else:
1123 1123 if i + off < len(self.series):
1124 1124 return self.series[i + off]
1125 1125 raise util.Abort(_("patch %s not in series") % patch)
1126 1126
1127 1127 def push(self, repo, patch=None, force=False, list=False,
1128 1128 mergeq=None, all=False, move=False, exact=False):
1129 1129 diffopts = self.diffopts()
1130 1130 wlock = repo.wlock()
1131 1131 try:
1132 1132 heads = []
1133 1133 for b, ls in repo.branchmap().iteritems():
1134 1134 heads += ls
1135 1135 if not heads:
1136 1136 heads = [nullid]
1137 1137 if repo.dirstate.p1() not in heads and not exact:
1138 1138 self.ui.status(_("(working directory not at a head)\n"))
1139 1139
1140 1140 if not self.series:
1141 1141 self.ui.warn(_('no patches in series\n'))
1142 1142 return 0
1143 1143
1144 1144 # Suppose our series file is: A B C and the current 'top'
1145 1145 # patch is B. qpush C should be performed (moving forward)
1146 1146 # qpush B is a NOP (no change) qpush A is an error (can't
1147 1147 # go backwards with qpush)
1148 1148 if patch:
1149 1149 patch = self.lookup(patch)
1150 1150 info = self.isapplied(patch)
1151 1151 if info and info[0] >= len(self.applied) - 1:
1152 1152 self.ui.warn(
1153 1153 _('qpush: %s is already at the top\n') % patch)
1154 1154 return 0
1155 1155
1156 1156 pushable, reason = self.pushable(patch)
1157 1157 if pushable:
1158 1158 if self.series.index(patch) < self.seriesend():
1159 1159 raise util.Abort(
1160 1160 _("cannot push to a previous patch: %s") % patch)
1161 1161 else:
1162 1162 if reason:
1163 1163 reason = _('guarded by %s') % reason
1164 1164 else:
1165 1165 reason = _('no matching guards')
1166 1166 self.ui.warn(_("cannot push '%s' - %s\n") % (patch, reason))
1167 1167 return 1
1168 1168 elif all:
1169 1169 patch = self.series[-1]
1170 1170 if self.isapplied(patch):
1171 1171 self.ui.warn(_('all patches are currently applied\n'))
1172 1172 return 0
1173 1173
1174 1174 # Following the above example, starting at 'top' of B:
1175 1175 # qpush should be performed (pushes C), but a subsequent
1176 1176 # qpush without an argument is an error (nothing to
1177 1177 # apply). This allows a loop of "...while hg qpush..." to
1178 1178 # work as it detects an error when done
1179 1179 start = self.seriesend()
1180 1180 if start == len(self.series):
1181 1181 self.ui.warn(_('patch series already fully applied\n'))
1182 1182 return 1
1183 1183 if not force:
1184 1184 self.checklocalchanges(repo, refresh=self.applied)
1185 1185
1186 1186 if exact:
1187 1187 if move:
1188 1188 raise util.Abort(_("cannot use --exact and --move together"))
1189 1189 if self.applied:
1190 1190 raise util.Abort(_("cannot push --exact with applied patches"))
1191 1191 root = self.series[start]
1192 1192 target = patchheader(self.join(root), self.plainmode).parent
1193 1193 if not target:
1194 1194 raise util.Abort(
1195 1195 _("%s does not have a parent recorded") % root)
1196 1196 if not repo[target] == repo['.']:
1197 1197 hg.update(repo, target)
1198 1198
1199 1199 if move:
1200 1200 if not patch:
1201 1201 raise util.Abort(_("please specify the patch to move"))
1202 1202 for i, rpn in enumerate(self.fullseries[start:]):
1203 1203 # strip markers for patch guards
1204 1204 if self.guard_re.split(rpn, 1)[0] == patch:
1205 1205 break
1206 1206 index = start + i
1207 1207 assert index < len(self.fullseries)
1208 1208 fullpatch = self.fullseries[index]
1209 1209 del self.fullseries[index]
1210 1210 self.fullseries.insert(start, fullpatch)
1211 1211 self.parseseries()
1212 1212 self.seriesdirty = True
1213 1213
1214 1214 self.applieddirty = True
1215 1215 if start > 0:
1216 1216 self.checktoppatch(repo)
1217 1217 if not patch:
1218 1218 patch = self.series[start]
1219 1219 end = start + 1
1220 1220 else:
1221 1221 end = self.series.index(patch, start) + 1
1222 1222
1223 1223 s = self.series[start:end]
1224 1224 all_files = set()
1225 1225 try:
1226 1226 if mergeq:
1227 1227 ret = self.mergepatch(repo, mergeq, s, diffopts)
1228 1228 else:
1229 1229 ret = self.apply(repo, s, list, all_files=all_files)
1230 1230 except:
1231 1231 self.ui.warn(_('cleaning up working directory...'))
1232 1232 node = repo.dirstate.p1()
1233 1233 hg.revert(repo, node, None)
1234 1234 # only remove unknown files that we know we touched or
1235 1235 # created while patching
1236 1236 for f in all_files:
1237 1237 if f not in repo.dirstate:
1238 1238 try:
1239 1239 util.unlinkpath(repo.wjoin(f))
1240 1240 except OSError, inst:
1241 1241 if inst.errno != errno.ENOENT:
1242 1242 raise
1243 1243 self.ui.warn(_('done\n'))
1244 1244 raise
1245 1245
1246 1246 if not self.applied:
1247 1247 return ret[0]
1248 1248 top = self.applied[-1].name
1249 1249 if ret[0] and ret[0] > 1:
1250 1250 msg = _("errors during apply, please fix and refresh %s\n")
1251 1251 self.ui.write(msg % top)
1252 1252 else:
1253 1253 self.ui.write(_("now at: %s\n") % top)
1254 1254 return ret[0]
1255 1255
1256 1256 finally:
1257 1257 wlock.release()
1258 1258
1259 1259 def pop(self, repo, patch=None, force=False, update=True, all=False):
1260 1260 wlock = repo.wlock()
1261 1261 try:
1262 1262 if patch:
1263 1263 # index, rev, patch
1264 1264 info = self.isapplied(patch)
1265 1265 if not info:
1266 1266 patch = self.lookup(patch)
1267 1267 info = self.isapplied(patch)
1268 1268 if not info:
1269 1269 raise util.Abort(_("patch %s is not applied") % patch)
1270 1270
1271 1271 if not self.applied:
1272 1272 # Allow qpop -a to work repeatedly,
1273 1273 # but not qpop without an argument
1274 1274 self.ui.warn(_("no patches applied\n"))
1275 1275 return not all
1276 1276
1277 1277 if all:
1278 1278 start = 0
1279 1279 elif patch:
1280 1280 start = info[0] + 1
1281 1281 else:
1282 1282 start = len(self.applied) - 1
1283 1283
1284 1284 if start >= len(self.applied):
1285 1285 self.ui.warn(_("qpop: %s is already at the top\n") % patch)
1286 1286 return
1287 1287
1288 1288 if not update:
1289 1289 parents = repo.dirstate.parents()
1290 1290 rr = [x.node for x in self.applied]
1291 1291 for p in parents:
1292 1292 if p in rr:
1293 1293 self.ui.warn(_("qpop: forcing dirstate update\n"))
1294 1294 update = True
1295 1295 else:
1296 1296 parents = [p.node() for p in repo[None].parents()]
1297 1297 needupdate = False
1298 1298 for entry in self.applied[start:]:
1299 1299 if entry.node in parents:
1300 1300 needupdate = True
1301 1301 break
1302 1302 update = needupdate
1303 1303
1304 1304 if not force and update:
1305 1305 self.checklocalchanges(repo)
1306 1306
1307 1307 self.applieddirty = True
1308 1308 end = len(self.applied)
1309 1309 rev = self.applied[start].node
1310 1310 if update:
1311 1311 top = self.checktoppatch(repo)[0]
1312 1312
1313 1313 try:
1314 1314 heads = repo.changelog.heads(rev)
1315 1315 except error.LookupError:
1316 1316 node = short(rev)
1317 1317 raise util.Abort(_('trying to pop unknown node %s') % node)
1318 1318
1319 1319 if heads != [self.applied[-1].node]:
1320 1320 raise util.Abort(_("popping would remove a revision not "
1321 1321 "managed by this patch queue"))
1322 1322 if not repo[self.applied[-1].node].mutable():
1323 1323 raise util.Abort(
1324 1324 _("popping would remove an immutable revision"),
1325 1325 hint=_('see "hg help phases" for details'))
1326 1326
1327 1327 # we know there are no local changes, so we can make a simplified
1328 1328 # form of hg.update.
1329 1329 if update:
1330 1330 qp = self.qparents(repo, rev)
1331 1331 ctx = repo[qp]
1332 1332 m, a, r, d = repo.status(qp, top)[:4]
1333 1333 if d:
1334 1334 raise util.Abort(_("deletions found between repo revs"))
1335 1335 for f in a:
1336 1336 try:
1337 1337 util.unlinkpath(repo.wjoin(f))
1338 1338 except OSError, e:
1339 1339 if e.errno != errno.ENOENT:
1340 1340 raise
1341 1341 repo.dirstate.drop(f)
1342 1342 for f in m + r:
1343 1343 fctx = ctx[f]
1344 1344 repo.wwrite(f, fctx.data(), fctx.flags())
1345 1345 repo.dirstate.normal(f)
1346 1346 repo.dirstate.setparents(qp, nullid)
1347 1347 for patch in reversed(self.applied[start:end]):
1348 1348 self.ui.status(_("popping %s\n") % patch.name)
1349 1349 del self.applied[start:end]
1350 1350 self.strip(repo, [rev], update=False, backup='strip')
1351 1351 if self.applied:
1352 1352 self.ui.write(_("now at: %s\n") % self.applied[-1].name)
1353 1353 else:
1354 1354 self.ui.write(_("patch queue now empty\n"))
1355 1355 finally:
1356 1356 wlock.release()
1357 1357
1358 1358 def diff(self, repo, pats, opts):
1359 1359 top, patch = self.checktoppatch(repo)
1360 1360 if not top:
1361 1361 self.ui.write(_("no patches applied\n"))
1362 1362 return
1363 1363 qp = self.qparents(repo, top)
1364 1364 if opts.get('reverse'):
1365 1365 node1, node2 = None, qp
1366 1366 else:
1367 1367 node1, node2 = qp, None
1368 1368 diffopts = self.diffopts(opts, patch)
1369 1369 self.printdiff(repo, diffopts, node1, node2, files=pats, opts=opts)
1370 1370
1371 1371 def refresh(self, repo, pats=None, **opts):
1372 1372 if not self.applied:
1373 1373 self.ui.write(_("no patches applied\n"))
1374 1374 return 1
1375 1375 msg = opts.get('msg', '').rstrip()
1376 1376 newuser = opts.get('user')
1377 1377 newdate = opts.get('date')
1378 1378 if newdate:
1379 1379 newdate = '%d %d' % util.parsedate(newdate)
1380 1380 wlock = repo.wlock()
1381 1381
1382 1382 try:
1383 1383 self.checktoppatch(repo)
1384 1384 (top, patchfn) = (self.applied[-1].node, self.applied[-1].name)
1385 1385 if repo.changelog.heads(top) != [top]:
1386 1386 raise util.Abort(_("cannot refresh a revision with children"))
1387 1387 if not repo[top].mutable():
1388 1388 raise util.Abort(_("cannot refresh immutable revision"),
1389 1389 hint=_('see "hg help phases" for details'))
1390 1390
1391 1391 inclsubs = self.checksubstate(repo)
1392 1392
1393 1393 cparents = repo.changelog.parents(top)
1394 1394 patchparent = self.qparents(repo, top)
1395 1395 ph = patchheader(self.join(patchfn), self.plainmode)
1396 1396 diffopts = self.diffopts({'git': opts.get('git')}, patchfn)
1397 1397 if msg:
1398 1398 ph.setmessage(msg)
1399 1399 if newuser:
1400 1400 ph.setuser(newuser)
1401 1401 if newdate:
1402 1402 ph.setdate(newdate)
1403 1403 ph.setparent(hex(patchparent))
1404 1404
1405 1405 # only commit new patch when write is complete
1406 1406 patchf = self.opener(patchfn, 'w', atomictemp=True)
1407 1407
1408 1408 comments = str(ph)
1409 1409 if comments:
1410 1410 patchf.write(comments)
1411 1411
1412 1412 # update the dirstate in place, strip off the qtip commit
1413 1413 # and then commit.
1414 1414 #
1415 1415 # this should really read:
1416 1416 # mm, dd, aa = repo.status(top, patchparent)[:3]
1417 1417 # but we do it backwards to take advantage of manifest/chlog
1418 1418 # caching against the next repo.status call
1419 1419 mm, aa, dd = repo.status(patchparent, top)[:3]
1420 1420 changes = repo.changelog.read(top)
1421 1421 man = repo.manifest.read(changes[0])
1422 1422 aaa = aa[:]
1423 1423 matchfn = scmutil.match(repo[None], pats, opts)
1424 1424 # in short mode, we only diff the files included in the
1425 1425 # patch already plus specified files
1426 1426 if opts.get('short'):
1427 1427 # if amending a patch, we start with existing
1428 1428 # files plus specified files - unfiltered
1429 1429 match = scmutil.matchfiles(repo, mm + aa + dd + matchfn.files())
1430 1430 # filter with inc/exl options
1431 1431 matchfn = scmutil.match(repo[None], opts=opts)
1432 1432 else:
1433 1433 match = scmutil.matchall(repo)
1434 1434 m, a, r, d = repo.status(match=match)[:4]
1435 1435 mm = set(mm)
1436 1436 aa = set(aa)
1437 1437 dd = set(dd)
1438 1438
1439 1439 # we might end up with files that were added between
1440 1440 # qtip and the dirstate parent, but then changed in the
1441 1441 # local dirstate. in this case, we want them to only
1442 1442 # show up in the added section
1443 1443 for x in m:
1444 1444 if x not in aa:
1445 1445 mm.add(x)
1446 1446 # we might end up with files added by the local dirstate that
1447 1447 # were deleted by the patch. In this case, they should only
1448 1448 # show up in the changed section.
1449 1449 for x in a:
1450 1450 if x in dd:
1451 1451 dd.remove(x)
1452 1452 mm.add(x)
1453 1453 else:
1454 1454 aa.add(x)
1455 1455 # make sure any files deleted in the local dirstate
1456 1456 # are not in the add or change column of the patch
1457 1457 forget = []
1458 1458 for x in d + r:
1459 1459 if x in aa:
1460 1460 aa.remove(x)
1461 1461 forget.append(x)
1462 1462 continue
1463 1463 else:
1464 1464 mm.discard(x)
1465 1465 dd.add(x)
1466 1466
1467 1467 m = list(mm)
1468 1468 r = list(dd)
1469 1469 a = list(aa)
1470 1470 c = [filter(matchfn, l) for l in (m, a, r)]
1471 1471 match = scmutil.matchfiles(repo, set(c[0] + c[1] + c[2] + inclsubs))
1472 1472 chunks = patchmod.diff(repo, patchparent, match=match,
1473 1473 changes=c, opts=diffopts)
1474 1474 for chunk in chunks:
1475 1475 patchf.write(chunk)
1476 1476
1477 1477 try:
1478 1478 if diffopts.git or diffopts.upgrade:
1479 1479 copies = {}
1480 1480 for dst in a:
1481 1481 src = repo.dirstate.copied(dst)
1482 1482 # during qfold, the source file for copies may
1483 1483 # be removed. Treat this as a simple add.
1484 1484 if src is not None and src in repo.dirstate:
1485 1485 copies.setdefault(src, []).append(dst)
1486 1486 repo.dirstate.add(dst)
1487 1487 # remember the copies between patchparent and qtip
1488 1488 for dst in aaa:
1489 1489 f = repo.file(dst)
1490 1490 src = f.renamed(man[dst])
1491 1491 if src:
1492 1492 copies.setdefault(src[0], []).extend(
1493 1493 copies.get(dst, []))
1494 1494 if dst in a:
1495 1495 copies[src[0]].append(dst)
1496 1496 # we can't copy a file created by the patch itself
1497 1497 if dst in copies:
1498 1498 del copies[dst]
1499 1499 for src, dsts in copies.iteritems():
1500 1500 for dst in dsts:
1501 1501 repo.dirstate.copy(src, dst)
1502 1502 else:
1503 1503 for dst in a:
1504 1504 repo.dirstate.add(dst)
1505 1505 # Drop useless copy information
1506 1506 for f in list(repo.dirstate.copies()):
1507 1507 repo.dirstate.copy(None, f)
1508 1508 for f in r:
1509 1509 repo.dirstate.remove(f)
1510 1510 # if the patch excludes a modified file, mark that
1511 1511 # file with mtime=0 so status can see it.
1512 1512 mm = []
1513 1513 for i in xrange(len(m)-1, -1, -1):
1514 1514 if not matchfn(m[i]):
1515 1515 mm.append(m[i])
1516 1516 del m[i]
1517 1517 for f in m:
1518 1518 repo.dirstate.normal(f)
1519 1519 for f in mm:
1520 1520 repo.dirstate.normallookup(f)
1521 1521 for f in forget:
1522 1522 repo.dirstate.drop(f)
1523 1523
1524 1524 if not msg:
1525 1525 if not ph.message:
1526 1526 message = "[mq]: %s\n" % patchfn
1527 1527 else:
1528 1528 message = "\n".join(ph.message)
1529 1529 else:
1530 1530 message = msg
1531 1531
1532 1532 user = ph.user or changes[1]
1533 1533
1534 1534 oldphase = repo[top].phase()
1535 1535
1536 1536 # assumes strip can roll itself back if interrupted
1537 1537 repo.dirstate.setparents(*cparents)
1538 1538 self.applied.pop()
1539 1539 self.applieddirty = True
1540 1540 self.strip(repo, [top], update=False,
1541 1541 backup='strip')
1542 1542 except:
1543 1543 repo.dirstate.invalidate()
1544 1544 raise
1545 1545
1546 1546 try:
1547 1547 # might be nice to attempt to roll back strip after this
1548 1548
1549 1549 # Ensure we create a new changeset in the same phase than
1550 1550 # the old one.
1551 1551 n = secretcommit(repo, oldphase, message, user, ph.date,
1552 1552 match=match, force=True)
1553 1553 # only write patch after a successful commit
1554 1554 patchf.close()
1555 1555 self.applied.append(statusentry(n, patchfn))
1556 1556 except:
1557 1557 ctx = repo[cparents[0]]
1558 1558 repo.dirstate.rebuild(ctx.node(), ctx.manifest())
1559 1559 self.savedirty()
1560 1560 self.ui.warn(_('refresh interrupted while patch was popped! '
1561 1561 '(revert --all, qpush to recover)\n'))
1562 1562 raise
1563 1563 finally:
1564 1564 wlock.release()
1565 1565 self.removeundo(repo)
1566 1566
1567 1567 def init(self, repo, create=False):
1568 1568 if not create and os.path.isdir(self.path):
1569 1569 raise util.Abort(_("patch queue directory already exists"))
1570 1570 try:
1571 1571 os.mkdir(self.path)
1572 1572 except OSError, inst:
1573 1573 if inst.errno != errno.EEXIST or not create:
1574 1574 raise
1575 1575 if create:
1576 1576 return self.qrepo(create=True)
1577 1577
1578 1578 def unapplied(self, repo, patch=None):
1579 1579 if patch and patch not in self.series:
1580 1580 raise util.Abort(_("patch %s is not in series file") % patch)
1581 1581 if not patch:
1582 1582 start = self.seriesend()
1583 1583 else:
1584 1584 start = self.series.index(patch) + 1
1585 1585 unapplied = []
1586 1586 for i in xrange(start, len(self.series)):
1587 1587 pushable, reason = self.pushable(i)
1588 1588 if pushable:
1589 1589 unapplied.append((i, self.series[i]))
1590 1590 self.explainpushable(i)
1591 1591 return unapplied
1592 1592
1593 1593 def qseries(self, repo, missing=None, start=0, length=None, status=None,
1594 1594 summary=False):
1595 1595 def displayname(pfx, patchname, state):
1596 1596 if pfx:
1597 1597 self.ui.write(pfx)
1598 1598 if summary:
1599 1599 ph = patchheader(self.join(patchname), self.plainmode)
1600 1600 msg = ph.message and ph.message[0] or ''
1601 1601 if self.ui.formatted():
1602 1602 width = self.ui.termwidth() - len(pfx) - len(patchname) - 2
1603 1603 if width > 0:
1604 1604 msg = util.ellipsis(msg, width)
1605 1605 else:
1606 1606 msg = ''
1607 1607 self.ui.write(patchname, label='qseries.' + state)
1608 1608 self.ui.write(': ')
1609 1609 self.ui.write(msg, label='qseries.message.' + state)
1610 1610 else:
1611 1611 self.ui.write(patchname, label='qseries.' + state)
1612 1612 self.ui.write('\n')
1613 1613
1614 1614 applied = set([p.name for p in self.applied])
1615 1615 if length is None:
1616 1616 length = len(self.series) - start
1617 1617 if not missing:
1618 1618 if self.ui.verbose:
1619 1619 idxwidth = len(str(start + length - 1))
1620 1620 for i in xrange(start, start + length):
1621 1621 patch = self.series[i]
1622 1622 if patch in applied:
1623 1623 char, state = 'A', 'applied'
1624 1624 elif self.pushable(i)[0]:
1625 1625 char, state = 'U', 'unapplied'
1626 1626 else:
1627 1627 char, state = 'G', 'guarded'
1628 1628 pfx = ''
1629 1629 if self.ui.verbose:
1630 1630 pfx = '%*d %s ' % (idxwidth, i, char)
1631 1631 elif status and status != char:
1632 1632 continue
1633 1633 displayname(pfx, patch, state)
1634 1634 else:
1635 1635 msng_list = []
1636 1636 for root, dirs, files in os.walk(self.path):
1637 1637 d = root[len(self.path) + 1:]
1638 1638 for f in files:
1639 1639 fl = os.path.join(d, f)
1640 1640 if (fl not in self.series and
1641 1641 fl not in (self.statuspath, self.seriespath,
1642 1642 self.guardspath)
1643 1643 and not fl.startswith('.')):
1644 1644 msng_list.append(fl)
1645 1645 for x in sorted(msng_list):
1646 1646 pfx = self.ui.verbose and ('D ') or ''
1647 1647 displayname(pfx, x, 'missing')
1648 1648
1649 1649 def issaveline(self, l):
1650 1650 if l.name == '.hg.patches.save.line':
1651 1651 return True
1652 1652
1653 1653 def qrepo(self, create=False):
1654 1654 ui = self.ui.copy()
1655 1655 ui.setconfig('paths', 'default', '', overlay=False)
1656 1656 ui.setconfig('paths', 'default-push', '', overlay=False)
1657 1657 if create or os.path.isdir(self.join(".hg")):
1658 1658 return hg.repository(ui, path=self.path, create=create)
1659 1659
1660 1660 def restore(self, repo, rev, delete=None, qupdate=None):
1661 1661 desc = repo[rev].description().strip()
1662 1662 lines = desc.splitlines()
1663 1663 i = 0
1664 1664 datastart = None
1665 1665 series = []
1666 1666 applied = []
1667 1667 qpp = None
1668 1668 for i, line in enumerate(lines):
1669 1669 if line == 'Patch Data:':
1670 1670 datastart = i + 1
1671 1671 elif line.startswith('Dirstate:'):
1672 1672 l = line.rstrip()
1673 1673 l = l[10:].split(' ')
1674 1674 qpp = [bin(x) for x in l]
1675 1675 elif datastart is not None:
1676 1676 l = line.rstrip()
1677 1677 n, name = l.split(':', 1)
1678 1678 if n:
1679 1679 applied.append(statusentry(bin(n), name))
1680 1680 else:
1681 1681 series.append(l)
1682 1682 if datastart is None:
1683 1683 self.ui.warn(_("No saved patch data found\n"))
1684 1684 return 1
1685 1685 self.ui.warn(_("restoring status: %s\n") % lines[0])
1686 1686 self.fullseries = series
1687 1687 self.applied = applied
1688 1688 self.parseseries()
1689 1689 self.seriesdirty = True
1690 1690 self.applieddirty = True
1691 1691 heads = repo.changelog.heads()
1692 1692 if delete:
1693 1693 if rev not in heads:
1694 1694 self.ui.warn(_("save entry has children, leaving it alone\n"))
1695 1695 else:
1696 1696 self.ui.warn(_("removing save entry %s\n") % short(rev))
1697 1697 pp = repo.dirstate.parents()
1698 1698 if rev in pp:
1699 1699 update = True
1700 1700 else:
1701 1701 update = False
1702 1702 self.strip(repo, [rev], update=update, backup='strip')
1703 1703 if qpp:
1704 1704 self.ui.warn(_("saved queue repository parents: %s %s\n") %
1705 1705 (short(qpp[0]), short(qpp[1])))
1706 1706 if qupdate:
1707 1707 self.ui.status(_("updating queue directory\n"))
1708 1708 r = self.qrepo()
1709 1709 if not r:
1710 1710 self.ui.warn(_("Unable to load queue repository\n"))
1711 1711 return 1
1712 1712 hg.clean(r, qpp[0])
1713 1713
1714 1714 def save(self, repo, msg=None):
1715 1715 if not self.applied:
1716 1716 self.ui.warn(_("save: no patches applied, exiting\n"))
1717 1717 return 1
1718 1718 if self.issaveline(self.applied[-1]):
1719 1719 self.ui.warn(_("status is already saved\n"))
1720 1720 return 1
1721 1721
1722 1722 if not msg:
1723 1723 msg = _("hg patches saved state")
1724 1724 else:
1725 1725 msg = "hg patches: " + msg.rstrip('\r\n')
1726 1726 r = self.qrepo()
1727 1727 if r:
1728 1728 pp = r.dirstate.parents()
1729 1729 msg += "\nDirstate: %s %s" % (hex(pp[0]), hex(pp[1]))
1730 1730 msg += "\n\nPatch Data:\n"
1731 1731 msg += ''.join('%s\n' % x for x in self.applied)
1732 1732 msg += ''.join(':%s\n' % x for x in self.fullseries)
1733 1733 n = repo.commit(msg, force=True)
1734 1734 if not n:
1735 1735 self.ui.warn(_("repo commit failed\n"))
1736 1736 return 1
1737 1737 self.applied.append(statusentry(n, '.hg.patches.save.line'))
1738 1738 self.applieddirty = True
1739 1739 self.removeundo(repo)
1740 1740
1741 1741 def fullseriesend(self):
1742 1742 if self.applied:
1743 1743 p = self.applied[-1].name
1744 1744 end = self.findseries(p)
1745 1745 if end is None:
1746 1746 return len(self.fullseries)
1747 1747 return end + 1
1748 1748 return 0
1749 1749
1750 1750 def seriesend(self, all_patches=False):
1751 1751 """If all_patches is False, return the index of the next pushable patch
1752 1752 in the series, or the series length. If all_patches is True, return the
1753 1753 index of the first patch past the last applied one.
1754 1754 """
1755 1755 end = 0
1756 1756 def next(start):
1757 1757 if all_patches or start >= len(self.series):
1758 1758 return start
1759 1759 for i in xrange(start, len(self.series)):
1760 1760 p, reason = self.pushable(i)
1761 1761 if p:
1762 1762 return i
1763 1763 self.explainpushable(i)
1764 1764 return len(self.series)
1765 1765 if self.applied:
1766 1766 p = self.applied[-1].name
1767 1767 try:
1768 1768 end = self.series.index(p)
1769 1769 except ValueError:
1770 1770 return 0
1771 1771 return next(end + 1)
1772 1772 return next(end)
1773 1773
1774 1774 def appliedname(self, index):
1775 1775 pname = self.applied[index].name
1776 1776 if not self.ui.verbose:
1777 1777 p = pname
1778 1778 else:
1779 1779 p = str(self.series.index(pname)) + " " + pname
1780 1780 return p
1781 1781
1782 1782 def qimport(self, repo, files, patchname=None, rev=None, existing=None,
1783 1783 force=None, git=False):
1784 1784 def checkseries(patchname):
1785 1785 if patchname in self.series:
1786 1786 raise util.Abort(_('patch %s is already in the series file')
1787 1787 % patchname)
1788 1788
1789 1789 if rev:
1790 1790 if files:
1791 1791 raise util.Abort(_('option "-r" not valid when importing '
1792 1792 'files'))
1793 1793 rev = scmutil.revrange(repo, rev)
1794 1794 rev.sort(reverse=True)
1795 1795 if (len(files) > 1 or len(rev) > 1) and patchname:
1796 1796 raise util.Abort(_('option "-n" not valid when importing multiple '
1797 1797 'patches'))
1798 1798 imported = []
1799 1799 if rev:
1800 1800 # If mq patches are applied, we can only import revisions
1801 1801 # that form a linear path to qbase.
1802 1802 # Otherwise, they should form a linear path to a head.
1803 1803 heads = repo.changelog.heads(repo.changelog.node(rev[-1]))
1804 1804 if len(heads) > 1:
1805 1805 raise util.Abort(_('revision %d is the root of more than one '
1806 1806 'branch') % rev[-1])
1807 1807 if self.applied:
1808 1808 base = repo.changelog.node(rev[0])
1809 1809 if base in [n.node for n in self.applied]:
1810 1810 raise util.Abort(_('revision %d is already managed')
1811 1811 % rev[0])
1812 1812 if heads != [self.applied[-1].node]:
1813 1813 raise util.Abort(_('revision %d is not the parent of '
1814 1814 'the queue') % rev[0])
1815 1815 base = repo.changelog.rev(self.applied[0].node)
1816 1816 lastparent = repo.changelog.parentrevs(base)[0]
1817 1817 else:
1818 1818 if heads != [repo.changelog.node(rev[0])]:
1819 1819 raise util.Abort(_('revision %d has unmanaged children')
1820 1820 % rev[0])
1821 1821 lastparent = None
1822 1822
1823 1823 diffopts = self.diffopts({'git': git})
1824 1824 for r in rev:
1825 1825 if not repo[r].mutable():
1826 1826 raise util.Abort(_('revision %d is not mutable') % r,
1827 1827 hint=_('see "hg help phases" for details'))
1828 1828 p1, p2 = repo.changelog.parentrevs(r)
1829 1829 n = repo.changelog.node(r)
1830 1830 if p2 != nullrev:
1831 1831 raise util.Abort(_('cannot import merge revision %d') % r)
1832 1832 if lastparent and lastparent != r:
1833 1833 raise util.Abort(_('revision %d is not the parent of %d')
1834 1834 % (r, lastparent))
1835 1835 lastparent = p1
1836 1836
1837 1837 if not patchname:
1838 1838 patchname = normname('%d.diff' % r)
1839 1839 checkseries(patchname)
1840 1840 self.checkpatchname(patchname, force)
1841 1841 self.fullseries.insert(0, patchname)
1842 1842
1843 1843 patchf = self.opener(patchname, "w")
1844 1844 cmdutil.export(repo, [n], fp=patchf, opts=diffopts)
1845 1845 patchf.close()
1846 1846
1847 1847 se = statusentry(n, patchname)
1848 1848 self.applied.insert(0, se)
1849 1849
1850 1850 self.added.append(patchname)
1851 1851 imported.append(patchname)
1852 1852 patchname = None
1853 1853 if rev and repo.ui.configbool('mq', 'secret', False):
1854 1854 # if we added anything with --rev, we must move the secret root
1855 1855 phases.retractboundary(repo, phases.secret, [n])
1856 1856 self.parseseries()
1857 1857 self.applieddirty = True
1858 1858 self.seriesdirty = True
1859 1859
1860 1860 for i, filename in enumerate(files):
1861 1861 if existing:
1862 1862 if filename == '-':
1863 1863 raise util.Abort(_('-e is incompatible with import from -'))
1864 1864 filename = normname(filename)
1865 1865 self.checkreservedname(filename)
1866 1866 originpath = self.join(filename)
1867 1867 if not os.path.isfile(originpath):
1868 1868 raise util.Abort(_("patch %s does not exist") % filename)
1869 1869
1870 1870 if patchname:
1871 1871 self.checkpatchname(patchname, force)
1872 1872
1873 1873 self.ui.write(_('renaming %s to %s\n')
1874 1874 % (filename, patchname))
1875 1875 util.rename(originpath, self.join(patchname))
1876 1876 else:
1877 1877 patchname = filename
1878 1878
1879 1879 else:
1880 1880 if filename == '-' and not patchname:
1881 1881 raise util.Abort(_('need --name to import a patch from -'))
1882 1882 elif not patchname:
1883 1883 patchname = normname(os.path.basename(filename.rstrip('/')))
1884 1884 self.checkpatchname(patchname, force)
1885 1885 try:
1886 1886 if filename == '-':
1887 1887 text = self.ui.fin.read()
1888 1888 else:
1889 1889 fp = url.open(self.ui, filename)
1890 1890 text = fp.read()
1891 1891 fp.close()
1892 1892 except (OSError, IOError):
1893 1893 raise util.Abort(_("unable to read file %s") % filename)
1894 1894 patchf = self.opener(patchname, "w")
1895 1895 patchf.write(text)
1896 1896 patchf.close()
1897 1897 if not force:
1898 1898 checkseries(patchname)
1899 1899 if patchname not in self.series:
1900 1900 index = self.fullseriesend() + i
1901 1901 self.fullseries[index:index] = [patchname]
1902 1902 self.parseseries()
1903 1903 self.seriesdirty = True
1904 1904 self.ui.warn(_("adding %s to series file\n") % patchname)
1905 1905 self.added.append(patchname)
1906 1906 imported.append(patchname)
1907 1907 patchname = None
1908 1908
1909 1909 self.removeundo(repo)
1910 1910 return imported
1911 1911
1912 1912 @command("qdelete|qremove|qrm",
1913 1913 [('k', 'keep', None, _('keep patch file')),
1914 1914 ('r', 'rev', [],
1915 1915 _('stop managing a revision (DEPRECATED)'), _('REV'))],
1916 1916 _('hg qdelete [-k] [PATCH]...'))
1917 1917 def delete(ui, repo, *patches, **opts):
1918 1918 """remove patches from queue
1919 1919
1920 1920 The patches must not be applied, and at least one patch is required. Exact
1921 1921 patch identifiers must be given. With -k/--keep, the patch files are
1922 1922 preserved in the patch directory.
1923 1923
1924 1924 To stop managing a patch and move it into permanent history,
1925 1925 use the :hg:`qfinish` command."""
1926 1926 q = repo.mq
1927 1927 q.delete(repo, patches, opts)
1928 1928 q.savedirty()
1929 1929 return 0
1930 1930
1931 1931 @command("qapplied",
1932 1932 [('1', 'last', None, _('show only the preceding applied patch'))
1933 1933 ] + seriesopts,
1934 1934 _('hg qapplied [-1] [-s] [PATCH]'))
1935 1935 def applied(ui, repo, patch=None, **opts):
1936 1936 """print the patches already applied
1937 1937
1938 1938 Returns 0 on success."""
1939 1939
1940 1940 q = repo.mq
1941 1941
1942 1942 if patch:
1943 1943 if patch not in q.series:
1944 1944 raise util.Abort(_("patch %s is not in series file") % patch)
1945 1945 end = q.series.index(patch) + 1
1946 1946 else:
1947 1947 end = q.seriesend(True)
1948 1948
1949 1949 if opts.get('last') and not end:
1950 1950 ui.write(_("no patches applied\n"))
1951 1951 return 1
1952 1952 elif opts.get('last') and end == 1:
1953 1953 ui.write(_("only one patch applied\n"))
1954 1954 return 1
1955 1955 elif opts.get('last'):
1956 1956 start = end - 2
1957 1957 end = 1
1958 1958 else:
1959 1959 start = 0
1960 1960
1961 1961 q.qseries(repo, length=end, start=start, status='A',
1962 1962 summary=opts.get('summary'))
1963 1963
1964 1964
1965 1965 @command("qunapplied",
1966 1966 [('1', 'first', None, _('show only the first patch'))] + seriesopts,
1967 1967 _('hg qunapplied [-1] [-s] [PATCH]'))
1968 1968 def unapplied(ui, repo, patch=None, **opts):
1969 1969 """print the patches not yet applied
1970 1970
1971 1971 Returns 0 on success."""
1972 1972
1973 1973 q = repo.mq
1974 1974 if patch:
1975 1975 if patch not in q.series:
1976 1976 raise util.Abort(_("patch %s is not in series file") % patch)
1977 1977 start = q.series.index(patch) + 1
1978 1978 else:
1979 1979 start = q.seriesend(True)
1980 1980
1981 1981 if start == len(q.series) and opts.get('first'):
1982 1982 ui.write(_("all patches applied\n"))
1983 1983 return 1
1984 1984
1985 1985 length = opts.get('first') and 1 or None
1986 1986 q.qseries(repo, start=start, length=length, status='U',
1987 1987 summary=opts.get('summary'))
1988 1988
1989 1989 @command("qimport",
1990 1990 [('e', 'existing', None, _('import file in patch directory')),
1991 1991 ('n', 'name', '',
1992 1992 _('name of patch file'), _('NAME')),
1993 1993 ('f', 'force', None, _('overwrite existing files')),
1994 1994 ('r', 'rev', [],
1995 1995 _('place existing revisions under mq control'), _('REV')),
1996 1996 ('g', 'git', None, _('use git extended diff format')),
1997 1997 ('P', 'push', None, _('qpush after importing'))],
1998 1998 _('hg qimport [-e] [-n NAME] [-f] [-g] [-P] [-r REV]... FILE...'))
1999 1999 def qimport(ui, repo, *filename, **opts):
2000 2000 """import a patch
2001 2001
2002 2002 The patch is inserted into the series after the last applied
2003 2003 patch. If no patches have been applied, qimport prepends the patch
2004 2004 to the series.
2005 2005
2006 2006 The patch will have the same name as its source file unless you
2007 2007 give it a new one with -n/--name.
2008 2008
2009 2009 You can register an existing patch inside the patch directory with
2010 2010 the -e/--existing flag.
2011 2011
2012 2012 With -f/--force, an existing patch of the same name will be
2013 2013 overwritten.
2014 2014
2015 2015 An existing changeset may be placed under mq control with -r/--rev
2016 2016 (e.g. qimport --rev tip -n patch will place tip under mq control).
2017 2017 With -g/--git, patches imported with --rev will use the git diff
2018 2018 format. See the diffs help topic for information on why this is
2019 2019 important for preserving rename/copy information and permission
2020 2020 changes. Use :hg:`qfinish` to remove changesets from mq control.
2021 2021
2022 2022 To import a patch from standard input, pass - as the patch file.
2023 2023 When importing from standard input, a patch name must be specified
2024 2024 using the --name flag.
2025 2025
2026 2026 To import an existing patch while renaming it::
2027 2027
2028 2028 hg qimport -e existing-patch -n new-name
2029 2029
2030 2030 Returns 0 if import succeeded.
2031 2031 """
2032 2032 lock = repo.lock() # cause this may move phase
2033 2033 try:
2034 2034 q = repo.mq
2035 2035 try:
2036 2036 imported = q.qimport(
2037 2037 repo, filename, patchname=opts.get('name'),
2038 2038 existing=opts.get('existing'), force=opts.get('force'),
2039 2039 rev=opts.get('rev'), git=opts.get('git'))
2040 2040 finally:
2041 2041 q.savedirty()
2042 2042
2043 2043
2044 2044 if imported and opts.get('push') and not opts.get('rev'):
2045 2045 return q.push(repo, imported[-1])
2046 2046 finally:
2047 2047 lock.release()
2048 2048 return 0
2049 2049
2050 2050 def qinit(ui, repo, create):
2051 2051 """initialize a new queue repository
2052 2052
2053 2053 This command also creates a series file for ordering patches, and
2054 2054 an mq-specific .hgignore file in the queue repository, to exclude
2055 2055 the status and guards files (these contain mostly transient state).
2056 2056
2057 2057 Returns 0 if initialization succeeded."""
2058 2058 q = repo.mq
2059 2059 r = q.init(repo, create)
2060 2060 q.savedirty()
2061 2061 if r:
2062 2062 if not os.path.exists(r.wjoin('.hgignore')):
2063 2063 fp = r.wopener('.hgignore', 'w')
2064 2064 fp.write('^\\.hg\n')
2065 2065 fp.write('^\\.mq\n')
2066 2066 fp.write('syntax: glob\n')
2067 2067 fp.write('status\n')
2068 2068 fp.write('guards\n')
2069 2069 fp.close()
2070 2070 if not os.path.exists(r.wjoin('series')):
2071 2071 r.wopener('series', 'w').close()
2072 2072 r[None].add(['.hgignore', 'series'])
2073 2073 commands.add(ui, r)
2074 2074 return 0
2075 2075
2076 2076 @command("^qinit",
2077 2077 [('c', 'create-repo', None, _('create queue repository'))],
2078 2078 _('hg qinit [-c]'))
2079 2079 def init(ui, repo, **opts):
2080 2080 """init a new queue repository (DEPRECATED)
2081 2081
2082 2082 The queue repository is unversioned by default. If
2083 2083 -c/--create-repo is specified, qinit will create a separate nested
2084 2084 repository for patches (qinit -c may also be run later to convert
2085 2085 an unversioned patch repository into a versioned one). You can use
2086 2086 qcommit to commit changes to this queue repository.
2087 2087
2088 2088 This command is deprecated. Without -c, it's implied by other relevant
2089 2089 commands. With -c, use :hg:`init --mq` instead."""
2090 2090 return qinit(ui, repo, create=opts.get('create_repo'))
2091 2091
2092 2092 @command("qclone",
2093 2093 [('', 'pull', None, _('use pull protocol to copy metadata')),
2094 2094 ('U', 'noupdate', None, _('do not update the new working directories')),
2095 2095 ('', 'uncompressed', None,
2096 2096 _('use uncompressed transfer (fast over LAN)')),
2097 2097 ('p', 'patches', '',
2098 2098 _('location of source patch repository'), _('REPO')),
2099 2099 ] + commands.remoteopts,
2100 2100 _('hg qclone [OPTION]... SOURCE [DEST]'))
2101 2101 def clone(ui, source, dest=None, **opts):
2102 2102 '''clone main and patch repository at same time
2103 2103
2104 2104 If source is local, destination will have no patches applied. If
2105 2105 source is remote, this command can not check if patches are
2106 2106 applied in source, so cannot guarantee that patches are not
2107 2107 applied in destination. If you clone remote repository, be sure
2108 2108 before that it has no patches applied.
2109 2109
2110 2110 Source patch repository is looked for in <src>/.hg/patches by
2111 2111 default. Use -p <url> to change.
2112 2112
2113 2113 The patch directory must be a nested Mercurial repository, as
2114 2114 would be created by :hg:`init --mq`.
2115 2115
2116 2116 Return 0 on success.
2117 2117 '''
2118 2118 def patchdir(repo):
2119 2119 """compute a patch repo url from a repo object"""
2120 2120 url = repo.url()
2121 2121 if url.endswith('/'):
2122 2122 url = url[:-1]
2123 2123 return url + '/.hg/patches'
2124 2124
2125 2125 # main repo (destination and sources)
2126 2126 if dest is None:
2127 2127 dest = hg.defaultdest(source)
2128 2128 sr = hg.repository(hg.remoteui(ui, opts), ui.expandpath(source))
2129 2129
2130 2130 # patches repo (source only)
2131 2131 if opts.get('patches'):
2132 2132 patchespath = ui.expandpath(opts.get('patches'))
2133 2133 else:
2134 2134 patchespath = patchdir(sr)
2135 2135 try:
2136 2136 hg.repository(ui, patchespath)
2137 2137 except error.RepoError:
2138 2138 raise util.Abort(_('versioned patch repository not found'
2139 2139 ' (see init --mq)'))
2140 2140 qbase, destrev = None, None
2141 2141 if sr.local():
2142 2142 if sr.mq.applied and sr[qbase].phase() != phases.secret:
2143 2143 qbase = sr.mq.applied[0].node
2144 2144 if not hg.islocal(dest):
2145 2145 heads = set(sr.heads())
2146 2146 destrev = list(heads.difference(sr.heads(qbase)))
2147 2147 destrev.append(sr.changelog.parents(qbase)[0])
2148 2148 elif sr.capable('lookup'):
2149 2149 try:
2150 2150 qbase = sr.lookup('qbase')
2151 2151 except error.RepoError:
2152 2152 pass
2153 2153
2154 2154 ui.note(_('cloning main repository\n'))
2155 2155 sr, dr = hg.clone(ui, opts, sr.url(), dest,
2156 2156 pull=opts.get('pull'),
2157 2157 rev=destrev,
2158 2158 update=False,
2159 2159 stream=opts.get('uncompressed'))
2160 2160
2161 2161 ui.note(_('cloning patch repository\n'))
2162 2162 hg.clone(ui, opts, opts.get('patches') or patchdir(sr), patchdir(dr),
2163 2163 pull=opts.get('pull'), update=not opts.get('noupdate'),
2164 2164 stream=opts.get('uncompressed'))
2165 2165
2166 2166 if dr.local():
2167 2167 if qbase:
2168 2168 ui.note(_('stripping applied patches from destination '
2169 2169 'repository\n'))
2170 2170 dr.mq.strip(dr, [qbase], update=False, backup=None)
2171 2171 if not opts.get('noupdate'):
2172 2172 ui.note(_('updating destination repository\n'))
2173 2173 hg.update(dr, dr.changelog.tip())
2174 2174
2175 2175 @command("qcommit|qci",
2176 2176 commands.table["^commit|ci"][1],
2177 2177 _('hg qcommit [OPTION]... [FILE]...'))
2178 2178 def commit(ui, repo, *pats, **opts):
2179 2179 """commit changes in the queue repository (DEPRECATED)
2180 2180
2181 2181 This command is deprecated; use :hg:`commit --mq` instead."""
2182 2182 q = repo.mq
2183 2183 r = q.qrepo()
2184 2184 if not r:
2185 2185 raise util.Abort('no queue repository')
2186 2186 commands.commit(r.ui, r, *pats, **opts)
2187 2187
2188 2188 @command("qseries",
2189 2189 [('m', 'missing', None, _('print patches not in series')),
2190 2190 ] + seriesopts,
2191 2191 _('hg qseries [-ms]'))
2192 2192 def series(ui, repo, **opts):
2193 2193 """print the entire series file
2194 2194
2195 2195 Returns 0 on success."""
2196 2196 repo.mq.qseries(repo, missing=opts.get('missing'), summary=opts.get('summary'))
2197 2197 return 0
2198 2198
2199 2199 @command("qtop", seriesopts, _('hg qtop [-s]'))
2200 2200 def top(ui, repo, **opts):
2201 2201 """print the name of the current patch
2202 2202
2203 2203 Returns 0 on success."""
2204 2204 q = repo.mq
2205 2205 t = q.applied and q.seriesend(True) or 0
2206 2206 if t:
2207 2207 q.qseries(repo, start=t - 1, length=1, status='A',
2208 2208 summary=opts.get('summary'))
2209 2209 else:
2210 2210 ui.write(_("no patches applied\n"))
2211 2211 return 1
2212 2212
2213 2213 @command("qnext", seriesopts, _('hg qnext [-s]'))
2214 2214 def next(ui, repo, **opts):
2215 2215 """print the name of the next pushable patch
2216 2216
2217 2217 Returns 0 on success."""
2218 2218 q = repo.mq
2219 2219 end = q.seriesend()
2220 2220 if end == len(q.series):
2221 2221 ui.write(_("all patches applied\n"))
2222 2222 return 1
2223 2223 q.qseries(repo, start=end, length=1, summary=opts.get('summary'))
2224 2224
2225 2225 @command("qprev", seriesopts, _('hg qprev [-s]'))
2226 2226 def prev(ui, repo, **opts):
2227 2227 """print the name of the preceding applied patch
2228 2228
2229 2229 Returns 0 on success."""
2230 2230 q = repo.mq
2231 2231 l = len(q.applied)
2232 2232 if l == 1:
2233 2233 ui.write(_("only one patch applied\n"))
2234 2234 return 1
2235 2235 if not l:
2236 2236 ui.write(_("no patches applied\n"))
2237 2237 return 1
2238 2238 idx = q.series.index(q.applied[-2].name)
2239 2239 q.qseries(repo, start=idx, length=1, status='A',
2240 2240 summary=opts.get('summary'))
2241 2241
2242 2242 def setupheaderopts(ui, opts):
2243 2243 if not opts.get('user') and opts.get('currentuser'):
2244 2244 opts['user'] = ui.username()
2245 2245 if not opts.get('date') and opts.get('currentdate'):
2246 2246 opts['date'] = "%d %d" % util.makedate()
2247 2247
2248 2248 @command("^qnew",
2249 2249 [('e', 'edit', None, _('edit commit message')),
2250 2250 ('f', 'force', None, _('import uncommitted changes (DEPRECATED)')),
2251 2251 ('g', 'git', None, _('use git extended diff format')),
2252 2252 ('U', 'currentuser', None, _('add "From: <current user>" to patch')),
2253 2253 ('u', 'user', '',
2254 2254 _('add "From: <USER>" to patch'), _('USER')),
2255 2255 ('D', 'currentdate', None, _('add "Date: <current date>" to patch')),
2256 2256 ('d', 'date', '',
2257 2257 _('add "Date: <DATE>" to patch'), _('DATE'))
2258 2258 ] + commands.walkopts + commands.commitopts,
2259 2259 _('hg qnew [-e] [-m TEXT] [-l FILE] PATCH [FILE]...'))
2260 2260 def new(ui, repo, patch, *args, **opts):
2261 2261 """create a new patch
2262 2262
2263 2263 qnew creates a new patch on top of the currently-applied patch (if
2264 2264 any). The patch will be initialized with any outstanding changes
2265 2265 in the working directory. You may also use -I/--include,
2266 2266 -X/--exclude, and/or a list of files after the patch name to add
2267 2267 only changes to matching files to the new patch, leaving the rest
2268 2268 as uncommitted modifications.
2269 2269
2270 2270 -u/--user and -d/--date can be used to set the (given) user and
2271 2271 date, respectively. -U/--currentuser and -D/--currentdate set user
2272 2272 to current user and date to current date.
2273 2273
2274 2274 -e/--edit, -m/--message or -l/--logfile set the patch header as
2275 2275 well as the commit message. If none is specified, the header is
2276 2276 empty and the commit message is '[mq]: PATCH'.
2277 2277
2278 2278 Use the -g/--git option to keep the patch in the git extended diff
2279 2279 format. Read the diffs help topic for more information on why this
2280 2280 is important for preserving permission changes and copy/rename
2281 2281 information.
2282 2282
2283 2283 Returns 0 on successful creation of a new patch.
2284 2284 """
2285 2285 msg = cmdutil.logmessage(ui, opts)
2286 2286 def getmsg():
2287 2287 return ui.edit(msg, opts.get('user') or ui.username())
2288 2288 q = repo.mq
2289 2289 opts['msg'] = msg
2290 2290 if opts.get('edit'):
2291 2291 opts['msg'] = getmsg
2292 2292 else:
2293 2293 opts['msg'] = msg
2294 2294 setupheaderopts(ui, opts)
2295 2295 q.new(repo, patch, *args, **opts)
2296 2296 q.savedirty()
2297 2297 return 0
2298 2298
2299 2299 @command("^qrefresh",
2300 2300 [('e', 'edit', None, _('edit commit message')),
2301 2301 ('g', 'git', None, _('use git extended diff format')),
2302 2302 ('s', 'short', None,
2303 2303 _('refresh only files already in the patch and specified files')),
2304 2304 ('U', 'currentuser', None,
2305 2305 _('add/update author field in patch with current user')),
2306 2306 ('u', 'user', '',
2307 2307 _('add/update author field in patch with given user'), _('USER')),
2308 2308 ('D', 'currentdate', None,
2309 2309 _('add/update date field in patch with current date')),
2310 2310 ('d', 'date', '',
2311 2311 _('add/update date field in patch with given date'), _('DATE'))
2312 2312 ] + commands.walkopts + commands.commitopts,
2313 2313 _('hg qrefresh [-I] [-X] [-e] [-m TEXT] [-l FILE] [-s] [FILE]...'))
2314 2314 def refresh(ui, repo, *pats, **opts):
2315 2315 """update the current patch
2316 2316
2317 2317 If any file patterns are provided, the refreshed patch will
2318 2318 contain only the modifications that match those patterns; the
2319 2319 remaining modifications will remain in the working directory.
2320 2320
2321 2321 If -s/--short is specified, files currently included in the patch
2322 2322 will be refreshed just like matched files and remain in the patch.
2323 2323
2324 2324 If -e/--edit is specified, Mercurial will start your configured editor for
2325 2325 you to enter a message. In case qrefresh fails, you will find a backup of
2326 2326 your message in ``.hg/last-message.txt``.
2327 2327
2328 2328 hg add/remove/copy/rename work as usual, though you might want to
2329 2329 use git-style patches (-g/--git or [diff] git=1) to track copies
2330 2330 and renames. See the diffs help topic for more information on the
2331 2331 git diff format.
2332 2332
2333 2333 Returns 0 on success.
2334 2334 """
2335 2335 q = repo.mq
2336 2336 message = cmdutil.logmessage(ui, opts)
2337 2337 if opts.get('edit'):
2338 2338 if not q.applied:
2339 2339 ui.write(_("no patches applied\n"))
2340 2340 return 1
2341 2341 if message:
2342 2342 raise util.Abort(_('option "-e" incompatible with "-m" or "-l"'))
2343 2343 patch = q.applied[-1].name
2344 2344 ph = patchheader(q.join(patch), q.plainmode)
2345 2345 message = ui.edit('\n'.join(ph.message), ph.user or ui.username())
2346 2346 # We don't want to lose the patch message if qrefresh fails (issue2062)
2347 2347 repo.savecommitmessage(message)
2348 2348 setupheaderopts(ui, opts)
2349 2349 wlock = repo.wlock()
2350 2350 try:
2351 2351 ret = q.refresh(repo, pats, msg=message, **opts)
2352 2352 q.savedirty()
2353 2353 return ret
2354 2354 finally:
2355 2355 wlock.release()
2356 2356
2357 2357 @command("^qdiff",
2358 2358 commands.diffopts + commands.diffopts2 + commands.walkopts,
2359 2359 _('hg qdiff [OPTION]... [FILE]...'))
2360 2360 def diff(ui, repo, *pats, **opts):
2361 2361 """diff of the current patch and subsequent modifications
2362 2362
2363 2363 Shows a diff which includes the current patch as well as any
2364 2364 changes which have been made in the working directory since the
2365 2365 last refresh (thus showing what the current patch would become
2366 2366 after a qrefresh).
2367 2367
2368 2368 Use :hg:`diff` if you only want to see the changes made since the
2369 2369 last qrefresh, or :hg:`export qtip` if you want to see changes
2370 2370 made by the current patch without including changes made since the
2371 2371 qrefresh.
2372 2372
2373 2373 Returns 0 on success.
2374 2374 """
2375 2375 repo.mq.diff(repo, pats, opts)
2376 2376 return 0
2377 2377
2378 2378 @command('qfold',
2379 2379 [('e', 'edit', None, _('edit patch header')),
2380 2380 ('k', 'keep', None, _('keep folded patch files')),
2381 2381 ] + commands.commitopts,
2382 2382 _('hg qfold [-e] [-k] [-m TEXT] [-l FILE] PATCH...'))
2383 2383 def fold(ui, repo, *files, **opts):
2384 2384 """fold the named patches into the current patch
2385 2385
2386 2386 Patches must not yet be applied. Each patch will be successively
2387 2387 applied to the current patch in the order given. If all the
2388 2388 patches apply successfully, the current patch will be refreshed
2389 2389 with the new cumulative patch, and the folded patches will be
2390 2390 deleted. With -k/--keep, the folded patch files will not be
2391 2391 removed afterwards.
2392 2392
2393 2393 The header for each folded patch will be concatenated with the
2394 2394 current patch header, separated by a line of ``* * *``.
2395 2395
2396 2396 Returns 0 on success."""
2397 2397 q = repo.mq
2398 2398 if not files:
2399 2399 raise util.Abort(_('qfold requires at least one patch name'))
2400 2400 if not q.checktoppatch(repo)[0]:
2401 2401 raise util.Abort(_('no patches applied'))
2402 2402 q.checklocalchanges(repo)
2403 2403
2404 2404 message = cmdutil.logmessage(ui, opts)
2405 2405 if opts.get('edit'):
2406 2406 if message:
2407 2407 raise util.Abort(_('option "-e" incompatible with "-m" or "-l"'))
2408 2408
2409 2409 parent = q.lookup('qtip')
2410 2410 patches = []
2411 2411 messages = []
2412 2412 for f in files:
2413 2413 p = q.lookup(f)
2414 2414 if p in patches or p == parent:
2415 2415 ui.warn(_('Skipping already folded patch %s\n') % p)
2416 2416 if q.isapplied(p):
2417 2417 raise util.Abort(_('qfold cannot fold already applied patch %s') % p)
2418 2418 patches.append(p)
2419 2419
2420 2420 for p in patches:
2421 2421 if not message:
2422 2422 ph = patchheader(q.join(p), q.plainmode)
2423 2423 if ph.message:
2424 2424 messages.append(ph.message)
2425 2425 pf = q.join(p)
2426 2426 (patchsuccess, files, fuzz) = q.patch(repo, pf)
2427 2427 if not patchsuccess:
2428 2428 raise util.Abort(_('error folding patch %s') % p)
2429 2429
2430 2430 if not message:
2431 2431 ph = patchheader(q.join(parent), q.plainmode)
2432 2432 message, user = ph.message, ph.user
2433 2433 for msg in messages:
2434 2434 message.append('* * *')
2435 2435 message.extend(msg)
2436 2436 message = '\n'.join(message)
2437 2437
2438 2438 if opts.get('edit'):
2439 2439 message = ui.edit(message, user or ui.username())
2440 2440
2441 2441 diffopts = q.patchopts(q.diffopts(), *patches)
2442 2442 wlock = repo.wlock()
2443 2443 try:
2444 2444 q.refresh(repo, msg=message, git=diffopts.git)
2445 2445 q.delete(repo, patches, opts)
2446 2446 q.savedirty()
2447 2447 finally:
2448 2448 wlock.release()
2449 2449
2450 2450 @command("qgoto",
2451 2451 [('f', 'force', None, _('overwrite any local changes'))],
2452 2452 _('hg qgoto [OPTION]... PATCH'))
2453 2453 def goto(ui, repo, patch, **opts):
2454 2454 '''push or pop patches until named patch is at top of stack
2455 2455
2456 2456 Returns 0 on success.'''
2457 2457 q = repo.mq
2458 2458 patch = q.lookup(patch)
2459 2459 if q.isapplied(patch):
2460 2460 ret = q.pop(repo, patch, force=opts.get('force'))
2461 2461 else:
2462 2462 ret = q.push(repo, patch, force=opts.get('force'))
2463 2463 q.savedirty()
2464 2464 return ret
2465 2465
2466 2466 @command("qguard",
2467 2467 [('l', 'list', None, _('list all patches and guards')),
2468 2468 ('n', 'none', None, _('drop all guards'))],
2469 2469 _('hg qguard [-l] [-n] [PATCH] [-- [+GUARD]... [-GUARD]...]'))
2470 2470 def guard(ui, repo, *args, **opts):
2471 2471 '''set or print guards for a patch
2472 2472
2473 2473 Guards control whether a patch can be pushed. A patch with no
2474 2474 guards is always pushed. A patch with a positive guard ("+foo") is
2475 2475 pushed only if the :hg:`qselect` command has activated it. A patch with
2476 2476 a negative guard ("-foo") is never pushed if the :hg:`qselect` command
2477 2477 has activated it.
2478 2478
2479 2479 With no arguments, print the currently active guards.
2480 2480 With arguments, set guards for the named patch.
2481 2481
2482 2482 .. note::
2483 2483 Specifying negative guards now requires '--'.
2484 2484
2485 2485 To set guards on another patch::
2486 2486
2487 2487 hg qguard other.patch -- +2.6.17 -stable
2488 2488
2489 2489 Returns 0 on success.
2490 2490 '''
2491 2491 def status(idx):
2492 2492 guards = q.seriesguards[idx] or ['unguarded']
2493 2493 if q.series[idx] in applied:
2494 2494 state = 'applied'
2495 2495 elif q.pushable(idx)[0]:
2496 2496 state = 'unapplied'
2497 2497 else:
2498 2498 state = 'guarded'
2499 2499 label = 'qguard.patch qguard.%s qseries.%s' % (state, state)
2500 2500 ui.write('%s: ' % ui.label(q.series[idx], label))
2501 2501
2502 2502 for i, guard in enumerate(guards):
2503 2503 if guard.startswith('+'):
2504 2504 ui.write(guard, label='qguard.positive')
2505 2505 elif guard.startswith('-'):
2506 2506 ui.write(guard, label='qguard.negative')
2507 2507 else:
2508 2508 ui.write(guard, label='qguard.unguarded')
2509 2509 if i != len(guards) - 1:
2510 2510 ui.write(' ')
2511 2511 ui.write('\n')
2512 2512 q = repo.mq
2513 2513 applied = set(p.name for p in q.applied)
2514 2514 patch = None
2515 2515 args = list(args)
2516 2516 if opts.get('list'):
2517 2517 if args or opts.get('none'):
2518 2518 raise util.Abort(_('cannot mix -l/--list with options or arguments'))
2519 2519 for i in xrange(len(q.series)):
2520 2520 status(i)
2521 2521 return
2522 2522 if not args or args[0][0:1] in '-+':
2523 2523 if not q.applied:
2524 2524 raise util.Abort(_('no patches applied'))
2525 2525 patch = q.applied[-1].name
2526 2526 if patch is None and args[0][0:1] not in '-+':
2527 2527 patch = args.pop(0)
2528 2528 if patch is None:
2529 2529 raise util.Abort(_('no patch to work with'))
2530 2530 if args or opts.get('none'):
2531 2531 idx = q.findseries(patch)
2532 2532 if idx is None:
2533 2533 raise util.Abort(_('no patch named %s') % patch)
2534 2534 q.setguards(idx, args)
2535 2535 q.savedirty()
2536 2536 else:
2537 2537 status(q.series.index(q.lookup(patch)))
2538 2538
2539 2539 @command("qheader", [], _('hg qheader [PATCH]'))
2540 2540 def header(ui, repo, patch=None):
2541 2541 """print the header of the topmost or specified patch
2542 2542
2543 2543 Returns 0 on success."""
2544 2544 q = repo.mq
2545 2545
2546 2546 if patch:
2547 2547 patch = q.lookup(patch)
2548 2548 else:
2549 2549 if not q.applied:
2550 2550 ui.write(_('no patches applied\n'))
2551 2551 return 1
2552 2552 patch = q.lookup('qtip')
2553 2553 ph = patchheader(q.join(patch), q.plainmode)
2554 2554
2555 2555 ui.write('\n'.join(ph.message) + '\n')
2556 2556
2557 2557 def lastsavename(path):
2558 2558 (directory, base) = os.path.split(path)
2559 2559 names = os.listdir(directory)
2560 2560 namere = re.compile("%s.([0-9]+)" % base)
2561 2561 maxindex = None
2562 2562 maxname = None
2563 2563 for f in names:
2564 2564 m = namere.match(f)
2565 2565 if m:
2566 2566 index = int(m.group(1))
2567 2567 if maxindex is None or index > maxindex:
2568 2568 maxindex = index
2569 2569 maxname = f
2570 2570 if maxname:
2571 2571 return (os.path.join(directory, maxname), maxindex)
2572 2572 return (None, None)
2573 2573
2574 2574 def savename(path):
2575 2575 (last, index) = lastsavename(path)
2576 2576 if last is None:
2577 2577 index = 0
2578 2578 newpath = path + ".%d" % (index + 1)
2579 2579 return newpath
2580 2580
2581 2581 @command("^qpush",
2582 2582 [('f', 'force', None, _('apply on top of local changes')),
2583 2583 ('e', 'exact', None, _('apply the target patch to its recorded parent')),
2584 2584 ('l', 'list', None, _('list patch name in commit text')),
2585 2585 ('a', 'all', None, _('apply all patches')),
2586 2586 ('m', 'merge', None, _('merge from another queue (DEPRECATED)')),
2587 2587 ('n', 'name', '',
2588 2588 _('merge queue name (DEPRECATED)'), _('NAME')),
2589 2589 ('', 'move', None, _('reorder patch series and apply only the patch'))],
2590 2590 _('hg qpush [-f] [-l] [-a] [--move] [PATCH | INDEX]'))
2591 2591 def push(ui, repo, patch=None, **opts):
2592 2592 """push the next patch onto the stack
2593 2593
2594 2594 When -f/--force is applied, all local changes in patched files
2595 2595 will be lost.
2596 2596
2597 2597 Return 0 on success.
2598 2598 """
2599 2599 q = repo.mq
2600 2600 mergeq = None
2601 2601
2602 2602 if opts.get('merge'):
2603 2603 if opts.get('name'):
2604 2604 newpath = repo.join(opts.get('name'))
2605 2605 else:
2606 2606 newpath, i = lastsavename(q.path)
2607 2607 if not newpath:
2608 2608 ui.warn(_("no saved queues found, please use -n\n"))
2609 2609 return 1
2610 2610 mergeq = queue(ui, repo.path, newpath)
2611 2611 ui.warn(_("merging with queue at: %s\n") % mergeq.path)
2612 2612 ret = q.push(repo, patch, force=opts.get('force'), list=opts.get('list'),
2613 2613 mergeq=mergeq, all=opts.get('all'), move=opts.get('move'),
2614 2614 exact=opts.get('exact'))
2615 2615 return ret
2616 2616
2617 2617 @command("^qpop",
2618 2618 [('a', 'all', None, _('pop all patches')),
2619 2619 ('n', 'name', '',
2620 2620 _('queue name to pop (DEPRECATED)'), _('NAME')),
2621 2621 ('f', 'force', None, _('forget any local changes to patched files'))],
2622 2622 _('hg qpop [-a] [-f] [PATCH | INDEX]'))
2623 2623 def pop(ui, repo, patch=None, **opts):
2624 2624 """pop the current patch off the stack
2625 2625
2626 2626 By default, pops off the top of the patch stack. If given a patch
2627 2627 name, keeps popping off patches until the named patch is at the
2628 2628 top of the stack.
2629 2629
2630 2630 Return 0 on success.
2631 2631 """
2632 2632 localupdate = True
2633 2633 if opts.get('name'):
2634 2634 q = queue(ui, repo.path, repo.join(opts.get('name')))
2635 2635 ui.warn(_('using patch queue: %s\n') % q.path)
2636 2636 localupdate = False
2637 2637 else:
2638 2638 q = repo.mq
2639 2639 ret = q.pop(repo, patch, force=opts.get('force'), update=localupdate,
2640 2640 all=opts.get('all'))
2641 2641 q.savedirty()
2642 2642 return ret
2643 2643
2644 2644 @command("qrename|qmv", [], _('hg qrename PATCH1 [PATCH2]'))
2645 2645 def rename(ui, repo, patch, name=None, **opts):
2646 2646 """rename a patch
2647 2647
2648 2648 With one argument, renames the current patch to PATCH1.
2649 2649 With two arguments, renames PATCH1 to PATCH2.
2650 2650
2651 2651 Returns 0 on success."""
2652 2652 q = repo.mq
2653 2653 if not name:
2654 2654 name = patch
2655 2655 patch = None
2656 2656
2657 2657 if patch:
2658 2658 patch = q.lookup(patch)
2659 2659 else:
2660 2660 if not q.applied:
2661 2661 ui.write(_('no patches applied\n'))
2662 2662 return
2663 2663 patch = q.lookup('qtip')
2664 2664 absdest = q.join(name)
2665 2665 if os.path.isdir(absdest):
2666 2666 name = normname(os.path.join(name, os.path.basename(patch)))
2667 2667 absdest = q.join(name)
2668 2668 q.checkpatchname(name)
2669 2669
2670 2670 ui.note(_('renaming %s to %s\n') % (patch, name))
2671 2671 i = q.findseries(patch)
2672 2672 guards = q.guard_re.findall(q.fullseries[i])
2673 2673 q.fullseries[i] = name + ''.join([' #' + g for g in guards])
2674 2674 q.parseseries()
2675 2675 q.seriesdirty = True
2676 2676
2677 2677 info = q.isapplied(patch)
2678 2678 if info:
2679 2679 q.applied[info[0]] = statusentry(info[1], name)
2680 2680 q.applieddirty = True
2681 2681
2682 2682 destdir = os.path.dirname(absdest)
2683 2683 if not os.path.isdir(destdir):
2684 2684 os.makedirs(destdir)
2685 2685 util.rename(q.join(patch), absdest)
2686 2686 r = q.qrepo()
2687 2687 if r and patch in r.dirstate:
2688 2688 wctx = r[None]
2689 2689 wlock = r.wlock()
2690 2690 try:
2691 2691 if r.dirstate[patch] == 'a':
2692 2692 r.dirstate.drop(patch)
2693 2693 r.dirstate.add(name)
2694 2694 else:
2695 2695 wctx.copy(patch, name)
2696 2696 wctx.forget([patch])
2697 2697 finally:
2698 2698 wlock.release()
2699 2699
2700 2700 q.savedirty()
2701 2701
2702 2702 @command("qrestore",
2703 2703 [('d', 'delete', None, _('delete save entry')),
2704 2704 ('u', 'update', None, _('update queue working directory'))],
2705 2705 _('hg qrestore [-d] [-u] REV'))
2706 2706 def restore(ui, repo, rev, **opts):
2707 2707 """restore the queue state saved by a revision (DEPRECATED)
2708 2708
2709 2709 This command is deprecated, use :hg:`rebase` instead."""
2710 2710 rev = repo.lookup(rev)
2711 2711 q = repo.mq
2712 2712 q.restore(repo, rev, delete=opts.get('delete'),
2713 2713 qupdate=opts.get('update'))
2714 2714 q.savedirty()
2715 2715 return 0
2716 2716
2717 2717 @command("qsave",
2718 2718 [('c', 'copy', None, _('copy patch directory')),
2719 2719 ('n', 'name', '',
2720 2720 _('copy directory name'), _('NAME')),
2721 2721 ('e', 'empty', None, _('clear queue status file')),
2722 2722 ('f', 'force', None, _('force copy'))] + commands.commitopts,
2723 2723 _('hg qsave [-m TEXT] [-l FILE] [-c] [-n NAME] [-e] [-f]'))
2724 2724 def save(ui, repo, **opts):
2725 2725 """save current queue state (DEPRECATED)
2726 2726
2727 2727 This command is deprecated, use :hg:`rebase` instead."""
2728 2728 q = repo.mq
2729 2729 message = cmdutil.logmessage(ui, opts)
2730 2730 ret = q.save(repo, msg=message)
2731 2731 if ret:
2732 2732 return ret
2733 2733 q.savedirty() # save to .hg/patches before copying
2734 2734 if opts.get('copy'):
2735 2735 path = q.path
2736 2736 if opts.get('name'):
2737 2737 newpath = os.path.join(q.basepath, opts.get('name'))
2738 2738 if os.path.exists(newpath):
2739 2739 if not os.path.isdir(newpath):
2740 2740 raise util.Abort(_('destination %s exists and is not '
2741 2741 'a directory') % newpath)
2742 2742 if not opts.get('force'):
2743 2743 raise util.Abort(_('destination %s exists, '
2744 2744 'use -f to force') % newpath)
2745 2745 else:
2746 2746 newpath = savename(path)
2747 2747 ui.warn(_("copy %s to %s\n") % (path, newpath))
2748 2748 util.copyfiles(path, newpath)
2749 2749 if opts.get('empty'):
2750 2750 del q.applied[:]
2751 2751 q.applieddirty = True
2752 2752 q.savedirty()
2753 2753 return 0
2754 2754
2755 2755 @command("strip",
2756 2756 [
2757 2757 ('r', 'rev', [], _('strip specified revision (optional, '
2758 2758 'can specify revisions without this '
2759 2759 'option)'), _('REV')),
2760 2760 ('f', 'force', None, _('force removal of changesets, discard '
2761 2761 'uncommitted changes (no backup)')),
2762 2762 ('b', 'backup', None, _('bundle only changesets with local revision'
2763 2763 ' number greater than REV which are not'
2764 2764 ' descendants of REV (DEPRECATED)')),
2765 ('n', 'no-backup', None, _('no backups')),
2765 ('', 'no-backup', None, _('no backups')),
2766 2766 ('', 'nobackup', None, _('no backups (DEPRECATED)')),
2767 ('n', '', None, _('ignored (DEPRECATED)')),
2767 2768 ('k', 'keep', None, _("do not modify working copy during strip"))],
2768 2769 _('hg strip [-k] [-f] [-n] REV...'))
2769 2770 def strip(ui, repo, *revs, **opts):
2770 2771 """strip changesets and all their descendants from the repository
2771 2772
2772 2773 The strip command removes the specified changesets and all their
2773 2774 descendants. If the working directory has uncommitted changes, the
2774 2775 operation is aborted unless the --force flag is supplied, in which
2775 2776 case changes will be discarded.
2776 2777
2777 2778 If a parent of the working directory is stripped, then the working
2778 2779 directory will automatically be updated to the most recent
2779 2780 available ancestor of the stripped parent after the operation
2780 2781 completes.
2781 2782
2782 2783 Any stripped changesets are stored in ``.hg/strip-backup`` as a
2783 2784 bundle (see :hg:`help bundle` and :hg:`help unbundle`). They can
2784 2785 be restored by running :hg:`unbundle .hg/strip-backup/BUNDLE`,
2785 2786 where BUNDLE is the bundle file created by the strip. Note that
2786 2787 the local revision numbers will in general be different after the
2787 2788 restore.
2788 2789
2789 2790 Use the --no-backup option to discard the backup bundle once the
2790 2791 operation completes.
2791 2792
2792 2793 Return 0 on success.
2793 2794 """
2794 2795 backup = 'all'
2795 2796 if opts.get('backup'):
2796 2797 backup = 'strip'
2797 2798 elif opts.get('no_backup') or opts.get('nobackup'):
2798 2799 backup = 'none'
2799 2800
2800 2801 cl = repo.changelog
2801 2802 revs = list(revs) + opts.get('rev')
2802 2803 revs = set(scmutil.revrange(repo, revs))
2803 2804 if not revs:
2804 2805 raise util.Abort(_('empty revision set'))
2805 2806
2806 2807 descendants = set(cl.descendants(*revs))
2807 2808 strippedrevs = revs.union(descendants)
2808 2809 roots = revs.difference(descendants)
2809 2810
2810 2811 update = False
2811 2812 # if one of the wdir parent is stripped we'll need
2812 2813 # to update away to an earlier revision
2813 2814 for p in repo.dirstate.parents():
2814 2815 if p != nullid and cl.rev(p) in strippedrevs:
2815 2816 update = True
2816 2817 break
2817 2818
2818 2819 rootnodes = set(cl.node(r) for r in roots)
2819 2820
2820 2821 q = repo.mq
2821 2822 if q.applied:
2822 2823 # refresh queue state if we're about to strip
2823 2824 # applied patches
2824 2825 if cl.rev(repo.lookup('qtip')) in strippedrevs:
2825 2826 q.applieddirty = True
2826 2827 start = 0
2827 2828 end = len(q.applied)
2828 2829 for i, statusentry in enumerate(q.applied):
2829 2830 if statusentry.node in rootnodes:
2830 2831 # if one of the stripped roots is an applied
2831 2832 # patch, only part of the queue is stripped
2832 2833 start = i
2833 2834 break
2834 2835 del q.applied[start:end]
2835 2836 q.savedirty()
2836 2837
2837 2838 revs = list(rootnodes)
2838 2839 if update and opts.get('keep'):
2839 2840 wlock = repo.wlock()
2840 2841 try:
2841 2842 urev = repo.mq.qparents(repo, revs[0])
2842 2843 repo.dirstate.rebuild(urev, repo[urev].manifest())
2843 2844 repo.dirstate.write()
2844 2845 update = False
2845 2846 finally:
2846 2847 wlock.release()
2847 2848
2848 2849 repo.mq.strip(repo, revs, backup=backup, update=update,
2849 2850 force=opts.get('force'))
2850 2851 return 0
2851 2852
2852 2853 @command("qselect",
2853 2854 [('n', 'none', None, _('disable all guards')),
2854 2855 ('s', 'series', None, _('list all guards in series file')),
2855 2856 ('', 'pop', None, _('pop to before first guarded applied patch')),
2856 2857 ('', 'reapply', None, _('pop, then reapply patches'))],
2857 2858 _('hg qselect [OPTION]... [GUARD]...'))
2858 2859 def select(ui, repo, *args, **opts):
2859 2860 '''set or print guarded patches to push
2860 2861
2861 2862 Use the :hg:`qguard` command to set or print guards on patch, then use
2862 2863 qselect to tell mq which guards to use. A patch will be pushed if
2863 2864 it has no guards or any positive guards match the currently
2864 2865 selected guard, but will not be pushed if any negative guards
2865 2866 match the current guard. For example::
2866 2867
2867 2868 qguard foo.patch -- -stable (negative guard)
2868 2869 qguard bar.patch +stable (positive guard)
2869 2870 qselect stable
2870 2871
2871 2872 This activates the "stable" guard. mq will skip foo.patch (because
2872 2873 it has a negative match) but push bar.patch (because it has a
2873 2874 positive match).
2874 2875
2875 2876 With no arguments, prints the currently active guards.
2876 2877 With one argument, sets the active guard.
2877 2878
2878 2879 Use -n/--none to deactivate guards (no other arguments needed).
2879 2880 When no guards are active, patches with positive guards are
2880 2881 skipped and patches with negative guards are pushed.
2881 2882
2882 2883 qselect can change the guards on applied patches. It does not pop
2883 2884 guarded patches by default. Use --pop to pop back to the last
2884 2885 applied patch that is not guarded. Use --reapply (which implies
2885 2886 --pop) to push back to the current patch afterwards, but skip
2886 2887 guarded patches.
2887 2888
2888 2889 Use -s/--series to print a list of all guards in the series file
2889 2890 (no other arguments needed). Use -v for more information.
2890 2891
2891 2892 Returns 0 on success.'''
2892 2893
2893 2894 q = repo.mq
2894 2895 guards = q.active()
2895 2896 if args or opts.get('none'):
2896 2897 old_unapplied = q.unapplied(repo)
2897 2898 old_guarded = [i for i in xrange(len(q.applied)) if
2898 2899 not q.pushable(i)[0]]
2899 2900 q.setactive(args)
2900 2901 q.savedirty()
2901 2902 if not args:
2902 2903 ui.status(_('guards deactivated\n'))
2903 2904 if not opts.get('pop') and not opts.get('reapply'):
2904 2905 unapplied = q.unapplied(repo)
2905 2906 guarded = [i for i in xrange(len(q.applied))
2906 2907 if not q.pushable(i)[0]]
2907 2908 if len(unapplied) != len(old_unapplied):
2908 2909 ui.status(_('number of unguarded, unapplied patches has '
2909 2910 'changed from %d to %d\n') %
2910 2911 (len(old_unapplied), len(unapplied)))
2911 2912 if len(guarded) != len(old_guarded):
2912 2913 ui.status(_('number of guarded, applied patches has changed '
2913 2914 'from %d to %d\n') %
2914 2915 (len(old_guarded), len(guarded)))
2915 2916 elif opts.get('series'):
2916 2917 guards = {}
2917 2918 noguards = 0
2918 2919 for gs in q.seriesguards:
2919 2920 if not gs:
2920 2921 noguards += 1
2921 2922 for g in gs:
2922 2923 guards.setdefault(g, 0)
2923 2924 guards[g] += 1
2924 2925 if ui.verbose:
2925 2926 guards['NONE'] = noguards
2926 2927 guards = guards.items()
2927 2928 guards.sort(key=lambda x: x[0][1:])
2928 2929 if guards:
2929 2930 ui.note(_('guards in series file:\n'))
2930 2931 for guard, count in guards:
2931 2932 ui.note('%2d ' % count)
2932 2933 ui.write(guard, '\n')
2933 2934 else:
2934 2935 ui.note(_('no guards in series file\n'))
2935 2936 else:
2936 2937 if guards:
2937 2938 ui.note(_('active guards:\n'))
2938 2939 for g in guards:
2939 2940 ui.write(g, '\n')
2940 2941 else:
2941 2942 ui.write(_('no active guards\n'))
2942 2943 reapply = opts.get('reapply') and q.applied and q.appliedname(-1)
2943 2944 popped = False
2944 2945 if opts.get('pop') or opts.get('reapply'):
2945 2946 for i in xrange(len(q.applied)):
2946 2947 pushable, reason = q.pushable(i)
2947 2948 if not pushable:
2948 2949 ui.status(_('popping guarded patches\n'))
2949 2950 popped = True
2950 2951 if i == 0:
2951 2952 q.pop(repo, all=True)
2952 2953 else:
2953 2954 q.pop(repo, str(i - 1))
2954 2955 break
2955 2956 if popped:
2956 2957 try:
2957 2958 if reapply:
2958 2959 ui.status(_('reapplying unguarded patches\n'))
2959 2960 q.push(repo, reapply)
2960 2961 finally:
2961 2962 q.savedirty()
2962 2963
2963 2964 @command("qfinish",
2964 2965 [('a', 'applied', None, _('finish all applied changesets'))],
2965 2966 _('hg qfinish [-a] [REV]...'))
2966 2967 def finish(ui, repo, *revrange, **opts):
2967 2968 """move applied patches into repository history
2968 2969
2969 2970 Finishes the specified revisions (corresponding to applied
2970 2971 patches) by moving them out of mq control into regular repository
2971 2972 history.
2972 2973
2973 2974 Accepts a revision range or the -a/--applied option. If --applied
2974 2975 is specified, all applied mq revisions are removed from mq
2975 2976 control. Otherwise, the given revisions must be at the base of the
2976 2977 stack of applied patches.
2977 2978
2978 2979 This can be especially useful if your changes have been applied to
2979 2980 an upstream repository, or if you are about to push your changes
2980 2981 to upstream.
2981 2982
2982 2983 Returns 0 on success.
2983 2984 """
2984 2985 if not opts.get('applied') and not revrange:
2985 2986 raise util.Abort(_('no revisions specified'))
2986 2987 elif opts.get('applied'):
2987 2988 revrange = ('qbase::qtip',) + revrange
2988 2989
2989 2990 q = repo.mq
2990 2991 if not q.applied:
2991 2992 ui.status(_('no patches applied\n'))
2992 2993 return 0
2993 2994
2994 2995 revs = scmutil.revrange(repo, revrange)
2995 2996 if repo['.'].rev() in revs and repo[None].files():
2996 2997 ui.warn(_('warning: uncommitted changes in the working directory\n'))
2997 2998 # queue.finish may changes phases but leave the responsability to lock the
2998 2999 # repo to the caller to avoid deadlock with wlock. This command code is
2999 3000 # responsability for this locking.
3000 3001 lock = repo.lock()
3001 3002 try:
3002 3003 q.finish(repo, revs)
3003 3004 q.savedirty()
3004 3005 finally:
3005 3006 lock.release()
3006 3007 return 0
3007 3008
3008 3009 @command("qqueue",
3009 3010 [('l', 'list', False, _('list all available queues')),
3010 3011 ('', 'active', False, _('print name of active queue')),
3011 3012 ('c', 'create', False, _('create new queue')),
3012 3013 ('', 'rename', False, _('rename active queue')),
3013 3014 ('', 'delete', False, _('delete reference to queue')),
3014 3015 ('', 'purge', False, _('delete queue, and remove patch dir')),
3015 3016 ],
3016 3017 _('[OPTION] [QUEUE]'))
3017 3018 def qqueue(ui, repo, name=None, **opts):
3018 3019 '''manage multiple patch queues
3019 3020
3020 3021 Supports switching between different patch queues, as well as creating
3021 3022 new patch queues and deleting existing ones.
3022 3023
3023 3024 Omitting a queue name or specifying -l/--list will show you the registered
3024 3025 queues - by default the "normal" patches queue is registered. The currently
3025 3026 active queue will be marked with "(active)". Specifying --active will print
3026 3027 only the name of the active queue.
3027 3028
3028 3029 To create a new queue, use -c/--create. The queue is automatically made
3029 3030 active, except in the case where there are applied patches from the
3030 3031 currently active queue in the repository. Then the queue will only be
3031 3032 created and switching will fail.
3032 3033
3033 3034 To delete an existing queue, use --delete. You cannot delete the currently
3034 3035 active queue.
3035 3036
3036 3037 Returns 0 on success.
3037 3038 '''
3038 3039 q = repo.mq
3039 3040 _defaultqueue = 'patches'
3040 3041 _allqueues = 'patches.queues'
3041 3042 _activequeue = 'patches.queue'
3042 3043
3043 3044 def _getcurrent():
3044 3045 cur = os.path.basename(q.path)
3045 3046 if cur.startswith('patches-'):
3046 3047 cur = cur[8:]
3047 3048 return cur
3048 3049
3049 3050 def _noqueues():
3050 3051 try:
3051 3052 fh = repo.opener(_allqueues, 'r')
3052 3053 fh.close()
3053 3054 except IOError:
3054 3055 return True
3055 3056
3056 3057 return False
3057 3058
3058 3059 def _getqueues():
3059 3060 current = _getcurrent()
3060 3061
3061 3062 try:
3062 3063 fh = repo.opener(_allqueues, 'r')
3063 3064 queues = [queue.strip() for queue in fh if queue.strip()]
3064 3065 fh.close()
3065 3066 if current not in queues:
3066 3067 queues.append(current)
3067 3068 except IOError:
3068 3069 queues = [_defaultqueue]
3069 3070
3070 3071 return sorted(queues)
3071 3072
3072 3073 def _setactive(name):
3073 3074 if q.applied:
3074 3075 raise util.Abort(_('patches applied - cannot set new queue active'))
3075 3076 _setactivenocheck(name)
3076 3077
3077 3078 def _setactivenocheck(name):
3078 3079 fh = repo.opener(_activequeue, 'w')
3079 3080 if name != 'patches':
3080 3081 fh.write(name)
3081 3082 fh.close()
3082 3083
3083 3084 def _addqueue(name):
3084 3085 fh = repo.opener(_allqueues, 'a')
3085 3086 fh.write('%s\n' % (name,))
3086 3087 fh.close()
3087 3088
3088 3089 def _queuedir(name):
3089 3090 if name == 'patches':
3090 3091 return repo.join('patches')
3091 3092 else:
3092 3093 return repo.join('patches-' + name)
3093 3094
3094 3095 def _validname(name):
3095 3096 for n in name:
3096 3097 if n in ':\\/.':
3097 3098 return False
3098 3099 return True
3099 3100
3100 3101 def _delete(name):
3101 3102 if name not in existing:
3102 3103 raise util.Abort(_('cannot delete queue that does not exist'))
3103 3104
3104 3105 current = _getcurrent()
3105 3106
3106 3107 if name == current:
3107 3108 raise util.Abort(_('cannot delete currently active queue'))
3108 3109
3109 3110 fh = repo.opener('patches.queues.new', 'w')
3110 3111 for queue in existing:
3111 3112 if queue == name:
3112 3113 continue
3113 3114 fh.write('%s\n' % (queue,))
3114 3115 fh.close()
3115 3116 util.rename(repo.join('patches.queues.new'), repo.join(_allqueues))
3116 3117
3117 3118 if not name or opts.get('list') or opts.get('active'):
3118 3119 current = _getcurrent()
3119 3120 if opts.get('active'):
3120 3121 ui.write('%s\n' % (current,))
3121 3122 return
3122 3123 for queue in _getqueues():
3123 3124 ui.write('%s' % (queue,))
3124 3125 if queue == current and not ui.quiet:
3125 3126 ui.write(_(' (active)\n'))
3126 3127 else:
3127 3128 ui.write('\n')
3128 3129 return
3129 3130
3130 3131 if not _validname(name):
3131 3132 raise util.Abort(
3132 3133 _('invalid queue name, may not contain the characters ":\\/."'))
3133 3134
3134 3135 existing = _getqueues()
3135 3136
3136 3137 if opts.get('create'):
3137 3138 if name in existing:
3138 3139 raise util.Abort(_('queue "%s" already exists') % name)
3139 3140 if _noqueues():
3140 3141 _addqueue(_defaultqueue)
3141 3142 _addqueue(name)
3142 3143 _setactive(name)
3143 3144 elif opts.get('rename'):
3144 3145 current = _getcurrent()
3145 3146 if name == current:
3146 3147 raise util.Abort(_('can\'t rename "%s" to its current name') % name)
3147 3148 if name in existing:
3148 3149 raise util.Abort(_('queue "%s" already exists') % name)
3149 3150
3150 3151 olddir = _queuedir(current)
3151 3152 newdir = _queuedir(name)
3152 3153
3153 3154 if os.path.exists(newdir):
3154 3155 raise util.Abort(_('non-queue directory "%s" already exists') %
3155 3156 newdir)
3156 3157
3157 3158 fh = repo.opener('patches.queues.new', 'w')
3158 3159 for queue in existing:
3159 3160 if queue == current:
3160 3161 fh.write('%s\n' % (name,))
3161 3162 if os.path.exists(olddir):
3162 3163 util.rename(olddir, newdir)
3163 3164 else:
3164 3165 fh.write('%s\n' % (queue,))
3165 3166 fh.close()
3166 3167 util.rename(repo.join('patches.queues.new'), repo.join(_allqueues))
3167 3168 _setactivenocheck(name)
3168 3169 elif opts.get('delete'):
3169 3170 _delete(name)
3170 3171 elif opts.get('purge'):
3171 3172 if name in existing:
3172 3173 _delete(name)
3173 3174 qdir = _queuedir(name)
3174 3175 if os.path.exists(qdir):
3175 3176 shutil.rmtree(qdir)
3176 3177 else:
3177 3178 if name not in existing:
3178 3179 raise util.Abort(_('use --create to create a new queue'))
3179 3180 _setactive(name)
3180 3181
3181 3182 def mqphasedefaults(repo, roots):
3182 3183 """callback used to set mq changeset as secret when no phase data exists"""
3183 3184 if repo.mq.applied:
3184 3185 if repo.ui.configbool('mq', 'secret', False):
3185 3186 mqphase = phases.secret
3186 3187 else:
3187 3188 mqphase = phases.draft
3188 3189 qbase = repo[repo.mq.applied[0].node]
3189 3190 roots[mqphase].add(qbase.node())
3190 3191 return roots
3191 3192
3192 3193 def reposetup(ui, repo):
3193 3194 class mqrepo(repo.__class__):
3194 3195 @util.propertycache
3195 3196 def mq(self):
3196 3197 return queue(self.ui, self.path)
3197 3198
3198 3199 def abortifwdirpatched(self, errmsg, force=False):
3199 3200 if self.mq.applied and not force:
3200 3201 parents = self.dirstate.parents()
3201 3202 patches = [s.node for s in self.mq.applied]
3202 3203 if parents[0] in patches or parents[1] in patches:
3203 3204 raise util.Abort(errmsg)
3204 3205
3205 3206 def commit(self, text="", user=None, date=None, match=None,
3206 3207 force=False, editor=False, extra={}):
3207 3208 self.abortifwdirpatched(
3208 3209 _('cannot commit over an applied mq patch'),
3209 3210 force)
3210 3211
3211 3212 return super(mqrepo, self).commit(text, user, date, match, force,
3212 3213 editor, extra)
3213 3214
3214 3215 def checkpush(self, force, revs):
3215 3216 if self.mq.applied and not force:
3216 3217 outapplied = [e.node for e in self.mq.applied]
3217 3218 if revs:
3218 3219 # Assume applied patches have no non-patch descendants and
3219 3220 # are not on remote already. Filtering any changeset not
3220 3221 # pushed.
3221 3222 heads = set(revs)
3222 3223 for node in reversed(outapplied):
3223 3224 if node in heads:
3224 3225 break
3225 3226 else:
3226 3227 outapplied.pop()
3227 3228 # looking for pushed and shared changeset
3228 3229 for node in outapplied:
3229 3230 if repo[node].phase() < phases.secret:
3230 3231 raise util.Abort(_('source has mq patches applied'))
3231 3232 # no non-secret patches pushed
3232 3233 super(mqrepo, self).checkpush(force, revs)
3233 3234
3234 3235 def _findtags(self):
3235 3236 '''augment tags from base class with patch tags'''
3236 3237 result = super(mqrepo, self)._findtags()
3237 3238
3238 3239 q = self.mq
3239 3240 if not q.applied:
3240 3241 return result
3241 3242
3242 3243 mqtags = [(patch.node, patch.name) for patch in q.applied]
3243 3244
3244 3245 try:
3245 3246 self.changelog.rev(mqtags[-1][0])
3246 3247 except error.LookupError:
3247 3248 self.ui.warn(_('mq status file refers to unknown node %s\n')
3248 3249 % short(mqtags[-1][0]))
3249 3250 return result
3250 3251
3251 3252 mqtags.append((mqtags[-1][0], 'qtip'))
3252 3253 mqtags.append((mqtags[0][0], 'qbase'))
3253 3254 mqtags.append((self.changelog.parents(mqtags[0][0])[0], 'qparent'))
3254 3255 tags = result[0]
3255 3256 for patch in mqtags:
3256 3257 if patch[1] in tags:
3257 3258 self.ui.warn(_('Tag %s overrides mq patch of the same name\n')
3258 3259 % patch[1])
3259 3260 else:
3260 3261 tags[patch[1]] = patch[0]
3261 3262
3262 3263 return result
3263 3264
3264 3265 def _branchtags(self, partial, lrev):
3265 3266 q = self.mq
3266 3267 cl = self.changelog
3267 3268 qbase = None
3268 3269 if not q.applied:
3269 3270 if getattr(self, '_committingpatch', False):
3270 3271 # Committing a new patch, must be tip
3271 3272 qbase = len(cl) - 1
3272 3273 else:
3273 3274 qbasenode = q.applied[0].node
3274 3275 try:
3275 3276 qbase = cl.rev(qbasenode)
3276 3277 except error.LookupError:
3277 3278 self.ui.warn(_('mq status file refers to unknown node %s\n')
3278 3279 % short(qbasenode))
3279 3280 if qbase is None:
3280 3281 return super(mqrepo, self)._branchtags(partial, lrev)
3281 3282
3282 3283 start = lrev + 1
3283 3284 if start < qbase:
3284 3285 # update the cache (excluding the patches) and save it
3285 3286 ctxgen = (self[r] for r in xrange(lrev + 1, qbase))
3286 3287 self._updatebranchcache(partial, ctxgen)
3287 3288 self._writebranchcache(partial, cl.node(qbase - 1), qbase - 1)
3288 3289 start = qbase
3289 3290 # if start = qbase, the cache is as updated as it should be.
3290 3291 # if start > qbase, the cache includes (part of) the patches.
3291 3292 # we might as well use it, but we won't save it.
3292 3293
3293 3294 # update the cache up to the tip
3294 3295 ctxgen = (self[r] for r in xrange(start, len(cl)))
3295 3296 self._updatebranchcache(partial, ctxgen)
3296 3297
3297 3298 return partial
3298 3299
3299 3300 if repo.local():
3300 3301 repo.__class__ = mqrepo
3301 3302
3302 3303 repo._phasedefaults.append(mqphasedefaults)
3303 3304
3304 3305 def mqimport(orig, ui, repo, *args, **kwargs):
3305 3306 if (hasattr(repo, 'abortifwdirpatched')
3306 3307 and not kwargs.get('no_commit', False)):
3307 3308 repo.abortifwdirpatched(_('cannot import over an applied patch'),
3308 3309 kwargs.get('force'))
3309 3310 return orig(ui, repo, *args, **kwargs)
3310 3311
3311 3312 def mqinit(orig, ui, *args, **kwargs):
3312 3313 mq = kwargs.pop('mq', None)
3313 3314
3314 3315 if not mq:
3315 3316 return orig(ui, *args, **kwargs)
3316 3317
3317 3318 if args:
3318 3319 repopath = args[0]
3319 3320 if not hg.islocal(repopath):
3320 3321 raise util.Abort(_('only a local queue repository '
3321 3322 'may be initialized'))
3322 3323 else:
3323 3324 repopath = cmdutil.findrepo(os.getcwd())
3324 3325 if not repopath:
3325 3326 raise util.Abort(_('there is no Mercurial repository here '
3326 3327 '(.hg not found)'))
3327 3328 repo = hg.repository(ui, repopath)
3328 3329 return qinit(ui, repo, True)
3329 3330
3330 3331 def mqcommand(orig, ui, repo, *args, **kwargs):
3331 3332 """Add --mq option to operate on patch repository instead of main"""
3332 3333
3333 3334 # some commands do not like getting unknown options
3334 3335 mq = kwargs.pop('mq', None)
3335 3336
3336 3337 if not mq:
3337 3338 return orig(ui, repo, *args, **kwargs)
3338 3339
3339 3340 q = repo.mq
3340 3341 r = q.qrepo()
3341 3342 if not r:
3342 3343 raise util.Abort(_('no queue repository'))
3343 3344 return orig(r.ui, r, *args, **kwargs)
3344 3345
3345 3346 def summary(orig, ui, repo, *args, **kwargs):
3346 3347 r = orig(ui, repo, *args, **kwargs)
3347 3348 q = repo.mq
3348 3349 m = []
3349 3350 a, u = len(q.applied), len(q.unapplied(repo))
3350 3351 if a:
3351 3352 m.append(ui.label(_("%d applied"), 'qseries.applied') % a)
3352 3353 if u:
3353 3354 m.append(ui.label(_("%d unapplied"), 'qseries.unapplied') % u)
3354 3355 if m:
3355 3356 ui.write("mq: %s\n" % ', '.join(m))
3356 3357 else:
3357 3358 ui.note(_("mq: (empty queue)\n"))
3358 3359 return r
3359 3360
3360 3361 def revsetmq(repo, subset, x):
3361 3362 """``mq()``
3362 3363 Changesets managed by MQ.
3363 3364 """
3364 3365 revset.getargs(x, 0, 0, _("mq takes no arguments"))
3365 3366 applied = set([repo[r.node].rev() for r in repo.mq.applied])
3366 3367 return [r for r in subset if r in applied]
3367 3368
3368 3369 def extsetup(ui):
3369 3370 revset.symbols['mq'] = revsetmq
3370 3371
3371 3372 # tell hggettext to extract docstrings from these functions:
3372 3373 i18nfunctions = [revsetmq]
3373 3374
3374 3375 def uisetup(ui):
3375 3376 mqopt = [('', 'mq', None, _("operate on patch repository"))]
3376 3377
3377 3378 extensions.wrapcommand(commands.table, 'import', mqimport)
3378 3379 extensions.wrapcommand(commands.table, 'summary', summary)
3379 3380
3380 3381 entry = extensions.wrapcommand(commands.table, 'init', mqinit)
3381 3382 entry[1].extend(mqopt)
3382 3383
3383 3384 nowrap = set(commands.norepo.split(" "))
3384 3385
3385 3386 def dotable(cmdtable):
3386 3387 for cmd in cmdtable.keys():
3387 3388 cmd = cmdutil.parsealiases(cmd)[0]
3388 3389 if cmd in nowrap:
3389 3390 continue
3390 3391 entry = extensions.wrapcommand(cmdtable, cmd, mqcommand)
3391 3392 entry[1].extend(mqopt)
3392 3393
3393 3394 dotable(commands.table)
3394 3395
3395 3396 for extname, extmodule in extensions.extensions():
3396 3397 if extmodule.__file__ != __file__:
3397 3398 dotable(getattr(extmodule, 'cmdtable', {}))
3398 3399
3399 3400
3400 3401 colortable = {'qguard.negative': 'red',
3401 3402 'qguard.positive': 'yellow',
3402 3403 'qguard.unguarded': 'green',
3403 3404 'qseries.applied': 'blue bold underline',
3404 3405 'qseries.guarded': 'black bold',
3405 3406 'qseries.missing': 'red bold',
3406 3407 'qseries.unapplied': 'black bold'}
General Comments 0
You need to be logged in to leave comments. Login now