mirror of
https://github.com/dw-0/kiauh.git
synced 2025-12-25 16:53:36 +05:00
* feat(obico): add obico extension Signed-off-by: Dominik Willner <th33xitus@gmail.com> * refactor: add obico to moonraker suffix blacklist Signed-off-by: Dominik Willner <th33xitus@gmail.com> * fix: correctly recognize the suffix of the instance Signed-off-by: Dominik Willner <th33xitus@gmail.com> * fix: fix logic of asking for linking Signed-off-by: Dominik Willner <th33xitus@gmail.com> * Squashed 'kiauh/core/submodules/simple_config_parser/' changes from 2698f60..7aa6586 7aa6586 fix: sections can have hyphens in their second word 44cedf5 fix(tests): fix whitespaces in expected output git-subtree-dir: kiauh/core/submodules/simple_config_parser git-subtree-split: 7aa658654eeb08fd53831effbfba4503a61e0eff * refactor: use SimpleConfigParser and finalize the code Signed-off-by: Dominik Willner <th33xitus@gmail.com> * fix: wrong condition in _load_config Signed-off-by: Dominik Willner <th33xitus@gmail.com> * Squashed 'kiauh/core/submodules/simple_config_parser/' changes from 7aa6586..47c353f 47c353f refactor: improve section regex dd904bc test: add more test cases git-subtree-dir: kiauh/core/submodules/simple_config_parser git-subtree-split: 47c353f4e91e6be9605394b174834e1f34c9cfdf * Squashed 'kiauh/core/submodules/simple_config_parser/' changes from 47c353f..3655330 3655330 refactor: use pop() for removing elements from lists and dicts 99733f1 refactor: add empty options dict to _all_options on section parsing git-subtree-dir: kiauh/core/submodules/simple_config_parser git-subtree-split: 3655330d2156e13acffc56fac070ab8716444c85 * refactor: improve config creations and patching Signed-off-by: Dominik Willner <th33xitus@gmail.com> --------- Signed-off-by: Dominik Willner <th33xitus@gmail.com>
387 lines
14 KiB
Python
387 lines
14 KiB
Python
# ======================================================================= #
|
|
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
|
# #
|
|
# This file is part of KIAUH - Klipper Installation And Update Helper #
|
|
# https://github.com/dw-0/kiauh #
|
|
# #
|
|
# This file may be distributed under the terms of the GNU GPLv3 license #
|
|
# ======================================================================= #
|
|
import shutil
|
|
from typing import List
|
|
|
|
from components.klipper.klipper import Klipper
|
|
from components.moonraker.moonraker import Moonraker
|
|
from core.instance_manager.instance_manager import InstanceManager
|
|
from core.submodules.simple_config_parser.src.simple_config_parser.simple_config_parser import (
|
|
SimpleConfigParser,
|
|
)
|
|
from extensions.base_extension import BaseExtension
|
|
from extensions.obico.moonraker_obico import (
|
|
OBICO_CFG_SAMPLE,
|
|
OBICO_DIR,
|
|
OBICO_ENV,
|
|
OBICO_LOG,
|
|
OBICO_MACROS_CFG,
|
|
OBICO_REPO,
|
|
OBICO_UPDATE_CFG,
|
|
OBICO_UPDATE_CFG_SAMPLE,
|
|
MoonrakerObico,
|
|
)
|
|
from utils.common import check_install_dependencies, moonraker_exists
|
|
from utils.config_utils import (
|
|
add_config_section,
|
|
remove_config_section,
|
|
)
|
|
from utils.fs_utils import remove_file
|
|
from utils.git_utils import git_clone_wrapper, git_pull_wrapper
|
|
from utils.input_utils import get_confirm, get_selection_input, get_string_input
|
|
from utils.logger import DialogType, Logger
|
|
from utils.sys_utils import (
|
|
cmd_sysctl_manage,
|
|
create_python_venv,
|
|
install_python_requirements,
|
|
parse_packages_from_file,
|
|
)
|
|
|
|
|
|
# noinspection PyMethodMayBeStatic
|
|
class ObicoExtension(BaseExtension):
|
|
server_url: str
|
|
|
|
def install_extension(self, **kwargs) -> None:
|
|
Logger.print_status("Installing Obico for Klipper ...")
|
|
|
|
# check if moonraker is installed. if not, notify the user and exit
|
|
if not moonraker_exists():
|
|
return
|
|
|
|
# if obico is already installed, ask if the user wants to repair an
|
|
# incomplete installation or link to the obico server
|
|
obico_im = InstanceManager(MoonrakerObico)
|
|
obico_instances: List[MoonrakerObico] = obico_im.instances
|
|
if obico_instances:
|
|
self._print_is_already_installed()
|
|
options = ["l", "L", "r", "R", "b", "B"]
|
|
action = get_selection_input("Perform action", option_list=options)
|
|
if action.lower() == "b":
|
|
Logger.print_info("Exiting Obico for Klipper installation ...")
|
|
return
|
|
elif action.lower() == "l":
|
|
unlinked_instances: List[MoonrakerObico] = [
|
|
obico for obico in obico_instances if not obico.is_linked
|
|
]
|
|
self._link_obico_instances(unlinked_instances)
|
|
return
|
|
else:
|
|
Logger.print_status("Re-Installing Obico for Klipper ...")
|
|
|
|
# let the user confirm installation
|
|
kl_im = InstanceManager(Klipper)
|
|
kl_instances: List[Klipper] = kl_im.instances
|
|
mr_im = InstanceManager(Moonraker)
|
|
mr_instances: List[Moonraker] = mr_im.instances
|
|
self._print_moonraker_instances(mr_instances)
|
|
if not get_confirm(
|
|
"Continue Obico for Klipper installation?",
|
|
default_choice=True,
|
|
allow_go_back=True,
|
|
):
|
|
return
|
|
|
|
try:
|
|
git_clone_wrapper(OBICO_REPO, OBICO_DIR)
|
|
self._install_dependencies()
|
|
|
|
# ask the user for the obico server url
|
|
self._get_server_url()
|
|
|
|
# create obico instances
|
|
for moonraker in mr_instances:
|
|
current_instance = MoonrakerObico(suffix=moonraker.suffix)
|
|
|
|
obico_im.current_instance = current_instance
|
|
obico_im.create_instance()
|
|
obico_im.enable_instance()
|
|
|
|
# create obico config
|
|
self._create_obico_cfg(current_instance, moonraker)
|
|
|
|
# create obico macros
|
|
self._create_obico_macros_cfg(moonraker)
|
|
|
|
# create obico update manager
|
|
self._create_obico_update_manager_cfg(moonraker)
|
|
|
|
obico_im.start_instance()
|
|
|
|
cmd_sysctl_manage("daemon-reload")
|
|
|
|
# add to klippers config
|
|
self._patch_printer_cfg(kl_instances)
|
|
kl_im.restart_all_instance()
|
|
|
|
# add to moonraker update manager
|
|
self._patch_moonraker_conf(mr_instances)
|
|
mr_im.restart_all_instance()
|
|
|
|
# check linking of / ask for linking instances
|
|
self._check_and_opt_link_instances()
|
|
|
|
Logger.print_dialog(
|
|
DialogType.SUCCESS,
|
|
["Obico for Klipper successfully installed!"],
|
|
center_content=True,
|
|
)
|
|
|
|
except Exception as e:
|
|
Logger.print_error(f"Error during Obico for Klipper installation:\n{e}")
|
|
|
|
def update_extension(self, **kwargs) -> None:
|
|
Logger.print_status("Updating Obico for Klipper ...")
|
|
try:
|
|
tb_im = InstanceManager(MoonrakerObico)
|
|
tb_im.stop_all_instance()
|
|
|
|
git_pull_wrapper(OBICO_REPO, OBICO_DIR)
|
|
self._install_dependencies()
|
|
|
|
tb_im.start_all_instance()
|
|
Logger.print_ok("Obico for Klipper successfully updated!")
|
|
|
|
except Exception as e:
|
|
Logger.print_error(f"Error during Obico for Klipper update:\n{e}")
|
|
|
|
def remove_extension(self, **kwargs) -> None:
|
|
Logger.print_status("Removing Obico for Klipper ...")
|
|
kl_im = InstanceManager(Klipper)
|
|
kl_instances: List[Klipper] = kl_im.instances
|
|
mr_im = InstanceManager(Moonraker)
|
|
mr_instances: List[Moonraker] = mr_im.instances
|
|
ob_im = InstanceManager(MoonrakerObico)
|
|
ob_instances: List[MoonrakerObico] = ob_im.instances
|
|
|
|
try:
|
|
self._remove_obico_instances(ob_im, ob_instances)
|
|
self._remove_obico_dir()
|
|
self._remove_obico_env()
|
|
remove_config_section(f"include {OBICO_MACROS_CFG}", kl_instances)
|
|
remove_config_section(f"include {OBICO_UPDATE_CFG}", mr_instances)
|
|
self._delete_obico_logs(ob_instances)
|
|
Logger.print_dialog(
|
|
DialogType.SUCCESS,
|
|
["Obico for Klipper successfully removed!"],
|
|
center_content=True,
|
|
)
|
|
|
|
except Exception as e:
|
|
Logger.print_error(f"Error during Obico for Klipper removal:\n{e}")
|
|
|
|
def _obico_server_url_prompt(self) -> None:
|
|
Logger.print_dialog(
|
|
DialogType.CUSTOM,
|
|
custom_title="Obico Server URL",
|
|
content=[
|
|
"You can use a self-hosted Obico Server or the Obico Cloud. "
|
|
"For more information, please visit:",
|
|
"https://obico.io.",
|
|
"\n\n",
|
|
"For the Obico Cloud, leave it as the default:",
|
|
"https://app.obico.io.",
|
|
"\n\n",
|
|
"For self-hosted server, specify:",
|
|
"http://server_ip:port",
|
|
"For instance, 'http://192.168.0.5:3334'.",
|
|
],
|
|
end="",
|
|
)
|
|
|
|
def _print_moonraker_instances(self, mr_instances) -> None:
|
|
mr_names = [f"● {moonraker.data_dir_name}" for moonraker in mr_instances]
|
|
if len(mr_names) > 1:
|
|
Logger.print_dialog(
|
|
DialogType.INFO,
|
|
[
|
|
"The following Moonraker instances were found:",
|
|
*mr_names,
|
|
"\n\n",
|
|
"The setup will apply the same names to Obico!",
|
|
],
|
|
end="",
|
|
)
|
|
|
|
def _print_is_already_installed(self) -> None:
|
|
Logger.print_dialog(
|
|
DialogType.INFO,
|
|
[
|
|
"Obico is already installed!",
|
|
"It is save to run the installer again to link your "
|
|
"printer or repair any issues.",
|
|
"\n\n",
|
|
"You can perform the following actions:",
|
|
"L) Link printer to the Obico server",
|
|
"R) Repair installation",
|
|
],
|
|
end="",
|
|
)
|
|
|
|
def _get_server_url(self) -> None:
|
|
self._obico_server_url_prompt()
|
|
pattern = r"^(http|https)://[a-zA-Z0-9./?=_%:-]*$"
|
|
self.server_url = get_string_input(
|
|
"Obico Server URL",
|
|
regex=pattern,
|
|
default="https://app.obico.io",
|
|
)
|
|
|
|
def _install_dependencies(self) -> None:
|
|
# install dependencies
|
|
script = OBICO_DIR.joinpath("install.sh")
|
|
package_list = parse_packages_from_file(script)
|
|
check_install_dependencies(package_list)
|
|
|
|
# create virtualenv
|
|
create_python_venv(OBICO_ENV)
|
|
requirements = OBICO_DIR.joinpath("requirements.txt")
|
|
install_python_requirements(OBICO_ENV, requirements)
|
|
|
|
def _create_obico_macros_cfg(self, moonraker) -> None:
|
|
macros_cfg = OBICO_DIR.joinpath(f"include_cfgs/{OBICO_MACROS_CFG}")
|
|
macros_target = moonraker.cfg_dir.joinpath(OBICO_MACROS_CFG)
|
|
if not macros_target.exists():
|
|
shutil.copy(macros_cfg, macros_target)
|
|
else:
|
|
Logger.print_info(
|
|
f"Obico's '{OBICO_MACROS_CFG}' in {moonraker.cfg_dir} already exists! Skipped ..."
|
|
)
|
|
|
|
def _create_obico_update_manager_cfg(self, moonraker) -> None:
|
|
update_cfg = OBICO_DIR.joinpath(OBICO_UPDATE_CFG_SAMPLE)
|
|
update_cfg_target = moonraker.cfg_dir.joinpath(OBICO_UPDATE_CFG)
|
|
if not update_cfg_target.exists():
|
|
shutil.copy(update_cfg, update_cfg_target)
|
|
else:
|
|
Logger.print_info(
|
|
f"Obico's '{OBICO_UPDATE_CFG}' in {moonraker.cfg_dir} already exists! Skipped ..."
|
|
)
|
|
|
|
def _create_obico_cfg(
|
|
self, current_instance: MoonrakerObico, moonraker: Moonraker
|
|
) -> None:
|
|
cfg_template = OBICO_DIR.joinpath(OBICO_CFG_SAMPLE)
|
|
cfg_target_file = current_instance.cfg_file
|
|
|
|
if not cfg_template.exists():
|
|
Logger.print_error(
|
|
f"Obico config template file {cfg_target_file} does not exist!"
|
|
)
|
|
return
|
|
|
|
if not cfg_target_file.exists():
|
|
shutil.copy(cfg_template, cfg_target_file)
|
|
self._patch_obico_cfg(moonraker, current_instance)
|
|
else:
|
|
Logger.print_info(
|
|
f"Obico config in {current_instance.cfg_dir} already exists! Skipped ..."
|
|
)
|
|
|
|
def _patch_obico_cfg(self, moonraker: Moonraker, obico: MoonrakerObico) -> None:
|
|
scp = SimpleConfigParser()
|
|
scp.read(obico.cfg_file)
|
|
scp.set("server", "url", self.server_url)
|
|
scp.set("moonraker", "port", str(moonraker.port))
|
|
scp.set("logging", "path", str(obico.log))
|
|
scp.write(obico.cfg_file)
|
|
|
|
def _patch_printer_cfg(self, klipper: List[Klipper]) -> None:
|
|
add_config_section(section=f"include {OBICO_MACROS_CFG}", instances=klipper)
|
|
|
|
def _patch_moonraker_conf(self, instances: List[Moonraker]) -> None:
|
|
add_config_section(section=f"include {OBICO_UPDATE_CFG}", instances=instances)
|
|
|
|
def _link_obico_instances(self, unlinked_instances):
|
|
for obico in unlinked_instances:
|
|
obico.link()
|
|
|
|
def _check_and_opt_link_instances(self):
|
|
Logger.print_status("Checking link status of Obico instances ...")
|
|
ob_im = InstanceManager(MoonrakerObico)
|
|
ob_instances: List[MoonrakerObico] = ob_im.instances
|
|
unlinked_instances: List[MoonrakerObico] = [
|
|
obico for obico in ob_instances if not obico.is_linked
|
|
]
|
|
if unlinked_instances:
|
|
Logger.print_dialog(
|
|
DialogType.INFO,
|
|
[
|
|
"The Obico instances for the following printers are not "
|
|
"linked to the server:",
|
|
*[f"● {obico.data_dir_name}" for obico in unlinked_instances],
|
|
"\n\n",
|
|
"It will take only 10 seconds to link the printer to the Obico server.",
|
|
"For more information visit:",
|
|
"https://www.obico.io/docs/user-guides/klipper-setup/",
|
|
"\n\n",
|
|
"If you don't want to link the printer now, you can restart the "
|
|
"linking process later by running this installer again.",
|
|
],
|
|
end="",
|
|
)
|
|
if not get_confirm("Do you want to link the printers now?"):
|
|
Logger.print_info("Linking to Obico server skipped ...")
|
|
return
|
|
|
|
self._link_obico_instances(unlinked_instances)
|
|
|
|
def _remove_obico_instances(
|
|
self,
|
|
instance_manager: InstanceManager,
|
|
instance_list: List[MoonrakerObico],
|
|
) -> None:
|
|
if not instance_list:
|
|
Logger.print_info("No Obico instances found. Skipped ...")
|
|
return
|
|
|
|
for instance in instance_list:
|
|
Logger.print_status(
|
|
f"Removing instance {instance.get_service_file_name()} ..."
|
|
)
|
|
instance_manager.current_instance = instance
|
|
instance_manager.stop_instance()
|
|
instance_manager.disable_instance()
|
|
instance_manager.delete_instance()
|
|
|
|
cmd_sysctl_manage("daemon-reload")
|
|
|
|
def _remove_obico_dir(self) -> None:
|
|
if not OBICO_DIR.exists():
|
|
Logger.print_info(f"'{OBICO_DIR}' does not exist. Skipped ...")
|
|
return
|
|
|
|
try:
|
|
shutil.rmtree(OBICO_DIR)
|
|
except OSError as e:
|
|
Logger.print_error(f"Unable to delete '{OBICO_DIR}':\n{e}")
|
|
|
|
def _remove_obico_env(self) -> None:
|
|
if not OBICO_ENV.exists():
|
|
Logger.print_info(f"'{OBICO_ENV}' does not exist. Skipped ...")
|
|
return
|
|
|
|
try:
|
|
shutil.rmtree(OBICO_ENV)
|
|
except OSError as e:
|
|
Logger.print_error(f"Unable to delete '{OBICO_ENV}':\n{e}")
|
|
|
|
def _delete_obico_logs(self, instances: List[MoonrakerObico]) -> None:
|
|
Logger.print_status("Removing Obico logs ...")
|
|
all_logfiles = []
|
|
for instance in instances:
|
|
all_logfiles = list(instance.log_dir.glob(f"{OBICO_LOG}*"))
|
|
if not all_logfiles:
|
|
Logger.print_info("No Obico logs found. Skipped ...")
|
|
return
|
|
|
|
for log in all_logfiles:
|
|
Logger.print_status(f"Remove '{log}'")
|
|
remove_file(log)
|