Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
C
cpython
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
cpython
Commits
171bb71b
Commit
171bb71b
authored
Oct 21, 2013
by
Guido van Rossum
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Switch subprocess stdin to a socketpair, attempting to fix issue #19293 (AIX hang).
parent
80152d73
Changes
2
Show whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
32 additions
and
4 deletions
+32
-4
Lib/asyncio/unix_events.py
Lib/asyncio/unix_events.py
+25
-4
Lib/test/test_asyncio/test_unix_events.py
Lib/test/test_asyncio/test_unix_events.py
+7
-0
No files found.
Lib/asyncio/unix_events.py
View file @
171bb71b
...
...
@@ -213,6 +213,9 @@ class _UnixReadPipeTransport(transports.ReadTransport):
self
.
_loop
=
loop
self
.
_pipe
=
pipe
self
.
_fileno
=
pipe
.
fileno
()
mode
=
os
.
fstat
(
self
.
_fileno
).
st_mode
if
not
(
stat
.
S_ISFIFO
(
mode
)
or
stat
.
S_ISSOCK
(
mode
)):
raise
ValueError
(
"Pipe transport is for pipes/sockets only."
)
_set_nonblocking
(
self
.
_fileno
)
self
.
_protocol
=
protocol
self
.
_closing
=
False
...
...
@@ -275,13 +278,21 @@ class _UnixWritePipeTransport(transports.WriteTransport):
self
.
_loop
=
loop
self
.
_pipe
=
pipe
self
.
_fileno
=
pipe
.
fileno
()
if
not
stat
.
S_ISFIFO
(
os
.
fstat
(
self
.
_fileno
).
st_mode
):
raise
ValueError
(
"Pipe transport is for pipes only."
)
mode
=
os
.
fstat
(
self
.
_fileno
).
st_mode
is_socket
=
stat
.
S_ISSOCK
(
mode
)
is_pipe
=
stat
.
S_ISFIFO
(
mode
)
if
not
(
is_socket
or
is_pipe
):
raise
ValueError
(
"Pipe transport is for pipes/sockets only."
)
_set_nonblocking
(
self
.
_fileno
)
self
.
_protocol
=
protocol
self
.
_buffer
=
[]
self
.
_conn_lost
=
0
self
.
_closing
=
False
# Set when close() or write_eof() called.
# On AIX, the reader trick only works for sockets.
# On other platforms it works for pipes and sockets.
# (Exception: OS X 10.4? Issue #19294.)
if
is_socket
or
not
sys
.
platform
.
startswith
(
"aix"
):
self
.
_loop
.
add_reader
(
self
.
_fileno
,
self
.
_read_ready
)
self
.
_loop
.
call_soon
(
self
.
_protocol
.
connection_made
,
self
)
...
...
@@ -289,7 +300,7 @@ class _UnixWritePipeTransport(transports.WriteTransport):
self
.
_loop
.
call_soon
(
waiter
.
set_result
,
None
)
def
_read_ready
(
self
):
#
pipe was closed by peer
#
Pipe was closed by peer.
self
.
_close
()
def
write
(
self
,
data
):
...
...
@@ -435,8 +446,15 @@ class _UnixSubprocessTransport(transports.SubprocessTransport):
self
.
_loop
=
loop
self
.
_pipes
=
{}
stdin_w
=
None
if
stdin
==
subprocess
.
PIPE
:
self
.
_pipes
[
STDIN
]
=
None
# Use a socket pair for stdin, since not all platforms
# support selecting read events on the write end of a
# socket (which we use in order to detect closing of the
# other end). Notably this is needed on AIX, and works
# just fine on other platforms.
stdin
,
stdin_w
=
self
.
_loop
.
_socketpair
()
if
stdout
==
subprocess
.
PIPE
:
self
.
_pipes
[
STDOUT
]
=
None
if
stderr
==
subprocess
.
PIPE
:
...
...
@@ -448,6 +466,9 @@ class _UnixSubprocessTransport(transports.SubprocessTransport):
self
.
_proc
=
subprocess
.
Popen
(
args
,
shell
=
shell
,
stdin
=
stdin
,
stdout
=
stdout
,
stderr
=
stderr
,
universal_newlines
=
False
,
bufsize
=
bufsize
,
**
kwargs
)
if
stdin_w
is
not
None
:
stdin
.
close
()
self
.
_proc
.
stdin
=
open
(
stdin_w
.
detach
(),
'rb'
,
buffering
=
bufsize
)
self
.
_extra
[
'subprocess'
]
=
self
.
_proc
def
close
(
self
):
...
...
Lib/test/test_asyncio/test_unix_events.py
View file @
171bb71b
...
...
@@ -312,6 +312,13 @@ class UnixReadPipeTransportTests(unittest.TestCase):
fcntl_patcher
.
start
()
self
.
addCleanup
(
fcntl_patcher
.
stop
)
fstat_patcher
=
unittest
.
mock
.
patch
(
'os.fstat'
)
m_fstat
=
fstat_patcher
.
start
()
st
=
unittest
.
mock
.
Mock
()
st
.
st_mode
=
stat
.
S_IFIFO
m_fstat
.
return_value
=
st
self
.
addCleanup
(
fstat_patcher
.
stop
)
def
test_ctor
(
self
):
tr
=
unix_events
.
_UnixReadPipeTransport
(
self
.
loop
,
self
.
pipe
,
self
.
protocol
)
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment