~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: 2007-11-03 01:53:30 UTC
  • mfrom: (2955.1.1 trunk)
  • Revision ID: pqm@pqm.ubuntu.com-20071103015330-pt1tec7wyxwwcey8
Fix #158972 don't use timeout for HttpServer

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2005, 2006, 2007, 2008 Canonical Ltd
 
1
# Copyright (C) 2005, 2006, 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
29
29
 
30
30
from cStringIO import StringIO
31
31
import os
32
 
import sys
33
32
 
34
33
from bzrlib.lazy_import import lazy_import
35
34
lazy_import(globals(), """
39
38
 
40
39
import bzrlib
41
40
from bzrlib import (
42
 
    config,
43
41
    errors,
44
42
    graph,
45
43
    lockable_files,
46
44
    lockdir,
47
 
    osutils,
48
45
    registry,
49
46
    remote,
50
47
    revision as _mod_revision,
51
48
    symbol_versioning,
52
49
    ui,
53
50
    urlutils,
54
 
    versionedfile,
55
 
    win32utils,
 
51
    xml4,
 
52
    xml5,
56
53
    workingtree,
57
54
    workingtree_4,
58
 
    xml4,
59
 
    xml5,
60
55
    )
61
56
from bzrlib.osutils import (
62
57
    sha_strings,
63
58
    sha_string,
64
59
    )
65
 
from bzrlib.repository import Repository
66
60
from bzrlib.smart.client import _SmartClient
67
61
from bzrlib.smart import protocol
 
62
from bzrlib.store.revision.text import TextRevisionStore
 
63
from bzrlib.store.text import TextStore
68
64
from bzrlib.store.versioned import WeaveStore
69
65
from bzrlib.transactions import WriteTransaction
70
66
from bzrlib.transport import (
82
78
from bzrlib.symbol_versioning import (
83
79
    deprecated_function,
84
80
    deprecated_method,
 
81
    zero_ninetyone,
85
82
    )
86
83
 
87
84
 
91
88
    BzrDir instances let you create or open any of the things that can be
92
89
    found within .bzr - checkouts, branches and repositories.
93
90
    
94
 
    :ivar transport:
 
91
    transport
95
92
        the transport which this bzr dir is rooted at (i.e. file:///.../.bzr/)
96
 
    :ivar root_transport:
 
93
    root_transport
97
94
        a transport connected to the directory this bzr was opened from
98
95
        (i.e. the parent directory holding the .bzr directory).
99
 
 
100
 
    Everything in the bzrdir should have the same file permissions.
101
96
    """
102
97
 
103
98
    def break_lock(self):
157
152
                format.get_format_description(),
158
153
                basedir)
159
154
 
160
 
    def clone(self, url, revision_id=None, force_new_repo=False,
161
 
              preserve_stacking=False):
 
155
    def clone(self, url, revision_id=None, force_new_repo=False):
162
156
        """Clone this bzrdir and its contents to url verbatim.
163
157
 
164
 
        :param url: The url create the clone at.  If url's last component does
165
 
            not exist, it will be created.
166
 
        :param revision_id: The tip revision-id to use for any branch or
167
 
            working tree.  If not None, then the clone operation may tune
 
158
        If url's last component does not exist, it will be created.
 
159
 
 
160
        if revision_id is not None, then the clone operation may tune
168
161
            itself to download less data.
169
 
        :param force_new_repo: Do not use a shared repository for the target
 
162
        :param force_new_repo: Do not use a shared repository for the target 
170
163
                               even if one is available.
171
 
        :param preserve_stacking: When cloning a stacked branch, stack the
172
 
            new branch on top of the other branch's stacked-on branch.
173
164
        """
174
165
        return self.clone_on_transport(get_transport(url),
175
166
                                       revision_id=revision_id,
176
 
                                       force_new_repo=force_new_repo,
177
 
                                       preserve_stacking=preserve_stacking)
 
167
                                       force_new_repo=force_new_repo)
178
168
 
179
169
    def clone_on_transport(self, transport, revision_id=None,
180
 
                           force_new_repo=False, preserve_stacking=False,
181
 
                           stacked_on=None):
 
170
                           force_new_repo=False):
182
171
        """Clone this bzrdir and its contents to transport verbatim.
183
172
 
184
 
        :param transport: The transport for the location to produce the clone
185
 
            at.  If the target directory does not exist, it will be created.
186
 
        :param revision_id: The tip revision-id to use for any branch or
187
 
            working tree.  If not None, then the clone operation may tune
 
173
        If the target directory does not exist, it will be created.
 
174
 
 
175
        if revision_id is not None, then the clone operation may tune
188
176
            itself to download less data.
189
 
        :param force_new_repo: Do not use a shared repository for the target,
 
177
        :param force_new_repo: Do not use a shared repository for the target 
190
178
                               even if one is available.
191
 
        :param preserve_stacking: When cloning a stacked branch, stack the
192
 
            new branch on top of the other branch's stacked-on branch.
193
179
        """
194
180
        transport.ensure_base()
195
 
        require_stacking = (stacked_on is not None)
196
 
        metadir = self.cloning_metadir(require_stacking)
197
 
        result = metadir.initialize_on_transport(transport)
198
 
        repository_policy = None
 
181
        result = self._format.initialize_on_transport(transport)
199
182
        try:
200
183
            local_repo = self.find_repository()
201
184
        except errors.NoRepositoryPresent:
202
185
            local_repo = None
203
 
        try:
204
 
            local_branch = self.open_branch()
205
 
        except errors.NotBranchError:
206
 
            local_branch = None
207
 
        else:
208
 
            # enable fallbacks when branch is not a branch reference
209
 
            if local_branch.repository.has_same_location(local_repo):
210
 
                local_repo = local_branch.repository
211
 
            if preserve_stacking:
212
 
                try:
213
 
                    stacked_on = local_branch.get_stacked_on_url()
214
 
                except (errors.UnstackableBranchFormat,
215
 
                        errors.UnstackableRepositoryFormat,
216
 
                        errors.NotStacked):
217
 
                    pass
218
 
 
219
186
        if local_repo:
220
187
            # may need to copy content in
221
 
            repository_policy = result.determine_repository_policy(
222
 
                force_new_repo, stacked_on, self.root_transport.base,
223
 
                require_stacking=require_stacking)
224
 
            make_working_trees = local_repo.make_working_trees()
225
 
            result_repo = repository_policy.acquire_repository(
226
 
                make_working_trees, local_repo.is_shared())
227
 
            result_repo.fetch(local_repo, revision_id=revision_id)
228
 
        else:
229
 
            result_repo = None
 
188
            if force_new_repo:
 
189
                result_repo = local_repo.clone(
 
190
                    result,
 
191
                    revision_id=revision_id)
 
192
                result_repo.set_make_working_trees(local_repo.make_working_trees())
 
193
            else:
 
194
                try:
 
195
                    result_repo = result.find_repository()
 
196
                    # fetch content this dir needs.
 
197
                    result_repo.fetch(local_repo, revision_id=revision_id)
 
198
                except errors.NoRepositoryPresent:
 
199
                    # needed to make one anyway.
 
200
                    result_repo = local_repo.clone(
 
201
                        result,
 
202
                        revision_id=revision_id)
 
203
                    result_repo.set_make_working_trees(local_repo.make_working_trees())
230
204
        # 1 if there is a branch present
231
205
        #   make sure its content is available in the target repository
232
206
        #   clone it.
233
 
        if local_branch is not None:
234
 
            result_branch = local_branch.clone(result, revision_id=revision_id)
235
 
            if repository_policy is not None:
236
 
                repository_policy.configure_branch(result_branch)
237
 
        if result_repo is None or result_repo.make_working_trees():
238
 
            try:
239
 
                self.open_workingtree().clone(result)
240
 
            except (errors.NoWorkingTree, errors.NotLocalUrl):
241
 
                pass
 
207
        try:
 
208
            self.open_branch().clone(result, revision_id=revision_id)
 
209
        except errors.NotBranchError:
 
210
            pass
 
211
        try:
 
212
            self.open_workingtree().clone(result)
 
213
        except (errors.NoWorkingTree, errors.NotLocalUrl):
 
214
            pass
242
215
        return result
243
216
 
244
217
    # TODO: This should be given a Transport, and should chdir up; otherwise
265
238
            format = BzrDirFormat.get_default_format()
266
239
        return format.initialize_on_transport(t)
267
240
 
268
 
    @staticmethod
269
 
    def find_bzrdirs(transport, evaluate=None, list_current=None):
270
 
        """Find bzrdirs recursively from current location.
271
 
 
272
 
        This is intended primarily as a building block for more sophisticated
273
 
        functionality, like finding trees under a directory, or finding
274
 
        branches that use a given repository.
275
 
        :param evaluate: An optional callable that yields recurse, value,
276
 
            where recurse controls whether this bzrdir is recursed into
277
 
            and value is the value to yield.  By default, all bzrdirs
278
 
            are recursed into, and the return value is the bzrdir.
279
 
        :param list_current: if supplied, use this function to list the current
280
 
            directory, instead of Transport.list_dir
281
 
        :return: a generator of found bzrdirs, or whatever evaluate returns.
282
 
        """
283
 
        if list_current is None:
284
 
            def list_current(transport):
285
 
                return transport.list_dir('')
286
 
        if evaluate is None:
287
 
            def evaluate(bzrdir):
288
 
                return True, bzrdir
289
 
 
290
 
        pending = [transport]
291
 
        while len(pending) > 0:
292
 
            current_transport = pending.pop()
293
 
            recurse = True
294
 
            try:
295
 
                bzrdir = BzrDir.open_from_transport(current_transport)
296
 
            except errors.NotBranchError:
297
 
                pass
298
 
            else:
299
 
                recurse, value = evaluate(bzrdir)
300
 
                yield value
301
 
            try:
302
 
                subdirs = list_current(current_transport)
303
 
            except errors.NoSuchFile:
304
 
                continue
305
 
            if recurse:
306
 
                for subdir in sorted(subdirs, reverse=True):
307
 
                    pending.append(current_transport.clone(subdir))
308
 
 
309
 
    @staticmethod
310
 
    def find_branches(transport):
311
 
        """Find all branches under a transport.
312
 
 
313
 
        This will find all branches below the transport, including branches
314
 
        inside other branches.  Where possible, it will use
315
 
        Repository.find_branches.
316
 
 
317
 
        To list all the branches that use a particular Repository, see
318
 
        Repository.find_branches
319
 
        """
320
 
        def evaluate(bzrdir):
321
 
            try:
322
 
                repository = bzrdir.open_repository()
323
 
            except errors.NoRepositoryPresent:
324
 
                pass
325
 
            else:
326
 
                return False, (None, repository)
327
 
            try:
328
 
                branch = bzrdir.open_branch()
329
 
            except errors.NotBranchError:
330
 
                return True, (None, None)
331
 
            else:
332
 
                return True, (branch, None)
333
 
        branches = []
334
 
        for branch, repo in BzrDir.find_bzrdirs(transport, evaluate=evaluate):
335
 
            if repo is not None:
336
 
                branches.extend(repo.find_branches())
337
 
            if branch is not None:
338
 
                branches.append(branch)
339
 
        return branches
340
 
 
341
 
    def destroy_repository(self):
342
 
        """Destroy the repository in this BzrDir"""
343
 
        raise NotImplementedError(self.destroy_repository)
344
 
 
345
241
    def create_branch(self):
346
242
        """Create a branch in this BzrDir.
347
243
 
375
271
        bzrdir._find_or_create_repository(force_new_repo)
376
272
        return bzrdir.create_branch()
377
273
 
378
 
    def determine_repository_policy(self, force_new_repo=False, stack_on=None,
379
 
                                    stack_on_pwd=None, require_stacking=False):
380
 
        """Return an object representing a policy to use.
381
 
 
382
 
        This controls whether a new repository is created, or a shared
383
 
        repository used instead.
384
 
 
385
 
        If stack_on is supplied, will not seek a containing shared repo.
386
 
 
387
 
        :param force_new_repo: If True, require a new repository to be created.
388
 
        :param stack_on: If supplied, the location to stack on.  If not
389
 
            supplied, a default_stack_on location may be used.
390
 
        :param stack_on_pwd: If stack_on is relative, the location it is
391
 
            relative to.
392
 
        """
393
 
        def repository_policy(found_bzrdir):
394
 
            stack_on = None
395
 
            stack_on_pwd = None
396
 
            config = found_bzrdir.get_config()
397
 
            stop = False
398
 
            if config is not None:
399
 
                stack_on = config.get_default_stack_on()
400
 
                if stack_on is not None:
401
 
                    stack_on_pwd = found_bzrdir.root_transport.base
402
 
                    stop = True
403
 
                    note('Using default stacking branch %s at %s', stack_on,
404
 
                         stack_on_pwd)
405
 
            # does it have a repository ?
406
 
            try:
407
 
                repository = found_bzrdir.open_repository()
408
 
            except errors.NoRepositoryPresent:
409
 
                repository = None
410
 
            else:
411
 
                if ((found_bzrdir.root_transport.base !=
412
 
                     self.root_transport.base) and not repository.is_shared()):
413
 
                    repository = None
414
 
                else:
415
 
                    stop = True
416
 
            if not stop:
417
 
                return None, False
418
 
            if repository:
419
 
                return UseExistingRepository(repository, stack_on,
420
 
                    stack_on_pwd, require_stacking=require_stacking), True
421
 
            else:
422
 
                return CreateRepository(self, stack_on, stack_on_pwd,
423
 
                    require_stacking=require_stacking), True
424
 
 
425
 
        if not force_new_repo:
426
 
            if stack_on is None:
427
 
                policy = self._find_containing(repository_policy)
428
 
                if policy is not None:
429
 
                    return policy
430
 
            else:
431
 
                try:
432
 
                    return UseExistingRepository(self.open_repository(),
433
 
                        stack_on, stack_on_pwd,
434
 
                        require_stacking=require_stacking)
435
 
                except errors.NoRepositoryPresent:
436
 
                    pass
437
 
        return CreateRepository(self, stack_on, stack_on_pwd,
438
 
                                require_stacking=require_stacking)
439
 
 
440
274
    def _find_or_create_repository(self, force_new_repo):
441
275
        """Create a new repository if needed, returning the repository."""
442
 
        policy = self.determine_repository_policy(force_new_repo)
443
 
        return policy.acquire_repository()
444
 
 
 
276
        if force_new_repo:
 
277
            return self.create_repository()
 
278
        try:
 
279
            return self.find_repository()
 
280
        except errors.NoRepositoryPresent:
 
281
            return self.create_repository()
 
282
        
445
283
    @staticmethod
446
284
    def create_branch_convenience(base, force_new_repo=False,
447
285
                                  force_new_tree=None, format=None,
488
326
        return result
489
327
 
490
328
    @staticmethod
 
329
    @deprecated_function(zero_ninetyone)
 
330
    def create_repository(base, shared=False, format=None):
 
331
        """Create a new BzrDir and Repository at the url 'base'.
 
332
 
 
333
        If no format is supplied, this will default to the current default
 
334
        BzrDirFormat by default, and use whatever repository format that that
 
335
        uses for bzrdirformat.create_repository.
 
336
 
 
337
        :param shared: Create a shared repository rather than a standalone
 
338
                       repository.
 
339
        The Repository object is returned.
 
340
 
 
341
        This must be overridden as an instance method in child classes, where
 
342
        it should take no parameters and construct whatever repository format
 
343
        that child class desires.
 
344
 
 
345
        This method is deprecated, please call create_repository on a bzrdir
 
346
        instance instead.
 
347
        """
 
348
        bzrdir = BzrDir.create(base, format)
 
349
        return bzrdir.create_repository(shared)
 
350
 
 
351
    @staticmethod
491
352
    def create_standalone_workingtree(base, format=None):
492
353
        """Create a new BzrDir, WorkingTree, Branch and Repository at 'base'.
493
354
 
509
370
                                               format=format).bzrdir
510
371
        return bzrdir.create_workingtree()
511
372
 
512
 
    def create_workingtree(self, revision_id=None, from_branch=None,
513
 
        accelerator_tree=None, hardlink=False):
 
373
    def create_workingtree(self, revision_id=None):
514
374
        """Create a working tree at this BzrDir.
515
375
        
516
 
        :param revision_id: create it as of this revision id.
517
 
        :param from_branch: override bzrdir branch (for lightweight checkouts)
518
 
        :param accelerator_tree: A tree which can be used for retrieving file
519
 
            contents more quickly than the revision tree, i.e. a workingtree.
520
 
            The revision tree will be used for cases where accelerator_tree's
521
 
            content is different.
 
376
        revision_id: create it as of this revision id.
522
377
        """
523
378
        raise NotImplementedError(self.create_workingtree)
524
379
 
562
417
        """
563
418
        raise NotImplementedError(self.destroy_workingtree_metadata)
564
419
 
565
 
    def _find_containing(self, evaluate):
566
 
        """Find something in a containing control directory.
567
 
 
568
 
        This method will scan containing control dirs, until it finds what
569
 
        it is looking for, decides that it will never find it, or runs out
570
 
        of containing control directories to check.
571
 
 
572
 
        It is used to implement find_repository and
573
 
        determine_repository_policy.
574
 
 
575
 
        :param evaluate: A function returning (value, stop).  If stop is True,
576
 
            the value will be returned.
577
 
        """
578
 
        found_bzrdir = self
579
 
        while True:
580
 
            result, stop = evaluate(found_bzrdir)
581
 
            if stop:
582
 
                return result
583
 
            next_transport = found_bzrdir.root_transport.clone('..')
584
 
            if (found_bzrdir.root_transport.base == next_transport.base):
585
 
                # top of the file system
586
 
                return None
587
 
            # find the next containing bzrdir
588
 
            try:
589
 
                found_bzrdir = BzrDir.open_containing_from_transport(
590
 
                    next_transport)[0]
591
 
            except errors.NotBranchError:
592
 
                return None
593
 
 
594
420
    def find_repository(self):
595
421
        """Find the repository that should be used.
596
422
 
598
424
        new branches as well as to hook existing branches up to their
599
425
        repository.
600
426
        """
601
 
        def usable_repository(found_bzrdir):
 
427
        try:
 
428
            return self.open_repository()
 
429
        except errors.NoRepositoryPresent:
 
430
            pass
 
431
        next_transport = self.root_transport.clone('..')
 
432
        while True:
 
433
            # find the next containing bzrdir
 
434
            try:
 
435
                found_bzrdir = BzrDir.open_containing_from_transport(
 
436
                    next_transport)[0]
 
437
            except errors.NotBranchError:
 
438
                # none found
 
439
                raise errors.NoRepositoryPresent(self)
602
440
            # does it have a repository ?
603
441
            try:
604
442
                repository = found_bzrdir.open_repository()
605
443
            except errors.NoRepositoryPresent:
606
 
                return None, False
607
 
            if found_bzrdir.root_transport.base == self.root_transport.base:
608
 
                return repository, True
609
 
            elif repository.is_shared():
610
 
                return repository, True
 
444
                next_transport = found_bzrdir.root_transport.clone('..')
 
445
                if (found_bzrdir.root_transport.base == next_transport.base):
 
446
                    # top of the file system
 
447
                    break
 
448
                else:
 
449
                    continue
 
450
            if ((found_bzrdir.root_transport.base ==
 
451
                 self.root_transport.base) or repository.is_shared()):
 
452
                return repository
611
453
            else:
612
 
                return None, True
613
 
 
614
 
        found_repo = self._find_containing(usable_repository)
615
 
        if found_repo is None:
616
 
            raise errors.NoRepositoryPresent(self)
617
 
        return found_repo
 
454
                raise errors.NoRepositoryPresent(self)
 
455
        raise errors.NoRepositoryPresent(self)
618
456
 
619
457
    def get_branch_reference(self):
620
458
        """Return the referenced URL for the branch in this bzrdir.
637
475
        guaranteed to point to an existing directory ready for use.
638
476
        """
639
477
        raise NotImplementedError(self.get_branch_transport)
640
 
 
641
 
    def _find_creation_modes(self):
642
 
        """Determine the appropriate modes for files and directories.
643
 
 
644
 
        They're always set to be consistent with the base directory,
645
 
        assuming that this transport allows setting modes.
646
 
        """
647
 
        # TODO: Do we need or want an option (maybe a config setting) to turn
648
 
        # this off or override it for particular locations? -- mbp 20080512
649
 
        if self._mode_check_done:
650
 
            return
651
 
        self._mode_check_done = True
652
 
        try:
653
 
            st = self.transport.stat('.')
654
 
        except errors.TransportNotPossible:
655
 
            self._dir_mode = None
656
 
            self._file_mode = None
657
 
        else:
658
 
            # Check the directory mode, but also make sure the created
659
 
            # directories and files are read-write for this user. This is
660
 
            # mostly a workaround for filesystems which lie about being able to
661
 
            # write to a directory (cygwin & win32)
662
 
            if (st.st_mode & 07777 == 00000):
663
 
                # FTP allows stat but does not return dir/file modes
664
 
                self._dir_mode = None
665
 
                self._file_mode = None
666
 
            else:
667
 
                self._dir_mode = (st.st_mode & 07777) | 00700
668
 
                # Remove the sticky and execute bits for files
669
 
                self._file_mode = self._dir_mode & ~07111
670
 
 
671
 
    def _get_file_mode(self):
672
 
        """Return Unix mode for newly created files, or None.
673
 
        """
674
 
        if not self._mode_check_done:
675
 
            self._find_creation_modes()
676
 
        return self._file_mode
677
 
 
678
 
    def _get_dir_mode(self):
679
 
        """Return Unix mode for newly created directories, or None.
680
 
        """
681
 
        if not self._mode_check_done:
682
 
            self._find_creation_modes()
683
 
        return self._dir_mode
684
478
        
685
479
    def get_repository_transport(self, repository_format):
686
480
        """Get the transport for use by repository format in this BzrDir.
707
501
        guaranteed to point to an existing directory ready for use.
708
502
        """
709
503
        raise NotImplementedError(self.get_workingtree_transport)
710
 
 
711
 
    def get_config(self):
712
 
        if getattr(self, '_get_config', None) is None:
713
 
            return None
714
 
        return self._get_config()
715
 
 
 
504
        
716
505
    def __init__(self, _transport, _format):
717
506
        """Initialize a Bzr control dir object.
718
507
        
725
514
        self._format = _format
726
515
        self.transport = _transport.clone('.bzr')
727
516
        self.root_transport = _transport
728
 
        self._mode_check_done = False
729
517
 
730
518
    def is_control_filename(self, filename):
731
519
        """True if filename is the name of a path which is reserved for bzrdir's.
866
654
                raise errors.NotBranchError(path=url)
867
655
            a_transport = new_t
868
656
 
869
 
    def _get_tree_branch(self):
870
 
        """Return the branch and tree, if any, for this bzrdir.
871
 
 
872
 
        Return None for tree if not present or inaccessible.
873
 
        Raise NotBranchError if no branch is present.
874
 
        :return: (tree, branch)
875
 
        """
876
 
        try:
877
 
            tree = self.open_workingtree()
878
 
        except (errors.NoWorkingTree, errors.NotLocalUrl):
879
 
            tree = None
880
 
            branch = self.open_branch()
881
 
        else:
882
 
            branch = tree.branch
883
 
        return tree, branch
884
 
 
885
 
    @classmethod
886
 
    def open_tree_or_branch(klass, location):
887
 
        """Return the branch and working tree at a location.
888
 
 
889
 
        If there is no tree at the location, tree will be None.
890
 
        If there is no branch at the location, an exception will be
891
 
        raised
892
 
        :return: (tree, branch)
893
 
        """
894
 
        bzrdir = klass.open(location)
895
 
        return bzrdir._get_tree_branch()
896
 
 
897
657
    @classmethod
898
658
    def open_containing_tree_or_branch(klass, location):
899
659
        """Return the branch and working tree contained by a location.
905
665
        relpath is the portion of the path that is contained by the branch.
906
666
        """
907
667
        bzrdir, relpath = klass.open_containing(location)
908
 
        tree, branch = bzrdir._get_tree_branch()
 
668
        try:
 
669
            tree = bzrdir.open_workingtree()
 
670
        except (errors.NoWorkingTree, errors.NotLocalUrl):
 
671
            tree = None
 
672
            branch = bzrdir.open_branch()
 
673
        else:
 
674
            branch = tree.branch
909
675
        return tree, branch, relpath
910
676
 
911
 
    @classmethod
912
 
    def open_containing_tree_branch_or_repository(klass, location):
913
 
        """Return the working tree, branch and repo contained by a location.
914
 
 
915
 
        Returns (tree, branch, repository, relpath).
916
 
        If there is no tree containing the location, tree will be None.
917
 
        If there is no branch containing the location, branch will be None.
918
 
        If there is no repository containing the location, repository will be
919
 
        None.
920
 
        relpath is the portion of the path that is contained by the innermost
921
 
        BzrDir.
922
 
 
923
 
        If no tree, branch or repository is found, a NotBranchError is raised.
924
 
        """
925
 
        bzrdir, relpath = klass.open_containing(location)
926
 
        try:
927
 
            tree, branch = bzrdir._get_tree_branch()
928
 
        except errors.NotBranchError:
929
 
            try:
930
 
                repo = bzrdir.find_repository()
931
 
                return None, None, repo, relpath
932
 
            except (errors.NoRepositoryPresent):
933
 
                raise errors.NotBranchError(location)
934
 
        return tree, branch, branch.repository, relpath
935
 
 
936
677
    def open_repository(self, _unsupported=False):
937
678
        """Open the repository object at this BzrDir if one is present.
938
679
 
946
687
        raise NotImplementedError(self.open_repository)
947
688
 
948
689
    def open_workingtree(self, _unsupported=False,
949
 
                         recommend_upgrade=True, from_branch=None):
 
690
            recommend_upgrade=True):
950
691
        """Open the workingtree object at this BzrDir if one is present.
951
692
 
952
693
        :param recommend_upgrade: Optional keyword parameter, when True (the
953
694
            default), emit through the ui module a recommendation that the user
954
695
            upgrade the working tree when the workingtree being opened is old
955
696
            (but still fully supported).
956
 
        :param from_branch: override bzrdir branch (for lightweight checkouts)
957
697
        """
958
698
        raise NotImplementedError(self.open_workingtree)
959
699
 
987
727
            return False
988
728
 
989
729
    def _cloning_metadir(self):
990
 
        """Produce a metadir suitable for cloning with.
991
 
        
992
 
        :returns: (destination_bzrdir_format, source_repository)
993
 
        """
 
730
        """Produce a metadir suitable for cloning with."""
994
731
        result_format = self._format.__class__()
995
732
        try:
996
733
            try:
997
734
                branch = self.open_branch()
998
735
                source_repository = branch.repository
999
 
                result_format._branch_format = branch._format
1000
736
            except errors.NotBranchError:
1001
737
                source_branch = None
1002
738
                source_repository = self.open_repository()
1020
756
            result_format.workingtree_format = tree._format.__class__()
1021
757
        return result_format, source_repository
1022
758
 
1023
 
    def cloning_metadir(self, require_stacking=False):
 
759
    def cloning_metadir(self):
1024
760
        """Produce a metadir suitable for cloning or sprouting with.
1025
761
 
1026
762
        These operations may produce workingtrees (yes, even though they're
1027
763
        "cloning" something that doesn't have a tree), so a viable workingtree
1028
764
        format must be selected.
1029
 
 
1030
 
        :require_stacking: If True, non-stackable formats will be upgraded
1031
 
            to similar stackable formats.
1032
 
        :returns: a BzrDirFormat with all component formats either set
1033
 
            appropriately or set to None if that component should not be
1034
 
            created.
1035
765
        """
1036
766
        format, repository = self._cloning_metadir()
1037
767
        if format._workingtree_format is None:
1039
769
                return format
1040
770
            tree_format = repository._format._matchingbzrdir.workingtree_format
1041
771
            format.workingtree_format = tree_format.__class__()
1042
 
        if (require_stacking and not
1043
 
            format.get_branch_format().supports_stacking()):
1044
 
            # We need to make a stacked branch, but the default format for the
1045
 
            # target doesn't support stacking.  So force a branch that *can*
1046
 
            # support stacking.
1047
 
            from bzrlib.branch import BzrBranchFormat7
1048
 
            format._branch_format = BzrBranchFormat7()
1049
 
            mutter("using %r for stacking" % (format._branch_format,))
1050
 
            from bzrlib.repofmt import pack_repo
1051
 
            if format.repository_format.rich_root_data:
1052
 
                bzrdir_format_name = '1.6.1-rich-root'
1053
 
                repo_format = pack_repo.RepositoryFormatKnitPack5RichRoot()
1054
 
            else:
1055
 
                bzrdir_format_name = '1.6'
1056
 
                repo_format = pack_repo.RepositoryFormatKnitPack5()
1057
 
            note('Source format does not support stacking, using format:'
1058
 
                 ' \'%s\'\n  %s\n',
1059
 
                 bzrdir_format_name, repo_format.get_format_description())
1060
 
            format.repository_format = repo_format
1061
772
        return format
1062
773
 
1063
774
    def checkout_metadir(self):
1064
775
        return self.cloning_metadir()
1065
776
 
1066
777
    def sprout(self, url, revision_id=None, force_new_repo=False,
1067
 
               recurse='down', possible_transports=None,
1068
 
               accelerator_tree=None, hardlink=False, stacked=False):
 
778
               recurse='down', possible_transports=None):
1069
779
        """Create a copy of this bzrdir prepared for use as a new line of
1070
780
        development.
1071
781
 
1078
788
 
1079
789
        if revision_id is not None, then the clone operation may tune
1080
790
            itself to download less data.
1081
 
        :param accelerator_tree: A tree which can be used for retrieving file
1082
 
            contents more quickly than the revision tree, i.e. a workingtree.
1083
 
            The revision tree will be used for cases where accelerator_tree's
1084
 
            content is different.
1085
 
        :param hardlink: If true, hard-link files from accelerator_tree,
1086
 
            where possible.
1087
 
        :param stacked: If true, create a stacked branch referring to the
1088
 
            location of this control directory.
1089
791
        """
1090
792
        target_transport = get_transport(url, possible_transports)
1091
793
        target_transport.ensure_base()
1092
 
        cloning_format = self.cloning_metadir(stacked)
1093
 
        # Create/update the result branch
 
794
        cloning_format = self.cloning_metadir()
1094
795
        result = cloning_format.initialize_on_transport(target_transport)
1095
796
        try:
1096
797
            source_branch = self.open_branch()
1097
798
            source_repository = source_branch.repository
1098
 
            if stacked:
1099
 
                stacked_branch_url = self.root_transport.base
1100
 
            else:
1101
 
                # if a stacked branch wasn't requested, we don't create one
1102
 
                # even if the origin was stacked
1103
 
                stacked_branch_url = None
1104
799
        except errors.NotBranchError:
1105
800
            source_branch = None
1106
801
            try:
1107
802
                source_repository = self.open_repository()
1108
803
            except errors.NoRepositoryPresent:
1109
804
                source_repository = None
1110
 
            stacked_branch_url = None
1111
 
        repository_policy = result.determine_repository_policy(
1112
 
            force_new_repo, stacked_branch_url, require_stacking=stacked)
1113
 
        result_repo = repository_policy.acquire_repository()
1114
 
        if source_repository is not None:
1115
 
            # Fetch while stacked to prevent unstacked fetch from
1116
 
            # Branch.sprout.
1117
 
            result_repo.fetch(source_repository, revision_id=revision_id)
1118
 
 
1119
 
        if source_branch is None:
1120
 
            # this is for sprouting a bzrdir without a branch; is that
1121
 
            # actually useful?
1122
 
            # Not especially, but it's part of the contract.
1123
 
            result_branch = result.create_branch()
1124
 
        else:
1125
 
            # Force NULL revision to avoid using repository before stacking
1126
 
            # is configured.
1127
 
            result_branch = source_branch.sprout(
1128
 
                result, revision_id=_mod_revision.NULL_REVISION)
1129
 
            parent_location = result_branch.get_parent()
1130
 
        mutter("created new branch %r" % (result_branch,))
1131
 
        repository_policy.configure_branch(result_branch)
 
805
        if force_new_repo:
 
806
            result_repo = None
 
807
        else:
 
808
            try:
 
809
                result_repo = result.find_repository()
 
810
            except errors.NoRepositoryPresent:
 
811
                result_repo = None
 
812
        if source_repository is None and result_repo is not None:
 
813
            pass
 
814
        elif source_repository is None and result_repo is None:
 
815
            # no repo available, make a new one
 
816
            result.create_repository()
 
817
        elif source_repository is not None and result_repo is None:
 
818
            # have source, and want to make a new target repo
 
819
            result_repo = source_repository.sprout(result,
 
820
                                                   revision_id=revision_id)
 
821
        else:
 
822
            # fetch needed content into target.
 
823
            if source_repository is not None:
 
824
                # would rather do 
 
825
                # source_repository.copy_content_into(result_repo,
 
826
                #                                     revision_id=revision_id)
 
827
                # so we can override the copy method
 
828
                result_repo.fetch(source_repository, revision_id=revision_id)
1132
829
        if source_branch is not None:
1133
 
            source_branch.copy_content_into(result_branch, revision_id)
1134
 
            # Override copy_content_into
1135
 
            result_branch.set_parent(parent_location)
1136
 
 
1137
 
        # Create/update the result working tree
 
830
            source_branch.sprout(result, revision_id=revision_id)
 
831
        else:
 
832
            result.create_branch()
1138
833
        if isinstance(target_transport, LocalTransport) and (
1139
834
            result_repo is None or result_repo.make_working_trees()):
1140
 
            wt = result.create_workingtree(accelerator_tree=accelerator_tree,
1141
 
                hardlink=hardlink)
 
835
            wt = result.create_workingtree()
1142
836
            wt.lock_write()
1143
837
            try:
1144
838
                if wt.path2id('') is None:
1155
849
                basis = wt.basis_tree()
1156
850
                basis.lock_read()
1157
851
                subtrees = basis.iter_references()
 
852
                recurse_branch = wt.branch
1158
853
            elif source_branch is not None:
1159
854
                basis = source_branch.basis_tree()
1160
855
                basis.lock_read()
1161
856
                subtrees = basis.iter_references()
 
857
                recurse_branch = source_branch
1162
858
            else:
1163
859
                subtrees = []
1164
860
                basis = None
1168
864
                    sublocation = source_branch.reference_parent(file_id, path)
1169
865
                    sublocation.bzrdir.sprout(target,
1170
866
                        basis.get_reference_revision(file_id, path),
1171
 
                        force_new_repo=force_new_repo, recurse=recurse,
1172
 
                        stacked=stacked)
 
867
                        force_new_repo=force_new_repo, recurse=recurse)
1173
868
            finally:
1174
869
                if basis is not None:
1175
870
                    basis.unlock()
1182
877
    def __init__(self, _transport, _format):
1183
878
        """See BzrDir.__init__."""
1184
879
        super(BzrDirPreSplitOut, self).__init__(_transport, _format)
 
880
        assert self._format._lock_class == lockable_files.TransportLock
 
881
        assert self._format._lock_file_name == 'branch-lock'
1185
882
        self._control_files = lockable_files.LockableFiles(
1186
883
                                            self.get_branch_transport(None),
1187
884
                                            self._format._lock_file_name,
1191
888
        """Pre-splitout bzrdirs do not suffer from stale locks."""
1192
889
        raise NotImplementedError(self.break_lock)
1193
890
 
1194
 
    def cloning_metadir(self, require_stacking=False):
1195
 
        """Produce a metadir suitable for cloning with."""
1196
 
        if require_stacking:
1197
 
            return format_registry.make_bzrdir('1.6')
1198
 
        return self._format.__class__()
1199
 
 
1200
 
    def clone(self, url, revision_id=None, force_new_repo=False,
1201
 
              preserve_stacking=False):
1202
 
        """See BzrDir.clone().
1203
 
 
1204
 
        force_new_repo has no effect, since this family of formats always
1205
 
        require a new repository.
1206
 
        preserve_stacking has no effect, since no source branch using this
1207
 
        family of formats can be stacked, so there is no stacking to preserve.
1208
 
        """
 
891
    def clone(self, url, revision_id=None, force_new_repo=False):
 
892
        """See BzrDir.clone()."""
 
893
        from bzrlib.workingtree import WorkingTreeFormat2
1209
894
        self._make_tail(url)
1210
895
        result = self._format._initialize_for_clone(url)
1211
896
        self.open_repository().clone(result, revision_id=revision_id)
1212
897
        from_branch = self.open_branch()
1213
898
        from_branch.clone(result, revision_id=revision_id)
1214
899
        try:
1215
 
            tree = self.open_workingtree()
 
900
            self.open_workingtree().clone(result)
1216
901
        except errors.NotLocalUrl:
1217
902
            # make a new one, this format always has to have one.
1218
 
            result._init_workingtree()
1219
 
        else:
1220
 
            tree.clone(result)
 
903
            try:
 
904
                WorkingTreeFormat2().initialize(result)
 
905
            except errors.NotLocalUrl:
 
906
                # but we cannot do it for remote trees.
 
907
                to_branch = result.open_branch()
 
908
                WorkingTreeFormat2().stub_initialize_remote(to_branch.control_files)
1221
909
        return result
1222
910
 
1223
911
    def create_branch(self):
1224
912
        """See BzrDir.create_branch."""
1225
 
        return self._format.get_branch_format().initialize(self)
 
913
        return self.open_branch()
1226
914
 
1227
915
    def destroy_branch(self):
1228
916
        """See BzrDir.destroy_branch."""
1234
922
            raise errors.IncompatibleFormat('shared repository', self._format)
1235
923
        return self.open_repository()
1236
924
 
1237
 
    def destroy_repository(self):
1238
 
        """See BzrDir.destroy_repository."""
1239
 
        raise errors.UnsupportedOperation(self.destroy_repository, self)
1240
 
 
1241
 
    def create_workingtree(self, revision_id=None, from_branch=None,
1242
 
                           accelerator_tree=None, hardlink=False):
 
925
    def create_workingtree(self, revision_id=None):
1243
926
        """See BzrDir.create_workingtree."""
1244
 
        # The workingtree is sometimes created when the bzrdir is created,
1245
 
        # but not when cloning.
1246
 
 
1247
927
        # this looks buggy but is not -really-
1248
928
        # because this format creates the workingtree when the bzrdir is
1249
929
        # created
1253
933
        # that can do wonky stuff here, and that only
1254
934
        # happens for creating checkouts, which cannot be 
1255
935
        # done on this format anyway. So - acceptable wart.
1256
 
        try:
1257
 
            result = self.open_workingtree(recommend_upgrade=False)
1258
 
        except errors.NoSuchFile:
1259
 
            result = self._init_workingtree()
 
936
        result = self.open_workingtree(recommend_upgrade=False)
1260
937
        if revision_id is not None:
1261
938
            if revision_id == _mod_revision.NULL_REVISION:
1262
939
                result.set_parent_ids([])
1264
941
                result.set_parent_ids([revision_id])
1265
942
        return result
1266
943
 
1267
 
    def _init_workingtree(self):
1268
 
        from bzrlib.workingtree import WorkingTreeFormat2
1269
 
        try:
1270
 
            return WorkingTreeFormat2().initialize(self)
1271
 
        except errors.NotLocalUrl:
1272
 
            # Even though we can't access the working tree, we need to
1273
 
            # create its control files.
1274
 
            return WorkingTreeFormat2()._stub_initialize_on_transport(
1275
 
                self.transport, self._control_files._file_mode)
1276
 
 
1277
944
    def destroy_workingtree(self):
1278
945
        """See BzrDir.destroy_workingtree."""
1279
946
        raise errors.UnsupportedOperation(self.destroy_workingtree, self)
1329
996
        return format.open(self, _found=True)
1330
997
 
1331
998
    def sprout(self, url, revision_id=None, force_new_repo=False,
1332
 
               possible_transports=None, accelerator_tree=None,
1333
 
               hardlink=False, stacked=False):
 
999
               possible_transports=None):
1334
1000
        """See BzrDir.sprout()."""
1335
 
        if stacked:
1336
 
            raise errors.UnstackableBranchFormat(
1337
 
                self._format, self.root_transport.base)
1338
1001
        from bzrlib.workingtree import WorkingTreeFormat2
1339
1002
        self._make_tail(url)
1340
1003
        result = self._format._initialize_for_clone(url)
1347
1010
        except errors.NotBranchError:
1348
1011
            pass
1349
1012
        # we always want a working tree
1350
 
        WorkingTreeFormat2().initialize(result,
1351
 
                                        accelerator_tree=accelerator_tree,
1352
 
                                        hardlink=hardlink)
 
1013
        WorkingTreeFormat2().initialize(result)
1353
1014
        return result
1354
1015
 
1355
1016
 
1439
1100
        """See BzrDir.create_repository."""
1440
1101
        return self._format.repository_format.initialize(self, shared)
1441
1102
 
1442
 
    def destroy_repository(self):
1443
 
        """See BzrDir.destroy_repository."""
1444
 
        self.transport.delete_tree('repository')
1445
 
 
1446
 
    def create_workingtree(self, revision_id=None, from_branch=None,
1447
 
                           accelerator_tree=None, hardlink=False):
 
1103
    def create_workingtree(self, revision_id=None):
1448
1104
        """See BzrDir.create_workingtree."""
1449
 
        return self._format.workingtree_format.initialize(
1450
 
            self, revision_id, from_branch=from_branch,
1451
 
            accelerator_tree=accelerator_tree, hardlink=hardlink)
 
1105
        from bzrlib.workingtree import WorkingTreeFormat
 
1106
        return self._format.workingtree_format.initialize(self, revision_id)
1452
1107
 
1453
1108
    def destroy_workingtree(self):
1454
1109
        """See BzrDir.destroy_workingtree."""
1578
1233
            basedir=self.root_transport.base)
1579
1234
        return format.open(self, _found=True)
1580
1235
 
1581
 
    def _get_config(self):
1582
 
        return config.BzrDirConfig(self.transport)
1583
 
 
1584
1236
 
1585
1237
class BzrDirFormat(object):
1586
1238
    """An encapsulation of the initialization and open routines for a format.
1648
1300
        try:
1649
1301
            return klass._formats[format_string]
1650
1302
        except KeyError:
1651
 
            raise errors.UnknownFormatError(format=format_string, kind='bzrdir')
 
1303
            raise errors.UnknownFormatError(format=format_string)
1652
1304
 
1653
1305
    @classmethod
1654
1306
    def get_default_format(klass):
1696
1348
                                      # FIXME: RBC 20060121 don't peek under
1697
1349
                                      # the covers
1698
1350
                                      mode=temp_control._dir_mode)
1699
 
        if sys.platform == 'win32' and isinstance(transport, LocalTransport):
1700
 
            win32utils.set_file_attr_hidden(transport._abspath('.bzr'))
1701
1351
        file_mode = temp_control._file_mode
1702
1352
        del temp_control
1703
 
        bzrdir_transport = transport.clone('.bzr')
1704
 
        utf8_files = [('README',
1705
 
                       "This is a Bazaar control directory.\n"
1706
 
                       "Do not change any files in this directory.\n"
1707
 
                       "See http://bazaar-vcs.org/ for more information about Bazaar.\n"),
 
1353
        mutter('created control directory in ' + transport.base)
 
1354
        control = transport.clone('.bzr')
 
1355
        utf8_files = [('README', 
 
1356
                       "This is a Bazaar-NG control directory.\n"
 
1357
                       "Do not change any files in this directory.\n"),
1708
1358
                      ('branch-format', self.get_format_string()),
1709
1359
                      ]
1710
1360
        # NB: no need to escape relative paths that are url safe.
1711
 
        control_files = lockable_files.LockableFiles(bzrdir_transport,
1712
 
            self._lock_file_name, self._lock_class)
 
1361
        control_files = lockable_files.LockableFiles(control,
 
1362
                            self._lock_file_name, self._lock_class)
1713
1363
        control_files.create_lock()
1714
1364
        control_files.lock_write()
1715
1365
        try:
1716
 
            for (filename, content) in utf8_files:
1717
 
                bzrdir_transport.put_bytes(filename, content,
1718
 
                    mode=file_mode)
 
1366
            for file, content in utf8_files:
 
1367
                control_files.put_utf8(file, content)
1719
1368
        finally:
1720
1369
            control_files.unlock()
1721
1370
        return self.open(transport, _found=True)
1800
1449
        klass._control_server_formats.append(format)
1801
1450
 
1802
1451
    @classmethod
 
1452
    @symbol_versioning.deprecated_method(symbol_versioning.zero_fourteen)
 
1453
    def set_default_format(klass, format):
 
1454
        klass._set_default_format(format)
 
1455
 
 
1456
    @classmethod
1803
1457
    def _set_default_format(klass, format):
1804
1458
        """Set default format (for testing behavior of defaults only)"""
1805
1459
        klass._default_format = format
1810
1464
 
1811
1465
    @classmethod
1812
1466
    def unregister_format(klass, format):
 
1467
        assert klass._formats[format.get_format_string()] is format
1813
1468
        del klass._formats[format.get_format_string()]
1814
1469
 
1815
1470
    @classmethod
1886
1541
        """See BzrDirFormat.get_format_string()."""
1887
1542
        return "Bazaar-NG branch, format 5\n"
1888
1543
 
1889
 
    def get_branch_format(self):
1890
 
        from bzrlib import branch
1891
 
        return branch.BzrBranchFormat4()
1892
 
 
1893
1544
    def get_format_description(self):
1894
1545
        """See BzrDirFormat.get_format_description()."""
1895
1546
        return "All-in-one format 5"
1909
1560
        """
1910
1561
        from bzrlib.branch import BzrBranchFormat4
1911
1562
        from bzrlib.repofmt.weaverepo import RepositoryFormat5
 
1563
        from bzrlib.workingtree import WorkingTreeFormat2
1912
1564
        result = (super(BzrDirFormat5, self).initialize_on_transport(transport))
1913
1565
        RepositoryFormat5().initialize(result, _internal=True)
1914
1566
        if not _cloning:
1915
1567
            branch = BzrBranchFormat4().initialize(result)
1916
 
            result._init_workingtree()
 
1568
            try:
 
1569
                WorkingTreeFormat2().initialize(result)
 
1570
            except errors.NotLocalUrl:
 
1571
                # Even though we can't access the working tree, we need to
 
1572
                # create its control files.
 
1573
                WorkingTreeFormat2().stub_initialize_remote(branch.control_files)
1917
1574
        return result
1918
1575
 
1919
1576
    def _open(self, transport):
1947
1604
        """See BzrDirFormat.get_format_description()."""
1948
1605
        return "All-in-one format 6"
1949
1606
 
1950
 
    def get_branch_format(self):
1951
 
        from bzrlib import branch
1952
 
        return branch.BzrBranchFormat4()
1953
 
 
1954
1607
    def get_converter(self, format=None):
1955
1608
        """See BzrDirFormat.get_converter()."""
1956
1609
        # there is one and only one upgrade path here.
1966
1619
        """
1967
1620
        from bzrlib.branch import BzrBranchFormat4
1968
1621
        from bzrlib.repofmt.weaverepo import RepositoryFormat6
 
1622
        from bzrlib.workingtree import WorkingTreeFormat2
1969
1623
        result = super(BzrDirFormat6, self).initialize_on_transport(transport)
1970
1624
        RepositoryFormat6().initialize(result, _internal=True)
1971
1625
        if not _cloning:
1972
1626
            branch = BzrBranchFormat4().initialize(result)
1973
 
            result._init_workingtree()
 
1627
            try:
 
1628
                WorkingTreeFormat2().initialize(result)
 
1629
            except errors.NotLocalUrl:
 
1630
                # Even though we can't access the working tree, we need to
 
1631
                # create its control files.
 
1632
                WorkingTreeFormat2().stub_initialize_remote(branch.control_files)
1974
1633
        return result
1975
1634
 
1976
1635
    def _open(self, transport):
2051
1710
        return RepositoryFormat.get_default_format()
2052
1711
 
2053
1712
    def __set_repository_format(self, value):
2054
 
        """Allow changing the repository format for metadir formats."""
 
1713
        """Allow changint the repository format for metadir formats."""
2055
1714
        self._repository_format = value
2056
1715
 
2057
1716
    repository_format = property(__return_repository_format, __set_repository_format)
2157
1816
        self.pb.note('  %6d revisions not present', len(self.absent_revisions))
2158
1817
        self.pb.note('  %6d texts', self.text_count)
2159
1818
        self._cleanup_spare_files_after_format4()
2160
 
        self.branch._transport.put_bytes(
2161
 
            'branch-format',
2162
 
            BzrDirFormat5().get_format_string(),
2163
 
            mode=self.bzrdir._get_file_mode())
 
1819
        self.branch.control_files.put_utf8('branch-format', BzrDirFormat5().get_format_string())
2164
1820
 
2165
1821
    def _cleanup_spare_files_after_format4(self):
2166
1822
        # FIXME working tree upgrade foo.
2175
1831
 
2176
1832
    def _convert_working_inv(self):
2177
1833
        inv = xml4.serializer_v4.read_inventory(
2178
 
                self.branch._transport.get('inventory'))
 
1834
                    self.branch.control_files.get('inventory'))
2179
1835
        new_inv_xml = xml5.serializer_v5.write_inventory_to_string(inv, working=True)
2180
 
        self.branch._transport.put_bytes('inventory', new_inv_xml,
2181
 
            mode=self.bzrdir._get_file_mode())
 
1836
        # FIXME inventory is a working tree change.
 
1837
        self.branch.control_files.put('inventory', StringIO(new_inv_xml))
2182
1838
 
2183
1839
    def _write_all_weaves(self):
2184
1840
        controlweaves = WeaveStore(self.bzrdir.transport, prefixed=False)
2204
1860
        self.bzrdir.transport.mkdir('revision-store')
2205
1861
        revision_transport = self.bzrdir.transport.clone('revision-store')
2206
1862
        # TODO permissions
2207
 
        from bzrlib.xml5 import serializer_v5
2208
 
        from bzrlib.repofmt.weaverepo import RevisionTextStore
2209
 
        revision_store = RevisionTextStore(revision_transport,
2210
 
            serializer_v5, False, versionedfile.PrefixMapper(),
2211
 
            lambda:True, lambda:True)
 
1863
        _revision_store = TextRevisionStore(TextStore(revision_transport,
 
1864
                                                      prefixed=False,
 
1865
                                                      compressed=True))
2212
1866
        try:
 
1867
            transaction = WriteTransaction()
2213
1868
            for i, rev_id in enumerate(self.converted_revs):
2214
1869
                self.pb.update('write revision', i, len(self.converted_revs))
2215
 
                text = serializer_v5.write_revision_to_string(
2216
 
                    self.revisions[rev_id])
2217
 
                key = (rev_id,)
2218
 
                revision_store.add_lines(key, None, osutils.split_lines(text))
 
1870
                _revision_store.add_revision(self.revisions[rev_id], transaction)
2219
1871
        finally:
2220
1872
            self.pb.clear()
2221
1873
            
2234
1886
                         rev_id)
2235
1887
            self.absent_revisions.add(rev_id)
2236
1888
        else:
2237
 
            rev = self.branch.repository.get_revision(rev_id)
 
1889
            rev = self.branch.repository._revision_store.get_revision(rev_id,
 
1890
                self.branch.repository.get_transaction())
2238
1891
            for parent_id in rev.parent_ids:
2239
1892
                self.known_revisions.add(parent_id)
2240
1893
                self.to_read.append(parent_id)
2241
1894
            self.revisions[rev_id] = rev
2242
1895
 
2243
1896
    def _load_old_inventory(self, rev_id):
 
1897
        assert rev_id not in self.converted_revs
2244
1898
        old_inv_xml = self.branch.repository.inventory_store.get(rev_id).read()
2245
1899
        inv = xml4.serializer_v4.read_inventory_from_string(old_inv_xml)
2246
1900
        inv.revision_id = rev_id
2247
1901
        rev = self.revisions[rev_id]
 
1902
        if rev.inventory_sha1:
 
1903
            assert rev.inventory_sha1 == sha_string(old_inv_xml), \
 
1904
                'inventory sha mismatch for {%s}' % rev_id
2248
1905
        return inv
2249
1906
 
2250
1907
    def _load_updated_inventory(self, rev_id):
 
1908
        assert rev_id in self.converted_revs
2251
1909
        inv_xml = self.inv_weave.get_text(rev_id)
2252
 
        inv = xml5.serializer_v5.read_inventory_from_string(inv_xml, rev_id)
 
1910
        inv = xml5.serializer_v5.read_inventory_from_string(inv_xml)
2253
1911
        return inv
2254
1912
 
2255
1913
    def _convert_one_rev(self, rev_id):
2263
1921
        self.converted_revs.add(rev_id)
2264
1922
 
2265
1923
    def _store_new_inv(self, rev, inv, present_parents):
 
1924
        # the XML is now updated with text versions
 
1925
        if __debug__:
 
1926
            entries = inv.iter_entries()
 
1927
            entries.next()
 
1928
            for path, ie in entries:
 
1929
                assert getattr(ie, 'revision', None) is not None, \
 
1930
                    'no revision on {%s} in {%s}' % \
 
1931
                    (file_id, rev.revision_id)
2266
1932
        new_inv_xml = xml5.serializer_v5.write_inventory_to_string(inv)
2267
1933
        new_inv_sha1 = sha_string(new_inv_xml)
2268
1934
        self.inv_weave.add_lines(rev.revision_id,
2297
1963
            self.text_weaves[file_id] = w
2298
1964
        text_changed = False
2299
1965
        parent_candiate_entries = ie.parent_candidates(parent_invs)
 
1966
        for old_revision in parent_candiate_entries.keys():
 
1967
            # if this fails, its a ghost ?
 
1968
            assert old_revision in self.converted_revs, \
 
1969
                "Revision {%s} not in converted_revs" % old_revision
2300
1970
        heads = graph.Graph(self).heads(parent_candiate_entries.keys())
2301
1971
        # XXX: Note that this is unordered - and this is tolerable because 
2302
1972
        # the previous code was also unordered.
2304
1974
            in heads)
2305
1975
        self.snapshot_ie(previous_entries, ie, w, rev_id)
2306
1976
        del ie.text_id
 
1977
        assert getattr(ie, 'revision', None) is not None
2307
1978
 
2308
 
    @symbol_versioning.deprecated_method(symbol_versioning.one_one)
2309
1979
    def get_parents(self, revision_ids):
2310
1980
        for revision_id in revision_ids:
2311
1981
            yield self.revisions[revision_id].parent_ids
2312
1982
 
2313
 
    def get_parent_map(self, revision_ids):
2314
 
        """See graph._StackedParentsProvider.get_parent_map"""
2315
 
        return dict((revision_id, self.revisions[revision_id])
2316
 
                    for revision_id in revision_ids
2317
 
                     if revision_id in self.revisions)
2318
 
 
2319
1983
    def snapshot_ie(self, previous_revisions, ie, w, rev_id):
2320
1984
        # TODO: convert this logic, which is ~= snapshot to
2321
1985
        # a call to:. This needs the path figured out. rather than a work_tree
2330
1994
                ie.revision = previous_ie.revision
2331
1995
                return
2332
1996
        if ie.has_text():
2333
 
            text = self.branch.repository._text_store.get(ie.text_id)
 
1997
            text = self.branch.repository.weave_store.get(ie.text_id)
2334
1998
            file_lines = text.readlines()
 
1999
            assert sha_strings(file_lines) == ie.text_sha1
 
2000
            assert sum(map(len, file_lines)) == ie.text_size
2335
2001
            w.add_lines(rev_id, previous_revisions, file_lines)
2336
2002
            self.text_count += 1
2337
2003
        else:
2384
2050
                if (filename.endswith(".weave") or
2385
2051
                    filename.endswith(".gz") or
2386
2052
                    filename.endswith(".sig")):
2387
 
                    file_id, suffix = os.path.splitext(filename)
 
2053
                    file_id = os.path.splitext(filename)[0]
2388
2054
                else:
2389
2055
                    file_id = filename
2390
 
                    suffix = ''
2391
 
                new_name = store._mapper.map((file_id,)) + suffix
 
2056
                prefix_dir = store.hash_prefix(file_id)
2392
2057
                # FIXME keep track of the dirs made RBC 20060121
2393
2058
                try:
2394
 
                    store_transport.move(filename, new_name)
 
2059
                    store_transport.move(filename, prefix_dir + '/' + filename)
2395
2060
                except errors.NoSuchFile: # catches missing dirs strangely enough
2396
 
                    store_transport.mkdir(osutils.dirname(new_name))
2397
 
                    store_transport.move(filename, new_name)
2398
 
        self.bzrdir.transport.put_bytes(
2399
 
            'branch-format',
2400
 
            BzrDirFormat6().get_format_string(),
2401
 
            mode=self.bzrdir._get_file_mode())
 
2061
                    store_transport.mkdir(prefix_dir)
 
2062
                    store_transport.move(filename, prefix_dir + '/' + filename)
 
2063
        self.bzrdir._control_files.put_utf8('branch-format', BzrDirFormat6().get_format_string())
2402
2064
 
2403
2065
 
2404
2066
class ConvertBzrDir6ToMeta(Converter):
2413
2075
        self.count = 0
2414
2076
        self.total = 20 # the steps we know about
2415
2077
        self.garbage_inventories = []
2416
 
        self.dir_mode = self.bzrdir._get_dir_mode()
2417
 
        self.file_mode = self.bzrdir._get_file_mode()
2418
2078
 
2419
2079
        self.pb.note('starting upgrade from format 6 to metadir')
2420
 
        self.bzrdir.transport.put_bytes(
2421
 
                'branch-format',
2422
 
                "Converting to format 6",
2423
 
                mode=self.file_mode)
 
2080
        self.bzrdir._control_files.put_utf8('branch-format', "Converting to format 6")
2424
2081
        # its faster to move specific files around than to open and use the apis...
2425
2082
        # first off, nuke ancestry.weave, it was never used.
2426
2083
        try:
2436
2093
            if name.startswith('basis-inventory.'):
2437
2094
                self.garbage_inventories.append(name)
2438
2095
        # create new directories for repository, working tree and branch
 
2096
        self.dir_mode = self.bzrdir._control_files._dir_mode
 
2097
        self.file_mode = self.bzrdir._control_files._file_mode
2439
2098
        repository_names = [('inventory.weave', True),
2440
2099
                            ('revision-store', True),
2441
2100
                            ('weaves', True)]
2489
2148
            for entry in checkout_files:
2490
2149
                self.move_entry('checkout', entry)
2491
2150
            if last_revision is not None:
2492
 
                self.bzrdir.transport.put_bytes(
 
2151
                self.bzrdir._control_files.put_utf8(
2493
2152
                    'checkout/last-revision', last_revision)
2494
 
        self.bzrdir.transport.put_bytes(
2495
 
            'branch-format',
2496
 
            BzrDirMetaFormat1().get_format_string(),
2497
 
            mode=self.file_mode)
 
2153
        self.bzrdir._control_files.put_utf8(
 
2154
            'branch-format', BzrDirMetaFormat1().get_format_string())
2498
2155
        return BzrDir.open(self.bzrdir.root_transport.base)
2499
2156
 
2500
2157
    def make_lock(self, name):
2518
2175
                raise
2519
2176
 
2520
2177
    def put_format(self, dirname, format):
2521
 
        self.bzrdir.transport.put_bytes('%s/format' % dirname,
2522
 
            format.get_format_string(),
2523
 
            self.file_mode)
 
2178
        self.bzrdir._control_files.put_utf8('%s/format' % dirname, format.get_format_string())
2524
2179
 
2525
2180
 
2526
2181
class ConvertMetaToMeta(Converter):
2556
2211
            pass
2557
2212
        else:
2558
2213
            # TODO: conversions of Branch and Tree should be done by
2559
 
            # InterXFormat lookups/some sort of registry.
 
2214
            # InterXFormat lookups
2560
2215
            # Avoid circular imports
2561
2216
            from bzrlib import branch as _mod_branch
2562
 
            old = branch._format.__class__
2563
 
            new = self.target_format.get_branch_format().__class__
2564
 
            while old != new:
2565
 
                if (old == _mod_branch.BzrBranchFormat5 and
2566
 
                    new in (_mod_branch.BzrBranchFormat6,
2567
 
                        _mod_branch.BzrBranchFormat7)):
2568
 
                    branch_converter = _mod_branch.Converter5to6()
2569
 
                elif (old == _mod_branch.BzrBranchFormat6 and
2570
 
                    new == _mod_branch.BzrBranchFormat7):
2571
 
                    branch_converter = _mod_branch.Converter6to7()
2572
 
                else:
2573
 
                    raise errors.BadConversionTarget("No converter", new)
 
2217
            if (branch._format.__class__ is _mod_branch.BzrBranchFormat5 and
 
2218
                self.target_format.get_branch_format().__class__ is
 
2219
                _mod_branch.BzrBranchFormat6):
 
2220
                branch_converter = _mod_branch.Converter5to6()
2574
2221
                branch_converter.convert(branch)
2575
 
                branch = self.bzrdir.open_branch()
2576
 
                old = branch._format.__class__
2577
2222
        try:
2578
2223
            tree = self.bzrdir.open_workingtree(recommend_upgrade=False)
2579
2224
        except (errors.NoWorkingTree, errors.NotLocalUrl):
2603
2248
    def probe_transport(klass, transport):
2604
2249
        """Return a RemoteBzrDirFormat object if it looks possible."""
2605
2250
        try:
2606
 
            medium = transport.get_smart_medium()
 
2251
            client = transport.get_smart_client()
2607
2252
        except (NotImplementedError, AttributeError,
2608
 
                errors.TransportNotPossible, errors.NoSmartMedium,
2609
 
                errors.SmartProtocolError):
 
2253
                errors.TransportNotPossible):
2610
2254
            # no smart server, so not a branch for this format type.
2611
2255
            raise errors.NotBranchError(path=transport.base)
2612
2256
        else:
2613
 
            # Decline to open it if the server doesn't support our required
2614
 
            # version (3) so that the VFS-based transport will do it.
2615
 
            if medium.should_probe():
2616
 
                try:
2617
 
                    server_version = medium.protocol_version()
2618
 
                except errors.SmartProtocolError:
2619
 
                    # Apparently there's no usable smart server there, even though
2620
 
                    # the medium supports the smart protocol.
2621
 
                    raise errors.NotBranchError(path=transport.base)
2622
 
                if server_version != '2':
2623
 
                    raise errors.NotBranchError(path=transport.base)
 
2257
            # Send a 'hello' request in protocol version one, and decline to
 
2258
            # open it if the server doesn't support our required version (2) so
 
2259
            # that the VFS-based transport will do it.
 
2260
            request = client.get_request()
 
2261
            smart_protocol = protocol.SmartClientRequestProtocolOne(request)
 
2262
            server_version = smart_protocol.query_version()
 
2263
            if server_version != 2:
 
2264
                raise errors.NotBranchError(path=transport.base)
2624
2265
            return klass()
2625
2266
 
2626
2267
    def initialize_on_transport(self, transport):
2627
2268
        try:
2628
2269
            # hand off the request to the smart server
2629
 
            client_medium = transport.get_smart_medium()
 
2270
            shared_medium = transport.get_shared_medium()
2630
2271
        except errors.NoSmartMedium:
2631
2272
            # TODO: lookup the local format from a server hint.
2632
2273
            local_dir_format = BzrDirMetaFormat1()
2633
2274
            return local_dir_format.initialize_on_transport(transport)
2634
 
        client = _SmartClient(client_medium)
 
2275
        client = _SmartClient(shared_medium)
2635
2276
        path = client.remote_path_from_transport(transport)
2636
 
        response = client.call('BzrDirFormat.initialize', path)
2637
 
        if response[0] != 'ok':
2638
 
            raise errors.SmartProtocolError('unexpected response code %s' % (response,))
 
2277
        response = _SmartClient(shared_medium).call('BzrDirFormat.initialize',
 
2278
                                                    path)
 
2279
        assert response[0] in ('ok', ), 'unexpected response code %s' % (response,)
2639
2280
        return remote.RemoteBzrDir(transport)
2640
2281
 
2641
2282
    def _open(self, transport):
2666
2307
    e.g. BzrDirMeta1 with weave repository.  Also, it's more user-oriented.
2667
2308
    """
2668
2309
 
2669
 
    def __init__(self):
2670
 
        """Create a BzrDirFormatRegistry."""
2671
 
        self._aliases = set()
2672
 
        super(BzrDirFormatRegistry, self).__init__()
2673
 
 
2674
 
    def aliases(self):
2675
 
        """Return a set of the format names which are aliases."""
2676
 
        return frozenset(self._aliases)
2677
 
 
2678
2310
    def register_metadir(self, key,
2679
2311
             repository_format, help, native=True, deprecated=False,
2680
2312
             branch_format=None,
2681
2313
             tree_format=None,
2682
2314
             hidden=False,
2683
 
             experimental=False,
2684
 
             alias=False):
 
2315
             experimental=False):
2685
2316
        """Register a metadir subformat.
2686
2317
 
2687
2318
        These all use a BzrDirMetaFormat1 bzrdir, but can be parameterized
2720
2351
                bd.repository_format = _load(repository_format)
2721
2352
            return bd
2722
2353
        self.register(key, helper, help, native, deprecated, hidden,
2723
 
            experimental, alias)
 
2354
            experimental)
2724
2355
 
2725
2356
    def register(self, key, factory, help, native=True, deprecated=False,
2726
 
                 hidden=False, experimental=False, alias=False):
 
2357
                 hidden=False, experimental=False):
2727
2358
        """Register a BzrDirFormat factory.
2728
2359
        
2729
2360
        The factory must be a callable that takes one parameter: the key.
2732
2363
        This function mainly exists to prevent the info object from being
2733
2364
        supplied directly.
2734
2365
        """
2735
 
        registry.Registry.register(self, key, factory, help,
 
2366
        registry.Registry.register(self, key, factory, help, 
2736
2367
            BzrDirFormatInfo(native, deprecated, hidden, experimental))
2737
 
        if alias:
2738
 
            self._aliases.add(key)
2739
2368
 
2740
2369
    def register_lazy(self, key, module_name, member_name, help, native=True,
2741
 
        deprecated=False, hidden=False, experimental=False, alias=False):
2742
 
        registry.Registry.register_lazy(self, key, module_name, member_name,
 
2370
                      deprecated=False, hidden=False, experimental=False):
 
2371
        registry.Registry.register_lazy(self, key, module_name, member_name, 
2743
2372
            help, BzrDirFormatInfo(native, deprecated, hidden, experimental))
2744
 
        if alias:
2745
 
            self._aliases.add(key)
2746
2373
 
2747
2374
    def set_default(self, key):
2748
2375
        """Set the 'default' key to be a clone of the supplied key.
2749
2376
        
2750
2377
        This method must be called once and only once.
2751
2378
        """
2752
 
        registry.Registry.register(self, 'default', self.get(key),
 
2379
        registry.Registry.register(self, 'default', self.get(key), 
2753
2380
            self.get_help(key), info=self.get_info(key))
2754
 
        self._aliases.add('default')
2755
2381
 
2756
2382
    def set_default_repository(self, key):
2757
2383
        """Set the FormatRegistry default and Repository default.
2763
2389
            self.remove('default')
2764
2390
        self.set_default(key)
2765
2391
        format = self.get('default')()
 
2392
        assert isinstance(format, BzrDirMetaFormat1)
2766
2393
 
2767
2394
    def make_bzrdir(self, key):
2768
2395
        return self.get(key)()
2820
2447
        return output
2821
2448
 
2822
2449
 
2823
 
class RepositoryAcquisitionPolicy(object):
2824
 
    """Abstract base class for repository acquisition policies.
2825
 
 
2826
 
    A repository acquisition policy decides how a BzrDir acquires a repository
2827
 
    for a branch that is being created.  The most basic policy decision is
2828
 
    whether to create a new repository or use an existing one.
2829
 
    """
2830
 
    def __init__(self, stack_on, stack_on_pwd, require_stacking):
2831
 
        """Constructor.
2832
 
 
2833
 
        :param stack_on: A location to stack on
2834
 
        :param stack_on_pwd: If stack_on is relative, the location it is
2835
 
            relative to.
2836
 
        :param require_stacking: If True, it is a failure to not stack.
2837
 
        """
2838
 
        self._stack_on = stack_on
2839
 
        self._stack_on_pwd = stack_on_pwd
2840
 
        self._require_stacking = require_stacking
2841
 
 
2842
 
    def configure_branch(self, branch):
2843
 
        """Apply any configuration data from this policy to the branch.
2844
 
 
2845
 
        Default implementation sets repository stacking.
2846
 
        """
2847
 
        if self._stack_on is None:
2848
 
            return
2849
 
        if self._stack_on_pwd is None:
2850
 
            stack_on = self._stack_on
2851
 
        else:
2852
 
            try:
2853
 
                stack_on = urlutils.rebase_url(self._stack_on,
2854
 
                    self._stack_on_pwd,
2855
 
                    branch.bzrdir.root_transport.base)
2856
 
            except errors.InvalidRebaseURLs:
2857
 
                stack_on = self._get_full_stack_on()
2858
 
        try:
2859
 
            branch.set_stacked_on_url(stack_on)
2860
 
        except errors.UnstackableBranchFormat:
2861
 
            if self._require_stacking:
2862
 
                raise
2863
 
 
2864
 
    def _get_full_stack_on(self):
2865
 
        """Get a fully-qualified URL for the stack_on location."""
2866
 
        if self._stack_on is None:
2867
 
            return None
2868
 
        if self._stack_on_pwd is None:
2869
 
            return self._stack_on
2870
 
        else:
2871
 
            return urlutils.join(self._stack_on_pwd, self._stack_on)
2872
 
 
2873
 
    def _add_fallback(self, repository):
2874
 
        """Add a fallback to the supplied repository, if stacking is set."""
2875
 
        stack_on = self._get_full_stack_on()
2876
 
        if stack_on is None:
2877
 
            return
2878
 
        stacked_dir = BzrDir.open(stack_on)
2879
 
        try:
2880
 
            stacked_repo = stacked_dir.open_branch().repository
2881
 
        except errors.NotBranchError:
2882
 
            stacked_repo = stacked_dir.open_repository()
2883
 
        try:
2884
 
            repository.add_fallback_repository(stacked_repo)
2885
 
        except errors.UnstackableRepositoryFormat:
2886
 
            if self._require_stacking:
2887
 
                raise
2888
 
 
2889
 
    def acquire_repository(self, make_working_trees=None, shared=False):
2890
 
        """Acquire a repository for this bzrdir.
2891
 
 
2892
 
        Implementations may create a new repository or use a pre-exising
2893
 
        repository.
2894
 
        :param make_working_trees: If creating a repository, set
2895
 
            make_working_trees to this value (if non-None)
2896
 
        :param shared: If creating a repository, make it shared if True
2897
 
        :return: A repository
2898
 
        """
2899
 
        raise NotImplemented(RepositoryAcquisitionPolicy.acquire_repository)
2900
 
 
2901
 
 
2902
 
class CreateRepository(RepositoryAcquisitionPolicy):
2903
 
    """A policy of creating a new repository"""
2904
 
 
2905
 
    def __init__(self, bzrdir, stack_on=None, stack_on_pwd=None,
2906
 
                 require_stacking=False):
2907
 
        """
2908
 
        Constructor.
2909
 
        :param bzrdir: The bzrdir to create the repository on.
2910
 
        :param stack_on: A location to stack on
2911
 
        :param stack_on_pwd: If stack_on is relative, the location it is
2912
 
            relative to.
2913
 
        """
2914
 
        RepositoryAcquisitionPolicy.__init__(self, stack_on, stack_on_pwd,
2915
 
                                             require_stacking)
2916
 
        self._bzrdir = bzrdir
2917
 
 
2918
 
    def acquire_repository(self, make_working_trees=None, shared=False):
2919
 
        """Implementation of RepositoryAcquisitionPolicy.acquire_repository
2920
 
 
2921
 
        Creates the desired repository in the bzrdir we already have.
2922
 
        """
2923
 
        repository = self._bzrdir.create_repository(shared=shared)
2924
 
        self._add_fallback(repository)
2925
 
        if make_working_trees is not None:
2926
 
            repository.set_make_working_trees(make_working_trees)
2927
 
        return repository
2928
 
 
2929
 
 
2930
 
class UseExistingRepository(RepositoryAcquisitionPolicy):
2931
 
    """A policy of reusing an existing repository"""
2932
 
 
2933
 
    def __init__(self, repository, stack_on=None, stack_on_pwd=None,
2934
 
                 require_stacking=False):
2935
 
        """Constructor.
2936
 
 
2937
 
        :param repository: The repository to use.
2938
 
        :param stack_on: A location to stack on
2939
 
        :param stack_on_pwd: If stack_on is relative, the location it is
2940
 
            relative to.
2941
 
        """
2942
 
        RepositoryAcquisitionPolicy.__init__(self, stack_on, stack_on_pwd,
2943
 
                                             require_stacking)
2944
 
        self._repository = repository
2945
 
 
2946
 
    def acquire_repository(self, make_working_trees=None, shared=False):
2947
 
        """Implementation of RepositoryAcquisitionPolicy.acquire_repository
2948
 
 
2949
 
        Returns an existing repository to use
2950
 
        """
2951
 
        self._add_fallback(self._repository)
2952
 
        return self._repository
2953
 
 
2954
 
 
2955
2450
format_registry = BzrDirFormatRegistry()
2956
2451
format_registry.register('weave', BzrDirFormat6,
2957
2452
    'Pre-0.8 format.  Slower than knit and does not'
2985
2480
    branch_format='bzrlib.branch.BzrBranchFormat6',
2986
2481
    tree_format='bzrlib.workingtree.WorkingTreeFormat4',
2987
2482
    )
2988
 
format_registry.register_metadir('rich-root',
2989
 
    'bzrlib.repofmt.knitrepo.RepositoryFormatKnit4',
2990
 
    help='New in 1.0.  Better handling of tree roots.  Incompatible with'
2991
 
        ' bzr < 1.0',
2992
 
    branch_format='bzrlib.branch.BzrBranchFormat6',
2993
 
    tree_format='bzrlib.workingtree.WorkingTreeFormat4',
2994
 
    )
2995
2483
format_registry.register_metadir('dirstate-with-subtree',
2996
2484
    'bzrlib.repofmt.knitrepo.RepositoryFormatKnit3',
2997
2485
    help='New in 0.15: Fast local operations and improved scaling for '
2999
2487
        'bzr branches. Incompatible with bzr < 0.15.',
3000
2488
    branch_format='bzrlib.branch.BzrBranchFormat6',
3001
2489
    tree_format='bzrlib.workingtree.WorkingTreeFormat4',
3002
 
    experimental=True,
3003
2490
    hidden=True,
3004
2491
    )
3005
 
format_registry.register_metadir('pack-0.92',
 
2492
format_registry.register_metadir('knitpack-experimental',
3006
2493
    'bzrlib.repofmt.pack_repo.RepositoryFormatKnitPack1',
3007
2494
    help='New in 0.92: Pack-based format with data compatible with '
3008
2495
        'dirstate-tags format repositories. Interoperates with '
3009
2496
        'bzr repositories before 0.92 but cannot be read by bzr < 0.92. '
3010
 
        'Previously called knitpack-experimental.  '
3011
 
        'For more information, see '
3012
 
        'http://doc.bazaar-vcs.org/latest/developers/packrepo.html.',
 
2497
        'NOTE: This format is experimental. Before using it, please read '
 
2498
        'http://doc.bazaar-vcs.org/latest/developers/knitpack.html.',
3013
2499
    branch_format='bzrlib.branch.BzrBranchFormat6',
3014
2500
    tree_format='bzrlib.workingtree.WorkingTreeFormat4',
 
2501
    experimental=True,
3015
2502
    )
3016
 
format_registry.register_metadir('pack-0.92-subtree',
 
2503
format_registry.register_metadir('knitpack-subtree-experimental',
3017
2504
    'bzrlib.repofmt.pack_repo.RepositoryFormatKnitPack3',
3018
2505
    help='New in 0.92: Pack-based format with data compatible with '
3019
2506
        'dirstate-with-subtree format repositories. Interoperates with '
3020
2507
        'bzr repositories before 0.92 but cannot be read by bzr < 0.92. '
3021
 
        'Previously called knitpack-experimental.  '
3022
 
        'For more information, see '
3023
 
        'http://doc.bazaar-vcs.org/latest/developers/packrepo.html.',
3024
 
    branch_format='bzrlib.branch.BzrBranchFormat6',
3025
 
    tree_format='bzrlib.workingtree.WorkingTreeFormat4',
3026
 
    hidden=True,
3027
 
    experimental=True,
3028
 
    )
3029
 
format_registry.register_metadir('rich-root-pack',
3030
 
    'bzrlib.repofmt.pack_repo.RepositoryFormatKnitPack4',
3031
 
    help='New in 1.0: Pack-based format with data compatible with '
3032
 
        'rich-root format repositories. Incompatible with'
3033
 
        ' bzr < 1.0',
3034
 
    branch_format='bzrlib.branch.BzrBranchFormat6',
3035
 
    tree_format='bzrlib.workingtree.WorkingTreeFormat4',
3036
 
    )
3037
 
format_registry.register_metadir('1.6',
3038
 
    'bzrlib.repofmt.pack_repo.RepositoryFormatKnitPack5',
3039
 
    help='A branch and pack based repository that supports stacking. ',
3040
 
    branch_format='bzrlib.branch.BzrBranchFormat7',
3041
 
    tree_format='bzrlib.workingtree.WorkingTreeFormat4',
3042
 
    )
3043
 
format_registry.register_metadir('1.6.1-rich-root',
3044
 
    'bzrlib.repofmt.pack_repo.RepositoryFormatKnitPack5RichRoot',
3045
 
    help='A branch and pack based repository that supports stacking '
3046
 
         'and rich root data (needed for bzr-svn). ',
3047
 
    branch_format='bzrlib.branch.BzrBranchFormat7',
3048
 
    tree_format='bzrlib.workingtree.WorkingTreeFormat4',
3049
 
    )
3050
 
# The following two formats should always just be aliases.
3051
 
format_registry.register_metadir('development',
3052
 
    'bzrlib.repofmt.pack_repo.RepositoryFormatPackDevelopment1',
3053
 
    help='Current development format. Can convert data to and from pack-0.92 '
3054
 
        '(and anything compatible with pack-0.92) format repositories. '
3055
 
        'Repositories and branches in this format can only be read by bzr.dev. '
3056
 
        'Please read '
3057
 
        'http://doc.bazaar-vcs.org/latest/developers/development-repo.html '
3058
 
        'before use.',
3059
 
    branch_format='bzrlib.branch.BzrBranchFormat7',
3060
 
    tree_format='bzrlib.workingtree.WorkingTreeFormat4',
3061
 
    experimental=True,
3062
 
    alias=True,
3063
 
    )
3064
 
format_registry.register_metadir('development-subtree',
3065
 
    'bzrlib.repofmt.pack_repo.RepositoryFormatPackDevelopment1Subtree',
3066
 
    help='Current development format, subtree variant. Can convert data to and '
3067
 
        'from pack-0.92-subtree (and anything compatible with '
3068
 
        'pack-0.92-subtree) format repositories. Repositories and branches in '
3069
 
        'this format can only be read by bzr.dev. Please read '
3070
 
        'http://doc.bazaar-vcs.org/latest/developers/development-repo.html '
3071
 
        'before use.',
3072
 
    branch_format='bzrlib.branch.BzrBranchFormat7',
3073
 
    tree_format='bzrlib.workingtree.WorkingTreeFormat4',
3074
 
    experimental=True,
3075
 
    alias=True,
3076
 
    )
3077
 
# And the development formats which the will have aliased one of follow:
3078
 
format_registry.register_metadir('development1',
3079
 
    'bzrlib.repofmt.pack_repo.RepositoryFormatPackDevelopment1',
3080
 
    help='A branch and pack based repository that supports stacking. '
3081
 
        'Please read '
3082
 
        'http://doc.bazaar-vcs.org/latest/developers/development-repo.html '
3083
 
        'before use.',
3084
 
    branch_format='bzrlib.branch.BzrBranchFormat7',
3085
 
    tree_format='bzrlib.workingtree.WorkingTreeFormat4',
3086
 
    hidden=True,
3087
 
    experimental=True,
3088
 
    )
3089
 
format_registry.register_metadir('development1-subtree',
3090
 
    'bzrlib.repofmt.pack_repo.RepositoryFormatPackDevelopment1Subtree',
3091
 
    help='A branch and pack based repository that supports stacking. '
3092
 
        'Please read '
3093
 
        'http://doc.bazaar-vcs.org/latest/developers/development-repo.html '
3094
 
        'before use.',
3095
 
    branch_format='bzrlib.branch.BzrBranchFormat7',
3096
 
    tree_format='bzrlib.workingtree.WorkingTreeFormat4',
3097
 
    hidden=True,
3098
 
    experimental=True,
3099
 
    )
3100
 
# The current format that is made on 'bzr init'.
3101
 
format_registry.set_default('pack-0.92')
 
2508
        'NOTE: This format is experimental. Before using it, please read '
 
2509
        'http://doc.bazaar-vcs.org/latest/developers/knitpack.html.',
 
2510
    branch_format='bzrlib.branch.BzrBranchFormat6',
 
2511
    tree_format='bzrlib.workingtree.WorkingTreeFormat4',
 
2512
    hidden=True,
 
2513
    experimental=True,
 
2514
    )
 
2515
format_registry.set_default('dirstate-tags')