5033405db7
Account for situations where no Content-Length or Transfer-Encoding headers are present in a response. The length of the response.content could still indicate a response was given. The earlier raise_for_status() should catch bad HTTP responses that return content. Closes-bug: #1766660 Change-Id: I6739e26a039a898982f5cdc96d19d6992ed37737
108 lines
3.2 KiB
Python
108 lines
3.2 KiB
Python
import functools
|
|
import json
|
|
from multiprocessing import Process
|
|
from multiprocessing import Queue
|
|
import traceback
|
|
|
|
from mock import Mock
|
|
import requests
|
|
from six.moves import socketserver
|
|
|
|
|
|
class TestsTimeoutException(Exception):
|
|
pass
|
|
|
|
|
|
def time_limit(seconds, fp, func, *args, **kwargs):
|
|
|
|
if fp:
|
|
if not hasattr(fp, 'write'):
|
|
raise TypeError("Expected 'file-like' object, got '%s'" % fp)
|
|
else:
|
|
def record(msg):
|
|
fp.write(msg)
|
|
else:
|
|
def record(msg):
|
|
return
|
|
|
|
def capture_results(msg_queue, func, *args, **kwargs):
|
|
try:
|
|
result = func(*args, **kwargs)
|
|
except Exception as e:
|
|
msg_queue.put(
|
|
"Running function '%s' resulted in exception '%s' with "
|
|
"message: '%s'\n" % (func.__name__, e.__class__.__name__, e))
|
|
# no point re-raising an exception from the subprocess, instead
|
|
# return False
|
|
return False
|
|
else:
|
|
msg_queue.put(
|
|
"Running function '%s' finished with result '%s', and"
|
|
"stack:\n%s\n" % (func.__name__, result,
|
|
traceback.format_stack()))
|
|
return result
|
|
|
|
messages = Queue()
|
|
# although creating a separate process is expensive it's the only way to
|
|
# ensure cross platform that we can cleanly terminate after timeout
|
|
p = Process(target=functools.partial(capture_results, messages, func),
|
|
args=args, kwargs=kwargs)
|
|
p.start()
|
|
p.join(seconds)
|
|
if p.is_alive():
|
|
p.terminate()
|
|
while not messages.empty():
|
|
record(messages.get())
|
|
record("Running function '%s' did not finish\n" % func.__name__)
|
|
|
|
raise TestsTimeoutException
|
|
else:
|
|
while not messages.empty():
|
|
record(messages.get())
|
|
record("Running function '%s' finished with exit code '%s'\n"
|
|
% (func.__name__, p.exitcode))
|
|
|
|
|
|
class NullServer(socketserver.TCPServer):
|
|
|
|
request_queue_size = 1
|
|
|
|
def __init__(self, server_address, *args, **kwargs):
|
|
# TCPServer is old style in python 2.x so cannot use
|
|
# super() correctly, explicitly call __init__.
|
|
|
|
# simply init'ing is sufficient to open the port, which
|
|
# with the server not started creates a black hole server
|
|
socketserver.TCPServer.__init__(
|
|
self, server_address, socketserver.BaseRequestHandler,
|
|
*args, **kwargs)
|
|
|
|
|
|
def build_response_mock(status_code, json_body=None, headers=None, **kwargs):
|
|
real_response = requests.Response()
|
|
real_response.status_code = status_code
|
|
|
|
text = None
|
|
if json_body is not None:
|
|
text = json.dumps(json_body)
|
|
if headers is not {}:
|
|
real_response.headers['content-length'] = len(text)
|
|
|
|
if headers is not None:
|
|
for k, v in headers.items():
|
|
real_response.headers[k] = v
|
|
|
|
for k, v in kwargs.items():
|
|
setattr(real_response, k, v)
|
|
|
|
response = Mock(wraps=real_response, autospec=True)
|
|
if text:
|
|
response.text = text
|
|
|
|
# for some reason, wraps cannot handle attributes which are dicts
|
|
# and accessed by key
|
|
response.headers = real_response.headers
|
|
response.content = text
|
|
|
|
return response
|