|
|
@ -1,3 +1,4 @@ |
|
|
|
import logging |
|
|
|
import subprocess |
|
|
|
import os |
|
|
|
import sys |
|
|
@ -9,10 +10,8 @@ from alice.runners.pyutils import glob_command |
|
|
|
|
|
|
|
# TODO: Handle config like PyPiConfig |
|
|
|
class PythonRunner: |
|
|
|
def __init__(self, params, config) -> None: |
|
|
|
self.verbose = params["verbose"] |
|
|
|
if self.verbose: |
|
|
|
print("[PythonRunner] Initializing") |
|
|
|
def __init__(self, config) -> None: |
|
|
|
logging.info("[PythonRunner] Initializing") |
|
|
|
self.workdir = config["workdir"] |
|
|
|
self.virtual_dir = os.path.abspath(os.path.join(self.workdir, "venv")) |
|
|
|
self.config = config |
|
|
@ -25,7 +24,7 @@ class PythonRunner: |
|
|
|
self.vpython = os.path.join(self.virtual_dir, "bin", "python3") |
|
|
|
|
|
|
|
if not os.path.exists(self.vpython): |
|
|
|
print("[PythonRunner] Initializing venv") |
|
|
|
logging.info("[PythonRunner] Initializing venv") |
|
|
|
with subprocess.Popen([sys.executable, "-m", "virtualenv", self.virtual_dir], |
|
|
|
stdout=subprocess.PIPE, stderr=subprocess.PIPE) as p: |
|
|
|
p.wait() |
|
|
@ -33,26 +32,22 @@ class PythonRunner: |
|
|
|
sys.stdout.buffer.write(p.stderr.read()) |
|
|
|
raise RunnerError("[PythonRunner] Could not create virtualenv") |
|
|
|
else: |
|
|
|
if self.verbose: |
|
|
|
print(f"[PythonRunner] Virtualenv initialized at {self.virtual_dir}") |
|
|
|
logging.info(f"[PythonRunner] Virtualenv initialized at {self.virtual_dir}") |
|
|
|
else: |
|
|
|
if self.verbose: |
|
|
|
print(f"[PythonRunner] Found virtualenv at {self.virtual_dir}") |
|
|
|
logging.info(f"[PythonRunner] Found virtualenv at {self.virtual_dir}") |
|
|
|
dependencies = self.config.get("dependencies", []) |
|
|
|
if len(dependencies) > 0: |
|
|
|
if self.verbose: |
|
|
|
print(f"[PythonRunner] Ensuring dependencies: {', '.join(dependencies)}") |
|
|
|
logging.info(f"[PythonRunner] Ensuring dependencies: {', '.join(dependencies)}") |
|
|
|
command = [self.vpython, "-m", "pip", "install"] + dependencies |
|
|
|
with subprocess.Popen(command, stdout=subprocess.PIPE, stderr=subprocess.PIPE) as p: |
|
|
|
p.wait() |
|
|
|
if p.returncode != 0: |
|
|
|
sys.stdout.buffer.write(p.stderr.read()) |
|
|
|
raise(RunnerError(f"[PythonRunner] Could not install dependencies: {dependencies} ({p.returncode})")) |
|
|
|
if self.verbose: |
|
|
|
print("[PythonRunner] Installation done") |
|
|
|
logging.info("[PythonRunner] Installation done") |
|
|
|
|
|
|
|
# Executes the given job in the one and only venv |
|
|
|
# parameter shall be the raw jobscpec |
|
|
|
# parameter is the raw jobscpec |
|
|
|
def run(self, job_spec): |
|
|
|
if "workdir" in job_spec: |
|
|
|
pwd = os.path.abspath(os.path.join(self.workdir, job_spec["workdir"])) |
|
|
@ -65,16 +60,14 @@ class PythonRunner: |
|
|
|
if "commands" in job_spec: |
|
|
|
commands = job_spec["commands"] |
|
|
|
for command in commands: |
|
|
|
if self.verbose: |
|
|
|
print(f"[PythonRunner] Raw command: {command}") |
|
|
|
logging.debug(f"[PythonRunner] Raw command: {command}") |
|
|
|
# TODO: only split if command is not an array |
|
|
|
if "*" in command: |
|
|
|
run_command = glob_command(shlex.split(command), pwd, self.verbose) |
|
|
|
run_command = glob_command(shlex.split(command), pwd) |
|
|
|
else: |
|
|
|
run_command = shlex.split(command) |
|
|
|
if self.verbose: |
|
|
|
print(f"[PythonRunner] Command to execute: {run_command}") |
|
|
|
print(f"[PythonRunner] Workdir: {pwd}") |
|
|
|
logging.info(f"[PythonRunner] Command to execute: {run_command}") |
|
|
|
logging.debug(f"[PythonRunner] Workdir: {pwd}") |
|
|
|
if os.path.isdir(pwd): |
|
|
|
with subprocess.Popen([self.vpython] + run_command, cwd=pwd, env=run_env) as p: |
|
|
|
p.wait() |
|
|
|