Commit e0558e65 authored by szager@chromium.org's avatar szager@chromium.org

Add nag_timer.

BUG=227537
R=maruel@chromium.org,cmp@chromium.org
NOTRY=true

Review URL: https://chromiumcodereview.appspot.com/14798003

git-svn-id: svn://svn.chromium.org/chrome/trunk/tools/depot_tools@197790 0039d316-1c4b-4281-b951-d872f2087c98
parent a3800b2d
...@@ -175,6 +175,7 @@ class Popen(subprocess.Popen): ...@@ -175,6 +175,7 @@ class Popen(subprocess.Popen):
self.stdin_is_void = False self.stdin_is_void = False
self.stdout_is_void = False self.stdout_is_void = False
self.stderr_is_void = False self.stderr_is_void = False
self.cmd_str = tmp_str
if kwargs.get('stdin') is VOID: if kwargs.get('stdin') is VOID:
kwargs['stdin'] = open(os.devnull, 'r') kwargs['stdin'] = open(os.devnull, 'r')
...@@ -190,6 +191,7 @@ class Popen(subprocess.Popen): ...@@ -190,6 +191,7 @@ class Popen(subprocess.Popen):
self.start = time.time() self.start = time.time()
self.timeout = None self.timeout = None
self.nag_timer = None
self.shell = kwargs.get('shell', None) self.shell = kwargs.get('shell', None)
# Silence pylint on MacOSX # Silence pylint on MacOSX
self.returncode = None self.returncode = None
...@@ -228,6 +230,8 @@ class Popen(subprocess.Popen): ...@@ -228,6 +230,8 @@ class Popen(subprocess.Popen):
# because of memory exhaustion. # because of memory exhaustion.
queue = Queue.Queue() queue = Queue.Queue()
done = threading.Event() done = threading.Event()
timer = []
last_output = [time.time()] * 2
def write_stdin(): def write_stdin():
try: try:
...@@ -249,10 +253,28 @@ class Popen(subprocess.Popen): ...@@ -249,10 +253,28 @@ class Popen(subprocess.Popen):
data = pipe.read(1) data = pipe.read(1)
if not data: if not data:
break break
last_output[0] = time.time()
queue.put((name, data)) queue.put((name, data))
finally: finally:
queue.put(name) queue.put(name)
def nag_fn():
now = time.time()
if done.is_set():
return
if last_output[0] == last_output[1]:
logging.warn(' No output for %.0f seconds from command:' % (
now - last_output[1]))
logging.warn(' %s' % self.cmd_str)
# Use 0.1 fudge factor in case:
# now ~= last_output[0] + self.nag_timer
sleep_time = self.nag_timer + last_output[0] - now - 0.1
while sleep_time < 0:
sleep_time += self.nag_timer
last_output[1] = last_output[0]
timer[0] = threading.Timer(sleep_time, nag_fn)
timer[0].start()
def timeout_fn(): def timeout_fn():
try: try:
done.wait(self.timeout) done.wait(self.timeout)
...@@ -290,6 +312,10 @@ class Popen(subprocess.Popen): ...@@ -290,6 +312,10 @@ class Popen(subprocess.Popen):
for t in threads.itervalues(): for t in threads.itervalues():
t.start() t.start()
if self.nag_timer:
timer.append(threading.Timer(self.nag_timer, nag_fn))
timer[0].start()
timed_out = False timed_out = False
try: try:
# This thread needs to be optimized for speed. # This thread needs to be optimized for speed.
...@@ -313,6 +339,8 @@ class Popen(subprocess.Popen): ...@@ -313,6 +339,8 @@ class Popen(subprocess.Popen):
finally: finally:
# Stop the threads. # Stop the threads.
done.set() done.set()
if timer:
timer[0].cancel()
if 'wait' in threads: if 'wait' in threads:
# Accelerate things, otherwise it would hang until the child process is # Accelerate things, otherwise it would hang until the child process is
# done. # done.
...@@ -324,16 +352,21 @@ class Popen(subprocess.Popen): ...@@ -324,16 +352,21 @@ class Popen(subprocess.Popen):
if timed_out: if timed_out:
self.returncode = TIMED_OUT self.returncode = TIMED_OUT
def communicate(self, input=None, timeout=None): # pylint: disable=W0221,W0622 # pylint: disable=W0221,W0622
def communicate(self, input=None, timeout=None, nag_timer=None):
"""Adds timeout and callbacks support. """Adds timeout and callbacks support.
Returns (stdout, stderr) like subprocess.Popen().communicate(). Returns (stdout, stderr) like subprocess.Popen().communicate().
- The process will be killed after |timeout| seconds and returncode set to - The process will be killed after |timeout| seconds and returncode set to
TIMED_OUT. TIMED_OUT.
- If the subprocess runs for |nag_timer| seconds without producing terminal
output, print a warning to stderr.
""" """
self.timeout = timeout self.timeout = timeout
if not self.timeout and not self.stdout_cb and not self.stderr_cb: self.nag_timer = nag_timer
if (not self.timeout and not self.nag_timer and
not self.stdout_cb and not self.stderr_cb):
return super(Popen, self).communicate(input) return super(Popen, self).communicate(input)
if self.timeout and self.shell: if self.timeout and self.shell:
...@@ -360,13 +393,15 @@ class Popen(subprocess.Popen): ...@@ -360,13 +393,15 @@ class Popen(subprocess.Popen):
return (stdout, stderr) return (stdout, stderr)
def communicate(args, timeout=None, **kwargs): def communicate(args, timeout=None, nag_timer=None, **kwargs):
"""Wraps subprocess.Popen().communicate() and add timeout support. """Wraps subprocess.Popen().communicate() and add timeout support.
Returns ((stdout, stderr), returncode). Returns ((stdout, stderr), returncode).
- The process will be killed after |timeout| seconds and returncode set to - The process will be killed after |timeout| seconds and returncode set to
TIMED_OUT. TIMED_OUT.
- If the subprocess runs for |nag_timer| seconds without producing terminal
output, print a warning to stderr.
- Automatically passes stdin content as input so do not specify stdin=PIPE. - Automatically passes stdin content as input so do not specify stdin=PIPE.
""" """
stdin = kwargs.pop('stdin', None) stdin = kwargs.pop('stdin', None)
...@@ -381,9 +416,9 @@ def communicate(args, timeout=None, **kwargs): ...@@ -381,9 +416,9 @@ def communicate(args, timeout=None, **kwargs):
proc = Popen(args, **kwargs) proc = Popen(args, **kwargs)
if stdin: if stdin:
return proc.communicate(stdin, timeout), proc.returncode return proc.communicate(stdin, timeout, nag_timer), proc.returncode
else: else:
return proc.communicate(None, timeout), proc.returncode return proc.communicate(None, timeout, nag_timer), proc.returncode
def call(args, **kwargs): def call(args, **kwargs):
......
...@@ -77,7 +77,8 @@ class DefaultsTest(auto_stub.TestCase): ...@@ -77,7 +77,8 @@ class DefaultsTest(auto_stub.TestCase):
results.update(kwargs) results.update(kwargs)
results['args'] = args results['args'] = args
@staticmethod @staticmethod
def communicate(input=None, timeout=None): # pylint: disable=W0622 # pylint: disable=W0622
def communicate(input=None, timeout=None, nag_timer=None):
return None, None return None, None
self.mock(subprocess2, 'Popen', fake_Popen) self.mock(subprocess2, 'Popen', fake_Popen)
return results return results
...@@ -591,7 +592,27 @@ class S2Test(BaseTestCase): ...@@ -591,7 +592,27 @@ class S2Test(BaseTestCase):
except Blow: except Blow:
self.assertNotEquals(0, proc.returncode) self.assertNotEquals(0, proc.returncode)
def test_nag_timer(self):
w = []
l = logging.getLogger()
class _Filter(logging.Filter):
def filter(self, record):
if record.levelno == logging.WARNING:
w.append(record.getMessage().lstrip())
return 0
f = _Filter()
l.addFilter(f)
proc = subprocess2.Popen(
self.exe + ['--stdout', '--sleep_first'], stdout=PIPE)
res = proc.communicate(nag_timer=3), proc.returncode
l.removeFilter(f)
self._check_res(res, 'A\nBB\nCCC\n', None, 0)
expected = ['No output for 3 seconds from command:', proc.cmd_str,
'No output for 6 seconds from command:', proc.cmd_str,
'No output for 9 seconds from command:', proc.cmd_str]
self.assertEquals(w, expected)
def child_main(args): def child_main(args):
if sys.platform == 'win32': if sys.platform == 'win32':
# Annoying, make sure the output is not translated on Windows. # Annoying, make sure the output is not translated on Windows.
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment