10 changed files with 246 additions and 0 deletions
@ -0,0 +1,6 @@ |
|||||
|
[build-system] |
||||
|
requires = [ |
||||
|
"setuptools>=42", |
||||
|
"wheel" |
||||
|
] |
||||
|
build-backend = "setuptools.build_meta" |
@ -0,0 +1,23 @@ |
|||||
|
[metadata] |
||||
|
name = alice |
||||
|
version = 0.0.1 |
||||
|
author = Daniel Gyulai |
||||
|
description = Alice CI framework |
||||
|
long_description = file: README.md |
||||
|
long_description_content_type = text/markdown |
||||
|
url = https://git.gyulai.cloud/gyulaid/alice |
||||
|
project_urls = |
||||
|
Bug Tracker = https://git.gyulai.cloud/gyulaid/alice/issues |
||||
|
classifiers = |
||||
|
Programming Language :: Python :: 3 |
||||
|
License :: OSI Approved :: MIT License |
||||
|
Operating System :: OS Independent |
||||
|
|
||||
|
[options] |
||||
|
package_dir = |
||||
|
= src |
||||
|
packages = find: |
||||
|
python_requires = >=3.6 |
||||
|
|
||||
|
[options.packages.find] |
||||
|
where = src |
@ -0,0 +1,3 @@ |
|||||
|
from .cli import main |
||||
|
|
||||
|
main() |
@ -0,0 +1,85 @@ |
|||||
|
# Sourcefor App class: |
||||
|
# https://stackoverflow.com/questions/57593111/how-to-call-pip-from-a-python-script-and-make-it-install-locally-to-that-script |
||||
|
import os |
||||
|
import sys |
||||
|
import subprocess |
||||
|
import argparse |
||||
|
|
||||
|
|
||||
|
class App: |
||||
|
def __init__(self, virtual_dir): |
||||
|
self.virtual_dir = virtual_dir |
||||
|
if os.name == "nt": |
||||
|
self.virtual_python = os.path.join(self.virtual_dir, "Scripts", "python.exe") |
||||
|
else: |
||||
|
self.virtual_python = os.path.join(self.virtual_dir, "bin", "python3") |
||||
|
|
||||
|
def install_virtual_env(self): |
||||
|
self.pip_install("virtualenv") |
||||
|
if not os.path.exists(self.virtual_python): |
||||
|
import subprocess |
||||
|
subprocess.call([sys.executable, "-m", "virtualenv", self.virtual_dir]) |
||||
|
else: |
||||
|
print("found virtual python: " + self.virtual_python) |
||||
|
|
||||
|
def is_venv(self): |
||||
|
return sys.prefix == self.virtual_dir |
||||
|
|
||||
|
def restart_under_venv(self): |
||||
|
print("Restarting under virtual environment " + self.virtual_dir) |
||||
|
with subprocess.Popen([self.virtual_python, __file__] + sys.argv[1:]) as p: |
||||
|
p.wait() |
||||
|
exit(p.returncode) |
||||
|
|
||||
|
def pip_install(self, package, import_name=None): |
||||
|
try: |
||||
|
if import_name is None: |
||||
|
__import__(package) |
||||
|
else: |
||||
|
__import__(import_name) |
||||
|
except: # noqa: E722 |
||||
|
subprocess.call([sys.executable, "-m", "pip", "install", package, "--upgrade"]) |
||||
|
|
||||
|
def __gen_env(self, param_list): |
||||
|
env_vars = {} |
||||
|
for item in param_list: |
||||
|
item_parts = item.split("=") |
||||
|
if len(item_parts) == 2: |
||||
|
env_vars[item_parts[0]] = item_parts[1] |
||||
|
return env_vars |
||||
|
|
||||
|
def run(self, args, repoDir): |
||||
|
if not self.is_venv(): |
||||
|
self.install_virtual_env() |
||||
|
self.restart_under_venv() |
||||
|
else: |
||||
|
print("Running under virtual environment") |
||||
|
self.pip_install("pyyaml", "yaml") |
||||
|
|
||||
|
from jobparser import JobParser |
||||
|
jobParser = JobParser(args.input, repoDir, self.virtual_python) |
||||
|
for name, import_name in jobParser.get_modules(): |
||||
|
self.pip_install(name, import_name) |
||||
|
|
||||
|
print("Begin pipeline steps...") |
||||
|
for step in args.steps: |
||||
|
if step in jobParser.jobs: |
||||
|
jobParser.jobs[step].run_commands(self.__gen_env(args.env)) |
||||
|
print(f"Step {step}: SUCCESS") |
||||
|
else: |
||||
|
raise Exception(f"Step {step} not found in {args.input}") |
||||
|
|
||||
|
def main(): |
||||
|
pathToScriptDir = os.path.dirname(os.path.realpath(__file__)) |
||||
|
repoDir = os.path.join(pathToScriptDir, "..") |
||||
|
app = App(os.path.join(pathToScriptDir, "venv")) |
||||
|
|
||||
|
parser = argparse.ArgumentParser() |
||||
|
parser.add_argument("steps", nargs='+') |
||||
|
parser.add_argument("-i", "--input", default="ci.yaml") |
||||
|
parser.add_argument("-e", "--env", nargs='*', default=[]) |
||||
|
args = parser.parse_args() |
||||
|
app.run(args, repoDir) |
||||
|
|
||||
|
if __name__ == "__main__": |
||||
|
main() |
@ -0,0 +1 @@ |
|||||
|
# TODO Implement |
@ -0,0 +1,2 @@ |
|||||
|
class NonZeroRetcode(Exception): |
||||
|
pass |
@ -0,0 +1,78 @@ |
|||||
|
import yaml |
||||
|
import shlex |
||||
|
from pythonrunner import PythonRunner |
||||
|
from exceptions import NonZeroRetcode |
||||
|
|
||||
|
|
||||
|
class DummyRunner(): |
||||
|
def __init__(self, type) -> None: |
||||
|
self.type = type |
||||
|
|
||||
|
def run(self, command, workdir=None, env=None): |
||||
|
raise Exception(f"Invalid runner type in config: {self.type}") |
||||
|
|
||||
|
|
||||
|
class Job(): |
||||
|
def __init__(self, type, repoDir, vpython, workspace, env={}) -> None: |
||||
|
self.runner = self.__get_runner(type, repoDir, vpython) |
||||
|
self.commands = [] |
||||
|
self.workspace = workspace |
||||
|
self.env = env |
||||
|
|
||||
|
def __get_runner(self, type, repoDir, vpython): |
||||
|
if type == "python": |
||||
|
return PythonRunner(repoDir, vpython) |
||||
|
else: |
||||
|
return DummyRunner(type) |
||||
|
|
||||
|
def run_commands(self, _env={}): |
||||
|
try: |
||||
|
if self.env is None: |
||||
|
env = _env.copy() |
||||
|
else: |
||||
|
env = self.env.copy() |
||||
|
env.update(_env) |
||||
|
for command in self.commands: |
||||
|
self.runner.run(command, self.workspace, env) |
||||
|
except NonZeroRetcode as n: |
||||
|
print(n) |
||||
|
exit(1) |
||||
|
|
||||
|
|
||||
|
class JobParser: |
||||
|
def __init__(self, file_path, repoDir, virtual_python) -> None: |
||||
|
with open(file_path) as f: |
||||
|
self.config = yaml.safe_load(f) |
||||
|
self.jobs = self.__get_jobs(repoDir, virtual_python) |
||||
|
|
||||
|
def __get_jobs(self, repoDir, virtual_python): |
||||
|
if "jobs" in self.config: |
||||
|
jobs = {} |
||||
|
for job_spec in self.config["jobs"]: |
||||
|
name = job_spec["name"] |
||||
|
if name in jobs: |
||||
|
raise Exception(f"Job with name {name} already exists!") |
||||
|
|
||||
|
job = Job(job_spec["type"], |
||||
|
repoDir, |
||||
|
virtual_python, |
||||
|
job_spec.get("workdir", None), |
||||
|
job_spec.get("env", None)) |
||||
|
|
||||
|
for cmd in job_spec["commands"]: |
||||
|
job.commands.append(shlex.split(cmd)) |
||||
|
jobs[name] = job |
||||
|
return jobs |
||||
|
|
||||
|
else: |
||||
|
raise Exception("No jobs defined in config") |
||||
|
|
||||
|
def get_modules(self): |
||||
|
modules = [] |
||||
|
if "runners" in self.config: |
||||
|
if "python" in self.config["runners"]: |
||||
|
if "dependencies" in self.config["runners"]["python"]: |
||||
|
for dep in self.config["runners"]["python"]["dependencies"]: |
||||
|
# (name, i_name) if i_name is defined, else (name, name) |
||||
|
modules.append((dep["name"], dep.get("import_name", dep["name"]))) |
||||
|
return modules |
@ -0,0 +1,48 @@ |
|||||
|
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}") |
Loading…
Reference in new issue