/brz/remove-bazaar

To get this branch, use:
bzr branch http://gegoxaren.bato24.eu/bzr/brz/remove-bazaar
0.358.2 by Jelmer Vernooij
Refresh copyright headers, add my email.
1
# Copyright (C) 2010-2018 Jelmer Vernooij <jelmer@jelmer.uk>
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
2
#
3
# This program is free software; you can redistribute it and/or modify
4
# it under the terms of the GNU General Public License as published by
5
# the Free Software Foundation; either version 2 of the License, or
6
# (at your option) any later version.
7
#
8
# This program is distributed in the hope that it will be useful,
9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
11
# GNU General Public License for more details.
12
#
13
# You should have received a copy of the GNU General Public License
14
# along with this program; if not, write to the Free Software
0.358.1 by Jelmer Vernooij
Fix FSF address.
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
16
17
"""A Git repository implementation that uses a Bazaar transport."""
18
0.200.1594 by Jelmer Vernooij
Use absolute_import everywhere.
19
from __future__ import absolute_import
20
6964.2.1 by Jelmer Vernooij
Initial work to support brz-git on python3.
21
from io import BytesIO
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
22
0.200.1340 by Jelmer Vernooij
Add basic support for alternates.
23
import os
0.284.5 by Jelmer Vernooij
Implement TransportRepo._determine_file_mode.
24
import sys
0.200.1340 by Jelmer Vernooij
Add basic support for alternates.
25
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
26
from dulwich.errors import (
0.246.7 by Jelmer Vernooij
more work.
27
    NoIndexPresent,
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
28
    )
0.410.1 by Jelmer Vernooij
use standard git infrastructure for locking.
29
from dulwich.file import (
30
    GitFile,
31
    FileLocked,
32
    )
0.246.4 by Jelmer Vernooij
more work on transportgit.
33
from dulwich.objects import (
34
    ShaFile,
35
    )
36
from dulwich.object_store import (
37
    PackBasedObjectStore,
38
    PACKDIR,
7268.1.1 by Jelmer Vernooij
Fix compatibility with newer versions of Dulwich.
39
    read_packs_file,
0.246.4 by Jelmer Vernooij
more work on transportgit.
40
    )
41
from dulwich.pack import (
0.200.1003 by Jelmer Vernooij
Initial work on supporting move_in_thin_pack.
42
    MemoryPackIndex,
0.246.7 by Jelmer Vernooij
more work.
43
    PackData,
0.246.4 by Jelmer Vernooij
more work on transportgit.
44
    Pack,
0.200.936 by Jelmer Vernooij
Test transportgit.
45
    iter_sha1,
0.246.7 by Jelmer Vernooij
more work.
46
    load_pack_index_file,
0.399.1 by Jelmer Vernooij
Fix thin pack fetching.
47
    write_pack_objects,
0.200.936 by Jelmer Vernooij
Test transportgit.
48
    write_pack_index_v2,
0.246.4 by Jelmer Vernooij
more work on transportgit.
49
    )
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
50
from dulwich.repo import (
51
    BaseRepo,
0.200.1775 by Jelmer Vernooij
Assume newer dulwich.
52
    InfoRefsContainer,
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
53
    RefsContainer,
0.200.1119 by Jelmer Vernooij
Refactor repository initialization.
54
    BASE_DIRECTORIES,
0.200.1776 by Jelmer Vernooij
Basic support for commondir.
55
    COMMONDIR,
0.398.1 by Jelmer Vernooij
Support reading .git files.
56
    CONTROLDIR,
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
57
    INDEX_FILENAME,
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
58
    OBJECTDIR,
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
59
    SYMREF,
60
    check_ref_format,
61
    read_packed_refs_with_peeled,
62
    read_packed_refs,
63
    write_packed_refs,
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
64
    )
65
6986.2.1 by Jelmer Vernooij
Move breezy.plugins.git to breezy.git.
66
from .. import (
7018.3.1 by Jelmer Vernooij
Fix git cache handling.
67
    osutils,
0.200.1340 by Jelmer Vernooij
Add basic support for alternates.
68
    transport as _mod_transport,
7018.3.2 by Jelmer Vernooij
Fix some git tests.
69
    urlutils,
70
    )
6986.2.1 by Jelmer Vernooij
Move breezy.plugins.git to breezy.git.
71
from ..errors import (
0.200.1718 by Jelmer Vernooij
Support AlreadyControlDirError.
72
    AlreadyControlDirError,
0.200.933 by Jelmer Vernooij
Allow directories to already exist :-)
73
    FileExists,
0.412.1 by Jelmer Vernooij
Implement Branch.break_lock.
74
    LockBroken,
0.410.1 by Jelmer Vernooij
use standard git infrastructure for locking.
75
    LockContention,
76
    NotLocalUrl,
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
77
    NoSuchFile,
0.398.1 by Jelmer Vernooij
Support reading .git files.
78
    ReadError,
0.200.946 by Jelmer Vernooij
Fix reading pack files over http.
79
    TransportNotPossible,
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
80
    )
81
6986.2.1 by Jelmer Vernooij
Move breezy.plugins.git to breezy.git.
82
from ..lock import LogicalLockResult
7411.1.1 by Jelmer Vernooij
Fix support for reading from a dumb git server.
83
from ..trace import warning
0.410.1 by Jelmer Vernooij
use standard git infrastructure for locking.
84
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
85
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
86
class TransportRefsContainer(RefsContainer):
87
    """Refs container that reads refs from a transport."""
88
0.311.3 by Jelmer Vernooij
Fix handling of lightweight checkouts.
89
    def __init__(self, transport, worktree_transport=None):
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
90
        self.transport = transport
0.311.3 by Jelmer Vernooij
Fix handling of lightweight checkouts.
91
        if worktree_transport is None:
92
            worktree_transport = transport
93
        self.worktree_transport = worktree_transport
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
94
        self._packed_refs = None
95
        self._peeled_refs = None
96
97
    def __repr__(self):
98
        return "%s(%r)" % (self.__class__.__name__, self.transport)
99
0.257.2 by Jelmer Vernooij
Fix transportgit.
100
    def _ensure_dir_exists(self, path):
7045.3.1 by Jelmer Vernooij
Fix another ~500 tests.
101
        for n in range(path.count("/")):
7143.15.2 by Jelmer Vernooij
Run autopep8.
102
            dirname = "/".join(path.split("/")[:n + 1])
0.257.2 by Jelmer Vernooij
Fix transportgit.
103
            try:
104
                self.transport.mkdir(dirname)
105
            except FileExists:
106
                pass
107
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
108
    def subkeys(self, base):
0.377.1 by Jelmer Vernooij
Fix some remote operations and add more tests.
109
        """Refs present in this container under a base.
110
111
        :param base: The base to return refs under.
112
        :return: A set of valid refs in this container under the base; the base
113
            prefix is stripped from the ref names returned.
114
        """
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
115
        keys = set()
0.377.1 by Jelmer Vernooij
Fix some remote operations and add more tests.
116
        base_len = len(base) + 1
117
        for refname in self.allkeys():
118
            if refname.startswith(base):
119
                keys.add(refname[base_len:])
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
120
        return keys
121
122
    def allkeys(self):
123
        keys = set()
0.200.1387 by Jelmer Vernooij
Avoid using HEAD.
124
        try:
0.311.3 by Jelmer Vernooij
Fix handling of lightweight checkouts.
125
            self.worktree_transport.get_bytes("HEAD")
0.200.1387 by Jelmer Vernooij
Avoid using HEAD.
126
        except NoSuchFile:
127
            pass
128
        else:
7018.3.2 by Jelmer Vernooij
Fix some git tests.
129
            keys.add(b"HEAD")
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
130
        try:
7143.15.2 by Jelmer Vernooij
Run autopep8.
131
            iter_files = list(self.transport.clone(
132
                "refs").iter_files_recursive())
0.257.2 by Jelmer Vernooij
Fix transportgit.
133
            for filename in iter_files:
7045.4.1 by Jelmer Vernooij
Some brz-git fixes.
134
                unquoted_filename = urlutils.unquote_to_bytes(filename)
7018.3.2 by Jelmer Vernooij
Fix some git tests.
135
                refname = osutils.pathjoin(b"refs", unquoted_filename)
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
136
                if check_ref_format(refname):
137
                    keys.add(refname)
0.257.2 by Jelmer Vernooij
Fix transportgit.
138
        except (TransportNotPossible, NoSuchFile):
139
            pass
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
140
        keys.update(self.get_packed_refs())
141
        return keys
142
143
    def get_packed_refs(self):
144
        """Get contents of the packed-refs file.
145
146
        :return: Dictionary mapping ref names to SHA1s
147
148
        :note: Will return an empty dictionary when no packed-refs file is
149
            present.
150
        """
151
        # TODO: invalidate the cache on repacking
152
        if self._packed_refs is None:
153
            # set both to empty because we want _peeled_refs to be
154
            # None if and only if _packed_refs is also None.
155
            self._packed_refs = {}
156
            self._peeled_refs = {}
157
            try:
158
                f = self.transport.get("packed-refs")
159
            except NoSuchFile:
160
                return {}
161
            try:
6964.2.1 by Jelmer Vernooij
Initial work to support brz-git on python3.
162
                first_line = next(iter(f)).rstrip()
7129.1.1 by Jelmer Vernooij
Fix bytiness.
163
                if (first_line.startswith(b"# pack-refs") and b" peeled" in
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
164
                        first_line):
165
                    for sha, name, peeled in read_packed_refs_with_peeled(f):
166
                        self._packed_refs[name] = sha
167
                        if peeled:
168
                            self._peeled_refs[name] = peeled
169
                else:
170
                    f.seek(0)
171
                    for sha, name in read_packed_refs(f):
172
                        self._packed_refs[name] = sha
173
            finally:
174
                f.close()
175
        return self._packed_refs
176
177
    def get_peeled(self, name):
178
        """Return the cached peeled value of a ref, if available.
179
180
        :param name: Name of the ref to peel
7143.15.3 by Jelmer Vernooij
Fix pep8 issues in breezy.git.
181
        :return: The peeled value of the ref. If the ref is known not point to
182
            a tag, this will be the SHA the ref refers to. If the ref may point
183
            to a tag, but no cached information is available, None is returned.
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
184
        """
185
        self.get_packed_refs()
186
        if self._peeled_refs is None or name not in self._packed_refs:
187
            # No cache: no peeled refs were read, or this ref is loose
188
            return None
189
        if name in self._peeled_refs:
190
            return self._peeled_refs[name]
191
        else:
192
            # Known not peelable
193
            return self[name]
194
195
    def read_loose_ref(self, name):
196
        """Read a reference file and return its contents.
197
198
        If the reference file a symbolic reference, only read the first line of
199
        the file. Otherwise, only read the first 40 bytes.
200
201
        :param name: the refname to read, relative to refpath
202
        :return: The contents of the ref file, or None if the file does not
203
            exist.
204
        :raises IOError: if any other error occurs
205
        """
0.311.3 by Jelmer Vernooij
Fix handling of lightweight checkouts.
206
        if name == b'HEAD':
207
            transport = self.worktree_transport
208
        else:
209
            transport = self.transport
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
210
        try:
7045.3.1 by Jelmer Vernooij
Fix another ~500 tests.
211
            f = transport.get(urlutils.quote_from_bytes(name))
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
212
        except NoSuchFile:
213
            return None
7027.4.7 by Jelmer Vernooij
Fix some tests.
214
        with f:
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
215
            header = f.read(len(SYMREF))
216
            if header == SYMREF:
217
                # Read only the first line
6964.2.1 by Jelmer Vernooij
Initial work to support brz-git on python3.
218
                return header + next(iter(f)).rstrip(b"\r\n")
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
219
            else:
220
                # Read only the first 40 bytes
7143.15.2 by Jelmer Vernooij
Run autopep8.
221
                return header + f.read(40 - len(SYMREF))
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
222
223
    def _remove_packed_ref(self, name):
224
        if self._packed_refs is None:
225
            return
226
        # reread cached refs from disk, while holding the lock
227
228
        self._packed_refs = None
229
        self.get_packed_refs()
230
231
        if name not in self._packed_refs:
232
            return
233
234
        del self._packed_refs[name]
235
        if name in self._peeled_refs:
236
            del self._peeled_refs[name]
7356.1.5 by Jelmer Vernooij
Use more ExitStacks.
237
        with self.transport.open_write_stream("packed-refs") as f:
0.200.954 by Jelmer Vernooij
Avoid writing to memory first unless strictly necessary.
238
            write_packed_refs(f, self._packed_refs, self._peeled_refs)
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
239
240
    def set_symbolic_ref(self, name, other):
241
        """Make a ref point at another ref.
242
243
        :param name: Name of the ref to set
244
        :param other: Name of the ref to point at
245
        """
246
        self._check_refname(name)
247
        self._check_refname(other)
0.311.3 by Jelmer Vernooij
Fix handling of lightweight checkouts.
248
        if name != b'HEAD':
249
            transport = self.transport
7045.3.1 by Jelmer Vernooij
Fix another ~500 tests.
250
            self._ensure_dir_exists(urlutils.quote_from_bytes(name))
0.311.3 by Jelmer Vernooij
Fix handling of lightweight checkouts.
251
        else:
252
            transport = self.worktree_transport
7143.15.2 by Jelmer Vernooij
Run autopep8.
253
        transport.put_bytes(urlutils.quote_from_bytes(
254
            name), SYMREF + other + b'\n')
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
255
256
    def set_if_equals(self, name, old_ref, new_ref):
257
        """Set a refname to new_ref only if it currently equals old_ref.
258
259
        This method follows all symbolic references, and can be used to perform
260
        an atomic compare-and-swap operation.
261
262
        :param name: The refname to set.
263
        :param old_ref: The old sha the refname must refer to, or None to set
264
            unconditionally.
265
        :param new_ref: The new sha the refname will refer to.
266
        :return: True if the set was successful, False otherwise.
267
        """
268
        try:
0.284.2 by Jelmer Vernooij
Use new RefsContainer.follow().
269
            realnames, _ = self.follow(name)
270
            realname = realnames[-1]
271
        except (KeyError, IndexError):
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
272
            realname = name
0.311.3 by Jelmer Vernooij
Fix handling of lightweight checkouts.
273
        if realname == b'HEAD':
274
            transport = self.worktree_transport
275
        else:
276
            transport = self.transport
7045.3.1 by Jelmer Vernooij
Fix another ~500 tests.
277
            self._ensure_dir_exists(urlutils.quote_from_bytes(realname))
7143.15.2 by Jelmer Vernooij
Run autopep8.
278
        transport.put_bytes(urlutils.quote_from_bytes(
279
            realname), new_ref + b"\n")
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
280
        return True
281
282
    def add_if_new(self, name, ref):
283
        """Add a new reference only if it does not already exist.
284
285
        This method follows symrefs, and only ensures that the last ref in the
286
        chain does not exist.
287
288
        :param name: The refname to set.
289
        :param ref: The new sha the refname will refer to.
290
        :return: True if the add was successful, False otherwise.
291
        """
292
        try:
0.284.2 by Jelmer Vernooij
Use new RefsContainer.follow().
293
            realnames, contents = self.follow(name)
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
294
            if contents is not None:
295
                return False
0.284.2 by Jelmer Vernooij
Use new RefsContainer.follow().
296
            realname = realnames[-1]
297
        except (KeyError, IndexError):
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
298
            realname = name
299
        self._check_refname(realname)
0.311.3 by Jelmer Vernooij
Fix handling of lightweight checkouts.
300
        if realname == b'HEAD':
301
            transport = self.worktree_transport
302
        else:
303
            transport = self.transport
7045.3.1 by Jelmer Vernooij
Fix another ~500 tests.
304
            self._ensure_dir_exists(urlutils.quote_from_bytes(realname))
7143.15.2 by Jelmer Vernooij
Run autopep8.
305
        transport.put_bytes(urlutils.quote_from_bytes(realname), ref + b"\n")
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
306
        return True
307
308
    def remove_if_equals(self, name, old_ref):
309
        """Remove a refname only if it currently equals old_ref.
310
311
        This method does not follow symbolic references. It can be used to
312
        perform an atomic compare-and-delete operation.
313
314
        :param name: The refname to delete.
7143.15.3 by Jelmer Vernooij
Fix pep8 issues in breezy.git.
315
        :param old_ref: The old sha the refname must refer to, or None to
316
            delete unconditionally.
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
317
        :return: True if the delete was successful, False otherwise.
318
        """
319
        self._check_refname(name)
320
        # may only be packed
0.311.3 by Jelmer Vernooij
Fix handling of lightweight checkouts.
321
        if name == b'HEAD':
322
            transport = self.worktree_transport
323
        else:
324
            transport = self.transport
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
325
        try:
7045.3.1 by Jelmer Vernooij
Fix another ~500 tests.
326
            transport.delete(urlutils.quote_from_bytes(name))
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
327
        except NoSuchFile:
328
            pass
329
        self._remove_packed_ref(name)
330
        return True
331
0.200.1459 by Jelmer Vernooij
Implement TransportRefsContainer.get.
332
    def get(self, name, default=None):
333
        try:
334
            return self[name]
335
        except KeyError:
336
            return default
337
0.412.1 by Jelmer Vernooij
Implement Branch.break_lock.
338
    def unlock_ref(self, name):
339
        if name == b"HEAD":
340
            transport = self.worktree_transport
341
        else:
342
            transport = self.transport
6973.12.3 by Jelmer Vernooij
Fixes.
343
        lockname = name + b".lock"
0.412.1 by Jelmer Vernooij
Implement Branch.break_lock.
344
        try:
7143.15.3 by Jelmer Vernooij
Fix pep8 issues in breezy.git.
345
            transport.delete(urlutils.quote_from_bytes(lockname))
0.412.1 by Jelmer Vernooij
Implement Branch.break_lock.
346
        except NoSuchFile:
347
            pass
348
0.383.1 by Jelmer Vernooij
Fix a bunch of locking issues.
349
    def lock_ref(self, name):
350
        if name == b"HEAD":
351
            transport = self.worktree_transport
352
        else:
353
            transport = self.transport
7045.3.1 by Jelmer Vernooij
Fix another ~500 tests.
354
        self._ensure_dir_exists(urlutils.quote_from_bytes(name))
355
        lockname = urlutils.quote_from_bytes(name + b".lock")
0.383.2 by Jelmer Vernooij
Fall back to (shudder) racy locking.
356
        try:
7143.15.3 by Jelmer Vernooij
Fix pep8 issues in breezy.git.
357
            local_path = transport.local_abspath(
7143.15.2 by Jelmer Vernooij
Run autopep8.
358
                urlutils.quote_from_bytes(name))
0.410.1 by Jelmer Vernooij
use standard git infrastructure for locking.
359
        except NotLocalUrl:
360
            # This is racy, but what can we do?
7143.15.3 by Jelmer Vernooij
Fix pep8 issues in breezy.git.
361
            if transport.has(lockname):
0.410.1 by Jelmer Vernooij
use standard git infrastructure for locking.
362
                raise LockContention(name)
7143.15.3 by Jelmer Vernooij
Fix pep8 issues in breezy.git.
363
            transport.put_bytes(lockname, b'Locked by brz-git')
364
            return LogicalLockResult(lambda: transport.delete(lockname))
0.410.1 by Jelmer Vernooij
use standard git infrastructure for locking.
365
        else:
366
            try:
367
                gf = GitFile(local_path, 'wb')
368
            except FileLocked as e:
369
                raise LockContention(name, e)
370
            else:
0.412.1 by Jelmer Vernooij
Implement Branch.break_lock.
371
                def unlock():
372
                    try:
7143.15.3 by Jelmer Vernooij
Fix pep8 issues in breezy.git.
373
                        transport.delete(lockname)
0.412.1 by Jelmer Vernooij
Implement Branch.break_lock.
374
                    except NoSuchFile:
375
                        raise LockBroken(lockname)
7143.15.3 by Jelmer Vernooij
Fix pep8 issues in breezy.git.
376
                    # GitFile.abort doesn't care if the lock has already
377
                    # disappeared
0.412.1 by Jelmer Vernooij
Implement Branch.break_lock.
378
                    gf.abort()
379
                return LogicalLockResult(unlock)
0.383.1 by Jelmer Vernooij
Fix a bunch of locking issues.
380
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
381
7045.4.2 by Jelmer Vernooij
Fix some more gitty tests.
382
# TODO(jelmer): Use upstream read_gitfile; unfortunately that expects strings
383
# rather than bytes..
384
def read_gitfile(f):
385
    """Read a ``.git`` file.
386
387
    The first line of the file should start with "gitdir: "
388
389
    :param f: File-like object to read from
390
    :return: A path
391
    """
392
    cs = f.read()
393
    if not cs.startswith(b"gitdir: "):
394
        raise ValueError("Expected file to start with 'gitdir: '")
395
    return cs[len(b"gitdir: "):].rstrip(b"\n")
396
397
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
398
class TransportRepo(BaseRepo):
399
0.200.1485 by Jelmer Vernooij
Keep track of refs text when opening bare repository.
400
    def __init__(self, transport, bare, refs_text=None):
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
401
        self.transport = transport
0.200.1387 by Jelmer Vernooij
Avoid using HEAD.
402
        self.bare = bare
0.398.1 by Jelmer Vernooij
Support reading .git files.
403
        try:
404
            with transport.get(CONTROLDIR) as f:
405
                path = read_gitfile(f)
406
        except (ReadError, NoSuchFile):
407
            if self.bare:
408
                self._controltransport = self.transport
409
            else:
410
                self._controltransport = self.transport.clone('.git')
0.200.1387 by Jelmer Vernooij
Avoid using HEAD.
411
        else:
7143.15.2 by Jelmer Vernooij
Run autopep8.
412
            self._controltransport = self.transport.clone(
413
                urlutils.quote_from_bytes(path))
0.200.1776 by Jelmer Vernooij
Basic support for commondir.
414
        commondir = self.get_named_file(COMMONDIR)
415
        if commondir is not None:
416
            with commondir:
417
                commondir = os.path.join(
418
                    self.controldir(),
419
                    commondir.read().rstrip(b"\r\n").decode(
420
                        sys.getfilesystemencoding()))
421
                self._commontransport = \
422
                    _mod_transport.get_transport_from_path(commondir)
423
        else:
424
            self._commontransport = self._controltransport
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
425
        object_store = TransportObjectStore(
0.200.1776 by Jelmer Vernooij
Basic support for commondir.
426
            self._commontransport.clone(OBJECTDIR))
0.200.1485 by Jelmer Vernooij
Keep track of refs text when opening bare repository.
427
        if refs_text is not None:
6964.2.1 by Jelmer Vernooij
Initial work to support brz-git on python3.
428
            refs_container = InfoRefsContainer(BytesIO(refs_text))
0.200.1572 by Jelmer Vernooij
Fix compatibility for fetching over dumb transport.
429
            try:
7143.15.2 by Jelmer Vernooij
Run autopep8.
430
                head = TransportRefsContainer(
7371.4.4 by Jelmer Vernooij
Pull in more fixes from janitor.
431
                    self._commontransport).read_loose_ref(b"HEAD")
0.200.1572 by Jelmer Vernooij
Fix compatibility for fetching over dumb transport.
432
            except KeyError:
433
                pass
434
            else:
7371.4.4 by Jelmer Vernooij
Pull in more fixes from janitor.
435
                refs_container._refs[b"HEAD"] = head
0.200.1485 by Jelmer Vernooij
Keep track of refs text when opening bare repository.
436
        else:
0.311.3 by Jelmer Vernooij
Fix handling of lightweight checkouts.
437
            refs_container = TransportRefsContainer(
7143.15.2 by Jelmer Vernooij
Run autopep8.
438
                self._commontransport, self._controltransport)
0.200.1636 by Jelmer Vernooij
Some formatting fixes.
439
        super(TransportRepo, self).__init__(object_store,
7143.15.2 by Jelmer Vernooij
Run autopep8.
440
                                            refs_container)
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
441
0.200.1658 by Jelmer Vernooij
Fix handling of ignores - return patterns that matched.
442
    def controldir(self):
443
        return self._controltransport.local_abspath('.')
444
0.404.1 by Jelmer Vernooij
Implement TransportRepo.commondir.
445
    def commondir(self):
446
        return self._commontransport.local_abspath('.')
447
0.200.1658 by Jelmer Vernooij
Fix handling of ignores - return patterns that matched.
448
    @property
449
    def path(self):
450
        return self.transport.local_abspath('.')
451
0.284.5 by Jelmer Vernooij
Implement TransportRepo._determine_file_mode.
452
    def _determine_file_mode(self):
453
        # Be consistent with bzr
454
        if sys.platform == 'win32':
455
            return False
456
        return True
457
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
458
    def get_named_file(self, path):
459
        """Get a file from the control dir with a specific name.
460
461
        Although the filename should be interpreted as a filename relative to
462
        the control dir in a disk-baked Repo, the object returned need not be
463
        pointing to a file in that location.
464
465
        :param path: The path to the file, relative to the control dir.
466
        :return: An open file object, or None if the file does not exist.
467
        """
468
        try:
469
            return self._controltransport.get(path.lstrip('/'))
470
        except NoSuchFile:
471
            return None
472
0.200.1119 by Jelmer Vernooij
Refactor repository initialization.
473
    def _put_named_file(self, relpath, contents):
474
        self._controltransport.put_bytes(relpath, contents)
475
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
476
    def index_path(self):
477
        """Return the path to the index file."""
478
        return self._controltransport.local_abspath(INDEX_FILENAME)
479
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
480
    def open_index(self):
481
        """Open the index for this repository."""
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
482
        from dulwich.index import Index
483
        if not self.has_index():
484
            raise NoIndexPresent()
485
        return Index(self.index_path())
486
487
    def has_index(self):
0.257.2 by Jelmer Vernooij
Fix transportgit.
488
        """Check if an index is present."""
489
        # Bare repos must never have index files; non-bare repos may have a
490
        # missing index file, which is treated as empty.
491
        return not self.bare
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
492
0.200.1546 by Jelmer Vernooij
Provide get_config.
493
    def get_config(self):
0.200.1547 by Jelmer Vernooij
Support setting branch nicks.
494
        from dulwich.config import ConfigFile
495
        try:
7027.4.7 by Jelmer Vernooij
Fix some tests.
496
            with self._controltransport.get('config') as f:
497
                return ConfigFile.from_file(f)
0.200.1547 by Jelmer Vernooij
Support setting branch nicks.
498
        except NoSuchFile:
499
            return ConfigFile()
500
501
    def get_config_stack(self):
502
        from dulwich.config import StackedConfig
0.200.1546 by Jelmer Vernooij
Provide get_config.
503
        backends = []
0.200.1547 by Jelmer Vernooij
Support setting branch nicks.
504
        p = self.get_config()
505
        if p is not None:
506
            backends.append(p)
507
            writable = p
508
        else:
509
            writable = None
0.200.1546 by Jelmer Vernooij
Provide get_config.
510
        backends.extend(StackedConfig.default_backends())
0.200.1547 by Jelmer Vernooij
Support setting branch nicks.
511
        return StackedConfig(backends, writable=writable)
0.200.1546 by Jelmer Vernooij
Provide get_config.
512
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
513
    def __repr__(self):
0.200.1031 by Jelmer Vernooij
Better __repr__.
514
        return "<%s for %r>" % (self.__class__.__name__, self.transport)
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
515
0.200.1119 by Jelmer Vernooij
Refactor repository initialization.
516
    @classmethod
517
    def init(cls, transport, bare=False):
518
        if not bare:
0.200.1718 by Jelmer Vernooij
Support AlreadyControlDirError.
519
            try:
520
                transport.mkdir(".git")
521
            except FileExists:
522
                raise AlreadyControlDirError(transport.base)
0.200.1119 by Jelmer Vernooij
Refactor repository initialization.
523
            control_transport = transport.clone(".git")
524
        else:
525
            control_transport = transport
526
        for d in BASE_DIRECTORIES:
0.200.1718 by Jelmer Vernooij
Support AlreadyControlDirError.
527
            try:
528
                control_transport.mkdir("/".join(d))
529
            except FileExists:
530
                pass
531
        try:
532
            control_transport.mkdir(OBJECTDIR)
533
        except FileExists:
534
            raise AlreadyControlDirError(transport.base)
0.200.1119 by Jelmer Vernooij
Refactor repository initialization.
535
        TransportObjectStore.init(control_transport.clone(OBJECTDIR))
0.200.1387 by Jelmer Vernooij
Avoid using HEAD.
536
        ret = cls(transport, bare)
6964.2.1 by Jelmer Vernooij
Initial work to support brz-git on python3.
537
        ret.refs.set_symbolic_ref(b"HEAD", b"refs/heads/master")
0.200.1119 by Jelmer Vernooij
Refactor repository initialization.
538
        ret._init_files(bare)
539
        return ret
540
0.246.4 by Jelmer Vernooij
more work on transportgit.
541
542
class TransportObjectStore(PackBasedObjectStore):
543
    """Git-style object store that exists on disk."""
544
545
    def __init__(self, transport):
546
        """Open an object store.
547
548
        :param transport: Transport to open data from
549
        """
550
        super(TransportObjectStore, self).__init__()
551
        self.transport = transport
552
        self.pack_transport = self.transport.clone(PACKDIR)
0.200.1340 by Jelmer Vernooij
Add basic support for alternates.
553
        self._alternates = None
0.200.1031 by Jelmer Vernooij
Better __repr__.
554
0.317.1 by Jelmer Vernooij
Implement TransportObjectStore.__eq__.
555
    def __eq__(self, other):
556
        if not isinstance(other, TransportObjectStore):
557
            return False
558
        return self.transport == other.transport
559
0.200.1031 by Jelmer Vernooij
Better __repr__.
560
    def __repr__(self):
561
        return "%s(%r)" % (self.__class__.__name__, self.transport)
562
0.200.1340 by Jelmer Vernooij
Add basic support for alternates.
563
    @property
564
    def alternates(self):
565
        if self._alternates is not None:
566
            return self._alternates
567
        self._alternates = []
568
        for path in self._read_alternate_paths():
569
            # FIXME: Check path
0.200.1342 by Jelmer Vernooij
Add basic support for alternates.
570
            t = _mod_transport.get_transport_from_path(path)
0.200.1340 by Jelmer Vernooij
Add basic support for alternates.
571
            self._alternates.append(self.__class__(t))
572
        return self._alternates
573
574
    def _read_alternate_paths(self):
575
        try:
576
            f = self.transport.get("info/alternates")
577
        except NoSuchFile:
578
            return []
579
        ret = []
7027.4.7 by Jelmer Vernooij
Fix some tests.
580
        with f:
0.200.1572 by Jelmer Vernooij
Fix compatibility for fetching over dumb transport.
581
            for l in f.read().splitlines():
7027.4.7 by Jelmer Vernooij
Fix some tests.
582
                if l[0] == b"#":
0.200.1340 by Jelmer Vernooij
Add basic support for alternates.
583
                    continue
584
                if os.path.isabs(l):
585
                    continue
586
                ret.append(l)
587
            return ret
588
0.281.2 by William Grant
Fix TransportObjectStore to actually have packs with the new cache dict.
589
    def _update_pack_cache(self):
7411.1.1 by Jelmer Vernooij
Fix support for reading from a dumb git server.
590
        pack_files = set(self._pack_names())
7268.1.1 by Jelmer Vernooij
Fix compatibility with newer versions of Dulwich.
591
        new_packs = []
592
        for basename in pack_files:
593
            pack_name = basename + ".pack"
594
            if basename not in self._pack_cache:
595
                try:
596
                    size = self.pack_transport.stat(pack_name).st_size
597
                except TransportNotPossible:
598
                    f = self.pack_transport.get(pack_name)
7411.1.1 by Jelmer Vernooij
Fix support for reading from a dumb git server.
599
                    # TODO(jelmer): Don't read entire file into memory?
600
                    f = BytesIO(f.read())
7268.1.1 by Jelmer Vernooij
Fix compatibility with newer versions of Dulwich.
601
                    pd = PackData(pack_name, f)
602
                else:
603
                    pd = PackData(
604
                        pack_name, self.pack_transport.get(pack_name),
605
                        size=size)
606
                idxname = basename + ".idx"
607
                idx = load_pack_index_file(
608
                    idxname, self.pack_transport.get(idxname))
609
                pack = Pack.from_objects(pd, idx)
610
                pack._basename = basename
611
                self._pack_cache[basename] = pack
612
                new_packs.append(pack)
613
        # Remove disappeared pack files
614
        for f in set(self._pack_cache) - pack_files:
615
            self._pack_cache.pop(f).close()
616
        return new_packs
0.281.2 by William Grant
Fix TransportObjectStore to actually have packs with the new cache dict.
617
0.256.1 by Jelmer Vernooij
Allow using manual listing for pack contents.
618
    def _pack_names(self):
7411.1.1 by Jelmer Vernooij
Fix support for reading from a dumb git server.
619
        pack_files = []
0.256.1 by Jelmer Vernooij
Allow using manual listing for pack contents.
620
        try:
7411.1.1 by Jelmer Vernooij
Fix support for reading from a dumb git server.
621
            dir_contents = self.pack_transport.list_dir(".")
622
            for name in dir_contents:
623
                if name.startswith("pack-") and name.endswith(".pack"):
624
                    # verify that idx exists first (otherwise the pack was not yet
625
                    # fully written)
626
                    idx_name = os.path.splitext(name)[0] + ".idx"
627
                    if idx_name in dir_contents:
628
                        pack_files.append(os.path.splitext(name)[0])
7131.2.1 by Jelmer Vernooij
Read packs file only if directory isn't listable.
629
        except TransportNotPossible:
630
            try:
631
                f = self.transport.get('info/packs')
632
            except NoSuchFile:
7411.1.1 by Jelmer Vernooij
Fix support for reading from a dumb git server.
633
                warning('No info/packs on remote host;'
634
                        'run \'git update-server-info\' on remote.')
7131.2.1 by Jelmer Vernooij
Read packs file only if directory isn't listable.
635
            else:
636
                with f:
7411.1.1 by Jelmer Vernooij
Fix support for reading from a dumb git server.
637
                    pack_files = [
638
                        os.path.splitext(name)[0]
639
                        for name in read_packs_file(f)]
7131.2.1 by Jelmer Vernooij
Read packs file only if directory isn't listable.
640
        except NoSuchFile:
7411.1.1 by Jelmer Vernooij
Fix support for reading from a dumb git server.
641
            pass
642
        return pack_files
0.256.1 by Jelmer Vernooij
Allow using manual listing for pack contents.
643
0.200.1671 by Jelmer Vernooij
Fix pack tests.
644
    def _remove_pack(self, pack):
645
        self.pack_transport.delete(os.path.basename(pack.index.path))
646
        self.pack_transport.delete(pack.data.filename)
7268.1.1 by Jelmer Vernooij
Fix compatibility with newer versions of Dulwich.
647
        try:
648
            del self._pack_cache[os.path.basename(pack._basename)]
649
        except KeyError:
650
            pass
0.246.4 by Jelmer Vernooij
more work on transportgit.
651
652
    def _iter_loose_objects(self):
653
        for base in self.transport.list_dir('.'):
654
            if len(base) != 2:
655
                continue
656
            for rest in self.transport.list_dir(base):
7143.15.2 by Jelmer Vernooij
Run autopep8.
657
                yield (base + rest).encode(sys.getfilesystemencoding())
0.246.4 by Jelmer Vernooij
more work on transportgit.
658
659
    def _split_loose_object(self, sha):
660
        return (sha[:2], sha[2:])
661
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
662
    def _remove_loose_object(self, sha):
7018.3.1 by Jelmer Vernooij
Fix git cache handling.
663
        path = osutils.joinpath(self._split_loose_object(sha))
7045.3.1 by Jelmer Vernooij
Fix another ~500 tests.
664
        self.transport.delete(urlutils.quote_from_bytes(path))
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
665
0.246.4 by Jelmer Vernooij
more work on transportgit.
666
    def _get_loose_object(self, sha):
7018.3.1 by Jelmer Vernooij
Fix git cache handling.
667
        path = osutils.joinpath(self._split_loose_object(sha))
0.246.4 by Jelmer Vernooij
more work on transportgit.
668
        try:
7045.3.1 by Jelmer Vernooij
Fix another ~500 tests.
669
            with self.transport.get(urlutils.quote_from_bytes(path)) as f:
7027.4.7 by Jelmer Vernooij
Fix some tests.
670
                return ShaFile.from_file(f)
0.246.4 by Jelmer Vernooij
more work on transportgit.
671
        except NoSuchFile:
672
            return None
673
674
    def add_object(self, obj):
675
        """Add a single object to this object store.
676
677
        :param obj: Object to add
678
        """
679
        (dir, file) = self._split_loose_object(obj.id)
0.200.933 by Jelmer Vernooij
Allow directories to already exist :-)
680
        try:
7045.3.1 by Jelmer Vernooij
Fix another ~500 tests.
681
            self.transport.mkdir(urlutils.quote_from_bytes(dir))
0.200.933 by Jelmer Vernooij
Allow directories to already exist :-)
682
        except FileExists:
683
            pass
7045.3.1 by Jelmer Vernooij
Fix another ~500 tests.
684
        path = urlutils.quote_from_bytes(osutils.pathjoin(dir, file))
0.246.4 by Jelmer Vernooij
more work on transportgit.
685
        if self.transport.has(path):
7143.15.2 by Jelmer Vernooij
Run autopep8.
686
            return  # Already there, no need to write again
0.246.4 by Jelmer Vernooij
more work on transportgit.
687
        self.transport.put_bytes(path, obj.as_legacy_object())
688
0.200.937 by Jelmer Vernooij
Avoid using os module in transportgit.
689
    def move_in_pack(self, f):
0.200.936 by Jelmer Vernooij
Test transportgit.
690
        """Move a specific file containing a pack into the pack directory.
691
692
        :note: The file should be on the same file system as the
693
            packs directory.
694
695
        :param path: Path to the pack file.
696
        """
0.200.937 by Jelmer Vernooij
Avoid using os module in transportgit.
697
        f.seek(0)
0.200.1682 by Jelmer Vernooij
Fix compatibility with newer versions of dulwich.
698
        p = PackData("", f, len(f.getvalue()))
0.200.936 by Jelmer Vernooij
Test transportgit.
699
        entries = p.sorted_entries()
7143.15.2 by Jelmer Vernooij
Run autopep8.
700
        basename = "pack-%s" % iter_sha1(entry[0]
701
                                         for entry in entries).decode('ascii')
0.200.1671 by Jelmer Vernooij
Fix pack tests.
702
        p._filename = basename + ".pack"
0.200.937 by Jelmer Vernooij
Avoid using os module in transportgit.
703
        f.seek(0)
704
        self.pack_transport.put_file(basename + ".pack", f)
7356.1.5 by Jelmer Vernooij
Use more ExitStacks.
705
        with self.pack_transport.open_write_stream(basename + ".idx") as idxfile:
0.200.954 by Jelmer Vernooij
Avoid writing to memory first unless strictly necessary.
706
            write_pack_index_v2(idxfile, entries, p.get_stored_checksum())
0.399.2 by Jelmer Vernooij
Fix tests.
707
        idxfile = self.pack_transport.get(basename + ".idx")
7143.15.2 by Jelmer Vernooij
Run autopep8.
708
        idx = load_pack_index_file(basename + ".idx", idxfile)
0.399.2 by Jelmer Vernooij
Fix tests.
709
        final_pack = Pack.from_objects(p, idx)
710
        final_pack._basename = basename
7268.1.1 by Jelmer Vernooij
Fix compatibility with newer versions of Dulwich.
711
        self._add_cached_pack(basename, final_pack)
0.399.2 by Jelmer Vernooij
Fix tests.
712
        return final_pack
0.200.1003 by Jelmer Vernooij
Initial work on supporting move_in_thin_pack.
713
714
    def move_in_thin_pack(self, f):
715
        """Move a specific file containing a pack into the pack directory.
716
717
        :note: The file should be on the same file system as the
718
            packs directory.
719
720
        :param path: Path to the pack file.
721
        """
722
        f.seek(0)
0.399.1 by Jelmer Vernooij
Fix thin pack fetching.
723
        p = Pack('', resolve_ext_ref=self.get_raw)
724
        p._data = PackData.from_file(f, len(f.getvalue()))
725
        p._data.pack = p
7143.15.2 by Jelmer Vernooij
Run autopep8.
726
        p._idx_load = lambda: MemoryPackIndex(
727
            p.data.sorted_entries(), p.data.get_stored_checksum())
0.200.1003 by Jelmer Vernooij
Initial work on supporting move_in_thin_pack.
728
0.399.1 by Jelmer Vernooij
Fix thin pack fetching.
729
        pack_sha = p.index.objects_sha1()
0.200.1003 by Jelmer Vernooij
Initial work on supporting move_in_thin_pack.
730
7356.1.5 by Jelmer Vernooij
Use more ExitStacks.
731
        with self.pack_transport.open_write_stream(
732
                "pack-%s.pack" % pack_sha.decode('ascii')) as datafile:
0.399.1 by Jelmer Vernooij
Fix thin pack fetching.
733
            entries, data_sum = write_pack_objects(datafile, p.pack_tuples())
734
        entries = sorted([(k, v[0], v[1]) for (k, v) in entries.items()])
7356.1.5 by Jelmer Vernooij
Use more ExitStacks.
735
        with self.pack_transport.open_write_stream(
736
                "pack-%s.idx" % pack_sha.decode('ascii')) as idxfile:
0.399.1 by Jelmer Vernooij
Fix thin pack fetching.
737
            write_pack_index_v2(idxfile, entries, data_sum)
0.200.1003 by Jelmer Vernooij
Initial work on supporting move_in_thin_pack.
738
0.246.4 by Jelmer Vernooij
more work on transportgit.
739
    def add_pack(self):
0.200.1636 by Jelmer Vernooij
Some formatting fixes.
740
        """Add a new pack to this object store.
0.246.4 by Jelmer Vernooij
more work on transportgit.
741
0.200.1636 by Jelmer Vernooij
Some formatting fixes.
742
        :return: Fileobject to write to and a commit function to
0.246.4 by Jelmer Vernooij
more work on transportgit.
743
            call when the pack is finished.
744
        """
6964.2.1 by Jelmer Vernooij
Initial work to support brz-git on python3.
745
        f = BytesIO()
7143.15.2 by Jelmer Vernooij
Run autopep8.
746
0.200.936 by Jelmer Vernooij
Test transportgit.
747
        def commit():
0.200.937 by Jelmer Vernooij
Avoid using os module in transportgit.
748
            if len(f.getvalue()) > 0:
749
                return self.move_in_pack(f)
0.200.936 by Jelmer Vernooij
Test transportgit.
750
            else:
751
                return None
7143.15.2 by Jelmer Vernooij
Run autopep8.
752
0.200.1626 by Jelmer Vernooij
Fix compatibility with dulwich 0.9.1.
753
        def abort():
754
            return None
755
        return f, commit, abort
0.200.934 by Jelmer Vernooij
Add init function.
756
757
    @classmethod
758
    def init(cls, transport):
0.200.1718 by Jelmer Vernooij
Support AlreadyControlDirError.
759
        try:
760
            transport.mkdir('info')
761
        except FileExists:
762
            pass
763
        try:
764
            transport.mkdir(PACKDIR)
765
        except FileExists:
766
            pass
0.200.934 by Jelmer Vernooij
Add init function.
767
        return cls(transport)