mirror of
https://gitlab.opencode.de/bmi/opendesk/deployment/opendesk.git
synced 2025-12-06 07:21:36 +01:00
fix(helmfile): Add script to ease local development of platform charts.
This commit is contained in:
7
.gitignore
vendored
7
.gitignore
vendored
@@ -14,3 +14,10 @@ helmfile/environments/prod/values.yaml.gotmpl
|
|||||||
|
|
||||||
# Ignore editor backup files
|
# Ignore editor backup files
|
||||||
*~
|
*~
|
||||||
|
|
||||||
|
# Ignore ./log directory and *.log files
|
||||||
|
logs
|
||||||
|
*.log
|
||||||
|
|
||||||
|
# Ignore backup files, e.g. created by the script that eases the local chart development
|
||||||
|
*.bak
|
||||||
|
|||||||
@@ -60,7 +60,9 @@
|
|||||||
"Nordeck",
|
"Nordeck",
|
||||||
"Nubus",
|
"Nubus",
|
||||||
"Souveräne",
|
"Souveräne",
|
||||||
"Arbeitsplatz"
|
"Arbeitsplatz",
|
||||||
|
"commandline",
|
||||||
|
"helmfiles"
|
||||||
],
|
],
|
||||||
"ignoreWords": [],
|
"ignoreWords": [],
|
||||||
"import": []
|
"import": []
|
||||||
|
|||||||
35
dev/README.md
Normal file
35
dev/README.md
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
<!--
|
||||||
|
SPDX-FileCopyrightText: 2024 Zentrum für Digitale Souveränität der Öffentlichen Verwaltung (ZenDiS) GmbH
|
||||||
|
SPDX-License-Identifier: Apache-2.0
|
||||||
|
-->
|
||||||
|
|
||||||
|
<h1>Tools for local development<h1>
|
||||||
|
|
||||||
|
* [charts-local.py](#charts-localpy)
|
||||||
|
* [Commandline parameter](#commandline-parameter)
|
||||||
|
* [`--branch`](#--branch)
|
||||||
|
* [`--revert`](#--revert)
|
||||||
|
|
||||||
|
# charts-local.py
|
||||||
|
|
||||||
|
This script helps you on cloning the platform development Helm charts and referencing them directly in the openDesk
|
||||||
|
Helmfile deployment for comfortable local development and deployment. The charts will be cloned into a directory
|
||||||
|
parallel to the `opendesk` repo that is named after the branch you are working in the `opendesk` repo with.
|
||||||
|
|
||||||
|
The script will create `.bak` copies of the helmfiles that have been touched.
|
||||||
|
|
||||||
|
Run the script with `-h` to get information about the script's parameter on commandline.
|
||||||
|
|
||||||
|
## Commandline parameter
|
||||||
|
|
||||||
|
### `--branch`
|
||||||
|
|
||||||
|
Optional parameter: Defines a branch for the `opendesk` repo to work with. The script will create the branch if it
|
||||||
|
does not exist yet. Otherwise it will switch to defined branch.
|
||||||
|
|
||||||
|
If parameter is omitted the current branch of the `opendesk` repo will be used.
|
||||||
|
|
||||||
|
### `--revert`
|
||||||
|
|
||||||
|
Reverts the changes in the helmfiles pointing to the local Helm charts by copying the backup files created by the
|
||||||
|
scripts itself back to their original location.
|
||||||
176
dev/charts-local.py
Executable file
176
dev/charts-local.py
Executable file
@@ -0,0 +1,176 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
# SPDX-FileCopyrightText: 2024 Zentrum für Digitale Souveränität der Öffentlichen Verwaltung (ZenDiS) GmbH
|
||||||
|
# SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
import os.path
|
||||||
|
import logging
|
||||||
|
import yaml
|
||||||
|
import sys
|
||||||
|
import shutil
|
||||||
|
import re
|
||||||
|
import configargparse
|
||||||
|
|
||||||
|
from pathlib import Path
|
||||||
|
from git import Repo
|
||||||
|
|
||||||
|
p = configargparse.ArgParser()
|
||||||
|
p.add('--branch', env_var='CHART_DEV_BRANCH', help='The branch you want to work with. Will be created by the script if it does not exist yet.')
|
||||||
|
p.add('--git_hostname', env_var='GIT_HOSTNAME', default='git@gitlab.opencode.de', help='Set the hostname for the chart git checkouts.')
|
||||||
|
p.add('--revert', default=False, action='store_true', help='Set this parameter if you want to revert the referencing of the local helm chart checkout paths in the helmfiles.')
|
||||||
|
p.add('--loglevel', env_var='LOGLEVEL', default='DEBUG', help='Set the loglevel: DEBUG, INFO, WARNING, ERROR, CRITICAL-')
|
||||||
|
options = p.parse_args()
|
||||||
|
|
||||||
|
script_path = os.path.dirname(os.path.realpath(__file__))
|
||||||
|
# some static definitions
|
||||||
|
log_path = script_path+'/../logs'
|
||||||
|
charts_yaml = script_path+'/../helmfile/environments/default/charts.yaml'
|
||||||
|
base_repo_path = script_path+'/..'
|
||||||
|
base_helmfile = base_repo_path+'/helmfile_generic.yaml'
|
||||||
|
helmfile_backup_extension = '.bak'
|
||||||
|
|
||||||
|
Path(log_path).mkdir(parents=True, exist_ok=True)
|
||||||
|
|
||||||
|
logFormatter = logging.Formatter("%(asctime)s %(levelname)-5.5s %(message)s")
|
||||||
|
rootLogger = logging.getLogger()
|
||||||
|
rootLogger.setLevel(options.loglevel)
|
||||||
|
|
||||||
|
fileHandler = logging.FileHandler("{0}/{1}.log".format(log_path, os.path.basename(__file__)))
|
||||||
|
fileHandler.setFormatter(logFormatter)
|
||||||
|
rootLogger.addHandler(fileHandler)
|
||||||
|
|
||||||
|
consoleHandler = logging.StreamHandler()
|
||||||
|
consoleHandler.setFormatter(logFormatter)
|
||||||
|
rootLogger.addHandler(consoleHandler)
|
||||||
|
|
||||||
|
logging.debug(f"Working with relative paths from script location: {script_path}")
|
||||||
|
logging.debug(f"Log directory: {log_path}")
|
||||||
|
logging.debug(f"charts.yaml : {charts_yaml}")
|
||||||
|
|
||||||
|
|
||||||
|
def create_or_switch_branch_base_repo():
|
||||||
|
base_repo = Repo(path=base_repo_path)
|
||||||
|
current_branch = base_repo.active_branch.name
|
||||||
|
if not options.branch:
|
||||||
|
branch = current_branch
|
||||||
|
logging.debug(f"No branch specified on commandline, working with current branch: {current_branch}")
|
||||||
|
else:
|
||||||
|
branch = options.branch
|
||||||
|
if branch in base_repo.branches:
|
||||||
|
if branch != current_branch:
|
||||||
|
logging.debug(f"Selected {branch} already exists, switching.")
|
||||||
|
# ToDo: Graceful handle: "Please commit your changes or stash them before you switch branches."
|
||||||
|
base_repo.git.switch(branch)
|
||||||
|
else:
|
||||||
|
logging.debug(f"Already on selected brach {branch}")
|
||||||
|
else:
|
||||||
|
logging.debug(f"Creating branch {branch} and switching")
|
||||||
|
base_repo.git.branch(branch)
|
||||||
|
base_repo.git.switch(branch)
|
||||||
|
return branch
|
||||||
|
|
||||||
|
|
||||||
|
def clone_charts_locally(branch):
|
||||||
|
charts_clone_path = script_path+'/../../'+branch.replace('/', '_')
|
||||||
|
charts_dict = {}
|
||||||
|
remote_dict = {}
|
||||||
|
if os.path.isdir(charts_clone_path):
|
||||||
|
logging.warning(f"Path {charts_clone_path} already exists, will not clone any charts.")
|
||||||
|
else:
|
||||||
|
logging.debug(f"creating directory {charts_clone_path} to clone charts into")
|
||||||
|
Path(charts_clone_path).mkdir(parents=True, exist_ok=True)
|
||||||
|
|
||||||
|
with open(charts_yaml, 'r') as file:
|
||||||
|
charts = yaml.safe_load(file)
|
||||||
|
for chart in charts['charts']:
|
||||||
|
if 'opendesk/components/platform-development/charts' in charts['charts'][chart]['repository']:
|
||||||
|
tag = charts['charts'][chart]['version']
|
||||||
|
logging.debug(f"Working on {chart} / tag {tag}")
|
||||||
|
repository = charts['charts'][chart]['repository']
|
||||||
|
git_url = options.git_hostname+':'+repository
|
||||||
|
chart_repo_path = charts_clone_path+'/'+charts['charts'][chart]['name']
|
||||||
|
if git_url in remote_dict:
|
||||||
|
logging.debug(f"{chart} located at {git_url} is already checked out to {remote_dict[git_url]}")
|
||||||
|
charts_dict[chart] = remote_dict[git_url]
|
||||||
|
else:
|
||||||
|
if os.path.isdir(chart_repo_path):
|
||||||
|
logging.debug(f"Already exists {chart_repo_path} leaving it unmodified")
|
||||||
|
else:
|
||||||
|
logging.debug(f"Cloning into {chart_repo_path}")
|
||||||
|
Repo.clone_from(git_url, chart_repo_path)
|
||||||
|
chart_repo = Repo(path=chart_repo_path)
|
||||||
|
chart_repo.git.checkout('v'+charts['charts'][chart]['version'])
|
||||||
|
charts_dict[chart] = chart_repo_path
|
||||||
|
remote_dict[git_url] = chart_repo_path
|
||||||
|
return charts_dict
|
||||||
|
|
||||||
|
|
||||||
|
def grep_yaml(file):
|
||||||
|
with open(file, 'r') as file:
|
||||||
|
content = ''
|
||||||
|
for line in file.readlines():
|
||||||
|
if not ': {{' in line and not '- {{' in line:
|
||||||
|
content += line
|
||||||
|
return yaml.safe_load(content)
|
||||||
|
|
||||||
|
|
||||||
|
def get_child_helmfiles():
|
||||||
|
child_helmfiles = []
|
||||||
|
root_helmfile = grep_yaml(base_helmfile)
|
||||||
|
for entry in root_helmfile['helmfiles']:
|
||||||
|
child_helmfiles.append(base_repo_path+'/'+entry['path'])
|
||||||
|
return child_helmfiles
|
||||||
|
|
||||||
|
|
||||||
|
def process_the_helmfiles(charts_dict):
|
||||||
|
chart_def_prefix = ' chart: "'
|
||||||
|
name_def_prefix = ' - name: "'
|
||||||
|
child_helmfiles = get_child_helmfiles()
|
||||||
|
for child_helmfile in child_helmfiles:
|
||||||
|
child_helmfile_updated = False
|
||||||
|
output = []
|
||||||
|
with open(child_helmfile, 'r') as file:
|
||||||
|
for line in file:
|
||||||
|
if chart_def_prefix in line:
|
||||||
|
for chart_ident in charts_dict:
|
||||||
|
if '.Values.charts.'+chart_ident+'.name' in line:
|
||||||
|
logging.debug(f"found match with {chart_ident} in {line.strip()}")
|
||||||
|
if name_def_prefix not in line_memory:
|
||||||
|
sys.exit(f"Script requires `name` definition before the actual `chart` definition. Not the case for '{chart_ident}'")
|
||||||
|
else:
|
||||||
|
name = re.search(rf"^{name_def_prefix}(.+)\"", line_memory).group(1)
|
||||||
|
line = chart_def_prefix+charts_dict[chart_ident]+'/charts/'+name+'" # replaced by local-dev script'+"\n"
|
||||||
|
child_helmfile_updated = True
|
||||||
|
break
|
||||||
|
output.append(line)
|
||||||
|
line_memory = line
|
||||||
|
if child_helmfile_updated:
|
||||||
|
child_helmfile_backup = child_helmfile+helmfile_backup_extension
|
||||||
|
logging.debug(f"Updated {child_helmfile}")
|
||||||
|
if os.path.isfile(child_helmfile_backup):
|
||||||
|
logging.debug("backup {child_helmfile_backup} already exists, will not create a new one.")
|
||||||
|
else:
|
||||||
|
logging.debug(f"creating backup {child_helmfile_backup}.")
|
||||||
|
shutil.copy2(child_helmfile, child_helmfile_backup)
|
||||||
|
with open(child_helmfile, 'w') as file:
|
||||||
|
file.writelines(output)
|
||||||
|
|
||||||
|
|
||||||
|
def revert_the_helmfiles():
|
||||||
|
child_helmfiles = get_child_helmfiles()
|
||||||
|
for child_helmfile in child_helmfiles:
|
||||||
|
child_helmfile_backup = child_helmfile+helmfile_backup_extension
|
||||||
|
if os.path.isfile(child_helmfile_backup):
|
||||||
|
logging.debug(f"Reverting {child_helmfile} from backup {child_helmfile_backup}")
|
||||||
|
os.rename(child_helmfile_backup, child_helmfile)
|
||||||
|
else:
|
||||||
|
logging.debug(f"Did not found the backup file {child_helmfile_backup}")
|
||||||
|
|
||||||
|
##
|
||||||
|
## Main program
|
||||||
|
##
|
||||||
|
if options.revert:
|
||||||
|
revert_the_helmfiles()
|
||||||
|
else:
|
||||||
|
branch = create_or_switch_branch_base_repo()
|
||||||
|
charts_dict = clone_charts_locally(branch)
|
||||||
|
process_the_helmfiles(charts_dict)
|
||||||
6
dev/requirements.txt
Normal file
6
dev/requirements.txt
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
# SPDX-FileCopyrightText: 2024 Zentrum für Digitale Souveränität der Öffentlichen Verwaltung (ZenDiS) GmbH
|
||||||
|
# SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
configargparse
|
||||||
|
pyyaml
|
||||||
|
GitPython
|
||||||
Reference in New Issue
Block a user