~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/smart/request.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:
14
14
# along with this program; if not, write to the Free Software
15
15
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
16
16
 
17
 
"""Basic server-side logic for dealing with requests.
18
 
 
19
 
**XXX**:
20
 
 
21
 
The class names are a little confusing: the protocol will instantiate a
22
 
SmartServerRequestHandler, whose dispatch_command method creates an instance of
23
 
a SmartServerRequest subclass.
24
 
 
25
 
The request_handlers registry tracks SmartServerRequest classes (rather than
26
 
SmartServerRequestHandler).
27
 
"""
 
17
"""Basic server-side logic for dealing with requests."""
 
18
 
28
19
 
29
20
import tempfile
30
21
 
33
24
    errors,
34
25
    registry,
35
26
    revision,
36
 
    urlutils,
37
27
    )
38
28
from bzrlib.bundle.serializer import write_bundle
39
29
 
40
30
 
41
31
class SmartServerRequest(object):
42
 
    """Base class for request handlers.
43
 
    
44
 
    To define a new request, subclass this class and override the `do` method
45
 
    (and if appropriate, `do_body` as well).  Request implementors should take
46
 
    care to call `translate_client_path` and `transport_from_client_path` as
47
 
    appropriate when dealing with paths received from the client.
48
 
    """
49
 
    # XXX: rename this class to BaseSmartServerRequestHandler ?  A request
50
 
    # *handler* is a different concept to the request.
 
32
    """Base class for request handlers."""
51
33
 
52
 
    def __init__(self, backing_transport, root_client_path='/'):
 
34
    def __init__(self, backing_transport):
53
35
        """Constructor.
54
36
 
55
37
        :param backing_transport: the base transport to be used when performing
56
38
            this request.
57
 
        :param root_client_path: the client path that maps to the root of
58
 
            backing_transport.  This is used to interpret relpaths received
59
 
            from the client.  Clients will not be able to refer to paths above
60
 
            this root.  If root_client_path is None, then no translation will
61
 
            be performed on client paths.  Default is '/'.
62
39
        """
63
40
        self._backing_transport = backing_transport
64
 
        if root_client_path is not None:
65
 
            if not root_client_path.startswith('/'):
66
 
                root_client_path = '/' + root_client_path
67
 
            if not root_client_path.endswith('/'):
68
 
                root_client_path += '/'
69
 
        self._root_client_path = root_client_path
70
41
 
71
42
    def _check_enabled(self):
72
43
        """Raises DisabledMethod if this method is disabled."""
95
66
 
96
67
    def do_body(self, body_bytes):
97
68
        """Called if the client sends a body with the request.
98
 
 
99
 
        The do() method is still called, and must have returned None.
100
69
        
101
70
        Must return a SmartServerResponse.
102
71
        """
 
72
        # TODO: if a client erroneously sends a request that shouldn't have a
 
73
        # body, what to do?  Probably SmartServerRequestHandler should catch
 
74
        # this NotImplementedError and translate it into a 'bad request' error
 
75
        # to send to the client.
103
76
        raise NotImplementedError(self.do_body)
104
77
 
105
 
    def do_chunk(self, chunk_bytes):
106
 
        """Called with each body chunk if the request has a streamed body.
107
 
 
108
 
        The do() method is still called, and must have returned None.
109
 
        """
110
 
        raise NotImplementedError(self.do_chunk)
111
 
 
112
 
    def do_end(self):
113
 
        """Called when the end of the request has been received."""
114
 
        pass
115
 
    
116
 
    def translate_client_path(self, client_path):
117
 
        """Translate a path received from a network client into a local
118
 
        relpath.
119
 
 
120
 
        All paths received from the client *must* be translated.
121
 
 
122
 
        :param client_path: the path from the client.
123
 
        :returns: a relpath that may be used with self._backing_transport
124
 
            (unlike the untranslated client_path, which must not be used with
125
 
            the backing transport).
126
 
        """
127
 
        if self._root_client_path is None:
128
 
            # no translation necessary!
129
 
            return client_path
130
 
        if not client_path.startswith('/'):
131
 
            client_path = '/' + client_path
132
 
        if client_path.startswith(self._root_client_path):
133
 
            path = client_path[len(self._root_client_path):]
134
 
            relpath = urlutils.joinpath('/', path)
135
 
            if not relpath.startswith('/'):
136
 
                raise ValueError(relpath)
137
 
            return '.' + relpath
138
 
        else:
139
 
            raise errors.PathNotChild(client_path, self._root_client_path)
140
 
 
141
 
    def transport_from_client_path(self, client_path):
142
 
        """Get a backing transport corresponding to the location referred to by
143
 
        a network client.
144
 
 
145
 
        :seealso: translate_client_path
146
 
        :returns: a transport cloned from self._backing_transport
147
 
        """
148
 
        relpath = self.translate_client_path(client_path)
149
 
        return self._backing_transport.clone(relpath)
150
 
 
151
78
 
152
79
class SmartServerResponse(object):
153
80
    """A response to a client request.
156
83
    SuccessfulSmartServerResponse and FailedSmartServerResponse as appropriate.
157
84
    """
158
85
 
159
 
    def __init__(self, args, body=None, body_stream=None):
160
 
        """Constructor.
161
 
 
162
 
        :param args: tuple of response arguments.
163
 
        :param body: string of a response body.
164
 
        :param body_stream: iterable of bytestrings to be streamed to the
165
 
            client.
166
 
        """
 
86
    def __init__(self, args, body=None):
167
87
        self.args = args
168
 
        if body is not None and body_stream is not None:
169
 
            raise errors.BzrError(
170
 
                "'body' and 'body_stream' are mutually exclusive.")
171
88
        self.body = body
172
 
        self.body_stream = body_stream
173
89
 
174
90
    def __eq__(self, other):
175
91
        if other is None:
176
92
            return False
177
 
        return (other.args == self.args and
178
 
                other.body == self.body and
179
 
                other.body_stream is self.body_stream)
 
93
        return other.args == self.args and other.body == self.body
180
94
 
181
95
    def __repr__(self):
182
 
        status = {True: 'OK', False: 'ERR'}[self.is_successful()]
183
 
        return "<SmartServerResponse status=%s args=%r body=%r>" % (status,
 
96
        return "<SmartServerResponse args=%r body=%r>" % (
184
97
            self.args, self.body)
185
98
 
186
99
 
217
130
    # TODO: Better way of representing the body for commands that take it,
218
131
    # and allow it to be streamed into the server.
219
132
 
220
 
    def __init__(self, backing_transport, commands, root_client_path):
 
133
    def __init__(self, backing_transport, commands):
221
134
        """Constructor.
222
135
 
223
136
        :param backing_transport: a Transport to handle requests for.
225
138
            subclasses. e.g. bzrlib.transport.smart.vfs.vfs_commands.
226
139
        """
227
140
        self._backing_transport = backing_transport
228
 
        self._root_client_path = root_client_path
229
141
        self._commands = commands
230
142
        self._body_bytes = ''
231
143
        self.response = None
254
166
        try:
255
167
            command = self._commands.get(cmd)
256
168
        except LookupError:
257
 
            raise errors.UnknownSmartMethod(cmd)
258
 
        self._command = command(self._backing_transport, self._root_client_path)
 
169
            raise errors.SmartProtocolError("bad request %r" % (cmd,))
 
170
        self._command = command(self._backing_transport)
259
171
        self._run_handler_code(self._command.execute, args, {})
260
172
 
261
173
    def _run_handler_code(self, callable, args, kwargs):
307
219
            else:
308
220
                raise
309
221
 
310
 
    def headers_received(self, headers):
311
 
        # Just a no-op at the moment.
312
 
        pass
313
 
 
314
 
    def args_received(self, args):
315
 
        cmd = args[0]
316
 
        args = args[1:]
317
 
        try:
318
 
            command = self._commands.get(cmd)
319
 
        except LookupError:
320
 
            raise errors.UnknownSmartMethod(cmd)
321
 
        self._command = command(self._backing_transport)
322
 
        self._run_handler_code(self._command.execute, args, {})
323
 
 
324
 
    def prefixed_body_received(self, body_bytes):
325
 
        """No more body data will be received."""
326
 
        self._run_handler_code(self._command.do_body, (body_bytes,), {})
327
 
        # cannot read after this.
328
 
        self.finished_reading = True
329
 
 
330
 
    def body_chunk_received(self, chunk_bytes):
331
 
        self._run_handler_code(self._command.do_chunk, (chunk_bytes,), {})
332
 
 
333
 
    def end_received(self):
334
 
        self._run_handler_code(self._command.do_end, (), {})
335
 
 
336
222
 
337
223
class HelloRequest(SmartServerRequest):
338
224
    """Answer a version request with the highest protocol version this server
348
234
 
349
235
    def do(self, path, revision_id):
350
236
        # open transport relative to our base
351
 
        t = self.transport_from_client_path(path)
 
237
        t = self._backing_transport.clone(path)
352
238
        control, extra_path = bzrdir.BzrDir.open_containing_from_transport(t)
353
239
        repo = control.open_repository()
354
240
        tmpf = tempfile.TemporaryFile()
383
269
request_handlers.register_lazy(
384
270
    'Branch.set_last_revision', 'bzrlib.smart.branch', 'SmartServerBranchRequestSetLastRevision')
385
271
request_handlers.register_lazy(
386
 
    'Branch.set_last_revision_info', 'bzrlib.smart.branch',
387
 
    'SmartServerBranchRequestSetLastRevisionInfo')
388
 
request_handlers.register_lazy(
389
 
    'Branch.set_last_revision_ex', 'bzrlib.smart.branch',
390
 
    'SmartServerBranchRequestSetLastRevisionEx')
391
 
request_handlers.register_lazy(
392
272
    'Branch.unlock', 'bzrlib.smart.branch', 'SmartServerBranchRequestUnlock')
393
273
request_handlers.register_lazy(
394
 
    'BzrDir.find_repository', 'bzrlib.smart.bzrdir', 'SmartServerRequestFindRepositoryV1')
395
 
request_handlers.register_lazy(
396
 
    'BzrDir.find_repositoryV2', 'bzrlib.smart.bzrdir', 'SmartServerRequestFindRepositoryV2')
 
274
    'BzrDir.find_repository', 'bzrlib.smart.bzrdir', 'SmartServerRequestFindRepository')
397
275
request_handlers.register_lazy(
398
276
    'BzrDirFormat.initialize', 'bzrlib.smart.bzrdir', 'SmartServerRequestInitializeBzrDir')
399
277
request_handlers.register_lazy(
427
305
request_handlers.register_lazy('Repository.gather_stats',
428
306
                               'bzrlib.smart.repository',
429
307
                               'SmartServerRepositoryGatherStats')
430
 
request_handlers.register_lazy('Repository.get_parent_map',
431
 
                               'bzrlib.smart.repository',
432
 
                               'SmartServerRepositoryGetParentMap')
 
308
request_handlers.register_lazy(
 
309
    'Repository.stream_knit_data_for_revisions', 'bzrlib.smart.repository',
 
310
    'SmartServerRepositoryStreamKnitDataForRevisions')
433
311
request_handlers.register_lazy(
434
312
    'Repository.get_revision_graph', 'bzrlib.smart.repository', 'SmartServerRepositoryGetRevisionGraph')
435
313
request_handlers.register_lazy(