~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/tests/test_smart_request.py

Latest bzr.dev

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2009 Canonical Ltd
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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
16
 
 
17
 
"""Tests for smart server request infrastructure (bzrlib.smart.request)."""
18
 
 
19
 
import threading
20
 
 
21
 
from bzrlib import errors
22
 
from bzrlib.bzrdir import BzrDir
23
 
from bzrlib.smart import request
24
 
from bzrlib.tests import TestCase, TestCaseWithMemoryTransport
25
 
from bzrlib.transport import get_transport
26
 
 
27
 
 
28
 
class NoBodyRequest(request.SmartServerRequest):
29
 
    """A request that does not implement do_body."""
30
 
 
31
 
    def do(self):
32
 
        return request.SuccessfulSmartServerResponse(('ok',))
33
 
 
34
 
 
35
 
class DoErrorRequest(request.SmartServerRequest):
36
 
    """A request that raises an error from self.do()."""
37
 
    
38
 
    def do(self):
39
 
        raise errors.NoSuchFile('xyzzy')
40
 
 
41
 
 
42
 
class ChunkErrorRequest(request.SmartServerRequest):
43
 
    """A request that raises an error from self.do_chunk()."""
44
 
    
45
 
    def do(self):
46
 
        """No-op."""
47
 
        pass
48
 
 
49
 
    def do_chunk(self, bytes):
50
 
        raise errors.NoSuchFile('xyzzy')
51
 
 
52
 
 
53
 
class EndErrorRequest(request.SmartServerRequest):
54
 
    """A request that raises an error from self.do_end()."""
55
 
    
56
 
    def do(self):
57
 
        """No-op."""
58
 
        pass
59
 
 
60
 
    def do_chunk(self, bytes):
61
 
        """No-op."""
62
 
        pass
63
 
        
64
 
    def do_end(self):
65
 
        raise errors.NoSuchFile('xyzzy')
66
 
 
67
 
 
68
 
class CheckJailRequest(request.SmartServerRequest):
69
 
 
70
 
    def __init__(self, *args):
71
 
        request.SmartServerRequest.__init__(self, *args)
72
 
        self.jail_transports_log = []
73
 
 
74
 
    def do(self):
75
 
        self.jail_transports_log.append(request.jail_info.transports)
76
 
 
77
 
    def do_chunk(self, bytes):
78
 
        self.jail_transports_log.append(request.jail_info.transports)
79
 
 
80
 
    def do_end(self):
81
 
        self.jail_transports_log.append(request.jail_info.transports)
82
 
 
83
 
 
84
 
class TestSmartRequest(TestCase):
85
 
 
86
 
    def test_request_class_without_do_body(self):
87
 
        """If a request has no body data, and the request's implementation does
88
 
        not override do_body, then no exception is raised.
89
 
        """
90
 
        # Create a SmartServerRequestHandler with a SmartServerRequest subclass
91
 
        # that does not implement do_body.
92
 
        handler = request.SmartServerRequestHandler(
93
 
            None, {'foo': NoBodyRequest}, '/')
94
 
        # Emulate a request with no body (i.e. just args).
95
 
        handler.args_received(('foo',))
96
 
        handler.end_received()
97
 
        # Request done, no exception was raised.
98
 
 
99
 
    def test_only_request_code_is_jailed(self):
100
 
        transport = 'dummy transport'
101
 
        handler = request.SmartServerRequestHandler(
102
 
            transport, {'foo': CheckJailRequest}, '/')
103
 
        handler.args_received(('foo',))
104
 
        self.assertEqual(None, request.jail_info.transports)
105
 
        handler.accept_body('bytes')
106
 
        self.assertEqual(None, request.jail_info.transports)
107
 
        handler.end_received()
108
 
        self.assertEqual(None, request.jail_info.transports)
109
 
        self.assertEqual(
110
 
            [[transport]] * 3, handler._command.jail_transports_log)
111
 
 
112
 
 
113
 
 
114
 
class TestSmartRequestHandlerErrorTranslation(TestCase):
115
 
    """Tests that SmartServerRequestHandler will translate exceptions raised by
116
 
    a SmartServerRequest into FailedSmartServerResponses.
117
 
    """
118
 
 
119
 
    def assertNoResponse(self, handler):
120
 
        self.assertEqual(None, handler.response)
121
 
 
122
 
    def assertResponseIsTranslatedError(self, handler):
123
 
        expected_translation = ('NoSuchFile', 'xyzzy')
124
 
        self.assertEqual(
125
 
            request.FailedSmartServerResponse(expected_translation),
126
 
            handler.response)
127
 
 
128
 
    def test_error_translation_from_args_received(self):
129
 
        handler = request.SmartServerRequestHandler(
130
 
            None, {'foo': DoErrorRequest}, '/')
131
 
        handler.args_received(('foo',))
132
 
        self.assertResponseIsTranslatedError(handler)
133
 
 
134
 
    def test_error_translation_from_chunk_received(self):
135
 
        handler = request.SmartServerRequestHandler(
136
 
            None, {'foo': ChunkErrorRequest}, '/')
137
 
        handler.args_received(('foo',))
138
 
        self.assertNoResponse(handler)
139
 
        handler.accept_body('bytes')
140
 
        self.assertResponseIsTranslatedError(handler)
141
 
 
142
 
    def test_error_translation_from_end_received(self):
143
 
        handler = request.SmartServerRequestHandler(
144
 
            None, {'foo': EndErrorRequest}, '/')
145
 
        handler.args_received(('foo',))
146
 
        self.assertNoResponse(handler)
147
 
        handler.end_received()
148
 
        self.assertResponseIsTranslatedError(handler)
149
 
 
150
 
 
151
 
class TestRequestHanderErrorTranslation(TestCase):
152
 
    """Tests for bzrlib.smart.request._translate_error."""
153
 
 
154
 
    def assertTranslationEqual(self, expected_tuple, error):
155
 
        self.assertEqual(expected_tuple, request._translate_error(error))
156
 
 
157
 
    def test_NoSuchFile(self):
158
 
        self.assertTranslationEqual(
159
 
            ('NoSuchFile', 'path'), errors.NoSuchFile('path'))
160
 
 
161
 
    def test_LockContention(self):
162
 
        # For now, LockContentions are always transmitted with no details.
163
 
        # Eventually they should include a relpath or url or something else to
164
 
        # identify which lock is busy.
165
 
        self.assertTranslationEqual(
166
 
            ('LockContention',), errors.LockContention('lock', 'msg'))
167
 
 
168
 
    def test_TokenMismatch(self):
169
 
        self.assertTranslationEqual(
170
 
            ('TokenMismatch', 'some-token', 'actual-token'),
171
 
            errors.TokenMismatch('some-token', 'actual-token'))
172
 
 
173
 
 
174
 
class TestRequestJail(TestCaseWithMemoryTransport):
175
 
    
176
 
    def test_jail(self):
177
 
        transport = self.get_transport('blah')
178
 
        req = request.SmartServerRequest(transport)
179
 
        self.assertEqual(None, request.jail_info.transports)
180
 
        req.setup_jail()
181
 
        self.assertEqual([transport], request.jail_info.transports)
182
 
        req.teardown_jail()
183
 
        self.assertEqual(None, request.jail_info.transports)
184
 
 
185
 
 
186
 
class TestJailHook(TestCaseWithMemoryTransport):
187
 
 
188
 
    def tearDown(self):
189
 
        request.jail_info.transports = None
190
 
        TestCaseWithMemoryTransport.tearDown(self)
191
 
 
192
 
    def test_jail_hook(self):
193
 
        request.jail_info.transports = None
194
 
        _pre_open_hook = request._pre_open_hook
195
 
        # Any transport is fine if jail_info.transports is None
196
 
        t = self.get_transport('foo')
197
 
        _pre_open_hook(t)
198
 
        # A transport in jail_info.transports is allowed
199
 
        request.jail_info.transports = [t]
200
 
        _pre_open_hook(t)
201
 
        # A child of a transport in jail_info is allowed
202
 
        _pre_open_hook(t.clone('child'))
203
 
        # A parent is not allowed
204
 
        self.assertRaises(errors.JailBreak, _pre_open_hook, t.clone('..'))
205
 
        # A completely unrelated transport is not allowed
206
 
        self.assertRaises(
207
 
            errors.JailBreak, _pre_open_hook, get_transport('http://host/'))
208
 
 
209
 
    def test_open_bzrdir_in_non_main_thread(self):
210
 
        """Opening a bzrdir in a non-main thread should work ok.
211
 
        
212
 
        This makes sure that the globally-installed
213
 
        bzrlib.smart.request._pre_open_hook, which uses a threading.local(),
214
 
        works in a newly created thread.
215
 
        """
216
 
        bzrdir = self.make_bzrdir('.')
217
 
        transport = bzrdir.root_transport
218
 
        thread_result = []
219
 
        def t():
220
 
            BzrDir.open_from_transport(transport)
221
 
            thread_result.append('ok')
222
 
        thread = threading.Thread(target=t)
223
 
        thread.start()
224
 
        thread.join()
225
 
        self.assertEqual(['ok'], thread_result)
226