fetch.py
128 lines
| 5.2 KiB
| text/x-python
|
PythonLexer
/ hgext / fetch.py
Vadim Gelfer
|
r2800 | # fetch.py - pull and merge remote changes | ||
# | ||||
# Copyright 2006 Vadim Gelfer <vadim.gelfer@gmail.com> | ||||
# | ||||
# This software may be used and distributed according to the terms | ||||
# of the GNU General Public License, incorporated herein by reference. | ||||
Dirkjan Ochtman
|
r6666 | '''pulling, updating and merging in one command''' | ||
Vadim Gelfer
|
r2800 | |||
Matt Mackall
|
r3891 | from mercurial.i18n import _ | ||
Joel Rosdahl
|
r6211 | from mercurial.node import nullid, short | ||
Joel Rosdahl
|
r6212 | from mercurial import commands, cmdutil, hg, util | ||
Vadim Gelfer
|
r2800 | |||
def fetch(ui, repo, source='default', **opts): | ||||
'''Pull changes from a remote repository, merge new changes if needed. | ||||
This finds all changes from the repository at the specified path | ||||
or URL and adds them to the local repository. | ||||
If the pulled changes add a new head, the head is automatically | ||||
merged, and the result of the merge is committed. Otherwise, the | ||||
Bryan O'Sullivan
|
r6206 | working directory is updated to include the new changes. | ||
When a merge occurs, the newly pulled changes are assumed to be | ||||
"authoritative". The head of the new changes is used as the first | ||||
parent, with local changes as the second. To switch the merge | ||||
order, use --switch-parent. | ||||
Thomas Arendsen Hein
|
r6163 | |||
See 'hg help dates' for a list of formats valid for -d/--date. | ||||
''' | ||||
Vadim Gelfer
|
r2800 | |||
Matt Mackall
|
r4917 | def postincoming(other, modheads): | ||
Vadim Gelfer
|
r2800 | if modheads == 0: | ||
return 0 | ||||
if modheads == 1: | ||||
Matt Mackall
|
r4917 | return hg.clean(repo, repo.changelog.tip()) | ||
Vadim Gelfer
|
r2800 | newheads = repo.heads(parent) | ||
newchildren = [n for n in repo.heads(parent) if n != parent] | ||||
newparent = parent | ||||
if newchildren: | ||||
newparent = newchildren[0] | ||||
Matt Mackall
|
r4917 | hg.clean(repo, newparent) | ||
Vadim Gelfer
|
r2800 | newheads = [n for n in repo.heads() if n != newparent] | ||
Bryan O'Sullivan
|
r6206 | if len(newheads) > 1: | ||
Vadim Gelfer
|
r2800 | ui.status(_('not merging with %d other new heads ' | ||
'(use "hg heads" and "hg merge" to merge them)') % | ||||
(len(newheads) - 1)) | ||||
Bryan O'Sullivan
|
r6206 | return | ||
err = False | ||||
if newheads: | ||||
# By default, we consider the repository we're pulling | ||||
# *from* as authoritative, so we merge our changes into | ||||
# theirs. | ||||
if opts['switch_parent']: | ||||
firstparent, secondparent = newparent, newheads[0] | ||||
else: | ||||
firstparent, secondparent = newheads[0], newparent | ||||
ui.status(_('updating to %d:%s\n') % | ||||
(repo.changelog.rev(firstparent), | ||||
short(firstparent))) | ||||
hg.clean(repo, firstparent) | ||||
ui.status(_('merging with %d:%s\n') % | ||||
(repo.changelog.rev(secondparent), short(secondparent))) | ||||
err = hg.merge(repo, secondparent, remind=False) | ||||
Vadim Gelfer
|
r2800 | if not err: | ||
Matt Mackall
|
r4917 | mod, add, rem = repo.status()[:3] | ||
Matt Mackall
|
r4549 | message = (cmdutil.logmessage(opts) or | ||
Bryan O'Sullivan
|
r5798 | (_('Automated merge with %s') % | ||
util.removeauth(other.url()))) | ||||
Bryan O'Sullivan
|
r6225 | force_editor = opts.get('force_editor') or opts.get('edit') | ||
Vadim Gelfer
|
r2800 | n = repo.commit(mod + add + rem, message, | ||
Bryan O'Sullivan
|
r6385 | opts['user'], opts['date'], force=True, | ||
Bryan O'Sullivan
|
r6225 | force_editor=force_editor) | ||
Vadim Gelfer
|
r2800 | ui.status(_('new changeset %d:%s merges remote changes ' | ||
'with local\n') % (repo.changelog.rev(n), | ||||
short(n))) | ||||
Bryan O'Sullivan
|
r6206 | |||
Matt Mackall
|
r4917 | def pull(): | ||
Matt Mackall
|
r4549 | cmdutil.setremoteconfig(ui, opts) | ||
Vadim Gelfer
|
r2800 | |||
other = hg.repository(ui, ui.expandpath(source)) | ||||
Bryan O'Sullivan
|
r5798 | ui.status(_('pulling from %s\n') % | ||
util.hidepassword(ui.expandpath(source))) | ||||
Vadim Gelfer
|
r2800 | revs = None | ||
Bryan O'Sullivan
|
r6207 | if opts['rev']: | ||
if not other.local(): | ||||
raise util.Abort(_("fetch -r doesn't work for remote " | ||||
"repositories yet")) | ||||
else: | ||||
revs = [other.lookup(rev) for rev in opts['rev']] | ||||
Matt Mackall
|
r4917 | modheads = repo.pull(other, heads=revs) | ||
return postincoming(other, modheads) | ||||
Thomas Arendsen Hein
|
r3223 | |||
Thomas Arendsen Hein
|
r6139 | date = opts.get('date') | ||
if date: | ||||
opts['date'] = util.parsedate(date) | ||||
Vadim Gelfer
|
r2800 | parent, p2 = repo.dirstate.parents() | ||
if parent != repo.changelog.tip(): | ||||
raise util.Abort(_('working dir not at tip ' | ||||
'(use "hg update" to check out tip)')) | ||||
if p2 != nullid: | ||||
raise util.Abort(_('outstanding uncommitted merge')) | ||||
Matt Mackall
|
r4915 | wlock = lock = None | ||
Vadim Gelfer
|
r2825 | try: | ||
Matt Mackall
|
r4915 | wlock = repo.wlock() | ||
lock = repo.lock() | ||||
Bryan O'Sullivan
|
r6226 | mod, add, rem, del_ = repo.status()[:4] | ||
Vadim Gelfer
|
r2827 | if mod or add or rem: | ||
raise util.Abort(_('outstanding uncommitted changes')) | ||||
Bryan O'Sullivan
|
r6226 | if del_: | ||
raise util.Abort(_('working directory is missing some files')) | ||||
Vadim Gelfer
|
r2827 | if len(repo.heads()) > 1: | ||
raise util.Abort(_('multiple heads in this repository ' | ||||
'(use "hg heads" and "hg merge" to merge)')) | ||||
Matt Mackall
|
r4917 | return pull() | ||
Vadim Gelfer
|
r2825 | finally: | ||
Matt Mackall
|
r4915 | del lock, wlock | ||
Vadim Gelfer
|
r2800 | |||
cmdtable = { | ||||
'fetch': | ||||
Thomas Arendsen Hein
|
r4730 | (fetch, | ||
Benoit Boissinot
|
r5147 | [('r', 'rev', [], _('a specific revision you would like to pull')), | ||
Bryan O'Sullivan
|
r6225 | ('e', 'edit', None, _('edit commit message')), | ||
('', 'force-editor', None, _('edit commit message (DEPRECATED)')), | ||||
Bryan O'Sullivan
|
r6206 | ('', 'switch-parent', None, _('switch parents when merging')), | ||
Benoit Boissinot
|
r5147 | ] + commands.commitopts + commands.commitopts2 + commands.remoteopts, | ||
Thomas Arendsen Hein
|
r4730 | _('hg fetch [SOURCE]')), | ||
} | ||||