6

我正在使用 Django (1.5.5)、selenium (2.41.0)、splinter (0.6.0) 和 phantomjs (1.9.7) 来运行实时测试。

虽然测试大部分都有效,但时不时(通常在 CircleCI 上,在本地 VM 中较少)它们会挂起,直到 CircleCI 超时或我手动杀死跑步者(Ctrl-C 即 KeyboardInterrupt 有效)。

这是我的基本测试类的外观:

class SplinterTestCase(LiveServerTestCase):
    @classmethod
    def setUpClass(cls):
        super(SplinterTestCase, cls).setUpClass()
        # start phantom just once per class, to speed up tests
        cls.phantom = splinter.Browser('phantomjs', load_images=False)

    @classmethod
    def tearDownClass(cls):
        cls.phantom.quit()
        super(SplinterTestCase, cls).tearDownClass()

    def login(self, *args, **kwargs):
        # perform a login using Django builtin "client", steal the session
        # cookie and inject it to phantomjs, avoiding the need to do the
        # login dance for each test
        from django.conf import settings
        cn = settings.SESSION_COOKIE_NAME

        self.django_client.login(*args, **kwargs)
        if cn in self.django_client.cookies:
            self.client.driver.add_cookie({
                'name': cn,
                'value': self.django_client.cookies[cn].value,
                'path': '/',
                'domain': 'localhost'
            })

    def setUp(self):
        # use phantom as the test client instead of Django's
        super(SplinterTestCase, self).setUp()
        self.django_client = self.client
        self.client = self.phantom

    def tearDown(self):
        # this seems to help somewhat (decreases the number of timeouts), but
        # doesn't solve it completely
        self.client.visit('about:config')
        super(SplinterTestCase, self).tearDown()

在 Ctrl-C 之后,这是我得到的堆栈跟踪:

Traceback (most recent call last):
  File "/usr/lib/python2.7/wsgiref/handlers.py", line 86, in run
    self.finish_response()
  File "/usr/lib/python2.7/wsgiref/handlers.py", line 127, in finish_response
    self.write(data)
  File "/usr/lib/python2.7/wsgiref/handlers.py", line 215, in write
    self._write(data)
  File "/usr/lib/python2.7/socket.py", line 324, in write
    self.flush()
  File "/usr/lib/python2.7/socket.py", line 303, in flush
    self._sock.sendall(view[write_offset:write_offset+buffer_size])
error: [Errno 104] Connection reset by peer
Traceback (most recent call last):
  File "/home/ubuntu/memo-angel/venv/local/lib/python2.7/site-packages/django/test/testcases.py", line 998, in _handle_request_noblock
    self.process_request(request, client_address)
  File "/usr/lib/python2.7/SocketServer.py", line 310, in process_request
    self.finish_request(request, client_address)
  File "/usr/lib/python2.7/SocketServer.py", line 323, in finish_request
    self.RequestHandlerClass(request, client_address, self)
  File "/home/ubuntu/memo-angel/venv/local/lib/python2.7/site-packages/django/core/servers/basehttp.py", line 150, in __init__
    super(WSGIRequestHandler, self).__init__(*args, **kwargs)
  File "/usr/lib/python2.7/SocketServer.py", line 640, in __init__
    self.finish()
  File "/usr/lib/python2.7/SocketServer.py", line 693, in finish
    self.wfile.flush()
  File "/usr/lib/python2.7/socket.py", line 303, in flush
    self._sock.sendall(view[write_offset:write_offset+buffer_size])
error: [Errno 32] Broken pipe

类似的问题可能已经在Django 中讨论过 splinter 和 phantomjs 非常缓慢,因为原始海报还提到“它只是冻结,直到我没有耐心等待它完成”。那里提到的答案是尝试将 phantomjs 启动/停止放在类设置/拆卸中,我在这里做了,但这并不能解决问题。

有没有人遇到过类似的问题,如果有,您的解决方法是什么?

4

1 回答 1

2

为了识别问题并让您的测试更快地失败,您可能需要在 setUpClass / setUp 中配置套接字超时:

import socket

...
socket.setdefaulttimeout(10)
于 2014-04-22T09:05:43.827 回答