~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/smart/repository.py

  • Committer: Canonical.com Patch Queue Manager
  • Date: 2007-11-04 18:51:39 UTC
  • mfrom: (2961.1.1 trunk)
  • Revision ID: pqm@pqm.ubuntu.com-20071104185139-kaio3sneodg2kp71
Authentication ring implementation (read-only)

Show diffs side-by-side

added added

removed removed

Lines of Context:
16
16
 
17
17
"""Server-side repository related request implmentations."""
18
18
 
19
 
import bz2
20
19
from cStringIO import StringIO
21
20
import os
22
21
import sys
23
22
import tempfile
24
23
import tarfile
25
24
 
26
 
from bzrlib import (
27
 
    errors,
28
 
    osutils,
29
 
    )
 
25
from bzrlib import errors
30
26
from bzrlib.bzrdir import BzrDir
31
 
from bzrlib.pack import ContainerSerialiser
 
27
from bzrlib.pack import ContainerWriter
32
28
from bzrlib.smart.request import (
33
29
    FailedSmartServerResponse,
34
30
    SmartServerRequest,
35
31
    SuccessfulSmartServerResponse,
36
32
    )
37
 
from bzrlib.repository import _strip_NULL_ghosts
38
 
from bzrlib import revision as _mod_revision
39
33
 
40
34
 
41
35
class SmartServerRepositoryRequest(SmartServerRequest):
44
38
    def do(self, path, *args):
45
39
        """Execute a repository request.
46
40
        
47
 
        All Repository requests take a path to the repository as their first
48
 
        argument.  The repository must be at the exact path given by the
49
 
        client - no searching is done.
 
41
        The repository must be at the exact path - no searching is done.
50
42
 
51
43
        The actual logic is delegated to self.do_repository_request.
52
44
 
53
 
        :param client_path: The path for the repository as received from the
54
 
            client.
55
 
        :return: A SmartServerResponse from self.do_repository_request().
 
45
        :param path: The path for the repository.
 
46
        :return: A smart server from self.do_repository_request().
56
47
        """
57
 
        transport = self.transport_from_client_path(path)
 
48
        transport = self._backing_transport.clone(path)
58
49
        bzrdir = BzrDir.open_from_transport(transport)
59
 
        # Save the repository for use with do_body.
60
 
        self._repository = bzrdir.open_repository()
61
 
        return self.do_repository_request(self._repository, *args)
62
 
 
63
 
    def do_repository_request(self, repository, *args):
64
 
        """Override to provide an implementation for a verb."""
65
 
        # No-op for verbs that take bodies (None as a result indicates a body
66
 
        # is expected)
67
 
        return None
68
 
 
69
 
    def recreate_search(self, repository, recipe_bytes):
70
 
        lines = recipe_bytes.split('\n')
71
 
        start_keys = set(lines[0].split(' '))
72
 
        exclude_keys = set(lines[1].split(' '))
73
 
        revision_count = int(lines[2])
74
 
        repository.lock_read()
75
 
        try:
76
 
            search = repository.get_graph()._make_breadth_first_searcher(
77
 
                start_keys)
78
 
            while True:
79
 
                try:
80
 
                    next_revs = search.next()
81
 
                except StopIteration:
82
 
                    break
83
 
                search.stop_searching_any(exclude_keys.intersection(next_revs))
84
 
            search_result = search.get_result()
85
 
            if search_result.get_recipe()[2] != revision_count:
86
 
                # we got back a different amount of data than expected, this
87
 
                # gets reported as NoSuchRevision, because less revisions
88
 
                # indicates missing revisions, and more should never happen as
89
 
                # the excludes list considers ghosts and ensures that ghost
90
 
                # filling races are not a problem.
91
 
                return (None, FailedSmartServerResponse(('NoSuchRevision',)))
92
 
            return (search, None)
93
 
        finally:
94
 
            repository.unlock()
95
 
 
96
 
 
97
 
class SmartServerRepositoryReadLocked(SmartServerRepositoryRequest):
98
 
    """Calls self.do_readlocked_repository_request."""
99
 
 
100
 
    def do_repository_request(self, repository, *args):
101
 
        """Read lock a repository for do_readlocked_repository_request."""
102
 
        repository.lock_read()
103
 
        try:
104
 
            return self.do_readlocked_repository_request(repository, *args)
105
 
        finally:
106
 
            repository.unlock()
107
 
 
108
 
 
109
 
class SmartServerRepositoryGetParentMap(SmartServerRepositoryRequest):
110
 
    """Bzr 1.2+ - get parent data for revisions during a graph search."""
111
 
    
112
 
    def do_repository_request(self, repository, *revision_ids):
113
 
        """Get parent details for some revisions.
114
 
        
115
 
        All the parents for revision_ids are returned. Additionally up to 64KB
116
 
        of additional parent data found by performing a breadth first search
117
 
        from revision_ids is returned. The verb takes a body containing the
118
 
        current search state, see do_body for details.
119
 
 
120
 
        :param repository: The repository to query in.
121
 
        :param revision_ids: The utf8 encoded revision_id to answer for.
122
 
        """
123
 
        self._revision_ids = revision_ids
124
 
        return None # Signal that we want a body.
125
 
 
126
 
    def do_body(self, body_bytes):
127
 
        """Process the current search state and perform the parent lookup.
128
 
 
129
 
        :return: A smart server response where the body contains an utf8
130
 
            encoded flattened list of the parents of the revisions (the same
131
 
            format as Repository.get_revision_graph) which has been bz2
132
 
            compressed.
133
 
        """
134
 
        repository = self._repository
135
 
        repository.lock_read()
136
 
        try:
137
 
            return self._do_repository_request(body_bytes)
138
 
        finally:
139
 
            repository.unlock()
140
 
 
141
 
    def _do_repository_request(self, body_bytes):
142
 
        repository = self._repository
143
 
        revision_ids = set(self._revision_ids)
144
 
        search, error = self.recreate_search(repository, body_bytes)
145
 
        if error is not None:
146
 
            return error
147
 
        # TODO might be nice to start up the search again; but thats not
148
 
        # written or tested yet.
149
 
        client_seen_revs = set(search.get_result().get_keys())
150
 
        # Always include the requested ids.
151
 
        client_seen_revs.difference_update(revision_ids)
152
 
        lines = []
153
 
        repo_graph = repository.get_graph()
154
 
        result = {}
155
 
        queried_revs = set()
156
 
        size_so_far = 0
157
 
        next_revs = revision_ids
158
 
        first_loop_done = False
159
 
        while next_revs:
160
 
            queried_revs.update(next_revs)
161
 
            parent_map = repo_graph.get_parent_map(next_revs)
162
 
            next_revs = set()
163
 
            for revision_id, parents in parent_map.iteritems():
164
 
                # adjust for the wire
165
 
                if parents == (_mod_revision.NULL_REVISION,):
166
 
                    parents = ()
167
 
                # prepare the next query
168
 
                next_revs.update(parents)
169
 
                if revision_id not in client_seen_revs:
170
 
                    # Client does not have this revision, give it to it.
171
 
                    # add parents to the result
172
 
                    result[revision_id] = parents
173
 
                    # Approximate the serialized cost of this revision_id.
174
 
                    size_so_far += 2 + len(revision_id) + sum(map(len, parents))
175
 
            # get all the directly asked for parents, and then flesh out to
176
 
            # 64K (compressed) or so. We do one level of depth at a time to
177
 
            # stay in sync with the client. The 250000 magic number is
178
 
            # estimated compression ratio taken from bzr.dev itself.
179
 
            if first_loop_done and size_so_far > 250000:
180
 
                next_revs = set()
181
 
                break
182
 
            # don't query things we've already queried
183
 
            next_revs.difference_update(queried_revs)
184
 
            first_loop_done = True
185
 
 
186
 
        # sorting trivially puts lexographically similar revision ids together.
187
 
        # Compression FTW.
188
 
        for revision, parents in sorted(result.items()):
189
 
            lines.append(' '.join((revision, ) + tuple(parents)))
190
 
 
191
 
        return SuccessfulSmartServerResponse(
192
 
            ('ok', ), bz2.compress('\n'.join(lines)))
193
 
 
194
 
 
195
 
class SmartServerRepositoryGetRevisionGraph(SmartServerRepositoryReadLocked):
196
 
    
197
 
    def do_readlocked_repository_request(self, repository, revision_id):
 
50
        repository = bzrdir.open_repository()
 
51
        return self.do_repository_request(repository, *args)
 
52
 
 
53
 
 
54
class SmartServerRepositoryGetRevisionGraph(SmartServerRepositoryRequest):
 
55
    
 
56
    def do_repository_request(self, repository, revision_id):
198
57
        """Return the result of repository.get_revision_graph(revision_id).
199
 
 
200
 
        Deprecated as of bzr 1.4, but supported for older clients.
201
58
        
202
59
        :param repository: The repository to query in.
203
60
        :param revision_id: The utf8 encoded revision_id to get a graph from.
208
65
            revision_id = None
209
66
 
210
67
        lines = []
211
 
        graph = repository.get_graph()
212
 
        if revision_id:
213
 
            search_ids = [revision_id]
214
 
        else:
215
 
            search_ids = repository.all_revision_ids()
216
 
        search = graph._make_breadth_first_searcher(search_ids)
217
 
        transitive_ids = set()
218
 
        map(transitive_ids.update, list(search))
219
 
        parent_map = graph.get_parent_map(transitive_ids)
220
 
        revision_graph = _strip_NULL_ghosts(parent_map)
221
 
        if revision_id and revision_id not in revision_graph:
 
68
        try:
 
69
            revision_graph = repository.get_revision_graph(revision_id)
 
70
        except errors.NoSuchRevision:
222
71
            # Note that we return an empty body, rather than omitting the body.
223
72
            # This way the client knows that it can always expect to find a body
224
73
            # in the response for this method, even in the error case.
260
109
              firstrev: 1234.230 0
261
110
              latestrev: 345.700 3600
262
111
              revisions: 2
 
112
              size:45
263
113
 
264
114
              But containing only fields returned by the gather_stats() call
265
115
        """
318
168
        except errors.LockFailed, e:
319
169
            return FailedSmartServerResponse(('LockFailed',
320
170
                str(e.lock), str(e.why)))
321
 
        if token is not None:
322
 
            repository.leave_lock_in_place()
 
171
        repository.leave_lock_in_place()
323
172
        repository.unlock()
324
173
        if token is None:
325
174
            token = ''
352
201
 
353
202
    def do_repository_request(self, repository, compression):
354
203
        from bzrlib import osutils
 
204
        repo_transport = repository.control_files._transport
355
205
        tmp_dirname, tmp_repo = self._copy_to_tempdir(repository)
356
206
        try:
357
207
            controldir_name = tmp_dirname + '/.bzr'
360
210
            osutils.rmtree(tmp_dirname)
361
211
 
362
212
    def _copy_to_tempdir(self, from_repo):
363
 
        tmp_dirname = osutils.mkdtemp(prefix='tmpbzrclone')
 
213
        tmp_dirname = tempfile.mkdtemp(prefix='tmpbzrclone')
364
214
        tmp_bzrdir = from_repo.bzrdir._format.initialize(tmp_dirname)
365
215
        tmp_repo = from_repo._format.initialize(tmp_bzrdir)
366
216
        from_repo.copy_content_into(tmp_repo)
373
223
            # all finished; write the tempfile out to the network
374
224
            temp.seek(0)
375
225
            return SuccessfulSmartServerResponse(('ok',), temp.read())
376
 
            # FIXME: Don't read the whole thing into memory here; rather stream
377
 
            # it out from the file onto the network. mbp 20070411
 
226
            # FIXME: Don't read the whole thing into memory here; rather stream it
 
227
            # out from the file onto the network. mbp 20070411
378
228
        finally:
379
229
            temp.close()
380
230
 
390
240
            dirname = dirname.encode(sys.getfilesystemencoding())
391
241
            # python's tarball module includes the whole path by default so
392
242
            # override it
393
 
            if not dirname.endswith('.bzr'):
394
 
                raise ValueError(dirname)
 
243
            assert dirname.endswith('.bzr')
395
244
            tarball.add(dirname, '.bzr') # recursive by default
396
245
        finally:
397
246
            tarball.close()
 
247
 
 
248
 
 
249
class SmartServerRepositoryStreamKnitDataForRevisions(SmartServerRepositoryRequest):
 
250
 
 
251
    def do_repository_request(self, repository, *revision_ids):
 
252
        stream = repository.get_data_stream(revision_ids)
 
253
        filelike = StringIO()
 
254
        pack = ContainerWriter(filelike.write)
 
255
        pack.begin()
 
256
        try:
 
257
            for name_tuple, bytes in stream:
 
258
                pack.add_bytes_record(bytes, [name_tuple])
 
259
        except errors.RevisionNotPresent, e:
 
260
            return FailedSmartServerResponse(('NoSuchRevision', e.revision_id))
 
261
        pack.end()
 
262
        return SuccessfulSmartServerResponse(('ok',), filelike.getvalue())
 
263