/brz/remove-bazaar

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