14
14
# along with this program; if not, write to the Free Software
15
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
17
from cStringIO import StringIO
17
from __future__ import absolute_import
34
from bzrlib.bundle import bundle_data, serializer as bundle_serializer
35
from bzrlib import bencode
34
versionedfile as _mod_versionedfile,
36
from ...bundle import bundle_data, serializer as bundle_serializer
37
from ...i18n import ngettext
38
from ...sixish import (
44
class _MPDiffInventoryGenerator(_mod_versionedfile._MPDiffGenerator):
45
"""Generate Inventory diffs serialized inventories."""
47
def __init__(self, repo, inventory_keys):
48
super(_MPDiffInventoryGenerator, self).__init__(repo.inventories,
54
"""Compute the diffs one at a time."""
55
# This is instead of compute_diffs() since we guarantee our ordering of
56
# inventories, we don't have to do any buffering
57
self._find_needed_keys()
58
# We actually use a slightly different ordering. We grab all of the
59
# parents first, and then grab the ordered requests.
60
needed_ids = [k[-1] for k in self.present_parents]
61
needed_ids.extend([k[-1] for k in self.ordered_keys])
62
inv_to_str = self.repo._serializer.write_inventory_to_string
63
for inv in self.repo.iter_inventories(needed_ids):
64
revision_id = inv.revision_id
66
if key in self.present_parents:
67
# Not a key we will transmit, which is a shame, since because
68
# of that bundles don't work with stacked branches
71
parent_ids = [k[-1] for k in self.parent_map[key]]
72
as_bytes = inv_to_str(inv)
73
self._process_one_record(key, (as_bytes,))
74
if parent_ids is None:
76
diff = self.diffs.pop(key)
77
sha1 = osutils.sha_string(as_bytes)
78
yield revision_id, parent_ids, sha1, diff
38
81
class BundleWriter(object):
216
259
if metadata['storage_kind'] == 'header':
219
_unused, bytes = iterator.next()
262
_unused, bytes = next(iterator)
220
263
yield (bytes, metadata) + self.decode_name(names[0][0])
281
324
parents = graph.get_parent_map(revision_ids)
282
325
self.revision_ids = [r for r in revision_ids if r in parents]
283
self.revision_keys = set([(revid,) for revid in self.revision_ids])
326
self.revision_keys = {(revid,) for revid in self.revision_ids}
285
328
def do_write(self):
286
329
"""Write all data to the bundle"""
287
trace.note('Bundling %d revision(s).', len(self.revision_ids))
330
trace.note(ngettext('Bundling %d revision.', 'Bundling %d revisions.',
331
len(self.revision_ids)), len(self.revision_ids))
288
332
self.repository.lock_read()
290
334
self.bundle.begin()
310
354
altered_fileids = self.repository.fileids_altered_by_revision_ids(
311
355
self.revision_ids)
312
for file_id, revision_ids in altered_fileids.iteritems():
356
for file_id, revision_ids in viewitems(altered_fileids):
313
357
for revision_id in revision_ids:
314
358
text_keys.append((file_id, revision_id))
315
359
self._add_mp_records_keys('file', self.repository.texts, text_keys)
350
394
inventory_key_order = [(r,) for r in revision_order]
351
parent_map = self.repository.inventories.get_parent_map(
353
missing_keys = set(inventory_key_order).difference(parent_map)
355
raise errors.RevisionNotPresent(list(missing_keys)[0],
356
self.repository.inventories)
357
inv_to_str = self.repository._serializer.write_inventory_to_string
358
# Make sure that we grab the parent texts first
360
map(just_parents.update, parent_map.itervalues())
361
just_parents.difference_update(parent_map)
362
# Ignore ghost parents
363
present_parents = self.repository.inventories.get_parent_map(
365
ghost_keys = just_parents.difference(present_parents)
366
needed_inventories = list(present_parents) + inventory_key_order
367
needed_inventories = [k[-1] for k in needed_inventories]
369
for inv in self.repository.iter_inventories(needed_inventories):
370
revision_id = inv.revision_id
372
as_bytes = inv_to_str(inv)
373
# The sha1 is validated as the xml/textual form, not as the
374
# form-in-the-repository
375
sha1 = osutils.sha_string(as_bytes)
376
as_lines = osutils.split_lines(as_bytes)
378
all_lines[key] = as_lines
379
if key in just_parents:
380
# We don't transmit those entries
382
# Create an mpdiff for this text, and add it to the output
383
parent_keys = parent_map[key]
384
# See the comment in VF.make_mpdiffs about how this effects
385
# ordering when there are ghosts present. I think we have a latent
387
parent_lines = [all_lines[p_key] for p_key in parent_keys
388
if p_key not in ghost_keys]
389
diff = multiparent.MultiParent.from_lines(
390
as_lines, parent_lines)
395
generator = _MPDiffInventoryGenerator(self.repository,
397
for revision_id, parent_ids, sha1, diff in generator.iter_diffs():
391
398
text = ''.join(diff.to_patch())
392
parent_ids = [k[-1] for k in parent_keys]
393
399
self.bundle.add_multiparent_record(text, sha1, parent_ids,
394
400
'inventory', revision_id, None)