You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1002 lines
35 KiB
1002 lines
35 KiB
5 years ago
|
import unittest
|
||
|
import io
|
||
|
|
||
|
|
||
|
class TestThreadedTaskDispatcher(unittest.TestCase):
|
||
|
def _makeOne(self):
|
||
|
from waitress.task import ThreadedTaskDispatcher
|
||
|
|
||
|
return ThreadedTaskDispatcher()
|
||
|
|
||
|
def test_handler_thread_task_raises(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.threads.add(0)
|
||
|
inst.logger = DummyLogger()
|
||
|
|
||
|
class BadDummyTask(DummyTask):
|
||
|
def service(self):
|
||
|
super(BadDummyTask, self).service()
|
||
|
inst.stop_count += 1
|
||
|
raise Exception
|
||
|
|
||
|
task = BadDummyTask()
|
||
|
inst.logger = DummyLogger()
|
||
|
inst.queue.append(task)
|
||
|
inst.active_count += 1
|
||
|
inst.handler_thread(0)
|
||
|
self.assertEqual(inst.stop_count, 0)
|
||
|
self.assertEqual(inst.active_count, 0)
|
||
|
self.assertEqual(inst.threads, set())
|
||
|
self.assertEqual(len(inst.logger.logged), 1)
|
||
|
|
||
|
def test_set_thread_count_increase(self):
|
||
|
inst = self._makeOne()
|
||
|
L = []
|
||
|
inst.start_new_thread = lambda *x: L.append(x)
|
||
|
inst.set_thread_count(1)
|
||
|
self.assertEqual(L, [(inst.handler_thread, (0,))])
|
||
|
|
||
|
def test_set_thread_count_increase_with_existing(self):
|
||
|
inst = self._makeOne()
|
||
|
L = []
|
||
|
inst.threads = {0}
|
||
|
inst.start_new_thread = lambda *x: L.append(x)
|
||
|
inst.set_thread_count(2)
|
||
|
self.assertEqual(L, [(inst.handler_thread, (1,))])
|
||
|
|
||
|
def test_set_thread_count_decrease(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.threads = {0, 1}
|
||
|
inst.set_thread_count(1)
|
||
|
self.assertEqual(inst.stop_count, 1)
|
||
|
|
||
|
def test_set_thread_count_same(self):
|
||
|
inst = self._makeOne()
|
||
|
L = []
|
||
|
inst.start_new_thread = lambda *x: L.append(x)
|
||
|
inst.threads = {0}
|
||
|
inst.set_thread_count(1)
|
||
|
self.assertEqual(L, [])
|
||
|
|
||
|
def test_add_task_with_idle_threads(self):
|
||
|
task = DummyTask()
|
||
|
inst = self._makeOne()
|
||
|
inst.threads.add(0)
|
||
|
inst.queue_logger = DummyLogger()
|
||
|
inst.add_task(task)
|
||
|
self.assertEqual(len(inst.queue), 1)
|
||
|
self.assertEqual(len(inst.queue_logger.logged), 0)
|
||
|
|
||
|
def test_add_task_with_all_busy_threads(self):
|
||
|
task = DummyTask()
|
||
|
inst = self._makeOne()
|
||
|
inst.queue_logger = DummyLogger()
|
||
|
inst.add_task(task)
|
||
|
self.assertEqual(len(inst.queue_logger.logged), 1)
|
||
|
inst.add_task(task)
|
||
|
self.assertEqual(len(inst.queue_logger.logged), 2)
|
||
|
|
||
|
def test_shutdown_one_thread(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.threads.add(0)
|
||
|
inst.logger = DummyLogger()
|
||
|
task = DummyTask()
|
||
|
inst.queue.append(task)
|
||
|
self.assertEqual(inst.shutdown(timeout=0.01), True)
|
||
|
self.assertEqual(
|
||
|
inst.logger.logged,
|
||
|
["1 thread(s) still running", "Canceling 1 pending task(s)",],
|
||
|
)
|
||
|
self.assertEqual(task.cancelled, True)
|
||
|
|
||
|
def test_shutdown_no_threads(self):
|
||
|
inst = self._makeOne()
|
||
|
self.assertEqual(inst.shutdown(timeout=0.01), True)
|
||
|
|
||
|
def test_shutdown_no_cancel_pending(self):
|
||
|
inst = self._makeOne()
|
||
|
self.assertEqual(inst.shutdown(cancel_pending=False, timeout=0.01), False)
|
||
|
|
||
|
|
||
|
class TestTask(unittest.TestCase):
|
||
|
def _makeOne(self, channel=None, request=None):
|
||
|
if channel is None:
|
||
|
channel = DummyChannel()
|
||
|
if request is None:
|
||
|
request = DummyParser()
|
||
|
from waitress.task import Task
|
||
|
|
||
|
return Task(channel, request)
|
||
|
|
||
|
def test_ctor_version_not_in_known(self):
|
||
|
request = DummyParser()
|
||
|
request.version = "8.4"
|
||
|
inst = self._makeOne(request=request)
|
||
|
self.assertEqual(inst.version, "1.0")
|
||
|
|
||
|
def test_build_response_header_bad_http_version(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.request = DummyParser()
|
||
|
inst.version = "8.4"
|
||
|
self.assertRaises(AssertionError, inst.build_response_header)
|
||
|
|
||
|
def test_build_response_header_v10_keepalive_no_content_length(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.request = DummyParser()
|
||
|
inst.request.headers["CONNECTION"] = "keep-alive"
|
||
|
inst.version = "1.0"
|
||
|
result = inst.build_response_header()
|
||
|
lines = filter_lines(result)
|
||
|
self.assertEqual(len(lines), 4)
|
||
|
self.assertEqual(lines[0], b"HTTP/1.0 200 OK")
|
||
|
self.assertEqual(lines[1], b"Connection: close")
|
||
|
self.assertTrue(lines[2].startswith(b"Date:"))
|
||
|
self.assertEqual(lines[3], b"Server: waitress")
|
||
|
self.assertEqual(inst.close_on_finish, True)
|
||
|
self.assertTrue(("Connection", "close") in inst.response_headers)
|
||
|
|
||
|
def test_build_response_header_v10_keepalive_with_content_length(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.request = DummyParser()
|
||
|
inst.request.headers["CONNECTION"] = "keep-alive"
|
||
|
inst.response_headers = [("Content-Length", "10")]
|
||
|
inst.version = "1.0"
|
||
|
inst.content_length = 0
|
||
|
result = inst.build_response_header()
|
||
|
lines = filter_lines(result)
|
||
|
self.assertEqual(len(lines), 5)
|
||
|
self.assertEqual(lines[0], b"HTTP/1.0 200 OK")
|
||
|
self.assertEqual(lines[1], b"Connection: Keep-Alive")
|
||
|
self.assertEqual(lines[2], b"Content-Length: 10")
|
||
|
self.assertTrue(lines[3].startswith(b"Date:"))
|
||
|
self.assertEqual(lines[4], b"Server: waitress")
|
||
|
self.assertEqual(inst.close_on_finish, False)
|
||
|
|
||
|
def test_build_response_header_v11_connection_closed_by_client(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.request = DummyParser()
|
||
|
inst.version = "1.1"
|
||
|
inst.request.headers["CONNECTION"] = "close"
|
||
|
result = inst.build_response_header()
|
||
|
lines = filter_lines(result)
|
||
|
self.assertEqual(len(lines), 5)
|
||
|
self.assertEqual(lines[0], b"HTTP/1.1 200 OK")
|
||
|
self.assertEqual(lines[1], b"Connection: close")
|
||
|
self.assertTrue(lines[2].startswith(b"Date:"))
|
||
|
self.assertEqual(lines[3], b"Server: waitress")
|
||
|
self.assertEqual(lines[4], b"Transfer-Encoding: chunked")
|
||
|
self.assertTrue(("Connection", "close") in inst.response_headers)
|
||
|
self.assertEqual(inst.close_on_finish, True)
|
||
|
|
||
|
def test_build_response_header_v11_connection_keepalive_by_client(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.request = DummyParser()
|
||
|
inst.request.headers["CONNECTION"] = "keep-alive"
|
||
|
inst.version = "1.1"
|
||
|
result = inst.build_response_header()
|
||
|
lines = filter_lines(result)
|
||
|
self.assertEqual(len(lines), 5)
|
||
|
self.assertEqual(lines[0], b"HTTP/1.1 200 OK")
|
||
|
self.assertEqual(lines[1], b"Connection: close")
|
||
|
self.assertTrue(lines[2].startswith(b"Date:"))
|
||
|
self.assertEqual(lines[3], b"Server: waitress")
|
||
|
self.assertEqual(lines[4], b"Transfer-Encoding: chunked")
|
||
|
self.assertTrue(("Connection", "close") in inst.response_headers)
|
||
|
self.assertEqual(inst.close_on_finish, True)
|
||
|
|
||
|
def test_build_response_header_v11_200_no_content_length(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.request = DummyParser()
|
||
|
inst.version = "1.1"
|
||
|
result = inst.build_response_header()
|
||
|
lines = filter_lines(result)
|
||
|
self.assertEqual(len(lines), 5)
|
||
|
self.assertEqual(lines[0], b"HTTP/1.1 200 OK")
|
||
|
self.assertEqual(lines[1], b"Connection: close")
|
||
|
self.assertTrue(lines[2].startswith(b"Date:"))
|
||
|
self.assertEqual(lines[3], b"Server: waitress")
|
||
|
self.assertEqual(lines[4], b"Transfer-Encoding: chunked")
|
||
|
self.assertEqual(inst.close_on_finish, True)
|
||
|
self.assertTrue(("Connection", "close") in inst.response_headers)
|
||
|
|
||
|
def test_build_response_header_v11_204_no_content_length_or_transfer_encoding(self):
|
||
|
# RFC 7230: MUST NOT send Transfer-Encoding or Content-Length
|
||
|
# for any response with a status code of 1xx or 204.
|
||
|
inst = self._makeOne()
|
||
|
inst.request = DummyParser()
|
||
|
inst.version = "1.1"
|
||
|
inst.status = "204 No Content"
|
||
|
result = inst.build_response_header()
|
||
|
lines = filter_lines(result)
|
||
|
self.assertEqual(len(lines), 4)
|
||
|
self.assertEqual(lines[0], b"HTTP/1.1 204 No Content")
|
||
|
self.assertEqual(lines[1], b"Connection: close")
|
||
|
self.assertTrue(lines[2].startswith(b"Date:"))
|
||
|
self.assertEqual(lines[3], b"Server: waitress")
|
||
|
self.assertEqual(inst.close_on_finish, True)
|
||
|
self.assertTrue(("Connection", "close") in inst.response_headers)
|
||
|
|
||
|
def test_build_response_header_v11_1xx_no_content_length_or_transfer_encoding(self):
|
||
|
# RFC 7230: MUST NOT send Transfer-Encoding or Content-Length
|
||
|
# for any response with a status code of 1xx or 204.
|
||
|
inst = self._makeOne()
|
||
|
inst.request = DummyParser()
|
||
|
inst.version = "1.1"
|
||
|
inst.status = "100 Continue"
|
||
|
result = inst.build_response_header()
|
||
|
lines = filter_lines(result)
|
||
|
self.assertEqual(len(lines), 4)
|
||
|
self.assertEqual(lines[0], b"HTTP/1.1 100 Continue")
|
||
|
self.assertEqual(lines[1], b"Connection: close")
|
||
|
self.assertTrue(lines[2].startswith(b"Date:"))
|
||
|
self.assertEqual(lines[3], b"Server: waitress")
|
||
|
self.assertEqual(inst.close_on_finish, True)
|
||
|
self.assertTrue(("Connection", "close") in inst.response_headers)
|
||
|
|
||
|
def test_build_response_header_v11_304_no_content_length_or_transfer_encoding(self):
|
||
|
# RFC 7230: MUST NOT send Transfer-Encoding or Content-Length
|
||
|
# for any response with a status code of 1xx, 204 or 304.
|
||
|
inst = self._makeOne()
|
||
|
inst.request = DummyParser()
|
||
|
inst.version = "1.1"
|
||
|
inst.status = "304 Not Modified"
|
||
|
result = inst.build_response_header()
|
||
|
lines = filter_lines(result)
|
||
|
self.assertEqual(len(lines), 4)
|
||
|
self.assertEqual(lines[0], b"HTTP/1.1 304 Not Modified")
|
||
|
self.assertEqual(lines[1], b"Connection: close")
|
||
|
self.assertTrue(lines[2].startswith(b"Date:"))
|
||
|
self.assertEqual(lines[3], b"Server: waitress")
|
||
|
self.assertEqual(inst.close_on_finish, True)
|
||
|
self.assertTrue(("Connection", "close") in inst.response_headers)
|
||
|
|
||
|
def test_build_response_header_via_added(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.request = DummyParser()
|
||
|
inst.version = "1.0"
|
||
|
inst.response_headers = [("Server", "abc")]
|
||
|
result = inst.build_response_header()
|
||
|
lines = filter_lines(result)
|
||
|
self.assertEqual(len(lines), 5)
|
||
|
self.assertEqual(lines[0], b"HTTP/1.0 200 OK")
|
||
|
self.assertEqual(lines[1], b"Connection: close")
|
||
|
self.assertTrue(lines[2].startswith(b"Date:"))
|
||
|
self.assertEqual(lines[3], b"Server: abc")
|
||
|
self.assertEqual(lines[4], b"Via: waitress")
|
||
|
|
||
|
def test_build_response_header_date_exists(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.request = DummyParser()
|
||
|
inst.version = "1.0"
|
||
|
inst.response_headers = [("Date", "date")]
|
||
|
result = inst.build_response_header()
|
||
|
lines = filter_lines(result)
|
||
|
self.assertEqual(len(lines), 4)
|
||
|
self.assertEqual(lines[0], b"HTTP/1.0 200 OK")
|
||
|
self.assertEqual(lines[1], b"Connection: close")
|
||
|
self.assertTrue(lines[2].startswith(b"Date:"))
|
||
|
self.assertEqual(lines[3], b"Server: waitress")
|
||
|
|
||
|
def test_build_response_header_preexisting_content_length(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.request = DummyParser()
|
||
|
inst.version = "1.1"
|
||
|
inst.content_length = 100
|
||
|
result = inst.build_response_header()
|
||
|
lines = filter_lines(result)
|
||
|
self.assertEqual(len(lines), 4)
|
||
|
self.assertEqual(lines[0], b"HTTP/1.1 200 OK")
|
||
|
self.assertEqual(lines[1], b"Content-Length: 100")
|
||
|
self.assertTrue(lines[2].startswith(b"Date:"))
|
||
|
self.assertEqual(lines[3], b"Server: waitress")
|
||
|
|
||
|
def test_remove_content_length_header(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.response_headers = [("Content-Length", "70")]
|
||
|
inst.remove_content_length_header()
|
||
|
self.assertEqual(inst.response_headers, [])
|
||
|
|
||
|
def test_remove_content_length_header_with_other(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.response_headers = [
|
||
|
("Content-Length", "70"),
|
||
|
("Content-Type", "text/html"),
|
||
|
]
|
||
|
inst.remove_content_length_header()
|
||
|
self.assertEqual(inst.response_headers, [("Content-Type", "text/html")])
|
||
|
|
||
|
def test_start(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.start()
|
||
|
self.assertTrue(inst.start_time)
|
||
|
|
||
|
def test_finish_didnt_write_header(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.wrote_header = False
|
||
|
inst.complete = True
|
||
|
inst.finish()
|
||
|
self.assertTrue(inst.channel.written)
|
||
|
|
||
|
def test_finish_wrote_header(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.wrote_header = True
|
||
|
inst.finish()
|
||
|
self.assertFalse(inst.channel.written)
|
||
|
|
||
|
def test_finish_chunked_response(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.wrote_header = True
|
||
|
inst.chunked_response = True
|
||
|
inst.finish()
|
||
|
self.assertEqual(inst.channel.written, b"0\r\n\r\n")
|
||
|
|
||
|
def test_write_wrote_header(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.wrote_header = True
|
||
|
inst.complete = True
|
||
|
inst.content_length = 3
|
||
|
inst.write(b"abc")
|
||
|
self.assertEqual(inst.channel.written, b"abc")
|
||
|
|
||
|
def test_write_header_not_written(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.wrote_header = False
|
||
|
inst.complete = True
|
||
|
inst.write(b"abc")
|
||
|
self.assertTrue(inst.channel.written)
|
||
|
self.assertEqual(inst.wrote_header, True)
|
||
|
|
||
|
def test_write_start_response_uncalled(self):
|
||
|
inst = self._makeOne()
|
||
|
self.assertRaises(RuntimeError, inst.write, b"")
|
||
|
|
||
|
def test_write_chunked_response(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.wrote_header = True
|
||
|
inst.chunked_response = True
|
||
|
inst.complete = True
|
||
|
inst.write(b"abc")
|
||
|
self.assertEqual(inst.channel.written, b"3\r\nabc\r\n")
|
||
|
|
||
|
def test_write_preexisting_content_length(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.wrote_header = True
|
||
|
inst.complete = True
|
||
|
inst.content_length = 1
|
||
|
inst.logger = DummyLogger()
|
||
|
inst.write(b"abc")
|
||
|
self.assertTrue(inst.channel.written)
|
||
|
self.assertEqual(inst.logged_write_excess, True)
|
||
|
self.assertEqual(len(inst.logger.logged), 1)
|
||
|
|
||
|
|
||
|
class TestWSGITask(unittest.TestCase):
|
||
|
def _makeOne(self, channel=None, request=None):
|
||
|
if channel is None:
|
||
|
channel = DummyChannel()
|
||
|
if request is None:
|
||
|
request = DummyParser()
|
||
|
from waitress.task import WSGITask
|
||
|
|
||
|
return WSGITask(channel, request)
|
||
|
|
||
|
def test_service(self):
|
||
|
inst = self._makeOne()
|
||
|
|
||
|
def execute():
|
||
|
inst.executed = True
|
||
|
|
||
|
inst.execute = execute
|
||
|
inst.complete = True
|
||
|
inst.service()
|
||
|
self.assertTrue(inst.start_time)
|
||
|
self.assertTrue(inst.close_on_finish)
|
||
|
self.assertTrue(inst.channel.written)
|
||
|
self.assertEqual(inst.executed, True)
|
||
|
|
||
|
def test_service_server_raises_socket_error(self):
|
||
|
import socket
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
|
||
|
def execute():
|
||
|
raise socket.error
|
||
|
|
||
|
inst.execute = execute
|
||
|
self.assertRaises(socket.error, inst.service)
|
||
|
self.assertTrue(inst.start_time)
|
||
|
self.assertTrue(inst.close_on_finish)
|
||
|
self.assertFalse(inst.channel.written)
|
||
|
|
||
|
def test_execute_app_calls_start_response_twice_wo_exc_info(self):
|
||
|
def app(environ, start_response):
|
||
|
start_response("200 OK", [])
|
||
|
start_response("200 OK", [])
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.channel.server.application = app
|
||
|
self.assertRaises(AssertionError, inst.execute)
|
||
|
|
||
|
def test_execute_app_calls_start_response_w_exc_info_complete(self):
|
||
|
def app(environ, start_response):
|
||
|
start_response("200 OK", [], [ValueError, ValueError(), None])
|
||
|
return [b"a"]
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.complete = True
|
||
|
inst.channel.server.application = app
|
||
|
inst.execute()
|
||
|
self.assertTrue(inst.complete)
|
||
|
self.assertEqual(inst.status, "200 OK")
|
||
|
self.assertTrue(inst.channel.written)
|
||
|
|
||
|
def test_execute_app_calls_start_response_w_excinf_headers_unwritten(self):
|
||
|
def app(environ, start_response):
|
||
|
start_response("200 OK", [], [ValueError, None, None])
|
||
|
return [b"a"]
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.wrote_header = False
|
||
|
inst.channel.server.application = app
|
||
|
inst.response_headers = [("a", "b")]
|
||
|
inst.execute()
|
||
|
self.assertTrue(inst.complete)
|
||
|
self.assertEqual(inst.status, "200 OK")
|
||
|
self.assertTrue(inst.channel.written)
|
||
|
self.assertFalse(("a", "b") in inst.response_headers)
|
||
|
|
||
|
def test_execute_app_calls_start_response_w_excinf_headers_written(self):
|
||
|
def app(environ, start_response):
|
||
|
start_response("200 OK", [], [ValueError, ValueError(), None])
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.complete = True
|
||
|
inst.wrote_header = True
|
||
|
inst.channel.server.application = app
|
||
|
self.assertRaises(ValueError, inst.execute)
|
||
|
|
||
|
def test_execute_bad_header_key(self):
|
||
|
def app(environ, start_response):
|
||
|
start_response("200 OK", [(None, "a")])
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.channel.server.application = app
|
||
|
self.assertRaises(AssertionError, inst.execute)
|
||
|
|
||
|
def test_execute_bad_header_value(self):
|
||
|
def app(environ, start_response):
|
||
|
start_response("200 OK", [("a", None)])
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.channel.server.application = app
|
||
|
self.assertRaises(AssertionError, inst.execute)
|
||
|
|
||
|
def test_execute_hopbyhop_header(self):
|
||
|
def app(environ, start_response):
|
||
|
start_response("200 OK", [("Connection", "close")])
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.channel.server.application = app
|
||
|
self.assertRaises(AssertionError, inst.execute)
|
||
|
|
||
|
def test_execute_bad_header_value_control_characters(self):
|
||
|
def app(environ, start_response):
|
||
|
start_response("200 OK", [("a", "\n")])
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.channel.server.application = app
|
||
|
self.assertRaises(ValueError, inst.execute)
|
||
|
|
||
|
def test_execute_bad_header_name_control_characters(self):
|
||
|
def app(environ, start_response):
|
||
|
start_response("200 OK", [("a\r", "value")])
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.channel.server.application = app
|
||
|
self.assertRaises(ValueError, inst.execute)
|
||
|
|
||
|
def test_execute_bad_status_control_characters(self):
|
||
|
def app(environ, start_response):
|
||
|
start_response("200 OK\r", [])
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.channel.server.application = app
|
||
|
self.assertRaises(ValueError, inst.execute)
|
||
|
|
||
|
def test_preserve_header_value_order(self):
|
||
|
def app(environ, start_response):
|
||
|
write = start_response("200 OK", [("C", "b"), ("A", "b"), ("A", "a")])
|
||
|
write(b"abc")
|
||
|
return []
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.channel.server.application = app
|
||
|
inst.execute()
|
||
|
self.assertTrue(b"A: b\r\nA: a\r\nC: b\r\n" in inst.channel.written)
|
||
|
|
||
|
def test_execute_bad_status_value(self):
|
||
|
def app(environ, start_response):
|
||
|
start_response(None, [])
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.channel.server.application = app
|
||
|
self.assertRaises(AssertionError, inst.execute)
|
||
|
|
||
|
def test_execute_with_content_length_header(self):
|
||
|
def app(environ, start_response):
|
||
|
start_response("200 OK", [("Content-Length", "1")])
|
||
|
return [b"a"]
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.channel.server.application = app
|
||
|
inst.execute()
|
||
|
self.assertEqual(inst.content_length, 1)
|
||
|
|
||
|
def test_execute_app_calls_write(self):
|
||
|
def app(environ, start_response):
|
||
|
write = start_response("200 OK", [("Content-Length", "3")])
|
||
|
write(b"abc")
|
||
|
return []
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.channel.server.application = app
|
||
|
inst.execute()
|
||
|
self.assertEqual(inst.channel.written[-3:], b"abc")
|
||
|
|
||
|
def test_execute_app_returns_len1_chunk_without_cl(self):
|
||
|
def app(environ, start_response):
|
||
|
start_response("200 OK", [])
|
||
|
return [b"abc"]
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.channel.server.application = app
|
||
|
inst.execute()
|
||
|
self.assertEqual(inst.content_length, 3)
|
||
|
|
||
|
def test_execute_app_returns_empty_chunk_as_first(self):
|
||
|
def app(environ, start_response):
|
||
|
start_response("200 OK", [])
|
||
|
return ["", b"abc"]
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.channel.server.application = app
|
||
|
inst.execute()
|
||
|
self.assertEqual(inst.content_length, None)
|
||
|
|
||
|
def test_execute_app_returns_too_many_bytes(self):
|
||
|
def app(environ, start_response):
|
||
|
start_response("200 OK", [("Content-Length", "1")])
|
||
|
return [b"abc"]
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.channel.server.application = app
|
||
|
inst.logger = DummyLogger()
|
||
|
inst.execute()
|
||
|
self.assertEqual(inst.close_on_finish, True)
|
||
|
self.assertEqual(len(inst.logger.logged), 1)
|
||
|
|
||
|
def test_execute_app_returns_too_few_bytes(self):
|
||
|
def app(environ, start_response):
|
||
|
start_response("200 OK", [("Content-Length", "3")])
|
||
|
return [b"a"]
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.channel.server.application = app
|
||
|
inst.logger = DummyLogger()
|
||
|
inst.execute()
|
||
|
self.assertEqual(inst.close_on_finish, True)
|
||
|
self.assertEqual(len(inst.logger.logged), 1)
|
||
|
|
||
|
def test_execute_app_do_not_warn_on_head(self):
|
||
|
def app(environ, start_response):
|
||
|
start_response("200 OK", [("Content-Length", "3")])
|
||
|
return [b""]
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.request.command = "HEAD"
|
||
|
inst.channel.server.application = app
|
||
|
inst.logger = DummyLogger()
|
||
|
inst.execute()
|
||
|
self.assertEqual(inst.close_on_finish, True)
|
||
|
self.assertEqual(len(inst.logger.logged), 0)
|
||
|
|
||
|
def test_execute_app_without_body_204_logged(self):
|
||
|
def app(environ, start_response):
|
||
|
start_response("204 No Content", [("Content-Length", "3")])
|
||
|
return [b"abc"]
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.channel.server.application = app
|
||
|
inst.logger = DummyLogger()
|
||
|
inst.execute()
|
||
|
self.assertEqual(inst.close_on_finish, True)
|
||
|
self.assertNotIn(b"abc", inst.channel.written)
|
||
|
self.assertNotIn(b"Content-Length", inst.channel.written)
|
||
|
self.assertNotIn(b"Transfer-Encoding", inst.channel.written)
|
||
|
self.assertEqual(len(inst.logger.logged), 1)
|
||
|
|
||
|
def test_execute_app_without_body_304_logged(self):
|
||
|
def app(environ, start_response):
|
||
|
start_response("304 Not Modified", [("Content-Length", "3")])
|
||
|
return [b"abc"]
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.channel.server.application = app
|
||
|
inst.logger = DummyLogger()
|
||
|
inst.execute()
|
||
|
self.assertEqual(inst.close_on_finish, True)
|
||
|
self.assertNotIn(b"abc", inst.channel.written)
|
||
|
self.assertNotIn(b"Content-Length", inst.channel.written)
|
||
|
self.assertNotIn(b"Transfer-Encoding", inst.channel.written)
|
||
|
self.assertEqual(len(inst.logger.logged), 1)
|
||
|
|
||
|
def test_execute_app_returns_closeable(self):
|
||
|
class closeable(list):
|
||
|
def close(self):
|
||
|
self.closed = True
|
||
|
|
||
|
foo = closeable([b"abc"])
|
||
|
|
||
|
def app(environ, start_response):
|
||
|
start_response("200 OK", [("Content-Length", "3")])
|
||
|
return foo
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.channel.server.application = app
|
||
|
inst.execute()
|
||
|
self.assertEqual(foo.closed, True)
|
||
|
|
||
|
def test_execute_app_returns_filewrapper_prepare_returns_True(self):
|
||
|
from waitress.buffers import ReadOnlyFileBasedBuffer
|
||
|
|
||
|
f = io.BytesIO(b"abc")
|
||
|
app_iter = ReadOnlyFileBasedBuffer(f, 8192)
|
||
|
|
||
|
def app(environ, start_response):
|
||
|
start_response("200 OK", [("Content-Length", "3")])
|
||
|
return app_iter
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.channel.server.application = app
|
||
|
inst.execute()
|
||
|
self.assertTrue(inst.channel.written) # header
|
||
|
self.assertEqual(inst.channel.otherdata, [app_iter])
|
||
|
|
||
|
def test_execute_app_returns_filewrapper_prepare_returns_True_nocl(self):
|
||
|
from waitress.buffers import ReadOnlyFileBasedBuffer
|
||
|
|
||
|
f = io.BytesIO(b"abc")
|
||
|
app_iter = ReadOnlyFileBasedBuffer(f, 8192)
|
||
|
|
||
|
def app(environ, start_response):
|
||
|
start_response("200 OK", [])
|
||
|
return app_iter
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.channel.server.application = app
|
||
|
inst.execute()
|
||
|
self.assertTrue(inst.channel.written) # header
|
||
|
self.assertEqual(inst.channel.otherdata, [app_iter])
|
||
|
self.assertEqual(inst.content_length, 3)
|
||
|
|
||
|
def test_execute_app_returns_filewrapper_prepare_returns_True_badcl(self):
|
||
|
from waitress.buffers import ReadOnlyFileBasedBuffer
|
||
|
|
||
|
f = io.BytesIO(b"abc")
|
||
|
app_iter = ReadOnlyFileBasedBuffer(f, 8192)
|
||
|
|
||
|
def app(environ, start_response):
|
||
|
start_response("200 OK", [])
|
||
|
return app_iter
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
inst.channel.server.application = app
|
||
|
inst.content_length = 10
|
||
|
inst.response_headers = [("Content-Length", "10")]
|
||
|
inst.execute()
|
||
|
self.assertTrue(inst.channel.written) # header
|
||
|
self.assertEqual(inst.channel.otherdata, [app_iter])
|
||
|
self.assertEqual(inst.content_length, 3)
|
||
|
self.assertEqual(dict(inst.response_headers)["Content-Length"], "3")
|
||
|
|
||
|
def test_get_environment_already_cached(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.environ = object()
|
||
|
self.assertEqual(inst.get_environment(), inst.environ)
|
||
|
|
||
|
def test_get_environment_path_startswith_more_than_one_slash(self):
|
||
|
inst = self._makeOne()
|
||
|
request = DummyParser()
|
||
|
request.path = "///abc"
|
||
|
inst.request = request
|
||
|
environ = inst.get_environment()
|
||
|
self.assertEqual(environ["PATH_INFO"], "/abc")
|
||
|
|
||
|
def test_get_environment_path_empty(self):
|
||
|
inst = self._makeOne()
|
||
|
request = DummyParser()
|
||
|
request.path = ""
|
||
|
inst.request = request
|
||
|
environ = inst.get_environment()
|
||
|
self.assertEqual(environ["PATH_INFO"], "")
|
||
|
|
||
|
def test_get_environment_no_query(self):
|
||
|
inst = self._makeOne()
|
||
|
request = DummyParser()
|
||
|
inst.request = request
|
||
|
environ = inst.get_environment()
|
||
|
self.assertEqual(environ["QUERY_STRING"], "")
|
||
|
|
||
|
def test_get_environment_with_query(self):
|
||
|
inst = self._makeOne()
|
||
|
request = DummyParser()
|
||
|
request.query = "abc"
|
||
|
inst.request = request
|
||
|
environ = inst.get_environment()
|
||
|
self.assertEqual(environ["QUERY_STRING"], "abc")
|
||
|
|
||
|
def test_get_environ_with_url_prefix_miss(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.channel.server.adj.url_prefix = "/foo"
|
||
|
request = DummyParser()
|
||
|
request.path = "/bar"
|
||
|
inst.request = request
|
||
|
environ = inst.get_environment()
|
||
|
self.assertEqual(environ["PATH_INFO"], "/bar")
|
||
|
self.assertEqual(environ["SCRIPT_NAME"], "/foo")
|
||
|
|
||
|
def test_get_environ_with_url_prefix_hit(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.channel.server.adj.url_prefix = "/foo"
|
||
|
request = DummyParser()
|
||
|
request.path = "/foo/fuz"
|
||
|
inst.request = request
|
||
|
environ = inst.get_environment()
|
||
|
self.assertEqual(environ["PATH_INFO"], "/fuz")
|
||
|
self.assertEqual(environ["SCRIPT_NAME"], "/foo")
|
||
|
|
||
|
def test_get_environ_with_url_prefix_empty_path(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.channel.server.adj.url_prefix = "/foo"
|
||
|
request = DummyParser()
|
||
|
request.path = "/foo"
|
||
|
inst.request = request
|
||
|
environ = inst.get_environment()
|
||
|
self.assertEqual(environ["PATH_INFO"], "")
|
||
|
self.assertEqual(environ["SCRIPT_NAME"], "/foo")
|
||
|
|
||
|
def test_get_environment_values(self):
|
||
|
import sys
|
||
|
|
||
|
inst = self._makeOne()
|
||
|
request = DummyParser()
|
||
|
request.headers = {
|
||
|
"CONTENT_TYPE": "abc",
|
||
|
"CONTENT_LENGTH": "10",
|
||
|
"X_FOO": "BAR",
|
||
|
"CONNECTION": "close",
|
||
|
}
|
||
|
request.query = "abc"
|
||
|
inst.request = request
|
||
|
environ = inst.get_environment()
|
||
|
|
||
|
# nail the keys of environ
|
||
|
self.assertEqual(
|
||
|
sorted(environ.keys()),
|
||
|
[
|
||
|
"CONTENT_LENGTH",
|
||
|
"CONTENT_TYPE",
|
||
|
"HTTP_CONNECTION",
|
||
|
"HTTP_X_FOO",
|
||
|
"PATH_INFO",
|
||
|
"QUERY_STRING",
|
||
|
"REMOTE_ADDR",
|
||
|
"REMOTE_HOST",
|
||
|
"REMOTE_PORT",
|
||
|
"REQUEST_METHOD",
|
||
|
"SCRIPT_NAME",
|
||
|
"SERVER_NAME",
|
||
|
"SERVER_PORT",
|
||
|
"SERVER_PROTOCOL",
|
||
|
"SERVER_SOFTWARE",
|
||
|
"wsgi.errors",
|
||
|
"wsgi.file_wrapper",
|
||
|
"wsgi.input",
|
||
|
"wsgi.input_terminated",
|
||
|
"wsgi.multiprocess",
|
||
|
"wsgi.multithread",
|
||
|
"wsgi.run_once",
|
||
|
"wsgi.url_scheme",
|
||
|
"wsgi.version",
|
||
|
],
|
||
|
)
|
||
|
|
||
|
self.assertEqual(environ["REQUEST_METHOD"], "GET")
|
||
|
self.assertEqual(environ["SERVER_PORT"], "80")
|
||
|
self.assertEqual(environ["SERVER_NAME"], "localhost")
|
||
|
self.assertEqual(environ["SERVER_SOFTWARE"], "waitress")
|
||
|
self.assertEqual(environ["SERVER_PROTOCOL"], "HTTP/1.0")
|
||
|
self.assertEqual(environ["SCRIPT_NAME"], "")
|
||
|
self.assertEqual(environ["HTTP_CONNECTION"], "close")
|
||
|
self.assertEqual(environ["PATH_INFO"], "/")
|
||
|
self.assertEqual(environ["QUERY_STRING"], "abc")
|
||
|
self.assertEqual(environ["REMOTE_ADDR"], "127.0.0.1")
|
||
|
self.assertEqual(environ["REMOTE_HOST"], "127.0.0.1")
|
||
|
self.assertEqual(environ["REMOTE_PORT"], "39830")
|
||
|
self.assertEqual(environ["CONTENT_TYPE"], "abc")
|
||
|
self.assertEqual(environ["CONTENT_LENGTH"], "10")
|
||
|
self.assertEqual(environ["HTTP_X_FOO"], "BAR")
|
||
|
self.assertEqual(environ["wsgi.version"], (1, 0))
|
||
|
self.assertEqual(environ["wsgi.url_scheme"], "http")
|
||
|
self.assertEqual(environ["wsgi.errors"], sys.stderr)
|
||
|
self.assertEqual(environ["wsgi.multithread"], True)
|
||
|
self.assertEqual(environ["wsgi.multiprocess"], False)
|
||
|
self.assertEqual(environ["wsgi.run_once"], False)
|
||
|
self.assertEqual(environ["wsgi.input"], "stream")
|
||
|
self.assertEqual(environ["wsgi.input_terminated"], True)
|
||
|
self.assertEqual(inst.environ, environ)
|
||
|
|
||
|
|
||
|
class TestErrorTask(unittest.TestCase):
|
||
|
def _makeOne(self, channel=None, request=None):
|
||
|
if channel is None:
|
||
|
channel = DummyChannel()
|
||
|
if request is None:
|
||
|
request = DummyParser()
|
||
|
request.error = self._makeDummyError()
|
||
|
from waitress.task import ErrorTask
|
||
|
|
||
|
return ErrorTask(channel, request)
|
||
|
|
||
|
def _makeDummyError(self):
|
||
|
from waitress.utilities import Error
|
||
|
|
||
|
e = Error("body")
|
||
|
e.code = 432
|
||
|
e.reason = "Too Ugly"
|
||
|
return e
|
||
|
|
||
|
def test_execute_http_10(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.execute()
|
||
|
lines = filter_lines(inst.channel.written)
|
||
|
self.assertEqual(len(lines), 9)
|
||
|
self.assertEqual(lines[0], b"HTTP/1.0 432 Too Ugly")
|
||
|
self.assertEqual(lines[1], b"Connection: close")
|
||
|
self.assertEqual(lines[2], b"Content-Length: 43")
|
||
|
self.assertEqual(lines[3], b"Content-Type: text/plain")
|
||
|
self.assertTrue(lines[4])
|
||
|
self.assertEqual(lines[5], b"Server: waitress")
|
||
|
self.assertEqual(lines[6], b"Too Ugly")
|
||
|
self.assertEqual(lines[7], b"body")
|
||
|
self.assertEqual(lines[8], b"(generated by waitress)")
|
||
|
|
||
|
def test_execute_http_11(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.version = "1.1"
|
||
|
inst.execute()
|
||
|
lines = filter_lines(inst.channel.written)
|
||
|
self.assertEqual(len(lines), 9)
|
||
|
self.assertEqual(lines[0], b"HTTP/1.1 432 Too Ugly")
|
||
|
self.assertEqual(lines[1], b"Connection: close")
|
||
|
self.assertEqual(lines[2], b"Content-Length: 43")
|
||
|
self.assertEqual(lines[3], b"Content-Type: text/plain")
|
||
|
self.assertTrue(lines[4])
|
||
|
self.assertEqual(lines[5], b"Server: waitress")
|
||
|
self.assertEqual(lines[6], b"Too Ugly")
|
||
|
self.assertEqual(lines[7], b"body")
|
||
|
self.assertEqual(lines[8], b"(generated by waitress)")
|
||
|
|
||
|
def test_execute_http_11_close(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.version = "1.1"
|
||
|
inst.request.headers["CONNECTION"] = "close"
|
||
|
inst.execute()
|
||
|
lines = filter_lines(inst.channel.written)
|
||
|
self.assertEqual(len(lines), 9)
|
||
|
self.assertEqual(lines[0], b"HTTP/1.1 432 Too Ugly")
|
||
|
self.assertEqual(lines[1], b"Connection: close")
|
||
|
self.assertEqual(lines[2], b"Content-Length: 43")
|
||
|
self.assertEqual(lines[3], b"Content-Type: text/plain")
|
||
|
self.assertTrue(lines[4])
|
||
|
self.assertEqual(lines[5], b"Server: waitress")
|
||
|
self.assertEqual(lines[6], b"Too Ugly")
|
||
|
self.assertEqual(lines[7], b"body")
|
||
|
self.assertEqual(lines[8], b"(generated by waitress)")
|
||
|
|
||
|
def test_execute_http_11_keep_forces_close(self):
|
||
|
inst = self._makeOne()
|
||
|
inst.version = "1.1"
|
||
|
inst.request.headers["CONNECTION"] = "keep-alive"
|
||
|
inst.execute()
|
||
|
lines = filter_lines(inst.channel.written)
|
||
|
self.assertEqual(len(lines), 9)
|
||
|
self.assertEqual(lines[0], b"HTTP/1.1 432 Too Ugly")
|
||
|
self.assertEqual(lines[1], b"Connection: close")
|
||
|
self.assertEqual(lines[2], b"Content-Length: 43")
|
||
|
self.assertEqual(lines[3], b"Content-Type: text/plain")
|
||
|
self.assertTrue(lines[4])
|
||
|
self.assertEqual(lines[5], b"Server: waitress")
|
||
|
self.assertEqual(lines[6], b"Too Ugly")
|
||
|
self.assertEqual(lines[7], b"body")
|
||
|
self.assertEqual(lines[8], b"(generated by waitress)")
|
||
|
|
||
|
|
||
|
class DummyTask(object):
|
||
|
serviced = False
|
||
|
cancelled = False
|
||
|
|
||
|
def service(self):
|
||
|
self.serviced = True
|
||
|
|
||
|
def cancel(self):
|
||
|
self.cancelled = True
|
||
|
|
||
|
|
||
|
class DummyAdj(object):
|
||
|
log_socket_errors = True
|
||
|
ident = "waitress"
|
||
|
host = "127.0.0.1"
|
||
|
port = 80
|
||
|
url_prefix = ""
|
||
|
|
||
|
|
||
|
class DummyServer(object):
|
||
|
server_name = "localhost"
|
||
|
effective_port = 80
|
||
|
|
||
|
def __init__(self):
|
||
|
self.adj = DummyAdj()
|
||
|
|
||
|
|
||
|
class DummyChannel(object):
|
||
|
closed_when_done = False
|
||
|
adj = DummyAdj()
|
||
|
creation_time = 0
|
||
|
addr = ("127.0.0.1", 39830)
|
||
|
|
||
|
def __init__(self, server=None):
|
||
|
if server is None:
|
||
|
server = DummyServer()
|
||
|
self.server = server
|
||
|
self.written = b""
|
||
|
self.otherdata = []
|
||
|
|
||
|
def write_soon(self, data):
|
||
|
if isinstance(data, bytes):
|
||
|
self.written += data
|
||
|
else:
|
||
|
self.otherdata.append(data)
|
||
|
return len(data)
|
||
|
|
||
|
|
||
|
class DummyParser(object):
|
||
|
version = "1.0"
|
||
|
command = "GET"
|
||
|
path = "/"
|
||
|
query = ""
|
||
|
url_scheme = "http"
|
||
|
expect_continue = False
|
||
|
headers_finished = False
|
||
|
|
||
|
def __init__(self):
|
||
|
self.headers = {}
|
||
|
|
||
|
def get_body_stream(self):
|
||
|
return "stream"
|
||
|
|
||
|
|
||
|
def filter_lines(s):
|
||
|
return list(filter(None, s.split(b"\r\n")))
|
||
|
|
||
|
|
||
|
class DummyLogger(object):
|
||
|
def __init__(self):
|
||
|
self.logged = []
|
||
|
|
||
|
def warning(self, msg, *args):
|
||
|
self.logged.append(msg % args)
|
||
|
|
||
|
def exception(self, msg, *args):
|
||
|
self.logged.append(msg % args)
|