~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/versionedfile.py

  • Committer: John Arbash Meinel
  • Date: 2008-09-05 02:29:34 UTC
  • mto: (3697.7.4 1.7)
  • mto: This revision was merged to the branch mainline in revision 3748.
  • Revision ID: john@arbash-meinel.com-20080905022934-s8692mbwpkdwi106
Cleanups to the algorithm documentation.

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2005, 2006, 2007, 2008 Canonical Ltd
 
1
# Copyright (C) 2005, 2006 Canonical Ltd
2
2
#
3
3
# Authors:
4
4
#   Johan Rydberg <jrydberg@gnu.org>
15
15
#
16
16
# You should have received a copy of the GNU General Public License
17
17
# along with this program; if not, write to the Free Software
18
 
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
 
18
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
19
19
 
20
20
"""Versioned text file storage api."""
21
21
 
22
22
from copy import copy
23
23
from cStringIO import StringIO
24
24
import os
25
 
import struct
 
25
import urllib
26
26
from zlib import adler32
27
27
 
28
28
from bzrlib.lazy_import import lazy_import
29
29
lazy_import(globals(), """
30
 
import urllib
31
30
 
32
31
from bzrlib import (
33
 
    annotate,
34
32
    errors,
35
 
    groupcompress,
36
 
    index,
37
 
    knit,
38
33
    osutils,
39
34
    multiparent,
40
35
    tsort,
41
36
    revision,
42
37
    ui,
43
38
    )
44
 
from bzrlib.graph import DictParentsProvider, Graph, StackedParentsProvider
 
39
from bzrlib.graph import DictParentsProvider, Graph, _StackedParentsProvider
45
40
from bzrlib.transport.memory import MemoryTransport
46
41
""")
47
42
from bzrlib.inter import InterObject
48
43
from bzrlib.registry import Registry
49
44
from bzrlib.symbol_versioning import *
50
45
from bzrlib.textmerge import TextMerge
51
 
from bzrlib import bencode
52
46
 
53
47
 
54
48
adapter_registry = Registry()
64
58
    'bzrlib.knit', 'FTAnnotatedToUnannotated')
65
59
adapter_registry.register_lazy(('knit-annotated-ft-gz', 'fulltext'),
66
60
    'bzrlib.knit', 'FTAnnotatedToFullText')
67
 
# adapter_registry.register_lazy(('knit-annotated-ft-gz', 'chunked'),
68
 
#     'bzrlib.knit', 'FTAnnotatedToChunked')
69
61
 
70
62
 
71
63
class ContentFactory(object):
72
64
    """Abstract interface for insertion and retrieval from a VersionedFile.
73
 
 
 
65
    
74
66
    :ivar sha1: None, or the sha1 of the content fulltext.
75
67
    :ivar storage_kind: The native storage kind of this factory. One of
76
68
        'mpdiff', 'knit-annotated-ft', 'knit-annotated-delta', 'knit-ft',
91
83
        self.parents = None
92
84
 
93
85
 
94
 
class ChunkedContentFactory(ContentFactory):
95
 
    """Static data content factory.
96
 
 
97
 
    This takes a 'chunked' list of strings. The only requirement on 'chunked' is
98
 
    that ''.join(lines) becomes a valid fulltext. A tuple of a single string
99
 
    satisfies this, as does a list of lines.
100
 
 
101
 
    :ivar sha1: None, or the sha1 of the content fulltext.
102
 
    :ivar storage_kind: The native storage kind of this factory. Always
103
 
        'chunked'
104
 
    :ivar key: The key of this content. Each key is a tuple with a single
105
 
        string in it.
106
 
    :ivar parents: A tuple of parent keys for self.key. If the object has
107
 
        no parent information, None (as opposed to () for an empty list of
108
 
        parents).
109
 
     """
110
 
 
111
 
    def __init__(self, key, parents, sha1, chunks):
112
 
        """Create a ContentFactory."""
113
 
        self.sha1 = sha1
114
 
        self.storage_kind = 'chunked'
115
 
        self.key = key
116
 
        self.parents = parents
117
 
        self._chunks = chunks
118
 
 
119
 
    def get_bytes_as(self, storage_kind):
120
 
        if storage_kind == 'chunked':
121
 
            return self._chunks
122
 
        elif storage_kind == 'fulltext':
123
 
            return ''.join(self._chunks)
124
 
        raise errors.UnavailableRepresentation(self.key, storage_kind,
125
 
            self.storage_kind)
126
 
 
127
 
 
128
86
class FulltextContentFactory(ContentFactory):
129
87
    """Static data content factory.
130
88
 
131
89
    This takes a fulltext when created and just returns that during
132
90
    get_bytes_as('fulltext').
133
 
 
 
91
    
134
92
    :ivar sha1: None, or the sha1 of the content fulltext.
135
93
    :ivar storage_kind: The native storage kind of this factory. Always
136
94
        'fulltext'.
152
110
    def get_bytes_as(self, storage_kind):
153
111
        if storage_kind == self.storage_kind:
154
112
            return self._text
155
 
        elif storage_kind == 'chunked':
156
 
            return [self._text]
157
113
        raise errors.UnavailableRepresentation(self.key, storage_kind,
158
114
            self.storage_kind)
159
115
 
160
116
 
161
117
class AbsentContentFactory(ContentFactory):
162
118
    """A placeholder content factory for unavailable texts.
163
 
 
 
119
    
164
120
    :ivar sha1: None.
165
121
    :ivar storage_kind: 'absent'.
166
122
    :ivar key: The key of this content. Each key is a tuple with a single
175
131
        self.key = key
176
132
        self.parents = None
177
133
 
178
 
    def get_bytes_as(self, storage_kind):
179
 
        raise ValueError('A request was made for key: %s, but that'
180
 
                         ' content is not available, and the calling'
181
 
                         ' code does not handle if it is missing.'
182
 
                         % (self.key,))
183
 
 
184
134
 
185
135
class AdapterFactory(ContentFactory):
186
136
    """A content factory to adapt between key prefix's."""
208
158
 
209
159
class VersionedFile(object):
210
160
    """Versioned text file storage.
211
 
 
 
161
    
212
162
    A versioned file manages versions of line-based text files,
213
163
    keeping track of the originating version for each line.
214
164
 
252
202
    def insert_record_stream(self, stream):
253
203
        """Insert a record stream into this versioned file.
254
204
 
255
 
        :param stream: A stream of records to insert.
 
205
        :param stream: A stream of records to insert. 
256
206
        :return: None
257
207
        :seealso VersionedFile.get_record_stream:
258
208
        """
277
227
            the data back accurately. (Checking the lines have been split
278
228
            correctly is expensive and extremely unlikely to catch bugs so it
279
229
            is not done at runtime unless check_content is True.)
280
 
        :param parent_texts: An optional dictionary containing the opaque
 
230
        :param parent_texts: An optional dictionary containing the opaque 
281
231
            representations of some or all of the parents of version_id to
282
232
            allow delta optimisations.  VERY IMPORTANT: the texts must be those
283
233
            returned by add_lines or data corruption can be caused.
311
261
        parent_texts=None, nostore_sha=None, random_id=False,
312
262
        check_content=True, left_matching_blocks=None):
313
263
        """Add lines to the versioned file, allowing ghosts to be present.
314
 
 
 
264
        
315
265
        This takes the same parameters as add_lines and returns the same.
316
266
        """
317
267
        self._check_write_ok()
341
291
 
342
292
    def get_format_signature(self):
343
293
        """Get a text description of the data encoding in this file.
344
 
 
 
294
        
345
295
        :since: 0.90
346
296
        """
347
297
        raise NotImplementedError(self.get_format_signature)
468
418
        if isinstance(version_ids, basestring):
469
419
            version_ids = [version_ids]
470
420
        raise NotImplementedError(self.get_ancestry)
471
 
 
 
421
        
472
422
    def get_ancestry_with_ghosts(self, version_ids):
473
423
        """Return a list of all ancestors of given version(s). This
474
424
        will not include the null revision.
475
425
 
476
426
        Must raise RevisionNotPresent if any of the given versions are
477
427
        not present in file history.
478
 
 
 
428
        
479
429
        Ghosts that are known about will be included in ancestry list,
480
430
        but are not explicitly marked.
481
431
        """
482
432
        raise NotImplementedError(self.get_ancestry_with_ghosts)
483
 
 
 
433
    
484
434
    def get_parent_map(self, version_ids):
485
435
        """Get a map of the parents of version_ids.
486
436
 
549
499
        unchanged   Alive in both a and b (possibly created in both)
550
500
        new-a       Created in a
551
501
        new-b       Created in b
552
 
        ghost-a     Killed in a, unborn in b
 
502
        ghost-a     Killed in a, unborn in b    
553
503
        ghost-b     Killed in b, unborn in a
554
504
        irrelevant  Not in either revision
555
505
        """
556
506
        raise NotImplementedError(VersionedFile.plan_merge)
557
 
 
 
507
        
558
508
    def weave_merge(self, plan, a_marker=TextMerge.A_MARKER,
559
509
                    b_marker=TextMerge.B_MARKER):
560
510
        return PlanWeaveMerge(plan, a_marker, b_marker).merge_lines()[0]
562
512
 
563
513
class RecordingVersionedFilesDecorator(object):
564
514
    """A minimal versioned files that records calls made on it.
565
 
 
 
515
    
566
516
    Only enough methods have been added to support tests using it to date.
567
517
 
568
518
    :ivar calls: A list of the calls made; can be reset at any time by
570
520
    """
571
521
 
572
522
    def __init__(self, backing_vf):
573
 
        """Create a RecordingVersionedFilesDecorator decorating backing_vf.
574
 
 
 
523
        """Create a RecordingVersionedFileDsecorator decorating backing_vf.
 
524
        
575
525
        :param backing_vf: The versioned file to answer all methods.
576
526
        """
577
527
        self._backing_vf = backing_vf
611
561
        return self._backing_vf.keys()
612
562
 
613
563
 
614
 
class OrderingVersionedFilesDecorator(RecordingVersionedFilesDecorator):
615
 
    """A VF that records calls, and returns keys in specific order.
616
 
 
617
 
    :ivar calls: A list of the calls made; can be reset at any time by
618
 
        assigning [] to it.
619
 
    """
620
 
 
621
 
    def __init__(self, backing_vf, key_priority):
622
 
        """Create a RecordingVersionedFilesDecorator decorating backing_vf.
623
 
 
624
 
        :param backing_vf: The versioned file to answer all methods.
625
 
        :param key_priority: A dictionary defining what order keys should be
626
 
            returned from an 'unordered' get_record_stream request.
627
 
            Keys with lower priority are returned first, keys not present in
628
 
            the map get an implicit priority of 0, and are returned in
629
 
            lexicographical order.
630
 
        """
631
 
        RecordingVersionedFilesDecorator.__init__(self, backing_vf)
632
 
        self._key_priority = key_priority
633
 
 
634
 
    def get_record_stream(self, keys, sort_order, include_delta_closure):
635
 
        self.calls.append(("get_record_stream", list(keys), sort_order,
636
 
            include_delta_closure))
637
 
        if sort_order == 'unordered':
638
 
            def sort_key(key):
639
 
                return (self._key_priority.get(key, 0), key)
640
 
            # Use a defined order by asking for the keys one-by-one from the
641
 
            # backing_vf
642
 
            for key in sorted(keys, key=sort_key):
643
 
                for record in self._backing_vf.get_record_stream([key],
644
 
                                'unordered', include_delta_closure):
645
 
                    yield record
646
 
        else:
647
 
            for record in self._backing_vf.get_record_stream(keys, sort_order,
648
 
                            include_delta_closure):
649
 
                yield record
650
 
 
651
 
 
652
564
class KeyMapper(object):
653
565
    """KeyMappers map between keys and underlying partitioned storage."""
654
566
 
663
575
 
664
576
    def unmap(self, partition_id):
665
577
        """Map a partitioned storage id back to a key prefix.
666
 
 
 
578
        
667
579
        :param partition_id: The underlying partition id.
668
580
        :return: As much of a key (or prefix) as is derivable from the partition
669
581
            id.
701
613
 
702
614
class PrefixMapper(URLEscapeMapper):
703
615
    """A key mapper that extracts the first component of a key.
704
 
 
 
616
    
705
617
    This mapper is for use with a transport based backend.
706
618
    """
707
619
 
740
652
 
741
653
class HashEscapedPrefixMapper(HashPrefixMapper):
742
654
    """Combines the escaped first component of a key with a hash.
743
 
 
 
655
    
744
656
    This mapper is for use with a transport based backend.
745
657
    """
746
658
 
802
714
        check_content=True):
803
715
        """Add a text to the store.
804
716
 
805
 
        :param key: The key tuple of the text to add. If the last element is
806
 
            None, a CHK string will be generated during the addition.
 
717
        :param key: The key tuple of the text to add.
807
718
        :param parents: The parents key tuples of the text to add.
808
719
        :param lines: A list of lines. Each line must be a bytestring. And all
809
720
            of them except the last must be terminated with \n and contain no
813
724
            the data back accurately. (Checking the lines have been split
814
725
            correctly is expensive and extremely unlikely to catch bugs so it
815
726
            is not done at runtime unless check_content is True.)
816
 
        :param parent_texts: An optional dictionary containing the opaque
 
727
        :param parent_texts: An optional dictionary containing the opaque 
817
728
            representations of some or all of the parents of version_id to
818
729
            allow delta optimisations.  VERY IMPORTANT: the texts must be those
819
730
            returned by add_lines or data corruption can be caused.
836
747
        """
837
748
        raise NotImplementedError(self.add_lines)
838
749
 
839
 
    def _add_text(self, key, parents, text, nostore_sha=None, random_id=False):
840
 
        """Add a text to the store.
841
 
 
842
 
        This is a private function for use by CommitBuilder.
843
 
 
844
 
        :param key: The key tuple of the text to add. If the last element is
845
 
            None, a CHK string will be generated during the addition.
846
 
        :param parents: The parents key tuples of the text to add.
847
 
        :param text: A string containing the text to be committed.
848
 
        :param nostore_sha: Raise ExistingContent and do not add the lines to
849
 
            the versioned file if the digest of the lines matches this.
850
 
        :param random_id: If True a random id has been selected rather than
851
 
            an id determined by some deterministic process such as a converter
852
 
            from a foreign VCS. When True the backend may choose not to check
853
 
            for uniqueness of the resulting key within the versioned file, so
854
 
            this should only be done when the result is expected to be unique
855
 
            anyway.
856
 
        :param check_content: If True, the lines supplied are verified to be
857
 
            bytestrings that are correctly formed lines.
858
 
        :return: The text sha1, the number of bytes in the text, and an opaque
859
 
                 representation of the inserted version which can be provided
860
 
                 back to future _add_text calls in the parent_texts dictionary.
861
 
        """
862
 
        # The default implementation just thunks over to .add_lines(),
863
 
        # inefficient, but it works.
864
 
        return self.add_lines(key, parents, osutils.split_lines(text),
865
 
                              nostore_sha=nostore_sha,
866
 
                              random_id=random_id,
867
 
                              check_content=True)
868
 
 
869
750
    def add_mpdiffs(self, records):
870
751
        """Add mpdiffs to this VersionedFile.
871
752
 
884
765
                                  if not mpvf.has_version(p))
885
766
        # It seems likely that adding all the present parents as fulltexts can
886
767
        # easily exhaust memory.
887
 
        chunks_to_lines = osutils.chunks_to_lines
 
768
        split_lines = osutils.split_lines
888
769
        for record in self.get_record_stream(needed_parents, 'unordered',
889
770
            True):
890
771
            if record.storage_kind == 'absent':
891
772
                continue
892
 
            mpvf.add_version(chunks_to_lines(record.get_bytes_as('chunked')),
 
773
            mpvf.add_version(split_lines(record.get_bytes_as('fulltext')),
893
774
                record.key, [])
894
775
        for (key, parent_keys, expected_sha1, mpdiff), lines in\
895
776
            zip(records, mpvf.get_line_list(versions)):
913
794
        raise NotImplementedError(self.annotate)
914
795
 
915
796
    def check(self, progress_bar=None):
916
 
        """Check this object for integrity.
917
 
        
918
 
        :param progress_bar: A progress bar to output as the check progresses.
919
 
        :param keys: Specific keys within the VersionedFiles to check. When
920
 
            this parameter is not None, check() becomes a generator as per
921
 
            get_record_stream. The difference to get_record_stream is that
922
 
            more or deeper checks will be performed.
923
 
        :return: None, or if keys was supplied a generator as per
924
 
            get_record_stream.
925
 
        """
 
797
        """Check this object for integrity."""
926
798
        raise NotImplementedError(self.check)
927
799
 
928
800
    @staticmethod
974
846
        """
975
847
        raise NotImplementedError(self.get_sha1s)
976
848
 
977
 
    has_key = index._has_key_from_parent_map
978
 
 
979
 
    def get_missing_compression_parent_keys(self):
980
 
        """Return an iterable of keys of missing compression parents.
981
 
 
982
 
        Check this after calling insert_record_stream to find out if there are
983
 
        any missing compression parents.  If there are, the records that
984
 
        depend on them are not able to be inserted safely. The precise
985
 
        behaviour depends on the concrete VersionedFiles class in use.
986
 
 
987
 
        Classes that do not support this will raise NotImplementedError.
988
 
        """
989
 
        raise NotImplementedError(self.get_missing_compression_parent_keys)
990
 
 
991
849
    def insert_record_stream(self, stream):
992
850
        """Insert a record stream into this container.
993
851
 
994
 
        :param stream: A stream of records to insert.
 
852
        :param stream: A stream of records to insert. 
995
853
        :return: None
996
854
        :seealso VersionedFile.get_record_stream:
997
855
        """
1041
899
        ghosts = maybe_ghosts - set(self.get_parent_map(maybe_ghosts))
1042
900
        knit_keys.difference_update(ghosts)
1043
901
        lines = {}
1044
 
        chunks_to_lines = osutils.chunks_to_lines
 
902
        split_lines = osutils.split_lines
1045
903
        for record in self.get_record_stream(knit_keys, 'topological', True):
1046
 
            lines[record.key] = chunks_to_lines(record.get_bytes_as('chunked'))
 
904
            lines[record.key] = split_lines(record.get_bytes_as('fulltext'))
1047
905
            # line_block_dict = {}
1048
906
            # for parent, blocks in record.extract_line_blocks():
1049
907
            #   line_blocks[parent] = blocks
1064
922
                parent_lines, left_parent_blocks))
1065
923
        return diffs
1066
924
 
1067
 
    missing_keys = index._missing_keys_from_parent_map
1068
 
 
1069
925
    def _extract_blocks(self, version_id, source, target):
1070
926
        return None
1071
927
 
1138
994
            result.append((prefix + (origin,), line))
1139
995
        return result
1140
996
 
1141
 
    def get_annotator(self):
1142
 
        return annotate.Annotator(self)
1143
 
 
1144
 
    def check(self, progress_bar=None, keys=None):
 
997
    def check(self, progress_bar=None):
1145
998
        """See VersionedFiles.check()."""
1146
 
        # XXX: This is over-enthusiastic but as we only thunk for Weaves today
1147
 
        # this is tolerable. Ideally we'd pass keys down to check() and 
1148
 
        # have the older VersiondFile interface updated too.
1149
999
        for prefix, vf in self._iter_all_components():
1150
1000
            vf.check()
1151
 
        if keys is not None:
1152
 
            return self.get_record_stream(keys, 'unordered', True)
1153
1001
 
1154
1002
    def get_parent_map(self, keys):
1155
1003
        """Get a map of the parents of keys.
1233
1081
    def insert_record_stream(self, stream):
1234
1082
        """Insert a record stream into this container.
1235
1083
 
1236
 
        :param stream: A stream of records to insert.
 
1084
        :param stream: A stream of records to insert. 
1237
1085
        :return: None
1238
1086
        :seealso VersionedFile.get_record_stream:
1239
1087
        """
1360
1208
                lines = self._lines[key]
1361
1209
                parents = self._parents[key]
1362
1210
                pending.remove(key)
1363
 
                yield ChunkedContentFactory(key, parents, None, lines)
 
1211
                yield FulltextContentFactory(key, parents, None,
 
1212
                    ''.join(lines))
1364
1213
        for versionedfile in self.fallback_versionedfiles:
1365
1214
            for record in versionedfile.get_record_stream(
1366
1215
                pending, 'unordered', True):
1387
1236
            result[revision.NULL_REVISION] = ()
1388
1237
        self._providers = self._providers[:1] + self.fallback_versionedfiles
1389
1238
        result.update(
1390
 
            StackedParentsProvider(self._providers).get_parent_map(keys))
 
1239
            _StackedParentsProvider(self._providers).get_parent_map(keys))
1391
1240
        for key, parents in result.iteritems():
1392
1241
            if parents == ():
1393
1242
                result[key] = (revision.NULL_REVISION,)
1396
1245
 
1397
1246
class PlanWeaveMerge(TextMerge):
1398
1247
    """Weave merge that takes a plan as its input.
1399
 
 
 
1248
    
1400
1249
    This exists so that VersionedFile.plan_merge is implementable.
1401
1250
    Most callers will want to use WeaveMerge instead.
1402
1251
    """
1423
1272
                yield(lines_a,)
1424
1273
            else:
1425
1274
                yield (lines_a, lines_b)
1426
 
 
 
1275
       
1427
1276
        # We previously considered either 'unchanged' or 'killed-both' lines
1428
1277
        # to be possible places to resynchronize.  However, assuming agreement
1429
1278
        # on killed-both lines may be too aggressive. -- mbp 20060324
1435
1284
                lines_a = []
1436
1285
                lines_b = []
1437
1286
                ch_a = ch_b = False
1438
 
 
 
1287
                
1439
1288
            if state == 'unchanged':
1440
1289
                if line:
1441
1290
                    yield ([line],)
1457
1306
            elif state == 'conflicted-b':
1458
1307
                ch_b = ch_a = True
1459
1308
                lines_b.append(line)
1460
 
            elif state == 'killed-both':
1461
 
                # This counts as a change, even though there is no associated
1462
 
                # line
1463
 
                ch_b = ch_a = True
1464
1309
            else:
1465
1310
                if state not in ('irrelevant', 'ghost-a', 'ghost-b',
1466
 
                        'killed-base'):
 
1311
                        'killed-base', 'killed-both'):
1467
1312
                    raise AssertionError(state)
1468
1313
        for struct in outstanding_struct():
1469
1314
            yield struct
1472
1317
class WeaveMerge(PlanWeaveMerge):
1473
1318
    """Weave merge that takes a VersionedFile and two versions as its input."""
1474
1319
 
1475
 
    def __init__(self, versionedfile, ver_a, ver_b,
 
1320
    def __init__(self, versionedfile, ver_a, ver_b, 
1476
1321
        a_marker=PlanWeaveMerge.A_MARKER, b_marker=PlanWeaveMerge.B_MARKER):
1477
1322
        plan = versionedfile.plan_merge(ver_a, ver_b)
1478
1323
        PlanWeaveMerge.__init__(self, plan, a_marker, b_marker)
1479
1324
 
1480
1325
 
1481
1326
class VirtualVersionedFiles(VersionedFiles):
1482
 
    """Dummy implementation for VersionedFiles that uses other functions for
 
1327
    """Dummy implementation for VersionedFiles that uses other functions for 
1483
1328
    obtaining fulltexts and parent maps.
1484
1329
 
1485
 
    This is always on the bottom of the stack and uses string keys
 
1330
    This is always on the bottom of the stack and uses string keys 
1486
1331
    (rather than tuples) internally.
1487
1332
    """
1488
1333
 
1490
1335
        """Create a VirtualVersionedFiles.
1491
1336
 
1492
1337
        :param get_parent_map: Same signature as Repository.get_parent_map.
1493
 
        :param get_lines: Should return lines for specified key or None if
 
1338
        :param get_lines: Should return lines for specified key or None if 
1494
1339
                          not available.
1495
1340
        """
1496
1341
        super(VirtualVersionedFiles, self).__init__()
1497
1342
        self._get_parent_map = get_parent_map
1498
1343
        self._get_lines = get_lines
1499
 
 
 
1344
        
1500
1345
    def check(self, progressbar=None):
1501
1346
        """See VersionedFiles.check.
1502
1347
 
1534
1379
            if lines is not None:
1535
1380
                if not isinstance(lines, list):
1536
1381
                    raise AssertionError
1537
 
                yield ChunkedContentFactory((k,), None,
 
1382
                yield FulltextContentFactory((k,), None, 
1538
1383
                        sha1=osutils.sha_strings(lines),
1539
 
                        chunks=lines)
 
1384
                        text=''.join(lines))
1540
1385
            else:
1541
1386
                yield AbsentContentFactory((k,))
1542
1387
 
1543
 
    def iter_lines_added_or_present_in_keys(self, keys, pb=None):
1544
 
        """See VersionedFile.iter_lines_added_or_present_in_versions()."""
1545
 
        for i, (key,) in enumerate(keys):
1546
 
            if pb is not None:
1547
 
                pb.update("Finding changed lines", i, len(keys))
1548
 
            for l in self._get_lines(key):
1549
 
                yield (l, key)
1550
 
 
1551
 
 
1552
 
def network_bytes_to_kind_and_offset(network_bytes):
1553
 
    """Strip of a record kind from the front of network_bytes.
1554
 
 
1555
 
    :param network_bytes: The bytes of a record.
1556
 
    :return: A tuple (storage_kind, offset_of_remaining_bytes)
1557
 
    """
1558
 
    line_end = network_bytes.find('\n')
1559
 
    storage_kind = network_bytes[:line_end]
1560
 
    return storage_kind, line_end + 1
1561
 
 
1562
 
 
1563
 
class NetworkRecordStream(object):
1564
 
    """A record_stream which reconstitures a serialised stream."""
1565
 
 
1566
 
    def __init__(self, bytes_iterator):
1567
 
        """Create a NetworkRecordStream.
1568
 
 
1569
 
        :param bytes_iterator: An iterator of bytes. Each item in this
1570
 
            iterator should have been obtained from a record_streams'
1571
 
            record.get_bytes_as(record.storage_kind) call.
1572
 
        """
1573
 
        self._bytes_iterator = bytes_iterator
1574
 
        self._kind_factory = {'knit-ft-gz':knit.knit_network_to_record,
1575
 
            'knit-delta-gz':knit.knit_network_to_record,
1576
 
            'knit-annotated-ft-gz':knit.knit_network_to_record,
1577
 
            'knit-annotated-delta-gz':knit.knit_network_to_record,
1578
 
            'knit-delta-closure':knit.knit_delta_closure_to_records,
1579
 
            'fulltext':fulltext_network_to_record,
1580
 
            'groupcompress-block':groupcompress.network_block_to_records,
1581
 
            }
1582
 
 
1583
 
    def read(self):
1584
 
        """Read the stream.
1585
 
 
1586
 
        :return: An iterator as per VersionedFiles.get_record_stream().
1587
 
        """
1588
 
        for bytes in self._bytes_iterator:
1589
 
            storage_kind, line_end = network_bytes_to_kind_and_offset(bytes)
1590
 
            for record in self._kind_factory[storage_kind](
1591
 
                storage_kind, bytes, line_end):
1592
 
                yield record
1593
 
 
1594
 
 
1595
 
def fulltext_network_to_record(kind, bytes, line_end):
1596
 
    """Convert a network fulltext record to record."""
1597
 
    meta_len, = struct.unpack('!L', bytes[line_end:line_end+4])
1598
 
    record_meta = bytes[line_end+4:line_end+4+meta_len]
1599
 
    key, parents = bencode.bdecode_as_tuple(record_meta)
1600
 
    if parents == 'nil':
1601
 
        parents = None
1602
 
    fulltext = bytes[line_end+4+meta_len:]
1603
 
    return [FulltextContentFactory(key, parents, None, fulltext)]
1604
 
 
1605
 
 
1606
 
def _length_prefix(bytes):
1607
 
    return struct.pack('!L', len(bytes))
1608
 
 
1609
 
 
1610
 
def record_to_fulltext_bytes(record):
1611
 
    if record.parents is None:
1612
 
        parents = 'nil'
1613
 
    else:
1614
 
        parents = record.parents
1615
 
    record_meta = bencode.bencode((record.key, parents))
1616
 
    record_content = record.get_bytes_as('fulltext')
1617
 
    return "fulltext\n%s%s%s" % (
1618
 
        _length_prefix(record_meta), record_meta, record_content)
1619
 
 
1620
 
 
1621
 
def sort_groupcompress(parent_map):
1622
 
    """Sort and group the keys in parent_map into groupcompress order.
1623
 
 
1624
 
    groupcompress is defined (currently) as reverse-topological order, grouped
1625
 
    by the key prefix.
1626
 
 
1627
 
    :return: A sorted-list of keys
1628
 
    """
1629
 
    # gc-optimal ordering is approximately reverse topological,
1630
 
    # properly grouped by file-id.
1631
 
    per_prefix_map = {}
1632
 
    for item in parent_map.iteritems():
1633
 
        key = item[0]
1634
 
        if isinstance(key, str) or len(key) == 1:
1635
 
            prefix = ''
1636
 
        else:
1637
 
            prefix = key[0]
1638
 
        try:
1639
 
            per_prefix_map[prefix].append(item)
1640
 
        except KeyError:
1641
 
            per_prefix_map[prefix] = [item]
1642
 
 
1643
 
    present_keys = []
1644
 
    for prefix in sorted(per_prefix_map):
1645
 
        present_keys.extend(reversed(tsort.topo_sort(per_prefix_map[prefix])))
1646
 
    return present_keys
 
1388
 
 
1389