You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

125 lines
5.2 KiB

import re
import signal
import subprocess
import sys
import unittest
import os
import time
import wx
from gooey.gui import events, processor
from gooey.gui.pubsub import pub
from gooey.gui.processor import ProcessController
class TestProcessor(unittest.TestCase):
def test_extract_progress(self):
# should pull out a number based on the supplied
# regex and expression
processor = ProcessController(r"^progress: (\d+)%$", None, False, 'utf-8')
self.assertEqual(processor._extract_progress(b'progress: 50%'), 50)
processor = ProcessController(r"total: (\d+)%$", None, False, 'utf-8')
self.assertEqual(processor._extract_progress(b'my cool total: 100%'), 100)
def test_extract_progress_returns_none_if_no_regex_supplied(self):
processor = ProcessController(None, None, False, 'utf-8')
self.assertIsNone(processor._extract_progress(b'Total progress: 100%'))
def test_extract_progress_returns_none_if_no_match_found(self):
processor = ProcessController(r'(\d+)%$', None, False, 'utf-8')
self.assertIsNone(processor._extract_progress(b'No match in dis string'))
def test_eval_progress(self):
# given a match in the string, should eval the result
regex = r'(\d+)/(\d+)$'
processor = ProcessController(regex, r'x[0] / x[1]', False,False, 'utf-8')
match = re.search(regex, '50/50')
self.assertEqual(processor._eval_progress(match), 1.0)
def test_eval_progress_returns_none_on_failure(self):
# given a match in the string, should eval the result
regex = r'(\d+)/(\d+)$'
processor = ProcessController(regex, r'x[0] *^/* x[1]', False, False,'utf-8')
match = re.search(regex, '50/50')
self.assertIsNone(processor._eval_progress(match))
def test_all_interrupts_halt_process(self):
"""
TODO: These tests are hella flaky. I'm confident that the feature works. However, getting
signals, subprocesses and unittest to all play together reliably is proving tricky. It
primarily seems to come down to how long the time.sleep() is before sending the shutdown
signal.
"""
cmd = 'python ' + os.path.join(os.getcwd(), 'files', 'infinite_loop.py')
try:
import _winapi
signals = [signal.SIGTERM, signal.CTRL_BREAK_EVENT, signal.CTRL_C_EVENT]
except ModuleNotFoundError:
signals = [signal.SIGTERM, signal.SIGINT]
try:
for sig in signals:
print('sig', sig)
processor = ProcessController(None, None, False, 'utf-8', True, shutdown_signal=sig)
processor.run(cmd)
self.assertTrue(processor.running())
# super-duper important sleep so that the
# signal is actually received by the child process
# see: https://stackoverflow.com/questions/32023719/how-to-simulate-a-terminal-ctrl-c-event-from-a-unittest
time.sleep(1)
processor.stop()
max_wait = time.time() + 4
while processor.running() and time.time() < max_wait:
time.sleep(0.1)
self.assertFalse(processor.running())
except KeyboardInterrupt:
pass
def test_ignore_sigint_family_signals(self):
try:
import _winapi
signals = [signal.CTRL_BREAK_EVENT, signal.CTRL_C_EVENT]
programs = ['ignore_break.py', 'ignore_interrupt.py']
except ModuleNotFoundError:
signals = [signal.SIGINT]
programs = ['ignore_interrupt.py']
for program, sig in zip(programs, signals):
cmd = sys.executable + ' ' + os.path.join(os.getcwd(), 'files', program)
process = processor = ProcessController(None, None, False, 'utf-8', True, shutdown_signal=sig, testmode=True)
process.run(cmd)
# super-duper important sleep so that the
# signal is actually received by the child process
# see: https://stackoverflow.com/questions/32023719/how-to-simulate-a-terminal-ctrl-c-event-from-a-unittest
time.sleep(1)
process.send_shutdown_signal()
# wait to give stdout enough time to write
time.sleep(1)
# now our signal should have been received, but rejected.
self.assertTrue(processor.running())
# so we sigterm to actually shut down the process.
process._send_signal(signal.SIGTERM)
# sanity wait
max_wait = time.time() + 2
while processor.running() and time.time() < max_wait:
time.sleep(0.1)
# now we should be shut down due to killing the process.
self.assertFalse(processor.running())
# and we'll see in the stdout out from the process that our
# interrupt was received
output = process._process.stdout.read().decode('utf-8')
self.assertIn("INTERRUPT", str(output))
# but indeed ignored. It continued running and writing to stdout after
# receiving the signal
self.assertTrue(output.index("INTERRUPT") < len(output))