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