~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/controldir.py

(jameinel) Allow 'bzr serve' to interpret SIGHUP as a graceful shutdown.
 (bug #795025) (John A Meinel)

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2010 Canonical Ltd
 
1
# Copyright (C) 2010, 2011 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
28
28
 
29
29
from bzrlib import (
30
30
    errors,
31
 
    graph,
32
 
    registry,
33
31
    revision as _mod_revision,
34
 
    symbol_versioning,
 
32
    transport as _mod_transport,
 
33
    ui,
35
34
    urlutils,
36
35
    )
37
36
from bzrlib.push import (
38
37
    PushResult,
39
38
    )
40
 
from bzrlib.trace import (
41
 
    mutter,
42
 
    )
43
 
from bzrlib.transport import (
44
 
    get_transport,
45
 
    local,
46
 
    )
47
39
 
48
40
""")
49
41
 
 
42
from bzrlib import registry
 
43
 
50
44
 
51
45
class ControlComponent(object):
52
46
    """Abstract base class for control directory components.
139
133
        """
140
134
        raise NotImplementedError(self.needs_format_conversion)
141
135
 
 
136
    def create_repository(self, shared=False):
 
137
        """Create a new repository in this control directory.
 
138
 
 
139
        :param shared: If a shared repository should be created
 
140
        :return: The newly created repository
 
141
        """
 
142
        raise NotImplementedError(self.create_repository)
 
143
 
142
144
    def destroy_repository(self):
143
145
        """Destroy the repository in this ControlDir."""
144
146
        raise NotImplementedError(self.destroy_repository)
145
147
 
146
 
    def create_branch(self, name=None):
 
148
    def create_branch(self, name=None, repository=None,
 
149
                      append_revisions_only=None):
147
150
        """Create a branch in this ControlDir.
148
151
 
149
152
        :param name: Name of the colocated branch to create, None for
150
153
            the default branch.
 
154
        :param append_revisions_only: Whether this branch should only allow
 
155
            appending new revisions to its history.
151
156
 
152
157
        The controldirs format will control what branch format is created.
153
158
        For more control see BranchFormatXX.create(a_controldir).
191
196
        """
192
197
        raise NotImplementedError(self.destroy_workingtree_metadata)
193
198
 
 
199
    def find_branch_format(self, name=None):
 
200
        """Find the branch 'format' for this bzrdir.
 
201
 
 
202
        This might be a synthetic object for e.g. RemoteBranch and SVN.
 
203
        """
 
204
        raise NotImplementedError(self.find_branch_format)
 
205
 
194
206
    def get_branch_reference(self, name=None):
195
207
        """Return the referenced URL for the branch in this controldir.
196
208
 
205
217
            raise errors.NoColocatedBranchSupport(self)
206
218
        return None
207
219
 
208
 
    def get_branch_transport(self, branch_format, name=None):
209
 
        """Get the transport for use by branch format in this ControlDir.
210
 
 
211
 
        Note that bzr dirs that do not support format strings will raise
212
 
        IncompatibleFormat if the branch format they are given has
213
 
        a format string, and vice versa.
214
 
 
215
 
        If branch_format is None, the transport is returned with no
216
 
        checking. If it is not None, then the returned transport is
217
 
        guaranteed to point to an existing directory ready for use.
218
 
        """
219
 
        raise NotImplementedError(self.get_branch_transport)
220
 
 
221
 
    def get_repository_transport(self, repository_format):
222
 
        """Get the transport for use by repository format in this ControlDir.
223
 
 
224
 
        Note that bzr dirs that do not support format strings will raise
225
 
        IncompatibleFormat if the repository format they are given has
226
 
        a format string, and vice versa.
227
 
 
228
 
        If repository_format is None, the transport is returned with no
229
 
        checking. If it is not None, then the returned transport is
230
 
        guaranteed to point to an existing directory ready for use.
231
 
        """
232
 
        raise NotImplementedError(self.get_repository_transport)
233
 
 
234
 
    def get_workingtree_transport(self, tree_format):
235
 
        """Get the transport for use by workingtree format in this ControlDir.
236
 
 
237
 
        Note that bzr dirs that do not support format strings will raise
238
 
        IncompatibleFormat if the workingtree format they are given has a
239
 
        format string, and vice versa.
240
 
 
241
 
        If workingtree_format is None, the transport is returned with no
242
 
        checking. If it is not None, then the returned transport is
243
 
        guaranteed to point to an existing directory ready for use.
244
 
        """
245
 
        raise NotImplementedError(self.get_workingtree_transport)
246
 
 
247
220
    def open_branch(self, name=None, unsupported=False,
248
221
                    ignore_fallbacks=False):
249
222
        """Open the branch object at this ControlDir if one is present.
263
236
        get at a repository.
264
237
 
265
238
        :param _unsupported: a private parameter, not part of the api.
 
239
 
266
240
        TODO: static convenience version of this?
267
241
        """
268
242
        raise NotImplementedError(self.open_repository)
302
276
        except errors.NotBranchError:
303
277
            return False
304
278
 
 
279
    def _get_selected_branch(self):
 
280
        """Return the name of the branch selected by the user.
 
281
 
 
282
        :return: Name of the branch selected by the user, or None.
 
283
        """
 
284
        branch = self.root_transport.get_segment_parameters().get("branch")
 
285
        if branch is not None:
 
286
            branch = urlutils.unescape(branch)
 
287
        return branch
 
288
 
305
289
    def has_workingtree(self):
306
290
        """Tell if this controldir contains a working tree.
307
291
 
351
335
        whether one existed before or not; and a local branch is always
352
336
        created.
353
337
 
354
 
        if revision_id is not None, then the clone operation may tune
355
 
            itself to download less data.
 
338
        :param revision_id: if revision_id is not None, then the clone
 
339
            operation may tune itself to download less data.
356
340
        :param accelerator_tree: A tree which can be used for retrieving file
357
341
            contents more quickly than the revision tree, i.e. a workingtree.
358
342
            The revision tree will be used for cases where accelerator_tree's
364
348
        :param create_tree_if_local: If true, a working-tree will be created
365
349
            when working locally.
366
350
        """
367
 
        target_transport = get_transport(url, possible_transports)
368
 
        target_transport.ensure_base()
369
 
        cloning_format = self.cloning_metadir(stacked)
370
 
        # Create/update the result branch
371
 
        result = cloning_format.initialize_on_transport(target_transport)
372
 
        # if a stacked branch wasn't requested, we don't create one
373
 
        # even if the origin was stacked
374
 
        stacked_branch_url = None
375
 
        if source_branch is not None:
376
 
            if stacked:
377
 
                stacked_branch_url = self.root_transport.base
378
 
            source_repository = source_branch.repository
379
 
        else:
380
 
            try:
381
 
                source_branch = self.open_branch()
382
 
                source_repository = source_branch.repository
383
 
                if stacked:
384
 
                    stacked_branch_url = self.root_transport.base
385
 
            except errors.NotBranchError:
386
 
                source_branch = None
387
 
                try:
388
 
                    source_repository = self.open_repository()
389
 
                except errors.NoRepositoryPresent:
390
 
                    source_repository = None
391
 
        repository_policy = result.determine_repository_policy(
392
 
            force_new_repo, stacked_branch_url, require_stacking=stacked)
393
 
        result_repo, is_new_repo = repository_policy.acquire_repository()
394
 
        is_stacked = stacked or (len(result_repo._fallback_repositories) != 0)
395
 
        if is_new_repo and revision_id is not None and not is_stacked:
396
 
            fetch_spec = graph.PendingAncestryResult(
397
 
                [revision_id], source_repository)
398
 
        else:
399
 
            fetch_spec = None
400
 
        if source_repository is not None:
401
 
            # Fetch while stacked to prevent unstacked fetch from
402
 
            # Branch.sprout.
403
 
            if fetch_spec is None:
404
 
                result_repo.fetch(source_repository, revision_id=revision_id)
405
 
            else:
406
 
                result_repo.fetch(source_repository, fetch_spec=fetch_spec)
407
 
 
408
 
        if source_branch is None:
409
 
            # this is for sprouting a controldir without a branch; is that
410
 
            # actually useful?
411
 
            # Not especially, but it's part of the contract.
412
 
            result_branch = result.create_branch()
413
 
        else:
414
 
            result_branch = source_branch.sprout(result,
415
 
                revision_id=revision_id, repository_policy=repository_policy)
416
 
        mutter("created new branch %r" % (result_branch,))
417
 
 
418
 
        # Create/update the result working tree
419
 
        if (create_tree_if_local and
420
 
            isinstance(target_transport, local.LocalTransport) and
421
 
            (result_repo is None or result_repo.make_working_trees())):
422
 
            wt = result.create_workingtree(accelerator_tree=accelerator_tree,
423
 
                hardlink=hardlink)
424
 
            wt.lock_write()
425
 
            try:
426
 
                if wt.path2id('') is None:
427
 
                    try:
428
 
                        wt.set_root_id(self.open_workingtree.get_root_id())
429
 
                    except errors.NoWorkingTree:
430
 
                        pass
431
 
            finally:
432
 
                wt.unlock()
433
 
        else:
434
 
            wt = None
435
 
        if recurse == 'down':
436
 
            if wt is not None:
437
 
                basis = wt.basis_tree()
438
 
                basis.lock_read()
439
 
                subtrees = basis.iter_references()
440
 
            elif result_branch is not None:
441
 
                basis = result_branch.basis_tree()
442
 
                basis.lock_read()
443
 
                subtrees = basis.iter_references()
444
 
            elif source_branch is not None:
445
 
                basis = source_branch.basis_tree()
446
 
                basis.lock_read()
447
 
                subtrees = basis.iter_references()
448
 
            else:
449
 
                subtrees = []
450
 
                basis = None
451
 
            try:
452
 
                for path, file_id in subtrees:
453
 
                    target = urlutils.join(url, urlutils.escape(path))
454
 
                    sublocation = source_branch.reference_parent(file_id, path)
455
 
                    sublocation.bzrdir.sprout(target,
456
 
                        basis.get_reference_revision(file_id, path),
457
 
                        force_new_repo=force_new_repo, recurse=recurse,
458
 
                        stacked=stacked)
459
 
            finally:
460
 
                if basis is not None:
461
 
                    basis.unlock()
462
 
        return result
 
351
        raise NotImplementedError(self.sprout)
463
352
 
464
353
    def push_branch(self, source, revision_id=None, overwrite=False, 
465
354
        remember=False, create_prefix=False):
481
370
        if br_to is None:
482
371
            # We have a repository but no branch, copy the revisions, and then
483
372
            # create a branch.
 
373
            if revision_id is None:
 
374
                # No revision supplied by the user, default to the branch
 
375
                # revision
 
376
                revision_id = source.last_revision()
484
377
            repository_to.fetch(source.repository, revision_id=revision_id)
485
378
            br_to = source.clone(self, revision_id=revision_id)
486
379
            if source.get_push_location() is None or remember:
564
457
        :param preserve_stacking: When cloning a stacked branch, stack the
565
458
            new branch on top of the other branch's stacked-on branch.
566
459
        """
567
 
        return self.clone_on_transport(get_transport(url),
 
460
        return self.clone_on_transport(_mod_transport.get_transport(url),
568
461
                                       revision_id=revision_id,
569
462
                                       force_new_repo=force_new_repo,
570
463
                                       preserve_stacking=preserve_stacking)
571
464
 
572
465
    def clone_on_transport(self, transport, revision_id=None,
573
466
        force_new_repo=False, preserve_stacking=False, stacked_on=None,
574
 
        create_prefix=False, use_existing_dir=True):
 
467
        create_prefix=False, use_existing_dir=True, no_tree=False):
575
468
        """Clone this bzrdir and its contents to transport verbatim.
576
469
 
577
470
        :param transport: The transport for the location to produce the clone
586
479
        :param create_prefix: Create any missing directories leading up to
587
480
            to_transport.
588
481
        :param use_existing_dir: Use an existing directory if one exists.
 
482
        :param no_tree: If set to true prevents creation of a working tree.
589
483
        """
590
484
        raise NotImplementedError(self.clone_on_transport)
591
485
 
592
486
 
 
487
class ControlComponentFormat(object):
 
488
    """A component that can live inside of a .bzr meta directory."""
 
489
 
 
490
    upgrade_recommended = False
 
491
 
 
492
    def get_format_string(self):
 
493
        """Return the format of this format, if usable in meta directories."""
 
494
        raise NotImplementedError(self.get_format_string)
 
495
 
 
496
    def get_format_description(self):
 
497
        """Return the short description for this format."""
 
498
        raise NotImplementedError(self.get_format_description)
 
499
 
 
500
    def is_supported(self):
 
501
        """Is this format supported?
 
502
 
 
503
        Supported formats must be initializable and openable.
 
504
        Unsupported formats may not support initialization or committing or
 
505
        some other features depending on the reason for not being supported.
 
506
        """
 
507
        return True
 
508
 
 
509
    def check_support_status(self, allow_unsupported, recommend_upgrade=True,
 
510
        basedir=None):
 
511
        """Give an error or warning on old formats.
 
512
 
 
513
        :param allow_unsupported: If true, allow opening
 
514
            formats that are strongly deprecated, and which may
 
515
            have limited functionality.
 
516
 
 
517
        :param recommend_upgrade: If true (default), warn
 
518
            the user through the ui object that they may wish
 
519
            to upgrade the object.
 
520
        """
 
521
        if not allow_unsupported and not self.is_supported():
 
522
            # see open_downlevel to open legacy branches.
 
523
            raise errors.UnsupportedFormatError(format=self)
 
524
        if recommend_upgrade and self.upgrade_recommended:
 
525
            ui.ui_factory.recommend_upgrade(
 
526
                self.get_format_description(), basedir)
 
527
 
 
528
 
 
529
class ControlComponentFormatRegistry(registry.FormatRegistry):
 
530
    """A registry for control components (branch, workingtree, repository)."""
 
531
 
 
532
    def __init__(self, other_registry=None):
 
533
        super(ControlComponentFormatRegistry, self).__init__(other_registry)
 
534
        self._extra_formats = []
 
535
 
 
536
    def register(self, format):
 
537
        """Register a new format."""
 
538
        super(ControlComponentFormatRegistry, self).register(
 
539
            format.get_format_string(), format)
 
540
 
 
541
    def remove(self, format):
 
542
        """Remove a registered format."""
 
543
        super(ControlComponentFormatRegistry, self).remove(
 
544
            format.get_format_string())
 
545
 
 
546
    def register_extra(self, format):
 
547
        """Register a format that can not be used in a metadir.
 
548
 
 
549
        This is mainly useful to allow custom repository formats, such as older
 
550
        Bazaar formats and foreign formats, to be tested.
 
551
        """
 
552
        self._extra_formats.append(registry._ObjectGetter(format))
 
553
 
 
554
    def remove_extra(self, format):
 
555
        """Remove an extra format.
 
556
        """
 
557
        self._extra_formats.remove(registry._ObjectGetter(format))
 
558
 
 
559
    def register_extra_lazy(self, module_name, member_name):
 
560
        """Register a format lazily.
 
561
        """
 
562
        self._extra_formats.append(
 
563
            registry._LazyObjectGetter(module_name, member_name))
 
564
 
 
565
    def _get_extra(self):
 
566
        """Return all "extra" formats, not usable in meta directories."""
 
567
        result = []
 
568
        for getter in self._extra_formats:
 
569
            f = getter.get_obj()
 
570
            if callable(f):
 
571
                f = f()
 
572
            result.append(f)
 
573
        return result
 
574
 
 
575
    def _get_all(self):
 
576
        """Return all formats, even those not usable in metadirs.
 
577
        """
 
578
        result = []
 
579
        for name in self.keys():
 
580
            fmt = self.get(name)
 
581
            if callable(fmt):
 
582
                fmt = fmt()
 
583
            result.append(fmt)
 
584
        return result + self._get_extra()
 
585
 
 
586
    def _get_all_modules(self):
 
587
        """Return a set of the modules providing objects."""
 
588
        modules = set()
 
589
        for name in self.keys():
 
590
            modules.add(self._get_module(name))
 
591
        for getter in self._extra_formats:
 
592
            modules.add(getter.get_module())
 
593
        return modules
 
594
 
 
595
 
 
596
class Converter(object):
 
597
    """Converts a disk format object from one format to another."""
 
598
 
 
599
    def convert(self, to_convert, pb):
 
600
        """Perform the conversion of to_convert, giving feedback via pb.
 
601
 
 
602
        :param to_convert: The disk object to convert.
 
603
        :param pb: a progress bar to use for progress information.
 
604
        """
 
605
 
 
606
    def step(self, message):
 
607
        """Update the pb by a step."""
 
608
        self.count +=1
 
609
        self.pb.update(message, self.count, self.total)
 
610
 
 
611
 
593
612
class ControlDirFormat(object):
594
613
    """An encapsulation of the initialization and open routines for a format.
595
614
 
614
633
    _default_format = None
615
634
    """The default format used for new control directories."""
616
635
 
617
 
    _formats = []
618
 
    """The registered control formats - .bzr, ....
619
 
 
620
 
    This is a list of ControlDirFormat objects.
621
 
    """
622
 
 
623
636
    _server_probers = []
624
637
    """The registered server format probers, e.g. RemoteBzrProber.
625
638
 
637
650
    """
638
651
 
639
652
    supports_workingtrees = True
 
653
    """Whether working trees can exist in control directories of this format.
 
654
    """
 
655
 
 
656
    fixed_components = False
 
657
    """Whether components can not change format independent of the control dir.
 
658
    """
 
659
 
 
660
    upgrade_recommended = False
 
661
    """Whether an upgrade from this format is recommended."""
640
662
 
641
663
    def get_format_description(self):
642
664
        """Return the short description for this format."""
659
681
    def is_supported(self):
660
682
        """Is this format supported?
661
683
 
662
 
        Supported formats must be initializable and openable.
 
684
        Supported formats must be openable.
663
685
        Unsupported formats may not support initialization or committing or
664
686
        some other features depending on the reason for not being supported.
665
687
        """
666
688
        return True
667
689
 
 
690
    def is_initializable(self):
 
691
        """Whether new control directories of this format can be initialized.
 
692
        """
 
693
        return self.is_supported()
 
694
 
 
695
    def check_support_status(self, allow_unsupported, recommend_upgrade=True,
 
696
        basedir=None):
 
697
        """Give an error or warning on old formats.
 
698
 
 
699
        :param allow_unsupported: If true, allow opening
 
700
            formats that are strongly deprecated, and which may
 
701
            have limited functionality.
 
702
 
 
703
        :param recommend_upgrade: If true (default), warn
 
704
            the user through the ui object that they may wish
 
705
            to upgrade the object.
 
706
        """
 
707
        if not allow_unsupported and not self.is_supported():
 
708
            # see open_downlevel to open legacy branches.
 
709
            raise errors.UnsupportedFormatError(format=self)
 
710
        if recommend_upgrade and self.upgrade_recommended:
 
711
            ui.ui_factory.recommend_upgrade(
 
712
                self.get_format_description(), basedir)
 
713
 
668
714
    def same_model(self, target_format):
669
715
        return (self.repository_format.rich_root_data ==
670
716
            target_format.rich_root_data)
674
720
        """Register a format that does not use '.bzr' for its control dir.
675
721
 
676
722
        """
677
 
        klass._formats.append(format)
 
723
        raise errors.BzrError("ControlDirFormat.register_format() has been "
 
724
            "removed in Bazaar 2.4. Please upgrade your plugins.")
678
725
 
679
726
    @classmethod
680
727
    def register_prober(klass, prober):
706
753
        return self.get_format_description().rstrip()
707
754
 
708
755
    @classmethod
709
 
    def unregister_format(klass, format):
710
 
        klass._formats.remove(format)
711
 
 
712
 
    @classmethod
713
756
    def known_formats(klass):
714
757
        """Return all the known formats.
715
758
        """
716
 
        return set(klass._formats)
 
759
        result = set()
 
760
        for prober_kls in klass._probers + klass._server_probers:
 
761
            result.update(prober_kls.known_formats())
 
762
        return result
717
763
 
718
764
    @classmethod
719
765
    def find_format(klass, transport, _server_formats=True):
741
787
        Subclasses should typically override initialize_on_transport
742
788
        instead of this method.
743
789
        """
744
 
        return self.initialize_on_transport(get_transport(url,
745
 
                                                          possible_transports))
 
790
        return self.initialize_on_transport(
 
791
            _mod_transport.get_transport(url, possible_transports))
 
792
 
746
793
    def initialize_on_transport(self, transport):
747
794
        """Initialize a new controldir in the base directory of a Transport."""
748
795
        raise NotImplementedError(self.initialize_on_transport)
808
855
 
809
856
 
810
857
class Prober(object):
811
 
    """Abstract class that can be used to detect a particular kind of 
 
858
    """Abstract class that can be used to detect a particular kind of
812
859
    control directory.
813
860
 
814
 
    At the moment this just contains a single method to probe a particular 
815
 
    transport, but it may be extended in the future to e.g. avoid 
 
861
    At the moment this just contains a single method to probe a particular
 
862
    transport, but it may be extended in the future to e.g. avoid
816
863
    multiple levels of probing for Subversion repositories.
 
864
 
 
865
    See BzrProber and RemoteBzrProber in bzrlib.bzrdir for the
 
866
    probers that detect .bzr/ directories and Bazaar smart servers,
 
867
    respectively.
 
868
 
 
869
    Probers should be registered using the register_server_prober or
 
870
    register_prober methods on ControlDirFormat.
817
871
    """
818
872
 
819
873
    def probe_transport(self, transport):
826
880
        """
827
881
        raise NotImplementedError(self.probe_transport)
828
882
 
 
883
    @classmethod
 
884
    def known_formats(cls):
 
885
        """Return the control dir formats known by this prober.
 
886
 
 
887
        Multiple probers can return the same formats, so this should
 
888
        return a set.
 
889
 
 
890
        :return: A set of known formats.
 
891
        """
 
892
        raise NotImplementedError(cls.known_formats)
 
893
 
829
894
 
830
895
class ControlDirFormatInfo(object):
831
896
 
840
905
    """Registry of user-selectable ControlDir subformats.
841
906
 
842
907
    Differs from ControlDirFormat._formats in that it provides sub-formats,
843
 
    e.g. ControlDirMeta1 with weave repository.  Also, it's more user-oriented.
 
908
    e.g. BzrDirMeta1 with weave repository.  Also, it's more user-oriented.
844
909
    """
845
910
 
846
911
    def __init__(self):