2023-01-19 21:13:57 +00:00
|
|
|
# Copyright © 2022, 2023 Cerc
|
2022-09-28 16:07:13 +00:00
|
|
|
|
|
|
|
# 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/>.
|
|
|
|
|
2023-01-19 21:13:57 +00:00
|
|
|
import os.path
|
|
|
|
import sys
|
2023-07-05 14:50:56 +00:00
|
|
|
import ruamel.yaml
|
2023-01-19 21:13:57 +00:00
|
|
|
from pathlib import Path
|
|
|
|
|
|
|
|
|
2022-09-28 16:07:13 +00:00
|
|
|
def include_exclude_check(s, include, exclude):
|
2022-10-04 01:37:18 +00:00
|
|
|
if include is None and exclude is None:
|
2022-09-28 16:07:13 +00:00
|
|
|
return True
|
2022-10-04 01:37:18 +00:00
|
|
|
if include is not None:
|
2022-09-28 16:07:13 +00:00
|
|
|
include_list = include.split(",")
|
|
|
|
return s in include_list
|
2022-10-04 01:37:18 +00:00
|
|
|
if exclude is not None:
|
2022-09-28 16:07:13 +00:00
|
|
|
exclude_list = exclude.split(",")
|
|
|
|
return s not in exclude_list
|
2023-01-19 21:13:57 +00:00
|
|
|
|
|
|
|
|
2023-06-27 22:58:41 +00:00
|
|
|
def get_stack_file_path(stack):
|
2023-01-19 21:13:57 +00:00
|
|
|
# In order to be compatible with Python 3.8 we need to use this hack to get the path:
|
|
|
|
# See: https://stackoverflow.com/questions/25389095/python-get-path-of-root-project-structure
|
|
|
|
stack_file_path = Path(__file__).absolute().parent.joinpath("data", "stacks", stack, "stack.yml")
|
2023-06-27 22:58:41 +00:00
|
|
|
return stack_file_path
|
|
|
|
|
|
|
|
|
|
|
|
# Caller can pass either the name of a stack, or a path to a stack file
|
|
|
|
def get_parsed_stack_config(stack):
|
|
|
|
stack_file_path = stack if isinstance(stack, os.PathLike) else get_stack_file_path(stack)
|
2023-01-19 21:13:57 +00:00
|
|
|
try:
|
|
|
|
with stack_file_path:
|
2023-07-05 14:50:56 +00:00
|
|
|
stack_config = get_yaml().load(open(stack_file_path, "r"))
|
2023-01-19 21:13:57 +00:00
|
|
|
return stack_config
|
|
|
|
except FileNotFoundError as error:
|
|
|
|
# We try here to generate a useful diagnostic error
|
|
|
|
# First check if the stack directory is present
|
|
|
|
stack_directory = stack_file_path.parent
|
|
|
|
if os.path.exists(stack_directory):
|
|
|
|
print(f"Error: stack.yml file is missing from stack: {stack}")
|
|
|
|
else:
|
|
|
|
print(f"Error: stack: {stack} does not exist")
|
|
|
|
print(f"Exiting, error: {error}")
|
|
|
|
sys.exit(1)
|
2023-06-27 22:58:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
def get_parsed_deployment_spec(spec_file):
|
|
|
|
spec_file_path = Path(spec_file)
|
|
|
|
try:
|
|
|
|
with spec_file_path:
|
2023-07-05 14:50:56 +00:00
|
|
|
deploy_spec = get_yaml().load(open(spec_file_path, "r"))
|
2023-06-27 22:58:41 +00:00
|
|
|
return deploy_spec
|
|
|
|
except FileNotFoundError as error:
|
|
|
|
# We try here to generate a useful diagnostic error
|
|
|
|
print(f"Error: spec file: {spec_file_path} does not exist")
|
|
|
|
print(f"Exiting, error: {error}")
|
|
|
|
sys.exit(1)
|
|
|
|
|
|
|
|
|
2023-07-05 14:50:56 +00:00
|
|
|
def get_yaml():
|
|
|
|
# See: https://stackoverflow.com/a/45701840/1701505
|
|
|
|
yaml = ruamel.yaml.YAML()
|
|
|
|
yaml.preserve_quotes = True
|
|
|
|
yaml.indent(sequence=3, offset=1)
|
|
|
|
return yaml
|
|
|
|
|
|
|
|
|
2023-06-27 22:58:41 +00:00
|
|
|
# TODO: this is fragile wrt to the subcommand depth
|
|
|
|
# See also: https://github.com/pallets/click/issues/108
|
|
|
|
def global_options(ctx):
|
|
|
|
return ctx.parent.parent.obj
|
|
|
|
|
|
|
|
|
|
|
|
# TODO: hack
|
|
|
|
def global_options2(ctx):
|
|
|
|
return ctx.parent.obj
|