Skip to content
This repository was archived by the owner on Nov 23, 2017. It is now read-only.

Commit 2608734

Browse files
committed
More robust check of socket.type
1 parent 8203372 commit 2608734

File tree

4 files changed

+57
-12
lines changed

4 files changed

+57
-12
lines changed

asyncio/base_events.py

Lines changed: 10 additions & 10 deletions
Original file line numberDiff line numberDiff line change
@@ -84,20 +84,18 @@ def _set_reuseport(sock):
8484
'SO_REUSEPORT defined but not implemented.')
8585

8686

87-
# Linux's sock.type is a bitmask that can include extra info about socket.
88-
_SOCKET_TYPE_MASK = 0
89-
if hasattr(socket, 'SOCK_NONBLOCK'):
90-
_SOCKET_TYPE_MASK |= socket.SOCK_NONBLOCK
91-
if hasattr(socket, 'SOCK_CLOEXEC'):
92-
_SOCKET_TYPE_MASK |= socket.SOCK_CLOEXEC
93-
94-
9587
def _is_stream_socket(sock_type):
96-
return (sock_type & ~_SOCKET_TYPE_MASK) == socket.SOCK_STREAM
88+
# Linux's socket.type is a bitmask that can include extra info
89+
# about socket, therefore we can't do simple
90+
# `sock_type == socket.SOCK_STREAM`.
91+
return (sock_type & socket.SOCK_STREAM) == socket.SOCK_STREAM
9792

9893

9994
def _is_dgram_socket(sock_type):
100-
return (sock_type & ~_SOCKET_TYPE_MASK) == socket.SOCK_DGRAM
95+
# Linux's socket.type is a bitmask that can include extra info
96+
# about socket, therefore we can't do simple
97+
# `sock_type == socket.SOCK_DGRAM`.
98+
return (sock_type & socket.SOCK_DGRAM) == socket.SOCK_DGRAM
10199

102100

103101
def _is_ip_socket(sock_family):
@@ -119,8 +117,10 @@ def _ipaddr_info(host, port, family, type, proto):
119117
return None
120118

121119
if _is_stream_socket(type):
120+
type = socket.SOCK_STREAM
122121
proto = socket.IPPROTO_TCP
123122
elif _is_dgram_socket(type):
123+
type = socket.SOCK_DGRAM
124124
proto = socket.IPPROTO_UDP
125125
else:
126126
return None

asyncio/unix_events.py

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -235,7 +235,7 @@ def create_unix_connection(self, protocol_factory, path, *,
235235
if sock is None:
236236
raise ValueError('no path and sock were specified')
237237
if (sock.family != socket.AF_UNIX or
238-
sock.type != socket.SOCK_STREAM):
238+
not base_events._is_stream_socket(sock.type)):
239239
raise ValueError(
240240
'A UNIX Domain Stream Socket was expected, got {!r}'
241241
.format(sock))
@@ -289,7 +289,7 @@ def create_unix_server(self, protocol_factory, path=None, *,
289289
'path was not specified, and no sock specified')
290290

291291
if (sock.family != socket.AF_UNIX or
292-
sock.type != socket.SOCK_STREAM):
292+
not base_events._is_stream_socket(sock.type)):
293293
raise ValueError(
294294
'A UNIX Domain Stream Socket was expected, got {!r}'
295295
.format(sock))

tests/test_base_events.py

Lines changed: 18 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -116,6 +116,13 @@ def test_ipaddr_info(self):
116116
self.assertIsNone(
117117
base_events._ipaddr_info('::3%lo0', 1, INET6, STREAM, TCP))
118118

119+
if hasattr(socket, 'SOCK_NONBLOCK'):
120+
self.assertEqual(
121+
(INET, STREAM, TCP, '', ('1.2.3.4', 1)),
122+
base_events._ipaddr_info(
123+
'1.2.3.4', 1, INET, STREAM | socket.SOCK_NONBLOCK, TCP))
124+
125+
119126
def test_port_parameter_types(self):
120127
# Test obscure kinds of arguments for "port".
121128
INET = socket.AF_INET
@@ -1058,6 +1065,17 @@ def test_create_server_wrong_sock(self):
10581065
'A TCP Stream Socket was expected'):
10591066
self.loop.run_until_complete(coro)
10601067

1068+
@unittest.skipUnless(hasattr(socket, 'SOCK_NONBLOCK'),
1069+
'no socket.SOCK_NONBLOCK (linux only)')
1070+
def test_create_server_stream_bittype(self):
1071+
sock = socket.socket(
1072+
socket.AF_INET, socket.SOCK_STREAM | socket.SOCK_NONBLOCK)
1073+
with sock:
1074+
coro = self.loop.create_server(lambda: None, sock=sock)
1075+
srv = self.loop.run_until_complete(coro)
1076+
srv.close()
1077+
self.loop.run_until_complete(srv.wait_closed())
1078+
10611079
def test_create_datagram_endpoint_wrong_sock(self):
10621080
sock = socket.socket(socket.AF_INET)
10631081
with sock:

tests/test_unix_events.py

Lines changed: 27 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -280,6 +280,33 @@ def test_create_unix_server_path_inetsock(self):
280280
'A UNIX Domain Stream.*was expected'):
281281
self.loop.run_until_complete(coro)
282282

283+
def test_create_unix_server_path_dgram(self):
284+
sock = socket.socket(socket.AF_UNIX, socket.SOCK_DGRAM)
285+
with sock:
286+
coro = self.loop.create_unix_server(lambda: None, path=None,
287+
sock=sock)
288+
with self.assertRaisesRegex(ValueError,
289+
'A UNIX Domain Stream.*was expected'):
290+
self.loop.run_until_complete(coro)
291+
292+
@unittest.skipUnless(hasattr(socket, 'SOCK_NONBLOCK'),
293+
'no socket.SOCK_NONBLOCK (linux only)')
294+
def test_create_unix_server_path_stream_bittype(self):
295+
sock = socket.socket(
296+
socket.AF_UNIX, socket.SOCK_STREAM | socket.SOCK_NONBLOCK)
297+
with tempfile.NamedTemporaryFile() as file:
298+
fn = file.name
299+
try:
300+
with sock:
301+
sock.bind(fn)
302+
coro = self.loop.create_unix_server(lambda: None, path=None,
303+
sock=sock)
304+
srv = self.loop.run_until_complete(coro)
305+
srv.close()
306+
self.loop.run_until_complete(srv.wait_closed())
307+
finally:
308+
os.unlink(fn)
309+
283310
def test_create_unix_connection_path_inetsock(self):
284311
sock = socket.socket()
285312
with sock:

0 commit comments

Comments
 (0)