/brz/remove-bazaar

To get this branch, use:
bzr branch http://gegoxaren.bato24.eu/bzr/brz/remove-bazaar
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
1
# Copyright (C) 2007 Canonical Ltd
0.200.910 by Jelmer Vernooij
update copyright years
2
# Copyright (C) 2010 Jelmer Vernooij
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
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., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
17
0.200.1012 by Jelmer Vernooij
Rename BzrDir to ControlDir.
18
"""An adapter between a Git control dir and a Bazaar ControlDir."""
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
19
0.200.1311 by Jelmer Vernooij
More work on colocated branch support.
20
import urllib
21
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
22
from bzrlib import (
0.239.13 by Jelmer Vernooij
Don't break "bzr info -v" when Dulwich is not installed.
23
    errors as bzr_errors,
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
24
    lockable_files,
0.200.1137 by Jelmer Vernooij
Support BzrProber.known_formats().
25
    trace,
0.200.1172 by Jelmer Vernooij
Provide GitDir._available_backup_name.
26
    osutils,
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
27
    urlutils,
28
    )
0.200.1137 by Jelmer Vernooij
Support BzrProber.known_formats().
29
from bzrlib.bzrdir import CreateRepository
30
from bzrlib.transport import do_catching_redirections
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
31
0.200.280 by Jelmer Vernooij
Support bzr.dev.
32
LockWarner = getattr(lockable_files, "_LockWarner", None)
33
0.200.1111 by Jelmer Vernooij
Drop support for Bazaar < 2.3.
34
from bzrlib.controldir import (
35
    ControlDir,
0.200.1137 by Jelmer Vernooij
Support BzrProber.known_formats().
36
    ControlDirFormat,
0.200.1111 by Jelmer Vernooij
Drop support for Bazaar < 2.3.
37
    format_registry,
38
    )
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
39
0.200.123 by Jelmer Vernooij
Use central git module.
40
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
41
class GitLock(object):
42
    """A lock that thunks through to Git."""
43
0.200.1130 by Jelmer Vernooij
Implement GitLock.lock_name.
44
    def __init__(self):
45
        self.lock_name = "git lock"
46
0.200.84 by Jelmer Vernooij
Fix lock_write argument.
47
    def lock_write(self, token=None):
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
48
        pass
49
50
    def lock_read(self):
51
        pass
52
53
    def unlock(self):
54
        pass
55
0.200.73 by Jelmer Vernooij
Implement GitLock.peek().
56
    def peek(self):
57
        pass
58
0.200.130 by Jelmer Vernooij
Make most tree inspection tests succeed.
59
    def validate_token(self, token):
60
        pass
61
0.200.629 by Jelmer Vernooij
Add GitLock.break_lock().
62
    def break_lock(self):
0.200.1254 by Jelmer Vernooij
break_lock is not implemented for git control directories.
63
        raise NotImplementedError(self.break_lock)
0.200.629 by Jelmer Vernooij
Add GitLock.break_lock().
64
0.200.1170 by Jelmer Vernooij
Implement GitLock.leave_lock_in_place and GitLock.dont_leave_lock_in_place.
65
    def dont_leave_in_place(self):
66
        raise NotImplementedError(self.dont_leave_in_place)
67
68
    def leave_in_place(self):
69
        raise NotImplementedError(self.leave_in_place)
70
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
71
72
class GitLockableFiles(lockable_files.LockableFiles):
73
    """Git specific lockable files abstraction."""
74
0.200.129 by Jelmer Vernooij
merge dulwich.
75
    def __init__(self, transport, lock):
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
76
        self._lock = lock
77
        self._transaction = None
78
        self._lock_mode = None
0.200.129 by Jelmer Vernooij
merge dulwich.
79
        self._transport = transport
0.200.280 by Jelmer Vernooij
Support bzr.dev.
80
        if LockWarner is None:
81
            # Bzr 1.13
82
            self._lock_count = 0
83
        else:
84
            self._lock_warner = LockWarner(repr(self))
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
85
86
0.200.1026 by Jelmer Vernooij
Fix typo.
87
class GitDirConfig(object):
0.200.1025 by Jelmer Vernooij
Implement GitDir.get_config().
88
89
    def get_default_stack_on(self):
90
        return None
91
92
    def set_default_stack_on(self, value):
93
        raise bzr_errors.BzrError("Cannot set configuration")
94
95
0.200.1137 by Jelmer Vernooij
Support BzrProber.known_formats().
96
class GitControlDirFormat(ControlDirFormat):
97
98
    _lock_class = lockable_files.TransportLock
99
100
    colocated_branches = True
101
    fixed_components = True
102
103
    def __eq__(self, other):
104
        return type(self) == type(other)
105
106
    def is_supported(self):
107
        return True
108
109
    def network_name(self):
110
        return "git"
111
112
0.200.1012 by Jelmer Vernooij
Rename BzrDir to ControlDir.
113
class GitDir(ControlDir):
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
114
    """An adapter to the '.git' dir used by git."""
115
0.200.148 by Jelmer Vernooij
Share more infrastructure between LocalGitDir and RemoteGitDir.
116
    def is_supported(self):
117
        return True
118
0.200.981 by Jelmer Vernooij
Mark git directories as not convertable (for now).
119
    def can_convert_format(self):
120
        return False
121
0.200.1025 by Jelmer Vernooij
Implement GitDir.get_config().
122
    def break_lock(self):
123
        pass
124
0.200.155 by Jelmer Vernooij
Fix formatting, remove catch-all for exceptions when opening local repositories.
125
    def cloning_metadir(self, stacked=False):
0.200.1013 by Jelmer Vernooij
More renames.
126
        return format_registry.make_bzrdir("default")
0.200.155 by Jelmer Vernooij
Fix formatting, remove catch-all for exceptions when opening local repositories.
127
0.200.1165 by Jelmer Vernooij
Implement GitDir.checkout_metadir.
128
    def checkout_metadir(self, stacked=False):
129
        return format_registry.make_bzrdir("default")
130
0.200.1310 by Jelmer Vernooij
Add _get_selected_ref method.
131
    def _get_selected_ref(self, branch):
132
        if branch is None and getattr(self, "_get_selected_branch", False):
133
            branch = self._get_selected_branch()
134
        if branch is not None:
0.200.1311 by Jelmer Vernooij
More work on colocated branch support.
135
            from bzrlib.plugins.git.refs import branch_name_to_ref
136
            return branch_name_to_ref(branch, None)
0.200.1310 by Jelmer Vernooij
Add _get_selected_ref method.
137
        segment_parameters = getattr(
138
            self.user_transport, "get_segment_parameters", lambda: {})()
0.200.1311 by Jelmer Vernooij
More work on colocated branch support.
139
        ref = segment_parameters.get("ref")
140
        if ref is not None:
141
            ref = urlutils.unescape(ref)
142
        return ref
0.200.1310 by Jelmer Vernooij
Add _get_selected_ref method.
143
0.200.1025 by Jelmer Vernooij
Implement GitDir.get_config().
144
    def get_config(self):
145
        return GitDirConfig()
146
0.200.1172 by Jelmer Vernooij
Provide GitDir._available_backup_name.
147
    def _available_backup_name(self, base):
148
        return osutils.available_backup_name(base, self.root_transport.has)
149
0.259.1 by Jelmer Vernooij
Provide custom GitDir.sprout() for bzr 2.4 compatibility.
150
    def sprout(self, url, revision_id=None, force_new_repo=False,
151
               recurse='down', possible_transports=None,
152
               accelerator_tree=None, hardlink=False, stacked=False,
153
               source_branch=None, create_tree_if_local=True):
154
        from bzrlib.repository import InterRepository
155
        from bzrlib.transport.local import LocalTransport
156
        from bzrlib.transport import get_transport
157
        target_transport = get_transport(url, possible_transports)
158
        target_transport.ensure_base()
159
        cloning_format = self.cloning_metadir()
160
        # Create/update the result branch
161
        result = cloning_format.initialize_on_transport(target_transport)
0.200.1373 by Jelmer Vernooij
Prevent accidentally removing branch.
162
        source_branch = self.open_branch()
0.259.1 by Jelmer Vernooij
Provide custom GitDir.sprout() for bzr 2.4 compatibility.
163
        source_repository = self.find_repository()
164
        try:
165
            result_repo = result.find_repository()
166
        except bzr_errors.NoRepositoryPresent:
167
            result_repo = result.create_repository()
168
            target_is_empty = True
169
        else:
170
            target_is_empty = None # Unknown
171
        if stacked:
172
            raise bzr_errors.IncompatibleRepositories(source_repository, result_repo)
173
        interrepo = InterRepository.get(source_repository, result_repo)
174
175
        if revision_id is not None:
0.259.4 by Jelmer Vernooij
Put determine_wants methods on InterRepo.
176
            determine_wants = interrepo.get_determine_wants_revids(
177
                [revision_id], include_tags=True)
0.259.1 by Jelmer Vernooij
Provide custom GitDir.sprout() for bzr 2.4 compatibility.
178
        else:
0.259.4 by Jelmer Vernooij
Put determine_wants methods on InterRepo.
179
            determine_wants = interrepo.determine_wants_all
0.259.1 by Jelmer Vernooij
Provide custom GitDir.sprout() for bzr 2.4 compatibility.
180
        interrepo.fetch_objects(determine_wants=determine_wants,
181
            mapping=source_branch.mapping)
182
        result_branch = source_branch.sprout(result,
183
            revision_id=revision_id, repository=result_repo)
0.200.1372 by Jelmer Vernooij
Fix formatting.
184
        if (create_tree_if_local
185
            and isinstance(target_transport, LocalTransport)
0.259.1 by Jelmer Vernooij
Provide custom GitDir.sprout() for bzr 2.4 compatibility.
186
            and (result_repo is None or result_repo.make_working_trees())):
187
            wt = result.create_workingtree(accelerator_tree=accelerator_tree,
188
                hardlink=hardlink, from_branch=result_branch)
189
            wt.lock_write()
190
            try:
191
                if wt.path2id('') is None:
192
                    try:
193
                        wt.set_root_id(self.open_workingtree.get_root_id())
194
                    except bzr_errors.NoWorkingTree:
195
                        pass
196
            finally:
197
                wt.unlock()
198
        return result
199
0.200.1117 by Jelmer Vernooij
Provide basic implementation of GitDir.clone_on_transport.
200
    def clone_on_transport(self, transport, revision_id=None,
201
        force_new_repo=False, preserve_stacking=False, stacked_on=None,
202
        create_prefix=False, use_existing_dir=True, no_tree=False):
203
        """See ControlDir.clone_on_transport."""
0.200.1171 by Jelmer Vernooij
Fix some more tests.
204
        from bzrlib.repository import InterRepository
205
        from bzrlib.plugins.git.mapping import default_mapping
0.200.1119 by Jelmer Vernooij
Refactor repository initialization.
206
        if no_tree:
207
            format = BareLocalGitControlDirFormat()
208
        else:
209
            format = LocalGitControlDirFormat()
210
        (target_repo, target_controldir, stacking, repo_policy) = format.initialize_on_transport_ex(transport, use_existing_dir=use_existing_dir, create_prefix=create_prefix, force_new_repo=force_new_repo)
211
        target_git_repo = target_repo._git
0.200.1117 by Jelmer Vernooij
Provide basic implementation of GitDir.clone_on_transport.
212
        source_repo = self.open_repository()
213
        source_git_repo = source_repo._git
0.200.1171 by Jelmer Vernooij
Fix some more tests.
214
        interrepo = InterRepository.get(source_repo, target_repo)
0.200.1117 by Jelmer Vernooij
Provide basic implementation of GitDir.clone_on_transport.
215
        if revision_id is not None:
0.200.1171 by Jelmer Vernooij
Fix some more tests.
216
            determine_wants = interrepo.get_determine_wants_revids([revision_id], include_tags=True)
0.200.1117 by Jelmer Vernooij
Provide basic implementation of GitDir.clone_on_transport.
217
        else:
0.200.1171 by Jelmer Vernooij
Fix some more tests.
218
            determine_wants = interrepo.determine_wants_all
219
        (pack_hint, _, refs) = interrepo.fetch_objects(determine_wants,
220
            mapping=default_mapping)
0.200.1117 by Jelmer Vernooij
Provide basic implementation of GitDir.clone_on_transport.
221
        for name, val in refs.iteritems():
222
            target_git_repo.refs[name] = val
223
        lockfiles = GitLockableFiles(transport, GitLock())
224
        return self.__class__(transport, lockfiles, target_git_repo, format)
225
0.259.2 by Jelmer Vernooij
Make sure RemoteGitDir.find_repository works.
226
    def find_repository(self):
227
        """Find the repository that should be used.
228
229
        This does not require a branch as we use it to find the repo for
230
        new branches as well as to hook existing branches up to their
231
        repository.
232
        """
233
        return self.open_repository()
234
0.200.148 by Jelmer Vernooij
Share more infrastructure between LocalGitDir and RemoteGitDir.
235
0.200.1137 by Jelmer Vernooij
Support BzrProber.known_formats().
236
class LocalGitControlDirFormat(GitControlDirFormat):
237
    """The .git directory control format."""
238
239
    bare = False
240
241
    @classmethod
242
    def _known_formats(self):
243
        return set([LocalGitControlDirFormat()])
244
245
    @property
246
    def repository_format(self):
247
        from bzrlib.plugins.git.repository import GitRepositoryFormat
248
        return GitRepositoryFormat()
249
250
    def get_branch_format(self):
251
        from bzrlib.plugins.git.branch import GitBranchFormat
252
        return GitBranchFormat()
253
254
    def open(self, transport, _found=None):
255
        """Open this directory.
256
257
        """
258
        from bzrlib.plugins.git.transportgit import TransportRepo
0.200.1387 by Jelmer Vernooij
Avoid using HEAD.
259
        gitrepo = TransportRepo(transport, self.bare)
0.200.1137 by Jelmer Vernooij
Support BzrProber.known_formats().
260
        lockfiles = GitLockableFiles(transport, GitLock())
261
        return LocalGitDir(transport, lockfiles, gitrepo, self)
262
263
    def get_format_description(self):
264
        return "Local Git Repository"
265
266
    def initialize_on_transport(self, transport):
267
        from bzrlib.plugins.git.transportgit import TransportRepo
268
        TransportRepo.init(transport, bare=self.bare)
269
        return self.open(transport)
270
271
    def initialize_on_transport_ex(self, transport, use_existing_dir=False,
272
        create_prefix=False, force_new_repo=False, stacked_on=None,
273
        stack_on_pwd=None, repo_format_name=None, make_working_trees=None,
274
        shared_repo=False, vfs_only=False):
275
        def make_directory(transport):
276
            transport.mkdir('.')
277
            return transport
278
        def redirected(transport, e, redirection_notice):
279
            trace.note(redirection_notice)
280
            return transport._redirected_to(e.source, e.target)
281
        try:
282
            transport = do_catching_redirections(make_directory, transport,
283
                redirected)
284
        except bzr_errors.FileExists:
285
            if not use_existing_dir:
286
                raise
287
        except bzr_errors.NoSuchFile:
288
            if not create_prefix:
289
                raise
290
            transport.create_prefix()
291
        controldir = self.initialize_on_transport(transport)
292
        repository = controldir.open_repository()
293
        repository.lock_write()
294
        return (repository, controldir, False, CreateRepository(controldir))
295
296
    def is_supported(self):
297
        return True
298
299
300
class BareLocalGitControlDirFormat(LocalGitControlDirFormat):
301
302
    bare = True
303
    supports_workingtrees = False
304
305
    def get_format_description(self):
306
        return "Local Git Repository (bare)"
307
308
0.200.148 by Jelmer Vernooij
Share more infrastructure between LocalGitDir and RemoteGitDir.
309
class LocalGitDir(GitDir):
310
    """An adapter to the '.git' dir used by git."""
311
0.239.13 by Jelmer Vernooij
Don't break "bzr info -v" when Dulwich is not installed.
312
    def _get_gitrepository_class(self):
313
        from bzrlib.plugins.git.repository import LocalGitRepository
314
        return LocalGitRepository
315
0.200.1313 by Jelmer Vernooij
Add __repr__
316
    def __repr__(self):
317
        return "<%s at %r>" % (
318
            self.__class__.__name__, self.root_transport.base)
319
0.239.13 by Jelmer Vernooij
Don't break "bzr info -v" when Dulwich is not installed.
320
    _gitrepository_class = property(_get_gitrepository_class)
0.202.2 by David Allouche
GitRepository.get_inventory and .revision_tree work for the null revision. Support for testing GitRepository without disk data.
321
0.200.1014 by Jelmer Vernooij
Fix tests.
322
    @property
323
    def user_transport(self):
324
        return self.root_transport
325
326
    @property
327
    def control_transport(self):
328
        return self.transport
329
0.200.90 by Jelmer Vernooij
Basic support for opening working trees.
330
    def __init__(self, transport, lockfiles, gitrepo, format):
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
331
        self._format = format
332
        self.root_transport = transport
0.200.1018 by Jelmer Vernooij
Fix use with new control dir API.
333
        self._mode_check_done = False
0.200.90 by Jelmer Vernooij
Basic support for opening working trees.
334
        self._git = gitrepo
335
        if gitrepo.bare:
336
            self.transport = transport
337
        else:
338
            self.transport = transport.clone('.git')
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
339
        self._lockfiles = lockfiles
0.200.381 by Jelmer Vernooij
Support working trees properly, status and ls.
340
        self._mode_check_done = None
341
342
    def is_control_filename(self, filename):
0.200.1126 by Jelmer Vernooij
Fix GitDir.is_control_filename.
343
        return (filename == '.git' or filename.startswith('.git/'))
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
344
0.200.1311 by Jelmer Vernooij
More work on colocated branch support.
345
    def _get_symref(self, ref):
346
        from dulwich.repo import SYMREF
347
        refcontents = self._git.refs.read_ref(ref)
348
        if refcontents is None: # no such ref
349
            return None
350
        if refcontents.startswith(SYMREF):
351
            return refcontents[len(SYMREF):].rstrip("\n")
352
        return None
353
354
    def set_branch_reference(self, name, target):
355
        ref = self._get_selected_ref(name)
356
        if ref is None:
357
            ref = "HEAD"
358
        if not getattr(target, "ref", None):
359
            raise bzr_errors.BzrError("Can only set symrefs to Git refs")
360
        self._git.refs.set_symbolic_ref(ref, target.ref)
361
362
    def get_branch_reference(self, name=None):
363
        ref = self._get_selected_ref(name)
364
        if ref is None:
365
            ref = "HEAD"
366
        target_ref = self._get_symref(ref)
367
        if target_ref is not None:
0.200.1377 by Jelmer Vernooij
Fix get_branch_reference.
368
            return urlutils.join_segment_parameters(
0.200.1379 by Jelmer Vernooij
Escape slashes.
369
                self.user_url.rstrip("/"), {"ref": urllib.quote(target_ref, '')})
0.200.1311 by Jelmer Vernooij
More work on colocated branch support.
370
        return None
371
372
    def find_branch_format(self, name=None):
373
        from bzrlib.plugins.git.branch import (
374
            GitBranchFormat,
375
            GitSymrefBranchFormat,
376
            )
377
        ref = self._get_selected_ref(name)
378
        if ref is None:
379
            ref = "HEAD"
380
        if self._get_symref(ref) is not None:
381
            return GitSymrefBranchFormat()
382
        else:
383
            return GitBranchFormat()
384
0.200.978 by Jelmer Vernooij
Allow name argument to get_branch_transport to be missing.
385
    def get_branch_transport(self, branch_format, name=None):
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
386
        if branch_format is None:
387
            return self.transport
0.200.1012 by Jelmer Vernooij
Rename BzrDir to ControlDir.
388
        if isinstance(branch_format, LocalGitControlDirFormat):
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
389
            return self.transport
0.239.13 by Jelmer Vernooij
Don't break "bzr info -v" when Dulwich is not installed.
390
        raise bzr_errors.IncompatibleFormat(branch_format, self._format)
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
391
0.200.887 by Jelmer Vernooij
get_branch_transport takes a name argument.
392
    def get_repository_transport(self, format):
393
        if format is None:
394
            return self.transport
0.200.1012 by Jelmer Vernooij
Rename BzrDir to ControlDir.
395
        if isinstance(format, LocalGitControlDirFormat):
0.200.887 by Jelmer Vernooij
get_branch_transport takes a name argument.
396
            return self.transport
397
        raise bzr_errors.IncompatibleFormat(format, self._format)
398
399
    def get_workingtree_transport(self, format):
400
        if format is None:
401
            return self.transport
0.200.1012 by Jelmer Vernooij
Rename BzrDir to ControlDir.
402
        if isinstance(format, LocalGitControlDirFormat):
0.200.887 by Jelmer Vernooij
get_branch_transport takes a name argument.
403
            return self.transport
404
        raise bzr_errors.IncompatibleFormat(format, self._format)
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
405
0.200.1148 by Jelmer Vernooij
Remove no longer necessary compatibility code for open_branch.
406
    def open_branch(self, name=None, unsupported=False, ignore_fallbacks=None):
0.200.57 by Jelmer Vernooij
Fix more tests.
407
        """'create' a branch for this dir."""
408
        repo = self.open_repository()
0.239.13 by Jelmer Vernooij
Don't break "bzr info -v" when Dulwich is not installed.
409
        from bzrlib.plugins.git.branch import LocalGitBranch
0.200.1310 by Jelmer Vernooij
Add _get_selected_ref method.
410
        ref = self._get_selected_ref(name)
0.200.1311 by Jelmer Vernooij
More work on colocated branch support.
411
        if ref is None:
412
            ref = "HEAD"
0.200.1312 by Jelmer Vernooij
Error out on missing branches.
413
        ref, sha = self._git.refs._follow(ref)
414
        if not ref in self._git.refs:
0.200.1311 by Jelmer Vernooij
More work on colocated branch support.
415
            raise bzr_errors.NotBranchError(self.root_transport.base,
416
                    bzrdir=self)
0.200.1310 by Jelmer Vernooij
Add _get_selected_ref method.
417
        return LocalGitBranch(self, repo, ref, self._lockfiles)
0.200.722 by Jelmer Vernooij
Implement GitDir.list_branches() and support name argument to open_branch.
418
0.200.724 by Jelmer Vernooij
support destroy_branch
419
    def destroy_branch(self, name=None):
0.200.1310 by Jelmer Vernooij
Add _get_selected_ref method.
420
        refname = self._get_selected_ref(name)
0.200.1364 by Jelmer Vernooij
Fix .destroy_branch.
421
        if refname is None:
422
            refname = "refs/heads/master"
423
        try:
424
            del self._git.refs[refname]
425
        except KeyError:
0.200.997 by Jelmer Vernooij
Implement BzrDir.needs_format_conversion.
426
            raise bzr_errors.NotBranchError(self.root_transport.base,
427
                    bzrdir=self)
0.200.724 by Jelmer Vernooij
support destroy_branch
428
0.200.980 by Jelmer Vernooij
Implement LocalGitBzrDir.destroy_repository().
429
    def destroy_repository(self):
430
        raise bzr_errors.UnsupportedOperation(self.destroy_repository, self)
431
0.200.986 by Jelmer Vernooij
Implement GitDir.destroy_workingtree.
432
    def destroy_workingtree(self):
433
        raise bzr_errors.UnsupportedOperation(self.destroy_workingtree, self)
434
0.200.997 by Jelmer Vernooij
Implement BzrDir.needs_format_conversion.
435
    def needs_format_conversion(self, format=None):
436
        return not isinstance(self._format, format.__class__)
437
0.200.722 by Jelmer Vernooij
Implement GitDir.list_branches() and support name argument to open_branch.
438
    def list_branches(self):
439
        ret = []
0.200.1313 by Jelmer Vernooij
Add __repr__
440
        for name in self._git.refs.keys():
0.200.832 by Jelmer Vernooij
Update to newer version of Dulwich, saner branch names.
441
            if name.startswith("refs/heads/"):
0.200.766 by Jelmer Vernooij
Only list actual branches, not tags.
442
                ret.append(self.open_branch(name=name))
0.200.722 by Jelmer Vernooij
Implement GitDir.list_branches() and support name argument to open_branch.
443
        return ret
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
444
0.200.1114 by Jelmer Vernooij
Properly raise exception when create_repository is called with shared=True
445
    def open_repository(self):
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
446
        """'open' a repository for this dir."""
0.202.2 by David Allouche
GitRepository.get_inventory and .revision_tree work for the null revision. Support for testing GitRepository without disk data.
447
        return self._gitrepository_class(self, self._lockfiles)
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
448
0.203.1 by Aaron Bentley
Make checkouts work
449
    def open_workingtree(self, recommend_upgrade=True):
0.246.5 by Jelmer Vernooij
Cope with has_index not existing.
450
        if not self._git.bare:
451
            from dulwich.errors import NoIndexPresent
0.200.803 by Jelmer Vernooij
Default to non-bare repositories when initializing a control directory.
452
            repo = self.open_repository()
0.246.5 by Jelmer Vernooij
Cope with has_index not existing.
453
            try:
0.200.803 by Jelmer Vernooij
Default to non-bare repositories when initializing a control directory.
454
                index = repo._git.open_index()
0.246.5 by Jelmer Vernooij
Cope with has_index not existing.
455
            except NoIndexPresent:
456
                pass
0.200.803 by Jelmer Vernooij
Default to non-bare repositories when initializing a control directory.
457
            else:
458
                from bzrlib.plugins.git.workingtree import GitWorkingTree
0.200.921 by Jelmer Vernooij
fix init tests.
459
                try:
460
                    branch = self.open_branch()
461
                except bzr_errors.NotBranchError:
462
                    pass
463
                else:
464
                    return GitWorkingTree(self, repo, branch, index)
0.200.392 by Jelmer Vernooij
Fix some tests now that working trees are supported.
465
        loc = urlutils.unescape_for_display(self.root_transport.base, 'ascii')
0.239.13 by Jelmer Vernooij
Don't break "bzr info -v" when Dulwich is not installed.
466
        raise bzr_errors.NoWorkingTree(loc)
0.200.18 by John Arbash Meinel
Start splitting up the Git{Branch,Dir,Repository} into separate modules, etc.
467
0.200.108 by Jelmer Vernooij
Support bzr init --git.
468
    def create_repository(self, shared=False):
0.200.1114 by Jelmer Vernooij
Properly raise exception when create_repository is called with shared=True
469
        from bzrlib.plugins.git.repository import GitRepositoryFormat
470
        if shared:
471
            raise bzr_errors.IncompatibleFormat(GitRepositoryFormat(), self._format)
0.200.108 by Jelmer Vernooij
Support bzr init --git.
472
        return self.open_repository()
0.200.288 by Jelmer Vernooij
Add test for init-repo.
473
0.200.1377 by Jelmer Vernooij
Fix get_branch_reference.
474
    def create_branch(self, name=None, repository=None,
475
                      append_revisions_only=None):
0.200.1310 by Jelmer Vernooij
Add _get_selected_ref method.
476
        refname = self._get_selected_ref(name)
0.200.891 by Jelmer Vernooij
Use ZERO_SHA constant where possible.
477
        from dulwich.protocol import ZERO_SHA
0.200.1311 by Jelmer Vernooij
More work on colocated branch support.
478
        # FIXME: This is a bit awkward. Perhaps we should have a
479
        # a separate method for changing the default branch?
480
        if refname is None:
481
            refname = "refs/heads/master"
0.200.1373 by Jelmer Vernooij
Prevent accidentally removing branch.
482
            set_head = True
483
        else:
484
            set_head = False
485
486
        if refname in self._git.refs:
487
            raise bzr_errors.AlreadyBranchError(self.base)
488
        self._git.refs[refname] = ZERO_SHA
489
        if set_head:
0.200.1311 by Jelmer Vernooij
More work on colocated branch support.
490
            self._git.refs.set_symbolic_ref("HEAD", refname)
0.200.1377 by Jelmer Vernooij
Fix get_branch_reference.
491
        branch = self.open_branch(name)
0.200.1378 by Jelmer Vernooij
Fix branch.
492
        if append_revisions_only:
0.200.1377 by Jelmer Vernooij
Fix get_branch_reference.
493
            branch.set_append_revisions_only(append_revisions_only)
494
        return branch
0.200.535 by Jelmer Vernooij
use standard version to check for index.
495
496
    def backup_bzrdir(self):
497
        if self._git.bare:
498
            self.root_transport.copy_tree(".git", ".git.backup")
499
            return (self.root_transport.abspath(".git"),
500
                    self.root_transport.abspath(".git.backup"))
501
        else:
0.239.13 by Jelmer Vernooij
Don't break "bzr info -v" when Dulwich is not installed.
502
            raise bzr_errors.BzrError("Unable to backup bare repositories")
0.200.535 by Jelmer Vernooij
use standard version to check for index.
503
504
    def create_workingtree(self, revision_id=None, from_branch=None,
505
        accelerator_tree=None, hardlink=False):
506
        if self._git.bare:
0.200.1038 by Jelmer Vernooij
Raise UnsupportedOperation on create_workingtree.
507
            raise bzr_errors.UnsupportedOperation(self.create_workingtree, self)
0.200.535 by Jelmer Vernooij
use standard version to check for index.
508
        from dulwich.index import write_index
0.200.613 by Jelmer Vernooij
Support creating working tree for existing git repo.
509
        from dulwich.pack import SHA1Writer
510
        f = open(self.transport.local_abspath("index"), 'w+')
511
        try:
512
            f = SHA1Writer(f)
513
            write_index(f, [])
514
        finally:
515
            f.close()
0.200.535 by Jelmer Vernooij
use standard version to check for index.
516
        return self.open_workingtree()
0.200.1015 by Jelmer Vernooij
Fix GitControlDir.find_repository().
517
0.200.1114 by Jelmer Vernooij
Properly raise exception when create_repository is called with shared=True
518
    def _find_or_create_repository(self, force_new_repo=None):
519
        return self.create_repository(shared=False)
520
0.200.1018 by Jelmer Vernooij
Fix use with new control dir API.
521
    def _find_creation_modes(self):
522
        """Determine the appropriate modes for files and directories.
523
524
        They're always set to be consistent with the base directory,
525
        assuming that this transport allows setting modes.
526
        """
527
        # TODO: Do we need or want an option (maybe a config setting) to turn
528
        # this off or override it for particular locations? -- mbp 20080512
529
        if self._mode_check_done:
530
            return
531
        self._mode_check_done = True
532
        try:
533
            st = self.transport.stat('.')
0.200.1116 by Jelmer Vernooij
Fix missing import.
534
        except bzr_errors.TransportNotPossible:
0.200.1018 by Jelmer Vernooij
Fix use with new control dir API.
535
            self._dir_mode = None
536
            self._file_mode = None
537
        else:
538
            # Check the directory mode, but also make sure the created
539
            # directories and files are read-write for this user. This is
540
            # mostly a workaround for filesystems which lie about being able to
541
            # write to a directory (cygwin & win32)
542
            if (st.st_mode & 07777 == 00000):
543
                # FTP allows stat but does not return dir/file modes
544
                self._dir_mode = None
545
                self._file_mode = None
546
            else:
547
                self._dir_mode = (st.st_mode & 07777) | 00700
548
                # Remove the sticky and execute bits for files
549
                self._file_mode = self._dir_mode & ~07111
550
551
    def _get_file_mode(self):
552
        """Return Unix mode for newly created files, or None.
553
        """
554
        if not self._mode_check_done:
555
            self._find_creation_modes()
556
        return self._file_mode
557
558
    def _get_dir_mode(self):
559
        """Return Unix mode for newly created directories, or None.
560
        """
561
        if not self._mode_check_done:
562
            self._find_creation_modes()
563
        return self._dir_mode
564