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