/brz/remove-bazaar

To get this branch, use:
bzr branch http://gegoxaren.bato24.eu/bzr/brz/remove-bazaar

« back to all changes in this revision

Viewing changes to breezy/git/dir.py

  • Committer: Jelmer Vernooij
  • Date: 2018-11-11 04:08:32 UTC
  • mto: (7143.16.20 even-more-cleanups)
  • mto: This revision was merged to the branch mainline in revision 7175.
  • Revision ID: jelmer@jelmer.uk-20181111040832-nsljjynzzwmznf3h
Run autopep8.

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
# Copyright (C) 2007 Canonical Ltd
 
2
# Copyright (C) 2010-2018 Jelmer Vernooij
 
3
#
 
4
# This program is free software; you can redistribute it and/or modify
 
5
# it under the terms of the GNU General Public License as published by
 
6
# the Free Software Foundation; either version 2 of the License, or
 
7
# (at your option) any later version.
 
8
#
 
9
# This program is distributed in the hope that it will be useful,
 
10
# but WITHOUT ANY WARRANTY; without even the implied warranty of
 
11
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 
12
# GNU General Public License for more details.
 
13
#
 
14
# You should have received a copy of the GNU General Public License
 
15
# along with this program; if not, write to the Free Software
 
16
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
 
17
 
 
18
"""An adapter between a Git control dir and a Bazaar ControlDir."""
 
19
 
 
20
from __future__ import absolute_import
 
21
 
 
22
from .. import (
 
23
    branch as _mod_branch,
 
24
    errors as bzr_errors,
 
25
    trace,
 
26
    osutils,
 
27
    urlutils,
 
28
    )
 
29
from ..sixish import (
 
30
    PY3,
 
31
    viewitems,
 
32
    )
 
33
from ..transport import (
 
34
    do_catching_redirections,
 
35
    get_transport_from_path,
 
36
    )
 
37
 
 
38
from ..controldir import (
 
39
    BranchReferenceLoop,
 
40
    ControlDir,
 
41
    ControlDirFormat,
 
42
    format_registry,
 
43
    RepositoryAcquisitionPolicy,
 
44
    )
 
45
 
 
46
from .push import (
 
47
    GitPushResult,
 
48
    )
 
49
from .transportgit import (
 
50
    OBJECTDIR,
 
51
    TransportObjectStore,
 
52
    )
 
53
 
 
54
 
 
55
class GitDirConfig(object):
 
56
 
 
57
    def get_default_stack_on(self):
 
58
        return None
 
59
 
 
60
    def set_default_stack_on(self, value):
 
61
        raise bzr_errors.BzrError("Cannot set configuration")
 
62
 
 
63
 
 
64
class GitControlDirFormat(ControlDirFormat):
 
65
 
 
66
    colocated_branches = True
 
67
    fixed_components = True
 
68
 
 
69
    def __eq__(self, other):
 
70
        return type(self) == type(other)
 
71
 
 
72
    def is_supported(self):
 
73
        return True
 
74
 
 
75
    def network_name(self):
 
76
        return b"git"
 
77
 
 
78
 
 
79
class UseExistingRepository(RepositoryAcquisitionPolicy):
 
80
    """A policy of reusing an existing repository"""
 
81
 
 
82
    def __init__(self, repository, stack_on=None, stack_on_pwd=None,
 
83
                 require_stacking=False):
 
84
        """Constructor.
 
85
 
 
86
        :param repository: The repository to use.
 
87
        :param stack_on: A location to stack on
 
88
        :param stack_on_pwd: If stack_on is relative, the location it is
 
89
            relative to.
 
90
        """
 
91
        super(UseExistingRepository, self).__init__(
 
92
            stack_on, stack_on_pwd, require_stacking)
 
93
        self._repository = repository
 
94
 
 
95
    def acquire_repository(self, make_working_trees=None, shared=False,
 
96
                           possible_transports=None):
 
97
        """Implementation of RepositoryAcquisitionPolicy.acquire_repository
 
98
 
 
99
        Returns an existing repository to use.
 
100
        """
 
101
        return self._repository, False
 
102
 
 
103
 
 
104
class GitDir(ControlDir):
 
105
    """An adapter to the '.git' dir used by git."""
 
106
 
 
107
    def is_supported(self):
 
108
        return True
 
109
 
 
110
    def can_convert_format(self):
 
111
        return False
 
112
 
 
113
    def break_lock(self):
 
114
        # There are no global locks, so nothing to break.
 
115
        raise NotImplementedError(self.break_lock)
 
116
 
 
117
    def cloning_metadir(self, stacked=False):
 
118
        return format_registry.make_controldir("git")
 
119
 
 
120
    def checkout_metadir(self, stacked=False):
 
121
        return format_registry.make_controldir("git")
 
122
 
 
123
    def _get_selected_ref(self, branch, ref=None):
 
124
        if ref is not None and branch is not None:
 
125
            raise bzr_errors.BzrError("can't specify both ref and branch")
 
126
        if ref is not None:
 
127
            return ref
 
128
        if branch is not None:
 
129
            from .refs import branch_name_to_ref
 
130
            return branch_name_to_ref(branch)
 
131
        segment_parameters = getattr(
 
132
            self.user_transport, "get_segment_parameters", lambda: {})()
 
133
        ref = segment_parameters.get("ref")
 
134
        if ref is not None:
 
135
            return urlutils.unquote_to_bytes(ref)
 
136
        if branch is None and getattr(self, "_get_selected_branch", False):
 
137
            branch = self._get_selected_branch()
 
138
            if branch is not None:
 
139
                from .refs import branch_name_to_ref
 
140
                return branch_name_to_ref(branch)
 
141
        return b"HEAD"
 
142
 
 
143
    def get_config(self):
 
144
        return GitDirConfig()
 
145
 
 
146
    def _available_backup_name(self, base):
 
147
        return osutils.available_backup_name(base, self.root_transport.has)
 
148
 
 
149
    def sprout(self, url, revision_id=None, force_new_repo=False,
 
150
               recurse='down', possible_transports=None,
 
151
               accelerator_tree=None, hardlink=False, stacked=False,
 
152
               source_branch=None, create_tree_if_local=True):
 
153
        from ..repository import InterRepository
 
154
        from ..transport.local import LocalTransport
 
155
        from ..transport import get_transport
 
156
        target_transport = get_transport(url, possible_transports)
 
157
        target_transport.ensure_base()
 
158
        cloning_format = self.cloning_metadir()
 
159
        # Create/update the result branch
 
160
        try:
 
161
            result = ControlDir.open_from_transport(target_transport)
 
162
        except bzr_errors.NotBranchError:
 
163
            result = cloning_format.initialize_on_transport(target_transport)
 
164
        source_branch = self.open_branch()
 
165
        source_repository = self.find_repository()
 
166
        try:
 
167
            result_repo = result.find_repository()
 
168
        except bzr_errors.NoRepositoryPresent:
 
169
            result_repo = result.create_repository()
 
170
            target_is_empty = True
 
171
        else:
 
172
            target_is_empty = None  # Unknown
 
173
        if stacked:
 
174
            raise _mod_branch.UnstackableBranchFormat(
 
175
                self._format, self.user_url)
 
176
        interrepo = InterRepository.get(source_repository, result_repo)
 
177
 
 
178
        if revision_id is not None:
 
179
            determine_wants = interrepo.get_determine_wants_revids(
 
180
                [revision_id], include_tags=True)
 
181
        else:
 
182
            determine_wants = interrepo.determine_wants_all
 
183
        interrepo.fetch_objects(determine_wants=determine_wants,
 
184
                                mapping=source_branch.mapping)
 
185
        result_branch = source_branch.sprout(result,
 
186
                                             revision_id=revision_id, repository=result_repo)
 
187
        if (create_tree_if_local and
 
188
            isinstance(target_transport, LocalTransport) and
 
189
                (result_repo is None or result_repo.make_working_trees())):
 
190
            wt = result.create_workingtree(accelerator_tree=accelerator_tree,
 
191
                                           hardlink=hardlink, from_branch=result_branch)
 
192
        return result
 
193
 
 
194
    def clone_on_transport(self, transport, revision_id=None,
 
195
                           force_new_repo=False, preserve_stacking=False, stacked_on=None,
 
196
                           create_prefix=False, use_existing_dir=True, no_tree=False):
 
197
        """See ControlDir.clone_on_transport."""
 
198
        from ..repository import InterRepository
 
199
        from .mapping import default_mapping
 
200
        if stacked_on is not None:
 
201
            raise _mod_branch.UnstackableBranchFormat(
 
202
                self._format, self.user_url)
 
203
        if no_tree:
 
204
            format = BareLocalGitControlDirFormat()
 
205
        else:
 
206
            format = LocalGitControlDirFormat()
 
207
        (target_repo, target_controldir, stacking,
 
208
         repo_policy) = format.initialize_on_transport_ex(
 
209
            transport, use_existing_dir=use_existing_dir,
 
210
            create_prefix=create_prefix,
 
211
            force_new_repo=force_new_repo)
 
212
        target_repo = target_controldir.find_repository()
 
213
        target_git_repo = target_repo._git
 
214
        source_repo = self.find_repository()
 
215
        source_git_repo = source_repo._git
 
216
        interrepo = InterRepository.get(source_repo, target_repo)
 
217
        if revision_id is not None:
 
218
            determine_wants = interrepo.get_determine_wants_revids(
 
219
                [revision_id], include_tags=True)
 
220
        else:
 
221
            determine_wants = interrepo.determine_wants_all
 
222
        (pack_hint, _, refs) = interrepo.fetch_objects(determine_wants,
 
223
                                                       mapping=default_mapping)
 
224
        for name, val in viewitems(refs):
 
225
            target_git_repo.refs[name] = val
 
226
        result_dir = self.__class__(transport, target_git_repo, format)
 
227
        if revision_id is not None:
 
228
            result_dir.open_branch().set_last_revision(revision_id)
 
229
        try:
 
230
            # Cheaper to check if the target is not local, than to try making
 
231
            # the tree and fail.
 
232
            result_dir.root_transport.local_abspath('.')
 
233
            if result_dir.open_repository().make_working_trees():
 
234
                self.open_workingtree().clone(result_dir, revision_id=revision_id)
 
235
        except (bzr_errors.NoWorkingTree, bzr_errors.NotLocalUrl):
 
236
            pass
 
237
 
 
238
        return result_dir
 
239
 
 
240
    def find_repository(self):
 
241
        """Find the repository that should be used.
 
242
 
 
243
        This does not require a branch as we use it to find the repo for
 
244
        new branches as well as to hook existing branches up to their
 
245
        repository.
 
246
        """
 
247
        return self._gitrepository_class(self._find_commondir())
 
248
 
 
249
    def get_refs_container(self):
 
250
        """Retrieve the refs container.
 
251
        """
 
252
        raise NotImplementedError(self.get_refs_container)
 
253
 
 
254
    def determine_repository_policy(self, force_new_repo=False, stack_on=None,
 
255
                                    stack_on_pwd=None, require_stacking=False):
 
256
        """Return an object representing a policy to use.
 
257
 
 
258
        This controls whether a new repository is created, and the format of
 
259
        that repository, or some existing shared repository used instead.
 
260
 
 
261
        If stack_on is supplied, will not seek a containing shared repo.
 
262
 
 
263
        :param force_new_repo: If True, require a new repository to be created.
 
264
        :param stack_on: If supplied, the location to stack on.  If not
 
265
            supplied, a default_stack_on location may be used.
 
266
        :param stack_on_pwd: If stack_on is relative, the location it is
 
267
            relative to.
 
268
        """
 
269
        return UseExistingRepository(self.find_repository())
 
270
 
 
271
    def get_branches(self):
 
272
        from .refs import ref_to_branch_name
 
273
        ret = {}
 
274
        for ref in self.get_refs_container().keys():
 
275
            try:
 
276
                branch_name = ref_to_branch_name(ref)
 
277
            except ValueError:
 
278
                continue
 
279
            except UnicodeDecodeError:
 
280
                trace.warning("Ignoring branch %r with unicode error ref", ref)
 
281
                continue
 
282
            ret[branch_name] = self.open_branch(ref=ref)
 
283
        return ret
 
284
 
 
285
    def list_branches(self):
 
286
        return list(self.get_branches().values())
 
287
 
 
288
    def push_branch(self, source, revision_id=None, overwrite=False,
 
289
                    remember=False, create_prefix=False, lossy=False,
 
290
                    name=None):
 
291
        """Push the source branch into this ControlDir."""
 
292
        push_result = GitPushResult()
 
293
        push_result.workingtree_updated = None
 
294
        push_result.master_branch = None
 
295
        push_result.source_branch = source
 
296
        push_result.stacked_on = None
 
297
        repo = self.find_repository()
 
298
        refname = self._get_selected_ref(name)
 
299
        from .branch import GitBranch
 
300
        if isinstance(source, GitBranch) and lossy:
 
301
            raise bzr_errors.LossyPushToSameVCS(source.controldir, self)
 
302
        target = self.open_branch(name, nascent_ok=True)
 
303
        push_result.branch_push_result = source.push(
 
304
            target, overwrite=overwrite, stop_revision=revision_id,
 
305
            lossy=lossy)
 
306
        push_result.new_revid = push_result.branch_push_result.new_revid
 
307
        push_result.old_revid = push_result.branch_push_result.old_revid
 
308
        push_result.target_branch = self.open_branch(name)
 
309
        if source.get_push_location() is None or remember:
 
310
            source.set_push_location(push_result.target_branch.base)
 
311
        return push_result
 
312
 
 
313
 
 
314
class LocalGitControlDirFormat(GitControlDirFormat):
 
315
    """The .git directory control format."""
 
316
 
 
317
    bare = False
 
318
 
 
319
    @classmethod
 
320
    def _known_formats(self):
 
321
        return set([LocalGitControlDirFormat()])
 
322
 
 
323
    @property
 
324
    def repository_format(self):
 
325
        from .repository import GitRepositoryFormat
 
326
        return GitRepositoryFormat()
 
327
 
 
328
    @property
 
329
    def workingtree_format(self):
 
330
        from .workingtree import GitWorkingTreeFormat
 
331
        return GitWorkingTreeFormat()
 
332
 
 
333
    def get_branch_format(self):
 
334
        from .branch import LocalGitBranchFormat
 
335
        return LocalGitBranchFormat()
 
336
 
 
337
    def open(self, transport, _found=None):
 
338
        """Open this directory.
 
339
 
 
340
        """
 
341
        from .transportgit import TransportRepo
 
342
 
 
343
        def _open(transport):
 
344
            return TransportRepo(transport, self.bare,
 
345
                                 refs_text=getattr(self, "_refs_text", None))
 
346
 
 
347
        def redirected(transport, e, redirection_notice):
 
348
            trace.note(redirection_notice)
 
349
            return transport._redirected_to(e.source, e.target)
 
350
        gitrepo = do_catching_redirections(_open, transport, redirected)
 
351
        if not gitrepo._controltransport.has('HEAD'):
 
352
            raise bzr_errors.NotBranchError(path=transport.base)
 
353
        return LocalGitDir(transport, gitrepo, self)
 
354
 
 
355
    def get_format_description(self):
 
356
        return "Local Git Repository"
 
357
 
 
358
    def initialize_on_transport(self, transport):
 
359
        from .transportgit import TransportRepo
 
360
        repo = TransportRepo.init(transport, bare=self.bare)
 
361
        return self.open(transport)
 
362
 
 
363
    def initialize_on_transport_ex(self, transport, use_existing_dir=False,
 
364
                                   create_prefix=False, force_new_repo=False, stacked_on=None,
 
365
                                   stack_on_pwd=None, repo_format_name=None, make_working_trees=None,
 
366
                                   shared_repo=False, vfs_only=False):
 
367
        def make_directory(transport):
 
368
            transport.mkdir('.')
 
369
            return transport
 
370
 
 
371
        def redirected(transport, e, redirection_notice):
 
372
            trace.note(redirection_notice)
 
373
            return transport._redirected_to(e.source, e.target)
 
374
        try:
 
375
            transport = do_catching_redirections(make_directory, transport,
 
376
                                                 redirected)
 
377
        except bzr_errors.FileExists:
 
378
            if not use_existing_dir:
 
379
                raise
 
380
        except bzr_errors.NoSuchFile:
 
381
            if not create_prefix:
 
382
                raise
 
383
            transport.create_prefix()
 
384
        controldir = self.initialize_on_transport(transport)
 
385
        if repo_format_name:
 
386
            result_repo = controldir.find_repository()
 
387
            repository_policy = UseExistingRepository(result_repo)
 
388
            result_repo.lock_write()
 
389
        else:
 
390
            result_repo = None
 
391
            repository_policy = None
 
392
        return (result_repo, controldir, False,
 
393
                repository_policy)
 
394
 
 
395
    def is_supported(self):
 
396
        return True
 
397
 
 
398
    def supports_transport(self, transport):
 
399
        try:
 
400
            external_url = transport.external_url()
 
401
        except bzr_errors.InProcessTransport:
 
402
            raise bzr_errors.NotBranchError(path=transport.base)
 
403
        return external_url.startswith("file:")
 
404
 
 
405
 
 
406
class BareLocalGitControlDirFormat(LocalGitControlDirFormat):
 
407
 
 
408
    bare = True
 
409
    supports_workingtrees = False
 
410
 
 
411
    def get_format_description(self):
 
412
        return "Local Git Repository (bare)"
 
413
 
 
414
 
 
415
class LocalGitDir(GitDir):
 
416
    """An adapter to the '.git' dir used by git."""
 
417
 
 
418
    def _get_gitrepository_class(self):
 
419
        from .repository import LocalGitRepository
 
420
        return LocalGitRepository
 
421
 
 
422
    def __repr__(self):
 
423
        return "<%s at %r>" % (
 
424
            self.__class__.__name__, self.root_transport.base)
 
425
 
 
426
    _gitrepository_class = property(_get_gitrepository_class)
 
427
 
 
428
    @property
 
429
    def user_transport(self):
 
430
        return self.root_transport
 
431
 
 
432
    @property
 
433
    def control_transport(self):
 
434
        return self._git._controltransport
 
435
 
 
436
    def __init__(self, transport, gitrepo, format):
 
437
        self._format = format
 
438
        self.root_transport = transport
 
439
        self._mode_check_done = False
 
440
        self._git = gitrepo
 
441
        if gitrepo.bare:
 
442
            self.transport = transport
 
443
        else:
 
444
            self.transport = transport.clone('.git')
 
445
        self._mode_check_done = None
 
446
 
 
447
    def is_control_filename(self, filename):
 
448
        return (filename == '.git'
 
449
                or filename.startswith('.git/')
 
450
                or filename.startswith('.git\\'))
 
451
 
 
452
    def _get_symref(self, ref):
 
453
        ref_chain, unused_sha = self._git.refs.follow(ref)
 
454
        if len(ref_chain) == 1:
 
455
            return None
 
456
        return ref_chain[1]
 
457
 
 
458
    def set_branch_reference(self, target_branch, name=None):
 
459
        ref = self._get_selected_ref(name)
 
460
        if self.control_transport.base == target_branch.controldir.control_transport.base:
 
461
            if ref == target_branch.ref:
 
462
                raise BranchReferenceLoop(target_branch)
 
463
            self._git.refs.set_symbolic_ref(ref, target_branch.ref)
 
464
        else:
 
465
            try:
 
466
                target_path = target_branch.controldir.control_transport.local_abspath(
 
467
                    '.')
 
468
            except bzr_errors.NotLocalUrl:
 
469
                raise bzr_errors.IncompatibleFormat(
 
470
                    target_branch._format, self._format)
 
471
            # TODO(jelmer): Do some consistency checking across branches..
 
472
            self.control_transport.put_bytes(
 
473
                'commondir', target_path.encode('utf-8'))
 
474
            # TODO(jelmer): Urgh, avoid mucking about with internals.
 
475
            self._git._commontransport = target_branch.repository._git._commontransport.clone()
 
476
            self._git.object_store = TransportObjectStore(
 
477
                self._git._commontransport.clone(OBJECTDIR))
 
478
            self._git.refs.transport = self._git._commontransport
 
479
            target_ref_chain, unused_sha = target_branch.controldir._git.refs.follow(
 
480
                target_branch.ref)
 
481
            for target_ref in target_ref_chain:
 
482
                if target_ref == b'HEAD':
 
483
                    continue
 
484
                break
 
485
            else:
 
486
                # Can't create a reference to something that is not a in a repository.
 
487
                raise bzr_errors.IncompatibleFormat(
 
488
                    self.set_branch_reference, self)
 
489
            self._git.refs.set_symbolic_ref(ref, target_ref)
 
490
 
 
491
    def get_branch_reference(self, name=None):
 
492
        ref = self._get_selected_ref(name)
 
493
        target_ref = self._get_symref(ref)
 
494
        if target_ref is not None:
 
495
            from .refs import ref_to_branch_name
 
496
            try:
 
497
                branch_name = ref_to_branch_name(target_ref)
 
498
            except ValueError:
 
499
                params = {'ref': urlutils.quote(
 
500
                    target_ref.decode('utf-8'), '')}
 
501
            else:
 
502
                if branch_name != '':
 
503
                    params = {'branch': urlutils.quote(branch_name, '')}
 
504
                else:
 
505
                    params = {}
 
506
            try:
 
507
                commondir = self.control_transport.get_bytes('commondir')
 
508
            except bzr_errors.NoSuchFile:
 
509
                base_url = self.user_url.rstrip('/')
 
510
            else:
 
511
                base_url = urlutils.local_path_to_url(
 
512
                    commondir.decode(osutils._fs_enc)).rstrip('/.git/') + '/'
 
513
            if not PY3:
 
514
                params = {k: v.encode('utf-8') for (k, v) in viewitems(params)}
 
515
            return urlutils.join_segment_parameters(base_url, params)
 
516
        return None
 
517
 
 
518
    def find_branch_format(self, name=None):
 
519
        from .branch import (
 
520
            LocalGitBranchFormat,
 
521
            )
 
522
        ref = self._get_selected_ref(name)
 
523
        return LocalGitBranchFormat()
 
524
 
 
525
    def get_branch_transport(self, branch_format, name=None):
 
526
        if branch_format is None:
 
527
            return self.transport
 
528
        if isinstance(branch_format, LocalGitControlDirFormat):
 
529
            return self.transport
 
530
        raise bzr_errors.IncompatibleFormat(branch_format, self._format)
 
531
 
 
532
    def get_repository_transport(self, format):
 
533
        if format is None:
 
534
            return self.transport
 
535
        if isinstance(format, LocalGitControlDirFormat):
 
536
            return self.transport
 
537
        raise bzr_errors.IncompatibleFormat(format, self._format)
 
538
 
 
539
    def get_workingtree_transport(self, format):
 
540
        if format is None:
 
541
            return self.transport
 
542
        if isinstance(format, LocalGitControlDirFormat):
 
543
            return self.transport
 
544
        raise bzr_errors.IncompatibleFormat(format, self._format)
 
545
 
 
546
    def open_branch(self, name=None, unsupported=False, ignore_fallbacks=None,
 
547
                    ref=None, possible_transports=None, nascent_ok=False):
 
548
        """'create' a branch for this dir."""
 
549
        repo = self.find_repository()
 
550
        from .branch import LocalGitBranch
 
551
        ref = self._get_selected_ref(name, ref)
 
552
        if not nascent_ok and ref not in self._git.refs:
 
553
            raise bzr_errors.NotBranchError(self.root_transport.base,
 
554
                                            controldir=self)
 
555
        ref_chain, unused_sha = self._git.refs.follow(ref)
 
556
        if ref_chain[-1] == b'HEAD':
 
557
            controldir = self
 
558
        else:
 
559
            controldir = self._find_commondir()
 
560
        return LocalGitBranch(controldir, repo, ref_chain[-1])
 
561
 
 
562
    def destroy_branch(self, name=None):
 
563
        refname = self._get_selected_ref(name)
 
564
        if refname == b'HEAD':
 
565
            # HEAD can't be removed
 
566
            raise bzr_errors.UnsupportedOperation(
 
567
                self.destroy_branch, self)
 
568
        try:
 
569
            del self._git.refs[refname]
 
570
        except KeyError:
 
571
            raise bzr_errors.NotBranchError(self.root_transport.base,
 
572
                                            controldir=self)
 
573
 
 
574
    def destroy_repository(self):
 
575
        raise bzr_errors.UnsupportedOperation(self.destroy_repository, self)
 
576
 
 
577
    def destroy_workingtree(self):
 
578
        raise bzr_errors.UnsupportedOperation(self.destroy_workingtree, self)
 
579
 
 
580
    def destroy_workingtree_metadata(self):
 
581
        raise bzr_errors.UnsupportedOperation(
 
582
            self.destroy_workingtree_metadata, self)
 
583
 
 
584
    def needs_format_conversion(self, format=None):
 
585
        return not isinstance(self._format, format.__class__)
 
586
 
 
587
    def open_repository(self):
 
588
        """'open' a repository for this dir."""
 
589
        if self.control_transport.has('commondir'):
 
590
            raise bzr_errors.NoRepositoryPresent(self)
 
591
        return self._gitrepository_class(self)
 
592
 
 
593
    def has_workingtree(self):
 
594
        return not self._git.bare
 
595
 
 
596
    def open_workingtree(self, recommend_upgrade=True, unsupported=False):
 
597
        if not self._git.bare:
 
598
            repo = self.find_repository()
 
599
            from .workingtree import GitWorkingTree
 
600
            branch = self.open_branch(ref=b'HEAD', nascent_ok=True)
 
601
            return GitWorkingTree(self, repo, branch)
 
602
        loc = urlutils.unescape_for_display(self.root_transport.base, 'ascii')
 
603
        raise bzr_errors.NoWorkingTree(loc)
 
604
 
 
605
    def create_repository(self, shared=False):
 
606
        from .repository import GitRepositoryFormat
 
607
        if shared:
 
608
            raise bzr_errors.IncompatibleFormat(
 
609
                GitRepositoryFormat(), self._format)
 
610
        return self.find_repository()
 
611
 
 
612
    def create_branch(self, name=None, repository=None,
 
613
                      append_revisions_only=None, ref=None):
 
614
        refname = self._get_selected_ref(name, ref)
 
615
        if refname != b'HEAD' and refname in self._git.refs:
 
616
            raise bzr_errors.AlreadyBranchError(self.user_url)
 
617
        repo = self.open_repository()
 
618
        if refname in self._git.refs:
 
619
            ref_chain, unused_sha = self._git.refs.follow(
 
620
                self._get_selected_ref(None))
 
621
            if ref_chain[0] == b'HEAD':
 
622
                refname = ref_chain[1]
 
623
        from .branch import LocalGitBranch
 
624
        branch = LocalGitBranch(self, repo, refname)
 
625
        if append_revisions_only:
 
626
            branch.set_append_revisions_only(append_revisions_only)
 
627
        return branch
 
628
 
 
629
    def backup_bzrdir(self):
 
630
        if not self._git.bare:
 
631
            self.root_transport.copy_tree(".git", ".git.backup")
 
632
            return (self.root_transport.abspath(".git"),
 
633
                    self.root_transport.abspath(".git.backup"))
 
634
        else:
 
635
            basename = urlutils.basename(self.root_transport.base)
 
636
            parent = self.root_transport.clone('..')
 
637
            parent.copy_tree(basename, basename + ".backup")
 
638
 
 
639
    def create_workingtree(self, revision_id=None, from_branch=None,
 
640
                           accelerator_tree=None, hardlink=False):
 
641
        if self._git.bare:
 
642
            raise bzr_errors.UnsupportedOperation(
 
643
                self.create_workingtree, self)
 
644
        if from_branch is None:
 
645
            from_branch = self.open_branch(nascent_ok=True)
 
646
        if revision_id is None:
 
647
            revision_id = from_branch.last_revision()
 
648
        repo = self.find_repository()
 
649
        from .workingtree import GitWorkingTree
 
650
        wt = GitWorkingTree(self, repo, from_branch)
 
651
        wt.set_last_revision(revision_id)
 
652
        wt._build_checkout_with_index()
 
653
        return wt
 
654
 
 
655
    def _find_or_create_repository(self, force_new_repo=None):
 
656
        return self.create_repository(shared=False)
 
657
 
 
658
    def _find_creation_modes(self):
 
659
        """Determine the appropriate modes for files and directories.
 
660
 
 
661
        They're always set to be consistent with the base directory,
 
662
        assuming that this transport allows setting modes.
 
663
        """
 
664
        # TODO: Do we need or want an option (maybe a config setting) to turn
 
665
        # this off or override it for particular locations? -- mbp 20080512
 
666
        if self._mode_check_done:
 
667
            return
 
668
        self._mode_check_done = True
 
669
        try:
 
670
            st = self.transport.stat('.')
 
671
        except bzr_errors.TransportNotPossible:
 
672
            self._dir_mode = None
 
673
            self._file_mode = None
 
674
        else:
 
675
            # Check the directory mode, but also make sure the created
 
676
            # directories and files are read-write for this user. This is
 
677
            # mostly a workaround for filesystems which lie about being able to
 
678
            # write to a directory (cygwin & win32)
 
679
            if (st.st_mode & 0o7777 == 0o0000):
 
680
                # FTP allows stat but does not return dir/file modes
 
681
                self._dir_mode = None
 
682
                self._file_mode = None
 
683
            else:
 
684
                self._dir_mode = (st.st_mode & 0o7777) | 0o0700
 
685
                # Remove the sticky and execute bits for files
 
686
                self._file_mode = self._dir_mode & ~0o7111
 
687
 
 
688
    def _get_file_mode(self):
 
689
        """Return Unix mode for newly created files, or None.
 
690
        """
 
691
        if not self._mode_check_done:
 
692
            self._find_creation_modes()
 
693
        return self._file_mode
 
694
 
 
695
    def _get_dir_mode(self):
 
696
        """Return Unix mode for newly created directories, or None.
 
697
        """
 
698
        if not self._mode_check_done:
 
699
            self._find_creation_modes()
 
700
        return self._dir_mode
 
701
 
 
702
    def get_refs_container(self):
 
703
        return self._git.refs
 
704
 
 
705
    def get_peeled(self, ref):
 
706
        return self._git.get_peeled(ref)
 
707
 
 
708
    def _find_commondir(self):
 
709
        try:
 
710
            commondir = self.control_transport.get_bytes('commondir')
 
711
        except bzr_errors.NoSuchFile:
 
712
            return self
 
713
        else:
 
714
            commondir = commondir.rstrip(b'/.git/').decode(osutils._fs_enc)
 
715
            return ControlDir.open_from_transport(get_transport_from_path(commondir))