18
18
from bzrlib import urlutils
19
19
from bzrlib.bzrdir import BzrDir, BzrDirFormat
20
from bzrlib.errors import BzrError, NoSuchFile, NotLocalUrl
20
from bzrlib.errors import NoSuchFile, NotLocalUrl
21
21
from bzrlib.lockable_files import TransportLock
22
22
from bzrlib.repository import Repository
23
23
from bzrlib.trace import info
24
24
from bzrlib.transport import Transport
26
from bzrlib.plugins.git import lazy_check_versions
26
from bzrlib.plugins.git import git
29
27
from bzrlib.plugins.git.branch import GitBranch
30
from bzrlib.plugins.git.errors import NoSuchRef
31
28
from bzrlib.plugins.git.dir import GitDir
32
29
from bzrlib.plugins.git.foreign import ForeignBranch
33
30
from bzrlib.plugins.git.repository import GitFormat, GitRepository
41
from dulwich.errors import GitProtocolError
42
from dulwich.pack import PackData, Pack, PackIndex
44
# Don't run any tests on GitSmartTransport as it is not intended to be
45
# a full implementation of Transport
46
def get_test_permutations():
35
from dulwich.pack import PackData
50
38
class GitSmartTransport(Transport):
55
43
assert scheme == "git"
56
44
hostport, self._path = urllib.splithost(loc)
57
45
(self._host, self._port) = urllib.splitnport(hostport, git.protocol.TCP_GIT_PORT)
58
self._client = _client
60
def has(self, relpath):
63
def _get_client(self):
64
if self._client is not None:
68
return git.client.TCPGitClient(self._host, self._port,
69
capabilities=["multi_ack", "side-band-64k", "ofs-delta", "side-band"])
46
if _client is not None:
47
self._client = _client
49
self._client = git.client.TCPGitClient(self._host, self._port)
71
51
def fetch_pack(self, determine_wants, graph_walker, pack_data, progress=None):
72
52
if progress is None:
73
53
def progress(text):
74
54
info("git: %s" % text)
75
client = self._get_client()
55
self._client.fetch_pack(self._path, determine_wants, graph_walker,
58
def fetch_objects(self, determine_wants, graph_walker, progress=None):
59
fd, path = tempfile.mkstemp(dir=self.pack_dir(), suffix=".pack")
60
self.fetch_pack(determine_wants, graph_walker, lambda x: os.write(fd, x), progress)
77
client.fetch_pack(self._path, determine_wants,
78
graph_walker, pack_data, progress)
79
except GitProtocolError, e:
64
for o in p.iterobjects():
82
69
def get(self, path):
83
70
raise NoSuchFile(path)
85
def abspath(self, relpath):
86
return urlutils.join(self.base, relpath)
88
72
def clone(self, offset=None):
89
73
"""See Transport.clone()."""
106
90
def open_repository(self):
107
91
return RemoteGitRepository(self, self._lockfiles)
109
def open_branch(self, _unsupported=False):
93
def open_branch(self):
110
94
repo = self.open_repository()
111
95
# TODO: Support for multiple branches in one bzrdir in bzrlib!
112
96
return RemoteGitBranch(self, repo, "HEAD", self._lockfiles)
115
99
raise NotLocalUrl(self.transport.base)
118
class EmptyObjectStoreIterator(dict):
120
def iterobjects(self):
124
class TemporaryPackIterator(Pack):
126
def __init__(self, path, resolve_ext_ref):
127
self.resolve_ext_ref = resolve_ext_ref
128
super(TemporaryPackIterator, self).__init__(path)
132
if self._idx is None:
133
self._data.create_index_v2(self._idx_path, self.resolve_ext_ref)
134
self._idx = PackIndex(self._idx_path)
138
os.remove(self._data_path)
139
os.remove(self._idx_path)
142
102
class RemoteGitRepository(GitRepository):
144
104
def __init__(self, gitdir, lockfiles):
149
109
self._transport.fetch_pack(determine_wants, graph_walker, pack_data,
152
def fetch_objects(self, determine_wants, graph_walker, resolve_ext_ref, progress=None):
153
fd, path = tempfile.mkstemp(suffix=".pack")
154
self.fetch_pack(determine_wants, graph_walker, lambda x: os.write(fd, x), progress)
156
if os.path.getsize(path) == 0:
157
return EmptyObjectStoreIterator()
158
return TemporaryPackIterator(path[:-len(".pack")], resolve_ext_ref)
161
113
class RemoteGitBranch(GitBranch):
163
115
def __init__(self, bzrdir, repository, name, lockfiles):
164
116
def determine_wants(heads):
165
if not name in heads:
166
raise NoSuchRef(name)
167
117
self._ref = heads[name]
168
118
bzrdir.root_transport.fetch_pack(determine_wants, None, lambda x: None,
169
119
lambda x: mutter("git: %s" % x))
172
122
def last_revision(self):
173
123
return self.mapping.revision_id_foreign_to_bzr(self._ref)
175
def _synchronize_history(self, destination, revision_id):
176
"""See Branch._synchronize_history()."""
177
destination.generate_revision_history(self.last_revision())