~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/bzrdir.py

  • Committer: Canonical.com Patch Queue Manager
  • Date: 2008-04-07 07:52:50 UTC
  • mfrom: (3340.1.1 208418-1.4)
  • Revision ID: pqm@pqm.ubuntu.com-20080407075250-phs53xnslo8boaeo
Return the correct knit serialisation method in _StreamAccess.
        (Andrew Bennetts, Martin Pool, Robert Collins)

Show diffs side-by-side

added added

removed removed

Lines of Context:
18
18
 
19
19
At format 7 this was split out into Branch, Repository and Checkout control
20
20
directories.
 
21
 
 
22
Note: This module has a lot of ``open`` functions/methods that return
 
23
references to in-memory objects. As a rule, there are no matching ``close``
 
24
methods. To free any associated resources, simply stop referencing the
 
25
objects returned.
21
26
"""
22
27
 
23
 
# TODO: remove unittest dependency; put that stuff inside the test suite
24
 
 
25
 
# TODO: Can we move specific formats into separate modules to make this file
26
 
# smaller?
 
28
# TODO: Move old formats into a plugin to make this file smaller.
27
29
 
28
30
from cStringIO import StringIO
29
31
import os
30
 
import textwrap
 
32
import sys
31
33
 
32
34
from bzrlib.lazy_import import lazy_import
33
35
lazy_import(globals(), """
34
 
from copy import deepcopy
35
36
from stat import S_ISDIR
36
 
import unittest
 
37
import textwrap
 
38
from warnings import warn
37
39
 
38
40
import bzrlib
39
41
from bzrlib import (
40
42
    errors,
 
43
    graph,
41
44
    lockable_files,
42
45
    lockdir,
43
46
    registry,
46
49
    symbol_versioning,
47
50
    ui,
48
51
    urlutils,
 
52
    win32utils,
 
53
    workingtree,
 
54
    workingtree_4,
49
55
    xml4,
50
56
    xml5,
51
 
    workingtree,
52
 
    workingtree_4,
53
57
    )
54
58
from bzrlib.osutils import (
55
 
    safe_unicode,
56
59
    sha_strings,
57
60
    sha_string,
58
61
    )
74
77
    note,
75
78
    )
76
79
from bzrlib.transport.local import LocalTransport
 
80
from bzrlib.symbol_versioning import (
 
81
    deprecated_function,
 
82
    deprecated_method,
 
83
    zero_ninetyone,
 
84
    )
77
85
 
78
86
 
79
87
class BzrDir(object):
85
93
    transport
86
94
        the transport which this bzr dir is rooted at (i.e. file:///.../.bzr/)
87
95
    root_transport
88
 
        a transport connected to the directory this bzr was opened from.
 
96
        a transport connected to the directory this bzr was opened from
 
97
        (i.e. the parent directory holding the .bzr directory).
89
98
    """
90
99
 
91
100
    def break_lock(self):
148
157
    def clone(self, url, revision_id=None, force_new_repo=False):
149
158
        """Clone this bzrdir and its contents to url verbatim.
150
159
 
151
 
        If urls last component does not exist, it will be created.
152
 
 
153
 
        if revision_id is not None, then the clone operation may tune
154
 
            itself to download less data.
155
 
        :param force_new_repo: Do not use a shared repository for the target 
156
 
                               even if one is available.
157
 
        """
158
 
        self._make_tail(url)
159
 
        result = self._format.initialize(url)
 
160
        If url's last component does not exist, it will be created.
 
161
 
 
162
        if revision_id is not None, then the clone operation may tune
 
163
            itself to download less data.
 
164
        :param force_new_repo: Do not use a shared repository for the target 
 
165
                               even if one is available.
 
166
        """
 
167
        return self.clone_on_transport(get_transport(url),
 
168
                                       revision_id=revision_id,
 
169
                                       force_new_repo=force_new_repo)
 
170
 
 
171
    def clone_on_transport(self, transport, revision_id=None,
 
172
                           force_new_repo=False):
 
173
        """Clone this bzrdir and its contents to transport verbatim.
 
174
 
 
175
        If the target directory does not exist, it will be created.
 
176
 
 
177
        if revision_id is not None, then the clone operation may tune
 
178
            itself to download less data.
 
179
        :param force_new_repo: Do not use a shared repository for the target 
 
180
                               even if one is available.
 
181
        """
 
182
        transport.ensure_base()
 
183
        result = self._format.initialize_on_transport(transport)
160
184
        try:
161
185
            local_repo = self.find_repository()
162
186
        except errors.NoRepositoryPresent:
187
211
        except errors.NotBranchError:
188
212
            pass
189
213
        try:
190
 
            self.open_workingtree().clone(result)
191
 
        except (errors.NoWorkingTree, errors.NotLocalUrl):
192
 
            pass
 
214
            result_repo = result.find_repository()
 
215
        except errors.NoRepositoryPresent:
 
216
            result_repo = None
 
217
        if result_repo is None or result_repo.make_working_trees():
 
218
            try:
 
219
                self.open_workingtree().clone(result)
 
220
            except (errors.NoWorkingTree, errors.NotLocalUrl):
 
221
                pass
193
222
        return result
194
223
 
195
224
    # TODO: This should be given a Transport, and should chdir up; otherwise
196
225
    # this will open a new connection.
197
226
    def _make_tail(self, url):
198
 
        head, tail = urlutils.split(url)
199
 
        if tail and tail != '.':
200
 
            t = get_transport(head)
201
 
            try:
202
 
                t.mkdir(tail)
203
 
            except errors.FileExists:
204
 
                pass
 
227
        t = get_transport(url)
 
228
        t.ensure_base()
205
229
 
206
 
    # TODO: Should take a Transport
207
230
    @classmethod
208
 
    def create(cls, base, format=None):
 
231
    def create(cls, base, format=None, possible_transports=None):
209
232
        """Create a new BzrDir at the url 'base'.
210
233
        
211
 
        This will call the current default formats initialize with base
212
 
        as the only parameter.
213
 
 
214
234
        :param format: If supplied, the format of branch to create.  If not
215
235
            supplied, the default is used.
 
236
        :param possible_transports: If supplied, a list of transports that 
 
237
            can be reused to share a remote connection.
216
238
        """
217
239
        if cls is not BzrDir:
218
240
            raise AssertionError("BzrDir.create always creates the default"
219
241
                " format, not one of %r" % cls)
220
 
        head, tail = urlutils.split(base)
221
 
        if tail and tail != '.':
222
 
            t = get_transport(head)
223
 
            try:
224
 
                t.mkdir(tail)
225
 
            except errors.FileExists:
226
 
                pass
 
242
        t = get_transport(base, possible_transports)
 
243
        t.ensure_base()
227
244
        if format is None:
228
245
            format = BzrDirFormat.get_default_format()
229
 
        return format.initialize(safe_unicode(base))
 
246
        return format.initialize_on_transport(t)
 
247
 
 
248
    @staticmethod
 
249
    def find_bzrdirs(transport, evaluate=None, list_current=None):
 
250
        """Find bzrdirs recursively from current location.
 
251
 
 
252
        This is intended primarily as a building block for more sophisticated
 
253
        functionality, like finding trees under a directory, or finding
 
254
        branches that use a given repository.
 
255
        :param evaluate: An optional callable that yields recurse, value,
 
256
            where recurse controls whether this bzrdir is recursed into
 
257
            and value is the value to yield.  By default, all bzrdirs
 
258
            are recursed into, and the return value is the bzrdir.
 
259
        :param list_current: if supplied, use this function to list the current
 
260
            directory, instead of Transport.list_dir
 
261
        :return: a generator of found bzrdirs, or whatever evaluate returns.
 
262
        """
 
263
        if list_current is None:
 
264
            def list_current(transport):
 
265
                return transport.list_dir('')
 
266
        if evaluate is None:
 
267
            def evaluate(bzrdir):
 
268
                return True, bzrdir
 
269
 
 
270
        pending = [transport]
 
271
        while len(pending) > 0:
 
272
            current_transport = pending.pop()
 
273
            recurse = True
 
274
            try:
 
275
                bzrdir = BzrDir.open_from_transport(current_transport)
 
276
            except errors.NotBranchError:
 
277
                pass
 
278
            else:
 
279
                recurse, value = evaluate(bzrdir)
 
280
                yield value
 
281
            try:
 
282
                subdirs = list_current(current_transport)
 
283
            except errors.NoSuchFile:
 
284
                continue
 
285
            if recurse:
 
286
                for subdir in sorted(subdirs, reverse=True):
 
287
                    pending.append(current_transport.clone(subdir))
 
288
 
 
289
    @staticmethod
 
290
    def find_branches(transport):
 
291
        """Find all branches under a transport.
 
292
 
 
293
        This will find all branches below the transport, including branches
 
294
        inside other branches.  Where possible, it will use
 
295
        Repository.find_branches.
 
296
 
 
297
        To list all the branches that use a particular Repository, see
 
298
        Repository.find_branches
 
299
        """
 
300
        def evaluate(bzrdir):
 
301
            try:
 
302
                repository = bzrdir.open_repository()
 
303
            except errors.NoRepositoryPresent:
 
304
                pass
 
305
            else:
 
306
                return False, (None, repository)
 
307
            try:
 
308
                branch = bzrdir.open_branch()
 
309
            except errors.NotBranchError:
 
310
                return True, (None, None)
 
311
            else:
 
312
                return True, (branch, None)
 
313
        branches = []
 
314
        for branch, repo in BzrDir.find_bzrdirs(transport, evaluate=evaluate):
 
315
            if repo is not None:
 
316
                branches.extend(repo.find_branches())
 
317
            if branch is not None:
 
318
                branches.append(branch)
 
319
        return branches
 
320
 
 
321
 
 
322
    def destroy_repository(self):
 
323
        """Destroy the repository in this BzrDir"""
 
324
        raise NotImplementedError(self.destroy_repository)
230
325
 
231
326
    def create_branch(self):
232
327
        """Create a branch in this BzrDir.
233
328
 
234
 
        The bzrdirs format will control what branch format is created.
 
329
        The bzrdir's format will control what branch format is created.
235
330
        For more control see BranchFormatXX.create(a_bzrdir).
236
331
        """
237
332
        raise NotImplementedError(self.create_branch)
238
333
 
 
334
    def destroy_branch(self):
 
335
        """Destroy the branch in this BzrDir"""
 
336
        raise NotImplementedError(self.destroy_branch)
 
337
 
239
338
    @staticmethod
240
339
    def create_branch_and_repo(base, force_new_repo=False, format=None):
241
340
        """Create a new BzrDir, Branch and Repository at the url 'base'.
242
341
 
243
 
        This will use the current default BzrDirFormat, and use whatever 
 
342
        This will use the current default BzrDirFormat unless one is
 
343
        specified, and use whatever 
244
344
        repository format that that uses via bzrdir.create_branch and
245
345
        create_repository. If a shared repository is available that is used
246
346
        preferentially.
249
349
 
250
350
        :param base: The URL to create the branch at.
251
351
        :param force_new_repo: If True a new repository is always created.
 
352
        :param format: If supplied, the format of branch to create.  If not
 
353
            supplied, the default is used.
252
354
        """
253
355
        bzrdir = BzrDir.create(base, format)
254
356
        bzrdir._find_or_create_repository(force_new_repo)
265
367
        
266
368
    @staticmethod
267
369
    def create_branch_convenience(base, force_new_repo=False,
268
 
                                  force_new_tree=None, format=None):
 
370
                                  force_new_tree=None, format=None,
 
371
                                  possible_transports=None):
269
372
        """Create a new BzrDir, Branch and Repository at the url 'base'.
270
373
 
271
374
        This is a convenience function - it will use an existing repository
272
375
        if possible, can be told explicitly whether to create a working tree or
273
376
        not.
274
377
 
275
 
        This will use the current default BzrDirFormat, and use whatever 
 
378
        This will use the current default BzrDirFormat unless one is
 
379
        specified, and use whatever 
276
380
        repository format that that uses via bzrdir.create_branch and
277
381
        create_repository. If a shared repository is available that is used
278
382
        preferentially. Whatever repository is used, its tree creation policy
287
391
        :param force_new_repo: If True a new repository is always created.
288
392
        :param force_new_tree: If True or False force creation of a tree or 
289
393
                               prevent such creation respectively.
290
 
        :param format: Override for the for the bzrdir format to create
 
394
        :param format: Override for the bzrdir format to create.
 
395
        :param possible_transports: An optional reusable transports list.
291
396
        """
292
397
        if force_new_tree:
293
398
            # check for non local urls
294
 
            t = get_transport(safe_unicode(base))
 
399
            t = get_transport(base, possible_transports)
295
400
            if not isinstance(t, LocalTransport):
296
401
                raise errors.NotLocalUrl(base)
297
 
        bzrdir = BzrDir.create(base, format)
 
402
        bzrdir = BzrDir.create(base, format, possible_transports)
298
403
        repo = bzrdir._find_or_create_repository(force_new_repo)
299
404
        result = bzrdir.create_branch()
300
 
        if force_new_tree or (repo.make_working_trees() and 
 
405
        if force_new_tree or (repo.make_working_trees() and
301
406
                              force_new_tree is None):
302
407
            try:
303
408
                bzrdir.create_workingtree()
304
409
            except errors.NotLocalUrl:
305
410
                pass
306
411
        return result
307
 
        
 
412
 
308
413
    @staticmethod
 
414
    @deprecated_function(zero_ninetyone)
309
415
    def create_repository(base, shared=False, format=None):
310
416
        """Create a new BzrDir and Repository at the url 'base'.
311
417
 
320
426
        This must be overridden as an instance method in child classes, where
321
427
        it should take no parameters and construct whatever repository format
322
428
        that child class desires.
 
429
 
 
430
        This method is deprecated, please call create_repository on a bzrdir
 
431
        instance instead.
323
432
        """
324
433
        bzrdir = BzrDir.create(base, format)
325
434
        return bzrdir.create_repository(shared)
330
439
 
331
440
        'base' must be a local path or a file:// url.
332
441
 
333
 
        This will use the current default BzrDirFormat, and use whatever 
 
442
        This will use the current default BzrDirFormat unless one is
 
443
        specified, and use whatever 
334
444
        repository format that that uses for bzrdirformat.create_workingtree,
335
445
        create_branch and create_repository.
336
446
 
 
447
        :param format: Override for the bzrdir format to create.
337
448
        :return: The WorkingTree object.
338
449
        """
339
 
        t = get_transport(safe_unicode(base))
 
450
        t = get_transport(base)
340
451
        if not isinstance(t, LocalTransport):
341
452
            raise errors.NotLocalUrl(base)
342
 
        bzrdir = BzrDir.create_branch_and_repo(safe_unicode(base),
 
453
        bzrdir = BzrDir.create_branch_and_repo(base,
343
454
                                               force_new_repo=True,
344
455
                                               format=format).bzrdir
345
456
        return bzrdir.create_workingtree()
346
457
 
347
 
    def create_workingtree(self, revision_id=None):
 
458
    def create_workingtree(self, revision_id=None, from_branch=None,
 
459
        accelerator_tree=None, hardlink=False):
348
460
        """Create a working tree at this BzrDir.
349
461
        
350
 
        revision_id: create it as of this revision id.
 
462
        :param revision_id: create it as of this revision id.
 
463
        :param from_branch: override bzrdir branch (for lightweight checkouts)
 
464
        :param accelerator_tree: A tree which can be used for retrieving file
 
465
            contents more quickly than the revision tree, i.e. a workingtree.
 
466
            The revision tree will be used for cases where accelerator_tree's
 
467
            content is different.
351
468
        """
352
469
        raise NotImplementedError(self.create_workingtree)
353
470
 
354
 
    def retire_bzrdir(self):
 
471
    def retire_bzrdir(self, limit=10000):
355
472
        """Permanently disable the bzrdir.
356
473
 
357
474
        This is done by renaming it to give the user some ability to recover
359
476
 
360
477
        This will have horrible consequences if anyone has anything locked or
361
478
        in use.
 
479
        :param limit: number of times to retry
362
480
        """
363
 
        for i in xrange(10000):
 
481
        i  = 0
 
482
        while True:
364
483
            try:
365
484
                to_path = '.bzr.retired.%d' % i
366
485
                self.root_transport.rename('.bzr', to_path)
367
486
                note("renamed %s to %s"
368
487
                    % (self.root_transport.abspath('.bzr'), to_path))
369
 
                break
 
488
                return
370
489
            except (errors.TransportError, IOError, errors.PathError):
371
 
                pass
 
490
                i += 1
 
491
                if i > limit:
 
492
                    raise
 
493
                else:
 
494
                    pass
372
495
 
373
496
    def destroy_workingtree(self):
374
497
        """Destroy the working tree at this BzrDir.
386
509
        raise NotImplementedError(self.destroy_workingtree_metadata)
387
510
 
388
511
    def find_repository(self):
389
 
        """Find the repository that should be used for a_bzrdir.
 
512
        """Find the repository that should be used.
390
513
 
391
514
        This does not require a branch as we use it to find the repo for
392
515
        new branches as well as to hook existing branches up to their
439
562
        a format string, and vice versa.
440
563
 
441
564
        If branch_format is None, the transport is returned with no 
442
 
        checking. if it is not None, then the returned transport is
 
565
        checking. If it is not None, then the returned transport is
443
566
        guaranteed to point to an existing directory ready for use.
444
567
        """
445
568
        raise NotImplementedError(self.get_branch_transport)
452
575
        a format string, and vice versa.
453
576
 
454
577
        If repository_format is None, the transport is returned with no 
455
 
        checking. if it is not None, then the returned transport is
 
578
        checking. If it is not None, then the returned transport is
456
579
        guaranteed to point to an existing directory ready for use.
457
580
        """
458
581
        raise NotImplementedError(self.get_repository_transport)
465
588
        format string, and vice versa.
466
589
 
467
590
        If workingtree_format is None, the transport is returned with no 
468
 
        checking. if it is not None, then the returned transport is
 
591
        checking. If it is not None, then the returned transport is
469
592
        guaranteed to point to an existing directory ready for use.
470
593
        """
471
594
        raise NotImplementedError(self.get_workingtree_transport)
497
620
        # this might be better on the BzrDirFormat class because it refers to 
498
621
        # all the possible bzrdir disk formats. 
499
622
        # This method is tested via the workingtree is_control_filename tests- 
500
 
        # it was extracted from WorkingTree.is_control_filename. If the methods
501
 
        # contract is extended beyond the current trivial  implementation please
 
623
        # it was extracted from WorkingTree.is_control_filename. If the method's
 
624
        # contract is extended beyond the current trivial implementation, please
502
625
        # add new tests for it to the appropriate place.
503
626
        return filename == '.bzr' or filename.startswith('.bzr/')
504
627
 
519
642
        return BzrDir.open(base, _unsupported=True)
520
643
        
521
644
    @staticmethod
522
 
    def open(base, _unsupported=False):
523
 
        """Open an existing bzrdir, rooted at 'base' (url)
 
645
    def open(base, _unsupported=False, possible_transports=None):
 
646
        """Open an existing bzrdir, rooted at 'base' (url).
524
647
        
525
 
        _unsupported is a private parameter to the BzrDir class.
 
648
        :param _unsupported: a private parameter to the BzrDir class.
526
649
        """
527
 
        t = get_transport(base)
 
650
        t = get_transport(base, possible_transports=possible_transports)
528
651
        return BzrDir.open_from_transport(t, _unsupported=_unsupported)
529
652
 
530
653
    @staticmethod
551
674
            note('%s is%s redirected to %s',
552
675
                 transport.base, e.permanently, target)
553
676
            # Let's try with a new transport
554
 
            qualified_target = e.get_target_url()[:-len(relpath)]
555
677
            # FIXME: If 'transport' has a qualifier, this should
556
678
            # be applied again to the new transport *iff* the
557
 
            # schemes used are the same. It's a bit tricky to
558
 
            # verify, so I'll punt for now
 
679
            # schemes used are the same. Uncomment this code
 
680
            # once the function (and tests) exist.
559
681
            # -- vila20070212
 
682
            #target = urlutils.copy_url_qualifiers(original, target)
560
683
            return get_transport(target)
561
684
 
562
685
        try:
580
703
        raise NotImplementedError(self.open_branch)
581
704
 
582
705
    @staticmethod
583
 
    def open_containing(url):
 
706
    def open_containing(url, possible_transports=None):
584
707
        """Open an existing branch which contains url.
585
708
        
586
709
        :param url: url to search from.
587
710
        See open_containing_from_transport for more detail.
588
711
        """
589
 
        return BzrDir.open_containing_from_transport(get_transport(url))
 
712
        transport = get_transport(url, possible_transports)
 
713
        return BzrDir.open_containing_from_transport(transport)
590
714
    
591
715
    @staticmethod
592
716
    def open_containing_from_transport(a_transport):
593
 
        """Open an existing branch which contains a_transport.base
 
717
        """Open an existing branch which contains a_transport.base.
594
718
 
595
719
        This probes for a branch at a_transport, and searches upwards from there.
596
720
 
621
745
                raise errors.NotBranchError(path=url)
622
746
            a_transport = new_t
623
747
 
 
748
    def _get_tree_branch(self):
 
749
        """Return the branch and tree, if any, for this bzrdir.
 
750
 
 
751
        Return None for tree if not present or inaccessible.
 
752
        Raise NotBranchError if no branch is present.
 
753
        :return: (tree, branch)
 
754
        """
 
755
        try:
 
756
            tree = self.open_workingtree()
 
757
        except (errors.NoWorkingTree, errors.NotLocalUrl):
 
758
            tree = None
 
759
            branch = self.open_branch()
 
760
        else:
 
761
            branch = tree.branch
 
762
        return tree, branch
 
763
 
 
764
    @classmethod
 
765
    def open_tree_or_branch(klass, location):
 
766
        """Return the branch and working tree at a location.
 
767
 
 
768
        If there is no tree at the location, tree will be None.
 
769
        If there is no branch at the location, an exception will be
 
770
        raised
 
771
        :return: (tree, branch)
 
772
        """
 
773
        bzrdir = klass.open(location)
 
774
        return bzrdir._get_tree_branch()
 
775
 
624
776
    @classmethod
625
777
    def open_containing_tree_or_branch(klass, location):
626
778
        """Return the branch and working tree contained by a location.
632
784
        relpath is the portion of the path that is contained by the branch.
633
785
        """
634
786
        bzrdir, relpath = klass.open_containing(location)
635
 
        try:
636
 
            tree = bzrdir.open_workingtree()
637
 
        except (errors.NoWorkingTree, errors.NotLocalUrl):
638
 
            tree = None
639
 
            branch = bzrdir.open_branch()
640
 
        else:
641
 
            branch = tree.branch
 
787
        tree, branch = bzrdir._get_tree_branch()
642
788
        return tree, branch, relpath
643
789
 
644
790
    def open_repository(self, _unsupported=False):
645
791
        """Open the repository object at this BzrDir if one is present.
646
792
 
647
 
        This will not follow the Branch object pointer - its strictly a direct
 
793
        This will not follow the Branch object pointer - it's strictly a direct
648
794
        open facility. Most client code should use open_branch().repository to
649
795
        get at a repository.
650
796
 
651
 
        _unsupported is a private parameter, not part of the api.
 
797
        :param _unsupported: a private parameter, not part of the api.
652
798
        TODO: static convenience version of this?
653
799
        """
654
800
        raise NotImplementedError(self.open_repository)
655
801
 
656
802
    def open_workingtree(self, _unsupported=False,
657
 
            recommend_upgrade=True):
 
803
                         recommend_upgrade=True, from_branch=None):
658
804
        """Open the workingtree object at this BzrDir if one is present.
659
805
 
660
806
        :param recommend_upgrade: Optional keyword parameter, when True (the
661
807
            default), emit through the ui module a recommendation that the user
662
808
            upgrade the working tree when the workingtree being opened is old
663
809
            (but still fully supported).
 
810
        :param from_branch: override bzrdir branch (for lightweight checkouts)
664
811
        """
665
812
        raise NotImplementedError(self.open_workingtree)
666
813
 
694
841
            return False
695
842
 
696
843
    def _cloning_metadir(self):
697
 
        """Produce a metadir suitable for cloning with"""
 
844
        """Produce a metadir suitable for cloning with."""
698
845
        result_format = self._format.__class__()
699
846
        try:
700
847
            try:
727
874
        """Produce a metadir suitable for cloning or sprouting with.
728
875
 
729
876
        These operations may produce workingtrees (yes, even though they're
730
 
        "cloning" something that doesn't have a tree, so a viable workingtree
 
877
        "cloning" something that doesn't have a tree), so a viable workingtree
731
878
        format must be selected.
732
879
        """
733
880
        format, repository = self._cloning_metadir()
742
889
        return self.cloning_metadir()
743
890
 
744
891
    def sprout(self, url, revision_id=None, force_new_repo=False,
745
 
               recurse='down'):
 
892
               recurse='down', possible_transports=None,
 
893
               accelerator_tree=None, hardlink=False):
746
894
        """Create a copy of this bzrdir prepared for use as a new line of
747
895
        development.
748
896
 
749
 
        If urls last component does not exist, it will be created.
 
897
        If url's last component does not exist, it will be created.
750
898
 
751
899
        Attributes related to the identity of the source branch like
752
900
        branch nickname will be cleaned, a working tree is created
755
903
 
756
904
        if revision_id is not None, then the clone operation may tune
757
905
            itself to download less data.
 
906
        :param accelerator_tree: A tree which can be used for retrieving file
 
907
            contents more quickly than the revision tree, i.e. a workingtree.
 
908
            The revision tree will be used for cases where accelerator_tree's
 
909
            content is different.
 
910
        :param hardlink: If true, hard-link files from accelerator_tree,
 
911
            where possible.
758
912
        """
759
 
        self._make_tail(url)
 
913
        target_transport = get_transport(url, possible_transports)
 
914
        target_transport.ensure_base()
760
915
        cloning_format = self.cloning_metadir()
761
 
        result = cloning_format.initialize(url)
 
916
        result = cloning_format.initialize_on_transport(target_transport)
762
917
        try:
763
918
            source_branch = self.open_branch()
764
919
            source_repository = source_branch.repository
782
937
            result.create_repository()
783
938
        elif source_repository is not None and result_repo is None:
784
939
            # have source, and want to make a new target repo
785
 
            result_repo = source_repository.sprout(result, revision_id=revision_id)
 
940
            result_repo = source_repository.sprout(result,
 
941
                                                   revision_id=revision_id)
786
942
        else:
787
943
            # fetch needed content into target.
788
944
            if source_repository is not None:
789
945
                # would rather do 
790
 
                # source_repository.copy_content_into(result_repo, revision_id=revision_id)
 
946
                # source_repository.copy_content_into(result_repo,
 
947
                #                                     revision_id=revision_id)
791
948
                # so we can override the copy method
792
949
                result_repo.fetch(source_repository, revision_id=revision_id)
793
950
        if source_branch is not None:
794
951
            source_branch.sprout(result, revision_id=revision_id)
795
952
        else:
796
953
            result.create_branch()
797
 
        # TODO: jam 20060426 we probably need a test in here in the
798
 
        #       case that the newly sprouted branch is a remote one
799
 
        if result_repo is None or result_repo.make_working_trees():
800
 
            wt = result.create_workingtree()
 
954
        if isinstance(target_transport, LocalTransport) and (
 
955
            result_repo is None or result_repo.make_working_trees()):
 
956
            wt = result.create_workingtree(accelerator_tree=accelerator_tree,
 
957
                hardlink=hardlink)
801
958
            wt.lock_write()
802
959
            try:
803
960
                if wt.path2id('') is None:
877
1034
        """See BzrDir.create_branch."""
878
1035
        return self.open_branch()
879
1036
 
 
1037
    def destroy_branch(self):
 
1038
        """See BzrDir.destroy_branch."""
 
1039
        raise errors.UnsupportedOperation(self.destroy_branch, self)
 
1040
 
880
1041
    def create_repository(self, shared=False):
881
1042
        """See BzrDir.create_repository."""
882
1043
        if shared:
883
1044
            raise errors.IncompatibleFormat('shared repository', self._format)
884
1045
        return self.open_repository()
885
1046
 
886
 
    def create_workingtree(self, revision_id=None):
 
1047
    def destroy_repository(self):
 
1048
        """See BzrDir.destroy_repository."""
 
1049
        raise errors.UnsupportedOperation(self.destroy_repository, self)
 
1050
 
 
1051
    def create_workingtree(self, revision_id=None, from_branch=None,
 
1052
                           accelerator_tree=None, hardlink=False):
887
1053
        """See BzrDir.create_workingtree."""
888
1054
        # this looks buggy but is not -really-
889
1055
        # because this format creates the workingtree when the bzrdir is
956
1122
        self._check_supported(format, unsupported)
957
1123
        return format.open(self, _found=True)
958
1124
 
959
 
    def sprout(self, url, revision_id=None, force_new_repo=False):
 
1125
    def sprout(self, url, revision_id=None, force_new_repo=False,
 
1126
               possible_transports=None, accelerator_tree=None,
 
1127
               hardlink=False):
960
1128
        """See BzrDir.sprout()."""
961
1129
        from bzrlib.workingtree import WorkingTreeFormat2
962
1130
        self._make_tail(url)
970
1138
        except errors.NotBranchError:
971
1139
            pass
972
1140
        # we always want a working tree
973
 
        WorkingTreeFormat2().initialize(result)
 
1141
        WorkingTreeFormat2().initialize(result,
 
1142
                                        accelerator_tree=accelerator_tree,
 
1143
                                        hardlink=hardlink)
974
1144
        return result
975
1145
 
976
1146
 
1052
1222
        """See BzrDir.create_branch."""
1053
1223
        return self._format.get_branch_format().initialize(self)
1054
1224
 
 
1225
    def destroy_branch(self):
 
1226
        """See BzrDir.create_branch."""
 
1227
        self.transport.delete_tree('branch')
 
1228
 
1055
1229
    def create_repository(self, shared=False):
1056
1230
        """See BzrDir.create_repository."""
1057
1231
        return self._format.repository_format.initialize(self, shared)
1058
1232
 
1059
 
    def create_workingtree(self, revision_id=None):
 
1233
    def destroy_repository(self):
 
1234
        """See BzrDir.destroy_repository."""
 
1235
        self.transport.delete_tree('repository')
 
1236
 
 
1237
    def create_workingtree(self, revision_id=None, from_branch=None,
 
1238
                           accelerator_tree=None, hardlink=False):
1060
1239
        """See BzrDir.create_workingtree."""
1061
 
        from bzrlib.workingtree import WorkingTreeFormat
1062
 
        return self._format.workingtree_format.initialize(self, revision_id)
 
1240
        return self._format.workingtree_format.initialize(
 
1241
            self, revision_id, from_branch=from_branch,
 
1242
            accelerator_tree=accelerator_tree, hardlink=hardlink)
1063
1243
 
1064
1244
    def destroy_workingtree(self):
1065
1245
        """See BzrDir.destroy_workingtree."""
1066
1246
        wt = self.open_workingtree(recommend_upgrade=False)
1067
1247
        repository = wt.branch.repository
1068
1248
        empty = repository.revision_tree(_mod_revision.NULL_REVISION)
1069
 
        wt.revert([], old_tree=empty)
 
1249
        wt.revert(old_tree=empty)
1070
1250
        self.destroy_workingtree_metadata()
1071
1251
 
1072
1252
    def destroy_workingtree_metadata(self):
1198
1378
     * a format string,
1199
1379
     * an open routine.
1200
1380
 
1201
 
    Formats are placed in an dict by their format string for reference 
 
1381
    Formats are placed in a dict by their format string for reference 
1202
1382
    during bzrdir opening. These should be subclasses of BzrDirFormat
1203
1383
    for consistency.
1204
1384
 
1256
1436
        try:
1257
1437
            return klass._formats[format_string]
1258
1438
        except KeyError:
1259
 
            raise errors.UnknownFormatError(format=format_string)
 
1439
            raise errors.UnknownFormatError(format=format_string, kind='bzrdir')
1260
1440
 
1261
1441
    @classmethod
1262
1442
    def get_default_format(klass):
1285
1465
        """
1286
1466
        raise NotImplementedError(self.get_converter)
1287
1467
 
1288
 
    def initialize(self, url):
 
1468
    def initialize(self, url, possible_transports=None):
1289
1469
        """Create a bzr control dir at this url and return an opened copy.
1290
1470
        
1291
1471
        Subclasses should typically override initialize_on_transport
1292
1472
        instead of this method.
1293
1473
        """
1294
 
        return self.initialize_on_transport(get_transport(url))
 
1474
        return self.initialize_on_transport(get_transport(url,
 
1475
                                                          possible_transports))
1295
1476
 
1296
1477
    def initialize_on_transport(self, transport):
1297
1478
        """Initialize a new bzrdir in the base directory of a Transport."""
1303
1484
                                      # FIXME: RBC 20060121 don't peek under
1304
1485
                                      # the covers
1305
1486
                                      mode=temp_control._dir_mode)
 
1487
        if sys.platform == 'win32' and isinstance(transport, LocalTransport):
 
1488
            win32utils.set_file_attr_hidden(transport._abspath('.bzr'))
1306
1489
        file_mode = temp_control._file_mode
1307
1490
        del temp_control
1308
1491
        mutter('created control directory in ' + transport.base)
1309
1492
        control = transport.clone('.bzr')
1310
1493
        utf8_files = [('README', 
1311
 
                       "This is a Bazaar-NG control directory.\n"
1312
 
                       "Do not change any files in this directory.\n"),
 
1494
                       "This is a Bazaar control directory.\n"
 
1495
                       "Do not change any files in this directory.\n"
 
1496
                       "See http://bazaar-vcs.org/ for more information about Bazaar.\n"),
1313
1497
                      ('branch-format', self.get_format_string()),
1314
1498
                      ]
1315
1499
        # NB: no need to escape relative paths that are url safe.
1414
1598
        klass._default_format = format
1415
1599
 
1416
1600
    def __str__(self):
1417
 
        return self.get_format_string()[:-1]
 
1601
        # Trim the newline
 
1602
        return self.get_format_string().rstrip()
1418
1603
 
1419
1604
    @classmethod
1420
1605
    def unregister_format(klass, format):
1664
1849
        return RepositoryFormat.get_default_format()
1665
1850
 
1666
1851
    def __set_repository_format(self, value):
1667
 
        """Allow changint the repository format for metadir formats."""
 
1852
        """Allow changing the repository format for metadir formats."""
1668
1853
        self._repository_format = value
1669
1854
 
1670
1855
    repository_format = property(__return_repository_format, __set_repository_format)
1694
1879
BzrDirFormat._default_format = __default_format
1695
1880
 
1696
1881
 
1697
 
class BzrDirTestProviderAdapter(object):
1698
 
    """A tool to generate a suite testing multiple bzrdir formats at once.
1699
 
 
1700
 
    This is done by copying the test once for each transport and injecting
1701
 
    the transport_server, transport_readonly_server, and bzrdir_format
1702
 
    classes into each copy. Each copy is also given a new id() to make it
1703
 
    easy to identify.
1704
 
    """
1705
 
 
1706
 
    def __init__(self, vfs_factory, transport_server, transport_readonly_server,
1707
 
        formats):
1708
 
        """Create an object to adapt tests.
1709
 
 
1710
 
        :param vfs_server: A factory to create a Transport Server which has
1711
 
            all the VFS methods working, and is writable.
1712
 
        """
1713
 
        self._vfs_factory = vfs_factory
1714
 
        self._transport_server = transport_server
1715
 
        self._transport_readonly_server = transport_readonly_server
1716
 
        self._formats = formats
1717
 
    
1718
 
    def adapt(self, test):
1719
 
        result = unittest.TestSuite()
1720
 
        for format in self._formats:
1721
 
            new_test = deepcopy(test)
1722
 
            new_test.vfs_transport_factory = self._vfs_factory
1723
 
            new_test.transport_server = self._transport_server
1724
 
            new_test.transport_readonly_server = self._transport_readonly_server
1725
 
            new_test.bzrdir_format = format
1726
 
            def make_new_test_id():
1727
 
                new_id = "%s(%s)" % (new_test.id(), format.__class__.__name__)
1728
 
                return lambda: new_id
1729
 
            new_test.id = make_new_test_id()
1730
 
            result.addTest(new_test)
1731
 
        return result
1732
 
 
1733
 
 
1734
1882
class Converter(object):
1735
1883
    """Converts a disk format object from one format to another."""
1736
1884
 
1823
1971
    def _convert_working_inv(self):
1824
1972
        inv = xml4.serializer_v4.read_inventory(
1825
1973
                    self.branch.control_files.get('inventory'))
1826
 
        new_inv_xml = xml5.serializer_v5.write_inventory_to_string(inv)
 
1974
        new_inv_xml = xml5.serializer_v5.write_inventory_to_string(inv, working=True)
1827
1975
        # FIXME inventory is a working tree change.
1828
1976
        self.branch.control_files.put('inventory', StringIO(new_inv_xml))
1829
1977
 
1898
2046
    def _load_updated_inventory(self, rev_id):
1899
2047
        assert rev_id in self.converted_revs
1900
2048
        inv_xml = self.inv_weave.get_text(rev_id)
1901
 
        inv = xml5.serializer_v5.read_inventory_from_string(inv_xml)
 
2049
        inv = xml5.serializer_v5.read_inventory_from_string(inv_xml, rev_id)
1902
2050
        return inv
1903
2051
 
1904
2052
    def _convert_one_rev(self, rev_id):
1908
2056
        present_parents = [p for p in rev.parent_ids
1909
2057
                           if p not in self.absent_revisions]
1910
2058
        self._convert_revision_contents(rev, inv, present_parents)
1911
 
        self._store_new_weave(rev, inv, present_parents)
 
2059
        self._store_new_inv(rev, inv, present_parents)
1912
2060
        self.converted_revs.add(rev_id)
1913
2061
 
1914
 
    def _store_new_weave(self, rev, inv, present_parents):
 
2062
    def _store_new_inv(self, rev, inv, present_parents):
1915
2063
        # the XML is now updated with text versions
1916
2064
        if __debug__:
1917
2065
            entries = inv.iter_entries()
1922
2070
                    (file_id, rev.revision_id)
1923
2071
        new_inv_xml = xml5.serializer_v5.write_inventory_to_string(inv)
1924
2072
        new_inv_sha1 = sha_string(new_inv_xml)
1925
 
        self.inv_weave.add_lines(rev.revision_id, 
 
2073
        self.inv_weave.add_lines(rev.revision_id,
1926
2074
                                 present_parents,
1927
2075
                                 new_inv_xml.splitlines(True))
1928
2076
        rev.inventory_sha1 = new_inv_sha1
1953
2101
            w = Weave(file_id)
1954
2102
            self.text_weaves[file_id] = w
1955
2103
        text_changed = False
1956
 
        previous_entries = ie.find_previous_heads(parent_invs,
1957
 
                                                  None,
1958
 
                                                  None,
1959
 
                                                  entry_vf=w)
1960
 
        for old_revision in previous_entries:
1961
 
                # if this fails, its a ghost ?
1962
 
                assert old_revision in self.converted_revs, \
1963
 
                    "Revision {%s} not in converted_revs" % old_revision
 
2104
        parent_candiate_entries = ie.parent_candidates(parent_invs)
 
2105
        for old_revision in parent_candiate_entries.keys():
 
2106
            # if this fails, its a ghost ?
 
2107
            assert old_revision in self.converted_revs, \
 
2108
                "Revision {%s} not in converted_revs" % old_revision
 
2109
        heads = graph.Graph(self).heads(parent_candiate_entries.keys())
 
2110
        # XXX: Note that this is unordered - and this is tolerable because 
 
2111
        # the previous code was also unordered.
 
2112
        previous_entries = dict((head, parent_candiate_entries[head]) for head
 
2113
            in heads)
1964
2114
        self.snapshot_ie(previous_entries, ie, w, rev_id)
1965
2115
        del ie.text_id
1966
2116
        assert getattr(ie, 'revision', None) is not None
1967
2117
 
 
2118
    @symbol_versioning.deprecated_method(symbol_versioning.one_one)
 
2119
    def get_parents(self, revision_ids):
 
2120
        for revision_id in revision_ids:
 
2121
            yield self.revisions[revision_id].parent_ids
 
2122
 
 
2123
    def get_parent_map(self, revision_ids):
 
2124
        """See graph._StackedParentsProvider.get_parent_map"""
 
2125
        return dict((revision_id, self.revisions[revision_id])
 
2126
                    for revision_id in revision_ids
 
2127
                     if revision_id in self.revisions)
 
2128
 
1968
2129
    def snapshot_ie(self, previous_revisions, ie, w, rev_id):
1969
2130
        # TODO: convert this logic, which is ~= snapshot to
1970
2131
        # a call to:. This needs the path figured out. rather than a work_tree
1979
2140
                ie.revision = previous_ie.revision
1980
2141
                return
1981
2142
        if ie.has_text():
1982
 
            text = self.branch.repository.text_store.get(ie.text_id)
 
2143
            text = self.branch.repository.weave_store.get(ie.text_id)
1983
2144
            file_lines = text.readlines()
1984
2145
            assert sha_strings(file_lines) == ie.text_sha1
1985
2146
            assert sum(map(len, file_lines)) == ie.text_size
2233
2394
    def probe_transport(klass, transport):
2234
2395
        """Return a RemoteBzrDirFormat object if it looks possible."""
2235
2396
        try:
2236
 
            client = transport.get_smart_client()
 
2397
            medium = transport.get_smart_medium()
2237
2398
        except (NotImplementedError, AttributeError,
2238
 
                errors.TransportNotPossible):
 
2399
                errors.TransportNotPossible, errors.NoSmartMedium):
2239
2400
            # no smart server, so not a branch for this format type.
2240
2401
            raise errors.NotBranchError(path=transport.base)
2241
2402
        else:
2242
 
            # Send a 'hello' request in protocol version one, and decline to
2243
 
            # open it if the server doesn't support our required version (2) so
2244
 
            # that the VFS-based transport will do it.
2245
 
            request = client.get_request()
2246
 
            smart_protocol = protocol.SmartClientRequestProtocolOne(request)
2247
 
            server_version = smart_protocol.query_version()
 
2403
            # Decline to open it if the server doesn't support our required
 
2404
            # version (2) so that the VFS-based transport will do it.
 
2405
            try:
 
2406
                server_version = medium.protocol_version()
 
2407
            except errors.SmartProtocolError:
 
2408
                # Apparently there's no usable smart server there, even though
 
2409
                # the medium supports the smart protocol.
 
2410
                raise errors.NotBranchError(path=transport.base)
2248
2411
            if server_version != 2:
2249
2412
                raise errors.NotBranchError(path=transport.base)
2250
2413
            return klass()
2252
2415
    def initialize_on_transport(self, transport):
2253
2416
        try:
2254
2417
            # hand off the request to the smart server
2255
 
            medium = transport.get_smart_medium()
 
2418
            client_medium = transport.get_smart_medium()
2256
2419
        except errors.NoSmartMedium:
2257
2420
            # TODO: lookup the local format from a server hint.
2258
2421
            local_dir_format = BzrDirMetaFormat1()
2259
2422
            return local_dir_format.initialize_on_transport(transport)
2260
 
        client = _SmartClient(medium)
 
2423
        client = _SmartClient(client_medium, transport.base)
2261
2424
        path = client.remote_path_from_transport(transport)
2262
 
        response = _SmartClient(medium).call('BzrDirFormat.initialize', path)
 
2425
        response = client.call('BzrDirFormat.initialize', path)
2263
2426
        assert response[0] in ('ok', ), 'unexpected response code %s' % (response,)
2264
2427
        return remote.RemoteBzrDir(transport)
2265
2428
 
2277
2440
 
2278
2441
class BzrDirFormatInfo(object):
2279
2442
 
2280
 
    def __init__(self, native, deprecated, hidden):
 
2443
    def __init__(self, native, deprecated, hidden, experimental):
2281
2444
        self.deprecated = deprecated
2282
2445
        self.native = native
2283
2446
        self.hidden = hidden
 
2447
        self.experimental = experimental
2284
2448
 
2285
2449
 
2286
2450
class BzrDirFormatRegistry(registry.Registry):
2290
2454
    e.g. BzrDirMeta1 with weave repository.  Also, it's more user-oriented.
2291
2455
    """
2292
2456
 
 
2457
    def __init__(self):
 
2458
        """Create a BzrDirFormatRegistry."""
 
2459
        self._aliases = set()
 
2460
        super(BzrDirFormatRegistry, self).__init__()
 
2461
 
 
2462
    def aliases(self):
 
2463
        """Return a set of the format names which are aliases."""
 
2464
        return frozenset(self._aliases)
 
2465
 
2293
2466
    def register_metadir(self, key,
2294
2467
             repository_format, help, native=True, deprecated=False,
2295
2468
             branch_format=None,
2296
2469
             tree_format=None,
2297
 
             hidden=False):
 
2470
             hidden=False,
 
2471
             experimental=False,
 
2472
             alias=False):
2298
2473
        """Register a metadir subformat.
2299
2474
 
2300
2475
        These all use a BzrDirMetaFormat1 bzrdir, but can be parameterized
2332
2507
            if repository_format is not None:
2333
2508
                bd.repository_format = _load(repository_format)
2334
2509
            return bd
2335
 
        self.register(key, helper, help, native, deprecated, hidden)
 
2510
        self.register(key, helper, help, native, deprecated, hidden,
 
2511
            experimental, alias)
2336
2512
 
2337
2513
    def register(self, key, factory, help, native=True, deprecated=False,
2338
 
                 hidden=False):
 
2514
                 hidden=False, experimental=False, alias=False):
2339
2515
        """Register a BzrDirFormat factory.
2340
2516
        
2341
2517
        The factory must be a callable that takes one parameter: the key.
2344
2520
        This function mainly exists to prevent the info object from being
2345
2521
        supplied directly.
2346
2522
        """
2347
 
        registry.Registry.register(self, key, factory, help, 
2348
 
            BzrDirFormatInfo(native, deprecated, hidden))
 
2523
        registry.Registry.register(self, key, factory, help,
 
2524
            BzrDirFormatInfo(native, deprecated, hidden, experimental))
 
2525
        if alias:
 
2526
            self._aliases.add(key)
2349
2527
 
2350
2528
    def register_lazy(self, key, module_name, member_name, help, native=True,
2351
 
                      deprecated=False, hidden=False):
2352
 
        registry.Registry.register_lazy(self, key, module_name, member_name, 
2353
 
            help, BzrDirFormatInfo(native, deprecated, hidden))
 
2529
        deprecated=False, hidden=False, experimental=False, alias=False):
 
2530
        registry.Registry.register_lazy(self, key, module_name, member_name,
 
2531
            help, BzrDirFormatInfo(native, deprecated, hidden, experimental))
 
2532
        if alias:
 
2533
            self._aliases.add(key)
2354
2534
 
2355
2535
    def set_default(self, key):
2356
2536
        """Set the 'default' key to be a clone of the supplied key.
2357
2537
        
2358
2538
        This method must be called once and only once.
2359
2539
        """
2360
 
        registry.Registry.register(self, 'default', self.get(key), 
 
2540
        registry.Registry.register(self, 'default', self.get(key),
2361
2541
            self.get_help(key), info=self.get_info(key))
 
2542
        self._aliases.add('default')
2362
2543
 
2363
2544
    def set_default_repository(self, key):
2364
2545
        """Set the FormatRegistry default and Repository default.
2377
2558
 
2378
2559
    def help_topic(self, topic):
2379
2560
        output = textwrap.dedent("""\
2380
 
            Bazaar directory formats
2381
 
            ------------------------
2382
 
 
2383
2561
            These formats can be used for creating branches, working trees, and
2384
2562
            repositories.
2385
2563
 
2386
2564
            """)
 
2565
        default_realkey = None
2387
2566
        default_help = self.get_help('default')
2388
2567
        help_pairs = []
2389
2568
        for key in self.keys():
2398
2577
        def wrapped(key, help, info):
2399
2578
            if info.native:
2400
2579
                help = '(native) ' + help
2401
 
            return '  %s:\n%s\n\n' % (key, 
 
2580
            return ':%s:\n%s\n\n' % (key, 
2402
2581
                    textwrap.fill(help, initial_indent='    ', 
2403
2582
                    subsequent_indent='    '))
2404
 
        output += wrapped('%s/default' % default_realkey, default_help,
2405
 
                          self.get_info('default'))
 
2583
        if default_realkey is not None:
 
2584
            output += wrapped(default_realkey, '(default) %s' % default_help,
 
2585
                              self.get_info('default'))
2406
2586
        deprecated_pairs = []
 
2587
        experimental_pairs = []
2407
2588
        for key, help in help_pairs:
2408
2589
            info = self.get_info(key)
2409
2590
            if info.hidden:
2410
2591
                continue
2411
2592
            elif info.deprecated:
2412
2593
                deprecated_pairs.append((key, help))
 
2594
            elif info.experimental:
 
2595
                experimental_pairs.append((key, help))
2413
2596
            else:
2414
2597
                output += wrapped(key, help, info)
 
2598
        if len(experimental_pairs) > 0:
 
2599
            output += "Experimental formats are shown below.\n\n"
 
2600
            for key, help in experimental_pairs:
 
2601
                info = self.get_info(key)
 
2602
                output += wrapped(key, help, info)
2415
2603
        if len(deprecated_pairs) > 0:
2416
 
            output += "Deprecated formats\n------------------\n\n"
 
2604
            output += "Deprecated formats are shown below.\n\n"
2417
2605
            for key, help in deprecated_pairs:
2418
2606
                info = self.get_info(key)
2419
2607
                output += wrapped(key, help, info)
2454
2642
    branch_format='bzrlib.branch.BzrBranchFormat6',
2455
2643
    tree_format='bzrlib.workingtree.WorkingTreeFormat4',
2456
2644
    )
 
2645
format_registry.register_metadir('rich-root',
 
2646
    'bzrlib.repofmt.knitrepo.RepositoryFormatKnit4',
 
2647
    help='New in 1.0.  Better handling of tree roots.  Incompatible with'
 
2648
        ' bzr < 1.0',
 
2649
    branch_format='bzrlib.branch.BzrBranchFormat6',
 
2650
    tree_format='bzrlib.workingtree.WorkingTreeFormat4',
 
2651
    )
2457
2652
format_registry.register_metadir('dirstate-with-subtree',
2458
2653
    'bzrlib.repofmt.knitrepo.RepositoryFormatKnit3',
2459
2654
    help='New in 0.15: Fast local operations and improved scaling for '
2461
2656
        'bzr branches. Incompatible with bzr < 0.15.',
2462
2657
    branch_format='bzrlib.branch.BzrBranchFormat6',
2463
2658
    tree_format='bzrlib.workingtree.WorkingTreeFormat4',
2464
 
    hidden=True,
2465
 
    )
2466
 
format_registry.set_default('dirstate')
 
2659
    experimental=True,
 
2660
    hidden=True,
 
2661
    )
 
2662
format_registry.register_metadir('pack-0.92',
 
2663
    'bzrlib.repofmt.pack_repo.RepositoryFormatKnitPack1',
 
2664
    help='New in 0.92: Pack-based format with data compatible with '
 
2665
        'dirstate-tags format repositories. Interoperates with '
 
2666
        'bzr repositories before 0.92 but cannot be read by bzr < 0.92. '
 
2667
        'Previously called knitpack-experimental.  '
 
2668
        'For more information, see '
 
2669
        'http://doc.bazaar-vcs.org/latest/developers/packrepo.html.',
 
2670
    branch_format='bzrlib.branch.BzrBranchFormat6',
 
2671
    tree_format='bzrlib.workingtree.WorkingTreeFormat4',
 
2672
    )
 
2673
format_registry.register_metadir('pack-0.92-subtree',
 
2674
    'bzrlib.repofmt.pack_repo.RepositoryFormatKnitPack3',
 
2675
    help='New in 0.92: Pack-based format with data compatible with '
 
2676
        'dirstate-with-subtree format repositories. Interoperates with '
 
2677
        'bzr repositories before 0.92 but cannot be read by bzr < 0.92. '
 
2678
        'Previously called knitpack-experimental.  '
 
2679
        'For more information, see '
 
2680
        'http://doc.bazaar-vcs.org/latest/developers/packrepo.html.',
 
2681
    branch_format='bzrlib.branch.BzrBranchFormat6',
 
2682
    tree_format='bzrlib.workingtree.WorkingTreeFormat4',
 
2683
    hidden=True,
 
2684
    experimental=True,
 
2685
    )
 
2686
format_registry.register_metadir('rich-root-pack',
 
2687
    'bzrlib.repofmt.pack_repo.RepositoryFormatKnitPack4',
 
2688
    help='New in 1.0: Pack-based format with data compatible with '
 
2689
        'rich-root format repositories. Incompatible with'
 
2690
        ' bzr < 1.0',
 
2691
    branch_format='bzrlib.branch.BzrBranchFormat6',
 
2692
    tree_format='bzrlib.workingtree.WorkingTreeFormat4',
 
2693
    )
 
2694
# The following two formats should always just be aliases.
 
2695
format_registry.register_metadir('development',
 
2696
    'bzrlib.repofmt.pack_repo.RepositoryFormatPackDevelopment0',
 
2697
    help='Current development format. Can convert data to and from pack-0.92 '
 
2698
        '(and anything compatible with pack-0.92) format repositories. '
 
2699
        'Repositories in this format can only be read by bzr.dev. '
 
2700
        'Please read '
 
2701
        'http://doc.bazaar-vcs.org/latest/developers/development-repo.html '
 
2702
        'before use.',
 
2703
    branch_format='bzrlib.branch.BzrBranchFormat6',
 
2704
    tree_format='bzrlib.workingtree.WorkingTreeFormat4',
 
2705
    experimental=True,
 
2706
    alias=True,
 
2707
    )
 
2708
format_registry.register_metadir('development-subtree',
 
2709
    'bzrlib.repofmt.pack_repo.RepositoryFormatPackDevelopment0Subtree',
 
2710
    help='Current development format, subtree variant. Can convert data to and '
 
2711
        'from pack-0.92 (and anything compatible with pack-0.92) format '
 
2712
        'repositories. Repositories in this format can only be read by '
 
2713
        'bzr.dev. Please read '
 
2714
        'http://doc.bazaar-vcs.org/latest/developers/development-repo.html '
 
2715
        'before use.',
 
2716
    branch_format='bzrlib.branch.BzrBranchFormat6',
 
2717
    tree_format='bzrlib.workingtree.WorkingTreeFormat4',
 
2718
    experimental=True,
 
2719
    alias=True,
 
2720
    )
 
2721
# And the development formats which the will have aliased one of follow:
 
2722
format_registry.register_metadir('development0',
 
2723
    'bzrlib.repofmt.pack_repo.RepositoryFormatPackDevelopment0',
 
2724
    help='Trivial rename of pack-0.92 to provide a development format. '
 
2725
        'Please read '
 
2726
        'http://doc.bazaar-vcs.org/latest/developers/development-repo.html '
 
2727
        'before use.',
 
2728
    branch_format='bzrlib.branch.BzrBranchFormat6',
 
2729
    tree_format='bzrlib.workingtree.WorkingTreeFormat4',
 
2730
    hidden=True,
 
2731
    experimental=True,
 
2732
    )
 
2733
format_registry.register_metadir('development0-subtree',
 
2734
    'bzrlib.repofmt.pack_repo.RepositoryFormatPackDevelopment0Subtree',
 
2735
    help='Trivial rename of pack-0.92-subtree to provide a development format. '
 
2736
        'Please read '
 
2737
        'http://doc.bazaar-vcs.org/latest/developers/development-repo.html '
 
2738
        'before use.',
 
2739
    branch_format='bzrlib.branch.BzrBranchFormat6',
 
2740
    tree_format='bzrlib.workingtree.WorkingTreeFormat4',
 
2741
    hidden=True,
 
2742
    experimental=True,
 
2743
    )
 
2744
format_registry.set_default('pack-0.92')