/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: Breezy landing bot
  • Author(s): Jelmer Vernooij
  • Date: 2018-11-16 11:31:40 UTC
  • mfrom: (7143.12.3 annotated-tags)
  • Revision ID: breezy.the.bot@gmail.com-20181116113140-618u04763u0dyxnh
Fix fetching of revisions that are referenced by annotated tags.

Merged from https://code.launchpad.net/~jelmer/brz/annotated-tags/+merge/358536

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 brz_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 brz_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 brz_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 brz_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 brz_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(self._format, self.user_url)
 
175
        interrepo = InterRepository.get(source_repository, result_repo)
 
176
 
 
177
        if revision_id is not None:
 
178
            determine_wants = interrepo.get_determine_wants_revids(
 
179
                [revision_id], include_tags=True)
 
180
        else:
 
181
            determine_wants = interrepo.determine_wants_all
 
182
        interrepo.fetch_objects(determine_wants=determine_wants,
 
183
            mapping=source_branch.mapping)
 
184
        result_branch = source_branch.sprout(result,
 
185
            revision_id=revision_id, repository=result_repo)
 
186
        if (create_tree_if_local
 
187
            and isinstance(target_transport, LocalTransport)
 
188
            and (result_repo is None or result_repo.make_working_trees())):
 
189
            wt = result.create_workingtree(accelerator_tree=accelerator_tree,
 
190
                hardlink=hardlink, from_branch=result_branch)
 
191
        return result
 
192
 
 
193
    def clone_on_transport(self, transport, revision_id=None,
 
194
        force_new_repo=False, preserve_stacking=False, stacked_on=None,
 
195
        create_prefix=False, use_existing_dir=True, no_tree=False):
 
196
        """See ControlDir.clone_on_transport."""
 
197
        from ..repository import InterRepository
 
198
        from .mapping import default_mapping
 
199
        if stacked_on is not None:
 
200
            raise _mod_branch.UnstackableBranchFormat(self._format, self.user_url)
 
201
        if no_tree:
 
202
            format = BareLocalGitControlDirFormat()
 
203
        else:
 
204
            format = LocalGitControlDirFormat()
 
205
        (target_repo, target_controldir, stacking,
 
206
                repo_policy) = format.initialize_on_transport_ex(
 
207
                        transport, use_existing_dir=use_existing_dir,
 
208
                        create_prefix=create_prefix,
 
209
                        force_new_repo=force_new_repo)
 
210
        target_repo = target_controldir.find_repository()
 
211
        target_git_repo = target_repo._git
 
212
        source_repo = self.find_repository()
 
213
        source_git_repo = source_repo._git
 
214
        interrepo = InterRepository.get(source_repo, target_repo)
 
215
        if revision_id is not None:
 
216
            determine_wants = interrepo.get_determine_wants_revids([revision_id], include_tags=True)
 
217
        else:
 
218
            determine_wants = interrepo.determine_wants_all
 
219
        (pack_hint, _, refs) = interrepo.fetch_objects(determine_wants,
 
220
            mapping=default_mapping)
 
221
        for name, val in viewitems(refs):
 
222
            target_git_repo.refs[name] = val
 
223
        result_dir = self.__class__(transport, target_git_repo, format)
 
224
        if revision_id is not None:
 
225
            result_dir.open_branch().set_last_revision(revision_id)
 
226
        try:
 
227
            # Cheaper to check if the target is not local, than to try making
 
228
            # the tree and fail.
 
229
            result_dir.root_transport.local_abspath('.')
 
230
            if result_dir.open_repository().make_working_trees():
 
231
                self.open_workingtree().clone(result_dir, revision_id=revision_id)
 
232
        except (brz_errors.NoWorkingTree, brz_errors.NotLocalUrl):
 
233
            pass
 
234
 
 
235
        return result_dir
 
236
 
 
237
    def find_repository(self):
 
238
        """Find the repository that should be used.
 
239
 
 
240
        This does not require a branch as we use it to find the repo for
 
241
        new branches as well as to hook existing branches up to their
 
242
        repository.
 
243
        """
 
244
        return self._gitrepository_class(self._find_commondir())
 
245
 
 
246
    def get_refs_container(self):
 
247
        """Retrieve the refs container.
 
248
        """
 
249
        raise NotImplementedError(self.get_refs_container)
 
250
 
 
251
    def determine_repository_policy(self, force_new_repo=False, stack_on=None,
 
252
                                    stack_on_pwd=None, require_stacking=False):
 
253
        """Return an object representing a policy to use.
 
254
 
 
255
        This controls whether a new repository is created, and the format of
 
256
        that repository, or some existing shared repository used instead.
 
257
 
 
258
        If stack_on is supplied, will not seek a containing shared repo.
 
259
 
 
260
        :param force_new_repo: If True, require a new repository to be created.
 
261
        :param stack_on: If supplied, the location to stack on.  If not
 
262
            supplied, a default_stack_on location may be used.
 
263
        :param stack_on_pwd: If stack_on is relative, the location it is
 
264
            relative to.
 
265
        """
 
266
        return UseExistingRepository(self.find_repository())
 
267
 
 
268
    def get_branches(self):
 
269
        from .refs import ref_to_branch_name
 
270
        ret = {}
 
271
        for ref in self.get_refs_container().keys():
 
272
            try:
 
273
                branch_name = ref_to_branch_name(ref)
 
274
            except ValueError:
 
275
                continue
 
276
            except UnicodeDecodeError:
 
277
                trace.warning("Ignoring branch %r with unicode error ref", ref)
 
278
                continue
 
279
            ret[branch_name] = self.open_branch(ref=ref)
 
280
        return ret
 
281
 
 
282
    def list_branches(self):
 
283
        return list(self.get_branches().values())
 
284
 
 
285
    def push_branch(self, source, revision_id=None, overwrite=False,
 
286
                    remember=False, create_prefix=False, lossy=False,
 
287
                    name=None):
 
288
        """Push the source branch into this ControlDir."""
 
289
        push_result = GitPushResult()
 
290
        push_result.workingtree_updated = None
 
291
        push_result.master_branch = None
 
292
        push_result.source_branch = source
 
293
        push_result.stacked_on = None
 
294
        repo = self.find_repository()
 
295
        refname = self._get_selected_ref(name)
 
296
        from .branch import GitBranch
 
297
        if isinstance(source, GitBranch) and lossy:
 
298
            raise brz_errors.LossyPushToSameVCS(source.controldir, self)
 
299
        target = self.open_branch(name, nascent_ok=True)
 
300
        push_result.branch_push_result = source.push(
 
301
                target, overwrite=overwrite, stop_revision=revision_id,
 
302
                lossy=lossy)
 
303
        push_result.new_revid = push_result.branch_push_result.new_revid
 
304
        push_result.old_revid = push_result.branch_push_result.old_revid
 
305
        push_result.target_branch = self.open_branch(name)
 
306
        if source.get_push_location() is None or remember:
 
307
            source.set_push_location(push_result.target_branch.base)
 
308
        return push_result
 
309
 
 
310
 
 
311
class LocalGitControlDirFormat(GitControlDirFormat):
 
312
    """The .git directory control format."""
 
313
 
 
314
    bare = False
 
315
 
 
316
    @classmethod
 
317
    def _known_formats(self):
 
318
        return set([LocalGitControlDirFormat()])
 
319
 
 
320
    @property
 
321
    def repository_format(self):
 
322
        from .repository import GitRepositoryFormat
 
323
        return GitRepositoryFormat()
 
324
 
 
325
    @property
 
326
    def workingtree_format(self):
 
327
        from .workingtree import GitWorkingTreeFormat
 
328
        return GitWorkingTreeFormat()
 
329
 
 
330
    def get_branch_format(self):
 
331
        from .branch import LocalGitBranchFormat
 
332
        return LocalGitBranchFormat()
 
333
 
 
334
    def open(self, transport, _found=None):
 
335
        """Open this directory.
 
336
 
 
337
        """
 
338
        from .transportgit import TransportRepo
 
339
        def _open(transport):
 
340
            return TransportRepo(transport, self.bare,
 
341
                refs_text=getattr(self, "_refs_text", None))
 
342
        def redirected(transport, e, redirection_notice):
 
343
            trace.note(redirection_notice)
 
344
            return transport._redirected_to(e.source, e.target)
 
345
        gitrepo = do_catching_redirections(_open, transport, redirected)
 
346
        if not gitrepo._controltransport.has('HEAD'):
 
347
            raise brz_errors.NotBranchError(path=transport.base)
 
348
        return LocalGitDir(transport, gitrepo, self)
 
349
 
 
350
    def get_format_description(self):
 
351
        return "Local Git Repository"
 
352
 
 
353
    def initialize_on_transport(self, transport):
 
354
        from .transportgit import TransportRepo
 
355
        repo = TransportRepo.init(transport, bare=self.bare)
 
356
        return self.open(transport)
 
357
 
 
358
    def initialize_on_transport_ex(self, transport, use_existing_dir=False,
 
359
        create_prefix=False, force_new_repo=False, stacked_on=None,
 
360
        stack_on_pwd=None, repo_format_name=None, make_working_trees=None,
 
361
        shared_repo=False, vfs_only=False):
 
362
        def make_directory(transport):
 
363
            transport.mkdir('.')
 
364
            return transport
 
365
        def redirected(transport, e, redirection_notice):
 
366
            trace.note(redirection_notice)
 
367
            return transport._redirected_to(e.source, e.target)
 
368
        try:
 
369
            transport = do_catching_redirections(make_directory, transport,
 
370
                redirected)
 
371
        except brz_errors.FileExists:
 
372
            if not use_existing_dir:
 
373
                raise
 
374
        except brz_errors.NoSuchFile:
 
375
            if not create_prefix:
 
376
                raise
 
377
            transport.create_prefix()
 
378
        controldir = self.initialize_on_transport(transport)
 
379
        if repo_format_name:
 
380
            result_repo = controldir.find_repository()
 
381
            repository_policy = UseExistingRepository(result_repo)
 
382
            result_repo.lock_write()
 
383
        else:
 
384
            result_repo = None
 
385
            repository_policy = None
 
386
        return (result_repo, controldir, False,
 
387
                repository_policy)
 
388
 
 
389
    def is_supported(self):
 
390
        return True
 
391
 
 
392
    def supports_transport(self, transport):
 
393
        try:
 
394
            external_url = transport.external_url()
 
395
        except brz_errors.InProcessTransport:
 
396
            raise brz_errors.NotBranchError(path=transport.base)
 
397
        return external_url.startswith("file:")
 
398
 
 
399
 
 
400
class BareLocalGitControlDirFormat(LocalGitControlDirFormat):
 
401
 
 
402
    bare = True
 
403
    supports_workingtrees = False
 
404
 
 
405
    def get_format_description(self):
 
406
        return "Local Git Repository (bare)"
 
407
 
 
408
 
 
409
class LocalGitDir(GitDir):
 
410
    """An adapter to the '.git' dir used by git."""
 
411
 
 
412
    def _get_gitrepository_class(self):
 
413
        from .repository import LocalGitRepository
 
414
        return LocalGitRepository
 
415
 
 
416
    def __repr__(self):
 
417
        return "<%s at %r>" % (
 
418
            self.__class__.__name__, self.root_transport.base)
 
419
 
 
420
    _gitrepository_class = property(_get_gitrepository_class)
 
421
 
 
422
    @property
 
423
    def user_transport(self):
 
424
        return self.root_transport
 
425
 
 
426
    @property
 
427
    def control_transport(self):
 
428
        return self._git._controltransport
 
429
 
 
430
    def __init__(self, transport, gitrepo, format):
 
431
        self._format = format
 
432
        self.root_transport = transport
 
433
        self._mode_check_done = False
 
434
        self._git = gitrepo
 
435
        if gitrepo.bare:
 
436
            self.transport = transport
 
437
        else:
 
438
            self.transport = transport.clone('.git')
 
439
        self._mode_check_done = None
 
440
 
 
441
    def is_control_filename(self, filename):
 
442
        return (filename == '.git' or
 
443
                filename.startswith('.git/') or
 
444
                filename.startswith('.git\\'))
 
445
 
 
446
    def _get_symref(self, ref):
 
447
        ref_chain, unused_sha = self._git.refs.follow(ref)
 
448
        if len(ref_chain) == 1:
 
449
            return None
 
450
        return ref_chain[1]
 
451
 
 
452
    def set_branch_reference(self, target_branch, name=None):
 
453
        ref = self._get_selected_ref(name)
 
454
        if self.control_transport.base == target_branch.controldir.control_transport.base:
 
455
            if ref == target_branch.ref:
 
456
                raise BranchReferenceLoop(target_branch)
 
457
            self._git.refs.set_symbolic_ref(ref, target_branch.ref)
 
458
        else:
 
459
            try:
 
460
                target_path = target_branch.controldir.control_transport.local_abspath('.')
 
461
            except brz_errors.NotLocalUrl:
 
462
                raise brz_errors.IncompatibleFormat(target_branch._format, self._format)
 
463
            # TODO(jelmer): Do some consistency checking across branches..
 
464
            self.control_transport.put_bytes('commondir', target_path.encode('utf-8'))
 
465
            # TODO(jelmer): Urgh, avoid mucking about with internals.
 
466
            self._git._commontransport = target_branch.repository._git._commontransport.clone()
 
467
            self._git.object_store = TransportObjectStore(self._git._commontransport.clone(OBJECTDIR))
 
468
            self._git.refs.transport = self._git._commontransport
 
469
            target_ref_chain, unused_sha = target_branch.controldir._git.refs.follow(target_branch.ref)
 
470
            for target_ref in target_ref_chain:
 
471
                if target_ref == b'HEAD':
 
472
                    continue
 
473
                break
 
474
            else:
 
475
                # Can't create a reference to something that is not a in a repository.
 
476
                raise brz_errors.IncompatibleFormat(self.set_branch_reference, self)
 
477
            self._git.refs.set_symbolic_ref(ref, target_ref)
 
478
 
 
479
    def get_branch_reference(self, name=None):
 
480
        ref = self._get_selected_ref(name)
 
481
        target_ref = self._get_symref(ref)
 
482
        if target_ref is not None:
 
483
            from .refs import ref_to_branch_name
 
484
            try:
 
485
                branch_name = ref_to_branch_name(target_ref)
 
486
            except ValueError:
 
487
                params = {'ref': urlutils.quote(target_ref.decode('utf-8'), '')}
 
488
            else:
 
489
                if branch_name != '':
 
490
                    params = {'branch': urlutils.quote(branch_name, '')}
 
491
                else:
 
492
                    params = {}
 
493
            try:
 
494
                commondir = self.control_transport.get_bytes('commondir')
 
495
            except brz_errors.NoSuchFile:
 
496
                base_url = self.user_url.rstrip('/')
 
497
            else:
 
498
                base_url = urlutils.local_path_to_url(commondir.decode(osutils._fs_enc)).rstrip('/.git/')+'/'
 
499
            if not PY3:
 
500
                params = {k: v.encode('utf-8') for (k, v) in viewitems(params)}
 
501
            return urlutils.join_segment_parameters(base_url, params)
 
502
        return None
 
503
 
 
504
    def find_branch_format(self, name=None):
 
505
        from .branch import (
 
506
            LocalGitBranchFormat,
 
507
            )
 
508
        ref = self._get_selected_ref(name)
 
509
        return LocalGitBranchFormat()
 
510
 
 
511
    def get_branch_transport(self, branch_format, name=None):
 
512
        if branch_format is None:
 
513
            return self.transport
 
514
        if isinstance(branch_format, LocalGitControlDirFormat):
 
515
            return self.transport
 
516
        raise brz_errors.IncompatibleFormat(branch_format, self._format)
 
517
 
 
518
    def get_repository_transport(self, format):
 
519
        if format is None:
 
520
            return self.transport
 
521
        if isinstance(format, LocalGitControlDirFormat):
 
522
            return self.transport
 
523
        raise brz_errors.IncompatibleFormat(format, self._format)
 
524
 
 
525
    def get_workingtree_transport(self, format):
 
526
        if format is None:
 
527
            return self.transport
 
528
        if isinstance(format, LocalGitControlDirFormat):
 
529
            return self.transport
 
530
        raise brz_errors.IncompatibleFormat(format, self._format)
 
531
 
 
532
    def open_branch(self, name=None, unsupported=False, ignore_fallbacks=None,
 
533
            ref=None, possible_transports=None, nascent_ok=False):
 
534
        """'create' a branch for this dir."""
 
535
        repo = self.find_repository()
 
536
        from .branch import LocalGitBranch
 
537
        ref = self._get_selected_ref(name, ref)
 
538
        if not nascent_ok and ref not in self._git.refs:
 
539
            raise brz_errors.NotBranchError(self.root_transport.base,
 
540
                    controldir=self)
 
541
        ref_chain, unused_sha = self._git.refs.follow(ref)
 
542
        if ref_chain[-1] == b'HEAD':
 
543
            controldir = self
 
544
        else:
 
545
            controldir = self._find_commondir()
 
546
        return LocalGitBranch(controldir, repo, ref_chain[-1])
 
547
 
 
548
    def destroy_branch(self, name=None):
 
549
        refname = self._get_selected_ref(name)
 
550
        if refname == b'HEAD':
 
551
            # HEAD can't be removed
 
552
            raise brz_errors.UnsupportedOperation(
 
553
                self.destroy_branch, self)
 
554
        try:
 
555
            del self._git.refs[refname]
 
556
        except KeyError:
 
557
            raise brz_errors.NotBranchError(self.root_transport.base,
 
558
                    controldir=self)
 
559
 
 
560
    def destroy_repository(self):
 
561
        raise brz_errors.UnsupportedOperation(self.destroy_repository, self)
 
562
 
 
563
    def destroy_workingtree(self):
 
564
        raise brz_errors.UnsupportedOperation(self.destroy_workingtree, self)
 
565
 
 
566
    def destroy_workingtree_metadata(self):
 
567
        raise brz_errors.UnsupportedOperation(self.destroy_workingtree_metadata, self)
 
568
 
 
569
    def needs_format_conversion(self, format=None):
 
570
        return not isinstance(self._format, format.__class__)
 
571
 
 
572
    def open_repository(self):
 
573
        """'open' a repository for this dir."""
 
574
        if self.control_transport.has('commondir'):
 
575
            raise brz_errors.NoRepositoryPresent(self)
 
576
        return self._gitrepository_class(self)
 
577
 
 
578
    def has_workingtree(self):
 
579
        return not self._git.bare
 
580
 
 
581
    def open_workingtree(self, recommend_upgrade=True, unsupported=False):
 
582
        if not self._git.bare:
 
583
            repo = self.find_repository()
 
584
            from .workingtree import GitWorkingTree
 
585
            branch = self.open_branch(ref=b'HEAD', nascent_ok=True)
 
586
            return GitWorkingTree(self, repo, branch)
 
587
        loc = urlutils.unescape_for_display(self.root_transport.base, 'ascii')
 
588
        raise brz_errors.NoWorkingTree(loc)
 
589
 
 
590
    def create_repository(self, shared=False):
 
591
        from .repository import GitRepositoryFormat
 
592
        if shared:
 
593
            raise brz_errors.IncompatibleFormat(GitRepositoryFormat(), self._format)
 
594
        return self.find_repository()
 
595
 
 
596
    def create_branch(self, name=None, repository=None,
 
597
                      append_revisions_only=None, ref=None):
 
598
        refname = self._get_selected_ref(name, ref)
 
599
        if refname != b'HEAD' and refname in self._git.refs:
 
600
            raise brz_errors.AlreadyBranchError(self.user_url)
 
601
        repo = self.open_repository()
 
602
        if refname in self._git.refs:
 
603
            ref_chain, unused_sha = self._git.refs.follow(self._get_selected_ref(None))
 
604
            if ref_chain[0] == b'HEAD':
 
605
                refname = ref_chain[1]
 
606
        from .branch import LocalGitBranch
 
607
        branch = LocalGitBranch(self, repo, refname)
 
608
        if append_revisions_only:
 
609
            branch.set_append_revisions_only(append_revisions_only)
 
610
        return branch
 
611
 
 
612
    def backup_bzrdir(self):
 
613
        if not self._git.bare:
 
614
            self.root_transport.copy_tree(".git", ".git.backup")
 
615
            return (self.root_transport.abspath(".git"),
 
616
                    self.root_transport.abspath(".git.backup"))
 
617
        else:
 
618
            basename = urlutils.basename(self.root_transport.base)
 
619
            parent = self.root_transport.clone('..')
 
620
            parent.copy_tree(basename, basename + ".backup")
 
621
 
 
622
    def create_workingtree(self, revision_id=None, from_branch=None,
 
623
        accelerator_tree=None, hardlink=False):
 
624
        if self._git.bare:
 
625
            raise brz_errors.UnsupportedOperation(self.create_workingtree, self)
 
626
        if from_branch is None:
 
627
            from_branch = self.open_branch(nascent_ok=True)
 
628
        if revision_id is None:
 
629
            revision_id = from_branch.last_revision()
 
630
        repo = self.find_repository()
 
631
        from .workingtree import GitWorkingTree
 
632
        wt = GitWorkingTree(self, repo, from_branch)
 
633
        wt.set_last_revision(revision_id)
 
634
        wt._build_checkout_with_index()
 
635
        return wt
 
636
 
 
637
    def _find_or_create_repository(self, force_new_repo=None):
 
638
        return self.create_repository(shared=False)
 
639
 
 
640
    def _find_creation_modes(self):
 
641
        """Determine the appropriate modes for files and directories.
 
642
 
 
643
        They're always set to be consistent with the base directory,
 
644
        assuming that this transport allows setting modes.
 
645
        """
 
646
        # TODO: Do we need or want an option (maybe a config setting) to turn
 
647
        # this off or override it for particular locations? -- mbp 20080512
 
648
        if self._mode_check_done:
 
649
            return
 
650
        self._mode_check_done = True
 
651
        try:
 
652
            st = self.transport.stat('.')
 
653
        except brz_errors.TransportNotPossible:
 
654
            self._dir_mode = None
 
655
            self._file_mode = None
 
656
        else:
 
657
            # Check the directory mode, but also make sure the created
 
658
            # directories and files are read-write for this user. This is
 
659
            # mostly a workaround for filesystems which lie about being able to
 
660
            # write to a directory (cygwin & win32)
 
661
            if (st.st_mode & 0o7777 == 0o0000):
 
662
                # FTP allows stat but does not return dir/file modes
 
663
                self._dir_mode = None
 
664
                self._file_mode = None
 
665
            else:
 
666
                self._dir_mode = (st.st_mode & 0o7777) | 0o0700
 
667
                # Remove the sticky and execute bits for files
 
668
                self._file_mode = self._dir_mode & ~0o7111
 
669
 
 
670
    def _get_file_mode(self):
 
671
        """Return Unix mode for newly created files, or None.
 
672
        """
 
673
        if not self._mode_check_done:
 
674
            self._find_creation_modes()
 
675
        return self._file_mode
 
676
 
 
677
    def _get_dir_mode(self):
 
678
        """Return Unix mode for newly created directories, or None.
 
679
        """
 
680
        if not self._mode_check_done:
 
681
            self._find_creation_modes()
 
682
        return self._dir_mode
 
683
 
 
684
    def get_refs_container(self):
 
685
        return self._git.refs
 
686
 
 
687
    def get_peeled(self, ref):
 
688
        return self._git.get_peeled(ref)
 
689
 
 
690
    def _find_commondir(self):
 
691
        try:
 
692
            commondir = self.control_transport.get_bytes('commondir')
 
693
        except brz_errors.NoSuchFile:
 
694
            return self
 
695
        else:
 
696
            commondir = commondir.rstrip(b'/.git/').decode(osutils._fs_enc)
 
697
            return ControlDir.open_from_transport(get_transport_from_path(commondir))