summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorKevin Van Brunt <kmvanbrunt@gmail.com>2019-11-18 22:34:57 -0500
committerKevin Van Brunt <kmvanbrunt@gmail.com>2019-11-18 22:34:57 -0500
commit43c74f7f09ef6095d21e9b5d83308e3904f636a9 (patch)
tree102ce235139e4df01cd3935661eca06b0c7b5615
parent73535e1ff82b49c594fc694ef0ea898d46742750 (diff)
downloadcmd2-git-43c74f7f09ef6095d21e9b5d83308e3904f636a9.tar.gz
Fixed bug where pipe processes were not being stopped by Ctrl-C on Windows
-rw-r--r--CHANGELOG.md2
-rw-r--r--cmd2/cmd2.py18
-rw-r--r--cmd2/utils.py6
-rw-r--r--tests/test_utils.py31
4 files changed, 35 insertions, 22 deletions
diff --git a/CHANGELOG.md b/CHANGELOG.md
index 630f6892..8264049b 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -1,6 +1,6 @@
## 0.9.21 (TBD, 2019)
* Bug Fixes
- * Fixed bug where pipe processes were not being stopped by Ctrl-C on Linux/Mac
+ * Fixed bug where pipe processes were not being stopped by Ctrl-C
* Enhancements
* Added `read_input()` function that is used to read from stdin. Unlike the Python built-in `input()`, it also has
an argument to disable tab completion while input is being entered.
diff --git a/cmd2/cmd2.py b/cmd2/cmd2.py
index 051e6f7c..d544cb5f 100644
--- a/cmd2/cmd2.py
+++ b/cmd2/cmd2.py
@@ -1824,24 +1824,22 @@ class Cmd(cmd.Cmd):
subproc_stdin = io.open(read_fd, 'r')
new_stdout = io.open(write_fd, 'w')
- # Set options to not forward signals to the pipe process. If a Ctrl-C event occurs,
- # our sigint handler will forward it only to the most recent pipe process. This makes
- # sure pipe processes close in the right order (most recent first).
+ # Create pipe process in a separate group to isolate our signals from it. If a Ctrl-C event occurs,
+ # our sigint handler will forward it only to the most recent pipe process. This makes sure pipe
+ # processes close in the right order (most recent first).
+ kwargs = dict()
if sys.platform == 'win32':
- creationflags = subprocess.CREATE_NEW_PROCESS_GROUP
- start_new_session = False
+ kwargs['creationflags'] = subprocess.CREATE_NEW_PROCESS_GROUP
else:
- creationflags = 0
- start_new_session = True
+ kwargs['start_new_session'] = True
# For any stream that is a StdSim, we will use a pipe so we can capture its output
proc = subprocess.Popen(statement.pipe_to,
stdin=subproc_stdin,
stdout=subprocess.PIPE if isinstance(self.stdout, utils.StdSim) else self.stdout,
stderr=subprocess.PIPE if isinstance(sys.stderr, utils.StdSim) else sys.stderr,
- creationflags=creationflags,
- start_new_session=start_new_session,
- shell=True)
+ shell=True,
+ **kwargs)
# Popen was called with shell=True so the user can chain pipe commands and redirect their output
# like: !ls -l | grep user | wc -l > out.txt. But this makes it difficult to know if the pipe process
diff --git a/cmd2/utils.py b/cmd2/utils.py
index 3155c64a..a1a0d377 100644
--- a/cmd2/utils.py
+++ b/cmd2/utils.py
@@ -517,10 +517,12 @@ class ProcReader(object):
self._err_thread.start()
def send_sigint(self) -> None:
- """Send a SIGINT to the process similar to if <Ctrl>+C were pressed."""
+ """Send a SIGINT to the process similar to if <Ctrl>+C were pressed"""
import signal
if sys.platform.startswith('win'):
- self._proc.send_signal(signal.CTRL_C_EVENT)
+ # cmd2 started the Windows process in a new process group. Therefore
+ # a CTRL_C_EVENT can't be sent to it. Send a CTRL_BREAK_EVENT instead.
+ self._proc.send_signal(signal.CTRL_BREAK_EVENT)
else:
# Since cmd2 uses shell=True in its Popen calls, we need to send the SIGINT to
# the whole process group to make sure it propagates further than the shell
diff --git a/tests/test_utils.py b/tests/test_utils.py
index 52adc7cd..e4b9169c 100644
--- a/tests/test_utils.py
+++ b/tests/test_utils.py
@@ -5,6 +5,7 @@ Unit testing for cmd2/utils.py module.
"""
import signal
import sys
+import time
import pytest
@@ -228,27 +229,31 @@ def pr_none():
import subprocess
# Start a long running process so we have time to run tests on it before it finishes
- # Put the new process into a separate group so signals sent to it won't interfere with this process
+ # Put the new process into a separate group so its signal are isolated from ours
+ kwargs = dict()
if sys.platform.startswith('win'):
command = 'timeout -t 5 /nobreak'
- creationflags = subprocess.CREATE_NEW_PROCESS_GROUP
- start_new_session = False
+ kwargs['creationflags'] = subprocess.CREATE_NEW_PROCESS_GROUP
else:
command = 'sleep 5'
- creationflags = 0
- start_new_session = True
+ kwargs['start_new_session'] = True
- proc = subprocess.Popen(command,
- creationflags=creationflags,
- start_new_session=start_new_session,
- shell=True)
+ proc = subprocess.Popen(command, shell=True, **kwargs)
pr = cu.ProcReader(proc, None, None)
return pr
def test_proc_reader_send_sigint(pr_none):
assert pr_none._proc.poll() is None
pr_none.send_sigint()
+
+ wait_start = time.monotonic()
pr_none.wait()
+ wait_finish = time.monotonic()
+
+ # Make sure the process exited before sleep of 5 seconds finished
+ # 3 seconds accounts for some delay but is long enough for the process to exit
+ assert wait_finish - wait_start < 3
+
ret_code = pr_none._proc.poll()
if sys.platform.startswith('win'):
assert ret_code is not None
@@ -258,7 +263,15 @@ def test_proc_reader_send_sigint(pr_none):
def test_proc_reader_terminate(pr_none):
assert pr_none._proc.poll() is None
pr_none.terminate()
+
+ wait_start = time.monotonic()
pr_none.wait()
+ wait_finish = time.monotonic()
+
+ # Make sure the process exited before sleep of 5 seconds finished
+ # 3 seconds accounts for some delay but is long enough for the process to exit
+ assert wait_finish - wait_start < 3
+
ret_code = pr_none._proc.poll()
if sys.platform.startswith('win'):
assert ret_code is not None