##// END OF EJS Templates
bundle2: lazy unbundle of part payload...
Pierre-Yves David -
r21019:3dc09f83 default
parent child Browse files
Show More
@@ -1,640 +1,662
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: (16 bits integer)
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. parameter 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 safefly 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 remains simple and we want to discourage any
56 56 crazy usage.
57 57 - Textual data allow easy human inspection of a the 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: (16 bits inter)
68 68
69 69 The total number of Bytes used by the part headers. 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 :typename: alphanumerical part name
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 arbitraty 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 :payload:
117 117
118 118 payload is a series of `<chunksize><chunkdata>`.
119 119
120 120 `chunksize` is a 32 bits integer, `chunkdata` are plain bytes (as much as
121 121 `chunksize` says)` The payload part is concluded by a zero size chunk.
122 122
123 123 The current implementation always produces either zero or one chunk.
124 124 This is an implementation limitation that will ultimatly be lifted.
125 125
126 126 Bundle processing
127 127 ============================
128 128
129 129 Each part is processed in order using a "part handler". Handler are registered
130 130 for a certain part type.
131 131
132 132 The matching of a part to its handler is case insensitive. The case of the
133 133 part type is used to know if a part is mandatory or advisory. If the Part type
134 134 contains any uppercase char it is considered mandatory. When no handler is
135 135 known for a Mandatory part, the process is aborted and an exception is raised.
136 136 If the part is advisory and no handler is known, the part is ignored. When the
137 137 process is aborted, the full bundle is still read from the stream to keep the
138 138 channel usable. But none of the part read from an abort are processed. In the
139 139 future, dropping the stream may become an option for channel we do not care to
140 140 preserve.
141 141 """
142 142
143 143 import util
144 144 import struct
145 145 import urllib
146 146 import string
147 147 import StringIO
148 148
149 149 import changegroup
150 150 from i18n import _
151 151
152 152 _pack = struct.pack
153 153 _unpack = struct.unpack
154 154
155 155 _magicstring = 'HG20'
156 156
157 157 _fstreamparamsize = '>H'
158 158 _fpartheadersize = '>H'
159 159 _fparttypesize = '>B'
160 160 _fpartid = '>I'
161 161 _fpayloadsize = '>I'
162 162 _fpartparamcount = '>BB'
163 163
164 164 preferedchunksize = 4096
165 165
166 166 def _makefpartparamsizes(nbparams):
167 167 """return a struct format to read part parameter sizes
168 168
169 169 The number parameters is variable so we need to build that format
170 170 dynamically.
171 171 """
172 172 return '>'+('BB'*nbparams)
173 173
174 174 parthandlermapping = {}
175 175
176 176 def parthandler(parttype):
177 177 """decorator that register a function as a bundle2 part handler
178 178
179 179 eg::
180 180
181 181 @parthandler('myparttype')
182 182 def myparttypehandler(...):
183 183 '''process a part of type "my part".'''
184 184 ...
185 185 """
186 186 def _decorator(func):
187 187 lparttype = parttype.lower() # enforce lower case matching.
188 188 assert lparttype not in parthandlermapping
189 189 parthandlermapping[lparttype] = func
190 190 return func
191 191 return _decorator
192 192
193 193 class unbundlerecords(object):
194 194 """keep record of what happens during and unbundle
195 195
196 196 New records are added using `records.add('cat', obj)`. Where 'cat' is a
197 197 category of record and obj is an arbitraty object.
198 198
199 199 `records['cat']` will return all entries of this category 'cat'.
200 200
201 201 Iterating on the object itself will yield `('category', obj)` tuples
202 202 for all entries.
203 203
204 204 All iterations happens in chronological order.
205 205 """
206 206
207 207 def __init__(self):
208 208 self._categories = {}
209 209 self._sequences = []
210 210 self._replies = {}
211 211
212 212 def add(self, category, entry, inreplyto=None):
213 213 """add a new record of a given category.
214 214
215 215 The entry can then be retrieved in the list returned by
216 216 self['category']."""
217 217 self._categories.setdefault(category, []).append(entry)
218 218 self._sequences.append((category, entry))
219 219 if inreplyto is not None:
220 220 self.getreplies(inreplyto).add(category, entry)
221 221
222 222 def getreplies(self, partid):
223 223 """get the subrecords that replies to a specific part"""
224 224 return self._replies.setdefault(partid, unbundlerecords())
225 225
226 226 def __getitem__(self, cat):
227 227 return tuple(self._categories.get(cat, ()))
228 228
229 229 def __iter__(self):
230 230 return iter(self._sequences)
231 231
232 232 def __len__(self):
233 233 return len(self._sequences)
234 234
235 235 def __nonzero__(self):
236 236 return bool(self._sequences)
237 237
238 238 class bundleoperation(object):
239 239 """an object that represents a single bundling process
240 240
241 241 Its purpose is to carry unbundle-related objects and states.
242 242
243 243 A new object should be created at the beginning of each bundle processing.
244 244 The object is to be returned by the processing function.
245 245
246 246 The object has very little content now it will ultimately contain:
247 247 * an access to the repo the bundle is applied to,
248 248 * a ui object,
249 249 * a way to retrieve a transaction to add changes to the repo,
250 250 * a way to record the result of processing each part,
251 251 * a way to construct a bundle response when applicable.
252 252 """
253 253
254 254 def __init__(self, repo, transactiongetter):
255 255 self.repo = repo
256 256 self.ui = repo.ui
257 257 self.records = unbundlerecords()
258 258 self.gettransaction = transactiongetter
259 259 self.reply = None
260 260
261 261 class TransactionUnavailable(RuntimeError):
262 262 pass
263 263
264 264 def _notransaction():
265 265 """default method to get a transaction while processing a bundle
266 266
267 267 Raise an exception to highlight the fact that no transaction was expected
268 268 to be created"""
269 269 raise TransactionUnavailable()
270 270
271 271 def processbundle(repo, unbundler, transactiongetter=_notransaction):
272 272 """This function process a bundle, apply effect to/from a repo
273 273
274 274 It iterates over each part then searches for and uses the proper handling
275 275 code to process the part. Parts are processed in order.
276 276
277 277 This is very early version of this function that will be strongly reworked
278 278 before final usage.
279 279
280 280 Unknown Mandatory part will abort the process.
281 281 """
282 282 op = bundleoperation(repo, transactiongetter)
283 283 # todo:
284 284 # - only create reply bundle if requested.
285 285 op.reply = bundle20(op.ui)
286 286 # todo:
287 287 # - replace this is a init function soon.
288 288 # - exception catching
289 289 unbundler.params
290 290 iterparts = iter(unbundler)
291 part = None
291 292 try:
292 293 for part in iterparts:
293 294 parttype = part.type
294 295 # part key are matched lower case
295 296 key = parttype.lower()
296 297 try:
297 298 handler = parthandlermapping[key]
298 299 op.ui.debug('found a handler for part %r\n' % parttype)
299 300 except KeyError:
300 301 if key != parttype: # mandatory parts
301 302 # todo:
302 303 # - use a more precise exception
303 304 raise
304 305 op.ui.debug('ignoring unknown advisory part %r\n' % key)
305 # todo:
306 # - consume the part once we use streaming
306 # consuming the part
307 part.read()
307 308 continue
308 309
309 310 # handler is called outside the above try block so that we don't
310 311 # risk catching KeyErrors from anything other than the
311 312 # parthandlermapping lookup (any KeyError raised by handler()
312 313 # itself represents a defect of a different variety).
313 314 handler(op, part)
315 part.read()
314 316 except Exception:
317 if part is not None:
318 # consume the bundle content
319 part.read()
315 320 for part in iterparts:
316 pass # consume the bundle content
321 # consume the bundle content
322 part.read()
317 323 raise
318 324 return op
319 325
320 326 class bundle20(object):
321 327 """represent an outgoing bundle2 container
322 328
323 329 Use the `addparam` method to add stream level parameter. and `addpart` to
324 330 populate it. Then call `getchunks` to retrieve all the binary chunks of
325 331 datathat compose the bundle2 container."""
326 332
327 333 def __init__(self, ui):
328 334 self.ui = ui
329 335 self._params = []
330 336 self._parts = []
331 337
332 338 def addparam(self, name, value=None):
333 339 """add a stream level parameter"""
334 340 if not name:
335 341 raise ValueError('empty parameter name')
336 342 if name[0] not in string.letters:
337 343 raise ValueError('non letter first character: %r' % name)
338 344 self._params.append((name, value))
339 345
340 346 def addpart(self, part):
341 347 """add a new part to the bundle2 container
342 348
343 349 Parts contains the actuall applicative payload."""
344 350 assert part.id is None
345 351 part.id = len(self._parts) # very cheap counter
346 352 self._parts.append(part)
347 353
348 354 def getchunks(self):
349 355 self.ui.debug('start emission of %s stream\n' % _magicstring)
350 356 yield _magicstring
351 357 param = self._paramchunk()
352 358 self.ui.debug('bundle parameter: %s\n' % param)
353 359 yield _pack(_fstreamparamsize, len(param))
354 360 if param:
355 361 yield param
356 362
357 363 self.ui.debug('start of parts\n')
358 364 for part in self._parts:
359 365 self.ui.debug('bundle part: "%s"\n' % part.type)
360 366 for chunk in part.getchunks():
361 367 yield chunk
362 368 self.ui.debug('end of bundle\n')
363 369 yield '\0\0'
364 370
365 371 def _paramchunk(self):
366 372 """return a encoded version of all stream parameters"""
367 373 blocks = []
368 374 for par, value in self._params:
369 375 par = urllib.quote(par)
370 376 if value is not None:
371 377 value = urllib.quote(value)
372 378 par = '%s=%s' % (par, value)
373 379 blocks.append(par)
374 380 return ' '.join(blocks)
375 381
376 382 class unpackermixin(object):
377 383 """A mixin to extract bytes and struct data from a stream"""
378 384
379 385 def __init__(self, fp):
380 386 self._fp = fp
381 387
382 388 def _unpack(self, format):
383 389 """unpack this struct format from the stream"""
384 390 data = self._readexact(struct.calcsize(format))
385 391 return _unpack(format, data)
386 392
387 393 def _readexact(self, size):
388 394 """read exactly <size> bytes from the stream"""
389 395 return changegroup.readexactly(self._fp, size)
390 396
391 397
392 398 class unbundle20(unpackermixin):
393 399 """interpret a bundle2 stream
394 400
395 401 (this will eventually yield parts)"""
396 402
397 403 def __init__(self, ui, fp):
398 404 self.ui = ui
399 405 super(unbundle20, self).__init__(fp)
400 406 header = self._readexact(4)
401 407 magic, version = header[0:2], header[2:4]
402 408 if magic != 'HG':
403 409 raise util.Abort(_('not a Mercurial bundle'))
404 410 if version != '20':
405 411 raise util.Abort(_('unknown bundle version %s') % version)
406 412 self.ui.debug('start processing of %s stream\n' % header)
407 413
408 414 @util.propertycache
409 415 def params(self):
410 416 """dictionnary of stream level parameters"""
411 417 self.ui.debug('reading bundle2 stream parameters\n')
412 418 params = {}
413 419 paramssize = self._unpack(_fstreamparamsize)[0]
414 420 if paramssize:
415 421 for p in self._readexact(paramssize).split(' '):
416 422 p = p.split('=', 1)
417 423 p = [urllib.unquote(i) for i in p]
418 424 if len(p) < 2:
419 425 p.append(None)
420 426 self._processparam(*p)
421 427 params[p[0]] = p[1]
422 428 return params
423 429
424 430 def _processparam(self, name, value):
425 431 """process a parameter, applying its effect if needed
426 432
427 433 Parameter starting with a lower case letter are advisory and will be
428 434 ignored when unknown. Those starting with an upper case letter are
429 435 mandatory and will this function will raise a KeyError when unknown.
430 436
431 437 Note: no option are currently supported. Any input will be either
432 438 ignored or failing.
433 439 """
434 440 if not name:
435 441 raise ValueError('empty parameter name')
436 442 if name[0] not in string.letters:
437 443 raise ValueError('non letter first character: %r' % name)
438 444 # Some logic will be later added here to try to process the option for
439 445 # a dict of known parameter.
440 446 if name[0].islower():
441 447 self.ui.debug("ignoring unknown parameter %r\n" % name)
442 448 else:
443 449 raise KeyError(name)
444 450
445 451
446 452 def __iter__(self):
447 453 """yield all parts contained in the stream"""
448 454 # make sure param have been loaded
449 455 self.params
450 456 self.ui.debug('start extraction of bundle2 parts\n')
451 457 headerblock = self._readpartheader()
452 458 while headerblock is not None:
453 459 part = unbundlepart(self.ui, headerblock, self._fp)
454 460 yield part
455 461 headerblock = self._readpartheader()
456 462 self.ui.debug('end of bundle2 stream\n')
457 463
458 464 def _readpartheader(self):
459 465 """reads a part header size and return the bytes blob
460 466
461 467 returns None if empty"""
462 468 headersize = self._unpack(_fpartheadersize)[0]
463 469 self.ui.debug('part header size: %i\n' % headersize)
464 470 if headersize:
465 471 return self._readexact(headersize)
466 472 return None
467 473
468 474
469 475 class bundlepart(object):
470 476 """A bundle2 part contains application level payload
471 477
472 478 The part `type` is used to route the part to the application level
473 479 handler.
474 480 """
475 481
476 482 def __init__(self, parttype, mandatoryparams=(), advisoryparams=(),
477 483 data=''):
478 484 self.id = None
479 485 self.type = parttype
480 486 self.data = data
481 487 self.mandatoryparams = mandatoryparams
482 488 self.advisoryparams = advisoryparams
483 489
484 490 def getchunks(self):
485 491 #### header
486 492 ## parttype
487 493 header = [_pack(_fparttypesize, len(self.type)),
488 494 self.type, _pack(_fpartid, self.id),
489 495 ]
490 496 ## parameters
491 497 # count
492 498 manpar = self.mandatoryparams
493 499 advpar = self.advisoryparams
494 500 header.append(_pack(_fpartparamcount, len(manpar), len(advpar)))
495 501 # size
496 502 parsizes = []
497 503 for key, value in manpar:
498 504 parsizes.append(len(key))
499 505 parsizes.append(len(value))
500 506 for key, value in advpar:
501 507 parsizes.append(len(key))
502 508 parsizes.append(len(value))
503 509 paramsizes = _pack(_makefpartparamsizes(len(parsizes) / 2), *parsizes)
504 510 header.append(paramsizes)
505 511 # key, value
506 512 for key, value in manpar:
507 513 header.append(key)
508 514 header.append(value)
509 515 for key, value in advpar:
510 516 header.append(key)
511 517 header.append(value)
512 518 ## finalize header
513 519 headerchunk = ''.join(header)
514 520 yield _pack(_fpartheadersize, len(headerchunk))
515 521 yield headerchunk
516 522 ## payload
517 523 for chunk in self._payloadchunks():
518 524 yield _pack(_fpayloadsize, len(chunk))
519 525 yield chunk
520 526 # end of payload
521 527 yield _pack(_fpayloadsize, 0)
522 528
523 529 def _payloadchunks(self):
524 530 """yield chunks of a the part payload
525 531
526 532 Exists to handle the different methods to provide data to a part."""
527 533 # we only support fixed size data now.
528 534 # This will be improved in the future.
529 535 if util.safehasattr(self.data, 'next'):
530 536 buff = util.chunkbuffer(self.data)
531 537 chunk = buff.read(preferedchunksize)
532 538 while chunk:
533 539 yield chunk
534 540 chunk = buff.read(preferedchunksize)
535 541 elif len(self.data):
536 542 yield self.data
537 543
538 544 class unbundlepart(unpackermixin):
539 545 """a bundle part read from a bundle"""
540 546
541 547 def __init__(self, ui, header, fp):
542 548 super(unbundlepart, self).__init__(fp)
543 549 self.ui = ui
544 550 # unbundle state attr
545 551 self._headerdata = header
546 552 self._headeroffset = 0
553 self._initialized = False
554 self.consumed = False
547 555 # part data
548 556 self.id = None
549 557 self.type = None
550 558 self.mandatoryparams = None
551 559 self.advisoryparams = None
552 self.data = None
553 self._readdata()
560 self._payloadstream = None
561 self._readheader()
554 562
555 563 def _fromheader(self, size):
556 564 """return the next <size> byte from the header"""
557 565 offset = self._headeroffset
558 566 data = self._headerdata[offset:(offset + size)]
559 self._headeroffset += size
567 self._headeroffset = offset + size
560 568 return data
561 569
562 570 def _unpackheader(self, format):
563 571 """read given format from header
564 572
565 573 This automatically compute the size of the format to read."""
566 574 data = self._fromheader(struct.calcsize(format))
567 575 return _unpack(format, data)
568 576
569 def _readdata(self):
577 def _readheader(self):
570 578 """read the header and setup the object"""
571 # some utility to help reading from the header block
572
573 579 typesize = self._unpackheader(_fparttypesize)[0]
574 580 self.type = self._fromheader(typesize)
575 581 self.ui.debug('part type: "%s"\n' % self.type)
576 582 self.id = self._unpackheader(_fpartid)[0]
577 583 self.ui.debug('part id: "%s"\n' % self.id)
578 584 ## reading parameters
579 585 # param count
580 586 mancount, advcount = self._unpackheader(_fpartparamcount)
581 587 self.ui.debug('part parameters: %i\n' % (mancount + advcount))
582 588 # param size
583 589 fparamsizes = _makefpartparamsizes(mancount + advcount)
584 590 paramsizes = self._unpackheader(fparamsizes)
585 591 # make it a list of couple again
586 592 paramsizes = zip(paramsizes[::2], paramsizes[1::2])
587 593 # split mandatory from advisory
588 594 mansizes = paramsizes[:mancount]
589 595 advsizes = paramsizes[mancount:]
590 596 # retrive param value
591 597 manparams = []
592 598 for key, value in mansizes:
593 599 manparams.append((self._fromheader(key), self._fromheader(value)))
594 600 advparams = []
595 601 for key, value in advsizes:
596 602 advparams.append((self._fromheader(key), self._fromheader(value)))
597 603 self.mandatoryparams = manparams
598 604 self.advisoryparams = advparams
599 605 ## part payload
600 payload = []
606 def payloadchunks():
601 607 payloadsize = self._unpack(_fpayloadsize)[0]
602 608 self.ui.debug('payload chunk size: %i\n' % payloadsize)
603 609 while payloadsize:
604 payload.append(self._readexact(payloadsize))
610 yield self._readexact(payloadsize)
605 611 payloadsize = self._unpack(_fpayloadsize)[0]
606 612 self.ui.debug('payload chunk size: %i\n' % payloadsize)
607 self.data = ''.join(payload)
613 self._payloadstream = util.chunkbuffer(payloadchunks())
614 # we read the data, tell it
615 self._initialized = True
616
617 def read(self, size=None):
618 """read payload data"""
619 if not self._initialized:
620 self._readheader()
621 if size is None:
622 data = self._payloadstream.read()
623 else:
624 data = self._payloadstream.read(size)
625 if size is None or len(data) < size:
626 self.consumed = True
627 return data
628
608 629
609 630 @parthandler('changegroup')
610 631 def handlechangegroup(op, inpart):
611 632 """apply a changegroup part on the repo
612 633
613 634 This is a very early implementation that will massive rework before being
614 635 inflicted to any end-user.
615 636 """
616 637 # Make sure we trigger a transaction creation
617 638 #
618 639 # The addchangegroup function will get a transaction object by itself, but
619 640 # we need to make sure we trigger the creation of a transaction object used
620 641 # for the whole processing scope.
621 642 op.gettransaction()
622 data = StringIO.StringIO(inpart.data)
643 data = StringIO.StringIO(inpart.read())
623 644 data.seek(0)
624 645 cg = changegroup.readbundle(data, 'bundle2part')
625 646 ret = changegroup.addchangegroup(op.repo, cg, 'bundle2', 'bundle2')
626 647 op.records.add('changegroup', {'return': ret})
627 648 if op.reply is not None:
628 649 # This is definitly not the final form of this
629 650 # return. But one need to start somewhere.
630 651 part = bundlepart('reply:changegroup', (),
631 652 [('in-reply-to', str(inpart.id)),
632 653 ('return', '%i' % ret)])
633 654 op.reply.addpart(part)
655 assert not inpart.read()
634 656
635 657 @parthandler('reply:changegroup')
636 658 def handlechangegroup(op, inpart):
637 659 p = dict(inpart.advisoryparams)
638 660 ret = int(p['return'])
639 661 op.records.add('changegroup', {'return': ret}, int(p['in-reply-to']))
640 662
@@ -1,678 +1,678
1 1
2 2 Create an extension to test bundle2 API
3 3
4 4 $ cat > bundle2.py << EOF
5 5 > """A small extension to test bundle2 implementation
6 6 >
7 7 > Current bundle2 implementation is far too limited to be used in any core
8 8 > code. We still need to be able to test it while it grow up.
9 9 > """
10 10 >
11 11 > import sys
12 12 > from mercurial import cmdutil
13 13 > from mercurial import util
14 14 > from mercurial import bundle2
15 15 > from mercurial import scmutil
16 16 > from mercurial import discovery
17 17 > from mercurial import changegroup
18 18 > cmdtable = {}
19 19 > command = cmdutil.command(cmdtable)
20 20 >
21 21 > ELEPHANTSSONG = """Patali Dirapata, Cromda Cromda Ripalo, Pata Pata, Ko Ko Ko
22 22 > Bokoro Dipoulito, Rondi Rondi Pepino, Pata Pata, Ko Ko Ko
23 23 > Emana Karassoli, Loucra Loucra Ponponto, Pata Pata, Ko Ko Ko."""
24 24 > assert len(ELEPHANTSSONG) == 178 # future test say 178 bytes, trust it.
25 25 >
26 26 > @bundle2.parthandler('test:song')
27 27 > def songhandler(op, part):
28 28 > """handle a "test:song" bundle2 part, printing the lyrics on stdin"""
29 29 > op.ui.write('The choir starts singing:\n')
30 30 > verses = 0
31 > for line in part.data.split('\n'):
31 > for line in part.read().split('\n'):
32 32 > op.ui.write(' %s\n' % line)
33 33 > verses += 1
34 34 > op.records.add('song', {'verses': verses})
35 35 >
36 36 > @bundle2.parthandler('test:ping')
37 37 > def pinghandler(op, part):
38 38 > op.ui.write('received ping request (id %i)\n' % part.id)
39 39 > if op.reply is not None:
40 40 > rpart = bundle2.bundlepart('test:pong',
41 41 > [('in-reply-to', str(part.id))])
42 42 > op.reply.addpart(rpart)
43 43 >
44 44 > @command('bundle2',
45 45 > [('', 'param', [], 'stream level parameter'),
46 46 > ('', 'unknown', False, 'include an unknown mandatory part in the bundle'),
47 47 > ('', 'parts', False, 'include some arbitrary parts to the bundle'),
48 48 > ('r', 'rev', [], 'includes those changeset in the bundle'),],
49 49 > '[OUTPUTFILE]')
50 50 > def cmdbundle2(ui, repo, path=None, **opts):
51 51 > """write a bundle2 container on standard ouput"""
52 52 > bundler = bundle2.bundle20(ui)
53 53 > for p in opts['param']:
54 54 > p = p.split('=', 1)
55 55 > try:
56 56 > bundler.addparam(*p)
57 57 > except ValueError, exc:
58 58 > raise util.Abort('%s' % exc)
59 59 >
60 60 > revs = opts['rev']
61 61 > if 'rev' in opts:
62 62 > revs = scmutil.revrange(repo, opts['rev'])
63 63 > if revs:
64 64 > # very crude version of a changegroup part creation
65 65 > bundled = repo.revs('%ld::%ld', revs, revs)
66 66 > headmissing = [c.node() for c in repo.set('heads(%ld)', revs)]
67 67 > headcommon = [c.node() for c in repo.set('parents(%ld) - %ld', revs, revs)]
68 68 > outgoing = discovery.outgoing(repo.changelog, headcommon, headmissing)
69 69 > cg = changegroup.getlocalbundle(repo, 'test:bundle2', outgoing, None)
70 70 > def cgchunks(cg=cg):
71 71 > yield 'HG10UN'
72 72 > for c in cg.getchunks():
73 73 > yield c
74 74 > part = bundle2.bundlepart('changegroup', data=cgchunks())
75 75 > bundler.addpart(part)
76 76 >
77 77 > if opts['parts']:
78 78 > part = bundle2.bundlepart('test:empty')
79 79 > bundler.addpart(part)
80 80 > # add a second one to make sure we handle multiple parts
81 81 > part = bundle2.bundlepart('test:empty')
82 82 > bundler.addpart(part)
83 83 > part = bundle2.bundlepart('test:song', data=ELEPHANTSSONG)
84 84 > bundler.addpart(part)
85 85 > part = bundle2.bundlepart('test:math',
86 86 > [('pi', '3.14'), ('e', '2.72')],
87 87 > [('cooking', 'raw')],
88 88 > '42')
89 89 > bundler.addpart(part)
90 90 > if opts['unknown']:
91 91 > part = bundle2.bundlepart('test:UNKNOWN',
92 92 > data='some random content')
93 93 > bundler.addpart(part)
94 94 > if opts['parts']:
95 95 > part = bundle2.bundlepart('test:ping')
96 96 > bundler.addpart(part)
97 97 >
98 98 > if path is None:
99 99 > file = sys.stdout
100 100 > else:
101 101 > file = open(path, 'w')
102 102 >
103 103 > for chunk in bundler.getchunks():
104 104 > file.write(chunk)
105 105 >
106 106 > @command('unbundle2', [], '')
107 107 > def cmdunbundle2(ui, repo, replypath=None):
108 108 > """process a bundle2 stream from stdin on the current repo"""
109 109 > try:
110 110 > tr = None
111 111 > lock = repo.lock()
112 112 > tr = repo.transaction('processbundle')
113 113 > try:
114 114 > unbundler = bundle2.unbundle20(ui, sys.stdin)
115 115 > op = bundle2.processbundle(repo, unbundler, lambda: tr)
116 116 > tr.close()
117 117 > except KeyError, exc:
118 118 > raise util.Abort('missing support for %s' % exc)
119 119 > finally:
120 120 > if tr is not None:
121 121 > tr.release()
122 122 > lock.release()
123 123 > remains = sys.stdin.read()
124 124 > ui.write('%i unread bytes\n' % len(remains))
125 125 > if op.records['song']:
126 126 > totalverses = sum(r['verses'] for r in op.records['song'])
127 127 > ui.write('%i total verses sung\n' % totalverses)
128 128 > for rec in op.records['changegroup']:
129 129 > ui.write('addchangegroup return: %i\n' % rec['return'])
130 130 > if op.reply is not None and replypath is not None:
131 131 > file = open(replypath, 'w')
132 132 > for chunk in op.reply.getchunks():
133 133 > file.write(chunk)
134 134 >
135 135 > @command('statbundle2', [], '')
136 136 > def cmdstatbundle2(ui, repo):
137 137 > """print statistic on the bundle2 container read from stdin"""
138 138 > unbundler = bundle2.unbundle20(ui, sys.stdin)
139 139 > try:
140 140 > params = unbundler.params
141 141 > except KeyError, exc:
142 142 > raise util.Abort('unknown parameters: %s' % exc)
143 143 > ui.write('options count: %i\n' % len(params))
144 144 > for key in sorted(params):
145 145 > ui.write('- %s\n' % key)
146 146 > value = params[key]
147 147 > if value is not None:
148 148 > ui.write(' %s\n' % value)
149 149 > count = 0
150 150 > for p in unbundler:
151 151 > count += 1
152 152 > ui.write(' :%s:\n' % p.type)
153 153 > ui.write(' mandatory: %i\n' % len(p.mandatoryparams))
154 154 > ui.write(' advisory: %i\n' % len(p.advisoryparams))
155 > ui.write(' payload: %i bytes\n' % len(p.data))
155 > ui.write(' payload: %i bytes\n' % len(p.read()))
156 156 > ui.write('parts count: %i\n' % count)
157 157 > EOF
158 158 $ cat >> $HGRCPATH << EOF
159 159 > [extensions]
160 160 > bundle2=$TESTTMP/bundle2.py
161 161 > [server]
162 162 > bundle2=True
163 163 > EOF
164 164
165 165 The extension requires a repo (currently unused)
166 166
167 167 $ hg init main
168 168 $ cd main
169 169 $ touch a
170 170 $ hg add a
171 171 $ hg commit -m 'a'
172 172
173 173
174 174 Empty bundle
175 175 =================
176 176
177 177 - no option
178 178 - no parts
179 179
180 180 Test bundling
181 181
182 182 $ hg bundle2
183 183 HG20\x00\x00\x00\x00 (no-eol) (esc)
184 184
185 185 Test unbundling
186 186
187 187 $ hg bundle2 | hg statbundle2
188 188 options count: 0
189 189 parts count: 0
190 190
191 191 Test old style bundle are detected and refused
192 192
193 193 $ hg bundle --all ../bundle.hg
194 194 1 changesets found
195 195 $ hg statbundle2 < ../bundle.hg
196 196 abort: unknown bundle version 10
197 197 [255]
198 198
199 199 Test parameters
200 200 =================
201 201
202 202 - some options
203 203 - no parts
204 204
205 205 advisory parameters, no value
206 206 -------------------------------
207 207
208 208 Simplest possible parameters form
209 209
210 210 Test generation simple option
211 211
212 212 $ hg bundle2 --param 'caution'
213 213 HG20\x00\x07caution\x00\x00 (no-eol) (esc)
214 214
215 215 Test unbundling
216 216
217 217 $ hg bundle2 --param 'caution' | hg statbundle2
218 218 options count: 1
219 219 - caution
220 220 parts count: 0
221 221
222 222 Test generation multiple option
223 223
224 224 $ hg bundle2 --param 'caution' --param 'meal'
225 225 HG20\x00\x0ccaution meal\x00\x00 (no-eol) (esc)
226 226
227 227 Test unbundling
228 228
229 229 $ hg bundle2 --param 'caution' --param 'meal' | hg statbundle2
230 230 options count: 2
231 231 - caution
232 232 - meal
233 233 parts count: 0
234 234
235 235 advisory parameters, with value
236 236 -------------------------------
237 237
238 238 Test generation
239 239
240 240 $ hg bundle2 --param 'caution' --param 'meal=vegan' --param 'elephants'
241 241 HG20\x00\x1ccaution meal=vegan elephants\x00\x00 (no-eol) (esc)
242 242
243 243 Test unbundling
244 244
245 245 $ hg bundle2 --param 'caution' --param 'meal=vegan' --param 'elephants' | hg statbundle2
246 246 options count: 3
247 247 - caution
248 248 - elephants
249 249 - meal
250 250 vegan
251 251 parts count: 0
252 252
253 253 parameter with special char in value
254 254 ---------------------------------------------------
255 255
256 256 Test generation
257 257
258 258 $ hg bundle2 --param 'e|! 7/=babar%#==tutu' --param simple
259 259 HG20\x00)e%7C%21%207/=babar%25%23%3D%3Dtutu simple\x00\x00 (no-eol) (esc)
260 260
261 261 Test unbundling
262 262
263 263 $ hg bundle2 --param 'e|! 7/=babar%#==tutu' --param simple | hg statbundle2
264 264 options count: 2
265 265 - e|! 7/
266 266 babar%#==tutu
267 267 - simple
268 268 parts count: 0
269 269
270 270 Test unknown mandatory option
271 271 ---------------------------------------------------
272 272
273 273 $ hg bundle2 --param 'Gravity' | hg statbundle2
274 274 abort: unknown parameters: 'Gravity'
275 275 [255]
276 276
277 277 Test debug output
278 278 ---------------------------------------------------
279 279
280 280 bundling debug
281 281
282 282 $ hg bundle2 --debug --param 'e|! 7/=babar%#==tutu' --param simple ../out.hg2
283 283 start emission of HG20 stream
284 284 bundle parameter: e%7C%21%207/=babar%25%23%3D%3Dtutu simple
285 285 start of parts
286 286 end of bundle
287 287
288 288 file content is ok
289 289
290 290 $ cat ../out.hg2
291 291 HG20\x00)e%7C%21%207/=babar%25%23%3D%3Dtutu simple\x00\x00 (no-eol) (esc)
292 292
293 293 unbundling debug
294 294
295 295 $ hg statbundle2 --debug < ../out.hg2
296 296 start processing of HG20 stream
297 297 reading bundle2 stream parameters
298 298 ignoring unknown parameter 'e|! 7/'
299 299 ignoring unknown parameter 'simple'
300 300 options count: 2
301 301 - e|! 7/
302 302 babar%#==tutu
303 303 - simple
304 304 start extraction of bundle2 parts
305 305 part header size: 0
306 306 end of bundle2 stream
307 307 parts count: 0
308 308
309 309
310 310 Test buggy input
311 311 ---------------------------------------------------
312 312
313 313 empty parameter name
314 314
315 315 $ hg bundle2 --param '' --quiet
316 316 abort: empty parameter name
317 317 [255]
318 318
319 319 bad parameter name
320 320
321 321 $ hg bundle2 --param 42babar
322 322 abort: non letter first character: '42babar'
323 323 [255]
324 324
325 325
326 326 Test part
327 327 =================
328 328
329 329 $ hg bundle2 --parts ../parts.hg2 --debug
330 330 start emission of HG20 stream
331 331 bundle parameter:
332 332 start of parts
333 333 bundle part: "test:empty"
334 334 bundle part: "test:empty"
335 335 bundle part: "test:song"
336 336 bundle part: "test:math"
337 337 bundle part: "test:ping"
338 338 end of bundle
339 339
340 340 $ cat ../parts.hg2
341 341 HG20\x00\x00\x00\x11 (esc)
342 342 test:empty\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x11 (esc)
343 343 test:empty\x00\x00\x00\x01\x00\x00\x00\x00\x00\x00\x00\x10 test:song\x00\x00\x00\x02\x00\x00\x00\x00\x00\xb2Patali Dirapata, Cromda Cromda Ripalo, Pata Pata, Ko Ko Ko (esc)
344 344 Bokoro Dipoulito, Rondi Rondi Pepino, Pata Pata, Ko Ko Ko
345 345 Emana Karassoli, Loucra Loucra Ponponto, Pata Pata, Ko Ko Ko.\x00\x00\x00\x00\x00+ test:math\x00\x00\x00\x03\x02\x01\x02\x04\x01\x04\x07\x03pi3.14e2.72cookingraw\x00\x00\x00\x0242\x00\x00\x00\x00\x00\x10 test:ping\x00\x00\x00\x04\x00\x00\x00\x00\x00\x00\x00\x00 (no-eol) (esc)
346 346
347 347
348 348 $ hg statbundle2 < ../parts.hg2
349 349 options count: 0
350 350 :test:empty:
351 351 mandatory: 0
352 352 advisory: 0
353 353 payload: 0 bytes
354 354 :test:empty:
355 355 mandatory: 0
356 356 advisory: 0
357 357 payload: 0 bytes
358 358 :test:song:
359 359 mandatory: 0
360 360 advisory: 0
361 361 payload: 178 bytes
362 362 :test:math:
363 363 mandatory: 2
364 364 advisory: 1
365 365 payload: 2 bytes
366 366 :test:ping:
367 367 mandatory: 0
368 368 advisory: 0
369 369 payload: 0 bytes
370 370 parts count: 5
371 371
372 372 $ hg statbundle2 --debug < ../parts.hg2
373 373 start processing of HG20 stream
374 374 reading bundle2 stream parameters
375 375 options count: 0
376 376 start extraction of bundle2 parts
377 377 part header size: 17
378 378 part type: "test:empty"
379 379 part id: "0"
380 380 part parameters: 0
381 payload chunk size: 0
382 381 :test:empty:
383 382 mandatory: 0
384 383 advisory: 0
384 payload chunk size: 0
385 385 payload: 0 bytes
386 386 part header size: 17
387 387 part type: "test:empty"
388 388 part id: "1"
389 389 part parameters: 0
390 payload chunk size: 0
391 390 :test:empty:
392 391 mandatory: 0
393 392 advisory: 0
393 payload chunk size: 0
394 394 payload: 0 bytes
395 395 part header size: 16
396 396 part type: "test:song"
397 397 part id: "2"
398 398 part parameters: 0
399 payload chunk size: 178
400 payload chunk size: 0
401 399 :test:song:
402 400 mandatory: 0
403 401 advisory: 0
402 payload chunk size: 178
403 payload chunk size: 0
404 404 payload: 178 bytes
405 405 part header size: 43
406 406 part type: "test:math"
407 407 part id: "3"
408 408 part parameters: 3
409 payload chunk size: 2
410 payload chunk size: 0
411 409 :test:math:
412 410 mandatory: 2
413 411 advisory: 1
412 payload chunk size: 2
413 payload chunk size: 0
414 414 payload: 2 bytes
415 415 part header size: 16
416 416 part type: "test:ping"
417 417 part id: "4"
418 418 part parameters: 0
419 payload chunk size: 0
420 419 :test:ping:
421 420 mandatory: 0
422 421 advisory: 0
422 payload chunk size: 0
423 423 payload: 0 bytes
424 424 part header size: 0
425 425 end of bundle2 stream
426 426 parts count: 5
427 427
428 428 Test actual unbundling of test part
429 429 =======================================
430 430
431 431 Process the bundle
432 432
433 433 $ hg unbundle2 --debug < ../parts.hg2
434 434 start processing of HG20 stream
435 435 reading bundle2 stream parameters
436 436 start extraction of bundle2 parts
437 437 part header size: 17
438 438 part type: "test:empty"
439 439 part id: "0"
440 440 part parameters: 0
441 ignoring unknown advisory part 'test:empty'
441 442 payload chunk size: 0
442 ignoring unknown advisory part 'test:empty'
443 443 part header size: 17
444 444 part type: "test:empty"
445 445 part id: "1"
446 446 part parameters: 0
447 ignoring unknown advisory part 'test:empty'
447 448 payload chunk size: 0
448 ignoring unknown advisory part 'test:empty'
449 449 part header size: 16
450 450 part type: "test:song"
451 451 part id: "2"
452 452 part parameters: 0
453 found a handler for part 'test:song'
454 The choir starts singing:
453 455 payload chunk size: 178
454 456 payload chunk size: 0
455 found a handler for part 'test:song'
456 The choir starts singing:
457 457 Patali Dirapata, Cromda Cromda Ripalo, Pata Pata, Ko Ko Ko
458 458 Bokoro Dipoulito, Rondi Rondi Pepino, Pata Pata, Ko Ko Ko
459 459 Emana Karassoli, Loucra Loucra Ponponto, Pata Pata, Ko Ko Ko.
460 460 part header size: 43
461 461 part type: "test:math"
462 462 part id: "3"
463 463 part parameters: 3
464 ignoring unknown advisory part 'test:math'
464 465 payload chunk size: 2
465 466 payload chunk size: 0
466 ignoring unknown advisory part 'test:math'
467 467 part header size: 16
468 468 part type: "test:ping"
469 469 part id: "4"
470 470 part parameters: 0
471 payload chunk size: 0
472 471 found a handler for part 'test:ping'
473 472 received ping request (id 4)
473 payload chunk size: 0
474 474 part header size: 0
475 475 end of bundle2 stream
476 476 0 unread bytes
477 477 3 total verses sung
478 478
479 479 Unbundle with an unknown mandatory part
480 480 (should abort)
481 481
482 482 $ hg bundle2 --parts --unknown ../unknown.hg2
483 483
484 484 $ hg unbundle2 < ../unknown.hg2
485 485 The choir starts singing:
486 486 Patali Dirapata, Cromda Cromda Ripalo, Pata Pata, Ko Ko Ko
487 487 Bokoro Dipoulito, Rondi Rondi Pepino, Pata Pata, Ko Ko Ko
488 488 Emana Karassoli, Loucra Loucra Ponponto, Pata Pata, Ko Ko Ko.
489 489 0 unread bytes
490 490 abort: missing support for 'test:unknown'
491 491 [255]
492 492
493 493 unbundle with a reply
494 494
495 495 $ hg unbundle2 ../reply.hg2 < ../parts.hg2
496 496 The choir starts singing:
497 497 Patali Dirapata, Cromda Cromda Ripalo, Pata Pata, Ko Ko Ko
498 498 Bokoro Dipoulito, Rondi Rondi Pepino, Pata Pata, Ko Ko Ko
499 499 Emana Karassoli, Loucra Loucra Ponponto, Pata Pata, Ko Ko Ko.
500 500 received ping request (id 4)
501 501 0 unread bytes
502 502 3 total verses sung
503 503
504 504 The reply is a bundle
505 505
506 506 $ cat ../reply.hg2
507 507 HG20\x00\x00\x00\x1e test:pong\x00\x00\x00\x00\x01\x00\x0b\x01in-reply-to4\x00\x00\x00\x00\x00\x00 (no-eol) (esc)
508 508
509 509 The reply is valid
510 510
511 511 $ hg statbundle2 < ../reply.hg2
512 512 options count: 0
513 513 :test:pong:
514 514 mandatory: 1
515 515 advisory: 0
516 516 payload: 0 bytes
517 517 parts count: 1
518 518
519 519 Support for changegroup
520 520 ===================================
521 521
522 522 $ hg unbundle $TESTDIR/bundles/rebase.hg
523 523 adding changesets
524 524 adding manifests
525 525 adding file changes
526 526 added 8 changesets with 7 changes to 7 files (+3 heads)
527 527 (run 'hg heads' to see heads, 'hg merge' to merge)
528 528
529 529 $ hg log -G
530 530 o changeset: 8:02de42196ebe
531 531 | tag: tip
532 532 | parent: 6:24b6387c8c8c
533 533 | user: Nicolas Dumazet <nicdumz.commits@gmail.com>
534 534 | date: Sat Apr 30 15:24:48 2011 +0200
535 535 | summary: H
536 536 |
537 537 | o changeset: 7:eea13746799a
538 538 |/| parent: 6:24b6387c8c8c
539 539 | | parent: 5:9520eea781bc
540 540 | | user: Nicolas Dumazet <nicdumz.commits@gmail.com>
541 541 | | date: Sat Apr 30 15:24:48 2011 +0200
542 542 | | summary: G
543 543 | |
544 544 o | changeset: 6:24b6387c8c8c
545 545 | | parent: 1:cd010b8cd998
546 546 | | user: Nicolas Dumazet <nicdumz.commits@gmail.com>
547 547 | | date: Sat Apr 30 15:24:48 2011 +0200
548 548 | | summary: F
549 549 | |
550 550 | o changeset: 5:9520eea781bc
551 551 |/ parent: 1:cd010b8cd998
552 552 | user: Nicolas Dumazet <nicdumz.commits@gmail.com>
553 553 | date: Sat Apr 30 15:24:48 2011 +0200
554 554 | summary: E
555 555 |
556 556 | o changeset: 4:32af7686d403
557 557 | | user: Nicolas Dumazet <nicdumz.commits@gmail.com>
558 558 | | date: Sat Apr 30 15:24:48 2011 +0200
559 559 | | summary: D
560 560 | |
561 561 | o changeset: 3:5fddd98957c8
562 562 | | user: Nicolas Dumazet <nicdumz.commits@gmail.com>
563 563 | | date: Sat Apr 30 15:24:48 2011 +0200
564 564 | | summary: C
565 565 | |
566 566 | o changeset: 2:42ccdea3bb16
567 567 |/ user: Nicolas Dumazet <nicdumz.commits@gmail.com>
568 568 | date: Sat Apr 30 15:24:48 2011 +0200
569 569 | summary: B
570 570 |
571 571 o changeset: 1:cd010b8cd998
572 572 parent: -1:000000000000
573 573 user: Nicolas Dumazet <nicdumz.commits@gmail.com>
574 574 date: Sat Apr 30 15:24:48 2011 +0200
575 575 summary: A
576 576
577 577 @ changeset: 0:3903775176ed
578 578 user: test
579 579 date: Thu Jan 01 00:00:00 1970 +0000
580 580 summary: a
581 581
582 582
583 583 $ hg bundle2 --debug --rev '8+7+5+4' ../rev.hg2
584 584 4 changesets found
585 585 list of changesets:
586 586 32af7686d403cf45b5d95f2d70cebea587ac806a
587 587 9520eea781bcca16c1e15acc0ba14335a0e8e5ba
588 588 eea13746799a9e0bfd88f29d3c2e9dc9389f524f
589 589 02de42196ebee42ef284b6780a87cdc96e8eaab6
590 590 start emission of HG20 stream
591 591 bundle parameter:
592 592 start of parts
593 593 bundle part: "changegroup"
594 594 bundling: 1/4 changesets (25.00%)
595 595 bundling: 2/4 changesets (50.00%)
596 596 bundling: 3/4 changesets (75.00%)
597 597 bundling: 4/4 changesets (100.00%)
598 598 bundling: 1/4 manifests (25.00%)
599 599 bundling: 2/4 manifests (50.00%)
600 600 bundling: 3/4 manifests (75.00%)
601 601 bundling: 4/4 manifests (100.00%)
602 602 bundling: D 1/3 files (33.33%)
603 603 bundling: E 2/3 files (66.67%)
604 604 bundling: H 3/3 files (100.00%)
605 605 end of bundle
606 606
607 607 $ cat ../rev.hg2
608 608 HG20\x00\x00\x00\x12\x0bchangegroup\x00\x00\x00\x00\x00\x00\x00\x00\x06\x19HG10UN\x00\x00\x00\xa42\xafv\x86\xd4\x03\xcfE\xb5\xd9_-p\xce\xbe\xa5\x87\xac\x80j_\xdd\xd9\x89W\xc8\xa5JMCm\xfe\x1d\xa9\xd8\x7f!\xa1\xb9{\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x002\xafv\x86\xd4\x03\xcfE\xb5\xd9_-p\xce\xbe\xa5\x87\xac\x80j\x00\x00\x00\x00\x00\x00\x00)\x00\x00\x00)6e1f4c47ecb533ffd0c8e52cdc88afb6cd39e20c (esc)
609 609 \x00\x00\x00f\x00\x00\x00h\x00\x00\x00\x02D (esc)
610 610 \x00\x00\x00i\x00\x00\x00j\x00\x00\x00\x01D\x00\x00\x00\xa4\x95 \xee\xa7\x81\xbc\xca\x16\xc1\xe1Z\xcc\x0b\xa1C5\xa0\xe8\xe5\xba\xcd\x01\x0b\x8c\xd9\x98\xf3\x98\x1aZ\x81\x15\xf9O\x8d\xa4\xabP`\x89\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x95 \xee\xa7\x81\xbc\xca\x16\xc1\xe1Z\xcc\x0b\xa1C5\xa0\xe8\xe5\xba\x00\x00\x00\x00\x00\x00\x00)\x00\x00\x00)4dece9c826f69490507b98c6383a3009b295837d (esc)
611 611 \x00\x00\x00f\x00\x00\x00h\x00\x00\x00\x02E (esc)
612 612 \x00\x00\x00i\x00\x00\x00j\x00\x00\x00\x01E\x00\x00\x00\xa2\xee\xa17Fy\x9a\x9e\x0b\xfd\x88\xf2\x9d<.\x9d\xc98\x9fRO$\xb68|\x8c\x8c\xae7\x17\x88\x80\xf3\xfa\x95\xde\xd3\xcb\x1c\xf7\x85\x95 \xee\xa7\x81\xbc\xca\x16\xc1\xe1Z\xcc\x0b\xa1C5\xa0\xe8\xe5\xba\xee\xa17Fy\x9a\x9e\x0b\xfd\x88\xf2\x9d<.\x9d\xc98\x9fRO\x00\x00\x00\x00\x00\x00\x00)\x00\x00\x00)365b93d57fdf4814e2b5911d6bacff2b12014441 (esc)
613 613 \x00\x00\x00f\x00\x00\x00h\x00\x00\x00\x00\x00\x00\x00i\x00\x00\x00j\x00\x00\x00\x01G\x00\x00\x00\xa4\x02\xdeB\x19n\xbe\xe4.\xf2\x84\xb6x (esc)
614 614 \x87\xcd\xc9n\x8e\xaa\xb6$\xb68|\x8c\x8c\xae7\x17\x88\x80\xf3\xfa\x95\xde\xd3\xcb\x1c\xf7\x85\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\xdeB\x19n\xbe\xe4.\xf2\x84\xb6x (esc)
615 615 \x87\xcd\xc9n\x8e\xaa\xb6\x00\x00\x00\x00\x00\x00\x00)\x00\x00\x00)8bee48edc7318541fc0013ee41b089276a8c24bf (esc)
616 616 \x00\x00\x00f\x00\x00\x00f\x00\x00\x00\x02H (esc)
617 617 \x00\x00\x00g\x00\x00\x00h\x00\x00\x00\x01H\x00\x00\x00\x00\x00\x00\x00\x8bn\x1fLG\xec\xb53\xff\xd0\xc8\xe5,\xdc\x88\xaf\xb6\xcd9\xe2\x0cf\xa5\xa0\x18\x17\xfd\xf5#\x9c'8\x02\xb5\xb7a\x8d\x05\x1c\x89\xe4\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x002\xafv\x86\xd4\x03\xcfE\xb5\xd9_-p\xce\xbe\xa5\x87\xac\x80j\x00\x00\x00\x81\x00\x00\x00\x81\x00\x00\x00+D\x00c3f1ca2924c16a19b0656a84900e504e5b0aec2d (esc)
618 618 \x00\x00\x00\x8bM\xec\xe9\xc8&\xf6\x94\x90P{\x98\xc68:0 \xb2\x95\x83}\x00}\x8c\x9d\x88\x84\x13%\xf5\xc6\xb0cq\xb3[N\x8a+\x1a\x83\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x95 \xee\xa7\x81\xbc\xca\x16\xc1\xe1Z\xcc\x0b\xa1C5\xa0\xe8\xe5\xba\x00\x00\x00+\x00\x00\x00\xac\x00\x00\x00+E\x009c6fd0350a6c0d0c49d4a9c5017cf07043f54e58 (esc)
619 619 \x00\x00\x00\x8b6[\x93\xd5\x7f\xdfH\x14\xe2\xb5\x91\x1dk\xac\xff+\x12\x01DA(\xa5\x84\xc6^\xf1!\xf8\x9e\xb6j\xb7\xd0\xbc\x15=\x80\x99\xe7\xceM\xec\xe9\xc8&\xf6\x94\x90P{\x98\xc68:0 \xb2\x95\x83}\xee\xa17Fy\x9a\x9e\x0b\xfd\x88\xf2\x9d<.\x9d\xc98\x9fRO\x00\x00\x00V\x00\x00\x00V\x00\x00\x00+F\x0022bfcfd62a21a3287edbd4d656218d0f525ed76a (esc)
620 620 \x00\x00\x00\x97\x8b\xeeH\xed\xc71\x85A\xfc\x00\x13\xeeA\xb0\x89'j\x8c$\xbf(\xa5\x84\xc6^\xf1!\xf8\x9e\xb6j\xb7\xd0\xbc\x15=\x80\x99\xe7\xce\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\xdeB\x19n\xbe\xe4.\xf2\x84\xb6x (esc)
621 621 \x87\xcd\xc9n\x8e\xaa\xb6\x00\x00\x00+\x00\x00\x00V\x00\x00\x00\x00\x00\x00\x00\x81\x00\x00\x00\x81\x00\x00\x00+H\x008500189e74a9e0475e822093bc7db0d631aeb0b4 (esc)
622 622 \x00\x00\x00\x00\x00\x00\x00\x05D\x00\x00\x00b\xc3\xf1\xca)$\xc1j\x19\xb0ej\x84\x90\x0ePN[ (esc)
623 623 \xec-\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x002\xafv\x86\xd4\x03\xcfE\xb5\xd9_-p\xce\xbe\xa5\x87\xac\x80j\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02D (esc)
624 624 \x00\x00\x00\x00\x00\x00\x00\x05E\x00\x00\x00b\x9co\xd05 (esc)
625 625 l\r (no-eol) (esc)
626 626 \x0cI\xd4\xa9\xc5\x01|\xf0pC\xf5NX\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x95 \xee\xa7\x81\xbc\xca\x16\xc1\xe1Z\xcc\x0b\xa1C5\xa0\xe8\xe5\xba\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02E (esc)
627 627 \x00\x00\x00\x00\x00\x00\x00\x05H\x00\x00\x00b\x85\x00\x18\x9et\xa9\xe0G^\x82 \x93\xbc}\xb0\xd61\xae\xb0\xb4\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\xdeB\x19n\xbe\xe4.\xf2\x84\xb6x (esc)
628 628 \x87\xcd\xc9n\x8e\xaa\xb6\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02H (esc)
629 629 \x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00 (no-eol) (esc)
630 630
631 631 $ hg unbundle2 ../rev-replay.hg2 < ../rev.hg2
632 632 adding changesets
633 633 adding manifests
634 634 adding file changes
635 635 added 0 changesets with 0 changes to 3 files
636 636 0 unread bytes
637 637 addchangegroup return: 1
638 638
639 639 $ cat ../rev-replay.hg2
640 640 HG20\x00\x00\x00/\x11reply:changegroup\x00\x00\x00\x00\x00\x02\x0b\x01\x06\x01in-reply-to0return1\x00\x00\x00\x00\x00\x00 (no-eol) (esc)
641 641
642 642 Real world exchange
643 643 =====================
644 644
645 645
646 646 clone --pull
647 647
648 648 $ cd ..
649 649 $ hg clone main other --pull --rev 9520eea781bc
650 650 adding changesets
651 651 adding manifests
652 652 adding file changes
653 653 added 2 changesets with 2 changes to 2 files
654 654 updating to branch default
655 655 2 files updated, 0 files merged, 0 files removed, 0 files unresolved
656 656 $ hg -R other log -G
657 657 @ changeset: 1:9520eea781bc
658 658 | tag: tip
659 659 | user: Nicolas Dumazet <nicdumz.commits@gmail.com>
660 660 | date: Sat Apr 30 15:24:48 2011 +0200
661 661 | summary: E
662 662 |
663 663 o changeset: 0:cd010b8cd998
664 664 user: Nicolas Dumazet <nicdumz.commits@gmail.com>
665 665 date: Sat Apr 30 15:24:48 2011 +0200
666 666 summary: A
667 667
668 668
669 669 pull
670 670
671 671 $ hg -R other pull
672 672 pulling from $TESTTMP/main (glob)
673 673 searching for changes
674 674 adding changesets
675 675 adding manifests
676 676 adding file changes
677 677 added 7 changesets with 6 changes to 6 files (+3 heads)
678 678 (run 'hg heads' to see heads, 'hg merge' to merge)
General Comments 0
You need to be logged in to leave comments. Login now