~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/repofmt/weaverepo.py

Merge up bzr.dev.

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2007-2011 Canonical Ltd
 
1
# Copyright (C) 2005, 2006, 2007 Canonical Ltd
2
2
#
3
3
# This program is free software; you can redistribute it and/or modify
4
4
# it under the terms of the GNU General Public License as published by
12
12
#
13
13
# You should have received a copy of the GNU General Public License
14
14
# along with this program; if not, write to the Free Software
15
 
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
 
15
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
16
16
 
17
17
"""Deprecated weave-based repository formats.
18
18
 
20
20
ghosts.
21
21
"""
22
22
 
23
 
from __future__ import absolute_import
24
 
 
25
 
import gzip
26
 
import os
27
 
from cStringIO import StringIO
28
 
 
29
 
from bzrlib.lazy_import import lazy_import
30
 
lazy_import(globals(), """
31
 
import itertools
32
 
 
33
 
from bzrlib import (
34
 
    xml5,
35
 
    graph as _mod_graph,
36
 
    ui,
37
 
    )
38
 
""")
39
 
from bzrlib import (
 
23
from StringIO import StringIO
 
24
 
 
25
from bzrlib import (
 
26
    bzrdir,
40
27
    debug,
41
28
    errors,
42
29
    lockable_files,
43
30
    lockdir,
44
31
    osutils,
45
 
    symbol_versioning,
46
 
    trace,
47
 
    tuned_gzip,
48
 
    urlutils,
49
 
    versionedfile,
 
32
    revision as _mod_revision,
50
33
    weave,
51
34
    weavefile,
 
35
    xml5,
52
36
    )
53
37
from bzrlib.decorators import needs_read_lock, needs_write_lock
54
38
from bzrlib.repository import (
55
 
    InterRepository,
56
 
    RepositoryFormatMetaDir,
 
39
    CommitBuilder,
 
40
    MetaDirVersionedFileRepository,
 
41
    MetaDirRepositoryFormat,
 
42
    Repository,
 
43
    RepositoryFormat,
57
44
    )
58
45
from bzrlib.store.text import TextStore
59
 
from bzrlib.versionedfile import (
60
 
    AbsentContentFactory,
61
 
    FulltextContentFactory,
62
 
    VersionedFiles,
63
 
    )
64
 
from bzrlib.vf_repository import (
65
 
    InterSameDataRepository,
66
 
    VersionedFileCommitBuilder,
67
 
    VersionedFileRepository,
68
 
    VersionedFileRepositoryFormat,
69
 
    MetaDirVersionedFileRepository,
70
 
    MetaDirVersionedFileRepositoryFormat,
71
 
    )
72
 
 
73
 
from bzrlib.plugins.weave_fmt import bzrdir as weave_bzrdir
74
 
 
75
 
 
76
 
class AllInOneRepository(VersionedFileRepository):
 
46
from bzrlib.trace import mutter
 
47
 
 
48
 
 
49
class AllInOneRepository(Repository):
77
50
    """Legacy support - the repository behaviour for all-in-one branches."""
78
51
 
79
 
    @property
80
 
    def _serializer(self):
81
 
        return xml5.serializer_v5
82
 
 
83
 
    def _escape(self, file_or_path):
84
 
        if not isinstance(file_or_path, basestring):
85
 
            file_or_path = '/'.join(file_or_path)
86
 
        if file_or_path == '':
87
 
            return u''
88
 
        return urlutils.escape(osutils.safe_unicode(file_or_path))
89
 
 
90
 
    def __init__(self, _format, a_bzrdir):
 
52
    _serializer = xml5.serializer_v5
 
53
 
 
54
    def __init__(self, _format, a_bzrdir, _revision_store, control_store, text_store):
91
55
        # we reuse one control files instance.
92
 
        dir_mode = a_bzrdir._get_dir_mode()
93
 
        file_mode = a_bzrdir._get_file_mode()
 
56
        dir_mode = a_bzrdir._control_files._dir_mode
 
57
        file_mode = a_bzrdir._control_files._file_mode
94
58
 
95
59
        def get_store(name, compressed=True, prefixed=False):
96
60
            # FIXME: This approach of assuming stores are all entirely compressed
97
 
            # or entirely uncompressed is tidy, but breaks upgrade from
98
 
            # some existing branches where there's a mixture; we probably
 
61
            # or entirely uncompressed is tidy, but breaks upgrade from 
 
62
            # some existing branches where there's a mixture; we probably 
99
63
            # still want the option to look for both.
100
 
            relpath = self._escape(name)
101
 
            store = TextStore(a_bzrdir.transport.clone(relpath),
 
64
            relpath = a_bzrdir._control_files._escape(name)
 
65
            store = TextStore(a_bzrdir._control_files._transport.clone(relpath),
102
66
                              prefixed=prefixed, compressed=compressed,
103
67
                              dir_mode=dir_mode,
104
68
                              file_mode=file_mode)
105
69
            return store
106
70
 
107
71
        # not broken out yet because the controlweaves|inventory_store
108
 
        # and texts bits are still different.
 
72
        # and text_store | weave_store bits are still different.
109
73
        if isinstance(_format, RepositoryFormat4):
110
 
            # cannot remove these - there is still no consistent api
 
74
            # cannot remove these - there is still no consistent api 
111
75
            # which allows access to this old info.
112
76
            self.inventory_store = get_store('inventory-store')
113
 
            self._text_store = get_store('text-store')
114
 
        super(AllInOneRepository, self).__init__(_format, a_bzrdir, a_bzrdir._control_files)
 
77
            text_store = get_store('text-store')
 
78
        super(AllInOneRepository, self).__init__(_format, a_bzrdir, a_bzrdir._control_files, _revision_store, control_store, text_store)
 
79
        if control_store is not None:
 
80
            control_store.get_scope = self.get_transaction
 
81
        text_store.get_scope = self.get_transaction
115
82
 
116
83
    @needs_read_lock
117
84
    def _all_possible_ids(self):
118
85
        """Return all the possible revisions that we could find."""
119
86
        if 'evil' in debug.debug_flags:
120
 
            trace.mutter_callsite(
121
 
                3, "_all_possible_ids scales with size of history.")
122
 
        return [key[-1] for key in self.inventories.keys()]
 
87
            mutter_callsite(3, "_all_possible_ids scales with size of history.")
 
88
        return self.get_inventory_weave().versions()
123
89
 
124
90
    @needs_read_lock
125
91
    def _all_revision_ids(self):
126
 
        """Returns a list of all the revision ids in the repository.
 
92
        """Returns a list of all the revision ids in the repository. 
127
93
 
128
 
        These are in as much topological order as the underlying store can
 
94
        These are in as much topological order as the underlying store can 
129
95
        present: for weaves ghosts may lead to a lack of correctness until
130
96
        the reweave updates the parents list.
131
97
        """
132
 
        return [key[-1] for key in self.revisions.keys()]
133
 
 
134
 
    def _activate_new_inventory(self):
135
 
        """Put a replacement inventory.new into use as inventories."""
136
 
        # Copy the content across
137
 
        t = self.bzrdir._control_files._transport
138
 
        t.copy('inventory.new.weave', 'inventory.weave')
139
 
        # delete the temp inventory
140
 
        t.delete('inventory.new.weave')
141
 
        # Check we can parse the new weave properly as a sanity check
142
 
        self.inventories.keys()
143
 
 
144
 
    def _backup_inventory(self):
145
 
        t = self.bzrdir._control_files._transport
146
 
        t.copy('inventory.weave', 'inventory.backup.weave')
147
 
 
148
 
    def _temp_inventories(self):
149
 
        t = self.bzrdir._control_files._transport
150
 
        return self._format._get_inventories(t, self, 'inventory.new')
 
98
        if self._revision_store.text_store.listable():
 
99
            return self._revision_store.all_revision_ids(self.get_transaction())
 
100
        result = self._all_possible_ids()
 
101
        # TODO: jam 20070210 Ensure that _all_possible_ids returns non-unicode
 
102
        #       ids. (It should, since _revision_store's API should change to
 
103
        #       return utf8 revision_ids)
 
104
        return self._eliminate_revisions_not_present(result)
 
105
 
 
106
    def _check_revision_parents(self, revision, inventory):
 
107
        """Private to Repository and Fetch.
 
108
        
 
109
        This checks the parentage of revision in an inventory weave for 
 
110
        consistency and is only applicable to inventory-weave-for-ancestry
 
111
        using repository formats & fetchers.
 
112
        """
 
113
        weave_parents = inventory.get_parent_map(
 
114
            [revision.revision_id])[revision.revision_id]
 
115
        parent_map = inventory.get_parent_map(revision.parent_ids)
 
116
        for parent_id in revision.parent_ids:
 
117
            if parent_id in parent_map:
 
118
                # this parent must not be a ghost.
 
119
                if not parent_id in weave_parents:
 
120
                    # but it is a ghost
 
121
                    raise errors.CorruptRepository(self)
151
122
 
152
123
    def get_commit_builder(self, branch, parents, config, timestamp=None,
153
124
                           timezone=None, committer=None, revprops=None,
154
 
                           revision_id=None, lossy=False):
 
125
                           revision_id=None):
155
126
        self._check_ascii_revisionid(revision_id, self.get_commit_builder)
156
 
        result = VersionedFileCommitBuilder(self, parents, config, timestamp,
157
 
            timezone, committer, revprops, revision_id, lossy=lossy)
 
127
        result = WeaveCommitBuilder(self, parents, config, timestamp, timezone,
 
128
                              committer, revprops, revision_id)
158
129
        self.start_write_group()
159
130
        return result
160
131
 
161
132
    @needs_read_lock
162
133
    def get_revisions(self, revision_ids):
163
134
        revs = self._get_revisions(revision_ids)
 
135
        # weave corruption can lead to absent revision markers that should be
 
136
        # present.
 
137
        # the following test is reasonably cheap (it needs a single weave read)
 
138
        # and the weave is cached in read transactions. In write transactions
 
139
        # it is not cached but typically we only read a small number of
 
140
        # revisions. For knits when they are introduced we will probably want
 
141
        # to ensure that caching write transactions are in use.
 
142
        inv = self.get_inventory_weave()
 
143
        for rev in revs:
 
144
            self._check_revision_parents(rev, inv)
164
145
        return revs
165
146
 
166
 
    def _inventory_add_lines(self, revision_id, parents, lines,
167
 
        check_content=True):
168
 
        """Store lines in inv_vf and return the sha1 of the inventory."""
169
 
        present_parents = self.get_graph().get_parent_map(parents)
170
 
        final_parents = []
171
 
        for parent in parents:
172
 
            if parent in present_parents:
173
 
                final_parents.append((parent,))
174
 
        return self.inventories.add_lines((revision_id,), final_parents, lines,
175
 
            check_content=check_content)[0]
 
147
    def has_revisions(self, revision_ids):
 
148
        """See Repository.has_revisions()."""
 
149
        result = set()
 
150
        transaction = self.get_transaction()
 
151
        for revision_id in revision_ids:
 
152
            if self._revision_store.has_revision_id(revision_id, transaction):
 
153
                result.add(revision_id)
 
154
        return result
176
155
 
 
156
    @needs_read_lock
177
157
    def is_shared(self):
178
158
        """AllInOne repositories cannot be shared."""
179
159
        return False
188
168
        :param new_value: True to restore the default, False to disable making
189
169
                          working trees.
190
170
        """
191
 
        raise errors.RepositoryUpgradeRequired(self.user_url)
 
171
        raise errors.RepositoryUpgradeRequired(self.bzrdir.root_transport.base)
192
172
 
193
173
    def make_working_trees(self):
194
174
        """Returns the policy for making working trees on new branches."""
195
175
        return True
196
176
 
 
177
    def revision_graph_can_have_wrong_parents(self):
 
178
        # XXX: This is an old format that we don't support full checking on, so
 
179
        # just claim that checking for this inconsistency is not required.
 
180
        return False
 
181
 
197
182
 
198
183
class WeaveMetaDirRepository(MetaDirVersionedFileRepository):
199
184
    """A subclass of MetaDirRepository to set weave specific policy."""
200
185
 
201
 
    def __init__(self, _format, a_bzrdir, control_files):
202
 
        super(WeaveMetaDirRepository, self).__init__(_format, a_bzrdir, control_files)
203
 
        self._serializer = _format._serializer
 
186
    _serializer = xml5.serializer_v5
204
187
 
205
188
    @needs_read_lock
206
189
    def _all_possible_ids(self):
207
190
        """Return all the possible revisions that we could find."""
208
191
        if 'evil' in debug.debug_flags:
209
 
            trace.mutter_callsite(
210
 
                3, "_all_possible_ids scales with size of history.")
211
 
        return [key[-1] for key in self.inventories.keys()]
 
192
            mutter_callsite(3, "_all_possible_ids scales with size of history.")
 
193
        return self.get_inventory_weave().versions()
212
194
 
213
195
    @needs_read_lock
214
196
    def _all_revision_ids(self):
215
 
        """Returns a list of all the revision ids in the repository.
 
197
        """Returns a list of all the revision ids in the repository. 
216
198
 
217
 
        These are in as much topological order as the underlying store can
 
199
        These are in as much topological order as the underlying store can 
218
200
        present: for weaves ghosts may lead to a lack of correctness until
219
201
        the reweave updates the parents list.
220
202
        """
221
 
        return [key[-1] for key in self.revisions.keys()]
222
 
 
223
 
    def _activate_new_inventory(self):
224
 
        """Put a replacement inventory.new into use as inventories."""
225
 
        # Copy the content across
226
 
        t = self._transport
227
 
        t.copy('inventory.new.weave', 'inventory.weave')
228
 
        # delete the temp inventory
229
 
        t.delete('inventory.new.weave')
230
 
        # Check we can parse the new weave properly as a sanity check
231
 
        self.inventories.keys()
232
 
 
233
 
    def _backup_inventory(self):
234
 
        t = self._transport
235
 
        t.copy('inventory.weave', 'inventory.backup.weave')
236
 
 
237
 
    def _temp_inventories(self):
238
 
        t = self._transport
239
 
        return self._format._get_inventories(t, self, 'inventory.new')
 
203
        if self._revision_store.text_store.listable():
 
204
            return self._revision_store.all_revision_ids(self.get_transaction())
 
205
        result = self._all_possible_ids()
 
206
        # TODO: jam 20070210 Ensure that _all_possible_ids returns non-unicode
 
207
        #       ids. (It should, since _revision_store's API should change to
 
208
        #       return utf8 revision_ids)
 
209
        return self._eliminate_revisions_not_present(result)
 
210
 
 
211
    def _check_revision_parents(self, revision, inventory):
 
212
        """Private to Repository and Fetch.
 
213
        
 
214
        This checks the parentage of revision in an inventory weave for 
 
215
        consistency and is only applicable to inventory-weave-for-ancestry
 
216
        using repository formats & fetchers.
 
217
        """
 
218
        weave_parents = inventory.get_parent_map(
 
219
            [revision.revision_id])[revision.revision_id]
 
220
        parent_map = inventory.get_parent_map(revision.parent_ids)
 
221
        for parent_id in revision.parent_ids:
 
222
            if parent_id in parent_map:
 
223
                # this parent must not be a ghost.
 
224
                if not parent_id in weave_parents:
 
225
                    # but it is a ghost
 
226
                    raise errors.CorruptRepository(self)
240
227
 
241
228
    def get_commit_builder(self, branch, parents, config, timestamp=None,
242
229
                           timezone=None, committer=None, revprops=None,
243
 
                           revision_id=None, lossy=False):
 
230
                           revision_id=None):
244
231
        self._check_ascii_revisionid(revision_id, self.get_commit_builder)
245
 
        result = VersionedFileCommitBuilder(self, parents, config, timestamp,
246
 
            timezone, committer, revprops, revision_id, lossy=lossy)
 
232
        result = WeaveCommitBuilder(self, parents, config, timestamp, timezone,
 
233
                              committer, revprops, revision_id)
247
234
        self.start_write_group()
248
235
        return result
249
236
 
250
237
    @needs_read_lock
251
238
    def get_revision(self, revision_id):
252
239
        """Return the Revision object for a named revision"""
 
240
        # TODO: jam 20070210 get_revision_reconcile should do this for us
253
241
        r = self.get_revision_reconcile(revision_id)
 
242
        # weave corruption can lead to absent revision markers that should be
 
243
        # present.
 
244
        # the following test is reasonably cheap (it needs a single weave read)
 
245
        # and the weave is cached in read transactions. In write transactions
 
246
        # it is not cached but typically we only read a small number of
 
247
        # revisions. For knits when they are introduced we will probably want
 
248
        # to ensure that caching write transactions are in use.
 
249
        inv = self.get_inventory_weave()
 
250
        self._check_revision_parents(r, inv)
254
251
        return r
255
252
 
256
 
    def _inventory_add_lines(self, revision_id, parents, lines,
257
 
        check_content=True):
258
 
        """Store lines in inv_vf and return the sha1 of the inventory."""
259
 
        present_parents = self.get_graph().get_parent_map(parents)
260
 
        final_parents = []
261
 
        for parent in parents:
262
 
            if parent in present_parents:
263
 
                final_parents.append((parent,))
264
 
        return self.inventories.add_lines((revision_id,), final_parents, lines,
265
 
            check_content=check_content)[0]
266
 
 
267
 
 
268
 
class PreSplitOutRepositoryFormat(VersionedFileRepositoryFormat):
 
253
    def has_revisions(self, revision_ids):
 
254
        """See Repository.has_revisions()."""
 
255
        result = set()
 
256
        transaction = self.get_transaction()
 
257
        for revision_id in revision_ids:
 
258
            if self._revision_store.has_revision_id(revision_id, transaction):
 
259
                result.add(revision_id)
 
260
        return result
 
261
 
 
262
    def revision_graph_can_have_wrong_parents(self):
 
263
        # XXX: This is an old format that we don't support full checking on, so
 
264
        # just claim that checking for this inconsistency is not required.
 
265
        return False
 
266
 
 
267
 
 
268
class PreSplitOutRepositoryFormat(RepositoryFormat):
269
269
    """Base class for the pre split out repository formats."""
270
270
 
271
271
    rich_root_data = False
272
272
    supports_tree_reference = False
273
273
    supports_ghosts = False
274
274
    supports_external_lookups = False
275
 
    supports_chks = False
276
 
    supports_nesting_repositories = True
277
 
    _fetch_order = 'topological'
278
 
    _fetch_reconcile = True
279
 
    fast_deltas = False
280
 
    supports_leaving_lock = False
281
 
    # XXX: This is an old format that we don't support full checking on, so
282
 
    # just claim that checking for this inconsistency is not required.
283
 
    revision_graph_can_have_wrong_parents = False
284
275
 
285
276
    def initialize(self, a_bzrdir, shared=False, _internal=False):
286
277
        """Create a weave repository."""
290
281
        if not _internal:
291
282
            # always initialized when the bzrdir is.
292
283
            return self.open(a_bzrdir, _found=True)
293
 
 
 
284
        
294
285
        # Create an empty weave
295
286
        sio = StringIO()
296
287
        weavefile.write_weave_v5(weave.Weave(), sio)
297
288
        empty_weave = sio.getvalue()
298
289
 
299
 
        trace.mutter('creating repository in %s.', a_bzrdir.transport.base)
300
 
 
 
290
        mutter('creating repository in %s.', a_bzrdir.transport.base)
 
291
        dirs = ['revision-store', 'weaves']
 
292
        files = [('inventory.weave', StringIO(empty_weave)),
 
293
                 ]
 
294
        
301
295
        # FIXME: RBC 20060125 don't peek under the covers
302
296
        # NB: no need to escape relative paths that are url safe.
303
297
        control_files = lockable_files.LockableFiles(a_bzrdir.transport,
304
 
            'branch-lock', lockable_files.TransportLock)
 
298
                                'branch-lock', lockable_files.TransportLock)
305
299
        control_files.create_lock()
306
300
        control_files.lock_write()
307
 
        transport = a_bzrdir.transport
 
301
        control_files._transport.mkdir_multi(dirs,
 
302
                mode=control_files._dir_mode)
308
303
        try:
309
 
            transport.mkdir_multi(['revision-store', 'weaves'],
310
 
                mode=a_bzrdir._get_dir_mode())
311
 
            transport.put_bytes_non_atomic('inventory.weave', empty_weave,
312
 
                mode=a_bzrdir._get_file_mode())
 
304
            for file, content in files:
 
305
                control_files.put(file, content)
313
306
        finally:
314
307
            control_files.unlock()
315
 
        repository = self.open(a_bzrdir, _found=True)
316
 
        self._run_post_repo_init_hooks(repository, a_bzrdir, shared)
317
 
        return repository
 
308
        return self.open(a_bzrdir, _found=True)
 
309
 
 
310
    def _get_control_store(self, repo_transport, control_files):
 
311
        """Return the control store for this repository."""
 
312
        return self._get_versioned_file_store('',
 
313
                                              repo_transport,
 
314
                                              control_files,
 
315
                                              prefixed=False)
 
316
 
 
317
    def _get_text_store(self, transport, control_files):
 
318
        """Get a store for file texts for this format."""
 
319
        raise NotImplementedError(self._get_text_store)
318
320
 
319
321
    def open(self, a_bzrdir, _found=False):
320
322
        """See RepositoryFormat.open()."""
323
325
            raise NotImplementedError
324
326
 
325
327
        repo_transport = a_bzrdir.get_repository_transport(None)
326
 
        result = AllInOneRepository(_format=self, a_bzrdir=a_bzrdir)
327
 
        result.revisions = self._get_revisions(repo_transport, result)
328
 
        result.signatures = self._get_signatures(repo_transport, result)
329
 
        result.inventories = self._get_inventories(repo_transport, result)
330
 
        result.texts = self._get_texts(repo_transport, result)
331
 
        result.chk_bytes = None
332
 
        return result
 
328
        control_files = a_bzrdir._control_files
 
329
        text_store = self._get_text_store(repo_transport, control_files)
 
330
        control_store = self._get_control_store(repo_transport, control_files)
 
331
        _revision_store = self._get_revision_store(repo_transport, control_files)
 
332
        return AllInOneRepository(_format=self,
 
333
                                  a_bzrdir=a_bzrdir,
 
334
                                  _revision_store=_revision_store,
 
335
                                  control_store=control_store,
 
336
                                  text_store=text_store)
333
337
 
334
 
    def is_deprecated(self):
335
 
        return True
 
338
    def check_conversion_target(self, target_format):
 
339
        pass
336
340
 
337
341
 
338
342
class RepositoryFormat4(PreSplitOutRepositoryFormat):
347
351
    has been removed.
348
352
    """
349
353
 
350
 
    supports_funky_characters = False
 
354
    _matchingbzrdir = bzrdir.BzrDirFormat4()
351
355
 
352
 
    _matchingbzrdir = weave_bzrdir.BzrDirFormat4()
 
356
    def __init__(self):
 
357
        super(RepositoryFormat4, self).__init__()
353
358
 
354
359
    def get_format_description(self):
355
360
        """See RepositoryFormat.get_format_description()."""
363
368
        """Format 4 is not supported.
364
369
 
365
370
        It is not supported because the model changed from 4 to 5 and the
366
 
        conversion logic is expensive - so doing it on the fly was not
 
371
        conversion logic is expensive - so doing it on the fly was not 
367
372
        feasible.
368
373
        """
369
374
        return False
370
375
 
371
 
    def _get_inventories(self, repo_transport, repo, name='inventory'):
372
 
        # No inventories store written so far.
373
 
        return None
374
 
 
375
 
    def _get_revisions(self, repo_transport, repo):
376
 
        from bzrlib.plugins.weave_fmt.xml4 import serializer_v4
377
 
        return RevisionTextStore(repo_transport.clone('revision-store'),
378
 
            serializer_v4, True, versionedfile.PrefixMapper(),
379
 
            repo.is_locked, repo.is_write_locked)
380
 
 
381
 
    def _get_signatures(self, repo_transport, repo):
382
 
        return SignatureTextStore(repo_transport.clone('revision-store'),
383
 
            False, versionedfile.PrefixMapper(),
384
 
            repo.is_locked, repo.is_write_locked)
385
 
 
386
 
    def _get_texts(self, repo_transport, repo):
387
 
        return None
 
376
    def _get_control_store(self, repo_transport, control_files):
 
377
        """Format 4 repositories have no formal control store at this point.
 
378
        
 
379
        This will cause any control-file-needing apis to fail - this is desired.
 
380
        """
 
381
        return None
 
382
    
 
383
    def _get_revision_store(self, repo_transport, control_files):
 
384
        """See RepositoryFormat._get_revision_store()."""
 
385
        from bzrlib.xml4 import serializer_v4
 
386
        return self._get_text_rev_store(repo_transport,
 
387
                                        control_files,
 
388
                                        'revision-store',
 
389
                                        serializer=serializer_v4)
 
390
 
 
391
    def _get_text_store(self, transport, control_files):
 
392
        """See RepositoryFormat._get_text_store()."""
388
393
 
389
394
 
390
395
class RepositoryFormat5(PreSplitOutRepositoryFormat):
397
402
    """
398
403
 
399
404
    _versionedfile_class = weave.WeaveFile
400
 
    _matchingbzrdir = weave_bzrdir.BzrDirFormat5()
401
 
    supports_funky_characters = False
 
405
    _matchingbzrdir = bzrdir.BzrDirFormat5()
402
406
 
403
 
    @property
404
 
    def _serializer(self):
405
 
        return xml5.serializer_v5
 
407
    def __init__(self):
 
408
        super(RepositoryFormat5, self).__init__()
406
409
 
407
410
    def get_format_description(self):
408
411
        """See RepositoryFormat.get_format_description()."""
409
412
        return "Weave repository format 5"
410
413
 
411
 
    def network_name(self):
412
 
        """The network name for this format is the control dirs disk label."""
413
 
        return self._matchingbzrdir.get_format_string()
414
 
 
415
 
    def _get_inventories(self, repo_transport, repo, name='inventory'):
416
 
        mapper = versionedfile.ConstantMapper(name)
417
 
        return versionedfile.ThunkedVersionedFiles(repo_transport,
418
 
            weave.WeaveFile, mapper, repo.is_locked)
419
 
 
420
 
    def _get_revisions(self, repo_transport, repo):
421
 
        return RevisionTextStore(repo_transport.clone('revision-store'),
422
 
            xml5.serializer_v5, False, versionedfile.PrefixMapper(),
423
 
            repo.is_locked, repo.is_write_locked)
424
 
 
425
 
    def _get_signatures(self, repo_transport, repo):
426
 
        return SignatureTextStore(repo_transport.clone('revision-store'),
427
 
            False, versionedfile.PrefixMapper(),
428
 
            repo.is_locked, repo.is_write_locked)
429
 
 
430
 
    def _get_texts(self, repo_transport, repo):
431
 
        mapper = versionedfile.PrefixMapper()
432
 
        base_transport = repo_transport.clone('weaves')
433
 
        return versionedfile.ThunkedVersionedFiles(base_transport,
434
 
            weave.WeaveFile, mapper, repo.is_locked)
 
414
    def _get_revision_store(self, repo_transport, control_files):
 
415
        """See RepositoryFormat._get_revision_store()."""
 
416
        """Return the revision store object for this a_bzrdir."""
 
417
        return self._get_text_rev_store(repo_transport,
 
418
                                        control_files,
 
419
                                        'revision-store',
 
420
                                        compressed=False)
 
421
 
 
422
    def _get_text_store(self, transport, control_files):
 
423
        """See RepositoryFormat._get_text_store()."""
 
424
        return self._get_versioned_file_store('weaves', transport, control_files, prefixed=False)
435
425
 
436
426
 
437
427
class RepositoryFormat6(PreSplitOutRepositoryFormat):
444
434
    """
445
435
 
446
436
    _versionedfile_class = weave.WeaveFile
447
 
    _matchingbzrdir = weave_bzrdir.BzrDirFormat6()
448
 
    supports_funky_characters = False
449
 
    @property
450
 
    def _serializer(self):
451
 
        return xml5.serializer_v5
 
437
    _matchingbzrdir = bzrdir.BzrDirFormat6()
 
438
 
 
439
    def __init__(self):
 
440
        super(RepositoryFormat6, self).__init__()
452
441
 
453
442
    def get_format_description(self):
454
443
        """See RepositoryFormat.get_format_description()."""
455
444
        return "Weave repository format 6"
456
445
 
457
 
    def network_name(self):
458
 
        """The network name for this format is the control dirs disk label."""
459
 
        return self._matchingbzrdir.get_format_string()
460
 
 
461
 
    def _get_inventories(self, repo_transport, repo, name='inventory'):
462
 
        mapper = versionedfile.ConstantMapper(name)
463
 
        return versionedfile.ThunkedVersionedFiles(repo_transport,
464
 
            weave.WeaveFile, mapper, repo.is_locked)
465
 
 
466
 
    def _get_revisions(self, repo_transport, repo):
467
 
        return RevisionTextStore(repo_transport.clone('revision-store'),
468
 
            xml5.serializer_v5, False, versionedfile.HashPrefixMapper(),
469
 
            repo.is_locked, repo.is_write_locked)
470
 
 
471
 
    def _get_signatures(self, repo_transport, repo):
472
 
        return SignatureTextStore(repo_transport.clone('revision-store'),
473
 
            False, versionedfile.HashPrefixMapper(),
474
 
            repo.is_locked, repo.is_write_locked)
475
 
 
476
 
    def _get_texts(self, repo_transport, repo):
477
 
        mapper = versionedfile.HashPrefixMapper()
478
 
        base_transport = repo_transport.clone('weaves')
479
 
        return versionedfile.ThunkedVersionedFiles(base_transport,
480
 
            weave.WeaveFile, mapper, repo.is_locked)
481
 
 
482
 
 
483
 
class RepositoryFormat7(MetaDirVersionedFileRepositoryFormat):
 
446
    def _get_revision_store(self, repo_transport, control_files):
 
447
        """See RepositoryFormat._get_revision_store()."""
 
448
        return self._get_text_rev_store(repo_transport,
 
449
                                        control_files,
 
450
                                        'revision-store',
 
451
                                        compressed=False,
 
452
                                        prefixed=True)
 
453
 
 
454
    def _get_text_store(self, transport, control_files):
 
455
        """See RepositoryFormat._get_text_store()."""
 
456
        return self._get_versioned_file_store('weaves', transport, control_files)
 
457
 
 
458
class RepositoryFormat7(MetaDirRepositoryFormat):
484
459
    """Bzr repository 7.
485
460
 
486
461
    This repository format has:
494
469
 
495
470
    _versionedfile_class = weave.WeaveFile
496
471
    supports_ghosts = False
497
 
    supports_chks = False
498
 
    supports_funky_characters = False
499
 
    revision_graph_can_have_wrong_parents = False
500
 
 
501
 
    _fetch_order = 'topological'
502
 
    _fetch_reconcile = True
503
 
    fast_deltas = False
504
 
    @property
505
 
    def _serializer(self):
506
 
        return xml5.serializer_v5
507
 
 
508
 
    @classmethod
509
 
    def get_format_string(cls):
 
472
 
 
473
    def _get_control_store(self, repo_transport, control_files):
 
474
        """Return the control store for this repository."""
 
475
        return self._get_versioned_file_store('',
 
476
                                              repo_transport,
 
477
                                              control_files,
 
478
                                              prefixed=False)
 
479
 
 
480
    def get_format_string(self):
510
481
        """See RepositoryFormat.get_format_string()."""
511
482
        return "Bazaar-NG Repository format 7"
512
483
 
514
485
        """See RepositoryFormat.get_format_description()."""
515
486
        return "Weave repository format 7"
516
487
 
517
 
    def _get_inventories(self, repo_transport, repo, name='inventory'):
518
 
        mapper = versionedfile.ConstantMapper(name)
519
 
        return versionedfile.ThunkedVersionedFiles(repo_transport,
520
 
            weave.WeaveFile, mapper, repo.is_locked)
521
 
 
522
 
    def _get_revisions(self, repo_transport, repo):
523
 
        return RevisionTextStore(repo_transport.clone('revision-store'),
524
 
            xml5.serializer_v5, True, versionedfile.HashPrefixMapper(),
525
 
            repo.is_locked, repo.is_write_locked)
526
 
 
527
 
    def _get_signatures(self, repo_transport, repo):
528
 
        return SignatureTextStore(repo_transport.clone('revision-store'),
529
 
            True, versionedfile.HashPrefixMapper(),
530
 
            repo.is_locked, repo.is_write_locked)
531
 
 
532
 
    def _get_texts(self, repo_transport, repo):
533
 
        mapper = versionedfile.HashPrefixMapper()
534
 
        base_transport = repo_transport.clone('weaves')
535
 
        return versionedfile.ThunkedVersionedFiles(base_transport,
536
 
            weave.WeaveFile, mapper, repo.is_locked)
 
488
    def check_conversion_target(self, target_format):
 
489
        pass
 
490
 
 
491
    def _get_revision_store(self, repo_transport, control_files):
 
492
        """See RepositoryFormat._get_revision_store()."""
 
493
        return self._get_text_rev_store(repo_transport,
 
494
                                        control_files,
 
495
                                        'revision-store',
 
496
                                        compressed=False,
 
497
                                        prefixed=True,
 
498
                                        )
 
499
 
 
500
    def _get_text_store(self, transport, control_files):
 
501
        """See RepositoryFormat._get_text_store()."""
 
502
        return self._get_versioned_file_store('weaves',
 
503
                                              transport,
 
504
                                              control_files)
537
505
 
538
506
    def initialize(self, a_bzrdir, shared=False):
539
507
        """Create a weave repository.
546
514
        weavefile.write_weave_v5(weave.Weave(), sio)
547
515
        empty_weave = sio.getvalue()
548
516
 
549
 
        trace.mutter('creating repository in %s.', a_bzrdir.transport.base)
 
517
        mutter('creating repository in %s.', a_bzrdir.transport.base)
550
518
        dirs = ['revision-store', 'weaves']
551
 
        files = [('inventory.weave', StringIO(empty_weave)),
 
519
        files = [('inventory.weave', StringIO(empty_weave)), 
552
520
                 ]
553
521
        utf8_files = [('format', self.get_format_string())]
554
 
 
 
522
 
555
523
        self._upload_blank_content(a_bzrdir, dirs, files, utf8_files, shared)
556
524
        return self.open(a_bzrdir=a_bzrdir, _found=True)
557
525
 
558
526
    def open(self, a_bzrdir, _found=False, _override_transport=None):
559
527
        """See RepositoryFormat.open().
560
 
 
 
528
        
561
529
        :param _override_transport: INTERNAL USE ONLY. Allows opening the
562
530
                                    repository at a slightly different url
563
531
                                    than normal. I.e. during 'upgrade'.
564
532
        """
565
533
        if not _found:
566
 
            format = RepositoryFormatMetaDir.find_format(a_bzrdir)
 
534
            format = RepositoryFormat.find_format(a_bzrdir)
567
535
        if _override_transport is not None:
568
536
            repo_transport = _override_transport
569
537
        else:
570
538
            repo_transport = a_bzrdir.get_repository_transport(None)
571
539
        control_files = lockable_files.LockableFiles(repo_transport,
572
540
                                'lock', lockdir.LockDir)
573
 
        result = WeaveMetaDirRepository(_format=self, a_bzrdir=a_bzrdir,
574
 
            control_files=control_files)
575
 
        result.revisions = self._get_revisions(repo_transport, result)
576
 
        result.signatures = self._get_signatures(repo_transport, result)
577
 
        result.inventories = self._get_inventories(repo_transport, result)
578
 
        result.texts = self._get_texts(repo_transport, result)
579
 
        result.chk_bytes = None
580
 
        result._transport = repo_transport
581
 
        return result
582
 
 
583
 
    def is_deprecated(self):
584
 
        return True
585
 
 
586
 
 
587
 
class TextVersionedFiles(VersionedFiles):
588
 
    """Just-a-bunch-of-files based VersionedFile stores."""
589
 
 
590
 
    def __init__(self, transport, compressed, mapper, is_locked, can_write):
591
 
        self._compressed = compressed
592
 
        self._transport = transport
593
 
        self._mapper = mapper
594
 
        if self._compressed:
595
 
            self._ext = '.gz'
596
 
        else:
597
 
            self._ext = ''
598
 
        self._is_locked = is_locked
599
 
        self._can_write = can_write
600
 
 
601
 
    def add_lines(self, key, parents, lines):
602
 
        """Add a revision to the store."""
603
 
        if not self._is_locked():
604
 
            raise errors.ObjectNotLocked(self)
605
 
        if not self._can_write():
606
 
            raise errors.ReadOnlyError(self)
607
 
        if '/' in key[-1]:
608
 
            raise ValueError('bad idea to put / in %r' % (key,))
609
 
        text = ''.join(lines)
610
 
        if self._compressed:
611
 
            text = tuned_gzip.bytes_to_gzip(text)
612
 
        path = self._map(key)
613
 
        self._transport.put_bytes_non_atomic(path, text, create_parent_dir=True)
614
 
 
615
 
    def insert_record_stream(self, stream):
616
 
        adapters = {}
617
 
        for record in stream:
618
 
            # Raise an error when a record is missing.
619
 
            if record.storage_kind == 'absent':
620
 
                raise errors.RevisionNotPresent([record.key[0]], self)
621
 
            # adapt to non-tuple interface
622
 
            if record.storage_kind == 'fulltext':
623
 
                self.add_lines(record.key, None,
624
 
                    osutils.split_lines(record.get_bytes_as('fulltext')))
625
 
            else:
626
 
                adapter_key = record.storage_kind, 'fulltext'
627
 
                try:
628
 
                    adapter = adapters[adapter_key]
629
 
                except KeyError:
630
 
                    adapter_factory = adapter_registry.get(adapter_key)
631
 
                    adapter = adapter_factory(self)
632
 
                    adapters[adapter_key] = adapter
633
 
                lines = osutils.split_lines(adapter.get_bytes(
634
 
                    record, record.get_bytes_as(record.storage_kind)))
635
 
                try:
636
 
                    self.add_lines(record.key, None, lines)
637
 
                except errors.RevisionAlreadyPresent:
638
 
                    pass
639
 
 
640
 
    def _load_text(self, key):
641
 
        if not self._is_locked():
642
 
            raise errors.ObjectNotLocked(self)
643
 
        path = self._map(key)
644
 
        try:
645
 
            text = self._transport.get_bytes(path)
646
 
            compressed = self._compressed
647
 
        except errors.NoSuchFile:
648
 
            if self._compressed:
649
 
                # try without the .gz
650
 
                path = path[:-3]
651
 
                try:
652
 
                    text = self._transport.get_bytes(path)
653
 
                    compressed = False
654
 
                except errors.NoSuchFile:
655
 
                    return None
656
 
            else:
657
 
                return None
658
 
        if compressed:
659
 
            text = gzip.GzipFile(mode='rb', fileobj=StringIO(text)).read()
660
 
        return text
661
 
 
662
 
    def _map(self, key):
663
 
        return self._mapper.map(key) + self._ext
664
 
 
665
 
 
666
 
class RevisionTextStore(TextVersionedFiles):
667
 
    """Legacy thunk for format 4 repositories."""
668
 
 
669
 
    def __init__(self, transport, serializer, compressed, mapper, is_locked,
670
 
        can_write):
671
 
        """Create a RevisionTextStore at transport with serializer."""
672
 
        TextVersionedFiles.__init__(self, transport, compressed, mapper,
673
 
            is_locked, can_write)
674
 
        self._serializer = serializer
675
 
 
676
 
    def _load_text_parents(self, key):
677
 
        text = self._load_text(key)
678
 
        if text is None:
679
 
            return None, None
680
 
        parents = self._serializer.read_revision_from_string(text).parent_ids
681
 
        return text, tuple((parent,) for parent in parents)
682
 
 
683
 
    def get_parent_map(self, keys):
684
 
        result = {}
685
 
        for key in keys:
686
 
            parents = self._load_text_parents(key)[1]
687
 
            if parents is None:
688
 
                continue
689
 
            result[key] = parents
690
 
        return result
691
 
 
692
 
    def get_known_graph_ancestry(self, keys):
693
 
        """Get a KnownGraph instance with the ancestry of keys."""
694
 
        keys = self.keys()
695
 
        parent_map = self.get_parent_map(keys)
696
 
        kg = _mod_graph.KnownGraph(parent_map)
697
 
        return kg
698
 
 
699
 
    def get_record_stream(self, keys, sort_order, include_delta_closure):
700
 
        for key in keys:
701
 
            text, parents = self._load_text_parents(key)
702
 
            if text is None:
703
 
                yield AbsentContentFactory(key)
704
 
            else:
705
 
                yield FulltextContentFactory(key, parents, None, text)
706
 
 
707
 
    def keys(self):
708
 
        if not self._is_locked():
709
 
            raise errors.ObjectNotLocked(self)
710
 
        relpaths = set()
711
 
        for quoted_relpath in self._transport.iter_files_recursive():
712
 
            relpath = urlutils.unquote(quoted_relpath)
713
 
            path, ext = os.path.splitext(relpath)
714
 
            if ext == '.gz':
715
 
                relpath = path
716
 
            if not relpath.endswith('.sig'):
717
 
                relpaths.add(relpath)
718
 
        paths = list(relpaths)
719
 
        return set([self._mapper.unmap(path) for path in paths])
720
 
 
721
 
 
722
 
class SignatureTextStore(TextVersionedFiles):
723
 
    """Legacy thunk for format 4-7 repositories."""
724
 
 
725
 
    def __init__(self, transport, compressed, mapper, is_locked, can_write):
726
 
        TextVersionedFiles.__init__(self, transport, compressed, mapper,
727
 
            is_locked, can_write)
728
 
        self._ext = '.sig' + self._ext
729
 
 
730
 
    def get_parent_map(self, keys):
731
 
        result = {}
732
 
        for key in keys:
733
 
            text = self._load_text(key)
734
 
            if text is None:
735
 
                continue
736
 
            result[key] = None
737
 
        return result
738
 
 
739
 
    def get_record_stream(self, keys, sort_order, include_delta_closure):
740
 
        for key in keys:
741
 
            text = self._load_text(key)
742
 
            if text is None:
743
 
                yield AbsentContentFactory(key)
744
 
            else:
745
 
                yield FulltextContentFactory(key, None, None, text)
746
 
 
747
 
    def keys(self):
748
 
        if not self._is_locked():
749
 
            raise errors.ObjectNotLocked(self)
750
 
        relpaths = set()
751
 
        for quoted_relpath in self._transport.iter_files_recursive():
752
 
            relpath = urlutils.unquote(quoted_relpath)
753
 
            path, ext = os.path.splitext(relpath)
754
 
            if ext == '.gz':
755
 
                relpath = path
756
 
            if not relpath.endswith('.sig'):
757
 
                continue
758
 
            relpaths.add(relpath[:-4])
759
 
        paths = list(relpaths)
760
 
        return set([self._mapper.unmap(path) for path in paths])
761
 
 
762
 
 
763
 
class InterWeaveRepo(InterSameDataRepository):
764
 
    """Optimised code paths between Weave based repositories.
765
 
    """
766
 
 
767
 
    @classmethod
768
 
    def _get_repo_format_to_test(self):
769
 
        return RepositoryFormat7()
770
 
 
771
 
    @staticmethod
772
 
    def is_compatible(source, target):
773
 
        """Be compatible with known Weave formats.
774
 
 
775
 
        We don't test for the stores being of specific types because that
776
 
        could lead to confusing results, and there is no need to be
777
 
        overly general.
778
 
        """
779
 
        try:
780
 
            return (isinstance(source._format, (RepositoryFormat5,
781
 
                                                RepositoryFormat6,
782
 
                                                RepositoryFormat7)) and
783
 
                    isinstance(target._format, (RepositoryFormat5,
784
 
                                                RepositoryFormat6,
785
 
                                                RepositoryFormat7)))
786
 
        except AttributeError:
787
 
            return False
788
 
 
789
 
    @needs_write_lock
790
 
    def copy_content(self, revision_id=None):
791
 
        """See InterRepository.copy_content()."""
792
 
        # weave specific optimised path:
793
 
        try:
794
 
            self.target.set_make_working_trees(self.source.make_working_trees())
795
 
        except (errors.RepositoryUpgradeRequired, NotImplemented):
796
 
            pass
797
 
        # FIXME do not peek!
798
 
        if self.source._transport.listable():
799
 
            pb = ui.ui_factory.nested_progress_bar()
800
 
            try:
801
 
                self.target.texts.insert_record_stream(
802
 
                    self.source.texts.get_record_stream(
803
 
                        self.source.texts.keys(), 'topological', False))
804
 
                pb.update('Copying inventory', 0, 1)
805
 
                self.target.inventories.insert_record_stream(
806
 
                    self.source.inventories.get_record_stream(
807
 
                        self.source.inventories.keys(), 'topological', False))
808
 
                self.target.signatures.insert_record_stream(
809
 
                    self.source.signatures.get_record_stream(
810
 
                        self.source.signatures.keys(),
811
 
                        'unordered', True))
812
 
                self.target.revisions.insert_record_stream(
813
 
                    self.source.revisions.get_record_stream(
814
 
                        self.source.revisions.keys(),
815
 
                        'topological', True))
816
 
            finally:
817
 
                pb.finished()
818
 
        else:
819
 
            self.target.fetch(self.source, revision_id=revision_id)
820
 
 
821
 
    @needs_read_lock
822
 
    def search_missing_revision_ids(self,
823
 
            revision_id=symbol_versioning.DEPRECATED_PARAMETER,
824
 
            find_ghosts=True, revision_ids=None, if_present_ids=None,
825
 
            limit=None):
826
 
        """See InterRepository.search_missing_revision_ids()."""
827
 
        # we want all revisions to satisfy revision_id in source.
828
 
        # but we don't want to stat every file here and there.
829
 
        # we want then, all revisions other needs to satisfy revision_id
830
 
        # checked, but not those that we have locally.
831
 
        # so the first thing is to get a subset of the revisions to
832
 
        # satisfy revision_id in source, and then eliminate those that
833
 
        # we do already have.
834
 
        # this is slow on high latency connection to self, but as this
835
 
        # disk format scales terribly for push anyway due to rewriting
836
 
        # inventory.weave, this is considered acceptable.
837
 
        # - RBC 20060209
838
 
        if symbol_versioning.deprecated_passed(revision_id):
839
 
            symbol_versioning.warn(
840
 
                'search_missing_revision_ids(revision_id=...) was '
841
 
                'deprecated in 2.4.  Use revision_ids=[...] instead.',
842
 
                DeprecationWarning, stacklevel=2)
843
 
            if revision_ids is not None:
844
 
                raise AssertionError(
845
 
                    'revision_ids is mutually exclusive with revision_id')
846
 
            if revision_id is not None:
847
 
                revision_ids = [revision_id]
848
 
        del revision_id
849
 
        source_ids_set = self._present_source_revisions_for(
850
 
            revision_ids, if_present_ids)
851
 
        # source_ids is the worst possible case we may need to pull.
852
 
        # now we want to filter source_ids against what we actually
853
 
        # have in target, but don't try to check for existence where we know
854
 
        # we do not have a revision as that would be pointless.
855
 
        target_ids = set(self.target._all_possible_ids())
856
 
        possibly_present_revisions = target_ids.intersection(source_ids_set)
857
 
        actually_present_revisions = set(
858
 
            self.target._eliminate_revisions_not_present(possibly_present_revisions))
859
 
        required_revisions = source_ids_set.difference(actually_present_revisions)
860
 
        if revision_ids is not None:
861
 
            # we used get_ancestry to determine source_ids then we are assured all
862
 
            # revisions referenced are present as they are installed in topological order.
863
 
            # and the tip revision was validated by get_ancestry.
864
 
            result_set = required_revisions
865
 
        else:
866
 
            # if we just grabbed the possibly available ids, then
867
 
            # we only have an estimate of whats available and need to validate
868
 
            # that against the revision records.
869
 
            result_set = set(
870
 
                self.source._eliminate_revisions_not_present(required_revisions))
871
 
        if limit is not None:
872
 
            topo_ordered = self.get_graph().iter_topo_order(result_set)
873
 
            result_set = set(itertools.islice(topo_ordered, limit))
874
 
        return self.source.revision_ids_to_search_result(result_set)
875
 
 
876
 
 
877
 
InterRepository.register_optimiser(InterWeaveRepo)
878
 
 
879
 
 
880
 
def get_extra_interrepo_test_combinations():
881
 
    from bzrlib.repofmt import knitrepo
882
 
    return [(InterRepository, RepositoryFormat5(),
883
 
        knitrepo.RepositoryFormatKnit3())]
 
541
        text_store = self._get_text_store(repo_transport, control_files)
 
542
        control_store = self._get_control_store(repo_transport, control_files)
 
543
        _revision_store = self._get_revision_store(repo_transport, control_files)
 
544
        return WeaveMetaDirRepository(_format=self,
 
545
            a_bzrdir=a_bzrdir,
 
546
            control_files=control_files,
 
547
            _revision_store=_revision_store,
 
548
            control_store=control_store,
 
549
            text_store=text_store)
 
550
 
 
551
 
 
552
class WeaveCommitBuilder(CommitBuilder):
 
553
    """A builder for weave based repos that don't support ghosts."""
 
554
 
 
555
    def _add_text_to_weave(self, file_id, new_lines, parents, nostore_sha):
 
556
        versionedfile = self.repository.weave_store.get_weave_or_empty(
 
557
            file_id, self.repository.get_transaction())
 
558
        result = versionedfile.add_lines(
 
559
            self._new_revision_id, parents, new_lines,
 
560
            nostore_sha=nostore_sha)[0:2]
 
561
        return result
 
562
 
 
563
 
 
564
_legacy_formats = [RepositoryFormat4(),
 
565
                   RepositoryFormat5(),
 
566
                   RepositoryFormat6()]