Mercurial > repos > shellac > sam_consensus_v3
comparison env/lib/python3.9/site-packages/urllib3/util/connection.py @ 0:4f3585e2f14b draft default tip
"planemo upload commit 60cee0fc7c0cda8592644e1aad72851dec82c959"
author | shellac |
---|---|
date | Mon, 22 Mar 2021 18:12:50 +0000 |
parents | |
children |
comparison
equal
deleted
inserted
replaced
-1:000000000000 | 0:4f3585e2f14b |
---|---|
1 from __future__ import absolute_import | |
2 | |
3 import socket | |
4 | |
5 from urllib3.exceptions import LocationParseError | |
6 | |
7 from ..contrib import _appengine_environ | |
8 from ..packages import six | |
9 from .wait import NoWayToWaitForSocketError, wait_for_read | |
10 | |
11 | |
12 def is_connection_dropped(conn): # Platform-specific | |
13 """ | |
14 Returns True if the connection is dropped and should be closed. | |
15 | |
16 :param conn: | |
17 :class:`http.client.HTTPConnection` object. | |
18 | |
19 Note: For platforms like AppEngine, this will always return ``False`` to | |
20 let the platform handle connection recycling transparently for us. | |
21 """ | |
22 sock = getattr(conn, "sock", False) | |
23 if sock is False: # Platform-specific: AppEngine | |
24 return False | |
25 if sock is None: # Connection already closed (such as by httplib). | |
26 return True | |
27 try: | |
28 # Returns True if readable, which here means it's been dropped | |
29 return wait_for_read(sock, timeout=0.0) | |
30 except NoWayToWaitForSocketError: # Platform-specific: AppEngine | |
31 return False | |
32 | |
33 | |
34 # This function is copied from socket.py in the Python 2.7 standard | |
35 # library test suite. Added to its signature is only `socket_options`. | |
36 # One additional modification is that we avoid binding to IPv6 servers | |
37 # discovered in DNS if the system doesn't have IPv6 functionality. | |
38 def create_connection( | |
39 address, | |
40 timeout=socket._GLOBAL_DEFAULT_TIMEOUT, | |
41 source_address=None, | |
42 socket_options=None, | |
43 ): | |
44 """Connect to *address* and return the socket object. | |
45 | |
46 Convenience function. Connect to *address* (a 2-tuple ``(host, | |
47 port)``) and return the socket object. Passing the optional | |
48 *timeout* parameter will set the timeout on the socket instance | |
49 before attempting to connect. If no *timeout* is supplied, the | |
50 global default timeout setting returned by :func:`socket.getdefaulttimeout` | |
51 is used. If *source_address* is set it must be a tuple of (host, port) | |
52 for the socket to bind as a source address before making the connection. | |
53 An host of '' or port 0 tells the OS to use the default. | |
54 """ | |
55 | |
56 host, port = address | |
57 if host.startswith("["): | |
58 host = host.strip("[]") | |
59 err = None | |
60 | |
61 # Using the value from allowed_gai_family() in the context of getaddrinfo lets | |
62 # us select whether to work with IPv4 DNS records, IPv6 records, or both. | |
63 # The original create_connection function always returns all records. | |
64 family = allowed_gai_family() | |
65 | |
66 try: | |
67 host.encode("idna") | |
68 except UnicodeError: | |
69 return six.raise_from( | |
70 LocationParseError(u"'%s', label empty or too long" % host), None | |
71 ) | |
72 | |
73 for res in socket.getaddrinfo(host, port, family, socket.SOCK_STREAM): | |
74 af, socktype, proto, canonname, sa = res | |
75 sock = None | |
76 try: | |
77 sock = socket.socket(af, socktype, proto) | |
78 | |
79 # If provided, set socket level options before connecting. | |
80 _set_socket_options(sock, socket_options) | |
81 | |
82 if timeout is not socket._GLOBAL_DEFAULT_TIMEOUT: | |
83 sock.settimeout(timeout) | |
84 if source_address: | |
85 sock.bind(source_address) | |
86 sock.connect(sa) | |
87 return sock | |
88 | |
89 except socket.error as e: | |
90 err = e | |
91 if sock is not None: | |
92 sock.close() | |
93 sock = None | |
94 | |
95 if err is not None: | |
96 raise err | |
97 | |
98 raise socket.error("getaddrinfo returns an empty list") | |
99 | |
100 | |
101 def _set_socket_options(sock, options): | |
102 if options is None: | |
103 return | |
104 | |
105 for opt in options: | |
106 sock.setsockopt(*opt) | |
107 | |
108 | |
109 def allowed_gai_family(): | |
110 """This function is designed to work in the context of | |
111 getaddrinfo, where family=socket.AF_UNSPEC is the default and | |
112 will perform a DNS search for both IPv6 and IPv4 records.""" | |
113 | |
114 family = socket.AF_INET | |
115 if HAS_IPV6: | |
116 family = socket.AF_UNSPEC | |
117 return family | |
118 | |
119 | |
120 def _has_ipv6(host): | |
121 """ Returns True if the system can bind an IPv6 address. """ | |
122 sock = None | |
123 has_ipv6 = False | |
124 | |
125 # App Engine doesn't support IPV6 sockets and actually has a quota on the | |
126 # number of sockets that can be used, so just early out here instead of | |
127 # creating a socket needlessly. | |
128 # See https://github.com/urllib3/urllib3/issues/1446 | |
129 if _appengine_environ.is_appengine_sandbox(): | |
130 return False | |
131 | |
132 if socket.has_ipv6: | |
133 # has_ipv6 returns true if cPython was compiled with IPv6 support. | |
134 # It does not tell us if the system has IPv6 support enabled. To | |
135 # determine that we must bind to an IPv6 address. | |
136 # https://github.com/urllib3/urllib3/pull/611 | |
137 # https://bugs.python.org/issue658327 | |
138 try: | |
139 sock = socket.socket(socket.AF_INET6) | |
140 sock.bind((host, 0)) | |
141 has_ipv6 = True | |
142 except Exception: | |
143 pass | |
144 | |
145 if sock: | |
146 sock.close() | |
147 return has_ipv6 | |
148 | |
149 | |
150 HAS_IPV6 = _has_ipv6("::1") |