/brz/remove-bazaar

To get this branch, use:
bzr branch http://gegoxaren.bato24.eu/bzr/brz/remove-bazaar
0.358.2 by Jelmer Vernooij
Refresh copyright headers, add my email.
1
# Copyright (C) 2010-2018 Jelmer Vernooij <jelmer@jelmer.uk>
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
0.358.1 by Jelmer Vernooij
Fix FSF address.
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
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
6964.2.1 by Jelmer Vernooij
Initial work to support brz-git on python3.
21
from io import BytesIO
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
22
0.200.1340 by Jelmer Vernooij
Add basic support for alternates.
23
import os
0.284.5 by Jelmer Vernooij
Implement TransportRepo._determine_file_mode.
24
import sys
0.200.1340 by Jelmer Vernooij
Add basic support for alternates.
25
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
26
from dulwich.errors import (
0.246.7 by Jelmer Vernooij
more work.
27
    NoIndexPresent,
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
28
    )
0.410.1 by Jelmer Vernooij
use standard git infrastructure for locking.
29
from dulwich.file import (
30
    GitFile,
31
    FileLocked,
32
    )
0.246.4 by Jelmer Vernooij
more work on transportgit.
33
from dulwich.objects import (
34
    ShaFile,
35
    )
36
from dulwich.object_store import (
37
    PackBasedObjectStore,
38
    PACKDIR,
7268.1.1 by Jelmer Vernooij
Fix compatibility with newer versions of Dulwich.
39
    read_packs_file,
0.246.4 by Jelmer Vernooij
more work on transportgit.
40
    )
41
from dulwich.pack import (
0.200.1003 by Jelmer Vernooij
Initial work on supporting move_in_thin_pack.
42
    MemoryPackIndex,
0.246.7 by Jelmer Vernooij
more work.
43
    PackData,
0.246.4 by Jelmer Vernooij
more work on transportgit.
44
    Pack,
0.200.936 by Jelmer Vernooij
Test transportgit.
45
    iter_sha1,
0.246.7 by Jelmer Vernooij
more work.
46
    load_pack_index_file,
0.399.1 by Jelmer Vernooij
Fix thin pack fetching.
47
    write_pack_objects,
0.200.936 by Jelmer Vernooij
Test transportgit.
48
    write_pack_index_v2,
0.246.4 by Jelmer Vernooij
more work on transportgit.
49
    )
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
50
from dulwich.repo import (
51
    BaseRepo,
0.200.1775 by Jelmer Vernooij
Assume newer dulwich.
52
    InfoRefsContainer,
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
53
    RefsContainer,
0.200.1119 by Jelmer Vernooij
Refactor repository initialization.
54
    BASE_DIRECTORIES,
0.200.1776 by Jelmer Vernooij
Basic support for commondir.
55
    COMMONDIR,
0.398.1 by Jelmer Vernooij
Support reading .git files.
56
    CONTROLDIR,
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
57
    INDEX_FILENAME,
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
58
    OBJECTDIR,
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
59
    SYMREF,
60
    check_ref_format,
61
    read_packed_refs_with_peeled,
62
    read_packed_refs,
63
    write_packed_refs,
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
64
    )
65
6986.2.1 by Jelmer Vernooij
Move breezy.plugins.git to breezy.git.
66
from .. import (
7018.3.1 by Jelmer Vernooij
Fix git cache handling.
67
    osutils,
0.200.1340 by Jelmer Vernooij
Add basic support for alternates.
68
    transport as _mod_transport,
7018.3.2 by Jelmer Vernooij
Fix some git tests.
69
    urlutils,
70
    )
6986.2.1 by Jelmer Vernooij
Move breezy.plugins.git to breezy.git.
71
from ..errors import (
0.200.1718 by Jelmer Vernooij
Support AlreadyControlDirError.
72
    AlreadyControlDirError,
0.200.933 by Jelmer Vernooij
Allow directories to already exist :-)
73
    FileExists,
0.412.1 by Jelmer Vernooij
Implement Branch.break_lock.
74
    LockBroken,
0.410.1 by Jelmer Vernooij
use standard git infrastructure for locking.
75
    LockContention,
76
    NotLocalUrl,
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
77
    NoSuchFile,
0.398.1 by Jelmer Vernooij
Support reading .git files.
78
    ReadError,
0.200.946 by Jelmer Vernooij
Fix reading pack files over http.
79
    TransportNotPossible,
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
80
    )
81
6986.2.1 by Jelmer Vernooij
Move breezy.plugins.git to breezy.git.
82
from ..lock import LogicalLockResult
7411.1.1 by Jelmer Vernooij
Fix support for reading from a dumb git server.
83
from ..trace import warning
0.410.1 by Jelmer Vernooij
use standard git infrastructure for locking.
84
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
85
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
86
class TransportRefsContainer(RefsContainer):
87
    """Refs container that reads refs from a transport."""
88
0.311.3 by Jelmer Vernooij
Fix handling of lightweight checkouts.
89
    def __init__(self, transport, worktree_transport=None):
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
90
        self.transport = transport
0.311.3 by Jelmer Vernooij
Fix handling of lightweight checkouts.
91
        if worktree_transport is None:
92
            worktree_transport = transport
93
        self.worktree_transport = worktree_transport
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
94
        self._packed_refs = None
95
        self._peeled_refs = None
96
97
    def __repr__(self):
98
        return "%s(%r)" % (self.__class__.__name__, self.transport)
99
0.257.2 by Jelmer Vernooij
Fix transportgit.
100
    def _ensure_dir_exists(self, path):
7045.3.1 by Jelmer Vernooij
Fix another ~500 tests.
101
        for n in range(path.count("/")):
7143.15.2 by Jelmer Vernooij
Run autopep8.
102
            dirname = "/".join(path.split("/")[:n + 1])
0.257.2 by Jelmer Vernooij
Fix transportgit.
103
            try:
104
                self.transport.mkdir(dirname)
105
            except FileExists:
106
                pass
107
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
108
    def subkeys(self, base):
0.377.1 by Jelmer Vernooij
Fix some remote operations and add more tests.
109
        """Refs present in this container under a base.
110
111
        :param base: The base to return refs under.
112
        :return: A set of valid refs in this container under the base; the base
113
            prefix is stripped from the ref names returned.
114
        """
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
115
        keys = set()
0.377.1 by Jelmer Vernooij
Fix some remote operations and add more tests.
116
        base_len = len(base) + 1
117
        for refname in self.allkeys():
118
            if refname.startswith(base):
119
                keys.add(refname[base_len:])
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
120
        return keys
121
122
    def allkeys(self):
123
        keys = set()
0.200.1387 by Jelmer Vernooij
Avoid using HEAD.
124
        try:
0.311.3 by Jelmer Vernooij
Fix handling of lightweight checkouts.
125
            self.worktree_transport.get_bytes("HEAD")
0.200.1387 by Jelmer Vernooij
Avoid using HEAD.
126
        except NoSuchFile:
127
            pass
128
        else:
7018.3.2 by Jelmer Vernooij
Fix some git tests.
129
            keys.add(b"HEAD")
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
130
        try:
7143.15.2 by Jelmer Vernooij
Run autopep8.
131
            iter_files = list(self.transport.clone(
132
                "refs").iter_files_recursive())
0.257.2 by Jelmer Vernooij
Fix transportgit.
133
            for filename in iter_files:
7045.4.1 by Jelmer Vernooij
Some brz-git fixes.
134
                unquoted_filename = urlutils.unquote_to_bytes(filename)
7018.3.2 by Jelmer Vernooij
Fix some git tests.
135
                refname = osutils.pathjoin(b"refs", unquoted_filename)
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
136
                if check_ref_format(refname):
137
                    keys.add(refname)
0.257.2 by Jelmer Vernooij
Fix transportgit.
138
        except (TransportNotPossible, NoSuchFile):
139
            pass
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
140
        keys.update(self.get_packed_refs())
141
        return keys
142
143
    def get_packed_refs(self):
144
        """Get contents of the packed-refs file.
145
146
        :return: Dictionary mapping ref names to SHA1s
147
148
        :note: Will return an empty dictionary when no packed-refs file is
149
            present.
150
        """
151
        # TODO: invalidate the cache on repacking
152
        if self._packed_refs is None:
153
            # set both to empty because we want _peeled_refs to be
154
            # None if and only if _packed_refs is also None.
155
            self._packed_refs = {}
156
            self._peeled_refs = {}
157
            try:
158
                f = self.transport.get("packed-refs")
159
            except NoSuchFile:
160
                return {}
161
            try:
6964.2.1 by Jelmer Vernooij
Initial work to support brz-git on python3.
162
                first_line = next(iter(f)).rstrip()
7129.1.1 by Jelmer Vernooij
Fix bytiness.
163
                if (first_line.startswith(b"# pack-refs") and b" peeled" in
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
164
                        first_line):
165
                    for sha, name, peeled in read_packed_refs_with_peeled(f):
166
                        self._packed_refs[name] = sha
167
                        if peeled:
168
                            self._peeled_refs[name] = peeled
169
                else:
170
                    f.seek(0)
171
                    for sha, name in read_packed_refs(f):
172
                        self._packed_refs[name] = sha
173
            finally:
174
                f.close()
175
        return self._packed_refs
176
177
    def get_peeled(self, name):
178
        """Return the cached peeled value of a ref, if available.
179
180
        :param name: Name of the ref to peel
7143.15.3 by Jelmer Vernooij
Fix pep8 issues in breezy.git.
181
        :return: The peeled value of the ref. If the ref is known not point to
182
            a tag, this will be the SHA the ref refers to. If the ref may point
183
            to a tag, but no cached information is available, None is returned.
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
184
        """
185
        self.get_packed_refs()
186
        if self._peeled_refs is None or name not in self._packed_refs:
187
            # No cache: no peeled refs were read, or this ref is loose
188
            return None
189
        if name in self._peeled_refs:
190
            return self._peeled_refs[name]
191
        else:
192
            # Known not peelable
193
            return self[name]
194
195
    def read_loose_ref(self, name):
196
        """Read a reference file and return its contents.
197
198
        If the reference file a symbolic reference, only read the first line of
199
        the file. Otherwise, only read the first 40 bytes.
200
201
        :param name: the refname to read, relative to refpath
202
        :return: The contents of the ref file, or None if the file does not
203
            exist.
204
        :raises IOError: if any other error occurs
205
        """
0.311.3 by Jelmer Vernooij
Fix handling of lightweight checkouts.
206
        if name == b'HEAD':
207
            transport = self.worktree_transport
208
        else:
209
            transport = self.transport
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
210
        try:
7045.3.1 by Jelmer Vernooij
Fix another ~500 tests.
211
            f = transport.get(urlutils.quote_from_bytes(name))
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
212
        except NoSuchFile:
213
            return None
7027.4.7 by Jelmer Vernooij
Fix some tests.
214
        with f:
7290.44.1 by Jelmer Vernooij
Don't throw an error attempting to read the parent of a ref.
215
            try:
216
                header = f.read(len(SYMREF))
217
            except ReadError:
218
                # probably a directory
219
                return None
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
220
            if header == SYMREF:
221
                # Read only the first line
6964.2.1 by Jelmer Vernooij
Initial work to support brz-git on python3.
222
                return header + next(iter(f)).rstrip(b"\r\n")
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
223
            else:
224
                # Read only the first 40 bytes
7143.15.2 by Jelmer Vernooij
Run autopep8.
225
                return header + f.read(40 - len(SYMREF))
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
226
227
    def _remove_packed_ref(self, name):
228
        if self._packed_refs is None:
229
            return
230
        # reread cached refs from disk, while holding the lock
231
232
        self._packed_refs = None
233
        self.get_packed_refs()
234
235
        if name not in self._packed_refs:
236
            return
237
238
        del self._packed_refs[name]
239
        if name in self._peeled_refs:
240
            del self._peeled_refs[name]
7356.1.5 by Jelmer Vernooij
Use more ExitStacks.
241
        with self.transport.open_write_stream("packed-refs") as f:
0.200.954 by Jelmer Vernooij
Avoid writing to memory first unless strictly necessary.
242
            write_packed_refs(f, self._packed_refs, self._peeled_refs)
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
243
244
    def set_symbolic_ref(self, name, other):
245
        """Make a ref point at another ref.
246
247
        :param name: Name of the ref to set
248
        :param other: Name of the ref to point at
249
        """
250
        self._check_refname(name)
251
        self._check_refname(other)
0.311.3 by Jelmer Vernooij
Fix handling of lightweight checkouts.
252
        if name != b'HEAD':
253
            transport = self.transport
7045.3.1 by Jelmer Vernooij
Fix another ~500 tests.
254
            self._ensure_dir_exists(urlutils.quote_from_bytes(name))
0.311.3 by Jelmer Vernooij
Fix handling of lightweight checkouts.
255
        else:
256
            transport = self.worktree_transport
7143.15.2 by Jelmer Vernooij
Run autopep8.
257
        transport.put_bytes(urlutils.quote_from_bytes(
258
            name), SYMREF + other + b'\n')
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
259
260
    def set_if_equals(self, name, old_ref, new_ref):
261
        """Set a refname to new_ref only if it currently equals old_ref.
262
263
        This method follows all symbolic references, and can be used to perform
264
        an atomic compare-and-swap operation.
265
266
        :param name: The refname to set.
267
        :param old_ref: The old sha the refname must refer to, or None to set
268
            unconditionally.
269
        :param new_ref: The new sha the refname will refer to.
270
        :return: True if the set was successful, False otherwise.
271
        """
272
        try:
0.284.2 by Jelmer Vernooij
Use new RefsContainer.follow().
273
            realnames, _ = self.follow(name)
274
            realname = realnames[-1]
275
        except (KeyError, IndexError):
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
276
            realname = name
0.311.3 by Jelmer Vernooij
Fix handling of lightweight checkouts.
277
        if realname == b'HEAD':
278
            transport = self.worktree_transport
279
        else:
280
            transport = self.transport
7045.3.1 by Jelmer Vernooij
Fix another ~500 tests.
281
            self._ensure_dir_exists(urlutils.quote_from_bytes(realname))
7143.15.2 by Jelmer Vernooij
Run autopep8.
282
        transport.put_bytes(urlutils.quote_from_bytes(
283
            realname), new_ref + b"\n")
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
284
        return True
285
286
    def add_if_new(self, name, ref):
287
        """Add a new reference only if it does not already exist.
288
289
        This method follows symrefs, and only ensures that the last ref in the
290
        chain does not exist.
291
292
        :param name: The refname to set.
293
        :param ref: The new sha the refname will refer to.
294
        :return: True if the add was successful, False otherwise.
295
        """
296
        try:
0.284.2 by Jelmer Vernooij
Use new RefsContainer.follow().
297
            realnames, contents = self.follow(name)
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
298
            if contents is not None:
299
                return False
0.284.2 by Jelmer Vernooij
Use new RefsContainer.follow().
300
            realname = realnames[-1]
301
        except (KeyError, IndexError):
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
302
            realname = name
303
        self._check_refname(realname)
0.311.3 by Jelmer Vernooij
Fix handling of lightweight checkouts.
304
        if realname == b'HEAD':
305
            transport = self.worktree_transport
306
        else:
307
            transport = self.transport
7045.3.1 by Jelmer Vernooij
Fix another ~500 tests.
308
            self._ensure_dir_exists(urlutils.quote_from_bytes(realname))
7143.15.2 by Jelmer Vernooij
Run autopep8.
309
        transport.put_bytes(urlutils.quote_from_bytes(realname), ref + b"\n")
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
310
        return True
311
312
    def remove_if_equals(self, name, old_ref):
313
        """Remove a refname only if it currently equals old_ref.
314
315
        This method does not follow symbolic references. It can be used to
316
        perform an atomic compare-and-delete operation.
317
318
        :param name: The refname to delete.
7143.15.3 by Jelmer Vernooij
Fix pep8 issues in breezy.git.
319
        :param old_ref: The old sha the refname must refer to, or None to
320
            delete unconditionally.
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
321
        :return: True if the delete was successful, False otherwise.
322
        """
323
        self._check_refname(name)
324
        # may only be packed
0.311.3 by Jelmer Vernooij
Fix handling of lightweight checkouts.
325
        if name == b'HEAD':
326
            transport = self.worktree_transport
327
        else:
328
            transport = self.transport
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
329
        try:
7045.3.1 by Jelmer Vernooij
Fix another ~500 tests.
330
            transport.delete(urlutils.quote_from_bytes(name))
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
331
        except NoSuchFile:
332
            pass
333
        self._remove_packed_ref(name)
334
        return True
335
0.200.1459 by Jelmer Vernooij
Implement TransportRefsContainer.get.
336
    def get(self, name, default=None):
337
        try:
338
            return self[name]
339
        except KeyError:
340
            return default
341
0.412.1 by Jelmer Vernooij
Implement Branch.break_lock.
342
    def unlock_ref(self, name):
343
        if name == b"HEAD":
344
            transport = self.worktree_transport
345
        else:
346
            transport = self.transport
6973.12.3 by Jelmer Vernooij
Fixes.
347
        lockname = name + b".lock"
0.412.1 by Jelmer Vernooij
Implement Branch.break_lock.
348
        try:
7143.15.3 by Jelmer Vernooij
Fix pep8 issues in breezy.git.
349
            transport.delete(urlutils.quote_from_bytes(lockname))
0.412.1 by Jelmer Vernooij
Implement Branch.break_lock.
350
        except NoSuchFile:
351
            pass
352
0.383.1 by Jelmer Vernooij
Fix a bunch of locking issues.
353
    def lock_ref(self, name):
354
        if name == b"HEAD":
355
            transport = self.worktree_transport
356
        else:
357
            transport = self.transport
7045.3.1 by Jelmer Vernooij
Fix another ~500 tests.
358
        self._ensure_dir_exists(urlutils.quote_from_bytes(name))
359
        lockname = urlutils.quote_from_bytes(name + b".lock")
0.383.2 by Jelmer Vernooij
Fall back to (shudder) racy locking.
360
        try:
7143.15.3 by Jelmer Vernooij
Fix pep8 issues in breezy.git.
361
            local_path = transport.local_abspath(
7143.15.2 by Jelmer Vernooij
Run autopep8.
362
                urlutils.quote_from_bytes(name))
0.410.1 by Jelmer Vernooij
use standard git infrastructure for locking.
363
        except NotLocalUrl:
364
            # This is racy, but what can we do?
7143.15.3 by Jelmer Vernooij
Fix pep8 issues in breezy.git.
365
            if transport.has(lockname):
0.410.1 by Jelmer Vernooij
use standard git infrastructure for locking.
366
                raise LockContention(name)
7143.15.3 by Jelmer Vernooij
Fix pep8 issues in breezy.git.
367
            transport.put_bytes(lockname, b'Locked by brz-git')
368
            return LogicalLockResult(lambda: transport.delete(lockname))
0.410.1 by Jelmer Vernooij
use standard git infrastructure for locking.
369
        else:
370
            try:
371
                gf = GitFile(local_path, 'wb')
372
            except FileLocked as e:
373
                raise LockContention(name, e)
374
            else:
0.412.1 by Jelmer Vernooij
Implement Branch.break_lock.
375
                def unlock():
376
                    try:
7143.15.3 by Jelmer Vernooij
Fix pep8 issues in breezy.git.
377
                        transport.delete(lockname)
0.412.1 by Jelmer Vernooij
Implement Branch.break_lock.
378
                    except NoSuchFile:
379
                        raise LockBroken(lockname)
7143.15.3 by Jelmer Vernooij
Fix pep8 issues in breezy.git.
380
                    # GitFile.abort doesn't care if the lock has already
381
                    # disappeared
0.412.1 by Jelmer Vernooij
Implement Branch.break_lock.
382
                    gf.abort()
383
                return LogicalLockResult(unlock)
0.383.1 by Jelmer Vernooij
Fix a bunch of locking issues.
384
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
385
7045.4.2 by Jelmer Vernooij
Fix some more gitty tests.
386
# TODO(jelmer): Use upstream read_gitfile; unfortunately that expects strings
387
# rather than bytes..
388
def read_gitfile(f):
389
    """Read a ``.git`` file.
390
391
    The first line of the file should start with "gitdir: "
392
393
    :param f: File-like object to read from
394
    :return: A path
395
    """
396
    cs = f.read()
397
    if not cs.startswith(b"gitdir: "):
398
        raise ValueError("Expected file to start with 'gitdir: '")
399
    return cs[len(b"gitdir: "):].rstrip(b"\n")
400
401
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
402
class TransportRepo(BaseRepo):
403
0.200.1485 by Jelmer Vernooij
Keep track of refs text when opening bare repository.
404
    def __init__(self, transport, bare, refs_text=None):
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
405
        self.transport = transport
0.200.1387 by Jelmer Vernooij
Avoid using HEAD.
406
        self.bare = bare
0.398.1 by Jelmer Vernooij
Support reading .git files.
407
        try:
408
            with transport.get(CONTROLDIR) as f:
409
                path = read_gitfile(f)
410
        except (ReadError, NoSuchFile):
411
            if self.bare:
412
                self._controltransport = self.transport
413
            else:
414
                self._controltransport = self.transport.clone('.git')
0.200.1387 by Jelmer Vernooij
Avoid using HEAD.
415
        else:
7143.15.2 by Jelmer Vernooij
Run autopep8.
416
            self._controltransport = self.transport.clone(
417
                urlutils.quote_from_bytes(path))
0.200.1776 by Jelmer Vernooij
Basic support for commondir.
418
        commondir = self.get_named_file(COMMONDIR)
419
        if commondir is not None:
420
            with commondir:
421
                commondir = os.path.join(
422
                    self.controldir(),
423
                    commondir.read().rstrip(b"\r\n").decode(
424
                        sys.getfilesystemencoding()))
425
                self._commontransport = \
426
                    _mod_transport.get_transport_from_path(commondir)
427
        else:
428
            self._commontransport = self._controltransport
7490.19.1 by Jelmer Vernooij
Add support for compression parameters in Git.
429
        config = self.get_config()
430
        object_store = TransportObjectStore.from_config(
431
            self._commontransport.clone(OBJECTDIR),
432
            config)
0.200.1485 by Jelmer Vernooij
Keep track of refs text when opening bare repository.
433
        if refs_text is not None:
6964.2.1 by Jelmer Vernooij
Initial work to support brz-git on python3.
434
            refs_container = InfoRefsContainer(BytesIO(refs_text))
0.200.1572 by Jelmer Vernooij
Fix compatibility for fetching over dumb transport.
435
            try:
7143.15.2 by Jelmer Vernooij
Run autopep8.
436
                head = TransportRefsContainer(
7371.4.4 by Jelmer Vernooij
Pull in more fixes from janitor.
437
                    self._commontransport).read_loose_ref(b"HEAD")
0.200.1572 by Jelmer Vernooij
Fix compatibility for fetching over dumb transport.
438
            except KeyError:
439
                pass
440
            else:
7371.4.4 by Jelmer Vernooij
Pull in more fixes from janitor.
441
                refs_container._refs[b"HEAD"] = head
0.200.1485 by Jelmer Vernooij
Keep track of refs text when opening bare repository.
442
        else:
0.311.3 by Jelmer Vernooij
Fix handling of lightweight checkouts.
443
            refs_container = TransportRefsContainer(
7143.15.2 by Jelmer Vernooij
Run autopep8.
444
                self._commontransport, self._controltransport)
0.200.1636 by Jelmer Vernooij
Some formatting fixes.
445
        super(TransportRepo, self).__init__(object_store,
7143.15.2 by Jelmer Vernooij
Run autopep8.
446
                                            refs_container)
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
447
0.200.1658 by Jelmer Vernooij
Fix handling of ignores - return patterns that matched.
448
    def controldir(self):
449
        return self._controltransport.local_abspath('.')
450
0.404.1 by Jelmer Vernooij
Implement TransportRepo.commondir.
451
    def commondir(self):
452
        return self._commontransport.local_abspath('.')
453
0.200.1658 by Jelmer Vernooij
Fix handling of ignores - return patterns that matched.
454
    @property
455
    def path(self):
456
        return self.transport.local_abspath('.')
457
0.284.5 by Jelmer Vernooij
Implement TransportRepo._determine_file_mode.
458
    def _determine_file_mode(self):
459
        # Be consistent with bzr
460
        if sys.platform == 'win32':
461
            return False
462
        return True
463
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
464
    def get_named_file(self, path):
465
        """Get a file from the control dir with a specific name.
466
467
        Although the filename should be interpreted as a filename relative to
468
        the control dir in a disk-baked Repo, the object returned need not be
469
        pointing to a file in that location.
470
471
        :param path: The path to the file, relative to the control dir.
472
        :return: An open file object, or None if the file does not exist.
473
        """
474
        try:
475
            return self._controltransport.get(path.lstrip('/'))
476
        except NoSuchFile:
477
            return None
478
0.200.1119 by Jelmer Vernooij
Refactor repository initialization.
479
    def _put_named_file(self, relpath, contents):
480
        self._controltransport.put_bytes(relpath, contents)
481
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
482
    def index_path(self):
483
        """Return the path to the index file."""
484
        return self._controltransport.local_abspath(INDEX_FILENAME)
485
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
486
    def open_index(self):
487
        """Open the index for this repository."""
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
488
        from dulwich.index import Index
489
        if not self.has_index():
490
            raise NoIndexPresent()
491
        return Index(self.index_path())
492
493
    def has_index(self):
0.257.2 by Jelmer Vernooij
Fix transportgit.
494
        """Check if an index is present."""
495
        # Bare repos must never have index files; non-bare repos may have a
496
        # missing index file, which is treated as empty.
497
        return not self.bare
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
498
0.200.1546 by Jelmer Vernooij
Provide get_config.
499
    def get_config(self):
0.200.1547 by Jelmer Vernooij
Support setting branch nicks.
500
        from dulwich.config import ConfigFile
501
        try:
7027.4.7 by Jelmer Vernooij
Fix some tests.
502
            with self._controltransport.get('config') as f:
503
                return ConfigFile.from_file(f)
0.200.1547 by Jelmer Vernooij
Support setting branch nicks.
504
        except NoSuchFile:
505
            return ConfigFile()
506
507
    def get_config_stack(self):
508
        from dulwich.config import StackedConfig
0.200.1546 by Jelmer Vernooij
Provide get_config.
509
        backends = []
0.200.1547 by Jelmer Vernooij
Support setting branch nicks.
510
        p = self.get_config()
511
        if p is not None:
512
            backends.append(p)
513
            writable = p
514
        else:
515
            writable = None
0.200.1546 by Jelmer Vernooij
Provide get_config.
516
        backends.extend(StackedConfig.default_backends())
0.200.1547 by Jelmer Vernooij
Support setting branch nicks.
517
        return StackedConfig(backends, writable=writable)
0.200.1546 by Jelmer Vernooij
Provide get_config.
518
7490.26.2 by Jelmer Vernooij
More shallow work.
519
    # Here for compatibility with dulwich < 0.19.17
520
    def generate_pack_data(self, have, want, progress=None, ofs_delta=None):
521
        """Generate pack data objects for a set of wants/haves.
522
523
        Args:
524
          have: List of SHA1s of objects that should not be sent
525
          want: List of SHA1s of objects that should be sent
526
          ofs_delta: Whether OFS deltas can be included
527
          progress: Optional progress reporting method
528
        """
7490.26.4 by Jelmer Vernooij
Fix flake.
529
        shallow = self.get_shallow()
7490.26.2 by Jelmer Vernooij
More shallow work.
530
        if shallow:
531
            return self.object_store.generate_pack_data(
532
                have, want, shallow=shallow,
533
                progress=progress, ofs_delta=ofs_delta)
534
        else:
535
            return self.object_store.generate_pack_data(
536
                have, want, progress=progress, ofs_delta=ofs_delta)
537
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
538
    def __repr__(self):
0.200.1031 by Jelmer Vernooij
Better __repr__.
539
        return "<%s for %r>" % (self.__class__.__name__, self.transport)
0.246.1 by Jelmer Vernooij
Add TransportRepo class.
540
0.200.1119 by Jelmer Vernooij
Refactor repository initialization.
541
    @classmethod
542
    def init(cls, transport, bare=False):
543
        if not bare:
0.200.1718 by Jelmer Vernooij
Support AlreadyControlDirError.
544
            try:
545
                transport.mkdir(".git")
546
            except FileExists:
547
                raise AlreadyControlDirError(transport.base)
0.200.1119 by Jelmer Vernooij
Refactor repository initialization.
548
            control_transport = transport.clone(".git")
549
        else:
550
            control_transport = transport
551
        for d in BASE_DIRECTORIES:
0.200.1718 by Jelmer Vernooij
Support AlreadyControlDirError.
552
            try:
553
                control_transport.mkdir("/".join(d))
554
            except FileExists:
555
                pass
556
        try:
557
            control_transport.mkdir(OBJECTDIR)
558
        except FileExists:
559
            raise AlreadyControlDirError(transport.base)
0.200.1119 by Jelmer Vernooij
Refactor repository initialization.
560
        TransportObjectStore.init(control_transport.clone(OBJECTDIR))
0.200.1387 by Jelmer Vernooij
Avoid using HEAD.
561
        ret = cls(transport, bare)
6964.2.1 by Jelmer Vernooij
Initial work to support brz-git on python3.
562
        ret.refs.set_symbolic_ref(b"HEAD", b"refs/heads/master")
0.200.1119 by Jelmer Vernooij
Refactor repository initialization.
563
        ret._init_files(bare)
564
        return ret
565
0.246.4 by Jelmer Vernooij
more work on transportgit.
566
567
class TransportObjectStore(PackBasedObjectStore):
568
    """Git-style object store that exists on disk."""
569
7490.19.1 by Jelmer Vernooij
Add support for compression parameters in Git.
570
    def __init__(self, transport,
571
                 loose_compression_level=-1, pack_compression_level=-1):
0.246.4 by Jelmer Vernooij
more work on transportgit.
572
        """Open an object store.
573
574
        :param transport: Transport to open data from
575
        """
576
        super(TransportObjectStore, self).__init__()
7490.19.1 by Jelmer Vernooij
Add support for compression parameters in Git.
577
        self.pack_compression_level = pack_compression_level
578
        self.loose_compression_level = loose_compression_level
0.246.4 by Jelmer Vernooij
more work on transportgit.
579
        self.transport = transport
580
        self.pack_transport = self.transport.clone(PACKDIR)
0.200.1340 by Jelmer Vernooij
Add basic support for alternates.
581
        self._alternates = None
0.200.1031 by Jelmer Vernooij
Better __repr__.
582
7490.19.1 by Jelmer Vernooij
Add support for compression parameters in Git.
583
    @classmethod
584
    def from_config(cls, path, config):
585
        try:
586
            default_compression_level = int(config.get(
587
                (b'core', ), b'compression').decode())
588
        except KeyError:
589
            default_compression_level = -1
590
        try:
591
            loose_compression_level = int(config.get(
592
                (b'core', ), b'looseCompression').decode())
593
        except KeyError:
594
            loose_compression_level = default_compression_level
595
        try:
596
            pack_compression_level = int(config.get(
597
                (b'core', ), 'packCompression').decode())
598
        except KeyError:
599
            pack_compression_level = default_compression_level
600
        return cls(path, loose_compression_level, pack_compression_level)
601
0.317.1 by Jelmer Vernooij
Implement TransportObjectStore.__eq__.
602
    def __eq__(self, other):
603
        if not isinstance(other, TransportObjectStore):
604
            return False
605
        return self.transport == other.transport
606
0.200.1031 by Jelmer Vernooij
Better __repr__.
607
    def __repr__(self):
608
        return "%s(%r)" % (self.__class__.__name__, self.transport)
609
0.200.1340 by Jelmer Vernooij
Add basic support for alternates.
610
    @property
611
    def alternates(self):
612
        if self._alternates is not None:
613
            return self._alternates
614
        self._alternates = []
615
        for path in self._read_alternate_paths():
616
            # FIXME: Check path
0.200.1342 by Jelmer Vernooij
Add basic support for alternates.
617
            t = _mod_transport.get_transport_from_path(path)
0.200.1340 by Jelmer Vernooij
Add basic support for alternates.
618
            self._alternates.append(self.__class__(t))
619
        return self._alternates
620
621
    def _read_alternate_paths(self):
622
        try:
623
            f = self.transport.get("info/alternates")
624
        except NoSuchFile:
625
            return []
626
        ret = []
7027.4.7 by Jelmer Vernooij
Fix some tests.
627
        with f:
0.200.1572 by Jelmer Vernooij
Fix compatibility for fetching over dumb transport.
628
            for l in f.read().splitlines():
7027.4.7 by Jelmer Vernooij
Fix some tests.
629
                if l[0] == b"#":
0.200.1340 by Jelmer Vernooij
Add basic support for alternates.
630
                    continue
631
                if os.path.isabs(l):
632
                    continue
633
                ret.append(l)
634
            return ret
635
0.281.2 by William Grant
Fix TransportObjectStore to actually have packs with the new cache dict.
636
    def _update_pack_cache(self):
7411.1.1 by Jelmer Vernooij
Fix support for reading from a dumb git server.
637
        pack_files = set(self._pack_names())
7268.1.1 by Jelmer Vernooij
Fix compatibility with newer versions of Dulwich.
638
        new_packs = []
639
        for basename in pack_files:
640
            pack_name = basename + ".pack"
641
            if basename not in self._pack_cache:
642
                try:
643
                    size = self.pack_transport.stat(pack_name).st_size
644
                except TransportNotPossible:
645
                    f = self.pack_transport.get(pack_name)
7411.1.1 by Jelmer Vernooij
Fix support for reading from a dumb git server.
646
                    # TODO(jelmer): Don't read entire file into memory?
647
                    f = BytesIO(f.read())
7268.1.1 by Jelmer Vernooij
Fix compatibility with newer versions of Dulwich.
648
                    pd = PackData(pack_name, f)
649
                else:
650
                    pd = PackData(
651
                        pack_name, self.pack_transport.get(pack_name),
652
                        size=size)
653
                idxname = basename + ".idx"
654
                idx = load_pack_index_file(
655
                    idxname, self.pack_transport.get(idxname))
656
                pack = Pack.from_objects(pd, idx)
657
                pack._basename = basename
658
                self._pack_cache[basename] = pack
659
                new_packs.append(pack)
660
        # Remove disappeared pack files
661
        for f in set(self._pack_cache) - pack_files:
662
            self._pack_cache.pop(f).close()
663
        return new_packs
0.281.2 by William Grant
Fix TransportObjectStore to actually have packs with the new cache dict.
664
0.256.1 by Jelmer Vernooij
Allow using manual listing for pack contents.
665
    def _pack_names(self):
7411.1.1 by Jelmer Vernooij
Fix support for reading from a dumb git server.
666
        pack_files = []
0.256.1 by Jelmer Vernooij
Allow using manual listing for pack contents.
667
        try:
7411.1.1 by Jelmer Vernooij
Fix support for reading from a dumb git server.
668
            dir_contents = self.pack_transport.list_dir(".")
669
            for name in dir_contents:
670
                if name.startswith("pack-") and name.endswith(".pack"):
671
                    # verify that idx exists first (otherwise the pack was not yet
672
                    # fully written)
673
                    idx_name = os.path.splitext(name)[0] + ".idx"
674
                    if idx_name in dir_contents:
675
                        pack_files.append(os.path.splitext(name)[0])
7131.2.1 by Jelmer Vernooij
Read packs file only if directory isn't listable.
676
        except TransportNotPossible:
677
            try:
678
                f = self.transport.get('info/packs')
679
            except NoSuchFile:
7411.1.1 by Jelmer Vernooij
Fix support for reading from a dumb git server.
680
                warning('No info/packs on remote host;'
681
                        'run \'git update-server-info\' on remote.')
7131.2.1 by Jelmer Vernooij
Read packs file only if directory isn't listable.
682
            else:
683
                with f:
7411.1.1 by Jelmer Vernooij
Fix support for reading from a dumb git server.
684
                    pack_files = [
685
                        os.path.splitext(name)[0]
686
                        for name in read_packs_file(f)]
7131.2.1 by Jelmer Vernooij
Read packs file only if directory isn't listable.
687
        except NoSuchFile:
7411.1.1 by Jelmer Vernooij
Fix support for reading from a dumb git server.
688
            pass
689
        return pack_files
0.256.1 by Jelmer Vernooij
Allow using manual listing for pack contents.
690
0.200.1671 by Jelmer Vernooij
Fix pack tests.
691
    def _remove_pack(self, pack):
692
        self.pack_transport.delete(os.path.basename(pack.index.path))
693
        self.pack_transport.delete(pack.data.filename)
7268.1.1 by Jelmer Vernooij
Fix compatibility with newer versions of Dulwich.
694
        try:
695
            del self._pack_cache[os.path.basename(pack._basename)]
696
        except KeyError:
697
            pass
0.246.4 by Jelmer Vernooij
more work on transportgit.
698
699
    def _iter_loose_objects(self):
700
        for base in self.transport.list_dir('.'):
701
            if len(base) != 2:
702
                continue
703
            for rest in self.transport.list_dir(base):
7143.15.2 by Jelmer Vernooij
Run autopep8.
704
                yield (base + rest).encode(sys.getfilesystemencoding())
0.246.4 by Jelmer Vernooij
more work on transportgit.
705
706
    def _split_loose_object(self, sha):
707
        return (sha[:2], sha[2:])
708
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
709
    def _remove_loose_object(self, sha):
7018.3.1 by Jelmer Vernooij
Fix git cache handling.
710
        path = osutils.joinpath(self._split_loose_object(sha))
7045.3.1 by Jelmer Vernooij
Fix another ~500 tests.
711
        self.transport.delete(urlutils.quote_from_bytes(path))
0.257.1 by Jelmer Vernooij
use transport repo objects even for local access.
712
0.246.4 by Jelmer Vernooij
more work on transportgit.
713
    def _get_loose_object(self, sha):
7018.3.1 by Jelmer Vernooij
Fix git cache handling.
714
        path = osutils.joinpath(self._split_loose_object(sha))
0.246.4 by Jelmer Vernooij
more work on transportgit.
715
        try:
7045.3.1 by Jelmer Vernooij
Fix another ~500 tests.
716
            with self.transport.get(urlutils.quote_from_bytes(path)) as f:
7027.4.7 by Jelmer Vernooij
Fix some tests.
717
                return ShaFile.from_file(f)
0.246.4 by Jelmer Vernooij
more work on transportgit.
718
        except NoSuchFile:
719
            return None
720
721
    def add_object(self, obj):
722
        """Add a single object to this object store.
723
724
        :param obj: Object to add
725
        """
726
        (dir, file) = self._split_loose_object(obj.id)
0.200.933 by Jelmer Vernooij
Allow directories to already exist :-)
727
        try:
7045.3.1 by Jelmer Vernooij
Fix another ~500 tests.
728
            self.transport.mkdir(urlutils.quote_from_bytes(dir))
0.200.933 by Jelmer Vernooij
Allow directories to already exist :-)
729
        except FileExists:
730
            pass
7045.3.1 by Jelmer Vernooij
Fix another ~500 tests.
731
        path = urlutils.quote_from_bytes(osutils.pathjoin(dir, file))
0.246.4 by Jelmer Vernooij
more work on transportgit.
732
        if self.transport.has(path):
7143.15.2 by Jelmer Vernooij
Run autopep8.
733
            return  # Already there, no need to write again
7490.19.1 by Jelmer Vernooij
Add support for compression parameters in Git.
734
        # Backwards compatibility with Dulwich < 0.20, which doesn't support
735
        # the compression_level parameter.
736
        if self.loose_compression_level not in (-1, None):
737
            raw_string = obj.as_legacy_object(
738
                compression_level=self.loose_compression_level)
739
        else:
740
            raw_string = obj.as_legacy_object()
741
        self.transport.put_bytes(path, raw_string)
0.246.4 by Jelmer Vernooij
more work on transportgit.
742
0.200.937 by Jelmer Vernooij
Avoid using os module in transportgit.
743
    def move_in_pack(self, f):
0.200.936 by Jelmer Vernooij
Test transportgit.
744
        """Move a specific file containing a pack into the pack directory.
745
746
        :note: The file should be on the same file system as the
747
            packs directory.
748
749
        :param path: Path to the pack file.
750
        """
0.200.937 by Jelmer Vernooij
Avoid using os module in transportgit.
751
        f.seek(0)
0.200.1682 by Jelmer Vernooij
Fix compatibility with newer versions of dulwich.
752
        p = PackData("", f, len(f.getvalue()))
0.200.936 by Jelmer Vernooij
Test transportgit.
753
        entries = p.sorted_entries()
7143.15.2 by Jelmer Vernooij
Run autopep8.
754
        basename = "pack-%s" % iter_sha1(entry[0]
755
                                         for entry in entries).decode('ascii')
0.200.1671 by Jelmer Vernooij
Fix pack tests.
756
        p._filename = basename + ".pack"
0.200.937 by Jelmer Vernooij
Avoid using os module in transportgit.
757
        f.seek(0)
758
        self.pack_transport.put_file(basename + ".pack", f)
7356.1.5 by Jelmer Vernooij
Use more ExitStacks.
759
        with self.pack_transport.open_write_stream(basename + ".idx") as idxfile:
0.200.954 by Jelmer Vernooij
Avoid writing to memory first unless strictly necessary.
760
            write_pack_index_v2(idxfile, entries, p.get_stored_checksum())
0.399.2 by Jelmer Vernooij
Fix tests.
761
        idxfile = self.pack_transport.get(basename + ".idx")
7143.15.2 by Jelmer Vernooij
Run autopep8.
762
        idx = load_pack_index_file(basename + ".idx", idxfile)
0.399.2 by Jelmer Vernooij
Fix tests.
763
        final_pack = Pack.from_objects(p, idx)
764
        final_pack._basename = basename
7268.1.1 by Jelmer Vernooij
Fix compatibility with newer versions of Dulwich.
765
        self._add_cached_pack(basename, final_pack)
0.399.2 by Jelmer Vernooij
Fix tests.
766
        return final_pack
0.200.1003 by Jelmer Vernooij
Initial work on supporting move_in_thin_pack.
767
768
    def move_in_thin_pack(self, f):
769
        """Move a specific file containing a pack into the pack directory.
770
771
        :note: The file should be on the same file system as the
772
            packs directory.
773
774
        :param path: Path to the pack file.
775
        """
776
        f.seek(0)
0.399.1 by Jelmer Vernooij
Fix thin pack fetching.
777
        p = Pack('', resolve_ext_ref=self.get_raw)
778
        p._data = PackData.from_file(f, len(f.getvalue()))
779
        p._data.pack = p
7143.15.2 by Jelmer Vernooij
Run autopep8.
780
        p._idx_load = lambda: MemoryPackIndex(
781
            p.data.sorted_entries(), p.data.get_stored_checksum())
0.200.1003 by Jelmer Vernooij
Initial work on supporting move_in_thin_pack.
782
0.399.1 by Jelmer Vernooij
Fix thin pack fetching.
783
        pack_sha = p.index.objects_sha1()
0.200.1003 by Jelmer Vernooij
Initial work on supporting move_in_thin_pack.
784
7356.1.5 by Jelmer Vernooij
Use more ExitStacks.
785
        with self.pack_transport.open_write_stream(
786
                "pack-%s.pack" % pack_sha.decode('ascii')) as datafile:
0.399.1 by Jelmer Vernooij
Fix thin pack fetching.
787
            entries, data_sum = write_pack_objects(datafile, p.pack_tuples())
788
        entries = sorted([(k, v[0], v[1]) for (k, v) in entries.items()])
7356.1.5 by Jelmer Vernooij
Use more ExitStacks.
789
        with self.pack_transport.open_write_stream(
790
                "pack-%s.idx" % pack_sha.decode('ascii')) as idxfile:
0.399.1 by Jelmer Vernooij
Fix thin pack fetching.
791
            write_pack_index_v2(idxfile, entries, data_sum)
0.200.1003 by Jelmer Vernooij
Initial work on supporting move_in_thin_pack.
792
0.246.4 by Jelmer Vernooij
more work on transportgit.
793
    def add_pack(self):
0.200.1636 by Jelmer Vernooij
Some formatting fixes.
794
        """Add a new pack to this object store.
0.246.4 by Jelmer Vernooij
more work on transportgit.
795
0.200.1636 by Jelmer Vernooij
Some formatting fixes.
796
        :return: Fileobject to write to and a commit function to
0.246.4 by Jelmer Vernooij
more work on transportgit.
797
            call when the pack is finished.
798
        """
6964.2.1 by Jelmer Vernooij
Initial work to support brz-git on python3.
799
        f = BytesIO()
7143.15.2 by Jelmer Vernooij
Run autopep8.
800
0.200.936 by Jelmer Vernooij
Test transportgit.
801
        def commit():
0.200.937 by Jelmer Vernooij
Avoid using os module in transportgit.
802
            if len(f.getvalue()) > 0:
803
                return self.move_in_pack(f)
0.200.936 by Jelmer Vernooij
Test transportgit.
804
            else:
805
                return None
7143.15.2 by Jelmer Vernooij
Run autopep8.
806
0.200.1626 by Jelmer Vernooij
Fix compatibility with dulwich 0.9.1.
807
        def abort():
808
            return None
809
        return f, commit, abort
0.200.934 by Jelmer Vernooij
Add init function.
810
811
    @classmethod
812
    def init(cls, transport):
0.200.1718 by Jelmer Vernooij
Support AlreadyControlDirError.
813
        try:
814
            transport.mkdir('info')
815
        except FileExists:
816
            pass
817
        try:
818
            transport.mkdir(PACKDIR)
819
        except FileExists:
820
            pass
0.200.934 by Jelmer Vernooij
Add init function.
821
        return cls(transport)