1 #!/usr/bin/env python 2 # 3 # Copyright 2011, Google Inc. 4 # All rights reserved. 5 # 6 # Redistribution and use in source and binary forms, with or without 7 # modification, are permitted provided that the following conditions are 8 # met: 9 # 10 # * Redistributions of source code must retain the above copyright 11 # notice, this list of conditions and the following disclaimer. 12 # * Redistributions in binary form must reproduce the above 13 # copyright notice, this list of conditions and the following disclaimer 14 # in the documentation and/or other materials provided with the 15 # distribution. 16 # * Neither the name of Google Inc. nor the names of its 17 # contributors may be used to endorse or promote products derived from 18 # this software without specific prior written permission. 19 # 20 # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 21 # "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 22 # LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR 23 # A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT 24 # OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, 25 # SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT 26 # LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 27 # DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 28 # THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 29 # (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 30 # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 31 32 33 """Tests for handshake module.""" 34 35 36 import unittest 37 38 import set_sys_path # Update sys.path to locate mod_pywebsocket module. 39 from mod_pywebsocket import common 40 from mod_pywebsocket.handshake._base import AbortedByUserException 41 from mod_pywebsocket.handshake._base import HandshakeException 42 from mod_pywebsocket.handshake._base import VersionException 43 from mod_pywebsocket.handshake.hybi import Handshaker 44 45 import mock 46 47 48 class RequestDefinition(object): 49 """A class for holding data for constructing opening handshake strings for 50 testing the opening handshake processor. 51 """ 52 53 def __init__(self, method, uri, headers): 54 self.method = method 55 self.uri = uri 56 self.headers = headers 57 58 59 def _create_good_request_def(): 60 return RequestDefinition( 61 'GET', '/demo', 62 {'Host': 'server.example.com', 63 'Upgrade': 'websocket', 64 'Connection': 'Upgrade', 65 'Sec-WebSocket-Key': 'dGhlIHNhbXBsZSBub25jZQ==', 66 'Sec-WebSocket-Origin': 'http://example.com', 67 'Sec-WebSocket-Version': '8'}) 68 69 70 def _create_request(request_def): 71 conn = mock.MockConn('') 72 return mock.MockRequest( 73 method=request_def.method, 74 uri=request_def.uri, 75 headers_in=request_def.headers, 76 connection=conn) 77 78 79 def _create_handshaker(request): 80 handshaker = Handshaker(request, mock.MockDispatcher()) 81 return handshaker 82 83 84 class SubprotocolChoosingDispatcher(object): 85 """A dispatcher for testing. This dispatcher sets the i-th subprotocol 86 of requested ones to ws_protocol where i is given on construction as index 87 argument. If index is negative, default_value will be set to ws_protocol. 88 """ 89 90 def __init__(self, index, default_value=None): 91 self.index = index 92 self.default_value = default_value 93 94 def do_extra_handshake(self, conn_context): 95 if self.index >= 0: 96 conn_context.ws_protocol = conn_context.ws_requested_protocols[ 97 self.index] 98 else: 99 conn_context.ws_protocol = self.default_value 100 101 def transfer_data(self, conn_context): 102 pass 103 104 105 class HandshakeAbortedException(Exception): 106 pass 107 108 109 class AbortingDispatcher(object): 110 """A dispatcher for testing. This dispatcher raises an exception in 111 do_extra_handshake to reject the request. 112 """ 113 114 def do_extra_handshake(self, conn_context): 115 raise HandshakeAbortedException('An exception to reject the request') 116 117 def transfer_data(self, conn_context): 118 pass 119 120 121 class AbortedByUserDispatcher(object): 122 """A dispatcher for testing. This dispatcher raises an 123 AbortedByUserException in do_extra_handshake to reject the request. 124 """ 125 126 def do_extra_handshake(self, conn_context): 127 raise AbortedByUserException('An AbortedByUserException to reject the ' 128 'request') 129 130 def transfer_data(self, conn_context): 131 pass 132 133 134 _EXPECTED_RESPONSE = ( 135 'HTTP/1.1 101 Switching Protocols\r\n' 136 'Upgrade: websocket\r\n' 137 'Connection: Upgrade\r\n' 138 'Sec-WebSocket-Accept: s3pPLMBiTxaQ9kYGzzhZRbK+xOo=\r\n\r\n') 139 140 141 class HandshakerTest(unittest.TestCase): 142 """A unittest for draft-ietf-hybi-thewebsocketprotocol-06 and later 143 handshake processor. 144 """ 145 146 def test_do_handshake(self): 147 request = _create_request(_create_good_request_def()) 148 dispatcher = mock.MockDispatcher() 149 handshaker = Handshaker(request, dispatcher) 150 handshaker.do_handshake() 151 152 self.assertTrue(dispatcher.do_extra_handshake_called) 153 154 self.assertEqual( 155 _EXPECTED_RESPONSE, request.connection.written_data()) 156 self.assertEqual('/demo', request.ws_resource) 157 self.assertEqual('http://example.com', request.ws_origin) 158 self.assertEqual(None, request.ws_protocol) 159 self.assertEqual(None, request.ws_extensions) 160 self.assertEqual(common.VERSION_HYBI08, request.ws_version) 161 162 def test_do_handshake_with_capitalized_value(self): 163 request_def = _create_good_request_def() 164 request_def.headers['upgrade'] = 'WEBSOCKET' 165 166 request = _create_request(request_def) 167 handshaker = _create_handshaker(request) 168 handshaker.do_handshake() 169 self.assertEqual( 170 _EXPECTED_RESPONSE, request.connection.written_data()) 171 172 request_def = _create_good_request_def() 173 request_def.headers['Connection'] = 'UPGRADE' 174 175 request = _create_request(request_def) 176 handshaker = _create_handshaker(request) 177 handshaker.do_handshake() 178 self.assertEqual( 179 _EXPECTED_RESPONSE, request.connection.written_data()) 180 181 def test_do_handshake_with_multiple_connection_values(self): 182 request_def = _create_good_request_def() 183 request_def.headers['Connection'] = 'Upgrade, keep-alive, , ' 184 185 request = _create_request(request_def) 186 handshaker = _create_handshaker(request) 187 handshaker.do_handshake() 188 self.assertEqual( 189 _EXPECTED_RESPONSE, request.connection.written_data()) 190 191 def test_aborting_handshake(self): 192 handshaker = Handshaker( 193 _create_request(_create_good_request_def()), 194 AbortingDispatcher()) 195 # do_extra_handshake raises an exception. Check that it's not caught by 196 # do_handshake. 197 self.assertRaises(HandshakeAbortedException, handshaker.do_handshake) 198 199 def test_do_handshake_with_protocol(self): 200 request_def = _create_good_request_def() 201 request_def.headers['Sec-WebSocket-Protocol'] = 'chat, superchat' 202 203 request = _create_request(request_def) 204 handshaker = Handshaker(request, SubprotocolChoosingDispatcher(0)) 205 handshaker.do_handshake() 206 207 EXPECTED_RESPONSE = ( 208 'HTTP/1.1 101 Switching Protocols\r\n' 209 'Upgrade: websocket\r\n' 210 'Connection: Upgrade\r\n' 211 'Sec-WebSocket-Accept: s3pPLMBiTxaQ9kYGzzhZRbK+xOo=\r\n' 212 'Sec-WebSocket-Protocol: chat\r\n\r\n') 213 214 self.assertEqual(EXPECTED_RESPONSE, request.connection.written_data()) 215 self.assertEqual('chat', request.ws_protocol) 216 217 def test_do_handshake_protocol_not_in_request_but_in_response(self): 218 request_def = _create_good_request_def() 219 request = _create_request(request_def) 220 handshaker = Handshaker( 221 request, SubprotocolChoosingDispatcher(-1, 'foobar')) 222 # No request has been made but ws_protocol is set. HandshakeException 223 # must be raised. 224 self.assertRaises(HandshakeException, handshaker.do_handshake) 225 226 def test_do_handshake_with_protocol_no_protocol_selection(self): 227 request_def = _create_good_request_def() 228 request_def.headers['Sec-WebSocket-Protocol'] = 'chat, superchat' 229 230 request = _create_request(request_def) 231 handshaker = _create_handshaker(request) 232 # ws_protocol is not set. HandshakeException must be raised. 233 self.assertRaises(HandshakeException, handshaker.do_handshake) 234 235 def test_do_handshake_with_extensions(self): 236 request_def = _create_good_request_def() 237 request_def.headers['Sec-WebSocket-Extensions'] = ( 238 'deflate-stream, unknown') 239 240 EXPECTED_RESPONSE = ( 241 'HTTP/1.1 101 Switching Protocols\r\n' 242 'Upgrade: websocket\r\n' 243 'Connection: Upgrade\r\n' 244 'Sec-WebSocket-Accept: s3pPLMBiTxaQ9kYGzzhZRbK+xOo=\r\n' 245 'Sec-WebSocket-Extensions: deflate-stream\r\n\r\n') 246 247 request = _create_request(request_def) 248 handshaker = _create_handshaker(request) 249 handshaker.do_handshake() 250 self.assertEqual(EXPECTED_RESPONSE, request.connection.written_data()) 251 self.assertEqual(1, len(request.ws_extensions)) 252 extension = request.ws_extensions[0] 253 self.assertEqual('deflate-stream', extension.name()) 254 self.assertEqual(0, len(extension.get_parameter_names())) 255 256 def test_do_handshake_with_quoted_extensions(self): 257 request_def = _create_good_request_def() 258 request_def.headers['Sec-WebSocket-Extensions'] = ( 259 'deflate-stream, , ' 260 'unknown; e = "mc^2"; ma="\r\n \\\rf "; pv=nrt') 261 262 request = _create_request(request_def) 263 handshaker = _create_handshaker(request) 264 self.assertRaises(HandshakeException, handshaker.do_handshake) 265 266 def test_do_handshake_with_optional_headers(self): 267 request_def = _create_good_request_def() 268 request_def.headers['EmptyValue'] = '' 269 request_def.headers['AKey'] = 'AValue' 270 271 request = _create_request(request_def) 272 handshaker = _create_handshaker(request) 273 handshaker.do_handshake() 274 self.assertEqual( 275 'AValue', request.headers_in['AKey']) 276 self.assertEqual( 277 '', request.headers_in['EmptyValue']) 278 279 def test_abort_extra_handshake(self): 280 handshaker = Handshaker( 281 _create_request(_create_good_request_def()), 282 AbortedByUserDispatcher()) 283 # do_extra_handshake raises an AbortedByUserException. Check that it's 284 # not caught by do_handshake. 285 self.assertRaises(AbortedByUserException, handshaker.do_handshake) 286 287 def test_bad_requests(self): 288 bad_cases = [ 289 ('HTTP request', 290 RequestDefinition( 291 'GET', '/demo', 292 {'Host': 'www.google.com', 293 'User-Agent': 294 'Mozilla/5.0 (Macintosh; U; Intel Mac OS X 10.5;' 295 ' en-US; rv:1.9.1.3) Gecko/20090824 Firefox/3.5.3' 296 ' GTB6 GTBA', 297 'Accept': 298 'text/html,application/xhtml+xml,application/xml;q=0.9,' 299 '*/*;q=0.8', 300 'Accept-Language': 'en-us,en;q=0.5', 301 'Accept-Encoding': 'gzip,deflate', 302 'Accept-Charset': 'ISO-8859-1,utf-8;q=0.7,*;q=0.7', 303 'Keep-Alive': '300', 304 'Connection': 'keep-alive'}), None, True)] 305 306 request_def = _create_good_request_def() 307 request_def.method = 'POST' 308 bad_cases.append(('Wrong method', request_def, None, True)) 309 310 request_def = _create_good_request_def() 311 del request_def.headers['Host'] 312 bad_cases.append(('Missing Host', request_def, None, True)) 313 314 request_def = _create_good_request_def() 315 del request_def.headers['Upgrade'] 316 bad_cases.append(('Missing Upgrade', request_def, None, True)) 317 318 request_def = _create_good_request_def() 319 request_def.headers['Upgrade'] = 'nonwebsocket' 320 bad_cases.append(('Wrong Upgrade', request_def, None, True)) 321 322 request_def = _create_good_request_def() 323 del request_def.headers['Connection'] 324 bad_cases.append(('Missing Connection', request_def, None, True)) 325 326 request_def = _create_good_request_def() 327 request_def.headers['Connection'] = 'Downgrade' 328 bad_cases.append(('Wrong Connection', request_def, None, True)) 329 330 request_def = _create_good_request_def() 331 del request_def.headers['Sec-WebSocket-Key'] 332 bad_cases.append(('Missing Sec-WebSocket-Key', request_def, 400, True)) 333 334 request_def = _create_good_request_def() 335 request_def.headers['Sec-WebSocket-Key'] = ( 336 'dGhlIHNhbXBsZSBub25jZQ==garbage') 337 bad_cases.append(('Wrong Sec-WebSocket-Key (with garbage on the tail)', 338 request_def, 400, True)) 339 340 request_def = _create_good_request_def() 341 request_def.headers['Sec-WebSocket-Key'] = 'YQ==' # BASE64 of 'a' 342 bad_cases.append( 343 ('Wrong Sec-WebSocket-Key (decoded value is not 16 octets long)', 344 request_def, 400, True)) 345 346 request_def = _create_good_request_def() 347 del request_def.headers['Sec-WebSocket-Version'] 348 bad_cases.append(('Missing Sec-WebSocket-Version', request_def, None, 349 True)) 350 351 request_def = _create_good_request_def() 352 request_def.headers['Sec-WebSocket-Version'] = '3' 353 bad_cases.append(('Wrong Sec-WebSocket-Version', request_def, None, 354 False)) 355 356 for (case_name, request_def, expected_status, 357 expect_handshake_exception) in bad_cases: 358 request = _create_request(request_def) 359 handshaker = Handshaker(request, mock.MockDispatcher()) 360 try: 361 handshaker.do_handshake() 362 self.fail('No exception thrown for \'%s\' case' % case_name) 363 except HandshakeException, e: 364 self.assertTrue(expect_handshake_exception) 365 self.assertEqual(expected_status, e.status) 366 except VersionException, e: 367 self.assertFalse(expect_handshake_exception) 368 369 370 if __name__ == '__main__': 371 unittest.main() 372 373 374 # vi:sts=4 sw=4 et 375