##// END OF EJS Templates
fix strip on windows...
Benoit Boissinot -
r3901:733d56b8 default
parent child Browse files
Show More
@@ -1,2191 +1,2192
1 1 # queue.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
6 6 # of the GNU General Public License, incorporated herein by reference.
7 7
8 8 '''patch management and development
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 prepare repository to work with patches qinit
20 20 create new patch qnew
21 21 import existing patch qimport
22 22
23 23 print patch series qseries
24 24 print applied patches qapplied
25 25 print name of top applied patch qtop
26 26
27 27 add known patch to applied stack qpush
28 28 remove patch from applied stack qpop
29 29 refresh contents of top applied patch qrefresh
30 30 '''
31 31
32 32 from mercurial.demandload import *
33 33 from mercurial.i18n import gettext as _
34 34 from mercurial import commands
35 35 demandload(globals(), "os sys re struct traceback errno bz2")
36 36 demandload(globals(), "mercurial:cmdutil,hg,patch,revlog,util,changegroup")
37 37
38 38 commands.norepo += " qclone qversion"
39 39
40 40 class statusentry:
41 41 def __init__(self, rev, name=None):
42 42 if not name:
43 43 fields = rev.split(':', 1)
44 44 if len(fields) == 2:
45 45 self.rev, self.name = fields
46 46 else:
47 47 self.rev, self.name = None, None
48 48 else:
49 49 self.rev, self.name = rev, name
50 50
51 51 def __str__(self):
52 52 return self.rev + ':' + self.name
53 53
54 54 class queue:
55 55 def __init__(self, ui, path, patchdir=None):
56 56 self.basepath = path
57 57 self.path = patchdir or os.path.join(path, "patches")
58 58 self.opener = util.opener(self.path)
59 59 self.ui = ui
60 60 self.applied = []
61 61 self.full_series = []
62 62 self.applied_dirty = 0
63 63 self.series_dirty = 0
64 64 self.series_path = "series"
65 65 self.status_path = "status"
66 66 self.guards_path = "guards"
67 67 self.active_guards = None
68 68 self.guards_dirty = False
69 69 self._diffopts = None
70 70
71 71 if os.path.exists(self.join(self.series_path)):
72 72 self.full_series = self.opener(self.series_path).read().splitlines()
73 73 self.parse_series()
74 74
75 75 if os.path.exists(self.join(self.status_path)):
76 76 lines = self.opener(self.status_path).read().splitlines()
77 77 self.applied = [statusentry(l) for l in lines]
78 78
79 79 def diffopts(self):
80 80 if self._diffopts is None:
81 81 self._diffopts = patch.diffopts(self.ui)
82 82 return self._diffopts
83 83
84 84 def join(self, *p):
85 85 return os.path.join(self.path, *p)
86 86
87 87 def find_series(self, patch):
88 88 pre = re.compile("(\s*)([^#]+)")
89 89 index = 0
90 90 for l in self.full_series:
91 91 m = pre.match(l)
92 92 if m:
93 93 s = m.group(2)
94 94 s = s.rstrip()
95 95 if s == patch:
96 96 return index
97 97 index += 1
98 98 return None
99 99
100 100 guard_re = re.compile(r'\s?#([-+][^-+# \t\r\n\f][^# \t\r\n\f]*)')
101 101
102 102 def parse_series(self):
103 103 self.series = []
104 104 self.series_guards = []
105 105 for l in self.full_series:
106 106 h = l.find('#')
107 107 if h == -1:
108 108 patch = l
109 109 comment = ''
110 110 elif h == 0:
111 111 continue
112 112 else:
113 113 patch = l[:h]
114 114 comment = l[h:]
115 115 patch = patch.strip()
116 116 if patch:
117 117 if patch in self.series:
118 118 raise util.Abort(_('%s appears more than once in %s') %
119 119 (patch, self.join(self.series_path)))
120 120 self.series.append(patch)
121 121 self.series_guards.append(self.guard_re.findall(comment))
122 122
123 123 def check_guard(self, guard):
124 124 bad_chars = '# \t\r\n\f'
125 125 first = guard[0]
126 126 for c in '-+':
127 127 if first == c:
128 128 return (_('guard %r starts with invalid character: %r') %
129 129 (guard, c))
130 130 for c in bad_chars:
131 131 if c in guard:
132 132 return _('invalid character in guard %r: %r') % (guard, c)
133 133
134 134 def set_active(self, guards):
135 135 for guard in guards:
136 136 bad = self.check_guard(guard)
137 137 if bad:
138 138 raise util.Abort(bad)
139 139 guards = dict.fromkeys(guards).keys()
140 140 guards.sort()
141 141 self.ui.debug('active guards: %s\n' % ' '.join(guards))
142 142 self.active_guards = guards
143 143 self.guards_dirty = True
144 144
145 145 def active(self):
146 146 if self.active_guards is None:
147 147 self.active_guards = []
148 148 try:
149 149 guards = self.opener(self.guards_path).read().split()
150 150 except IOError, err:
151 151 if err.errno != errno.ENOENT: raise
152 152 guards = []
153 153 for i, guard in enumerate(guards):
154 154 bad = self.check_guard(guard)
155 155 if bad:
156 156 self.ui.warn('%s:%d: %s\n' %
157 157 (self.join(self.guards_path), i + 1, bad))
158 158 else:
159 159 self.active_guards.append(guard)
160 160 return self.active_guards
161 161
162 162 def set_guards(self, idx, guards):
163 163 for g in guards:
164 164 if len(g) < 2:
165 165 raise util.Abort(_('guard %r too short') % g)
166 166 if g[0] not in '-+':
167 167 raise util.Abort(_('guard %r starts with invalid char') % g)
168 168 bad = self.check_guard(g[1:])
169 169 if bad:
170 170 raise util.Abort(bad)
171 171 drop = self.guard_re.sub('', self.full_series[idx])
172 172 self.full_series[idx] = drop + ''.join([' #' + g for g in guards])
173 173 self.parse_series()
174 174 self.series_dirty = True
175 175
176 176 def pushable(self, idx):
177 177 if isinstance(idx, str):
178 178 idx = self.series.index(idx)
179 179 patchguards = self.series_guards[idx]
180 180 if not patchguards:
181 181 return True, None
182 182 default = False
183 183 guards = self.active()
184 184 exactneg = [g for g in patchguards if g[0] == '-' and g[1:] in guards]
185 185 if exactneg:
186 186 return False, exactneg[0]
187 187 pos = [g for g in patchguards if g[0] == '+']
188 188 exactpos = [g for g in pos if g[1:] in guards]
189 189 if pos:
190 190 if exactpos:
191 191 return True, exactpos[0]
192 192 return False, pos
193 193 return True, ''
194 194
195 195 def explain_pushable(self, idx, all_patches=False):
196 196 write = all_patches and self.ui.write or self.ui.warn
197 197 if all_patches or self.ui.verbose:
198 198 if isinstance(idx, str):
199 199 idx = self.series.index(idx)
200 200 pushable, why = self.pushable(idx)
201 201 if all_patches and pushable:
202 202 if why is None:
203 203 write(_('allowing %s - no guards in effect\n') %
204 204 self.series[idx])
205 205 else:
206 206 if not why:
207 207 write(_('allowing %s - no matching negative guards\n') %
208 208 self.series[idx])
209 209 else:
210 210 write(_('allowing %s - guarded by %r\n') %
211 211 (self.series[idx], why))
212 212 if not pushable:
213 213 if why:
214 214 write(_('skipping %s - guarded by %r\n') %
215 215 (self.series[idx], why))
216 216 else:
217 217 write(_('skipping %s - no matching guards\n') %
218 218 self.series[idx])
219 219
220 220 def save_dirty(self):
221 221 def write_list(items, path):
222 222 fp = self.opener(path, 'w')
223 223 for i in items:
224 224 print >> fp, i
225 225 fp.close()
226 226 if self.applied_dirty: write_list(map(str, self.applied), self.status_path)
227 227 if self.series_dirty: write_list(self.full_series, self.series_path)
228 228 if self.guards_dirty: write_list(self.active_guards, self.guards_path)
229 229
230 230 def readheaders(self, patch):
231 231 def eatdiff(lines):
232 232 while lines:
233 233 l = lines[-1]
234 234 if (l.startswith("diff -") or
235 235 l.startswith("Index:") or
236 236 l.startswith("===========")):
237 237 del lines[-1]
238 238 else:
239 239 break
240 240 def eatempty(lines):
241 241 while lines:
242 242 l = lines[-1]
243 243 if re.match('\s*$', l):
244 244 del lines[-1]
245 245 else:
246 246 break
247 247
248 248 pf = self.join(patch)
249 249 message = []
250 250 comments = []
251 251 user = None
252 252 date = None
253 253 format = None
254 254 subject = None
255 255 diffstart = 0
256 256
257 257 for line in file(pf):
258 258 line = line.rstrip()
259 259 if line.startswith('diff --git'):
260 260 diffstart = 2
261 261 break
262 262 if diffstart:
263 263 if line.startswith('+++ '):
264 264 diffstart = 2
265 265 break
266 266 if line.startswith("--- "):
267 267 diffstart = 1
268 268 continue
269 269 elif format == "hgpatch":
270 270 # parse values when importing the result of an hg export
271 271 if line.startswith("# User "):
272 272 user = line[7:]
273 273 elif line.startswith("# Date "):
274 274 date = line[7:]
275 275 elif not line.startswith("# ") and line:
276 276 message.append(line)
277 277 format = None
278 278 elif line == '# HG changeset patch':
279 279 format = "hgpatch"
280 280 elif (format != "tagdone" and (line.startswith("Subject: ") or
281 281 line.startswith("subject: "))):
282 282 subject = line[9:]
283 283 format = "tag"
284 284 elif (format != "tagdone" and (line.startswith("From: ") or
285 285 line.startswith("from: "))):
286 286 user = line[6:]
287 287 format = "tag"
288 288 elif format == "tag" and line == "":
289 289 # when looking for tags (subject: from: etc) they
290 290 # end once you find a blank line in the source
291 291 format = "tagdone"
292 292 elif message or line:
293 293 message.append(line)
294 294 comments.append(line)
295 295
296 296 eatdiff(message)
297 297 eatdiff(comments)
298 298 eatempty(message)
299 299 eatempty(comments)
300 300
301 301 # make sure message isn't empty
302 302 if format and format.startswith("tag") and subject:
303 303 message.insert(0, "")
304 304 message.insert(0, subject)
305 305 return (message, comments, user, date, diffstart > 1)
306 306
307 307 def printdiff(self, repo, node1, node2=None, files=None,
308 308 fp=None, changes=None, opts={}):
309 309 fns, matchfn, anypats = cmdutil.matchpats(repo, files, opts)
310 310
311 311 patch.diff(repo, node1, node2, fns, match=matchfn,
312 312 fp=fp, changes=changes, opts=self.diffopts())
313 313
314 314 def mergeone(self, repo, mergeq, head, patch, rev, wlock):
315 315 # first try just applying the patch
316 316 (err, n) = self.apply(repo, [ patch ], update_status=False,
317 317 strict=True, merge=rev, wlock=wlock)
318 318
319 319 if err == 0:
320 320 return (err, n)
321 321
322 322 if n is None:
323 323 raise util.Abort(_("apply failed for patch %s") % patch)
324 324
325 325 self.ui.warn("patch didn't work out, merging %s\n" % patch)
326 326
327 327 # apply failed, strip away that rev and merge.
328 328 hg.clean(repo, head, wlock=wlock)
329 329 self.strip(repo, n, update=False, backup='strip', wlock=wlock)
330 330
331 331 c = repo.changelog.read(rev)
332 332 ret = hg.merge(repo, rev, wlock=wlock)
333 333 if ret:
334 334 raise util.Abort(_("update returned %d") % ret)
335 335 n = repo.commit(None, c[4], c[1], force=1, wlock=wlock)
336 336 if n == None:
337 337 raise util.Abort(_("repo commit failed"))
338 338 try:
339 339 message, comments, user, date, patchfound = mergeq.readheaders(patch)
340 340 except:
341 341 raise util.Abort(_("unable to read %s") % patch)
342 342
343 343 patchf = self.opener(patch, "w")
344 344 if comments:
345 345 comments = "\n".join(comments) + '\n\n'
346 346 patchf.write(comments)
347 347 self.printdiff(repo, head, n, fp=patchf)
348 348 patchf.close()
349 349 return (0, n)
350 350
351 351 def qparents(self, repo, rev=None):
352 352 if rev is None:
353 353 (p1, p2) = repo.dirstate.parents()
354 354 if p2 == revlog.nullid:
355 355 return p1
356 356 if len(self.applied) == 0:
357 357 return None
358 358 return revlog.bin(self.applied[-1].rev)
359 359 pp = repo.changelog.parents(rev)
360 360 if pp[1] != revlog.nullid:
361 361 arevs = [ x.rev for x in self.applied ]
362 362 p0 = revlog.hex(pp[0])
363 363 p1 = revlog.hex(pp[1])
364 364 if p0 in arevs:
365 365 return pp[0]
366 366 if p1 in arevs:
367 367 return pp[1]
368 368 return pp[0]
369 369
370 370 def mergepatch(self, repo, mergeq, series, wlock):
371 371 if len(self.applied) == 0:
372 372 # each of the patches merged in will have two parents. This
373 373 # can confuse the qrefresh, qdiff, and strip code because it
374 374 # needs to know which parent is actually in the patch queue.
375 375 # so, we insert a merge marker with only one parent. This way
376 376 # the first patch in the queue is never a merge patch
377 377 #
378 378 pname = ".hg.patches.merge.marker"
379 379 n = repo.commit(None, '[mq]: merge marker', user=None, force=1,
380 380 wlock=wlock)
381 381 self.applied.append(statusentry(revlog.hex(n), pname))
382 382 self.applied_dirty = 1
383 383
384 384 head = self.qparents(repo)
385 385
386 386 for patch in series:
387 387 patch = mergeq.lookup(patch, strict=True)
388 388 if not patch:
389 389 self.ui.warn("patch %s does not exist\n" % patch)
390 390 return (1, None)
391 391 pushable, reason = self.pushable(patch)
392 392 if not pushable:
393 393 self.explain_pushable(patch, all_patches=True)
394 394 continue
395 395 info = mergeq.isapplied(patch)
396 396 if not info:
397 397 self.ui.warn("patch %s is not applied\n" % patch)
398 398 return (1, None)
399 399 rev = revlog.bin(info[1])
400 400 (err, head) = self.mergeone(repo, mergeq, head, patch, rev, wlock)
401 401 if head:
402 402 self.applied.append(statusentry(revlog.hex(head), patch))
403 403 self.applied_dirty = 1
404 404 if err:
405 405 return (err, head)
406 406 return (0, head)
407 407
408 408 def patch(self, repo, patchfile):
409 409 '''Apply patchfile to the working directory.
410 410 patchfile: file name of patch'''
411 411 files = {}
412 412 try:
413 413 fuzz = patch.patch(patchfile, self.ui, strip=1, cwd=repo.root,
414 414 files=files)
415 415 except Exception, inst:
416 416 self.ui.note(str(inst) + '\n')
417 417 if not self.ui.verbose:
418 418 self.ui.warn("patch failed, unable to continue (try -v)\n")
419 419 return (False, files, False)
420 420
421 421 return (True, files, fuzz)
422 422
423 423 def apply(self, repo, series, list=False, update_status=True,
424 424 strict=False, patchdir=None, merge=None, wlock=None):
425 425 # TODO unify with commands.py
426 426 if not patchdir:
427 427 patchdir = self.path
428 428 err = 0
429 429 if not wlock:
430 430 wlock = repo.wlock()
431 431 lock = repo.lock()
432 432 tr = repo.transaction()
433 433 n = None
434 434 for patchname in series:
435 435 pushable, reason = self.pushable(patchname)
436 436 if not pushable:
437 437 self.explain_pushable(patchname, all_patches=True)
438 438 continue
439 439 self.ui.warn("applying %s\n" % patchname)
440 440 pf = os.path.join(patchdir, patchname)
441 441
442 442 try:
443 443 message, comments, user, date, patchfound = self.readheaders(patchname)
444 444 except:
445 445 self.ui.warn("Unable to read %s\n" % patchname)
446 446 err = 1
447 447 break
448 448
449 449 if not message:
450 450 message = "imported patch %s\n" % patchname
451 451 else:
452 452 if list:
453 453 message.append("\nimported patch %s" % patchname)
454 454 message = '\n'.join(message)
455 455
456 456 (patcherr, files, fuzz) = self.patch(repo, pf)
457 457 patcherr = not patcherr
458 458
459 459 if merge and files:
460 460 # Mark as merged and update dirstate parent info
461 461 repo.dirstate.update(repo.dirstate.filterfiles(files.keys()), 'm')
462 462 p1, p2 = repo.dirstate.parents()
463 463 repo.dirstate.setparents(p1, merge)
464 464 files = patch.updatedir(self.ui, repo, files, wlock=wlock)
465 465 n = repo.commit(files, message, user, date, force=1, lock=lock,
466 466 wlock=wlock)
467 467
468 468 if n == None:
469 469 raise util.Abort(_("repo commit failed"))
470 470
471 471 if update_status:
472 472 self.applied.append(statusentry(revlog.hex(n), patchname))
473 473
474 474 if patcherr:
475 475 if not patchfound:
476 476 self.ui.warn("patch %s is empty\n" % patchname)
477 477 err = 0
478 478 else:
479 479 self.ui.warn("patch failed, rejects left in working dir\n")
480 480 err = 1
481 481 break
482 482
483 483 if fuzz and strict:
484 484 self.ui.warn("fuzz found when applying patch, stopping\n")
485 485 err = 1
486 486 break
487 487 tr.close()
488 488 return (err, n)
489 489
490 490 def delete(self, repo, patches, opts):
491 491 realpatches = []
492 492 for patch in patches:
493 493 patch = self.lookup(patch, strict=True)
494 494 info = self.isapplied(patch)
495 495 if info:
496 496 raise util.Abort(_("cannot delete applied patch %s") % patch)
497 497 if patch not in self.series:
498 498 raise util.Abort(_("patch %s not in series file") % patch)
499 499 realpatches.append(patch)
500 500
501 501 appliedbase = 0
502 502 if opts.get('rev'):
503 503 if not self.applied:
504 504 raise util.Abort(_('no patches applied'))
505 505 revs = cmdutil.revrange(repo, opts['rev'])
506 506 if len(revs) > 1 and revs[0] > revs[1]:
507 507 revs.reverse()
508 508 for rev in revs:
509 509 if appliedbase >= len(self.applied):
510 510 raise util.Abort(_("revision %d is not managed") % rev)
511 511
512 512 base = revlog.bin(self.applied[appliedbase].rev)
513 513 node = repo.changelog.node(rev)
514 514 if node != base:
515 515 raise util.Abort(_("cannot delete revision %d above "
516 516 "applied patches") % rev)
517 517 realpatches.append(self.applied[appliedbase].name)
518 518 appliedbase += 1
519 519
520 520 if not opts.get('keep'):
521 521 r = self.qrepo()
522 522 if r:
523 523 r.remove(realpatches, True)
524 524 else:
525 525 for p in realpatches:
526 526 os.unlink(self.join(p))
527 527
528 528 if appliedbase:
529 529 del self.applied[:appliedbase]
530 530 self.applied_dirty = 1
531 531 indices = [self.find_series(p) for p in realpatches]
532 532 indices.sort()
533 533 for i in indices[-1::-1]:
534 534 del self.full_series[i]
535 535 self.parse_series()
536 536 self.series_dirty = 1
537 537
538 538 def check_toppatch(self, repo):
539 539 if len(self.applied) > 0:
540 540 top = revlog.bin(self.applied[-1].rev)
541 541 pp = repo.dirstate.parents()
542 542 if top not in pp:
543 543 raise util.Abort(_("queue top not at same revision as working directory"))
544 544 return top
545 545 return None
546 546 def check_localchanges(self, repo, force=False, refresh=True):
547 547 m, a, r, d = repo.status()[:4]
548 548 if m or a or r or d:
549 549 if not force:
550 550 if refresh:
551 551 raise util.Abort(_("local changes found, refresh first"))
552 552 else:
553 553 raise util.Abort(_("local changes found"))
554 554 return m, a, r, d
555 555 def new(self, repo, patch, msg=None, force=None):
556 556 if os.path.exists(self.join(patch)):
557 557 raise util.Abort(_('patch "%s" already exists') % patch)
558 558 m, a, r, d = self.check_localchanges(repo, force)
559 559 commitfiles = m + a + r
560 560 self.check_toppatch(repo)
561 561 wlock = repo.wlock()
562 562 insert = self.full_series_end()
563 563 if msg:
564 564 n = repo.commit(commitfiles, "[mq]: %s" % msg, force=True,
565 565 wlock=wlock)
566 566 else:
567 567 n = repo.commit(commitfiles,
568 568 "New patch: %s" % patch, force=True, wlock=wlock)
569 569 if n == None:
570 570 raise util.Abort(_("repo commit failed"))
571 571 self.full_series[insert:insert] = [patch]
572 572 self.applied.append(statusentry(revlog.hex(n), patch))
573 573 self.parse_series()
574 574 self.series_dirty = 1
575 575 self.applied_dirty = 1
576 576 p = self.opener(patch, "w")
577 577 if msg:
578 578 msg = msg + "\n"
579 579 p.write(msg)
580 580 p.close()
581 581 wlock = None
582 582 r = self.qrepo()
583 583 if r: r.add([patch])
584 584 if commitfiles:
585 585 self.refresh(repo, short=True)
586 586
587 587 def strip(self, repo, rev, update=True, backup="all", wlock=None):
588 588 def limitheads(chlog, stop):
589 589 """return the list of all nodes that have no children"""
590 590 p = {}
591 591 h = []
592 592 stoprev = 0
593 593 if stop in chlog.nodemap:
594 594 stoprev = chlog.rev(stop)
595 595
596 596 for r in xrange(chlog.count() - 1, -1, -1):
597 597 n = chlog.node(r)
598 598 if n not in p:
599 599 h.append(n)
600 600 if n == stop:
601 601 break
602 602 if r < stoprev:
603 603 break
604 604 for pn in chlog.parents(n):
605 605 p[pn] = 1
606 606 return h
607 607
608 608 def bundle(cg):
609 609 backupdir = repo.join("strip-backup")
610 610 if not os.path.isdir(backupdir):
611 611 os.mkdir(backupdir)
612 612 name = os.path.join(backupdir, "%s" % revlog.short(rev))
613 613 name = savename(name)
614 614 self.ui.warn("saving bundle to %s\n" % name)
615 615 return changegroup.writebundle(cg, name, "HG10BZ")
616 616
617 617 def stripall(rev, revnum):
618 618 cl = repo.changelog
619 619 c = cl.read(rev)
620 620 mm = repo.manifest.read(c[0])
621 621 seen = {}
622 622
623 623 for x in xrange(revnum, cl.count()):
624 624 c = cl.read(cl.node(x))
625 625 for f in c[3]:
626 626 if f in seen:
627 627 continue
628 628 seen[f] = 1
629 629 if f in mm:
630 630 filerev = mm[f]
631 631 else:
632 632 filerev = 0
633 633 seen[f] = filerev
634 634 # we go in two steps here so the strip loop happens in a
635 635 # sensible order. When stripping many files, this helps keep
636 636 # our disk access patterns under control.
637 637 seen_list = seen.keys()
638 638 seen_list.sort()
639 639 for f in seen_list:
640 640 ff = repo.file(f)
641 641 filerev = seen[f]
642 642 if filerev != 0:
643 643 if filerev in ff.nodemap:
644 644 filerev = ff.rev(filerev)
645 645 else:
646 646 filerev = 0
647 647 ff.strip(filerev, revnum)
648 648
649 649 if not wlock:
650 650 wlock = repo.wlock()
651 651 lock = repo.lock()
652 652 chlog = repo.changelog
653 653 # TODO delete the undo files, and handle undo of merge sets
654 654 pp = chlog.parents(rev)
655 655 revnum = chlog.rev(rev)
656 656
657 657 if update:
658 658 self.check_localchanges(repo, refresh=False)
659 659 urev = self.qparents(repo, rev)
660 660 hg.clean(repo, urev, wlock=wlock)
661 661 repo.dirstate.write()
662 662
663 663 # save is a list of all the branches we are truncating away
664 664 # that we actually want to keep. changegroup will be used
665 665 # to preserve them and add them back after the truncate
666 666 saveheads = []
667 667 savebases = {}
668 668
669 669 heads = limitheads(chlog, rev)
670 670 seen = {}
671 671
672 672 # search through all the heads, finding those where the revision
673 673 # we want to strip away is an ancestor. Also look for merges
674 674 # that might be turned into new heads by the strip.
675 675 while heads:
676 676 h = heads.pop()
677 677 n = h
678 678 while True:
679 679 seen[n] = 1
680 680 pp = chlog.parents(n)
681 681 if pp[1] != revlog.nullid and chlog.rev(pp[1]) > revnum:
682 682 if pp[1] not in seen:
683 683 heads.append(pp[1])
684 684 if pp[0] == revlog.nullid:
685 685 break
686 686 if chlog.rev(pp[0]) < revnum:
687 687 break
688 688 n = pp[0]
689 689 if n == rev:
690 690 break
691 691 r = chlog.reachable(h, rev)
692 692 if rev not in r:
693 693 saveheads.append(h)
694 694 for x in r:
695 695 if chlog.rev(x) > revnum:
696 696 savebases[x] = 1
697 697
698 698 # create a changegroup for all the branches we need to keep
699 699 if backup == "all":
700 700 backupch = repo.changegroupsubset([rev], chlog.heads(), 'strip')
701 701 bundle(backupch)
702 702 if saveheads:
703 703 backupch = repo.changegroupsubset(savebases.keys(), saveheads, 'strip')
704 704 chgrpfile = bundle(backupch)
705 chgrpfile = 'file:%s' % chgrpfile
705 706
706 707 stripall(rev, revnum)
707 708
708 709 change = chlog.read(rev)
709 710 chlog.strip(revnum, revnum)
710 711 repo.manifest.strip(repo.manifest.rev(change[0]), revnum)
711 712 if saveheads:
712 713 self.ui.status("adding branch\n")
713 714 commands.unbundle(self.ui, repo, chgrpfile, update=False)
714 715 if backup != "strip":
715 716 os.unlink(chgrpfile)
716 717
717 718 def isapplied(self, patch):
718 719 """returns (index, rev, patch)"""
719 720 for i in xrange(len(self.applied)):
720 721 a = self.applied[i]
721 722 if a.name == patch:
722 723 return (i, a.rev, a.name)
723 724 return None
724 725
725 726 # if the exact patch name does not exist, we try a few
726 727 # variations. If strict is passed, we try only #1
727 728 #
728 729 # 1) a number to indicate an offset in the series file
729 730 # 2) a unique substring of the patch name was given
730 731 # 3) patchname[-+]num to indicate an offset in the series file
731 732 def lookup(self, patch, strict=False):
732 733 patch = patch and str(patch)
733 734
734 735 def partial_name(s):
735 736 if s in self.series:
736 737 return s
737 738 matches = [x for x in self.series if s in x]
738 739 if len(matches) > 1:
739 740 self.ui.warn(_('patch name "%s" is ambiguous:\n') % s)
740 741 for m in matches:
741 742 self.ui.warn(' %s\n' % m)
742 743 return None
743 744 if matches:
744 745 return matches[0]
745 746 if len(self.series) > 0 and len(self.applied) > 0:
746 747 if s == 'qtip':
747 748 return self.series[self.series_end(True)-1]
748 749 if s == 'qbase':
749 750 return self.series[0]
750 751 return None
751 752 if patch == None:
752 753 return None
753 754
754 755 # we don't want to return a partial match until we make
755 756 # sure the file name passed in does not exist (checked below)
756 757 res = partial_name(patch)
757 758 if res and res == patch:
758 759 return res
759 760
760 761 if not os.path.isfile(self.join(patch)):
761 762 try:
762 763 sno = int(patch)
763 764 except(ValueError, OverflowError):
764 765 pass
765 766 else:
766 767 if sno < len(self.series):
767 768 return self.series[sno]
768 769 if not strict:
769 770 # return any partial match made above
770 771 if res:
771 772 return res
772 773 minus = patch.rfind('-')
773 774 if minus >= 0:
774 775 res = partial_name(patch[:minus])
775 776 if res:
776 777 i = self.series.index(res)
777 778 try:
778 779 off = int(patch[minus+1:] or 1)
779 780 except(ValueError, OverflowError):
780 781 pass
781 782 else:
782 783 if i - off >= 0:
783 784 return self.series[i - off]
784 785 plus = patch.rfind('+')
785 786 if plus >= 0:
786 787 res = partial_name(patch[:plus])
787 788 if res:
788 789 i = self.series.index(res)
789 790 try:
790 791 off = int(patch[plus+1:] or 1)
791 792 except(ValueError, OverflowError):
792 793 pass
793 794 else:
794 795 if i + off < len(self.series):
795 796 return self.series[i + off]
796 797 raise util.Abort(_("patch %s not in series") % patch)
797 798
798 799 def push(self, repo, patch=None, force=False, list=False,
799 800 mergeq=None, wlock=None):
800 801 if not wlock:
801 802 wlock = repo.wlock()
802 803 patch = self.lookup(patch)
803 804 if patch and self.isapplied(patch):
804 805 raise util.Abort(_("patch %s is already applied") % patch)
805 806 if self.series_end() == len(self.series):
806 807 raise util.Abort(_("patch series fully applied"))
807 808 if not force:
808 809 self.check_localchanges(repo)
809 810
810 811 self.applied_dirty = 1;
811 812 start = self.series_end()
812 813 if start > 0:
813 814 self.check_toppatch(repo)
814 815 if not patch:
815 816 patch = self.series[start]
816 817 end = start + 1
817 818 else:
818 819 end = self.series.index(patch, start) + 1
819 820 s = self.series[start:end]
820 821 if mergeq:
821 822 ret = self.mergepatch(repo, mergeq, s, wlock)
822 823 else:
823 824 ret = self.apply(repo, s, list, wlock=wlock)
824 825 top = self.applied[-1].name
825 826 if ret[0]:
826 827 self.ui.write("Errors during apply, please fix and refresh %s\n" %
827 828 top)
828 829 else:
829 830 self.ui.write("Now at: %s\n" % top)
830 831 return ret[0]
831 832
832 833 def pop(self, repo, patch=None, force=False, update=True, all=False,
833 834 wlock=None):
834 835 def getfile(f, rev):
835 836 t = repo.file(f).read(rev)
836 837 try:
837 838 repo.wfile(f, "w").write(t)
838 839 except IOError:
839 840 try:
840 841 os.makedirs(os.path.dirname(repo.wjoin(f)))
841 842 except OSError, err:
842 843 if err.errno != errno.EEXIST: raise
843 844 repo.wfile(f, "w").write(t)
844 845
845 846 if not wlock:
846 847 wlock = repo.wlock()
847 848 if patch:
848 849 # index, rev, patch
849 850 info = self.isapplied(patch)
850 851 if not info:
851 852 patch = self.lookup(patch)
852 853 info = self.isapplied(patch)
853 854 if not info:
854 855 raise util.Abort(_("patch %s is not applied") % patch)
855 856 if len(self.applied) == 0:
856 857 raise util.Abort(_("no patches applied"))
857 858
858 859 if not update:
859 860 parents = repo.dirstate.parents()
860 861 rr = [ revlog.bin(x.rev) for x in self.applied ]
861 862 for p in parents:
862 863 if p in rr:
863 864 self.ui.warn("qpop: forcing dirstate update\n")
864 865 update = True
865 866
866 867 if not force and update:
867 868 self.check_localchanges(repo)
868 869
869 870 self.applied_dirty = 1;
870 871 end = len(self.applied)
871 872 if not patch:
872 873 if all:
873 874 popi = 0
874 875 else:
875 876 popi = len(self.applied) - 1
876 877 else:
877 878 popi = info[0] + 1
878 879 if popi >= end:
879 880 self.ui.warn("qpop: %s is already at the top\n" % patch)
880 881 return
881 882 info = [ popi ] + [self.applied[popi].rev, self.applied[popi].name]
882 883
883 884 start = info[0]
884 885 rev = revlog.bin(info[1])
885 886
886 887 # we know there are no local changes, so we can make a simplified
887 888 # form of hg.update.
888 889 if update:
889 890 top = self.check_toppatch(repo)
890 891 qp = self.qparents(repo, rev)
891 892 changes = repo.changelog.read(qp)
892 893 mmap = repo.manifest.read(changes[0])
893 894 m, a, r, d, u = repo.status(qp, top)[:5]
894 895 if d:
895 896 raise util.Abort("deletions found between repo revs")
896 897 for f in m:
897 898 getfile(f, mmap[f])
898 899 for f in r:
899 900 getfile(f, mmap[f])
900 901 util.set_exec(repo.wjoin(f), mmap.execf(f))
901 902 repo.dirstate.update(m + r, 'n')
902 903 for f in a:
903 904 try:
904 905 os.unlink(repo.wjoin(f))
905 906 except OSError, e:
906 907 if e.errno != errno.ENOENT:
907 908 raise
908 909 try: os.removedirs(os.path.dirname(repo.wjoin(f)))
909 910 except: pass
910 911 if a:
911 912 repo.dirstate.forget(a)
912 913 repo.dirstate.setparents(qp, revlog.nullid)
913 914 self.strip(repo, rev, update=False, backup='strip', wlock=wlock)
914 915 del self.applied[start:end]
915 916 if len(self.applied):
916 917 self.ui.write("Now at: %s\n" % self.applied[-1].name)
917 918 else:
918 919 self.ui.write("Patch queue now empty\n")
919 920
920 921 def diff(self, repo, pats, opts):
921 922 top = self.check_toppatch(repo)
922 923 if not top:
923 924 self.ui.write("No patches applied\n")
924 925 return
925 926 qp = self.qparents(repo, top)
926 927 if opts.get('git'):
927 928 self.diffopts().git = True
928 929 self.printdiff(repo, qp, files=pats, opts=opts)
929 930
930 931 def refresh(self, repo, pats=None, **opts):
931 932 if len(self.applied) == 0:
932 933 self.ui.write("No patches applied\n")
933 934 return 1
934 935 wlock = repo.wlock()
935 936 self.check_toppatch(repo)
936 937 (top, patchfn) = (self.applied[-1].rev, self.applied[-1].name)
937 938 top = revlog.bin(top)
938 939 cparents = repo.changelog.parents(top)
939 940 patchparent = self.qparents(repo, top)
940 941 message, comments, user, date, patchfound = self.readheaders(patchfn)
941 942
942 943 patchf = self.opener(patchfn, "w")
943 944 msg = opts.get('msg', '').rstrip()
944 945 if msg:
945 946 if comments:
946 947 # Remove existing message.
947 948 ci = 0
948 949 for mi in xrange(len(message)):
949 950 while message[mi] != comments[ci]:
950 951 ci += 1
951 952 del comments[ci]
952 953 comments.append(msg)
953 954 if comments:
954 955 comments = "\n".join(comments) + '\n\n'
955 956 patchf.write(comments)
956 957
957 958 if opts.get('git'):
958 959 self.diffopts().git = True
959 960 fns, matchfn, anypats = cmdutil.matchpats(repo, pats, opts)
960 961 tip = repo.changelog.tip()
961 962 if top == tip:
962 963 # if the top of our patch queue is also the tip, there is an
963 964 # optimization here. We update the dirstate in place and strip
964 965 # off the tip commit. Then just commit the current directory
965 966 # tree. We can also send repo.commit the list of files
966 967 # changed to speed up the diff
967 968 #
968 969 # in short mode, we only diff the files included in the
969 970 # patch already
970 971 #
971 972 # this should really read:
972 973 # mm, dd, aa, aa2, uu = repo.status(tip, patchparent)[:5]
973 974 # but we do it backwards to take advantage of manifest/chlog
974 975 # caching against the next repo.status call
975 976 #
976 977 mm, aa, dd, aa2, uu = repo.status(patchparent, tip)[:5]
977 978 changes = repo.changelog.read(tip)
978 979 man = repo.manifest.read(changes[0])
979 980 aaa = aa[:]
980 981 if opts.get('short'):
981 982 filelist = mm + aa + dd
982 983 else:
983 984 filelist = None
984 985 m, a, r, d, u = repo.status(files=filelist)[:5]
985 986
986 987 # we might end up with files that were added between tip and
987 988 # the dirstate parent, but then changed in the local dirstate.
988 989 # in this case, we want them to only show up in the added section
989 990 for x in m:
990 991 if x not in aa:
991 992 mm.append(x)
992 993 # we might end up with files added by the local dirstate that
993 994 # were deleted by the patch. In this case, they should only
994 995 # show up in the changed section.
995 996 for x in a:
996 997 if x in dd:
997 998 del dd[dd.index(x)]
998 999 mm.append(x)
999 1000 else:
1000 1001 aa.append(x)
1001 1002 # make sure any files deleted in the local dirstate
1002 1003 # are not in the add or change column of the patch
1003 1004 forget = []
1004 1005 for x in d + r:
1005 1006 if x in aa:
1006 1007 del aa[aa.index(x)]
1007 1008 forget.append(x)
1008 1009 continue
1009 1010 elif x in mm:
1010 1011 del mm[mm.index(x)]
1011 1012 dd.append(x)
1012 1013
1013 1014 m = util.unique(mm)
1014 1015 r = util.unique(dd)
1015 1016 a = util.unique(aa)
1016 1017 filelist = filter(matchfn, util.unique(m + r + a))
1017 1018 patch.diff(repo, patchparent, files=filelist, match=matchfn,
1018 1019 fp=patchf, changes=(m, a, r, [], u),
1019 1020 opts=self.diffopts())
1020 1021 patchf.close()
1021 1022
1022 1023 repo.dirstate.setparents(*cparents)
1023 1024 copies = {}
1024 1025 for dst in a:
1025 1026 src = repo.dirstate.copied(dst)
1026 1027 if src is None:
1027 1028 continue
1028 1029 copies.setdefault(src, []).append(dst)
1029 1030 repo.dirstate.update(a, 'a')
1030 1031 # remember the copies between patchparent and tip
1031 1032 # this may be slow, so don't do it if we're not tracking copies
1032 1033 if self.diffopts().git:
1033 1034 for dst in aaa:
1034 1035 f = repo.file(dst)
1035 1036 src = f.renamed(man[dst])
1036 1037 if src:
1037 1038 copies[src[0]] = copies.get(dst, [])
1038 1039 if dst in a:
1039 1040 copies[src[0]].append(dst)
1040 1041 # we can't copy a file created by the patch itself
1041 1042 if dst in copies:
1042 1043 del copies[dst]
1043 1044 for src, dsts in copies.iteritems():
1044 1045 for dst in dsts:
1045 1046 repo.dirstate.copy(src, dst)
1046 1047 repo.dirstate.update(r, 'r')
1047 1048 # if the patch excludes a modified file, mark that file with mtime=0
1048 1049 # so status can see it.
1049 1050 mm = []
1050 1051 for i in xrange(len(m)-1, -1, -1):
1051 1052 if not matchfn(m[i]):
1052 1053 mm.append(m[i])
1053 1054 del m[i]
1054 1055 repo.dirstate.update(m, 'n')
1055 1056 repo.dirstate.update(mm, 'n', st_mtime=0)
1056 1057 repo.dirstate.forget(forget)
1057 1058
1058 1059 if not msg:
1059 1060 if not message:
1060 1061 message = "patch queue: %s\n" % patchfn
1061 1062 else:
1062 1063 message = "\n".join(message)
1063 1064 else:
1064 1065 message = msg
1065 1066
1066 1067 self.strip(repo, top, update=False, backup='strip', wlock=wlock)
1067 1068 n = repo.commit(filelist, message, changes[1], force=1, wlock=wlock)
1068 1069 self.applied[-1] = statusentry(revlog.hex(n), patchfn)
1069 1070 self.applied_dirty = 1
1070 1071 else:
1071 1072 self.printdiff(repo, patchparent, fp=patchf)
1072 1073 patchf.close()
1073 1074 added = repo.status()[1]
1074 1075 for a in added:
1075 1076 f = repo.wjoin(a)
1076 1077 try:
1077 1078 os.unlink(f)
1078 1079 except OSError, e:
1079 1080 if e.errno != errno.ENOENT:
1080 1081 raise
1081 1082 try: os.removedirs(os.path.dirname(f))
1082 1083 except: pass
1083 1084 # forget the file copies in the dirstate
1084 1085 # push should readd the files later on
1085 1086 repo.dirstate.forget(added)
1086 1087 self.pop(repo, force=True, wlock=wlock)
1087 1088 self.push(repo, force=True, wlock=wlock)
1088 1089
1089 1090 def init(self, repo, create=False):
1090 1091 if os.path.isdir(self.path):
1091 1092 raise util.Abort(_("patch queue directory already exists"))
1092 1093 os.mkdir(self.path)
1093 1094 if create:
1094 1095 return self.qrepo(create=True)
1095 1096
1096 1097 def unapplied(self, repo, patch=None):
1097 1098 if patch and patch not in self.series:
1098 1099 raise util.Abort(_("patch %s is not in series file") % patch)
1099 1100 if not patch:
1100 1101 start = self.series_end()
1101 1102 else:
1102 1103 start = self.series.index(patch) + 1
1103 1104 unapplied = []
1104 1105 for i in xrange(start, len(self.series)):
1105 1106 pushable, reason = self.pushable(i)
1106 1107 if pushable:
1107 1108 unapplied.append((i, self.series[i]))
1108 1109 self.explain_pushable(i)
1109 1110 return unapplied
1110 1111
1111 1112 def qseries(self, repo, missing=None, start=0, length=0, status=None,
1112 1113 summary=False):
1113 1114 def displayname(patchname):
1114 1115 if summary:
1115 1116 msg = self.readheaders(patchname)[0]
1116 1117 msg = msg and ': ' + msg[0] or ': '
1117 1118 else:
1118 1119 msg = ''
1119 1120 return '%s%s' % (patchname, msg)
1120 1121
1121 1122 def pname(i):
1122 1123 if status == 'A':
1123 1124 return self.applied[i].name
1124 1125 else:
1125 1126 return self.series[i]
1126 1127
1127 1128 applied = dict.fromkeys([p.name for p in self.applied])
1128 1129 if not length:
1129 1130 length = len(self.series) - start
1130 1131 if not missing:
1131 1132 for i in xrange(start, start+length):
1132 1133 pfx = ''
1133 1134 patch = pname(i)
1134 1135 if self.ui.verbose:
1135 1136 if patch in applied:
1136 1137 stat = 'A'
1137 1138 elif self.pushable(i)[0]:
1138 1139 stat = 'U'
1139 1140 else:
1140 1141 stat = 'G'
1141 1142 pfx = '%d %s ' % (i, stat)
1142 1143 self.ui.write('%s%s\n' % (pfx, displayname(patch)))
1143 1144 else:
1144 1145 msng_list = []
1145 1146 for root, dirs, files in os.walk(self.path):
1146 1147 d = root[len(self.path) + 1:]
1147 1148 for f in files:
1148 1149 fl = os.path.join(d, f)
1149 1150 if (fl not in self.series and
1150 1151 fl not in (self.status_path, self.series_path)
1151 1152 and not fl.startswith('.')):
1152 1153 msng_list.append(fl)
1153 1154 msng_list.sort()
1154 1155 for x in msng_list:
1155 1156 pfx = self.ui.verbose and ('D ') or ''
1156 1157 self.ui.write("%s%s\n" % (pfx, displayname(x)))
1157 1158
1158 1159 def issaveline(self, l):
1159 1160 if l.name == '.hg.patches.save.line':
1160 1161 return True
1161 1162
1162 1163 def qrepo(self, create=False):
1163 1164 if create or os.path.isdir(self.join(".hg")):
1164 1165 return hg.repository(self.ui, path=self.path, create=create)
1165 1166
1166 1167 def restore(self, repo, rev, delete=None, qupdate=None):
1167 1168 c = repo.changelog.read(rev)
1168 1169 desc = c[4].strip()
1169 1170 lines = desc.splitlines()
1170 1171 i = 0
1171 1172 datastart = None
1172 1173 series = []
1173 1174 applied = []
1174 1175 qpp = None
1175 1176 for i in xrange(0, len(lines)):
1176 1177 if lines[i] == 'Patch Data:':
1177 1178 datastart = i + 1
1178 1179 elif lines[i].startswith('Dirstate:'):
1179 1180 l = lines[i].rstrip()
1180 1181 l = l[10:].split(' ')
1181 1182 qpp = [ hg.bin(x) for x in l ]
1182 1183 elif datastart != None:
1183 1184 l = lines[i].rstrip()
1184 1185 se = statusentry(l)
1185 1186 file_ = se.name
1186 1187 if se.rev:
1187 1188 applied.append(se)
1188 1189 else:
1189 1190 series.append(file_)
1190 1191 if datastart == None:
1191 1192 self.ui.warn("No saved patch data found\n")
1192 1193 return 1
1193 1194 self.ui.warn("restoring status: %s\n" % lines[0])
1194 1195 self.full_series = series
1195 1196 self.applied = applied
1196 1197 self.parse_series()
1197 1198 self.series_dirty = 1
1198 1199 self.applied_dirty = 1
1199 1200 heads = repo.changelog.heads()
1200 1201 if delete:
1201 1202 if rev not in heads:
1202 1203 self.ui.warn("save entry has children, leaving it alone\n")
1203 1204 else:
1204 1205 self.ui.warn("removing save entry %s\n" % hg.short(rev))
1205 1206 pp = repo.dirstate.parents()
1206 1207 if rev in pp:
1207 1208 update = True
1208 1209 else:
1209 1210 update = False
1210 1211 self.strip(repo, rev, update=update, backup='strip')
1211 1212 if qpp:
1212 1213 self.ui.warn("saved queue repository parents: %s %s\n" %
1213 1214 (hg.short(qpp[0]), hg.short(qpp[1])))
1214 1215 if qupdate:
1215 1216 print "queue directory updating"
1216 1217 r = self.qrepo()
1217 1218 if not r:
1218 1219 self.ui.warn("Unable to load queue repository\n")
1219 1220 return 1
1220 1221 hg.clean(r, qpp[0])
1221 1222
1222 1223 def save(self, repo, msg=None):
1223 1224 if len(self.applied) == 0:
1224 1225 self.ui.warn("save: no patches applied, exiting\n")
1225 1226 return 1
1226 1227 if self.issaveline(self.applied[-1]):
1227 1228 self.ui.warn("status is already saved\n")
1228 1229 return 1
1229 1230
1230 1231 ar = [ ':' + x for x in self.full_series ]
1231 1232 if not msg:
1232 1233 msg = "hg patches saved state"
1233 1234 else:
1234 1235 msg = "hg patches: " + msg.rstrip('\r\n')
1235 1236 r = self.qrepo()
1236 1237 if r:
1237 1238 pp = r.dirstate.parents()
1238 1239 msg += "\nDirstate: %s %s" % (hg.hex(pp[0]), hg.hex(pp[1]))
1239 1240 msg += "\n\nPatch Data:\n"
1240 1241 text = msg + "\n".join([str(x) for x in self.applied]) + '\n' + (ar and
1241 1242 "\n".join(ar) + '\n' or "")
1242 1243 n = repo.commit(None, text, user=None, force=1)
1243 1244 if not n:
1244 1245 self.ui.warn("repo commit failed\n")
1245 1246 return 1
1246 1247 self.applied.append(statusentry(revlog.hex(n),'.hg.patches.save.line'))
1247 1248 self.applied_dirty = 1
1248 1249
1249 1250 def full_series_end(self):
1250 1251 if len(self.applied) > 0:
1251 1252 p = self.applied[-1].name
1252 1253 end = self.find_series(p)
1253 1254 if end == None:
1254 1255 return len(self.full_series)
1255 1256 return end + 1
1256 1257 return 0
1257 1258
1258 1259 def series_end(self, all_patches=False):
1259 1260 end = 0
1260 1261 def next(start):
1261 1262 if all_patches:
1262 1263 return start
1263 1264 i = start
1264 1265 while i < len(self.series):
1265 1266 p, reason = self.pushable(i)
1266 1267 if p:
1267 1268 break
1268 1269 self.explain_pushable(i)
1269 1270 i += 1
1270 1271 return i
1271 1272 if len(self.applied) > 0:
1272 1273 p = self.applied[-1].name
1273 1274 try:
1274 1275 end = self.series.index(p)
1275 1276 except ValueError:
1276 1277 return 0
1277 1278 return next(end + 1)
1278 1279 return next(end)
1279 1280
1280 1281 def appliedname(self, index):
1281 1282 pname = self.applied[index].name
1282 1283 if not self.ui.verbose:
1283 1284 p = pname
1284 1285 else:
1285 1286 p = str(self.series.index(pname)) + " " + pname
1286 1287 return p
1287 1288
1288 1289 def qimport(self, repo, files, patchname=None, rev=None, existing=None,
1289 1290 force=None, git=False):
1290 1291 def checkseries(patchname):
1291 1292 if patchname in self.series:
1292 1293 raise util.Abort(_('patch %s is already in the series file')
1293 1294 % patchname)
1294 1295 def checkfile(patchname):
1295 1296 if not force and os.path.exists(self.join(patchname)):
1296 1297 raise util.Abort(_('patch "%s" already exists')
1297 1298 % patchname)
1298 1299
1299 1300 if rev:
1300 1301 if files:
1301 1302 raise util.Abort(_('option "-r" not valid when importing '
1302 1303 'files'))
1303 1304 rev = cmdutil.revrange(repo, rev)
1304 1305 rev.sort(lambda x, y: cmp(y, x))
1305 1306 if (len(files) > 1 or len(rev) > 1) and patchname:
1306 1307 raise util.Abort(_('option "-n" not valid when importing multiple '
1307 1308 'patches'))
1308 1309 i = 0
1309 1310 added = []
1310 1311 if rev:
1311 1312 # If mq patches are applied, we can only import revisions
1312 1313 # that form a linear path to qbase.
1313 1314 # Otherwise, they should form a linear path to a head.
1314 1315 heads = repo.changelog.heads(repo.changelog.node(rev[-1]))
1315 1316 if len(heads) > 1:
1316 1317 raise util.Abort(_('revision %d is the root of more than one '
1317 1318 'branch') % rev[-1])
1318 1319 if self.applied:
1319 1320 base = revlog.hex(repo.changelog.node(rev[0]))
1320 1321 if base in [n.rev for n in self.applied]:
1321 1322 raise util.Abort(_('revision %d is already managed')
1322 1323 % rev[0])
1323 1324 if heads != [revlog.bin(self.applied[-1].rev)]:
1324 1325 raise util.Abort(_('revision %d is not the parent of '
1325 1326 'the queue') % rev[0])
1326 1327 base = repo.changelog.rev(revlog.bin(self.applied[0].rev))
1327 1328 lastparent = repo.changelog.parentrevs(base)[0]
1328 1329 else:
1329 1330 if heads != [repo.changelog.node(rev[0])]:
1330 1331 raise util.Abort(_('revision %d has unmanaged children')
1331 1332 % rev[0])
1332 1333 lastparent = None
1333 1334
1334 1335 if git:
1335 1336 self.diffopts().git = True
1336 1337
1337 1338 for r in rev:
1338 1339 p1, p2 = repo.changelog.parentrevs(r)
1339 1340 n = repo.changelog.node(r)
1340 1341 if p2 != revlog.nullrev:
1341 1342 raise util.Abort(_('cannot import merge revision %d') % r)
1342 1343 if lastparent and lastparent != r:
1343 1344 raise util.Abort(_('revision %d is not the parent of %d')
1344 1345 % (r, lastparent))
1345 1346 lastparent = p1
1346 1347
1347 1348 if not patchname:
1348 1349 patchname = '%d.diff' % r
1349 1350 checkseries(patchname)
1350 1351 checkfile(patchname)
1351 1352 self.full_series.insert(0, patchname)
1352 1353
1353 1354 patchf = self.opener(patchname, "w")
1354 1355 patch.export(repo, [n], fp=patchf, opts=self.diffopts())
1355 1356 patchf.close()
1356 1357
1357 1358 se = statusentry(revlog.hex(n), patchname)
1358 1359 self.applied.insert(0, se)
1359 1360
1360 1361 added.append(patchname)
1361 1362 patchname = None
1362 1363 self.parse_series()
1363 1364 self.applied_dirty = 1
1364 1365
1365 1366 for filename in files:
1366 1367 if existing:
1367 1368 if filename == '-':
1368 1369 raise util.Abort(_('-e is incompatible with import from -'))
1369 1370 if not patchname:
1370 1371 patchname = filename
1371 1372 if not os.path.isfile(self.join(patchname)):
1372 1373 raise util.Abort(_("patch %s does not exist") % patchname)
1373 1374 else:
1374 1375 try:
1375 1376 if filename == '-':
1376 1377 if not patchname:
1377 1378 raise util.Abort(_('need --name to import a patch from -'))
1378 1379 text = sys.stdin.read()
1379 1380 else:
1380 1381 text = file(filename).read()
1381 1382 except IOError:
1382 1383 raise util.Abort(_("unable to read %s") % patchname)
1383 1384 if not patchname:
1384 1385 patchname = os.path.basename(filename)
1385 1386 checkfile(patchname)
1386 1387 patchf = self.opener(patchname, "w")
1387 1388 patchf.write(text)
1388 1389 checkseries(patchname)
1389 1390 index = self.full_series_end() + i
1390 1391 self.full_series[index:index] = [patchname]
1391 1392 self.parse_series()
1392 1393 self.ui.warn("adding %s to series file\n" % patchname)
1393 1394 i += 1
1394 1395 added.append(patchname)
1395 1396 patchname = None
1396 1397 self.series_dirty = 1
1397 1398 qrepo = self.qrepo()
1398 1399 if qrepo:
1399 1400 qrepo.add(added)
1400 1401
1401 1402 def delete(ui, repo, *patches, **opts):
1402 1403 """remove patches from queue
1403 1404
1404 1405 With --rev, mq will stop managing the named revisions. The
1405 1406 patches must be applied and at the base of the stack. This option
1406 1407 is useful when the patches have been applied upstream.
1407 1408
1408 1409 Otherwise, the patches must not be applied.
1409 1410
1410 1411 With --keep, the patch files are preserved in the patch directory."""
1411 1412 q = repo.mq
1412 1413 q.delete(repo, patches, opts)
1413 1414 q.save_dirty()
1414 1415 return 0
1415 1416
1416 1417 def applied(ui, repo, patch=None, **opts):
1417 1418 """print the patches already applied"""
1418 1419 q = repo.mq
1419 1420 if patch:
1420 1421 if patch not in q.series:
1421 1422 raise util.Abort(_("patch %s is not in series file") % patch)
1422 1423 end = q.series.index(patch) + 1
1423 1424 else:
1424 1425 end = len(q.applied)
1425 1426 if not end:
1426 1427 return
1427 1428
1428 1429 return q.qseries(repo, length=end, status='A', summary=opts.get('summary'))
1429 1430
1430 1431 def unapplied(ui, repo, patch=None, **opts):
1431 1432 """print the patches not yet applied"""
1432 1433 q = repo.mq
1433 1434 if patch:
1434 1435 if patch not in q.series:
1435 1436 raise util.Abort(_("patch %s is not in series file") % patch)
1436 1437 start = q.series.index(patch) + 1
1437 1438 else:
1438 1439 start = q.series_end()
1439 1440 q.qseries(repo, start=start, summary=opts.get('summary'))
1440 1441
1441 1442 def qimport(ui, repo, *filename, **opts):
1442 1443 """import a patch
1443 1444
1444 1445 The patch will have the same name as its source file unless you
1445 1446 give it a new one with --name.
1446 1447
1447 1448 You can register an existing patch inside the patch directory
1448 1449 with the --existing flag.
1449 1450
1450 1451 With --force, an existing patch of the same name will be overwritten.
1451 1452
1452 1453 An existing changeset may be placed under mq control with --rev
1453 1454 (e.g. qimport --rev tip -n patch will place tip under mq control).
1454 1455 With --git, patches imported with --rev will use the git diff
1455 1456 format.
1456 1457 """
1457 1458 q = repo.mq
1458 1459 q.qimport(repo, filename, patchname=opts['name'],
1459 1460 existing=opts['existing'], force=opts['force'], rev=opts['rev'],
1460 1461 git=opts['git'])
1461 1462 q.save_dirty()
1462 1463 return 0
1463 1464
1464 1465 def init(ui, repo, **opts):
1465 1466 """init a new queue repository
1466 1467
1467 1468 The queue repository is unversioned by default. If -c is
1468 1469 specified, qinit will create a separate nested repository
1469 1470 for patches. Use qcommit to commit changes to this queue
1470 1471 repository."""
1471 1472 q = repo.mq
1472 1473 r = q.init(repo, create=opts['create_repo'])
1473 1474 q.save_dirty()
1474 1475 if r:
1475 1476 fp = r.wopener('.hgignore', 'w')
1476 1477 print >> fp, 'syntax: glob'
1477 1478 print >> fp, 'status'
1478 1479 print >> fp, 'guards'
1479 1480 fp.close()
1480 1481 r.wopener('series', 'w').close()
1481 1482 r.add(['.hgignore', 'series'])
1482 1483 return 0
1483 1484
1484 1485 def clone(ui, source, dest=None, **opts):
1485 1486 '''clone main and patch repository at same time
1486 1487
1487 1488 If source is local, destination will have no patches applied. If
1488 1489 source is remote, this command can not check if patches are
1489 1490 applied in source, so cannot guarantee that patches are not
1490 1491 applied in destination. If you clone remote repository, be sure
1491 1492 before that it has no patches applied.
1492 1493
1493 1494 Source patch repository is looked for in <src>/.hg/patches by
1494 1495 default. Use -p <url> to change.
1495 1496 '''
1496 1497 commands.setremoteconfig(ui, opts)
1497 1498 if dest is None:
1498 1499 dest = hg.defaultdest(source)
1499 1500 sr = hg.repository(ui, ui.expandpath(source))
1500 1501 qbase, destrev = None, None
1501 1502 if sr.local():
1502 1503 reposetup(ui, sr)
1503 1504 if sr.mq.applied:
1504 1505 qbase = revlog.bin(sr.mq.applied[0].rev)
1505 1506 if not hg.islocal(dest):
1506 1507 destrev = sr.parents(qbase)[0]
1507 1508 ui.note(_('cloning main repo\n'))
1508 1509 sr, dr = hg.clone(ui, sr, dest,
1509 1510 pull=opts['pull'],
1510 1511 rev=destrev,
1511 1512 update=False,
1512 1513 stream=opts['uncompressed'])
1513 1514 ui.note(_('cloning patch repo\n'))
1514 1515 spr, dpr = hg.clone(ui, opts['patches'] or (sr.url() + '/.hg/patches'),
1515 1516 dr.url() + '/.hg/patches',
1516 1517 pull=opts['pull'],
1517 1518 update=not opts['noupdate'],
1518 1519 stream=opts['uncompressed'])
1519 1520 if dr.local():
1520 1521 if qbase:
1521 1522 ui.note(_('stripping applied patches from destination repo\n'))
1522 1523 reposetup(ui, dr)
1523 1524 dr.mq.strip(dr, qbase, update=False, backup=None)
1524 1525 if not opts['noupdate']:
1525 1526 ui.note(_('updating destination repo\n'))
1526 1527 hg.update(dr, dr.changelog.tip())
1527 1528
1528 1529 def commit(ui, repo, *pats, **opts):
1529 1530 """commit changes in the queue repository"""
1530 1531 q = repo.mq
1531 1532 r = q.qrepo()
1532 1533 if not r: raise util.Abort('no queue repository')
1533 1534 commands.commit(r.ui, r, *pats, **opts)
1534 1535
1535 1536 def series(ui, repo, **opts):
1536 1537 """print the entire series file"""
1537 1538 repo.mq.qseries(repo, missing=opts['missing'], summary=opts['summary'])
1538 1539 return 0
1539 1540
1540 1541 def top(ui, repo, **opts):
1541 1542 """print the name of the current patch"""
1542 1543 q = repo.mq
1543 1544 t = len(q.applied)
1544 1545 if t:
1545 1546 return q.qseries(repo, start=t-1, length=1, status='A',
1546 1547 summary=opts.get('summary'))
1547 1548 else:
1548 1549 ui.write("No patches applied\n")
1549 1550 return 1
1550 1551
1551 1552 def next(ui, repo, **opts):
1552 1553 """print the name of the next patch"""
1553 1554 q = repo.mq
1554 1555 end = q.series_end()
1555 1556 if end == len(q.series):
1556 1557 ui.write("All patches applied\n")
1557 1558 return 1
1558 1559 return q.qseries(repo, start=end, length=1, summary=opts.get('summary'))
1559 1560
1560 1561 def prev(ui, repo, **opts):
1561 1562 """print the name of the previous patch"""
1562 1563 q = repo.mq
1563 1564 l = len(q.applied)
1564 1565 if l == 1:
1565 1566 ui.write("Only one patch applied\n")
1566 1567 return 1
1567 1568 if not l:
1568 1569 ui.write("No patches applied\n")
1569 1570 return 1
1570 1571 return q.qseries(repo, start=l-2, length=1, status='A',
1571 1572 summary=opts.get('summary'))
1572 1573
1573 1574 def new(ui, repo, patch, **opts):
1574 1575 """create a new patch
1575 1576
1576 1577 qnew creates a new patch on top of the currently-applied patch
1577 1578 (if any). It will refuse to run if there are any outstanding
1578 1579 changes unless -f is specified, in which case the patch will
1579 1580 be initialised with them.
1580 1581
1581 1582 -e, -m or -l set the patch header as well as the commit message.
1582 1583 If none is specified, the patch header is empty and the
1583 1584 commit message is 'New patch: PATCH'"""
1584 1585 q = repo.mq
1585 1586 message = commands.logmessage(opts)
1586 1587 if opts['edit']:
1587 1588 message = ui.edit(message, ui.username())
1588 1589 q.new(repo, patch, msg=message, force=opts['force'])
1589 1590 q.save_dirty()
1590 1591 return 0
1591 1592
1592 1593 def refresh(ui, repo, *pats, **opts):
1593 1594 """update the current patch
1594 1595
1595 1596 If any file patterns are provided, the refreshed patch will contain only
1596 1597 the modifications that match those patterns; the remaining modifications
1597 1598 will remain in the working directory.
1598 1599 """
1599 1600 q = repo.mq
1600 1601 message = commands.logmessage(opts)
1601 1602 if opts['edit']:
1602 1603 if message:
1603 1604 raise util.Abort(_('option "-e" incompatible with "-m" or "-l"'))
1604 1605 patch = q.applied[-1].name
1605 1606 (message, comment, user, date, hasdiff) = q.readheaders(patch)
1606 1607 message = ui.edit('\n'.join(message), user or ui.username())
1607 1608 ret = q.refresh(repo, pats, msg=message, **opts)
1608 1609 q.save_dirty()
1609 1610 return ret
1610 1611
1611 1612 def diff(ui, repo, *pats, **opts):
1612 1613 """diff of the current patch"""
1613 1614 repo.mq.diff(repo, pats, opts)
1614 1615 return 0
1615 1616
1616 1617 def fold(ui, repo, *files, **opts):
1617 1618 """fold the named patches into the current patch
1618 1619
1619 1620 Patches must not yet be applied. Each patch will be successively
1620 1621 applied to the current patch in the order given. If all the
1621 1622 patches apply successfully, the current patch will be refreshed
1622 1623 with the new cumulative patch, and the folded patches will
1623 1624 be deleted. With -k/--keep, the folded patch files will not
1624 1625 be removed afterwards.
1625 1626
1626 1627 The header for each folded patch will be concatenated with
1627 1628 the current patch header, separated by a line of '* * *'."""
1628 1629
1629 1630 q = repo.mq
1630 1631
1631 1632 if not files:
1632 1633 raise util.Abort(_('qfold requires at least one patch name'))
1633 1634 if not q.check_toppatch(repo):
1634 1635 raise util.Abort(_('No patches applied'))
1635 1636
1636 1637 message = commands.logmessage(opts)
1637 1638 if opts['edit']:
1638 1639 if message:
1639 1640 raise util.Abort(_('option "-e" incompatible with "-m" or "-l"'))
1640 1641
1641 1642 parent = q.lookup('qtip')
1642 1643 patches = []
1643 1644 messages = []
1644 1645 for f in files:
1645 1646 p = q.lookup(f)
1646 1647 if p in patches or p == parent:
1647 1648 ui.warn(_('Skipping already folded patch %s') % p)
1648 1649 if q.isapplied(p):
1649 1650 raise util.Abort(_('qfold cannot fold already applied patch %s') % p)
1650 1651 patches.append(p)
1651 1652
1652 1653 for p in patches:
1653 1654 if not message:
1654 1655 messages.append(q.readheaders(p)[0])
1655 1656 pf = q.join(p)
1656 1657 (patchsuccess, files, fuzz) = q.patch(repo, pf)
1657 1658 if not patchsuccess:
1658 1659 raise util.Abort(_('Error folding patch %s') % p)
1659 1660 patch.updatedir(ui, repo, files)
1660 1661
1661 1662 if not message:
1662 1663 message, comments, user = q.readheaders(parent)[0:3]
1663 1664 for msg in messages:
1664 1665 message.append('* * *')
1665 1666 message.extend(msg)
1666 1667 message = '\n'.join(message)
1667 1668
1668 1669 if opts['edit']:
1669 1670 message = ui.edit(message, user or ui.username())
1670 1671
1671 1672 q.refresh(repo, msg=message)
1672 1673 q.delete(repo, patches, opts)
1673 1674 q.save_dirty()
1674 1675
1675 1676 def guard(ui, repo, *args, **opts):
1676 1677 '''set or print guards for a patch
1677 1678
1678 1679 Guards control whether a patch can be pushed. A patch with no
1679 1680 guards is always pushed. A patch with a positive guard ("+foo") is
1680 1681 pushed only if the qselect command has activated it. A patch with
1681 1682 a negative guard ("-foo") is never pushed if the qselect command
1682 1683 has activated it.
1683 1684
1684 1685 With no arguments, print the currently active guards.
1685 1686 With arguments, set guards for the named patch.
1686 1687
1687 1688 To set a negative guard "-foo" on topmost patch ("--" is needed so
1688 1689 hg will not interpret "-foo" as an option):
1689 1690 hg qguard -- -foo
1690 1691
1691 1692 To set guards on another patch:
1692 1693 hg qguard other.patch +2.6.17 -stable
1693 1694 '''
1694 1695 def status(idx):
1695 1696 guards = q.series_guards[idx] or ['unguarded']
1696 1697 ui.write('%s: %s\n' % (q.series[idx], ' '.join(guards)))
1697 1698 q = repo.mq
1698 1699 patch = None
1699 1700 args = list(args)
1700 1701 if opts['list']:
1701 1702 if args or opts['none']:
1702 1703 raise util.Abort(_('cannot mix -l/--list with options or arguments'))
1703 1704 for i in xrange(len(q.series)):
1704 1705 status(i)
1705 1706 return
1706 1707 if not args or args[0][0:1] in '-+':
1707 1708 if not q.applied:
1708 1709 raise util.Abort(_('no patches applied'))
1709 1710 patch = q.applied[-1].name
1710 1711 if patch is None and args[0][0:1] not in '-+':
1711 1712 patch = args.pop(0)
1712 1713 if patch is None:
1713 1714 raise util.Abort(_('no patch to work with'))
1714 1715 if args or opts['none']:
1715 1716 q.set_guards(q.find_series(patch), args)
1716 1717 q.save_dirty()
1717 1718 else:
1718 1719 status(q.series.index(q.lookup(patch)))
1719 1720
1720 1721 def header(ui, repo, patch=None):
1721 1722 """Print the header of the topmost or specified patch"""
1722 1723 q = repo.mq
1723 1724
1724 1725 if patch:
1725 1726 patch = q.lookup(patch)
1726 1727 else:
1727 1728 if not q.applied:
1728 1729 ui.write('No patches applied\n')
1729 1730 return 1
1730 1731 patch = q.lookup('qtip')
1731 1732 message = repo.mq.readheaders(patch)[0]
1732 1733
1733 1734 ui.write('\n'.join(message) + '\n')
1734 1735
1735 1736 def lastsavename(path):
1736 1737 (directory, base) = os.path.split(path)
1737 1738 names = os.listdir(directory)
1738 1739 namere = re.compile("%s.([0-9]+)" % base)
1739 1740 maxindex = None
1740 1741 maxname = None
1741 1742 for f in names:
1742 1743 m = namere.match(f)
1743 1744 if m:
1744 1745 index = int(m.group(1))
1745 1746 if maxindex == None or index > maxindex:
1746 1747 maxindex = index
1747 1748 maxname = f
1748 1749 if maxname:
1749 1750 return (os.path.join(directory, maxname), maxindex)
1750 1751 return (None, None)
1751 1752
1752 1753 def savename(path):
1753 1754 (last, index) = lastsavename(path)
1754 1755 if last is None:
1755 1756 index = 0
1756 1757 newpath = path + ".%d" % (index + 1)
1757 1758 return newpath
1758 1759
1759 1760 def push(ui, repo, patch=None, **opts):
1760 1761 """push the next patch onto the stack"""
1761 1762 q = repo.mq
1762 1763 mergeq = None
1763 1764
1764 1765 if opts['all']:
1765 1766 if not q.series:
1766 1767 raise util.Abort(_('no patches in series'))
1767 1768 patch = q.series[-1]
1768 1769 if opts['merge']:
1769 1770 if opts['name']:
1770 1771 newpath = opts['name']
1771 1772 else:
1772 1773 newpath, i = lastsavename(q.path)
1773 1774 if not newpath:
1774 1775 ui.warn("no saved queues found, please use -n\n")
1775 1776 return 1
1776 1777 mergeq = queue(ui, repo.join(""), newpath)
1777 1778 ui.warn("merging with queue at: %s\n" % mergeq.path)
1778 1779 ret = q.push(repo, patch, force=opts['force'], list=opts['list'],
1779 1780 mergeq=mergeq)
1780 1781 q.save_dirty()
1781 1782 return ret
1782 1783
1783 1784 def pop(ui, repo, patch=None, **opts):
1784 1785 """pop the current patch off the stack"""
1785 1786 localupdate = True
1786 1787 if opts['name']:
1787 1788 q = queue(ui, repo.join(""), repo.join(opts['name']))
1788 1789 ui.warn('using patch queue: %s\n' % q.path)
1789 1790 localupdate = False
1790 1791 else:
1791 1792 q = repo.mq
1792 1793 q.pop(repo, patch, force=opts['force'], update=localupdate, all=opts['all'])
1793 1794 q.save_dirty()
1794 1795 return 0
1795 1796
1796 1797 def rename(ui, repo, patch, name=None, **opts):
1797 1798 """rename a patch
1798 1799
1799 1800 With one argument, renames the current patch to PATCH1.
1800 1801 With two arguments, renames PATCH1 to PATCH2."""
1801 1802
1802 1803 q = repo.mq
1803 1804
1804 1805 if not name:
1805 1806 name = patch
1806 1807 patch = None
1807 1808
1808 1809 if patch:
1809 1810 patch = q.lookup(patch)
1810 1811 else:
1811 1812 if not q.applied:
1812 1813 ui.write(_('No patches applied\n'))
1813 1814 return
1814 1815 patch = q.lookup('qtip')
1815 1816 absdest = q.join(name)
1816 1817 if os.path.isdir(absdest):
1817 1818 name = os.path.join(name, os.path.basename(patch))
1818 1819 absdest = q.join(name)
1819 1820 if os.path.exists(absdest):
1820 1821 raise util.Abort(_('%s already exists') % absdest)
1821 1822
1822 1823 if name in q.series:
1823 1824 raise util.Abort(_('A patch named %s already exists in the series file') % name)
1824 1825
1825 1826 if ui.verbose:
1826 1827 ui.write('Renaming %s to %s\n' % (patch, name))
1827 1828 i = q.find_series(patch)
1828 1829 guards = q.guard_re.findall(q.full_series[i])
1829 1830 q.full_series[i] = name + ''.join([' #' + g for g in guards])
1830 1831 q.parse_series()
1831 1832 q.series_dirty = 1
1832 1833
1833 1834 info = q.isapplied(patch)
1834 1835 if info:
1835 1836 q.applied[info[0]] = statusentry(info[1], name)
1836 1837 q.applied_dirty = 1
1837 1838
1838 1839 util.rename(q.join(patch), absdest)
1839 1840 r = q.qrepo()
1840 1841 if r:
1841 1842 wlock = r.wlock()
1842 1843 if r.dirstate.state(name) == 'r':
1843 1844 r.undelete([name], wlock)
1844 1845 r.copy(patch, name, wlock)
1845 1846 r.remove([patch], False, wlock)
1846 1847
1847 1848 q.save_dirty()
1848 1849
1849 1850 def restore(ui, repo, rev, **opts):
1850 1851 """restore the queue state saved by a rev"""
1851 1852 rev = repo.lookup(rev)
1852 1853 q = repo.mq
1853 1854 q.restore(repo, rev, delete=opts['delete'],
1854 1855 qupdate=opts['update'])
1855 1856 q.save_dirty()
1856 1857 return 0
1857 1858
1858 1859 def save(ui, repo, **opts):
1859 1860 """save current queue state"""
1860 1861 q = repo.mq
1861 1862 message = commands.logmessage(opts)
1862 1863 ret = q.save(repo, msg=message)
1863 1864 if ret:
1864 1865 return ret
1865 1866 q.save_dirty()
1866 1867 if opts['copy']:
1867 1868 path = q.path
1868 1869 if opts['name']:
1869 1870 newpath = os.path.join(q.basepath, opts['name'])
1870 1871 if os.path.exists(newpath):
1871 1872 if not os.path.isdir(newpath):
1872 1873 raise util.Abort(_('destination %s exists and is not '
1873 1874 'a directory') % newpath)
1874 1875 if not opts['force']:
1875 1876 raise util.Abort(_('destination %s exists, '
1876 1877 'use -f to force') % newpath)
1877 1878 else:
1878 1879 newpath = savename(path)
1879 1880 ui.warn("copy %s to %s\n" % (path, newpath))
1880 1881 util.copyfiles(path, newpath)
1881 1882 if opts['empty']:
1882 1883 try:
1883 1884 os.unlink(q.join(q.status_path))
1884 1885 except:
1885 1886 pass
1886 1887 return 0
1887 1888
1888 1889 def strip(ui, repo, rev, **opts):
1889 1890 """strip a revision and all later revs on the same branch"""
1890 1891 rev = repo.lookup(rev)
1891 1892 backup = 'all'
1892 1893 if opts['backup']:
1893 1894 backup = 'strip'
1894 1895 elif opts['nobackup']:
1895 1896 backup = 'none'
1896 1897 update = repo.dirstate.parents()[0] != revlog.nullid
1897 1898 repo.mq.strip(repo, rev, backup=backup, update=update)
1898 1899 return 0
1899 1900
1900 1901 def select(ui, repo, *args, **opts):
1901 1902 '''set or print guarded patches to push
1902 1903
1903 1904 Use the qguard command to set or print guards on patch, then use
1904 1905 qselect to tell mq which guards to use. A patch will be pushed if it
1905 1906 has no guards or any positive guards match the currently selected guard,
1906 1907 but will not be pushed if any negative guards match the current guard.
1907 1908 For example:
1908 1909
1909 1910 qguard foo.patch -stable (negative guard)
1910 1911 qguard bar.patch +stable (positive guard)
1911 1912 qselect stable
1912 1913
1913 1914 This activates the "stable" guard. mq will skip foo.patch (because
1914 1915 it has a negative match) but push bar.patch (because it
1915 1916 has a positive match).
1916 1917
1917 1918 With no arguments, prints the currently active guards.
1918 1919 With one argument, sets the active guard.
1919 1920
1920 1921 Use -n/--none to deactivate guards (no other arguments needed).
1921 1922 When no guards are active, patches with positive guards are skipped
1922 1923 and patches with negative guards are pushed.
1923 1924
1924 1925 qselect can change the guards on applied patches. It does not pop
1925 1926 guarded patches by default. Use --pop to pop back to the last applied
1926 1927 patch that is not guarded. Use --reapply (which implies --pop) to push
1927 1928 back to the current patch afterwards, but skip guarded patches.
1928 1929
1929 1930 Use -s/--series to print a list of all guards in the series file (no
1930 1931 other arguments needed). Use -v for more information.'''
1931 1932
1932 1933 q = repo.mq
1933 1934 guards = q.active()
1934 1935 if args or opts['none']:
1935 1936 old_unapplied = q.unapplied(repo)
1936 1937 old_guarded = [i for i in xrange(len(q.applied)) if
1937 1938 not q.pushable(i)[0]]
1938 1939 q.set_active(args)
1939 1940 q.save_dirty()
1940 1941 if not args:
1941 1942 ui.status(_('guards deactivated\n'))
1942 1943 if not opts['pop'] and not opts['reapply']:
1943 1944 unapplied = q.unapplied(repo)
1944 1945 guarded = [i for i in xrange(len(q.applied))
1945 1946 if not q.pushable(i)[0]]
1946 1947 if len(unapplied) != len(old_unapplied):
1947 1948 ui.status(_('number of unguarded, unapplied patches has '
1948 1949 'changed from %d to %d\n') %
1949 1950 (len(old_unapplied), len(unapplied)))
1950 1951 if len(guarded) != len(old_guarded):
1951 1952 ui.status(_('number of guarded, applied patches has changed '
1952 1953 'from %d to %d\n') %
1953 1954 (len(old_guarded), len(guarded)))
1954 1955 elif opts['series']:
1955 1956 guards = {}
1956 1957 noguards = 0
1957 1958 for gs in q.series_guards:
1958 1959 if not gs:
1959 1960 noguards += 1
1960 1961 for g in gs:
1961 1962 guards.setdefault(g, 0)
1962 1963 guards[g] += 1
1963 1964 if ui.verbose:
1964 1965 guards['NONE'] = noguards
1965 1966 guards = guards.items()
1966 1967 guards.sort(lambda a, b: cmp(a[0][1:], b[0][1:]))
1967 1968 if guards:
1968 1969 ui.note(_('guards in series file:\n'))
1969 1970 for guard, count in guards:
1970 1971 ui.note('%2d ' % count)
1971 1972 ui.write(guard, '\n')
1972 1973 else:
1973 1974 ui.note(_('no guards in series file\n'))
1974 1975 else:
1975 1976 if guards:
1976 1977 ui.note(_('active guards:\n'))
1977 1978 for g in guards:
1978 1979 ui.write(g, '\n')
1979 1980 else:
1980 1981 ui.write(_('no active guards\n'))
1981 1982 reapply = opts['reapply'] and q.applied and q.appliedname(-1)
1982 1983 popped = False
1983 1984 if opts['pop'] or opts['reapply']:
1984 1985 for i in xrange(len(q.applied)):
1985 1986 pushable, reason = q.pushable(i)
1986 1987 if not pushable:
1987 1988 ui.status(_('popping guarded patches\n'))
1988 1989 popped = True
1989 1990 if i == 0:
1990 1991 q.pop(repo, all=True)
1991 1992 else:
1992 1993 q.pop(repo, i-1)
1993 1994 break
1994 1995 if popped:
1995 1996 try:
1996 1997 if reapply:
1997 1998 ui.status(_('reapplying unguarded patches\n'))
1998 1999 q.push(repo, reapply)
1999 2000 finally:
2000 2001 q.save_dirty()
2001 2002
2002 2003 def reposetup(ui, repo):
2003 2004 class mqrepo(repo.__class__):
2004 2005 def abort_if_wdir_patched(self, errmsg, force=False):
2005 2006 if self.mq.applied and not force:
2006 2007 parent = revlog.hex(self.dirstate.parents()[0])
2007 2008 if parent in [s.rev for s in self.mq.applied]:
2008 2009 raise util.Abort(errmsg)
2009 2010
2010 2011 def commit(self, *args, **opts):
2011 2012 if len(args) >= 6:
2012 2013 force = args[5]
2013 2014 else:
2014 2015 force = opts.get('force')
2015 2016 self.abort_if_wdir_patched(
2016 2017 _('cannot commit over an applied mq patch'),
2017 2018 force)
2018 2019
2019 2020 return super(mqrepo, self).commit(*args, **opts)
2020 2021
2021 2022 def push(self, remote, force=False, revs=None):
2022 2023 if self.mq.applied and not force:
2023 2024 raise util.Abort(_('source has mq patches applied'))
2024 2025 return super(mqrepo, self).push(remote, force, revs)
2025 2026
2026 2027 def tags(self):
2027 2028 if self.tagscache:
2028 2029 return self.tagscache
2029 2030
2030 2031 tagscache = super(mqrepo, self).tags()
2031 2032
2032 2033 q = self.mq
2033 2034 if not q.applied:
2034 2035 return tagscache
2035 2036
2036 2037 mqtags = [(patch.rev, patch.name) for patch in q.applied]
2037 2038 mqtags.append((mqtags[-1][0], 'qtip'))
2038 2039 mqtags.append((mqtags[0][0], 'qbase'))
2039 2040 for patch in mqtags:
2040 2041 if patch[1] in tagscache:
2041 2042 self.ui.warn('Tag %s overrides mq patch of the same name\n' % patch[1])
2042 2043 else:
2043 2044 tagscache[patch[1]] = revlog.bin(patch[0])
2044 2045
2045 2046 return tagscache
2046 2047
2047 2048 def _branchtags(self):
2048 2049 q = self.mq
2049 2050 if not q.applied:
2050 2051 return super(mqrepo, self)._branchtags()
2051 2052
2052 2053 self.branchcache = {} # avoid recursion in changectx
2053 2054 cl = self.changelog
2054 2055 partial, last, lrev = self._readbranchcache()
2055 2056
2056 2057 qbase = cl.rev(revlog.bin(q.applied[0].rev))
2057 2058 start = lrev + 1
2058 2059 if start < qbase:
2059 2060 # update the cache (excluding the patches) and save it
2060 2061 self._updatebranchcache(partial, lrev+1, qbase)
2061 2062 self._writebranchcache(partial, cl.node(qbase-1), qbase-1)
2062 2063 start = qbase
2063 2064 # if start = qbase, the cache is as updated as it should be.
2064 2065 # if start > qbase, the cache includes (part of) the patches.
2065 2066 # we might as well use it, but we won't save it.
2066 2067
2067 2068 # update the cache up to the tip
2068 2069 self._updatebranchcache(partial, start, cl.count())
2069 2070
2070 2071 return partial
2071 2072
2072 2073 if repo.local():
2073 2074 repo.__class__ = mqrepo
2074 2075 repo.mq = queue(ui, repo.join(""))
2075 2076
2076 2077 seriesopts = [('s', 'summary', None, _('print first line of patch header'))]
2077 2078
2078 2079 cmdtable = {
2079 2080 "qapplied": (applied, [] + seriesopts, 'hg qapplied [-s] [PATCH]'),
2080 2081 "qclone": (clone,
2081 2082 [('', 'pull', None, _('use pull protocol to copy metadata')),
2082 2083 ('U', 'noupdate', None, _('do not update the new working directories')),
2083 2084 ('', 'uncompressed', None,
2084 2085 _('use uncompressed transfer (fast over LAN)')),
2085 2086 ('e', 'ssh', '', _('specify ssh command to use')),
2086 2087 ('p', 'patches', '', _('location of source patch repo')),
2087 2088 ('', 'remotecmd', '',
2088 2089 _('specify hg command to run on the remote side'))],
2089 2090 'hg qclone [OPTION]... SOURCE [DEST]'),
2090 2091 "qcommit|qci":
2091 2092 (commit,
2092 2093 commands.table["^commit|ci"][1],
2093 2094 'hg qcommit [OPTION]... [FILE]...'),
2094 2095 "^qdiff": (diff,
2095 2096 [('g', 'git', None, _('use git extended diff format')),
2096 2097 ('I', 'include', [], _('include names matching the given patterns')),
2097 2098 ('X', 'exclude', [], _('exclude names matching the given patterns'))],
2098 2099 'hg qdiff [-I] [-X] [FILE]...'),
2099 2100 "qdelete|qremove|qrm":
2100 2101 (delete,
2101 2102 [('k', 'keep', None, _('keep patch file')),
2102 2103 ('r', 'rev', [], _('stop managing a revision'))],
2103 2104 'hg qdelete [-k] [-r REV]... PATCH...'),
2104 2105 'qfold':
2105 2106 (fold,
2106 2107 [('e', 'edit', None, _('edit patch header')),
2107 2108 ('k', 'keep', None, _('keep folded patch files'))
2108 2109 ] + commands.commitopts,
2109 2110 'hg qfold [-e] [-m <text>] [-l <file] PATCH...'),
2110 2111 'qguard': (guard, [('l', 'list', None, _('list all patches and guards')),
2111 2112 ('n', 'none', None, _('drop all guards'))],
2112 2113 'hg qguard [PATCH] [+GUARD...] [-GUARD...]'),
2113 2114 'qheader': (header, [],
2114 2115 _('hg qheader [PATCH]')),
2115 2116 "^qimport":
2116 2117 (qimport,
2117 2118 [('e', 'existing', None, 'import file in patch dir'),
2118 2119 ('n', 'name', '', 'patch file name'),
2119 2120 ('f', 'force', None, 'overwrite existing files'),
2120 2121 ('r', 'rev', [], 'place existing revisions under mq control'),
2121 2122 ('g', 'git', None, _('use git extended diff format'))],
2122 2123 'hg qimport [-e] [-n NAME] [-f] [-g] [-r REV]... FILE...'),
2123 2124 "^qinit":
2124 2125 (init,
2125 2126 [('c', 'create-repo', None, 'create queue repository')],
2126 2127 'hg qinit [-c]'),
2127 2128 "qnew":
2128 2129 (new,
2129 2130 [('e', 'edit', None, _('edit commit message')),
2130 2131 ('f', 'force', None, _('import uncommitted changes into patch'))
2131 2132 ] + commands.commitopts,
2132 2133 'hg qnew [-e] [-m TEXT] [-l FILE] [-f] PATCH'),
2133 2134 "qnext": (next, [] + seriesopts, 'hg qnext [-s]'),
2134 2135 "qprev": (prev, [] + seriesopts, 'hg qprev [-s]'),
2135 2136 "^qpop":
2136 2137 (pop,
2137 2138 [('a', 'all', None, 'pop all patches'),
2138 2139 ('n', 'name', '', 'queue name to pop'),
2139 2140 ('f', 'force', None, 'forget any local changes')],
2140 2141 'hg qpop [-a] [-n NAME] [-f] [PATCH | INDEX]'),
2141 2142 "^qpush":
2142 2143 (push,
2143 2144 [('f', 'force', None, 'apply if the patch has rejects'),
2144 2145 ('l', 'list', None, 'list patch name in commit text'),
2145 2146 ('a', 'all', None, 'apply all patches'),
2146 2147 ('m', 'merge', None, 'merge from another queue'),
2147 2148 ('n', 'name', '', 'merge queue name')],
2148 2149 'hg qpush [-f] [-l] [-a] [-m] [-n NAME] [PATCH | INDEX]'),
2149 2150 "^qrefresh":
2150 2151 (refresh,
2151 2152 [('e', 'edit', None, _('edit commit message')),
2152 2153 ('g', 'git', None, _('use git extended diff format')),
2153 2154 ('s', 'short', None, 'refresh only files already in the patch'),
2154 2155 ('I', 'include', [], _('include names matching the given patterns')),
2155 2156 ('X', 'exclude', [], _('exclude names matching the given patterns'))
2156 2157 ] + commands.commitopts,
2157 2158 'hg qrefresh [-I] [-X] [-e] [-m TEXT] [-l FILE] [-s] FILES...'),
2158 2159 'qrename|qmv':
2159 2160 (rename, [], 'hg qrename PATCH1 [PATCH2]'),
2160 2161 "qrestore":
2161 2162 (restore,
2162 2163 [('d', 'delete', None, 'delete save entry'),
2163 2164 ('u', 'update', None, 'update queue working dir')],
2164 2165 'hg qrestore [-d] [-u] REV'),
2165 2166 "qsave":
2166 2167 (save,
2167 2168 [('c', 'copy', None, 'copy patch directory'),
2168 2169 ('n', 'name', '', 'copy directory name'),
2169 2170 ('e', 'empty', None, 'clear queue status file'),
2170 2171 ('f', 'force', None, 'force copy')] + commands.commitopts,
2171 2172 'hg qsave [-m TEXT] [-l FILE] [-c] [-n NAME] [-e] [-f]'),
2172 2173 "qselect": (select,
2173 2174 [('n', 'none', None, _('disable all guards')),
2174 2175 ('s', 'series', None, _('list all guards in series file')),
2175 2176 ('', 'pop', None,
2176 2177 _('pop to before first guarded applied patch')),
2177 2178 ('', 'reapply', None, _('pop, then reapply patches'))],
2178 2179 'hg qselect [OPTION...] [GUARD...]'),
2179 2180 "qseries":
2180 2181 (series,
2181 2182 [('m', 'missing', None, 'print patches not in series')] + seriesopts,
2182 2183 'hg qseries [-ms]'),
2183 2184 "^strip":
2184 2185 (strip,
2185 2186 [('f', 'force', None, 'force multi-head removal'),
2186 2187 ('b', 'backup', None, 'bundle unrelated changesets'),
2187 2188 ('n', 'nobackup', None, 'no backups')],
2188 2189 'hg strip [-f] [-b] [-n] REV'),
2189 2190 "qtop": (top, [] + seriesopts, 'hg qtop [-s]'),
2190 2191 "qunapplied": (unapplied, [] + seriesopts, 'hg qunapplied [-s] [PATCH]'),
2191 2192 }
General Comments 0
You need to be logged in to leave comments. Login now