~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/bundle/serializer/v4.py

Compare URLs in RemoteRepository.__eq__, rather than '_client' attributes.

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
# Copyright (C) 2007 Canonical Ltd
 
2
#
 
3
# This program is free software; you can redistribute it and/or modify
 
4
# it under the terms of the GNU General Public License as published by
 
5
# the Free Software Foundation; either version 2 of the License, or
 
6
# (at your option) any later version.
 
7
#
 
8
# This program is distributed in the hope that it will be useful,
 
9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
 
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 
11
# GNU General Public License for more details.
 
12
#
 
13
# You should have received a copy of the GNU General Public License
 
14
# along with this program; if not, write to the Free Software
 
15
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 
16
 
 
17
from cStringIO import StringIO
 
18
import bz2
 
19
import re
 
20
 
 
21
from bzrlib import (
 
22
    diff,
 
23
    errors,
 
24
    iterablefile,
 
25
    multiparent,
 
26
    osutils,
 
27
    pack,
 
28
    revision as _mod_revision,
 
29
    trace,
 
30
    xml_serializer,
 
31
    )
 
32
from bzrlib.bundle import bundle_data, serializer
 
33
from bzrlib.util import bencode
 
34
 
 
35
 
 
36
class BundleWriter(object):
 
37
    """Writer for bundle-format files.
 
38
 
 
39
    This serves roughly the same purpose as ContainerReader, but acts as a
 
40
    layer on top of it.
 
41
 
 
42
    Provides ways of writing the specific record types supported this bundle
 
43
    format.
 
44
    """
 
45
 
 
46
    def __init__(self, fileobj):
 
47
        self._container = pack.ContainerWriter(self._write_encoded)
 
48
        self._fileobj = fileobj
 
49
        self._compressor = bz2.BZ2Compressor()
 
50
 
 
51
    def _write_encoded(self, bytes):
 
52
        """Write bzip2-encoded bytes to the file"""
 
53
        self._fileobj.write(self._compressor.compress(bytes))
 
54
 
 
55
    def begin(self):
 
56
        """Start writing the bundle"""
 
57
        self._fileobj.write(serializer._get_bundle_header(
 
58
            serializer.v4_string))
 
59
        self._fileobj.write('#\n')
 
60
        self._container.begin()
 
61
 
 
62
    def end(self):
 
63
        """Finish writing the bundle"""
 
64
        self._container.end()
 
65
        self._fileobj.write(self._compressor.flush())
 
66
 
 
67
    def add_multiparent_record(self, mp_bytes, sha1, parents, repo_kind,
 
68
                               revision_id, file_id):
 
69
        """Add a record for a multi-parent diff
 
70
 
 
71
        :mp_bytes: A multi-parent diff, as a bytestring
 
72
        :sha1: The sha1 hash of the fulltext
 
73
        :parents: a list of revision-ids of the parents
 
74
        :repo_kind: The kind of object in the repository.  May be 'file' or
 
75
            'inventory'
 
76
        :revision_id: The revision id of the mpdiff being added.
 
77
        :file_id: The file-id of the file, or None for inventories.
 
78
        """
 
79
        metadata = {'parents': parents,
 
80
                    'storage_kind': 'mpdiff',
 
81
                    'sha1': sha1}
 
82
        self._add_record(mp_bytes, metadata, repo_kind, revision_id, file_id)
 
83
 
 
84
    def add_fulltext_record(self, bytes, parents, repo_kind, revision_id):
 
85
        """Add a record for a fulltext
 
86
 
 
87
        :bytes: The fulltext, as a bytestring
 
88
        :parents: a list of revision-ids of the parents
 
89
        :repo_kind: The kind of object in the repository.  May be 'revision' or
 
90
            'signature'
 
91
        :revision_id: The revision id of the fulltext being added.
 
92
        """
 
93
        metadata = {'parents': parents,
 
94
                    'storage_kind': 'mpdiff'}
 
95
        self._add_record(bytes, {'parents': parents,
 
96
            'storage_kind': 'fulltext'}, repo_kind, revision_id, None)
 
97
 
 
98
    def add_info_record(self, **kwargs):
 
99
        """Add an info record to the bundle
 
100
 
 
101
        Any parameters may be supplied, except 'self' and 'storage_kind'.
 
102
        Values must be lists, strings, integers, dicts, or a combination.
 
103
        """
 
104
        kwargs['storage_kind'] = 'header'
 
105
        self._add_record(None, kwargs, 'info', None, None)
 
106
 
 
107
    @staticmethod
 
108
    def encode_name(content_kind, revision_id, file_id=None):
 
109
        """Encode semantic ids as a container name"""
 
110
        assert content_kind in ('revision', 'file', 'inventory', 'signature',
 
111
                                'info')
 
112
 
 
113
        if content_kind == 'file':
 
114
            assert file_id is not None
 
115
        else:
 
116
            assert file_id is None
 
117
        if content_kind == 'info':
 
118
            assert revision_id is None
 
119
        else:
 
120
            assert revision_id is not None
 
121
        names = [n.replace('/', '//') for n in
 
122
                 (content_kind, revision_id, file_id) if n is not None]
 
123
        return '/'.join(names)
 
124
 
 
125
    def _add_record(self, bytes, metadata, repo_kind, revision_id, file_id):
 
126
        """Add a bundle record to the container.
 
127
 
 
128
        Most bundle records are recorded as header/body pairs, with the
 
129
        body being nameless.  Records with storage_kind 'header' have no
 
130
        body.
 
131
        """
 
132
        name = self.encode_name(repo_kind, revision_id, file_id)
 
133
        encoded_metadata = bencode.bencode(metadata)
 
134
        self._container.add_bytes_record(encoded_metadata, [name])
 
135
        if metadata['storage_kind'] != 'header':
 
136
            self._container.add_bytes_record(bytes, [])
 
137
 
 
138
 
 
139
class BundleReader(object):
 
140
    """Reader for bundle-format files.
 
141
 
 
142
    This serves roughly the same purpose as ContainerReader, but acts as a
 
143
    layer on top of it, providing metadata, a semantic name, and a record
 
144
    body
 
145
    """
 
146
 
 
147
    def __init__(self, fileobj):
 
148
        line = fileobj.readline()
 
149
        if line != '\n':
 
150
            fileobj.readline()
 
151
        self.patch_lines = []
 
152
        self._container = pack.ContainerReader(
 
153
            iterablefile.IterableFile(self.iter_decode(fileobj)))
 
154
 
 
155
    @staticmethod
 
156
    def iter_decode(fileobj):
 
157
        """Iterate through decoded fragments of the file"""
 
158
        decompressor = bz2.BZ2Decompressor()
 
159
        for line in fileobj:
 
160
            yield decompressor.decompress(line)
 
161
 
 
162
    @staticmethod
 
163
    def decode_name(name):
 
164
        """Decode a name from its container form into a semantic form
 
165
 
 
166
        :retval: content_kind, revision_id, file_id
 
167
        """
 
168
        segments = re.split('(//?)', name)
 
169
        names = ['']
 
170
        for segment in segments:
 
171
            if segment == '//':
 
172
                names[-1] += '/'
 
173
            elif segment == '/':
 
174
                names.append('')
 
175
            else:
 
176
                names[-1] += segment
 
177
        content_kind = names[0]
 
178
        revision_id = None
 
179
        file_id = None
 
180
        if len(names) > 1:
 
181
            revision_id = names[1]
 
182
        if len(names) > 2:
 
183
            file_id = names[2]
 
184
        return content_kind, revision_id, file_id
 
185
 
 
186
    def iter_records(self):
 
187
        """Iterate through bundle records
 
188
 
 
189
        :return: a generator of (bytes, metadata, content_kind, revision_id,
 
190
            file_id)
 
191
        """
 
192
        iterator = self._container.iter_records()
 
193
        for names, meta_bytes in iterator:
 
194
            if len(names) != 1:
 
195
                raise errors.BadBundle('Record has %d names instead of 1'
 
196
                                       % len(names))
 
197
            metadata = bencode.bdecode(meta_bytes(None))
 
198
            if metadata['storage_kind'] == 'header':
 
199
                bytes = None
 
200
            else:
 
201
                _unused, bytes = iterator.next()
 
202
                bytes = bytes(None)
 
203
            yield (bytes, metadata) + self.decode_name(names[0])
 
204
 
 
205
 
 
206
class BundleSerializerV4(serializer.BundleSerializer):
 
207
    """Implement the high-level bundle interface"""
 
208
 
 
209
    def write(self, repository, revision_ids, forced_bases, fileobj):
 
210
        """Write a bundle to a file-like object
 
211
 
 
212
        For backwards-compatibility only
 
213
        """
 
214
        write_op = BundleWriteOperation.from_old_args(repository, revision_ids,
 
215
                                                      forced_bases, fileobj)
 
216
        return write_op.do_write()
 
217
 
 
218
    def write_bundle(self, repository, target, base, fileobj):
 
219
        """Write a bundle to a file object
 
220
 
 
221
        :param repository: The repository to retrieve revision data from
 
222
        :param target: The head revision to include ancestors of
 
223
        :param base: The ancestor of the target to stop including acestors
 
224
            at.
 
225
        :param fileobj: The file-like object to write to
 
226
        """
 
227
        write_op =  BundleWriteOperation(base, target, repository, fileobj)
 
228
        return write_op.do_write()
 
229
 
 
230
    def read(self, file):
 
231
        """return a reader object for a given file"""
 
232
        bundle = BundleInfoV4(file, self)
 
233
        return bundle
 
234
 
 
235
    @staticmethod
 
236
    def get_source_serializer(info):
 
237
        """Retrieve the serializer for a given info object"""
 
238
        return xml_serializer.format_registry.get(info['serializer'])
 
239
 
 
240
 
 
241
class BundleWriteOperation(object):
 
242
    """Perform the operation of writing revisions to a bundle"""
 
243
 
 
244
    @classmethod
 
245
    def from_old_args(cls, repository, revision_ids, forced_bases, fileobj):
 
246
        """Create a BundleWriteOperation from old-style arguments"""
 
247
        base, target = cls.get_base_target(revision_ids, forced_bases,
 
248
                                           repository)
 
249
        return BundleWriteOperation(base, target, repository, fileobj,
 
250
                                    revision_ids)
 
251
 
 
252
    def __init__(self, base, target, repository, fileobj, revision_ids=None):
 
253
        self.base = base
 
254
        self.target = target
 
255
        self.repository = repository
 
256
        bundle = BundleWriter(fileobj)
 
257
        self.bundle = bundle
 
258
        self.base_ancestry = set(repository.get_ancestry(base,
 
259
                                                         topo_sorted=False))
 
260
        if revision_ids is not None:
 
261
            self.revision_ids = revision_ids
 
262
        else:
 
263
            revision_ids = set(repository.get_ancestry(target,
 
264
                                                       topo_sorted=False))
 
265
            self.revision_ids = revision_ids.difference(self.base_ancestry)
 
266
 
 
267
    def do_write(self):
 
268
        """Write all data to the bundle"""
 
269
        self.bundle.begin()
 
270
        self.write_info()
 
271
        self.write_files()
 
272
        self.write_revisions()
 
273
        self.bundle.end()
 
274
        return self.revision_ids
 
275
 
 
276
    def write_info(self):
 
277
        """Write format info"""
 
278
        serializer_format = self.repository.get_serializer_format()
 
279
        supports_rich_root = {True: 1, False: 0}[
 
280
            self.repository.supports_rich_root()]
 
281
        self.bundle.add_info_record(serializer=serializer_format,
 
282
                                    supports_rich_root=supports_rich_root)
 
283
 
 
284
    def iter_file_revisions(self):
 
285
        """Iterate through all relevant revisions of all files.
 
286
 
 
287
        This is the correct implementation, but is not compatible with bzr.dev,
 
288
        because certain old revisions were not converted correctly, and have
 
289
        the wrong "revision" marker in inventories.
 
290
        """
 
291
        transaction = self.repository.get_transaction()
 
292
        altered = self.repository.fileids_altered_by_revision_ids(
 
293
            self.revision_ids)
 
294
        for file_id, file_revision_ids in altered.iteritems():
 
295
            vf = self.repository.weave_store.get_weave(file_id, transaction)
 
296
            yield vf, file_id, file_revision_ids
 
297
 
 
298
    def iter_file_revisions_aggressive(self):
 
299
        """Iterate through all relevant revisions of all files.
 
300
 
 
301
        This uses the standard iter_file_revisions to determine what revisions
 
302
        are referred to by inventories, but then uses the versionedfile to
 
303
        determine what the build-dependencies of each required revision.
 
304
 
 
305
        All build dependencies which are not ancestors of the base revision
 
306
        are emitted.
 
307
        """
 
308
        for vf, file_id, file_revision_ids in self.iter_file_revisions():
 
309
            new_revision_ids = set()
 
310
            pending = list(file_revision_ids)
 
311
            while len(pending) > 0:
 
312
                revision_id = pending.pop()
 
313
                if revision_id in new_revision_ids:
 
314
                    continue
 
315
                if revision_id in self.base_ancestry:
 
316
                    continue
 
317
                new_revision_ids.add(revision_id)
 
318
                pending.extend(vf.get_parents(revision_id))
 
319
            yield vf, file_id, new_revision_ids
 
320
 
 
321
    def write_files(self):
 
322
        """Write bundle records for all revisions of all files"""
 
323
        for vf, file_id, revision_ids in self.iter_file_revisions_aggressive():
 
324
            self.add_mp_records('file', file_id, vf, revision_ids)
 
325
 
 
326
    def write_revisions(self):
 
327
        """Write bundle records for all revisions and signatures"""
 
328
        inv_vf = self.repository.get_inventory_weave()
 
329
        revision_order = list(multiparent.topo_iter(inv_vf, self.revision_ids))
 
330
        if self.target is not None and self.target in self.revision_ids:
 
331
            revision_order.remove(self.target)
 
332
            revision_order.append(self.target)
 
333
        self.add_mp_records('inventory', None, inv_vf, revision_order)
 
334
        parents_list = self.repository.get_parents(revision_order)
 
335
        for parents, revision_id in zip(parents_list, revision_order):
 
336
            revision_text = self.repository.get_revision_xml(revision_id)
 
337
            self.bundle.add_fulltext_record(revision_text, parents,
 
338
                                       'revision', revision_id)
 
339
            try:
 
340
                self.bundle.add_fulltext_record(
 
341
                    self.repository.get_signature_text(
 
342
                    revision_id), parents, 'signature', revision_id)
 
343
            except errors.NoSuchRevision:
 
344
                pass
 
345
 
 
346
    @staticmethod
 
347
    def get_base_target(revision_ids, forced_bases, repository):
 
348
        """Determine the base and target from old-style revision ids"""
 
349
        if len(revision_ids) == 0:
 
350
            return None, None
 
351
        target = revision_ids[0]
 
352
        base = forced_bases.get(target)
 
353
        if base is None:
 
354
            parents = repository.get_revision(target).parent_ids
 
355
            if len(parents) == 0:
 
356
                base = _mod_revision.NULL_REVISION
 
357
            else:
 
358
                base = parents[0]
 
359
        return base, target
 
360
 
 
361
    def add_mp_records(self, repo_kind, file_id, vf, revision_ids):
 
362
        """Add multi-parent diff records to a bundle"""
 
363
        revision_ids = list(multiparent.topo_iter(vf, revision_ids))
 
364
        mpdiffs = vf.make_mpdiffs(revision_ids)
 
365
        sha1s = vf.get_sha1s(revision_ids)
 
366
        for mpdiff, revision_id, sha1, in zip(mpdiffs, revision_ids, sha1s):
 
367
            parents = vf.get_parents(revision_id)
 
368
            text = ''.join(mpdiff.to_patch())
 
369
            self.bundle.add_multiparent_record(text, sha1, parents, repo_kind,
 
370
                                               revision_id, file_id)
 
371
 
 
372
 
 
373
class BundleInfoV4(object):
 
374
 
 
375
    """Provide (most of) the BundleInfo interface"""
 
376
    def __init__(self, fileobj, serializer):
 
377
        self._fileobj = fileobj
 
378
        self._serializer = serializer
 
379
        self.__real_revisions = None
 
380
        self.__revisions = None
 
381
 
 
382
    def install(self, repository):
 
383
        return self.install_revisions(repository)
 
384
 
 
385
    def install_revisions(self, repository):
 
386
        """Install this bundle's revisions into the specified repository"""
 
387
        repository.lock_write()
 
388
        try:
 
389
            ri = RevisionInstaller(self.get_bundle_reader(),
 
390
                                   self._serializer, repository)
 
391
            return ri.install()
 
392
        finally:
 
393
            repository.unlock()
 
394
 
 
395
    def get_merge_request(self, target_repo):
 
396
        """Provide data for performing a merge
 
397
 
 
398
        Returns suggested base, suggested target, and patch verification status
 
399
        """
 
400
        return None, self.target, 'inapplicable'
 
401
 
 
402
    def get_bundle_reader(self):
 
403
        self._fileobj.seek(0)
 
404
        return BundleReader(self._fileobj)
 
405
 
 
406
    def _get_real_revisions(self):
 
407
        if self.__real_revisions is None:
 
408
            self.__real_revisions = []
 
409
            bundle_reader = self.get_bundle_reader()
 
410
            for bytes, metadata, repo_kind, revision_id, file_id in \
 
411
                bundle_reader.iter_records():
 
412
                if repo_kind == 'info':
 
413
                    serializer =\
 
414
                        self._serializer.get_source_serializer(metadata)
 
415
                if repo_kind == 'revision':
 
416
                    rev = serializer.read_revision_from_string(bytes)
 
417
                    self.__real_revisions.append(rev)
 
418
        return self.__real_revisions
 
419
    real_revisions = property(_get_real_revisions)
 
420
 
 
421
    def _get_revisions(self):
 
422
        if self.__revisions is None:
 
423
            self.__revisions = []
 
424
            for revision in self.real_revisions:
 
425
                self.__revisions.append(
 
426
                    bundle_data.RevisionInfo.from_revision(revision))
 
427
        return self.__revisions
 
428
 
 
429
    revisions = property(_get_revisions)
 
430
 
 
431
    def _get_target(self):
 
432
        return self.revisions[-1].revision_id
 
433
 
 
434
    target = property(_get_target)
 
435
 
 
436
 
 
437
class RevisionInstaller(object):
 
438
    """Installs revisions into a repository"""
 
439
 
 
440
    def __init__(self, container, serializer, repository):
 
441
        self._container = container
 
442
        self._serializer = serializer
 
443
        self._repository = repository
 
444
        self._info = None
 
445
 
 
446
    def install(self):
 
447
        """Perform the installation"""
 
448
        current_file = None
 
449
        current_versionedfile = None
 
450
        pending_file_records = []
 
451
        added_inv = set()
 
452
        target_revision = None
 
453
        for bytes, metadata, repo_kind, revision_id, file_id in\
 
454
            self._container.iter_records():
 
455
            if repo_kind == 'info':
 
456
                assert self._info is None
 
457
                self._handle_info(metadata)
 
458
            if repo_kind != 'file':
 
459
                self._install_mp_records(current_versionedfile,
 
460
                    pending_file_records)
 
461
                current_file = None
 
462
                current_versionedfile = None
 
463
                pending_file_records = []
 
464
                if repo_kind == 'inventory':
 
465
                    self._install_inventory(revision_id, metadata, bytes)
 
466
                if repo_kind == 'revision':
 
467
                    target_revision = revision_id
 
468
                    self._install_revision(revision_id, metadata, bytes)
 
469
                if repo_kind == 'signature':
 
470
                    self._install_signature(revision_id, metadata, bytes)
 
471
            if repo_kind == 'file':
 
472
                if file_id != current_file:
 
473
                    self._install_mp_records(current_versionedfile,
 
474
                        pending_file_records)
 
475
                    current_file = file_id
 
476
                    current_versionedfile = \
 
477
                        self._repository.weave_store.get_weave_or_empty(
 
478
                        file_id, self._repository.get_transaction())
 
479
                    pending_file_records = []
 
480
                if revision_id in current_versionedfile:
 
481
                    continue
 
482
                pending_file_records.append((revision_id, metadata, bytes))
 
483
        self._install_mp_records(current_versionedfile, pending_file_records)
 
484
        return target_revision
 
485
 
 
486
    def _handle_info(self, info):
 
487
        """Extract data from an info record"""
 
488
        self._info = info
 
489
        self._source_serializer = self._serializer.get_source_serializer(info)
 
490
        if (info['supports_rich_root'] == 0 and
 
491
            self._repository.supports_rich_root()):
 
492
            self.update_root = True
 
493
        else:
 
494
            self.update_root = False
 
495
 
 
496
    def _install_mp_records(self, versionedfile, records):
 
497
        if len(records) == 0:
 
498
            return
 
499
        d_func = multiparent.MultiParent.from_patch
 
500
        vf_records = [(r, m['parents'], m['sha1'], d_func(t)) for r, m, t in
 
501
                      records if r not in versionedfile]
 
502
        versionedfile.add_mpdiffs(vf_records)
 
503
 
 
504
    def _install_inventory(self, revision_id, metadata, text):
 
505
        vf = self._repository.get_inventory_weave()
 
506
        if revision_id in vf:
 
507
            return
 
508
        parent_ids = metadata['parents']
 
509
        if self._info['serializer'] == self._repository._serializer.format_num:
 
510
            return self._install_mp_records(vf, [(revision_id, metadata,
 
511
                                                  text)])
 
512
        parents = [self._repository.get_inventory(p)
 
513
                   for p in parent_ids]
 
514
        parent_texts = [self._source_serializer.write_inventory_to_string(p)
 
515
                        for p in parents]
 
516
        target_lines = multiparent.MultiParent.from_patch(text).to_lines(
 
517
            parent_texts)
 
518
        sha1 = osutils.sha_strings(target_lines)
 
519
        if sha1 != metadata['sha1']:
 
520
            raise errors.BadBundle("Can't convert to target format")
 
521
        target_inv = self._source_serializer.read_inventory_from_string(
 
522
            ''.join(target_lines))
 
523
        self._handle_root(target_inv, parent_ids)
 
524
        try:
 
525
            self._repository.add_inventory(revision_id, target_inv, parent_ids)
 
526
        except errors.UnsupportedInventoryKind:
 
527
            raise errors.IncompatibleRevision(repr(self._repository))
 
528
 
 
529
    def _handle_root(self, target_inv, parent_ids):
 
530
        revision_id = target_inv.revision_id
 
531
        if self.update_root:
 
532
            target_inv.root.revision = revision_id
 
533
            store = self._repository.weave_store
 
534
            transaction = self._repository.get_transaction()
 
535
            vf = store.get_weave_or_empty(target_inv.root.file_id, transaction)
 
536
            vf.add_lines(revision_id, parent_ids, [])
 
537
        elif not self._repository.supports_rich_root():
 
538
            if target_inv.root.revision != revision_id:
 
539
                raise errors.IncompatibleRevision(repr(self._repository))
 
540
 
 
541
 
 
542
    def _install_revision(self, revision_id, metadata, text):
 
543
        if self._repository.has_revision(revision_id):
 
544
            return
 
545
        self._repository._add_revision_text(revision_id, text)
 
546
 
 
547
    def _install_signature(self, revision_id, metadata, text):
 
548
        transaction = self._repository.get_transaction()
 
549
        if self._repository._revision_store.has_signature(revision_id,
 
550
                                                          transaction):
 
551
            return
 
552
        self._repository._revision_store.add_revision_signature_text(
 
553
            revision_id, text, transaction)