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.

567 lines
16 KiB

  1. """Utilities shared by tests."""
  2. import collections
  3. import contextlib
  4. import io
  5. import logging
  6. import os
  7. import re
  8. import selectors
  9. import socket
  10. import socketserver
  11. import sys
  12. import tempfile
  13. import threading
  14. import time
  15. import unittest
  16. import weakref
  17. from unittest import mock
  18. from http.server import HTTPServer
  19. from wsgiref.simple_server import WSGIRequestHandler, WSGIServer
  20. try:
  21. import ssl
  22. except ImportError: # pragma: no cover
  23. ssl = None
  24. from asyncio import base_events
  25. from asyncio import events
  26. from asyncio import format_helpers
  27. from asyncio import futures
  28. from asyncio import tasks
  29. from asyncio.log import logger
  30. from test import support
  31. def data_file(filename):
  32. if hasattr(support, 'TEST_HOME_DIR'):
  33. fullname = os.path.join(support.TEST_HOME_DIR, filename)
  34. if os.path.isfile(fullname):
  35. return fullname
  36. fullname = os.path.join(os.path.dirname(__file__), filename)
  37. if os.path.isfile(fullname):
  38. return fullname
  39. raise FileNotFoundError(filename)
  40. ONLYCERT = data_file('ssl_cert.pem')
  41. ONLYKEY = data_file('ssl_key.pem')
  42. SIGNED_CERTFILE = data_file('keycert3.pem')
  43. SIGNING_CA = data_file('pycacert.pem')
  44. PEERCERT = {
  45. 'OCSP': ('http://testca.pythontest.net/testca/ocsp/',),
  46. 'caIssuers': ('http://testca.pythontest.net/testca/pycacert.cer',),
  47. 'crlDistributionPoints': ('http://testca.pythontest.net/testca/revocation.crl',),
  48. 'issuer': ((('countryName', 'XY'),),
  49. (('organizationName', 'Python Software Foundation CA'),),
  50. (('commonName', 'our-ca-server'),)),
  51. 'notAfter': 'Nov 28 19:09:06 2027 GMT',
  52. 'notBefore': 'Jan 19 19:09:06 2018 GMT',
  53. 'serialNumber': '82EDBF41C880919C',
  54. 'subject': ((('countryName', 'XY'),),
  55. (('localityName', 'Castle Anthrax'),),
  56. (('organizationName', 'Python Software Foundation'),),
  57. (('commonName', 'localhost'),)),
  58. 'subjectAltName': (('DNS', 'localhost'),),
  59. 'version': 3
  60. }
  61. def simple_server_sslcontext():
  62. server_context = ssl.SSLContext(ssl.PROTOCOL_TLS_SERVER)
  63. server_context.load_cert_chain(ONLYCERT, ONLYKEY)
  64. server_context.check_hostname = False
  65. server_context.verify_mode = ssl.CERT_NONE
  66. return server_context
  67. def simple_client_sslcontext():
  68. client_context = ssl.SSLContext(ssl.PROTOCOL_TLS_CLIENT)
  69. client_context.check_hostname = False
  70. client_context.verify_mode = ssl.CERT_NONE
  71. return client_context
  72. def dummy_ssl_context():
  73. if ssl is None:
  74. return None
  75. else:
  76. return ssl.SSLContext(ssl.PROTOCOL_TLS)
  77. def run_briefly(loop):
  78. async def once():
  79. pass
  80. gen = once()
  81. t = loop.create_task(gen)
  82. # Don't log a warning if the task is not done after run_until_complete().
  83. # It occurs if the loop is stopped or if a task raises a BaseException.
  84. t._log_destroy_pending = False
  85. try:
  86. loop.run_until_complete(t)
  87. finally:
  88. gen.close()
  89. def run_until(loop, pred, timeout=30):
  90. deadline = time.time() + timeout
  91. while not pred():
  92. if timeout is not None:
  93. timeout = deadline - time.time()
  94. if timeout <= 0:
  95. raise futures.TimeoutError()
  96. loop.run_until_complete(tasks.sleep(0.001, loop=loop))
  97. def run_once(loop):
  98. """Legacy API to run once through the event loop.
  99. This is the recommended pattern for test code. It will poll the
  100. selector once and run all callbacks scheduled in response to I/O
  101. events.
  102. """
  103. loop.call_soon(loop.stop)
  104. loop.run_forever()
  105. class SilentWSGIRequestHandler(WSGIRequestHandler):
  106. def get_stderr(self):
  107. return io.StringIO()
  108. def log_message(self, format, *args):
  109. pass
  110. class SilentWSGIServer(WSGIServer):
  111. request_timeout = 2
  112. def get_request(self):
  113. request, client_addr = super().get_request()
  114. request.settimeout(self.request_timeout)
  115. return request, client_addr
  116. def handle_error(self, request, client_address):
  117. pass
  118. class SSLWSGIServerMixin:
  119. def finish_request(self, request, client_address):
  120. # The relative location of our test directory (which
  121. # contains the ssl key and certificate files) differs
  122. # between the stdlib and stand-alone asyncio.
  123. # Prefer our own if we can find it.
  124. here = os.path.join(os.path.dirname(__file__), '..', 'tests')
  125. if not os.path.isdir(here):
  126. here = os.path.join(os.path.dirname(os.__file__),
  127. 'test', 'test_asyncio')
  128. keyfile = os.path.join(here, 'ssl_key.pem')
  129. certfile = os.path.join(here, 'ssl_cert.pem')
  130. context = ssl.SSLContext()
  131. context.load_cert_chain(certfile, keyfile)
  132. ssock = context.wrap_socket(request, server_side=True)
  133. try:
  134. self.RequestHandlerClass(ssock, client_address, self)
  135. ssock.close()
  136. except OSError:
  137. # maybe socket has been closed by peer
  138. pass
  139. class SSLWSGIServer(SSLWSGIServerMixin, SilentWSGIServer):
  140. pass
  141. def _run_test_server(*, address, use_ssl=False, server_cls, server_ssl_cls):
  142. def app(environ, start_response):
  143. status = '200 OK'
  144. headers = [('Content-type', 'text/plain')]
  145. start_response(status, headers)
  146. return [b'Test message']
  147. # Run the test WSGI server in a separate thread in order not to
  148. # interfere with event handling in the main thread
  149. server_class = server_ssl_cls if use_ssl else server_cls
  150. httpd = server_class(address, SilentWSGIRequestHandler)
  151. httpd.set_app(app)
  152. httpd.address = httpd.server_address
  153. server_thread = threading.Thread(
  154. target=lambda: httpd.serve_forever(poll_interval=0.05))
  155. server_thread.start()
  156. try:
  157. yield httpd
  158. finally:
  159. httpd.shutdown()
  160. httpd.server_close()
  161. server_thread.join()
  162. if hasattr(socket, 'AF_UNIX'):
  163. class UnixHTTPServer(socketserver.UnixStreamServer, HTTPServer):
  164. def server_bind(self):
  165. socketserver.UnixStreamServer.server_bind(self)
  166. self.server_name = '127.0.0.1'
  167. self.server_port = 80
  168. class UnixWSGIServer(UnixHTTPServer, WSGIServer):
  169. request_timeout = 2
  170. def server_bind(self):
  171. UnixHTTPServer.server_bind(self)
  172. self.setup_environ()
  173. def get_request(self):
  174. request, client_addr = super().get_request()
  175. request.settimeout(self.request_timeout)
  176. # Code in the stdlib expects that get_request
  177. # will return a socket and a tuple (host, port).
  178. # However, this isn't true for UNIX sockets,
  179. # as the second return value will be a path;
  180. # hence we return some fake data sufficient
  181. # to get the tests going
  182. return request, ('127.0.0.1', '')
  183. class SilentUnixWSGIServer(UnixWSGIServer):
  184. def handle_error(self, request, client_address):
  185. pass
  186. class UnixSSLWSGIServer(SSLWSGIServerMixin, SilentUnixWSGIServer):
  187. pass
  188. def gen_unix_socket_path():
  189. with tempfile.NamedTemporaryFile() as file:
  190. return file.name
  191. @contextlib.contextmanager
  192. def unix_socket_path():
  193. path = gen_unix_socket_path()
  194. try:
  195. yield path
  196. finally:
  197. try:
  198. os.unlink(path)
  199. except OSError:
  200. pass
  201. @contextlib.contextmanager
  202. def run_test_unix_server(*, use_ssl=False):
  203. with unix_socket_path() as path:
  204. yield from _run_test_server(address=path, use_ssl=use_ssl,
  205. server_cls=SilentUnixWSGIServer,
  206. server_ssl_cls=UnixSSLWSGIServer)
  207. @contextlib.contextmanager
  208. def run_test_server(*, host='127.0.0.1', port=0, use_ssl=False):
  209. yield from _run_test_server(address=(host, port), use_ssl=use_ssl,
  210. server_cls=SilentWSGIServer,
  211. server_ssl_cls=SSLWSGIServer)
  212. def make_test_protocol(base):
  213. dct = {}
  214. for name in dir(base):
  215. if name.startswith('__') and name.endswith('__'):
  216. # skip magic names
  217. continue
  218. dct[name] = MockCallback(return_value=None)
  219. return type('TestProtocol', (base,) + base.__bases__, dct)()
  220. class TestSelector(selectors.BaseSelector):
  221. def __init__(self):
  222. self.keys = {}
  223. def register(self, fileobj, events, data=None):
  224. key = selectors.SelectorKey(fileobj, 0, events, data)
  225. self.keys[fileobj] = key
  226. return key
  227. def unregister(self, fileobj):
  228. return self.keys.pop(fileobj)
  229. def select(self, timeout):
  230. return []
  231. def get_map(self):
  232. return self.keys
  233. class TestLoop(base_events.BaseEventLoop):
  234. """Loop for unittests.
  235. It manages self time directly.
  236. If something scheduled to be executed later then
  237. on next loop iteration after all ready handlers done
  238. generator passed to __init__ is calling.
  239. Generator should be like this:
  240. def gen():
  241. ...
  242. when = yield ...
  243. ... = yield time_advance
  244. Value returned by yield is absolute time of next scheduled handler.
  245. Value passed to yield is time advance to move loop's time forward.
  246. """
  247. def __init__(self, gen=None):
  248. super().__init__()
  249. if gen is None:
  250. def gen():
  251. yield
  252. self._check_on_close = False
  253. else:
  254. self._check_on_close = True
  255. self._gen = gen()
  256. next(self._gen)
  257. self._time = 0
  258. self._clock_resolution = 1e-9
  259. self._timers = []
  260. self._selector = TestSelector()
  261. self.readers = {}
  262. self.writers = {}
  263. self.reset_counters()
  264. self._transports = weakref.WeakValueDictionary()
  265. def time(self):
  266. return self._time
  267. def advance_time(self, advance):
  268. """Move test time forward."""
  269. if advance:
  270. self._time += advance
  271. def close(self):
  272. super().close()
  273. if self._check_on_close:
  274. try:
  275. self._gen.send(0)
  276. except StopIteration:
  277. pass
  278. else: # pragma: no cover
  279. raise AssertionError("Time generator is not finished")
  280. def _add_reader(self, fd, callback, *args):
  281. self.readers[fd] = events.Handle(callback, args, self, None)
  282. def _remove_reader(self, fd):
  283. self.remove_reader_count[fd] += 1
  284. if fd in self.readers:
  285. del self.readers[fd]
  286. return True
  287. else:
  288. return False
  289. def assert_reader(self, fd, callback, *args):
  290. if fd not in self.readers:
  291. raise AssertionError(f'fd {fd} is not registered')
  292. handle = self.readers[fd]
  293. if handle._callback != callback:
  294. raise AssertionError(
  295. f'unexpected callback: {handle._callback} != {callback}')
  296. if handle._args != args:
  297. raise AssertionError(
  298. f'unexpected callback args: {handle._args} != {args}')
  299. def assert_no_reader(self, fd):
  300. if fd in self.readers:
  301. raise AssertionError(f'fd {fd} is registered')
  302. def _add_writer(self, fd, callback, *args):
  303. self.writers[fd] = events.Handle(callback, args, self, None)
  304. def _remove_writer(self, fd):
  305. self.remove_writer_count[fd] += 1
  306. if fd in self.writers:
  307. del self.writers[fd]
  308. return True
  309. else:
  310. return False
  311. def assert_writer(self, fd, callback, *args):
  312. assert fd in self.writers, 'fd {} is not registered'.format(fd)
  313. handle = self.writers[fd]
  314. assert handle._callback == callback, '{!r} != {!r}'.format(
  315. handle._callback, callback)
  316. assert handle._args == args, '{!r} != {!r}'.format(
  317. handle._args, args)
  318. def _ensure_fd_no_transport(self, fd):
  319. if not isinstance(fd, int):
  320. try:
  321. fd = int(fd.fileno())
  322. except (AttributeError, TypeError, ValueError):
  323. # This code matches selectors._fileobj_to_fd function.
  324. raise ValueError("Invalid file object: "
  325. "{!r}".format(fd)) from None
  326. try:
  327. transport = self._transports[fd]
  328. except KeyError:
  329. pass
  330. else:
  331. raise RuntimeError(
  332. 'File descriptor {!r} is used by transport {!r}'.format(
  333. fd, transport))
  334. def add_reader(self, fd, callback, *args):
  335. """Add a reader callback."""
  336. self._ensure_fd_no_transport(fd)
  337. return self._add_reader(fd, callback, *args)
  338. def remove_reader(self, fd):
  339. """Remove a reader callback."""
  340. self._ensure_fd_no_transport(fd)
  341. return self._remove_reader(fd)
  342. def add_writer(self, fd, callback, *args):
  343. """Add a writer callback.."""
  344. self._ensure_fd_no_transport(fd)
  345. return self._add_writer(fd, callback, *args)
  346. def remove_writer(self, fd):
  347. """Remove a writer callback."""
  348. self._ensure_fd_no_transport(fd)
  349. return self._remove_writer(fd)
  350. def reset_counters(self):
  351. self.remove_reader_count = collections.defaultdict(int)
  352. self.remove_writer_count = collections.defaultdict(int)
  353. def _run_once(self):
  354. super()._run_once()
  355. for when in self._timers:
  356. advance = self._gen.send(when)
  357. self.advance_time(advance)
  358. self._timers = []
  359. def call_at(self, when, callback, *args, context=None):
  360. self._timers.append(when)
  361. return super().call_at(when, callback, *args, context=context)
  362. def _process_events(self, event_list):
  363. return
  364. def _write_to_self(self):
  365. pass
  366. def MockCallback(**kwargs):
  367. return mock.Mock(spec=['__call__'], **kwargs)
  368. class MockPattern(str):
  369. """A regex based str with a fuzzy __eq__.
  370. Use this helper with 'mock.assert_called_with', or anywhere
  371. where a regex comparison between strings is needed.
  372. For instance:
  373. mock_call.assert_called_with(MockPattern('spam.*ham'))
  374. """
  375. def __eq__(self, other):
  376. return bool(re.search(str(self), other, re.S))
  377. class MockInstanceOf:
  378. def __init__(self, type):
  379. self._type = type
  380. def __eq__(self, other):
  381. return isinstance(other, self._type)
  382. def get_function_source(func):
  383. source = format_helpers._get_function_source(func)
  384. if source is None:
  385. raise ValueError("unable to get the source of %r" % (func,))
  386. return source
  387. class TestCase(unittest.TestCase):
  388. @staticmethod
  389. def close_loop(loop):
  390. executor = loop._default_executor
  391. if executor is not None:
  392. executor.shutdown(wait=True)
  393. loop.close()
  394. def set_event_loop(self, loop, *, cleanup=True):
  395. assert loop is not None
  396. # ensure that the event loop is passed explicitly in asyncio
  397. events.set_event_loop(None)
  398. if cleanup:
  399. self.addCleanup(self.close_loop, loop)
  400. def new_test_loop(self, gen=None):
  401. loop = TestLoop(gen)
  402. self.set_event_loop(loop)
  403. return loop
  404. def unpatch_get_running_loop(self):
  405. events._get_running_loop = self._get_running_loop
  406. def setUp(self):
  407. self._get_running_loop = events._get_running_loop
  408. events._get_running_loop = lambda: None
  409. self._thread_cleanup = support.threading_setup()
  410. def tearDown(self):
  411. self.unpatch_get_running_loop()
  412. events.set_event_loop(None)
  413. # Detect CPython bug #23353: ensure that yield/yield-from is not used
  414. # in an except block of a generator
  415. self.assertEqual(sys.exc_info(), (None, None, None))
  416. self.doCleanups()
  417. support.threading_cleanup(*self._thread_cleanup)
  418. support.reap_children()
  419. @contextlib.contextmanager
  420. def disable_logger():
  421. """Context manager to disable asyncio logger.
  422. For example, it can be used to ignore warnings in debug mode.
  423. """
  424. old_level = logger.level
  425. try:
  426. logger.setLevel(logging.CRITICAL+1)
  427. yield
  428. finally:
  429. logger.setLevel(old_level)
  430. def mock_nonblocking_socket(proto=socket.IPPROTO_TCP, type=socket.SOCK_STREAM,
  431. family=socket.AF_INET):
  432. """Create a mock of a non-blocking socket."""
  433. sock = mock.MagicMock(socket.socket)
  434. sock.proto = proto
  435. sock.type = type
  436. sock.family = family
  437. sock.gettimeout.return_value = 0.0
  438. return sock