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