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