2012-08-16 19:01:16 +08:00
|
|
|
from __future__ import unicode_literals
|
|
|
|
|
|
|
|
from io import BytesIO
|
|
|
|
|
2013-03-19 06:06:24 +08:00
|
|
|
from django.core.servers.basehttp import ServerHandler, MAX_SOCKET_CHUNK_SIZE
|
2010-10-11 20:55:17 +08:00
|
|
|
from django.utils.unittest import TestCase
|
2009-05-07 23:39:06 +08:00
|
|
|
|
|
|
|
|
|
|
|
class DummyHandler(object):
|
2013-03-19 06:06:24 +08:00
|
|
|
def log_request(self, *args, **kwargs):
|
2009-05-07 23:39:06 +08:00
|
|
|
pass
|
|
|
|
|
2013-03-19 06:06:24 +08:00
|
|
|
|
2009-05-07 23:39:06 +08:00
|
|
|
class FileWrapperHandler(ServerHandler):
|
|
|
|
def __init__(self, *args, **kwargs):
|
2013-03-19 06:06:24 +08:00
|
|
|
super(FileWrapperHandler, self).__init__(*args, **kwargs)
|
2009-05-07 23:39:06 +08:00
|
|
|
self.request_handler = DummyHandler()
|
|
|
|
self._used_sendfile = False
|
|
|
|
|
|
|
|
def sendfile(self):
|
|
|
|
self._used_sendfile = True
|
|
|
|
return True
|
|
|
|
|
2013-03-19 06:06:24 +08:00
|
|
|
|
2009-05-07 23:39:06 +08:00
|
|
|
def wsgi_app(environ, start_response):
|
2012-08-16 19:01:16 +08:00
|
|
|
start_response(str('200 OK'), [(str('Content-Type'), str('text/plain'))])
|
|
|
|
return [b'Hello World!']
|
2009-05-07 23:39:06 +08:00
|
|
|
|
2013-03-19 06:06:24 +08:00
|
|
|
|
2009-05-07 23:39:06 +08:00
|
|
|
def wsgi_app_file_wrapper(environ, start_response):
|
2012-08-16 19:01:16 +08:00
|
|
|
start_response(str('200 OK'), [(str('Content-Type'), str('text/plain'))])
|
|
|
|
return environ['wsgi.file_wrapper'](BytesIO(b'foo'))
|
2009-05-07 23:39:06 +08:00
|
|
|
|
2013-03-19 06:06:24 +08:00
|
|
|
|
2009-05-07 23:39:06 +08:00
|
|
|
class WSGIFileWrapperTests(TestCase):
|
|
|
|
"""
|
|
|
|
Test that the wsgi.file_wrapper works for the builting server.
|
2013-03-19 06:06:24 +08:00
|
|
|
|
|
|
|
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.
|
2009-05-07 23:39:06 +08:00
|
|
|
"""
|
|
|
|
|
|
|
|
def test_file_wrapper_uses_sendfile(self):
|
|
|
|
env = {'SERVER_PROTOCOL': 'HTTP/1.0'}
|
2012-08-16 19:01:16 +08:00
|
|
|
handler = FileWrapperHandler(None, BytesIO(), BytesIO(), env)
|
2009-05-07 23:39:06 +08:00
|
|
|
handler.run(wsgi_app_file_wrapper)
|
2011-03-03 23:04:39 +08:00
|
|
|
self.assertTrue(handler._used_sendfile)
|
2012-08-16 19:01:16 +08:00
|
|
|
self.assertEqual(handler.stdout.getvalue(), b'')
|
|
|
|
self.assertEqual(handler.stderr.getvalue(), b'')
|
2009-05-07 23:39:06 +08:00
|
|
|
|
|
|
|
def test_file_wrapper_no_sendfile(self):
|
|
|
|
env = {'SERVER_PROTOCOL': 'HTTP/1.0'}
|
2012-08-16 19:01:16 +08:00
|
|
|
handler = FileWrapperHandler(None, BytesIO(), BytesIO(), env)
|
2009-05-07 23:39:06 +08:00
|
|
|
handler.run(wsgi_app)
|
2010-12-04 15:28:12 +08:00
|
|
|
self.assertFalse(handler._used_sendfile)
|
2012-08-16 19:01:16 +08:00
|
|
|
self.assertEqual(handler.stdout.getvalue().splitlines()[-1], b'Hello World!')
|
|
|
|
self.assertEqual(handler.stderr.getvalue(), b'')
|
2013-03-19 06:06:24 +08:00
|
|
|
|
|
|
|
|
|
|
|
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)
|