/brz/remove-bazaar

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