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