1
# Copyright (C) 2006-2009 Canonical Ltd
3
# Authors: Robert Collins <robert.collins@canonical.com>
4
# Jelmer Vernooij <jelmer@samba.org>
5
# John Carr <john.carr@unrouted.co.uk>
7
# This program is free software; you can redistribute it and/or modify
8
# it under the terms of the GNU General Public License as published by
9
# the Free Software Foundation; either version 2 of the License, or
10
# (at your option) any later version.
12
# This program is distributed in the hope that it will be useful,
13
# but WITHOUT ANY WARRANTY; without even the implied warranty of
14
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15
# GNU General Public License for more details.
17
# You should have received a copy of the GNU General Public License
18
# along with this program; if not, write to the Free Software
19
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
22
"""A GIT branch and repository format implementation for bzr."""
31
bzr_compatible_versions,
32
bzr_plugin_version as version_info,
33
dulwich_minimum_version,
36
if version_info[3] == 'final':
37
version_string = '%d.%d.%d' % version_info[:3]
39
version_string = '%d.%d.%d%s%d' % version_info
40
__version__ = version_string
42
bzrlib.api.require_any_api(bzrlib, bzr_compatible_versions)
50
from bzrlib.controldir import (
58
from bzrlib.bzrdir import (
64
ControlDirFormat = BzrDirFormat
65
has_controldir = False
69
from bzrlib.foreign import (
72
from bzrlib.help_topics import (
75
from bzrlib.lockable_files import (
78
from bzrlib.transport import (
79
register_lazy_transport,
80
register_transport_proto,
82
from bzrlib.commands import (
85
from bzrlib.version_info_formats.format_rio import (
86
RioVersionInfoBuilder,
88
from bzrlib.send import (
89
format_registry as send_format_registry,
93
if getattr(sys, "frozen", None):
94
# allow import additional libs from ./_lib for bzr.exe only
95
sys.path.append(os.path.normpath(
96
os.path.join(os.path.dirname(__file__), '_lib')))
101
from dulwich import __version__ as dulwich_version
103
raise bzr_errors.DependencyNotPresent("dulwich",
104
"bzr-git: Please install dulwich, https://launchpad.net/dulwich")
106
if dulwich_version < dulwich_minimum_version:
107
raise bzr_errors.DependencyNotPresent("dulwich",
108
"bzr-git: Dulwich is too old; at least %d.%d.%d is required" %
109
dulwich_minimum_version)
112
_versions_checked = False
113
def lazy_check_versions():
114
global _versions_checked
115
if _versions_checked:
118
_versions_checked = True
120
format_registry.register_lazy('git',
121
"bzrlib.plugins.git.dir", "LocalGitControlDirFormat",
122
help='GIT repository.', native=False, experimental=True,
125
from bzrlib.revisionspec import revspec_registry
126
revspec_registry.register_lazy("git:", "bzrlib.plugins.git.revspec",
130
from bzrlib.revisionspec import dwim_revspecs
134
from bzrlib.plugins.git.revspec import RevisionSpec_git
135
dwim_revspecs.append(RevisionSpec_git)
138
class GitControlDirFormat(ControlDirFormat):
140
_lock_class = TransportLock
142
colocated_branches = True
144
def __eq__(self, other):
145
return type(self) == type(other)
147
def is_supported(self):
150
def network_name(self):
154
class LocalGitControlDirFormat(GitControlDirFormat):
155
"""The .git directory control format."""
158
def _known_formats(self):
159
return set([LocalGitControlDirFormat()])
161
def open(self, transport, _found=None):
162
"""Open this directory.
165
lazy_check_versions()
166
from bzrlib.plugins.git.transportgit import TransportRepo
167
gitrepo = TransportRepo(transport)
168
from bzrlib.plugins.git.dir import LocalGitDir, GitLockableFiles, GitLock
169
lockfiles = GitLockableFiles(transport, GitLock())
170
return LocalGitDir(transport, lockfiles, gitrepo, self)
173
def probe_transport(klass, transport):
175
if not transport.has_any(['info/refs', '.git/branches',
177
raise bzr_errors.NotBranchError(path=transport.base)
178
except bzr_errors.NoSuchFile:
179
raise bzr_errors.NotBranchError(path=transport.base)
180
from bzrlib import urlutils
181
if urlutils.split(transport.base)[1] == ".git":
182
raise bzr_errors.NotBranchError(path=transport.base)
183
lazy_check_versions()
187
format.open(transport)
189
except dulwich.errors.NotGitRepository, e:
190
raise bzr_errors.NotBranchError(path=transport.base)
191
raise bzr_errors.NotBranchError(path=transport.base)
193
def get_format_description(self):
194
return "Local Git Repository"
196
def get_format_string(self):
197
return "Local Git Repository"
199
def initialize_on_transport(self, transport):
200
from bzrlib.transport.local import LocalTransport
202
if not isinstance(transport, LocalTransport):
203
raise NotImplementedError(self.initialize,
204
"Can't create Git Repositories/branches on "
205
"non-local transports")
206
lazy_check_versions()
207
from dulwich.repo import Repo
208
Repo.init(transport.local_abspath(".").encode(osutils._fs_enc))
209
return self.open(transport)
211
def is_supported(self):
215
class RemoteGitControlDirFormat(GitControlDirFormat):
216
"""The .git directory control format."""
219
def _known_formats(self):
220
return set([RemoteGitControlDirFormat()])
222
def open(self, transport, _found=None):
223
"""Open this directory.
226
# we dont grok readonly - git isn't integrated with transport.
228
if url.startswith('readonly+'):
229
url = url[len('readonly+'):]
230
if (not url.startswith("git://") and not url.startswith("git+")):
231
raise bzr_errors.NotBranchError(transport.base)
232
from bzrlib.plugins.git.remote import RemoteGitDir, GitSmartTransport
233
if not isinstance(transport, GitSmartTransport):
234
raise bzr_errors.NotBranchError(transport.base)
235
from bzrlib.plugins.git.dir import GitLockableFiles, GitLock
236
lockfiles = GitLockableFiles(transport, GitLock())
237
return RemoteGitDir(transport, lockfiles, self)
240
def probe_transport(klass, transport):
241
"""Our format is present if the transport ends in '.not/'."""
243
if url.startswith('readonly+'):
244
url = url[len('readonly+'):]
245
if (not url.startswith("git://") and not url.startswith("git+")):
246
raise bzr_errors.NotBranchError(transport.base)
247
# little ugly, but works
249
from bzrlib.plugins.git.remote import GitSmartTransport
250
if not isinstance(transport, GitSmartTransport):
251
raise bzr_errors.NotBranchError(transport.base)
254
def get_format_description(self):
255
return "Remote Git Repository"
257
def get_format_string(self):
258
return "Remote Git Repository"
260
def initialize_on_transport(self, transport):
261
raise bzr_errors.UninitializableFormat(self)
264
ControlDirFormat.register_control_format(LocalGitControlDirFormat)
265
ControlDirFormat.register_control_format(RemoteGitControlDirFormat)
267
register_transport_proto('git://',
268
help="Access using the Git smart server protocol.")
269
register_transport_proto('git+ssh://',
270
help="Access using the Git smart server protocol over SSH.")
272
register_lazy_transport("git://", 'bzrlib.plugins.git.remote',
273
'TCPGitSmartTransport')
274
register_lazy_transport("git+ssh://", 'bzrlib.plugins.git.remote',
275
'SSHGitSmartTransport')
277
foreign_vcs_registry.register_lazy("git",
278
"bzrlib.plugins.git.mapping", "foreign_git", "Stupid content tracker")
280
plugin_cmds.register_lazy("cmd_git_import", [], "bzrlib.plugins.git.commands")
281
plugin_cmds.register_lazy("cmd_git_object", ["git-objects", "git-cat"],
282
"bzrlib.plugins.git.commands")
283
plugin_cmds.register_lazy("cmd_git_refs", [], "bzrlib.plugins.git.commands")
284
plugin_cmds.register_lazy("cmd_git_apply", [], "bzrlib.plugins.git.commands")
286
def update_stanza(rev, stanza):
287
mapping = getattr(rev, "mapping", None)
288
if mapping is not None and mapping.revid_prefix.startswith("git-"):
289
stanza.add("git-commit", rev.foreign_revid)
292
rio_hooks = getattr(RioVersionInfoBuilder, "hooks", None)
293
if rio_hooks is not None:
294
rio_hooks.install_named_hook('revision', update_stanza, None)
297
from bzrlib.transport import transport_server_registry
298
transport_server_registry.register_lazy('git',
299
'bzrlib.plugins.git.server',
301
'Git Smart server protocol over TCP. (default port: 9418)')
304
from bzrlib.repository import (
305
network_format_registry as repository_network_format_registry,
307
repository_network_format_registry.register_lazy('git',
308
'bzrlib.plugins.git.repository', 'GitRepositoryFormat')
310
from bzrlib.bzrdir import (
311
network_format_registry as bzrdir_network_format_registry,
313
bzrdir_network_format_registry.register('git', GitControlDirFormat)
315
send_format_registry.register_lazy('git', 'bzrlib.plugins.git.send',
316
'send_git', 'Git am-style diff format')
318
topic_registry.register_lazy('git',
319
'bzrlib.plugins.git.help',
320
'help_git', 'Using Bazaar with Git')
323
from bzrlib.diff import format_registry as diff_format_registry
327
diff_format_registry.register_lazy('git', 'bzrlib.plugins.git.send',
328
'GitDiffTree', 'Git am-style diff format')
331
from bzrlib.plugins.git import tests
332
return tests.test_suite()