##// END OF EJS Templates
bundle2: handle compression in _forwardchunks...
Joerg Sonnenberger -
r42319:29569f2d default
parent child Browse files
Show More
@@ -1,2326 +1,2335 b''
1 1 # bundle2.py - generic container format to transmit arbitrary data.
2 2 #
3 3 # Copyright 2013 Facebook, Inc.
4 4 #
5 5 # This software may be used and distributed according to the terms of the
6 6 # GNU General Public License version 2 or any later version.
7 7 """Handling of the new bundle2 format
8 8
9 9 The goal of bundle2 is to act as an atomically packet to transmit a set of
10 10 payloads in an application agnostic way. It consist in a sequence of "parts"
11 11 that will be handed to and processed by the application layer.
12 12
13 13
14 14 General format architecture
15 15 ===========================
16 16
17 17 The format is architectured as follow
18 18
19 19 - magic string
20 20 - stream level parameters
21 21 - payload parts (any number)
22 22 - end of stream marker.
23 23
24 24 the Binary format
25 25 ============================
26 26
27 27 All numbers are unsigned and big-endian.
28 28
29 29 stream level parameters
30 30 ------------------------
31 31
32 32 Binary format is as follow
33 33
34 34 :params size: int32
35 35
36 36 The total number of Bytes used by the parameters
37 37
38 38 :params value: arbitrary number of Bytes
39 39
40 40 A blob of `params size` containing the serialized version of all stream level
41 41 parameters.
42 42
43 43 The blob contains a space separated list of parameters. Parameters with value
44 44 are stored in the form `<name>=<value>`. Both name and value are urlquoted.
45 45
46 46 Empty name are obviously forbidden.
47 47
48 48 Name MUST start with a letter. If this first letter is lower case, the
49 49 parameter is advisory and can be safely ignored. However when the first
50 50 letter is capital, the parameter is mandatory and the bundling process MUST
51 51 stop if he is not able to proceed it.
52 52
53 53 Stream parameters use a simple textual format for two main reasons:
54 54
55 55 - Stream level parameters should remain simple and we want to discourage any
56 56 crazy usage.
57 57 - Textual data allow easy human inspection of a bundle2 header in case of
58 58 troubles.
59 59
60 60 Any Applicative level options MUST go into a bundle2 part instead.
61 61
62 62 Payload part
63 63 ------------------------
64 64
65 65 Binary format is as follow
66 66
67 67 :header size: int32
68 68
69 69 The total number of Bytes used by the part header. When the header is empty
70 70 (size = 0) this is interpreted as the end of stream marker.
71 71
72 72 :header:
73 73
74 74 The header defines how to interpret the part. It contains two piece of
75 75 data: the part type, and the part parameters.
76 76
77 77 The part type is used to route an application level handler, that can
78 78 interpret payload.
79 79
80 80 Part parameters are passed to the application level handler. They are
81 81 meant to convey information that will help the application level object to
82 82 interpret the part payload.
83 83
84 84 The binary format of the header is has follow
85 85
86 86 :typesize: (one byte)
87 87
88 88 :parttype: alphanumerical part name (restricted to [a-zA-Z0-9_:-]*)
89 89
90 90 :partid: A 32bits integer (unique in the bundle) that can be used to refer
91 91 to this part.
92 92
93 93 :parameters:
94 94
95 95 Part's parameter may have arbitrary content, the binary structure is::
96 96
97 97 <mandatory-count><advisory-count><param-sizes><param-data>
98 98
99 99 :mandatory-count: 1 byte, number of mandatory parameters
100 100
101 101 :advisory-count: 1 byte, number of advisory parameters
102 102
103 103 :param-sizes:
104 104
105 105 N couple of bytes, where N is the total number of parameters. Each
106 106 couple contains (<size-of-key>, <size-of-value) for one parameter.
107 107
108 108 :param-data:
109 109
110 110 A blob of bytes from which each parameter key and value can be
111 111 retrieved using the list of size couples stored in the previous
112 112 field.
113 113
114 114 Mandatory parameters comes first, then the advisory ones.
115 115
116 116 Each parameter's key MUST be unique within the part.
117 117
118 118 :payload:
119 119
120 120 payload is a series of `<chunksize><chunkdata>`.
121 121
122 122 `chunksize` is an int32, `chunkdata` are plain bytes (as much as
123 123 `chunksize` says)` The payload part is concluded by a zero size chunk.
124 124
125 125 The current implementation always produces either zero or one chunk.
126 126 This is an implementation limitation that will ultimately be lifted.
127 127
128 128 `chunksize` can be negative to trigger special case processing. No such
129 129 processing is in place yet.
130 130
131 131 Bundle processing
132 132 ============================
133 133
134 134 Each part is processed in order using a "part handler". Handler are registered
135 135 for a certain part type.
136 136
137 137 The matching of a part to its handler is case insensitive. The case of the
138 138 part type is used to know if a part is mandatory or advisory. If the Part type
139 139 contains any uppercase char it is considered mandatory. When no handler is
140 140 known for a Mandatory part, the process is aborted and an exception is raised.
141 141 If the part is advisory and no handler is known, the part is ignored. When the
142 142 process is aborted, the full bundle is still read from the stream to keep the
143 143 channel usable. But none of the part read from an abort are processed. In the
144 144 future, dropping the stream may become an option for channel we do not care to
145 145 preserve.
146 146 """
147 147
148 148 from __future__ import absolute_import, division
149 149
150 150 import collections
151 151 import errno
152 152 import os
153 153 import re
154 154 import string
155 155 import struct
156 156 import sys
157 157
158 158 from .i18n import _
159 159 from . import (
160 160 bookmarks,
161 161 changegroup,
162 162 encoding,
163 163 error,
164 164 node as nodemod,
165 165 obsolete,
166 166 phases,
167 167 pushkey,
168 168 pycompat,
169 169 streamclone,
170 170 tags,
171 171 url,
172 172 util,
173 173 )
174 174 from .utils import (
175 175 stringutil,
176 176 )
177 177
178 178 urlerr = util.urlerr
179 179 urlreq = util.urlreq
180 180
181 181 _pack = struct.pack
182 182 _unpack = struct.unpack
183 183
184 184 _fstreamparamsize = '>i'
185 185 _fpartheadersize = '>i'
186 186 _fparttypesize = '>B'
187 187 _fpartid = '>I'
188 188 _fpayloadsize = '>i'
189 189 _fpartparamcount = '>BB'
190 190
191 191 preferedchunksize = 32768
192 192
193 193 _parttypeforbidden = re.compile('[^a-zA-Z0-9_:-]')
194 194
195 195 def outdebug(ui, message):
196 196 """debug regarding output stream (bundling)"""
197 197 if ui.configbool('devel', 'bundle2.debug'):
198 198 ui.debug('bundle2-output: %s\n' % message)
199 199
200 200 def indebug(ui, message):
201 201 """debug on input stream (unbundling)"""
202 202 if ui.configbool('devel', 'bundle2.debug'):
203 203 ui.debug('bundle2-input: %s\n' % message)
204 204
205 205 def validateparttype(parttype):
206 206 """raise ValueError if a parttype contains invalid character"""
207 207 if _parttypeforbidden.search(parttype):
208 208 raise ValueError(parttype)
209 209
210 210 def _makefpartparamsizes(nbparams):
211 211 """return a struct format to read part parameter sizes
212 212
213 213 The number parameters is variable so we need to build that format
214 214 dynamically.
215 215 """
216 216 return '>'+('BB'*nbparams)
217 217
218 218 parthandlermapping = {}
219 219
220 220 def parthandler(parttype, params=()):
221 221 """decorator that register a function as a bundle2 part handler
222 222
223 223 eg::
224 224
225 225 @parthandler('myparttype', ('mandatory', 'param', 'handled'))
226 226 def myparttypehandler(...):
227 227 '''process a part of type "my part".'''
228 228 ...
229 229 """
230 230 validateparttype(parttype)
231 231 def _decorator(func):
232 232 lparttype = parttype.lower() # enforce lower case matching.
233 233 assert lparttype not in parthandlermapping
234 234 parthandlermapping[lparttype] = func
235 235 func.params = frozenset(params)
236 236 return func
237 237 return _decorator
238 238
239 239 class unbundlerecords(object):
240 240 """keep record of what happens during and unbundle
241 241
242 242 New records are added using `records.add('cat', obj)`. Where 'cat' is a
243 243 category of record and obj is an arbitrary object.
244 244
245 245 `records['cat']` will return all entries of this category 'cat'.
246 246
247 247 Iterating on the object itself will yield `('category', obj)` tuples
248 248 for all entries.
249 249
250 250 All iterations happens in chronological order.
251 251 """
252 252
253 253 def __init__(self):
254 254 self._categories = {}
255 255 self._sequences = []
256 256 self._replies = {}
257 257
258 258 def add(self, category, entry, inreplyto=None):
259 259 """add a new record of a given category.
260 260
261 261 The entry can then be retrieved in the list returned by
262 262 self['category']."""
263 263 self._categories.setdefault(category, []).append(entry)
264 264 self._sequences.append((category, entry))
265 265 if inreplyto is not None:
266 266 self.getreplies(inreplyto).add(category, entry)
267 267
268 268 def getreplies(self, partid):
269 269 """get the records that are replies to a specific part"""
270 270 return self._replies.setdefault(partid, unbundlerecords())
271 271
272 272 def __getitem__(self, cat):
273 273 return tuple(self._categories.get(cat, ()))
274 274
275 275 def __iter__(self):
276 276 return iter(self._sequences)
277 277
278 278 def __len__(self):
279 279 return len(self._sequences)
280 280
281 281 def __nonzero__(self):
282 282 return bool(self._sequences)
283 283
284 284 __bool__ = __nonzero__
285 285
286 286 class bundleoperation(object):
287 287 """an object that represents a single bundling process
288 288
289 289 Its purpose is to carry unbundle-related objects and states.
290 290
291 291 A new object should be created at the beginning of each bundle processing.
292 292 The object is to be returned by the processing function.
293 293
294 294 The object has very little content now it will ultimately contain:
295 295 * an access to the repo the bundle is applied to,
296 296 * a ui object,
297 297 * a way to retrieve a transaction to add changes to the repo,
298 298 * a way to record the result of processing each part,
299 299 * a way to construct a bundle response when applicable.
300 300 """
301 301
302 302 def __init__(self, repo, transactiongetter, captureoutput=True, source=''):
303 303 self.repo = repo
304 304 self.ui = repo.ui
305 305 self.records = unbundlerecords()
306 306 self.reply = None
307 307 self.captureoutput = captureoutput
308 308 self.hookargs = {}
309 309 self._gettransaction = transactiongetter
310 310 # carries value that can modify part behavior
311 311 self.modes = {}
312 312 self.source = source
313 313
314 314 def gettransaction(self):
315 315 transaction = self._gettransaction()
316 316
317 317 if self.hookargs:
318 318 # the ones added to the transaction supercede those added
319 319 # to the operation.
320 320 self.hookargs.update(transaction.hookargs)
321 321 transaction.hookargs = self.hookargs
322 322
323 323 # mark the hookargs as flushed. further attempts to add to
324 324 # hookargs will result in an abort.
325 325 self.hookargs = None
326 326
327 327 return transaction
328 328
329 329 def addhookargs(self, hookargs):
330 330 if self.hookargs is None:
331 331 raise error.ProgrammingError('attempted to add hookargs to '
332 332 'operation after transaction started')
333 333 self.hookargs.update(hookargs)
334 334
335 335 class TransactionUnavailable(RuntimeError):
336 336 pass
337 337
338 338 def _notransaction():
339 339 """default method to get a transaction while processing a bundle
340 340
341 341 Raise an exception to highlight the fact that no transaction was expected
342 342 to be created"""
343 343 raise TransactionUnavailable()
344 344
345 345 def applybundle(repo, unbundler, tr, source, url=None, **kwargs):
346 346 # transform me into unbundler.apply() as soon as the freeze is lifted
347 347 if isinstance(unbundler, unbundle20):
348 348 tr.hookargs['bundle2'] = '1'
349 349 if source is not None and 'source' not in tr.hookargs:
350 350 tr.hookargs['source'] = source
351 351 if url is not None and 'url' not in tr.hookargs:
352 352 tr.hookargs['url'] = url
353 353 return processbundle(repo, unbundler, lambda: tr, source=source)
354 354 else:
355 355 # the transactiongetter won't be used, but we might as well set it
356 356 op = bundleoperation(repo, lambda: tr, source=source)
357 357 _processchangegroup(op, unbundler, tr, source, url, **kwargs)
358 358 return op
359 359
360 360 class partiterator(object):
361 361 def __init__(self, repo, op, unbundler):
362 362 self.repo = repo
363 363 self.op = op
364 364 self.unbundler = unbundler
365 365 self.iterator = None
366 366 self.count = 0
367 367 self.current = None
368 368
369 369 def __enter__(self):
370 370 def func():
371 371 itr = enumerate(self.unbundler.iterparts())
372 372 for count, p in itr:
373 373 self.count = count
374 374 self.current = p
375 375 yield p
376 376 p.consume()
377 377 self.current = None
378 378 self.iterator = func()
379 379 return self.iterator
380 380
381 381 def __exit__(self, type, exc, tb):
382 382 if not self.iterator:
383 383 return
384 384
385 385 # Only gracefully abort in a normal exception situation. User aborts
386 386 # like Ctrl+C throw a KeyboardInterrupt which is not a base Exception,
387 387 # and should not gracefully cleanup.
388 388 if isinstance(exc, Exception):
389 389 # Any exceptions seeking to the end of the bundle at this point are
390 390 # almost certainly related to the underlying stream being bad.
391 391 # And, chances are that the exception we're handling is related to
392 392 # getting in that bad state. So, we swallow the seeking error and
393 393 # re-raise the original error.
394 394 seekerror = False
395 395 try:
396 396 if self.current:
397 397 # consume the part content to not corrupt the stream.
398 398 self.current.consume()
399 399
400 400 for part in self.iterator:
401 401 # consume the bundle content
402 402 part.consume()
403 403 except Exception:
404 404 seekerror = True
405 405
406 406 # Small hack to let caller code distinguish exceptions from bundle2
407 407 # processing from processing the old format. This is mostly needed
408 408 # to handle different return codes to unbundle according to the type
409 409 # of bundle. We should probably clean up or drop this return code
410 410 # craziness in a future version.
411 411 exc.duringunbundle2 = True
412 412 salvaged = []
413 413 replycaps = None
414 414 if self.op.reply is not None:
415 415 salvaged = self.op.reply.salvageoutput()
416 416 replycaps = self.op.reply.capabilities
417 417 exc._replycaps = replycaps
418 418 exc._bundle2salvagedoutput = salvaged
419 419
420 420 # Re-raising from a variable loses the original stack. So only use
421 421 # that form if we need to.
422 422 if seekerror:
423 423 raise exc
424 424
425 425 self.repo.ui.debug('bundle2-input-bundle: %i parts total\n' %
426 426 self.count)
427 427
428 428 def processbundle(repo, unbundler, transactiongetter=None, op=None, source=''):
429 429 """This function process a bundle, apply effect to/from a repo
430 430
431 431 It iterates over each part then searches for and uses the proper handling
432 432 code to process the part. Parts are processed in order.
433 433
434 434 Unknown Mandatory part will abort the process.
435 435
436 436 It is temporarily possible to provide a prebuilt bundleoperation to the
437 437 function. This is used to ensure output is properly propagated in case of
438 438 an error during the unbundling. This output capturing part will likely be
439 439 reworked and this ability will probably go away in the process.
440 440 """
441 441 if op is None:
442 442 if transactiongetter is None:
443 443 transactiongetter = _notransaction
444 444 op = bundleoperation(repo, transactiongetter, source=source)
445 445 # todo:
446 446 # - replace this is a init function soon.
447 447 # - exception catching
448 448 unbundler.params
449 449 if repo.ui.debugflag:
450 450 msg = ['bundle2-input-bundle:']
451 451 if unbundler.params:
452 452 msg.append(' %i params' % len(unbundler.params))
453 453 if op._gettransaction is None or op._gettransaction is _notransaction:
454 454 msg.append(' no-transaction')
455 455 else:
456 456 msg.append(' with-transaction')
457 457 msg.append('\n')
458 458 repo.ui.debug(''.join(msg))
459 459
460 460 processparts(repo, op, unbundler)
461 461
462 462 return op
463 463
464 464 def processparts(repo, op, unbundler):
465 465 with partiterator(repo, op, unbundler) as parts:
466 466 for part in parts:
467 467 _processpart(op, part)
468 468
469 469 def _processchangegroup(op, cg, tr, source, url, **kwargs):
470 470 ret = cg.apply(op.repo, tr, source, url, **kwargs)
471 471 op.records.add('changegroup', {
472 472 'return': ret,
473 473 })
474 474 return ret
475 475
476 476 def _gethandler(op, part):
477 477 status = 'unknown' # used by debug output
478 478 try:
479 479 handler = parthandlermapping.get(part.type)
480 480 if handler is None:
481 481 status = 'unsupported-type'
482 482 raise error.BundleUnknownFeatureError(parttype=part.type)
483 483 indebug(op.ui, 'found a handler for part %s' % part.type)
484 484 unknownparams = part.mandatorykeys - handler.params
485 485 if unknownparams:
486 486 unknownparams = list(unknownparams)
487 487 unknownparams.sort()
488 488 status = 'unsupported-params (%s)' % ', '.join(unknownparams)
489 489 raise error.BundleUnknownFeatureError(parttype=part.type,
490 490 params=unknownparams)
491 491 status = 'supported'
492 492 except error.BundleUnknownFeatureError as exc:
493 493 if part.mandatory: # mandatory parts
494 494 raise
495 495 indebug(op.ui, 'ignoring unsupported advisory part %s' % exc)
496 496 return # skip to part processing
497 497 finally:
498 498 if op.ui.debugflag:
499 499 msg = ['bundle2-input-part: "%s"' % part.type]
500 500 if not part.mandatory:
501 501 msg.append(' (advisory)')
502 502 nbmp = len(part.mandatorykeys)
503 503 nbap = len(part.params) - nbmp
504 504 if nbmp or nbap:
505 505 msg.append(' (params:')
506 506 if nbmp:
507 507 msg.append(' %i mandatory' % nbmp)
508 508 if nbap:
509 509 msg.append(' %i advisory' % nbmp)
510 510 msg.append(')')
511 511 msg.append(' %s\n' % status)
512 512 op.ui.debug(''.join(msg))
513 513
514 514 return handler
515 515
516 516 def _processpart(op, part):
517 517 """process a single part from a bundle
518 518
519 519 The part is guaranteed to have been fully consumed when the function exits
520 520 (even if an exception is raised)."""
521 521 handler = _gethandler(op, part)
522 522 if handler is None:
523 523 return
524 524
525 525 # handler is called outside the above try block so that we don't
526 526 # risk catching KeyErrors from anything other than the
527 527 # parthandlermapping lookup (any KeyError raised by handler()
528 528 # itself represents a defect of a different variety).
529 529 output = None
530 530 if op.captureoutput and op.reply is not None:
531 531 op.ui.pushbuffer(error=True, subproc=True)
532 532 output = ''
533 533 try:
534 534 handler(op, part)
535 535 finally:
536 536 if output is not None:
537 537 output = op.ui.popbuffer()
538 538 if output:
539 539 outpart = op.reply.newpart('output', data=output,
540 540 mandatory=False)
541 541 outpart.addparam(
542 542 'in-reply-to', pycompat.bytestr(part.id), mandatory=False)
543 543
544 544 def decodecaps(blob):
545 545 """decode a bundle2 caps bytes blob into a dictionary
546 546
547 547 The blob is a list of capabilities (one per line)
548 548 Capabilities may have values using a line of the form::
549 549
550 550 capability=value1,value2,value3
551 551
552 552 The values are always a list."""
553 553 caps = {}
554 554 for line in blob.splitlines():
555 555 if not line:
556 556 continue
557 557 if '=' not in line:
558 558 key, vals = line, ()
559 559 else:
560 560 key, vals = line.split('=', 1)
561 561 vals = vals.split(',')
562 562 key = urlreq.unquote(key)
563 563 vals = [urlreq.unquote(v) for v in vals]
564 564 caps[key] = vals
565 565 return caps
566 566
567 567 def encodecaps(caps):
568 568 """encode a bundle2 caps dictionary into a bytes blob"""
569 569 chunks = []
570 570 for ca in sorted(caps):
571 571 vals = caps[ca]
572 572 ca = urlreq.quote(ca)
573 573 vals = [urlreq.quote(v) for v in vals]
574 574 if vals:
575 575 ca = "%s=%s" % (ca, ','.join(vals))
576 576 chunks.append(ca)
577 577 return '\n'.join(chunks)
578 578
579 579 bundletypes = {
580 580 "": ("", 'UN'), # only when using unbundle on ssh and old http servers
581 581 # since the unification ssh accepts a header but there
582 582 # is no capability signaling it.
583 583 "HG20": (), # special-cased below
584 584 "HG10UN": ("HG10UN", 'UN'),
585 585 "HG10BZ": ("HG10", 'BZ'),
586 586 "HG10GZ": ("HG10GZ", 'GZ'),
587 587 }
588 588
589 589 # hgweb uses this list to communicate its preferred type
590 590 bundlepriority = ['HG10GZ', 'HG10BZ', 'HG10UN']
591 591
592 592 class bundle20(object):
593 593 """represent an outgoing bundle2 container
594 594
595 595 Use the `addparam` method to add stream level parameter. and `newpart` to
596 596 populate it. Then call `getchunks` to retrieve all the binary chunks of
597 597 data that compose the bundle2 container."""
598 598
599 599 _magicstring = 'HG20'
600 600
601 601 def __init__(self, ui, capabilities=()):
602 602 self.ui = ui
603 603 self._params = []
604 604 self._parts = []
605 605 self.capabilities = dict(capabilities)
606 606 self._compengine = util.compengines.forbundletype('UN')
607 607 self._compopts = None
608 608 # If compression is being handled by a consumer of the raw
609 609 # data (e.g. the wire protocol), unsetting this flag tells
610 610 # consumers that the bundle is best left uncompressed.
611 611 self.prefercompressed = True
612 612
613 613 def setcompression(self, alg, compopts=None):
614 614 """setup core part compression to <alg>"""
615 615 if alg in (None, 'UN'):
616 616 return
617 617 assert not any(n.lower() == 'compression' for n, v in self._params)
618 618 self.addparam('Compression', alg)
619 619 self._compengine = util.compengines.forbundletype(alg)
620 620 self._compopts = compopts
621 621
622 622 @property
623 623 def nbparts(self):
624 624 """total number of parts added to the bundler"""
625 625 return len(self._parts)
626 626
627 627 # methods used to defines the bundle2 content
628 628 def addparam(self, name, value=None):
629 629 """add a stream level parameter"""
630 630 if not name:
631 631 raise error.ProgrammingError(b'empty parameter name')
632 632 if name[0:1] not in pycompat.bytestr(string.ascii_letters):
633 633 raise error.ProgrammingError(b'non letter first character: %s'
634 634 % name)
635 635 self._params.append((name, value))
636 636
637 637 def addpart(self, part):
638 638 """add a new part to the bundle2 container
639 639
640 640 Parts contains the actual applicative payload."""
641 641 assert part.id is None
642 642 part.id = len(self._parts) # very cheap counter
643 643 self._parts.append(part)
644 644
645 645 def newpart(self, typeid, *args, **kwargs):
646 646 """create a new part and add it to the containers
647 647
648 648 As the part is directly added to the containers. For now, this means
649 649 that any failure to properly initialize the part after calling
650 650 ``newpart`` should result in a failure of the whole bundling process.
651 651
652 652 You can still fall back to manually create and add if you need better
653 653 control."""
654 654 part = bundlepart(typeid, *args, **kwargs)
655 655 self.addpart(part)
656 656 return part
657 657
658 658 # methods used to generate the bundle2 stream
659 659 def getchunks(self):
660 660 if self.ui.debugflag:
661 661 msg = ['bundle2-output-bundle: "%s",' % self._magicstring]
662 662 if self._params:
663 663 msg.append(' (%i params)' % len(self._params))
664 664 msg.append(' %i parts total\n' % len(self._parts))
665 665 self.ui.debug(''.join(msg))
666 666 outdebug(self.ui, 'start emission of %s stream' % self._magicstring)
667 667 yield self._magicstring
668 668 param = self._paramchunk()
669 669 outdebug(self.ui, 'bundle parameter: %s' % param)
670 670 yield _pack(_fstreamparamsize, len(param))
671 671 if param:
672 672 yield param
673 673 for chunk in self._compengine.compressstream(self._getcorechunk(),
674 674 self._compopts):
675 675 yield chunk
676 676
677 677 def _paramchunk(self):
678 678 """return a encoded version of all stream parameters"""
679 679 blocks = []
680 680 for par, value in self._params:
681 681 par = urlreq.quote(par)
682 682 if value is not None:
683 683 value = urlreq.quote(value)
684 684 par = '%s=%s' % (par, value)
685 685 blocks.append(par)
686 686 return ' '.join(blocks)
687 687
688 688 def _getcorechunk(self):
689 689 """yield chunk for the core part of the bundle
690 690
691 691 (all but headers and parameters)"""
692 692 outdebug(self.ui, 'start of parts')
693 693 for part in self._parts:
694 694 outdebug(self.ui, 'bundle part: "%s"' % part.type)
695 695 for chunk in part.getchunks(ui=self.ui):
696 696 yield chunk
697 697 outdebug(self.ui, 'end of bundle')
698 698 yield _pack(_fpartheadersize, 0)
699 699
700 700
701 701 def salvageoutput(self):
702 702 """return a list with a copy of all output parts in the bundle
703 703
704 704 This is meant to be used during error handling to make sure we preserve
705 705 server output"""
706 706 salvaged = []
707 707 for part in self._parts:
708 708 if part.type.startswith('output'):
709 709 salvaged.append(part.copy())
710 710 return salvaged
711 711
712 712
713 713 class unpackermixin(object):
714 714 """A mixin to extract bytes and struct data from a stream"""
715 715
716 716 def __init__(self, fp):
717 717 self._fp = fp
718 718
719 719 def _unpack(self, format):
720 720 """unpack this struct format from the stream
721 721
722 722 This method is meant for internal usage by the bundle2 protocol only.
723 723 They directly manipulate the low level stream including bundle2 level
724 724 instruction.
725 725
726 726 Do not use it to implement higher-level logic or methods."""
727 727 data = self._readexact(struct.calcsize(format))
728 728 return _unpack(format, data)
729 729
730 730 def _readexact(self, size):
731 731 """read exactly <size> bytes from the stream
732 732
733 733 This method is meant for internal usage by the bundle2 protocol only.
734 734 They directly manipulate the low level stream including bundle2 level
735 735 instruction.
736 736
737 737 Do not use it to implement higher-level logic or methods."""
738 738 return changegroup.readexactly(self._fp, size)
739 739
740 740 def getunbundler(ui, fp, magicstring=None):
741 741 """return a valid unbundler object for a given magicstring"""
742 742 if magicstring is None:
743 743 magicstring = changegroup.readexactly(fp, 4)
744 744 magic, version = magicstring[0:2], magicstring[2:4]
745 745 if magic != 'HG':
746 746 ui.debug(
747 747 "error: invalid magic: %r (version %r), should be 'HG'\n"
748 748 % (magic, version))
749 749 raise error.Abort(_('not a Mercurial bundle'))
750 750 unbundlerclass = formatmap.get(version)
751 751 if unbundlerclass is None:
752 752 raise error.Abort(_('unknown bundle version %s') % version)
753 753 unbundler = unbundlerclass(ui, fp)
754 754 indebug(ui, 'start processing of %s stream' % magicstring)
755 755 return unbundler
756 756
757 757 class unbundle20(unpackermixin):
758 758 """interpret a bundle2 stream
759 759
760 760 This class is fed with a binary stream and yields parts through its
761 761 `iterparts` methods."""
762 762
763 763 _magicstring = 'HG20'
764 764
765 765 def __init__(self, ui, fp):
766 766 """If header is specified, we do not read it out of the stream."""
767 767 self.ui = ui
768 768 self._compengine = util.compengines.forbundletype('UN')
769 769 self._compressed = None
770 770 super(unbundle20, self).__init__(fp)
771 771
772 772 @util.propertycache
773 773 def params(self):
774 774 """dictionary of stream level parameters"""
775 775 indebug(self.ui, 'reading bundle2 stream parameters')
776 776 params = {}
777 777 paramssize = self._unpack(_fstreamparamsize)[0]
778 778 if paramssize < 0:
779 779 raise error.BundleValueError('negative bundle param size: %i'
780 780 % paramssize)
781 781 if paramssize:
782 782 params = self._readexact(paramssize)
783 783 params = self._processallparams(params)
784 784 return params
785 785
786 786 def _processallparams(self, paramsblock):
787 787 """"""
788 788 params = util.sortdict()
789 789 for p in paramsblock.split(' '):
790 790 p = p.split('=', 1)
791 791 p = [urlreq.unquote(i) for i in p]
792 792 if len(p) < 2:
793 793 p.append(None)
794 794 self._processparam(*p)
795 795 params[p[0]] = p[1]
796 796 return params
797 797
798 798
799 799 def _processparam(self, name, value):
800 800 """process a parameter, applying its effect if needed
801 801
802 802 Parameter starting with a lower case letter are advisory and will be
803 803 ignored when unknown. Those starting with an upper case letter are
804 804 mandatory and will this function will raise a KeyError when unknown.
805 805
806 806 Note: no option are currently supported. Any input will be either
807 807 ignored or failing.
808 808 """
809 809 if not name:
810 810 raise ValueError(r'empty parameter name')
811 811 if name[0:1] not in pycompat.bytestr(string.ascii_letters):
812 812 raise ValueError(r'non letter first character: %s' % name)
813 813 try:
814 814 handler = b2streamparamsmap[name.lower()]
815 815 except KeyError:
816 816 if name[0:1].islower():
817 817 indebug(self.ui, "ignoring unknown parameter %s" % name)
818 818 else:
819 819 raise error.BundleUnknownFeatureError(params=(name,))
820 820 else:
821 821 handler(self, name, value)
822 822
823 823 def _forwardchunks(self):
824 824 """utility to transfer a bundle2 as binary
825 825
826 826 This is made necessary by the fact the 'getbundle' command over 'ssh'
827 827 have no way to know then the reply end, relying on the bundle to be
828 828 interpreted to know its end. This is terrible and we are sorry, but we
829 829 needed to move forward to get general delta enabled.
830 830 """
831 831 yield self._magicstring
832 832 assert 'params' not in vars(self)
833 833 paramssize = self._unpack(_fstreamparamsize)[0]
834 834 if paramssize < 0:
835 835 raise error.BundleValueError('negative bundle param size: %i'
836 836 % paramssize)
837 yield _pack(_fstreamparamsize, paramssize)
838 837 if paramssize:
839 838 params = self._readexact(paramssize)
840 839 self._processallparams(params)
841 yield params
842 assert self._compengine.bundletype()[1] == 'UN'
840 # The payload itself is decompressed below, so drop
841 # the compression parameter passed down to compensate.
842 outparams = []
843 for p in params.split(' '):
844 k, v = p.split('=', 1)
845 if k.lower() != 'compression':
846 outparams.append(p)
847 outparams = ' '.join(outparams)
848 yield _pack(_fstreamparamsize, len(outparams))
849 yield outparams
850 else:
851 yield _pack(_fstreamparamsize, paramssize)
843 852 # From there, payload might need to be decompressed
844 853 self._fp = self._compengine.decompressorreader(self._fp)
845 854 emptycount = 0
846 855 while emptycount < 2:
847 856 # so we can brainlessly loop
848 857 assert _fpartheadersize == _fpayloadsize
849 858 size = self._unpack(_fpartheadersize)[0]
850 859 yield _pack(_fpartheadersize, size)
851 860 if size:
852 861 emptycount = 0
853 862 else:
854 863 emptycount += 1
855 864 continue
856 865 if size == flaginterrupt:
857 866 continue
858 867 elif size < 0:
859 868 raise error.BundleValueError('negative chunk size: %i')
860 869 yield self._readexact(size)
861 870
862 871
863 872 def iterparts(self, seekable=False):
864 873 """yield all parts contained in the stream"""
865 874 cls = seekableunbundlepart if seekable else unbundlepart
866 875 # make sure param have been loaded
867 876 self.params
868 877 # From there, payload need to be decompressed
869 878 self._fp = self._compengine.decompressorreader(self._fp)
870 879 indebug(self.ui, 'start extraction of bundle2 parts')
871 880 headerblock = self._readpartheader()
872 881 while headerblock is not None:
873 882 part = cls(self.ui, headerblock, self._fp)
874 883 yield part
875 884 # Ensure part is fully consumed so we can start reading the next
876 885 # part.
877 886 part.consume()
878 887
879 888 headerblock = self._readpartheader()
880 889 indebug(self.ui, 'end of bundle2 stream')
881 890
882 891 def _readpartheader(self):
883 892 """reads a part header size and return the bytes blob
884 893
885 894 returns None if empty"""
886 895 headersize = self._unpack(_fpartheadersize)[0]
887 896 if headersize < 0:
888 897 raise error.BundleValueError('negative part header size: %i'
889 898 % headersize)
890 899 indebug(self.ui, 'part header size: %i' % headersize)
891 900 if headersize:
892 901 return self._readexact(headersize)
893 902 return None
894 903
895 904 def compressed(self):
896 905 self.params # load params
897 906 return self._compressed
898 907
899 908 def close(self):
900 909 """close underlying file"""
901 910 if util.safehasattr(self._fp, 'close'):
902 911 return self._fp.close()
903 912
904 913 formatmap = {'20': unbundle20}
905 914
906 915 b2streamparamsmap = {}
907 916
908 917 def b2streamparamhandler(name):
909 918 """register a handler for a stream level parameter"""
910 919 def decorator(func):
911 920 assert name not in formatmap
912 921 b2streamparamsmap[name] = func
913 922 return func
914 923 return decorator
915 924
916 925 @b2streamparamhandler('compression')
917 926 def processcompression(unbundler, param, value):
918 927 """read compression parameter and install payload decompression"""
919 928 if value not in util.compengines.supportedbundletypes:
920 929 raise error.BundleUnknownFeatureError(params=(param,),
921 930 values=(value,))
922 931 unbundler._compengine = util.compengines.forbundletype(value)
923 932 if value is not None:
924 933 unbundler._compressed = True
925 934
926 935 class bundlepart(object):
927 936 """A bundle2 part contains application level payload
928 937
929 938 The part `type` is used to route the part to the application level
930 939 handler.
931 940
932 941 The part payload is contained in ``part.data``. It could be raw bytes or a
933 942 generator of byte chunks.
934 943
935 944 You can add parameters to the part using the ``addparam`` method.
936 945 Parameters can be either mandatory (default) or advisory. Remote side
937 946 should be able to safely ignore the advisory ones.
938 947
939 948 Both data and parameters cannot be modified after the generation has begun.
940 949 """
941 950
942 951 def __init__(self, parttype, mandatoryparams=(), advisoryparams=(),
943 952 data='', mandatory=True):
944 953 validateparttype(parttype)
945 954 self.id = None
946 955 self.type = parttype
947 956 self._data = data
948 957 self._mandatoryparams = list(mandatoryparams)
949 958 self._advisoryparams = list(advisoryparams)
950 959 # checking for duplicated entries
951 960 self._seenparams = set()
952 961 for pname, __ in self._mandatoryparams + self._advisoryparams:
953 962 if pname in self._seenparams:
954 963 raise error.ProgrammingError('duplicated params: %s' % pname)
955 964 self._seenparams.add(pname)
956 965 # status of the part's generation:
957 966 # - None: not started,
958 967 # - False: currently generated,
959 968 # - True: generation done.
960 969 self._generated = None
961 970 self.mandatory = mandatory
962 971
963 972 def __repr__(self):
964 973 cls = "%s.%s" % (self.__class__.__module__, self.__class__.__name__)
965 974 return ('<%s object at %x; id: %s; type: %s; mandatory: %s>'
966 975 % (cls, id(self), self.id, self.type, self.mandatory))
967 976
968 977 def copy(self):
969 978 """return a copy of the part
970 979
971 980 The new part have the very same content but no partid assigned yet.
972 981 Parts with generated data cannot be copied."""
973 982 assert not util.safehasattr(self.data, 'next')
974 983 return self.__class__(self.type, self._mandatoryparams,
975 984 self._advisoryparams, self._data, self.mandatory)
976 985
977 986 # methods used to defines the part content
978 987 @property
979 988 def data(self):
980 989 return self._data
981 990
982 991 @data.setter
983 992 def data(self, data):
984 993 if self._generated is not None:
985 994 raise error.ReadOnlyPartError('part is being generated')
986 995 self._data = data
987 996
988 997 @property
989 998 def mandatoryparams(self):
990 999 # make it an immutable tuple to force people through ``addparam``
991 1000 return tuple(self._mandatoryparams)
992 1001
993 1002 @property
994 1003 def advisoryparams(self):
995 1004 # make it an immutable tuple to force people through ``addparam``
996 1005 return tuple(self._advisoryparams)
997 1006
998 1007 def addparam(self, name, value='', mandatory=True):
999 1008 """add a parameter to the part
1000 1009
1001 1010 If 'mandatory' is set to True, the remote handler must claim support
1002 1011 for this parameter or the unbundling will be aborted.
1003 1012
1004 1013 The 'name' and 'value' cannot exceed 255 bytes each.
1005 1014 """
1006 1015 if self._generated is not None:
1007 1016 raise error.ReadOnlyPartError('part is being generated')
1008 1017 if name in self._seenparams:
1009 1018 raise ValueError('duplicated params: %s' % name)
1010 1019 self._seenparams.add(name)
1011 1020 params = self._advisoryparams
1012 1021 if mandatory:
1013 1022 params = self._mandatoryparams
1014 1023 params.append((name, value))
1015 1024
1016 1025 # methods used to generates the bundle2 stream
1017 1026 def getchunks(self, ui):
1018 1027 if self._generated is not None:
1019 1028 raise error.ProgrammingError('part can only be consumed once')
1020 1029 self._generated = False
1021 1030
1022 1031 if ui.debugflag:
1023 1032 msg = ['bundle2-output-part: "%s"' % self.type]
1024 1033 if not self.mandatory:
1025 1034 msg.append(' (advisory)')
1026 1035 nbmp = len(self.mandatoryparams)
1027 1036 nbap = len(self.advisoryparams)
1028 1037 if nbmp or nbap:
1029 1038 msg.append(' (params:')
1030 1039 if nbmp:
1031 1040 msg.append(' %i mandatory' % nbmp)
1032 1041 if nbap:
1033 1042 msg.append(' %i advisory' % nbmp)
1034 1043 msg.append(')')
1035 1044 if not self.data:
1036 1045 msg.append(' empty payload')
1037 1046 elif (util.safehasattr(self.data, 'next')
1038 1047 or util.safehasattr(self.data, '__next__')):
1039 1048 msg.append(' streamed payload')
1040 1049 else:
1041 1050 msg.append(' %i bytes payload' % len(self.data))
1042 1051 msg.append('\n')
1043 1052 ui.debug(''.join(msg))
1044 1053
1045 1054 #### header
1046 1055 if self.mandatory:
1047 1056 parttype = self.type.upper()
1048 1057 else:
1049 1058 parttype = self.type.lower()
1050 1059 outdebug(ui, 'part %s: "%s"' % (pycompat.bytestr(self.id), parttype))
1051 1060 ## parttype
1052 1061 header = [_pack(_fparttypesize, len(parttype)),
1053 1062 parttype, _pack(_fpartid, self.id),
1054 1063 ]
1055 1064 ## parameters
1056 1065 # count
1057 1066 manpar = self.mandatoryparams
1058 1067 advpar = self.advisoryparams
1059 1068 header.append(_pack(_fpartparamcount, len(manpar), len(advpar)))
1060 1069 # size
1061 1070 parsizes = []
1062 1071 for key, value in manpar:
1063 1072 parsizes.append(len(key))
1064 1073 parsizes.append(len(value))
1065 1074 for key, value in advpar:
1066 1075 parsizes.append(len(key))
1067 1076 parsizes.append(len(value))
1068 1077 paramsizes = _pack(_makefpartparamsizes(len(parsizes) // 2), *parsizes)
1069 1078 header.append(paramsizes)
1070 1079 # key, value
1071 1080 for key, value in manpar:
1072 1081 header.append(key)
1073 1082 header.append(value)
1074 1083 for key, value in advpar:
1075 1084 header.append(key)
1076 1085 header.append(value)
1077 1086 ## finalize header
1078 1087 try:
1079 1088 headerchunk = ''.join(header)
1080 1089 except TypeError:
1081 1090 raise TypeError(r'Found a non-bytes trying to '
1082 1091 r'build bundle part header: %r' % header)
1083 1092 outdebug(ui, 'header chunk size: %i' % len(headerchunk))
1084 1093 yield _pack(_fpartheadersize, len(headerchunk))
1085 1094 yield headerchunk
1086 1095 ## payload
1087 1096 try:
1088 1097 for chunk in self._payloadchunks():
1089 1098 outdebug(ui, 'payload chunk size: %i' % len(chunk))
1090 1099 yield _pack(_fpayloadsize, len(chunk))
1091 1100 yield chunk
1092 1101 except GeneratorExit:
1093 1102 # GeneratorExit means that nobody is listening for our
1094 1103 # results anyway, so just bail quickly rather than trying
1095 1104 # to produce an error part.
1096 1105 ui.debug('bundle2-generatorexit\n')
1097 1106 raise
1098 1107 except BaseException as exc:
1099 1108 bexc = stringutil.forcebytestr(exc)
1100 1109 # backup exception data for later
1101 1110 ui.debug('bundle2-input-stream-interrupt: encoding exception %s'
1102 1111 % bexc)
1103 1112 tb = sys.exc_info()[2]
1104 1113 msg = 'unexpected error: %s' % bexc
1105 1114 interpart = bundlepart('error:abort', [('message', msg)],
1106 1115 mandatory=False)
1107 1116 interpart.id = 0
1108 1117 yield _pack(_fpayloadsize, -1)
1109 1118 for chunk in interpart.getchunks(ui=ui):
1110 1119 yield chunk
1111 1120 outdebug(ui, 'closing payload chunk')
1112 1121 # abort current part payload
1113 1122 yield _pack(_fpayloadsize, 0)
1114 1123 pycompat.raisewithtb(exc, tb)
1115 1124 # end of payload
1116 1125 outdebug(ui, 'closing payload chunk')
1117 1126 yield _pack(_fpayloadsize, 0)
1118 1127 self._generated = True
1119 1128
1120 1129 def _payloadchunks(self):
1121 1130 """yield chunks of a the part payload
1122 1131
1123 1132 Exists to handle the different methods to provide data to a part."""
1124 1133 # we only support fixed size data now.
1125 1134 # This will be improved in the future.
1126 1135 if (util.safehasattr(self.data, 'next')
1127 1136 or util.safehasattr(self.data, '__next__')):
1128 1137 buff = util.chunkbuffer(self.data)
1129 1138 chunk = buff.read(preferedchunksize)
1130 1139 while chunk:
1131 1140 yield chunk
1132 1141 chunk = buff.read(preferedchunksize)
1133 1142 elif len(self.data):
1134 1143 yield self.data
1135 1144
1136 1145
1137 1146 flaginterrupt = -1
1138 1147
1139 1148 class interrupthandler(unpackermixin):
1140 1149 """read one part and process it with restricted capability
1141 1150
1142 1151 This allows to transmit exception raised on the producer size during part
1143 1152 iteration while the consumer is reading a part.
1144 1153
1145 1154 Part processed in this manner only have access to a ui object,"""
1146 1155
1147 1156 def __init__(self, ui, fp):
1148 1157 super(interrupthandler, self).__init__(fp)
1149 1158 self.ui = ui
1150 1159
1151 1160 def _readpartheader(self):
1152 1161 """reads a part header size and return the bytes blob
1153 1162
1154 1163 returns None if empty"""
1155 1164 headersize = self._unpack(_fpartheadersize)[0]
1156 1165 if headersize < 0:
1157 1166 raise error.BundleValueError('negative part header size: %i'
1158 1167 % headersize)
1159 1168 indebug(self.ui, 'part header size: %i\n' % headersize)
1160 1169 if headersize:
1161 1170 return self._readexact(headersize)
1162 1171 return None
1163 1172
1164 1173 def __call__(self):
1165 1174
1166 1175 self.ui.debug('bundle2-input-stream-interrupt:'
1167 1176 ' opening out of band context\n')
1168 1177 indebug(self.ui, 'bundle2 stream interruption, looking for a part.')
1169 1178 headerblock = self._readpartheader()
1170 1179 if headerblock is None:
1171 1180 indebug(self.ui, 'no part found during interruption.')
1172 1181 return
1173 1182 part = unbundlepart(self.ui, headerblock, self._fp)
1174 1183 op = interruptoperation(self.ui)
1175 1184 hardabort = False
1176 1185 try:
1177 1186 _processpart(op, part)
1178 1187 except (SystemExit, KeyboardInterrupt):
1179 1188 hardabort = True
1180 1189 raise
1181 1190 finally:
1182 1191 if not hardabort:
1183 1192 part.consume()
1184 1193 self.ui.debug('bundle2-input-stream-interrupt:'
1185 1194 ' closing out of band context\n')
1186 1195
1187 1196 class interruptoperation(object):
1188 1197 """A limited operation to be use by part handler during interruption
1189 1198
1190 1199 It only have access to an ui object.
1191 1200 """
1192 1201
1193 1202 def __init__(self, ui):
1194 1203 self.ui = ui
1195 1204 self.reply = None
1196 1205 self.captureoutput = False
1197 1206
1198 1207 @property
1199 1208 def repo(self):
1200 1209 raise error.ProgrammingError('no repo access from stream interruption')
1201 1210
1202 1211 def gettransaction(self):
1203 1212 raise TransactionUnavailable('no repo access from stream interruption')
1204 1213
1205 1214 def decodepayloadchunks(ui, fh):
1206 1215 """Reads bundle2 part payload data into chunks.
1207 1216
1208 1217 Part payload data consists of framed chunks. This function takes
1209 1218 a file handle and emits those chunks.
1210 1219 """
1211 1220 dolog = ui.configbool('devel', 'bundle2.debug')
1212 1221 debug = ui.debug
1213 1222
1214 1223 headerstruct = struct.Struct(_fpayloadsize)
1215 1224 headersize = headerstruct.size
1216 1225 unpack = headerstruct.unpack
1217 1226
1218 1227 readexactly = changegroup.readexactly
1219 1228 read = fh.read
1220 1229
1221 1230 chunksize = unpack(readexactly(fh, headersize))[0]
1222 1231 indebug(ui, 'payload chunk size: %i' % chunksize)
1223 1232
1224 1233 # changegroup.readexactly() is inlined below for performance.
1225 1234 while chunksize:
1226 1235 if chunksize >= 0:
1227 1236 s = read(chunksize)
1228 1237 if len(s) < chunksize:
1229 1238 raise error.Abort(_('stream ended unexpectedly '
1230 1239 ' (got %d bytes, expected %d)') %
1231 1240 (len(s), chunksize))
1232 1241
1233 1242 yield s
1234 1243 elif chunksize == flaginterrupt:
1235 1244 # Interrupt "signal" detected. The regular stream is interrupted
1236 1245 # and a bundle2 part follows. Consume it.
1237 1246 interrupthandler(ui, fh)()
1238 1247 else:
1239 1248 raise error.BundleValueError(
1240 1249 'negative payload chunk size: %s' % chunksize)
1241 1250
1242 1251 s = read(headersize)
1243 1252 if len(s) < headersize:
1244 1253 raise error.Abort(_('stream ended unexpectedly '
1245 1254 ' (got %d bytes, expected %d)') %
1246 1255 (len(s), chunksize))
1247 1256
1248 1257 chunksize = unpack(s)[0]
1249 1258
1250 1259 # indebug() inlined for performance.
1251 1260 if dolog:
1252 1261 debug('bundle2-input: payload chunk size: %i\n' % chunksize)
1253 1262
1254 1263 class unbundlepart(unpackermixin):
1255 1264 """a bundle part read from a bundle"""
1256 1265
1257 1266 def __init__(self, ui, header, fp):
1258 1267 super(unbundlepart, self).__init__(fp)
1259 1268 self._seekable = (util.safehasattr(fp, 'seek') and
1260 1269 util.safehasattr(fp, 'tell'))
1261 1270 self.ui = ui
1262 1271 # unbundle state attr
1263 1272 self._headerdata = header
1264 1273 self._headeroffset = 0
1265 1274 self._initialized = False
1266 1275 self.consumed = False
1267 1276 # part data
1268 1277 self.id = None
1269 1278 self.type = None
1270 1279 self.mandatoryparams = None
1271 1280 self.advisoryparams = None
1272 1281 self.params = None
1273 1282 self.mandatorykeys = ()
1274 1283 self._readheader()
1275 1284 self._mandatory = None
1276 1285 self._pos = 0
1277 1286
1278 1287 def _fromheader(self, size):
1279 1288 """return the next <size> byte from the header"""
1280 1289 offset = self._headeroffset
1281 1290 data = self._headerdata[offset:(offset + size)]
1282 1291 self._headeroffset = offset + size
1283 1292 return data
1284 1293
1285 1294 def _unpackheader(self, format):
1286 1295 """read given format from header
1287 1296
1288 1297 This automatically compute the size of the format to read."""
1289 1298 data = self._fromheader(struct.calcsize(format))
1290 1299 return _unpack(format, data)
1291 1300
1292 1301 def _initparams(self, mandatoryparams, advisoryparams):
1293 1302 """internal function to setup all logic related parameters"""
1294 1303 # make it read only to prevent people touching it by mistake.
1295 1304 self.mandatoryparams = tuple(mandatoryparams)
1296 1305 self.advisoryparams = tuple(advisoryparams)
1297 1306 # user friendly UI
1298 1307 self.params = util.sortdict(self.mandatoryparams)
1299 1308 self.params.update(self.advisoryparams)
1300 1309 self.mandatorykeys = frozenset(p[0] for p in mandatoryparams)
1301 1310
1302 1311 def _readheader(self):
1303 1312 """read the header and setup the object"""
1304 1313 typesize = self._unpackheader(_fparttypesize)[0]
1305 1314 self.type = self._fromheader(typesize)
1306 1315 indebug(self.ui, 'part type: "%s"' % self.type)
1307 1316 self.id = self._unpackheader(_fpartid)[0]
1308 1317 indebug(self.ui, 'part id: "%s"' % pycompat.bytestr(self.id))
1309 1318 # extract mandatory bit from type
1310 1319 self.mandatory = (self.type != self.type.lower())
1311 1320 self.type = self.type.lower()
1312 1321 ## reading parameters
1313 1322 # param count
1314 1323 mancount, advcount = self._unpackheader(_fpartparamcount)
1315 1324 indebug(self.ui, 'part parameters: %i' % (mancount + advcount))
1316 1325 # param size
1317 1326 fparamsizes = _makefpartparamsizes(mancount + advcount)
1318 1327 paramsizes = self._unpackheader(fparamsizes)
1319 1328 # make it a list of couple again
1320 1329 paramsizes = list(zip(paramsizes[::2], paramsizes[1::2]))
1321 1330 # split mandatory from advisory
1322 1331 mansizes = paramsizes[:mancount]
1323 1332 advsizes = paramsizes[mancount:]
1324 1333 # retrieve param value
1325 1334 manparams = []
1326 1335 for key, value in mansizes:
1327 1336 manparams.append((self._fromheader(key), self._fromheader(value)))
1328 1337 advparams = []
1329 1338 for key, value in advsizes:
1330 1339 advparams.append((self._fromheader(key), self._fromheader(value)))
1331 1340 self._initparams(manparams, advparams)
1332 1341 ## part payload
1333 1342 self._payloadstream = util.chunkbuffer(self._payloadchunks())
1334 1343 # we read the data, tell it
1335 1344 self._initialized = True
1336 1345
1337 1346 def _payloadchunks(self):
1338 1347 """Generator of decoded chunks in the payload."""
1339 1348 return decodepayloadchunks(self.ui, self._fp)
1340 1349
1341 1350 def consume(self):
1342 1351 """Read the part payload until completion.
1343 1352
1344 1353 By consuming the part data, the underlying stream read offset will
1345 1354 be advanced to the next part (or end of stream).
1346 1355 """
1347 1356 if self.consumed:
1348 1357 return
1349 1358
1350 1359 chunk = self.read(32768)
1351 1360 while chunk:
1352 1361 self._pos += len(chunk)
1353 1362 chunk = self.read(32768)
1354 1363
1355 1364 def read(self, size=None):
1356 1365 """read payload data"""
1357 1366 if not self._initialized:
1358 1367 self._readheader()
1359 1368 if size is None:
1360 1369 data = self._payloadstream.read()
1361 1370 else:
1362 1371 data = self._payloadstream.read(size)
1363 1372 self._pos += len(data)
1364 1373 if size is None or len(data) < size:
1365 1374 if not self.consumed and self._pos:
1366 1375 self.ui.debug('bundle2-input-part: total payload size %i\n'
1367 1376 % self._pos)
1368 1377 self.consumed = True
1369 1378 return data
1370 1379
1371 1380 class seekableunbundlepart(unbundlepart):
1372 1381 """A bundle2 part in a bundle that is seekable.
1373 1382
1374 1383 Regular ``unbundlepart`` instances can only be read once. This class
1375 1384 extends ``unbundlepart`` to enable bi-directional seeking within the
1376 1385 part.
1377 1386
1378 1387 Bundle2 part data consists of framed chunks. Offsets when seeking
1379 1388 refer to the decoded data, not the offsets in the underlying bundle2
1380 1389 stream.
1381 1390
1382 1391 To facilitate quickly seeking within the decoded data, instances of this
1383 1392 class maintain a mapping between offsets in the underlying stream and
1384 1393 the decoded payload. This mapping will consume memory in proportion
1385 1394 to the number of chunks within the payload (which almost certainly
1386 1395 increases in proportion with the size of the part).
1387 1396 """
1388 1397 def __init__(self, ui, header, fp):
1389 1398 # (payload, file) offsets for chunk starts.
1390 1399 self._chunkindex = []
1391 1400
1392 1401 super(seekableunbundlepart, self).__init__(ui, header, fp)
1393 1402
1394 1403 def _payloadchunks(self, chunknum=0):
1395 1404 '''seek to specified chunk and start yielding data'''
1396 1405 if len(self._chunkindex) == 0:
1397 1406 assert chunknum == 0, 'Must start with chunk 0'
1398 1407 self._chunkindex.append((0, self._tellfp()))
1399 1408 else:
1400 1409 assert chunknum < len(self._chunkindex), (
1401 1410 'Unknown chunk %d' % chunknum)
1402 1411 self._seekfp(self._chunkindex[chunknum][1])
1403 1412
1404 1413 pos = self._chunkindex[chunknum][0]
1405 1414
1406 1415 for chunk in decodepayloadchunks(self.ui, self._fp):
1407 1416 chunknum += 1
1408 1417 pos += len(chunk)
1409 1418 if chunknum == len(self._chunkindex):
1410 1419 self._chunkindex.append((pos, self._tellfp()))
1411 1420
1412 1421 yield chunk
1413 1422
1414 1423 def _findchunk(self, pos):
1415 1424 '''for a given payload position, return a chunk number and offset'''
1416 1425 for chunk, (ppos, fpos) in enumerate(self._chunkindex):
1417 1426 if ppos == pos:
1418 1427 return chunk, 0
1419 1428 elif ppos > pos:
1420 1429 return chunk - 1, pos - self._chunkindex[chunk - 1][0]
1421 1430 raise ValueError('Unknown chunk')
1422 1431
1423 1432 def tell(self):
1424 1433 return self._pos
1425 1434
1426 1435 def seek(self, offset, whence=os.SEEK_SET):
1427 1436 if whence == os.SEEK_SET:
1428 1437 newpos = offset
1429 1438 elif whence == os.SEEK_CUR:
1430 1439 newpos = self._pos + offset
1431 1440 elif whence == os.SEEK_END:
1432 1441 if not self.consumed:
1433 1442 # Can't use self.consume() here because it advances self._pos.
1434 1443 chunk = self.read(32768)
1435 1444 while chunk:
1436 1445 chunk = self.read(32768)
1437 1446 newpos = self._chunkindex[-1][0] - offset
1438 1447 else:
1439 1448 raise ValueError('Unknown whence value: %r' % (whence,))
1440 1449
1441 1450 if newpos > self._chunkindex[-1][0] and not self.consumed:
1442 1451 # Can't use self.consume() here because it advances self._pos.
1443 1452 chunk = self.read(32768)
1444 1453 while chunk:
1445 1454 chunk = self.read(32668)
1446 1455
1447 1456 if not 0 <= newpos <= self._chunkindex[-1][0]:
1448 1457 raise ValueError('Offset out of range')
1449 1458
1450 1459 if self._pos != newpos:
1451 1460 chunk, internaloffset = self._findchunk(newpos)
1452 1461 self._payloadstream = util.chunkbuffer(self._payloadchunks(chunk))
1453 1462 adjust = self.read(internaloffset)
1454 1463 if len(adjust) != internaloffset:
1455 1464 raise error.Abort(_('Seek failed\n'))
1456 1465 self._pos = newpos
1457 1466
1458 1467 def _seekfp(self, offset, whence=0):
1459 1468 """move the underlying file pointer
1460 1469
1461 1470 This method is meant for internal usage by the bundle2 protocol only.
1462 1471 They directly manipulate the low level stream including bundle2 level
1463 1472 instruction.
1464 1473
1465 1474 Do not use it to implement higher-level logic or methods."""
1466 1475 if self._seekable:
1467 1476 return self._fp.seek(offset, whence)
1468 1477 else:
1469 1478 raise NotImplementedError(_('File pointer is not seekable'))
1470 1479
1471 1480 def _tellfp(self):
1472 1481 """return the file offset, or None if file is not seekable
1473 1482
1474 1483 This method is meant for internal usage by the bundle2 protocol only.
1475 1484 They directly manipulate the low level stream including bundle2 level
1476 1485 instruction.
1477 1486
1478 1487 Do not use it to implement higher-level logic or methods."""
1479 1488 if self._seekable:
1480 1489 try:
1481 1490 return self._fp.tell()
1482 1491 except IOError as e:
1483 1492 if e.errno == errno.ESPIPE:
1484 1493 self._seekable = False
1485 1494 else:
1486 1495 raise
1487 1496 return None
1488 1497
1489 1498 # These are only the static capabilities.
1490 1499 # Check the 'getrepocaps' function for the rest.
1491 1500 capabilities = {'HG20': (),
1492 1501 'bookmarks': (),
1493 1502 'error': ('abort', 'unsupportedcontent', 'pushraced',
1494 1503 'pushkey'),
1495 1504 'listkeys': (),
1496 1505 'pushkey': (),
1497 1506 'digests': tuple(sorted(util.DIGESTS.keys())),
1498 1507 'remote-changegroup': ('http', 'https'),
1499 1508 'hgtagsfnodes': (),
1500 1509 'rev-branch-cache': (),
1501 1510 'phases': ('heads',),
1502 1511 'stream': ('v2',),
1503 1512 }
1504 1513
1505 1514 def getrepocaps(repo, allowpushback=False, role=None):
1506 1515 """return the bundle2 capabilities for a given repo
1507 1516
1508 1517 Exists to allow extensions (like evolution) to mutate the capabilities.
1509 1518
1510 1519 The returned value is used for servers advertising their capabilities as
1511 1520 well as clients advertising their capabilities to servers as part of
1512 1521 bundle2 requests. The ``role`` argument specifies which is which.
1513 1522 """
1514 1523 if role not in ('client', 'server'):
1515 1524 raise error.ProgrammingError('role argument must be client or server')
1516 1525
1517 1526 caps = capabilities.copy()
1518 1527 caps['changegroup'] = tuple(sorted(
1519 1528 changegroup.supportedincomingversions(repo)))
1520 1529 if obsolete.isenabled(repo, obsolete.exchangeopt):
1521 1530 supportedformat = tuple('V%i' % v for v in obsolete.formats)
1522 1531 caps['obsmarkers'] = supportedformat
1523 1532 if allowpushback:
1524 1533 caps['pushback'] = ()
1525 1534 cpmode = repo.ui.config('server', 'concurrent-push-mode')
1526 1535 if cpmode == 'check-related':
1527 1536 caps['checkheads'] = ('related',)
1528 1537 if 'phases' in repo.ui.configlist('devel', 'legacy.exchange'):
1529 1538 caps.pop('phases')
1530 1539
1531 1540 # Don't advertise stream clone support in server mode if not configured.
1532 1541 if role == 'server':
1533 1542 streamsupported = repo.ui.configbool('server', 'uncompressed',
1534 1543 untrusted=True)
1535 1544 featuresupported = repo.ui.configbool('server', 'bundle2.stream')
1536 1545
1537 1546 if not streamsupported or not featuresupported:
1538 1547 caps.pop('stream')
1539 1548 # Else always advertise support on client, because payload support
1540 1549 # should always be advertised.
1541 1550
1542 1551 return caps
1543 1552
1544 1553 def bundle2caps(remote):
1545 1554 """return the bundle capabilities of a peer as dict"""
1546 1555 raw = remote.capable('bundle2')
1547 1556 if not raw and raw != '':
1548 1557 return {}
1549 1558 capsblob = urlreq.unquote(remote.capable('bundle2'))
1550 1559 return decodecaps(capsblob)
1551 1560
1552 1561 def obsmarkersversion(caps):
1553 1562 """extract the list of supported obsmarkers versions from a bundle2caps dict
1554 1563 """
1555 1564 obscaps = caps.get('obsmarkers', ())
1556 1565 return [int(c[1:]) for c in obscaps if c.startswith('V')]
1557 1566
1558 1567 def writenewbundle(ui, repo, source, filename, bundletype, outgoing, opts,
1559 1568 vfs=None, compression=None, compopts=None):
1560 1569 if bundletype.startswith('HG10'):
1561 1570 cg = changegroup.makechangegroup(repo, outgoing, '01', source)
1562 1571 return writebundle(ui, cg, filename, bundletype, vfs=vfs,
1563 1572 compression=compression, compopts=compopts)
1564 1573 elif not bundletype.startswith('HG20'):
1565 1574 raise error.ProgrammingError('unknown bundle type: %s' % bundletype)
1566 1575
1567 1576 caps = {}
1568 1577 if 'obsolescence' in opts:
1569 1578 caps['obsmarkers'] = ('V1',)
1570 1579 bundle = bundle20(ui, caps)
1571 1580 bundle.setcompression(compression, compopts)
1572 1581 _addpartsfromopts(ui, repo, bundle, source, outgoing, opts)
1573 1582 chunkiter = bundle.getchunks()
1574 1583
1575 1584 return changegroup.writechunks(ui, chunkiter, filename, vfs=vfs)
1576 1585
1577 1586 def _addpartsfromopts(ui, repo, bundler, source, outgoing, opts):
1578 1587 # We should eventually reconcile this logic with the one behind
1579 1588 # 'exchange.getbundle2partsgenerator'.
1580 1589 #
1581 1590 # The type of input from 'getbundle' and 'writenewbundle' are a bit
1582 1591 # different right now. So we keep them separated for now for the sake of
1583 1592 # simplicity.
1584 1593
1585 1594 # we might not always want a changegroup in such bundle, for example in
1586 1595 # stream bundles
1587 1596 if opts.get('changegroup', True):
1588 1597 cgversion = opts.get('cg.version')
1589 1598 if cgversion is None:
1590 1599 cgversion = changegroup.safeversion(repo)
1591 1600 cg = changegroup.makechangegroup(repo, outgoing, cgversion, source)
1592 1601 part = bundler.newpart('changegroup', data=cg.getchunks())
1593 1602 part.addparam('version', cg.version)
1594 1603 if 'clcount' in cg.extras:
1595 1604 part.addparam('nbchanges', '%d' % cg.extras['clcount'],
1596 1605 mandatory=False)
1597 1606 if opts.get('phases') and repo.revs('%ln and secret()',
1598 1607 outgoing.missingheads):
1599 1608 part.addparam('targetphase', '%d' % phases.secret, mandatory=False)
1600 1609
1601 1610 if opts.get('streamv2', False):
1602 1611 addpartbundlestream2(bundler, repo, stream=True)
1603 1612
1604 1613 if opts.get('tagsfnodescache', True):
1605 1614 addparttagsfnodescache(repo, bundler, outgoing)
1606 1615
1607 1616 if opts.get('revbranchcache', True):
1608 1617 addpartrevbranchcache(repo, bundler, outgoing)
1609 1618
1610 1619 if opts.get('obsolescence', False):
1611 1620 obsmarkers = repo.obsstore.relevantmarkers(outgoing.missing)
1612 1621 buildobsmarkerspart(bundler, obsmarkers)
1613 1622
1614 1623 if opts.get('phases', False):
1615 1624 headsbyphase = phases.subsetphaseheads(repo, outgoing.missing)
1616 1625 phasedata = phases.binaryencode(headsbyphase)
1617 1626 bundler.newpart('phase-heads', data=phasedata)
1618 1627
1619 1628 def addparttagsfnodescache(repo, bundler, outgoing):
1620 1629 # we include the tags fnode cache for the bundle changeset
1621 1630 # (as an optional parts)
1622 1631 cache = tags.hgtagsfnodescache(repo.unfiltered())
1623 1632 chunks = []
1624 1633
1625 1634 # .hgtags fnodes are only relevant for head changesets. While we could
1626 1635 # transfer values for all known nodes, there will likely be little to
1627 1636 # no benefit.
1628 1637 #
1629 1638 # We don't bother using a generator to produce output data because
1630 1639 # a) we only have 40 bytes per head and even esoteric numbers of heads
1631 1640 # consume little memory (1M heads is 40MB) b) we don't want to send the
1632 1641 # part if we don't have entries and knowing if we have entries requires
1633 1642 # cache lookups.
1634 1643 for node in outgoing.missingheads:
1635 1644 # Don't compute missing, as this may slow down serving.
1636 1645 fnode = cache.getfnode(node, computemissing=False)
1637 1646 if fnode is not None:
1638 1647 chunks.extend([node, fnode])
1639 1648
1640 1649 if chunks:
1641 1650 bundler.newpart('hgtagsfnodes', data=''.join(chunks))
1642 1651
1643 1652 def addpartrevbranchcache(repo, bundler, outgoing):
1644 1653 # we include the rev branch cache for the bundle changeset
1645 1654 # (as an optional parts)
1646 1655 cache = repo.revbranchcache()
1647 1656 cl = repo.unfiltered().changelog
1648 1657 branchesdata = collections.defaultdict(lambda: (set(), set()))
1649 1658 for node in outgoing.missing:
1650 1659 branch, close = cache.branchinfo(cl.rev(node))
1651 1660 branchesdata[branch][close].add(node)
1652 1661
1653 1662 def generate():
1654 1663 for branch, (nodes, closed) in sorted(branchesdata.items()):
1655 1664 utf8branch = encoding.fromlocal(branch)
1656 1665 yield rbcstruct.pack(len(utf8branch), len(nodes), len(closed))
1657 1666 yield utf8branch
1658 1667 for n in sorted(nodes):
1659 1668 yield n
1660 1669 for n in sorted(closed):
1661 1670 yield n
1662 1671
1663 1672 bundler.newpart('cache:rev-branch-cache', data=generate(),
1664 1673 mandatory=False)
1665 1674
1666 1675 def _formatrequirementsspec(requirements):
1667 1676 requirements = [req for req in requirements if req != "shared"]
1668 1677 return urlreq.quote(','.join(sorted(requirements)))
1669 1678
1670 1679 def _formatrequirementsparams(requirements):
1671 1680 requirements = _formatrequirementsspec(requirements)
1672 1681 params = "%s%s" % (urlreq.quote("requirements="), requirements)
1673 1682 return params
1674 1683
1675 1684 def addpartbundlestream2(bundler, repo, **kwargs):
1676 1685 if not kwargs.get(r'stream', False):
1677 1686 return
1678 1687
1679 1688 if not streamclone.allowservergeneration(repo):
1680 1689 raise error.Abort(_('stream data requested but server does not allow '
1681 1690 'this feature'),
1682 1691 hint=_('well-behaved clients should not be '
1683 1692 'requesting stream data from servers not '
1684 1693 'advertising it; the client may be buggy'))
1685 1694
1686 1695 # Stream clones don't compress well. And compression undermines a
1687 1696 # goal of stream clones, which is to be fast. Communicate the desire
1688 1697 # to avoid compression to consumers of the bundle.
1689 1698 bundler.prefercompressed = False
1690 1699
1691 1700 # get the includes and excludes
1692 1701 includepats = kwargs.get(r'includepats')
1693 1702 excludepats = kwargs.get(r'excludepats')
1694 1703
1695 1704 narrowstream = repo.ui.configbool('experimental',
1696 1705 'server.stream-narrow-clones')
1697 1706
1698 1707 if (includepats or excludepats) and not narrowstream:
1699 1708 raise error.Abort(_('server does not support narrow stream clones'))
1700 1709
1701 1710 includeobsmarkers = False
1702 1711 if repo.obsstore:
1703 1712 remoteversions = obsmarkersversion(bundler.capabilities)
1704 1713 if not remoteversions:
1705 1714 raise error.Abort(_('server has obsolescence markers, but client '
1706 1715 'cannot receive them via stream clone'))
1707 1716 elif repo.obsstore._version in remoteversions:
1708 1717 includeobsmarkers = True
1709 1718
1710 1719 filecount, bytecount, it = streamclone.generatev2(repo, includepats,
1711 1720 excludepats,
1712 1721 includeobsmarkers)
1713 1722 requirements = _formatrequirementsspec(repo.requirements)
1714 1723 part = bundler.newpart('stream2', data=it)
1715 1724 part.addparam('bytecount', '%d' % bytecount, mandatory=True)
1716 1725 part.addparam('filecount', '%d' % filecount, mandatory=True)
1717 1726 part.addparam('requirements', requirements, mandatory=True)
1718 1727
1719 1728 def buildobsmarkerspart(bundler, markers):
1720 1729 """add an obsmarker part to the bundler with <markers>
1721 1730
1722 1731 No part is created if markers is empty.
1723 1732 Raises ValueError if the bundler doesn't support any known obsmarker format.
1724 1733 """
1725 1734 if not markers:
1726 1735 return None
1727 1736
1728 1737 remoteversions = obsmarkersversion(bundler.capabilities)
1729 1738 version = obsolete.commonversion(remoteversions)
1730 1739 if version is None:
1731 1740 raise ValueError('bundler does not support common obsmarker format')
1732 1741 stream = obsolete.encodemarkers(markers, True, version=version)
1733 1742 return bundler.newpart('obsmarkers', data=stream)
1734 1743
1735 1744 def writebundle(ui, cg, filename, bundletype, vfs=None, compression=None,
1736 1745 compopts=None):
1737 1746 """Write a bundle file and return its filename.
1738 1747
1739 1748 Existing files will not be overwritten.
1740 1749 If no filename is specified, a temporary file is created.
1741 1750 bz2 compression can be turned off.
1742 1751 The bundle file will be deleted in case of errors.
1743 1752 """
1744 1753
1745 1754 if bundletype == "HG20":
1746 1755 bundle = bundle20(ui)
1747 1756 bundle.setcompression(compression, compopts)
1748 1757 part = bundle.newpart('changegroup', data=cg.getchunks())
1749 1758 part.addparam('version', cg.version)
1750 1759 if 'clcount' in cg.extras:
1751 1760 part.addparam('nbchanges', '%d' % cg.extras['clcount'],
1752 1761 mandatory=False)
1753 1762 chunkiter = bundle.getchunks()
1754 1763 else:
1755 1764 # compression argument is only for the bundle2 case
1756 1765 assert compression is None
1757 1766 if cg.version != '01':
1758 1767 raise error.Abort(_('old bundle types only supports v1 '
1759 1768 'changegroups'))
1760 1769 header, comp = bundletypes[bundletype]
1761 1770 if comp not in util.compengines.supportedbundletypes:
1762 1771 raise error.Abort(_('unknown stream compression type: %s')
1763 1772 % comp)
1764 1773 compengine = util.compengines.forbundletype(comp)
1765 1774 def chunkiter():
1766 1775 yield header
1767 1776 for chunk in compengine.compressstream(cg.getchunks(), compopts):
1768 1777 yield chunk
1769 1778 chunkiter = chunkiter()
1770 1779
1771 1780 # parse the changegroup data, otherwise we will block
1772 1781 # in case of sshrepo because we don't know the end of the stream
1773 1782 return changegroup.writechunks(ui, chunkiter, filename, vfs=vfs)
1774 1783
1775 1784 def combinechangegroupresults(op):
1776 1785 """logic to combine 0 or more addchangegroup results into one"""
1777 1786 results = [r.get('return', 0)
1778 1787 for r in op.records['changegroup']]
1779 1788 changedheads = 0
1780 1789 result = 1
1781 1790 for ret in results:
1782 1791 # If any changegroup result is 0, return 0
1783 1792 if ret == 0:
1784 1793 result = 0
1785 1794 break
1786 1795 if ret < -1:
1787 1796 changedheads += ret + 1
1788 1797 elif ret > 1:
1789 1798 changedheads += ret - 1
1790 1799 if changedheads > 0:
1791 1800 result = 1 + changedheads
1792 1801 elif changedheads < 0:
1793 1802 result = -1 + changedheads
1794 1803 return result
1795 1804
1796 1805 @parthandler('changegroup', ('version', 'nbchanges', 'treemanifest',
1797 1806 'targetphase'))
1798 1807 def handlechangegroup(op, inpart):
1799 1808 """apply a changegroup part on the repo
1800 1809
1801 1810 This is a very early implementation that will massive rework before being
1802 1811 inflicted to any end-user.
1803 1812 """
1804 1813 from . import localrepo
1805 1814
1806 1815 tr = op.gettransaction()
1807 1816 unpackerversion = inpart.params.get('version', '01')
1808 1817 # We should raise an appropriate exception here
1809 1818 cg = changegroup.getunbundler(unpackerversion, inpart, None)
1810 1819 # the source and url passed here are overwritten by the one contained in
1811 1820 # the transaction.hookargs argument. So 'bundle2' is a placeholder
1812 1821 nbchangesets = None
1813 1822 if 'nbchanges' in inpart.params:
1814 1823 nbchangesets = int(inpart.params.get('nbchanges'))
1815 1824 if ('treemanifest' in inpart.params and
1816 1825 'treemanifest' not in op.repo.requirements):
1817 1826 if len(op.repo.changelog) != 0:
1818 1827 raise error.Abort(_(
1819 1828 "bundle contains tree manifests, but local repo is "
1820 1829 "non-empty and does not use tree manifests"))
1821 1830 op.repo.requirements.add('treemanifest')
1822 1831 op.repo.svfs.options = localrepo.resolvestorevfsoptions(
1823 1832 op.repo.ui, op.repo.requirements, op.repo.features)
1824 1833 op.repo._writerequirements()
1825 1834 extrakwargs = {}
1826 1835 targetphase = inpart.params.get('targetphase')
1827 1836 if targetphase is not None:
1828 1837 extrakwargs[r'targetphase'] = int(targetphase)
1829 1838 ret = _processchangegroup(op, cg, tr, 'bundle2', 'bundle2',
1830 1839 expectedtotal=nbchangesets, **extrakwargs)
1831 1840 if op.reply is not None:
1832 1841 # This is definitely not the final form of this
1833 1842 # return. But one need to start somewhere.
1834 1843 part = op.reply.newpart('reply:changegroup', mandatory=False)
1835 1844 part.addparam(
1836 1845 'in-reply-to', pycompat.bytestr(inpart.id), mandatory=False)
1837 1846 part.addparam('return', '%i' % ret, mandatory=False)
1838 1847 assert not inpart.read()
1839 1848
1840 1849 _remotechangegroupparams = tuple(['url', 'size', 'digests'] +
1841 1850 ['digest:%s' % k for k in util.DIGESTS.keys()])
1842 1851 @parthandler('remote-changegroup', _remotechangegroupparams)
1843 1852 def handleremotechangegroup(op, inpart):
1844 1853 """apply a bundle10 on the repo, given an url and validation information
1845 1854
1846 1855 All the information about the remote bundle to import are given as
1847 1856 parameters. The parameters include:
1848 1857 - url: the url to the bundle10.
1849 1858 - size: the bundle10 file size. It is used to validate what was
1850 1859 retrieved by the client matches the server knowledge about the bundle.
1851 1860 - digests: a space separated list of the digest types provided as
1852 1861 parameters.
1853 1862 - digest:<digest-type>: the hexadecimal representation of the digest with
1854 1863 that name. Like the size, it is used to validate what was retrieved by
1855 1864 the client matches what the server knows about the bundle.
1856 1865
1857 1866 When multiple digest types are given, all of them are checked.
1858 1867 """
1859 1868 try:
1860 1869 raw_url = inpart.params['url']
1861 1870 except KeyError:
1862 1871 raise error.Abort(_('remote-changegroup: missing "%s" param') % 'url')
1863 1872 parsed_url = util.url(raw_url)
1864 1873 if parsed_url.scheme not in capabilities['remote-changegroup']:
1865 1874 raise error.Abort(_('remote-changegroup does not support %s urls') %
1866 1875 parsed_url.scheme)
1867 1876
1868 1877 try:
1869 1878 size = int(inpart.params['size'])
1870 1879 except ValueError:
1871 1880 raise error.Abort(_('remote-changegroup: invalid value for param "%s"')
1872 1881 % 'size')
1873 1882 except KeyError:
1874 1883 raise error.Abort(_('remote-changegroup: missing "%s" param') % 'size')
1875 1884
1876 1885 digests = {}
1877 1886 for typ in inpart.params.get('digests', '').split():
1878 1887 param = 'digest:%s' % typ
1879 1888 try:
1880 1889 value = inpart.params[param]
1881 1890 except KeyError:
1882 1891 raise error.Abort(_('remote-changegroup: missing "%s" param') %
1883 1892 param)
1884 1893 digests[typ] = value
1885 1894
1886 1895 real_part = util.digestchecker(url.open(op.ui, raw_url), size, digests)
1887 1896
1888 1897 tr = op.gettransaction()
1889 1898 from . import exchange
1890 1899 cg = exchange.readbundle(op.repo.ui, real_part, raw_url)
1891 1900 if not isinstance(cg, changegroup.cg1unpacker):
1892 1901 raise error.Abort(_('%s: not a bundle version 1.0') %
1893 1902 util.hidepassword(raw_url))
1894 1903 ret = _processchangegroup(op, cg, tr, 'bundle2', 'bundle2')
1895 1904 if op.reply is not None:
1896 1905 # This is definitely not the final form of this
1897 1906 # return. But one need to start somewhere.
1898 1907 part = op.reply.newpart('reply:changegroup')
1899 1908 part.addparam(
1900 1909 'in-reply-to', pycompat.bytestr(inpart.id), mandatory=False)
1901 1910 part.addparam('return', '%i' % ret, mandatory=False)
1902 1911 try:
1903 1912 real_part.validate()
1904 1913 except error.Abort as e:
1905 1914 raise error.Abort(_('bundle at %s is corrupted:\n%s') %
1906 1915 (util.hidepassword(raw_url), bytes(e)))
1907 1916 assert not inpart.read()
1908 1917
1909 1918 @parthandler('reply:changegroup', ('return', 'in-reply-to'))
1910 1919 def handlereplychangegroup(op, inpart):
1911 1920 ret = int(inpart.params['return'])
1912 1921 replyto = int(inpart.params['in-reply-to'])
1913 1922 op.records.add('changegroup', {'return': ret}, replyto)
1914 1923
1915 1924 @parthandler('check:bookmarks')
1916 1925 def handlecheckbookmarks(op, inpart):
1917 1926 """check location of bookmarks
1918 1927
1919 1928 This part is to be used to detect push race regarding bookmark, it
1920 1929 contains binary encoded (bookmark, node) tuple. If the local state does
1921 1930 not marks the one in the part, a PushRaced exception is raised
1922 1931 """
1923 1932 bookdata = bookmarks.binarydecode(inpart)
1924 1933
1925 1934 msgstandard = ('remote repository changed while pushing - please try again '
1926 1935 '(bookmark "%s" move from %s to %s)')
1927 1936 msgmissing = ('remote repository changed while pushing - please try again '
1928 1937 '(bookmark "%s" is missing, expected %s)')
1929 1938 msgexist = ('remote repository changed while pushing - please try again '
1930 1939 '(bookmark "%s" set on %s, expected missing)')
1931 1940 for book, node in bookdata:
1932 1941 currentnode = op.repo._bookmarks.get(book)
1933 1942 if currentnode != node:
1934 1943 if node is None:
1935 1944 finalmsg = msgexist % (book, nodemod.short(currentnode))
1936 1945 elif currentnode is None:
1937 1946 finalmsg = msgmissing % (book, nodemod.short(node))
1938 1947 else:
1939 1948 finalmsg = msgstandard % (book, nodemod.short(node),
1940 1949 nodemod.short(currentnode))
1941 1950 raise error.PushRaced(finalmsg)
1942 1951
1943 1952 @parthandler('check:heads')
1944 1953 def handlecheckheads(op, inpart):
1945 1954 """check that head of the repo did not change
1946 1955
1947 1956 This is used to detect a push race when using unbundle.
1948 1957 This replaces the "heads" argument of unbundle."""
1949 1958 h = inpart.read(20)
1950 1959 heads = []
1951 1960 while len(h) == 20:
1952 1961 heads.append(h)
1953 1962 h = inpart.read(20)
1954 1963 assert not h
1955 1964 # Trigger a transaction so that we are guaranteed to have the lock now.
1956 1965 if op.ui.configbool('experimental', 'bundle2lazylocking'):
1957 1966 op.gettransaction()
1958 1967 if sorted(heads) != sorted(op.repo.heads()):
1959 1968 raise error.PushRaced('remote repository changed while pushing - '
1960 1969 'please try again')
1961 1970
1962 1971 @parthandler('check:updated-heads')
1963 1972 def handlecheckupdatedheads(op, inpart):
1964 1973 """check for race on the heads touched by a push
1965 1974
1966 1975 This is similar to 'check:heads' but focus on the heads actually updated
1967 1976 during the push. If other activities happen on unrelated heads, it is
1968 1977 ignored.
1969 1978
1970 1979 This allow server with high traffic to avoid push contention as long as
1971 1980 unrelated parts of the graph are involved."""
1972 1981 h = inpart.read(20)
1973 1982 heads = []
1974 1983 while len(h) == 20:
1975 1984 heads.append(h)
1976 1985 h = inpart.read(20)
1977 1986 assert not h
1978 1987 # trigger a transaction so that we are guaranteed to have the lock now.
1979 1988 if op.ui.configbool('experimental', 'bundle2lazylocking'):
1980 1989 op.gettransaction()
1981 1990
1982 1991 currentheads = set()
1983 1992 for ls in op.repo.branchmap().iterheads():
1984 1993 currentheads.update(ls)
1985 1994
1986 1995 for h in heads:
1987 1996 if h not in currentheads:
1988 1997 raise error.PushRaced('remote repository changed while pushing - '
1989 1998 'please try again')
1990 1999
1991 2000 @parthandler('check:phases')
1992 2001 def handlecheckphases(op, inpart):
1993 2002 """check that phase boundaries of the repository did not change
1994 2003
1995 2004 This is used to detect a push race.
1996 2005 """
1997 2006 phasetonodes = phases.binarydecode(inpart)
1998 2007 unfi = op.repo.unfiltered()
1999 2008 cl = unfi.changelog
2000 2009 phasecache = unfi._phasecache
2001 2010 msg = ('remote repository changed while pushing - please try again '
2002 2011 '(%s is %s expected %s)')
2003 2012 for expectedphase, nodes in enumerate(phasetonodes):
2004 2013 for n in nodes:
2005 2014 actualphase = phasecache.phase(unfi, cl.rev(n))
2006 2015 if actualphase != expectedphase:
2007 2016 finalmsg = msg % (nodemod.short(n),
2008 2017 phases.phasenames[actualphase],
2009 2018 phases.phasenames[expectedphase])
2010 2019 raise error.PushRaced(finalmsg)
2011 2020
2012 2021 @parthandler('output')
2013 2022 def handleoutput(op, inpart):
2014 2023 """forward output captured on the server to the client"""
2015 2024 for line in inpart.read().splitlines():
2016 2025 op.ui.status(_('remote: %s\n') % line)
2017 2026
2018 2027 @parthandler('replycaps')
2019 2028 def handlereplycaps(op, inpart):
2020 2029 """Notify that a reply bundle should be created
2021 2030
2022 2031 The payload contains the capabilities information for the reply"""
2023 2032 caps = decodecaps(inpart.read())
2024 2033 if op.reply is None:
2025 2034 op.reply = bundle20(op.ui, caps)
2026 2035
2027 2036 class AbortFromPart(error.Abort):
2028 2037 """Sub-class of Abort that denotes an error from a bundle2 part."""
2029 2038
2030 2039 @parthandler('error:abort', ('message', 'hint'))
2031 2040 def handleerrorabort(op, inpart):
2032 2041 """Used to transmit abort error over the wire"""
2033 2042 raise AbortFromPart(inpart.params['message'],
2034 2043 hint=inpart.params.get('hint'))
2035 2044
2036 2045 @parthandler('error:pushkey', ('namespace', 'key', 'new', 'old', 'ret',
2037 2046 'in-reply-to'))
2038 2047 def handleerrorpushkey(op, inpart):
2039 2048 """Used to transmit failure of a mandatory pushkey over the wire"""
2040 2049 kwargs = {}
2041 2050 for name in ('namespace', 'key', 'new', 'old', 'ret'):
2042 2051 value = inpart.params.get(name)
2043 2052 if value is not None:
2044 2053 kwargs[name] = value
2045 2054 raise error.PushkeyFailed(inpart.params['in-reply-to'],
2046 2055 **pycompat.strkwargs(kwargs))
2047 2056
2048 2057 @parthandler('error:unsupportedcontent', ('parttype', 'params'))
2049 2058 def handleerrorunsupportedcontent(op, inpart):
2050 2059 """Used to transmit unknown content error over the wire"""
2051 2060 kwargs = {}
2052 2061 parttype = inpart.params.get('parttype')
2053 2062 if parttype is not None:
2054 2063 kwargs['parttype'] = parttype
2055 2064 params = inpart.params.get('params')
2056 2065 if params is not None:
2057 2066 kwargs['params'] = params.split('\0')
2058 2067
2059 2068 raise error.BundleUnknownFeatureError(**pycompat.strkwargs(kwargs))
2060 2069
2061 2070 @parthandler('error:pushraced', ('message',))
2062 2071 def handleerrorpushraced(op, inpart):
2063 2072 """Used to transmit push race error over the wire"""
2064 2073 raise error.ResponseError(_('push failed:'), inpart.params['message'])
2065 2074
2066 2075 @parthandler('listkeys', ('namespace',))
2067 2076 def handlelistkeys(op, inpart):
2068 2077 """retrieve pushkey namespace content stored in a bundle2"""
2069 2078 namespace = inpart.params['namespace']
2070 2079 r = pushkey.decodekeys(inpart.read())
2071 2080 op.records.add('listkeys', (namespace, r))
2072 2081
2073 2082 @parthandler('pushkey', ('namespace', 'key', 'old', 'new'))
2074 2083 def handlepushkey(op, inpart):
2075 2084 """process a pushkey request"""
2076 2085 dec = pushkey.decode
2077 2086 namespace = dec(inpart.params['namespace'])
2078 2087 key = dec(inpart.params['key'])
2079 2088 old = dec(inpart.params['old'])
2080 2089 new = dec(inpart.params['new'])
2081 2090 # Grab the transaction to ensure that we have the lock before performing the
2082 2091 # pushkey.
2083 2092 if op.ui.configbool('experimental', 'bundle2lazylocking'):
2084 2093 op.gettransaction()
2085 2094 ret = op.repo.pushkey(namespace, key, old, new)
2086 2095 record = {'namespace': namespace,
2087 2096 'key': key,
2088 2097 'old': old,
2089 2098 'new': new}
2090 2099 op.records.add('pushkey', record)
2091 2100 if op.reply is not None:
2092 2101 rpart = op.reply.newpart('reply:pushkey')
2093 2102 rpart.addparam(
2094 2103 'in-reply-to', pycompat.bytestr(inpart.id), mandatory=False)
2095 2104 rpart.addparam('return', '%i' % ret, mandatory=False)
2096 2105 if inpart.mandatory and not ret:
2097 2106 kwargs = {}
2098 2107 for key in ('namespace', 'key', 'new', 'old', 'ret'):
2099 2108 if key in inpart.params:
2100 2109 kwargs[key] = inpart.params[key]
2101 2110 raise error.PushkeyFailed(partid='%d' % inpart.id,
2102 2111 **pycompat.strkwargs(kwargs))
2103 2112
2104 2113 @parthandler('bookmarks')
2105 2114 def handlebookmark(op, inpart):
2106 2115 """transmit bookmark information
2107 2116
2108 2117 The part contains binary encoded bookmark information.
2109 2118
2110 2119 The exact behavior of this part can be controlled by the 'bookmarks' mode
2111 2120 on the bundle operation.
2112 2121
2113 2122 When mode is 'apply' (the default) the bookmark information is applied as
2114 2123 is to the unbundling repository. Make sure a 'check:bookmarks' part is
2115 2124 issued earlier to check for push races in such update. This behavior is
2116 2125 suitable for pushing.
2117 2126
2118 2127 When mode is 'records', the information is recorded into the 'bookmarks'
2119 2128 records of the bundle operation. This behavior is suitable for pulling.
2120 2129 """
2121 2130 changes = bookmarks.binarydecode(inpart)
2122 2131
2123 2132 pushkeycompat = op.repo.ui.configbool('server', 'bookmarks-pushkey-compat')
2124 2133 bookmarksmode = op.modes.get('bookmarks', 'apply')
2125 2134
2126 2135 if bookmarksmode == 'apply':
2127 2136 tr = op.gettransaction()
2128 2137 bookstore = op.repo._bookmarks
2129 2138 if pushkeycompat:
2130 2139 allhooks = []
2131 2140 for book, node in changes:
2132 2141 hookargs = tr.hookargs.copy()
2133 2142 hookargs['pushkeycompat'] = '1'
2134 2143 hookargs['namespace'] = 'bookmarks'
2135 2144 hookargs['key'] = book
2136 2145 hookargs['old'] = nodemod.hex(bookstore.get(book, ''))
2137 2146 hookargs['new'] = nodemod.hex(node if node is not None else '')
2138 2147 allhooks.append(hookargs)
2139 2148
2140 2149 for hookargs in allhooks:
2141 2150 op.repo.hook('prepushkey', throw=True,
2142 2151 **pycompat.strkwargs(hookargs))
2143 2152
2144 2153 bookstore.applychanges(op.repo, op.gettransaction(), changes)
2145 2154
2146 2155 if pushkeycompat:
2147 2156 def runhook():
2148 2157 for hookargs in allhooks:
2149 2158 op.repo.hook('pushkey', **pycompat.strkwargs(hookargs))
2150 2159 op.repo._afterlock(runhook)
2151 2160
2152 2161 elif bookmarksmode == 'records':
2153 2162 for book, node in changes:
2154 2163 record = {'bookmark': book, 'node': node}
2155 2164 op.records.add('bookmarks', record)
2156 2165 else:
2157 2166 raise error.ProgrammingError('unkown bookmark mode: %s' % bookmarksmode)
2158 2167
2159 2168 @parthandler('phase-heads')
2160 2169 def handlephases(op, inpart):
2161 2170 """apply phases from bundle part to repo"""
2162 2171 headsbyphase = phases.binarydecode(inpart)
2163 2172 phases.updatephases(op.repo.unfiltered(), op.gettransaction, headsbyphase)
2164 2173
2165 2174 @parthandler('reply:pushkey', ('return', 'in-reply-to'))
2166 2175 def handlepushkeyreply(op, inpart):
2167 2176 """retrieve the result of a pushkey request"""
2168 2177 ret = int(inpart.params['return'])
2169 2178 partid = int(inpart.params['in-reply-to'])
2170 2179 op.records.add('pushkey', {'return': ret}, partid)
2171 2180
2172 2181 @parthandler('obsmarkers')
2173 2182 def handleobsmarker(op, inpart):
2174 2183 """add a stream of obsmarkers to the repo"""
2175 2184 tr = op.gettransaction()
2176 2185 markerdata = inpart.read()
2177 2186 if op.ui.config('experimental', 'obsmarkers-exchange-debug'):
2178 2187 op.ui.write(('obsmarker-exchange: %i bytes received\n')
2179 2188 % len(markerdata))
2180 2189 # The mergemarkers call will crash if marker creation is not enabled.
2181 2190 # we want to avoid this if the part is advisory.
2182 2191 if not inpart.mandatory and op.repo.obsstore.readonly:
2183 2192 op.repo.ui.debug('ignoring obsolescence markers, feature not enabled\n')
2184 2193 return
2185 2194 new = op.repo.obsstore.mergemarkers(tr, markerdata)
2186 2195 op.repo.invalidatevolatilesets()
2187 2196 if new:
2188 2197 op.repo.ui.status(_('%i new obsolescence markers\n') % new)
2189 2198 op.records.add('obsmarkers', {'new': new})
2190 2199 if op.reply is not None:
2191 2200 rpart = op.reply.newpart('reply:obsmarkers')
2192 2201 rpart.addparam(
2193 2202 'in-reply-to', pycompat.bytestr(inpart.id), mandatory=False)
2194 2203 rpart.addparam('new', '%i' % new, mandatory=False)
2195 2204
2196 2205
2197 2206 @parthandler('reply:obsmarkers', ('new', 'in-reply-to'))
2198 2207 def handleobsmarkerreply(op, inpart):
2199 2208 """retrieve the result of a pushkey request"""
2200 2209 ret = int(inpart.params['new'])
2201 2210 partid = int(inpart.params['in-reply-to'])
2202 2211 op.records.add('obsmarkers', {'new': ret}, partid)
2203 2212
2204 2213 @parthandler('hgtagsfnodes')
2205 2214 def handlehgtagsfnodes(op, inpart):
2206 2215 """Applies .hgtags fnodes cache entries to the local repo.
2207 2216
2208 2217 Payload is pairs of 20 byte changeset nodes and filenodes.
2209 2218 """
2210 2219 # Grab the transaction so we ensure that we have the lock at this point.
2211 2220 if op.ui.configbool('experimental', 'bundle2lazylocking'):
2212 2221 op.gettransaction()
2213 2222 cache = tags.hgtagsfnodescache(op.repo.unfiltered())
2214 2223
2215 2224 count = 0
2216 2225 while True:
2217 2226 node = inpart.read(20)
2218 2227 fnode = inpart.read(20)
2219 2228 if len(node) < 20 or len(fnode) < 20:
2220 2229 op.ui.debug('ignoring incomplete received .hgtags fnodes data\n')
2221 2230 break
2222 2231 cache.setfnode(node, fnode)
2223 2232 count += 1
2224 2233
2225 2234 cache.write()
2226 2235 op.ui.debug('applied %i hgtags fnodes cache entries\n' % count)
2227 2236
2228 2237 rbcstruct = struct.Struct('>III')
2229 2238
2230 2239 @parthandler('cache:rev-branch-cache')
2231 2240 def handlerbc(op, inpart):
2232 2241 """receive a rev-branch-cache payload and update the local cache
2233 2242
2234 2243 The payload is a series of data related to each branch
2235 2244
2236 2245 1) branch name length
2237 2246 2) number of open heads
2238 2247 3) number of closed heads
2239 2248 4) open heads nodes
2240 2249 5) closed heads nodes
2241 2250 """
2242 2251 total = 0
2243 2252 rawheader = inpart.read(rbcstruct.size)
2244 2253 cache = op.repo.revbranchcache()
2245 2254 cl = op.repo.unfiltered().changelog
2246 2255 while rawheader:
2247 2256 header = rbcstruct.unpack(rawheader)
2248 2257 total += header[1] + header[2]
2249 2258 utf8branch = inpart.read(header[0])
2250 2259 branch = encoding.tolocal(utf8branch)
2251 2260 for x in pycompat.xrange(header[1]):
2252 2261 node = inpart.read(20)
2253 2262 rev = cl.rev(node)
2254 2263 cache.setdata(branch, rev, node, False)
2255 2264 for x in pycompat.xrange(header[2]):
2256 2265 node = inpart.read(20)
2257 2266 rev = cl.rev(node)
2258 2267 cache.setdata(branch, rev, node, True)
2259 2268 rawheader = inpart.read(rbcstruct.size)
2260 2269 cache.write()
2261 2270
2262 2271 @parthandler('pushvars')
2263 2272 def bundle2getvars(op, part):
2264 2273 '''unbundle a bundle2 containing shellvars on the server'''
2265 2274 # An option to disable unbundling on server-side for security reasons
2266 2275 if op.ui.configbool('push', 'pushvars.server'):
2267 2276 hookargs = {}
2268 2277 for key, value in part.advisoryparams:
2269 2278 key = key.upper()
2270 2279 # We want pushed variables to have USERVAR_ prepended so we know
2271 2280 # they came from the --pushvar flag.
2272 2281 key = "USERVAR_" + key
2273 2282 hookargs[key] = value
2274 2283 op.addhookargs(hookargs)
2275 2284
2276 2285 @parthandler('stream2', ('requirements', 'filecount', 'bytecount'))
2277 2286 def handlestreamv2bundle(op, part):
2278 2287
2279 2288 requirements = urlreq.unquote(part.params['requirements']).split(',')
2280 2289 filecount = int(part.params['filecount'])
2281 2290 bytecount = int(part.params['bytecount'])
2282 2291
2283 2292 repo = op.repo
2284 2293 if len(repo):
2285 2294 msg = _('cannot apply stream clone to non empty repository')
2286 2295 raise error.Abort(msg)
2287 2296
2288 2297 repo.ui.debug('applying stream bundle\n')
2289 2298 streamclone.applybundlev2(repo, part, filecount, bytecount,
2290 2299 requirements)
2291 2300
2292 2301 def widen_bundle(repo, oldmatcher, newmatcher, common, known, cgversion,
2293 2302 ellipses):
2294 2303 """generates bundle2 for widening a narrow clone
2295 2304
2296 2305 repo is the localrepository instance
2297 2306 oldmatcher matches what the client already has
2298 2307 newmatcher matches what the client needs (including what it already has)
2299 2308 common is set of common heads between server and client
2300 2309 known is a set of revs known on the client side (used in ellipses)
2301 2310 cgversion is the changegroup version to send
2302 2311 ellipses is boolean value telling whether to send ellipses data or not
2303 2312
2304 2313 returns bundle2 of the data required for extending
2305 2314 """
2306 2315 bundler = bundle20(repo.ui)
2307 2316 commonnodes = set()
2308 2317 cl = repo.changelog
2309 2318 for r in repo.revs("::%ln", common):
2310 2319 commonnodes.add(cl.node(r))
2311 2320 if commonnodes:
2312 2321 # XXX: we should only send the filelogs (and treemanifest). user
2313 2322 # already has the changelog and manifest
2314 2323 packer = changegroup.getbundler(cgversion, repo,
2315 2324 oldmatcher=oldmatcher,
2316 2325 matcher=newmatcher,
2317 2326 fullnodes=commonnodes)
2318 2327 cgdata = packer.generate({nodemod.nullid}, list(commonnodes),
2319 2328 False, 'narrow_widen', changelog=False)
2320 2329
2321 2330 part = bundler.newpart('changegroup', data=cgdata)
2322 2331 part.addparam('version', cgversion)
2323 2332 if 'treemanifest' in repo.requirements:
2324 2333 part.addparam('treemanifest', '1')
2325 2334
2326 2335 return bundler
@@ -1,157 +1,189 b''
1 1 #require no-chg
2 2
3 3 $ hg init repo
4 4 $ cd repo
5 5 $ echo foo > foo
6 6 $ hg ci -qAm 'add foo'
7 7 $ echo >> foo
8 8 $ hg ci -m 'change foo'
9 9 $ hg up -qC 0
10 10 $ echo bar > bar
11 11 $ hg ci -qAm 'add bar'
12 12
13 13 $ hg log
14 14 changeset: 2:effea6de0384
15 15 tag: tip
16 16 parent: 0:bbd179dfa0a7
17 17 user: test
18 18 date: Thu Jan 01 00:00:00 1970 +0000
19 19 summary: add bar
20 20
21 21 changeset: 1:ed1b79f46b9a
22 22 user: test
23 23 date: Thu Jan 01 00:00:00 1970 +0000
24 24 summary: change foo
25 25
26 26 changeset: 0:bbd179dfa0a7
27 27 user: test
28 28 date: Thu Jan 01 00:00:00 1970 +0000
29 29 summary: add foo
30 30
31 31 $ cd ..
32 32
33 33 Test pullbundle functionality
34 34
35 35 $ cd repo
36 36 $ cat <<EOF > .hg/hgrc
37 37 > [server]
38 38 > pullbundle = True
39 39 > [extensions]
40 40 > blackbox =
41 41 > EOF
42 42 $ hg bundle --base null -r 0 .hg/0.hg
43 43 1 changesets found
44 44 $ hg bundle --base 0 -r 1 .hg/1.hg
45 45 1 changesets found
46 46 $ hg bundle --base 1 -r 2 .hg/2.hg
47 47 1 changesets found
48 48 $ cat <<EOF > .hg/pullbundles.manifest
49 49 > 2.hg BUNDLESPEC=none-v2 heads=effea6de0384e684f44435651cb7bd70b8735bd4 bases=bbd179dfa0a71671c253b3ae0aa1513b60d199fa
50 50 > 1.hg BUNDLESPEC=bzip2-v2 heads=ed1b79f46b9a29f5a6efa59cf12fcfca43bead5a bases=bbd179dfa0a71671c253b3ae0aa1513b60d199fa
51 51 > 0.hg BUNDLESPEC=gzip-v2 heads=bbd179dfa0a71671c253b3ae0aa1513b60d199fa
52 52 > EOF
53 53 $ hg --config blackbox.track=debug --debug serve -p $HGPORT2 -d --pid-file=../repo.pid
54 54 listening at http://*:$HGPORT2/ (bound to $LOCALIP:$HGPORT2) (glob) (?)
55 55 $ cat ../repo.pid >> $DAEMON_PIDS
56 56 $ cd ..
57 57 $ hg clone -r 0 http://localhost:$HGPORT2/ repo.pullbundle
58 58 adding changesets
59 59 adding manifests
60 60 adding file changes
61 61 added 1 changesets with 1 changes to 1 files
62 62 new changesets bbd179dfa0a7 (1 drafts)
63 63 updating to branch default
64 64 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
65 65 $ cd repo.pullbundle
66 66 $ hg pull -r 1
67 67 pulling from http://localhost:$HGPORT2/
68 68 searching for changes
69 69 adding changesets
70 70 adding manifests
71 71 adding file changes
72 72 added 1 changesets with 1 changes to 1 files
73 73 new changesets ed1b79f46b9a (1 drafts)
74 74 (run 'hg update' to get a working copy)
75 75 $ hg pull -r 2
76 76 pulling from http://localhost:$HGPORT2/
77 77 searching for changes
78 78 adding changesets
79 79 adding manifests
80 80 adding file changes
81 81 added 1 changesets with 1 changes to 1 files (+1 heads)
82 82 new changesets effea6de0384 (1 drafts)
83 83 (run 'hg heads' to see heads, 'hg merge' to merge)
84 84 $ cd ..
85 85 $ killdaemons.py
86 86 $ grep 'sending pullbundle ' repo/.hg/blackbox.log
87 87 * sending pullbundle "0.hg" (glob)
88 88 * sending pullbundle "1.hg" (glob)
89 89 * sending pullbundle "2.hg" (glob)
90 90 $ rm repo/.hg/blackbox.log
91 91
92 92 Test pullbundle functionality for incremental pulls
93 93
94 94 $ cd repo
95 95 $ hg --config blackbox.track=debug --debug serve -p $HGPORT2 -d --pid-file=../repo.pid
96 96 listening at http://*:$HGPORT2/ (bound to $LOCALIP:$HGPORT2) (glob) (?)
97 97 $ cat ../repo.pid >> $DAEMON_PIDS
98 98 $ cd ..
99 99 $ hg clone http://localhost:$HGPORT2/ repo.pullbundle2
100 100 requesting all changes
101 101 adding changesets
102 102 adding manifests
103 103 adding file changes
104 104 added 1 changesets with 1 changes to 1 files
105 105 adding changesets
106 106 adding manifests
107 107 adding file changes
108 108 added 1 changesets with 1 changes to 1 files
109 109 adding changesets
110 110 adding manifests
111 111 adding file changes
112 112 added 1 changesets with 1 changes to 1 files (+1 heads)
113 113 new changesets bbd179dfa0a7:ed1b79f46b9a (3 drafts)
114 114 updating to branch default
115 115 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
116 116 $ killdaemons.py
117 117 $ grep 'sending pullbundle ' repo/.hg/blackbox.log
118 118 * sending pullbundle "0.hg" (glob)
119 119 * sending pullbundle "2.hg" (glob)
120 120 * sending pullbundle "1.hg" (glob)
121 121 $ rm repo/.hg/blackbox.log
122 122
123 Test pullbundle functionality for incoming
124
125 $ cd repo
126 $ hg --config blackbox.track=debug --debug serve -p $HGPORT2 -d --pid-file=../repo.pid
127 listening at http://*:$HGPORT2/ (bound to $LOCALIP:$HGPORT2) (glob) (?)
128 $ cat ../repo.pid >> $DAEMON_PIDS
129 $ cd ..
130 $ hg clone http://localhost:$HGPORT2/ repo.pullbundle2a -r 0
131 adding changesets
132 adding manifests
133 adding file changes
134 added 1 changesets with 1 changes to 1 files
135 new changesets bbd179dfa0a7 (1 drafts)
136 updating to branch default
137 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
138 $ cd repo.pullbundle2a
139 $ hg incoming -r ed1b79f46b9a
140 comparing with http://localhost:$HGPORT2/
141 searching for changes
142 changeset: 1:ed1b79f46b9a
143 tag: tip
144 user: test
145 date: Thu Jan 01 00:00:00 1970 +0000
146 summary: change foo
147
148 $ cd ..
149 $ killdaemons.py
150 $ grep 'sending pullbundle ' repo/.hg/blackbox.log
151 * sending pullbundle "0.hg" (glob)
152 * sending pullbundle "1.hg" (glob)
153 $ rm repo/.hg/blackbox.log
154
123 155 Test recovery from misconfigured server sending no new data
124 156
125 157 $ cd repo
126 158 $ cat <<EOF > .hg/pullbundles.manifest
127 159 > 0.hg heads=ed1b79f46b9a29f5a6efa59cf12fcfca43bead5a bases=bbd179dfa0a71671c253b3ae0aa1513b60d199fa
128 160 > 0.hg heads=bbd179dfa0a71671c253b3ae0aa1513b60d199fa
129 161 > EOF
130 162 $ hg --config blackbox.track=debug --debug serve -p $HGPORT2 -d --pid-file=../repo.pid
131 163 listening at http://*:$HGPORT2/ (bound to $LOCALIP:$HGPORT2) (glob) (?)
132 164 $ cat ../repo.pid >> $DAEMON_PIDS
133 165 $ cd ..
134 166 $ hg clone -r 0 http://localhost:$HGPORT2/ repo.pullbundle3
135 167 adding changesets
136 168 adding manifests
137 169 adding file changes
138 170 added 1 changesets with 1 changes to 1 files
139 171 new changesets bbd179dfa0a7 (1 drafts)
140 172 updating to branch default
141 173 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
142 174 $ cd repo.pullbundle3
143 175 $ hg pull -r 1
144 176 pulling from http://localhost:$HGPORT2/
145 177 searching for changes
146 178 adding changesets
147 179 adding manifests
148 180 adding file changes
149 181 added 0 changesets with 0 changes to 1 files
150 182 abort: 00changelog.i@ed1b79f46b9a: no node!
151 183 [255]
152 184 $ cd ..
153 185 $ killdaemons.py
154 186 $ grep 'sending pullbundle ' repo/.hg/blackbox.log
155 187 * sending pullbundle "0.hg" (glob)
156 188 * sending pullbundle "0.hg" (glob)
157 189 $ rm repo/.hg/blackbox.log
General Comments 0
You need to be logged in to leave comments. Login now