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