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