##// END OF EJS Templates
bundle2: allow extensions to plug into the push process...
Pierre-Yves David -
r21149:c0d96bce default
parent child Browse files
Show More
@@ -1,721 +1,737
1 # exchange.py - utility to exchange data between repos.
1 # exchange.py - utility to exchange data between repos.
2 #
2 #
3 # Copyright 2005-2007 Matt Mackall <mpm@selenic.com>
3 # Copyright 2005-2007 Matt Mackall <mpm@selenic.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 from i18n import _
8 from i18n import _
9 from node import hex, nullid
9 from node import hex, nullid
10 import errno, urllib
10 import errno, urllib
11 import util, scmutil, changegroup, base85
11 import util, scmutil, changegroup, base85
12 import discovery, phases, obsolete, bookmarks, bundle2
12 import discovery, phases, obsolete, bookmarks, bundle2
13
13
14 def readbundle(ui, fh, fname, vfs=None):
14 def readbundle(ui, fh, fname, vfs=None):
15 header = changegroup.readexactly(fh, 4)
15 header = changegroup.readexactly(fh, 4)
16
16
17 alg = None
17 alg = None
18 if not fname:
18 if not fname:
19 fname = "stream"
19 fname = "stream"
20 if not header.startswith('HG') and header.startswith('\0'):
20 if not header.startswith('HG') and header.startswith('\0'):
21 fh = changegroup.headerlessfixup(fh, header)
21 fh = changegroup.headerlessfixup(fh, header)
22 header = "HG10"
22 header = "HG10"
23 alg = 'UN'
23 alg = 'UN'
24 elif vfs:
24 elif vfs:
25 fname = vfs.join(fname)
25 fname = vfs.join(fname)
26
26
27 magic, version = header[0:2], header[2:4]
27 magic, version = header[0:2], header[2:4]
28
28
29 if magic != 'HG':
29 if magic != 'HG':
30 raise util.Abort(_('%s: not a Mercurial bundle') % fname)
30 raise util.Abort(_('%s: not a Mercurial bundle') % fname)
31 if version == '10':
31 if version == '10':
32 if alg is None:
32 if alg is None:
33 alg = changegroup.readexactly(fh, 2)
33 alg = changegroup.readexactly(fh, 2)
34 return changegroup.unbundle10(fh, alg)
34 return changegroup.unbundle10(fh, alg)
35 elif version == '2X':
35 elif version == '2X':
36 return bundle2.unbundle20(ui, fh, header=magic + version)
36 return bundle2.unbundle20(ui, fh, header=magic + version)
37 else:
37 else:
38 raise util.Abort(_('%s: unknown bundle version %s') % (fname, version))
38 raise util.Abort(_('%s: unknown bundle version %s') % (fname, version))
39
39
40
40
41 class pushoperation(object):
41 class pushoperation(object):
42 """A object that represent a single push operation
42 """A object that represent a single push operation
43
43
44 It purpose is to carry push related state and very common operation.
44 It purpose is to carry push related state and very common operation.
45
45
46 A new should be created at the beginning of each push and discarded
46 A new should be created at the beginning of each push and discarded
47 afterward.
47 afterward.
48 """
48 """
49
49
50 def __init__(self, repo, remote, force=False, revs=None, newbranch=False):
50 def __init__(self, repo, remote, force=False, revs=None, newbranch=False):
51 # repo we push from
51 # repo we push from
52 self.repo = repo
52 self.repo = repo
53 self.ui = repo.ui
53 self.ui = repo.ui
54 # repo we push to
54 # repo we push to
55 self.remote = remote
55 self.remote = remote
56 # force option provided
56 # force option provided
57 self.force = force
57 self.force = force
58 # revs to be pushed (None is "all")
58 # revs to be pushed (None is "all")
59 self.revs = revs
59 self.revs = revs
60 # allow push of new branch
60 # allow push of new branch
61 self.newbranch = newbranch
61 self.newbranch = newbranch
62 # did a local lock get acquired?
62 # did a local lock get acquired?
63 self.locallocked = None
63 self.locallocked = None
64 # Integer version of the push result
64 # Integer version of the push result
65 # - None means nothing to push
65 # - None means nothing to push
66 # - 0 means HTTP error
66 # - 0 means HTTP error
67 # - 1 means we pushed and remote head count is unchanged *or*
67 # - 1 means we pushed and remote head count is unchanged *or*
68 # we have outgoing changesets but refused to push
68 # we have outgoing changesets but refused to push
69 # - other values as described by addchangegroup()
69 # - other values as described by addchangegroup()
70 self.ret = None
70 self.ret = None
71 # discover.outgoing object (contains common and outgoing data)
71 # discover.outgoing object (contains common and outgoing data)
72 self.outgoing = None
72 self.outgoing = None
73 # all remote heads before the push
73 # all remote heads before the push
74 self.remoteheads = None
74 self.remoteheads = None
75 # testable as a boolean indicating if any nodes are missing locally.
75 # testable as a boolean indicating if any nodes are missing locally.
76 self.incoming = None
76 self.incoming = None
77 # set of all heads common after changeset bundle push
77 # set of all heads common after changeset bundle push
78 self.commonheads = None
78 self.commonheads = None
79
79
80 def push(repo, remote, force=False, revs=None, newbranch=False):
80 def push(repo, remote, force=False, revs=None, newbranch=False):
81 '''Push outgoing changesets (limited by revs) from a local
81 '''Push outgoing changesets (limited by revs) from a local
82 repository to remote. Return an integer:
82 repository to remote. Return an integer:
83 - None means nothing to push
83 - None means nothing to push
84 - 0 means HTTP error
84 - 0 means HTTP error
85 - 1 means we pushed and remote head count is unchanged *or*
85 - 1 means we pushed and remote head count is unchanged *or*
86 we have outgoing changesets but refused to push
86 we have outgoing changesets but refused to push
87 - other values as described by addchangegroup()
87 - other values as described by addchangegroup()
88 '''
88 '''
89 pushop = pushoperation(repo, remote, force, revs, newbranch)
89 pushop = pushoperation(repo, remote, force, revs, newbranch)
90 if pushop.remote.local():
90 if pushop.remote.local():
91 missing = (set(pushop.repo.requirements)
91 missing = (set(pushop.repo.requirements)
92 - pushop.remote.local().supported)
92 - pushop.remote.local().supported)
93 if missing:
93 if missing:
94 msg = _("required features are not"
94 msg = _("required features are not"
95 " supported in the destination:"
95 " supported in the destination:"
96 " %s") % (', '.join(sorted(missing)))
96 " %s") % (', '.join(sorted(missing)))
97 raise util.Abort(msg)
97 raise util.Abort(msg)
98
98
99 # there are two ways to push to remote repo:
99 # there are two ways to push to remote repo:
100 #
100 #
101 # addchangegroup assumes local user can lock remote
101 # addchangegroup assumes local user can lock remote
102 # repo (local filesystem, old ssh servers).
102 # repo (local filesystem, old ssh servers).
103 #
103 #
104 # unbundle assumes local user cannot lock remote repo (new ssh
104 # unbundle assumes local user cannot lock remote repo (new ssh
105 # servers, http servers).
105 # servers, http servers).
106
106
107 if not pushop.remote.canpush():
107 if not pushop.remote.canpush():
108 raise util.Abort(_("destination does not support push"))
108 raise util.Abort(_("destination does not support push"))
109 # get local lock as we might write phase data
109 # get local lock as we might write phase data
110 locallock = None
110 locallock = None
111 try:
111 try:
112 locallock = pushop.repo.lock()
112 locallock = pushop.repo.lock()
113 pushop.locallocked = True
113 pushop.locallocked = True
114 except IOError, err:
114 except IOError, err:
115 pushop.locallocked = False
115 pushop.locallocked = False
116 if err.errno != errno.EACCES:
116 if err.errno != errno.EACCES:
117 raise
117 raise
118 # source repo cannot be locked.
118 # source repo cannot be locked.
119 # We do not abort the push, but just disable the local phase
119 # We do not abort the push, but just disable the local phase
120 # synchronisation.
120 # synchronisation.
121 msg = 'cannot lock source repository: %s\n' % err
121 msg = 'cannot lock source repository: %s\n' % err
122 pushop.ui.debug(msg)
122 pushop.ui.debug(msg)
123 try:
123 try:
124 pushop.repo.checkpush(pushop)
124 pushop.repo.checkpush(pushop)
125 lock = None
125 lock = None
126 unbundle = pushop.remote.capable('unbundle')
126 unbundle = pushop.remote.capable('unbundle')
127 if not unbundle:
127 if not unbundle:
128 lock = pushop.remote.lock()
128 lock = pushop.remote.lock()
129 try:
129 try:
130 _pushdiscovery(pushop)
130 _pushdiscovery(pushop)
131 if _pushcheckoutgoing(pushop):
131 if _pushcheckoutgoing(pushop):
132 pushop.repo.prepushoutgoinghooks(pushop.repo,
132 pushop.repo.prepushoutgoinghooks(pushop.repo,
133 pushop.remote,
133 pushop.remote,
134 pushop.outgoing)
134 pushop.outgoing)
135 if (pushop.repo.ui.configbool('experimental', 'bundle2-exp',
135 if (pushop.repo.ui.configbool('experimental', 'bundle2-exp',
136 False)
136 False)
137 and pushop.remote.capable('bundle2-exp')):
137 and pushop.remote.capable('bundle2-exp')):
138 _pushbundle2(pushop)
138 _pushbundle2(pushop)
139 else:
139 else:
140 _pushchangeset(pushop)
140 _pushchangeset(pushop)
141 _pushcomputecommonheads(pushop)
141 _pushcomputecommonheads(pushop)
142 _pushsyncphase(pushop)
142 _pushsyncphase(pushop)
143 _pushobsolete(pushop)
143 _pushobsolete(pushop)
144 finally:
144 finally:
145 if lock is not None:
145 if lock is not None:
146 lock.release()
146 lock.release()
147 finally:
147 finally:
148 if locallock is not None:
148 if locallock is not None:
149 locallock.release()
149 locallock.release()
150
150
151 _pushbookmark(pushop)
151 _pushbookmark(pushop)
152 return pushop.ret
152 return pushop.ret
153
153
154 def _pushdiscovery(pushop):
154 def _pushdiscovery(pushop):
155 # discovery
155 # discovery
156 unfi = pushop.repo.unfiltered()
156 unfi = pushop.repo.unfiltered()
157 fci = discovery.findcommonincoming
157 fci = discovery.findcommonincoming
158 commoninc = fci(unfi, pushop.remote, force=pushop.force)
158 commoninc = fci(unfi, pushop.remote, force=pushop.force)
159 common, inc, remoteheads = commoninc
159 common, inc, remoteheads = commoninc
160 fco = discovery.findcommonoutgoing
160 fco = discovery.findcommonoutgoing
161 outgoing = fco(unfi, pushop.remote, onlyheads=pushop.revs,
161 outgoing = fco(unfi, pushop.remote, onlyheads=pushop.revs,
162 commoninc=commoninc, force=pushop.force)
162 commoninc=commoninc, force=pushop.force)
163 pushop.outgoing = outgoing
163 pushop.outgoing = outgoing
164 pushop.remoteheads = remoteheads
164 pushop.remoteheads = remoteheads
165 pushop.incoming = inc
165 pushop.incoming = inc
166
166
167 def _pushcheckoutgoing(pushop):
167 def _pushcheckoutgoing(pushop):
168 outgoing = pushop.outgoing
168 outgoing = pushop.outgoing
169 unfi = pushop.repo.unfiltered()
169 unfi = pushop.repo.unfiltered()
170 if not outgoing.missing:
170 if not outgoing.missing:
171 # nothing to push
171 # nothing to push
172 scmutil.nochangesfound(unfi.ui, unfi, outgoing.excluded)
172 scmutil.nochangesfound(unfi.ui, unfi, outgoing.excluded)
173 return False
173 return False
174 # something to push
174 # something to push
175 if not pushop.force:
175 if not pushop.force:
176 # if repo.obsstore == False --> no obsolete
176 # if repo.obsstore == False --> no obsolete
177 # then, save the iteration
177 # then, save the iteration
178 if unfi.obsstore:
178 if unfi.obsstore:
179 # this message are here for 80 char limit reason
179 # this message are here for 80 char limit reason
180 mso = _("push includes obsolete changeset: %s!")
180 mso = _("push includes obsolete changeset: %s!")
181 mst = "push includes %s changeset: %s!"
181 mst = "push includes %s changeset: %s!"
182 # plain versions for i18n tool to detect them
182 # plain versions for i18n tool to detect them
183 _("push includes unstable changeset: %s!")
183 _("push includes unstable changeset: %s!")
184 _("push includes bumped changeset: %s!")
184 _("push includes bumped changeset: %s!")
185 _("push includes divergent changeset: %s!")
185 _("push includes divergent changeset: %s!")
186 # If we are to push if there is at least one
186 # If we are to push if there is at least one
187 # obsolete or unstable changeset in missing, at
187 # obsolete or unstable changeset in missing, at
188 # least one of the missinghead will be obsolete or
188 # least one of the missinghead will be obsolete or
189 # unstable. So checking heads only is ok
189 # unstable. So checking heads only is ok
190 for node in outgoing.missingheads:
190 for node in outgoing.missingheads:
191 ctx = unfi[node]
191 ctx = unfi[node]
192 if ctx.obsolete():
192 if ctx.obsolete():
193 raise util.Abort(mso % ctx)
193 raise util.Abort(mso % ctx)
194 elif ctx.troubled():
194 elif ctx.troubled():
195 raise util.Abort(_(mst)
195 raise util.Abort(_(mst)
196 % (ctx.troubles()[0],
196 % (ctx.troubles()[0],
197 ctx))
197 ctx))
198 newbm = pushop.ui.configlist('bookmarks', 'pushing')
198 newbm = pushop.ui.configlist('bookmarks', 'pushing')
199 discovery.checkheads(unfi, pushop.remote, outgoing,
199 discovery.checkheads(unfi, pushop.remote, outgoing,
200 pushop.remoteheads,
200 pushop.remoteheads,
201 pushop.newbranch,
201 pushop.newbranch,
202 bool(pushop.incoming),
202 bool(pushop.incoming),
203 newbm)
203 newbm)
204 return True
204 return True
205
205
206 def _pushbundle2(pushop):
206 def _pushbundle2(pushop):
207 """push data to the remote using bundle2
207 """push data to the remote using bundle2
208
208
209 The only currently supported type of data is changegroup but this will
209 The only currently supported type of data is changegroup but this will
210 evolve in the future."""
210 evolve in the future."""
211 # Send known head to the server for race detection.
211 # Send known head to the server for race detection.
212 capsblob = urllib.unquote(pushop.remote.capable('bundle2-exp'))
212 capsblob = urllib.unquote(pushop.remote.capable('bundle2-exp'))
213 caps = bundle2.decodecaps(capsblob)
213 caps = bundle2.decodecaps(capsblob)
214 bundler = bundle2.bundle20(pushop.ui, caps)
214 bundler = bundle2.bundle20(pushop.ui, caps)
215 # create reply capability
215 # create reply capability
216 capsblob = bundle2.encodecaps(pushop.repo.bundle2caps)
216 capsblob = bundle2.encodecaps(pushop.repo.bundle2caps)
217 bundler.addpart(bundle2.bundlepart('b2x:replycaps', data=capsblob))
217 bundler.addpart(bundle2.bundlepart('b2x:replycaps', data=capsblob))
218 if not pushop.force:
218 if not pushop.force:
219 part = bundle2.bundlepart('B2X:CHECK:HEADS',
219 part = bundle2.bundlepart('B2X:CHECK:HEADS',
220 data=iter(pushop.remoteheads))
220 data=iter(pushop.remoteheads))
221 bundler.addpart(part)
221 bundler.addpart(part)
222 extrainfo = _pushbundle2extraparts(pushop, bundler)
222 # add the changegroup bundle
223 # add the changegroup bundle
223 cg = changegroup.getlocalbundle(pushop.repo, 'push', pushop.outgoing)
224 cg = changegroup.getlocalbundle(pushop.repo, 'push', pushop.outgoing)
224 cgpart = bundle2.bundlepart('B2X:CHANGEGROUP', data=cg.getchunks())
225 cgpart = bundle2.bundlepart('B2X:CHANGEGROUP', data=cg.getchunks())
225 bundler.addpart(cgpart)
226 bundler.addpart(cgpart)
226 stream = util.chunkbuffer(bundler.getchunks())
227 stream = util.chunkbuffer(bundler.getchunks())
227 reply = pushop.remote.unbundle(stream, ['force'], 'push')
228 reply = pushop.remote.unbundle(stream, ['force'], 'push')
228 try:
229 try:
229 op = bundle2.processbundle(pushop.repo, reply)
230 op = bundle2.processbundle(pushop.repo, reply)
230 except KeyError, exc:
231 except KeyError, exc:
231 raise util.Abort('missing support for %s' % exc)
232 raise util.Abort('missing support for %s' % exc)
232 cgreplies = op.records.getreplies(cgpart.id)
233 cgreplies = op.records.getreplies(cgpart.id)
233 assert len(cgreplies['changegroup']) == 1
234 assert len(cgreplies['changegroup']) == 1
234 pushop.ret = cgreplies['changegroup'][0]['return']
235 pushop.ret = cgreplies['changegroup'][0]['return']
236 _pushbundle2extrareply(pushop, op, extrainfo)
237
238 def _pushbundle2extraparts(pushop, bundler):
239 """hook function to let extensions add parts
240
241 Return a dict to let extensions pass data to the reply processing.
242 """
243 return {}
244
245 def _pushbundle2extrareply(pushop, op, extrainfo):
246 """hook function to let extensions react to part replies
247
248 The dict from _pushbundle2extrareply is fed to this function.
249 """
250 pass
235
251
236 def _pushchangeset(pushop):
252 def _pushchangeset(pushop):
237 """Make the actual push of changeset bundle to remote repo"""
253 """Make the actual push of changeset bundle to remote repo"""
238 outgoing = pushop.outgoing
254 outgoing = pushop.outgoing
239 unbundle = pushop.remote.capable('unbundle')
255 unbundle = pushop.remote.capable('unbundle')
240 # TODO: get bundlecaps from remote
256 # TODO: get bundlecaps from remote
241 bundlecaps = None
257 bundlecaps = None
242 # create a changegroup from local
258 # create a changegroup from local
243 if pushop.revs is None and not (outgoing.excluded
259 if pushop.revs is None and not (outgoing.excluded
244 or pushop.repo.changelog.filteredrevs):
260 or pushop.repo.changelog.filteredrevs):
245 # push everything,
261 # push everything,
246 # use the fast path, no race possible on push
262 # use the fast path, no race possible on push
247 bundler = changegroup.bundle10(pushop.repo, bundlecaps)
263 bundler = changegroup.bundle10(pushop.repo, bundlecaps)
248 cg = changegroup.getsubset(pushop.repo,
264 cg = changegroup.getsubset(pushop.repo,
249 outgoing,
265 outgoing,
250 bundler,
266 bundler,
251 'push',
267 'push',
252 fastpath=True)
268 fastpath=True)
253 else:
269 else:
254 cg = changegroup.getlocalbundle(pushop.repo, 'push', outgoing,
270 cg = changegroup.getlocalbundle(pushop.repo, 'push', outgoing,
255 bundlecaps)
271 bundlecaps)
256
272
257 # apply changegroup to remote
273 # apply changegroup to remote
258 if unbundle:
274 if unbundle:
259 # local repo finds heads on server, finds out what
275 # local repo finds heads on server, finds out what
260 # revs it must push. once revs transferred, if server
276 # revs it must push. once revs transferred, if server
261 # finds it has different heads (someone else won
277 # finds it has different heads (someone else won
262 # commit/push race), server aborts.
278 # commit/push race), server aborts.
263 if pushop.force:
279 if pushop.force:
264 remoteheads = ['force']
280 remoteheads = ['force']
265 else:
281 else:
266 remoteheads = pushop.remoteheads
282 remoteheads = pushop.remoteheads
267 # ssh: return remote's addchangegroup()
283 # ssh: return remote's addchangegroup()
268 # http: return remote's addchangegroup() or 0 for error
284 # http: return remote's addchangegroup() or 0 for error
269 pushop.ret = pushop.remote.unbundle(cg, remoteheads,
285 pushop.ret = pushop.remote.unbundle(cg, remoteheads,
270 'push')
286 'push')
271 else:
287 else:
272 # we return an integer indicating remote head count
288 # we return an integer indicating remote head count
273 # change
289 # change
274 pushop.ret = pushop.remote.addchangegroup(cg, 'push', pushop.repo.url())
290 pushop.ret = pushop.remote.addchangegroup(cg, 'push', pushop.repo.url())
275
291
276 def _pushcomputecommonheads(pushop):
292 def _pushcomputecommonheads(pushop):
277 unfi = pushop.repo.unfiltered()
293 unfi = pushop.repo.unfiltered()
278 if pushop.ret:
294 if pushop.ret:
279 # push succeed, synchronize target of the push
295 # push succeed, synchronize target of the push
280 cheads = pushop.outgoing.missingheads
296 cheads = pushop.outgoing.missingheads
281 elif pushop.revs is None:
297 elif pushop.revs is None:
282 # All out push fails. synchronize all common
298 # All out push fails. synchronize all common
283 cheads = pushop.outgoing.commonheads
299 cheads = pushop.outgoing.commonheads
284 else:
300 else:
285 # I want cheads = heads(::missingheads and ::commonheads)
301 # I want cheads = heads(::missingheads and ::commonheads)
286 # (missingheads is revs with secret changeset filtered out)
302 # (missingheads is revs with secret changeset filtered out)
287 #
303 #
288 # This can be expressed as:
304 # This can be expressed as:
289 # cheads = ( (missingheads and ::commonheads)
305 # cheads = ( (missingheads and ::commonheads)
290 # + (commonheads and ::missingheads))"
306 # + (commonheads and ::missingheads))"
291 # )
307 # )
292 #
308 #
293 # while trying to push we already computed the following:
309 # while trying to push we already computed the following:
294 # common = (::commonheads)
310 # common = (::commonheads)
295 # missing = ((commonheads::missingheads) - commonheads)
311 # missing = ((commonheads::missingheads) - commonheads)
296 #
312 #
297 # We can pick:
313 # We can pick:
298 # * missingheads part of common (::commonheads)
314 # * missingheads part of common (::commonheads)
299 common = set(pushop.outgoing.common)
315 common = set(pushop.outgoing.common)
300 nm = pushop.repo.changelog.nodemap
316 nm = pushop.repo.changelog.nodemap
301 cheads = [node for node in pushop.revs if nm[node] in common]
317 cheads = [node for node in pushop.revs if nm[node] in common]
302 # and
318 # and
303 # * commonheads parents on missing
319 # * commonheads parents on missing
304 revset = unfi.set('%ln and parents(roots(%ln))',
320 revset = unfi.set('%ln and parents(roots(%ln))',
305 pushop.outgoing.commonheads,
321 pushop.outgoing.commonheads,
306 pushop.outgoing.missing)
322 pushop.outgoing.missing)
307 cheads.extend(c.node() for c in revset)
323 cheads.extend(c.node() for c in revset)
308 pushop.commonheads = cheads
324 pushop.commonheads = cheads
309
325
310 def _pushsyncphase(pushop):
326 def _pushsyncphase(pushop):
311 """synchronise phase information locally and remotely"""
327 """synchronise phase information locally and remotely"""
312 unfi = pushop.repo.unfiltered()
328 unfi = pushop.repo.unfiltered()
313 cheads = pushop.commonheads
329 cheads = pushop.commonheads
314 if pushop.ret:
330 if pushop.ret:
315 # push succeed, synchronize target of the push
331 # push succeed, synchronize target of the push
316 cheads = pushop.outgoing.missingheads
332 cheads = pushop.outgoing.missingheads
317 elif pushop.revs is None:
333 elif pushop.revs is None:
318 # All out push fails. synchronize all common
334 # All out push fails. synchronize all common
319 cheads = pushop.outgoing.commonheads
335 cheads = pushop.outgoing.commonheads
320 else:
336 else:
321 # I want cheads = heads(::missingheads and ::commonheads)
337 # I want cheads = heads(::missingheads and ::commonheads)
322 # (missingheads is revs with secret changeset filtered out)
338 # (missingheads is revs with secret changeset filtered out)
323 #
339 #
324 # This can be expressed as:
340 # This can be expressed as:
325 # cheads = ( (missingheads and ::commonheads)
341 # cheads = ( (missingheads and ::commonheads)
326 # + (commonheads and ::missingheads))"
342 # + (commonheads and ::missingheads))"
327 # )
343 # )
328 #
344 #
329 # while trying to push we already computed the following:
345 # while trying to push we already computed the following:
330 # common = (::commonheads)
346 # common = (::commonheads)
331 # missing = ((commonheads::missingheads) - commonheads)
347 # missing = ((commonheads::missingheads) - commonheads)
332 #
348 #
333 # We can pick:
349 # We can pick:
334 # * missingheads part of common (::commonheads)
350 # * missingheads part of common (::commonheads)
335 common = set(pushop.outgoing.common)
351 common = set(pushop.outgoing.common)
336 nm = pushop.repo.changelog.nodemap
352 nm = pushop.repo.changelog.nodemap
337 cheads = [node for node in pushop.revs if nm[node] in common]
353 cheads = [node for node in pushop.revs if nm[node] in common]
338 # and
354 # and
339 # * commonheads parents on missing
355 # * commonheads parents on missing
340 revset = unfi.set('%ln and parents(roots(%ln))',
356 revset = unfi.set('%ln and parents(roots(%ln))',
341 pushop.outgoing.commonheads,
357 pushop.outgoing.commonheads,
342 pushop.outgoing.missing)
358 pushop.outgoing.missing)
343 cheads.extend(c.node() for c in revset)
359 cheads.extend(c.node() for c in revset)
344 pushop.commonheads = cheads
360 pushop.commonheads = cheads
345 # even when we don't push, exchanging phase data is useful
361 # even when we don't push, exchanging phase data is useful
346 remotephases = pushop.remote.listkeys('phases')
362 remotephases = pushop.remote.listkeys('phases')
347 if (pushop.ui.configbool('ui', '_usedassubrepo', False)
363 if (pushop.ui.configbool('ui', '_usedassubrepo', False)
348 and remotephases # server supports phases
364 and remotephases # server supports phases
349 and pushop.ret is None # nothing was pushed
365 and pushop.ret is None # nothing was pushed
350 and remotephases.get('publishing', False)):
366 and remotephases.get('publishing', False)):
351 # When:
367 # When:
352 # - this is a subrepo push
368 # - this is a subrepo push
353 # - and remote support phase
369 # - and remote support phase
354 # - and no changeset was pushed
370 # - and no changeset was pushed
355 # - and remote is publishing
371 # - and remote is publishing
356 # We may be in issue 3871 case!
372 # We may be in issue 3871 case!
357 # We drop the possible phase synchronisation done by
373 # We drop the possible phase synchronisation done by
358 # courtesy to publish changesets possibly locally draft
374 # courtesy to publish changesets possibly locally draft
359 # on the remote.
375 # on the remote.
360 remotephases = {'publishing': 'True'}
376 remotephases = {'publishing': 'True'}
361 if not remotephases: # old server or public only reply from non-publishing
377 if not remotephases: # old server or public only reply from non-publishing
362 _localphasemove(pushop, cheads)
378 _localphasemove(pushop, cheads)
363 # don't push any phase data as there is nothing to push
379 # don't push any phase data as there is nothing to push
364 else:
380 else:
365 ana = phases.analyzeremotephases(pushop.repo, cheads,
381 ana = phases.analyzeremotephases(pushop.repo, cheads,
366 remotephases)
382 remotephases)
367 pheads, droots = ana
383 pheads, droots = ana
368 ### Apply remote phase on local
384 ### Apply remote phase on local
369 if remotephases.get('publishing', False):
385 if remotephases.get('publishing', False):
370 _localphasemove(pushop, cheads)
386 _localphasemove(pushop, cheads)
371 else: # publish = False
387 else: # publish = False
372 _localphasemove(pushop, pheads)
388 _localphasemove(pushop, pheads)
373 _localphasemove(pushop, cheads, phases.draft)
389 _localphasemove(pushop, cheads, phases.draft)
374 ### Apply local phase on remote
390 ### Apply local phase on remote
375
391
376 # Get the list of all revs draft on remote by public here.
392 # Get the list of all revs draft on remote by public here.
377 # XXX Beware that revset break if droots is not strictly
393 # XXX Beware that revset break if droots is not strictly
378 # XXX root we may want to ensure it is but it is costly
394 # XXX root we may want to ensure it is but it is costly
379 outdated = unfi.set('heads((%ln::%ln) and public())',
395 outdated = unfi.set('heads((%ln::%ln) and public())',
380 droots, cheads)
396 droots, cheads)
381 for newremotehead in outdated:
397 for newremotehead in outdated:
382 r = pushop.remote.pushkey('phases',
398 r = pushop.remote.pushkey('phases',
383 newremotehead.hex(),
399 newremotehead.hex(),
384 str(phases.draft),
400 str(phases.draft),
385 str(phases.public))
401 str(phases.public))
386 if not r:
402 if not r:
387 pushop.ui.warn(_('updating %s to public failed!\n')
403 pushop.ui.warn(_('updating %s to public failed!\n')
388 % newremotehead)
404 % newremotehead)
389
405
390 def _localphasemove(pushop, nodes, phase=phases.public):
406 def _localphasemove(pushop, nodes, phase=phases.public):
391 """move <nodes> to <phase> in the local source repo"""
407 """move <nodes> to <phase> in the local source repo"""
392 if pushop.locallocked:
408 if pushop.locallocked:
393 phases.advanceboundary(pushop.repo, phase, nodes)
409 phases.advanceboundary(pushop.repo, phase, nodes)
394 else:
410 else:
395 # repo is not locked, do not change any phases!
411 # repo is not locked, do not change any phases!
396 # Informs the user that phases should have been moved when
412 # Informs the user that phases should have been moved when
397 # applicable.
413 # applicable.
398 actualmoves = [n for n in nodes if phase < pushop.repo[n].phase()]
414 actualmoves = [n for n in nodes if phase < pushop.repo[n].phase()]
399 phasestr = phases.phasenames[phase]
415 phasestr = phases.phasenames[phase]
400 if actualmoves:
416 if actualmoves:
401 pushop.ui.status(_('cannot lock source repo, skipping '
417 pushop.ui.status(_('cannot lock source repo, skipping '
402 'local %s phase update\n') % phasestr)
418 'local %s phase update\n') % phasestr)
403
419
404 def _pushobsolete(pushop):
420 def _pushobsolete(pushop):
405 """utility function to push obsolete markers to a remote"""
421 """utility function to push obsolete markers to a remote"""
406 pushop.ui.debug('try to push obsolete markers to remote\n')
422 pushop.ui.debug('try to push obsolete markers to remote\n')
407 repo = pushop.repo
423 repo = pushop.repo
408 remote = pushop.remote
424 remote = pushop.remote
409 if (obsolete._enabled and repo.obsstore and
425 if (obsolete._enabled and repo.obsstore and
410 'obsolete' in remote.listkeys('namespaces')):
426 'obsolete' in remote.listkeys('namespaces')):
411 rslts = []
427 rslts = []
412 remotedata = repo.listkeys('obsolete')
428 remotedata = repo.listkeys('obsolete')
413 for key in sorted(remotedata, reverse=True):
429 for key in sorted(remotedata, reverse=True):
414 # reverse sort to ensure we end with dump0
430 # reverse sort to ensure we end with dump0
415 data = remotedata[key]
431 data = remotedata[key]
416 rslts.append(remote.pushkey('obsolete', key, '', data))
432 rslts.append(remote.pushkey('obsolete', key, '', data))
417 if [r for r in rslts if not r]:
433 if [r for r in rslts if not r]:
418 msg = _('failed to push some obsolete markers!\n')
434 msg = _('failed to push some obsolete markers!\n')
419 repo.ui.warn(msg)
435 repo.ui.warn(msg)
420
436
421 def _pushbookmark(pushop):
437 def _pushbookmark(pushop):
422 """Update bookmark position on remote"""
438 """Update bookmark position on remote"""
423 ui = pushop.ui
439 ui = pushop.ui
424 repo = pushop.repo.unfiltered()
440 repo = pushop.repo.unfiltered()
425 remote = pushop.remote
441 remote = pushop.remote
426 ui.debug("checking for updated bookmarks\n")
442 ui.debug("checking for updated bookmarks\n")
427 revnums = map(repo.changelog.rev, pushop.revs or [])
443 revnums = map(repo.changelog.rev, pushop.revs or [])
428 ancestors = [a for a in repo.changelog.ancestors(revnums, inclusive=True)]
444 ancestors = [a for a in repo.changelog.ancestors(revnums, inclusive=True)]
429 (addsrc, adddst, advsrc, advdst, diverge, differ, invalid
445 (addsrc, adddst, advsrc, advdst, diverge, differ, invalid
430 ) = bookmarks.compare(repo, repo._bookmarks, remote.listkeys('bookmarks'),
446 ) = bookmarks.compare(repo, repo._bookmarks, remote.listkeys('bookmarks'),
431 srchex=hex)
447 srchex=hex)
432
448
433 for b, scid, dcid in advsrc:
449 for b, scid, dcid in advsrc:
434 if ancestors and repo[scid].rev() not in ancestors:
450 if ancestors and repo[scid].rev() not in ancestors:
435 continue
451 continue
436 if remote.pushkey('bookmarks', b, dcid, scid):
452 if remote.pushkey('bookmarks', b, dcid, scid):
437 ui.status(_("updating bookmark %s\n") % b)
453 ui.status(_("updating bookmark %s\n") % b)
438 else:
454 else:
439 ui.warn(_('updating bookmark %s failed!\n') % b)
455 ui.warn(_('updating bookmark %s failed!\n') % b)
440
456
441 class pulloperation(object):
457 class pulloperation(object):
442 """A object that represent a single pull operation
458 """A object that represent a single pull operation
443
459
444 It purpose is to carry push related state and very common operation.
460 It purpose is to carry push related state and very common operation.
445
461
446 A new should be created at the beginning of each pull and discarded
462 A new should be created at the beginning of each pull and discarded
447 afterward.
463 afterward.
448 """
464 """
449
465
450 def __init__(self, repo, remote, heads=None, force=False):
466 def __init__(self, repo, remote, heads=None, force=False):
451 # repo we pull into
467 # repo we pull into
452 self.repo = repo
468 self.repo = repo
453 # repo we pull from
469 # repo we pull from
454 self.remote = remote
470 self.remote = remote
455 # revision we try to pull (None is "all")
471 # revision we try to pull (None is "all")
456 self.heads = heads
472 self.heads = heads
457 # do we force pull?
473 # do we force pull?
458 self.force = force
474 self.force = force
459 # the name the pull transaction
475 # the name the pull transaction
460 self._trname = 'pull\n' + util.hidepassword(remote.url())
476 self._trname = 'pull\n' + util.hidepassword(remote.url())
461 # hold the transaction once created
477 # hold the transaction once created
462 self._tr = None
478 self._tr = None
463 # set of common changeset between local and remote before pull
479 # set of common changeset between local and remote before pull
464 self.common = None
480 self.common = None
465 # set of pulled head
481 # set of pulled head
466 self.rheads = None
482 self.rheads = None
467 # list of missing changeset to fetch remotely
483 # list of missing changeset to fetch remotely
468 self.fetch = None
484 self.fetch = None
469 # result of changegroup pulling (used as return code by pull)
485 # result of changegroup pulling (used as return code by pull)
470 self.cgresult = None
486 self.cgresult = None
471 # list of step remaining todo (related to future bundle2 usage)
487 # list of step remaining todo (related to future bundle2 usage)
472 self.todosteps = set(['changegroup', 'phases', 'obsmarkers'])
488 self.todosteps = set(['changegroup', 'phases', 'obsmarkers'])
473
489
474 @util.propertycache
490 @util.propertycache
475 def pulledsubset(self):
491 def pulledsubset(self):
476 """heads of the set of changeset target by the pull"""
492 """heads of the set of changeset target by the pull"""
477 # compute target subset
493 # compute target subset
478 if self.heads is None:
494 if self.heads is None:
479 # We pulled every thing possible
495 # We pulled every thing possible
480 # sync on everything common
496 # sync on everything common
481 c = set(self.common)
497 c = set(self.common)
482 ret = list(self.common)
498 ret = list(self.common)
483 for n in self.rheads:
499 for n in self.rheads:
484 if n not in c:
500 if n not in c:
485 ret.append(n)
501 ret.append(n)
486 return ret
502 return ret
487 else:
503 else:
488 # We pulled a specific subset
504 # We pulled a specific subset
489 # sync on this subset
505 # sync on this subset
490 return self.heads
506 return self.heads
491
507
492 def gettransaction(self):
508 def gettransaction(self):
493 """get appropriate pull transaction, creating it if needed"""
509 """get appropriate pull transaction, creating it if needed"""
494 if self._tr is None:
510 if self._tr is None:
495 self._tr = self.repo.transaction(self._trname)
511 self._tr = self.repo.transaction(self._trname)
496 return self._tr
512 return self._tr
497
513
498 def closetransaction(self):
514 def closetransaction(self):
499 """close transaction if created"""
515 """close transaction if created"""
500 if self._tr is not None:
516 if self._tr is not None:
501 self._tr.close()
517 self._tr.close()
502
518
503 def releasetransaction(self):
519 def releasetransaction(self):
504 """release transaction if created"""
520 """release transaction if created"""
505 if self._tr is not None:
521 if self._tr is not None:
506 self._tr.release()
522 self._tr.release()
507
523
508 def pull(repo, remote, heads=None, force=False):
524 def pull(repo, remote, heads=None, force=False):
509 pullop = pulloperation(repo, remote, heads, force)
525 pullop = pulloperation(repo, remote, heads, force)
510 if pullop.remote.local():
526 if pullop.remote.local():
511 missing = set(pullop.remote.requirements) - pullop.repo.supported
527 missing = set(pullop.remote.requirements) - pullop.repo.supported
512 if missing:
528 if missing:
513 msg = _("required features are not"
529 msg = _("required features are not"
514 " supported in the destination:"
530 " supported in the destination:"
515 " %s") % (', '.join(sorted(missing)))
531 " %s") % (', '.join(sorted(missing)))
516 raise util.Abort(msg)
532 raise util.Abort(msg)
517
533
518 lock = pullop.repo.lock()
534 lock = pullop.repo.lock()
519 try:
535 try:
520 _pulldiscovery(pullop)
536 _pulldiscovery(pullop)
521 if (pullop.repo.ui.configbool('server', 'bundle2', False)
537 if (pullop.repo.ui.configbool('server', 'bundle2', False)
522 and pullop.remote.capable('bundle2-exp')):
538 and pullop.remote.capable('bundle2-exp')):
523 _pullbundle2(pullop)
539 _pullbundle2(pullop)
524 if 'changegroup' in pullop.todosteps:
540 if 'changegroup' in pullop.todosteps:
525 _pullchangeset(pullop)
541 _pullchangeset(pullop)
526 if 'phases' in pullop.todosteps:
542 if 'phases' in pullop.todosteps:
527 _pullphase(pullop)
543 _pullphase(pullop)
528 if 'obsmarkers' in pullop.todosteps:
544 if 'obsmarkers' in pullop.todosteps:
529 _pullobsolete(pullop)
545 _pullobsolete(pullop)
530 pullop.closetransaction()
546 pullop.closetransaction()
531 finally:
547 finally:
532 pullop.releasetransaction()
548 pullop.releasetransaction()
533 lock.release()
549 lock.release()
534
550
535 return pullop.cgresult
551 return pullop.cgresult
536
552
537 def _pulldiscovery(pullop):
553 def _pulldiscovery(pullop):
538 """discovery phase for the pull
554 """discovery phase for the pull
539
555
540 Current handle changeset discovery only, will change handle all discovery
556 Current handle changeset discovery only, will change handle all discovery
541 at some point."""
557 at some point."""
542 tmp = discovery.findcommonincoming(pullop.repo.unfiltered(),
558 tmp = discovery.findcommonincoming(pullop.repo.unfiltered(),
543 pullop.remote,
559 pullop.remote,
544 heads=pullop.heads,
560 heads=pullop.heads,
545 force=pullop.force)
561 force=pullop.force)
546 pullop.common, pullop.fetch, pullop.rheads = tmp
562 pullop.common, pullop.fetch, pullop.rheads = tmp
547
563
548 def _pullbundle2(pullop):
564 def _pullbundle2(pullop):
549 """pull data using bundle2
565 """pull data using bundle2
550
566
551 For now, the only supported data are changegroup."""
567 For now, the only supported data are changegroup."""
552 kwargs = {'bundlecaps': set(['HG2X'])}
568 kwargs = {'bundlecaps': set(['HG2X'])}
553 capsblob = bundle2.encodecaps(pullop.repo.bundle2caps)
569 capsblob = bundle2.encodecaps(pullop.repo.bundle2caps)
554 kwargs['bundlecaps'].add('bundle2=' + urllib.quote(capsblob))
570 kwargs['bundlecaps'].add('bundle2=' + urllib.quote(capsblob))
555 # pulling changegroup
571 # pulling changegroup
556 pullop.todosteps.remove('changegroup')
572 pullop.todosteps.remove('changegroup')
557 if not pullop.fetch:
573 if not pullop.fetch:
558 pullop.repo.ui.status(_("no changes found\n"))
574 pullop.repo.ui.status(_("no changes found\n"))
559 pullop.cgresult = 0
575 pullop.cgresult = 0
560 else:
576 else:
561 kwargs['common'] = pullop.common
577 kwargs['common'] = pullop.common
562 kwargs['heads'] = pullop.heads or pullop.rheads
578 kwargs['heads'] = pullop.heads or pullop.rheads
563 if pullop.heads is None and list(pullop.common) == [nullid]:
579 if pullop.heads is None and list(pullop.common) == [nullid]:
564 pullop.repo.ui.status(_("requesting all changes\n"))
580 pullop.repo.ui.status(_("requesting all changes\n"))
565 if kwargs.keys() == ['format']:
581 if kwargs.keys() == ['format']:
566 return # nothing to pull
582 return # nothing to pull
567 bundle = pullop.remote.getbundle('pull', **kwargs)
583 bundle = pullop.remote.getbundle('pull', **kwargs)
568 try:
584 try:
569 op = bundle2.processbundle(pullop.repo, bundle, pullop.gettransaction)
585 op = bundle2.processbundle(pullop.repo, bundle, pullop.gettransaction)
570 except KeyError, exc:
586 except KeyError, exc:
571 raise util.Abort('missing support for %s' % exc)
587 raise util.Abort('missing support for %s' % exc)
572 assert len(op.records['changegroup']) == 1
588 assert len(op.records['changegroup']) == 1
573 pullop.cgresult = op.records['changegroup'][0]['return']
589 pullop.cgresult = op.records['changegroup'][0]['return']
574
590
575 def _pullchangeset(pullop):
591 def _pullchangeset(pullop):
576 """pull changeset from unbundle into the local repo"""
592 """pull changeset from unbundle into the local repo"""
577 # We delay the open of the transaction as late as possible so we
593 # We delay the open of the transaction as late as possible so we
578 # don't open transaction for nothing or you break future useful
594 # don't open transaction for nothing or you break future useful
579 # rollback call
595 # rollback call
580 pullop.todosteps.remove('changegroup')
596 pullop.todosteps.remove('changegroup')
581 if not pullop.fetch:
597 if not pullop.fetch:
582 pullop.repo.ui.status(_("no changes found\n"))
598 pullop.repo.ui.status(_("no changes found\n"))
583 pullop.cgresult = 0
599 pullop.cgresult = 0
584 return
600 return
585 pullop.gettransaction()
601 pullop.gettransaction()
586 if pullop.heads is None and list(pullop.common) == [nullid]:
602 if pullop.heads is None and list(pullop.common) == [nullid]:
587 pullop.repo.ui.status(_("requesting all changes\n"))
603 pullop.repo.ui.status(_("requesting all changes\n"))
588 elif pullop.heads is None and pullop.remote.capable('changegroupsubset'):
604 elif pullop.heads is None and pullop.remote.capable('changegroupsubset'):
589 # issue1320, avoid a race if remote changed after discovery
605 # issue1320, avoid a race if remote changed after discovery
590 pullop.heads = pullop.rheads
606 pullop.heads = pullop.rheads
591
607
592 if pullop.remote.capable('getbundle'):
608 if pullop.remote.capable('getbundle'):
593 # TODO: get bundlecaps from remote
609 # TODO: get bundlecaps from remote
594 cg = pullop.remote.getbundle('pull', common=pullop.common,
610 cg = pullop.remote.getbundle('pull', common=pullop.common,
595 heads=pullop.heads or pullop.rheads)
611 heads=pullop.heads or pullop.rheads)
596 elif pullop.heads is None:
612 elif pullop.heads is None:
597 cg = pullop.remote.changegroup(pullop.fetch, 'pull')
613 cg = pullop.remote.changegroup(pullop.fetch, 'pull')
598 elif not pullop.remote.capable('changegroupsubset'):
614 elif not pullop.remote.capable('changegroupsubset'):
599 raise util.Abort(_("partial pull cannot be done because "
615 raise util.Abort(_("partial pull cannot be done because "
600 "other repository doesn't support "
616 "other repository doesn't support "
601 "changegroupsubset."))
617 "changegroupsubset."))
602 else:
618 else:
603 cg = pullop.remote.changegroupsubset(pullop.fetch, pullop.heads, 'pull')
619 cg = pullop.remote.changegroupsubset(pullop.fetch, pullop.heads, 'pull')
604 pullop.cgresult = changegroup.addchangegroup(pullop.repo, cg, 'pull',
620 pullop.cgresult = changegroup.addchangegroup(pullop.repo, cg, 'pull',
605 pullop.remote.url())
621 pullop.remote.url())
606
622
607 def _pullphase(pullop):
623 def _pullphase(pullop):
608 # Get remote phases data from remote
624 # Get remote phases data from remote
609 pullop.todosteps.remove('phases')
625 pullop.todosteps.remove('phases')
610 remotephases = pullop.remote.listkeys('phases')
626 remotephases = pullop.remote.listkeys('phases')
611 publishing = bool(remotephases.get('publishing', False))
627 publishing = bool(remotephases.get('publishing', False))
612 if remotephases and not publishing:
628 if remotephases and not publishing:
613 # remote is new and unpublishing
629 # remote is new and unpublishing
614 pheads, _dr = phases.analyzeremotephases(pullop.repo,
630 pheads, _dr = phases.analyzeremotephases(pullop.repo,
615 pullop.pulledsubset,
631 pullop.pulledsubset,
616 remotephases)
632 remotephases)
617 phases.advanceboundary(pullop.repo, phases.public, pheads)
633 phases.advanceboundary(pullop.repo, phases.public, pheads)
618 phases.advanceboundary(pullop.repo, phases.draft,
634 phases.advanceboundary(pullop.repo, phases.draft,
619 pullop.pulledsubset)
635 pullop.pulledsubset)
620 else:
636 else:
621 # Remote is old or publishing all common changesets
637 # Remote is old or publishing all common changesets
622 # should be seen as public
638 # should be seen as public
623 phases.advanceboundary(pullop.repo, phases.public,
639 phases.advanceboundary(pullop.repo, phases.public,
624 pullop.pulledsubset)
640 pullop.pulledsubset)
625
641
626 def _pullobsolete(pullop):
642 def _pullobsolete(pullop):
627 """utility function to pull obsolete markers from a remote
643 """utility function to pull obsolete markers from a remote
628
644
629 The `gettransaction` is function that return the pull transaction, creating
645 The `gettransaction` is function that return the pull transaction, creating
630 one if necessary. We return the transaction to inform the calling code that
646 one if necessary. We return the transaction to inform the calling code that
631 a new transaction have been created (when applicable).
647 a new transaction have been created (when applicable).
632
648
633 Exists mostly to allow overriding for experimentation purpose"""
649 Exists mostly to allow overriding for experimentation purpose"""
634 pullop.todosteps.remove('obsmarkers')
650 pullop.todosteps.remove('obsmarkers')
635 tr = None
651 tr = None
636 if obsolete._enabled:
652 if obsolete._enabled:
637 pullop.repo.ui.debug('fetching remote obsolete markers\n')
653 pullop.repo.ui.debug('fetching remote obsolete markers\n')
638 remoteobs = pullop.remote.listkeys('obsolete')
654 remoteobs = pullop.remote.listkeys('obsolete')
639 if 'dump0' in remoteobs:
655 if 'dump0' in remoteobs:
640 tr = pullop.gettransaction()
656 tr = pullop.gettransaction()
641 for key in sorted(remoteobs, reverse=True):
657 for key in sorted(remoteobs, reverse=True):
642 if key.startswith('dump'):
658 if key.startswith('dump'):
643 data = base85.b85decode(remoteobs[key])
659 data = base85.b85decode(remoteobs[key])
644 pullop.repo.obsstore.mergemarkers(tr, data)
660 pullop.repo.obsstore.mergemarkers(tr, data)
645 pullop.repo.invalidatevolatilesets()
661 pullop.repo.invalidatevolatilesets()
646 return tr
662 return tr
647
663
648 def getbundle(repo, source, heads=None, common=None, bundlecaps=None):
664 def getbundle(repo, source, heads=None, common=None, bundlecaps=None):
649 """return a full bundle (with potentially multiple kind of parts)
665 """return a full bundle (with potentially multiple kind of parts)
650
666
651 Could be a bundle HG10 or a bundle HG2X depending on bundlecaps
667 Could be a bundle HG10 or a bundle HG2X depending on bundlecaps
652 passed. For now, the bundle can contain only changegroup, but this will
668 passed. For now, the bundle can contain only changegroup, but this will
653 changes when more part type will be available for bundle2.
669 changes when more part type will be available for bundle2.
654
670
655 This is different from changegroup.getbundle that only returns an HG10
671 This is different from changegroup.getbundle that only returns an HG10
656 changegroup bundle. They may eventually get reunited in the future when we
672 changegroup bundle. They may eventually get reunited in the future when we
657 have a clearer idea of the API we what to query different data.
673 have a clearer idea of the API we what to query different data.
658
674
659 The implementation is at a very early stage and will get massive rework
675 The implementation is at a very early stage and will get massive rework
660 when the API of bundle is refined.
676 when the API of bundle is refined.
661 """
677 """
662 # build bundle here.
678 # build bundle here.
663 cg = changegroup.getbundle(repo, source, heads=heads,
679 cg = changegroup.getbundle(repo, source, heads=heads,
664 common=common, bundlecaps=bundlecaps)
680 common=common, bundlecaps=bundlecaps)
665 if bundlecaps is None or 'HG2X' not in bundlecaps:
681 if bundlecaps is None or 'HG2X' not in bundlecaps:
666 return cg
682 return cg
667 # very crude first implementation,
683 # very crude first implementation,
668 # the bundle API will change and the generation will be done lazily.
684 # the bundle API will change and the generation will be done lazily.
669 b2caps = {}
685 b2caps = {}
670 for bcaps in bundlecaps:
686 for bcaps in bundlecaps:
671 if bcaps.startswith('bundle2='):
687 if bcaps.startswith('bundle2='):
672 blob = urllib.unquote(bcaps[len('bundle2='):])
688 blob = urllib.unquote(bcaps[len('bundle2='):])
673 b2caps.update(bundle2.decodecaps(blob))
689 b2caps.update(bundle2.decodecaps(blob))
674 bundler = bundle2.bundle20(repo.ui, b2caps)
690 bundler = bundle2.bundle20(repo.ui, b2caps)
675 part = bundle2.bundlepart('b2x:changegroup', data=cg.getchunks())
691 part = bundle2.bundlepart('b2x:changegroup', data=cg.getchunks())
676 bundler.addpart(part)
692 bundler.addpart(part)
677 return util.chunkbuffer(bundler.getchunks())
693 return util.chunkbuffer(bundler.getchunks())
678
694
679 class PushRaced(RuntimeError):
695 class PushRaced(RuntimeError):
680 """An exception raised during unbundling that indicate a push race"""
696 """An exception raised during unbundling that indicate a push race"""
681
697
682 def check_heads(repo, their_heads, context):
698 def check_heads(repo, their_heads, context):
683 """check if the heads of a repo have been modified
699 """check if the heads of a repo have been modified
684
700
685 Used by peer for unbundling.
701 Used by peer for unbundling.
686 """
702 """
687 heads = repo.heads()
703 heads = repo.heads()
688 heads_hash = util.sha1(''.join(sorted(heads))).digest()
704 heads_hash = util.sha1(''.join(sorted(heads))).digest()
689 if not (their_heads == ['force'] or their_heads == heads or
705 if not (their_heads == ['force'] or their_heads == heads or
690 their_heads == ['hashed', heads_hash]):
706 their_heads == ['hashed', heads_hash]):
691 # someone else committed/pushed/unbundled while we
707 # someone else committed/pushed/unbundled while we
692 # were transferring data
708 # were transferring data
693 raise PushRaced('repository changed while %s - '
709 raise PushRaced('repository changed while %s - '
694 'please try again' % context)
710 'please try again' % context)
695
711
696 def unbundle(repo, cg, heads, source, url):
712 def unbundle(repo, cg, heads, source, url):
697 """Apply a bundle to a repo.
713 """Apply a bundle to a repo.
698
714
699 this function makes sure the repo is locked during the application and have
715 this function makes sure the repo is locked during the application and have
700 mechanism to check that no push race occurred between the creation of the
716 mechanism to check that no push race occurred between the creation of the
701 bundle and its application.
717 bundle and its application.
702
718
703 If the push was raced as PushRaced exception is raised."""
719 If the push was raced as PushRaced exception is raised."""
704 r = 0
720 r = 0
705 # need a transaction when processing a bundle2 stream
721 # need a transaction when processing a bundle2 stream
706 tr = None
722 tr = None
707 lock = repo.lock()
723 lock = repo.lock()
708 try:
724 try:
709 check_heads(repo, heads, 'uploading changes')
725 check_heads(repo, heads, 'uploading changes')
710 # push can proceed
726 # push can proceed
711 if util.safehasattr(cg, 'params'):
727 if util.safehasattr(cg, 'params'):
712 tr = repo.transaction('unbundle')
728 tr = repo.transaction('unbundle')
713 r = bundle2.processbundle(repo, cg, lambda: tr).reply
729 r = bundle2.processbundle(repo, cg, lambda: tr).reply
714 tr.close()
730 tr.close()
715 else:
731 else:
716 r = changegroup.addchangegroup(repo, cg, source, url)
732 r = changegroup.addchangegroup(repo, cg, source, url)
717 finally:
733 finally:
718 if tr is not None:
734 if tr is not None:
719 tr.release()
735 tr.release()
720 lock.release()
736 lock.release()
721 return r
737 return r
General Comments 0
You need to be logged in to leave comments. Login now