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