|
|
@ -2,19 +2,21 @@ import subprocess |
|
|
|
import os |
|
|
|
import sys |
|
|
|
import shlex |
|
|
|
from tabnanny import verbose |
|
|
|
|
|
|
|
from alice.exceptions import NonZeroRetcode, RunnerError, ConfigException |
|
|
|
|
|
|
|
|
|
|
|
# same venv across all runs! |
|
|
|
class PythonRunner(): |
|
|
|
def __init__(self, workdir, defaults) -> None: |
|
|
|
def __init__(self, workdir, defaults, verbose) -> None: |
|
|
|
self.workdir = workdir |
|
|
|
self.virtual_dir = os.path.abspath(os.path.join(workdir, "venv")) |
|
|
|
self.config = defaults |
|
|
|
self.env_vars = os.environ.copy() |
|
|
|
for env_var in defaults["env"]: |
|
|
|
self.env_vars[env_var["name"]] = env_var["value"] |
|
|
|
self.verbose = verbose |
|
|
|
|
|
|
|
self.__init_venv() |
|
|
|
|
|
|
@ -30,11 +32,13 @@ class PythonRunner(): |
|
|
|
p.wait() |
|
|
|
if p.returncode != 0: |
|
|
|
sys.stdout.buffer.write(p.stderr.read()) |
|
|
|
raise RunnerError("PythonRunner: Could not create virtualenv") |
|
|
|
raise RunnerError("[PythonRunner] Could not create virtualenv") |
|
|
|
else: |
|
|
|
print(f"PythonRunner: Virtualenv initialized at {self.virtual_dir}") |
|
|
|
if self.verbose: |
|
|
|
print(f"[PythonRunner] Virtualenv initialized at {self.virtual_dir}") |
|
|
|
else: |
|
|
|
print(f"PythonRunner: Found virtualenv at {self.virtual_dir}") |
|
|
|
if self.verbose: |
|
|
|
print(f"[PythonRunner] Found virtualenv at {self.virtual_dir}") |
|
|
|
|
|
|
|
# Stores common defaults for all jobs - all types! |
|
|
|
# Also - dependency install by config is only allowed in this step |
|
|
@ -47,25 +51,35 @@ class PythonRunner(): |
|
|
|
p.wait() |
|
|
|
if p.returncode != 0: |
|
|
|
sys.stdout.buffer.write(p.stderr.read()) |
|
|
|
raise(RunnerError(f"PythonRunner: Could not install dependency: {dependency} ({p.returncode})")) |
|
|
|
raise(RunnerError(f"[PythonRunner] Could not install dependency: {dependency} ({p.returncode})")) |
|
|
|
if "env" in config: |
|
|
|
for env_var in config["env"]: |
|
|
|
self.env_vars[env_var["name"]] = env_var["value"] |
|
|
|
if "workdir" in config and config["workdir"] is not None: |
|
|
|
self.workdir = os.path.join(self.workdir, config["workdir"]) |
|
|
|
|
|
|
|
def __ghetto_glob(self, command): |
|
|
|
def __ghetto_glob(self, command, workdir): |
|
|
|
if self.verbose: |
|
|
|
print(f"[PythonRunner][Globbing] Starting command: {' '.join(command)}") |
|
|
|
new_command = [] |
|
|
|
for item in command: |
|
|
|
if "*" in item: |
|
|
|
dir = os.path.abspath(os.path.dirname(item)) |
|
|
|
if self.verbose: |
|
|
|
print(f"[PythonRunner][Globbing] Found item: [{item}]") |
|
|
|
dir = os.path.abspath(os.path.join(workdir, os.path.dirname(item))) |
|
|
|
base_name = os.path.basename(item) |
|
|
|
if os.path.isdir(dir): |
|
|
|
item_parts = base_name.split("*") |
|
|
|
for file in os.listdir(dir): |
|
|
|
# TODO: Fix ordering! A*B = B*A = AB* |
|
|
|
if item_parts[0] in file and item_parts[1] in file: |
|
|
|
new_command.append(os.path.join(dir, file)) |
|
|
|
new_item = os.path.join(dir, file) |
|
|
|
if self.verbose: |
|
|
|
print(f"[PythonRunner][Globbing] Substitute: {new_item}") |
|
|
|
new_command.append(new_item) |
|
|
|
else: |
|
|
|
if self.verbose: |
|
|
|
print(f"[PythonRunner][Globbing] Dir not exists: {dir}") |
|
|
|
else: |
|
|
|
new_command.append(item) |
|
|
|
return new_command |
|
|
@ -84,14 +98,19 @@ class PythonRunner(): |
|
|
|
if "commands" in job_spec: |
|
|
|
commands = job_spec["commands"] |
|
|
|
for command in commands: |
|
|
|
if self.verbose: |
|
|
|
print(f"[PythonRunner] Raw command: {command}") |
|
|
|
# TODO: only split if command is not an array |
|
|
|
run_command = self.__ghetto_glob(shlex.split(command)) |
|
|
|
run_command = self.__ghetto_glob(shlex.split(command), pwd) |
|
|
|
if self.verbose: |
|
|
|
print(f"[PythonRunner] Command to execute: {run_command}") |
|
|
|
print(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() |
|
|
|
if p.returncode != 0: |
|
|
|
raise NonZeroRetcode(f"Command {command} returned code {p.returncode}") |
|
|
|
else: |
|
|
|
raise RunnerError(f"PythonRunner: Invalid path for shell command: {pwd}") |
|
|
|
raise RunnerError(f"[PythonRunner] Invalid path for shell command: {pwd}") |
|
|
|
else: |
|
|
|
raise ConfigException(f"PythonRunner: No commands specified in step {job_spec['name']}") |
|
|
|
raise ConfigException(f"[PythonRunner] No commands specified in step {job_spec['name']}") |
|
|
|