forked from cerc-io/stack-orchestrator
129 lines
5.3 KiB
Python
129 lines
5.3 KiB
Python
# Copyright © 2023 Vulcanize
|
|
|
|
# This program is free software: you can redistribute it and/or modify
|
|
# it under the terms of the GNU Affero General Public License as published by
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
# (at your option) any later version.
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU Affero General Public License for more details.
|
|
|
|
# You should have received a copy of the GNU Affero General Public License
|
|
# along with this program. If not, see <http:#www.gnu.org/licenses/>.
|
|
|
|
from pathlib import Path
|
|
from kubernetes import client, config
|
|
|
|
from app.deploy.deployer import Deployer, DeployerConfigGenerator
|
|
from app.deploy.k8s.helpers import create_cluster, destroy_cluster, load_images_into_kind
|
|
from app.deploy.k8s.helpers import pods_in_deployment, log_stream_from_string, generate_kind_config
|
|
from app.deploy.k8s.cluster_info import ClusterInfo
|
|
from app.opts import opts
|
|
|
|
|
|
class K8sDeployer(Deployer):
|
|
name: str = "k8s"
|
|
core_api: client.CoreV1Api
|
|
apps_api: client.AppsV1Api
|
|
k8s_namespace: str = "default"
|
|
kind_cluster_name: str
|
|
cluster_info : ClusterInfo
|
|
|
|
def __init__(self, compose_files, compose_project_name, compose_env_file) -> None:
|
|
if (opts.o.debug):
|
|
print(f"Compose files: {compose_files}")
|
|
print(f"Project name: {compose_project_name}")
|
|
print(f"Env file: {compose_env_file}")
|
|
self.kind_cluster_name = compose_project_name
|
|
self.cluster_info = ClusterInfo()
|
|
self.cluster_info.int_from_pod_files(compose_files)
|
|
|
|
def connect_api(self):
|
|
config.load_kube_config(context=f"kind-{self.kind_cluster_name}")
|
|
self.core_api = client.CoreV1Api()
|
|
self.apps_api = client.AppsV1Api()
|
|
|
|
def up(self, detach, services):
|
|
# Create the kind cluster
|
|
# HACK: pass in the config file path here
|
|
create_cluster(self.kind_cluster_name, "./test-deployment-dir/kind-config.yml")
|
|
self.connect_api()
|
|
# Ensure the referenced containers are copied into kind
|
|
load_images_into_kind(self.kind_cluster_name, self.cluster_info.image_set)
|
|
# Figure out the PVCs for this deployment
|
|
pvcs = self.cluster_info.get_pvcs()
|
|
for pvc in pvcs:
|
|
if opts.o.debug:
|
|
print(f"Sending this: {pvc}")
|
|
pvc_resp = self.core_api.create_namespaced_persistent_volume_claim(body=pvc, namespace=self.k8s_namespace)
|
|
if opts.o.debug:
|
|
print("PVCs created:")
|
|
print(f"{pvc_resp}")
|
|
# Process compose files into a Deployment
|
|
deployment = self.cluster_info.get_deployment()
|
|
# Create the k8s objects
|
|
if opts.o.debug:
|
|
print(f"Sending this: {deployment}")
|
|
deployment_resp = self.apps_api.create_namespaced_deployment(
|
|
body=deployment, namespace=self.k8s_namespace
|
|
)
|
|
if opts.o.debug:
|
|
print("Deployment created:")
|
|
print(f"{deployment_resp.metadata.namespace} {deployment_resp.metadata.name} \
|
|
{deployment_resp.metadata.generation} {deployment_resp.spec.template.spec.containers[0].image}")
|
|
|
|
def down(self, timeout, volumes):
|
|
# Delete the k8s objects
|
|
# Destroy the kind cluster
|
|
destroy_cluster(self.kind_cluster_name)
|
|
|
|
def ps(self):
|
|
self.connect_api()
|
|
# Call whatever API we need to get the running container list
|
|
ret = self.core_api.list_pod_for_all_namespaces(watch=False)
|
|
if ret.items:
|
|
for i in ret.items:
|
|
print("%s\t%s\t%s" % (i.status.pod_ip, i.metadata.namespace, i.metadata.name))
|
|
ret = self.core_api.list_node(pretty=True, watch=False)
|
|
return []
|
|
|
|
def port(self, service, private_port):
|
|
# Since we handle the port mapping, need to figure out where this comes from
|
|
# Also look into whether it makes sense to get ports for k8s
|
|
pass
|
|
|
|
def execute(self, service_name, command, tty, envs):
|
|
# Call the API to execute a command in a running container
|
|
pass
|
|
|
|
def logs(self, services, tail, follow, stream):
|
|
self.connect_api()
|
|
pods = pods_in_deployment(self.core_api, "test-deployment")
|
|
if len(pods) > 1:
|
|
print("Warning: more than one pod in the deployment")
|
|
k8s_pod_name = pods[0]
|
|
log_data = self.core_api.read_namespaced_pod_log(k8s_pod_name, namespace="default", container="test")
|
|
return log_stream_from_string(log_data)
|
|
|
|
def run(self, image, command, user, volumes, entrypoint=None):
|
|
# We need to figure out how to do this -- check why we're being called first
|
|
pass
|
|
|
|
|
|
class K8sDeployerConfigGenerator(DeployerConfigGenerator):
|
|
config_file_name: str = "kind-config.yml"
|
|
|
|
def __init__(self) -> None:
|
|
super().__init__()
|
|
|
|
def generate(self, deployment_dir: Path):
|
|
# Check the file isn't already there
|
|
# Get the config file contents
|
|
content = generate_kind_config(deployment_dir)
|
|
config_file = deployment_dir.joinpath(self.config_file_name)
|
|
# Write the file
|
|
with open(config_file, "w") as output_file:
|
|
output_file.write(content)
|