Browse Source
Co-authored-by: gyulaid <gyulaid@gyulai.cloud> Reviewed-on: #8 Co-authored-by: Daniel Gyulai <gyulaid@gyulai.cloud> Co-committed-by: Daniel Gyulai <gyulaid@gyulai.cloud>pull/10/head 0.0.2
11 changed files with 237 additions and 150 deletions
@ -1,3 +1,3 @@ |
|||||
from .cli import main |
from cli import main |
||||
|
|
||||
main() |
main() |
||||
|
@ -1,2 +1,10 @@ |
|||||
class NonZeroRetcode(Exception): |
class NonZeroRetcode(Exception): |
||||
pass |
pass |
||||
|
|
||||
|
|
||||
|
class RunnerError(Exception): |
||||
|
pass |
||||
|
|
||||
|
|
||||
|
class ConfigException(Exception): |
||||
|
pass |
||||
|
@ -1,48 +0,0 @@ |
|||||
import subprocess |
|
||||
import os |
|
||||
|
|
||||
from exceptions import NonZeroRetcode |
|
||||
|
|
||||
|
|
||||
class PythonRunner(): |
|
||||
def __init__(self, repo, vpython) -> None: |
|
||||
self.vpython = vpython |
|
||||
self.repopath = repo |
|
||||
|
|
||||
def __get_env(self, overrides): |
|
||||
env = os.environ.copy() |
|
||||
if overrides is not None: |
|
||||
for key, value in overrides.items(): |
|
||||
env[key] = value |
|
||||
return env |
|
||||
|
|
||||
def ghetto_glob(self, command): |
|
||||
new_command = [] |
|
||||
for item in command: |
|
||||
if "*" in item: |
|
||||
dir = os.path.abspath(os.path.dirname(item)) |
|
||||
base_name = os.path.basename(item) |
|
||||
if os.path.isdir(dir): |
|
||||
item_parts = base_name.split("*") |
|
||||
print(item_parts) |
|
||||
for file in os.listdir(dir): |
|
||||
if item_parts[0] in file and item_parts[1] in file: |
|
||||
new_command.append(os.path.join(dir, file)) |
|
||||
else: |
|
||||
new_command.append(item) |
|
||||
return new_command |
|
||||
|
|
||||
def run(self, command, workdir=None, env=None): |
|
||||
if workdir is not None: |
|
||||
pwd = os.path.abspath(os.path.join(self.repopath, workdir)) |
|
||||
else: |
|
||||
pwd = self.repopath |
|
||||
run_env = self.__get_env(env) |
|
||||
run_command = self.ghetto_glob(command) |
|
||||
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 Exception(f"Invalid path for shell command: {pwd}") |
|
@ -0,0 +1,34 @@ |
|||||
|
from runners.pythonrunner import PythonRunner |
||||
|
from os import getcwd |
||||
|
|
||||
|
|
||||
|
class Factory(): |
||||
|
def __init__(self) -> None: |
||||
|
self.runnertypes = self.__load_runners() |
||||
|
self.runners = {} |
||||
|
self.workdir = getcwd() |
||||
|
self.globals = {} |
||||
|
|
||||
|
def __load_runners(self): |
||||
|
# TODO: Runners can be imported via cli too |
||||
|
# module = __import__("module_file") |
||||
|
# my_class = getattr(module, "class_name") |
||||
|
|
||||
|
return {"python": PythonRunner} |
||||
|
|
||||
|
def set_globals(self, globals): |
||||
|
self.globals = globals |
||||
|
|
||||
|
def update_globals(self, update): |
||||
|
if "env" in update: |
||||
|
self.globals["env"].update(update["env"]) |
||||
|
|
||||
|
def update_runners(self, config): |
||||
|
for runnertype, runnerconfig in config.items(): |
||||
|
if runnertype != "global": |
||||
|
self.get_runner(runnertype).update_config(runnerconfig) |
||||
|
|
||||
|
def get_runner(self, runnertype): |
||||
|
if runnertype not in self.runners: |
||||
|
self.runners[runnertype] = self.runnertypes[runnertype](self.workdir, self.globals) |
||||
|
return self.runners[runnertype] |
@ -0,0 +1,96 @@ |
|||||
|
import subprocess |
||||
|
import os |
||||
|
import sys |
||||
|
import shlex |
||||
|
|
||||
|
from exceptions import NonZeroRetcode, RunnerError, ConfigException |
||||
|
|
||||
|
|
||||
|
# same venv across all runs! |
||||
|
class PythonRunner(): |
||||
|
def __init__(self, workdir, defaults) -> 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.__init_venv() |
||||
|
|
||||
|
def __init_venv(self): |
||||
|
if os.name == "nt": # Windows |
||||
|
self.vpython = os.path.join(self.virtual_dir, "Scripts", "python.exe") |
||||
|
else: # Linux & Mac |
||||
|
self.vpython = os.path.join(self.virtual_dir, "bin", "python3") |
||||
|
|
||||
|
if not os.path.exists(self.vpython): |
||||
|
with subprocess.Popen([sys.executable, "-m", "virtualenv", self.virtual_dir], |
||||
|
stdout=subprocess.PIPE, stderr=subprocess.PIPE) as p: |
||||
|
p.wait() |
||||
|
if p.returncode != 0: |
||||
|
sys.stdout.buffer.write(p.stderr.read()) |
||||
|
raise RunnerError("PythonRunner: Could not create virtualenv") |
||||
|
else: |
||||
|
print(f"PythonRunner: Virtualenv initialized at {self.virtual_dir}") |
||||
|
else: |
||||
|
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 |
||||
|
def update_config(self, config): |
||||
|
if "dependencies" in config: |
||||
|
for dependency in config["dependencies"]: |
||||
|
# TODO: Check what happens with fixed version |
||||
|
with subprocess.Popen([self.vpython, "-m", "pip", "install", dependency, "--upgrade"], 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 dependency: {dependency} ({p.returncode})")) |
||||
|
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): |
||||
|
new_command = [] |
||||
|
for item in command: |
||||
|
if "*" in item: |
||||
|
dir = os.path.abspath(os.path.dirname(item)) |
||||
|
base_name = os.path.basename(item) |
||||
|
if os.path.isdir(dir): |
||||
|
item_parts = base_name.split("*") |
||||
|
print(item_parts) |
||||
|
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)) |
||||
|
else: |
||||
|
new_command.append(item) |
||||
|
return new_command |
||||
|
|
||||
|
# Executes the given job in the one and only venv |
||||
|
# parameter shall be 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"])) |
||||
|
else: |
||||
|
pwd = self.workdir |
||||
|
run_env = self.env_vars.copy() |
||||
|
if "env" in job_spec: |
||||
|
for env_var in job_spec["env"]: |
||||
|
run_env[env_var["name"]] = env_var["value"] |
||||
|
if "commands" in job_spec: |
||||
|
commands = job_spec["commands"] |
||||
|
for command in commands: |
||||
|
# TODO: only split if command is not an array |
||||
|
run_command = self.__ghetto_glob(shlex.split(command)) |
||||
|
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}") |
||||
|
else: |
||||
|
raise ConfigException(f"PythonRunner: No commands specified in step {job_spec['name']}") |
@ -0,0 +1,29 @@ |
|||||
|
runners: |
||||
|
global: |
||||
|
env: |
||||
|
- name: A |
||||
|
value: A |
||||
|
- name: B |
||||
|
value: B |
||||
|
- name: C |
||||
|
value: C |
||||
|
workdir: packages |
||||
|
python: |
||||
|
env: |
||||
|
- name: A |
||||
|
value: D |
||||
|
dependencies: |
||||
|
- flake8 |
||||
|
- build |
||||
|
jobs: |
||||
|
- name: env |
||||
|
type: python |
||||
|
env: |
||||
|
- name: B |
||||
|
value: E |
||||
|
commands: |
||||
|
- "-c \"import os; print(os.environ)\"" |
||||
|
- name: lint |
||||
|
workdir: alice-ci |
||||
|
commands: |
||||
|
- "-m flake8 --ignore E501" |
Loading…
Reference in new issue