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