10 changed files with 323 additions and 63 deletions
@ -0,0 +1,38 @@ |
|||
import logging |
|||
import os |
|||
|
|||
from alice.exceptions import ConfigException |
|||
|
|||
|
|||
class ConfigHolder: |
|||
__instance = None |
|||
file_name = ".alice" |
|||
|
|||
@staticmethod |
|||
def getInstance(): |
|||
""" Static access method. """ |
|||
if ConfigHolder.__instance is None: |
|||
ConfigHolder() |
|||
return ConfigHolder.__instance |
|||
|
|||
def __init__(self): |
|||
""" Virtually private constructor. """ |
|||
if ConfigHolder.__instance is not None: |
|||
raise Exception("This class is a singleton!") |
|||
else: |
|||
ConfigHolder.__instance = self |
|||
config = os.path.abspath(os.path.join(os.getcwd(), self.file_name)) |
|||
self.vars = {} |
|||
if os.path.isfile(config): |
|||
with open(config) as f: |
|||
for line in f: |
|||
items = line.split("=") |
|||
if len(items) > 1: |
|||
self.vars[items[0]] = line.replace(f"{items[0]}=", "") |
|||
logging.debug(f"Loaded from {self.file_name}: {self.vars}") |
|||
|
|||
def get(self, key): |
|||
try: |
|||
self.vars[key] |
|||
except KeyError: |
|||
raise ConfigException(f"{key} not defined in .conf!") |
@ -1,19 +1,223 @@ |
|||
# TODO Implement |
|||
class DockerConfig: |
|||
from enum import Enum |
|||
import json |
|||
import logging |
|||
from os import path, getcwd |
|||
import docker |
|||
|
|||
from alice.runners.pyutils import grab_from, gen_dict |
|||
from alice.exceptions import ConfigException, NonZeroRetcode, RunnerError |
|||
|
|||
|
|||
class ImageSource(Enum): |
|||
NONE = 1 |
|||
BUILD = 2 |
|||
PULL = 3 |
|||
|
|||
|
|||
def get_user(config, default): |
|||
if "credentials" in config: |
|||
if "username" in config["credentials"]: |
|||
data = config["credentials"]["username"] |
|||
if isinstance(data, str): |
|||
return data |
|||
else: |
|||
return grab_from(data) |
|||
return default |
|||
|
|||
|
|||
def get_pass(config, default): |
|||
if "credentials" in config: |
|||
if "password" in config["credentials"]: |
|||
data = config["credentials"]["password"] |
|||
if isinstance(data, str): |
|||
return data |
|||
else: |
|||
return grab_from(data) |
|||
return default |
|||
|
|||
|
|||
def get_provider(config, default, default_type): |
|||
if "image" in config: |
|||
build = False |
|||
pull = False |
|||
candidate_type = default_type |
|||
if "build" in config["image"]: |
|||
build = True |
|||
if default_type == ImageSource.BUILD: |
|||
candidate = default.copy(config["image"]["build"]) |
|||
else: |
|||
candidate = Builder(config["image"]["build"]) |
|||
candidate_type = ImageSource.BUILD |
|||
elif "pull" in config["image"]: |
|||
pull = True |
|||
if default_type == ImageSource.PULL: |
|||
candidate = default.copy(config["image"]["pull"]) |
|||
else: |
|||
candidate = Puller(config["image"]["pull"]) |
|||
candidate_type = ImageSource.PULL |
|||
|
|||
if build and pull: |
|||
raise ConfigException("[DockerRunner] Can't build and pull the same image!") |
|||
|
|||
return candidate, candidate_type |
|||
return default, default_type |
|||
|
|||
|
|||
class Tagger: |
|||
def __init__(self, config={}) -> None: |
|||
pass |
|||
self.name = config.get("name", None) |
|||
self.username = get_user(config, None) |
|||
self.password = get_pass(config, None) |
|||
self.publish = config.get("publish", False) |
|||
|
|||
def copy(self, job_config): |
|||
t = Tagger() |
|||
t.name = job_config.get("name", self.name) |
|||
t.username = get_user(job_config, self.username) |
|||
t.password = get_pass(job_config, self.password) |
|||
t.publish = job_config.get("publish", self.publish) |
|||
return t |
|||
|
|||
def __str__(self) -> str: |
|||
data = { |
|||
"name": self.name, |
|||
"publish": self.publish, |
|||
"credentials": { |
|||
"username": self.username, |
|||
"password": self.password |
|||
} |
|||
} |
|||
return f"{data}" |
|||
|
|||
|
|||
class Builder(): |
|||
def __init__(self, config) -> None: |
|||
self.dir = path.abspath(config.get("dir", getcwd())) |
|||
self.dockerfile = config.get("dockerfile", None) |
|||
self.name = config.get("name", None) |
|||
self.args = gen_dict(config.get("args", [])) |
|||
|
|||
def copy(self, job_config): |
|||
pass |
|||
b = Builder({}) |
|||
b.dir = path.abspath(path.join(self.dir, job_config.get("dir", "."))) |
|||
b.dockerfile = job_config.get("dockerfile", self.dockerfile) |
|||
b.name = job_config.get("name", self.name) |
|||
b.args = self.args.copy().update(gen_dict(job_config.get("args", []))) |
|||
return b |
|||
|
|||
def __str__(self) -> str: |
|||
data = { |
|||
"type": "builder", |
|||
"dir": self.dir, |
|||
"dockerfile": self.dockerfile, |
|||
"name": self.name, |
|||
"args": self.args |
|||
} |
|||
return json.dumps(data) |
|||
|
|||
def prepare(self, client): |
|||
print(f"[DockerRunner] Building image {self.name}") |
|||
if self.dockerfile is None: |
|||
self.dockerfile = "Dockerfile" |
|||
try: |
|||
image, log = client.images.build(path=self.dir, |
|||
dockerfile=self.dockerfile, |
|||
tag=self.name, |
|||
buildargs=self.args, |
|||
labels={"builder": "alice-ci"}) |
|||
for i in log: |
|||
logging.debug(i) |
|||
return image |
|||
except docker.errors.BuildError as e: |
|||
raise RunnerError(f"[DockerRunner] Build failed: {e}") |
|||
except docker.errors.APIError as e: |
|||
raise RunnerError(f"[DockerRunner] Error: {e}") |
|||
|
|||
|
|||
class Puller(): |
|||
def __init__(self, config={}) -> None: |
|||
self.name = config.get("name", None) |
|||
self.username = get_user(config, None) |
|||
self.password = get_pass(config, None) |
|||
|
|||
def copy(self, job_config={}): |
|||
p = Puller() |
|||
p.name = job_config.get("name", self.name) |
|||
p.username = get_user(job_config, self.username) |
|||
p.password = get_pass(job_config, self.password) |
|||
|
|||
def __str__(self) -> str: |
|||
data = { |
|||
"name": self.name, |
|||
"credentials": { |
|||
"username": self.username, |
|||
"password": self.password |
|||
} |
|||
} |
|||
return f"{data}" |
|||
|
|||
def prepare(self, client): |
|||
print(f"[DockerRunner] Pulling image {self.name}") |
|||
return client.images.pull(self.name) |
|||
|
|||
|
|||
class DockerConfig: |
|||
def __init__(self, config={}) -> None: |
|||
self.username = get_user(config, None) |
|||
self.password = get_pass(config, None) |
|||
self.image_provider, self.provider_type = get_provider(config, None, ImageSource.NONE) |
|||
self.tagger = Tagger(config.get("tag", {})) |
|||
self.commands = config.get("commands", []) |
|||
|
|||
def copy(self, job_config={}): |
|||
d = DockerConfig() |
|||
d.username = get_user(job_config, self.username) |
|||
d.password = get_pass(job_config, self.password) |
|||
d.image_provider, d.provider_type = get_provider(job_config, self.image_provider, self.provider_type) |
|||
d.tagger = self.tagger.copy(job_config.get("tag", {})) |
|||
d.commands = self.commands.copy() + job_config.get("commands", []) |
|||
return d |
|||
|
|||
def __str__(self) -> str: |
|||
data = { |
|||
"credentials": { |
|||
"username": {self.username}, |
|||
"password": {self.password} |
|||
}, |
|||
"image": self.image_provider.__str__(), |
|||
"commands": self.commands, |
|||
"tag": self.tagger.__str__() |
|||
} |
|||
return f"{data}" |
|||
|
|||
|
|||
# Supported tasks: |
|||
# - Build image |
|||
# - Push image to repo |
|||
# - Run arbitrary code in image |
|||
class DockerRunner(): |
|||
def __init__(self) -> None: |
|||
pass |
|||
def __init__(self, config) -> None: |
|||
logging.info("[DockerRunner] Initializing") |
|||
self.config = DockerConfig(config) |
|||
self.client = docker.from_env() |
|||
|
|||
def run(self, job_spec): |
|||
pass |
|||
job_config = self.config.copy(job_spec) |
|||
if job_config.image_provider is None: |
|||
raise RunnerError("[DockerRunner] No image provider configured!") |
|||
image = job_config.image_provider.prepare(self.client) |
|||
logging.info(f"[DockerRunner] Image: {image.tags} ({image.id})") |
|||
|
|||
if len(job_config.commands) > 0: |
|||
container = self.client.containers.run(image=image.id, |
|||
entrypoint=["sleep", "infinity"], |
|||
detach=True, |
|||
auto_remove=True) |
|||
try: |
|||
for i in job_config.commands: |
|||
command = ["/bin/sh", "-c", i] |
|||
logging.debug(f"[DockerRunner] Command array: {command}") |
|||
code, output = container.exec_run(cmd=command) |
|||
for line in output.decode("UTF-8").splitlines(): |
|||
print(f"[{job_spec['name']}] {line}") |
|||
if code != 0: |
|||
raise NonZeroRetcode(f"Command {i} returned code {code}") |
|||
finally: |
|||
if container is not None: |
|||
container.stop() |
|||
|
@ -0,0 +1,9 @@ |
|||
FROM ubuntu:latest |
|||
|
|||
RUN apt update && apt install -y python3 |
|||
|
|||
ADD hello.py /opt/hello.py |
|||
|
|||
#ENTRYPOINT [ "/bin/sh", "-c" ] |
|||
|
|||
#CMD ["/usr/local/python/bin/python3", "/opt/hello.py"] |
@ -0,0 +1,2 @@ |
|||
if __name__ == "__main__": |
|||
print("Hi Mom!") |
Loading…
Reference in new issue