1

我有一个模拟来自不存在的远程服务器的请求的测试:

    @httpretty.activate
    def test_no_such_host(self):

        def no_such_host(request, uri, headers):
            raise requests.ConnectionError()

        httpretty.register_uri(
                'GET',
                EXAMPLE_WEBFINGER_URL,
                status=200,
                headers = {
                    'Content-Type': 'application/jrd+json',
                    },
                body = no_such_host,
                )

        webfinger = get_webfinger(
                EXAMPLE_USERNAME,
                EXAMPLE_HOSTNAME,
                )

        self.assertEqual(
                webfinger.url,
                None,
                )

get_webfinger实际上确实抓住了ConnectionError。当它运行时,测试通过——但它也报告ConnectionError异常来自 httpretty 线程:

Creating test database for alias 'default'...
System check identified no issues (0 silenced).
Exception in thread Thread-1:
Traceback (most recent call last):
  File "/usr/lib/python3.6/threading.py", line 916, in _bootstrap_inner
    self.run()
  File "/usr/lib/python3.6/threading.py", line 864, in run
    self._target(*self._args, **self._kwargs)
  File "/home/marnanel/alpha/lib/python3.6/site-packages/httpretty/core.py", line 781, in fill_filekind
    status, headers, self.body = self.callable_body(self.request, self.info.full_url(), headers)
  File "/home/marnanel/proj/kepi/kepi/sombrero_sendpub/tests/test_webfinger.py", line 183, in no_such_host
    raise requests.ConnectionError()
requests.exceptions.ConnectionError

.
----------------------------------------------------------------------
Ran 1 test in 0.117s

OK
Destroying test database for alias 'default'...

单元测试或正在测试的代码中没有线程。这是许多类似测试之一。那些不涉及异常的都运行得很好。

我怎样才能让它停止打印异常?

4

1 回答 1

2

HTTPretty 使用线程来模拟套接字超时,但没有正确处理异常,请参阅问题 #334。后者提出了一种处理异常的方法,但维护者(目前)还没有采纳。

但是,您看到的消息是由默认threading.excepthook()实现打印的。您可以设置自己的钩子;无操作 lambda 将使错误静音:

threading.excepthook = lambda a: None

您可以使用上下文管理器暂时禁用钩子:

import threading
from contextlib import contextmanager

@contextmanager
def suppress_thread_exceptions():
    orig = threading.excepthook
    threading.excepthook = lambda a: None
    try:
        yield
    finally:
        threading.excepthook = orig

然后在您的测试中使用上述内容:

with suppress_thread_exceptions():
    # use HTTPretty
于 2020-08-01T15:17:10.997 回答