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