refactor - This PR closes #3 #8
Merged
gyulaid
merged 2 commits from refactor
into master
3 years ago
11 changed files with 237 additions and 150 deletions
@ -1,3 +1,3 @@ |
|||
from .cli import main |
|||
from cli import main |
|||
|
|||
main() |
|||
|
@ -1,2 +1,10 @@ |
|||
class NonZeroRetcode(Exception): |
|||
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