##// END OF EJS Templates
fetch: wrap docstrings at 70 characters
Martin Geisler -
r9258:1aeb2249 default
parent child Browse files
Show More
@@ -1,147 +1,148
1 # fetch.py - pull and merge remote changes
1 # fetch.py - pull and merge remote changes
2 #
2 #
3 # Copyright 2006 Vadim Gelfer <vadim.gelfer@gmail.com>
3 # Copyright 2006 Vadim Gelfer <vadim.gelfer@gmail.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, incorporated herein by reference.
6 # GNU General Public License version 2, incorporated herein by reference.
7
7
8 '''pull, update and merge in one command'''
8 '''pull, update and merge in one command'''
9
9
10 from mercurial.i18n import _
10 from mercurial.i18n import _
11 from mercurial.node import nullid, short
11 from mercurial.node import nullid, short
12 from mercurial import commands, cmdutil, hg, util, url, error
12 from mercurial import commands, cmdutil, hg, util, url, error
13 from mercurial.lock import release
13 from mercurial.lock import release
14
14
15 def fetch(ui, repo, source='default', **opts):
15 def fetch(ui, repo, source='default', **opts):
16 '''pull changes from a remote repository, merge new changes if needed.
16 '''pull changes from a remote repository, merge new changes if needed.
17
17
18 This finds all changes from the repository at the specified path or URL
18 This finds all changes from the repository at the specified path
19 and adds them to the local repository.
19 or URL and adds them to the local repository.
20
20
21 If the pulled changes add a new branch head, the head is automatically
21 If the pulled changes add a new branch head, the head is
22 merged, and the result of the merge is committed. Otherwise, the working
22 automatically merged, and the result of the merge is committed.
23 directory is updated to include the new changes.
23 Otherwise, the working directory is updated to include the new
24 changes.
24
25
25 When a merge occurs, the newly pulled changes are assumed to be
26 When a merge occurs, the newly pulled changes are assumed to be
26 "authoritative". The head of the new changes is used as the first parent,
27 "authoritative". The head of the new changes is used as the first
27 with local changes as the second. To switch the merge order, use
28 parent, with local changes as the second. To switch the merge
28 --switch-parent.
29 order, use --switch-parent.
29
30
30 See 'hg help dates' for a list of formats valid for -d/--date.
31 See 'hg help dates' for a list of formats valid for -d/--date.
31 '''
32 '''
32
33
33 date = opts.get('date')
34 date = opts.get('date')
34 if date:
35 if date:
35 opts['date'] = util.parsedate(date)
36 opts['date'] = util.parsedate(date)
36
37
37 parent, p2 = repo.dirstate.parents()
38 parent, p2 = repo.dirstate.parents()
38 branch = repo.dirstate.branch()
39 branch = repo.dirstate.branch()
39 branchnode = repo.branchtags().get(branch)
40 branchnode = repo.branchtags().get(branch)
40 if parent != branchnode:
41 if parent != branchnode:
41 raise util.Abort(_('working dir not at branch tip '
42 raise util.Abort(_('working dir not at branch tip '
42 '(use "hg update" to check out branch tip)'))
43 '(use "hg update" to check out branch tip)'))
43
44
44 if p2 != nullid:
45 if p2 != nullid:
45 raise util.Abort(_('outstanding uncommitted merge'))
46 raise util.Abort(_('outstanding uncommitted merge'))
46
47
47 wlock = lock = None
48 wlock = lock = None
48 try:
49 try:
49 wlock = repo.wlock()
50 wlock = repo.wlock()
50 lock = repo.lock()
51 lock = repo.lock()
51 mod, add, rem, del_ = repo.status()[:4]
52 mod, add, rem, del_ = repo.status()[:4]
52
53
53 if mod or add or rem:
54 if mod or add or rem:
54 raise util.Abort(_('outstanding uncommitted changes'))
55 raise util.Abort(_('outstanding uncommitted changes'))
55 if del_:
56 if del_:
56 raise util.Abort(_('working directory is missing some files'))
57 raise util.Abort(_('working directory is missing some files'))
57 bheads = repo.branchheads(branch)
58 bheads = repo.branchheads(branch)
58 bheads = [head for head in bheads if len(repo[head].children()) == 0]
59 bheads = [head for head in bheads if len(repo[head].children()) == 0]
59 if len(bheads) > 1:
60 if len(bheads) > 1:
60 raise util.Abort(_('multiple heads in this branch '
61 raise util.Abort(_('multiple heads in this branch '
61 '(use "hg heads ." and "hg merge" to merge)'))
62 '(use "hg heads ." and "hg merge" to merge)'))
62
63
63 other = hg.repository(cmdutil.remoteui(repo, opts),
64 other = hg.repository(cmdutil.remoteui(repo, opts),
64 ui.expandpath(source))
65 ui.expandpath(source))
65 ui.status(_('pulling from %s\n') %
66 ui.status(_('pulling from %s\n') %
66 url.hidepassword(ui.expandpath(source)))
67 url.hidepassword(ui.expandpath(source)))
67 revs = None
68 revs = None
68 if opts['rev']:
69 if opts['rev']:
69 try:
70 try:
70 revs = [other.lookup(rev) for rev in opts['rev']]
71 revs = [other.lookup(rev) for rev in opts['rev']]
71 except error.CapabilityError:
72 except error.CapabilityError:
72 err = _("Other repository doesn't support revision lookup, "
73 err = _("Other repository doesn't support revision lookup, "
73 "so a rev cannot be specified.")
74 "so a rev cannot be specified.")
74 raise util.Abort(err)
75 raise util.Abort(err)
75
76
76 # Are there any changes at all?
77 # Are there any changes at all?
77 modheads = repo.pull(other, heads=revs)
78 modheads = repo.pull(other, heads=revs)
78 if modheads == 0:
79 if modheads == 0:
79 return 0
80 return 0
80
81
81 # Is this a simple fast-forward along the current branch?
82 # Is this a simple fast-forward along the current branch?
82 newheads = repo.branchheads(branch)
83 newheads = repo.branchheads(branch)
83 newheads = [head for head in newheads if len(repo[head].children()) == 0]
84 newheads = [head for head in newheads if len(repo[head].children()) == 0]
84 newchildren = repo.changelog.nodesbetween([parent], newheads)[2]
85 newchildren = repo.changelog.nodesbetween([parent], newheads)[2]
85 if len(newheads) == 1:
86 if len(newheads) == 1:
86 if newchildren[0] != parent:
87 if newchildren[0] != parent:
87 return hg.clean(repo, newchildren[0])
88 return hg.clean(repo, newchildren[0])
88 else:
89 else:
89 return
90 return
90
91
91 # Are there more than one additional branch heads?
92 # Are there more than one additional branch heads?
92 newchildren = [n for n in newchildren if n != parent]
93 newchildren = [n for n in newchildren if n != parent]
93 newparent = parent
94 newparent = parent
94 if newchildren:
95 if newchildren:
95 newparent = newchildren[0]
96 newparent = newchildren[0]
96 hg.clean(repo, newparent)
97 hg.clean(repo, newparent)
97 newheads = [n for n in newheads if n != newparent]
98 newheads = [n for n in newheads if n != newparent]
98 if len(newheads) > 1:
99 if len(newheads) > 1:
99 ui.status(_('not merging with %d other new branch heads '
100 ui.status(_('not merging with %d other new branch heads '
100 '(use "hg heads ." and "hg merge" to merge them)\n') %
101 '(use "hg heads ." and "hg merge" to merge them)\n') %
101 (len(newheads) - 1))
102 (len(newheads) - 1))
102 return
103 return
103
104
104 # Otherwise, let's merge.
105 # Otherwise, let's merge.
105 err = False
106 err = False
106 if newheads:
107 if newheads:
107 # By default, we consider the repository we're pulling
108 # By default, we consider the repository we're pulling
108 # *from* as authoritative, so we merge our changes into
109 # *from* as authoritative, so we merge our changes into
109 # theirs.
110 # theirs.
110 if opts['switch_parent']:
111 if opts['switch_parent']:
111 firstparent, secondparent = newparent, newheads[0]
112 firstparent, secondparent = newparent, newheads[0]
112 else:
113 else:
113 firstparent, secondparent = newheads[0], newparent
114 firstparent, secondparent = newheads[0], newparent
114 ui.status(_('updating to %d:%s\n') %
115 ui.status(_('updating to %d:%s\n') %
115 (repo.changelog.rev(firstparent),
116 (repo.changelog.rev(firstparent),
116 short(firstparent)))
117 short(firstparent)))
117 hg.clean(repo, firstparent)
118 hg.clean(repo, firstparent)
118 ui.status(_('merging with %d:%s\n') %
119 ui.status(_('merging with %d:%s\n') %
119 (repo.changelog.rev(secondparent), short(secondparent)))
120 (repo.changelog.rev(secondparent), short(secondparent)))
120 err = hg.merge(repo, secondparent, remind=False)
121 err = hg.merge(repo, secondparent, remind=False)
121
122
122 if not err:
123 if not err:
123 # we don't translate commit messages
124 # we don't translate commit messages
124 message = (cmdutil.logmessage(opts) or
125 message = (cmdutil.logmessage(opts) or
125 ('Automated merge with %s' %
126 ('Automated merge with %s' %
126 url.removeauth(other.url())))
127 url.removeauth(other.url())))
127 editor = cmdutil.commiteditor
128 editor = cmdutil.commiteditor
128 if opts.get('force_editor') or opts.get('edit'):
129 if opts.get('force_editor') or opts.get('edit'):
129 editor = cmdutil.commitforceeditor
130 editor = cmdutil.commitforceeditor
130 n = repo.commit(message, opts['user'], opts['date'], editor=editor)
131 n = repo.commit(message, opts['user'], opts['date'], editor=editor)
131 ui.status(_('new changeset %d:%s merges remote changes '
132 ui.status(_('new changeset %d:%s merges remote changes '
132 'with local\n') % (repo.changelog.rev(n),
133 'with local\n') % (repo.changelog.rev(n),
133 short(n)))
134 short(n)))
134
135
135 finally:
136 finally:
136 release(lock, wlock)
137 release(lock, wlock)
137
138
138 cmdtable = {
139 cmdtable = {
139 'fetch':
140 'fetch':
140 (fetch,
141 (fetch,
141 [('r', 'rev', [], _('a specific revision you would like to pull')),
142 [('r', 'rev', [], _('a specific revision you would like to pull')),
142 ('e', 'edit', None, _('edit commit message')),
143 ('e', 'edit', None, _('edit commit message')),
143 ('', 'force-editor', None, _('edit commit message (DEPRECATED)')),
144 ('', 'force-editor', None, _('edit commit message (DEPRECATED)')),
144 ('', 'switch-parent', None, _('switch parents when merging')),
145 ('', 'switch-parent', None, _('switch parents when merging')),
145 ] + commands.commitopts + commands.commitopts2 + commands.remoteopts,
146 ] + commands.commitopts + commands.commitopts2 + commands.remoteopts,
146 _('hg fetch [SOURCE]')),
147 _('hg fetch [SOURCE]')),
147 }
148 }
General Comments 0
You need to be logged in to leave comments. Login now