mirror of
				https://github.com/django/django.git
				synced 2025-10-25 14:46:09 +00:00 
			
		
		
		
	Fixed #18972 -- Refactored bundled wsgi server's chunking algorithm.
Thanks to amosonn at yahoo.com for the report, @doda for the initial patch and @datagrok for the revamped logic and test case.
This commit is contained in:
		
				
					committed by
					
						 Simon Charette
						Simon Charette
					
				
			
			
				
	
			
			
			
						parent
						
							80e68ee2ff
						
					
				
				
					commit
					a7960bcb35
				
			| @@ -9,7 +9,7 @@ been reviewed for security issues. DON'T USE IT FOR PRODUCTION USE! | |||||||
|  |  | ||||||
| from __future__ import unicode_literals | from __future__ import unicode_literals | ||||||
|  |  | ||||||
| import os | from io import BytesIO | ||||||
| import socket | import socket | ||||||
| import sys | import sys | ||||||
| import traceback | import traceback | ||||||
| @@ -26,7 +26,13 @@ from django.core.wsgi import get_wsgi_application | |||||||
| from django.utils.module_loading import import_by_path | from django.utils.module_loading import import_by_path | ||||||
| from django.utils import six | from django.utils import six | ||||||
|  |  | ||||||
| __all__ = ['WSGIServer', 'WSGIRequestHandler'] | __all__ = ('WSGIServer', 'WSGIRequestHandler', 'MAX_SOCKET_CHUNK_SIZE') | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # If data is too large, socket will choke, so write chunks no larger than 32MB | ||||||
|  | # at a time. The rationale behind the 32MB can be found on Django's Trac: | ||||||
|  | # https://code.djangoproject.com/ticket/5596#comment:4 | ||||||
|  | MAX_SOCKET_CHUNK_SIZE = 32 * 1024 * 1024  # 32 MB | ||||||
|  |  | ||||||
|  |  | ||||||
| def get_internal_wsgi_application(): | def get_internal_wsgi_application(): | ||||||
| @@ -78,19 +84,9 @@ class ServerHandler(simple_server.ServerHandler, object): | |||||||
|             self.bytes_sent += len(data) |             self.bytes_sent += len(data) | ||||||
|  |  | ||||||
|         # XXX check Content-Length and truncate if too many bytes written? |         # XXX check Content-Length and truncate if too many bytes written? | ||||||
|  |         data = BytesIO(data) | ||||||
|         # If data is too large, socket will choke, so write chunks no larger |         for chunk in iter(lambda: data.read(MAX_SOCKET_CHUNK_SIZE), b''): | ||||||
|         # than 32MB at a time. |             self._write(chunk) | ||||||
|         length = len(data) |  | ||||||
|         if length > 33554432: |  | ||||||
|             offset = 0 |  | ||||||
|             while offset < length: |  | ||||||
|                 chunk_size = min(33554432, length) |  | ||||||
|                 self._write(data[offset:offset+chunk_size]) |  | ||||||
|                 self._flush() |  | ||||||
|                 offset += chunk_size |  | ||||||
|         else: |  | ||||||
|             self._write(data) |  | ||||||
|             self._flush() |             self._flush() | ||||||
|  |  | ||||||
|     def error_output(self, environ, start_response): |     def error_output(self, environ, start_response): | ||||||
|   | |||||||
| @@ -2,22 +2,18 @@ from __future__ import unicode_literals | |||||||
|  |  | ||||||
| from io import BytesIO | from io import BytesIO | ||||||
|  |  | ||||||
| from django.core.servers.basehttp import ServerHandler | from django.core.servers.basehttp import ServerHandler, MAX_SOCKET_CHUNK_SIZE | ||||||
| from django.utils.unittest import TestCase | from django.utils.unittest import TestCase | ||||||
|  |  | ||||||
| # |  | ||||||
| # Tests for #9659: wsgi.file_wrapper in the builtin server. |  | ||||||
| # We need to mock a couple of handlers and keep track of what |  | ||||||
| # gets called when using a couple kinds of WSGI apps. |  | ||||||
| # |  | ||||||
|  |  | ||||||
| class DummyHandler(object): | class DummyHandler(object): | ||||||
|     def log_request(*args, **kwargs): |     def log_request(self, *args, **kwargs): | ||||||
|         pass |         pass | ||||||
|  |  | ||||||
|  |  | ||||||
| class FileWrapperHandler(ServerHandler): | class FileWrapperHandler(ServerHandler): | ||||||
|     def __init__(self, *args, **kwargs): |     def __init__(self, *args, **kwargs): | ||||||
|         ServerHandler.__init__(self, *args, **kwargs) |         super(FileWrapperHandler, self).__init__(*args, **kwargs) | ||||||
|         self.request_handler = DummyHandler() |         self.request_handler = DummyHandler() | ||||||
|         self._used_sendfile = False |         self._used_sendfile = False | ||||||
|  |  | ||||||
| @@ -25,17 +21,24 @@ class FileWrapperHandler(ServerHandler): | |||||||
|         self._used_sendfile = True |         self._used_sendfile = True | ||||||
|         return True |         return True | ||||||
|  |  | ||||||
|  |  | ||||||
| def wsgi_app(environ, start_response): | def wsgi_app(environ, start_response): | ||||||
|     start_response(str('200 OK'), [(str('Content-Type'), str('text/plain'))]) |     start_response(str('200 OK'), [(str('Content-Type'), str('text/plain'))]) | ||||||
|     return [b'Hello World!'] |     return [b'Hello World!'] | ||||||
|  |  | ||||||
|  |  | ||||||
| def wsgi_app_file_wrapper(environ, start_response): | def wsgi_app_file_wrapper(environ, start_response): | ||||||
|     start_response(str('200 OK'), [(str('Content-Type'), str('text/plain'))]) |     start_response(str('200 OK'), [(str('Content-Type'), str('text/plain'))]) | ||||||
|     return environ['wsgi.file_wrapper'](BytesIO(b'foo')) |     return environ['wsgi.file_wrapper'](BytesIO(b'foo')) | ||||||
|  |  | ||||||
|  |  | ||||||
| class WSGIFileWrapperTests(TestCase): | class WSGIFileWrapperTests(TestCase): | ||||||
|     """ |     """ | ||||||
|     Test that the wsgi.file_wrapper works for the builting server. |     Test that the wsgi.file_wrapper works for the builting server. | ||||||
|  |  | ||||||
|  |     Tests for #9659: wsgi.file_wrapper in the builtin server. | ||||||
|  |     We need to mock a couple of handlers and keep track of what | ||||||
|  |     gets called when using a couple kinds of WSGI apps. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     def test_file_wrapper_uses_sendfile(self): |     def test_file_wrapper_uses_sendfile(self): | ||||||
| @@ -53,3 +56,47 @@ class WSGIFileWrapperTests(TestCase): | |||||||
|         self.assertFalse(handler._used_sendfile) |         self.assertFalse(handler._used_sendfile) | ||||||
|         self.assertEqual(handler.stdout.getvalue().splitlines()[-1], b'Hello World!') |         self.assertEqual(handler.stdout.getvalue().splitlines()[-1], b'Hello World!') | ||||||
|         self.assertEqual(handler.stderr.getvalue(), b'') |         self.assertEqual(handler.stderr.getvalue(), b'') | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class WriteChunkCounterHandler(ServerHandler): | ||||||
|  |     """ | ||||||
|  |     Server handler that counts the number of chunks written after headers were | ||||||
|  |     sent. Used to make sure large response body chunking works properly. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     def __init__(self, *args, **kwargs): | ||||||
|  |         super(WriteChunkCounterHandler, self).__init__(*args, **kwargs) | ||||||
|  |         self.request_handler = DummyHandler() | ||||||
|  |         self.headers_written = False | ||||||
|  |         self.write_chunk_counter = 0 | ||||||
|  |  | ||||||
|  |     def send_headers(self): | ||||||
|  |         super(WriteChunkCounterHandler, self).send_headers() | ||||||
|  |         self.headers_written = True | ||||||
|  |  | ||||||
|  |     def _write(self, data): | ||||||
|  |         if self.headers_written: | ||||||
|  |             self.write_chunk_counter += 1 | ||||||
|  |         self.stdout.write(data) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def send_big_data_app(environ, start_response): | ||||||
|  |     start_response(str('200 OK'), [(str('Content-Type'), str('text/plain'))]) | ||||||
|  |     # Return a blob of data that is 1.5 times the maximum chunk size. | ||||||
|  |     return [b'x' * (MAX_SOCKET_CHUNK_SIZE + MAX_SOCKET_CHUNK_SIZE // 2)] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ServerHandlerChunksProperly(TestCase): | ||||||
|  |     """ | ||||||
|  |     Test that the ServerHandler chunks data properly. | ||||||
|  |  | ||||||
|  |     Tests for #18972: The logic that performs the math to break data into | ||||||
|  |     32MB (MAX_SOCKET_CHUNK_SIZE) chunks was flawed, BUT it didn't actually | ||||||
|  |     cause any problems. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     def test_chunked_data(self): | ||||||
|  |         env = {'SERVER_PROTOCOL': 'HTTP/1.0'} | ||||||
|  |         handler = WriteChunkCounterHandler(None, BytesIO(), BytesIO(), env) | ||||||
|  |         handler.run(send_big_data_app) | ||||||
|  |         self.assertEqual(handler.write_chunk_counter, 2) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user