##// END OF EJS Templates
rebase: lose the comparison to `nullmerge`...
Pierre-Yves David -
r18446:c83d36b8 default
parent child Browse files
Show More
@@ -1,741 +1,741 b''
1 # rebase.py - rebasing feature for mercurial
1 # rebase.py - rebasing feature for mercurial
2 #
2 #
3 # Copyright 2008 Stefano Tortarolo <stefano.tortarolo at gmail dot com>
3 # Copyright 2008 Stefano Tortarolo <stefano.tortarolo at gmail dot com>
4 #
4 #
5 # This software may be used and distributed according to the terms of the
5 # This software may be used and distributed according to the terms of the
6 # GNU General Public License version 2 or any later version.
6 # GNU General Public License version 2 or any later version.
7
7
8 '''command to move sets of revisions to a different ancestor
8 '''command to move sets of revisions to a different ancestor
9
9
10 This extension lets you rebase changesets in an existing Mercurial
10 This extension lets you rebase changesets in an existing Mercurial
11 repository.
11 repository.
12
12
13 For more information:
13 For more information:
14 http://mercurial.selenic.com/wiki/RebaseExtension
14 http://mercurial.selenic.com/wiki/RebaseExtension
15 '''
15 '''
16
16
17 from mercurial import hg, util, repair, merge, cmdutil, commands, bookmarks
17 from mercurial import hg, util, repair, merge, cmdutil, commands, bookmarks
18 from mercurial import extensions, patch, scmutil, phases, obsolete
18 from mercurial import extensions, patch, scmutil, phases, obsolete
19 from mercurial.commands import templateopts
19 from mercurial.commands import templateopts
20 from mercurial.node import nullrev
20 from mercurial.node import nullrev
21 from mercurial.lock import release
21 from mercurial.lock import release
22 from mercurial.i18n import _
22 from mercurial.i18n import _
23 import os, errno
23 import os, errno
24
24
25 nullmerge = -2
25 nullmerge = -2
26
26
27 cmdtable = {}
27 cmdtable = {}
28 command = cmdutil.command(cmdtable)
28 command = cmdutil.command(cmdtable)
29 testedwith = 'internal'
29 testedwith = 'internal'
30
30
31 @command('rebase',
31 @command('rebase',
32 [('s', 'source', '',
32 [('s', 'source', '',
33 _('rebase from the specified changeset'), _('REV')),
33 _('rebase from the specified changeset'), _('REV')),
34 ('b', 'base', '',
34 ('b', 'base', '',
35 _('rebase from the base of the specified changeset '
35 _('rebase from the base of the specified changeset '
36 '(up to greatest common ancestor of base and dest)'),
36 '(up to greatest common ancestor of base and dest)'),
37 _('REV')),
37 _('REV')),
38 ('r', 'rev', [],
38 ('r', 'rev', [],
39 _('rebase these revisions'),
39 _('rebase these revisions'),
40 _('REV')),
40 _('REV')),
41 ('d', 'dest', '',
41 ('d', 'dest', '',
42 _('rebase onto the specified changeset'), _('REV')),
42 _('rebase onto the specified changeset'), _('REV')),
43 ('', 'collapse', False, _('collapse the rebased changesets')),
43 ('', 'collapse', False, _('collapse the rebased changesets')),
44 ('m', 'message', '',
44 ('m', 'message', '',
45 _('use text as collapse commit message'), _('TEXT')),
45 _('use text as collapse commit message'), _('TEXT')),
46 ('e', 'edit', False, _('invoke editor on commit messages')),
46 ('e', 'edit', False, _('invoke editor on commit messages')),
47 ('l', 'logfile', '',
47 ('l', 'logfile', '',
48 _('read collapse commit message from file'), _('FILE')),
48 _('read collapse commit message from file'), _('FILE')),
49 ('', 'keep', False, _('keep original changesets')),
49 ('', 'keep', False, _('keep original changesets')),
50 ('', 'keepbranches', False, _('keep original branch names')),
50 ('', 'keepbranches', False, _('keep original branch names')),
51 ('D', 'detach', False, _('(DEPRECATED)')),
51 ('D', 'detach', False, _('(DEPRECATED)')),
52 ('t', 'tool', '', _('specify merge tool')),
52 ('t', 'tool', '', _('specify merge tool')),
53 ('c', 'continue', False, _('continue an interrupted rebase')),
53 ('c', 'continue', False, _('continue an interrupted rebase')),
54 ('a', 'abort', False, _('abort an interrupted rebase'))] +
54 ('a', 'abort', False, _('abort an interrupted rebase'))] +
55 templateopts,
55 templateopts,
56 _('[-s REV | -b REV] [-d REV] [OPTION]'))
56 _('[-s REV | -b REV] [-d REV] [OPTION]'))
57 def rebase(ui, repo, **opts):
57 def rebase(ui, repo, **opts):
58 """move changeset (and descendants) to a different branch
58 """move changeset (and descendants) to a different branch
59
59
60 Rebase uses repeated merging to graft changesets from one part of
60 Rebase uses repeated merging to graft changesets from one part of
61 history (the source) onto another (the destination). This can be
61 history (the source) onto another (the destination). This can be
62 useful for linearizing *local* changes relative to a master
62 useful for linearizing *local* changes relative to a master
63 development tree.
63 development tree.
64
64
65 You should not rebase changesets that have already been shared
65 You should not rebase changesets that have already been shared
66 with others. Doing so will force everybody else to perform the
66 with others. Doing so will force everybody else to perform the
67 same rebase or they will end up with duplicated changesets after
67 same rebase or they will end up with duplicated changesets after
68 pulling in your rebased changesets.
68 pulling in your rebased changesets.
69
69
70 If you don't specify a destination changeset (``-d/--dest``),
70 If you don't specify a destination changeset (``-d/--dest``),
71 rebase uses the tipmost head of the current named branch as the
71 rebase uses the tipmost head of the current named branch as the
72 destination. (The destination changeset is not modified by
72 destination. (The destination changeset is not modified by
73 rebasing, but new changesets are added as its descendants.)
73 rebasing, but new changesets are added as its descendants.)
74
74
75 You can specify which changesets to rebase in two ways: as a
75 You can specify which changesets to rebase in two ways: as a
76 "source" changeset or as a "base" changeset. Both are shorthand
76 "source" changeset or as a "base" changeset. Both are shorthand
77 for a topologically related set of changesets (the "source
77 for a topologically related set of changesets (the "source
78 branch"). If you specify source (``-s/--source``), rebase will
78 branch"). If you specify source (``-s/--source``), rebase will
79 rebase that changeset and all of its descendants onto dest. If you
79 rebase that changeset and all of its descendants onto dest. If you
80 specify base (``-b/--base``), rebase will select ancestors of base
80 specify base (``-b/--base``), rebase will select ancestors of base
81 back to but not including the common ancestor with dest. Thus,
81 back to but not including the common ancestor with dest. Thus,
82 ``-b`` is less precise but more convenient than ``-s``: you can
82 ``-b`` is less precise but more convenient than ``-s``: you can
83 specify any changeset in the source branch, and rebase will select
83 specify any changeset in the source branch, and rebase will select
84 the whole branch. If you specify neither ``-s`` nor ``-b``, rebase
84 the whole branch. If you specify neither ``-s`` nor ``-b``, rebase
85 uses the parent of the working directory as the base.
85 uses the parent of the working directory as the base.
86
86
87 By default, rebase recreates the changesets in the source branch
87 By default, rebase recreates the changesets in the source branch
88 as descendants of dest and then destroys the originals. Use
88 as descendants of dest and then destroys the originals. Use
89 ``--keep`` to preserve the original source changesets. Some
89 ``--keep`` to preserve the original source changesets. Some
90 changesets in the source branch (e.g. merges from the destination
90 changesets in the source branch (e.g. merges from the destination
91 branch) may be dropped if they no longer contribute any change.
91 branch) may be dropped if they no longer contribute any change.
92
92
93 One result of the rules for selecting the destination changeset
93 One result of the rules for selecting the destination changeset
94 and source branch is that, unlike ``merge``, rebase will do
94 and source branch is that, unlike ``merge``, rebase will do
95 nothing if you are at the latest (tipmost) head of a named branch
95 nothing if you are at the latest (tipmost) head of a named branch
96 with two heads. You need to explicitly specify source and/or
96 with two heads. You need to explicitly specify source and/or
97 destination (or ``update`` to the other head, if it's the head of
97 destination (or ``update`` to the other head, if it's the head of
98 the intended source branch).
98 the intended source branch).
99
99
100 If a rebase is interrupted to manually resolve a merge, it can be
100 If a rebase is interrupted to manually resolve a merge, it can be
101 continued with --continue/-c or aborted with --abort/-a.
101 continued with --continue/-c or aborted with --abort/-a.
102
102
103 Returns 0 on success, 1 if nothing to rebase.
103 Returns 0 on success, 1 if nothing to rebase.
104 """
104 """
105 originalwd = target = None
105 originalwd = target = None
106 external = nullrev
106 external = nullrev
107 state = {}
107 state = {}
108 skipped = set()
108 skipped = set()
109 targetancestors = set()
109 targetancestors = set()
110
110
111 editor = None
111 editor = None
112 if opts.get('edit'):
112 if opts.get('edit'):
113 editor = cmdutil.commitforceeditor
113 editor = cmdutil.commitforceeditor
114
114
115 lock = wlock = None
115 lock = wlock = None
116 try:
116 try:
117 wlock = repo.wlock()
117 wlock = repo.wlock()
118 lock = repo.lock()
118 lock = repo.lock()
119
119
120 # Validate input and define rebasing points
120 # Validate input and define rebasing points
121 destf = opts.get('dest', None)
121 destf = opts.get('dest', None)
122 srcf = opts.get('source', None)
122 srcf = opts.get('source', None)
123 basef = opts.get('base', None)
123 basef = opts.get('base', None)
124 revf = opts.get('rev', [])
124 revf = opts.get('rev', [])
125 contf = opts.get('continue')
125 contf = opts.get('continue')
126 abortf = opts.get('abort')
126 abortf = opts.get('abort')
127 collapsef = opts.get('collapse', False)
127 collapsef = opts.get('collapse', False)
128 collapsemsg = cmdutil.logmessage(ui, opts)
128 collapsemsg = cmdutil.logmessage(ui, opts)
129 extrafn = opts.get('extrafn') # internal, used by e.g. hgsubversion
129 extrafn = opts.get('extrafn') # internal, used by e.g. hgsubversion
130 keepf = opts.get('keep', False)
130 keepf = opts.get('keep', False)
131 keepbranchesf = opts.get('keepbranches', False)
131 keepbranchesf = opts.get('keepbranches', False)
132 # keepopen is not meant for use on the command line, but by
132 # keepopen is not meant for use on the command line, but by
133 # other extensions
133 # other extensions
134 keepopen = opts.get('keepopen', False)
134 keepopen = opts.get('keepopen', False)
135
135
136 if collapsemsg and not collapsef:
136 if collapsemsg and not collapsef:
137 raise util.Abort(
137 raise util.Abort(
138 _('message can only be specified with collapse'))
138 _('message can only be specified with collapse'))
139
139
140 if contf or abortf:
140 if contf or abortf:
141 if contf and abortf:
141 if contf and abortf:
142 raise util.Abort(_('cannot use both abort and continue'))
142 raise util.Abort(_('cannot use both abort and continue'))
143 if collapsef:
143 if collapsef:
144 raise util.Abort(
144 raise util.Abort(
145 _('cannot use collapse with continue or abort'))
145 _('cannot use collapse with continue or abort'))
146 if srcf or basef or destf:
146 if srcf or basef or destf:
147 raise util.Abort(
147 raise util.Abort(
148 _('abort and continue do not allow specifying revisions'))
148 _('abort and continue do not allow specifying revisions'))
149 if opts.get('tool', False):
149 if opts.get('tool', False):
150 ui.warn(_('tool option will be ignored\n'))
150 ui.warn(_('tool option will be ignored\n'))
151
151
152 (originalwd, target, state, skipped, collapsef, keepf,
152 (originalwd, target, state, skipped, collapsef, keepf,
153 keepbranchesf, external) = restorestatus(repo)
153 keepbranchesf, external) = restorestatus(repo)
154 if abortf:
154 if abortf:
155 return abort(repo, originalwd, target, state)
155 return abort(repo, originalwd, target, state)
156 else:
156 else:
157 if srcf and basef:
157 if srcf and basef:
158 raise util.Abort(_('cannot specify both a '
158 raise util.Abort(_('cannot specify both a '
159 'source and a base'))
159 'source and a base'))
160 if revf and basef:
160 if revf and basef:
161 raise util.Abort(_('cannot specify both a '
161 raise util.Abort(_('cannot specify both a '
162 'revision and a base'))
162 'revision and a base'))
163 if revf and srcf:
163 if revf and srcf:
164 raise util.Abort(_('cannot specify both a '
164 raise util.Abort(_('cannot specify both a '
165 'revision and a source'))
165 'revision and a source'))
166
166
167 cmdutil.bailifchanged(repo)
167 cmdutil.bailifchanged(repo)
168
168
169 if not destf:
169 if not destf:
170 # Destination defaults to the latest revision in the
170 # Destination defaults to the latest revision in the
171 # current branch
171 # current branch
172 branch = repo[None].branch()
172 branch = repo[None].branch()
173 dest = repo[branch]
173 dest = repo[branch]
174 else:
174 else:
175 dest = scmutil.revsingle(repo, destf)
175 dest = scmutil.revsingle(repo, destf)
176
176
177 if revf:
177 if revf:
178 rebaseset = repo.revs('%lr', revf)
178 rebaseset = repo.revs('%lr', revf)
179 elif srcf:
179 elif srcf:
180 src = scmutil.revrange(repo, [srcf])
180 src = scmutil.revrange(repo, [srcf])
181 rebaseset = repo.revs('(%ld)::', src)
181 rebaseset = repo.revs('(%ld)::', src)
182 else:
182 else:
183 base = scmutil.revrange(repo, [basef or '.'])
183 base = scmutil.revrange(repo, [basef or '.'])
184 rebaseset = repo.revs(
184 rebaseset = repo.revs(
185 '(children(ancestor(%ld, %d)) and ::(%ld))::',
185 '(children(ancestor(%ld, %d)) and ::(%ld))::',
186 base, dest, base)
186 base, dest, base)
187 if rebaseset:
187 if rebaseset:
188 root = min(rebaseset)
188 root = min(rebaseset)
189 else:
189 else:
190 root = None
190 root = None
191
191
192 if not rebaseset:
192 if not rebaseset:
193 repo.ui.debug('base is ancestor of destination\n')
193 repo.ui.debug('base is ancestor of destination\n')
194 result = None
194 result = None
195 elif (not (keepf or obsolete._enabled)
195 elif (not (keepf or obsolete._enabled)
196 and repo.revs('first(children(%ld) - %ld)',
196 and repo.revs('first(children(%ld) - %ld)',
197 rebaseset, rebaseset)):
197 rebaseset, rebaseset)):
198 raise util.Abort(
198 raise util.Abort(
199 _("can't remove original changesets with"
199 _("can't remove original changesets with"
200 " unrebased descendants"),
200 " unrebased descendants"),
201 hint=_('use --keep to keep original changesets'))
201 hint=_('use --keep to keep original changesets'))
202 elif not keepf and not repo[root].mutable():
202 elif not keepf and not repo[root].mutable():
203 raise util.Abort(_("can't rebase immutable changeset %s")
203 raise util.Abort(_("can't rebase immutable changeset %s")
204 % repo[root],
204 % repo[root],
205 hint=_('see hg help phases for details'))
205 hint=_('see hg help phases for details'))
206 else:
206 else:
207 result = buildstate(repo, dest, rebaseset, collapsef)
207 result = buildstate(repo, dest, rebaseset, collapsef)
208
208
209 if not result:
209 if not result:
210 # Empty state built, nothing to rebase
210 # Empty state built, nothing to rebase
211 ui.status(_('nothing to rebase\n'))
211 ui.status(_('nothing to rebase\n'))
212 return 1
212 return 1
213 else:
213 else:
214 originalwd, target, state = result
214 originalwd, target, state = result
215 if collapsef:
215 if collapsef:
216 targetancestors = repo.changelog.ancestors([target],
216 targetancestors = repo.changelog.ancestors([target],
217 inclusive=True)
217 inclusive=True)
218 external = checkexternal(repo, state, targetancestors)
218 external = checkexternal(repo, state, targetancestors)
219
219
220 if keepbranchesf:
220 if keepbranchesf:
221 assert not extrafn, 'cannot use both keepbranches and extrafn'
221 assert not extrafn, 'cannot use both keepbranches and extrafn'
222 def extrafn(ctx, extra):
222 def extrafn(ctx, extra):
223 extra['branch'] = ctx.branch()
223 extra['branch'] = ctx.branch()
224 if collapsef:
224 if collapsef:
225 branches = set()
225 branches = set()
226 for rev in state:
226 for rev in state:
227 branches.add(repo[rev].branch())
227 branches.add(repo[rev].branch())
228 if len(branches) > 1:
228 if len(branches) > 1:
229 raise util.Abort(_('cannot collapse multiple named '
229 raise util.Abort(_('cannot collapse multiple named '
230 'branches'))
230 'branches'))
231
231
232
232
233 # Rebase
233 # Rebase
234 if not targetancestors:
234 if not targetancestors:
235 targetancestors = repo.changelog.ancestors([target], inclusive=True)
235 targetancestors = repo.changelog.ancestors([target], inclusive=True)
236
236
237 # Keep track of the current bookmarks in order to reset them later
237 # Keep track of the current bookmarks in order to reset them later
238 currentbookmarks = repo._bookmarks.copy()
238 currentbookmarks = repo._bookmarks.copy()
239 activebookmark = repo._bookmarkcurrent
239 activebookmark = repo._bookmarkcurrent
240 if activebookmark:
240 if activebookmark:
241 bookmarks.unsetcurrent(repo)
241 bookmarks.unsetcurrent(repo)
242
242
243 sortedstate = sorted(state)
243 sortedstate = sorted(state)
244 total = len(sortedstate)
244 total = len(sortedstate)
245 pos = 0
245 pos = 0
246 for rev in sortedstate:
246 for rev in sortedstate:
247 pos += 1
247 pos += 1
248 if state[rev] == -1:
248 if state[rev] == -1:
249 ui.progress(_("rebasing"), pos, ("%d:%s" % (rev, repo[rev])),
249 ui.progress(_("rebasing"), pos, ("%d:%s" % (rev, repo[rev])),
250 _('changesets'), total)
250 _('changesets'), total)
251 storestatus(repo, originalwd, target, state, collapsef, keepf,
251 storestatus(repo, originalwd, target, state, collapsef, keepf,
252 keepbranchesf, external)
252 keepbranchesf, external)
253 p1, p2 = defineparents(repo, rev, target, state,
253 p1, p2 = defineparents(repo, rev, target, state,
254 targetancestors)
254 targetancestors)
255 if len(repo.parents()) == 2:
255 if len(repo.parents()) == 2:
256 repo.ui.debug('resuming interrupted rebase\n')
256 repo.ui.debug('resuming interrupted rebase\n')
257 else:
257 else:
258 try:
258 try:
259 ui.setconfig('ui', 'forcemerge', opts.get('tool', ''))
259 ui.setconfig('ui', 'forcemerge', opts.get('tool', ''))
260 stats = rebasenode(repo, rev, p1, state, collapsef)
260 stats = rebasenode(repo, rev, p1, state, collapsef)
261 if stats and stats[3] > 0:
261 if stats and stats[3] > 0:
262 raise util.Abort(_('unresolved conflicts (see hg '
262 raise util.Abort(_('unresolved conflicts (see hg '
263 'resolve, then hg rebase --continue)'))
263 'resolve, then hg rebase --continue)'))
264 finally:
264 finally:
265 ui.setconfig('ui', 'forcemerge', '')
265 ui.setconfig('ui', 'forcemerge', '')
266 cmdutil.duplicatecopies(repo, rev, target)
266 cmdutil.duplicatecopies(repo, rev, target)
267 if not collapsef:
267 if not collapsef:
268 newrev = concludenode(repo, rev, p1, p2, extrafn=extrafn,
268 newrev = concludenode(repo, rev, p1, p2, extrafn=extrafn,
269 editor=editor)
269 editor=editor)
270 else:
270 else:
271 # Skip commit if we are collapsing
271 # Skip commit if we are collapsing
272 repo.setparents(repo[p1].node())
272 repo.setparents(repo[p1].node())
273 newrev = None
273 newrev = None
274 # Update the state
274 # Update the state
275 if newrev is not None:
275 if newrev is not None:
276 state[rev] = repo[newrev].rev()
276 state[rev] = repo[newrev].rev()
277 else:
277 else:
278 if not collapsef:
278 if not collapsef:
279 ui.note(_('no changes, revision %d skipped\n') % rev)
279 ui.note(_('no changes, revision %d skipped\n') % rev)
280 ui.debug('next revision set to %s\n' % p1)
280 ui.debug('next revision set to %s\n' % p1)
281 skipped.add(rev)
281 skipped.add(rev)
282 state[rev] = p1
282 state[rev] = p1
283
283
284 ui.progress(_('rebasing'), None)
284 ui.progress(_('rebasing'), None)
285 ui.note(_('rebase merging completed\n'))
285 ui.note(_('rebase merging completed\n'))
286
286
287 if collapsef and not keepopen:
287 if collapsef and not keepopen:
288 p1, p2 = defineparents(repo, min(state), target,
288 p1, p2 = defineparents(repo, min(state), target,
289 state, targetancestors)
289 state, targetancestors)
290 if collapsemsg:
290 if collapsemsg:
291 commitmsg = collapsemsg
291 commitmsg = collapsemsg
292 else:
292 else:
293 commitmsg = 'Collapsed revision'
293 commitmsg = 'Collapsed revision'
294 for rebased in state:
294 for rebased in state:
295 if rebased not in skipped and state[rebased] != nullmerge:
295 if rebased not in skipped and state[rebased] > nullmerge:
296 commitmsg += '\n* %s' % repo[rebased].description()
296 commitmsg += '\n* %s' % repo[rebased].description()
297 commitmsg = ui.edit(commitmsg, repo.ui.username())
297 commitmsg = ui.edit(commitmsg, repo.ui.username())
298 newrev = concludenode(repo, rev, p1, external, commitmsg=commitmsg,
298 newrev = concludenode(repo, rev, p1, external, commitmsg=commitmsg,
299 extrafn=extrafn, editor=editor)
299 extrafn=extrafn, editor=editor)
300
300
301 if 'qtip' in repo.tags():
301 if 'qtip' in repo.tags():
302 updatemq(repo, state, skipped, **opts)
302 updatemq(repo, state, skipped, **opts)
303
303
304 if currentbookmarks:
304 if currentbookmarks:
305 # Nodeids are needed to reset bookmarks
305 # Nodeids are needed to reset bookmarks
306 nstate = {}
306 nstate = {}
307 for k, v in state.iteritems():
307 for k, v in state.iteritems():
308 if v != nullmerge:
308 if v > nullmerge:
309 nstate[repo[k].node()] = repo[v].node()
309 nstate[repo[k].node()] = repo[v].node()
310
310
311 if not keepf:
311 if not keepf:
312 collapsedas = None
312 collapsedas = None
313 if collapsef:
313 if collapsef:
314 collapsedas = newrev
314 collapsedas = newrev
315 clearrebased(ui, repo, state, skipped, collapsedas)
315 clearrebased(ui, repo, state, skipped, collapsedas)
316
316
317 if currentbookmarks:
317 if currentbookmarks:
318 updatebookmarks(repo, nstate, currentbookmarks, **opts)
318 updatebookmarks(repo, nstate, currentbookmarks, **opts)
319
319
320 clearstatus(repo)
320 clearstatus(repo)
321 ui.note(_("rebase completed\n"))
321 ui.note(_("rebase completed\n"))
322 util.unlinkpath(repo.sjoin('undo'), ignoremissing=True)
322 util.unlinkpath(repo.sjoin('undo'), ignoremissing=True)
323 if skipped:
323 if skipped:
324 ui.note(_("%d revisions have been skipped\n") % len(skipped))
324 ui.note(_("%d revisions have been skipped\n") % len(skipped))
325
325
326 if (activebookmark and
326 if (activebookmark and
327 repo['tip'].node() == repo._bookmarks[activebookmark]):
327 repo['tip'].node() == repo._bookmarks[activebookmark]):
328 bookmarks.setcurrent(repo, activebookmark)
328 bookmarks.setcurrent(repo, activebookmark)
329
329
330 finally:
330 finally:
331 release(lock, wlock)
331 release(lock, wlock)
332
332
333 def checkexternal(repo, state, targetancestors):
333 def checkexternal(repo, state, targetancestors):
334 """Check whether one or more external revisions need to be taken in
334 """Check whether one or more external revisions need to be taken in
335 consideration. In the latter case, abort.
335 consideration. In the latter case, abort.
336 """
336 """
337 external = nullrev
337 external = nullrev
338 source = min(state)
338 source = min(state)
339 for rev in state:
339 for rev in state:
340 if rev == source:
340 if rev == source:
341 continue
341 continue
342 # Check externals and fail if there are more than one
342 # Check externals and fail if there are more than one
343 for p in repo[rev].parents():
343 for p in repo[rev].parents():
344 if (p.rev() not in state
344 if (p.rev() not in state
345 and p.rev() not in targetancestors):
345 and p.rev() not in targetancestors):
346 if external != nullrev:
346 if external != nullrev:
347 raise util.Abort(_('unable to collapse, there is more '
347 raise util.Abort(_('unable to collapse, there is more '
348 'than one external parent'))
348 'than one external parent'))
349 external = p.rev()
349 external = p.rev()
350 return external
350 return external
351
351
352 def concludenode(repo, rev, p1, p2, commitmsg=None, editor=None, extrafn=None):
352 def concludenode(repo, rev, p1, p2, commitmsg=None, editor=None, extrafn=None):
353 'Commit the changes and store useful information in extra'
353 'Commit the changes and store useful information in extra'
354 try:
354 try:
355 repo.setparents(repo[p1].node(), repo[p2].node())
355 repo.setparents(repo[p1].node(), repo[p2].node())
356 ctx = repo[rev]
356 ctx = repo[rev]
357 if commitmsg is None:
357 if commitmsg is None:
358 commitmsg = ctx.description()
358 commitmsg = ctx.description()
359 extra = {'rebase_source': ctx.hex()}
359 extra = {'rebase_source': ctx.hex()}
360 if extrafn:
360 if extrafn:
361 extrafn(ctx, extra)
361 extrafn(ctx, extra)
362 # Commit might fail if unresolved files exist
362 # Commit might fail if unresolved files exist
363 newrev = repo.commit(text=commitmsg, user=ctx.user(),
363 newrev = repo.commit(text=commitmsg, user=ctx.user(),
364 date=ctx.date(), extra=extra, editor=editor)
364 date=ctx.date(), extra=extra, editor=editor)
365 repo.dirstate.setbranch(repo[newrev].branch())
365 repo.dirstate.setbranch(repo[newrev].branch())
366 targetphase = max(ctx.phase(), phases.draft)
366 targetphase = max(ctx.phase(), phases.draft)
367 # retractboundary doesn't overwrite upper phase inherited from parent
367 # retractboundary doesn't overwrite upper phase inherited from parent
368 newnode = repo[newrev].node()
368 newnode = repo[newrev].node()
369 if newnode:
369 if newnode:
370 phases.retractboundary(repo, targetphase, [newnode])
370 phases.retractboundary(repo, targetphase, [newnode])
371 return newrev
371 return newrev
372 except util.Abort:
372 except util.Abort:
373 # Invalidate the previous setparents
373 # Invalidate the previous setparents
374 repo.dirstate.invalidate()
374 repo.dirstate.invalidate()
375 raise
375 raise
376
376
377 def rebasenode(repo, rev, p1, state, collapse):
377 def rebasenode(repo, rev, p1, state, collapse):
378 'Rebase a single revision'
378 'Rebase a single revision'
379 # Merge phase
379 # Merge phase
380 # Update to target and merge it with local
380 # Update to target and merge it with local
381 if repo['.'].rev() != repo[p1].rev():
381 if repo['.'].rev() != repo[p1].rev():
382 repo.ui.debug(" update to %d:%s\n" % (repo[p1].rev(), repo[p1]))
382 repo.ui.debug(" update to %d:%s\n" % (repo[p1].rev(), repo[p1]))
383 merge.update(repo, p1, False, True, False)
383 merge.update(repo, p1, False, True, False)
384 else:
384 else:
385 repo.ui.debug(" already in target\n")
385 repo.ui.debug(" already in target\n")
386 repo.dirstate.write()
386 repo.dirstate.write()
387 repo.ui.debug(" merge against %d:%s\n" % (repo[rev].rev(), repo[rev]))
387 repo.ui.debug(" merge against %d:%s\n" % (repo[rev].rev(), repo[rev]))
388 base = None
388 base = None
389 if repo[rev].rev() != repo[min(state)].rev():
389 if repo[rev].rev() != repo[min(state)].rev():
390 base = repo[rev].p1().node()
390 base = repo[rev].p1().node()
391 # When collapsing in-place, the parent is the common ancestor, we
391 # When collapsing in-place, the parent is the common ancestor, we
392 # have to allow merging with it.
392 # have to allow merging with it.
393 return merge.update(repo, rev, True, True, False, base, collapse)
393 return merge.update(repo, rev, True, True, False, base, collapse)
394
394
395 def defineparents(repo, rev, target, state, targetancestors):
395 def defineparents(repo, rev, target, state, targetancestors):
396 'Return the new parent relationship of the revision that will be rebased'
396 'Return the new parent relationship of the revision that will be rebased'
397 parents = repo[rev].parents()
397 parents = repo[rev].parents()
398 p1 = p2 = nullrev
398 p1 = p2 = nullrev
399
399
400 P1n = parents[0].rev()
400 P1n = parents[0].rev()
401 if P1n in targetancestors:
401 if P1n in targetancestors:
402 p1 = target
402 p1 = target
403 elif P1n in state:
403 elif P1n in state:
404 if state[P1n] == nullmerge:
404 if state[P1n] == nullmerge:
405 p1 = target
405 p1 = target
406 else:
406 else:
407 p1 = state[P1n]
407 p1 = state[P1n]
408 else: # P1n external
408 else: # P1n external
409 p1 = target
409 p1 = target
410 p2 = P1n
410 p2 = P1n
411
411
412 if len(parents) == 2 and parents[1].rev() not in targetancestors:
412 if len(parents) == 2 and parents[1].rev() not in targetancestors:
413 P2n = parents[1].rev()
413 P2n = parents[1].rev()
414 # interesting second parent
414 # interesting second parent
415 if P2n in state:
415 if P2n in state:
416 if p1 == target: # P1n in targetancestors or external
416 if p1 == target: # P1n in targetancestors or external
417 p1 = state[P2n]
417 p1 = state[P2n]
418 else:
418 else:
419 p2 = state[P2n]
419 p2 = state[P2n]
420 else: # P2n external
420 else: # P2n external
421 if p2 != nullrev: # P1n external too => rev is a merged revision
421 if p2 != nullrev: # P1n external too => rev is a merged revision
422 raise util.Abort(_('cannot use revision %d as base, result '
422 raise util.Abort(_('cannot use revision %d as base, result '
423 'would have 3 parents') % rev)
423 'would have 3 parents') % rev)
424 p2 = P2n
424 p2 = P2n
425 repo.ui.debug(" future parents are %d and %d\n" %
425 repo.ui.debug(" future parents are %d and %d\n" %
426 (repo[p1].rev(), repo[p2].rev()))
426 (repo[p1].rev(), repo[p2].rev()))
427 return p1, p2
427 return p1, p2
428
428
429 def isagitpatch(repo, patchname):
429 def isagitpatch(repo, patchname):
430 'Return true if the given patch is in git format'
430 'Return true if the given patch is in git format'
431 mqpatch = os.path.join(repo.mq.path, patchname)
431 mqpatch = os.path.join(repo.mq.path, patchname)
432 for line in patch.linereader(file(mqpatch, 'rb')):
432 for line in patch.linereader(file(mqpatch, 'rb')):
433 if line.startswith('diff --git'):
433 if line.startswith('diff --git'):
434 return True
434 return True
435 return False
435 return False
436
436
437 def updatemq(repo, state, skipped, **opts):
437 def updatemq(repo, state, skipped, **opts):
438 'Update rebased mq patches - finalize and then import them'
438 'Update rebased mq patches - finalize and then import them'
439 mqrebase = {}
439 mqrebase = {}
440 mq = repo.mq
440 mq = repo.mq
441 original_series = mq.fullseries[:]
441 original_series = mq.fullseries[:]
442 skippedpatches = set()
442 skippedpatches = set()
443
443
444 for p in mq.applied:
444 for p in mq.applied:
445 rev = repo[p.node].rev()
445 rev = repo[p.node].rev()
446 if rev in state:
446 if rev in state:
447 repo.ui.debug('revision %d is an mq patch (%s), finalize it.\n' %
447 repo.ui.debug('revision %d is an mq patch (%s), finalize it.\n' %
448 (rev, p.name))
448 (rev, p.name))
449 mqrebase[rev] = (p.name, isagitpatch(repo, p.name))
449 mqrebase[rev] = (p.name, isagitpatch(repo, p.name))
450 else:
450 else:
451 # Applied but not rebased, not sure this should happen
451 # Applied but not rebased, not sure this should happen
452 skippedpatches.add(p.name)
452 skippedpatches.add(p.name)
453
453
454 if mqrebase:
454 if mqrebase:
455 mq.finish(repo, mqrebase.keys())
455 mq.finish(repo, mqrebase.keys())
456
456
457 # We must start import from the newest revision
457 # We must start import from the newest revision
458 for rev in sorted(mqrebase, reverse=True):
458 for rev in sorted(mqrebase, reverse=True):
459 if rev not in skipped:
459 if rev not in skipped:
460 name, isgit = mqrebase[rev]
460 name, isgit = mqrebase[rev]
461 repo.ui.debug('import mq patch %d (%s)\n' % (state[rev], name))
461 repo.ui.debug('import mq patch %d (%s)\n' % (state[rev], name))
462 mq.qimport(repo, (), patchname=name, git=isgit,
462 mq.qimport(repo, (), patchname=name, git=isgit,
463 rev=[str(state[rev])])
463 rev=[str(state[rev])])
464 else:
464 else:
465 # Rebased and skipped
465 # Rebased and skipped
466 skippedpatches.add(mqrebase[rev][0])
466 skippedpatches.add(mqrebase[rev][0])
467
467
468 # Patches were either applied and rebased and imported in
468 # Patches were either applied and rebased and imported in
469 # order, applied and removed or unapplied. Discard the removed
469 # order, applied and removed or unapplied. Discard the removed
470 # ones while preserving the original series order and guards.
470 # ones while preserving the original series order and guards.
471 newseries = [s for s in original_series
471 newseries = [s for s in original_series
472 if mq.guard_re.split(s, 1)[0] not in skippedpatches]
472 if mq.guard_re.split(s, 1)[0] not in skippedpatches]
473 mq.fullseries[:] = newseries
473 mq.fullseries[:] = newseries
474 mq.seriesdirty = True
474 mq.seriesdirty = True
475 mq.savedirty()
475 mq.savedirty()
476
476
477 def updatebookmarks(repo, nstate, originalbookmarks, **opts):
477 def updatebookmarks(repo, nstate, originalbookmarks, **opts):
478 'Move bookmarks to their correct changesets'
478 'Move bookmarks to their correct changesets'
479 marks = repo._bookmarks
479 marks = repo._bookmarks
480 for k, v in originalbookmarks.iteritems():
480 for k, v in originalbookmarks.iteritems():
481 if v in nstate:
481 if v in nstate:
482 if nstate[v] != nullmerge:
482 if nstate[v] > nullmerge:
483 # update the bookmarks for revs that have moved
483 # update the bookmarks for revs that have moved
484 marks[k] = nstate[v]
484 marks[k] = nstate[v]
485
485
486 marks.write()
486 marks.write()
487
487
488 def storestatus(repo, originalwd, target, state, collapse, keep, keepbranches,
488 def storestatus(repo, originalwd, target, state, collapse, keep, keepbranches,
489 external):
489 external):
490 'Store the current status to allow recovery'
490 'Store the current status to allow recovery'
491 f = repo.opener("rebasestate", "w")
491 f = repo.opener("rebasestate", "w")
492 f.write(repo[originalwd].hex() + '\n')
492 f.write(repo[originalwd].hex() + '\n')
493 f.write(repo[target].hex() + '\n')
493 f.write(repo[target].hex() + '\n')
494 f.write(repo[external].hex() + '\n')
494 f.write(repo[external].hex() + '\n')
495 f.write('%d\n' % int(collapse))
495 f.write('%d\n' % int(collapse))
496 f.write('%d\n' % int(keep))
496 f.write('%d\n' % int(keep))
497 f.write('%d\n' % int(keepbranches))
497 f.write('%d\n' % int(keepbranches))
498 for d, v in state.iteritems():
498 for d, v in state.iteritems():
499 oldrev = repo[d].hex()
499 oldrev = repo[d].hex()
500 if v != nullmerge:
500 if v > nullmerge:
501 newrev = repo[v].hex()
501 newrev = repo[v].hex()
502 else:
502 else:
503 newrev = v
503 newrev = v
504 f.write("%s:%s\n" % (oldrev, newrev))
504 f.write("%s:%s\n" % (oldrev, newrev))
505 f.close()
505 f.close()
506 repo.ui.debug('rebase status stored\n')
506 repo.ui.debug('rebase status stored\n')
507
507
508 def clearstatus(repo):
508 def clearstatus(repo):
509 'Remove the status files'
509 'Remove the status files'
510 util.unlinkpath(repo.join("rebasestate"), ignoremissing=True)
510 util.unlinkpath(repo.join("rebasestate"), ignoremissing=True)
511
511
512 def restorestatus(repo):
512 def restorestatus(repo):
513 'Restore a previously stored status'
513 'Restore a previously stored status'
514 try:
514 try:
515 target = None
515 target = None
516 collapse = False
516 collapse = False
517 external = nullrev
517 external = nullrev
518 state = {}
518 state = {}
519 f = repo.opener("rebasestate")
519 f = repo.opener("rebasestate")
520 for i, l in enumerate(f.read().splitlines()):
520 for i, l in enumerate(f.read().splitlines()):
521 if i == 0:
521 if i == 0:
522 originalwd = repo[l].rev()
522 originalwd = repo[l].rev()
523 elif i == 1:
523 elif i == 1:
524 target = repo[l].rev()
524 target = repo[l].rev()
525 elif i == 2:
525 elif i == 2:
526 external = repo[l].rev()
526 external = repo[l].rev()
527 elif i == 3:
527 elif i == 3:
528 collapse = bool(int(l))
528 collapse = bool(int(l))
529 elif i == 4:
529 elif i == 4:
530 keep = bool(int(l))
530 keep = bool(int(l))
531 elif i == 5:
531 elif i == 5:
532 keepbranches = bool(int(l))
532 keepbranches = bool(int(l))
533 else:
533 else:
534 oldrev, newrev = l.split(':')
534 oldrev, newrev = l.split(':')
535 if newrev != str(nullmerge):
535 if newrev != str(nullmerge):
536 state[repo[oldrev].rev()] = repo[newrev].rev()
536 state[repo[oldrev].rev()] = repo[newrev].rev()
537 else:
537 else:
538 state[repo[oldrev].rev()] = int(newrev)
538 state[repo[oldrev].rev()] = int(newrev)
539 skipped = set()
539 skipped = set()
540 # recompute the set of skipped revs
540 # recompute the set of skipped revs
541 if not collapse:
541 if not collapse:
542 seen = set([target])
542 seen = set([target])
543 for old, new in sorted(state.items()):
543 for old, new in sorted(state.items()):
544 if new != nullrev and new in seen:
544 if new != nullrev and new in seen:
545 skipped.add(old)
545 skipped.add(old)
546 seen.add(new)
546 seen.add(new)
547 repo.ui.debug('computed skipped revs: %s\n' % skipped)
547 repo.ui.debug('computed skipped revs: %s\n' % skipped)
548 repo.ui.debug('rebase status resumed\n')
548 repo.ui.debug('rebase status resumed\n')
549 return (originalwd, target, state, skipped,
549 return (originalwd, target, state, skipped,
550 collapse, keep, keepbranches, external)
550 collapse, keep, keepbranches, external)
551 except IOError, err:
551 except IOError, err:
552 if err.errno != errno.ENOENT:
552 if err.errno != errno.ENOENT:
553 raise
553 raise
554 raise util.Abort(_('no rebase in progress'))
554 raise util.Abort(_('no rebase in progress'))
555
555
556 def abort(repo, originalwd, target, state):
556 def abort(repo, originalwd, target, state):
557 'Restore the repository to its original state'
557 'Restore the repository to its original state'
558 dstates = [s for s in state.values() if s != nullrev]
558 dstates = [s for s in state.values() if s != nullrev]
559 immutable = [d for d in dstates if not repo[d].mutable()]
559 immutable = [d for d in dstates if not repo[d].mutable()]
560 if immutable:
560 if immutable:
561 raise util.Abort(_("can't abort rebase due to immutable changesets %s")
561 raise util.Abort(_("can't abort rebase due to immutable changesets %s")
562 % ', '.join(str(repo[r]) for r in immutable),
562 % ', '.join(str(repo[r]) for r in immutable),
563 hint=_('see hg help phases for details'))
563 hint=_('see hg help phases for details'))
564
564
565 descendants = set()
565 descendants = set()
566 if dstates:
566 if dstates:
567 descendants = set(repo.changelog.descendants(dstates))
567 descendants = set(repo.changelog.descendants(dstates))
568 if descendants - set(dstates):
568 if descendants - set(dstates):
569 repo.ui.warn(_("warning: new changesets detected on target branch, "
569 repo.ui.warn(_("warning: new changesets detected on target branch, "
570 "can't abort\n"))
570 "can't abort\n"))
571 return -1
571 return -1
572 else:
572 else:
573 # Strip from the first rebased revision
573 # Strip from the first rebased revision
574 merge.update(repo, repo[originalwd].rev(), False, True, False)
574 merge.update(repo, repo[originalwd].rev(), False, True, False)
575 rebased = filter(lambda x: x > -1 and x != target, state.values())
575 rebased = filter(lambda x: x > -1 and x != target, state.values())
576 if rebased:
576 if rebased:
577 strippoints = [c.node() for c in repo.set('roots(%ld)', rebased)]
577 strippoints = [c.node() for c in repo.set('roots(%ld)', rebased)]
578 # no backup of rebased cset versions needed
578 # no backup of rebased cset versions needed
579 repair.strip(repo.ui, repo, strippoints)
579 repair.strip(repo.ui, repo, strippoints)
580 clearstatus(repo)
580 clearstatus(repo)
581 repo.ui.warn(_('rebase aborted\n'))
581 repo.ui.warn(_('rebase aborted\n'))
582 return 0
582 return 0
583
583
584 def buildstate(repo, dest, rebaseset, collapse):
584 def buildstate(repo, dest, rebaseset, collapse):
585 '''Define which revisions are going to be rebased and where
585 '''Define which revisions are going to be rebased and where
586
586
587 repo: repo
587 repo: repo
588 dest: context
588 dest: context
589 rebaseset: set of rev
589 rebaseset: set of rev
590 '''
590 '''
591
591
592 # This check isn't strictly necessary, since mq detects commits over an
592 # This check isn't strictly necessary, since mq detects commits over an
593 # applied patch. But it prevents messing up the working directory when
593 # applied patch. But it prevents messing up the working directory when
594 # a partially completed rebase is blocked by mq.
594 # a partially completed rebase is blocked by mq.
595 if 'qtip' in repo.tags() and (dest.node() in
595 if 'qtip' in repo.tags() and (dest.node() in
596 [s.node for s in repo.mq.applied]):
596 [s.node for s in repo.mq.applied]):
597 raise util.Abort(_('cannot rebase onto an applied mq patch'))
597 raise util.Abort(_('cannot rebase onto an applied mq patch'))
598
598
599 roots = list(repo.set('roots(%ld)', rebaseset))
599 roots = list(repo.set('roots(%ld)', rebaseset))
600 if not roots:
600 if not roots:
601 raise util.Abort(_('no matching revisions'))
601 raise util.Abort(_('no matching revisions'))
602 roots.sort()
602 roots.sort()
603 state = {}
603 state = {}
604 detachset = set()
604 detachset = set()
605 for root in roots:
605 for root in roots:
606 commonbase = root.ancestor(dest)
606 commonbase = root.ancestor(dest)
607 if commonbase == root:
607 if commonbase == root:
608 raise util.Abort(_('source is ancestor of destination'))
608 raise util.Abort(_('source is ancestor of destination'))
609 if commonbase == dest:
609 if commonbase == dest:
610 samebranch = root.branch() == dest.branch()
610 samebranch = root.branch() == dest.branch()
611 if not collapse and samebranch and root in dest.children():
611 if not collapse and samebranch and root in dest.children():
612 repo.ui.debug('source is a child of destination\n')
612 repo.ui.debug('source is a child of destination\n')
613 return None
613 return None
614
614
615 repo.ui.debug('rebase onto %d starting from %s\n' % (dest, roots))
615 repo.ui.debug('rebase onto %d starting from %s\n' % (dest, roots))
616 state.update(dict.fromkeys(rebaseset, nullrev))
616 state.update(dict.fromkeys(rebaseset, nullrev))
617 # Rebase tries to turn <dest> into a parent of <root> while
617 # Rebase tries to turn <dest> into a parent of <root> while
618 # preserving the number of parents of rebased changesets:
618 # preserving the number of parents of rebased changesets:
619 #
619 #
620 # - A changeset with a single parent will always be rebased as a
620 # - A changeset with a single parent will always be rebased as a
621 # changeset with a single parent.
621 # changeset with a single parent.
622 #
622 #
623 # - A merge will be rebased as merge unless its parents are both
623 # - A merge will be rebased as merge unless its parents are both
624 # ancestors of <dest> or are themselves in the rebased set and
624 # ancestors of <dest> or are themselves in the rebased set and
625 # pruned while rebased.
625 # pruned while rebased.
626 #
626 #
627 # If one parent of <root> is an ancestor of <dest>, the rebased
627 # If one parent of <root> is an ancestor of <dest>, the rebased
628 # version of this parent will be <dest>. This is always true with
628 # version of this parent will be <dest>. This is always true with
629 # --base option.
629 # --base option.
630 #
630 #
631 # Otherwise, we need to *replace* the original parents with
631 # Otherwise, we need to *replace* the original parents with
632 # <dest>. This "detaches" the rebased set from its former location
632 # <dest>. This "detaches" the rebased set from its former location
633 # and rebases it onto <dest>. Changes introduced by ancestors of
633 # and rebases it onto <dest>. Changes introduced by ancestors of
634 # <root> not common with <dest> (the detachset, marked as
634 # <root> not common with <dest> (the detachset, marked as
635 # nullmerge) are "removed" from the rebased changesets.
635 # nullmerge) are "removed" from the rebased changesets.
636 #
636 #
637 # - If <root> has a single parent, set it to <dest>.
637 # - If <root> has a single parent, set it to <dest>.
638 #
638 #
639 # - If <root> is a merge, we cannot decide which parent to
639 # - If <root> is a merge, we cannot decide which parent to
640 # replace, the rebase operation is not clearly defined.
640 # replace, the rebase operation is not clearly defined.
641 #
641 #
642 # The table below sums up this behavior:
642 # The table below sums up this behavior:
643 #
643 #
644 # +------------------+----------------------+-------------------------+
644 # +------------------+----------------------+-------------------------+
645 # | | one parent | merge |
645 # | | one parent | merge |
646 # +------------------+----------------------+-------------------------+
646 # +------------------+----------------------+-------------------------+
647 # | parent in | new parent is <dest> | parents in ::<dest> are |
647 # | parent in | new parent is <dest> | parents in ::<dest> are |
648 # | ::<dest> | | remapped to <dest> |
648 # | ::<dest> | | remapped to <dest> |
649 # +------------------+----------------------+-------------------------+
649 # +------------------+----------------------+-------------------------+
650 # | unrelated source | new parent is <dest> | ambiguous, abort |
650 # | unrelated source | new parent is <dest> | ambiguous, abort |
651 # +------------------+----------------------+-------------------------+
651 # +------------------+----------------------+-------------------------+
652 #
652 #
653 # The actual abort is handled by `defineparents`
653 # The actual abort is handled by `defineparents`
654 if len(root.parents()) <= 1:
654 if len(root.parents()) <= 1:
655 # ancestors of <root> not ancestors of <dest>
655 # ancestors of <root> not ancestors of <dest>
656 detachset.update(repo.changelog.findmissingrevs([commonbase.rev()],
656 detachset.update(repo.changelog.findmissingrevs([commonbase.rev()],
657 [root.rev()]))
657 [root.rev()]))
658 for r in detachset:
658 for r in detachset:
659 if r not in state:
659 if r not in state:
660 state[r] = nullmerge
660 state[r] = nullmerge
661 return repo['.'].rev(), dest.rev(), state
661 return repo['.'].rev(), dest.rev(), state
662
662
663 def clearrebased(ui, repo, state, skipped, collapsedas=None):
663 def clearrebased(ui, repo, state, skipped, collapsedas=None):
664 """dispose of rebased revision at the end of the rebase
664 """dispose of rebased revision at the end of the rebase
665
665
666 If `collapsedas` is not None, the rebase was a collapse whose result if the
666 If `collapsedas` is not None, the rebase was a collapse whose result if the
667 `collapsedas` node."""
667 `collapsedas` node."""
668 if obsolete._enabled:
668 if obsolete._enabled:
669 markers = []
669 markers = []
670 for rev, newrev in sorted(state.items()):
670 for rev, newrev in sorted(state.items()):
671 if newrev >= 0:
671 if newrev >= 0:
672 if rev in skipped:
672 if rev in skipped:
673 succs = ()
673 succs = ()
674 elif collapsedas is not None:
674 elif collapsedas is not None:
675 succs = (repo[collapsedas],)
675 succs = (repo[collapsedas],)
676 else:
676 else:
677 succs = (repo[newrev],)
677 succs = (repo[newrev],)
678 markers.append((repo[rev], succs))
678 markers.append((repo[rev], succs))
679 if markers:
679 if markers:
680 obsolete.createmarkers(repo, markers)
680 obsolete.createmarkers(repo, markers)
681 else:
681 else:
682 rebased = [rev for rev in state if state[rev] != nullmerge]
682 rebased = [rev for rev in state if state[rev] > nullmerge]
683 if rebased:
683 if rebased:
684 stripped = []
684 stripped = []
685 for root in repo.set('roots(%ld)', rebased):
685 for root in repo.set('roots(%ld)', rebased):
686 if set(repo.changelog.descendants([root.rev()])) - set(state):
686 if set(repo.changelog.descendants([root.rev()])) - set(state):
687 ui.warn(_("warning: new changesets detected "
687 ui.warn(_("warning: new changesets detected "
688 "on source branch, not stripping\n"))
688 "on source branch, not stripping\n"))
689 else:
689 else:
690 stripped.append(root.node())
690 stripped.append(root.node())
691 if stripped:
691 if stripped:
692 # backup the old csets by default
692 # backup the old csets by default
693 repair.strip(ui, repo, stripped, "all")
693 repair.strip(ui, repo, stripped, "all")
694
694
695
695
696 def pullrebase(orig, ui, repo, *args, **opts):
696 def pullrebase(orig, ui, repo, *args, **opts):
697 'Call rebase after pull if the latter has been invoked with --rebase'
697 'Call rebase after pull if the latter has been invoked with --rebase'
698 if opts.get('rebase'):
698 if opts.get('rebase'):
699 if opts.get('update'):
699 if opts.get('update'):
700 del opts['update']
700 del opts['update']
701 ui.debug('--update and --rebase are not compatible, ignoring '
701 ui.debug('--update and --rebase are not compatible, ignoring '
702 'the update flag\n')
702 'the update flag\n')
703
703
704 movemarkfrom = repo['.'].node()
704 movemarkfrom = repo['.'].node()
705 cmdutil.bailifchanged(repo)
705 cmdutil.bailifchanged(repo)
706 revsprepull = len(repo)
706 revsprepull = len(repo)
707 origpostincoming = commands.postincoming
707 origpostincoming = commands.postincoming
708 def _dummy(*args, **kwargs):
708 def _dummy(*args, **kwargs):
709 pass
709 pass
710 commands.postincoming = _dummy
710 commands.postincoming = _dummy
711 try:
711 try:
712 orig(ui, repo, *args, **opts)
712 orig(ui, repo, *args, **opts)
713 finally:
713 finally:
714 commands.postincoming = origpostincoming
714 commands.postincoming = origpostincoming
715 revspostpull = len(repo)
715 revspostpull = len(repo)
716 if revspostpull > revsprepull:
716 if revspostpull > revsprepull:
717 # --rev option from pull conflict with rebase own --rev
717 # --rev option from pull conflict with rebase own --rev
718 # dropping it
718 # dropping it
719 if 'rev' in opts:
719 if 'rev' in opts:
720 del opts['rev']
720 del opts['rev']
721 rebase(ui, repo, **opts)
721 rebase(ui, repo, **opts)
722 branch = repo[None].branch()
722 branch = repo[None].branch()
723 dest = repo[branch].rev()
723 dest = repo[branch].rev()
724 if dest != repo['.'].rev():
724 if dest != repo['.'].rev():
725 # there was nothing to rebase we force an update
725 # there was nothing to rebase we force an update
726 hg.update(repo, dest)
726 hg.update(repo, dest)
727 if bookmarks.update(repo, [movemarkfrom], repo['.'].node()):
727 if bookmarks.update(repo, [movemarkfrom], repo['.'].node()):
728 ui.status(_("updating bookmark %s\n")
728 ui.status(_("updating bookmark %s\n")
729 % repo._bookmarkcurrent)
729 % repo._bookmarkcurrent)
730 else:
730 else:
731 if opts.get('tool'):
731 if opts.get('tool'):
732 raise util.Abort(_('--tool can only be used with --rebase'))
732 raise util.Abort(_('--tool can only be used with --rebase'))
733 orig(ui, repo, *args, **opts)
733 orig(ui, repo, *args, **opts)
734
734
735 def uisetup(ui):
735 def uisetup(ui):
736 'Replace pull with a decorator to provide --rebase option'
736 'Replace pull with a decorator to provide --rebase option'
737 entry = extensions.wrapcommand(commands.table, 'pull', pullrebase)
737 entry = extensions.wrapcommand(commands.table, 'pull', pullrebase)
738 entry[1].append(('', 'rebase', None,
738 entry[1].append(('', 'rebase', None,
739 _("rebase working directory to branch head")))
739 _("rebase working directory to branch head")))
740 entry[1].append(('t', 'tool', '',
740 entry[1].append(('t', 'tool', '',
741 _("specify merge tool for rebase")))
741 _("specify merge tool for rebase")))
General Comments 0
You need to be logged in to leave comments. Login now