import re import socket import subprocess import time import pytest from unit.applications.lang.python import TestApplicationPython prerequisites = {'modules': {'python': 'any'}} class TestSettings(TestApplicationPython): def sysctl(self): try: out = subprocess.check_output( ['sysctl', '-a'], stderr=subprocess.STDOUT ).decode() except FileNotFoundError: pytest.skip('requires sysctl') return out def test_settings_large_header_buffer_size(self): self.load('empty') def set_buffer_size(size): assert 'success' in self.conf( {'http': {'large_header_buffer_size': size}}, 'settings', ) def header_value(size, expect=200): headers = {'Host': 'a' * (size - 1), 'Connection': 'close'} assert self.get(headers=headers)['status'] == expect set_buffer_size(4096) header_value(4096) header_value(4097, 431) set_buffer_size(16384) header_value(16384) header_value(16385, 431) def test_settings_large_header_buffers(self): self.load('empty') def set_buffers(buffers): assert 'success' in self.conf( {'http': {'large_header_buffers': buffers}}, 'settings', ) def big_headers(headers_num, expect=200): headers = {'Host': 'localhost', 'Connection': 'close'} for i in range(headers_num): headers[f'Custom-header-{i}'] = 'a' * 8000 assert self.get(headers=headers)['status'] == expect set_buffers(1) big_headers(1) big_headers(2, 431) set_buffers(2) big_headers(2) big_headers(3, 431) set_buffers(8) big_headers(8) big_headers(9, 431) @pytest.mark.skip('not yet') def test_settings_large_header_buffer_invalid(self): def check_error(conf): assert 'error' in self.conf({'http': conf}, 'settings') check_error({'large_header_buffer_size': -1}) check_error({'large_header_buffer_size': 0}) check_error({'large_header_buffers': -1}) check_error({'large_header_buffers': 0}) def test_settings_server_version(self): self.load('empty') assert self.get()['headers']['Server'].startswith('Unit/') assert 'success' in self.conf( {"http": {"server_version": False}}, 'settings' ), 'remove version' assert self.get()['headers']['Server'] == 'Unit' assert 'success' in self.conf( {"http": {"server_version": True}}, 'settings' ), 'add version' assert self.get()['headers']['Server'].startswith('Unit/') def test_settings_header_read_timeout(self): self.load('empty') def req(): (_, sock) = self.http( b"""GET / HTTP/1.1 """, start=True, read_timeout=1, raw=True, ) time.sleep(3) return self.http( b"""Host: localhost Connection: close """, sock=sock, raw=True, ) assert 'success' in self.conf( {'http': {'header_read_timeout': 2}}, 'settings' ) assert req()['status'] == 408, 'status header read timeout' assert 'success' in self.conf( {'http': {'header_read_timeout': 7}}, 'settings' ) assert req()['status'] == 200, 'status header read timeout 2' def test_settings_header_read_timeout_update(self): self.load('empty') assert 'success' in self.conf( {'http': {'header_read_timeout': 4}}, 'settings' ) sock = self.http( b"""GET / HTTP/1.1 """, raw=True, no_recv=True, ) time.sleep(2) sock = self.http( b"""Host: localhost """, sock=sock, raw=True, no_recv=True, ) time.sleep(2) (resp, sock) = self.http( b"""X-Blah: blah """, start=True, sock=sock, read_timeout=1, raw=True, ) if len(resp) != 0: sock.close() else: time.sleep(2) resp = self.http( b"""Connection: close """, sock=sock, raw=True, ) assert resp['status'] == 408, 'status header read timeout update' def test_settings_body_read_timeout(self): self.load('empty') def req(): (_, sock) = self.http( b"""POST / HTTP/1.1 Host: localhost Content-Length: 10 Connection: close """, start=True, raw_resp=True, read_timeout=1, raw=True, ) time.sleep(3) return self.http(b"""0123456789""", sock=sock, raw=True) assert 'success' in self.conf( {'http': {'body_read_timeout': 2}}, 'settings' ) assert req()['status'] == 408, 'status body read timeout' assert 'success' in self.conf( {'http': {'body_read_timeout': 7}}, 'settings' ) assert req()['status'] == 200, 'status body read timeout 2' def test_settings_body_read_timeout_update(self): self.load('empty') assert 'success' in self.conf( {'http': {'body_read_timeout': 4}}, 'settings' ) (resp, sock) = self.http( b"""POST / HTTP/1.1 Host: localhost Content-Length: 10 Connection: close """, start=True, read_timeout=1, raw=True, ) time.sleep(2) (resp, sock) = self.http( b"""012""", start=True, sock=sock, read_timeout=1, raw=True ) time.sleep(2) (resp, sock) = self.http( b"""345""", start=True, sock=sock, read_timeout=1, raw=True ) time.sleep(2) resp = self.http(b"""6789""", sock=sock, raw=True) assert resp['status'] == 200, 'status body read timeout update' def test_settings_send_timeout(self, temp_dir): self.load('body_generate') def req(addr, data_len): sock = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM) sock.connect(addr) req = f"""GET / HTTP/1.1 Host: localhost X-Length: {data_len} Connection: close """ sock.sendall(req.encode()) data = sock.recv(16).decode() time.sleep(3) data += self.recvall(sock).decode() sock.close() return data sysctl_out = self.sysctl() values = re.findall( r'net.core.[rw]mem_(?:max|default).*?(\d+)', sysctl_out ) values = [int(v) for v in values] data_len = 1048576 if len(values) == 0 else 10 * max(values) addr = f'{temp_dir}/sock' assert 'success' in self.conf( {f'unix:{addr}': {'application': 'body_generate'}}, 'listeners' ) assert 'success' in self.conf({'http': {'send_timeout': 1}}, 'settings') data = req(addr, data_len) assert re.search(r'200 OK', data), 'send timeout status' assert len(data) < data_len, 'send timeout data ' self.conf({'http': {'send_timeout': 7}}, 'settings') data = req(addr, data_len) assert re.search(r'200 OK', data), 'send timeout status 2' assert len(data) > data_len, 'send timeout data 2' def test_settings_idle_timeout(self): self.load('empty') def req(): (_, sock) = self.get( headers={'Host': 'localhost', 'Connection': 'keep-alive'}, start=True, read_timeout=1, ) time.sleep(3) return self.get(sock=sock) assert self.get()['status'] == 200, 'init' assert 'success' in self.conf({'http': {'idle_timeout': 2}}, 'settings') assert req()['status'] == 408, 'status idle timeout' assert 'success' in self.conf({'http': {'idle_timeout': 7}}, 'settings') assert req()['status'] == 200, 'status idle timeout 2' def test_settings_idle_timeout_2(self): self.load('empty') def req(): sock = self.http(b'', raw=True, no_recv=True) time.sleep(3) return self.get(sock=sock) assert self.get()['status'] == 200, 'init' assert 'success' in self.conf({'http': {'idle_timeout': 1}}, 'settings') assert req()['status'] == 408, 'status idle timeout' assert 'success' in self.conf({'http': {'idle_timeout': 7}}, 'settings') assert req()['status'] == 200, 'status idle timeout 2' def test_settings_max_body_size(self): self.load('empty') assert 'success' in self.conf( {'http': {'max_body_size': 5}}, 'settings' ) assert self.post(body='01234')['status'] == 200, 'status size' assert self.post(body='012345')['status'] == 413, 'status size max' def test_settings_max_body_size_large(self): self.load('mirror') assert 'success' in self.conf( {'http': {'max_body_size': 32 * 1024 * 1024}}, 'settings' ) body = '0123456789abcdef' * 4 * 64 * 1024 resp = self.post(body=body, read_buffer_size=1024 * 1024) assert resp['status'] == 200, 'status size 4' assert resp['body'] == body, 'status body 4' body = '0123456789abcdef' * 8 * 64 * 1024 resp = self.post(body=body, read_buffer_size=1024 * 1024) assert resp['status'] == 200, 'status size 8' assert resp['body'] == body, 'status body 8' body = '0123456789abcdef' * 16 * 64 * 1024 resp = self.post(body=body, read_buffer_size=1024 * 1024) assert resp['status'] == 200, 'status size 16' assert resp['body'] == body, 'status body 16' body = '0123456789abcdef' * 32 * 64 * 1024 resp = self.post(body=body, read_buffer_size=1024 * 1024) assert resp['status'] == 200, 'status size 32' assert resp['body'] == body, 'status body 32' @pytest.mark.skip('not yet') def test_settings_negative_value(self): assert 'error' in self.conf( {'http': {'max_body_size': -1}}, 'settings' ), 'settings negative value' def test_settings_body_buffer_size(self): self.load('mirror') assert 'success' in self.conf( { 'http': { 'max_body_size': 64 * 1024 * 1024, 'body_buffer_size': 32 * 1024 * 1024, } }, 'settings', ) body = '0123456789abcdef' resp = self.post(body=body) assert bool(resp), 'response from application' assert resp['status'] == 200, 'status' assert resp['body'] == body, 'body' body = '0123456789abcdef' * 1024 * 1024 resp = self.post(body=body, read_buffer_size=1024 * 1024) assert bool(resp), 'response from application 2' assert resp['status'] == 200, 'status 2' assert resp['body'] == body, 'body 2' body = '0123456789abcdef' * 2 * 1024 * 1024 resp = self.post(body=body, read_buffer_size=1024 * 1024) assert bool(resp), 'response from application 3' assert resp['status'] == 200, 'status 3' assert resp['body'] == body, 'body 3' body = '0123456789abcdef' * 3 * 1024 * 1024 resp = self.post(body=body, read_buffer_size=1024 * 1024) assert bool(resp), 'response from application 4' assert resp['status'] == 200, 'status 4' assert resp['body'] == body, 'body 4' def test_settings_log_route(self, findall, search_in_file, wait_for_record): def count_fallbacks(): return len(findall(r'"fallback" taken')) def check_record(template): assert search_in_file(template) is not None def check_no_record(template): assert search_in_file(template) is None def template_req_line(url): return rf'\[notice\].*http request line "GET {url} HTTP/1\.1"' def template_selected(route): return rf'\[notice\].*"{route}" selected' def template_discarded(route): return rf'\[info\].*"{route}" discarded' def wait_for_request_log(status, uri, route): assert self.get(url=uri)['status'] == status assert wait_for_record(template_req_line(uri)) is not None assert wait_for_record(template_selected(route)) is not None # routes array assert 'success' in self.conf( { "listeners": {"*:7080": {"pass": "routes"}}, "routes": [ { "match": { "uri": "/zero", }, "action": {"return": 200}, }, { "action": {"return": 201}, }, ], "applications": {}, "settings": {"http": {"log_route": True}}, } ) wait_for_request_log(200, '/zero', 'routes/0') check_no_record(r'discarded') wait_for_request_log(201, '/one', 'routes/1') check_record(template_discarded('routes/0')) # routes object assert 'success' in self.conf( { "listeners": {"*:7080": {"pass": "routes/main"}}, "routes": { "main": [ { "match": { "uri": "/named_route", }, "action": {"return": 200}, }, { "action": {"return": 201}, }, ] }, "applications": {}, "settings": {"http": {"log_route": True}}, } ) wait_for_request_log(200, '/named_route', 'routes/main/0') check_no_record(template_discarded('routes/main')) wait_for_request_log(201, '/unnamed_route', 'routes/main/1') check_record(template_discarded('routes/main/0')) # routes sequence assert 'success' in self.conf( { "listeners": {"*:7080": {"pass": "routes/first"}}, "routes": { "first": [ { "action": {"pass": "routes/second"}, }, ], "second": [ { "action": {"return": 200}, }, ], }, "applications": {}, "settings": {"http": {"log_route": True}}, } ) wait_for_request_log(200, '/sequence', 'routes/second/0') check_record(template_selected('routes/first/0')) # fallback assert 'success' in self.conf( { "listeners": {"*:7080": {"pass": "routes/fall"}}, "routes": { "fall": [ { "action": { "share": "/blah", "fallback": {"pass": "routes/fall2"}, }, }, ], "fall2": [ { "action": {"return": 200}, }, ], }, "applications": {}, "settings": {"http": {"log_route": True}}, } ) wait_for_request_log(200, '/', 'routes/fall2/0') assert count_fallbacks() == 1 check_record(template_selected('routes/fall/0')) assert self.head()['status'] == 200 assert count_fallbacks() == 2 # disable log assert 'success' in self.conf({"log_route": False}, 'settings/http') url = '/disable_logging' assert self.get(url=url)['status'] == 200 time.sleep(1) check_no_record(template_req_line(url)) # total assert len(findall(r'\[notice\].*http request line')) == 7 assert len(findall(r'\[notice\].*selected')) == 10 assert len(findall(r'\[info\].*discarded')) == 2