2016-02-25 20:03:07 +01:00
|
|
|
#! /usr/bin/env python
|
|
|
|
# encoding: utf-8
|
2018-01-01 20:53:49 +01:00
|
|
|
# Thomas Nagy, 2016-2018 (ita)
|
2016-02-25 20:03:07 +01:00
|
|
|
|
2016-09-03 18:13:38 +02:00
|
|
|
import os, sys, traceback, base64, signal
|
2016-02-25 20:03:07 +01:00
|
|
|
try:
|
|
|
|
import cPickle
|
|
|
|
except ImportError:
|
|
|
|
import pickle as cPickle
|
|
|
|
|
|
|
|
try:
|
|
|
|
import subprocess32 as subprocess
|
|
|
|
except ImportError:
|
|
|
|
import subprocess
|
|
|
|
|
2016-08-25 07:48:14 +02:00
|
|
|
try:
|
|
|
|
TimeoutExpired = subprocess.TimeoutExpired
|
|
|
|
except AttributeError:
|
2017-04-22 22:12:11 +02:00
|
|
|
class TimeoutExpired(Exception):
|
2016-08-25 07:48:14 +02:00
|
|
|
pass
|
|
|
|
|
2016-02-26 20:11:58 +01:00
|
|
|
def run():
|
2016-02-26 14:25:54 +01:00
|
|
|
txt = sys.stdin.readline().strip()
|
2016-02-25 20:03:07 +01:00
|
|
|
if not txt:
|
2016-02-25 22:58:11 +01:00
|
|
|
# parent process probably ended
|
2016-02-26 20:11:58 +01:00
|
|
|
sys.exit(1)
|
2016-02-26 14:25:54 +01:00
|
|
|
[cmd, kwargs, cargs] = cPickle.loads(base64.b64decode(txt))
|
2016-02-25 20:03:07 +01:00
|
|
|
cargs = cargs or {}
|
|
|
|
|
2019-05-27 18:05:02 +02:00
|
|
|
if not 'close_fds' in kwargs:
|
|
|
|
# workers have no fds
|
|
|
|
kwargs['close_fds'] = False
|
|
|
|
|
2016-02-25 20:03:07 +01:00
|
|
|
ret = 1
|
2016-02-26 20:11:58 +01:00
|
|
|
out, err, ex, trace = (None, None, None, None)
|
2016-02-25 20:03:07 +01:00
|
|
|
try:
|
|
|
|
proc = subprocess.Popen(cmd, **kwargs)
|
2016-08-25 07:48:14 +02:00
|
|
|
try:
|
|
|
|
out, err = proc.communicate(**cargs)
|
|
|
|
except TimeoutExpired:
|
2016-09-03 20:22:43 +02:00
|
|
|
if kwargs.get('start_new_session') and hasattr(os, 'killpg'):
|
2016-09-03 18:13:38 +02:00
|
|
|
os.killpg(proc.pid, signal.SIGKILL)
|
2016-09-03 20:22:43 +02:00
|
|
|
else:
|
2016-09-03 18:13:38 +02:00
|
|
|
proc.kill()
|
2016-08-28 11:03:04 +02:00
|
|
|
out, err = proc.communicate()
|
2016-09-03 22:24:29 +02:00
|
|
|
exc = TimeoutExpired(proc.args, timeout=cargs['timeout'], output=out)
|
|
|
|
exc.stderr = err
|
|
|
|
raise exc
|
2016-02-25 20:03:07 +01:00
|
|
|
ret = proc.returncode
|
2016-08-28 10:48:41 +02:00
|
|
|
except Exception as e:
|
2016-02-25 20:03:07 +01:00
|
|
|
exc_type, exc_value, tb = sys.exc_info()
|
|
|
|
exc_lines = traceback.format_exception(exc_type, exc_value, tb)
|
2016-02-26 20:11:58 +01:00
|
|
|
trace = str(cmd) + '\n' + ''.join(exc_lines)
|
|
|
|
ex = e.__class__.__name__
|
2016-02-25 20:03:07 +01:00
|
|
|
|
|
|
|
# it is just text so maybe we do not need to pickle()
|
2016-02-26 20:11:58 +01:00
|
|
|
tmp = [ret, out, err, ex, trace]
|
2016-02-26 14:25:54 +01:00
|
|
|
obj = base64.b64encode(cPickle.dumps(tmp))
|
|
|
|
sys.stdout.write(obj.decode())
|
|
|
|
sys.stdout.write('\n')
|
2016-02-25 20:03:07 +01:00
|
|
|
sys.stdout.flush()
|
|
|
|
|
2016-02-26 20:11:58 +01:00
|
|
|
while 1:
|
|
|
|
try:
|
|
|
|
run()
|
|
|
|
except KeyboardInterrupt:
|
|
|
|
break
|
|
|
|
|