Errors from socket.connect() should be treated the same way as async failures.
On freebsd non-blocking connect() may return certain errors (such as ECONNREFUSED from localhost) immediately instead of returning EINPROGRESS and then giving the error later. Also improve the test for ipv6 compatibility, since freebsd returns a different error than other platforms when ipv6 is not available.
This commit is contained in:
parent
4b346bdde8
commit
420ad0869a
|
@ -119,9 +119,18 @@ class IOStream(object):
|
||||||
try:
|
try:
|
||||||
self.socket.connect(address)
|
self.socket.connect(address)
|
||||||
except socket.error, e:
|
except socket.error, e:
|
||||||
# In non-blocking mode connect() always raises an exception
|
# In non-blocking mode we expect connect() to raise an
|
||||||
|
# exception with EINPROGRESS or EWOULDBLOCK.
|
||||||
|
#
|
||||||
|
# On freebsd, other errors such as ECONNREFUSED may be
|
||||||
|
# returned immediately when attempting to connect to
|
||||||
|
# localhost, so handle them the same way as an error
|
||||||
|
# reported later in _handle_connect.
|
||||||
if e.args[0] not in (errno.EINPROGRESS, errno.EWOULDBLOCK):
|
if e.args[0] not in (errno.EINPROGRESS, errno.EWOULDBLOCK):
|
||||||
raise
|
logging.warning("Connect error on fd %d: %s",
|
||||||
|
self.socket.fileno(), e)
|
||||||
|
self.close()
|
||||||
|
return
|
||||||
self._connect_callback = stack_context.wrap(callback)
|
self._connect_callback = stack_context.wrap(callback)
|
||||||
self._add_io_state(self.io_loop.WRITE)
|
self._add_io_state(self.io_loop.WRITE)
|
||||||
|
|
||||||
|
|
|
@ -141,11 +141,15 @@ class SimpleHTTPClientTestCase(AsyncHTTPTestCase, LogTrapTestCase):
|
||||||
self.assertEqual(str(response.error), "HTTP 599: Timeout")
|
self.assertEqual(str(response.error), "HTTP 599: Timeout")
|
||||||
|
|
||||||
def test_ipv6(self):
|
def test_ipv6(self):
|
||||||
|
if not socket.has_ipv6:
|
||||||
|
# python compiled without ipv6 support, so skip this test
|
||||||
|
return
|
||||||
try:
|
try:
|
||||||
self.http_server.listen(self.get_http_port(), address='::1')
|
self.http_server.listen(self.get_http_port(), address='::1')
|
||||||
except socket.gaierror, e:
|
except socket.gaierror, e:
|
||||||
if e.errno == socket.EAI_ADDRFAMILY:
|
if e.errno == socket.EAI_ADDRFAMILY:
|
||||||
# ipv6 is not configured on this system, so skip this test
|
# python supports ipv6, but it's not configured on the network
|
||||||
|
# interface, so skip this test.
|
||||||
return
|
return
|
||||||
raise
|
raise
|
||||||
url = self.get_url("/hello").replace("localhost", "[::1]")
|
url = self.get_url("/hello").replace("localhost", "[::1]")
|
||||||
|
|
Loading…
Reference in New Issue