/brz/remove-bazaar

To get this branch, use:
bzr branch http://gegoxaren.bato24.eu/bzr/brz/remove-bazaar
0.200.1 by Robert Collins
Commit initial content.
1
# Copyright (C) 2006 Canonical Ltd
2
# Authors: Robert Collins <robert.collins@canonical.com>
3
#
4
# This program is free software; you can redistribute it and/or modify
5
# it under the terms of the GNU General Public License as published by
6
# the Free Software Foundation; either version 2 of the License, or
7
# (at your option) any later version.
8
#
9
# This program is distributed in the hope that it will be useful,
10
# but WITHOUT ANY WARRANTY; without even the implied warranty of
11
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12
# GNU General Public License for more details.
13
#
14
# You should have received a copy of the GNU General Public License
15
# along with this program; if not, write to the Free Software
16
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
17
18
19
"""A GIT branch and repository format implementation for bzr."""
20
21
22
import stgit
23
import stgit.git as git
24
25
from bzrlib.decorators import *
26
import bzrlib.branch
27
import bzrlib.bzrdir
28
import bzrlib.errors as errors
29
import bzrlib.repository
30
from bzrlib.revision import Revision
31
32
33
def gitrevid_from_bzr(revision_id):
34
    return revision_id[4:]
35
36
37
def bzrrevid_from_git(revision_id):
38
    return "git:" + revision_id
39
40
41
class GitLock(object):
42
    """A lock that thunks through to Git."""
43
44
    def lock_write(self):
45
        pass
46
47
    def lock_read(self):
48
        pass
49
50
    def unlock(self):
51
        pass
52
53
54
class GitLockableFiles(bzrlib.lockable_files.LockableFiles):
55
    """Git specific lockable files abstraction."""
56
57
    def __init__(self, lock):
58
        self._lock = lock
59
        self._transaction = None
60
        self._lock_mode = None
61
        self._lock_count = 0
62
63
64
class GitDir(bzrlib.bzrdir.BzrDir):
65
    """An adapter to the '.git' dir used by git."""
66
67
    def __init__(self, transport, lockfiles, format):
68
        self._format = format
69
        self.root_transport = transport
70
        self.transport = transport.clone('.git')
71
        self._lockfiles = lockfiles
72
73
    def get_branch_transport(self, branch_format):
74
        if branch_format is None:
75
            return self.transport
76
        if isinstance(branch_format, GitBzrDirFormat):
77
            return self.transport
78
        raise errors.IncompatibleFormat(branch_format, self._format)
79
80
    get_repository_transport = get_branch_transport
81
    get_workingtree_transport = get_branch_transport
82
83
    def is_supported(self):
84
        return True
85
86
    def open_branch(self, ignored=None):
87
        """'crate' a branch for this dir."""
88
        return GitBranch(self, self._lockfiles)
89
90
    def open_repository(self, shared=False):
91
        """'open' a repository for this dir."""
92
        return GitRepository(self._gitrepo, self, self._lockfiles)
93
94
95
class GitBzrDirFormat(bzrlib.bzrdir.BzrDirFormat):
96
    """The .git directory control format."""
97
98
    @classmethod
99
    def _known_formats(self):
100
        return set([GitBzrDirFormat()])
101
102
    def open(self, transport, _create=False, _found=None):
103
        """Open this directory.
104
        
105
        :param _create: create the git dir on the fly. private to GitDirFormat.
106
        """
107
        # we dont grok readonly - git isn't integrated with transport.
108
        url = transport.base
109
        if url.startswith('readonly+'):
110
            url = url[len('readonly+'):]
111
        if url.startswith('file://'):
112
            url = url[len('file://'):]
113
        url = url.encode('utf8')
114
        lockfiles = GitLockableFiles(GitLock())
115
        return GitDir(transport, lockfiles, self)
116
117
    @classmethod
118
    def probe_transport(klass, transport):
119
        """Our format is present if the transport ends in '.not/'."""
120
        # little ugly, but works
121
        format = klass() 
122
        # try a manual probe first, its a little faster perhaps ?
123
        if transport.has('.git'):
124
            return format
125
        # delegate to the main opening code. This pays a double rtt cost at the
126
        # moment, so perhaps we want probe_transport to return the opened thing
127
        # rather than an openener ? or we could return a curried thing with the
128
        # dir to open already instantiated ? Needs more thought.
129
        try:
130
            format.open(transport)
131
            return format
132
        except Exception, e:
133
            raise errors.NotBranchError(path=transport.base)
134
        raise errors.NotBranchError(path=transport.base)
135
136
137
bzrlib.bzrdir.BzrDirFormat.register_control_format(GitBzrDirFormat)
138
139
140
class GitBranch(bzrlib.branch.Branch):
141
    """An adapter to git repositories for bzr Branch objects."""
142
143
    def __init__(self, gitdir, lockfiles):
144
        self.bzrdir = gitdir
145
        self.control_files = lockfiles
146
        self.repository = GitRepository(gitdir, lockfiles)
147
        self.base = gitdir.root_transport.base
148
149
    def lock_write(self):
150
        self.control_files.lock_write()
151
152
    @needs_read_lock
153
    def last_revision(self):
154
        # perhaps should escape this ?
155
        return bzrrevid_from_git(git.get_head())
156
157
    def lock_read(self):
158
        self.control_files.lock_read()
159
160
    def unlock(self):
161
        self.control_files.unlock()
162
163
164
class GitRepository(bzrlib.repository.Repository):
165
    """An adapter to git repositories for bzr."""
166
167
    def __init__(self, gitdir, lockfiles):
168
        self.bzrdir = gitdir
169
        self.control_files = lockfiles
170
171
    def get_revision(self, revision_id):
172
        raw = stgit.git._output_lines('git-rev-list --header --max-count=1 %s' % gitrevid_from_bzr(revision_id))
173
        # first field is the rev itself.
174
        # then its 'field value'
175
        # until the EOF??
176
        parents = []
177
        log = []
178
        in_log = False
179
        committer = None
180
        for field in raw[1:]:
181
            #if field.startswith('author '):
182
            #    committer = field[7:]
183
            if field.startswith('parent '):
184
                parents.append(bzrrevid_from_git(field.split()[1]))
185
            elif field.startswith('committer '):
186
                commit_fields = field.split()
187
                if committer is None:
188
                    committer = ' '.join(commit_fields[1:-3])
189
                timestamp = commit_fields[-2]
190
                timezone = commit_fields[-1]
191
            elif in_log:
192
                log.append(field)
193
            elif field == '\n':
194
                in_log = True
195
196
        log = ''.join(log)
197
        result = Revision(revision_id)
198
        result.parent_ids = parents
199
        result.message = log
200
        result.inventory_sha1 = ""
201
        result.timezone = timezone and int(timezone)
202
        result.timestamp = float(timestamp)
203
        result.committer = committer 
204
        return result