Thomas E Lackey
1def279d26
All checks were successful
Lint Checks / Run linter (push) Successful in 33s
Publish / Build and publish (push) Successful in 1m7s
Smoke Test / Run basic test suite (push) Successful in 3m51s
Webapp Test / Run webapp test suite (push) Successful in 4m30s
Deploy Test / Run deploy test suite (push) Successful in 4m42s
Fixturenet-Laconicd-Test / Run Laconicd fixturenet and Laconic CLI tests (push) Successful in 12m52s
K8s Deploy Test / Run deploy test suite on kind/k8s (push) Successful in 7m27s
Database Test / Run database hosting test on kind/k8s (push) Successful in 9m35s
Container Registry Test / Run contaier registry hosting test on kind/k8s (push) Successful in 3m54s
External Stack Test / Run external stack test suite (push) Successful in 5m19s
NodePort example: ``` network: ports: caddy: - 1234 - 32020:2020 ``` Replicas example: ``` replicas: 2 ``` This also adds an optimization for k8s where if a directory matching the name of a configmap exists in beneath config/ in the stack, its contents will be copied into the corresponding configmap. For example: ``` # Config files in the stack ❯ ls stack-orchestrator/config/caddyconfig Caddyfile Caddyfile.one-req-per-upstream-example # ConfigMap in the spec ❯ cat foo.yml | grep config ... configmaps: caddyconfig: ./configmaps/caddyconfig # Create the deployment ❯ laconic-so --stack ~/cerc/caddy-ethcache/stack-orchestrator/stacks/caddy-ethcache deploy create --spec-file foo.yml # The files from beneath config/<config_map_name> have been copied to the ConfigMap directory from the spec. ❯ ls deployment-001/configmaps/caddyconfig Caddyfile Caddyfile.one-req-per-upstream-example ``` Reviewed-on: #913 Reviewed-by: David Boreham <dboreham@noreply.git.vdb.to> Co-authored-by: Thomas E Lackey <telackey@bozemanpass.com> Co-committed-by: Thomas E Lackey <telackey@bozemanpass.com>
144 lines
4.2 KiB
Python
144 lines
4.2 KiB
Python
# Copyright © 2022, 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/>.
|
|
|
|
import typing
|
|
import humanfriendly
|
|
|
|
from pathlib import Path
|
|
|
|
from stack_orchestrator.util import get_yaml
|
|
from stack_orchestrator import constants
|
|
|
|
|
|
class ResourceLimits:
|
|
cpus: float = None
|
|
memory: int = None
|
|
storage: int = None
|
|
|
|
def __init__(self, obj=None):
|
|
if obj is None:
|
|
obj = {}
|
|
if "cpus" in obj:
|
|
self.cpus = float(obj["cpus"])
|
|
if "memory" in obj:
|
|
self.memory = humanfriendly.parse_size(obj["memory"])
|
|
if "storage" in obj:
|
|
self.storage = humanfriendly.parse_size(obj["storage"])
|
|
|
|
def __len__(self):
|
|
return len(self.__dict__)
|
|
|
|
def __iter__(self):
|
|
for k in self.__dict__:
|
|
yield k, self.__dict__[k]
|
|
|
|
def __repr__(self):
|
|
return str(self.__dict__)
|
|
|
|
|
|
class Resources:
|
|
limits: ResourceLimits = None
|
|
reservations: ResourceLimits = None
|
|
|
|
def __init__(self, obj=None):
|
|
if obj is None:
|
|
obj = {}
|
|
if "reservations" in obj:
|
|
self.reservations = ResourceLimits(obj["reservations"])
|
|
if "limits" in obj:
|
|
self.limits = ResourceLimits(obj["limits"])
|
|
|
|
def __len__(self):
|
|
return len(self.__dict__)
|
|
|
|
def __iter__(self):
|
|
for k in self.__dict__:
|
|
yield k, self.__dict__[k]
|
|
|
|
def __repr__(self):
|
|
return str(self.__dict__)
|
|
|
|
|
|
class Spec:
|
|
|
|
obj: typing.Any
|
|
file_path: Path
|
|
|
|
def __init__(self, file_path: Path = None, obj=None) -> None:
|
|
if obj is None:
|
|
obj = {}
|
|
self.file_path = file_path
|
|
self.obj = obj
|
|
|
|
def __getitem__(self, item):
|
|
return self.obj[item]
|
|
|
|
def __contains__(self, item):
|
|
return item in self.obj
|
|
|
|
def get(self, item, default=None):
|
|
return self.obj.get(item, default)
|
|
|
|
def init_from_file(self, file_path: Path):
|
|
with file_path:
|
|
self.obj = get_yaml().load(open(file_path, "r"))
|
|
self.file_path = file_path
|
|
|
|
def get_image_registry(self):
|
|
return self.obj.get(constants.image_registry_key)
|
|
|
|
def get_volumes(self):
|
|
return self.obj.get(constants.volumes_key, {})
|
|
|
|
def get_configmaps(self):
|
|
return self.obj.get(constants.configmaps_key, {})
|
|
|
|
def get_container_resources(self):
|
|
return Resources(self.obj.get(constants.resources_key, {}).get("containers", {}))
|
|
|
|
def get_volume_resources(self):
|
|
return Resources(self.obj.get(constants.resources_key, {}).get(constants.volumes_key, {}))
|
|
|
|
def get_http_proxy(self):
|
|
return self.obj.get(constants.network_key, {}).get(constants.http_proxy_key, [])
|
|
|
|
def get_annotations(self):
|
|
return self.obj.get(constants.annotations_key, {})
|
|
|
|
def get_replicas(self):
|
|
return self.obj.get(constants.replicas_key, 1)
|
|
|
|
def get_labels(self):
|
|
return self.obj.get(constants.labels_key, {})
|
|
|
|
def get_privileged(self):
|
|
return "true" == str(self.obj.get(constants.security_key, {}).get("privileged", "false")).lower()
|
|
|
|
def get_capabilities(self):
|
|
return self.obj.get(constants.security_key, {}).get("capabilities", [])
|
|
|
|
def get_deployment_type(self):
|
|
return self.obj.get(constants.deploy_to_key)
|
|
|
|
def is_kubernetes_deployment(self):
|
|
return self.get_deployment_type() in [constants.k8s_kind_deploy_type,
|
|
constants.k8s_deploy_type]
|
|
|
|
def is_kind_deployment(self):
|
|
return self.get_deployment_type() in [constants.k8s_kind_deploy_type]
|
|
|
|
def is_docker_deployment(self):
|
|
return self.get_deployment_type() in [constants.compose_deploy_type]
|