Move socket.create_connection to green.socket to avoid blocking getaddrinfo() call
Given the order of imports and monkey-patching, even when using greendns the `getaddrinfo()` call inside of _socket_nodns.create_connection would use the unpatched (i.e. non-green) version of getaddrinfo, inadvertantly blocking the process in some situations
This commit is contained in:
@@ -13,7 +13,7 @@ from eventlet.greenio import _GLOBAL_DEFAULT_TIMEOUT
|
|||||||
from eventlet.greenio import _fileobject
|
from eventlet.greenio import _fileobject
|
||||||
|
|
||||||
__all__ = __socket.__all__
|
__all__ = __socket.__all__
|
||||||
__patched__ = ['fromfd', 'socketpair', 'create_connection', 'ssl', 'socket']
|
__patched__ = ['fromfd', 'socketpair', 'ssl', 'socket']
|
||||||
try:
|
try:
|
||||||
__original_fromfd__ = __socket.fromfd
|
__original_fromfd__ = __socket.fromfd
|
||||||
def fromfd(*args):
|
def fromfd(*args):
|
||||||
@@ -31,40 +31,6 @@ except AttributeError:
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
def create_connection(address,
|
|
||||||
timeout=_GLOBAL_DEFAULT_TIMEOUT,
|
|
||||||
source_address=None):
|
|
||||||
"""Connect to *address* and return the socket object.
|
|
||||||
|
|
||||||
Convenience function. Connect to *address* (a 2-tuple ``(host,
|
|
||||||
port)``) and return the socket object. Passing the optional
|
|
||||||
*timeout* parameter will set the timeout on the socket instance
|
|
||||||
before attempting to connect. If no *timeout* is supplied, the
|
|
||||||
global default timeout setting returned by :func:`getdefaulttimeout`
|
|
||||||
is used.
|
|
||||||
"""
|
|
||||||
|
|
||||||
msg = "getaddrinfo returns an empty list"
|
|
||||||
host, port = address
|
|
||||||
for res in getaddrinfo(host, port, 0, SOCK_STREAM):
|
|
||||||
af, socktype, proto, canonname, sa = res
|
|
||||||
sock = None
|
|
||||||
try:
|
|
||||||
sock = socket(af, socktype, proto)
|
|
||||||
if timeout is not _GLOBAL_DEFAULT_TIMEOUT:
|
|
||||||
sock.settimeout(timeout)
|
|
||||||
if source_address:
|
|
||||||
sock.bind(source_address)
|
|
||||||
sock.connect(sa)
|
|
||||||
return sock
|
|
||||||
|
|
||||||
except error, msg:
|
|
||||||
if sock is not None:
|
|
||||||
sock.close()
|
|
||||||
|
|
||||||
raise error, msg
|
|
||||||
|
|
||||||
|
|
||||||
def _convert_to_sslerror(ex):
|
def _convert_to_sslerror(ex):
|
||||||
""" Transliterates SSL.SysCallErrors to socket.sslerrors"""
|
""" Transliterates SSL.SysCallErrors to socket.sslerrors"""
|
||||||
return sslerror((ex.args[0], ex.args[1]))
|
return sslerror((ex.args[0], ex.args[1]))
|
||||||
|
@@ -8,14 +8,14 @@ globals().update(dict([(var, getattr(__socket, var))
|
|||||||
if not var.startswith('__')]))
|
if not var.startswith('__')]))
|
||||||
|
|
||||||
__all__ = __socket.__all__
|
__all__ = __socket.__all__
|
||||||
__patched__ = __socket.__patched__ + ['gethostbyname', 'getaddrinfo']
|
__patched__ = __socket.__patched__ + ['gethostbyname', 'getaddrinfo', 'create_connection',]
|
||||||
|
|
||||||
|
|
||||||
greendns = None
|
greendns = None
|
||||||
if os.environ.get("EVENTLET_NO_GREENDNS",'').lower() != "yes":
|
if os.environ.get("EVENTLET_NO_GREENDNS",'').lower() != "yes":
|
||||||
try:
|
try:
|
||||||
from eventlet.support import greendns
|
from eventlet.support import greendns
|
||||||
except ImportError:
|
except ImportError, ex:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
__original_gethostbyname__ = __socket.gethostbyname
|
__original_gethostbyname__ = __socket.gethostbyname
|
||||||
@@ -60,6 +60,39 @@ else:
|
|||||||
if greendns:
|
if greendns:
|
||||||
gethostbyname_ex = greendns.gethostbyname_ex
|
gethostbyname_ex = greendns.gethostbyname_ex
|
||||||
getnameinfo = greendns.getnameinfo
|
getnameinfo = greendns.getnameinfo
|
||||||
__patched__ + ['gethostbyname_ex', 'getnameinfo']
|
__patched__ = __patched__ + ['gethostbyname_ex', 'getnameinfo']
|
||||||
|
|
||||||
|
def create_connection(address,
|
||||||
|
timeout=_GLOBAL_DEFAULT_TIMEOUT,
|
||||||
|
source_address=None):
|
||||||
|
"""Connect to *address* and return the socket object.
|
||||||
|
|
||||||
|
Convenience function. Connect to *address* (a 2-tuple ``(host,
|
||||||
|
port)``) and return the socket object. Passing the optional
|
||||||
|
*timeout* parameter will set the timeout on the socket instance
|
||||||
|
before attempting to connect. If no *timeout* is supplied, the
|
||||||
|
global default timeout setting returned by :func:`getdefaulttimeout`
|
||||||
|
is used.
|
||||||
|
"""
|
||||||
|
|
||||||
|
msg = "getaddrinfo returns an empty list"
|
||||||
|
host, port = address
|
||||||
|
for res in getaddrinfo(host, port, 0, SOCK_STREAM):
|
||||||
|
af, socktype, proto, canonname, sa = res
|
||||||
|
sock = None
|
||||||
|
try:
|
||||||
|
sock = socket(af, socktype, proto)
|
||||||
|
if timeout is not _GLOBAL_DEFAULT_TIMEOUT:
|
||||||
|
sock.settimeout(timeout)
|
||||||
|
if source_address:
|
||||||
|
sock.bind(source_address)
|
||||||
|
sock.connect(sa)
|
||||||
|
return sock
|
||||||
|
|
||||||
|
except error, msg:
|
||||||
|
if sock is not None:
|
||||||
|
sock.close()
|
||||||
|
|
||||||
|
raise error, msg
|
||||||
|
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user