mirror of
https://github.com/dw-0/kiauh.git
synced 2025-12-24 00:03:42 +05:00
Compare commits
97 Commits
v5.1.2
...
03b940e569
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
03b940e569 | ||
|
|
ad56b51e70 | ||
|
|
c6999f1990 | ||
|
|
bc30cf418b | ||
|
|
ee81ee4c0c | ||
|
|
35911604af | ||
|
|
77f1089041 | ||
|
|
7820155094 | ||
|
|
c28d5c28b9 | ||
|
|
cda6d31a7c | ||
|
|
9a657daffd | ||
|
|
85b4b68f16 | ||
|
|
dfbce3b489 | ||
|
|
f3b0e45e39 | ||
|
|
83e5d9c0d5 | ||
|
|
8f44187568 | ||
|
|
625a808484 | ||
|
|
ad0dbf63b8 | ||
|
|
9dedf38079 | ||
|
|
1b4c76d080 | ||
|
|
d20d82aeac | ||
|
|
16a28ffda0 | ||
|
|
a9367cc064 | ||
|
|
b165d88855 | ||
|
|
6c59d58193 | ||
|
|
b4f5c3c1ac | ||
|
|
b69ecbc9b5 | ||
|
|
fc9fa39eee | ||
|
|
142b4498a3 | ||
|
|
012b6c4bb7 | ||
|
|
8aeb01aca0 | ||
|
|
da533fdd67 | ||
|
|
8cb0754296 | ||
|
|
7a6590e86a | ||
|
|
2f0feb317e | ||
|
|
b9479db766 | ||
|
|
14132fc34b | ||
|
|
3d5e83d5ab | ||
|
|
edd5f5c6fd | ||
|
|
8ff0b9d81d | ||
|
|
22e8e314db | ||
|
|
12bd8eb799 | ||
|
|
4915896099 | ||
|
|
cd38970bbd | ||
|
|
b8640f45a6 | ||
|
|
5fb4444f03 | ||
|
|
926ba1acb4 | ||
|
|
c2e7ee98df | ||
|
|
3865266da1 | ||
|
|
b83f642a13 | ||
|
|
30b4414469 | ||
|
|
1178d3c730 | ||
|
|
59d8867c8c | ||
|
|
80a953a587 | ||
|
|
a80f0bb0e8 | ||
|
|
78cefddb2e | ||
|
|
b20613819e | ||
|
|
1836beab42 | ||
|
|
545397f598 | ||
|
|
f709cf84e7 | ||
|
|
f62c10dc8b | ||
|
|
e121ba8a62 | ||
|
|
9a1a66aa64 | ||
|
|
420b193f4b | ||
|
|
de20f0c412 | ||
|
|
57f34b07c6 | ||
|
|
e35e44a76a | ||
|
|
bfb10c742b | ||
|
|
458c89a78a | ||
|
|
6128e35d45 | ||
|
|
279d000bb0 | ||
|
|
a4a3d5eecb | ||
|
|
1392ca9f82 | ||
|
|
47121f6875 | ||
|
|
d0d2404132 | ||
|
|
6ed5395f17 | ||
|
|
be805c169b | ||
|
|
eaf12db27e | ||
|
|
fe8767113b | ||
|
|
2148d95cf4 | ||
|
|
682be48e8d | ||
|
|
68369753fd | ||
|
|
44ed3b6ddf | ||
|
|
e12e578098 | ||
|
|
515a42f098 | ||
|
|
f9ecad0eca | ||
|
|
fb09acf660 | ||
|
|
093da73dd1 | ||
|
|
c9e8c4807e | ||
|
|
09e874214b | ||
|
|
623bd7553b | ||
|
|
1e0c74b549 | ||
|
|
358c666da9 | ||
|
|
84a530be7d | ||
|
|
bfff3019cb | ||
|
|
2a100c2934 | ||
|
|
ce0daa52ae |
@@ -1,15 +0,0 @@
|
|||||||
root = true
|
|
||||||
|
|
||||||
[*]
|
|
||||||
indent_style = space
|
|
||||||
indent_size = 4
|
|
||||||
insert_final_newline = true
|
|
||||||
trim_trailing_whitespace = true
|
|
||||||
charset = utf-8
|
|
||||||
end_of_line = lf
|
|
||||||
|
|
||||||
[*.py]
|
|
||||||
max_line_length = 88
|
|
||||||
|
|
||||||
[*.sh]
|
|
||||||
indent_size = 2
|
|
||||||
6
.gitignore
vendored
6
.gitignore
vendored
@@ -1,11 +1,7 @@
|
|||||||
.idea
|
.idea
|
||||||
.vscode
|
.vscode
|
||||||
|
.idea
|
||||||
.pytest_cache
|
.pytest_cache
|
||||||
.jupyter
|
|
||||||
*.ipynb
|
|
||||||
*.ipynb_checkpoints
|
|
||||||
*.tmp
|
|
||||||
__pycache__
|
|
||||||
.kiauh-env
|
.kiauh-env
|
||||||
*.code-workspace
|
*.code-workspace
|
||||||
*.iml
|
*.iml
|
||||||
|
|||||||
16
README.md
16
README.md
@@ -154,20 +154,18 @@ prompt and confirm by hitting ENTER.
|
|||||||
<tr>
|
<tr>
|
||||||
<th><h3><a href="https://github.com/Clon1998/mobileraker_companion">Mobileraker's Companion</a></h3></th>
|
<th><h3><a href="https://github.com/Clon1998/mobileraker_companion">Mobileraker's Companion</a></h3></th>
|
||||||
<th><h3><a href="https://octoeverywhere.com/?source=kiauh_readme">OctoEverywhere For Klipper</a></h3></th>
|
<th><h3><a href="https://octoeverywhere.com/?source=kiauh_readme">OctoEverywhere For Klipper</a></h3></th>
|
||||||
<th><h3><a href="https://github.com/crysxd/OctoApp-Plugin">OctoApp For Klipper</a></h3></th>
|
|
||||||
<th><h3></h3></th>
|
<th><h3></h3></th>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
<tr>
|
<tr>
|
||||||
<th><a href="https://github.com/Clon1998/mobileraker_companion"><img src="https://raw.githubusercontent.com/Clon1998/mobileraker/master/assets/icon/mr_appicon.png" alt="OctoEverywhere Logo" height="64"></a></th>
|
<th><a href="https://github.com/Clon1998/mobileraker_companion"><img src="https://raw.githubusercontent.com/Clon1998/mobileraker/master/assets/icon/mr_appicon.png" alt="OctoEverywhere Logo" height="64"></th>
|
||||||
<th><a href="https://octoeverywhere.com/?source=kiauh_readme"><img src="https://octoeverywhere.com/img/logo.svg" alt="OctoEverywhere Logo" height="64"></a></th>
|
<th><a href="https://octoeverywhere.com/?source=kiauh_readme"><img src="https://octoeverywhere.com/img/logo.svg" alt="OctoEverywhere Logo" height="64"></a></th>
|
||||||
<th><a href="https://octoapp.eu/?source=kiauh_readme"><img src="https://octoapp.eu/octoapp.webp" alt="OctoApp Logo" height="64"></a></th>
|
<th></th>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
<tr>
|
<tr>
|
||||||
<th>by <a href="https://github.com/Clon1998">Patrick Schmidt</a></th>
|
<th>by <a href="https://github.com/Clon1998">Patrick Schmidt</a></th>
|
||||||
<th>by <a href="https://github.com/QuinnDamerell">Quinn Damerell</a></th>
|
<th>by <a href="https://github.com/QuinnDamerell">Quinn Damerell</a></th>
|
||||||
<th>by <a href="https://github.com/crysxd">Christian Würthner</a></th>
|
|
||||||
<th></th>
|
<th></th>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
@@ -176,16 +174,6 @@ prompt and confirm by hitting ENTER.
|
|||||||
|
|
||||||
<hr>
|
<hr>
|
||||||
|
|
||||||
<h2 align="center">🎖️ Contributors 🎖️</h2>
|
|
||||||
|
|
||||||
<div align="center">
|
|
||||||
<a href="https://github.com/dw-0/kiauh/graphs/contributors">
|
|
||||||
<img src="https://contrib.rocks/image?repo=dw-0/kiauh" alt=""/>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<hr>
|
|
||||||
|
|
||||||
<h2 align="center">✨ Credits ✨</h2>
|
<h2 align="center">✨ Credits ✨</h2>
|
||||||
|
|
||||||
* A big thank you to [lixxbox](https://github.com/lixxbox) for that awesome KIAUH-Logo!
|
* A big thank you to [lixxbox](https://github.com/lixxbox) for that awesome KIAUH-Logo!
|
||||||
|
|||||||
@@ -1,18 +0,0 @@
|
|||||||
[kiauh]
|
|
||||||
backup_before_update: False
|
|
||||||
|
|
||||||
[klipper]
|
|
||||||
repo_url: https://github.com/Klipper3d/klipper
|
|
||||||
branch: master
|
|
||||||
|
|
||||||
[moonraker]
|
|
||||||
repo_url: https://github.com/Arksine/moonraker
|
|
||||||
branch: master
|
|
||||||
|
|
||||||
[mainsail]
|
|
||||||
port: 80
|
|
||||||
unstable_releases: False
|
|
||||||
|
|
||||||
[fluidd]
|
|
||||||
port: 80
|
|
||||||
unstable_releases: False
|
|
||||||
@@ -2,47 +2,6 @@
|
|||||||
|
|
||||||
This document covers possible important changes to KIAUH.
|
This document covers possible important changes to KIAUH.
|
||||||
|
|
||||||
### 2024-08-31 (v6.0.0-alpha.1)
|
|
||||||
Long time no see, but here we are again!
|
|
||||||
A lot has happened in the background, but now it is time to take it out into the wild.
|
|
||||||
|
|
||||||
#### KIAUH has now reached version 6! Well, at least in an alpha state...
|
|
||||||
|
|
||||||
The project has seen a complete rewrite of the script from scratch in Python.
|
|
||||||
It requires Python 3.8 or newer to run. Because this update is still in an alpha state, bugs may or will occur.
|
|
||||||
During startup, you will be asked if you want to start the new version 6 or the old version 5.
|
|
||||||
As long as version 6 is in a pre-release state, version 5 will still be available. If there are any critical issues
|
|
||||||
with the new version that were overlooked, you can always switch back to the old version.
|
|
||||||
|
|
||||||
In case you selected not to get asked about which version to start (option 3 or 4 in the startup dialog) and you want to
|
|
||||||
revert that decision, you will find a line called `version_to_launch=` within the `.kiauh.ini` file in your home directory.
|
|
||||||
Just delete that line, save the file and restart KIAUH. KIAUH will then ask you again which version you want to start.
|
|
||||||
|
|
||||||
Here is a list of the most important changes to KIAUH in regard to version 6:
|
|
||||||
- The majority of features available in KIAUH v5 are still available; they just got migrated from Bash to Python.
|
|
||||||
- It is now possible to add new/remove instances to/from existing multi-instance installations of Klipper and Moonraker
|
|
||||||
- KIAUH now has an Extension-System. This allows contributors to add new installers to KIAUH without having to modify the main script.
|
|
||||||
- You will now find some of the features that were previously available in the Installer-Menu in the Extensions-Menu.
|
|
||||||
- The current extensions are:
|
|
||||||
- G-Code Shell Command (previously found in the Advanced-Menu)
|
|
||||||
- Mainsail Theme Installer (previously found in the Advanced-Menu)
|
|
||||||
- Klipper-Backup (new in v6!)
|
|
||||||
- Moonraker Telegram Bot (previously found in the Installer-Menu)
|
|
||||||
- PrettyGCode for Klipper (previously found in the Installer-Menu)
|
|
||||||
- Obico for Klipper (previously found in the Installer-Menu)
|
|
||||||
- The following additional extensions are planned, but not yet available:
|
|
||||||
- Spoolman (available in v5 in the Installer-Menu)
|
|
||||||
- OctoApp (available in v5 in the Installer-Menu)
|
|
||||||
- KIAUH has its own config file now
|
|
||||||
- The file has some default values for the currently supported options
|
|
||||||
- There might be more options in the future
|
|
||||||
- It is located in KIAUH's root directory and is called `default.kiauh.cfg`
|
|
||||||
- DO NOT EDIT the default file directly, instead make a copy of it and call it `kiauh.cfg`
|
|
||||||
- Settings changed via the Advanced-Menu will be written to the `kiauh.cfg`
|
|
||||||
- Support for OctoPrint was removed
|
|
||||||
|
|
||||||
Feel free to give version 6 a try and report any bugs or issues you encounter! Every feedback is appreciated.
|
|
||||||
|
|
||||||
### 2023-06-17
|
### 2023-06-17
|
||||||
KIAUH has now added support for installing Mobileraker's companion!
|
KIAUH has now added support for installing Mobileraker's companion!
|
||||||
Mobileraker is a free and Open Source Android and iOS App for Klipper, utilizing the Moonraker API, allowing you
|
Mobileraker is a free and Open Source Android and iOS App for Klipper, utilizing the Moonraker API, allowing you
|
||||||
|
|||||||
16
kiauh.cfg.example
Normal file
16
kiauh.cfg.example
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
[kiauh]
|
||||||
|
backup_before_update: False
|
||||||
|
|
||||||
|
[klipper]
|
||||||
|
repository_url: https://github.com/Klipper3d/klipper
|
||||||
|
branch: master
|
||||||
|
method: https
|
||||||
|
|
||||||
|
[moonraker]
|
||||||
|
repository_url: https://github.com/Arksine/moonraker
|
||||||
|
branch: master
|
||||||
|
method: https
|
||||||
|
|
||||||
|
[mainsail]
|
||||||
|
default_port: 80
|
||||||
|
unstable_releases: False
|
||||||
237
kiauh.sh
237
kiauh.sh
@@ -12,166 +12,97 @@
|
|||||||
set -e
|
set -e
|
||||||
clear
|
clear
|
||||||
|
|
||||||
# make sure we have the correct permissions while running the script
|
function main() {
|
||||||
umask 022
|
local python_command
|
||||||
|
|
||||||
### sourcing all additional scripts
|
|
||||||
KIAUH_SRCDIR="$(dirname -- "$(readlink -f "${BASH_SOURCE[0]}")")"
|
|
||||||
for script in "${KIAUH_SRCDIR}/scripts/"*.sh; do . "${script}"; done
|
|
||||||
for script in "${KIAUH_SRCDIR}/scripts/ui/"*.sh; do . "${script}"; done
|
|
||||||
|
|
||||||
#===================================================#
|
|
||||||
#=================== UPDATE KIAUH ==================#
|
|
||||||
#===================================================#
|
|
||||||
|
|
||||||
function update_kiauh() {
|
|
||||||
status_msg "Updating KIAUH ..."
|
|
||||||
|
|
||||||
cd "${KIAUH_SRCDIR}"
|
|
||||||
git reset --hard && git pull
|
|
||||||
|
|
||||||
ok_msg "Update complete! Please restart KIAUH."
|
|
||||||
exit 0
|
|
||||||
}
|
|
||||||
|
|
||||||
#===================================================#
|
|
||||||
#=================== KIAUH STATUS ==================#
|
|
||||||
#===================================================#
|
|
||||||
|
|
||||||
function kiauh_update_avail() {
|
|
||||||
[[ ! -d "${KIAUH_SRCDIR}/.git" ]] && return
|
|
||||||
local origin head
|
|
||||||
|
|
||||||
cd "${KIAUH_SRCDIR}"
|
|
||||||
|
|
||||||
### abort if not on master branch
|
|
||||||
! git branch -a | grep -q "\* master" && return
|
|
||||||
|
|
||||||
### compare commit hash
|
|
||||||
git fetch -q
|
|
||||||
origin=$(git rev-parse --short=8 origin/master)
|
|
||||||
head=$(git rev-parse --short=8 HEAD)
|
|
||||||
|
|
||||||
if [[ ${origin} != "${head}" ]]; then
|
|
||||||
echo "true"
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
function save_startup_version() {
|
|
||||||
local launch_version
|
|
||||||
|
|
||||||
echo "${1}"
|
|
||||||
|
|
||||||
sed -i "/^version_to_launch=/d" "${INI_FILE}"
|
|
||||||
sed -i '$a'"version_to_launch=${1}" "${INI_FILE}"
|
|
||||||
}
|
|
||||||
|
|
||||||
function kiauh_update_dialog() {
|
|
||||||
[[ ! $(kiauh_update_avail) == "true" ]] && return
|
|
||||||
top_border
|
|
||||||
echo -e "|${green} New KIAUH update available! ${white}|"
|
|
||||||
hr
|
|
||||||
echo -e "|${green} View Changelog: https://git.io/JnmlX ${white}|"
|
|
||||||
blank_line
|
|
||||||
echo -e "|${yellow} It is recommended to keep KIAUH up to date. Updates ${white}|"
|
|
||||||
echo -e "|${yellow} usually contain bugfixes, important changes or new ${white}|"
|
|
||||||
echo -e "|${yellow} features. Please consider updating! ${white}|"
|
|
||||||
bottom_border
|
|
||||||
|
|
||||||
local yn
|
|
||||||
read -p "${cyan}###### Do you want to update now? (Y/n):${white} " yn
|
|
||||||
while true; do
|
|
||||||
case "${yn}" in
|
|
||||||
Y|y|Yes|yes|"")
|
|
||||||
do_action "update_kiauh"
|
|
||||||
break;;
|
|
||||||
N|n|No|no)
|
|
||||||
break;;
|
|
||||||
*)
|
|
||||||
deny_action "kiauh_update_dialog";;
|
|
||||||
esac
|
|
||||||
done
|
|
||||||
}
|
|
||||||
|
|
||||||
function launch_kiauh_v5() {
|
|
||||||
main_menu
|
|
||||||
}
|
|
||||||
|
|
||||||
function launch_kiauh_v6() {
|
|
||||||
local entrypoint
|
local entrypoint
|
||||||
|
|
||||||
if ! command -v python3 &>/dev/null || [[ $(python3 -V | cut -d " " -f2 | cut -d "." -f2) -lt 8 ]]; then
|
if command -v python3 &>/dev/null; then
|
||||||
echo "Python 3.8 or higher is not installed!"
|
python_command="python3"
|
||||||
echo "Please install Python 3.8 or higher and try again."
|
elif command -v python &>/dev/null; then
|
||||||
|
python_command="python"
|
||||||
|
else
|
||||||
|
echo "Python is not installed. Please install Python and try again."
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
entrypoint=$(dirname "$(readlink -f "${BASH_SOURCE[0]}")")
|
entrypoint=$(dirname "$(readlink -f "${BASH_SOURCE[0]}")")
|
||||||
|
|
||||||
export PYTHONPATH="${entrypoint}"
|
${python_command} "${entrypoint}/kiauh.py"
|
||||||
|
|
||||||
clear
|
|
||||||
python3 "${entrypoint}/kiauh.py"
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function main() {
|
|
||||||
read_kiauh_ini "${FUNCNAME[0]}"
|
|
||||||
|
|
||||||
if [[ ${version_to_launch} -eq 5 ]]; then
|
|
||||||
launch_kiauh_v5
|
|
||||||
elif [[ ${version_to_launch} -eq 6 ]]; then
|
|
||||||
launch_kiauh_v6
|
|
||||||
else
|
|
||||||
top_border
|
|
||||||
echo -e "| ${green}KIAUH v6.0.0-alpha1 is available now!${white} |"
|
|
||||||
hr
|
|
||||||
echo -e "| View Changelog: ${magenta}https://git.io/JnmlX${white} |"
|
|
||||||
blank_line
|
|
||||||
echo -e "| KIAUH v6 was completely rewritten from the ground up. |"
|
|
||||||
echo -e "| It's based on Python 3.8 and has many improvements. |"
|
|
||||||
blank_line
|
|
||||||
echo -e "| ${yellow}NOTE: Version 6 is still in alpha, so bugs may occur!${white} |"
|
|
||||||
echo -e "| ${yellow}Yet, your feedback and bug reports are very much${white} |"
|
|
||||||
echo -e "| ${yellow}appreciated and will help finalize the release.${white} |"
|
|
||||||
hr
|
|
||||||
echo -e "| Would you like to try out KIAUH v6? |"
|
|
||||||
echo -e "| 1) Yes |"
|
|
||||||
echo -e "| 2) No |"
|
|
||||||
echo -e "| 3) Yes, remember my choice for next time |"
|
|
||||||
echo -e "| 4) No, remember my choice for next time |"
|
|
||||||
quit_footer
|
|
||||||
while true; do
|
|
||||||
read -p "${cyan}###### Select action:${white} " -e input
|
|
||||||
case "${input}" in
|
|
||||||
1)
|
|
||||||
launch_kiauh_v6
|
|
||||||
break;;
|
|
||||||
2)
|
|
||||||
launch_kiauh_v5
|
|
||||||
break;;
|
|
||||||
3)
|
|
||||||
save_startup_version 6
|
|
||||||
launch_kiauh_v6
|
|
||||||
break;;
|
|
||||||
4)
|
|
||||||
save_startup_version 5
|
|
||||||
launch_kiauh_v5
|
|
||||||
break;;
|
|
||||||
Q|q)
|
|
||||||
echo -e "${green}###### Happy printing! ######${white}"; echo
|
|
||||||
exit 0;;
|
|
||||||
*)
|
|
||||||
error_msg "Invalid Input!\n";;
|
|
||||||
esac
|
|
||||||
done && input=""
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
check_if_ratos
|
|
||||||
check_euid
|
|
||||||
init_logfile
|
|
||||||
set_globals
|
|
||||||
kiauh_update_dialog
|
|
||||||
read_kiauh_ini
|
|
||||||
init_ini
|
|
||||||
main
|
main
|
||||||
|
|
||||||
|
#### sourcing all additional scripts
|
||||||
|
#KIAUH_SRCDIR="$(dirname -- "$(readlink -f "${BASH_SOURCE[0]}")")"
|
||||||
|
#for script in "${KIAUH_SRCDIR}/scripts/"*.sh; do . "${script}"; done
|
||||||
|
#for script in "${KIAUH_SRCDIR}/scripts/ui/"*.sh; do . "${script}"; done
|
||||||
|
#
|
||||||
|
##===================================================#
|
||||||
|
##=================== UPDATE KIAUH ==================#
|
||||||
|
##===================================================#
|
||||||
|
#
|
||||||
|
#function update_kiauh() {
|
||||||
|
# status_msg "Updating KIAUH ..."
|
||||||
|
#
|
||||||
|
# cd "${KIAUH_SRCDIR}"
|
||||||
|
# git reset --hard && git pull
|
||||||
|
#
|
||||||
|
# ok_msg "Update complete! Please restart KIAUH."
|
||||||
|
# exit 0
|
||||||
|
#}
|
||||||
|
#
|
||||||
|
##===================================================#
|
||||||
|
##=================== KIAUH STATUS ==================#
|
||||||
|
##===================================================#
|
||||||
|
#
|
||||||
|
#function kiauh_update_avail() {
|
||||||
|
# [[ ! -d "${KIAUH_SRCDIR}/.git" ]] && return
|
||||||
|
# local origin head
|
||||||
|
#
|
||||||
|
# cd "${KIAUH_SRCDIR}"
|
||||||
|
#
|
||||||
|
# ### abort if not on master branch
|
||||||
|
# ! git branch -a | grep -q "\* master" && return
|
||||||
|
#
|
||||||
|
# ### compare commit hash
|
||||||
|
# git fetch -q
|
||||||
|
# origin=$(git rev-parse --short=8 origin/master)
|
||||||
|
# head=$(git rev-parse --short=8 HEAD)
|
||||||
|
#
|
||||||
|
# if [[ ${origin} != "${head}" ]]; then
|
||||||
|
# echo "true"
|
||||||
|
# fi
|
||||||
|
#}
|
||||||
|
#
|
||||||
|
#function kiauh_update_dialog() {
|
||||||
|
# [[ ! $(kiauh_update_avail) == "true" ]] && return
|
||||||
|
# top_border
|
||||||
|
# echo -e "|${green} New KIAUH update available! ${white}|"
|
||||||
|
# hr
|
||||||
|
# echo -e "|${green} View Changelog: https://git.io/JnmlX ${white}|"
|
||||||
|
# blank_line
|
||||||
|
# echo -e "|${yellow} It is recommended to keep KIAUH up to date. Updates ${white}|"
|
||||||
|
# echo -e "|${yellow} usually contain bugfixes, important changes or new ${white}|"
|
||||||
|
# echo -e "|${yellow} features. Please consider updating! ${white}|"
|
||||||
|
# bottom_border
|
||||||
|
#
|
||||||
|
# local yn
|
||||||
|
# read -p "${cyan}###### Do you want to update now? (Y/n):${white} " yn
|
||||||
|
# while true; do
|
||||||
|
# case "${yn}" in
|
||||||
|
# Y|y|Yes|yes|"")
|
||||||
|
# do_action "update_kiauh"
|
||||||
|
# break;;
|
||||||
|
# N|n|No|no)
|
||||||
|
# break;;
|
||||||
|
# *)
|
||||||
|
# deny_action "kiauh_update_dialog";;
|
||||||
|
# esac
|
||||||
|
# done
|
||||||
|
#}
|
||||||
|
#
|
||||||
|
#check_euid
|
||||||
|
#init_logfile
|
||||||
|
#set_globals
|
||||||
|
#kiauh_update_dialog
|
||||||
|
#main_menu
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
# ======================================================================= #
|
# ======================================================================= #
|
||||||
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
||||||
# #
|
# #
|
||||||
@@ -7,9 +9,8 @@
|
|||||||
# This file may be distributed under the terms of the GNU GPLv3 license #
|
# This file may be distributed under the terms of the GNU GPLv3 license #
|
||||||
# ======================================================================= #
|
# ======================================================================= #
|
||||||
|
|
||||||
import sys
|
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
|
|
||||||
PROJECT_ROOT = Path(__file__).resolve().parent.parent
|
APPLICATION_ROOT = Path(__file__).resolve().parent.parent
|
||||||
APPLICATION_ROOT = Path(__file__).resolve().parent
|
KIAUH_CFG = APPLICATION_ROOT.joinpath("kiauh.cfg")
|
||||||
sys.path.append(str(APPLICATION_ROOT))
|
KIAUH_BACKUP_DIR = Path.home().joinpath("kiauh-backups")
|
||||||
|
|||||||
@@ -1,30 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
|
|
||||||
from pathlib import Path
|
|
||||||
|
|
||||||
from core.backup_manager import BACKUP_ROOT_DIR
|
|
||||||
from core.constants import SYSTEMD
|
|
||||||
|
|
||||||
# repo
|
|
||||||
CROWSNEST_REPO = "https://github.com/mainsail-crew/crowsnest.git"
|
|
||||||
|
|
||||||
# names
|
|
||||||
CROWSNEST_SERVICE_NAME = "crowsnest.service"
|
|
||||||
|
|
||||||
# directories
|
|
||||||
CROWSNEST_DIR = Path.home().joinpath("crowsnest")
|
|
||||||
CROWSNEST_BACKUP_DIR = BACKUP_ROOT_DIR.joinpath("crowsnest-backups")
|
|
||||||
|
|
||||||
# files
|
|
||||||
CROWSNEST_MULTI_CONFIG = CROWSNEST_DIR.joinpath("tools/.config")
|
|
||||||
CROWSNEST_INSTALL_SCRIPT = CROWSNEST_DIR.joinpath("tools/install.sh")
|
|
||||||
CROWSNEST_BIN_FILE = Path("/usr/local/bin/crowsnest")
|
|
||||||
CROWSNEST_LOGROTATE_FILE = Path("/etc/logrotate.d/crowsnest")
|
|
||||||
CROWSNEST_SERVICE_FILE = SYSTEMD.joinpath(CROWSNEST_SERVICE_NAME)
|
|
||||||
@@ -1,178 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import shutil
|
|
||||||
import time
|
|
||||||
from pathlib import Path
|
|
||||||
from subprocess import CalledProcessError, run
|
|
||||||
from typing import List
|
|
||||||
|
|
||||||
from components.crowsnest import (
|
|
||||||
CROWSNEST_BACKUP_DIR,
|
|
||||||
CROWSNEST_BIN_FILE,
|
|
||||||
CROWSNEST_DIR,
|
|
||||||
CROWSNEST_INSTALL_SCRIPT,
|
|
||||||
CROWSNEST_LOGROTATE_FILE,
|
|
||||||
CROWSNEST_MULTI_CONFIG,
|
|
||||||
CROWSNEST_REPO,
|
|
||||||
CROWSNEST_SERVICE_FILE,
|
|
||||||
CROWSNEST_SERVICE_NAME,
|
|
||||||
)
|
|
||||||
from components.klipper.klipper import Klipper
|
|
||||||
from core.backup_manager.backup_manager import BackupManager
|
|
||||||
from core.constants import CURRENT_USER
|
|
||||||
from core.logger import DialogType, Logger
|
|
||||||
from core.settings.kiauh_settings import KiauhSettings
|
|
||||||
from core.types import ComponentStatus
|
|
||||||
from utils.common import (
|
|
||||||
check_install_dependencies,
|
|
||||||
get_install_status,
|
|
||||||
)
|
|
||||||
from utils.git_utils import (
|
|
||||||
git_clone_wrapper,
|
|
||||||
git_pull_wrapper,
|
|
||||||
)
|
|
||||||
from utils.input_utils import get_confirm
|
|
||||||
from utils.instance_utils import get_instances
|
|
||||||
from utils.sys_utils import (
|
|
||||||
cmd_sysctl_service,
|
|
||||||
parse_packages_from_file,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def install_crowsnest() -> None:
|
|
||||||
# Step 1: Clone crowsnest repo
|
|
||||||
git_clone_wrapper(CROWSNEST_REPO, CROWSNEST_DIR, "master")
|
|
||||||
|
|
||||||
# Step 2: Install dependencies
|
|
||||||
check_install_dependencies({"make"})
|
|
||||||
|
|
||||||
# Step 3: Check for Multi Instance
|
|
||||||
instances: List[Klipper] = get_instances(Klipper)
|
|
||||||
|
|
||||||
if len(instances) > 1:
|
|
||||||
print_multi_instance_warning(instances)
|
|
||||||
|
|
||||||
if not get_confirm("Do you want to continue with the installation?"):
|
|
||||||
Logger.print_info("Crowsnest installation aborted!")
|
|
||||||
return
|
|
||||||
|
|
||||||
Logger.print_status("Launching crowsnest's install configurator ...")
|
|
||||||
time.sleep(3)
|
|
||||||
configure_multi_instance()
|
|
||||||
|
|
||||||
# Step 4: Launch crowsnest installer
|
|
||||||
Logger.print_status("Launching crowsnest installer ...")
|
|
||||||
Logger.print_info("Installer will prompt you for sudo password!")
|
|
||||||
try:
|
|
||||||
run(
|
|
||||||
f"sudo make install BASE_USER={CURRENT_USER}",
|
|
||||||
cwd=CROWSNEST_DIR,
|
|
||||||
shell=True,
|
|
||||||
check=True,
|
|
||||||
)
|
|
||||||
except CalledProcessError as e:
|
|
||||||
Logger.print_error(f"Something went wrong! Please try again...\n{e}")
|
|
||||||
return
|
|
||||||
|
|
||||||
|
|
||||||
def print_multi_instance_warning(instances: List[Klipper]) -> None:
|
|
||||||
Logger.print_dialog(
|
|
||||||
DialogType.WARNING,
|
|
||||||
[
|
|
||||||
"Multi instance install detected!",
|
|
||||||
"\n\n",
|
|
||||||
"Crowsnest is NOT designed to support multi instances. A workaround "
|
|
||||||
"for this is to choose the most used instance as a 'master' and use "
|
|
||||||
"this instance to set up your 'crowsnest.conf' and steering it's service.",
|
|
||||||
"\n\n",
|
|
||||||
"The following instances were found:",
|
|
||||||
*[f"● {instance.data_dir.name}" for instance in instances],
|
|
||||||
],
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def configure_multi_instance() -> None:
|
|
||||||
try:
|
|
||||||
run(
|
|
||||||
"make config",
|
|
||||||
cwd=CROWSNEST_DIR,
|
|
||||||
shell=True,
|
|
||||||
check=True,
|
|
||||||
)
|
|
||||||
except CalledProcessError as e:
|
|
||||||
Logger.print_error(f"Something went wrong! Please try again...\n{e}")
|
|
||||||
if CROWSNEST_MULTI_CONFIG.exists():
|
|
||||||
Path.unlink(CROWSNEST_MULTI_CONFIG)
|
|
||||||
return
|
|
||||||
|
|
||||||
if not CROWSNEST_MULTI_CONFIG.exists():
|
|
||||||
Logger.print_error("Generating .config failed, installation aborted")
|
|
||||||
|
|
||||||
|
|
||||||
def update_crowsnest() -> None:
|
|
||||||
try:
|
|
||||||
cmd_sysctl_service(CROWSNEST_SERVICE_NAME, "stop")
|
|
||||||
|
|
||||||
if not CROWSNEST_DIR.exists():
|
|
||||||
git_clone_wrapper(CROWSNEST_REPO, CROWSNEST_DIR, "master")
|
|
||||||
else:
|
|
||||||
Logger.print_status("Updating Crowsnest ...")
|
|
||||||
|
|
||||||
settings = KiauhSettings()
|
|
||||||
if settings.kiauh.backup_before_update:
|
|
||||||
bm = BackupManager()
|
|
||||||
bm.backup_directory(
|
|
||||||
CROWSNEST_DIR.name,
|
|
||||||
source=CROWSNEST_DIR,
|
|
||||||
target=CROWSNEST_BACKUP_DIR,
|
|
||||||
)
|
|
||||||
|
|
||||||
git_pull_wrapper(CROWSNEST_REPO, CROWSNEST_DIR)
|
|
||||||
|
|
||||||
deps = parse_packages_from_file(CROWSNEST_INSTALL_SCRIPT)
|
|
||||||
check_install_dependencies({*deps})
|
|
||||||
|
|
||||||
cmd_sysctl_service(CROWSNEST_SERVICE_NAME, "restart")
|
|
||||||
|
|
||||||
Logger.print_ok("Crowsnest updated successfully.", end="\n\n")
|
|
||||||
except CalledProcessError as e:
|
|
||||||
Logger.print_error(f"Something went wrong! Please try again...\n{e}")
|
|
||||||
return
|
|
||||||
|
|
||||||
|
|
||||||
def get_crowsnest_status() -> ComponentStatus:
|
|
||||||
files = [
|
|
||||||
CROWSNEST_BIN_FILE,
|
|
||||||
CROWSNEST_LOGROTATE_FILE,
|
|
||||||
CROWSNEST_SERVICE_FILE,
|
|
||||||
]
|
|
||||||
return get_install_status(CROWSNEST_DIR, files=files)
|
|
||||||
|
|
||||||
|
|
||||||
def remove_crowsnest() -> None:
|
|
||||||
if not CROWSNEST_DIR.exists():
|
|
||||||
Logger.print_info("Crowsnest does not seem to be installed! Skipping ...")
|
|
||||||
return
|
|
||||||
|
|
||||||
try:
|
|
||||||
run(
|
|
||||||
"make uninstall",
|
|
||||||
cwd=CROWSNEST_DIR,
|
|
||||||
shell=True,
|
|
||||||
check=True,
|
|
||||||
)
|
|
||||||
except CalledProcessError as e:
|
|
||||||
Logger.print_error(f"Something went wrong! Please try again...\n{e}")
|
|
||||||
return
|
|
||||||
|
|
||||||
Logger.print_status("Removing crowsnest directory ...")
|
|
||||||
shutil.rmtree(CROWSNEST_DIR)
|
|
||||||
Logger.print_ok("Directory removed!")
|
|
||||||
@@ -1,36 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
|
|
||||||
from pathlib import Path
|
|
||||||
|
|
||||||
from core.backup_manager import BACKUP_ROOT_DIR
|
|
||||||
|
|
||||||
MODULE_PATH = Path(__file__).resolve().parent
|
|
||||||
|
|
||||||
# names
|
|
||||||
KLIPPER_LOG_NAME = "klippy.log"
|
|
||||||
KLIPPER_CFG_NAME = "printer.cfg"
|
|
||||||
KLIPPER_SERIAL_NAME = "klippy.serial"
|
|
||||||
KLIPPER_UDS_NAME = "klippy.sock"
|
|
||||||
KLIPPER_ENV_FILE_NAME = "klipper.env"
|
|
||||||
KLIPPER_SERVICE_NAME = "klipper.service"
|
|
||||||
|
|
||||||
# directories
|
|
||||||
KLIPPER_DIR = Path.home().joinpath("klipper")
|
|
||||||
KLIPPER_ENV_DIR = Path.home().joinpath("klippy-env")
|
|
||||||
KLIPPER_BACKUP_DIR = BACKUP_ROOT_DIR.joinpath("klipper-backups")
|
|
||||||
|
|
||||||
# files
|
|
||||||
KLIPPER_REQ_FILE = KLIPPER_DIR.joinpath("scripts/klippy-requirements.txt")
|
|
||||||
KLIPPER_INSTALL_SCRIPT = KLIPPER_DIR.joinpath("scripts/install-ubuntu-22.04.sh")
|
|
||||||
KLIPPER_SERVICE_TEMPLATE = MODULE_PATH.joinpath(f"assets/{KLIPPER_SERVICE_NAME}")
|
|
||||||
KLIPPER_ENV_FILE_TEMPLATE = MODULE_PATH.joinpath(f"assets/{KLIPPER_ENV_FILE_NAME}")
|
|
||||||
|
|
||||||
|
|
||||||
EXIT_KLIPPER_SETUP = "Exiting Klipper setup ..."
|
|
||||||
@@ -1,142 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
from dataclasses import dataclass, field
|
|
||||||
from pathlib import Path
|
|
||||||
from subprocess import CalledProcessError
|
|
||||||
|
|
||||||
from components.klipper import (
|
|
||||||
KLIPPER_CFG_NAME,
|
|
||||||
KLIPPER_DIR,
|
|
||||||
KLIPPER_ENV_DIR,
|
|
||||||
KLIPPER_ENV_FILE_NAME,
|
|
||||||
KLIPPER_ENV_FILE_TEMPLATE,
|
|
||||||
KLIPPER_LOG_NAME,
|
|
||||||
KLIPPER_SERIAL_NAME,
|
|
||||||
KLIPPER_SERVICE_TEMPLATE,
|
|
||||||
KLIPPER_UDS_NAME,
|
|
||||||
)
|
|
||||||
from core.constants import CURRENT_USER
|
|
||||||
from core.instance_manager.base_instance import BaseInstance
|
|
||||||
from core.logger import Logger
|
|
||||||
from utils.fs_utils import create_folders, get_data_dir
|
|
||||||
from utils.sys_utils import get_service_file_path
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyMethodMayBeStatic
|
|
||||||
@dataclass(repr=True)
|
|
||||||
class Klipper:
|
|
||||||
suffix: str
|
|
||||||
base: BaseInstance = field(init=False, repr=False)
|
|
||||||
service_file_path: Path = field(init=False)
|
|
||||||
log_file_name: str = KLIPPER_LOG_NAME
|
|
||||||
klipper_dir: Path = KLIPPER_DIR
|
|
||||||
env_dir: Path = KLIPPER_ENV_DIR
|
|
||||||
data_dir: Path = field(init=False)
|
|
||||||
cfg_file: Path = field(init=False)
|
|
||||||
env_file: Path = field(init=False)
|
|
||||||
serial: Path = field(init=False)
|
|
||||||
uds: Path = field(init=False)
|
|
||||||
|
|
||||||
def __post_init__(self):
|
|
||||||
self.base: BaseInstance = BaseInstance(Klipper, self.suffix)
|
|
||||||
self.base.log_file_name = self.log_file_name
|
|
||||||
|
|
||||||
self.service_file_path: Path = get_service_file_path(Klipper, self.suffix)
|
|
||||||
self.data_dir: Path = get_data_dir(Klipper, self.suffix)
|
|
||||||
self.cfg_file: Path = self.base.cfg_dir.joinpath(KLIPPER_CFG_NAME)
|
|
||||||
self.env_file: Path = self.base.sysd_dir.joinpath(KLIPPER_ENV_FILE_NAME)
|
|
||||||
self.serial: Path = self.base.comms_dir.joinpath(KLIPPER_SERIAL_NAME)
|
|
||||||
self.uds: Path = self.base.comms_dir.joinpath(KLIPPER_UDS_NAME)
|
|
||||||
|
|
||||||
def create(self) -> None:
|
|
||||||
from utils.sys_utils import create_env_file, create_service_file
|
|
||||||
|
|
||||||
Logger.print_status("Creating new Klipper Instance ...")
|
|
||||||
|
|
||||||
try:
|
|
||||||
create_folders(self.base.base_folders)
|
|
||||||
|
|
||||||
create_service_file(
|
|
||||||
name=self.service_file_path.name,
|
|
||||||
content=self._prep_service_file_content(),
|
|
||||||
)
|
|
||||||
|
|
||||||
create_env_file(
|
|
||||||
path=self.base.sysd_dir.joinpath(KLIPPER_ENV_FILE_NAME),
|
|
||||||
content=self._prep_env_file_content(),
|
|
||||||
)
|
|
||||||
|
|
||||||
except CalledProcessError as e:
|
|
||||||
Logger.print_error(f"Error creating instance: {e}")
|
|
||||||
raise
|
|
||||||
except OSError as e:
|
|
||||||
Logger.print_error(f"Error creating env file: {e}")
|
|
||||||
raise
|
|
||||||
|
|
||||||
def _prep_service_file_content(self) -> str:
|
|
||||||
template = KLIPPER_SERVICE_TEMPLATE
|
|
||||||
|
|
||||||
try:
|
|
||||||
with open(template, "r") as template_file:
|
|
||||||
template_content = template_file.read()
|
|
||||||
except FileNotFoundError:
|
|
||||||
Logger.print_error(f"Unable to open {template} - File not found")
|
|
||||||
raise
|
|
||||||
|
|
||||||
service_content = template_content.replace(
|
|
||||||
"%USER%",
|
|
||||||
CURRENT_USER,
|
|
||||||
)
|
|
||||||
service_content = service_content.replace(
|
|
||||||
"%KLIPPER_DIR%",
|
|
||||||
self.klipper_dir.as_posix(),
|
|
||||||
)
|
|
||||||
service_content = service_content.replace(
|
|
||||||
"%ENV%",
|
|
||||||
self.env_dir.as_posix(),
|
|
||||||
)
|
|
||||||
service_content = service_content.replace(
|
|
||||||
"%ENV_FILE%",
|
|
||||||
self.base.sysd_dir.joinpath(KLIPPER_ENV_FILE_NAME).as_posix(),
|
|
||||||
)
|
|
||||||
return service_content
|
|
||||||
|
|
||||||
def _prep_env_file_content(self) -> str:
|
|
||||||
template = KLIPPER_ENV_FILE_TEMPLATE
|
|
||||||
|
|
||||||
try:
|
|
||||||
with open(template, "r") as env_file:
|
|
||||||
env_template_file_content = env_file.read()
|
|
||||||
except FileNotFoundError:
|
|
||||||
Logger.print_error(f"Unable to open {template} - File not found")
|
|
||||||
raise
|
|
||||||
|
|
||||||
env_file_content = env_template_file_content.replace(
|
|
||||||
"%KLIPPER_DIR%", self.klipper_dir.as_posix()
|
|
||||||
)
|
|
||||||
env_file_content = env_file_content.replace(
|
|
||||||
"%CFG%",
|
|
||||||
f"{self.base.cfg_dir}/{KLIPPER_CFG_NAME}",
|
|
||||||
)
|
|
||||||
env_file_content = env_file_content.replace(
|
|
||||||
"%SERIAL%",
|
|
||||||
self.serial.as_posix() if self.serial else "",
|
|
||||||
)
|
|
||||||
env_file_content = env_file_content.replace(
|
|
||||||
"%LOG%",
|
|
||||||
self.base.log_dir.joinpath(self.log_file_name).as_posix(),
|
|
||||||
)
|
|
||||||
env_file_content = env_file_content.replace(
|
|
||||||
"%UDS%",
|
|
||||||
self.uds.as_posix() if self.uds else "",
|
|
||||||
)
|
|
||||||
|
|
||||||
return env_file_content
|
|
||||||
@@ -1,114 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 textwrap
|
|
||||||
from enum import Enum, unique
|
|
||||||
from typing import List
|
|
||||||
|
|
||||||
from core.constants import (
|
|
||||||
COLOR_CYAN,
|
|
||||||
COLOR_GREEN,
|
|
||||||
COLOR_YELLOW,
|
|
||||||
RESET_FORMAT,
|
|
||||||
)
|
|
||||||
from core.menus.base_menu import print_back_footer
|
|
||||||
from utils.instance_type import InstanceType
|
|
||||||
|
|
||||||
|
|
||||||
@unique
|
|
||||||
class DisplayType(Enum):
|
|
||||||
SERVICE_NAME = "SERVICE_NAME"
|
|
||||||
PRINTER_NAME = "PRINTER_NAME"
|
|
||||||
|
|
||||||
|
|
||||||
def print_instance_overview(
|
|
||||||
instances: List[InstanceType],
|
|
||||||
display_type: DisplayType = DisplayType.SERVICE_NAME,
|
|
||||||
show_headline=True,
|
|
||||||
show_index=False,
|
|
||||||
start_index=0,
|
|
||||||
show_select_all=False,
|
|
||||||
) -> None:
|
|
||||||
dialog = "╔═══════════════════════════════════════════════════════╗\n"
|
|
||||||
if show_headline:
|
|
||||||
d_type = (
|
|
||||||
"Klipper instances"
|
|
||||||
if display_type is DisplayType.SERVICE_NAME
|
|
||||||
else "printer directories"
|
|
||||||
)
|
|
||||||
headline = f"{COLOR_GREEN}The following {d_type} were found:{RESET_FORMAT}"
|
|
||||||
dialog += f"║{headline:^64}║\n"
|
|
||||||
dialog += "╟───────────────────────────────────────────────────────╢\n"
|
|
||||||
|
|
||||||
if show_select_all:
|
|
||||||
select_all = f"{COLOR_YELLOW}a) Select all{RESET_FORMAT}"
|
|
||||||
dialog += f"║ {select_all:<63}║\n"
|
|
||||||
dialog += "║ ║\n"
|
|
||||||
|
|
||||||
for i, s in enumerate(instances):
|
|
||||||
if display_type is DisplayType.SERVICE_NAME:
|
|
||||||
name = s.service_file_path.stem
|
|
||||||
else:
|
|
||||||
name = s.data_dir
|
|
||||||
line = f"{COLOR_CYAN}{f'{i + start_index})' if show_index else '●'} {name}{RESET_FORMAT}"
|
|
||||||
dialog += f"║ {line:<63}║\n"
|
|
||||||
dialog += "╟───────────────────────────────────────────────────────╢\n"
|
|
||||||
|
|
||||||
print(dialog, end="")
|
|
||||||
print_back_footer()
|
|
||||||
|
|
||||||
|
|
||||||
def print_select_instance_count_dialog() -> None:
|
|
||||||
line1 = f"{COLOR_YELLOW}WARNING:{RESET_FORMAT}"
|
|
||||||
line2 = f"{COLOR_YELLOW}Setting up too many instances may crash your system.{RESET_FORMAT}"
|
|
||||||
dialog = textwrap.dedent(
|
|
||||||
f"""
|
|
||||||
╔═══════════════════════════════════════════════════════╗
|
|
||||||
║ Please select the number of Klipper instances to set ║
|
|
||||||
║ up. The number of Klipper instances will determine ║
|
|
||||||
║ the amount of printers you can run from this host. ║
|
|
||||||
║ ║
|
|
||||||
║ {line1:<63}║
|
|
||||||
║ {line2:<63}║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
"""
|
|
||||||
)[1:]
|
|
||||||
|
|
||||||
print(dialog, end="")
|
|
||||||
print_back_footer()
|
|
||||||
|
|
||||||
|
|
||||||
def print_select_custom_name_dialog() -> None:
|
|
||||||
line1 = f"{COLOR_YELLOW}INFO:{RESET_FORMAT}"
|
|
||||||
line2 = f"{COLOR_YELLOW}Only alphanumeric characters are allowed!{RESET_FORMAT}"
|
|
||||||
dialog = textwrap.dedent(
|
|
||||||
f"""
|
|
||||||
╔═══════════════════════════════════════════════════════╗
|
|
||||||
║ Do you want to assign a custom name to each instance? ║
|
|
||||||
║ ║
|
|
||||||
║ Assigning a custom name will create a Klipper service ║
|
|
||||||
║ and a printer directory with the chosen name. ║
|
|
||||||
║ ║
|
|
||||||
║ Example for custom name 'kiauh': ║
|
|
||||||
║ ● Klipper service: klipper-kiauh.service ║
|
|
||||||
║ ● Printer directory: printer_kiauh_data ║
|
|
||||||
║ ║
|
|
||||||
║ If skipped, each instance will get an index assigned ║
|
|
||||||
║ in ascending order, starting at '1' in case of a new ║
|
|
||||||
║ installation. Otherwise, the index will be derived ║
|
|
||||||
║ from amount of already existing instances. ║
|
|
||||||
║ ║
|
|
||||||
║ {line1:<63}║
|
|
||||||
║ {line2:<63}║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
"""
|
|
||||||
)[1:]
|
|
||||||
|
|
||||||
print(dialog, end="")
|
|
||||||
print_back_footer()
|
|
||||||
@@ -1,95 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
from typing import List
|
|
||||||
|
|
||||||
from components.klipper import KLIPPER_DIR, KLIPPER_ENV_DIR
|
|
||||||
from components.klipper.klipper import Klipper
|
|
||||||
from components.klipper.klipper_dialogs import print_instance_overview
|
|
||||||
from core.instance_manager.instance_manager import InstanceManager
|
|
||||||
from core.logger import Logger
|
|
||||||
from utils.fs_utils import run_remove_routines
|
|
||||||
from utils.input_utils import get_selection_input
|
|
||||||
from utils.instance_utils import get_instances
|
|
||||||
from utils.sys_utils import unit_file_exists
|
|
||||||
|
|
||||||
|
|
||||||
def run_klipper_removal(
|
|
||||||
remove_service: bool,
|
|
||||||
remove_dir: bool,
|
|
||||||
remove_env: bool,
|
|
||||||
) -> None:
|
|
||||||
klipper_instances: List[Klipper] = get_instances(Klipper)
|
|
||||||
|
|
||||||
if remove_service:
|
|
||||||
Logger.print_status("Removing Klipper instances ...")
|
|
||||||
if klipper_instances:
|
|
||||||
instances_to_remove = select_instances_to_remove(klipper_instances)
|
|
||||||
remove_instances(instances_to_remove)
|
|
||||||
else:
|
|
||||||
Logger.print_info("No Klipper Services installed! Skipped ...")
|
|
||||||
|
|
||||||
if (remove_dir or remove_env) and unit_file_exists("klipper", suffix="service"):
|
|
||||||
Logger.print_info("There are still other Klipper services installed:")
|
|
||||||
Logger.print_info(f"● '{KLIPPER_DIR}' was not removed.", prefix=False)
|
|
||||||
Logger.print_info(f"● '{KLIPPER_ENV_DIR}' was not removed.", prefix=False)
|
|
||||||
else:
|
|
||||||
if remove_dir:
|
|
||||||
Logger.print_status("Removing Klipper local repository ...")
|
|
||||||
run_remove_routines(KLIPPER_DIR)
|
|
||||||
if remove_env:
|
|
||||||
Logger.print_status("Removing Klipper Python environment ...")
|
|
||||||
run_remove_routines(KLIPPER_ENV_DIR)
|
|
||||||
|
|
||||||
|
|
||||||
def select_instances_to_remove(instances: List[Klipper]) -> List[Klipper] | None:
|
|
||||||
start_index = 1
|
|
||||||
options = [str(i + start_index) for i in range(len(instances))]
|
|
||||||
options.extend(["a", "b"])
|
|
||||||
instance_map = {options[i]: instances[i] for i in range(len(instances))}
|
|
||||||
|
|
||||||
print_instance_overview(
|
|
||||||
instances,
|
|
||||||
start_index=start_index,
|
|
||||||
show_index=True,
|
|
||||||
show_select_all=True,
|
|
||||||
)
|
|
||||||
selection = get_selection_input("Select Klipper instance to remove", options)
|
|
||||||
|
|
||||||
instances_to_remove = []
|
|
||||||
if selection == "b":
|
|
||||||
return None
|
|
||||||
elif selection == "a":
|
|
||||||
instances_to_remove.extend(instances)
|
|
||||||
else:
|
|
||||||
instances_to_remove.append(instance_map[selection])
|
|
||||||
|
|
||||||
return instances_to_remove
|
|
||||||
|
|
||||||
|
|
||||||
def remove_instances(
|
|
||||||
instance_list: List[Klipper] | None,
|
|
||||||
) -> None:
|
|
||||||
if not instance_list:
|
|
||||||
return
|
|
||||||
|
|
||||||
for instance in instance_list:
|
|
||||||
Logger.print_status(f"Removing instance {instance.service_file_path.stem} ...")
|
|
||||||
InstanceManager.remove(instance)
|
|
||||||
delete_klipper_env_file(instance)
|
|
||||||
|
|
||||||
|
|
||||||
def delete_klipper_env_file(instance: Klipper):
|
|
||||||
Logger.print_status(f"Remove '{instance.env_file}'")
|
|
||||||
if not instance.env_file.exists():
|
|
||||||
msg = f"Env file in {instance.base.sysd_dir} not found. Skipped ..."
|
|
||||||
Logger.print_info(msg)
|
|
||||||
return
|
|
||||||
run_remove_routines(instance.env_file)
|
|
||||||
@@ -1,239 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
from pathlib import Path
|
|
||||||
from typing import Dict, List, Tuple
|
|
||||||
|
|
||||||
from components.klipper import (
|
|
||||||
EXIT_KLIPPER_SETUP,
|
|
||||||
KLIPPER_DIR,
|
|
||||||
KLIPPER_ENV_DIR,
|
|
||||||
KLIPPER_INSTALL_SCRIPT,
|
|
||||||
KLIPPER_REQ_FILE,
|
|
||||||
)
|
|
||||||
from components.klipper.klipper import Klipper
|
|
||||||
from components.klipper.klipper_dialogs import (
|
|
||||||
print_select_custom_name_dialog,
|
|
||||||
)
|
|
||||||
from components.klipper.klipper_utils import (
|
|
||||||
assign_custom_name,
|
|
||||||
backup_klipper_dir,
|
|
||||||
check_user_groups,
|
|
||||||
create_example_printer_cfg,
|
|
||||||
get_install_count,
|
|
||||||
handle_disruptive_system_packages,
|
|
||||||
)
|
|
||||||
from components.moonraker.moonraker import Moonraker
|
|
||||||
from components.webui_client.client_utils import (
|
|
||||||
get_existing_clients,
|
|
||||||
)
|
|
||||||
from core.instance_manager.instance_manager import InstanceManager
|
|
||||||
from core.logger import DialogType, Logger
|
|
||||||
from core.settings.kiauh_settings import KiauhSettings
|
|
||||||
from utils.common import check_install_dependencies
|
|
||||||
from utils.git_utils import git_clone_wrapper, git_pull_wrapper
|
|
||||||
from utils.input_utils import get_confirm
|
|
||||||
from utils.instance_utils import get_instances
|
|
||||||
from utils.sys_utils import (
|
|
||||||
cmd_sysctl_manage,
|
|
||||||
cmd_sysctl_service,
|
|
||||||
create_python_venv,
|
|
||||||
install_python_requirements,
|
|
||||||
parse_packages_from_file,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def install_klipper() -> None:
|
|
||||||
Logger.print_status("Installing Klipper ...")
|
|
||||||
|
|
||||||
klipper_list: List[Klipper] = get_instances(Klipper)
|
|
||||||
moonraker_list: List[Moonraker] = get_instances(Moonraker)
|
|
||||||
match_moonraker: bool = False
|
|
||||||
|
|
||||||
# if there are more moonraker instances than klipper instances, ask the user to
|
|
||||||
# match the klipper instance count to the count of moonraker instances with the same suffix
|
|
||||||
if len(moonraker_list) > len(klipper_list):
|
|
||||||
is_confirmed = display_moonraker_info(moonraker_list)
|
|
||||||
if not is_confirmed:
|
|
||||||
Logger.print_status(EXIT_KLIPPER_SETUP)
|
|
||||||
return
|
|
||||||
match_moonraker = True
|
|
||||||
|
|
||||||
install_count, name_dict = get_install_count_and_name_dict(
|
|
||||||
klipper_list, moonraker_list
|
|
||||||
)
|
|
||||||
|
|
||||||
if install_count == 0:
|
|
||||||
Logger.print_status(EXIT_KLIPPER_SETUP)
|
|
||||||
return
|
|
||||||
|
|
||||||
is_multi_install = install_count > 1 or (len(name_dict) >= 1 and install_count >= 1)
|
|
||||||
if not name_dict and install_count == 1:
|
|
||||||
name_dict = {0: ""}
|
|
||||||
elif is_multi_install and not match_moonraker:
|
|
||||||
custom_names = use_custom_names_or_go_back()
|
|
||||||
if custom_names is None:
|
|
||||||
Logger.print_status(EXIT_KLIPPER_SETUP)
|
|
||||||
return
|
|
||||||
|
|
||||||
handle_instance_names(install_count, name_dict, custom_names)
|
|
||||||
|
|
||||||
create_example_cfg = get_confirm("Create example printer.cfg?")
|
|
||||||
# run the actual installation
|
|
||||||
try:
|
|
||||||
run_klipper_setup(klipper_list, name_dict, create_example_cfg)
|
|
||||||
except Exception as e:
|
|
||||||
Logger.print_error(e)
|
|
||||||
Logger.print_error("Klipper installation failed!")
|
|
||||||
return
|
|
||||||
|
|
||||||
|
|
||||||
def run_klipper_setup(
|
|
||||||
klipper_list: List[Klipper], name_dict: Dict[int, str], create_example_cfg: bool
|
|
||||||
) -> None:
|
|
||||||
if not klipper_list:
|
|
||||||
setup_klipper_prerequesites()
|
|
||||||
|
|
||||||
for i in name_dict:
|
|
||||||
# skip this iteration if there is already an instance with the name
|
|
||||||
if name_dict[i] in [n.suffix for n in klipper_list]:
|
|
||||||
continue
|
|
||||||
|
|
||||||
instance = Klipper(suffix=name_dict[i])
|
|
||||||
instance.create()
|
|
||||||
cmd_sysctl_service(instance.service_file_path.name, "enable")
|
|
||||||
|
|
||||||
if create_example_cfg:
|
|
||||||
# if a client-config is installed, include it in the new example cfg
|
|
||||||
clients = get_existing_clients()
|
|
||||||
create_example_printer_cfg(instance, clients)
|
|
||||||
|
|
||||||
cmd_sysctl_service(instance.service_file_path.name, "start")
|
|
||||||
|
|
||||||
cmd_sysctl_manage("daemon-reload")
|
|
||||||
|
|
||||||
# step 4: check/handle conflicting packages/services
|
|
||||||
handle_disruptive_system_packages()
|
|
||||||
|
|
||||||
# step 5: check for required group membership
|
|
||||||
check_user_groups()
|
|
||||||
|
|
||||||
|
|
||||||
def handle_instance_names(
|
|
||||||
install_count: int, name_dict: Dict[int, str], custom_names: bool
|
|
||||||
) -> None:
|
|
||||||
for i in range(install_count): # 3
|
|
||||||
key: int = len(name_dict.keys()) + 1
|
|
||||||
if custom_names:
|
|
||||||
assign_custom_name(key, name_dict)
|
|
||||||
else:
|
|
||||||
name_dict[key] = str(len(name_dict) + 1)
|
|
||||||
|
|
||||||
|
|
||||||
def get_install_count_and_name_dict(
|
|
||||||
klipper_list: List[Klipper], moonraker_list: List[Moonraker]
|
|
||||||
) -> Tuple[int, Dict[int, str]]:
|
|
||||||
install_count: int | None
|
|
||||||
if len(moonraker_list) > len(klipper_list):
|
|
||||||
install_count = len(moonraker_list)
|
|
||||||
name_dict = {i: moonraker.suffix for i, moonraker in enumerate(moonraker_list)}
|
|
||||||
else:
|
|
||||||
install_count = get_install_count()
|
|
||||||
name_dict = {i: klipper.suffix for i, klipper in enumerate(klipper_list)}
|
|
||||||
|
|
||||||
if install_count is None:
|
|
||||||
Logger.print_status(EXIT_KLIPPER_SETUP)
|
|
||||||
return 0, {}
|
|
||||||
|
|
||||||
return install_count, name_dict
|
|
||||||
|
|
||||||
|
|
||||||
def setup_klipper_prerequesites() -> None:
|
|
||||||
settings = KiauhSettings()
|
|
||||||
repo = settings.klipper.repo_url
|
|
||||||
branch = settings.klipper.branch
|
|
||||||
|
|
||||||
git_clone_wrapper(repo, KLIPPER_DIR, branch)
|
|
||||||
|
|
||||||
# install klipper dependencies and create python virtualenv
|
|
||||||
try:
|
|
||||||
install_klipper_packages()
|
|
||||||
if create_python_venv(KLIPPER_ENV_DIR):
|
|
||||||
install_python_requirements(KLIPPER_ENV_DIR, KLIPPER_REQ_FILE)
|
|
||||||
except Exception:
|
|
||||||
Logger.print_error("Error during installation of Klipper requirements!")
|
|
||||||
raise
|
|
||||||
|
|
||||||
|
|
||||||
def install_klipper_packages() -> None:
|
|
||||||
script = KLIPPER_INSTALL_SCRIPT
|
|
||||||
packages = parse_packages_from_file(script)
|
|
||||||
|
|
||||||
# Add dbus requirement for DietPi distro
|
|
||||||
if Path("/boot/dietpi/.version").exists():
|
|
||||||
packages.append("dbus")
|
|
||||||
|
|
||||||
check_install_dependencies({*packages})
|
|
||||||
|
|
||||||
|
|
||||||
def update_klipper() -> None:
|
|
||||||
Logger.print_dialog(
|
|
||||||
DialogType.WARNING,
|
|
||||||
[
|
|
||||||
"Do NOT continue if there are ongoing prints running!",
|
|
||||||
"All Klipper instances will be restarted during the update process and "
|
|
||||||
"ongoing prints WILL FAIL.",
|
|
||||||
],
|
|
||||||
)
|
|
||||||
|
|
||||||
if not get_confirm("Update Klipper now?"):
|
|
||||||
return
|
|
||||||
|
|
||||||
settings = KiauhSettings()
|
|
||||||
if settings.kiauh.backup_before_update:
|
|
||||||
backup_klipper_dir()
|
|
||||||
|
|
||||||
instances = get_instances(Klipper)
|
|
||||||
InstanceManager.stop_all(instances)
|
|
||||||
|
|
||||||
git_pull_wrapper(repo=settings.klipper.repo_url, target_dir=KLIPPER_DIR)
|
|
||||||
|
|
||||||
# install possible new system packages
|
|
||||||
install_klipper_packages()
|
|
||||||
# install possible new python dependencies
|
|
||||||
install_python_requirements(KLIPPER_ENV_DIR, KLIPPER_REQ_FILE)
|
|
||||||
|
|
||||||
InstanceManager.start_all(instances)
|
|
||||||
|
|
||||||
|
|
||||||
def use_custom_names_or_go_back() -> bool | None:
|
|
||||||
print_select_custom_name_dialog()
|
|
||||||
_input: bool | None = get_confirm(
|
|
||||||
"Assign custom names?",
|
|
||||||
False,
|
|
||||||
allow_go_back=True,
|
|
||||||
)
|
|
||||||
return _input
|
|
||||||
|
|
||||||
|
|
||||||
def display_moonraker_info(moonraker_list: List[Moonraker]) -> bool:
|
|
||||||
# todo: only show the klipper instances that are not already installed
|
|
||||||
Logger.print_dialog(
|
|
||||||
DialogType.INFO,
|
|
||||||
[
|
|
||||||
"Existing Moonraker instances detected:",
|
|
||||||
*[f"● {m.service_file_path.stem}" for m in moonraker_list],
|
|
||||||
"\n\n",
|
|
||||||
"The following Klipper instances will be installed:",
|
|
||||||
*[f"● klipper-{m.suffix}" for m in moonraker_list],
|
|
||||||
],
|
|
||||||
)
|
|
||||||
_input: bool = get_confirm("Proceed with installation?")
|
|
||||||
return _input
|
|
||||||
@@ -1,196 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import grp
|
|
||||||
import os
|
|
||||||
import shutil
|
|
||||||
from subprocess import CalledProcessError, run
|
|
||||||
from typing import Dict, List
|
|
||||||
|
|
||||||
from components.klipper import (
|
|
||||||
KLIPPER_BACKUP_DIR,
|
|
||||||
KLIPPER_DIR,
|
|
||||||
KLIPPER_ENV_DIR,
|
|
||||||
MODULE_PATH,
|
|
||||||
)
|
|
||||||
from components.klipper.klipper import Klipper
|
|
||||||
from components.klipper.klipper_dialogs import (
|
|
||||||
print_instance_overview,
|
|
||||||
print_select_instance_count_dialog,
|
|
||||||
)
|
|
||||||
from components.webui_client.base_data import BaseWebClient
|
|
||||||
from components.webui_client.client_config.client_config_setup import (
|
|
||||||
create_client_config_symlink,
|
|
||||||
)
|
|
||||||
from core.backup_manager.backup_manager import BackupManager
|
|
||||||
from core.constants import CURRENT_USER
|
|
||||||
from core.instance_manager.base_instance import SUFFIX_BLACKLIST
|
|
||||||
from core.logger import DialogType, Logger
|
|
||||||
from core.submodules.simple_config_parser.src.simple_config_parser.simple_config_parser import (
|
|
||||||
SimpleConfigParser,
|
|
||||||
)
|
|
||||||
from core.types import ComponentStatus
|
|
||||||
from utils.common import get_install_status
|
|
||||||
from utils.input_utils import get_confirm, get_number_input, get_string_input
|
|
||||||
from utils.instance_utils import get_instances
|
|
||||||
from utils.sys_utils import cmd_sysctl_service
|
|
||||||
|
|
||||||
|
|
||||||
def get_klipper_status() -> ComponentStatus:
|
|
||||||
return get_install_status(KLIPPER_DIR, KLIPPER_ENV_DIR, Klipper)
|
|
||||||
|
|
||||||
|
|
||||||
def add_to_existing() -> bool | None:
|
|
||||||
kl_instances: List[Klipper] = get_instances(Klipper)
|
|
||||||
print_instance_overview(kl_instances)
|
|
||||||
_input: bool | None = get_confirm("Add new instances?", allow_go_back=True)
|
|
||||||
return _input
|
|
||||||
|
|
||||||
|
|
||||||
def get_install_count() -> int | None:
|
|
||||||
"""
|
|
||||||
Print a dialog for selecting the amount of Klipper instances
|
|
||||||
to set up with an option to navigate back. Returns None if the
|
|
||||||
user selected to go back, otherwise an integer greater or equal than 1 |
|
|
||||||
:return: Integer >= 1 or None
|
|
||||||
"""
|
|
||||||
kl_instances = get_instances(Klipper)
|
|
||||||
print_select_instance_count_dialog()
|
|
||||||
question = (
|
|
||||||
f"Number of"
|
|
||||||
f"{' additional' if len(kl_instances) > 0 else ''} "
|
|
||||||
f"Klipper instances to set up"
|
|
||||||
)
|
|
||||||
_input: int | None = get_number_input(question, 1, default=1, allow_go_back=True)
|
|
||||||
return _input
|
|
||||||
|
|
||||||
|
|
||||||
def assign_custom_name(key: int, name_dict: Dict[int, str]) -> None:
|
|
||||||
existing_names = []
|
|
||||||
existing_names.extend(SUFFIX_BLACKLIST)
|
|
||||||
existing_names.extend(name_dict[n] for n in name_dict)
|
|
||||||
pattern = r"^[a-zA-Z0-9]+$"
|
|
||||||
|
|
||||||
question = f"Enter name for instance {key}"
|
|
||||||
name_dict[key] = get_string_input(question, exclude=existing_names, regex=pattern)
|
|
||||||
|
|
||||||
|
|
||||||
def check_user_groups() -> None:
|
|
||||||
user_groups = [grp.getgrgid(gid).gr_name for gid in os.getgroups()]
|
|
||||||
missing_groups = [g for g in ["tty", "dialout"] if g not in user_groups]
|
|
||||||
|
|
||||||
if not missing_groups:
|
|
||||||
return
|
|
||||||
|
|
||||||
Logger.print_dialog(
|
|
||||||
DialogType.ATTENTION,
|
|
||||||
[
|
|
||||||
"Your current user is not in group:",
|
|
||||||
*[f"● {g}" for g in missing_groups],
|
|
||||||
"\n\n",
|
|
||||||
"It is possible that you won't be able to successfully connect and/or "
|
|
||||||
"flash the controller board without your user being a member of that "
|
|
||||||
"group. If you want to add the current user to the group(s) listed above, "
|
|
||||||
"answer with 'Y'. Else skip with 'n'.",
|
|
||||||
"\n\n",
|
|
||||||
"INFO:",
|
|
||||||
"Relog required for group assignments to take effect!",
|
|
||||||
],
|
|
||||||
)
|
|
||||||
|
|
||||||
if not get_confirm(f"Add user '{CURRENT_USER}' to group(s) now?"):
|
|
||||||
log = "Skipped adding user to required groups. You might encounter issues."
|
|
||||||
Logger.warn(log)
|
|
||||||
return
|
|
||||||
|
|
||||||
try:
|
|
||||||
for group in missing_groups:
|
|
||||||
Logger.print_status(f"Adding user '{CURRENT_USER}' to group {group} ...")
|
|
||||||
command = ["sudo", "usermod", "-a", "-G", group, CURRENT_USER]
|
|
||||||
run(command, check=True)
|
|
||||||
Logger.print_ok(f"Group {group} assigned to user '{CURRENT_USER}'.")
|
|
||||||
except CalledProcessError as e:
|
|
||||||
Logger.print_error(f"Unable to add user to usergroups: {e}")
|
|
||||||
raise
|
|
||||||
|
|
||||||
log = "Remember to relog/restart this machine for the group(s) to be applied!"
|
|
||||||
Logger.print_warn(log)
|
|
||||||
|
|
||||||
|
|
||||||
def handle_disruptive_system_packages() -> None:
|
|
||||||
services = []
|
|
||||||
|
|
||||||
command = ["systemctl", "is-enabled", "brltty"]
|
|
||||||
brltty_status = run(command, capture_output=True, text=True)
|
|
||||||
|
|
||||||
command = ["systemctl", "is-enabled", "brltty-udev"]
|
|
||||||
brltty_udev_status = run(command, capture_output=True, text=True)
|
|
||||||
|
|
||||||
command = ["systemctl", "is-enabled", "ModemManager"]
|
|
||||||
modem_manager_status = run(command, capture_output=True, text=True)
|
|
||||||
|
|
||||||
if "enabled" in brltty_status.stdout:
|
|
||||||
services.append("brltty")
|
|
||||||
if "enabled" in brltty_udev_status.stdout:
|
|
||||||
services.append("brltty-udev")
|
|
||||||
if "enabled" in modem_manager_status.stdout:
|
|
||||||
services.append("ModemManager")
|
|
||||||
|
|
||||||
for service in services if services else []:
|
|
||||||
try:
|
|
||||||
cmd_sysctl_service(service, "mask")
|
|
||||||
except CalledProcessError:
|
|
||||||
Logger.print_dialog(
|
|
||||||
DialogType.WARNING,
|
|
||||||
[
|
|
||||||
f"KIAUH was unable to mask the {service} system service. "
|
|
||||||
"Please fix the problem manually. Otherwise, this may have "
|
|
||||||
"undesirable effects on the operation of Klipper."
|
|
||||||
],
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def create_example_printer_cfg(
|
|
||||||
instance: Klipper, clients: List[BaseWebClient] | None = None
|
|
||||||
) -> None:
|
|
||||||
Logger.print_status(f"Creating example printer.cfg in '{instance.base.cfg_dir}'")
|
|
||||||
if instance.cfg_file.is_file():
|
|
||||||
Logger.print_info(f"'{instance.cfg_file}' already exists.")
|
|
||||||
return
|
|
||||||
|
|
||||||
source = MODULE_PATH.joinpath("assets/printer.cfg")
|
|
||||||
target = instance.cfg_file
|
|
||||||
try:
|
|
||||||
shutil.copy(source, target)
|
|
||||||
except OSError as e:
|
|
||||||
Logger.print_error(f"Unable to create example printer.cfg:\n{e}")
|
|
||||||
return
|
|
||||||
|
|
||||||
scp = SimpleConfigParser()
|
|
||||||
scp.read_file(target)
|
|
||||||
scp.set_option("virtual_sdcard", "path", str(instance.base.gcodes_dir))
|
|
||||||
|
|
||||||
# include existing client configs in the example config
|
|
||||||
if clients is not None and len(clients) > 0:
|
|
||||||
for c in clients:
|
|
||||||
client_config = c.client_config
|
|
||||||
section = client_config.config_section
|
|
||||||
scp.add_section(section=section)
|
|
||||||
create_client_config_symlink(client_config, [instance])
|
|
||||||
|
|
||||||
scp.write_file(target)
|
|
||||||
|
|
||||||
Logger.print_ok(f"Example printer.cfg created in '{instance.base.cfg_dir}'")
|
|
||||||
|
|
||||||
|
|
||||||
def backup_klipper_dir() -> None:
|
|
||||||
bm = BackupManager()
|
|
||||||
bm.backup_directory("klipper", source=KLIPPER_DIR, target=KLIPPER_BACKUP_DIR)
|
|
||||||
bm.backup_directory("klippy-env", source=KLIPPER_ENV_DIR, target=KLIPPER_BACKUP_DIR)
|
|
||||||
@@ -1,118 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import textwrap
|
|
||||||
from typing import Type
|
|
||||||
|
|
||||||
from components.klipper import klipper_remove
|
|
||||||
from core.constants import COLOR_CYAN, COLOR_RED, RESET_FORMAT
|
|
||||||
from core.menus import FooterType, Option
|
|
||||||
from core.menus.base_menu import BaseMenu
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyUnusedLocal
|
|
||||||
class KlipperRemoveMenu(BaseMenu):
|
|
||||||
def __init__(self, previous_menu: Type[BaseMenu] | None = None):
|
|
||||||
super().__init__()
|
|
||||||
self.previous_menu: Type[BaseMenu] | None = previous_menu
|
|
||||||
self.footer_type = FooterType.BACK
|
|
||||||
self.remove_klipper_service = False
|
|
||||||
self.remove_klipper_dir = False
|
|
||||||
self.remove_klipper_env = False
|
|
||||||
self.selection_state = False
|
|
||||||
|
|
||||||
def set_previous_menu(self, previous_menu: Type[BaseMenu] | None) -> None:
|
|
||||||
from core.menus.remove_menu import RemoveMenu
|
|
||||||
|
|
||||||
self.previous_menu = previous_menu if previous_menu is not None else RemoveMenu
|
|
||||||
|
|
||||||
def set_options(self) -> None:
|
|
||||||
self.options = {
|
|
||||||
"a": Option(method=self.toggle_all),
|
|
||||||
"1": Option(method=self.toggle_remove_klipper_service),
|
|
||||||
"2": Option(method=self.toggle_remove_klipper_dir),
|
|
||||||
"3": Option(method=self.toggle_remove_klipper_env),
|
|
||||||
"c": Option(method=self.run_removal_process),
|
|
||||||
}
|
|
||||||
|
|
||||||
def print_menu(self) -> None:
|
|
||||||
header = " [ Remove Klipper ] "
|
|
||||||
color = COLOR_RED
|
|
||||||
count = 62 - len(color) - len(RESET_FORMAT)
|
|
||||||
checked = f"[{COLOR_CYAN}x{RESET_FORMAT}]"
|
|
||||||
unchecked = "[ ]"
|
|
||||||
o1 = checked if self.remove_klipper_service else unchecked
|
|
||||||
o2 = checked if self.remove_klipper_dir else unchecked
|
|
||||||
o3 = checked if self.remove_klipper_env else unchecked
|
|
||||||
menu = textwrap.dedent(
|
|
||||||
f"""
|
|
||||||
╔═══════════════════════════════════════════════════════╗
|
|
||||||
║ {color}{header:~^{count}}{RESET_FORMAT} ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ Enter a number and hit enter to select / deselect ║
|
|
||||||
║ the specific option for removal. ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ a) {self._get_selection_state_str():37} ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ 1) {o1} Remove Service ║
|
|
||||||
║ 2) {o2} Remove Local Repository ║
|
|
||||||
║ 3) {o3} Remove Python Environment ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ C) Continue ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
"""
|
|
||||||
)[1:]
|
|
||||||
print(menu, end="")
|
|
||||||
|
|
||||||
def toggle_all(self, **kwargs) -> None:
|
|
||||||
self.selection_state = not self.selection_state
|
|
||||||
self.remove_klipper_service = self.selection_state
|
|
||||||
self.remove_klipper_dir = self.selection_state
|
|
||||||
self.remove_klipper_env = self.selection_state
|
|
||||||
|
|
||||||
def toggle_remove_klipper_service(self, **kwargs) -> None:
|
|
||||||
self.remove_klipper_service = not self.remove_klipper_service
|
|
||||||
|
|
||||||
def toggle_remove_klipper_dir(self, **kwargs) -> None:
|
|
||||||
self.remove_klipper_dir = not self.remove_klipper_dir
|
|
||||||
|
|
||||||
def toggle_remove_klipper_env(self, **kwargs) -> None:
|
|
||||||
self.remove_klipper_env = not self.remove_klipper_env
|
|
||||||
|
|
||||||
def run_removal_process(self, **kwargs) -> None:
|
|
||||||
if (
|
|
||||||
not self.remove_klipper_service
|
|
||||||
and not self.remove_klipper_dir
|
|
||||||
and not self.remove_klipper_env
|
|
||||||
):
|
|
||||||
error = f"{COLOR_RED}Nothing selected! Select options to remove first.{RESET_FORMAT}"
|
|
||||||
print(error)
|
|
||||||
return
|
|
||||||
|
|
||||||
klipper_remove.run_klipper_removal(
|
|
||||||
self.remove_klipper_service,
|
|
||||||
self.remove_klipper_dir,
|
|
||||||
self.remove_klipper_env,
|
|
||||||
)
|
|
||||||
|
|
||||||
self.remove_klipper_service = False
|
|
||||||
self.remove_klipper_dir = False
|
|
||||||
self.remove_klipper_env = False
|
|
||||||
|
|
||||||
self._go_back()
|
|
||||||
|
|
||||||
def _get_selection_state_str(self) -> str:
|
|
||||||
return (
|
|
||||||
"Select everything" if not self.selection_state else "Deselect everything"
|
|
||||||
)
|
|
||||||
|
|
||||||
def _go_back(self, **kwargs) -> None:
|
|
||||||
if self.previous_menu is not None:
|
|
||||||
self.previous_menu().run()
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
|
|
||||||
from components.klipper import KLIPPER_DIR
|
|
||||||
|
|
||||||
SD_FLASH_SCRIPT = KLIPPER_DIR.joinpath("scripts/flash-sdcard.sh")
|
|
||||||
@@ -1,211 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 re
|
|
||||||
from subprocess import (
|
|
||||||
DEVNULL,
|
|
||||||
PIPE,
|
|
||||||
STDOUT,
|
|
||||||
CalledProcessError,
|
|
||||||
Popen,
|
|
||||||
check_output,
|
|
||||||
run,
|
|
||||||
)
|
|
||||||
from typing import List
|
|
||||||
|
|
||||||
from components.klipper import KLIPPER_DIR
|
|
||||||
from components.klipper.klipper import Klipper
|
|
||||||
from components.klipper_firmware import SD_FLASH_SCRIPT
|
|
||||||
from components.klipper_firmware.flash_options import (
|
|
||||||
FlashMethod,
|
|
||||||
FlashOptions,
|
|
||||||
)
|
|
||||||
from core.instance_manager.instance_manager import InstanceManager
|
|
||||||
from core.logger import Logger
|
|
||||||
from utils.instance_utils import get_instances
|
|
||||||
from utils.sys_utils import log_process
|
|
||||||
|
|
||||||
|
|
||||||
def find_firmware_file() -> bool:
|
|
||||||
target = KLIPPER_DIR.joinpath("out")
|
|
||||||
target_exists: bool = target.exists()
|
|
||||||
|
|
||||||
f1 = "klipper.elf.hex"
|
|
||||||
f2 = "klipper.elf"
|
|
||||||
f3 = "klipper.bin"
|
|
||||||
f4 = "klipper.uf2"
|
|
||||||
fw_file_exists: bool = (
|
|
||||||
(target.joinpath(f1).exists() and target.joinpath(f2).exists())
|
|
||||||
or target.joinpath(f3).exists()
|
|
||||||
or target.joinpath(f4).exists()
|
|
||||||
)
|
|
||||||
|
|
||||||
return target_exists and fw_file_exists
|
|
||||||
|
|
||||||
|
|
||||||
def find_usb_device_by_id() -> List[str]:
|
|
||||||
try:
|
|
||||||
command = "find /dev/serial/by-id/*"
|
|
||||||
output = check_output(command, shell=True, text=True, stderr=DEVNULL)
|
|
||||||
return output.splitlines()
|
|
||||||
except CalledProcessError as e:
|
|
||||||
Logger.print_error("Unable to find a USB device!")
|
|
||||||
Logger.print_error(e, prefix=False)
|
|
||||||
return []
|
|
||||||
|
|
||||||
|
|
||||||
def find_uart_device() -> List[str]:
|
|
||||||
try:
|
|
||||||
cmd = "find /dev -maxdepth 1"
|
|
||||||
output = check_output(cmd, shell=True, text=True, stderr=DEVNULL)
|
|
||||||
device_list = []
|
|
||||||
if output:
|
|
||||||
pattern = r"^/dev/tty(AMA0|S0)$"
|
|
||||||
devices = output.splitlines()
|
|
||||||
device_list = [d for d in devices if re.search(pattern, d)]
|
|
||||||
return device_list
|
|
||||||
except CalledProcessError as e:
|
|
||||||
Logger.print_error("Unable to find a UART device!")
|
|
||||||
Logger.print_error(e, prefix=False)
|
|
||||||
return []
|
|
||||||
|
|
||||||
|
|
||||||
def find_usb_dfu_device() -> List[str]:
|
|
||||||
try:
|
|
||||||
output = check_output("lsusb", shell=True, text=True, stderr=DEVNULL)
|
|
||||||
device_list = []
|
|
||||||
if output:
|
|
||||||
devices = output.splitlines()
|
|
||||||
device_list = [d.split(" ")[5] for d in devices if "DFU" in d]
|
|
||||||
return device_list
|
|
||||||
|
|
||||||
except CalledProcessError as e:
|
|
||||||
Logger.print_error("Unable to find a USB DFU device!")
|
|
||||||
Logger.print_error(e, prefix=False)
|
|
||||||
return []
|
|
||||||
|
|
||||||
|
|
||||||
def find_usb_rp2_boot_device() -> List[str]:
|
|
||||||
try:
|
|
||||||
output = check_output("lsusb", shell=True, text=True, stderr=DEVNULL)
|
|
||||||
device_list = []
|
|
||||||
if output:
|
|
||||||
devices = output.splitlines()
|
|
||||||
device_list = [d.split(" ")[5] for d in devices if "RP2 Boot" in d]
|
|
||||||
return device_list
|
|
||||||
|
|
||||||
except CalledProcessError as e:
|
|
||||||
Logger.print_error("Unable to find a USB RP2 Boot device!")
|
|
||||||
Logger.print_error(e, prefix=False)
|
|
||||||
return []
|
|
||||||
|
|
||||||
|
|
||||||
def get_sd_flash_board_list() -> List[str]:
|
|
||||||
if not KLIPPER_DIR.exists() or not SD_FLASH_SCRIPT.exists():
|
|
||||||
return []
|
|
||||||
|
|
||||||
try:
|
|
||||||
cmd = f"{SD_FLASH_SCRIPT} -l"
|
|
||||||
blist: List[str] = check_output(cmd, shell=True, text=True).splitlines()[1:]
|
|
||||||
return blist
|
|
||||||
except CalledProcessError as e:
|
|
||||||
Logger.print_error(f"An unexpected error occured:\n{e}")
|
|
||||||
return []
|
|
||||||
|
|
||||||
|
|
||||||
def start_flash_process(flash_options: FlashOptions) -> None:
|
|
||||||
Logger.print_status(f"Flashing '{flash_options.selected_mcu}' ...")
|
|
||||||
try:
|
|
||||||
if not flash_options.flash_method:
|
|
||||||
raise Exception("Missing value for flash_method!")
|
|
||||||
if not flash_options.flash_command:
|
|
||||||
raise Exception("Missing value for flash_command!")
|
|
||||||
if not flash_options.selected_mcu:
|
|
||||||
raise Exception("Missing value for selected_mcu!")
|
|
||||||
if not flash_options.connection_type:
|
|
||||||
raise Exception("Missing value for connection_type!")
|
|
||||||
if (
|
|
||||||
flash_options.flash_method == FlashMethod.SD_CARD
|
|
||||||
and not flash_options.selected_board
|
|
||||||
):
|
|
||||||
raise Exception("Missing value for selected_board!")
|
|
||||||
|
|
||||||
if flash_options.flash_method is FlashMethod.REGULAR:
|
|
||||||
cmd = [
|
|
||||||
"make",
|
|
||||||
flash_options.flash_command.value,
|
|
||||||
f"FLASH_DEVICE={flash_options.selected_mcu}",
|
|
||||||
]
|
|
||||||
elif flash_options.flash_method is FlashMethod.SD_CARD:
|
|
||||||
if not SD_FLASH_SCRIPT.exists():
|
|
||||||
raise Exception("Unable to find Klippers sdcard flash script!")
|
|
||||||
cmd = [
|
|
||||||
SD_FLASH_SCRIPT.as_posix(),
|
|
||||||
f"-b {flash_options.selected_baudrate}",
|
|
||||||
flash_options.selected_mcu,
|
|
||||||
flash_options.selected_board,
|
|
||||||
]
|
|
||||||
else:
|
|
||||||
raise Exception("Invalid value for flash_method!")
|
|
||||||
|
|
||||||
instances = get_instances(Klipper)
|
|
||||||
InstanceManager.stop_all(instances)
|
|
||||||
|
|
||||||
process = Popen(cmd, cwd=KLIPPER_DIR, stdout=PIPE, stderr=STDOUT, text=True)
|
|
||||||
log_process(process)
|
|
||||||
|
|
||||||
InstanceManager.start_all(instances)
|
|
||||||
|
|
||||||
rc = process.returncode
|
|
||||||
if rc != 0:
|
|
||||||
raise Exception(f"Flashing failed with returncode: {rc}")
|
|
||||||
else:
|
|
||||||
Logger.print_ok("Flashing successfull!", start="\n", end="\n\n")
|
|
||||||
|
|
||||||
except (Exception, CalledProcessError):
|
|
||||||
Logger.print_error("Flashing failed!", start="\n")
|
|
||||||
Logger.print_error("See the console output above!", end="\n\n")
|
|
||||||
|
|
||||||
|
|
||||||
def run_make_clean() -> None:
|
|
||||||
try:
|
|
||||||
run(
|
|
||||||
"make clean",
|
|
||||||
cwd=KLIPPER_DIR,
|
|
||||||
shell=True,
|
|
||||||
check=True,
|
|
||||||
)
|
|
||||||
except CalledProcessError as e:
|
|
||||||
Logger.print_error(f"Unexpected error:\n{e}")
|
|
||||||
raise
|
|
||||||
|
|
||||||
|
|
||||||
def run_make_menuconfig() -> None:
|
|
||||||
try:
|
|
||||||
run(
|
|
||||||
"make PYTHON=python3 menuconfig",
|
|
||||||
cwd=KLIPPER_DIR,
|
|
||||||
shell=True,
|
|
||||||
check=True,
|
|
||||||
)
|
|
||||||
except CalledProcessError as e:
|
|
||||||
Logger.print_error(f"Unexpected error:\n{e}")
|
|
||||||
raise
|
|
||||||
|
|
||||||
|
|
||||||
def run_make() -> None:
|
|
||||||
try:
|
|
||||||
run(
|
|
||||||
"make PYTHON=python3",
|
|
||||||
cwd=KLIPPER_DIR,
|
|
||||||
shell=True,
|
|
||||||
check=True,
|
|
||||||
)
|
|
||||||
except CalledProcessError as e:
|
|
||||||
Logger.print_error(f"Unexpected error:\n{e}")
|
|
||||||
raise
|
|
||||||
@@ -1,106 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
from dataclasses import field
|
|
||||||
from enum import Enum
|
|
||||||
from typing import List
|
|
||||||
|
|
||||||
|
|
||||||
class FlashMethod(Enum):
|
|
||||||
REGULAR = "Regular"
|
|
||||||
SD_CARD = "SD Card"
|
|
||||||
|
|
||||||
|
|
||||||
class FlashCommand(Enum):
|
|
||||||
FLASH = "flash"
|
|
||||||
SERIAL_FLASH = "serialflash"
|
|
||||||
|
|
||||||
|
|
||||||
class ConnectionType(Enum):
|
|
||||||
USB = "USB"
|
|
||||||
USB_DFU = "USB (DFU)"
|
|
||||||
USB_RP2040 = "USB (RP2040)"
|
|
||||||
UART = "UART"
|
|
||||||
|
|
||||||
|
|
||||||
class FlashOptions:
|
|
||||||
_instance = None
|
|
||||||
_flash_method: FlashMethod | None = None
|
|
||||||
_flash_command: FlashCommand | None = None
|
|
||||||
_connection_type: ConnectionType | None = None
|
|
||||||
_mcu_list: List[str] = field(default_factory=list)
|
|
||||||
_selected_mcu: str = ""
|
|
||||||
_selected_board: str = ""
|
|
||||||
_selected_baudrate: int = 250000
|
|
||||||
|
|
||||||
def __new__(cls, *args, **kwargs):
|
|
||||||
if not cls._instance:
|
|
||||||
cls._instance = super(FlashOptions, cls).__new__(cls, *args, **kwargs)
|
|
||||||
return cls._instance
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def destroy(cls) -> None:
|
|
||||||
cls._instance = None
|
|
||||||
|
|
||||||
@property
|
|
||||||
def flash_method(self) -> FlashMethod | None:
|
|
||||||
return self._flash_method
|
|
||||||
|
|
||||||
@flash_method.setter
|
|
||||||
def flash_method(self, value: FlashMethod | None):
|
|
||||||
self._flash_method = value
|
|
||||||
|
|
||||||
@property
|
|
||||||
def flash_command(self) -> FlashCommand | None:
|
|
||||||
return self._flash_command
|
|
||||||
|
|
||||||
@flash_command.setter
|
|
||||||
def flash_command(self, value: FlashCommand | None):
|
|
||||||
self._flash_command = value
|
|
||||||
|
|
||||||
@property
|
|
||||||
def connection_type(self) -> ConnectionType | None:
|
|
||||||
return self._connection_type
|
|
||||||
|
|
||||||
@connection_type.setter
|
|
||||||
def connection_type(self, value: ConnectionType | None):
|
|
||||||
self._connection_type = value
|
|
||||||
|
|
||||||
@property
|
|
||||||
def mcu_list(self) -> List[str]:
|
|
||||||
return self._mcu_list
|
|
||||||
|
|
||||||
@mcu_list.setter
|
|
||||||
def mcu_list(self, value: List[str]) -> None:
|
|
||||||
self._mcu_list = value
|
|
||||||
|
|
||||||
@property
|
|
||||||
def selected_mcu(self) -> str:
|
|
||||||
return self._selected_mcu
|
|
||||||
|
|
||||||
@selected_mcu.setter
|
|
||||||
def selected_mcu(self, value: str) -> None:
|
|
||||||
self._selected_mcu = value
|
|
||||||
|
|
||||||
@property
|
|
||||||
def selected_board(self) -> str:
|
|
||||||
return self._selected_board
|
|
||||||
|
|
||||||
@selected_board.setter
|
|
||||||
def selected_board(self, value: str) -> None:
|
|
||||||
self._selected_board = value
|
|
||||||
|
|
||||||
@property
|
|
||||||
def selected_baudrate(self) -> int:
|
|
||||||
return self._selected_baudrate
|
|
||||||
|
|
||||||
@selected_baudrate.setter
|
|
||||||
def selected_baudrate(self, value: int) -> None:
|
|
||||||
self._selected_baudrate = value
|
|
||||||
@@ -1,115 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import textwrap
|
|
||||||
from typing import List, Set, Type
|
|
||||||
|
|
||||||
from components.klipper import KLIPPER_DIR
|
|
||||||
from components.klipper_firmware.firmware_utils import (
|
|
||||||
run_make,
|
|
||||||
run_make_clean,
|
|
||||||
run_make_menuconfig,
|
|
||||||
)
|
|
||||||
from core.constants import COLOR_CYAN, COLOR_GREEN, COLOR_RED, RESET_FORMAT
|
|
||||||
from core.logger import Logger
|
|
||||||
from core.menus import Option
|
|
||||||
from core.menus.base_menu import BaseMenu
|
|
||||||
from utils.sys_utils import (
|
|
||||||
check_package_install,
|
|
||||||
install_system_packages,
|
|
||||||
update_system_package_lists,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyUnusedLocal
|
|
||||||
# noinspection PyMethodMayBeStatic
|
|
||||||
class KlipperBuildFirmwareMenu(BaseMenu):
|
|
||||||
def __init__(self, previous_menu: Type[BaseMenu] | None = None):
|
|
||||||
super().__init__()
|
|
||||||
self.previous_menu: Type[BaseMenu] | None = previous_menu
|
|
||||||
self.deps: Set[str] = {"build-essential", "dpkg-dev", "make"}
|
|
||||||
self.missing_deps: List[str] = check_package_install(self.deps)
|
|
||||||
|
|
||||||
def set_previous_menu(self, previous_menu: Type[BaseMenu] | None) -> None:
|
|
||||||
from core.menus.advanced_menu import AdvancedMenu
|
|
||||||
|
|
||||||
self.previous_menu = (
|
|
||||||
previous_menu if previous_menu is not None else AdvancedMenu
|
|
||||||
)
|
|
||||||
|
|
||||||
def set_options(self) -> None:
|
|
||||||
if len(self.missing_deps) == 0:
|
|
||||||
self.input_label_txt = "Press ENTER to continue"
|
|
||||||
self.default_option = Option(method=self.start_build_process)
|
|
||||||
else:
|
|
||||||
self.input_label_txt = "Press ENTER to install dependencies"
|
|
||||||
self.default_option = Option(method=self.install_missing_deps)
|
|
||||||
|
|
||||||
def print_menu(self) -> None:
|
|
||||||
header = " [ Build Firmware Menu ] "
|
|
||||||
color = COLOR_CYAN
|
|
||||||
count = 62 - len(color) - len(RESET_FORMAT)
|
|
||||||
menu = textwrap.dedent(
|
|
||||||
f"""
|
|
||||||
╔═══════════════════════════════════════════════════════╗
|
|
||||||
║ {color}{header:~^{count}}{RESET_FORMAT} ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ The following dependencies are required: ║
|
|
||||||
║ ║
|
|
||||||
"""
|
|
||||||
)[1:]
|
|
||||||
|
|
||||||
for d in self.deps:
|
|
||||||
status_ok = f"{COLOR_GREEN}*INSTALLED*{RESET_FORMAT}"
|
|
||||||
status_missing = f"{COLOR_RED}*MISSING*{RESET_FORMAT}"
|
|
||||||
status = status_missing if d in self.missing_deps else status_ok
|
|
||||||
padding = 39 - len(d) + len(status) + (len(status_ok) - len(status))
|
|
||||||
d = f" {COLOR_CYAN}● {d}{RESET_FORMAT}"
|
|
||||||
menu += f"║ {d}{status:>{padding}} ║\n"
|
|
||||||
menu += "║ ║\n"
|
|
||||||
|
|
||||||
if len(self.missing_deps) == 0:
|
|
||||||
line = f"{COLOR_GREEN}All dependencies are met!{RESET_FORMAT}"
|
|
||||||
else:
|
|
||||||
line = f"{COLOR_RED}Dependencies are missing!{RESET_FORMAT}"
|
|
||||||
|
|
||||||
menu += f"║ {line:<62} ║\n"
|
|
||||||
menu += "╟───────────────────────────────────────────────────────╢\n"
|
|
||||||
|
|
||||||
print(menu, end="")
|
|
||||||
|
|
||||||
def install_missing_deps(self, **kwargs) -> None:
|
|
||||||
try:
|
|
||||||
update_system_package_lists(silent=False)
|
|
||||||
Logger.print_status("Installing system packages...")
|
|
||||||
install_system_packages(self.missing_deps)
|
|
||||||
except Exception as e:
|
|
||||||
Logger.print_error(e)
|
|
||||||
Logger.print_error("Installing dependencies failed!")
|
|
||||||
finally:
|
|
||||||
# restart this menu
|
|
||||||
KlipperBuildFirmwareMenu().run()
|
|
||||||
|
|
||||||
def start_build_process(self, **kwargs) -> None:
|
|
||||||
try:
|
|
||||||
run_make_clean()
|
|
||||||
run_make_menuconfig()
|
|
||||||
run_make()
|
|
||||||
|
|
||||||
Logger.print_ok("Firmware successfully built!")
|
|
||||||
Logger.print_ok(f"Firmware file located in '{KLIPPER_DIR}/out'!")
|
|
||||||
|
|
||||||
except Exception as e:
|
|
||||||
Logger.print_error(e)
|
|
||||||
Logger.print_error("Building Klipper Firmware failed!")
|
|
||||||
|
|
||||||
finally:
|
|
||||||
if self.previous_menu is not None:
|
|
||||||
self.previous_menu().run()
|
|
||||||
@@ -1,111 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import textwrap
|
|
||||||
from typing import Type
|
|
||||||
|
|
||||||
from components.klipper_firmware.flash_options import FlashMethod, FlashOptions
|
|
||||||
from core.constants import COLOR_RED, RESET_FORMAT
|
|
||||||
from core.menus import FooterType, Option
|
|
||||||
from core.menus.base_menu import BaseMenu
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyUnusedLocal
|
|
||||||
# noinspection PyMethodMayBeStatic
|
|
||||||
class KlipperNoFirmwareErrorMenu(BaseMenu):
|
|
||||||
def __init__(self, previous_menu: Type[BaseMenu] | None = None):
|
|
||||||
super().__init__()
|
|
||||||
self.previous_menu: Type[BaseMenu] | None = previous_menu
|
|
||||||
|
|
||||||
self.flash_options = FlashOptions()
|
|
||||||
self.footer_type = FooterType.BLANK
|
|
||||||
self.input_label_txt = "Press ENTER to go back to [Advanced Menu]"
|
|
||||||
|
|
||||||
def set_previous_menu(self, previous_menu: Type[BaseMenu] | None) -> None:
|
|
||||||
self.previous_menu = previous_menu
|
|
||||||
|
|
||||||
def set_options(self) -> None:
|
|
||||||
self.default_option = Option(method=self.go_back)
|
|
||||||
|
|
||||||
def print_menu(self) -> None:
|
|
||||||
header = "!!! NO FIRMWARE FILE FOUND !!!"
|
|
||||||
color = COLOR_RED
|
|
||||||
count = 62 - len(color) - len(RESET_FORMAT)
|
|
||||||
line1 = f"{color}Unable to find a compiled firmware file!{RESET_FORMAT}"
|
|
||||||
menu = textwrap.dedent(
|
|
||||||
f"""
|
|
||||||
╔═══════════════════════════════════════════════════════╗
|
|
||||||
║ {color}{header:^{count}}{RESET_FORMAT} ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ {line1:<62} ║
|
|
||||||
║ ║
|
|
||||||
║ Make sure, that: ║
|
|
||||||
║ ● the folder '~/klipper/out' and its content exist ║
|
|
||||||
║ ● the folder contains the following file: ║
|
|
||||||
"""
|
|
||||||
)[1:]
|
|
||||||
|
|
||||||
if self.flash_options.flash_method is FlashMethod.REGULAR:
|
|
||||||
menu += "║ ● 'klipper.elf' ║\n"
|
|
||||||
menu += "║ ● 'klipper.elf.hex' ║\n"
|
|
||||||
else:
|
|
||||||
menu += "║ ● 'klipper.bin' ║\n"
|
|
||||||
|
|
||||||
print(menu, end="")
|
|
||||||
|
|
||||||
def go_back(self, **kwargs) -> None:
|
|
||||||
from core.menus.advanced_menu import AdvancedMenu
|
|
||||||
|
|
||||||
AdvancedMenu().run()
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyUnusedLocal
|
|
||||||
# noinspection PyMethodMayBeStatic
|
|
||||||
class KlipperNoBoardTypesErrorMenu(BaseMenu):
|
|
||||||
def __init__(self, previous_menu: Type[BaseMenu] | None = None):
|
|
||||||
super().__init__()
|
|
||||||
self.previous_menu: Type[BaseMenu] | None = previous_menu
|
|
||||||
self.footer_type = FooterType.BLANK
|
|
||||||
self.input_label_txt = "Press ENTER to go back to [Main Menu]"
|
|
||||||
|
|
||||||
def set_previous_menu(self, previous_menu: Type[BaseMenu] | None) -> None:
|
|
||||||
self.previous_menu = previous_menu
|
|
||||||
|
|
||||||
def set_options(self) -> None:
|
|
||||||
self.default_option = Option(method=self.go_back)
|
|
||||||
|
|
||||||
def print_menu(self) -> None:
|
|
||||||
header = "!!! ERROR GETTING BOARD LIST !!!"
|
|
||||||
color = COLOR_RED
|
|
||||||
count = 62 - len(color) - len(RESET_FORMAT)
|
|
||||||
line1 = f"{color}Reading the list of supported boards failed!{RESET_FORMAT}"
|
|
||||||
menu = textwrap.dedent(
|
|
||||||
f"""
|
|
||||||
╔═══════════════════════════════════════════════════════╗
|
|
||||||
║ {color}{header:^{count}}{RESET_FORMAT} ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ {line1:<62} ║
|
|
||||||
║ ║
|
|
||||||
║ Make sure, that: ║
|
|
||||||
║ ● the folder '~/klipper' and all its content exist ║
|
|
||||||
║ ● the content of folder '~/klipper' is not currupted ║
|
|
||||||
║ ● the file '~/klipper/scripts/flash-sd.py' exist ║
|
|
||||||
║ ● your current user has access to those files/folders ║
|
|
||||||
║ ║
|
|
||||||
║ If in doubt or this process continues to fail, please ║
|
|
||||||
║ consider to download Klipper again. ║
|
|
||||||
"""
|
|
||||||
)[1:]
|
|
||||||
print(menu, end="")
|
|
||||||
|
|
||||||
def go_back(self, **kwargs) -> None:
|
|
||||||
from core.menus.main_menu import MainMenu
|
|
||||||
|
|
||||||
MainMenu().run()
|
|
||||||
@@ -1,185 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import textwrap
|
|
||||||
from typing import Type
|
|
||||||
|
|
||||||
from core.constants import COLOR_CYAN, COLOR_YELLOW, RESET_FORMAT
|
|
||||||
from core.menus.base_menu import BaseMenu
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection DuplicatedCode
|
|
||||||
class KlipperFlashMethodHelpMenu(BaseMenu):
|
|
||||||
def __init__(self, previous_menu: Type[BaseMenu] | None = None):
|
|
||||||
super().__init__()
|
|
||||||
self.previous_menu: Type[BaseMenu] | None = previous_menu
|
|
||||||
|
|
||||||
def set_previous_menu(self, previous_menu: Type[BaseMenu] | None) -> None:
|
|
||||||
from components.klipper_firmware.menus.klipper_flash_menu import (
|
|
||||||
KlipperFlashMethodMenu,
|
|
||||||
)
|
|
||||||
|
|
||||||
self.previous_menu = (
|
|
||||||
previous_menu if previous_menu is not None else KlipperFlashMethodMenu
|
|
||||||
)
|
|
||||||
|
|
||||||
def set_options(self) -> None:
|
|
||||||
pass
|
|
||||||
|
|
||||||
def print_menu(self) -> None:
|
|
||||||
header = " < ? > Help: Flash MCU < ? > "
|
|
||||||
color = COLOR_YELLOW
|
|
||||||
count = 62 - len(color) - len(RESET_FORMAT)
|
|
||||||
subheader1 = f"{COLOR_CYAN}Regular flashing method:{RESET_FORMAT}"
|
|
||||||
subheader2 = f"{COLOR_CYAN}Updating via SD-Card Update:{RESET_FORMAT}"
|
|
||||||
menu = textwrap.dedent(
|
|
||||||
f"""
|
|
||||||
╔═══════════════════════════════════════════════════════╗
|
|
||||||
║ {color}{header:~^{count}}{RESET_FORMAT} ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ {subheader1:<62} ║
|
|
||||||
║ The default method to flash controller boards which ║
|
|
||||||
║ are connected and updated over USB and not by placing ║
|
|
||||||
║ a compiled firmware file onto an internal SD-Card. ║
|
|
||||||
║ ║
|
|
||||||
║ Common controllers that get flashed that way are: ║
|
|
||||||
║ - Arduino Mega 2560 ║
|
|
||||||
║ - Fysetc F6 / S6 (used without a Display + SD-Slot) ║
|
|
||||||
║ ║
|
|
||||||
║ {subheader2:<62} ║
|
|
||||||
║ Many popular controller boards ship with a bootloader ║
|
|
||||||
║ capable of updating the firmware via SD-Card. ║
|
|
||||||
║ Choose this method if your controller board supports ║
|
|
||||||
║ this way of updating. This method ONLY works for up- ║
|
|
||||||
║ grading firmware. The initial flashing procedure must ║
|
|
||||||
║ be done manually per the instructions that apply to ║
|
|
||||||
║ your controller board. ║
|
|
||||||
║ ║
|
|
||||||
║ Common controllers that can be flashed that way are: ║
|
|
||||||
║ - BigTreeTech SKR 1.3 / 1.4 (Turbo) / E3 / Mini E3 ║
|
|
||||||
║ - Fysetc F6 / S6 (used with a Display + SD-Slot) ║
|
|
||||||
║ - Fysetc Spider ║
|
|
||||||
║ ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
"""
|
|
||||||
)[1:]
|
|
||||||
print(menu, end="")
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection DuplicatedCode
|
|
||||||
class KlipperFlashCommandHelpMenu(BaseMenu):
|
|
||||||
def __init__(self, previous_menu: Type[BaseMenu] | None = None):
|
|
||||||
super().__init__()
|
|
||||||
self.previous_menu: Type[BaseMenu] | None = previous_menu
|
|
||||||
|
|
||||||
def set_previous_menu(self, previous_menu: Type[BaseMenu] | None) -> None:
|
|
||||||
from components.klipper_firmware.menus.klipper_flash_menu import (
|
|
||||||
KlipperFlashCommandMenu,
|
|
||||||
)
|
|
||||||
|
|
||||||
self.previous_menu = (
|
|
||||||
previous_menu if previous_menu is not None else KlipperFlashCommandMenu
|
|
||||||
)
|
|
||||||
|
|
||||||
def set_options(self) -> None:
|
|
||||||
pass
|
|
||||||
|
|
||||||
def print_menu(self) -> None:
|
|
||||||
header = " < ? > Help: Flash MCU < ? > "
|
|
||||||
color = COLOR_YELLOW
|
|
||||||
count = 62 - len(color) - len(RESET_FORMAT)
|
|
||||||
subheader1 = f"{COLOR_CYAN}make flash:{RESET_FORMAT}"
|
|
||||||
subheader2 = f"{COLOR_CYAN}make serialflash:{RESET_FORMAT}"
|
|
||||||
menu = textwrap.dedent(
|
|
||||||
f"""
|
|
||||||
╔═══════════════════════════════════════════════════════╗
|
|
||||||
║ {color}{header:~^{count}}{RESET_FORMAT} ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ {subheader1:<62} ║
|
|
||||||
║ The default command to flash controller board, it ║
|
|
||||||
║ will detect selected microcontroller and use suitable ║
|
|
||||||
║ tool for flashing it. ║
|
|
||||||
║ ║
|
|
||||||
║ {subheader2:<62} ║
|
|
||||||
║ Special command to flash STM32 microcontrollers in ║
|
|
||||||
║ DFU mode but connected via serial. stm32flash command ║
|
|
||||||
║ will be used internally. ║
|
|
||||||
║ ║
|
|
||||||
"""
|
|
||||||
)[1:]
|
|
||||||
print(menu, end="")
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection DuplicatedCode
|
|
||||||
class KlipperMcuConnectionHelpMenu(BaseMenu):
|
|
||||||
def __init__(self, previous_menu: Type[BaseMenu] | None = None):
|
|
||||||
super().__init__()
|
|
||||||
self.previous_menu: Type[BaseMenu] | None = previous_menu
|
|
||||||
|
|
||||||
def set_previous_menu(self, previous_menu: Type[BaseMenu] | None) -> None:
|
|
||||||
from components.klipper_firmware.menus.klipper_flash_menu import (
|
|
||||||
KlipperSelectMcuConnectionMenu,
|
|
||||||
)
|
|
||||||
|
|
||||||
self.previous_menu = (
|
|
||||||
previous_menu
|
|
||||||
if previous_menu is not None
|
|
||||||
else KlipperSelectMcuConnectionMenu
|
|
||||||
)
|
|
||||||
|
|
||||||
def set_options(self) -> None:
|
|
||||||
pass
|
|
||||||
|
|
||||||
def print_menu(self) -> None:
|
|
||||||
header = " < ? > Help: Flash MCU < ? > "
|
|
||||||
color = COLOR_YELLOW
|
|
||||||
count = 62 - len(color) - len(RESET_FORMAT)
|
|
||||||
subheader1 = f"{COLOR_CYAN}USB:{RESET_FORMAT}"
|
|
||||||
subheader2 = f"{COLOR_CYAN}UART:{RESET_FORMAT}"
|
|
||||||
subheader3 = f"{COLOR_CYAN}USB DFU:{RESET_FORMAT}"
|
|
||||||
subheader4 = f"{COLOR_CYAN}USB RP2040 Boot:{RESET_FORMAT}"
|
|
||||||
menu = textwrap.dedent(
|
|
||||||
f"""
|
|
||||||
╔═══════════════════════════════════════════════════════╗
|
|
||||||
║ {color}{header:~^{count}}{RESET_FORMAT} ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ {subheader1:<62} ║
|
|
||||||
║ Selecting USB as the connection method will scan the ║
|
|
||||||
║ USB ports for connected controller boards. This will ║
|
|
||||||
║ be similar to the 'ls /dev/serial/by-id/*' command ║
|
|
||||||
║ suggested by the official Klipper documentation for ║
|
|
||||||
║ determining successfull USB connections! ║
|
|
||||||
║ ║
|
|
||||||
║ {subheader2:<62} ║
|
|
||||||
║ Selecting UART as the connection method will list all ║
|
|
||||||
║ possible UART serial ports. Note: This method ALWAYS ║
|
|
||||||
║ returns something as it seems impossible to determine ║
|
|
||||||
║ if a valid Klipper controller board is connected or ║
|
|
||||||
║ not. Because of that, you MUST know which UART serial ║
|
|
||||||
║ port your controller board is connected to when using ║
|
|
||||||
║ this connection method. ║
|
|
||||||
║ ║
|
|
||||||
║ {subheader3:<62} ║
|
|
||||||
║ Selecting USB DFU as the connection method will scan ║
|
|
||||||
║ the USB ports for connected controller boards in ║
|
|
||||||
║ STM32 DFU mode, which is usually done by holding down ║
|
|
||||||
║ the BOOT button or setting a special jumper on the ║
|
|
||||||
║ board before powering up. ║
|
|
||||||
║ ║
|
|
||||||
║ {subheader4:<62} ║
|
|
||||||
║ Selecting USB RP2 Boot as the connection method will ║
|
|
||||||
║ scan the USB ports for connected RP2040 controller ║
|
|
||||||
║ boards in Boot mode, which is usually done by holding ║
|
|
||||||
║ down the BOOT button before powering up. ║
|
|
||||||
║ ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
"""
|
|
||||||
)[1:]
|
|
||||||
print(menu, end="")
|
|
||||||
@@ -1,481 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import textwrap
|
|
||||||
import time
|
|
||||||
from typing import Type
|
|
||||||
|
|
||||||
from components.klipper_firmware.firmware_utils import (
|
|
||||||
find_firmware_file,
|
|
||||||
find_uart_device,
|
|
||||||
find_usb_device_by_id,
|
|
||||||
find_usb_dfu_device,
|
|
||||||
find_usb_rp2_boot_device,
|
|
||||||
get_sd_flash_board_list,
|
|
||||||
start_flash_process,
|
|
||||||
)
|
|
||||||
from components.klipper_firmware.flash_options import (
|
|
||||||
ConnectionType,
|
|
||||||
FlashCommand,
|
|
||||||
FlashMethod,
|
|
||||||
FlashOptions,
|
|
||||||
)
|
|
||||||
from components.klipper_firmware.menus.klipper_flash_error_menu import (
|
|
||||||
KlipperNoBoardTypesErrorMenu,
|
|
||||||
KlipperNoFirmwareErrorMenu,
|
|
||||||
)
|
|
||||||
from components.klipper_firmware.menus.klipper_flash_help_menu import (
|
|
||||||
KlipperFlashCommandHelpMenu,
|
|
||||||
KlipperFlashMethodHelpMenu,
|
|
||||||
KlipperMcuConnectionHelpMenu,
|
|
||||||
)
|
|
||||||
from core.constants import COLOR_CYAN, COLOR_RED, COLOR_YELLOW, RESET_FORMAT
|
|
||||||
from core.logger import DialogType, Logger
|
|
||||||
from core.menus import FooterType, Option
|
|
||||||
from core.menus.base_menu import BaseMenu
|
|
||||||
from utils.input_utils import get_number_input
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyUnusedLocal
|
|
||||||
# noinspection PyMethodMayBeStatic
|
|
||||||
class KlipperFlashMethodMenu(BaseMenu):
|
|
||||||
def __init__(self, previous_menu: Type[BaseMenu] | None = None):
|
|
||||||
super().__init__()
|
|
||||||
self.help_menu = KlipperFlashMethodHelpMenu
|
|
||||||
self.input_label_txt = "Select flash method"
|
|
||||||
self.footer_type = FooterType.BACK_HELP
|
|
||||||
self.flash_options = FlashOptions()
|
|
||||||
|
|
||||||
def set_previous_menu(self, previous_menu: Type[BaseMenu] | None) -> None:
|
|
||||||
from core.menus.advanced_menu import AdvancedMenu
|
|
||||||
|
|
||||||
self.previous_menu = (
|
|
||||||
previous_menu if previous_menu is not None else AdvancedMenu
|
|
||||||
)
|
|
||||||
|
|
||||||
def set_options(self) -> None:
|
|
||||||
self.options = {
|
|
||||||
"1": Option(self.select_regular),
|
|
||||||
"2": Option(self.select_sdcard),
|
|
||||||
}
|
|
||||||
|
|
||||||
def print_menu(self) -> None:
|
|
||||||
header = " [ MCU Flash Menu ] "
|
|
||||||
subheader = f"{COLOR_YELLOW}ATTENTION:{RESET_FORMAT}"
|
|
||||||
subline1 = f"{COLOR_YELLOW}Make sure to select the correct method for the MCU!{RESET_FORMAT}"
|
|
||||||
subline2 = f"{COLOR_YELLOW}Not all MCUs support both methods!{RESET_FORMAT}"
|
|
||||||
|
|
||||||
color = COLOR_CYAN
|
|
||||||
count = 62 - len(color) - len(RESET_FORMAT)
|
|
||||||
menu = textwrap.dedent(
|
|
||||||
f"""
|
|
||||||
╔═══════════════════════════════════════════════════════╗
|
|
||||||
║ {color}{header:~^{count}}{RESET_FORMAT} ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ Select the flash method for flashing the MCU. ║
|
|
||||||
║ ║
|
|
||||||
║ {subheader:<62} ║
|
|
||||||
║ {subline1:<62} ║
|
|
||||||
║ {subline2:<62} ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ 1) Regular flashing method ║
|
|
||||||
║ 2) Updating via SD-Card Update ║
|
|
||||||
╟───────────────────────────┬───────────────────────────╢
|
|
||||||
"""
|
|
||||||
)[1:]
|
|
||||||
print(menu, end="")
|
|
||||||
|
|
||||||
def select_regular(self, **kwargs):
|
|
||||||
self.flash_options.flash_method = FlashMethod.REGULAR
|
|
||||||
self.goto_next_menu()
|
|
||||||
|
|
||||||
def select_sdcard(self, **kwargs):
|
|
||||||
self.flash_options.flash_method = FlashMethod.SD_CARD
|
|
||||||
self.goto_next_menu()
|
|
||||||
|
|
||||||
def goto_next_menu(self, **kwargs):
|
|
||||||
if find_firmware_file():
|
|
||||||
KlipperFlashCommandMenu(previous_menu=self.__class__).run()
|
|
||||||
else:
|
|
||||||
KlipperNoFirmwareErrorMenu().run()
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyUnusedLocal
|
|
||||||
# noinspection PyMethodMayBeStatic
|
|
||||||
class KlipperFlashCommandMenu(BaseMenu):
|
|
||||||
def __init__(self, previous_menu: Type[BaseMenu] | None = None):
|
|
||||||
super().__init__()
|
|
||||||
self.help_menu = KlipperFlashCommandHelpMenu
|
|
||||||
self.input_label_txt = "Select flash command"
|
|
||||||
self.footer_type = FooterType.BACK_HELP
|
|
||||||
self.flash_options = FlashOptions()
|
|
||||||
|
|
||||||
def set_previous_menu(self, previous_menu: Type[BaseMenu] | None) -> None:
|
|
||||||
self.previous_menu = (
|
|
||||||
previous_menu if previous_menu is not None else KlipperFlashMethodMenu
|
|
||||||
)
|
|
||||||
|
|
||||||
def set_options(self) -> None:
|
|
||||||
self.options = {
|
|
||||||
"1": Option(self.select_flash),
|
|
||||||
"2": Option(self.select_serialflash),
|
|
||||||
}
|
|
||||||
self.default_option = Option(self.select_flash)
|
|
||||||
|
|
||||||
def print_menu(self) -> None:
|
|
||||||
menu = textwrap.dedent(
|
|
||||||
"""
|
|
||||||
╔═══════════════════════════════════════════════════════╗
|
|
||||||
║ Which flash command to use for flashing the MCU? ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ 1) make flash (default) ║
|
|
||||||
║ 2) make serialflash (stm32flash) ║
|
|
||||||
╟───────────────────────────┬───────────────────────────╢
|
|
||||||
"""
|
|
||||||
)[1:]
|
|
||||||
print(menu, end="")
|
|
||||||
|
|
||||||
def select_flash(self, **kwargs):
|
|
||||||
self.flash_options.flash_command = FlashCommand.FLASH
|
|
||||||
self.goto_next_menu()
|
|
||||||
|
|
||||||
def select_serialflash(self, **kwargs):
|
|
||||||
self.flash_options.flash_command = FlashCommand.SERIAL_FLASH
|
|
||||||
self.goto_next_menu()
|
|
||||||
|
|
||||||
def goto_next_menu(self, **kwargs):
|
|
||||||
KlipperSelectMcuConnectionMenu(previous_menu=self.__class__).run()
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyUnusedLocal
|
|
||||||
# noinspection PyMethodMayBeStatic
|
|
||||||
class KlipperSelectMcuConnectionMenu(BaseMenu):
|
|
||||||
def __init__(
|
|
||||||
self, previous_menu: Type[BaseMenu] | None = None, standalone: bool = False
|
|
||||||
):
|
|
||||||
super().__init__()
|
|
||||||
self.previous_menu: Type[BaseMenu] | None = previous_menu
|
|
||||||
self.__standalone = standalone
|
|
||||||
self.help_menu = KlipperMcuConnectionHelpMenu
|
|
||||||
self.input_label_txt = "Select connection type"
|
|
||||||
self.footer_type = FooterType.BACK_HELP
|
|
||||||
self.flash_options = FlashOptions()
|
|
||||||
|
|
||||||
def set_previous_menu(self, previous_menu: Type[BaseMenu] | None) -> None:
|
|
||||||
self.previous_menu = (
|
|
||||||
previous_menu if previous_menu is not None else KlipperFlashCommandMenu
|
|
||||||
)
|
|
||||||
|
|
||||||
def set_options(self) -> None:
|
|
||||||
self.options = {
|
|
||||||
"1": Option(method=self.select_usb),
|
|
||||||
"2": Option(method=self.select_dfu),
|
|
||||||
"3": Option(method=self.select_usb_dfu),
|
|
||||||
"4": Option(method=self.select_usb_rp2040),
|
|
||||||
}
|
|
||||||
|
|
||||||
def print_menu(self) -> None:
|
|
||||||
header = "Make sure that the controller board is connected now!"
|
|
||||||
color = COLOR_YELLOW
|
|
||||||
count = 62 - len(color) - len(RESET_FORMAT)
|
|
||||||
menu = textwrap.dedent(
|
|
||||||
f"""
|
|
||||||
╔═══════════════════════════════════════════════════════╗
|
|
||||||
║ {color}{header:^{count}}{RESET_FORMAT} ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ How is the controller board connected to the host? ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ 1) USB ║
|
|
||||||
║ 2) UART ║
|
|
||||||
║ 3) USB (DFU mode) ║
|
|
||||||
║ 4) USB (RP2040 mode) ║
|
|
||||||
╟───────────────────────────┬───────────────────────────╢
|
|
||||||
"""
|
|
||||||
)[1:]
|
|
||||||
print(menu, end="")
|
|
||||||
|
|
||||||
def select_usb(self, **kwargs):
|
|
||||||
self.flash_options.connection_type = ConnectionType.USB
|
|
||||||
self.get_mcu_list()
|
|
||||||
|
|
||||||
def select_dfu(self, **kwargs):
|
|
||||||
self.flash_options.connection_type = ConnectionType.UART
|
|
||||||
self.get_mcu_list()
|
|
||||||
|
|
||||||
def select_usb_dfu(self, **kwargs):
|
|
||||||
self.flash_options.connection_type = ConnectionType.USB_DFU
|
|
||||||
self.get_mcu_list()
|
|
||||||
|
|
||||||
def select_usb_rp2040(self, **kwargs):
|
|
||||||
self.flash_options.connection_type = ConnectionType.USB_RP2040
|
|
||||||
self.get_mcu_list()
|
|
||||||
|
|
||||||
def get_mcu_list(self, **kwargs):
|
|
||||||
conn_type = self.flash_options.connection_type
|
|
||||||
|
|
||||||
if conn_type is ConnectionType.USB:
|
|
||||||
Logger.print_status("Identifying MCU connected via USB ...")
|
|
||||||
self.flash_options.mcu_list = find_usb_device_by_id()
|
|
||||||
elif conn_type is ConnectionType.UART:
|
|
||||||
Logger.print_status("Identifying MCU possibly connected via UART ...")
|
|
||||||
self.flash_options.mcu_list = find_uart_device()
|
|
||||||
elif conn_type is ConnectionType.USB_DFU:
|
|
||||||
Logger.print_status("Identifying MCU connected via USB in DFU mode ...")
|
|
||||||
self.flash_options.mcu_list = find_usb_dfu_device()
|
|
||||||
elif conn_type is ConnectionType.USB_RP2040:
|
|
||||||
Logger.print_status("Identifying MCU connected via USB in RP2 Boot mode ...")
|
|
||||||
self.flash_options.mcu_list = find_usb_rp2_boot_device()
|
|
||||||
|
|
||||||
if len(self.flash_options.mcu_list) < 1:
|
|
||||||
Logger.print_warn("No MCUs found!")
|
|
||||||
Logger.print_warn("Make sure they are connected and repeat this step.")
|
|
||||||
|
|
||||||
# if standalone is True, we only display the MCUs to the user and return
|
|
||||||
if self.__standalone and len(self.flash_options.mcu_list) > 0:
|
|
||||||
Logger.print_ok("The following MCUs were found:", prefix=False)
|
|
||||||
for i, mcu in enumerate(self.flash_options.mcu_list):
|
|
||||||
print(f" ● MCU #{i}: {COLOR_CYAN}{mcu}{RESET_FORMAT}")
|
|
||||||
time.sleep(3)
|
|
||||||
return
|
|
||||||
|
|
||||||
self.goto_next_menu()
|
|
||||||
|
|
||||||
def goto_next_menu(self, **kwargs):
|
|
||||||
KlipperSelectMcuIdMenu(previous_menu=self.__class__).run()
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyUnusedLocal
|
|
||||||
# noinspection PyMethodMayBeStatic
|
|
||||||
class KlipperSelectMcuIdMenu(BaseMenu):
|
|
||||||
def __init__(self, previous_menu: Type[BaseMenu] | None = None):
|
|
||||||
super().__init__()
|
|
||||||
self.flash_options = FlashOptions()
|
|
||||||
self.mcu_list = self.flash_options.mcu_list
|
|
||||||
self.input_label_txt = "Select MCU to flash"
|
|
||||||
self.footer_type = FooterType.BACK
|
|
||||||
|
|
||||||
def set_previous_menu(self, previous_menu: Type[BaseMenu] | None) -> None:
|
|
||||||
self.previous_menu = (
|
|
||||||
previous_menu
|
|
||||||
if previous_menu is not None
|
|
||||||
else KlipperSelectMcuConnectionMenu
|
|
||||||
)
|
|
||||||
|
|
||||||
def set_options(self) -> None:
|
|
||||||
self.options = {
|
|
||||||
f"{i}": Option(self.flash_mcu, f"{i}") for i in range(len(self.mcu_list))
|
|
||||||
}
|
|
||||||
|
|
||||||
def print_menu(self) -> None:
|
|
||||||
header = "!!! ATTENTION !!!"
|
|
||||||
header2 = f"[{COLOR_CYAN}List of detected MCUs{RESET_FORMAT}]"
|
|
||||||
color = COLOR_RED
|
|
||||||
count = 62 - len(color) - len(RESET_FORMAT)
|
|
||||||
menu = textwrap.dedent(
|
|
||||||
f"""
|
|
||||||
╔═══════════════════════════════════════════════════════╗
|
|
||||||
║ {color}{header:^{count}}{RESET_FORMAT} ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ Make sure, to select the correct MCU! ║
|
|
||||||
║ ONLY flash a firmware created for the respective MCU! ║
|
|
||||||
║ ║
|
|
||||||
╟{header2:─^64}╢
|
|
||||||
║ ║
|
|
||||||
"""
|
|
||||||
)[1:]
|
|
||||||
|
|
||||||
for i, mcu in enumerate(self.mcu_list):
|
|
||||||
mcu = mcu.split("/")[-1]
|
|
||||||
menu += f"║ {i}) {COLOR_CYAN}{mcu:<51}{RESET_FORMAT}║\n"
|
|
||||||
|
|
||||||
menu += textwrap.dedent(
|
|
||||||
"""
|
|
||||||
║ ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
"""
|
|
||||||
)[1:]
|
|
||||||
print(menu, end="")
|
|
||||||
|
|
||||||
def flash_mcu(self, **kwargs):
|
|
||||||
try:
|
|
||||||
index: int | None = kwargs.get("opt_index", None)
|
|
||||||
if index is None:
|
|
||||||
raise Exception("opt_index is None")
|
|
||||||
|
|
||||||
index = int(index)
|
|
||||||
selected_mcu = self.mcu_list[index]
|
|
||||||
self.flash_options.selected_mcu = selected_mcu
|
|
||||||
|
|
||||||
if self.flash_options.flash_method == FlashMethod.SD_CARD:
|
|
||||||
KlipperSelectSDFlashBoardMenu(previous_menu=self.__class__).run()
|
|
||||||
elif self.flash_options.flash_method == FlashMethod.REGULAR:
|
|
||||||
KlipperFlashOverviewMenu(previous_menu=self.__class__).run()
|
|
||||||
except Exception as e:
|
|
||||||
Logger.print_error(e)
|
|
||||||
Logger.print_error("Flashing failed!")
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyUnusedLocal
|
|
||||||
# noinspection PyMethodMayBeStatic
|
|
||||||
class KlipperSelectSDFlashBoardMenu(BaseMenu):
|
|
||||||
def __init__(self, previous_menu: Type[BaseMenu] | None = None):
|
|
||||||
super().__init__()
|
|
||||||
self.flash_options = FlashOptions()
|
|
||||||
self.available_boards = get_sd_flash_board_list()
|
|
||||||
self.input_label_txt = "Select board type"
|
|
||||||
|
|
||||||
def set_previous_menu(self, previous_menu: Type[BaseMenu] | None) -> None:
|
|
||||||
self.previous_menu = (
|
|
||||||
previous_menu if previous_menu is not None else KlipperSelectMcuIdMenu
|
|
||||||
)
|
|
||||||
|
|
||||||
def set_options(self) -> None:
|
|
||||||
self.options = {
|
|
||||||
f"{i}": Option(self.board_select, f"{i}")
|
|
||||||
for i in range(len(self.available_boards))
|
|
||||||
}
|
|
||||||
|
|
||||||
def print_menu(self) -> None:
|
|
||||||
if len(self.available_boards) < 1:
|
|
||||||
KlipperNoBoardTypesErrorMenu().run()
|
|
||||||
else:
|
|
||||||
menu = textwrap.dedent(
|
|
||||||
"""
|
|
||||||
╔═══════════════════════════════════════════════════════╗
|
|
||||||
║ Please select the type of board that corresponds to ║
|
|
||||||
║ the currently selected MCU ID you chose before. ║
|
|
||||||
║ ║
|
|
||||||
║ The following boards are currently supported: ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
"""
|
|
||||||
)[1:]
|
|
||||||
|
|
||||||
for i, board in enumerate(self.available_boards):
|
|
||||||
line = f" {i}) {board}"
|
|
||||||
menu += f"║{line:<55}║\n"
|
|
||||||
menu += "╟───────────────────────────────────────────────────────╢"
|
|
||||||
print(menu, end="")
|
|
||||||
|
|
||||||
def board_select(self, **kwargs):
|
|
||||||
try:
|
|
||||||
index: int | None = kwargs.get("opt_index", None)
|
|
||||||
if index is None:
|
|
||||||
raise Exception("opt_index is None")
|
|
||||||
|
|
||||||
index = int(index)
|
|
||||||
self.flash_options.selected_board = self.available_boards[index]
|
|
||||||
self.baudrate_select()
|
|
||||||
except Exception as e:
|
|
||||||
Logger.print_error(e)
|
|
||||||
Logger.print_error("Board selection failed!")
|
|
||||||
|
|
||||||
def baudrate_select(self, **kwargs):
|
|
||||||
Logger.print_dialog(
|
|
||||||
DialogType.CUSTOM,
|
|
||||||
[
|
|
||||||
"If your board is flashed with firmware that connects "
|
|
||||||
"at a custom baud rate, please change it now.",
|
|
||||||
"\n\n",
|
|
||||||
"If you are unsure, stick to the default 250000!",
|
|
||||||
],
|
|
||||||
)
|
|
||||||
self.flash_options.selected_baudrate = get_number_input(
|
|
||||||
question="Please set the baud rate",
|
|
||||||
default=250000,
|
|
||||||
min_count=0,
|
|
||||||
allow_go_back=True,
|
|
||||||
)
|
|
||||||
KlipperFlashOverviewMenu(previous_menu=self.__class__).run()
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyUnusedLocal
|
|
||||||
# noinspection PyMethodMayBeStatic
|
|
||||||
class KlipperFlashOverviewMenu(BaseMenu):
|
|
||||||
def __init__(self, previous_menu: Type[BaseMenu] | None = None):
|
|
||||||
super().__init__()
|
|
||||||
self.flash_options = FlashOptions()
|
|
||||||
self.input_label_txt = "Perform action (default=Y)"
|
|
||||||
|
|
||||||
def set_previous_menu(self, previous_menu: Type[BaseMenu] | None) -> None:
|
|
||||||
self.previous_menu: Type[BaseMenu] | None = previous_menu
|
|
||||||
|
|
||||||
def set_options(self) -> None:
|
|
||||||
self.options = {
|
|
||||||
"y": Option(self.execute_flash),
|
|
||||||
"n": Option(self.abort_process),
|
|
||||||
}
|
|
||||||
|
|
||||||
self.default_option = Option(self.execute_flash)
|
|
||||||
|
|
||||||
def print_menu(self) -> None:
|
|
||||||
header = "!!! ATTENTION !!!"
|
|
||||||
color = COLOR_RED
|
|
||||||
count = 62 - len(color) - len(RESET_FORMAT)
|
|
||||||
|
|
||||||
method = self.flash_options.flash_method.value
|
|
||||||
command = self.flash_options.flash_command.value
|
|
||||||
conn_type = self.flash_options.connection_type.value
|
|
||||||
mcu = self.flash_options.selected_mcu.split("/")[-1]
|
|
||||||
board = self.flash_options.selected_board
|
|
||||||
baudrate = self.flash_options.selected_baudrate
|
|
||||||
subheader = f"[{COLOR_CYAN}Overview{RESET_FORMAT}]"
|
|
||||||
menu = textwrap.dedent(
|
|
||||||
f"""
|
|
||||||
╔═══════════════════════════════════════════════════════╗
|
|
||||||
║ {color}{header:^{count}}{RESET_FORMAT} ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ Before contuining the flashing process, please check ║
|
|
||||||
║ if all parameters were set correctly! Once you made ║
|
|
||||||
║ sure everything is correct, start the process. If any ║
|
|
||||||
║ parameter needs to be changed, you can go back (B) ║
|
|
||||||
║ step by step or abort and start from the beginning. ║
|
|
||||||
║{subheader:─^64}║
|
|
||||||
║ ║
|
|
||||||
"""
|
|
||||||
)[1:]
|
|
||||||
|
|
||||||
menu += textwrap.dedent(
|
|
||||||
f"""
|
|
||||||
║ MCU: {COLOR_CYAN}{mcu:<48}{RESET_FORMAT} ║
|
|
||||||
║ Connection: {COLOR_CYAN}{conn_type:<41}{RESET_FORMAT} ║
|
|
||||||
║ Flash method: {COLOR_CYAN}{method:<39}{RESET_FORMAT} ║
|
|
||||||
║ Flash command: {COLOR_CYAN}{command:<38}{RESET_FORMAT} ║
|
|
||||||
"""
|
|
||||||
)[1:]
|
|
||||||
|
|
||||||
if self.flash_options.flash_method is FlashMethod.SD_CARD:
|
|
||||||
menu += textwrap.dedent(
|
|
||||||
f"""
|
|
||||||
║ Board type: {COLOR_CYAN}{board:<41}{RESET_FORMAT} ║
|
|
||||||
║ Baudrate: {COLOR_CYAN}{baudrate:<43}{RESET_FORMAT} ║
|
|
||||||
"""
|
|
||||||
)[1:]
|
|
||||||
|
|
||||||
menu += textwrap.dedent(
|
|
||||||
"""
|
|
||||||
║ ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ Y) Start flash process ║
|
|
||||||
║ N) Abort - Return to Advanced Menu ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
"""
|
|
||||||
)[1:]
|
|
||||||
print(menu, end="")
|
|
||||||
|
|
||||||
def execute_flash(self, **kwargs):
|
|
||||||
start_flash_process(self.flash_options)
|
|
||||||
Logger.print_info("Returning to MCU Flash Menu in 5 seconds ...")
|
|
||||||
time.sleep(5)
|
|
||||||
KlipperFlashMethodMenu().run()
|
|
||||||
|
|
||||||
def abort_process(self, **kwargs):
|
|
||||||
from core.menus.advanced_menu import AdvancedMenu
|
|
||||||
|
|
||||||
AdvancedMenu().run()
|
|
||||||
@@ -1,34 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from pathlib import Path
|
|
||||||
|
|
||||||
from core.backup_manager import BACKUP_ROOT_DIR
|
|
||||||
from core.constants import SYSTEMD
|
|
||||||
|
|
||||||
# repo
|
|
||||||
KLIPPERSCREEN_REPO = "https://github.com/KlipperScreen/KlipperScreen.git"
|
|
||||||
|
|
||||||
# names
|
|
||||||
KLIPPERSCREEN_SERVICE_NAME = "KlipperScreen.service"
|
|
||||||
KLIPPERSCREEN_UPDATER_SECTION_NAME = "update_manager KlipperScreen"
|
|
||||||
KLIPPERSCREEN_LOG_NAME = "KlipperScreen.log"
|
|
||||||
|
|
||||||
# directories
|
|
||||||
KLIPPERSCREEN_DIR = Path.home().joinpath("KlipperScreen")
|
|
||||||
KLIPPERSCREEN_ENV_DIR = Path.home().joinpath(".KlipperScreen-env")
|
|
||||||
KLIPPERSCREEN_BACKUP_DIR = BACKUP_ROOT_DIR.joinpath("klipperscreen-backups")
|
|
||||||
|
|
||||||
# files
|
|
||||||
KLIPPERSCREEN_REQ_FILE = KLIPPERSCREEN_DIR.joinpath(
|
|
||||||
"scripts/KlipperScreen-requirements.txt"
|
|
||||||
)
|
|
||||||
KLIPPERSCREEN_INSTALL_SCRIPT = KLIPPERSCREEN_DIR.joinpath(
|
|
||||||
"scripts/KlipperScreen-install.sh"
|
|
||||||
)
|
|
||||||
KLIPPERSCREEN_SERVICE_FILE = SYSTEMD.joinpath(KLIPPERSCREEN_SERVICE_NAME)
|
|
||||||
@@ -1,206 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 pathlib import Path
|
|
||||||
from subprocess import CalledProcessError, run
|
|
||||||
from typing import List
|
|
||||||
|
|
||||||
from components.klipper.klipper import Klipper
|
|
||||||
from components.klipperscreen import (
|
|
||||||
KLIPPERSCREEN_BACKUP_DIR,
|
|
||||||
KLIPPERSCREEN_DIR,
|
|
||||||
KLIPPERSCREEN_ENV_DIR,
|
|
||||||
KLIPPERSCREEN_INSTALL_SCRIPT,
|
|
||||||
KLIPPERSCREEN_LOG_NAME,
|
|
||||||
KLIPPERSCREEN_REPO,
|
|
||||||
KLIPPERSCREEN_REQ_FILE,
|
|
||||||
KLIPPERSCREEN_SERVICE_FILE,
|
|
||||||
KLIPPERSCREEN_SERVICE_NAME,
|
|
||||||
KLIPPERSCREEN_UPDATER_SECTION_NAME,
|
|
||||||
)
|
|
||||||
from components.moonraker.moonraker import Moonraker
|
|
||||||
from core.backup_manager.backup_manager import BackupManager
|
|
||||||
from core.constants import SYSTEMD
|
|
||||||
from core.instance_manager.instance_manager import InstanceManager
|
|
||||||
from core.logger import DialogType, Logger
|
|
||||||
from core.settings.kiauh_settings import KiauhSettings
|
|
||||||
from core.types import ComponentStatus
|
|
||||||
from utils.common import (
|
|
||||||
check_install_dependencies,
|
|
||||||
get_install_status,
|
|
||||||
)
|
|
||||||
from utils.config_utils import add_config_section, remove_config_section
|
|
||||||
from utils.fs_utils import remove_with_sudo
|
|
||||||
from utils.git_utils import (
|
|
||||||
git_clone_wrapper,
|
|
||||||
git_pull_wrapper,
|
|
||||||
)
|
|
||||||
from utils.input_utils import get_confirm
|
|
||||||
from utils.instance_utils import get_instances
|
|
||||||
from utils.sys_utils import (
|
|
||||||
check_python_version,
|
|
||||||
cmd_sysctl_service,
|
|
||||||
install_python_requirements,
|
|
||||||
remove_system_service,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def install_klipperscreen() -> None:
|
|
||||||
Logger.print_status("Installing KlipperScreen ...")
|
|
||||||
|
|
||||||
if not check_python_version(3, 7):
|
|
||||||
return
|
|
||||||
|
|
||||||
mr_instances = get_instances(Moonraker)
|
|
||||||
if not mr_instances:
|
|
||||||
Logger.print_dialog(
|
|
||||||
DialogType.WARNING,
|
|
||||||
[
|
|
||||||
"Moonraker not found! KlipperScreen will not properly work "
|
|
||||||
"without a working Moonraker installation.",
|
|
||||||
"\n\n",
|
|
||||||
"KlipperScreens update manager configuration for Moonraker "
|
|
||||||
"will not be added to any moonraker.conf.",
|
|
||||||
],
|
|
||||||
)
|
|
||||||
if not get_confirm(
|
|
||||||
"Continue KlipperScreen installation?",
|
|
||||||
default_choice=False,
|
|
||||||
allow_go_back=True,
|
|
||||||
):
|
|
||||||
return
|
|
||||||
|
|
||||||
check_install_dependencies()
|
|
||||||
|
|
||||||
git_clone_wrapper(KLIPPERSCREEN_REPO, KLIPPERSCREEN_DIR)
|
|
||||||
|
|
||||||
try:
|
|
||||||
run(KLIPPERSCREEN_INSTALL_SCRIPT.as_posix(), shell=True, check=True)
|
|
||||||
if mr_instances:
|
|
||||||
patch_klipperscreen_update_manager(mr_instances)
|
|
||||||
InstanceManager.restart_all(mr_instances)
|
|
||||||
else:
|
|
||||||
Logger.print_info(
|
|
||||||
"Moonraker is not installed! Cannot add "
|
|
||||||
"KlipperScreen to update manager!"
|
|
||||||
)
|
|
||||||
Logger.print_ok("KlipperScreen successfully installed!")
|
|
||||||
except CalledProcessError as e:
|
|
||||||
Logger.print_error(f"Error installing KlipperScreen:\n{e}")
|
|
||||||
return
|
|
||||||
|
|
||||||
|
|
||||||
def patch_klipperscreen_update_manager(instances: List[Moonraker]) -> None:
|
|
||||||
add_config_section(
|
|
||||||
section=KLIPPERSCREEN_UPDATER_SECTION_NAME,
|
|
||||||
instances=instances,
|
|
||||||
options=[
|
|
||||||
("type", "git_repo"),
|
|
||||||
("path", KLIPPERSCREEN_DIR.as_posix()),
|
|
||||||
("origin", KLIPPERSCREEN_REPO),
|
|
||||||
("managed_services", "KlipperScreen"),
|
|
||||||
("env", f"{KLIPPERSCREEN_ENV_DIR}/bin/python"),
|
|
||||||
("requirements", KLIPPERSCREEN_REQ_FILE.as_posix()),
|
|
||||||
("install_script", KLIPPERSCREEN_INSTALL_SCRIPT.as_posix()),
|
|
||||||
],
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def update_klipperscreen() -> None:
|
|
||||||
if not KLIPPERSCREEN_DIR.exists():
|
|
||||||
Logger.print_info("KlipperScreen does not seem to be installed! Skipping ...")
|
|
||||||
return
|
|
||||||
|
|
||||||
try:
|
|
||||||
Logger.print_status("Updating KlipperScreen ...")
|
|
||||||
|
|
||||||
cmd_sysctl_service(KLIPPERSCREEN_SERVICE_NAME, "stop")
|
|
||||||
|
|
||||||
settings = KiauhSettings()
|
|
||||||
if settings.kiauh.backup_before_update:
|
|
||||||
backup_klipperscreen_dir()
|
|
||||||
|
|
||||||
git_pull_wrapper(KLIPPERSCREEN_REPO, KLIPPERSCREEN_DIR)
|
|
||||||
|
|
||||||
install_python_requirements(KLIPPERSCREEN_ENV_DIR, KLIPPERSCREEN_REQ_FILE)
|
|
||||||
|
|
||||||
cmd_sysctl_service(KLIPPERSCREEN_SERVICE_NAME, "start")
|
|
||||||
|
|
||||||
Logger.print_ok("KlipperScreen updated successfully.", end="\n\n")
|
|
||||||
except CalledProcessError as e:
|
|
||||||
Logger.print_error(f"Error updating KlipperScreen:\n{e}")
|
|
||||||
return
|
|
||||||
|
|
||||||
|
|
||||||
def get_klipperscreen_status() -> ComponentStatus:
|
|
||||||
return get_install_status(
|
|
||||||
KLIPPERSCREEN_DIR,
|
|
||||||
KLIPPERSCREEN_ENV_DIR,
|
|
||||||
files=[SYSTEMD.joinpath(KLIPPERSCREEN_SERVICE_NAME)],
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def remove_klipperscreen() -> None:
|
|
||||||
Logger.print_status("Removing KlipperScreen ...")
|
|
||||||
try:
|
|
||||||
if KLIPPERSCREEN_DIR.exists():
|
|
||||||
Logger.print_status("Removing KlipperScreen directory ...")
|
|
||||||
shutil.rmtree(KLIPPERSCREEN_DIR)
|
|
||||||
Logger.print_ok("KlipperScreen directory successfully removed!")
|
|
||||||
else:
|
|
||||||
Logger.print_warn("KlipperScreen directory not found!")
|
|
||||||
|
|
||||||
if KLIPPERSCREEN_ENV_DIR.exists():
|
|
||||||
Logger.print_status("Removing KlipperScreen environment ...")
|
|
||||||
shutil.rmtree(KLIPPERSCREEN_ENV_DIR)
|
|
||||||
Logger.print_ok("KlipperScreen environment successfully removed!")
|
|
||||||
else:
|
|
||||||
Logger.print_warn("KlipperScreen environment not found!")
|
|
||||||
|
|
||||||
if KLIPPERSCREEN_SERVICE_FILE.exists():
|
|
||||||
remove_system_service(KLIPPERSCREEN_SERVICE_NAME)
|
|
||||||
|
|
||||||
logfile = Path(f"/tmp/{KLIPPERSCREEN_LOG_NAME}")
|
|
||||||
if logfile.exists():
|
|
||||||
Logger.print_status("Removing KlipperScreen log file ...")
|
|
||||||
remove_with_sudo(logfile)
|
|
||||||
Logger.print_ok("KlipperScreen log file successfully removed!")
|
|
||||||
|
|
||||||
kl_instances: List[Klipper] = get_instances(Klipper)
|
|
||||||
for instance in kl_instances:
|
|
||||||
logfile = instance.base.log_dir.joinpath(KLIPPERSCREEN_LOG_NAME)
|
|
||||||
if logfile.exists():
|
|
||||||
Logger.print_status(f"Removing {logfile} ...")
|
|
||||||
Path(logfile).unlink()
|
|
||||||
Logger.print_ok(f"{logfile} successfully removed!")
|
|
||||||
|
|
||||||
mr_instances: List[Moonraker] = get_instances(Moonraker)
|
|
||||||
if mr_instances:
|
|
||||||
Logger.print_status("Removing KlipperScreen from update manager ...")
|
|
||||||
remove_config_section("update_manager KlipperScreen", mr_instances)
|
|
||||||
Logger.print_ok("KlipperScreen successfully removed from update manager!")
|
|
||||||
|
|
||||||
Logger.print_ok("KlipperScreen successfully removed!")
|
|
||||||
|
|
||||||
except Exception as e:
|
|
||||||
Logger.print_error(f"Error removing KlipperScreen:\n{e}")
|
|
||||||
|
|
||||||
|
|
||||||
def backup_klipperscreen_dir() -> None:
|
|
||||||
bm = BackupManager()
|
|
||||||
bm.backup_directory(
|
|
||||||
KLIPPERSCREEN_DIR.name,
|
|
||||||
source=KLIPPERSCREEN_DIR,
|
|
||||||
target=KLIPPERSCREEN_BACKUP_DIR,
|
|
||||||
)
|
|
||||||
bm.backup_directory(
|
|
||||||
KLIPPERSCREEN_ENV_DIR.name,
|
|
||||||
source=KLIPPERSCREEN_ENV_DIR,
|
|
||||||
target=KLIPPERSCREEN_BACKUP_DIR,
|
|
||||||
)
|
|
||||||
@@ -1,70 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import textwrap
|
|
||||||
from typing import Type
|
|
||||||
|
|
||||||
from components.log_uploads.log_upload_utils import get_logfile_list, upload_logfile
|
|
||||||
from core.constants import COLOR_YELLOW, RESET_FORMAT
|
|
||||||
from core.logger import Logger
|
|
||||||
from core.menus import Option
|
|
||||||
from core.menus.base_menu import BaseMenu
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyMethodMayBeStatic
|
|
||||||
class LogUploadMenu(BaseMenu):
|
|
||||||
def __init__(self, previous_menu: Type[BaseMenu] | None = None):
|
|
||||||
super().__init__()
|
|
||||||
self.previous_menu: Type[BaseMenu] | None = previous_menu
|
|
||||||
self.logfile_list = get_logfile_list()
|
|
||||||
|
|
||||||
def set_previous_menu(self, previous_menu: Type[BaseMenu] | None) -> None:
|
|
||||||
from core.menus.main_menu import MainMenu
|
|
||||||
|
|
||||||
self.previous_menu = previous_menu if previous_menu is not None else MainMenu
|
|
||||||
|
|
||||||
def set_options(self) -> None:
|
|
||||||
self.options = {
|
|
||||||
f"{index}": Option(self.upload, opt_index=f"{index}")
|
|
||||||
for index in range(len(self.logfile_list))
|
|
||||||
}
|
|
||||||
|
|
||||||
def print_menu(self) -> None:
|
|
||||||
header = " [ Log Upload ] "
|
|
||||||
color = COLOR_YELLOW
|
|
||||||
count = 62 - len(color) - len(RESET_FORMAT)
|
|
||||||
menu = textwrap.dedent(
|
|
||||||
f"""
|
|
||||||
╔═══════════════════════════════════════════════════════╗
|
|
||||||
║ {color}{header:~^{count}}{RESET_FORMAT} ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ You can select the following logfiles for uploading: ║
|
|
||||||
║ ║
|
|
||||||
"""
|
|
||||||
)[1:]
|
|
||||||
|
|
||||||
for logfile in enumerate(self.logfile_list):
|
|
||||||
line = f"{logfile[0]}) {logfile[1].get('display_name')}"
|
|
||||||
menu += f"║ {line:<54}║\n"
|
|
||||||
menu += "╟───────────────────────────────────────────────────────╢\n"
|
|
||||||
|
|
||||||
print(menu, end="")
|
|
||||||
|
|
||||||
def upload(self, **kwargs):
|
|
||||||
try:
|
|
||||||
index: int | None = kwargs.get("opt_index", None)
|
|
||||||
if index is None:
|
|
||||||
raise Exception("opt_index is None")
|
|
||||||
|
|
||||||
index = int(index)
|
|
||||||
upload_logfile(self.logfile_list[index])
|
|
||||||
except Exception as e:
|
|
||||||
Logger.print_error(e)
|
|
||||||
Logger.print_error("Log upload failed!")
|
|
||||||
@@ -1,128 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import textwrap
|
|
||||||
from typing import Type
|
|
||||||
|
|
||||||
from components.moonraker import moonraker_remove
|
|
||||||
from core.constants import COLOR_CYAN, COLOR_RED, RESET_FORMAT
|
|
||||||
from core.menus import Option
|
|
||||||
from core.menus.base_menu import BaseMenu
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyUnusedLocal
|
|
||||||
class MoonrakerRemoveMenu(BaseMenu):
|
|
||||||
def __init__(self, previous_menu: Type[BaseMenu] | None = None):
|
|
||||||
super().__init__()
|
|
||||||
self.previous_menu: Type[BaseMenu] | None = previous_menu
|
|
||||||
self.remove_moonraker_service = False
|
|
||||||
self.remove_moonraker_dir = False
|
|
||||||
self.remove_moonraker_env = False
|
|
||||||
self.remove_moonraker_polkit = False
|
|
||||||
self.selection_state = False
|
|
||||||
|
|
||||||
def set_previous_menu(self, previous_menu: Type[BaseMenu] | None) -> None:
|
|
||||||
from core.menus.remove_menu import RemoveMenu
|
|
||||||
|
|
||||||
self.previous_menu = previous_menu if previous_menu is not None else RemoveMenu
|
|
||||||
|
|
||||||
def set_options(self) -> None:
|
|
||||||
self.options = {
|
|
||||||
"a": Option(method=self.toggle_all),
|
|
||||||
"1": Option(method=self.toggle_remove_moonraker_service),
|
|
||||||
"2": Option(method=self.toggle_remove_moonraker_dir),
|
|
||||||
"3": Option(method=self.toggle_remove_moonraker_env),
|
|
||||||
"4": Option(method=self.toggle_remove_moonraker_polkit),
|
|
||||||
"c": Option(method=self.run_removal_process),
|
|
||||||
}
|
|
||||||
|
|
||||||
def print_menu(self) -> None:
|
|
||||||
header = " [ Remove Moonraker ] "
|
|
||||||
color = COLOR_RED
|
|
||||||
count = 62 - len(color) - len(RESET_FORMAT)
|
|
||||||
checked = f"[{COLOR_CYAN}x{RESET_FORMAT}]"
|
|
||||||
unchecked = "[ ]"
|
|
||||||
o1 = checked if self.remove_moonraker_service else unchecked
|
|
||||||
o2 = checked if self.remove_moonraker_dir else unchecked
|
|
||||||
o3 = checked if self.remove_moonraker_env else unchecked
|
|
||||||
o4 = checked if self.remove_moonraker_polkit else unchecked
|
|
||||||
menu = textwrap.dedent(
|
|
||||||
f"""
|
|
||||||
╔═══════════════════════════════════════════════════════╗
|
|
||||||
║ {color}{header:~^{count}}{RESET_FORMAT} ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ Enter a number and hit enter to select / deselect ║
|
|
||||||
║ the specific option for removal. ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ a) {self._get_selection_state_str():37} ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ 1) {o1} Remove Service ║
|
|
||||||
║ 2) {o2} Remove Local Repository ║
|
|
||||||
║ 3) {o3} Remove Python Environment ║
|
|
||||||
║ 4) {o4} Remove Policy Kit Rules ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ C) Continue ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
"""
|
|
||||||
)[1:]
|
|
||||||
print(menu, end="")
|
|
||||||
|
|
||||||
def toggle_all(self, **kwargs) -> None:
|
|
||||||
self.selection_state = not self.selection_state
|
|
||||||
self.remove_moonraker_service = self.selection_state
|
|
||||||
self.remove_moonraker_dir = self.selection_state
|
|
||||||
self.remove_moonraker_env = self.selection_state
|
|
||||||
self.remove_moonraker_polkit = self.selection_state
|
|
||||||
|
|
||||||
def toggle_remove_moonraker_service(self, **kwargs) -> None:
|
|
||||||
self.remove_moonraker_service = not self.remove_moonraker_service
|
|
||||||
|
|
||||||
def toggle_remove_moonraker_dir(self, **kwargs) -> None:
|
|
||||||
self.remove_moonraker_dir = not self.remove_moonraker_dir
|
|
||||||
|
|
||||||
def toggle_remove_moonraker_env(self, **kwargs) -> None:
|
|
||||||
self.remove_moonraker_env = not self.remove_moonraker_env
|
|
||||||
|
|
||||||
def toggle_remove_moonraker_polkit(self, **kwargs) -> None:
|
|
||||||
self.remove_moonraker_polkit = not self.remove_moonraker_polkit
|
|
||||||
|
|
||||||
def run_removal_process(self, **kwargs) -> None:
|
|
||||||
if (
|
|
||||||
not self.remove_moonraker_service
|
|
||||||
and not self.remove_moonraker_dir
|
|
||||||
and not self.remove_moonraker_env
|
|
||||||
and not self.remove_moonraker_polkit
|
|
||||||
):
|
|
||||||
error = f"{COLOR_RED}Nothing selected! Select options to remove first.{RESET_FORMAT}"
|
|
||||||
print(error)
|
|
||||||
return
|
|
||||||
|
|
||||||
moonraker_remove.run_moonraker_removal(
|
|
||||||
self.remove_moonraker_service,
|
|
||||||
self.remove_moonraker_dir,
|
|
||||||
self.remove_moonraker_env,
|
|
||||||
self.remove_moonraker_polkit,
|
|
||||||
)
|
|
||||||
|
|
||||||
self.remove_moonraker_service = False
|
|
||||||
self.remove_moonraker_dir = False
|
|
||||||
self.remove_moonraker_env = False
|
|
||||||
self.remove_moonraker_polkit = False
|
|
||||||
|
|
||||||
self._go_back()
|
|
||||||
|
|
||||||
def _get_selection_state_str(self) -> str:
|
|
||||||
return (
|
|
||||||
"Select everything" if not self.selection_state else "Deselect everything"
|
|
||||||
)
|
|
||||||
|
|
||||||
def _go_back(self, **kwargs) -> None:
|
|
||||||
if self.previous_menu is not None:
|
|
||||||
self.previous_menu().run()
|
|
||||||
@@ -1,146 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
from dataclasses import dataclass, field
|
|
||||||
from pathlib import Path
|
|
||||||
from subprocess import CalledProcessError
|
|
||||||
|
|
||||||
from components.klipper.klipper import Klipper
|
|
||||||
from components.moonraker import (
|
|
||||||
MOONRAKER_CFG_NAME,
|
|
||||||
MOONRAKER_DIR,
|
|
||||||
MOONRAKER_ENV_DIR,
|
|
||||||
MOONRAKER_ENV_FILE_NAME,
|
|
||||||
MOONRAKER_ENV_FILE_TEMPLATE,
|
|
||||||
MOONRAKER_LOG_NAME,
|
|
||||||
MOONRAKER_SERVICE_TEMPLATE,
|
|
||||||
)
|
|
||||||
from core.constants import CURRENT_USER
|
|
||||||
from core.instance_manager.base_instance import BaseInstance
|
|
||||||
from core.logger import Logger
|
|
||||||
from core.submodules.simple_config_parser.src.simple_config_parser.simple_config_parser import (
|
|
||||||
SimpleConfigParser,
|
|
||||||
)
|
|
||||||
from utils.fs_utils import create_folders
|
|
||||||
from utils.sys_utils import get_service_file_path
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyMethodMayBeStatic
|
|
||||||
@dataclass
|
|
||||||
class Moonraker:
|
|
||||||
suffix: str
|
|
||||||
base: BaseInstance = field(init=False, repr=False)
|
|
||||||
service_file_path: Path = field(init=False)
|
|
||||||
log_file_name: str = MOONRAKER_LOG_NAME
|
|
||||||
moonraker_dir: Path = MOONRAKER_DIR
|
|
||||||
env_dir: Path = MOONRAKER_ENV_DIR
|
|
||||||
data_dir: Path = field(init=False)
|
|
||||||
cfg_file: Path = field(init=False)
|
|
||||||
env_file: Path = field(init=False)
|
|
||||||
backup_dir: Path = field(init=False)
|
|
||||||
certs_dir: Path = field(init=False)
|
|
||||||
db_dir: Path = field(init=False)
|
|
||||||
port: int | None = field(init=False)
|
|
||||||
|
|
||||||
def __post_init__(self):
|
|
||||||
self.base: BaseInstance = BaseInstance(Klipper, self.suffix)
|
|
||||||
self.base.log_file_name = self.log_file_name
|
|
||||||
|
|
||||||
self.service_file_path: Path = get_service_file_path(Moonraker, self.suffix)
|
|
||||||
self.data_dir: Path = self.base.data_dir
|
|
||||||
self.cfg_file: Path = self.base.cfg_dir.joinpath(MOONRAKER_CFG_NAME)
|
|
||||||
self.env_file: Path = self.base.sysd_dir.joinpath(MOONRAKER_ENV_FILE_NAME)
|
|
||||||
self.backup_dir: Path = self.base.data_dir.joinpath("backup")
|
|
||||||
self.certs_dir: Path = self.base.data_dir.joinpath("certs")
|
|
||||||
self.db_dir: Path = self.base.data_dir.joinpath("database")
|
|
||||||
self.port: int | None = self._get_port()
|
|
||||||
|
|
||||||
def create(self) -> None:
|
|
||||||
from utils.sys_utils import create_env_file, create_service_file
|
|
||||||
|
|
||||||
Logger.print_status("Creating new Moonraker Instance ...")
|
|
||||||
|
|
||||||
try:
|
|
||||||
create_folders(self.base.base_folders)
|
|
||||||
|
|
||||||
create_service_file(
|
|
||||||
name=self.service_file_path.name,
|
|
||||||
content=self._prep_service_file_content(),
|
|
||||||
)
|
|
||||||
create_env_file(
|
|
||||||
path=self.base.sysd_dir.joinpath(MOONRAKER_ENV_FILE_NAME),
|
|
||||||
content=self._prep_env_file_content(),
|
|
||||||
)
|
|
||||||
|
|
||||||
except CalledProcessError as e:
|
|
||||||
Logger.print_error(f"Error creating instance: {e}")
|
|
||||||
raise
|
|
||||||
except OSError as e:
|
|
||||||
Logger.print_error(f"Error creating env file: {e}")
|
|
||||||
raise
|
|
||||||
|
|
||||||
def _prep_service_file_content(self) -> str:
|
|
||||||
template = MOONRAKER_SERVICE_TEMPLATE
|
|
||||||
|
|
||||||
try:
|
|
||||||
with open(template, "r") as template_file:
|
|
||||||
template_content = template_file.read()
|
|
||||||
except FileNotFoundError:
|
|
||||||
Logger.print_error(f"Unable to open {template} - File not found")
|
|
||||||
raise
|
|
||||||
|
|
||||||
service_content = template_content.replace(
|
|
||||||
"%USER%",
|
|
||||||
CURRENT_USER,
|
|
||||||
)
|
|
||||||
service_content = service_content.replace(
|
|
||||||
"%MOONRAKER_DIR%",
|
|
||||||
self.moonraker_dir.as_posix(),
|
|
||||||
)
|
|
||||||
service_content = service_content.replace(
|
|
||||||
"%ENV%",
|
|
||||||
self.env_dir.as_posix(),
|
|
||||||
)
|
|
||||||
service_content = service_content.replace(
|
|
||||||
"%ENV_FILE%",
|
|
||||||
self.base.sysd_dir.joinpath(MOONRAKER_ENV_FILE_NAME).as_posix(),
|
|
||||||
)
|
|
||||||
return service_content
|
|
||||||
|
|
||||||
def _prep_env_file_content(self) -> str:
|
|
||||||
template = MOONRAKER_ENV_FILE_TEMPLATE
|
|
||||||
|
|
||||||
try:
|
|
||||||
with open(template, "r") as env_file:
|
|
||||||
env_template_file_content = env_file.read()
|
|
||||||
except FileNotFoundError:
|
|
||||||
Logger.print_error(f"Unable to open {template} - File not found")
|
|
||||||
raise
|
|
||||||
|
|
||||||
env_file_content = env_template_file_content.replace(
|
|
||||||
"%MOONRAKER_DIR%",
|
|
||||||
self.moonraker_dir.as_posix(),
|
|
||||||
)
|
|
||||||
env_file_content = env_file_content.replace(
|
|
||||||
"%PRINTER_DATA%",
|
|
||||||
self.base.data_dir.as_posix(),
|
|
||||||
)
|
|
||||||
|
|
||||||
return env_file_content
|
|
||||||
|
|
||||||
def _get_port(self) -> int | None:
|
|
||||||
if not self.cfg_file or not self.cfg_file.is_file():
|
|
||||||
return None
|
|
||||||
|
|
||||||
scp = SimpleConfigParser()
|
|
||||||
scp.read_file(self.cfg_file)
|
|
||||||
port: int | None = scp.getint("server", "port", fallback=None)
|
|
||||||
|
|
||||||
return port
|
|
||||||
@@ -1,121 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
from subprocess import DEVNULL, PIPE, CalledProcessError, run
|
|
||||||
from typing import List
|
|
||||||
|
|
||||||
from components.klipper.klipper_dialogs import print_instance_overview
|
|
||||||
from components.moonraker import MOONRAKER_DIR, MOONRAKER_ENV_DIR
|
|
||||||
from components.moonraker.moonraker import Moonraker
|
|
||||||
from core.instance_manager.instance_manager import InstanceManager
|
|
||||||
from core.logger import Logger
|
|
||||||
from utils.fs_utils import run_remove_routines
|
|
||||||
from utils.input_utils import get_selection_input
|
|
||||||
from utils.instance_utils import get_instances
|
|
||||||
from utils.sys_utils import unit_file_exists
|
|
||||||
|
|
||||||
|
|
||||||
def run_moonraker_removal(
|
|
||||||
remove_service: bool,
|
|
||||||
remove_dir: bool,
|
|
||||||
remove_env: bool,
|
|
||||||
remove_polkit: bool,
|
|
||||||
) -> None:
|
|
||||||
instances = get_instances(Moonraker)
|
|
||||||
|
|
||||||
if remove_service:
|
|
||||||
Logger.print_status("Removing Moonraker instances ...")
|
|
||||||
if instances:
|
|
||||||
instances_to_remove = select_instances_to_remove(instances)
|
|
||||||
remove_instances(instances_to_remove)
|
|
||||||
else:
|
|
||||||
Logger.print_info("No Moonraker Services installed! Skipped ...")
|
|
||||||
|
|
||||||
delete_remaining: bool = remove_polkit or remove_dir or remove_env
|
|
||||||
if delete_remaining and unit_file_exists("moonraker", suffix="service"):
|
|
||||||
Logger.print_info("There are still other Moonraker services installed")
|
|
||||||
Logger.print_info(
|
|
||||||
"● Moonraker PolicyKit rules were not removed.", prefix=False
|
|
||||||
)
|
|
||||||
Logger.print_info(f"● '{MOONRAKER_DIR}' was not removed.", prefix=False)
|
|
||||||
Logger.print_info(f"● '{MOONRAKER_ENV_DIR}' was not removed.", prefix=False)
|
|
||||||
else:
|
|
||||||
if remove_polkit:
|
|
||||||
Logger.print_status("Removing all Moonraker policykit rules ...")
|
|
||||||
remove_polkit_rules()
|
|
||||||
if remove_dir:
|
|
||||||
Logger.print_status("Removing Moonraker local repository ...")
|
|
||||||
run_remove_routines(MOONRAKER_DIR)
|
|
||||||
if remove_env:
|
|
||||||
Logger.print_status("Removing Moonraker Python environment ...")
|
|
||||||
run_remove_routines(MOONRAKER_ENV_DIR)
|
|
||||||
|
|
||||||
|
|
||||||
def select_instances_to_remove(
|
|
||||||
instances: List[Moonraker],
|
|
||||||
) -> List[Moonraker] | None:
|
|
||||||
start_index = 1
|
|
||||||
options = [str(i + start_index) for i in range(len(instances))]
|
|
||||||
options.extend(["a", "b"])
|
|
||||||
instance_map = {options[i]: instances[i] for i in range(len(instances))}
|
|
||||||
|
|
||||||
print_instance_overview(
|
|
||||||
instances,
|
|
||||||
start_index=start_index,
|
|
||||||
show_index=True,
|
|
||||||
show_select_all=True,
|
|
||||||
)
|
|
||||||
selection = get_selection_input("Select Moonraker instance to remove", options)
|
|
||||||
|
|
||||||
instances_to_remove = []
|
|
||||||
if selection == "b":
|
|
||||||
return None
|
|
||||||
elif selection == "a":
|
|
||||||
instances_to_remove.extend(instances)
|
|
||||||
else:
|
|
||||||
instances_to_remove.append(instance_map[selection])
|
|
||||||
|
|
||||||
return instances_to_remove
|
|
||||||
|
|
||||||
|
|
||||||
def remove_instances(
|
|
||||||
instance_list: List[Moonraker] | None,
|
|
||||||
) -> None:
|
|
||||||
if not instance_list:
|
|
||||||
Logger.print_info("No Moonraker instances found. Skipped ...")
|
|
||||||
return
|
|
||||||
for instance in instance_list:
|
|
||||||
Logger.print_status(f"Removing instance {instance.service_file_path.stem} ...")
|
|
||||||
InstanceManager.remove(instance)
|
|
||||||
delete_moonraker_env_file(instance)
|
|
||||||
|
|
||||||
|
|
||||||
def remove_polkit_rules() -> None:
|
|
||||||
if not MOONRAKER_DIR.exists():
|
|
||||||
log = "Cannot remove policykit rules. Moonraker directory not found."
|
|
||||||
Logger.print_warn(log)
|
|
||||||
return
|
|
||||||
|
|
||||||
try:
|
|
||||||
cmd = [f"{MOONRAKER_DIR}/scripts/set-policykit-rules.sh", "--clear"]
|
|
||||||
run(cmd, stderr=PIPE, stdout=DEVNULL, check=True)
|
|
||||||
except CalledProcessError as e:
|
|
||||||
Logger.print_error(f"Error while removing policykit rules: {e}")
|
|
||||||
|
|
||||||
Logger.print_ok("Policykit rules successfully removed!")
|
|
||||||
|
|
||||||
|
|
||||||
def delete_moonraker_env_file(instance: Moonraker):
|
|
||||||
Logger.print_status(f"Remove '{instance.env_file}'")
|
|
||||||
if not instance.env_file.exists():
|
|
||||||
msg = f"Env file in {instance.base.sysd_dir} not found. Skipped ..."
|
|
||||||
Logger.print_info(msg)
|
|
||||||
return
|
|
||||||
run_remove_routines(instance.env_file)
|
|
||||||
@@ -1,219 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import json
|
|
||||||
import subprocess
|
|
||||||
from typing import List
|
|
||||||
|
|
||||||
from components.klipper.klipper import Klipper
|
|
||||||
from components.moonraker import (
|
|
||||||
EXIT_MOONRAKER_SETUP,
|
|
||||||
MOONRAKER_DEPS_JSON_FILE,
|
|
||||||
MOONRAKER_DIR,
|
|
||||||
MOONRAKER_ENV_DIR,
|
|
||||||
MOONRAKER_INSTALL_SCRIPT,
|
|
||||||
MOONRAKER_REQ_FILE,
|
|
||||||
MOONRAKER_SPEEDUPS_REQ_FILE,
|
|
||||||
POLKIT_FILE,
|
|
||||||
POLKIT_LEGACY_FILE,
|
|
||||||
POLKIT_SCRIPT,
|
|
||||||
POLKIT_USR_FILE,
|
|
||||||
)
|
|
||||||
from components.moonraker.moonraker import Moonraker
|
|
||||||
from components.moonraker.moonraker_dialogs import print_moonraker_overview
|
|
||||||
from components.moonraker.moonraker_utils import (
|
|
||||||
backup_moonraker_dir,
|
|
||||||
create_example_moonraker_conf,
|
|
||||||
)
|
|
||||||
from components.webui_client.client_utils import (
|
|
||||||
enable_mainsail_remotemode,
|
|
||||||
get_existing_clients,
|
|
||||||
)
|
|
||||||
from components.webui_client.mainsail_data import MainsailData
|
|
||||||
from core.instance_manager.instance_manager import InstanceManager
|
|
||||||
from core.logger import Logger
|
|
||||||
from core.settings.kiauh_settings import KiauhSettings
|
|
||||||
from utils.common import check_install_dependencies
|
|
||||||
from utils.fs_utils import check_file_exist
|
|
||||||
from utils.git_utils import git_clone_wrapper, git_pull_wrapper
|
|
||||||
from utils.input_utils import (
|
|
||||||
get_confirm,
|
|
||||||
get_selection_input,
|
|
||||||
)
|
|
||||||
from utils.instance_utils import get_instances
|
|
||||||
from utils.sys_utils import (
|
|
||||||
check_python_version,
|
|
||||||
cmd_sysctl_manage,
|
|
||||||
cmd_sysctl_service,
|
|
||||||
create_python_venv,
|
|
||||||
install_python_requirements,
|
|
||||||
parse_packages_from_file,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def install_moonraker() -> None:
|
|
||||||
klipper_list: List[Klipper] = get_instances(Klipper)
|
|
||||||
|
|
||||||
if not check_moonraker_install_requirements(klipper_list):
|
|
||||||
return
|
|
||||||
|
|
||||||
moonraker_list: List[Moonraker] = get_instances(Moonraker)
|
|
||||||
instances: List[Moonraker] = []
|
|
||||||
selected_option: str | Klipper
|
|
||||||
|
|
||||||
if len(klipper_list) == 1:
|
|
||||||
instances.append(Moonraker(klipper_list[0].suffix))
|
|
||||||
else:
|
|
||||||
print_moonraker_overview(
|
|
||||||
klipper_list,
|
|
||||||
moonraker_list,
|
|
||||||
show_index=True,
|
|
||||||
show_select_all=True,
|
|
||||||
)
|
|
||||||
options = {str(i + 1): k for i, k in enumerate(klipper_list)}
|
|
||||||
additional_options = {"a": None, "b": None}
|
|
||||||
options = {**options, **additional_options}
|
|
||||||
question = "Select Klipper instance to setup Moonraker for"
|
|
||||||
selected_option = get_selection_input(question, options)
|
|
||||||
|
|
||||||
if selected_option == "b":
|
|
||||||
Logger.print_status(EXIT_MOONRAKER_SETUP)
|
|
||||||
return
|
|
||||||
|
|
||||||
if selected_option == "a":
|
|
||||||
instances.extend([Moonraker(k.suffix) for k in klipper_list])
|
|
||||||
else:
|
|
||||||
klipper_instance: Klipper | None = options.get(selected_option)
|
|
||||||
if klipper_instance is None:
|
|
||||||
raise Exception("Error selecting instance!")
|
|
||||||
instances.append(Moonraker(klipper_instance.suffix))
|
|
||||||
|
|
||||||
create_example_cfg = get_confirm("Create example moonraker.conf?")
|
|
||||||
|
|
||||||
try:
|
|
||||||
check_install_dependencies()
|
|
||||||
setup_moonraker_prerequesites()
|
|
||||||
install_moonraker_polkit()
|
|
||||||
|
|
||||||
used_ports_map = {m.suffix: m.port for m in moonraker_list}
|
|
||||||
for instance in instances:
|
|
||||||
instance.create()
|
|
||||||
cmd_sysctl_service(instance.service_file_path.name, "enable")
|
|
||||||
|
|
||||||
if create_example_cfg:
|
|
||||||
# if a webclient and/or it's config is installed, patch
|
|
||||||
# its update section to the config
|
|
||||||
clients = get_existing_clients()
|
|
||||||
create_example_moonraker_conf(instance, used_ports_map, clients)
|
|
||||||
|
|
||||||
cmd_sysctl_service(instance.service_file_path.name, "start")
|
|
||||||
|
|
||||||
cmd_sysctl_manage("daemon-reload")
|
|
||||||
|
|
||||||
# if mainsail is installed, and we installed
|
|
||||||
# multiple moonraker instances, we enable mainsails remote mode
|
|
||||||
if MainsailData().client_dir.exists() and len(moonraker_list) > 1:
|
|
||||||
enable_mainsail_remotemode()
|
|
||||||
|
|
||||||
except Exception as e:
|
|
||||||
Logger.print_error(f"Error while installing Moonraker: {e}")
|
|
||||||
return
|
|
||||||
|
|
||||||
|
|
||||||
def check_moonraker_install_requirements(klipper_list: List[Klipper]) -> bool:
|
|
||||||
def check_klipper_instances() -> bool:
|
|
||||||
if len(klipper_list) >= 1:
|
|
||||||
return True
|
|
||||||
|
|
||||||
Logger.print_warn("Klipper not installed!")
|
|
||||||
Logger.print_warn("Moonraker cannot be installed! Install Klipper first.")
|
|
||||||
return False
|
|
||||||
|
|
||||||
return check_python_version(3, 7) and check_klipper_instances()
|
|
||||||
|
|
||||||
|
|
||||||
def setup_moonraker_prerequesites() -> None:
|
|
||||||
settings = KiauhSettings()
|
|
||||||
repo = settings.moonraker.repo_url
|
|
||||||
branch = settings.moonraker.branch
|
|
||||||
|
|
||||||
git_clone_wrapper(repo, MOONRAKER_DIR, branch)
|
|
||||||
|
|
||||||
# install moonraker dependencies and create python virtualenv
|
|
||||||
install_moonraker_packages()
|
|
||||||
if create_python_venv(MOONRAKER_ENV_DIR):
|
|
||||||
install_python_requirements(MOONRAKER_ENV_DIR, MOONRAKER_REQ_FILE)
|
|
||||||
install_python_requirements(MOONRAKER_ENV_DIR, MOONRAKER_SPEEDUPS_REQ_FILE)
|
|
||||||
|
|
||||||
|
|
||||||
def install_moonraker_packages() -> None:
|
|
||||||
moonraker_deps = []
|
|
||||||
|
|
||||||
if MOONRAKER_DEPS_JSON_FILE.exists():
|
|
||||||
with open(MOONRAKER_DEPS_JSON_FILE, "r") as deps:
|
|
||||||
moonraker_deps = json.load(deps).get("debian", [])
|
|
||||||
elif MOONRAKER_INSTALL_SCRIPT.exists():
|
|
||||||
moonraker_deps = parse_packages_from_file(MOONRAKER_INSTALL_SCRIPT)
|
|
||||||
|
|
||||||
if not moonraker_deps:
|
|
||||||
raise ValueError("Error reading Moonraker dependencies!")
|
|
||||||
|
|
||||||
check_install_dependencies({*moonraker_deps})
|
|
||||||
|
|
||||||
|
|
||||||
def install_moonraker_polkit() -> None:
|
|
||||||
Logger.print_status("Installing Moonraker policykit rules ...")
|
|
||||||
|
|
||||||
legacy_file_exists = check_file_exist(POLKIT_LEGACY_FILE, True)
|
|
||||||
polkit_file_exists = check_file_exist(POLKIT_FILE, True)
|
|
||||||
usr_file_exists = check_file_exist(POLKIT_USR_FILE, True)
|
|
||||||
|
|
||||||
if legacy_file_exists or (polkit_file_exists and usr_file_exists):
|
|
||||||
Logger.print_info("Moonraker policykit rules are already installed.")
|
|
||||||
return
|
|
||||||
|
|
||||||
try:
|
|
||||||
command = [POLKIT_SCRIPT, "--disable-systemctl"]
|
|
||||||
result = subprocess.run(
|
|
||||||
command,
|
|
||||||
stderr=subprocess.PIPE,
|
|
||||||
stdout=subprocess.DEVNULL,
|
|
||||||
text=True,
|
|
||||||
)
|
|
||||||
if result.returncode != 0 or result.stderr:
|
|
||||||
Logger.print_error(f"{result.stderr}", False)
|
|
||||||
Logger.print_error("Installing Moonraker policykit rules failed!")
|
|
||||||
return
|
|
||||||
|
|
||||||
Logger.print_ok("Moonraker policykit rules successfully installed!")
|
|
||||||
except subprocess.CalledProcessError as e:
|
|
||||||
log = f"Error while installing Moonraker policykit rules: {e.stderr.decode()}"
|
|
||||||
Logger.print_error(log)
|
|
||||||
|
|
||||||
|
|
||||||
def update_moonraker() -> None:
|
|
||||||
if not get_confirm("Update Moonraker now?"):
|
|
||||||
return
|
|
||||||
|
|
||||||
settings = KiauhSettings()
|
|
||||||
if settings.kiauh.backup_before_update:
|
|
||||||
backup_moonraker_dir()
|
|
||||||
|
|
||||||
instances = get_instances(Moonraker)
|
|
||||||
InstanceManager.stop_all(instances)
|
|
||||||
|
|
||||||
git_pull_wrapper(repo=settings.moonraker.repo_url, target_dir=MOONRAKER_DIR)
|
|
||||||
|
|
||||||
# install possible new system packages
|
|
||||||
install_moonraker_packages()
|
|
||||||
# install possible new python dependencies
|
|
||||||
install_python_requirements(MOONRAKER_ENV_DIR, MOONRAKER_REQ_FILE)
|
|
||||||
|
|
||||||
InstanceManager.start_all(instances)
|
|
||||||
@@ -1,140 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 Dict, List, Optional
|
|
||||||
|
|
||||||
from components.moonraker import (
|
|
||||||
MODULE_PATH,
|
|
||||||
MOONRAKER_BACKUP_DIR,
|
|
||||||
MOONRAKER_DB_BACKUP_DIR,
|
|
||||||
MOONRAKER_DEFAULT_PORT,
|
|
||||||
MOONRAKER_DIR,
|
|
||||||
MOONRAKER_ENV_DIR,
|
|
||||||
)
|
|
||||||
from components.moonraker.moonraker import Moonraker
|
|
||||||
from components.webui_client.base_data import BaseWebClient
|
|
||||||
from core.backup_manager.backup_manager import BackupManager
|
|
||||||
from core.logger import Logger
|
|
||||||
from core.submodules.simple_config_parser.src.simple_config_parser.simple_config_parser import (
|
|
||||||
SimpleConfigParser,
|
|
||||||
)
|
|
||||||
from core.types import ComponentStatus
|
|
||||||
from utils.common import get_install_status
|
|
||||||
from utils.instance_utils import get_instances
|
|
||||||
from utils.sys_utils import (
|
|
||||||
get_ipv4_addr,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def get_moonraker_status() -> ComponentStatus:
|
|
||||||
return get_install_status(MOONRAKER_DIR, MOONRAKER_ENV_DIR, Moonraker)
|
|
||||||
|
|
||||||
|
|
||||||
def create_example_moonraker_conf(
|
|
||||||
instance: Moonraker,
|
|
||||||
ports_map: Dict[str, int],
|
|
||||||
clients: Optional[List[BaseWebClient]] = None,
|
|
||||||
) -> None:
|
|
||||||
Logger.print_status(f"Creating example moonraker.conf in '{instance.base.cfg_dir}'")
|
|
||||||
if instance.cfg_file.is_file():
|
|
||||||
Logger.print_info(f"'{instance.cfg_file}' already exists.")
|
|
||||||
return
|
|
||||||
|
|
||||||
source = MODULE_PATH.joinpath("assets/moonraker.conf")
|
|
||||||
target = instance.cfg_file
|
|
||||||
try:
|
|
||||||
shutil.copy(source, target)
|
|
||||||
except OSError as e:
|
|
||||||
Logger.print_error(f"Unable to create example moonraker.conf:\n{e}")
|
|
||||||
return
|
|
||||||
|
|
||||||
ports = [
|
|
||||||
ports_map.get(instance)
|
|
||||||
for instance in ports_map
|
|
||||||
if ports_map.get(instance) is not None
|
|
||||||
]
|
|
||||||
if ports_map.get(instance.suffix) is None:
|
|
||||||
# this could be improved to not increment the max value of the ports list and assign it as the port
|
|
||||||
# as it can lead to situation where the port for e.g. instance moonraker-2 becomes 7128 if the port
|
|
||||||
# of moonraker-1 is 7125 and moonraker-3 is 7127 and there are moonraker.conf files for moonraker-1
|
|
||||||
# and moonraker-3 already. though, there does not seem to be a very reliable way of always assigning
|
|
||||||
# the correct port to each instance and the user will likely be required to correct the value manually.
|
|
||||||
port = max(ports) + 1 if ports else MOONRAKER_DEFAULT_PORT
|
|
||||||
else:
|
|
||||||
port = ports_map.get(instance.suffix)
|
|
||||||
|
|
||||||
ports_map[instance.suffix] = port
|
|
||||||
|
|
||||||
ip = get_ipv4_addr().split(".")[:2]
|
|
||||||
ip.extend(["0", "0/16"])
|
|
||||||
uds = instance.base.comms_dir.joinpath("klippy.sock")
|
|
||||||
|
|
||||||
scp = SimpleConfigParser()
|
|
||||||
scp.read_file(target)
|
|
||||||
trusted_clients: List[str] = [
|
|
||||||
f" {'.'.join(ip)}\n",
|
|
||||||
*scp.getval("authorization", "trusted_clients"),
|
|
||||||
]
|
|
||||||
|
|
||||||
scp.set_option("server", "port", str(port))
|
|
||||||
scp.set_option("server", "klippy_uds_address", str(uds))
|
|
||||||
scp.set_option("authorization", "trusted_clients", trusted_clients)
|
|
||||||
|
|
||||||
# add existing client and client configs in the update section
|
|
||||||
if clients is not None and len(clients) > 0:
|
|
||||||
for c in clients:
|
|
||||||
# client part
|
|
||||||
c_section = f"update_manager {c.name}"
|
|
||||||
c_options = [
|
|
||||||
("type", "web"),
|
|
||||||
("channel", "stable"),
|
|
||||||
("repo", c.repo_path),
|
|
||||||
("path", c.client_dir),
|
|
||||||
]
|
|
||||||
scp.add_section(section=c_section)
|
|
||||||
for option in c_options:
|
|
||||||
scp.set_option(c_section, option[0], option[1])
|
|
||||||
|
|
||||||
# client config part
|
|
||||||
c_config = c.client_config
|
|
||||||
if c_config.config_dir.exists():
|
|
||||||
c_config_section = f"update_manager {c_config.name}"
|
|
||||||
c_config_options = [
|
|
||||||
("type", "git_repo"),
|
|
||||||
("primary_branch", "master"),
|
|
||||||
("path", c_config.config_dir),
|
|
||||||
("origin", c_config.repo_url),
|
|
||||||
("managed_services", "klipper"),
|
|
||||||
]
|
|
||||||
scp.add_section(section=c_config_section)
|
|
||||||
for option in c_config_options:
|
|
||||||
scp.set_option(c_config_section, option[0], option[1])
|
|
||||||
|
|
||||||
scp.write_file(target)
|
|
||||||
Logger.print_ok(f"Example moonraker.conf created in '{instance.base.cfg_dir}'")
|
|
||||||
|
|
||||||
|
|
||||||
def backup_moonraker_dir() -> None:
|
|
||||||
bm = BackupManager()
|
|
||||||
bm.backup_directory("moonraker", source=MOONRAKER_DIR, target=MOONRAKER_BACKUP_DIR)
|
|
||||||
bm.backup_directory(
|
|
||||||
"moonraker-env", source=MOONRAKER_ENV_DIR, target=MOONRAKER_BACKUP_DIR
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def backup_moonraker_db_dir() -> None:
|
|
||||||
instances: List[Moonraker] = get_instances(Moonraker)
|
|
||||||
bm = BackupManager()
|
|
||||||
|
|
||||||
for instance in instances:
|
|
||||||
name = f"database-{instance.data_dir.name}"
|
|
||||||
bm.backup_directory(
|
|
||||||
name, source=instance.db_dir, target=MOONRAKER_DB_BACKUP_DIR
|
|
||||||
)
|
|
||||||
@@ -1,56 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
from abc import ABC
|
|
||||||
from dataclasses import dataclass
|
|
||||||
from enum import Enum
|
|
||||||
from pathlib import Path
|
|
||||||
|
|
||||||
|
|
||||||
class WebClientType(Enum):
|
|
||||||
MAINSAIL: str = "mainsail"
|
|
||||||
FLUIDD: str = "fluidd"
|
|
||||||
|
|
||||||
|
|
||||||
class WebClientConfigType(Enum):
|
|
||||||
MAINSAIL: str = "mainsail-config"
|
|
||||||
FLUIDD: str = "fluidd-config"
|
|
||||||
|
|
||||||
|
|
||||||
@dataclass()
|
|
||||||
class BaseWebClient(ABC):
|
|
||||||
"""Base class for webclient data"""
|
|
||||||
|
|
||||||
client: WebClientType
|
|
||||||
name: str
|
|
||||||
display_name: str
|
|
||||||
client_dir: Path
|
|
||||||
config_file: Path
|
|
||||||
backup_dir: Path
|
|
||||||
repo_path: str
|
|
||||||
download_url: str
|
|
||||||
nginx_access_log: Path
|
|
||||||
nginx_error_log: Path
|
|
||||||
client_config: BaseWebClientConfig
|
|
||||||
|
|
||||||
|
|
||||||
@dataclass()
|
|
||||||
class BaseWebClientConfig(ABC):
|
|
||||||
"""Base class for webclient config data"""
|
|
||||||
|
|
||||||
client_config: WebClientConfigType
|
|
||||||
name: str
|
|
||||||
display_name: str
|
|
||||||
config_filename: str
|
|
||||||
config_dir: Path
|
|
||||||
backup_dir: Path
|
|
||||||
repo_url: str
|
|
||||||
config_section: str
|
|
||||||
@@ -1,43 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
|
|
||||||
|
|
||||||
from typing import List
|
|
||||||
|
|
||||||
from components.klipper.klipper import Klipper
|
|
||||||
from components.moonraker.moonraker import Moonraker
|
|
||||||
from components.webui_client.base_data import BaseWebClientConfig
|
|
||||||
from core.logger import Logger
|
|
||||||
from utils.config_utils import remove_config_section
|
|
||||||
from utils.fs_utils import run_remove_routines
|
|
||||||
from utils.instance_utils import get_instances
|
|
||||||
|
|
||||||
|
|
||||||
def run_client_config_removal(
|
|
||||||
client_config: BaseWebClientConfig,
|
|
||||||
kl_instances: List[Klipper],
|
|
||||||
mr_instances: List[Moonraker],
|
|
||||||
) -> None:
|
|
||||||
remove_client_config_dir(client_config)
|
|
||||||
remove_client_config_symlink(client_config)
|
|
||||||
remove_config_section(f"update_manager {client_config.name}", mr_instances)
|
|
||||||
remove_config_section(client_config.config_section, kl_instances)
|
|
||||||
|
|
||||||
|
|
||||||
def remove_client_config_dir(client_config: BaseWebClientConfig) -> None:
|
|
||||||
Logger.print_status(f"Removing {client_config.display_name} ...")
|
|
||||||
run_remove_routines(client_config.config_dir)
|
|
||||||
|
|
||||||
|
|
||||||
def remove_client_config_symlink(client_config: BaseWebClientConfig) -> None:
|
|
||||||
instances: List[Klipper] = get_instances(Klipper)
|
|
||||||
for instance in instances:
|
|
||||||
run_remove_routines(
|
|
||||||
instance.base.cfg_dir.joinpath(client_config.config_filename)
|
|
||||||
)
|
|
||||||
@@ -1,125 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import shutil
|
|
||||||
import subprocess
|
|
||||||
from pathlib import Path
|
|
||||||
from typing import List
|
|
||||||
|
|
||||||
from components.klipper.klipper import Klipper
|
|
||||||
from components.moonraker.moonraker import Moonraker
|
|
||||||
from components.webui_client.base_data import BaseWebClient, BaseWebClientConfig
|
|
||||||
from components.webui_client.client_dialogs import (
|
|
||||||
print_client_already_installed_dialog,
|
|
||||||
)
|
|
||||||
from components.webui_client.client_utils import (
|
|
||||||
backup_client_config_data,
|
|
||||||
detect_client_cfg_conflict,
|
|
||||||
)
|
|
||||||
from core.instance_manager.instance_manager import InstanceManager
|
|
||||||
from core.logger import Logger
|
|
||||||
from core.settings.kiauh_settings import KiauhSettings
|
|
||||||
from utils.common import backup_printer_config_dir
|
|
||||||
from utils.config_utils import add_config_section, add_config_section_at_top
|
|
||||||
from utils.fs_utils import create_symlink
|
|
||||||
from utils.git_utils import git_clone_wrapper, git_pull_wrapper
|
|
||||||
from utils.input_utils import get_confirm
|
|
||||||
from utils.instance_utils import get_instances
|
|
||||||
|
|
||||||
|
|
||||||
def install_client_config(client_data: BaseWebClient) -> None:
|
|
||||||
client_config: BaseWebClientConfig = client_data.client_config
|
|
||||||
display_name = client_config.display_name
|
|
||||||
|
|
||||||
if detect_client_cfg_conflict(client_data):
|
|
||||||
Logger.print_info("Another Client-Config is already installed! Skipped ...")
|
|
||||||
return
|
|
||||||
|
|
||||||
if client_config.config_dir.exists():
|
|
||||||
print_client_already_installed_dialog(display_name)
|
|
||||||
if get_confirm(f"Re-install {display_name}?", allow_go_back=True):
|
|
||||||
shutil.rmtree(client_config.config_dir)
|
|
||||||
else:
|
|
||||||
return
|
|
||||||
|
|
||||||
mr_instances: List[Moonraker] = get_instances(Moonraker)
|
|
||||||
kl_instances = get_instances(Klipper)
|
|
||||||
|
|
||||||
try:
|
|
||||||
download_client_config(client_config)
|
|
||||||
create_client_config_symlink(client_config, kl_instances)
|
|
||||||
|
|
||||||
backup_printer_config_dir()
|
|
||||||
|
|
||||||
add_config_section(
|
|
||||||
section=f"update_manager {client_config.name}",
|
|
||||||
instances=mr_instances,
|
|
||||||
options=[
|
|
||||||
("type", "git_repo"),
|
|
||||||
("primary_branch", "master"),
|
|
||||||
("path", str(client_config.config_dir)),
|
|
||||||
("origin", str(client_config.repo_url)),
|
|
||||||
("managed_services", "klipper"),
|
|
||||||
],
|
|
||||||
)
|
|
||||||
add_config_section_at_top(client_config.config_section, kl_instances)
|
|
||||||
InstanceManager.restart_all(kl_instances)
|
|
||||||
|
|
||||||
except Exception as e:
|
|
||||||
Logger.print_error(f"{display_name} installation failed!\n{e}")
|
|
||||||
return
|
|
||||||
|
|
||||||
Logger.print_ok(f"{display_name} installation complete!", start="\n")
|
|
||||||
|
|
||||||
|
|
||||||
def download_client_config(client_config: BaseWebClientConfig) -> None:
|
|
||||||
try:
|
|
||||||
Logger.print_status(f"Downloading {client_config.display_name} ...")
|
|
||||||
repo = client_config.repo_url
|
|
||||||
target_dir = client_config.config_dir
|
|
||||||
git_clone_wrapper(repo, target_dir)
|
|
||||||
except Exception:
|
|
||||||
Logger.print_error(f"Downloading {client_config.display_name} failed!")
|
|
||||||
raise
|
|
||||||
|
|
||||||
|
|
||||||
def update_client_config(client: BaseWebClient) -> None:
|
|
||||||
client_config: BaseWebClientConfig = client.client_config
|
|
||||||
|
|
||||||
Logger.print_status(f"Updating {client_config.display_name} ...")
|
|
||||||
|
|
||||||
if not client_config.config_dir.exists():
|
|
||||||
Logger.print_info(
|
|
||||||
f"Unable to update {client_config.display_name}. Directory does not exist! Skipping ..."
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
settings = KiauhSettings()
|
|
||||||
if settings.kiauh.backup_before_update:
|
|
||||||
backup_client_config_data(client)
|
|
||||||
|
|
||||||
git_pull_wrapper(client_config.repo_url, client_config.config_dir)
|
|
||||||
|
|
||||||
Logger.print_ok(f"Successfully updated {client_config.display_name}.")
|
|
||||||
Logger.print_info("Restart Klipper to reload the configuration!")
|
|
||||||
|
|
||||||
|
|
||||||
def create_client_config_symlink(
|
|
||||||
client_config: BaseWebClientConfig, klipper_instances: List[Klipper]
|
|
||||||
) -> None:
|
|
||||||
for instance in klipper_instances:
|
|
||||||
Logger.print_status(f"Create symlink for {client_config.config_filename} ...")
|
|
||||||
source = Path(client_config.config_dir, client_config.config_filename)
|
|
||||||
target = instance.base.cfg_dir
|
|
||||||
Logger.print_status(f"Linking {source} to {target}")
|
|
||||||
try:
|
|
||||||
create_symlink(source, target)
|
|
||||||
except subprocess.CalledProcessError:
|
|
||||||
Logger.print_error("Creating symlink failed!")
|
|
||||||
@@ -1,93 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
|
|
||||||
from typing import List
|
|
||||||
|
|
||||||
from components.webui_client.base_data import BaseWebClient
|
|
||||||
from core.logger import DialogType, Logger
|
|
||||||
|
|
||||||
|
|
||||||
def print_moonraker_not_found_dialog(name: str) -> None:
|
|
||||||
Logger.print_dialog(
|
|
||||||
DialogType.WARNING,
|
|
||||||
[
|
|
||||||
"No local Moonraker installation was found!",
|
|
||||||
"\n\n",
|
|
||||||
f"It is possible to install {name} without a local Moonraker installation. "
|
|
||||||
"If you continue, you need to make sure, that Moonraker is installed on "
|
|
||||||
f"another machine in your network. Otherwise {name} will NOT work "
|
|
||||||
"correctly.",
|
|
||||||
],
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def print_client_already_installed_dialog(name: str) -> None:
|
|
||||||
Logger.print_dialog(
|
|
||||||
DialogType.WARNING,
|
|
||||||
[
|
|
||||||
f"{name} seems to be already installed!",
|
|
||||||
f"If you continue, your current {name} installation will be overwritten.",
|
|
||||||
],
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def print_client_port_select_dialog(
|
|
||||||
name: str, port: int, ports_in_use: List[int]
|
|
||||||
) -> None:
|
|
||||||
dialog_content: List[str] = [
|
|
||||||
f"Please select the port, {name} should be served on. If your are unsure "
|
|
||||||
f"what to select, hit Enter to apply the suggested value of: {port}",
|
|
||||||
"\n\n",
|
|
||||||
f"In case you need {name} to be served on a specific port, you can set it "
|
|
||||||
f"now. Make sure that the port is not already used by another application "
|
|
||||||
f"on your system!",
|
|
||||||
]
|
|
||||||
|
|
||||||
if ports_in_use:
|
|
||||||
dialog_content.extend(
|
|
||||||
[
|
|
||||||
"\n\n",
|
|
||||||
"The following ports were found to be in use already:",
|
|
||||||
*[f"● {port}" for port in ports_in_use],
|
|
||||||
]
|
|
||||||
)
|
|
||||||
|
|
||||||
Logger.print_dialog(DialogType.CUSTOM, dialog_content)
|
|
||||||
|
|
||||||
|
|
||||||
def print_install_client_config_dialog(client: BaseWebClient) -> None:
|
|
||||||
name = client.display_name
|
|
||||||
url = client.client_config.repo_url.replace(".git", "")
|
|
||||||
Logger.print_dialog(
|
|
||||||
DialogType.INFO,
|
|
||||||
[
|
|
||||||
f"It is recommended to use special macros in order to have {name} fully "
|
|
||||||
f"functional and working.",
|
|
||||||
"\n\n",
|
|
||||||
f"The recommended macros for {name} can be seen here:",
|
|
||||||
url,
|
|
||||||
"\n\n",
|
|
||||||
"If you already use these macros skip this step. Otherwise you should "
|
|
||||||
"consider to answer with 'Y' to download the recommended macros.",
|
|
||||||
],
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def print_ipv6_warning_dialog() -> None:
|
|
||||||
Logger.print_dialog(
|
|
||||||
DialogType.WARNING,
|
|
||||||
[
|
|
||||||
"It looks like IPv6 is enabled on this system!",
|
|
||||||
"This may cause issues with the installation of NGINX in the following "
|
|
||||||
"steps! It is recommended to disable IPv6 on your system to avoid this issue.",
|
|
||||||
"\n\n",
|
|
||||||
"If you think this warning is a false alarm, and you are sure that "
|
|
||||||
"IPv6 is disabled, you can continue with the installation.",
|
|
||||||
],
|
|
||||||
)
|
|
||||||
@@ -1,85 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from typing import List
|
|
||||||
|
|
||||||
from components.klipper.klipper import Klipper
|
|
||||||
from components.moonraker.moonraker import Moonraker
|
|
||||||
from components.webui_client.base_data import (
|
|
||||||
BaseWebClient,
|
|
||||||
)
|
|
||||||
from components.webui_client.client_config.client_config_remove import (
|
|
||||||
run_client_config_removal,
|
|
||||||
)
|
|
||||||
from core.backup_manager.backup_manager import BackupManager
|
|
||||||
from core.constants import NGINX_SITES_AVAILABLE, NGINX_SITES_ENABLED
|
|
||||||
from core.logger import Logger
|
|
||||||
from utils.config_utils import remove_config_section
|
|
||||||
from utils.fs_utils import (
|
|
||||||
remove_with_sudo,
|
|
||||||
run_remove_routines,
|
|
||||||
)
|
|
||||||
from utils.instance_utils import get_instances
|
|
||||||
|
|
||||||
|
|
||||||
def run_client_removal(
|
|
||||||
client: BaseWebClient,
|
|
||||||
remove_client: bool,
|
|
||||||
remove_client_cfg: bool,
|
|
||||||
backup_config: bool,
|
|
||||||
) -> None:
|
|
||||||
mr_instances: List[Moonraker] = get_instances(Moonraker)
|
|
||||||
kl_instances: List[Klipper] = get_instances(Klipper)
|
|
||||||
|
|
||||||
if backup_config:
|
|
||||||
bm = BackupManager()
|
|
||||||
bm.backup_file(client.config_file)
|
|
||||||
|
|
||||||
if remove_client:
|
|
||||||
client_name = client.name
|
|
||||||
remove_client_dir(client)
|
|
||||||
remove_client_nginx_config(client_name)
|
|
||||||
remove_client_nginx_logs(client, kl_instances)
|
|
||||||
|
|
||||||
section = f"update_manager {client_name}"
|
|
||||||
remove_config_section(section, mr_instances)
|
|
||||||
|
|
||||||
if remove_client_cfg:
|
|
||||||
run_client_config_removal(
|
|
||||||
client.client_config,
|
|
||||||
kl_instances,
|
|
||||||
mr_instances,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def remove_client_dir(client: BaseWebClient) -> None:
|
|
||||||
Logger.print_status(f"Removing {client.display_name} ...")
|
|
||||||
run_remove_routines(client.client_dir)
|
|
||||||
|
|
||||||
|
|
||||||
def remove_client_nginx_config(name: str) -> None:
|
|
||||||
Logger.print_status(f"Removing NGINX config for {name.capitalize()} ...")
|
|
||||||
|
|
||||||
remove_with_sudo(NGINX_SITES_AVAILABLE.joinpath(name))
|
|
||||||
remove_with_sudo(NGINX_SITES_ENABLED.joinpath(name))
|
|
||||||
|
|
||||||
|
|
||||||
def remove_client_nginx_logs(client: BaseWebClient, instances: List[Klipper]) -> None:
|
|
||||||
Logger.print_status(f"Removing NGINX logs for {client.display_name} ...")
|
|
||||||
|
|
||||||
remove_with_sudo(client.nginx_access_log)
|
|
||||||
remove_with_sudo(client.nginx_error_log)
|
|
||||||
|
|
||||||
if not instances:
|
|
||||||
return
|
|
||||||
|
|
||||||
for instance in instances:
|
|
||||||
run_remove_routines(
|
|
||||||
instance.base.log_dir.joinpath(client.nginx_access_log.name)
|
|
||||||
)
|
|
||||||
run_remove_routines(instance.base.log_dir.joinpath(client.nginx_error_log.name))
|
|
||||||
@@ -1,172 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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
|
|
||||||
import tempfile
|
|
||||||
from pathlib import Path
|
|
||||||
from typing import List
|
|
||||||
|
|
||||||
from components.klipper.klipper import Klipper
|
|
||||||
from components.moonraker.moonraker import Moonraker
|
|
||||||
from components.webui_client import MODULE_PATH
|
|
||||||
from components.webui_client.base_data import (
|
|
||||||
BaseWebClient,
|
|
||||||
BaseWebClientConfig,
|
|
||||||
WebClientType,
|
|
||||||
)
|
|
||||||
from components.webui_client.client_config.client_config_setup import (
|
|
||||||
install_client_config,
|
|
||||||
)
|
|
||||||
from components.webui_client.client_dialogs import (
|
|
||||||
print_install_client_config_dialog,
|
|
||||||
print_moonraker_not_found_dialog,
|
|
||||||
)
|
|
||||||
from components.webui_client.client_utils import (
|
|
||||||
copy_common_vars_nginx_cfg,
|
|
||||||
copy_upstream_nginx_cfg,
|
|
||||||
create_nginx_cfg,
|
|
||||||
detect_client_cfg_conflict,
|
|
||||||
enable_mainsail_remotemode,
|
|
||||||
get_client_port_selection,
|
|
||||||
symlink_webui_nginx_log,
|
|
||||||
)
|
|
||||||
from core.instance_manager.instance_manager import InstanceManager
|
|
||||||
from core.logger import Logger
|
|
||||||
from utils.common import check_install_dependencies
|
|
||||||
from utils.config_utils import add_config_section
|
|
||||||
from utils.fs_utils import unzip
|
|
||||||
from utils.input_utils import get_confirm
|
|
||||||
from utils.instance_utils import get_instances
|
|
||||||
from utils.sys_utils import (
|
|
||||||
cmd_sysctl_service,
|
|
||||||
download_file,
|
|
||||||
get_ipv4_addr,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def install_client(client: BaseWebClient) -> None:
|
|
||||||
if client is None:
|
|
||||||
raise ValueError("Missing parameter client_data!")
|
|
||||||
|
|
||||||
if client.client_dir.exists():
|
|
||||||
Logger.print_info(
|
|
||||||
f"{client.display_name} seems to be already installed! Skipped ..."
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
mr_instances: List[Moonraker] = get_instances(Moonraker)
|
|
||||||
|
|
||||||
enable_remotemode = False
|
|
||||||
if not mr_instances:
|
|
||||||
print_moonraker_not_found_dialog(client.display_name)
|
|
||||||
if not get_confirm(f"Continue {client.display_name} installation?"):
|
|
||||||
return
|
|
||||||
|
|
||||||
# if moonraker is not installed or multiple instances
|
|
||||||
# are installed we enable mainsails remote mode
|
|
||||||
if (
|
|
||||||
client.client == WebClientType.MAINSAIL
|
|
||||||
and not mr_instances
|
|
||||||
or len(mr_instances) > 1
|
|
||||||
):
|
|
||||||
enable_remotemode = True
|
|
||||||
|
|
||||||
kl_instances = get_instances(Klipper)
|
|
||||||
install_client_cfg = False
|
|
||||||
client_config: BaseWebClientConfig = client.client_config
|
|
||||||
if (
|
|
||||||
kl_instances
|
|
||||||
and not client_config.config_dir.exists()
|
|
||||||
and not detect_client_cfg_conflict(client)
|
|
||||||
):
|
|
||||||
print_install_client_config_dialog(client)
|
|
||||||
question = f"Download the recommended {client_config.display_name}?"
|
|
||||||
install_client_cfg = get_confirm(question, allow_go_back=False)
|
|
||||||
|
|
||||||
port: int = get_client_port_selection(client)
|
|
||||||
|
|
||||||
check_install_dependencies({"nginx"})
|
|
||||||
|
|
||||||
try:
|
|
||||||
download_client(client)
|
|
||||||
if enable_remotemode and client.client == WebClientType.MAINSAIL:
|
|
||||||
enable_mainsail_remotemode()
|
|
||||||
if mr_instances:
|
|
||||||
add_config_section(
|
|
||||||
section=f"update_manager {client.name}",
|
|
||||||
instances=mr_instances,
|
|
||||||
options=[
|
|
||||||
("type", "web"),
|
|
||||||
("channel", "stable"),
|
|
||||||
("repo", str(client.repo_path)),
|
|
||||||
("path", str(client.client_dir)),
|
|
||||||
],
|
|
||||||
)
|
|
||||||
InstanceManager.restart_all(mr_instances)
|
|
||||||
if install_client_cfg and kl_instances:
|
|
||||||
install_client_config(client)
|
|
||||||
|
|
||||||
copy_upstream_nginx_cfg()
|
|
||||||
copy_common_vars_nginx_cfg()
|
|
||||||
create_nginx_cfg(
|
|
||||||
display_name=client.display_name,
|
|
||||||
cfg_name=client.name,
|
|
||||||
template_src=MODULE_PATH.joinpath("assets/nginx_cfg"),
|
|
||||||
PORT=port,
|
|
||||||
ROOT_DIR=client.client_dir,
|
|
||||||
NAME=client.name,
|
|
||||||
)
|
|
||||||
|
|
||||||
if kl_instances:
|
|
||||||
symlink_webui_nginx_log(client, kl_instances)
|
|
||||||
cmd_sysctl_service("nginx", "restart")
|
|
||||||
|
|
||||||
except Exception as e:
|
|
||||||
Logger.print_error(f"{client.display_name} installation failed!\n{e}")
|
|
||||||
return
|
|
||||||
|
|
||||||
log = f"Open {client.display_name} now on: http://{get_ipv4_addr()}:{port}"
|
|
||||||
Logger.print_ok(f"{client.display_name} installation complete!", start="\n")
|
|
||||||
Logger.print_ok(log, prefix=False, end="\n\n")
|
|
||||||
|
|
||||||
|
|
||||||
def download_client(client: BaseWebClient) -> None:
|
|
||||||
zipfile = f"{client.name.lower()}.zip"
|
|
||||||
target = Path().home().joinpath(zipfile)
|
|
||||||
try:
|
|
||||||
Logger.print_status(
|
|
||||||
f"Downloading {client.display_name} from {client.download_url} ..."
|
|
||||||
)
|
|
||||||
download_file(client.download_url, target, True)
|
|
||||||
Logger.print_ok("Download complete!")
|
|
||||||
|
|
||||||
Logger.print_status(f"Extracting {zipfile} ...")
|
|
||||||
unzip(target, client.client_dir)
|
|
||||||
target.unlink(missing_ok=True)
|
|
||||||
Logger.print_ok("OK!")
|
|
||||||
|
|
||||||
except Exception:
|
|
||||||
Logger.print_error(f"Downloading {client.display_name} failed!")
|
|
||||||
raise
|
|
||||||
|
|
||||||
|
|
||||||
def update_client(client: BaseWebClient) -> None:
|
|
||||||
Logger.print_status(f"Updating {client.display_name} ...")
|
|
||||||
if not client.client_dir.exists():
|
|
||||||
Logger.print_info(
|
|
||||||
f"Unable to update {client.display_name}. Directory does not exist! Skipping ..."
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
with tempfile.NamedTemporaryFile(suffix=".json") as tmp_file:
|
|
||||||
Logger.print_status(
|
|
||||||
f"Creating temporary backup of {client.config_file} as {tmp_file.name} ..."
|
|
||||||
)
|
|
||||||
shutil.copy(client.config_file, tmp_file.name)
|
|
||||||
download_client(client)
|
|
||||||
shutil.copy(tmp_file.name, client.config_file)
|
|
||||||
@@ -1,402 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import json
|
|
||||||
import re
|
|
||||||
import shutil
|
|
||||||
from pathlib import Path
|
|
||||||
from subprocess import PIPE, CalledProcessError, run
|
|
||||||
from typing import List, get_args
|
|
||||||
|
|
||||||
from components.klipper.klipper import Klipper
|
|
||||||
from components.webui_client import MODULE_PATH
|
|
||||||
from components.webui_client.base_data import (
|
|
||||||
BaseWebClient,
|
|
||||||
WebClientType,
|
|
||||||
)
|
|
||||||
from components.webui_client.client_dialogs import print_client_port_select_dialog
|
|
||||||
from components.webui_client.fluidd_data import FluiddData
|
|
||||||
from components.webui_client.mainsail_data import MainsailData
|
|
||||||
from core.backup_manager.backup_manager import BackupManager
|
|
||||||
from core.constants import (
|
|
||||||
COLOR_CYAN,
|
|
||||||
COLOR_YELLOW,
|
|
||||||
NGINX_CONFD,
|
|
||||||
NGINX_SITES_AVAILABLE,
|
|
||||||
NGINX_SITES_ENABLED,
|
|
||||||
RESET_FORMAT,
|
|
||||||
)
|
|
||||||
from core.logger import Logger
|
|
||||||
from core.settings.kiauh_settings import KiauhSettings, WebUiSettings
|
|
||||||
from core.submodules.simple_config_parser.src.simple_config_parser.simple_config_parser import (
|
|
||||||
SimpleConfigParser,
|
|
||||||
)
|
|
||||||
from core.types import ComponentStatus
|
|
||||||
from utils.common import get_install_status
|
|
||||||
from utils.fs_utils import create_symlink, remove_file
|
|
||||||
from utils.git_utils import (
|
|
||||||
get_latest_remote_tag,
|
|
||||||
get_latest_unstable_tag,
|
|
||||||
)
|
|
||||||
from utils.input_utils import get_number_input
|
|
||||||
from utils.instance_utils import get_instances
|
|
||||||
|
|
||||||
|
|
||||||
def get_client_status(
|
|
||||||
client: BaseWebClient, fetch_remote: bool = False
|
|
||||||
) -> ComponentStatus:
|
|
||||||
files = [
|
|
||||||
NGINX_SITES_AVAILABLE.joinpath(client.name),
|
|
||||||
NGINX_CONFD.joinpath("upstreams.conf"),
|
|
||||||
NGINX_CONFD.joinpath("common_vars.conf"),
|
|
||||||
]
|
|
||||||
comp_status: ComponentStatus = get_install_status(client.client_dir, files=files)
|
|
||||||
|
|
||||||
# if the client dir does not exist, set the status to not
|
|
||||||
# installed even if the other files are present
|
|
||||||
if not client.client_dir.exists():
|
|
||||||
comp_status.status = 0
|
|
||||||
|
|
||||||
comp_status.local = get_local_client_version(client)
|
|
||||||
comp_status.remote = get_remote_client_version(client) if fetch_remote else None
|
|
||||||
return comp_status
|
|
||||||
|
|
||||||
|
|
||||||
def get_client_config_status(client: BaseWebClient) -> ComponentStatus:
|
|
||||||
return get_install_status(client.client_config.config_dir)
|
|
||||||
|
|
||||||
|
|
||||||
def get_current_client_config() -> str:
|
|
||||||
mainsail, fluidd = MainsailData(), FluiddData()
|
|
||||||
clients: List[BaseWebClient] = [mainsail, fluidd]
|
|
||||||
installed = [c for c in clients if c.client_config.config_dir.exists()]
|
|
||||||
|
|
||||||
if not installed:
|
|
||||||
return f"{COLOR_CYAN}-{RESET_FORMAT}"
|
|
||||||
elif len(installed) == 1:
|
|
||||||
cfg = installed[0].client_config
|
|
||||||
return f"{COLOR_CYAN}{cfg.display_name}{RESET_FORMAT}"
|
|
||||||
|
|
||||||
# at this point, both client config folders exists, so we need to check
|
|
||||||
# which are actually included in the printer.cfg of all klipper instances
|
|
||||||
mainsail_includes, fluidd_includes = [], []
|
|
||||||
klipper_instances: List[Klipper] = get_instances(Klipper)
|
|
||||||
for instance in klipper_instances:
|
|
||||||
scp = SimpleConfigParser()
|
|
||||||
scp.read_file(instance.cfg_file)
|
|
||||||
includes_mainsail = scp.has_section(mainsail.client_config.config_section)
|
|
||||||
includes_fluidd = scp.has_section(fluidd.client_config.config_section)
|
|
||||||
|
|
||||||
if includes_mainsail:
|
|
||||||
mainsail_includes.append(instance)
|
|
||||||
if includes_fluidd:
|
|
||||||
fluidd_includes.append(instance)
|
|
||||||
|
|
||||||
# if both are included in the same file, we have a potential conflict
|
|
||||||
if includes_mainsail and includes_fluidd:
|
|
||||||
return f"{COLOR_YELLOW}Conflict!{RESET_FORMAT}"
|
|
||||||
|
|
||||||
if not mainsail_includes and not fluidd_includes:
|
|
||||||
# there are no includes at all, even though the client config folders exist
|
|
||||||
return f"{COLOR_CYAN}-{RESET_FORMAT}"
|
|
||||||
elif len(fluidd_includes) > len(mainsail_includes):
|
|
||||||
# there are more instances that include fluidd than mainsail
|
|
||||||
return f"{COLOR_CYAN}{fluidd.client_config.display_name}{RESET_FORMAT}"
|
|
||||||
else:
|
|
||||||
# there are the same amount of non-conflicting includes for each config
|
|
||||||
# or more instances include mainsail than fluidd
|
|
||||||
return f"{COLOR_CYAN}{mainsail.client_config.display_name}{RESET_FORMAT}"
|
|
||||||
|
|
||||||
|
|
||||||
def enable_mainsail_remotemode() -> None:
|
|
||||||
Logger.print_status("Enable Mainsails remote mode ...")
|
|
||||||
c_json = MainsailData().client_dir.joinpath("config.json")
|
|
||||||
with open(c_json, "r") as f:
|
|
||||||
config_data = json.load(f)
|
|
||||||
|
|
||||||
if config_data["instancesDB"] == "browser":
|
|
||||||
Logger.print_info("Remote mode already configured. Skipped ...")
|
|
||||||
return
|
|
||||||
|
|
||||||
Logger.print_status("Setting instance storage location to 'browser' ...")
|
|
||||||
config_data["instancesDB"] = "browser"
|
|
||||||
|
|
||||||
with open(c_json, "w") as f:
|
|
||||||
json.dump(config_data, f, indent=4)
|
|
||||||
Logger.print_ok("Mainsails remote mode enabled!")
|
|
||||||
|
|
||||||
|
|
||||||
def symlink_webui_nginx_log(
|
|
||||||
client: BaseWebClient, klipper_instances: List[Klipper]
|
|
||||||
) -> None:
|
|
||||||
Logger.print_status("Link NGINX logs into log directory ...")
|
|
||||||
access_log = client.nginx_access_log
|
|
||||||
error_log = client.nginx_error_log
|
|
||||||
|
|
||||||
for instance in klipper_instances:
|
|
||||||
desti_access = instance.base.log_dir.joinpath(access_log.name)
|
|
||||||
if not desti_access.exists():
|
|
||||||
desti_access.symlink_to(access_log)
|
|
||||||
|
|
||||||
desti_error = instance.base.log_dir.joinpath(error_log.name)
|
|
||||||
if not desti_error.exists():
|
|
||||||
desti_error.symlink_to(error_log)
|
|
||||||
|
|
||||||
|
|
||||||
def get_local_client_version(client: BaseWebClient) -> str | None:
|
|
||||||
relinfo_file = client.client_dir.joinpath("release_info.json")
|
|
||||||
version_file = client.client_dir.joinpath(".version")
|
|
||||||
|
|
||||||
if not client.client_dir.exists():
|
|
||||||
return None
|
|
||||||
if not relinfo_file.is_file() and not version_file.is_file():
|
|
||||||
return "n/a"
|
|
||||||
|
|
||||||
if relinfo_file.is_file():
|
|
||||||
with open(relinfo_file, "r") as f:
|
|
||||||
return str(json.load(f)["version"])
|
|
||||||
else:
|
|
||||||
with open(version_file, "r") as f:
|
|
||||||
return f.readlines()[0]
|
|
||||||
|
|
||||||
|
|
||||||
def get_remote_client_version(client: BaseWebClient) -> str | None:
|
|
||||||
try:
|
|
||||||
if (tag := get_latest_remote_tag(client.repo_path)) != "":
|
|
||||||
return str(tag)
|
|
||||||
return None
|
|
||||||
except Exception:
|
|
||||||
return None
|
|
||||||
|
|
||||||
|
|
||||||
def backup_client_data(client: BaseWebClient) -> None:
|
|
||||||
name = client.name
|
|
||||||
src = client.client_dir
|
|
||||||
dest = client.backup_dir
|
|
||||||
|
|
||||||
with open(src.joinpath(".version"), "r") as v:
|
|
||||||
version = v.readlines()[0]
|
|
||||||
|
|
||||||
bm = BackupManager()
|
|
||||||
bm.backup_directory(f"{name}-{version}", src, dest)
|
|
||||||
bm.backup_file(client.config_file, dest)
|
|
||||||
bm.backup_file(NGINX_SITES_AVAILABLE.joinpath(name), dest)
|
|
||||||
|
|
||||||
|
|
||||||
def backup_client_config_data(client: BaseWebClient) -> None:
|
|
||||||
client_config = client.client_config
|
|
||||||
name = client_config.name
|
|
||||||
source = client_config.config_dir
|
|
||||||
target = client_config.backup_dir
|
|
||||||
bm = BackupManager()
|
|
||||||
bm.backup_directory(name, source, target)
|
|
||||||
|
|
||||||
|
|
||||||
def get_existing_clients() -> List[BaseWebClient]:
|
|
||||||
clients = list(get_args(WebClientType))
|
|
||||||
installed_clients: List[BaseWebClient] = []
|
|
||||||
for client in clients:
|
|
||||||
if client.client_dir.exists():
|
|
||||||
installed_clients.append(client)
|
|
||||||
|
|
||||||
return installed_clients
|
|
||||||
|
|
||||||
|
|
||||||
def detect_client_cfg_conflict(curr_client: BaseWebClient) -> bool:
|
|
||||||
"""
|
|
||||||
Check if any other client configs are present on the system.
|
|
||||||
It is usually not harmful, but chances are they can conflict each other.
|
|
||||||
Multiple client configs are, at least, redundant to have them installed
|
|
||||||
:param curr_client: The client name to check for the conflict
|
|
||||||
:return: True, if other client configs were found, else False
|
|
||||||
"""
|
|
||||||
|
|
||||||
mainsail_cfg_status: ComponentStatus = get_client_config_status(MainsailData())
|
|
||||||
fluidd_cfg_status: ComponentStatus = get_client_config_status(FluiddData())
|
|
||||||
|
|
||||||
if curr_client.client == WebClientType.MAINSAIL and fluidd_cfg_status.status == 2:
|
|
||||||
return True
|
|
||||||
if curr_client.client == WebClientType.FLUIDD and mainsail_cfg_status.status == 2:
|
|
||||||
return True
|
|
||||||
|
|
||||||
return False
|
|
||||||
|
|
||||||
|
|
||||||
def get_download_url(base_url: str, client: BaseWebClient) -> str:
|
|
||||||
settings = KiauhSettings()
|
|
||||||
use_unstable = settings.get(client.name, "unstable_releases")
|
|
||||||
stable_url = f"{base_url}/latest/download/{client.name}.zip"
|
|
||||||
|
|
||||||
if not use_unstable:
|
|
||||||
return stable_url
|
|
||||||
|
|
||||||
try:
|
|
||||||
unstable_tag = get_latest_unstable_tag(client.repo_path)
|
|
||||||
if unstable_tag == "":
|
|
||||||
raise Exception
|
|
||||||
return f"{base_url}/download/{unstable_tag}/{client.name}.zip"
|
|
||||||
except Exception:
|
|
||||||
return stable_url
|
|
||||||
|
|
||||||
|
|
||||||
#################################################
|
|
||||||
## NGINX RELATED FUNCTIONS
|
|
||||||
#################################################
|
|
||||||
|
|
||||||
|
|
||||||
def copy_upstream_nginx_cfg() -> None:
|
|
||||||
"""
|
|
||||||
Creates an upstream.conf in /etc/nginx/conf.d
|
|
||||||
:return: None
|
|
||||||
"""
|
|
||||||
source = MODULE_PATH.joinpath("assets/upstreams.conf")
|
|
||||||
target = NGINX_CONFD.joinpath("upstreams.conf")
|
|
||||||
try:
|
|
||||||
command = ["sudo", "cp", source, target]
|
|
||||||
run(command, stderr=PIPE, check=True)
|
|
||||||
except CalledProcessError as e:
|
|
||||||
log = f"Unable to create upstreams.conf: {e.stderr.decode()}"
|
|
||||||
Logger.print_error(log)
|
|
||||||
raise
|
|
||||||
|
|
||||||
|
|
||||||
def copy_common_vars_nginx_cfg() -> None:
|
|
||||||
"""
|
|
||||||
Creates a common_vars.conf in /etc/nginx/conf.d
|
|
||||||
:return: None
|
|
||||||
"""
|
|
||||||
source = MODULE_PATH.joinpath("assets/common_vars.conf")
|
|
||||||
target = NGINX_CONFD.joinpath("common_vars.conf")
|
|
||||||
try:
|
|
||||||
command = ["sudo", "cp", source, target]
|
|
||||||
run(command, stderr=PIPE, check=True)
|
|
||||||
except CalledProcessError as e:
|
|
||||||
log = f"Unable to create upstreams.conf: {e.stderr.decode()}"
|
|
||||||
Logger.print_error(log)
|
|
||||||
raise
|
|
||||||
|
|
||||||
|
|
||||||
def generate_nginx_cfg_from_template(name: str, template_src: Path, **kwargs) -> None:
|
|
||||||
"""
|
|
||||||
Creates an NGINX config from a template file and
|
|
||||||
replaces all placeholders passed as kwargs. A placeholder must be defined
|
|
||||||
in the template file as %{placeholder}%.
|
|
||||||
:param name: name of the config to create
|
|
||||||
:param template_src: the path to the template file
|
|
||||||
:return: None
|
|
||||||
"""
|
|
||||||
tmp = Path.home().joinpath(f"{name}.tmp")
|
|
||||||
shutil.copy(template_src, tmp)
|
|
||||||
with open(tmp, "r+") as f:
|
|
||||||
content = f.read()
|
|
||||||
|
|
||||||
for key, value in kwargs.items():
|
|
||||||
content = content.replace(f"%{key}%", str(value))
|
|
||||||
|
|
||||||
f.seek(0)
|
|
||||||
f.write(content)
|
|
||||||
f.truncate()
|
|
||||||
|
|
||||||
target = NGINX_SITES_AVAILABLE.joinpath(name)
|
|
||||||
try:
|
|
||||||
command = ["sudo", "mv", tmp, target]
|
|
||||||
run(command, stderr=PIPE, check=True)
|
|
||||||
except CalledProcessError as e:
|
|
||||||
log = f"Unable to create '{target}': {e.stderr.decode()}"
|
|
||||||
Logger.print_error(log)
|
|
||||||
raise
|
|
||||||
|
|
||||||
|
|
||||||
def create_nginx_cfg(
|
|
||||||
display_name: str,
|
|
||||||
cfg_name: str,
|
|
||||||
template_src: Path,
|
|
||||||
**kwargs,
|
|
||||||
) -> None:
|
|
||||||
from utils.sys_utils import set_nginx_permissions
|
|
||||||
|
|
||||||
try:
|
|
||||||
Logger.print_status(f"Creating NGINX config for {display_name} ...")
|
|
||||||
|
|
||||||
source = NGINX_SITES_AVAILABLE.joinpath(cfg_name)
|
|
||||||
target = NGINX_SITES_ENABLED.joinpath(cfg_name)
|
|
||||||
remove_file(Path("/etc/nginx/sites-enabled/default"), True)
|
|
||||||
generate_nginx_cfg_from_template(cfg_name, template_src=template_src, **kwargs)
|
|
||||||
create_symlink(source, target, True)
|
|
||||||
set_nginx_permissions()
|
|
||||||
|
|
||||||
Logger.print_ok(f"NGINX config for {display_name} successfully created.")
|
|
||||||
except Exception:
|
|
||||||
Logger.print_error(f"Creating NGINX config for {display_name} failed!")
|
|
||||||
raise
|
|
||||||
|
|
||||||
|
|
||||||
def read_ports_from_nginx_configs() -> List[int]:
|
|
||||||
"""
|
|
||||||
Helper function to iterate over all NGINX configs and read all ports defined for listen
|
|
||||||
:return: A sorted list of listen ports
|
|
||||||
"""
|
|
||||||
if not NGINX_SITES_ENABLED.exists():
|
|
||||||
return []
|
|
||||||
|
|
||||||
port_list = []
|
|
||||||
for config in NGINX_SITES_ENABLED.iterdir():
|
|
||||||
if not config.is_file():
|
|
||||||
continue
|
|
||||||
|
|
||||||
with open(config, "r") as cfg:
|
|
||||||
lines = cfg.readlines()
|
|
||||||
|
|
||||||
for line in lines:
|
|
||||||
line = re.sub(
|
|
||||||
r"default_server|http://|https://|[;\[\]]",
|
|
||||||
"",
|
|
||||||
line.strip(),
|
|
||||||
)
|
|
||||||
if line.startswith("listen"):
|
|
||||||
if ":" not in line:
|
|
||||||
port_list.append(line.split()[-1])
|
|
||||||
else:
|
|
||||||
port_list.append(line.split(":")[-1])
|
|
||||||
|
|
||||||
ports_to_ints_list = [int(port) for port in port_list]
|
|
||||||
return sorted(ports_to_ints_list, key=lambda x: int(x))
|
|
||||||
|
|
||||||
|
|
||||||
def get_client_port_selection(client: BaseWebClient) -> int:
|
|
||||||
settings = KiauhSettings()
|
|
||||||
default_port: int = int(settings.get(client.name, "port"))
|
|
||||||
|
|
||||||
ports_in_use: List[int] = read_ports_from_nginx_configs()
|
|
||||||
next_free_port: int = get_next_free_port(ports_in_use)
|
|
||||||
|
|
||||||
port: int = next_free_port if default_port in ports_in_use else default_port
|
|
||||||
|
|
||||||
print_client_port_select_dialog(client.display_name, port, ports_in_use)
|
|
||||||
|
|
||||||
while True:
|
|
||||||
question = f"Configure {client.display_name} for port"
|
|
||||||
port_input = get_number_input(question, min_count=80, default=port)
|
|
||||||
|
|
||||||
if port_input not in ports_in_use:
|
|
||||||
client_settings: WebUiSettings = settings[client.name]
|
|
||||||
client_settings.port = port_input
|
|
||||||
settings.save()
|
|
||||||
|
|
||||||
return port_input
|
|
||||||
|
|
||||||
Logger.print_error("This port is already in use. Please select another one.")
|
|
||||||
|
|
||||||
|
|
||||||
def get_next_free_port(ports_in_use: List[int]) -> int:
|
|
||||||
valid_ports = set(range(80, 7125))
|
|
||||||
used_ports = set(map(int, ports_in_use))
|
|
||||||
|
|
||||||
return min(valid_ports - used_ports)
|
|
||||||
@@ -1,56 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
from dataclasses import dataclass
|
|
||||||
from pathlib import Path
|
|
||||||
|
|
||||||
from components.webui_client.base_data import (
|
|
||||||
BaseWebClient,
|
|
||||||
BaseWebClientConfig,
|
|
||||||
WebClientConfigType,
|
|
||||||
WebClientType,
|
|
||||||
)
|
|
||||||
from core.backup_manager import BACKUP_ROOT_DIR
|
|
||||||
|
|
||||||
|
|
||||||
@dataclass()
|
|
||||||
class FluiddConfigWeb(BaseWebClientConfig):
|
|
||||||
client_config: WebClientConfigType = WebClientConfigType.FLUIDD
|
|
||||||
name: str = client_config.value
|
|
||||||
display_name: str = name.title()
|
|
||||||
config_dir: Path = Path.home().joinpath("fluidd-config")
|
|
||||||
config_filename: str = "fluidd.cfg"
|
|
||||||
config_section: str = f"include {config_filename}"
|
|
||||||
backup_dir: Path = BACKUP_ROOT_DIR.joinpath("fluidd-config-backups")
|
|
||||||
repo_url: str = "https://github.com/fluidd-core/fluidd-config.git"
|
|
||||||
|
|
||||||
|
|
||||||
@dataclass()
|
|
||||||
class FluiddData(BaseWebClient):
|
|
||||||
BASE_DL_URL = "https://github.com/fluidd-core/fluidd/releases"
|
|
||||||
|
|
||||||
client: WebClientType = WebClientType.FLUIDD
|
|
||||||
name: str = client.value
|
|
||||||
display_name: str = name.capitalize()
|
|
||||||
client_dir: Path = Path.home().joinpath("fluidd")
|
|
||||||
config_file: Path = client_dir.joinpath("config.json")
|
|
||||||
backup_dir: Path = BACKUP_ROOT_DIR.joinpath("fluidd-backups")
|
|
||||||
repo_path: str = "fluidd-core/fluidd"
|
|
||||||
nginx_access_log: Path = Path("/var/log/nginx/fluidd-access.log")
|
|
||||||
nginx_error_log: Path = Path("/var/log/nginx/fluidd-error.log")
|
|
||||||
client_config: BaseWebClientConfig = None
|
|
||||||
download_url: str | None = None
|
|
||||||
|
|
||||||
def __post_init__(self):
|
|
||||||
from components.webui_client.client_utils import get_download_url
|
|
||||||
|
|
||||||
self.client_config = FluiddConfigWeb()
|
|
||||||
self.download_url = get_download_url(self.BASE_DL_URL, self)
|
|
||||||
@@ -1,56 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
from dataclasses import dataclass
|
|
||||||
from pathlib import Path
|
|
||||||
|
|
||||||
from components.webui_client.base_data import (
|
|
||||||
BaseWebClient,
|
|
||||||
BaseWebClientConfig,
|
|
||||||
WebClientConfigType,
|
|
||||||
WebClientType,
|
|
||||||
)
|
|
||||||
from core.backup_manager import BACKUP_ROOT_DIR
|
|
||||||
|
|
||||||
|
|
||||||
@dataclass()
|
|
||||||
class MainsailConfigWeb(BaseWebClientConfig):
|
|
||||||
client_config: WebClientConfigType = WebClientConfigType.MAINSAIL
|
|
||||||
name: str = client_config.value
|
|
||||||
display_name: str = name.title()
|
|
||||||
config_dir: Path = Path.home().joinpath("mainsail-config")
|
|
||||||
config_filename: str = "mainsail.cfg"
|
|
||||||
config_section: str = f"include {config_filename}"
|
|
||||||
backup_dir: Path = BACKUP_ROOT_DIR.joinpath("mainsail-config-backups")
|
|
||||||
repo_url: str = "https://github.com/mainsail-crew/mainsail-config.git"
|
|
||||||
|
|
||||||
|
|
||||||
@dataclass()
|
|
||||||
class MainsailData(BaseWebClient):
|
|
||||||
BASE_DL_URL: str = "https://github.com/mainsail-crew/mainsail/releases"
|
|
||||||
|
|
||||||
client: WebClientType = WebClientType.MAINSAIL
|
|
||||||
name: str = WebClientType.MAINSAIL.value
|
|
||||||
display_name: str = name.capitalize()
|
|
||||||
client_dir: Path = Path.home().joinpath("mainsail")
|
|
||||||
config_file: Path = client_dir.joinpath("config.json")
|
|
||||||
backup_dir: Path = BACKUP_ROOT_DIR.joinpath("mainsail-backups")
|
|
||||||
repo_path: str = "mainsail-crew/mainsail"
|
|
||||||
nginx_access_log: Path = Path("/var/log/nginx/mainsail-access.log")
|
|
||||||
nginx_error_log: Path = Path("/var/log/nginx/mainsail-error.log")
|
|
||||||
client_config: BaseWebClientConfig = None
|
|
||||||
download_url: str | None = None
|
|
||||||
|
|
||||||
def __post_init__(self):
|
|
||||||
from components.webui_client.client_utils import get_download_url
|
|
||||||
|
|
||||||
self.client_config = MainsailConfigWeb()
|
|
||||||
self.download_url = get_download_url(self.BASE_DL_URL, self)
|
|
||||||
@@ -1,126 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import textwrap
|
|
||||||
from typing import Type
|
|
||||||
|
|
||||||
from components.webui_client import client_remove
|
|
||||||
from components.webui_client.base_data import BaseWebClient
|
|
||||||
from core.constants import COLOR_CYAN, COLOR_RED, RESET_FORMAT
|
|
||||||
from core.menus import Option
|
|
||||||
from core.menus.base_menu import BaseMenu
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyUnusedLocal
|
|
||||||
class ClientRemoveMenu(BaseMenu):
|
|
||||||
def __init__(
|
|
||||||
self, client: BaseWebClient, previous_menu: Type[BaseMenu] | None = None
|
|
||||||
):
|
|
||||||
super().__init__()
|
|
||||||
self.previous_menu: Type[BaseMenu] | None = previous_menu
|
|
||||||
self.client: BaseWebClient = client
|
|
||||||
self.remove_client: bool = False
|
|
||||||
self.remove_client_cfg: bool = False
|
|
||||||
self.backup_config_json: bool = False
|
|
||||||
self.selection_state: bool = False
|
|
||||||
|
|
||||||
def set_previous_menu(self, previous_menu: Type[BaseMenu] | None) -> None:
|
|
||||||
from core.menus.remove_menu import RemoveMenu
|
|
||||||
|
|
||||||
self.previous_menu = previous_menu if previous_menu is not None else RemoveMenu
|
|
||||||
|
|
||||||
def set_options(self) -> None:
|
|
||||||
self.options = {
|
|
||||||
"a": Option(method=self.toggle_all),
|
|
||||||
"1": Option(method=self.toggle_rm_client),
|
|
||||||
"2": Option(method=self.toggle_rm_client_config),
|
|
||||||
"3": Option(method=self.toggle_backup_config_json),
|
|
||||||
"c": Option(method=self.run_removal_process),
|
|
||||||
}
|
|
||||||
|
|
||||||
def print_menu(self) -> None:
|
|
||||||
client_name = self.client.display_name
|
|
||||||
client_config = self.client.client_config
|
|
||||||
client_config_name = client_config.display_name
|
|
||||||
|
|
||||||
header = f" [ Remove {client_name} ] "
|
|
||||||
color = COLOR_RED
|
|
||||||
count = 62 - len(color) - len(RESET_FORMAT)
|
|
||||||
checked = f"[{COLOR_CYAN}x{RESET_FORMAT}]"
|
|
||||||
unchecked = "[ ]"
|
|
||||||
o1 = checked if self.remove_client else unchecked
|
|
||||||
o2 = checked if self.remove_client_cfg else unchecked
|
|
||||||
o3 = checked if self.backup_config_json else unchecked
|
|
||||||
menu = textwrap.dedent(
|
|
||||||
f"""
|
|
||||||
╔═══════════════════════════════════════════════════════╗
|
|
||||||
║ {color}{header:~^{count}}{RESET_FORMAT} ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ Enter a number and hit enter to select / deselect ║
|
|
||||||
║ the specific option for removal. ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ a) {self._get_selection_state_str():37} ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ 1) {o1} Remove {client_name:16} ║
|
|
||||||
║ 2) {o2} Remove {client_config_name:24} ║
|
|
||||||
║ 3) {o3} Backup config.json ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ C) Continue ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
"""
|
|
||||||
)[1:]
|
|
||||||
print(menu, end="")
|
|
||||||
|
|
||||||
def toggle_all(self, **kwargs) -> None:
|
|
||||||
self.selection_state = not self.selection_state
|
|
||||||
self.remove_client = self.selection_state
|
|
||||||
self.remove_client_cfg = self.selection_state
|
|
||||||
self.backup_config_json = self.selection_state
|
|
||||||
|
|
||||||
def toggle_rm_client(self, **kwargs) -> None:
|
|
||||||
self.remove_client = not self.remove_client
|
|
||||||
|
|
||||||
def toggle_rm_client_config(self, **kwargs) -> None:
|
|
||||||
self.remove_client_cfg = not self.remove_client_cfg
|
|
||||||
|
|
||||||
def toggle_backup_config_json(self, **kwargs) -> None:
|
|
||||||
self.backup_config_json = not self.backup_config_json
|
|
||||||
|
|
||||||
def run_removal_process(self, **kwargs) -> None:
|
|
||||||
if (
|
|
||||||
not self.remove_client
|
|
||||||
and not self.remove_client_cfg
|
|
||||||
and not self.backup_config_json
|
|
||||||
):
|
|
||||||
error = f"{COLOR_RED}Nothing selected ...{RESET_FORMAT}"
|
|
||||||
print(error)
|
|
||||||
return
|
|
||||||
|
|
||||||
client_remove.run_client_removal(
|
|
||||||
client=self.client,
|
|
||||||
remove_client=self.remove_client,
|
|
||||||
remove_client_cfg=self.remove_client_cfg,
|
|
||||||
backup_config=self.backup_config_json,
|
|
||||||
)
|
|
||||||
|
|
||||||
self.remove_client = False
|
|
||||||
self.remove_client_cfg = False
|
|
||||||
self.backup_config_json = False
|
|
||||||
|
|
||||||
self._go_back()
|
|
||||||
|
|
||||||
def _get_selection_state_str(self) -> str:
|
|
||||||
return (
|
|
||||||
"Select everything" if not self.selection_state else "Deselect everything"
|
|
||||||
)
|
|
||||||
|
|
||||||
def _go_back(self, **kwargs) -> None:
|
|
||||||
if self.previous_menu is not None:
|
|
||||||
self.previous_menu().run()
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
|
|
||||||
from pathlib import Path
|
|
||||||
|
|
||||||
BACKUP_ROOT_DIR = Path.home().joinpath("kiauh-backups")
|
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
# ======================================================================= #
|
# ======================================================================= #
|
||||||
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
||||||
# #
|
# #
|
||||||
@@ -6,27 +8,20 @@
|
|||||||
# #
|
# #
|
||||||
# This file may be distributed under the terms of the GNU GPLv3 license #
|
# This file may be distributed under the terms of the GNU GPLv3 license #
|
||||||
# ======================================================================= #
|
# ======================================================================= #
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import shutil
|
import shutil
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from typing import List
|
from typing import List
|
||||||
|
|
||||||
from core.backup_manager import BACKUP_ROOT_DIR
|
from kiauh import KIAUH_BACKUP_DIR
|
||||||
from core.logger import Logger
|
from kiauh.utils.common import get_current_date
|
||||||
from utils.common import get_current_date
|
from kiauh.utils.logger import Logger
|
||||||
|
|
||||||
|
|
||||||
class BackupManagerException(Exception):
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyUnusedLocal
|
|
||||||
# noinspection PyMethodMayBeStatic
|
# noinspection PyMethodMayBeStatic
|
||||||
class BackupManager:
|
class BackupManager:
|
||||||
def __init__(self, backup_root_dir: Path = BACKUP_ROOT_DIR):
|
def __init__(self, backup_root_dir: Path = KIAUH_BACKUP_DIR):
|
||||||
self._backup_root_dir: Path = backup_root_dir
|
self._backup_root_dir = backup_root_dir
|
||||||
self._ignore_folders: List[str] = []
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def backup_root_dir(self) -> Path:
|
def backup_root_dir(self) -> Path:
|
||||||
@@ -36,23 +31,15 @@ class BackupManager:
|
|||||||
def backup_root_dir(self, value: Path):
|
def backup_root_dir(self, value: Path):
|
||||||
self._backup_root_dir = value
|
self._backup_root_dir = value
|
||||||
|
|
||||||
@property
|
def backup_file(
|
||||||
def ignore_folders(self) -> List[str]:
|
self, files: List[Path] = None, target: Path = None, custom_filename=None
|
||||||
return self._ignore_folders
|
):
|
||||||
|
if not files:
|
||||||
@ignore_folders.setter
|
raise ValueError("Parameter 'files' cannot be None or an empty List!")
|
||||||
def ignore_folders(self, value: List[str]):
|
|
||||||
self._ignore_folders = value
|
|
||||||
|
|
||||||
def backup_file(self, file: Path, target: Path | None = None, custom_filename=None):
|
|
||||||
Logger.print_status(f"Creating backup of {file} ...")
|
|
||||||
|
|
||||||
if not file.exists():
|
|
||||||
Logger.print_info("File does not exist! Skipping ...")
|
|
||||||
return
|
|
||||||
|
|
||||||
target = self.backup_root_dir if target is None else target
|
target = self.backup_root_dir if target is None else target
|
||||||
|
for file in files:
|
||||||
|
Logger.print_status(f"Creating backup of {file} ...")
|
||||||
if Path(file).is_file():
|
if Path(file).is_file():
|
||||||
date = get_current_date().get("date")
|
date = get_current_date().get("date")
|
||||||
time = get_current_date().get("time")
|
time = get_current_date().get("time")
|
||||||
@@ -61,38 +48,25 @@ class BackupManager:
|
|||||||
try:
|
try:
|
||||||
Path(target).mkdir(exist_ok=True)
|
Path(target).mkdir(exist_ok=True)
|
||||||
shutil.copyfile(file, target.joinpath(filename))
|
shutil.copyfile(file, target.joinpath(filename))
|
||||||
Logger.print_ok("Backup successful!")
|
|
||||||
except OSError as e:
|
except OSError as e:
|
||||||
Logger.print_error(f"Unable to backup '{file}':\n{e}")
|
Logger.print_error(f"Unable to backup '{file}':\n{e}")
|
||||||
|
continue
|
||||||
else:
|
else:
|
||||||
Logger.print_info(f"File '{file}' not found ...")
|
Logger.print_info(f"File '{file}' not found ...")
|
||||||
|
|
||||||
def backup_directory(
|
def backup_directory(self, name: str, source: Path, target: Path = None) -> None:
|
||||||
self, name: str, source: Path, target: Path | None = None
|
|
||||||
) -> Path | None:
|
|
||||||
Logger.print_status(f"Creating backup of {name} in {target} ...")
|
|
||||||
|
|
||||||
if source is None or not Path(source).exists():
|
if source is None or not Path(source).exists():
|
||||||
Logger.print_info("Source directory does not exist! Skipping ...")
|
raise OSError
|
||||||
return
|
|
||||||
|
|
||||||
target = self.backup_root_dir if target is None else target
|
target = self.backup_root_dir if target is None else target
|
||||||
try:
|
try:
|
||||||
|
log = f"Creating backup of {name} in {target} ..."
|
||||||
|
Logger.print_status(log)
|
||||||
date = get_current_date().get("date")
|
date = get_current_date().get("date")
|
||||||
time = get_current_date().get("time")
|
time = get_current_date().get("time")
|
||||||
backup_target = target.joinpath(f"{name.lower()}-{date}-{time}")
|
shutil.copytree(source, target.joinpath(f"{name}-{date}-{time}"))
|
||||||
shutil.copytree(source, backup_target, ignore=self.ignore_folders_func)
|
|
||||||
Logger.print_ok("Backup successful!")
|
|
||||||
|
|
||||||
return backup_target
|
|
||||||
|
|
||||||
except OSError as e:
|
except OSError as e:
|
||||||
Logger.print_error(f"Unable to backup directory '{source}':\n{e}")
|
Logger.print_error(f"Unable to backup directory '{source}':\n{e}")
|
||||||
raise BackupManagerException(f"Unable to backup directory '{source}':\n{e}")
|
return
|
||||||
|
|
||||||
def ignore_folders_func(self, dirpath, filenames) -> List[str]:
|
Logger.print_ok("Backup successfull!")
|
||||||
return (
|
|
||||||
[f for f in filenames if f in self._ignore_folders]
|
|
||||||
if self._ignore_folders
|
|
||||||
else []
|
|
||||||
)
|
|
||||||
|
|||||||
85
kiauh/core/config_manager/config_manager.py
Normal file
85
kiauh/core/config_manager/config_manager.py
Normal file
@@ -0,0 +1,85 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
# ======================================================================= #
|
||||||
|
# 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 configparser
|
||||||
|
from pathlib import Path
|
||||||
|
from typing import Union
|
||||||
|
|
||||||
|
from kiauh.utils.logger import Logger
|
||||||
|
|
||||||
|
|
||||||
|
# noinspection PyMethodMayBeStatic
|
||||||
|
class ConfigManager:
|
||||||
|
def __init__(self, cfg_file: Path):
|
||||||
|
self.config_file = cfg_file
|
||||||
|
self.config = CustomConfigParser()
|
||||||
|
|
||||||
|
if cfg_file.is_file():
|
||||||
|
self.read_config()
|
||||||
|
|
||||||
|
def read_config(self) -> None:
|
||||||
|
if not self.config_file:
|
||||||
|
Logger.print_error("Unable to read config file. File not found.")
|
||||||
|
return
|
||||||
|
|
||||||
|
self.config.read_file(open(self.config_file, "r"))
|
||||||
|
|
||||||
|
def write_config(self) -> None:
|
||||||
|
with open(self.config_file, "w") as cfg:
|
||||||
|
self.config.write(cfg)
|
||||||
|
|
||||||
|
def get_value(self, section: str, key: str, silent=True) -> Union[str, bool, None]:
|
||||||
|
if not self.config.has_section(section):
|
||||||
|
if not silent:
|
||||||
|
log = f"Section not defined. Unable to read section: [{section}]."
|
||||||
|
Logger.print_error(log)
|
||||||
|
return None
|
||||||
|
|
||||||
|
if not self.config.has_option(section, key):
|
||||||
|
if not silent:
|
||||||
|
log = f"Option not defined in section [{section}]. Unable to read option: '{key}'."
|
||||||
|
Logger.print_error(log)
|
||||||
|
return None
|
||||||
|
|
||||||
|
value = self.config.get(section, key)
|
||||||
|
if value == "True" or value == "true":
|
||||||
|
return True
|
||||||
|
elif value == "False" or value == "false":
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
return value
|
||||||
|
|
||||||
|
def set_value(self, section: str, key: str, value: str):
|
||||||
|
self.config.set(section, key, value)
|
||||||
|
|
||||||
|
|
||||||
|
class CustomConfigParser(configparser.ConfigParser):
|
||||||
|
"""
|
||||||
|
A custom ConfigParser class overwriting the write() method of configparser.Configparser.
|
||||||
|
Key and value will be delimited by a ": ".
|
||||||
|
Note the whitespace AFTER the colon, which is the whole reason for that overwrite.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def write(self, fp, space_around_delimiters=False):
|
||||||
|
if self._defaults:
|
||||||
|
fp.write("[%s]\n" % configparser.DEFAULTSECT)
|
||||||
|
for key, value in self._defaults.items():
|
||||||
|
fp.write("%s: %s\n" % (key, str(value).replace("\n", "\n\t")))
|
||||||
|
fp.write("\n")
|
||||||
|
for section in self._sections:
|
||||||
|
fp.write("[%s]\n" % section)
|
||||||
|
for key, value in self._sections[section].items():
|
||||||
|
if key == "__name__":
|
||||||
|
continue
|
||||||
|
if (value is not None) or (self._optcre == self.OPTCRE):
|
||||||
|
key = ": ".join((key, str(value).replace("\n", "\n\t")))
|
||||||
|
fp.write("%s\n" % key)
|
||||||
|
fp.write("\n")
|
||||||
@@ -1,24 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import warnings
|
|
||||||
from typing import Callable
|
|
||||||
|
|
||||||
|
|
||||||
def deprecated(info: str = "", replaced_by: Callable | None = None) -> Callable:
|
|
||||||
def decorator(func) -> Callable:
|
|
||||||
def wrapper(*args, **kwargs):
|
|
||||||
msg = f"{info}{replaced_by.__name__ if replaced_by else ''}"
|
|
||||||
warnings.warn(msg, category=DeprecationWarning, stacklevel=2)
|
|
||||||
return func(*args, **kwargs)
|
|
||||||
|
|
||||||
return wrapper
|
|
||||||
|
|
||||||
return decorator
|
|
||||||
@@ -1,3 +1,5 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
# ======================================================================= #
|
# ======================================================================= #
|
||||||
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
||||||
# #
|
# #
|
||||||
@@ -7,52 +9,153 @@
|
|||||||
# This file may be distributed under the terms of the GNU GPLv3 license #
|
# This file may be distributed under the terms of the GNU GPLv3 license #
|
||||||
# ======================================================================= #
|
# ======================================================================= #
|
||||||
|
|
||||||
from __future__ import annotations
|
from abc import abstractmethod, ABC
|
||||||
|
|
||||||
import re
|
|
||||||
from dataclasses import dataclass, field
|
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from typing import List
|
from typing import List, Type, TypeVar
|
||||||
|
|
||||||
from utils.fs_utils import get_data_dir
|
from kiauh.utils.constants import SYSTEMD, CURRENT_USER
|
||||||
|
|
||||||
SUFFIX_BLACKLIST: List[str] = ["None", "mcu", "obico", "bambu", "companion"]
|
B = TypeVar(name="B", bound="BaseInstance", covariant=True)
|
||||||
|
|
||||||
|
|
||||||
@dataclass(repr=True)
|
class BaseInstance(ABC):
|
||||||
class BaseInstance:
|
@classmethod
|
||||||
instance_type: type
|
def blacklist(cls) -> List[str]:
|
||||||
suffix: str
|
return []
|
||||||
log_file_name: str | None = None
|
|
||||||
data_dir: Path = field(init=False)
|
|
||||||
base_folders: List[Path] = field(init=False)
|
|
||||||
cfg_dir: Path = field(init=False)
|
|
||||||
log_dir: Path = field(init=False)
|
|
||||||
gcodes_dir: Path = field(init=False)
|
|
||||||
comms_dir: Path = field(init=False)
|
|
||||||
sysd_dir: Path = field(init=False)
|
|
||||||
is_legacy_instance: bool = field(init=False)
|
|
||||||
|
|
||||||
def __post_init__(self):
|
def __init__(
|
||||||
self.data_dir = get_data_dir(self.instance_type, self.suffix)
|
self,
|
||||||
# the following attributes require the data_dir to be set
|
suffix: str,
|
||||||
self.cfg_dir = self.data_dir.joinpath("config")
|
instance_type: B = B,
|
||||||
self.log_dir = self.data_dir.joinpath("logs")
|
):
|
||||||
self.gcodes_dir = self.data_dir.joinpath("gcodes")
|
self._instance_type = instance_type
|
||||||
self.comms_dir = self.data_dir.joinpath("comms")
|
self._suffix = suffix
|
||||||
self.sysd_dir = self.data_dir.joinpath("systemd")
|
self._user = CURRENT_USER
|
||||||
self.is_legacy_instance = self._set_is_legacy_instance()
|
self._data_dir_name = self.get_data_dir_name_from_suffix()
|
||||||
self.base_folders = [
|
self._data_dir = Path.home().joinpath(f"{self._data_dir_name}_data")
|
||||||
|
self._cfg_dir = self.data_dir.joinpath("config")
|
||||||
|
self._log_dir = self.data_dir.joinpath("logs")
|
||||||
|
self._comms_dir = self.data_dir.joinpath("comms")
|
||||||
|
self._sysd_dir = self.data_dir.joinpath("systemd")
|
||||||
|
self._gcodes_dir = self.data_dir.joinpath("gcodes")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def instance_type(self) -> Type["BaseInstance"]:
|
||||||
|
return self._instance_type
|
||||||
|
|
||||||
|
@instance_type.setter
|
||||||
|
def instance_type(self, value: Type["BaseInstance"]) -> None:
|
||||||
|
self._instance_type = value
|
||||||
|
|
||||||
|
@property
|
||||||
|
def suffix(self) -> str:
|
||||||
|
return self._suffix
|
||||||
|
|
||||||
|
@suffix.setter
|
||||||
|
def suffix(self, value: str) -> None:
|
||||||
|
self._suffix = value
|
||||||
|
|
||||||
|
@property
|
||||||
|
def user(self) -> str:
|
||||||
|
return self._user
|
||||||
|
|
||||||
|
@user.setter
|
||||||
|
def user(self, value: str) -> None:
|
||||||
|
self._user = value
|
||||||
|
|
||||||
|
@property
|
||||||
|
def data_dir_name(self) -> str:
|
||||||
|
return self._data_dir_name
|
||||||
|
|
||||||
|
@data_dir_name.setter
|
||||||
|
def data_dir_name(self, value: str) -> None:
|
||||||
|
self._data_dir_name = value
|
||||||
|
|
||||||
|
@property
|
||||||
|
def data_dir(self) -> Path:
|
||||||
|
return self._data_dir
|
||||||
|
|
||||||
|
@data_dir.setter
|
||||||
|
def data_dir(self, value: str) -> None:
|
||||||
|
self._data_dir = value
|
||||||
|
|
||||||
|
@property
|
||||||
|
def cfg_dir(self) -> Path:
|
||||||
|
return self._cfg_dir
|
||||||
|
|
||||||
|
@cfg_dir.setter
|
||||||
|
def cfg_dir(self, value: str) -> None:
|
||||||
|
self._cfg_dir = value
|
||||||
|
|
||||||
|
@property
|
||||||
|
def log_dir(self) -> Path:
|
||||||
|
return self._log_dir
|
||||||
|
|
||||||
|
@log_dir.setter
|
||||||
|
def log_dir(self, value: str) -> None:
|
||||||
|
self._log_dir = value
|
||||||
|
|
||||||
|
@property
|
||||||
|
def comms_dir(self) -> Path:
|
||||||
|
return self._comms_dir
|
||||||
|
|
||||||
|
@comms_dir.setter
|
||||||
|
def comms_dir(self, value: str) -> None:
|
||||||
|
self._comms_dir = value
|
||||||
|
|
||||||
|
@property
|
||||||
|
def sysd_dir(self) -> Path:
|
||||||
|
return self._sysd_dir
|
||||||
|
|
||||||
|
@sysd_dir.setter
|
||||||
|
def sysd_dir(self, value: str) -> None:
|
||||||
|
self._sysd_dir = value
|
||||||
|
|
||||||
|
@property
|
||||||
|
def gcodes_dir(self) -> Path:
|
||||||
|
return self._gcodes_dir
|
||||||
|
|
||||||
|
@gcodes_dir.setter
|
||||||
|
def gcodes_dir(self, value: str) -> None:
|
||||||
|
self._gcodes_dir = value
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def create(self) -> None:
|
||||||
|
raise NotImplementedError("Subclasses must implement the create method")
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def delete(self) -> None:
|
||||||
|
raise NotImplementedError("Subclasses must implement the delete method")
|
||||||
|
|
||||||
|
def create_folders(self, add_dirs: List[Path] = None) -> None:
|
||||||
|
dirs = [
|
||||||
self.data_dir,
|
self.data_dir,
|
||||||
self.cfg_dir,
|
self.cfg_dir,
|
||||||
self.log_dir,
|
self.log_dir,
|
||||||
self.gcodes_dir,
|
|
||||||
self.comms_dir,
|
self.comms_dir,
|
||||||
self.sysd_dir,
|
self.sysd_dir,
|
||||||
]
|
]
|
||||||
|
|
||||||
def _set_is_legacy_instance(self) -> bool:
|
if add_dirs:
|
||||||
legacy_pattern = r"^(?!printer)(.+)_data"
|
dirs.extend(add_dirs)
|
||||||
match = re.search(legacy_pattern, self.data_dir.name)
|
|
||||||
|
|
||||||
return True if (match and self.suffix != "") else False
|
for _dir in dirs:
|
||||||
|
_dir.mkdir(exist_ok=True)
|
||||||
|
|
||||||
|
def get_service_file_name(self, extension: bool = False) -> str:
|
||||||
|
name = f"{self.__class__.__name__.lower()}"
|
||||||
|
if self.suffix != "":
|
||||||
|
name += f"-{self.suffix}"
|
||||||
|
|
||||||
|
return name if not extension else f"{name}.service"
|
||||||
|
|
||||||
|
def get_service_file_path(self) -> Path:
|
||||||
|
return SYSTEMD.joinpath(self.get_service_file_name(extension=True))
|
||||||
|
|
||||||
|
def get_data_dir_name_from_suffix(self) -> str:
|
||||||
|
if self._suffix == "":
|
||||||
|
return "printer"
|
||||||
|
elif self._suffix.isdigit():
|
||||||
|
return f"printer_{self._suffix}"
|
||||||
|
else:
|
||||||
|
return self._suffix
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
# ======================================================================= #
|
# ======================================================================= #
|
||||||
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
||||||
# #
|
# #
|
||||||
@@ -6,103 +8,207 @@
|
|||||||
# #
|
# #
|
||||||
# This file may be distributed under the terms of the GNU GPLv3 license #
|
# This file may be distributed under the terms of the GNU GPLv3 license #
|
||||||
# ======================================================================= #
|
# ======================================================================= #
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
|
import re
|
||||||
|
import subprocess
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from subprocess import CalledProcessError
|
from typing import List, Optional, Union, TypeVar
|
||||||
from typing import List
|
|
||||||
|
|
||||||
from core.logger import Logger
|
from kiauh.core.instance_manager.base_instance import BaseInstance
|
||||||
from utils.instance_type import InstanceType
|
from kiauh.utils.constants import SYSTEMD
|
||||||
from utils.sys_utils import cmd_sysctl_service
|
from kiauh.utils.logger import Logger
|
||||||
|
|
||||||
|
|
||||||
|
I = TypeVar(name="I", bound=BaseInstance, covariant=True)
|
||||||
|
|
||||||
|
|
||||||
|
# noinspection PyMethodMayBeStatic
|
||||||
class InstanceManager:
|
class InstanceManager:
|
||||||
@staticmethod
|
def __init__(self, instance_type: I) -> None:
|
||||||
def enable(instance: InstanceType) -> None:
|
self._instance_type = instance_type
|
||||||
service_name: str = instance.service_file_path.name
|
self._current_instance: Optional[I] = None
|
||||||
|
self._instance_suffix: Optional[str] = None
|
||||||
|
self._instance_service: Optional[str] = None
|
||||||
|
self._instance_service_full: Optional[str] = None
|
||||||
|
self._instance_service_path: Optional[str] = None
|
||||||
|
self._instances: List[I] = []
|
||||||
|
|
||||||
|
@property
|
||||||
|
def instance_type(self) -> I:
|
||||||
|
return self._instance_type
|
||||||
|
|
||||||
|
@instance_type.setter
|
||||||
|
def instance_type(self, value: I):
|
||||||
|
self._instance_type = value
|
||||||
|
|
||||||
|
@property
|
||||||
|
def current_instance(self) -> I:
|
||||||
|
return self._current_instance
|
||||||
|
|
||||||
|
@current_instance.setter
|
||||||
|
def current_instance(self, value: I) -> None:
|
||||||
|
self._current_instance = value
|
||||||
|
self.instance_suffix = value.suffix
|
||||||
|
self.instance_service = value.get_service_file_name()
|
||||||
|
self.instance_service_path = value.get_service_file_path()
|
||||||
|
|
||||||
|
@property
|
||||||
|
def instance_suffix(self) -> str:
|
||||||
|
return self._instance_suffix
|
||||||
|
|
||||||
|
@instance_suffix.setter
|
||||||
|
def instance_suffix(self, value: str):
|
||||||
|
self._instance_suffix = value
|
||||||
|
|
||||||
|
@property
|
||||||
|
def instance_service(self) -> str:
|
||||||
|
return self._instance_service
|
||||||
|
|
||||||
|
@instance_service.setter
|
||||||
|
def instance_service(self, value: str):
|
||||||
|
self._instance_service = value
|
||||||
|
|
||||||
|
@property
|
||||||
|
def instance_service_full(self) -> str:
|
||||||
|
return f"{self._instance_service}.service"
|
||||||
|
|
||||||
|
@property
|
||||||
|
def instance_service_path(self) -> str:
|
||||||
|
return self._instance_service_path
|
||||||
|
|
||||||
|
@instance_service_path.setter
|
||||||
|
def instance_service_path(self, value: str):
|
||||||
|
self._instance_service_path = value
|
||||||
|
|
||||||
|
@property
|
||||||
|
def instances(self) -> List[I]:
|
||||||
|
return self.find_instances()
|
||||||
|
|
||||||
|
@instances.setter
|
||||||
|
def instances(self, value: List[I]):
|
||||||
|
self._instances = value
|
||||||
|
|
||||||
|
def create_instance(self) -> None:
|
||||||
|
if self.current_instance is not None:
|
||||||
try:
|
try:
|
||||||
cmd_sysctl_service(service_name, "enable")
|
self.current_instance.create()
|
||||||
except CalledProcessError as e:
|
except (OSError, subprocess.CalledProcessError) as e:
|
||||||
Logger.print_error(f"Error enabling service {service_name}:")
|
Logger.print_error(f"Creating instance failed: {e}")
|
||||||
|
raise
|
||||||
|
else:
|
||||||
|
raise ValueError("current_instance cannot be None")
|
||||||
|
|
||||||
|
def delete_instance(self) -> None:
|
||||||
|
if self.current_instance is not None:
|
||||||
|
try:
|
||||||
|
self.current_instance.delete()
|
||||||
|
except (OSError, subprocess.CalledProcessError) as e:
|
||||||
|
Logger.print_error(f"Removing instance failed: {e}")
|
||||||
|
raise
|
||||||
|
else:
|
||||||
|
raise ValueError("current_instance cannot be None")
|
||||||
|
|
||||||
|
def enable_instance(self) -> None:
|
||||||
|
Logger.print_status(f"Enabling {self.instance_service_full} ...")
|
||||||
|
try:
|
||||||
|
command = ["sudo", "systemctl", "enable", self.instance_service_full]
|
||||||
|
if subprocess.run(command, check=True):
|
||||||
|
Logger.print_ok(f"{self.instance_service_full} enabled.")
|
||||||
|
except subprocess.CalledProcessError as e:
|
||||||
|
Logger.print_error(f"Error enabling service {self.instance_service_full}:")
|
||||||
Logger.print_error(f"{e}")
|
Logger.print_error(f"{e}")
|
||||||
|
|
||||||
@staticmethod
|
def disable_instance(self) -> None:
|
||||||
def disable(instance: InstanceType) -> None:
|
Logger.print_status(f"Disabling {self.instance_service_full} ...")
|
||||||
service_name: str = instance.service_file_path.name
|
|
||||||
try:
|
try:
|
||||||
cmd_sysctl_service(service_name, "disable")
|
command = ["sudo", "systemctl", "disable", self.instance_service_full]
|
||||||
except CalledProcessError as e:
|
if subprocess.run(command, check=True):
|
||||||
Logger.print_error(f"Error disabling {service_name}: {e}")
|
Logger.print_ok(f"{self.instance_service_full} disabled.")
|
||||||
|
except subprocess.CalledProcessError as e:
|
||||||
|
Logger.print_error(f"Error disabling {self.instance_service_full}:")
|
||||||
|
Logger.print_error(f"{e}")
|
||||||
|
|
||||||
|
def start_instance(self) -> None:
|
||||||
|
Logger.print_status(f"Starting {self.instance_service_full} ...")
|
||||||
|
try:
|
||||||
|
command = ["sudo", "systemctl", "start", self.instance_service_full]
|
||||||
|
if subprocess.run(command, check=True):
|
||||||
|
Logger.print_ok(f"{self.instance_service_full} started.")
|
||||||
|
except subprocess.CalledProcessError as e:
|
||||||
|
Logger.print_error(f"Error starting {self.instance_service_full}:")
|
||||||
|
Logger.print_error(f"{e}")
|
||||||
|
|
||||||
|
def restart_instance(self) -> None:
|
||||||
|
Logger.print_status(f"Restarting {self.instance_service_full} ...")
|
||||||
|
try:
|
||||||
|
command = ["sudo", "systemctl", "restart", self.instance_service_full]
|
||||||
|
if subprocess.run(command, check=True):
|
||||||
|
Logger.print_ok(f"{self.instance_service_full} restarted.")
|
||||||
|
except subprocess.CalledProcessError as e:
|
||||||
|
Logger.print_error(f"Error restarting {self.instance_service_full}:")
|
||||||
|
Logger.print_error(f"{e}")
|
||||||
|
|
||||||
|
def start_all_instance(self) -> None:
|
||||||
|
for instance in self.instances:
|
||||||
|
self.current_instance = instance
|
||||||
|
self.start_instance()
|
||||||
|
|
||||||
|
def restart_all_instance(self) -> None:
|
||||||
|
for instance in self.instances:
|
||||||
|
self.current_instance = instance
|
||||||
|
self.restart_instance()
|
||||||
|
|
||||||
|
def stop_instance(self) -> None:
|
||||||
|
Logger.print_status(f"Stopping {self.instance_service_full} ...")
|
||||||
|
try:
|
||||||
|
command = ["sudo", "systemctl", "stop", self.instance_service_full]
|
||||||
|
if subprocess.run(command, check=True):
|
||||||
|
Logger.print_ok(f"{self.instance_service_full} stopped.")
|
||||||
|
except subprocess.CalledProcessError as e:
|
||||||
|
Logger.print_error(f"Error stopping {self.instance_service_full}:")
|
||||||
|
Logger.print_error(f"{e}")
|
||||||
raise
|
raise
|
||||||
|
|
||||||
@staticmethod
|
def stop_all_instance(self) -> None:
|
||||||
def start(instance: InstanceType) -> None:
|
for instance in self.instances:
|
||||||
service_name: str = instance.service_file_path.name
|
self.current_instance = instance
|
||||||
|
self.stop_instance()
|
||||||
|
|
||||||
|
def reload_daemon(self) -> None:
|
||||||
|
Logger.print_status("Reloading systemd manager configuration ...")
|
||||||
try:
|
try:
|
||||||
cmd_sysctl_service(service_name, "start")
|
command = ["sudo", "systemctl", "daemon-reload"]
|
||||||
except CalledProcessError as e:
|
if subprocess.run(command, check=True):
|
||||||
Logger.print_error(f"Error starting {service_name}: {e}")
|
Logger.print_ok("Systemd manager configuration reloaded")
|
||||||
|
except subprocess.CalledProcessError as e:
|
||||||
|
Logger.print_error("Error reloading systemd manager configuration:")
|
||||||
|
Logger.print_error(f"{e}")
|
||||||
raise
|
raise
|
||||||
|
|
||||||
@staticmethod
|
def find_instances(self) -> List[I]:
|
||||||
def stop(instance: InstanceType) -> None:
|
name = self.instance_type.__name__.lower()
|
||||||
name: str = instance.service_file_path.name
|
pattern = re.compile(f"^{name}(-[0-9a-zA-Z]+)?.service$")
|
||||||
try:
|
excluded = self.instance_type.blacklist()
|
||||||
cmd_sysctl_service(name, "stop")
|
|
||||||
except CalledProcessError as e:
|
|
||||||
Logger.print_error(f"Error stopping {name}: {e}")
|
|
||||||
raise
|
|
||||||
|
|
||||||
@staticmethod
|
service_list = [
|
||||||
def restart(instance: InstanceType) -> None:
|
Path(SYSTEMD, service)
|
||||||
name: str = instance.service_file_path.name
|
for service in SYSTEMD.iterdir()
|
||||||
try:
|
if pattern.search(service.name)
|
||||||
cmd_sysctl_service(name, "restart")
|
and not any(s in service.name for s in excluded)
|
||||||
except CalledProcessError as e:
|
]
|
||||||
Logger.print_error(f"Error restarting {name}: {e}")
|
|
||||||
raise
|
|
||||||
|
|
||||||
@staticmethod
|
instance_list = [
|
||||||
def start_all(instances: List[InstanceType]) -> None:
|
self.instance_type(suffix=self._get_instance_suffix(service))
|
||||||
for instance in instances:
|
for service in service_list
|
||||||
InstanceManager.start(instance)
|
]
|
||||||
|
|
||||||
@staticmethod
|
return sorted(instance_list, key=lambda x: self._sort_instance_list(x.suffix))
|
||||||
def stop_all(instances: List[InstanceType]) -> None:
|
|
||||||
for instance in instances:
|
|
||||||
InstanceManager.stop(instance)
|
|
||||||
|
|
||||||
@staticmethod
|
def _get_instance_suffix(self, file_path: Path) -> str:
|
||||||
def restart_all(instances: List[InstanceType]) -> None:
|
return file_path.stem.split("-")[-1] if "-" in file_path.stem else ""
|
||||||
for instance in instances:
|
|
||||||
InstanceManager.restart(instance)
|
|
||||||
|
|
||||||
@staticmethod
|
def _sort_instance_list(self, s: Union[int, str, None]):
|
||||||
def remove(instance: InstanceType) -> None:
|
if s is None:
|
||||||
from utils.fs_utils import run_remove_routines
|
|
||||||
from utils.sys_utils import remove_system_service
|
|
||||||
|
|
||||||
try:
|
|
||||||
# remove the service file
|
|
||||||
service_file_path: Path = instance.service_file_path
|
|
||||||
if service_file_path is not None:
|
|
||||||
remove_system_service(service_file_path.name)
|
|
||||||
|
|
||||||
# then remove all the log files
|
|
||||||
if (
|
|
||||||
not instance.log_file_name
|
|
||||||
or not instance.base.log_dir
|
|
||||||
or not instance.base.log_dir.exists()
|
|
||||||
):
|
|
||||||
return
|
return
|
||||||
|
|
||||||
files = instance.base.log_dir.iterdir()
|
return int(s) if s.isdigit() else s
|
||||||
logs = [f for f in files if f.name.startswith(instance.log_file_name)]
|
|
||||||
for log in logs:
|
|
||||||
Logger.print_status(f"Remove '{log}'")
|
|
||||||
run_remove_routines(log)
|
|
||||||
|
|
||||||
except Exception as e:
|
|
||||||
Logger.print_error(f"Error removing service: {e}")
|
|
||||||
raise
|
|
||||||
|
|||||||
8
kiauh/core/instance_manager/name_scheme.py
Normal file
8
kiauh/core/instance_manager/name_scheme.py
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
from enum import unique, Enum
|
||||||
|
|
||||||
|
|
||||||
|
@unique
|
||||||
|
class NameScheme(Enum):
|
||||||
|
SINGLE = "SINGLE"
|
||||||
|
INDEX = "INDEX"
|
||||||
|
CUSTOM = "CUSTOM"
|
||||||
@@ -1,194 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import textwrap
|
|
||||||
from enum import Enum
|
|
||||||
from typing import List
|
|
||||||
|
|
||||||
from core.constants import (
|
|
||||||
COLOR_CYAN,
|
|
||||||
COLOR_GREEN,
|
|
||||||
COLOR_MAGENTA,
|
|
||||||
COLOR_RED,
|
|
||||||
COLOR_WHITE,
|
|
||||||
COLOR_YELLOW,
|
|
||||||
RESET_FORMAT,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
class DialogType(Enum):
|
|
||||||
INFO = ("INFO", COLOR_WHITE)
|
|
||||||
SUCCESS = ("SUCCESS", COLOR_GREEN)
|
|
||||||
ATTENTION = ("ATTENTION", COLOR_YELLOW)
|
|
||||||
WARNING = ("WARNING", COLOR_YELLOW)
|
|
||||||
ERROR = ("ERROR", COLOR_RED)
|
|
||||||
CUSTOM = (None, None)
|
|
||||||
|
|
||||||
|
|
||||||
class DialogCustomColor(Enum):
|
|
||||||
WHITE = COLOR_WHITE
|
|
||||||
GREEN = COLOR_GREEN
|
|
||||||
YELLOW = COLOR_YELLOW
|
|
||||||
RED = COLOR_RED
|
|
||||||
CYAN = COLOR_CYAN
|
|
||||||
MAGENTA = COLOR_MAGENTA
|
|
||||||
|
|
||||||
|
|
||||||
LINE_WIDTH = 53
|
|
||||||
|
|
||||||
|
|
||||||
class Logger:
|
|
||||||
@staticmethod
|
|
||||||
def info(msg) -> None:
|
|
||||||
# log to kiauh.log
|
|
||||||
pass
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def warn(msg) -> None:
|
|
||||||
# log to kiauh.log
|
|
||||||
pass
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def error(msg) -> None:
|
|
||||||
# log to kiauh.log
|
|
||||||
pass
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def print_info(msg, prefix=True, start="", end="\n") -> None:
|
|
||||||
message = f"[INFO] {msg}" if prefix else msg
|
|
||||||
print(f"{COLOR_WHITE}{start}{message}{RESET_FORMAT}", end=end)
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def print_ok(msg: str = "Success!", prefix=True, start="", end="\n") -> None:
|
|
||||||
message = f"[OK] {msg}" if prefix else msg
|
|
||||||
print(f"{COLOR_GREEN}{start}{message}{RESET_FORMAT}", end=end)
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def print_warn(msg, prefix=True, start="", end="\n") -> None:
|
|
||||||
message = f"[WARN] {msg}" if prefix else msg
|
|
||||||
print(f"{COLOR_YELLOW}{start}{message}{RESET_FORMAT}", end=end)
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def print_error(msg, prefix=True, start="", end="\n") -> None:
|
|
||||||
message = f"[ERROR] {msg}" if prefix else msg
|
|
||||||
print(f"{COLOR_RED}{start}{message}{RESET_FORMAT}", end=end)
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def print_status(msg, prefix=True, start="", end="\n") -> None:
|
|
||||||
message = f"\n###### {msg}" if prefix else msg
|
|
||||||
print(f"{COLOR_MAGENTA}{start}{message}{RESET_FORMAT}", end=end)
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def print_dialog(
|
|
||||||
title: DialogType,
|
|
||||||
content: List[str],
|
|
||||||
center_content: bool = False,
|
|
||||||
custom_title: str | None = None,
|
|
||||||
custom_color: DialogCustomColor | None = None,
|
|
||||||
margin_top: int = 0,
|
|
||||||
margin_bottom: int = 0,
|
|
||||||
) -> None:
|
|
||||||
"""
|
|
||||||
Prints a dialog with the given title and content.
|
|
||||||
Those dialogs should be used to display verbose messages to the user which
|
|
||||||
require simple interaction like confirmation or input. Do not use this for
|
|
||||||
navigating through the application.
|
|
||||||
|
|
||||||
:param title: The type of the dialog.
|
|
||||||
:param content: The content of the dialog.
|
|
||||||
:param center_content: Whether to center the content or not.
|
|
||||||
:param custom_title: A custom title for the dialog.
|
|
||||||
:param custom_color: A custom color for the dialog.
|
|
||||||
:param margin_top: The number of empty lines to print before the dialog.
|
|
||||||
:param margin_bottom: The number of empty lines to print after the dialog.
|
|
||||||
"""
|
|
||||||
dialog_color = Logger._get_dialog_color(title, custom_color)
|
|
||||||
dialog_title = Logger._get_dialog_title(title, custom_title)
|
|
||||||
dialog_title_formatted = Logger._format_dialog_title(dialog_title)
|
|
||||||
dialog_content = Logger.format_content(content, LINE_WIDTH, center_content)
|
|
||||||
top = Logger._format_top_border(dialog_color)
|
|
||||||
bottom = Logger._format_bottom_border()
|
|
||||||
|
|
||||||
print("\n" * margin_top)
|
|
||||||
print(
|
|
||||||
f"{top}{dialog_title_formatted}{dialog_content}{bottom}",
|
|
||||||
end="",
|
|
||||||
)
|
|
||||||
print("\n" * margin_bottom)
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def _get_dialog_title(
|
|
||||||
title: DialogType, custom_title: str | None = None
|
|
||||||
) -> str | None:
|
|
||||||
if title == DialogType.CUSTOM and custom_title:
|
|
||||||
return f"[ {custom_title} ]"
|
|
||||||
return f"[ {title.value[0]} ]" if title.value[0] else None
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def _get_dialog_color(
|
|
||||||
title: DialogType, custom_color: DialogCustomColor | None = None
|
|
||||||
) -> str:
|
|
||||||
if title == DialogType.CUSTOM and custom_color:
|
|
||||||
return str(custom_color.value)
|
|
||||||
|
|
||||||
color: str = title.value[1] if title.value[1] else DialogCustomColor.WHITE.value
|
|
||||||
|
|
||||||
return color
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def _format_top_border(color: str) -> str:
|
|
||||||
return f"{color}┏━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┓"
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def _format_bottom_border() -> str:
|
|
||||||
return (
|
|
||||||
f"\n┗━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┛{RESET_FORMAT}"
|
|
||||||
)
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def _format_dialog_title(title: str | None) -> str:
|
|
||||||
if title is not None:
|
|
||||||
return textwrap.dedent(f"""
|
|
||||||
┃ {title:^{LINE_WIDTH}} ┃
|
|
||||||
┠───────────────────────────────────────────────────────┨
|
|
||||||
""")
|
|
||||||
else:
|
|
||||||
return "\n"
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def format_content(
|
|
||||||
content: List[str],
|
|
||||||
line_width: int,
|
|
||||||
center_content: bool = False,
|
|
||||||
border_left: str = "┃",
|
|
||||||
border_right: str = "┃",
|
|
||||||
) -> str:
|
|
||||||
wrapper = textwrap.TextWrapper(line_width)
|
|
||||||
|
|
||||||
lines = []
|
|
||||||
for i, c in enumerate(content):
|
|
||||||
paragraph = wrapper.wrap(c)
|
|
||||||
lines.extend(paragraph)
|
|
||||||
|
|
||||||
# add a full blank line if we have a double newline
|
|
||||||
# character unless we are at the end of the list
|
|
||||||
if c == "\n\n" and i < len(content) - 1:
|
|
||||||
lines.append(" " * line_width)
|
|
||||||
|
|
||||||
if not center_content:
|
|
||||||
formatted_lines = [
|
|
||||||
f"{border_left} {line:<{line_width}} {border_right}" for line in lines
|
|
||||||
]
|
|
||||||
else:
|
|
||||||
formatted_lines = [
|
|
||||||
f"{border_left} {line:^{line_width}} {border_right}" for line in lines
|
|
||||||
]
|
|
||||||
|
|
||||||
return "\n".join(formatted_lines)
|
|
||||||
@@ -1,3 +1,5 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
# ======================================================================= #
|
# ======================================================================= #
|
||||||
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
||||||
# #
|
# #
|
||||||
@@ -6,32 +8,7 @@
|
|||||||
# #
|
# #
|
||||||
# This file may be distributed under the terms of the GNU GPLv3 license #
|
# This file may be distributed under the terms of the GNU GPLv3 license #
|
||||||
# ======================================================================= #
|
# ======================================================================= #
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
from dataclasses import dataclass
|
QUIT_FOOTER = "quit"
|
||||||
from enum import Enum
|
BACK_FOOTER = "back"
|
||||||
from typing import Any, Callable, Type
|
BACK_HELP_FOOTER = "back_help"
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
|
||||||
class Option:
|
|
||||||
"""
|
|
||||||
Represents a menu option.
|
|
||||||
:param method: Method that will be used to call the menu option
|
|
||||||
:param opt_index: Can be used to pass the user input to the menu option
|
|
||||||
:param opt_data: Can be used to pass any additional data to the menu option
|
|
||||||
"""
|
|
||||||
|
|
||||||
def __repr__(self):
|
|
||||||
return f"Option(method={self.method.__name__}, opt_index={self.opt_index}, opt_data={self.opt_data})"
|
|
||||||
|
|
||||||
method: Type[Callable]
|
|
||||||
opt_index: str = ""
|
|
||||||
opt_data: Any = None
|
|
||||||
|
|
||||||
|
|
||||||
class FooterType(Enum):
|
|
||||||
QUIT = "QUIT"
|
|
||||||
BACK = "BACK"
|
|
||||||
BACK_HELP = "BACK_HELP"
|
|
||||||
BLANK = "BLANK"
|
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
# ======================================================================= #
|
# ======================================================================= #
|
||||||
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
||||||
# #
|
# #
|
||||||
@@ -6,93 +8,35 @@
|
|||||||
# #
|
# #
|
||||||
# This file may be distributed under the terms of the GNU GPLv3 license #
|
# This file may be distributed under the terms of the GNU GPLv3 license #
|
||||||
# ======================================================================= #
|
# ======================================================================= #
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import textwrap
|
import textwrap
|
||||||
from typing import Type
|
|
||||||
|
|
||||||
from components.klipper import KLIPPER_DIR
|
from kiauh.core.menus import BACK_FOOTER
|
||||||
from components.klipper.klipper import Klipper
|
from kiauh.core.menus.base_menu import BaseMenu
|
||||||
from components.klipper_firmware.menus.klipper_build_menu import (
|
from kiauh.utils.constants import COLOR_YELLOW, RESET_FORMAT
|
||||||
KlipperBuildFirmwareMenu,
|
|
||||||
)
|
|
||||||
from components.klipper_firmware.menus.klipper_flash_menu import (
|
|
||||||
KlipperFlashMethodMenu,
|
|
||||||
KlipperSelectMcuConnectionMenu,
|
|
||||||
)
|
|
||||||
from components.moonraker import MOONRAKER_DIR
|
|
||||||
from components.moonraker.moonraker import Moonraker
|
|
||||||
from core.constants import COLOR_YELLOW, RESET_FORMAT
|
|
||||||
from core.menus import Option
|
|
||||||
from core.menus.base_menu import BaseMenu
|
|
||||||
from procedures.system import change_system_hostname
|
|
||||||
from utils.git_utils import rollback_repository
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyUnusedLocal
|
|
||||||
# noinspection PyMethodMayBeStatic
|
|
||||||
class AdvancedMenu(BaseMenu):
|
class AdvancedMenu(BaseMenu):
|
||||||
def __init__(self, previous_menu: Type[BaseMenu] | None = None) -> None:
|
def __init__(self):
|
||||||
super().__init__()
|
super().__init__(header=True, options={}, footer_type=BACK_FOOTER)
|
||||||
self.previous_menu: Type[BaseMenu] | None = previous_menu
|
|
||||||
|
|
||||||
def set_previous_menu(self, previous_menu: Type[BaseMenu] | None) -> None:
|
def print_menu(self):
|
||||||
from core.menus.main_menu import MainMenu
|
|
||||||
|
|
||||||
self.previous_menu = previous_menu if previous_menu is not None else MainMenu
|
|
||||||
|
|
||||||
def set_options(self) -> None:
|
|
||||||
self.options = {
|
|
||||||
"1": Option(method=self.build),
|
|
||||||
"2": Option(method=self.flash),
|
|
||||||
"3": Option(method=self.build_flash),
|
|
||||||
"4": Option(method=self.get_id),
|
|
||||||
"5": Option(method=self.klipper_rollback),
|
|
||||||
"6": Option(method=self.moonraker_rollback),
|
|
||||||
"7": Option(method=self.change_hostname),
|
|
||||||
}
|
|
||||||
|
|
||||||
def print_menu(self) -> None:
|
|
||||||
header = " [ Advanced Menu ] "
|
header = " [ Advanced Menu ] "
|
||||||
color = COLOR_YELLOW
|
color = COLOR_YELLOW
|
||||||
count = 62 - len(color) - len(RESET_FORMAT)
|
count = 62 - len(color) - len(RESET_FORMAT)
|
||||||
menu = textwrap.dedent(
|
menu = textwrap.dedent(
|
||||||
f"""
|
f"""
|
||||||
╔═══════════════════════════════════════════════════════╗
|
/=======================================================\\
|
||||||
║ {color}{header:~^{count}}{RESET_FORMAT} ║
|
| {color}{header:~^{count}}{RESET_FORMAT} |
|
||||||
╟───────────────────────────┬───────────────────────────╢
|
|-------------------------------------------------------|
|
||||||
║ Klipper Firmware: │ Repository Rollback: ║
|
| Klipper & API: | Mainsail: |
|
||||||
║ 1) [Build] │ 5) [Klipper] ║
|
| 0) [Rollback] | 5) [Theme installer] |
|
||||||
║ 2) [Flash] │ 6) [Moonraker] ║
|
| | |
|
||||||
║ 3) [Build + Flash] │ ║
|
| Firmware: | System: |
|
||||||
║ 4) [Get MCU ID] │ System: ║
|
| 1) [Build only] | 6) [Change hostname] |
|
||||||
║ │ 7) [Change hostname] ║
|
| 2) [Flash only] | |
|
||||||
╟───────────────────────────┴───────────────────────────╢
|
| 3) [Build + Flash] | Extras: |
|
||||||
|
| 4) [Get MCU ID] | 7) [G-Code Shell Command] |
|
||||||
"""
|
"""
|
||||||
)[1:]
|
)[1:]
|
||||||
print(menu, end="")
|
print(menu, end="")
|
||||||
|
|
||||||
def klipper_rollback(self, **kwargs) -> None:
|
|
||||||
rollback_repository(KLIPPER_DIR, Klipper)
|
|
||||||
|
|
||||||
def moonraker_rollback(self, **kwargs) -> None:
|
|
||||||
rollback_repository(MOONRAKER_DIR, Moonraker)
|
|
||||||
|
|
||||||
def build(self, **kwargs) -> None:
|
|
||||||
KlipperBuildFirmwareMenu(previous_menu=self.__class__).run()
|
|
||||||
|
|
||||||
def flash(self, **kwargs) -> None:
|
|
||||||
KlipperFlashMethodMenu(previous_menu=self.__class__).run()
|
|
||||||
|
|
||||||
def build_flash(self, **kwargs) -> None:
|
|
||||||
KlipperBuildFirmwareMenu(previous_menu=KlipperFlashMethodMenu).run()
|
|
||||||
KlipperFlashMethodMenu(previous_menu=self.__class__).run()
|
|
||||||
|
|
||||||
def get_id(self, **kwargs) -> None:
|
|
||||||
KlipperSelectMcuConnectionMenu(
|
|
||||||
previous_menu=self.__class__,
|
|
||||||
standalone=True,
|
|
||||||
).run()
|
|
||||||
|
|
||||||
def change_hostname(self, **kwargs) -> None:
|
|
||||||
change_system_hostname()
|
|
||||||
|
|||||||
@@ -1,108 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import textwrap
|
|
||||||
from typing import Type
|
|
||||||
|
|
||||||
from components.klipper.klipper_utils import backup_klipper_dir
|
|
||||||
from components.klipperscreen.klipperscreen import backup_klipperscreen_dir
|
|
||||||
from components.moonraker.moonraker_utils import (
|
|
||||||
backup_moonraker_db_dir,
|
|
||||||
backup_moonraker_dir,
|
|
||||||
)
|
|
||||||
from components.webui_client.client_utils import (
|
|
||||||
backup_client_config_data,
|
|
||||||
backup_client_data,
|
|
||||||
)
|
|
||||||
from components.webui_client.fluidd_data import FluiddData
|
|
||||||
from components.webui_client.mainsail_data import MainsailData
|
|
||||||
from core.constants import COLOR_CYAN, COLOR_YELLOW, RESET_FORMAT
|
|
||||||
from core.menus import Option
|
|
||||||
from core.menus.base_menu import BaseMenu
|
|
||||||
from utils.common import backup_printer_config_dir
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyUnusedLocal
|
|
||||||
# noinspection PyMethodMayBeStatic
|
|
||||||
class BackupMenu(BaseMenu):
|
|
||||||
def __init__(self, previous_menu: Type[BaseMenu] | None = None) -> None:
|
|
||||||
super().__init__()
|
|
||||||
self.previous_menu: Type[BaseMenu] | None = previous_menu
|
|
||||||
|
|
||||||
def set_previous_menu(self, previous_menu: Type[BaseMenu] | None) -> None:
|
|
||||||
from core.menus.main_menu import MainMenu
|
|
||||||
|
|
||||||
self.previous_menu = previous_menu if previous_menu is not None else MainMenu
|
|
||||||
|
|
||||||
def set_options(self) -> None:
|
|
||||||
self.options = {
|
|
||||||
"1": Option(method=self.backup_klipper),
|
|
||||||
"2": Option(method=self.backup_moonraker),
|
|
||||||
"3": Option(method=self.backup_printer_config),
|
|
||||||
"4": Option(method=self.backup_moonraker_db),
|
|
||||||
"5": Option(method=self.backup_mainsail),
|
|
||||||
"6": Option(method=self.backup_fluidd),
|
|
||||||
"7": Option(method=self.backup_mainsail_config),
|
|
||||||
"8": Option(method=self.backup_fluidd_config),
|
|
||||||
"9": Option(method=self.backup_klipperscreen),
|
|
||||||
}
|
|
||||||
|
|
||||||
def print_menu(self) -> None:
|
|
||||||
header = " [ Backup Menu ] "
|
|
||||||
line1 = f"{COLOR_YELLOW}INFO: Backups are located in '~/kiauh-backups'{RESET_FORMAT}"
|
|
||||||
color = COLOR_CYAN
|
|
||||||
count = 62 - len(color) - len(RESET_FORMAT)
|
|
||||||
menu = textwrap.dedent(
|
|
||||||
f"""
|
|
||||||
╔═══════════════════════════════════════════════════════╗
|
|
||||||
║ {color}{header:~^{count}}{RESET_FORMAT} ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ {line1:^62} ║
|
|
||||||
╟───────────────────────────┬───────────────────────────╢
|
|
||||||
║ Klipper & Moonraker API: │ Client-Config: ║
|
|
||||||
║ 1) [Klipper] │ 7) [Mainsail-Config] ║
|
|
||||||
║ 2) [Moonraker] │ 8) [Fluidd-Config] ║
|
|
||||||
║ 3) [Config Folder] │ ║
|
|
||||||
║ 4) [Moonraker Database] │ Touchscreen GUI: ║
|
|
||||||
║ │ 9) [KlipperScreen] ║
|
|
||||||
║ Webinterface: │ ║
|
|
||||||
║ 5) [Mainsail] │ ║
|
|
||||||
║ 6) [Fluidd] │ ║
|
|
||||||
╟───────────────────────────┴───────────────────────────╢
|
|
||||||
"""
|
|
||||||
)[1:]
|
|
||||||
print(menu, end="")
|
|
||||||
|
|
||||||
def backup_klipper(self, **kwargs) -> None:
|
|
||||||
backup_klipper_dir()
|
|
||||||
|
|
||||||
def backup_moonraker(self, **kwargs) -> None:
|
|
||||||
backup_moonraker_dir()
|
|
||||||
|
|
||||||
def backup_printer_config(self, **kwargs) -> None:
|
|
||||||
backup_printer_config_dir()
|
|
||||||
|
|
||||||
def backup_moonraker_db(self, **kwargs) -> None:
|
|
||||||
backup_moonraker_db_dir()
|
|
||||||
|
|
||||||
def backup_mainsail(self, **kwargs) -> None:
|
|
||||||
backup_client_data(MainsailData())
|
|
||||||
|
|
||||||
def backup_fluidd(self, **kwargs) -> None:
|
|
||||||
backup_client_data(FluiddData())
|
|
||||||
|
|
||||||
def backup_mainsail_config(self, **kwargs) -> None:
|
|
||||||
backup_client_config_data(MainsailData())
|
|
||||||
|
|
||||||
def backup_fluidd_config(self, **kwargs) -> None:
|
|
||||||
backup_client_config_data(FluiddData())
|
|
||||||
|
|
||||||
def backup_klipperscreen(self, **kwargs) -> None:
|
|
||||||
backup_klipperscreen_dir()
|
|
||||||
@@ -1,3 +1,5 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
# ======================================================================= #
|
# ======================================================================= #
|
||||||
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
||||||
# #
|
# #
|
||||||
@@ -7,32 +9,28 @@
|
|||||||
# This file may be distributed under the terms of the GNU GPLv3 license #
|
# This file may be distributed under the terms of the GNU GPLv3 license #
|
||||||
# ======================================================================= #
|
# ======================================================================= #
|
||||||
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import subprocess
|
import subprocess
|
||||||
import sys
|
import sys
|
||||||
import textwrap
|
import textwrap
|
||||||
import traceback
|
from abc import abstractmethod, ABC
|
||||||
from abc import abstractmethod
|
from typing import Dict, Any, Literal
|
||||||
from typing import Dict, Type
|
|
||||||
|
|
||||||
from core.constants import (
|
from kiauh.core.menus import QUIT_FOOTER, BACK_FOOTER, BACK_HELP_FOOTER
|
||||||
COLOR_CYAN,
|
from kiauh.utils.constants import (
|
||||||
COLOR_GREEN,
|
COLOR_GREEN,
|
||||||
COLOR_RED,
|
|
||||||
COLOR_YELLOW,
|
COLOR_YELLOW,
|
||||||
|
COLOR_RED,
|
||||||
|
COLOR_CYAN,
|
||||||
RESET_FORMAT,
|
RESET_FORMAT,
|
||||||
)
|
)
|
||||||
from core.logger import Logger
|
from kiauh.utils.logger import Logger
|
||||||
from core.menus import FooterType, Option
|
|
||||||
from utils.input_utils import get_selection_input
|
|
||||||
|
|
||||||
|
|
||||||
def clear() -> None:
|
def clear():
|
||||||
subprocess.call("clear", shell=True)
|
subprocess.call("clear", shell=True)
|
||||||
|
|
||||||
|
|
||||||
def print_header() -> None:
|
def print_header():
|
||||||
line1 = " [ KIAUH ] "
|
line1 = " [ KIAUH ] "
|
||||||
line2 = "Klipper Installation And Update Helper"
|
line2 = "Klipper Installation And Update Helper"
|
||||||
line3 = ""
|
line3 = ""
|
||||||
@@ -40,43 +38,45 @@ def print_header() -> None:
|
|||||||
count = 62 - len(color) - len(RESET_FORMAT)
|
count = 62 - len(color) - len(RESET_FORMAT)
|
||||||
header = textwrap.dedent(
|
header = textwrap.dedent(
|
||||||
f"""
|
f"""
|
||||||
╔═══════════════════════════════════════════════════════╗
|
/=======================================================\\
|
||||||
║ {color}{line1:~^{count}}{RESET_FORMAT} ║
|
| {color}{line1:~^{count}}{RESET_FORMAT} |
|
||||||
║ {color}{line2:^{count}}{RESET_FORMAT} ║
|
| {color}{line2:^{count}}{RESET_FORMAT} |
|
||||||
║ {color}{line3:~^{count}}{RESET_FORMAT} ║
|
| {color}{line3:~^{count}}{RESET_FORMAT} |
|
||||||
╚═══════════════════════════════════════════════════════╝
|
\=======================================================/
|
||||||
"""
|
"""
|
||||||
)[1:]
|
)[1:]
|
||||||
print(header, end="")
|
print(header, end="")
|
||||||
|
|
||||||
|
|
||||||
def print_quit_footer() -> None:
|
def print_quit_footer():
|
||||||
text = "Q) Quit"
|
text = "Q) Quit"
|
||||||
color = COLOR_RED
|
color = COLOR_RED
|
||||||
count = 62 - len(color) - len(RESET_FORMAT)
|
count = 62 - len(color) - len(RESET_FORMAT)
|
||||||
footer = textwrap.dedent(
|
footer = textwrap.dedent(
|
||||||
f"""
|
f"""
|
||||||
║ {color}{text:^{count}}{RESET_FORMAT} ║
|
|-------------------------------------------------------|
|
||||||
╚═══════════════════════════════════════════════════════╝
|
| {color}{text:^{count}}{RESET_FORMAT} |
|
||||||
|
\=======================================================/
|
||||||
"""
|
"""
|
||||||
)[1:]
|
)[1:]
|
||||||
print(footer, end="")
|
print(footer, end="")
|
||||||
|
|
||||||
|
|
||||||
def print_back_footer() -> None:
|
def print_back_footer():
|
||||||
text = "B) « Back"
|
text = "B) « Back"
|
||||||
color = COLOR_GREEN
|
color = COLOR_GREEN
|
||||||
count = 62 - len(color) - len(RESET_FORMAT)
|
count = 62 - len(color) - len(RESET_FORMAT)
|
||||||
footer = textwrap.dedent(
|
footer = textwrap.dedent(
|
||||||
f"""
|
f"""
|
||||||
║ {color}{text:^{count}}{RESET_FORMAT} ║
|
|-------------------------------------------------------|
|
||||||
╚═══════════════════════════════════════════════════════╝
|
| {color}{text:^{count}}{RESET_FORMAT} |
|
||||||
|
\=======================================================/
|
||||||
"""
|
"""
|
||||||
)[1:]
|
)[1:]
|
||||||
print(footer, end="")
|
print(footer, end="")
|
||||||
|
|
||||||
|
|
||||||
def print_back_help_footer() -> None:
|
def print_back_help_footer():
|
||||||
text1 = "B) « Back"
|
text1 = "B) « Back"
|
||||||
text2 = "H) Help [?]"
|
text2 = "H) Help [?]"
|
||||||
color1 = COLOR_GREEN
|
color1 = COLOR_GREEN
|
||||||
@@ -84,115 +84,100 @@ def print_back_help_footer() -> None:
|
|||||||
count = 34 - len(color1) - len(RESET_FORMAT)
|
count = 34 - len(color1) - len(RESET_FORMAT)
|
||||||
footer = textwrap.dedent(
|
footer = textwrap.dedent(
|
||||||
f"""
|
f"""
|
||||||
║ {color1}{text1:^{count}}{RESET_FORMAT} │ {color2}{text2:^{count}}{RESET_FORMAT} ║
|
|-------------------------------------------------------|
|
||||||
╚═══════════════════════════╧═══════════════════════════╝
|
| {color1}{text1:^{count}}{RESET_FORMAT} | {color2}{text2:^{count}}{RESET_FORMAT} |
|
||||||
|
\=======================================================/
|
||||||
"""
|
"""
|
||||||
)[1:]
|
)[1:]
|
||||||
print(footer, end="")
|
print(footer, end="")
|
||||||
|
|
||||||
|
|
||||||
def print_blank_footer() -> None:
|
class BaseMenu(ABC):
|
||||||
print("╚═══════════════════════════════════════════════════════╝")
|
NAVI_OPTIONS = {"quit": ["q"], "back": ["b"], "back_help": ["b", "h"]}
|
||||||
|
|
||||||
|
def __init__(
|
||||||
class PostInitCaller(type):
|
self,
|
||||||
def __call__(cls, *args, **kwargs):
|
options: Dict[int, Any],
|
||||||
obj = type.__call__(cls, *args, **kwargs)
|
options_offset: int = 0,
|
||||||
obj.__post_init__()
|
header: bool = True,
|
||||||
return obj
|
footer_type: Literal[
|
||||||
|
"QUIT_FOOTER", "BACK_FOOTER", "BACK_HELP_FOOTER"
|
||||||
|
] = QUIT_FOOTER,
|
||||||
# noinspection PyUnusedLocal
|
):
|
||||||
# noinspection PyMethodMayBeStatic
|
self.options = options
|
||||||
class BaseMenu(metaclass=PostInitCaller):
|
self.options_offset = options_offset
|
||||||
options: Dict[str, Option] = {}
|
self.header = header
|
||||||
options_offset: int = 0
|
self.footer_type = footer_type
|
||||||
default_option: Option = None
|
|
||||||
input_label_txt: str = "Perform action"
|
|
||||||
header: bool = False
|
|
||||||
previous_menu: Type[BaseMenu] | None = None
|
|
||||||
help_menu: Type[BaseMenu] | None = None
|
|
||||||
footer_type: FooterType = FooterType.BACK
|
|
||||||
|
|
||||||
def __init__(self, **kwargs) -> None:
|
|
||||||
if type(self) is BaseMenu:
|
|
||||||
raise NotImplementedError("BaseMenu cannot be instantiated directly.")
|
|
||||||
|
|
||||||
def __post_init__(self) -> None:
|
|
||||||
self.set_previous_menu(self.previous_menu)
|
|
||||||
self.set_options()
|
|
||||||
|
|
||||||
# conditionally add options based on footer type
|
|
||||||
if self.footer_type is FooterType.QUIT:
|
|
||||||
self.options["q"] = Option(method=self.__exit)
|
|
||||||
if self.footer_type is FooterType.BACK:
|
|
||||||
self.options["b"] = Option(method=self.__go_back)
|
|
||||||
if self.footer_type is FooterType.BACK_HELP:
|
|
||||||
self.options["b"] = Option(method=self.__go_back)
|
|
||||||
self.options["h"] = Option(method=self.__go_to_help)
|
|
||||||
# if defined, add the default option to the options dict
|
|
||||||
if self.default_option is not None:
|
|
||||||
self.options[""] = self.default_option
|
|
||||||
|
|
||||||
def __go_back(self, **kwargs) -> None:
|
|
||||||
if self.previous_menu is None:
|
|
||||||
return
|
|
||||||
self.previous_menu().run()
|
|
||||||
|
|
||||||
def __go_to_help(self, **kwargs) -> None:
|
|
||||||
if self.help_menu is None:
|
|
||||||
return
|
|
||||||
self.help_menu(previous_menu=self.__class__).run()
|
|
||||||
|
|
||||||
def __exit(self, **kwargs) -> None:
|
|
||||||
Logger.print_ok("###### Happy printing!", False)
|
|
||||||
sys.exit(0)
|
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def set_previous_menu(self, previous_menu: Type[BaseMenu] | None) -> None:
|
def print_menu(self):
|
||||||
raise NotImplementedError
|
raise NotImplementedError("Subclasses must implement the print_menu method")
|
||||||
|
|
||||||
@abstractmethod
|
def print_footer(self):
|
||||||
def set_options(self) -> None:
|
footer_type_map = {
|
||||||
raise NotImplementedError
|
QUIT_FOOTER: print_quit_footer,
|
||||||
|
BACK_FOOTER: print_back_footer,
|
||||||
|
BACK_HELP_FOOTER: print_back_help_footer,
|
||||||
|
}
|
||||||
|
footer_function = footer_type_map.get(self.footer_type, print_quit_footer)
|
||||||
|
footer_function()
|
||||||
|
|
||||||
@abstractmethod
|
def display(self):
|
||||||
def print_menu(self) -> None:
|
# clear()
|
||||||
raise NotImplementedError
|
|
||||||
|
|
||||||
def print_footer(self) -> None:
|
|
||||||
if self.footer_type is FooterType.QUIT:
|
|
||||||
print_quit_footer()
|
|
||||||
elif self.footer_type is FooterType.BACK:
|
|
||||||
print_back_footer()
|
|
||||||
elif self.footer_type is FooterType.BACK_HELP:
|
|
||||||
print_back_help_footer()
|
|
||||||
elif self.footer_type is FooterType.BLANK:
|
|
||||||
print_blank_footer()
|
|
||||||
else:
|
|
||||||
raise NotImplementedError("FooterType not correctly implemented!")
|
|
||||||
|
|
||||||
def display_menu(self) -> None:
|
|
||||||
if self.header:
|
if self.header:
|
||||||
print_header()
|
print_header()
|
||||||
self.print_menu()
|
self.print_menu()
|
||||||
self.print_footer()
|
self.print_footer()
|
||||||
|
|
||||||
def run(self) -> None:
|
def handle_user_input(self):
|
||||||
"""Start the menu lifecycle. When this function returns, the lifecycle of the menu ends."""
|
while True:
|
||||||
try:
|
choice = input(f"{COLOR_CYAN}###### Perform action: {RESET_FORMAT}")
|
||||||
self.display_menu()
|
|
||||||
option = get_selection_input(self.input_label_txt, self.options)
|
|
||||||
selected_option: Option = self.options.get(option)
|
|
||||||
|
|
||||||
selected_option.method(
|
if choice.isdigit() and 0 <= int(choice) < len(self.options):
|
||||||
opt_index=selected_option.opt_index,
|
return choice
|
||||||
opt_data=selected_option.opt_data,
|
elif choice.isalpha() and (
|
||||||
|
self.footer_type in self.NAVI_OPTIONS
|
||||||
|
and choice.lower() in self.NAVI_OPTIONS[self.footer_type]
|
||||||
|
):
|
||||||
|
return choice
|
||||||
|
else:
|
||||||
|
error_msg = (
|
||||||
|
"Invalid input!"
|
||||||
|
if choice.isalpha() or (not self.options and len(self.options) < 1)
|
||||||
|
else f"Invalid input! Select a number between {min(self.options)} and {max(self.options)}."
|
||||||
|
)
|
||||||
|
Logger.print_error(error_msg, False)
|
||||||
|
|
||||||
|
def start(self):
|
||||||
|
while True:
|
||||||
|
self.display()
|
||||||
|
choice = self.handle_user_input()
|
||||||
|
|
||||||
|
if choice == "q":
|
||||||
|
Logger.print_ok("###### Happy printing!", False)
|
||||||
|
sys.exit(0)
|
||||||
|
elif choice == "b":
|
||||||
|
return
|
||||||
|
elif choice == "p":
|
||||||
|
print("help!")
|
||||||
|
else:
|
||||||
|
self.execute_option(int(choice))
|
||||||
|
|
||||||
|
def execute_option(self, choice):
|
||||||
|
option = self.options.get(choice, None)
|
||||||
|
|
||||||
|
if isinstance(option, type) and issubclass(option, BaseMenu):
|
||||||
|
self.navigate_to_submenu(option)
|
||||||
|
elif callable(option):
|
||||||
|
option(opt_index=choice)
|
||||||
|
elif option is None:
|
||||||
|
raise NotImplementedError(f"No implementation for option {choice}")
|
||||||
|
else:
|
||||||
|
raise TypeError(
|
||||||
|
f"Type {type(option)} of option {choice} not of type BaseMenu or Method"
|
||||||
)
|
)
|
||||||
|
|
||||||
self.run()
|
def navigate_to_submenu(self, submenu_class):
|
||||||
|
submenu = submenu_class()
|
||||||
except Exception as e:
|
submenu.previous_menu = self
|
||||||
Logger.print_error(
|
submenu.start()
|
||||||
f"An unexpected error occured:\n{e}\n{traceback.format_exc()}"
|
|
||||||
)
|
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
# ======================================================================= #
|
# ======================================================================= #
|
||||||
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
||||||
# #
|
# #
|
||||||
@@ -6,93 +8,91 @@
|
|||||||
# #
|
# #
|
||||||
# This file may be distributed under the terms of the GNU GPLv3 license #
|
# This file may be distributed under the terms of the GNU GPLv3 license #
|
||||||
# ======================================================================= #
|
# ======================================================================= #
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import textwrap
|
import textwrap
|
||||||
from typing import Type
|
|
||||||
|
|
||||||
from components.crowsnest.crowsnest import install_crowsnest
|
from kiauh.core.menus import BACK_FOOTER
|
||||||
from components.klipper import klipper_setup
|
from kiauh.core.menus.base_menu import BaseMenu
|
||||||
from components.klipperscreen.klipperscreen import install_klipperscreen
|
from kiauh.modules.klipper import klipper_setup
|
||||||
from components.moonraker import moonraker_setup
|
from kiauh.modules.mainsail import mainsail_setup
|
||||||
from components.webui_client import client_setup
|
from kiauh.modules.moonraker import moonraker_setup
|
||||||
from components.webui_client.client_config import client_config_setup
|
from kiauh.utils.constants import COLOR_GREEN, RESET_FORMAT
|
||||||
from components.webui_client.fluidd_data import FluiddData
|
|
||||||
from components.webui_client.mainsail_data import MainsailData
|
|
||||||
from core.constants import COLOR_GREEN, RESET_FORMAT
|
|
||||||
from core.menus import Option
|
|
||||||
from core.menus.base_menu import BaseMenu
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyUnusedLocal
|
# noinspection PyUnusedLocal
|
||||||
# noinspection PyMethodMayBeStatic
|
# noinspection PyMethodMayBeStatic
|
||||||
class InstallMenu(BaseMenu):
|
class InstallMenu(BaseMenu):
|
||||||
def __init__(self, previous_menu: Type[BaseMenu] | None = None) -> None:
|
def __init__(self):
|
||||||
super().__init__()
|
super().__init__(
|
||||||
self.previous_menu: Type[BaseMenu] | None = previous_menu
|
header=True,
|
||||||
|
options={
|
||||||
|
1: self.install_klipper,
|
||||||
|
2: self.install_moonraker,
|
||||||
|
3: self.install_mainsail,
|
||||||
|
4: self.install_fluidd,
|
||||||
|
5: self.install_klipperscreen,
|
||||||
|
6: self.install_pretty_gcode,
|
||||||
|
7: self.install_telegram_bot,
|
||||||
|
8: self.install_obico,
|
||||||
|
9: self.install_octoeverywhere,
|
||||||
|
10: self.install_mobileraker,
|
||||||
|
11: self.install_crowsnest,
|
||||||
|
},
|
||||||
|
footer_type=BACK_FOOTER,
|
||||||
|
)
|
||||||
|
|
||||||
def set_previous_menu(self, previous_menu: Type[BaseMenu] | None) -> None:
|
def print_menu(self):
|
||||||
from core.menus.main_menu import MainMenu
|
|
||||||
|
|
||||||
self.previous_menu = previous_menu if previous_menu is not None else MainMenu
|
|
||||||
|
|
||||||
def set_options(self) -> None:
|
|
||||||
self.options = {
|
|
||||||
"1": Option(method=self.install_klipper),
|
|
||||||
"2": Option(method=self.install_moonraker),
|
|
||||||
"3": Option(method=self.install_mainsail),
|
|
||||||
"4": Option(method=self.install_fluidd),
|
|
||||||
"5": Option(method=self.install_mainsail_config),
|
|
||||||
"6": Option(method=self.install_fluidd_config),
|
|
||||||
"7": Option(method=self.install_klipperscreen),
|
|
||||||
"8": Option(method=self.install_crowsnest),
|
|
||||||
}
|
|
||||||
|
|
||||||
def print_menu(self) -> None:
|
|
||||||
header = " [ Installation Menu ] "
|
header = " [ Installation Menu ] "
|
||||||
color = COLOR_GREEN
|
color = COLOR_GREEN
|
||||||
count = 62 - len(color) - len(RESET_FORMAT)
|
count = 62 - len(color) - len(RESET_FORMAT)
|
||||||
menu = textwrap.dedent(
|
menu = textwrap.dedent(
|
||||||
f"""
|
f"""
|
||||||
╔═══════════════════════════════════════════════════════╗
|
/=======================================================\\
|
||||||
║ {color}{header:~^{count}}{RESET_FORMAT} ║
|
| {color}{header:~^{count}}{RESET_FORMAT} |
|
||||||
╟───────────────────────────┬───────────────────────────╢
|
|-------------------------------------------------------|
|
||||||
║ Firmware & API: │ Touchscreen GUI: ║
|
| Firmware & API: | Other: |
|
||||||
║ 1) [Klipper] │ 7) [KlipperScreen] ║
|
| 1) [Klipper] | 6) [PrettyGCode] |
|
||||||
║ 2) [Moonraker] │ ║
|
| 2) [Moonraker] | 7) [Telegram Bot] |
|
||||||
║ │ Webcam Streamer: ║
|
| | 8) $(obico_install_title) |
|
||||||
║ Webinterface: │ 8) [Crowsnest] ║
|
| Klipper Webinterface: | 9) [OctoEverywhere] |
|
||||||
║ 3) [Mainsail] │ ║
|
| 3) [Mainsail] | 10) [Mobileraker] |
|
||||||
║ 4) [Fluidd] │ ║
|
| 4) [Fluidd] | |
|
||||||
║ │ ║
|
| | Webcam Streamer: |
|
||||||
║ Client-Config: │ ║
|
| Touchscreen GUI: | 11) [Crowsnest] |
|
||||||
║ 5) [Mainsail-Config] │ ║
|
| 5) [KlipperScreen] | |
|
||||||
║ 6) [Fluidd-Config] │ ║
|
|
||||||
╟───────────────────────────┴───────────────────────────╢
|
|
||||||
"""
|
"""
|
||||||
)[1:]
|
)[1:]
|
||||||
print(menu, end="")
|
print(menu, end="")
|
||||||
|
|
||||||
def install_klipper(self, **kwargs) -> None:
|
def install_klipper(self, **kwargs):
|
||||||
klipper_setup.install_klipper()
|
klipper_setup.install_klipper()
|
||||||
|
|
||||||
def install_moonraker(self, **kwargs) -> None:
|
def install_moonraker(self, **kwargs):
|
||||||
moonraker_setup.install_moonraker()
|
moonraker_setup.install_moonraker()
|
||||||
|
|
||||||
def install_mainsail(self, **kwargs) -> None:
|
def install_mainsail(self, **kwargs):
|
||||||
client_setup.install_client(MainsailData())
|
mainsail_setup.install_mainsail()
|
||||||
|
|
||||||
def install_mainsail_config(self, **kwargs) -> None:
|
def install_fluidd(self, **kwargs):
|
||||||
client_config_setup.install_client_config(MainsailData())
|
print("install_fluidd")
|
||||||
|
|
||||||
def install_fluidd(self, **kwargs) -> None:
|
def install_klipperscreen(self, **kwargs):
|
||||||
client_setup.install_client(FluiddData())
|
print("install_klipperscreen")
|
||||||
|
|
||||||
def install_fluidd_config(self, **kwargs) -> None:
|
def install_pretty_gcode(self, **kwargs):
|
||||||
client_config_setup.install_client_config(FluiddData())
|
print("install_pretty_gcode")
|
||||||
|
|
||||||
def install_klipperscreen(self, **kwargs) -> None:
|
def install_telegram_bot(self, **kwargs):
|
||||||
install_klipperscreen()
|
print("install_telegram_bot")
|
||||||
|
|
||||||
def install_crowsnest(self, **kwargs) -> None:
|
def install_obico(self, **kwargs):
|
||||||
install_crowsnest()
|
print("install_obico")
|
||||||
|
|
||||||
|
def install_octoeverywhere(self, **kwargs):
|
||||||
|
print("install_octoeverywhere")
|
||||||
|
|
||||||
|
def install_mobileraker(self, **kwargs):
|
||||||
|
print("install_mobileraker")
|
||||||
|
|
||||||
|
def install_crowsnest(self, **kwargs):
|
||||||
|
print("install_crowsnest")
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
# ======================================================================= #
|
# ======================================================================= #
|
||||||
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
||||||
# #
|
# #
|
||||||
@@ -6,183 +8,120 @@
|
|||||||
# #
|
# #
|
||||||
# This file may be distributed under the terms of the GNU GPLv3 license #
|
# This file may be distributed under the terms of the GNU GPLv3 license #
|
||||||
# ======================================================================= #
|
# ======================================================================= #
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import sys
|
|
||||||
import textwrap
|
import textwrap
|
||||||
from typing import Callable, Type
|
|
||||||
|
|
||||||
from components.crowsnest.crowsnest import get_crowsnest_status
|
from kiauh.core.menus import QUIT_FOOTER
|
||||||
from components.klipper.klipper_utils import get_klipper_status
|
from kiauh.core.menus.advanced_menu import AdvancedMenu
|
||||||
from components.klipperscreen.klipperscreen import get_klipperscreen_status
|
from kiauh.core.menus.base_menu import BaseMenu
|
||||||
from components.log_uploads.menus.log_upload_menu import LogUploadMenu
|
from kiauh.core.menus.install_menu import InstallMenu
|
||||||
from components.moonraker.moonraker_utils import get_moonraker_status
|
from kiauh.core.menus.remove_menu import RemoveMenu
|
||||||
from components.webui_client.client_utils import (
|
from kiauh.core.menus.settings_menu import SettingsMenu
|
||||||
get_client_status,
|
from kiauh.core.menus.update_menu import UpdateMenu
|
||||||
get_current_client_config,
|
from kiauh.modules.klipper.klipper_utils import get_klipper_status
|
||||||
)
|
from kiauh.modules.log_uploads.menus.log_upload_menu import LogUploadMenu
|
||||||
from components.webui_client.fluidd_data import FluiddData
|
from kiauh.modules.mainsail.mainsail_utils import get_mainsail_status
|
||||||
from components.webui_client.mainsail_data import MainsailData
|
from kiauh.modules.moonraker.moonraker_utils import get_moonraker_status
|
||||||
from core.constants import (
|
from kiauh.utils.constants import (
|
||||||
COLOR_CYAN,
|
|
||||||
COLOR_GREEN,
|
|
||||||
COLOR_MAGENTA,
|
COLOR_MAGENTA,
|
||||||
COLOR_RED,
|
COLOR_CYAN,
|
||||||
COLOR_YELLOW,
|
|
||||||
RESET_FORMAT,
|
RESET_FORMAT,
|
||||||
|
COLOR_RED,
|
||||||
|
COLOR_GREEN,
|
||||||
|
COLOR_YELLOW,
|
||||||
)
|
)
|
||||||
from core.logger import Logger
|
|
||||||
from core.menus import FooterType
|
|
||||||
from core.menus.advanced_menu import AdvancedMenu
|
|
||||||
from core.menus.backup_menu import BackupMenu
|
|
||||||
from core.menus.base_menu import BaseMenu, Option
|
|
||||||
from core.menus.install_menu import InstallMenu
|
|
||||||
from core.menus.remove_menu import RemoveMenu
|
|
||||||
from core.menus.settings_menu import SettingsMenu
|
|
||||||
from core.menus.update_menu import UpdateMenu
|
|
||||||
from core.types import ComponentStatus, StatusMap, StatusText
|
|
||||||
from extensions.extensions_menu import ExtensionsMenu
|
|
||||||
from utils.common import get_kiauh_version, trunc_string
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyUnusedLocal
|
|
||||||
# noinspection PyMethodMayBeStatic
|
|
||||||
class MainMenu(BaseMenu):
|
class MainMenu(BaseMenu):
|
||||||
def __init__(self) -> None:
|
def __init__(self):
|
||||||
super().__init__()
|
super().__init__(
|
||||||
|
header=True,
|
||||||
self.header: bool = True
|
options={
|
||||||
self.footer_type: FooterType = FooterType.QUIT
|
0: LogUploadMenu,
|
||||||
|
1: InstallMenu,
|
||||||
self.version = ""
|
2: UpdateMenu,
|
||||||
self.kl_status, self.kl_owner, self.kl_repo = "", "", ""
|
3: RemoveMenu,
|
||||||
self.mr_status, self.mr_owner, self.mr_repo = "", "", ""
|
4: AdvancedMenu,
|
||||||
self.ms_status, self.fl_status, self.ks_status = "", "", ""
|
5: None,
|
||||||
self.cn_status, self.cc_status = "", ""
|
6: SettingsMenu,
|
||||||
self._init_status()
|
},
|
||||||
|
footer_type=QUIT_FOOTER,
|
||||||
def set_previous_menu(self, previous_menu: Type[BaseMenu] | None) -> None:
|
|
||||||
"""MainMenu does not have a previous menu"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
def set_options(self) -> None:
|
|
||||||
self.options = {
|
|
||||||
"0": Option(method=self.log_upload_menu),
|
|
||||||
"1": Option(method=self.install_menu),
|
|
||||||
"2": Option(method=self.update_menu),
|
|
||||||
"3": Option(method=self.remove_menu),
|
|
||||||
"4": Option(method=self.advanced_menu),
|
|
||||||
"5": Option(method=self.backup_menu),
|
|
||||||
"e": Option(method=self.extension_menu),
|
|
||||||
"s": Option(method=self.settings_menu),
|
|
||||||
}
|
|
||||||
|
|
||||||
def _init_status(self) -> None:
|
|
||||||
status_vars = ["kl", "mr", "ms", "fl", "ks", "cn"]
|
|
||||||
for var in status_vars:
|
|
||||||
setattr(
|
|
||||||
self,
|
|
||||||
f"{var}_status",
|
|
||||||
f"{COLOR_RED}Not installed{RESET_FORMAT}",
|
|
||||||
)
|
)
|
||||||
|
self.kl_status = ""
|
||||||
|
self.kl_repo = ""
|
||||||
|
self.mr_status = ""
|
||||||
|
self.mr_repo = ""
|
||||||
|
self.ms_status = ""
|
||||||
|
self.fl_status = ""
|
||||||
|
self.ks_status = ""
|
||||||
|
self.mb_status = ""
|
||||||
|
self.cn_status = ""
|
||||||
|
self.tg_status = ""
|
||||||
|
self.ob_status = ""
|
||||||
|
self.oe_status = ""
|
||||||
|
self.init_status()
|
||||||
|
|
||||||
def _fetch_status(self) -> None:
|
def init_status(self) -> None:
|
||||||
self.version = get_kiauh_version()
|
status_vars = ["kl", "mr", "ms", "fl", "ks", "mb", "cn", "tg", "ob", "oe"]
|
||||||
self._get_component_status("kl", get_klipper_status)
|
for var in status_vars:
|
||||||
self._get_component_status("mr", get_moonraker_status)
|
setattr(self, f"{var}_status", f"{COLOR_RED}Not installed!{RESET_FORMAT}")
|
||||||
self._get_component_status("ms", get_client_status, MainsailData())
|
|
||||||
self._get_component_status("fl", get_client_status, FluiddData())
|
|
||||||
self._get_component_status("ks", get_klipperscreen_status)
|
|
||||||
self._get_component_status("cn", get_crowsnest_status)
|
|
||||||
self.cc_status = get_current_client_config()
|
|
||||||
|
|
||||||
def _get_component_status(self, name: str, status_fn: Callable, *args) -> None:
|
def fetch_status(self) -> None:
|
||||||
status_data: ComponentStatus = status_fn(*args)
|
# klipper
|
||||||
code: int = status_data.status
|
klipper_status = get_klipper_status()
|
||||||
status: StatusText = StatusMap[code]
|
kl_status = klipper_status.get("status")
|
||||||
owner: str = trunc_string(status_data.owner, 23)
|
kl_code = klipper_status.get("status_code")
|
||||||
repo: str = trunc_string(status_data.repo, 23)
|
kl_instances = f" {klipper_status.get('instances')}" if kl_code == 1 else ""
|
||||||
instance_count: int = status_data.instances
|
self.kl_status = self.format_status_by_code(kl_code, kl_status, kl_instances)
|
||||||
|
self.kl_repo = f"{COLOR_CYAN}{klipper_status.get('repo')}{RESET_FORMAT}"
|
||||||
|
# moonraker
|
||||||
|
moonraker_status = get_moonraker_status()
|
||||||
|
mr_status = moonraker_status.get("status")
|
||||||
|
mr_code = moonraker_status.get("status_code")
|
||||||
|
mr_instances = f" {moonraker_status.get('instances')}" if mr_code == 1 else ""
|
||||||
|
self.mr_status = self.format_status_by_code(mr_code, mr_status, mr_instances)
|
||||||
|
self.mr_repo = f"{COLOR_CYAN}{moonraker_status.get('repo')}{RESET_FORMAT}"
|
||||||
|
# mainsail
|
||||||
|
self.ms_status = get_mainsail_status()
|
||||||
|
|
||||||
count_txt: str = ""
|
def format_status_by_code(self, code: int, status: str, count: str) -> str:
|
||||||
if instance_count > 0 and code == 2:
|
if code == 1:
|
||||||
count_txt = f": {instance_count}"
|
return f"{COLOR_GREEN}{status}{count}{RESET_FORMAT}"
|
||||||
|
|
||||||
setattr(self, f"{name}_status", self._format_by_code(code, status, count_txt))
|
|
||||||
setattr(self, f"{name}_owner", f"{COLOR_CYAN}{owner}{RESET_FORMAT}")
|
|
||||||
setattr(self, f"{name}_repo", f"{COLOR_CYAN}{repo}{RESET_FORMAT}")
|
|
||||||
|
|
||||||
def _format_by_code(self, code: int, status: str, count: str) -> str:
|
|
||||||
color = COLOR_RED
|
|
||||||
if code == 0:
|
|
||||||
color = COLOR_RED
|
|
||||||
elif code == 1:
|
|
||||||
color = COLOR_YELLOW
|
|
||||||
elif code == 2:
|
elif code == 2:
|
||||||
color = COLOR_GREEN
|
return f"{COLOR_RED}{status}{count}{RESET_FORMAT}"
|
||||||
|
|
||||||
return f"{color}{status}{count}{RESET_FORMAT}"
|
return f"{COLOR_YELLOW}{status}{count}{RESET_FORMAT}"
|
||||||
|
|
||||||
def print_menu(self) -> None:
|
def print_menu(self):
|
||||||
self._fetch_status()
|
self.fetch_status()
|
||||||
|
|
||||||
header = " [ Main Menu ] "
|
header = " [ Main Menu ] "
|
||||||
footer1 = f"{COLOR_CYAN}{self.version}{RESET_FORMAT}"
|
footer1 = "KIAUH v6.0.0"
|
||||||
footer2 = f"Changelog: {COLOR_MAGENTA}https://git.io/JnmlX{RESET_FORMAT}"
|
footer2 = f"Changelog: {COLOR_MAGENTA}https://git.io/JnmlX{RESET_FORMAT}"
|
||||||
color = COLOR_CYAN
|
color = COLOR_CYAN
|
||||||
count = 62 - len(color) - len(RESET_FORMAT)
|
count = 62 - len(color) - len(RESET_FORMAT)
|
||||||
pad1 = 32
|
|
||||||
pad2 = 26
|
|
||||||
menu = textwrap.dedent(
|
menu = textwrap.dedent(
|
||||||
f"""
|
f"""
|
||||||
╔═══════════════════════════════════════════════════════╗
|
/=======================================================\\
|
||||||
║ {color}{header:~^{count}}{RESET_FORMAT} ║
|
| {color}{header:~^{count}}{RESET_FORMAT} |
|
||||||
╟──────────────────┬────────────────────────────────────╢
|
|-------------------------------------------------------|
|
||||||
║ 0) [Log-Upload] │ Klipper: {self.kl_status:<{pad1}} ║
|
| 0) [Log-Upload] | Klipper: {self.kl_status:<32} |
|
||||||
║ │ Owner: {self.kl_owner:<{pad1}} ║
|
| | Repo: {self.kl_repo:<32} |
|
||||||
║ 1) [Install] │ Repo: {self.kl_repo:<{pad1}} ║
|
| 1) [Install] |------------------------------------|
|
||||||
║ 2) [Update] ├────────────────────────────────────╢
|
| 2) [Update] | Moonraker: {self.mr_status:<32} |
|
||||||
║ 3) [Remove] │ Moonraker: {self.mr_status:<{pad1}} ║
|
| 3) [Remove] | Repo: {self.mr_repo:<32} |
|
||||||
║ 4) [Advanced] │ Owner: {self.mr_owner:<{pad1}} ║
|
| 4) [Advanced] |------------------------------------|
|
||||||
║ 5) [Backup] │ Repo: {self.mr_repo:<{pad1}} ║
|
| 5) [Backup] | Mainsail: {self.ms_status:<26} |
|
||||||
║ ├────────────────────────────────────╢
|
| | Fluidd: {self.fl_status:<26} |
|
||||||
║ S) [Settings] │ Mainsail: {self.ms_status:<{pad2}} ║
|
| 6) [Settings] | KlipperScreen: {self.ks_status:<26} |
|
||||||
║ │ Fluidd: {self.fl_status:<{pad2}} ║
|
| | Mobileraker: {self.mb_status:<26} |
|
||||||
║ Community: │ Client-Config: {self.cc_status:<{pad2}} ║
|
| | |
|
||||||
║ E) [Extensions] │ ║
|
| | Crowsnest: {self.cn_status:<26} |
|
||||||
║ │ KlipperScreen: {self.ks_status:<{pad2}} ║
|
| | Telegram Bot: {self.tg_status:<26} |
|
||||||
║ │ Crowsnest: {self.cn_status:<{pad2}} ║
|
| | Obico: {self.ob_status:<26} |
|
||||||
╟──────────────────┼────────────────────────────────────╢
|
| | OctoEverywhere: {self.oe_status:<26} |
|
||||||
║ {footer1:^25} │ {footer2:^43} ║
|
|-------------------------------------------------------|
|
||||||
╟──────────────────┴────────────────────────────────────╢
|
| {COLOR_CYAN}{footer1:^16}{RESET_FORMAT} | {footer2:^43} |
|
||||||
"""
|
"""
|
||||||
)[1:]
|
)[1:]
|
||||||
print(menu, end="")
|
print(menu, end="")
|
||||||
|
|
||||||
def exit(self, **kwargs) -> None:
|
|
||||||
Logger.print_ok("###### Happy printing!", False)
|
|
||||||
sys.exit(0)
|
|
||||||
|
|
||||||
def log_upload_menu(self, **kwargs) -> None:
|
|
||||||
LogUploadMenu().run()
|
|
||||||
|
|
||||||
def install_menu(self, **kwargs) -> None:
|
|
||||||
InstallMenu(previous_menu=self.__class__).run()
|
|
||||||
|
|
||||||
def update_menu(self, **kwargs) -> None:
|
|
||||||
UpdateMenu(previous_menu=self.__class__).run()
|
|
||||||
|
|
||||||
def remove_menu(self, **kwargs) -> None:
|
|
||||||
RemoveMenu(previous_menu=self.__class__).run()
|
|
||||||
|
|
||||||
def advanced_menu(self, **kwargs) -> None:
|
|
||||||
AdvancedMenu(previous_menu=self.__class__).run()
|
|
||||||
|
|
||||||
def backup_menu(self, **kwargs) -> None:
|
|
||||||
BackupMenu(previous_menu=self.__class__).run()
|
|
||||||
|
|
||||||
def settings_menu(self, **kwargs) -> None:
|
|
||||||
SettingsMenu(previous_menu=self.__class__).run()
|
|
||||||
|
|
||||||
def extension_menu(self, **kwargs) -> None:
|
|
||||||
ExtensionsMenu(previous_menu=self.__class__).run()
|
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
# ======================================================================= #
|
# ======================================================================= #
|
||||||
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
||||||
# #
|
# #
|
||||||
@@ -6,84 +8,97 @@
|
|||||||
# #
|
# #
|
||||||
# This file may be distributed under the terms of the GNU GPLv3 license #
|
# This file may be distributed under the terms of the GNU GPLv3 license #
|
||||||
# ======================================================================= #
|
# ======================================================================= #
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import textwrap
|
import textwrap
|
||||||
from typing import Type
|
|
||||||
|
|
||||||
from components.crowsnest.crowsnest import remove_crowsnest
|
from kiauh.core.menus import BACK_FOOTER
|
||||||
from components.klipper.menus.klipper_remove_menu import KlipperRemoveMenu
|
from kiauh.core.menus.base_menu import BaseMenu
|
||||||
from components.klipperscreen.klipperscreen import remove_klipperscreen
|
from kiauh.modules.klipper.menus.klipper_remove_menu import KlipperRemoveMenu
|
||||||
from components.moonraker.menus.moonraker_remove_menu import (
|
from kiauh.modules.mainsail.menus.mainsail_remove_menu import MainsailRemoveMenu
|
||||||
MoonrakerRemoveMenu,
|
from kiauh.modules.moonraker.menus.moonraker_remove_menu import MoonrakerRemoveMenu
|
||||||
)
|
from kiauh.utils.constants import COLOR_RED, RESET_FORMAT
|
||||||
from components.webui_client.fluidd_data import FluiddData
|
|
||||||
from components.webui_client.mainsail_data import MainsailData
|
|
||||||
from components.webui_client.menus.client_remove_menu import ClientRemoveMenu
|
|
||||||
from core.constants import COLOR_RED, RESET_FORMAT
|
|
||||||
from core.menus import Option
|
|
||||||
from core.menus.base_menu import BaseMenu
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyUnusedLocal
|
# noinspection PyUnusedLocal
|
||||||
# noinspection PyMethodMayBeStatic
|
# noinspection PyMethodMayBeStatic
|
||||||
class RemoveMenu(BaseMenu):
|
class RemoveMenu(BaseMenu):
|
||||||
def __init__(self, previous_menu: Type[BaseMenu] | None = None) -> None:
|
def __init__(self):
|
||||||
super().__init__()
|
super().__init__(
|
||||||
self.previous_menu: Type[BaseMenu] | None = previous_menu
|
header=True,
|
||||||
|
options={
|
||||||
|
1: KlipperRemoveMenu,
|
||||||
|
2: MoonrakerRemoveMenu,
|
||||||
|
3: MainsailRemoveMenu,
|
||||||
|
5: self.remove_fluidd,
|
||||||
|
6: self.remove_klipperscreen,
|
||||||
|
7: self.remove_crowsnest,
|
||||||
|
8: self.remove_mjpgstreamer,
|
||||||
|
9: self.remove_pretty_gcode,
|
||||||
|
10: self.remove_telegram_bot,
|
||||||
|
11: self.remove_obico,
|
||||||
|
12: self.remove_octoeverywhere,
|
||||||
|
13: self.remove_mobileraker,
|
||||||
|
14: self.remove_nginx,
|
||||||
|
},
|
||||||
|
footer_type=BACK_FOOTER,
|
||||||
|
)
|
||||||
|
|
||||||
def set_previous_menu(self, previous_menu: Type[BaseMenu] | None) -> None:
|
def print_menu(self):
|
||||||
from core.menus.main_menu import MainMenu
|
|
||||||
|
|
||||||
self.previous_menu = previous_menu if previous_menu is not None else MainMenu
|
|
||||||
|
|
||||||
def set_options(self) -> None:
|
|
||||||
self.options = {
|
|
||||||
"1": Option(method=self.remove_klipper),
|
|
||||||
"2": Option(method=self.remove_moonraker),
|
|
||||||
"3": Option(method=self.remove_mainsail),
|
|
||||||
"4": Option(method=self.remove_fluidd),
|
|
||||||
"5": Option(method=self.remove_klipperscreen),
|
|
||||||
"6": Option(method=self.remove_crowsnest),
|
|
||||||
}
|
|
||||||
|
|
||||||
def print_menu(self) -> None:
|
|
||||||
header = " [ Remove Menu ] "
|
header = " [ Remove Menu ] "
|
||||||
color = COLOR_RED
|
color = COLOR_RED
|
||||||
count = 62 - len(color) - len(RESET_FORMAT)
|
count = 62 - len(color) - len(RESET_FORMAT)
|
||||||
menu = textwrap.dedent(
|
menu = textwrap.dedent(
|
||||||
f"""
|
f"""
|
||||||
╔═══════════════════════════════════════════════════════╗
|
/=======================================================\\
|
||||||
║ {color}{header:~^{count}}{RESET_FORMAT} ║
|
| {color}{header:~^{count}}{RESET_FORMAT} |
|
||||||
╟───────────────────────────────────────────────────────╢
|
|-------------------------------------------------------|
|
||||||
║ INFO: Configurations and/or any backups will be kept! ║
|
| INFO: Configurations and/or any backups will be kept! |
|
||||||
╟───────────────────────────┬───────────────────────────╢
|
|-------------------------------------------------------|
|
||||||
║ Firmware & API: │ Touchscreen GUI: ║
|
| Firmware & API: | Webcam Streamer: |
|
||||||
║ 1) [Klipper] │ 5) [KlipperScreen] ║
|
| 1) [Klipper] | 6) [Crowsnest] |
|
||||||
║ 2) [Moonraker] │ ║
|
| 2) [Moonraker] | 7) [MJPG-Streamer] |
|
||||||
║ │ Webcam Streamer: ║
|
| | |
|
||||||
║ Klipper Webinterface: │ 6) [Crowsnest] ║
|
| Klipper Webinterface: | Other: |
|
||||||
║ 3) [Mainsail] │ ║
|
| 3) [Mainsail] | 8) [PrettyGCode] |
|
||||||
║ 4) [Fluidd] │ ║
|
| 4) [Fluidd] | 9) [Telegram Bot] |
|
||||||
╟───────────────────────────┴───────────────────────────╢
|
| | 10) [Obico for Klipper] |
|
||||||
|
| Touchscreen GUI: | 11) [OctoEverywhere] |
|
||||||
|
| 5) [KlipperScreen] | 12) [Mobileraker] |
|
||||||
|
| | 13) [NGINX] |
|
||||||
|
| | |
|
||||||
"""
|
"""
|
||||||
)[1:]
|
)[1:]
|
||||||
print(menu, end="")
|
print(menu, end="")
|
||||||
|
|
||||||
def remove_klipper(self, **kwargs) -> None:
|
def remove_fluidd(self, **kwargs):
|
||||||
KlipperRemoveMenu(previous_menu=self.__class__).run()
|
print("remove_fluidd")
|
||||||
|
|
||||||
def remove_moonraker(self, **kwargs) -> None:
|
def remove_fluidd_config(self, **kwargs):
|
||||||
MoonrakerRemoveMenu(previous_menu=self.__class__).run()
|
print("remove_fluidd_config")
|
||||||
|
|
||||||
def remove_mainsail(self, **kwargs) -> None:
|
def remove_klipperscreen(self, **kwargs):
|
||||||
ClientRemoveMenu(previous_menu=self.__class__, client=MainsailData()).run()
|
print("remove_klipperscreen")
|
||||||
|
|
||||||
def remove_fluidd(self, **kwargs) -> None:
|
def remove_crowsnest(self, **kwargs):
|
||||||
ClientRemoveMenu(previous_menu=self.__class__, client=FluiddData()).run()
|
print("remove_crowsnest")
|
||||||
|
|
||||||
def remove_klipperscreen(self, **kwargs) -> None:
|
def remove_mjpgstreamer(self, **kwargs):
|
||||||
remove_klipperscreen()
|
print("remove_mjpgstreamer")
|
||||||
|
|
||||||
def remove_crowsnest(self, **kwargs) -> None:
|
def remove_pretty_gcode(self, **kwargs):
|
||||||
remove_crowsnest()
|
print("remove_pretty_gcode")
|
||||||
|
|
||||||
|
def remove_telegram_bot(self, **kwargs):
|
||||||
|
print("remove_telegram_bot")
|
||||||
|
|
||||||
|
def remove_obico(self, **kwargs):
|
||||||
|
print("remove_obico")
|
||||||
|
|
||||||
|
def remove_octoeverywhere(self, **kwargs):
|
||||||
|
print("remove_octoeverywhere")
|
||||||
|
|
||||||
|
def remove_mobileraker(self, **kwargs):
|
||||||
|
print("remove_mobileraker")
|
||||||
|
|
||||||
|
def remove_nginx(self, **kwargs):
|
||||||
|
print("remove_nginx")
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
# ======================================================================= #
|
# ======================================================================= #
|
||||||
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
||||||
# #
|
# #
|
||||||
@@ -6,184 +8,26 @@
|
|||||||
# #
|
# #
|
||||||
# This file may be distributed under the terms of the GNU GPLv3 license #
|
# This file may be distributed under the terms of the GNU GPLv3 license #
|
||||||
# ======================================================================= #
|
# ======================================================================= #
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import textwrap
|
from kiauh.core.menus.base_menu import BaseMenu
|
||||||
from typing import Literal, Tuple, Type
|
|
||||||
|
|
||||||
from core.constants import COLOR_CYAN, COLOR_GREEN, RESET_FORMAT
|
|
||||||
from core.logger import DialogType, Logger
|
|
||||||
from core.menus import Option
|
|
||||||
from core.menus.base_menu import BaseMenu
|
|
||||||
from core.settings.kiauh_settings import KiauhSettings, RepoSettings
|
|
||||||
from procedures.switch_repo import run_switch_repo_routine
|
|
||||||
from utils.input_utils import get_confirm, get_string_input
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyUnusedLocal
|
|
||||||
# noinspection PyMethodMayBeStatic
|
# noinspection PyMethodMayBeStatic
|
||||||
class SettingsMenu(BaseMenu):
|
class SettingsMenu(BaseMenu):
|
||||||
def __init__(self, previous_menu: Type[BaseMenu] | None = None) -> None:
|
def __init__(self):
|
||||||
super().__init__()
|
super().__init__(header=True, options={})
|
||||||
self.previous_menu: Type[BaseMenu] | None = previous_menu
|
|
||||||
self.klipper_repo: str | None = None
|
|
||||||
self.moonraker_repo: str | None = None
|
|
||||||
self.mainsail_unstable: bool | None = None
|
|
||||||
self.fluidd_unstable: bool | None = None
|
|
||||||
self.auto_backups_enabled: bool | None = None
|
|
||||||
self._load_settings()
|
|
||||||
|
|
||||||
def set_previous_menu(self, previous_menu: Type[BaseMenu] | None) -> None:
|
def print_menu(self):
|
||||||
from core.menus.main_menu import MainMenu
|
print("self")
|
||||||
|
|
||||||
self.previous_menu = previous_menu if previous_menu is not None else MainMenu
|
def execute_option_p(self):
|
||||||
|
# Implement the functionality for Option P
|
||||||
|
print("Executing Option P")
|
||||||
|
|
||||||
def set_options(self) -> None:
|
def execute_option_q(self):
|
||||||
self.options = {
|
# Implement the functionality for Option Q
|
||||||
"1": Option(method=self.set_klipper_repo),
|
print("Executing Option Q")
|
||||||
"2": Option(method=self.set_moonraker_repo),
|
|
||||||
"3": Option(method=self.toggle_mainsail_release),
|
|
||||||
"4": Option(method=self.toggle_fluidd_release),
|
|
||||||
"5": Option(method=self.toggle_backup_before_update),
|
|
||||||
}
|
|
||||||
|
|
||||||
def print_menu(self) -> None:
|
def execute_option_r(self):
|
||||||
header = " [ KIAUH Settings ] "
|
# Implement the functionality for Option R
|
||||||
color = COLOR_CYAN
|
print("Executing Option R")
|
||||||
count = 62 - len(color) - len(RESET_FORMAT)
|
|
||||||
checked = f"[{COLOR_GREEN}x{RESET_FORMAT}]"
|
|
||||||
unchecked = "[ ]"
|
|
||||||
o1 = checked if self.mainsail_unstable else unchecked
|
|
||||||
o2 = checked if self.fluidd_unstable else unchecked
|
|
||||||
o3 = checked if self.auto_backups_enabled else unchecked
|
|
||||||
menu = textwrap.dedent(
|
|
||||||
f"""
|
|
||||||
╔═══════════════════════════════════════════════════════╗
|
|
||||||
║ {color}{header:~^{count}}{RESET_FORMAT} ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ Klipper source repository: ║
|
|
||||||
║ ● {self.klipper_repo:<67} ║
|
|
||||||
║ ║
|
|
||||||
║ Moonraker source repository: ║
|
|
||||||
║ ● {self.moonraker_repo:<67} ║
|
|
||||||
║ ║
|
|
||||||
║ Install unstable Webinterface releases: ║
|
|
||||||
║ {o1} Mainsail ║
|
|
||||||
║ {o2} Fluidd ║
|
|
||||||
║ ║
|
|
||||||
║ Auto-Backup: ║
|
|
||||||
║ {o3} Automatic backup before update ║
|
|
||||||
║ ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
║ 1) Set Klipper source repository ║
|
|
||||||
║ 2) Set Moonraker source repository ║
|
|
||||||
║ ║
|
|
||||||
║ 3) Toggle unstable Mainsail releases ║
|
|
||||||
║ 4) Toggle unstable Fluidd releases ║
|
|
||||||
║ ║
|
|
||||||
║ 5) Toggle automatic backups before updates ║
|
|
||||||
╟───────────────────────────────────────────────────────╢
|
|
||||||
"""
|
|
||||||
)[1:]
|
|
||||||
print(menu, end="")
|
|
||||||
|
|
||||||
def _load_settings(self) -> None:
|
|
||||||
self.settings = KiauhSettings()
|
|
||||||
|
|
||||||
self._format_repo_str("klipper")
|
|
||||||
self._format_repo_str("moonraker")
|
|
||||||
|
|
||||||
self.auto_backups_enabled = self.settings.kiauh.backup_before_update
|
|
||||||
self.mainsail_unstable = self.settings.mainsail.unstable_releases
|
|
||||||
self.fluidd_unstable = self.settings.fluidd.unstable_releases
|
|
||||||
|
|
||||||
def _format_repo_str(self, repo_name: Literal["klipper", "moonraker"]) -> None:
|
|
||||||
repo: RepoSettings = self.settings[repo_name]
|
|
||||||
repo_str = f"{'/'.join(repo.repo_url.rsplit('/', 2)[-2:])}"
|
|
||||||
branch_str = f"({COLOR_CYAN}@ {repo.branch}{RESET_FORMAT})"
|
|
||||||
|
|
||||||
setattr(
|
|
||||||
self,
|
|
||||||
f"{repo_name}_repo",
|
|
||||||
f"{COLOR_CYAN}{repo_str}{RESET_FORMAT} {branch_str}",
|
|
||||||
)
|
|
||||||
|
|
||||||
def _gather_input(self) -> Tuple[str, str]:
|
|
||||||
Logger.print_dialog(
|
|
||||||
DialogType.ATTENTION,
|
|
||||||
[
|
|
||||||
"There is no input validation in place! Make sure your the input is "
|
|
||||||
"valid and has no typos or invalid characters! For the change to take "
|
|
||||||
"effect, the new repository will be cloned. A backup of the old "
|
|
||||||
"repository will be created.",
|
|
||||||
"\n\n",
|
|
||||||
"Make sure you don't have any ongoing prints running, as the services "
|
|
||||||
"will be restarted during this process! You will loose any ongoing print!",
|
|
||||||
],
|
|
||||||
)
|
|
||||||
repo = get_string_input(
|
|
||||||
"Enter new repository URL",
|
|
||||||
allow_special_chars=True,
|
|
||||||
)
|
|
||||||
branch = get_string_input(
|
|
||||||
"Enter new branch name",
|
|
||||||
allow_special_chars=True,
|
|
||||||
)
|
|
||||||
|
|
||||||
return repo, branch
|
|
||||||
|
|
||||||
def _set_repo(self, repo_name: Literal["klipper", "moonraker"]) -> None:
|
|
||||||
repo_url, branch = self._gather_input()
|
|
||||||
display_name = repo_name.capitalize()
|
|
||||||
Logger.print_dialog(
|
|
||||||
DialogType.CUSTOM,
|
|
||||||
[
|
|
||||||
f"New {display_name} repository URL:",
|
|
||||||
f"● {repo_url}",
|
|
||||||
f"New {display_name} repository branch:",
|
|
||||||
f"● {branch}",
|
|
||||||
],
|
|
||||||
)
|
|
||||||
|
|
||||||
if get_confirm("Apply changes?", allow_go_back=True):
|
|
||||||
repo: RepoSettings = self.settings[repo_name]
|
|
||||||
repo.repo_url = repo_url
|
|
||||||
repo.branch = branch
|
|
||||||
|
|
||||||
self.settings.save()
|
|
||||||
self._load_settings()
|
|
||||||
|
|
||||||
Logger.print_ok("Changes saved!")
|
|
||||||
else:
|
|
||||||
Logger.print_info(
|
|
||||||
f"Skipping change of {display_name} source repository ..."
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
Logger.print_status(f"Switching to {display_name}'s new source repository ...")
|
|
||||||
self._switch_repo(repo_name)
|
|
||||||
|
|
||||||
def _switch_repo(self, name: Literal["klipper", "moonraker"]) -> None:
|
|
||||||
repo: RepoSettings = self.settings[name]
|
|
||||||
run_switch_repo_routine(name, repo)
|
|
||||||
|
|
||||||
def set_klipper_repo(self, **kwargs) -> None:
|
|
||||||
self._set_repo("klipper")
|
|
||||||
|
|
||||||
def set_moonraker_repo(self, **kwargs) -> None:
|
|
||||||
self._set_repo("moonraker")
|
|
||||||
|
|
||||||
def toggle_mainsail_release(self, **kwargs) -> None:
|
|
||||||
self.mainsail_unstable = not self.mainsail_unstable
|
|
||||||
self.settings.mainsail.unstable_releases = self.mainsail_unstable
|
|
||||||
self.settings.save()
|
|
||||||
|
|
||||||
def toggle_fluidd_release(self, **kwargs) -> None:
|
|
||||||
self.fluidd_unstable = not self.fluidd_unstable
|
|
||||||
self.settings.fluidd.unstable_releases = self.fluidd_unstable
|
|
||||||
self.settings.save()
|
|
||||||
|
|
||||||
def toggle_backup_before_update(self, **kwargs) -> None:
|
|
||||||
self.auto_backups_enabled = not self.auto_backups_enabled
|
|
||||||
self.settings.kiauh.backup_before_update = self.auto_backups_enabled
|
|
||||||
self.settings.save()
|
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
# ======================================================================= #
|
# ======================================================================= #
|
||||||
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
# Copyright (C) 2020 - 2024 Dominik Willner <th33xitus@gmail.com> #
|
||||||
# #
|
# #
|
||||||
@@ -6,331 +8,155 @@
|
|||||||
# #
|
# #
|
||||||
# This file may be distributed under the terms of the GNU GPLv3 license #
|
# This file may be distributed under the terms of the GNU GPLv3 license #
|
||||||
# ======================================================================= #
|
# ======================================================================= #
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import textwrap
|
import textwrap
|
||||||
from typing import Callable, List, Type
|
|
||||||
|
|
||||||
from components.crowsnest.crowsnest import get_crowsnest_status, update_crowsnest
|
from kiauh.core.menus import BACK_FOOTER
|
||||||
from components.klipper.klipper_setup import update_klipper
|
from kiauh.core.menus.base_menu import BaseMenu
|
||||||
from components.klipper.klipper_utils import (
|
from kiauh.modules.klipper.klipper_setup import update_klipper
|
||||||
|
from kiauh.modules.klipper.klipper_utils import (
|
||||||
get_klipper_status,
|
get_klipper_status,
|
||||||
)
|
)
|
||||||
from components.klipperscreen.klipperscreen import (
|
from kiauh.modules.mainsail.mainsail_setup import update_mainsail
|
||||||
get_klipperscreen_status,
|
from kiauh.modules.mainsail.mainsail_utils import (
|
||||||
update_klipperscreen,
|
get_mainsail_local_version,
|
||||||
)
|
get_mainsail_remote_version,
|
||||||
from components.moonraker.moonraker_setup import update_moonraker
|
|
||||||
from components.moonraker.moonraker_utils import get_moonraker_status
|
|
||||||
from components.webui_client.client_config.client_config_setup import (
|
|
||||||
update_client_config,
|
|
||||||
)
|
|
||||||
from components.webui_client.client_setup import update_client
|
|
||||||
from components.webui_client.client_utils import (
|
|
||||||
get_client_config_status,
|
|
||||||
get_client_status,
|
|
||||||
)
|
|
||||||
from components.webui_client.fluidd_data import FluiddData
|
|
||||||
from components.webui_client.mainsail_data import MainsailData
|
|
||||||
from core.constants import (
|
|
||||||
COLOR_GREEN,
|
|
||||||
COLOR_RED,
|
|
||||||
COLOR_YELLOW,
|
|
||||||
RESET_FORMAT,
|
|
||||||
)
|
|
||||||
from core.logger import DialogType, Logger
|
|
||||||
from core.menus import Option
|
|
||||||
from core.menus.base_menu import BaseMenu
|
|
||||||
from core.spinner import Spinner
|
|
||||||
from core.types import ComponentStatus
|
|
||||||
from utils.input_utils import get_confirm
|
|
||||||
from utils.sys_utils import (
|
|
||||||
get_upgradable_packages,
|
|
||||||
update_system_package_lists,
|
|
||||||
upgrade_system_packages,
|
|
||||||
)
|
)
|
||||||
|
from kiauh.modules.moonraker.moonraker_setup import update_moonraker
|
||||||
|
from kiauh.modules.moonraker.moonraker_utils import get_moonraker_status
|
||||||
|
from kiauh.utils.constants import COLOR_GREEN, RESET_FORMAT, COLOR_YELLOW, COLOR_WHITE
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyUnusedLocal
|
# noinspection PyUnusedLocal
|
||||||
# noinspection PyMethodMayBeStatic
|
# noinspection PyMethodMayBeStatic
|
||||||
class UpdateMenu(BaseMenu):
|
class UpdateMenu(BaseMenu):
|
||||||
def __init__(self, previous_menu: Type[BaseMenu] | None = None) -> None:
|
def __init__(self):
|
||||||
super().__init__()
|
super().__init__(
|
||||||
self.previous_menu: Type[BaseMenu] | None = previous_menu
|
header=True,
|
||||||
|
options={
|
||||||
self.packages: List[str] = []
|
0: self.update_all,
|
||||||
self.package_count: int = 0
|
1: self.update_klipper,
|
||||||
|
2: self.update_moonraker,
|
||||||
self.klipper_local = self.klipper_remote = ""
|
3: self.update_mainsail,
|
||||||
self.moonraker_local = self.moonraker_remote = ""
|
4: self.update_fluidd,
|
||||||
self.mainsail_local = self.mainsail_remote = ""
|
5: self.update_klipperscreen,
|
||||||
self.mainsail_config_local = self.mainsail_config_remote = ""
|
6: self.update_pgc_for_klipper,
|
||||||
self.fluidd_local = self.fluidd_remote = ""
|
7: self.update_telegram_bot,
|
||||||
self.fluidd_config_local = self.fluidd_config_remote = ""
|
8: self.update_moonraker_obico,
|
||||||
self.klipperscreen_local = self.klipperscreen_remote = ""
|
9: self.update_octoeverywhere,
|
||||||
self.crowsnest_local = self.crowsnest_remote = ""
|
10: self.update_mobileraker,
|
||||||
|
11: self.update_crowsnest,
|
||||||
self.mainsail_data = MainsailData()
|
12: self.upgrade_system_packages,
|
||||||
self.fluidd_data = FluiddData()
|
|
||||||
self.status_data = {
|
|
||||||
"klipper": {
|
|
||||||
"display_name": "Klipper",
|
|
||||||
"installed": False,
|
|
||||||
"local": None,
|
|
||||||
"remote": None,
|
|
||||||
},
|
},
|
||||||
"moonraker": {
|
footer_type=BACK_FOOTER,
|
||||||
"display_name": "Moonraker",
|
)
|
||||||
"installed": False,
|
self.kl_local = f"{COLOR_WHITE}{RESET_FORMAT}"
|
||||||
"local": None,
|
self.kl_remote = f"{COLOR_WHITE}{RESET_FORMAT}"
|
||||||
"remote": None,
|
self.mr_local = f"{COLOR_WHITE}{RESET_FORMAT}"
|
||||||
},
|
self.mr_remote = f"{COLOR_WHITE}{RESET_FORMAT}"
|
||||||
"mainsail": {
|
self.ms_local = f"{COLOR_WHITE}{RESET_FORMAT}"
|
||||||
"display_name": "Mainsail",
|
self.ms_remote = f"{COLOR_WHITE}{RESET_FORMAT}"
|
||||||
"installed": False,
|
|
||||||
"local": None,
|
|
||||||
"remote": None,
|
|
||||||
},
|
|
||||||
"mainsail_config": {
|
|
||||||
"display_name": "Mainsail-Config",
|
|
||||||
"installed": False,
|
|
||||||
"local": None,
|
|
||||||
"remote": None,
|
|
||||||
},
|
|
||||||
"fluidd": {
|
|
||||||
"display_name": "Fluidd",
|
|
||||||
"installed": False,
|
|
||||||
"local": None,
|
|
||||||
"remote": None,
|
|
||||||
},
|
|
||||||
"fluidd_config": {
|
|
||||||
"display_name": "Fluidd-Config",
|
|
||||||
"installed": False,
|
|
||||||
"local": None,
|
|
||||||
"remote": None,
|
|
||||||
},
|
|
||||||
"klipperscreen": {
|
|
||||||
"display_name": "KlipperScreen",
|
|
||||||
"installed": False,
|
|
||||||
"local": None,
|
|
||||||
"remote": None,
|
|
||||||
},
|
|
||||||
"crowsnest": {
|
|
||||||
"display_name": "Crowsnest",
|
|
||||||
"installed": False,
|
|
||||||
"local": None,
|
|
||||||
"remote": None,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
def set_previous_menu(self, previous_menu: Type[BaseMenu] | None) -> None:
|
def print_menu(self):
|
||||||
from core.menus.main_menu import MainMenu
|
self.fetch_update_status()
|
||||||
|
|
||||||
self.previous_menu = previous_menu if previous_menu is not None else MainMenu
|
|
||||||
|
|
||||||
def set_options(self) -> None:
|
|
||||||
self.options = {
|
|
||||||
"a": Option(self.update_all),
|
|
||||||
"1": Option(self.update_klipper),
|
|
||||||
"2": Option(self.update_moonraker),
|
|
||||||
"3": Option(self.update_mainsail),
|
|
||||||
"4": Option(self.update_fluidd),
|
|
||||||
"5": Option(self.update_mainsail_config),
|
|
||||||
"6": Option(self.update_fluidd_config),
|
|
||||||
"7": Option(self.update_klipperscreen),
|
|
||||||
"8": Option(self.update_crowsnest),
|
|
||||||
"9": Option(self.upgrade_system_packages),
|
|
||||||
}
|
|
||||||
|
|
||||||
def print_menu(self) -> None:
|
|
||||||
spinner = Spinner("Loading update menu, please wait", color="green")
|
|
||||||
spinner.start()
|
|
||||||
|
|
||||||
self._fetch_update_status()
|
|
||||||
|
|
||||||
spinner.stop()
|
|
||||||
|
|
||||||
header = " [ Update Menu ] "
|
header = " [ Update Menu ] "
|
||||||
color = COLOR_GREEN
|
color = COLOR_GREEN
|
||||||
count = 62 - len(color) - len(RESET_FORMAT)
|
count = 62 - len(color) - len(RESET_FORMAT)
|
||||||
|
|
||||||
sysupgrades: str = "No upgrades available."
|
|
||||||
padding = 29
|
|
||||||
if self.package_count > 0:
|
|
||||||
sysupgrades = (
|
|
||||||
f"{COLOR_GREEN}{self.package_count} upgrades available!{RESET_FORMAT}"
|
|
||||||
)
|
|
||||||
padding = 38
|
|
||||||
|
|
||||||
menu = textwrap.dedent(
|
menu = textwrap.dedent(
|
||||||
f"""
|
f"""
|
||||||
╔═══════════════════════════════════════════════════════╗
|
/=======================================================\\
|
||||||
║ {color}{header:~^{count}}{RESET_FORMAT} ║
|
| {color}{header:~^{count}}{RESET_FORMAT} |
|
||||||
╟───────────────────────┬───────────────┬───────────────╢
|
|-------------------------------------------------------|
|
||||||
║ a) Update all │ │ ║
|
| 0) Update all | | |
|
||||||
║ │ Current: │ Latest: ║
|
| | Current: | Latest: |
|
||||||
║ Klipper & API: ├───────────────┼───────────────╢
|
| Klipper & API: |---------------|---------------|
|
||||||
║ 1) Klipper │ {self.klipper_local:<22} │ {self.klipper_remote:<22} ║
|
| 1) Klipper | {self.kl_local:<22} | {self.kl_remote:<22} |
|
||||||
║ 2) Moonraker │ {self.moonraker_local:<22} │ {self.moonraker_remote:<22} ║
|
| 2) Moonraker | {self.mr_local:<22} | {self.mr_remote:<22} |
|
||||||
║ │ │ ║
|
| | | |
|
||||||
║ Webinterface: ├───────────────┼───────────────╢
|
| Klipper Webinterface: |---------------|---------------|
|
||||||
║ 3) Mainsail │ {self.mainsail_local:<22} │ {self.mainsail_remote:<22} ║
|
| 3) Mainsail | {self.ms_local:<22} | {self.ms_remote:<22} |
|
||||||
║ 4) Fluidd │ {self.fluidd_local:<22} │ {self.fluidd_remote:<22} ║
|
| 4) Fluidd | | |
|
||||||
║ │ │ ║
|
| | | |
|
||||||
║ Client-Config: ├───────────────┼───────────────╢
|
| Touchscreen GUI: |---------------|---------------|
|
||||||
║ 5) Mainsail-Config │ {self.mainsail_config_local:<22} │ {self.mainsail_config_remote:<22} ║
|
| 5) KlipperScreen | | |
|
||||||
║ 6) Fluidd-Config │ {self.fluidd_config_local:<22} │ {self.fluidd_config_remote:<22} ║
|
| | | |
|
||||||
║ │ │ ║
|
| Other: |---------------|---------------|
|
||||||
║ Other: ├───────────────┼───────────────╢
|
| 6) PrettyGCode | | |
|
||||||
║ 7) KlipperScreen │ {self.klipperscreen_local:<22} │ {self.klipperscreen_remote:<22} ║
|
| 7) Telegram Bot | | |
|
||||||
║ 8) Crowsnest │ {self.crowsnest_local:<22} │ {self.crowsnest_remote:<22} ║
|
| 8) Obico for Klipper | | |
|
||||||
║ ├───────────────┴───────────────╢
|
| 9) OctoEverywhere | | |
|
||||||
║ 9) System │ {sysupgrades:^{padding}} ║
|
| 10) Mobileraker | | |
|
||||||
╟───────────────────────┴───────────────────────────────╢
|
| 11) Crowsnest | | |
|
||||||
|
| |-------------------------------|
|
||||||
|
| 12) System | |
|
||||||
"""
|
"""
|
||||||
)[1:]
|
)[1:]
|
||||||
print(menu, end="")
|
print(menu, end="")
|
||||||
|
|
||||||
def update_all(self, **kwargs) -> None:
|
def update_all(self, **kwargs):
|
||||||
Logger.print_status("Updating all components ...")
|
print("update_all")
|
||||||
self.update_klipper()
|
|
||||||
self.update_moonraker()
|
|
||||||
self.update_mainsail()
|
|
||||||
self.update_mainsail_config()
|
|
||||||
self.update_fluidd()
|
|
||||||
self.update_fluidd_config()
|
|
||||||
self.update_klipperscreen()
|
|
||||||
self.update_crowsnest()
|
|
||||||
self.upgrade_system_packages()
|
|
||||||
|
|
||||||
def update_klipper(self, **kwargs) -> None:
|
def update_klipper(self, **kwargs):
|
||||||
self._run_update_routine("klipper", update_klipper)
|
update_klipper()
|
||||||
|
|
||||||
def update_moonraker(self, **kwargs) -> None:
|
def update_moonraker(self, **kwargs):
|
||||||
self._run_update_routine("moonraker", update_moonraker)
|
update_moonraker()
|
||||||
|
|
||||||
def update_mainsail(self, **kwargs) -> None:
|
def update_mainsail(self, **kwargs):
|
||||||
self._run_update_routine(
|
update_mainsail()
|
||||||
"mainsail",
|
|
||||||
update_client,
|
|
||||||
self.mainsail_data,
|
|
||||||
)
|
|
||||||
|
|
||||||
def update_mainsail_config(self, **kwargs) -> None:
|
def update_fluidd(self, **kwargs):
|
||||||
self._run_update_routine(
|
print("update_fluidd")
|
||||||
"mainsail_config",
|
|
||||||
update_client_config,
|
|
||||||
self.mainsail_data,
|
|
||||||
)
|
|
||||||
|
|
||||||
def update_fluidd(self, **kwargs) -> None:
|
def update_klipperscreen(self, **kwargs):
|
||||||
self._run_update_routine(
|
print("update_klipperscreen")
|
||||||
"fluidd",
|
|
||||||
update_client,
|
|
||||||
self.fluidd_data,
|
|
||||||
)
|
|
||||||
|
|
||||||
def update_fluidd_config(self, **kwargs) -> None:
|
def update_pgc_for_klipper(self, **kwargs):
|
||||||
self._run_update_routine(
|
print("update_pgc_for_klipper")
|
||||||
"fluidd_config",
|
|
||||||
update_client_config,
|
|
||||||
self.fluidd_data,
|
|
||||||
)
|
|
||||||
|
|
||||||
def update_klipperscreen(self, **kwargs) -> None:
|
def update_telegram_bot(self, **kwargs):
|
||||||
self._run_update_routine("klipperscreen", update_klipperscreen)
|
print("update_telegram_bot")
|
||||||
|
|
||||||
def update_crowsnest(self, **kwargs) -> None:
|
def update_moonraker_obico(self, **kwargs):
|
||||||
self._run_update_routine("crowsnest", update_crowsnest)
|
print("update_moonraker_obico")
|
||||||
|
|
||||||
def upgrade_system_packages(self, **kwargs) -> None:
|
def update_octoeverywhere(self, **kwargs):
|
||||||
self._run_system_updates()
|
print("update_octoeverywhere")
|
||||||
|
|
||||||
def _fetch_update_status(self) -> None:
|
def update_mobileraker(self, **kwargs):
|
||||||
self._set_status_data("klipper", get_klipper_status)
|
print("update_mobileraker")
|
||||||
self._set_status_data("moonraker", get_moonraker_status)
|
|
||||||
self._set_status_data("mainsail", get_client_status, self.mainsail_data, True)
|
|
||||||
self._set_status_data(
|
|
||||||
"mainsail_config", get_client_config_status, self.mainsail_data
|
|
||||||
)
|
|
||||||
self._set_status_data("fluidd", get_client_status, self.fluidd_data, True)
|
|
||||||
self._set_status_data(
|
|
||||||
"fluidd_config", get_client_config_status, self.fluidd_data
|
|
||||||
)
|
|
||||||
self._set_status_data("klipperscreen", get_klipperscreen_status)
|
|
||||||
self._set_status_data("crowsnest", get_crowsnest_status)
|
|
||||||
|
|
||||||
update_system_package_lists(silent=True)
|
def update_crowsnest(self, **kwargs):
|
||||||
self.packages = get_upgradable_packages()
|
print("update_crowsnest")
|
||||||
self.package_count = len(self.packages)
|
|
||||||
|
|
||||||
def _format_local_status(self, local_version, remote_version) -> str:
|
def upgrade_system_packages(self, **kwargs):
|
||||||
color = COLOR_RED
|
print("upgrade_system_packages")
|
||||||
if not local_version:
|
|
||||||
color = COLOR_RED
|
|
||||||
elif local_version == remote_version:
|
|
||||||
color = COLOR_GREEN
|
|
||||||
elif local_version != remote_version:
|
|
||||||
color = COLOR_YELLOW
|
|
||||||
|
|
||||||
return f"{color}{local_version or '-'}{RESET_FORMAT}"
|
def fetch_update_status(self):
|
||||||
|
# klipper
|
||||||
def _set_status_data(self, name: str, status_fn: Callable, *args) -> None:
|
kl_status = get_klipper_status()
|
||||||
comp_status: ComponentStatus = status_fn(*args)
|
self.kl_local = kl_status.get("local")
|
||||||
|
self.kl_remote = kl_status.get("remote")
|
||||||
self.status_data[name]["installed"] = True if comp_status.status == 2 else False
|
if self.kl_local == self.kl_remote:
|
||||||
self.status_data[name]["local"] = comp_status.local
|
self.kl_local = f"{COLOR_GREEN}{self.kl_local}{RESET_FORMAT}"
|
||||||
self.status_data[name]["remote"] = comp_status.remote
|
else:
|
||||||
|
self.kl_local = f"{COLOR_YELLOW}{self.kl_local}{RESET_FORMAT}"
|
||||||
self._set_status_string(name)
|
self.kl_remote = f"{COLOR_GREEN}{self.kl_remote}{RESET_FORMAT}"
|
||||||
|
# moonraker
|
||||||
def _set_status_string(self, name: str) -> None:
|
mr_status = get_moonraker_status()
|
||||||
local_status = self.status_data[name].get("local", None)
|
self.mr_local = mr_status.get("local")
|
||||||
remote_status = self.status_data[name].get("remote", None)
|
self.mr_remote = mr_status.get("remote")
|
||||||
|
if self.mr_local == self.mr_remote:
|
||||||
color = COLOR_GREEN if remote_status else COLOR_RED
|
self.mr_local = f"{COLOR_GREEN}{self.mr_local}{RESET_FORMAT}"
|
||||||
local_txt = self._format_local_status(local_status, remote_status)
|
else:
|
||||||
remote_txt = f"{color}{remote_status or '-'}{RESET_FORMAT}"
|
self.mr_local = f"{COLOR_YELLOW}{self.mr_local}{RESET_FORMAT}"
|
||||||
|
self.mr_remote = f"{COLOR_GREEN}{self.mr_remote}{RESET_FORMAT}"
|
||||||
setattr(self, f"{name}_local", local_txt)
|
# mainsail
|
||||||
setattr(self, f"{name}_remote", remote_txt)
|
self.ms_local = get_mainsail_local_version()
|
||||||
|
self.ms_remote = get_mainsail_remote_version()
|
||||||
def _check_is_installed(self, name: str) -> bool:
|
if self.ms_local == self.ms_remote:
|
||||||
return self.status_data[name]["installed"]
|
self.ms_local = f"{COLOR_GREEN}{self.ms_local}{RESET_FORMAT}"
|
||||||
|
else:
|
||||||
def _is_update_available(self, name: str) -> bool:
|
self.ms_local = f"{COLOR_YELLOW}{self.ms_local}{RESET_FORMAT}"
|
||||||
return self.status_data[name]["local"] != self.status_data[name]["remote"]
|
self.ms_remote = f"{COLOR_GREEN}{self.ms_remote}{RESET_FORMAT}"
|
||||||
|
|
||||||
def _run_update_routine(self, name: str, update_fn: Callable, *args) -> None:
|
|
||||||
display_name = self.status_data[name]["display_name"]
|
|
||||||
is_installed = self._check_is_installed(name)
|
|
||||||
is_update_available = self._is_update_available(name)
|
|
||||||
|
|
||||||
if not is_installed:
|
|
||||||
Logger.print_info(f"{display_name} is not installed! Skipped ...")
|
|
||||||
return
|
|
||||||
elif not is_update_available:
|
|
||||||
Logger.print_info(f"{display_name} is already up to date! Skipped ...")
|
|
||||||
return
|
|
||||||
|
|
||||||
update_fn(*args)
|
|
||||||
|
|
||||||
def _run_system_updates(self) -> None:
|
|
||||||
if not self.packages:
|
|
||||||
Logger.print_info("No system upgrades available!")
|
|
||||||
return
|
|
||||||
|
|
||||||
try:
|
|
||||||
pkgs: str = ", ".join(self.packages)
|
|
||||||
Logger.print_dialog(
|
|
||||||
DialogType.CUSTOM,
|
|
||||||
["The following packages will be upgraded:", "\n\n", pkgs],
|
|
||||||
custom_title="UPGRADABLE SYSTEM UPDATES",
|
|
||||||
)
|
|
||||||
if not get_confirm("Continue?"):
|
|
||||||
return
|
|
||||||
Logger.print_status("Upgrading system packages ...")
|
|
||||||
upgrade_system_packages(self.packages)
|
|
||||||
except Exception as e:
|
|
||||||
Logger.print_error(f"Error upgrading system packages:\n{e}")
|
|
||||||
raise
|
|
||||||
|
|||||||
170
kiauh/core/repo_manager/repo_manager.py
Normal file
170
kiauh/core/repo_manager/repo_manager.py
Normal file
@@ -0,0 +1,170 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
# ======================================================================= #
|
||||||
|
# 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
|
||||||
|
import subprocess
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
from kiauh.utils.input_utils import get_confirm
|
||||||
|
from kiauh.utils.logger import Logger
|
||||||
|
|
||||||
|
|
||||||
|
# noinspection PyMethodMayBeStatic
|
||||||
|
class RepoManager:
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
repo: str,
|
||||||
|
target_dir: str,
|
||||||
|
branch: str = None,
|
||||||
|
):
|
||||||
|
self._repo = repo
|
||||||
|
self._branch = branch if branch is not None else "master"
|
||||||
|
self._method = self._get_method()
|
||||||
|
self._target_dir = target_dir
|
||||||
|
|
||||||
|
@property
|
||||||
|
def repo(self) -> str:
|
||||||
|
return self._repo
|
||||||
|
|
||||||
|
@repo.setter
|
||||||
|
def repo(self, value) -> None:
|
||||||
|
self._repo = value
|
||||||
|
|
||||||
|
@property
|
||||||
|
def branch(self) -> str:
|
||||||
|
return self._branch
|
||||||
|
|
||||||
|
@branch.setter
|
||||||
|
def branch(self, value) -> None:
|
||||||
|
self._branch = value
|
||||||
|
|
||||||
|
@property
|
||||||
|
def method(self) -> str:
|
||||||
|
return self._method
|
||||||
|
|
||||||
|
@method.setter
|
||||||
|
def method(self, value) -> None:
|
||||||
|
self._method = value
|
||||||
|
|
||||||
|
@property
|
||||||
|
def target_dir(self) -> str:
|
||||||
|
return self._target_dir
|
||||||
|
|
||||||
|
@target_dir.setter
|
||||||
|
def target_dir(self, value) -> None:
|
||||||
|
self._target_dir = value
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_repo_name(repo: Path) -> str:
|
||||||
|
"""
|
||||||
|
Helper method to extract the organisation and name of a repository |
|
||||||
|
:param repo: repository to extract the values from
|
||||||
|
:return: String in form of "<orga>/<name>"
|
||||||
|
"""
|
||||||
|
if not repo.exists() and not repo.joinpath(".git").exists():
|
||||||
|
return "-"
|
||||||
|
|
||||||
|
try:
|
||||||
|
cmd = ["git", "-C", repo, "config", "--get", "remote.origin.url"]
|
||||||
|
result = subprocess.check_output(cmd, stderr=subprocess.DEVNULL)
|
||||||
|
return "/".join(result.decode().strip().split("/")[-2:])
|
||||||
|
except subprocess.CalledProcessError:
|
||||||
|
return "-"
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_local_commit(repo: Path) -> str:
|
||||||
|
if not repo.exists() and not repo.joinpath(".git").exists():
|
||||||
|
return "-"
|
||||||
|
|
||||||
|
try:
|
||||||
|
cmd = f"cd {repo} && git describe HEAD --always --tags | cut -d '-' -f 1,2"
|
||||||
|
return subprocess.check_output(cmd, shell=True, text=True).strip()
|
||||||
|
except subprocess.CalledProcessError:
|
||||||
|
return "-"
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_remote_commit(repo: Path) -> str:
|
||||||
|
if not repo.exists() and not repo.joinpath(".git").exists():
|
||||||
|
return "-"
|
||||||
|
|
||||||
|
try:
|
||||||
|
# get locally checked out branch
|
||||||
|
branch_cmd = f"cd {repo} && git branch | grep -E '\*'"
|
||||||
|
branch = subprocess.check_output(branch_cmd, shell=True, text=True)
|
||||||
|
branch = branch.split("*")[-1].strip()
|
||||||
|
cmd = f"cd {repo} && git describe 'origin/{branch}' --always --tags | cut -d '-' -f 1,2"
|
||||||
|
return subprocess.check_output(cmd, shell=True, text=True).strip()
|
||||||
|
except subprocess.CalledProcessError:
|
||||||
|
return "-"
|
||||||
|
|
||||||
|
def clone_repo(self):
|
||||||
|
log = f"Cloning repository from '{self.repo}' with method '{self.method}'"
|
||||||
|
Logger.print_status(log)
|
||||||
|
try:
|
||||||
|
if Path(self.target_dir).exists():
|
||||||
|
question = f"'{self.target_dir}' already exists. Overwrite?"
|
||||||
|
if not get_confirm(question, default_choice=False):
|
||||||
|
Logger.print_info("Skipping re-clone of repository.")
|
||||||
|
return
|
||||||
|
shutil.rmtree(self.target_dir)
|
||||||
|
|
||||||
|
self._clone()
|
||||||
|
self._checkout()
|
||||||
|
except subprocess.CalledProcessError:
|
||||||
|
log = "An unexpected error occured during cloning of the repository."
|
||||||
|
Logger.print_error(log)
|
||||||
|
return
|
||||||
|
except OSError as e:
|
||||||
|
Logger.print_error(f"Error removing existing repository: {e.strerror}")
|
||||||
|
return
|
||||||
|
|
||||||
|
def pull_repo(self) -> None:
|
||||||
|
Logger.print_status(f"Updating repository '{self.repo}' ...")
|
||||||
|
try:
|
||||||
|
self._pull()
|
||||||
|
except subprocess.CalledProcessError:
|
||||||
|
log = "An unexpected error occured during updating the repository."
|
||||||
|
Logger.print_error(log)
|
||||||
|
return
|
||||||
|
|
||||||
|
def _clone(self):
|
||||||
|
try:
|
||||||
|
command = ["git", "clone", self.repo, self.target_dir]
|
||||||
|
subprocess.run(command, check=True)
|
||||||
|
|
||||||
|
Logger.print_ok("Clone successfull!")
|
||||||
|
except subprocess.CalledProcessError as e:
|
||||||
|
log = f"Error cloning repository {self.repo}: {e.stderr.decode()}"
|
||||||
|
Logger.print_error(log)
|
||||||
|
raise
|
||||||
|
|
||||||
|
def _checkout(self):
|
||||||
|
try:
|
||||||
|
command = ["git", "checkout", f"{self.branch}"]
|
||||||
|
subprocess.run(command, cwd=self.target_dir, check=True)
|
||||||
|
|
||||||
|
Logger.print_ok("Checkout successfull!")
|
||||||
|
except subprocess.CalledProcessError as e:
|
||||||
|
log = f"Error checking out branch {self.branch}: {e.stderr.decode()}"
|
||||||
|
Logger.print_error(log)
|
||||||
|
raise
|
||||||
|
|
||||||
|
def _pull(self) -> None:
|
||||||
|
try:
|
||||||
|
command = ["git", "pull"]
|
||||||
|
subprocess.run(command, cwd=self.target_dir, check=True)
|
||||||
|
except subprocess.CalledProcessError as e:
|
||||||
|
log = f"Error on git pull: {e.stderr.decode()}"
|
||||||
|
Logger.print_error(log)
|
||||||
|
raise
|
||||||
|
|
||||||
|
def _get_method(self) -> str:
|
||||||
|
return "ssh" if self.repo.startswith("git") else "https"
|
||||||
@@ -1,199 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# 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 #
|
|
||||||
# ======================================================================= #
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
from dataclasses import dataclass, field
|
|
||||||
from typing import Any
|
|
||||||
|
|
||||||
from core.logger import DialogType, Logger
|
|
||||||
from core.submodules.simple_config_parser.src.simple_config_parser.simple_config_parser import (
|
|
||||||
NoOptionError,
|
|
||||||
NoSectionError,
|
|
||||||
SimpleConfigParser,
|
|
||||||
)
|
|
||||||
from utils.sys_utils import kill
|
|
||||||
|
|
||||||
from kiauh import PROJECT_ROOT
|
|
||||||
|
|
||||||
DEFAULT_CFG = PROJECT_ROOT.joinpath("default.kiauh.cfg")
|
|
||||||
CUSTOM_CFG = PROJECT_ROOT.joinpath("kiauh.cfg")
|
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
|
||||||
class AppSettings:
|
|
||||||
backup_before_update: bool | None = field(default=None)
|
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
|
||||||
class RepoSettings:
|
|
||||||
repo_url: str | None = field(default=None)
|
|
||||||
branch: str | None = field(default=None)
|
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
|
||||||
class WebUiSettings:
|
|
||||||
port: str | None = field(default=None)
|
|
||||||
unstable_releases: bool | None = field(default=None)
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyUnusedLocal
|
|
||||||
# noinspection PyMethodMayBeStatic
|
|
||||||
class KiauhSettings:
|
|
||||||
_instance = None
|
|
||||||
|
|
||||||
def __new__(cls, *args, **kwargs) -> "KiauhSettings":
|
|
||||||
if cls._instance is None:
|
|
||||||
cls._instance = super(KiauhSettings, cls).__new__(cls, *args, **kwargs)
|
|
||||||
return cls._instance
|
|
||||||
|
|
||||||
def __repr__(self) -> str:
|
|
||||||
return (
|
|
||||||
f"KiauhSettings(kiauh={self.kiauh}, klipper={self.klipper},"
|
|
||||||
f" moonraker={self.moonraker}, mainsail={self.mainsail},"
|
|
||||||
f" fluidd={self.fluidd})"
|
|
||||||
)
|
|
||||||
|
|
||||||
def __getitem__(self, item: str) -> Any:
|
|
||||||
return getattr(self, item)
|
|
||||||
|
|
||||||
def __init__(self) -> None:
|
|
||||||
if not hasattr(self, "__initialized"):
|
|
||||||
self.__initialized = False
|
|
||||||
if self.__initialized:
|
|
||||||
return
|
|
||||||
self.__initialized = True
|
|
||||||
self.config = SimpleConfigParser()
|
|
||||||
self.kiauh = AppSettings()
|
|
||||||
self.klipper = RepoSettings()
|
|
||||||
self.moonraker = RepoSettings()
|
|
||||||
self.mainsail = WebUiSettings()
|
|
||||||
self.fluidd = WebUiSettings()
|
|
||||||
|
|
||||||
self._load_config()
|
|
||||||
|
|
||||||
def get(self, section: str, option: str) -> str | int | bool:
|
|
||||||
"""
|
|
||||||
Get a value from the settings state by providing the section and option name as
|
|
||||||
strings. Prefer direct access to the properties, as it is usually safer!
|
|
||||||
:param section: The section name as string.
|
|
||||||
:param option: The option name as string.
|
|
||||||
:return: The value of the option as string, int or bool.
|
|
||||||
"""
|
|
||||||
|
|
||||||
try:
|
|
||||||
section = getattr(self, section)
|
|
||||||
value = getattr(section, option)
|
|
||||||
return value # type: ignore
|
|
||||||
except AttributeError:
|
|
||||||
raise
|
|
||||||
|
|
||||||
def save(self) -> None:
|
|
||||||
self._set_config_options_state()
|
|
||||||
self.config.write_file(CUSTOM_CFG)
|
|
||||||
self._load_config()
|
|
||||||
|
|
||||||
def _load_config(self) -> None:
|
|
||||||
if not CUSTOM_CFG.exists() and not DEFAULT_CFG.exists():
|
|
||||||
self._kill()
|
|
||||||
|
|
||||||
cfg = CUSTOM_CFG if CUSTOM_CFG.exists() else DEFAULT_CFG
|
|
||||||
self.config.read_file(cfg)
|
|
||||||
|
|
||||||
self._validate_cfg()
|
|
||||||
self._apply_settings_from_file()
|
|
||||||
|
|
||||||
def _validate_cfg(self) -> None:
|
|
||||||
try:
|
|
||||||
self._validate_bool("kiauh", "backup_before_update")
|
|
||||||
|
|
||||||
self._validate_str("klipper", "repo_url")
|
|
||||||
self._validate_str("klipper", "branch")
|
|
||||||
|
|
||||||
self._validate_int("mainsail", "port")
|
|
||||||
self._validate_bool("mainsail", "unstable_releases")
|
|
||||||
|
|
||||||
self._validate_int("fluidd", "port")
|
|
||||||
self._validate_bool("fluidd", "unstable_releases")
|
|
||||||
|
|
||||||
except ValueError:
|
|
||||||
err = f"Invalid value for option '{self._v_option}' in section '{self._v_section}'"
|
|
||||||
Logger.print_error(err)
|
|
||||||
kill()
|
|
||||||
except NoSectionError:
|
|
||||||
err = f"Missing section '{self._v_section}' in config file"
|
|
||||||
Logger.print_error(err)
|
|
||||||
kill()
|
|
||||||
except NoOptionError:
|
|
||||||
err = f"Missing option '{self._v_option}' in section '{self._v_section}'"
|
|
||||||
Logger.print_error(err)
|
|
||||||
kill()
|
|
||||||
|
|
||||||
def _validate_bool(self, section: str, option: str) -> None:
|
|
||||||
self._v_section, self._v_option = (section, option)
|
|
||||||
(bool(self.config.getboolean(section, option)))
|
|
||||||
|
|
||||||
def _validate_int(self, section: str, option: str) -> None:
|
|
||||||
self._v_section, self._v_option = (section, option)
|
|
||||||
int(self.config.getint(section, option))
|
|
||||||
|
|
||||||
def _validate_str(self, section: str, option: str) -> None:
|
|
||||||
self._v_section, self._v_option = (section, option)
|
|
||||||
v = self.config.getval(section, option)
|
|
||||||
if v.isdigit() or v.lower() == "true" or v.lower() == "false":
|
|
||||||
raise ValueError
|
|
||||||
|
|
||||||
def _apply_settings_from_file(self) -> None:
|
|
||||||
self.kiauh.backup_before_update = self.config.getboolean(
|
|
||||||
"kiauh", "backup_before_update"
|
|
||||||
)
|
|
||||||
self.klipper.repo_url = self.config.getval("klipper", "repo_url")
|
|
||||||
self.klipper.branch = self.config.getval("klipper", "branch")
|
|
||||||
self.moonraker.repo_url = self.config.getval("moonraker", "repo_url")
|
|
||||||
self.moonraker.branch = self.config.getval("moonraker", "branch")
|
|
||||||
self.mainsail.port = self.config.getint("mainsail", "port")
|
|
||||||
self.mainsail.unstable_releases = self.config.getboolean(
|
|
||||||
"mainsail", "unstable_releases"
|
|
||||||
)
|
|
||||||
self.fluidd.port = self.config.getint("fluidd", "port")
|
|
||||||
self.fluidd.unstable_releases = self.config.getboolean(
|
|
||||||
"fluidd", "unstable_releases"
|
|
||||||
)
|
|
||||||
|
|
||||||
def _set_config_options_state(self) -> None:
|
|
||||||
self.config.set_option(
|
|
||||||
"kiauh",
|
|
||||||
"backup_before_update",
|
|
||||||
str(self.kiauh.backup_before_update),
|
|
||||||
)
|
|
||||||
self.config.set_option("klipper", "repo_url", self.klipper.repo_url)
|
|
||||||
self.config.set_option("klipper", "branch", self.klipper.branch)
|
|
||||||
self.config.set_option("moonraker", "repo_url", self.moonraker.repo_url)
|
|
||||||
self.config.set_option("moonraker", "branch", self.moonraker.branch)
|
|
||||||
self.config.set_option("mainsail", "port", str(self.mainsail.port))
|
|
||||||
self.config.set_option(
|
|
||||||
"mainsail",
|
|
||||||
"unstable_releases",
|
|
||||||
str(self.mainsail.unstable_releases),
|
|
||||||
)
|
|
||||||
self.config.set_option("fluidd", "port", str(self.fluidd.port))
|
|
||||||
self.config.set_option(
|
|
||||||
"fluidd", "unstable_releases", str(self.fluidd.unstable_releases)
|
|
||||||
)
|
|
||||||
|
|
||||||
def _kill(self) -> None:
|
|
||||||
Logger.print_dialog(
|
|
||||||
DialogType.ERROR,
|
|
||||||
[
|
|
||||||
"No KIAUH configuration file found! Please make sure you have at least "
|
|
||||||
"one of the following configuration files in KIAUH's root directory:",
|
|
||||||
"● default.kiauh.cfg",
|
|
||||||
"● kiauh.cfg",
|
|
||||||
],
|
|
||||||
)
|
|
||||||
kill()
|
|
||||||
@@ -1,61 +0,0 @@
|
|||||||
import sys
|
|
||||||
import threading
|
|
||||||
import time
|
|
||||||
from typing import List, Literal
|
|
||||||
|
|
||||||
from core.constants import (
|
|
||||||
COLOR_GREEN,
|
|
||||||
COLOR_RED,
|
|
||||||
COLOR_WHITE,
|
|
||||||
COLOR_YELLOW,
|
|
||||||
RESET_FORMAT,
|
|
||||||
)
|
|
||||||
|
|
||||||
SpinnerColor = Literal["white", "red", "green", "yellow"]
|
|
||||||
|
|
||||||
|
|
||||||
class Spinner:
|
|
||||||
def __init__(
|
|
||||||
self,
|
|
||||||
message: str = "Loading",
|
|
||||||
color: SpinnerColor = "white",
|
|
||||||
interval: float = 0.2,
|
|
||||||
) -> None:
|
|
||||||
self.message = f"{message} ..."
|
|
||||||
self.interval = interval
|
|
||||||
self._stop_event = threading.Event()
|
|
||||||
self._thread = threading.Thread(target=self._animate)
|
|
||||||
self._color = ""
|
|
||||||
self._set_color(color)
|
|
||||||
|
|
||||||
def _animate(self) -> None:
|
|
||||||
animation: List[str] = ["⠋", "⠙", "⠹", "⠸", "⠼", "⠴", "⠦", "⠧", "⠇", "⠏"]
|
|
||||||
while not self._stop_event.is_set():
|
|
||||||
for char in animation:
|
|
||||||
sys.stdout.write(f"\r{self._color}{char}{RESET_FORMAT} {self.message}")
|
|
||||||
sys.stdout.flush()
|
|
||||||
time.sleep(self.interval)
|
|
||||||
if self._stop_event.is_set():
|
|
||||||
break
|
|
||||||
sys.stdout.write("\r" + " " * (len(self.message) + 1) + "\r")
|
|
||||||
sys.stdout.flush()
|
|
||||||
|
|
||||||
def _set_color(self, color: SpinnerColor) -> None:
|
|
||||||
if color == "white":
|
|
||||||
self._color = COLOR_WHITE
|
|
||||||
elif color == "red":
|
|
||||||
self._color = COLOR_RED
|
|
||||||
elif color == "green":
|
|
||||||
self._color = COLOR_GREEN
|
|
||||||
elif color == "yellow":
|
|
||||||
self._color = COLOR_YELLOW
|
|
||||||
|
|
||||||
def start(self) -> None:
|
|
||||||
self._stop_event.clear()
|
|
||||||
if not self._thread.is_alive():
|
|
||||||
self._thread = threading.Thread(target=self._animate)
|
|
||||||
self._thread.start()
|
|
||||||
|
|
||||||
def stop(self) -> None:
|
|
||||||
self._stop_event.set()
|
|
||||||
self._thread.join()
|
|
||||||
@@ -1,13 +0,0 @@
|
|||||||
# see https://editorconfig.org/
|
|
||||||
root = true
|
|
||||||
|
|
||||||
[*]
|
|
||||||
end_of_line = lf
|
|
||||||
trim_trailing_whitespace = true
|
|
||||||
indent_style = space
|
|
||||||
insert_final_newline = true
|
|
||||||
indent_size = 4
|
|
||||||
charset = utf-8
|
|
||||||
|
|
||||||
[*.py]
|
|
||||||
max_line_length = 88
|
|
||||||
@@ -1,13 +0,0 @@
|
|||||||
*.py[cod]
|
|
||||||
*.pyc
|
|
||||||
__pycache__
|
|
||||||
.pytest_cache/
|
|
||||||
|
|
||||||
.idea/
|
|
||||||
.vscode/
|
|
||||||
|
|
||||||
.venv*/
|
|
||||||
venv*/
|
|
||||||
|
|
||||||
.coverage
|
|
||||||
htmlcov/
|
|
||||||
@@ -1,674 +0,0 @@
|
|||||||
GNU GENERAL PUBLIC LICENSE
|
|
||||||
Version 3, 29 June 2007
|
|
||||||
|
|
||||||
Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/>
|
|
||||||
Everyone is permitted to copy and distribute verbatim copies
|
|
||||||
of this license document, but changing it is not allowed.
|
|
||||||
|
|
||||||
Preamble
|
|
||||||
|
|
||||||
The GNU General Public License is a free, copyleft license for
|
|
||||||
software and other kinds of works.
|
|
||||||
|
|
||||||
The licenses for most software and other practical works are designed
|
|
||||||
to take away your freedom to share and change the works. By contrast,
|
|
||||||
the GNU General Public License is intended to guarantee your freedom to
|
|
||||||
share and change all versions of a program--to make sure it remains free
|
|
||||||
software for all its users. We, the Free Software Foundation, use the
|
|
||||||
GNU General Public License for most of our software; it applies also to
|
|
||||||
any other work released this way by its authors. You can apply it to
|
|
||||||
your programs, too.
|
|
||||||
|
|
||||||
When we speak of free software, we are referring to freedom, not
|
|
||||||
price. Our General Public Licenses are designed to make sure that you
|
|
||||||
have the freedom to distribute copies of free software (and charge for
|
|
||||||
them if you wish), that you receive source code or can get it if you
|
|
||||||
want it, that you can change the software or use pieces of it in new
|
|
||||||
free programs, and that you know you can do these things.
|
|
||||||
|
|
||||||
To protect your rights, we need to prevent others from denying you
|
|
||||||
these rights or asking you to surrender the rights. Therefore, you have
|
|
||||||
certain responsibilities if you distribute copies of the software, or if
|
|
||||||
you modify it: responsibilities to respect the freedom of others.
|
|
||||||
|
|
||||||
For example, if you distribute copies of such a program, whether
|
|
||||||
gratis or for a fee, you must pass on to the recipients the same
|
|
||||||
freedoms that you received. You must make sure that they, too, receive
|
|
||||||
or can get the source code. And you must show them these terms so they
|
|
||||||
know their rights.
|
|
||||||
|
|
||||||
Developers that use the GNU GPL protect your rights with two steps:
|
|
||||||
(1) assert copyright on the software, and (2) offer you this License
|
|
||||||
giving you legal permission to copy, distribute and/or modify it.
|
|
||||||
|
|
||||||
For the developers' and authors' protection, the GPL clearly explains
|
|
||||||
that there is no warranty for this free software. For both users' and
|
|
||||||
authors' sake, the GPL requires that modified versions be marked as
|
|
||||||
changed, so that their problems will not be attributed erroneously to
|
|
||||||
authors of previous versions.
|
|
||||||
|
|
||||||
Some devices are designed to deny users access to install or run
|
|
||||||
modified versions of the software inside them, although the manufacturer
|
|
||||||
can do so. This is fundamentally incompatible with the aim of
|
|
||||||
protecting users' freedom to change the software. The systematic
|
|
||||||
pattern of such abuse occurs in the area of products for individuals to
|
|
||||||
use, which is precisely where it is most unacceptable. Therefore, we
|
|
||||||
have designed this version of the GPL to prohibit the practice for those
|
|
||||||
products. If such problems arise substantially in other domains, we
|
|
||||||
stand ready to extend this provision to those domains in future versions
|
|
||||||
of the GPL, as needed to protect the freedom of users.
|
|
||||||
|
|
||||||
Finally, every program is threatened constantly by software patents.
|
|
||||||
States should not allow patents to restrict development and use of
|
|
||||||
software on general-purpose computers, but in those that do, we wish to
|
|
||||||
avoid the special danger that patents applied to a free program could
|
|
||||||
make it effectively proprietary. To prevent this, the GPL assures that
|
|
||||||
patents cannot be used to render the program non-free.
|
|
||||||
|
|
||||||
The precise terms and conditions for copying, distribution and
|
|
||||||
modification follow.
|
|
||||||
|
|
||||||
TERMS AND CONDITIONS
|
|
||||||
|
|
||||||
0. Definitions.
|
|
||||||
|
|
||||||
"This License" refers to version 3 of the GNU General Public License.
|
|
||||||
|
|
||||||
"Copyright" also means copyright-like laws that apply to other kinds of
|
|
||||||
works, such as semiconductor masks.
|
|
||||||
|
|
||||||
"The Program" refers to any copyrightable work licensed under this
|
|
||||||
License. Each licensee is addressed as "you". "Licensees" and
|
|
||||||
"recipients" may be individuals or organizations.
|
|
||||||
|
|
||||||
To "modify" a work means to copy from or adapt all or part of the work
|
|
||||||
in a fashion requiring copyright permission, other than the making of an
|
|
||||||
exact copy. The resulting work is called a "modified version" of the
|
|
||||||
earlier work or a work "based on" the earlier work.
|
|
||||||
|
|
||||||
A "covered work" means either the unmodified Program or a work based
|
|
||||||
on the Program.
|
|
||||||
|
|
||||||
To "propagate" a work means to do anything with it that, without
|
|
||||||
permission, would make you directly or secondarily liable for
|
|
||||||
infringement under applicable copyright law, except executing it on a
|
|
||||||
computer or modifying a private copy. Propagation includes copying,
|
|
||||||
distribution (with or without modification), making available to the
|
|
||||||
public, and in some countries other activities as well.
|
|
||||||
|
|
||||||
To "convey" a work means any kind of propagation that enables other
|
|
||||||
parties to make or receive copies. Mere interaction with a user through
|
|
||||||
a computer network, with no transfer of a copy, is not conveying.
|
|
||||||
|
|
||||||
An interactive user interface displays "Appropriate Legal Notices"
|
|
||||||
to the extent that it includes a convenient and prominently visible
|
|
||||||
feature that (1) displays an appropriate copyright notice, and (2)
|
|
||||||
tells the user that there is no warranty for the work (except to the
|
|
||||||
extent that warranties are provided), that licensees may convey the
|
|
||||||
work under this License, and how to view a copy of this License. If
|
|
||||||
the interface presents a list of user commands or options, such as a
|
|
||||||
menu, a prominent item in the list meets this criterion.
|
|
||||||
|
|
||||||
1. Source Code.
|
|
||||||
|
|
||||||
The "source code" for a work means the preferred form of the work
|
|
||||||
for making modifications to it. "Object code" means any non-source
|
|
||||||
form of a work.
|
|
||||||
|
|
||||||
A "Standard Interface" means an interface that either is an official
|
|
||||||
standard defined by a recognized standards body, or, in the case of
|
|
||||||
interfaces specified for a particular programming language, one that
|
|
||||||
is widely used among developers working in that language.
|
|
||||||
|
|
||||||
The "System Libraries" of an executable work include anything, other
|
|
||||||
than the work as a whole, that (a) is included in the normal form of
|
|
||||||
packaging a Major Component, but which is not part of that Major
|
|
||||||
Component, and (b) serves only to enable use of the work with that
|
|
||||||
Major Component, or to implement a Standard Interface for which an
|
|
||||||
implementation is available to the public in source code form. A
|
|
||||||
"Major Component", in this context, means a major essential component
|
|
||||||
(kernel, window system, and so on) of the specific operating system
|
|
||||||
(if any) on which the executable work runs, or a compiler used to
|
|
||||||
produce the work, or an object code interpreter used to run it.
|
|
||||||
|
|
||||||
The "Corresponding Source" for a work in object code form means all
|
|
||||||
the source code needed to generate, install, and (for an executable
|
|
||||||
work) run the object code and to modify the work, including scripts to
|
|
||||||
control those activities. However, it does not include the work's
|
|
||||||
System Libraries, or general-purpose tools or generally available free
|
|
||||||
programs which are used unmodified in performing those activities but
|
|
||||||
which are not part of the work. For example, Corresponding Source
|
|
||||||
includes interface definition files associated with source files for
|
|
||||||
the work, and the source code for shared libraries and dynamically
|
|
||||||
linked subprograms that the work is specifically designed to require,
|
|
||||||
such as by intimate data communication or control flow between those
|
|
||||||
subprograms and other parts of the work.
|
|
||||||
|
|
||||||
The Corresponding Source need not include anything that users
|
|
||||||
can regenerate automatically from other parts of the Corresponding
|
|
||||||
Source.
|
|
||||||
|
|
||||||
The Corresponding Source for a work in source code form is that
|
|
||||||
same work.
|
|
||||||
|
|
||||||
2. Basic Permissions.
|
|
||||||
|
|
||||||
All rights granted under this License are granted for the term of
|
|
||||||
copyright on the Program, and are irrevocable provided the stated
|
|
||||||
conditions are met. This License explicitly affirms your unlimited
|
|
||||||
permission to run the unmodified Program. The output from running a
|
|
||||||
covered work is covered by this License only if the output, given its
|
|
||||||
content, constitutes a covered work. This License acknowledges your
|
|
||||||
rights of fair use or other equivalent, as provided by copyright law.
|
|
||||||
|
|
||||||
You may make, run and propagate covered works that you do not
|
|
||||||
convey, without conditions so long as your license otherwise remains
|
|
||||||
in force. You may convey covered works to others for the sole purpose
|
|
||||||
of having them make modifications exclusively for you, or provide you
|
|
||||||
with facilities for running those works, provided that you comply with
|
|
||||||
the terms of this License in conveying all material for which you do
|
|
||||||
not control copyright. Those thus making or running the covered works
|
|
||||||
for you must do so exclusively on your behalf, under your direction
|
|
||||||
and control, on terms that prohibit them from making any copies of
|
|
||||||
your copyrighted material outside their relationship with you.
|
|
||||||
|
|
||||||
Conveying under any other circumstances is permitted solely under
|
|
||||||
the conditions stated below. Sublicensing is not allowed; section 10
|
|
||||||
makes it unnecessary.
|
|
||||||
|
|
||||||
3. Protecting Users' Legal Rights From Anti-Circumvention Law.
|
|
||||||
|
|
||||||
No covered work shall be deemed part of an effective technological
|
|
||||||
measure under any applicable law fulfilling obligations under article
|
|
||||||
11 of the WIPO copyright treaty adopted on 20 December 1996, or
|
|
||||||
similar laws prohibiting or restricting circumvention of such
|
|
||||||
measures.
|
|
||||||
|
|
||||||
When you convey a covered work, you waive any legal power to forbid
|
|
||||||
circumvention of technological measures to the extent such circumvention
|
|
||||||
is effected by exercising rights under this License with respect to
|
|
||||||
the covered work, and you disclaim any intention to limit operation or
|
|
||||||
modification of the work as a means of enforcing, against the work's
|
|
||||||
users, your or third parties' legal rights to forbid circumvention of
|
|
||||||
technological measures.
|
|
||||||
|
|
||||||
4. Conveying Verbatim Copies.
|
|
||||||
|
|
||||||
You may convey verbatim copies of the Program's source code as you
|
|
||||||
receive it, in any medium, provided that you conspicuously and
|
|
||||||
appropriately publish on each copy an appropriate copyright notice;
|
|
||||||
keep intact all notices stating that this License and any
|
|
||||||
non-permissive terms added in accord with section 7 apply to the code;
|
|
||||||
keep intact all notices of the absence of any warranty; and give all
|
|
||||||
recipients a copy of this License along with the Program.
|
|
||||||
|
|
||||||
You may charge any price or no price for each copy that you convey,
|
|
||||||
and you may offer support or warranty protection for a fee.
|
|
||||||
|
|
||||||
5. Conveying Modified Source Versions.
|
|
||||||
|
|
||||||
You may convey a work based on the Program, or the modifications to
|
|
||||||
produce it from the Program, in the form of source code under the
|
|
||||||
terms of section 4, provided that you also meet all of these conditions:
|
|
||||||
|
|
||||||
a) The work must carry prominent notices stating that you modified
|
|
||||||
it, and giving a relevant date.
|
|
||||||
|
|
||||||
b) The work must carry prominent notices stating that it is
|
|
||||||
released under this License and any conditions added under section
|
|
||||||
7. This requirement modifies the requirement in section 4 to
|
|
||||||
"keep intact all notices".
|
|
||||||
|
|
||||||
c) You must license the entire work, as a whole, under this
|
|
||||||
License to anyone who comes into possession of a copy. This
|
|
||||||
License will therefore apply, along with any applicable section 7
|
|
||||||
additional terms, to the whole of the work, and all its parts,
|
|
||||||
regardless of how they are packaged. This License gives no
|
|
||||||
permission to license the work in any other way, but it does not
|
|
||||||
invalidate such permission if you have separately received it.
|
|
||||||
|
|
||||||
d) If the work has interactive user interfaces, each must display
|
|
||||||
Appropriate Legal Notices; however, if the Program has interactive
|
|
||||||
interfaces that do not display Appropriate Legal Notices, your
|
|
||||||
work need not make them do so.
|
|
||||||
|
|
||||||
A compilation of a covered work with other separate and independent
|
|
||||||
works, which are not by their nature extensions of the covered work,
|
|
||||||
and which are not combined with it such as to form a larger program,
|
|
||||||
in or on a volume of a storage or distribution medium, is called an
|
|
||||||
"aggregate" if the compilation and its resulting copyright are not
|
|
||||||
used to limit the access or legal rights of the compilation's users
|
|
||||||
beyond what the individual works permit. Inclusion of a covered work
|
|
||||||
in an aggregate does not cause this License to apply to the other
|
|
||||||
parts of the aggregate.
|
|
||||||
|
|
||||||
6. Conveying Non-Source Forms.
|
|
||||||
|
|
||||||
You may convey a covered work in object code form under the terms
|
|
||||||
of sections 4 and 5, provided that you also convey the
|
|
||||||
machine-readable Corresponding Source under the terms of this License,
|
|
||||||
in one of these ways:
|
|
||||||
|
|
||||||
a) Convey the object code in, or embodied in, a physical product
|
|
||||||
(including a physical distribution medium), accompanied by the
|
|
||||||
Corresponding Source fixed on a durable physical medium
|
|
||||||
customarily used for software interchange.
|
|
||||||
|
|
||||||
b) Convey the object code in, or embodied in, a physical product
|
|
||||||
(including a physical distribution medium), accompanied by a
|
|
||||||
written offer, valid for at least three years and valid for as
|
|
||||||
long as you offer spare parts or customer support for that product
|
|
||||||
model, to give anyone who possesses the object code either (1) a
|
|
||||||
copy of the Corresponding Source for all the software in the
|
|
||||||
product that is covered by this License, on a durable physical
|
|
||||||
medium customarily used for software interchange, for a price no
|
|
||||||
more than your reasonable cost of physically performing this
|
|
||||||
conveying of source, or (2) access to copy the
|
|
||||||
Corresponding Source from a network server at no charge.
|
|
||||||
|
|
||||||
c) Convey individual copies of the object code with a copy of the
|
|
||||||
written offer to provide the Corresponding Source. This
|
|
||||||
alternative is allowed only occasionally and noncommercially, and
|
|
||||||
only if you received the object code with such an offer, in accord
|
|
||||||
with subsection 6b.
|
|
||||||
|
|
||||||
d) Convey the object code by offering access from a designated
|
|
||||||
place (gratis or for a charge), and offer equivalent access to the
|
|
||||||
Corresponding Source in the same way through the same place at no
|
|
||||||
further charge. You need not require recipients to copy the
|
|
||||||
Corresponding Source along with the object code. If the place to
|
|
||||||
copy the object code is a network server, the Corresponding Source
|
|
||||||
may be on a different server (operated by you or a third party)
|
|
||||||
that supports equivalent copying facilities, provided you maintain
|
|
||||||
clear directions next to the object code saying where to find the
|
|
||||||
Corresponding Source. Regardless of what server hosts the
|
|
||||||
Corresponding Source, you remain obligated to ensure that it is
|
|
||||||
available for as long as needed to satisfy these requirements.
|
|
||||||
|
|
||||||
e) Convey the object code using peer-to-peer transmission, provided
|
|
||||||
you inform other peers where the object code and Corresponding
|
|
||||||
Source of the work are being offered to the general public at no
|
|
||||||
charge under subsection 6d.
|
|
||||||
|
|
||||||
A separable portion of the object code, whose source code is excluded
|
|
||||||
from the Corresponding Source as a System Library, need not be
|
|
||||||
included in conveying the object code work.
|
|
||||||
|
|
||||||
A "User Product" is either (1) a "consumer product", which means any
|
|
||||||
tangible personal property which is normally used for personal, family,
|
|
||||||
or household purposes, or (2) anything designed or sold for incorporation
|
|
||||||
into a dwelling. In determining whether a product is a consumer product,
|
|
||||||
doubtful cases shall be resolved in favor of coverage. For a particular
|
|
||||||
product received by a particular user, "normally used" refers to a
|
|
||||||
typical or common use of that class of product, regardless of the status
|
|
||||||
of the particular user or of the way in which the particular user
|
|
||||||
actually uses, or expects or is expected to use, the product. A product
|
|
||||||
is a consumer product regardless of whether the product has substantial
|
|
||||||
commercial, industrial or non-consumer uses, unless such uses represent
|
|
||||||
the only significant mode of use of the product.
|
|
||||||
|
|
||||||
"Installation Information" for a User Product means any methods,
|
|
||||||
procedures, authorization keys, or other information required to install
|
|
||||||
and execute modified versions of a covered work in that User Product from
|
|
||||||
a modified version of its Corresponding Source. The information must
|
|
||||||
suffice to ensure that the continued functioning of the modified object
|
|
||||||
code is in no case prevented or interfered with solely because
|
|
||||||
modification has been made.
|
|
||||||
|
|
||||||
If you convey an object code work under this section in, or with, or
|
|
||||||
specifically for use in, a User Product, and the conveying occurs as
|
|
||||||
part of a transaction in which the right of possession and use of the
|
|
||||||
User Product is transferred to the recipient in perpetuity or for a
|
|
||||||
fixed term (regardless of how the transaction is characterized), the
|
|
||||||
Corresponding Source conveyed under this section must be accompanied
|
|
||||||
by the Installation Information. But this requirement does not apply
|
|
||||||
if neither you nor any third party retains the ability to install
|
|
||||||
modified object code on the User Product (for example, the work has
|
|
||||||
been installed in ROM).
|
|
||||||
|
|
||||||
The requirement to provide Installation Information does not include a
|
|
||||||
requirement to continue to provide support service, warranty, or updates
|
|
||||||
for a work that has been modified or installed by the recipient, or for
|
|
||||||
the User Product in which it has been modified or installed. Access to a
|
|
||||||
network may be denied when the modification itself materially and
|
|
||||||
adversely affects the operation of the network or violates the rules and
|
|
||||||
protocols for communication across the network.
|
|
||||||
|
|
||||||
Corresponding Source conveyed, and Installation Information provided,
|
|
||||||
in accord with this section must be in a format that is publicly
|
|
||||||
documented (and with an implementation available to the public in
|
|
||||||
source code form), and must require no special password or key for
|
|
||||||
unpacking, reading or copying.
|
|
||||||
|
|
||||||
7. Additional Terms.
|
|
||||||
|
|
||||||
"Additional permissions" are terms that supplement the terms of this
|
|
||||||
License by making exceptions from one or more of its conditions.
|
|
||||||
Additional permissions that are applicable to the entire Program shall
|
|
||||||
be treated as though they were included in this License, to the extent
|
|
||||||
that they are valid under applicable law. If additional permissions
|
|
||||||
apply only to part of the Program, that part may be used separately
|
|
||||||
under those permissions, but the entire Program remains governed by
|
|
||||||
this License without regard to the additional permissions.
|
|
||||||
|
|
||||||
When you convey a copy of a covered work, you may at your option
|
|
||||||
remove any additional permissions from that copy, or from any part of
|
|
||||||
it. (Additional permissions may be written to require their own
|
|
||||||
removal in certain cases when you modify the work.) You may place
|
|
||||||
additional permissions on material, added by you to a covered work,
|
|
||||||
for which you have or can give appropriate copyright permission.
|
|
||||||
|
|
||||||
Notwithstanding any other provision of this License, for material you
|
|
||||||
add to a covered work, you may (if authorized by the copyright holders of
|
|
||||||
that material) supplement the terms of this License with terms:
|
|
||||||
|
|
||||||
a) Disclaiming warranty or limiting liability differently from the
|
|
||||||
terms of sections 15 and 16 of this License; or
|
|
||||||
|
|
||||||
b) Requiring preservation of specified reasonable legal notices or
|
|
||||||
author attributions in that material or in the Appropriate Legal
|
|
||||||
Notices displayed by works containing it; or
|
|
||||||
|
|
||||||
c) Prohibiting misrepresentation of the origin of that material, or
|
|
||||||
requiring that modified versions of such material be marked in
|
|
||||||
reasonable ways as different from the original version; or
|
|
||||||
|
|
||||||
d) Limiting the use for publicity purposes of names of licensors or
|
|
||||||
authors of the material; or
|
|
||||||
|
|
||||||
e) Declining to grant rights under trademark law for use of some
|
|
||||||
trade names, trademarks, or service marks; or
|
|
||||||
|
|
||||||
f) Requiring indemnification of licensors and authors of that
|
|
||||||
material by anyone who conveys the material (or modified versions of
|
|
||||||
it) with contractual assumptions of liability to the recipient, for
|
|
||||||
any liability that these contractual assumptions directly impose on
|
|
||||||
those licensors and authors.
|
|
||||||
|
|
||||||
All other non-permissive additional terms are considered "further
|
|
||||||
restrictions" within the meaning of section 10. If the Program as you
|
|
||||||
received it, or any part of it, contains a notice stating that it is
|
|
||||||
governed by this License along with a term that is a further
|
|
||||||
restriction, you may remove that term. If a license document contains
|
|
||||||
a further restriction but permits relicensing or conveying under this
|
|
||||||
License, you may add to a covered work material governed by the terms
|
|
||||||
of that license document, provided that the further restriction does
|
|
||||||
not survive such relicensing or conveying.
|
|
||||||
|
|
||||||
If you add terms to a covered work in accord with this section, you
|
|
||||||
must place, in the relevant source files, a statement of the
|
|
||||||
additional terms that apply to those files, or a notice indicating
|
|
||||||
where to find the applicable terms.
|
|
||||||
|
|
||||||
Additional terms, permissive or non-permissive, may be stated in the
|
|
||||||
form of a separately written license, or stated as exceptions;
|
|
||||||
the above requirements apply either way.
|
|
||||||
|
|
||||||
8. Termination.
|
|
||||||
|
|
||||||
You may not propagate or modify a covered work except as expressly
|
|
||||||
provided under this License. Any attempt otherwise to propagate or
|
|
||||||
modify it is void, and will automatically terminate your rights under
|
|
||||||
this License (including any patent licenses granted under the third
|
|
||||||
paragraph of section 11).
|
|
||||||
|
|
||||||
However, if you cease all violation of this License, then your
|
|
||||||
license from a particular copyright holder is reinstated (a)
|
|
||||||
provisionally, unless and until the copyright holder explicitly and
|
|
||||||
finally terminates your license, and (b) permanently, if the copyright
|
|
||||||
holder fails to notify you of the violation by some reasonable means
|
|
||||||
prior to 60 days after the cessation.
|
|
||||||
|
|
||||||
Moreover, your license from a particular copyright holder is
|
|
||||||
reinstated permanently if the copyright holder notifies you of the
|
|
||||||
violation by some reasonable means, this is the first time you have
|
|
||||||
received notice of violation of this License (for any work) from that
|
|
||||||
copyright holder, and you cure the violation prior to 30 days after
|
|
||||||
your receipt of the notice.
|
|
||||||
|
|
||||||
Termination of your rights under this section does not terminate the
|
|
||||||
licenses of parties who have received copies or rights from you under
|
|
||||||
this License. If your rights have been terminated and not permanently
|
|
||||||
reinstated, you do not qualify to receive new licenses for the same
|
|
||||||
material under section 10.
|
|
||||||
|
|
||||||
9. Acceptance Not Required for Having Copies.
|
|
||||||
|
|
||||||
You are not required to accept this License in order to receive or
|
|
||||||
run a copy of the Program. Ancillary propagation of a covered work
|
|
||||||
occurring solely as a consequence of using peer-to-peer transmission
|
|
||||||
to receive a copy likewise does not require acceptance. However,
|
|
||||||
nothing other than this License grants you permission to propagate or
|
|
||||||
modify any covered work. These actions infringe copyright if you do
|
|
||||||
not accept this License. Therefore, by modifying or propagating a
|
|
||||||
covered work, you indicate your acceptance of this License to do so.
|
|
||||||
|
|
||||||
10. Automatic Licensing of Downstream Recipients.
|
|
||||||
|
|
||||||
Each time you convey a covered work, the recipient automatically
|
|
||||||
receives a license from the original licensors, to run, modify and
|
|
||||||
propagate that work, subject to this License. You are not responsible
|
|
||||||
for enforcing compliance by third parties with this License.
|
|
||||||
|
|
||||||
An "entity transaction" is a transaction transferring control of an
|
|
||||||
organization, or substantially all assets of one, or subdividing an
|
|
||||||
organization, or merging organizations. If propagation of a covered
|
|
||||||
work results from an entity transaction, each party to that
|
|
||||||
transaction who receives a copy of the work also receives whatever
|
|
||||||
licenses to the work the party's predecessor in interest had or could
|
|
||||||
give under the previous paragraph, plus a right to possession of the
|
|
||||||
Corresponding Source of the work from the predecessor in interest, if
|
|
||||||
the predecessor has it or can get it with reasonable efforts.
|
|
||||||
|
|
||||||
You may not impose any further restrictions on the exercise of the
|
|
||||||
rights granted or affirmed under this License. For example, you may
|
|
||||||
not impose a license fee, royalty, or other charge for exercise of
|
|
||||||
rights granted under this License, and you may not initiate litigation
|
|
||||||
(including a cross-claim or counterclaim in a lawsuit) alleging that
|
|
||||||
any patent claim is infringed by making, using, selling, offering for
|
|
||||||
sale, or importing the Program or any portion of it.
|
|
||||||
|
|
||||||
11. Patents.
|
|
||||||
|
|
||||||
A "contributor" is a copyright holder who authorizes use under this
|
|
||||||
License of the Program or a work on which the Program is based. The
|
|
||||||
work thus licensed is called the contributor's "contributor version".
|
|
||||||
|
|
||||||
A contributor's "essential patent claims" are all patent claims
|
|
||||||
owned or controlled by the contributor, whether already acquired or
|
|
||||||
hereafter acquired, that would be infringed by some manner, permitted
|
|
||||||
by this License, of making, using, or selling its contributor version,
|
|
||||||
but do not include claims that would be infringed only as a
|
|
||||||
consequence of further modification of the contributor version. For
|
|
||||||
purposes of this definition, "control" includes the right to grant
|
|
||||||
patent sublicenses in a manner consistent with the requirements of
|
|
||||||
this License.
|
|
||||||
|
|
||||||
Each contributor grants you a non-exclusive, worldwide, royalty-free
|
|
||||||
patent license under the contributor's essential patent claims, to
|
|
||||||
make, use, sell, offer for sale, import and otherwise run, modify and
|
|
||||||
propagate the contents of its contributor version.
|
|
||||||
|
|
||||||
In the following three paragraphs, a "patent license" is any express
|
|
||||||
agreement or commitment, however denominated, not to enforce a patent
|
|
||||||
(such as an express permission to practice a patent or covenant not to
|
|
||||||
sue for patent infringement). To "grant" such a patent license to a
|
|
||||||
party means to make such an agreement or commitment not to enforce a
|
|
||||||
patent against the party.
|
|
||||||
|
|
||||||
If you convey a covered work, knowingly relying on a patent license,
|
|
||||||
and the Corresponding Source of the work is not available for anyone
|
|
||||||
to copy, free of charge and under the terms of this License, through a
|
|
||||||
publicly available network server or other readily accessible means,
|
|
||||||
then you must either (1) cause the Corresponding Source to be so
|
|
||||||
available, or (2) arrange to deprive yourself of the benefit of the
|
|
||||||
patent license for this particular work, or (3) arrange, in a manner
|
|
||||||
consistent with the requirements of this License, to extend the patent
|
|
||||||
license to downstream recipients. "Knowingly relying" means you have
|
|
||||||
actual knowledge that, but for the patent license, your conveying the
|
|
||||||
covered work in a country, or your recipient's use of the covered work
|
|
||||||
in a country, would infringe one or more identifiable patents in that
|
|
||||||
country that you have reason to believe are valid.
|
|
||||||
|
|
||||||
If, pursuant to or in connection with a single transaction or
|
|
||||||
arrangement, you convey, or propagate by procuring conveyance of, a
|
|
||||||
covered work, and grant a patent license to some of the parties
|
|
||||||
receiving the covered work authorizing them to use, propagate, modify
|
|
||||||
or convey a specific copy of the covered work, then the patent license
|
|
||||||
you grant is automatically extended to all recipients of the covered
|
|
||||||
work and works based on it.
|
|
||||||
|
|
||||||
A patent license is "discriminatory" if it does not include within
|
|
||||||
the scope of its coverage, prohibits the exercise of, or is
|
|
||||||
conditioned on the non-exercise of one or more of the rights that are
|
|
||||||
specifically granted under this License. You may not convey a covered
|
|
||||||
work if you are a party to an arrangement with a third party that is
|
|
||||||
in the business of distributing software, under which you make payment
|
|
||||||
to the third party based on the extent of your activity of conveying
|
|
||||||
the work, and under which the third party grants, to any of the
|
|
||||||
parties who would receive the covered work from you, a discriminatory
|
|
||||||
patent license (a) in connection with copies of the covered work
|
|
||||||
conveyed by you (or copies made from those copies), or (b) primarily
|
|
||||||
for and in connection with specific products or compilations that
|
|
||||||
contain the covered work, unless you entered into that arrangement,
|
|
||||||
or that patent license was granted, prior to 28 March 2007.
|
|
||||||
|
|
||||||
Nothing in this License shall be construed as excluding or limiting
|
|
||||||
any implied license or other defenses to infringement that may
|
|
||||||
otherwise be available to you under applicable patent law.
|
|
||||||
|
|
||||||
12. No Surrender of Others' Freedom.
|
|
||||||
|
|
||||||
If conditions are imposed on you (whether by court order, agreement or
|
|
||||||
otherwise) that contradict the conditions of this License, they do not
|
|
||||||
excuse you from the conditions of this License. If you cannot convey a
|
|
||||||
covered work so as to satisfy simultaneously your obligations under this
|
|
||||||
License and any other pertinent obligations, then as a consequence you may
|
|
||||||
not convey it at all. For example, if you agree to terms that obligate you
|
|
||||||
to collect a royalty for further conveying from those to whom you convey
|
|
||||||
the Program, the only way you could satisfy both those terms and this
|
|
||||||
License would be to refrain entirely from conveying the Program.
|
|
||||||
|
|
||||||
13. Use with the GNU Affero General Public License.
|
|
||||||
|
|
||||||
Notwithstanding any other provision of this License, you have
|
|
||||||
permission to link or combine any covered work with a work licensed
|
|
||||||
under version 3 of the GNU Affero General Public License into a single
|
|
||||||
combined work, and to convey the resulting work. The terms of this
|
|
||||||
License will continue to apply to the part which is the covered work,
|
|
||||||
but the special requirements of the GNU Affero General Public License,
|
|
||||||
section 13, concerning interaction through a network will apply to the
|
|
||||||
combination as such.
|
|
||||||
|
|
||||||
14. Revised Versions of this License.
|
|
||||||
|
|
||||||
The Free Software Foundation may publish revised and/or new versions of
|
|
||||||
the GNU General Public License from time to time. Such new versions will
|
|
||||||
be similar in spirit to the present version, but may differ in detail to
|
|
||||||
address new problems or concerns.
|
|
||||||
|
|
||||||
Each version is given a distinguishing version number. If the
|
|
||||||
Program specifies that a certain numbered version of the GNU General
|
|
||||||
Public License "or any later version" applies to it, you have the
|
|
||||||
option of following the terms and conditions either of that numbered
|
|
||||||
version or of any later version published by the Free Software
|
|
||||||
Foundation. If the Program does not specify a version number of the
|
|
||||||
GNU General Public License, you may choose any version ever published
|
|
||||||
by the Free Software Foundation.
|
|
||||||
|
|
||||||
If the Program specifies that a proxy can decide which future
|
|
||||||
versions of the GNU General Public License can be used, that proxy's
|
|
||||||
public statement of acceptance of a version permanently authorizes you
|
|
||||||
to choose that version for the Program.
|
|
||||||
|
|
||||||
Later license versions may give you additional or different
|
|
||||||
permissions. However, no additional obligations are imposed on any
|
|
||||||
author or copyright holder as a result of your choosing to follow a
|
|
||||||
later version.
|
|
||||||
|
|
||||||
15. Disclaimer of Warranty.
|
|
||||||
|
|
||||||
THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
|
|
||||||
APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
|
|
||||||
HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
|
|
||||||
OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
|
|
||||||
THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
||||||
PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
|
|
||||||
IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
|
|
||||||
ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
|
|
||||||
|
|
||||||
16. Limitation of Liability.
|
|
||||||
|
|
||||||
IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
|
||||||
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
|
|
||||||
THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
|
|
||||||
GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
|
|
||||||
USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
|
|
||||||
DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
|
|
||||||
PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
|
|
||||||
EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
|
|
||||||
SUCH DAMAGES.
|
|
||||||
|
|
||||||
17. Interpretation of Sections 15 and 16.
|
|
||||||
|
|
||||||
If the disclaimer of warranty and limitation of liability provided
|
|
||||||
above cannot be given local legal effect according to their terms,
|
|
||||||
reviewing courts shall apply local law that most closely approximates
|
|
||||||
an absolute waiver of all civil liability in connection with the
|
|
||||||
Program, unless a warranty or assumption of liability accompanies a
|
|
||||||
copy of the Program in return for a fee.
|
|
||||||
|
|
||||||
END OF TERMS AND CONDITIONS
|
|
||||||
|
|
||||||
How to Apply These Terms to Your New Programs
|
|
||||||
|
|
||||||
If you develop a new program, and you want it to be of the greatest
|
|
||||||
possible use to the public, the best way to achieve this is to make it
|
|
||||||
free software which everyone can redistribute and change under these terms.
|
|
||||||
|
|
||||||
To do so, attach the following notices to the program. It is safest
|
|
||||||
to attach them to the start of each source file to most effectively
|
|
||||||
state the exclusion of warranty; and each file should have at least
|
|
||||||
the "copyright" line and a pointer to where the full notice is found.
|
|
||||||
|
|
||||||
<one line to give the program's name and a brief idea of what it does.>
|
|
||||||
Copyright (C) <year> <name of author>
|
|
||||||
|
|
||||||
This program is free software: you can redistribute it and/or modify
|
|
||||||
it under the terms of the GNU 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 General Public License for more details.
|
|
||||||
|
|
||||||
You should have received a copy of the GNU General Public License
|
|
||||||
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
||||||
|
|
||||||
Also add information on how to contact you by electronic and paper mail.
|
|
||||||
|
|
||||||
If the program does terminal interaction, make it output a short
|
|
||||||
notice like this when it starts in an interactive mode:
|
|
||||||
|
|
||||||
<program> Copyright (C) <year> <name of author>
|
|
||||||
This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
|
||||||
This is free software, and you are welcome to redistribute it
|
|
||||||
under certain conditions; type `show c' for details.
|
|
||||||
|
|
||||||
The hypothetical commands `show w' and `show c' should show the appropriate
|
|
||||||
parts of the General Public License. Of course, your program's commands
|
|
||||||
might be different; for a GUI interface, you would use an "about box".
|
|
||||||
|
|
||||||
You should also get your employer (if you work as a programmer) or school,
|
|
||||||
if any, to sign a "copyright disclaimer" for the program, if necessary.
|
|
||||||
For more information on this, and how to apply and follow the GNU GPL, see
|
|
||||||
<https://www.gnu.org/licenses/>.
|
|
||||||
|
|
||||||
The GNU General Public License does not permit incorporating your program
|
|
||||||
into proprietary programs. If your program is a subroutine library, you
|
|
||||||
may consider it more useful to permit linking proprietary applications with
|
|
||||||
the library. If this is what you want to do, use the GNU Lesser General
|
|
||||||
Public License instead of this License. But first, please read
|
|
||||||
<https://www.gnu.org/licenses/why-not-lgpl.html>.
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
# Simple Config Parser
|
|
||||||
|
|
||||||
A custom config parser inspired by Python's configparser module.
|
|
||||||
Specialized for handling Klipper style config files.
|
|
||||||
|
|
||||||
|
|
||||||
@@ -1,66 +0,0 @@
|
|||||||
[project]
|
|
||||||
name = "simple-config-parser"
|
|
||||||
version = "0.0.1"
|
|
||||||
description = "A simple config parser for Python"
|
|
||||||
authors = [
|
|
||||||
{name = "Dominik Willner", email = "th33xitus@gmail.com"},
|
|
||||||
]
|
|
||||||
readme = "README.md"
|
|
||||||
license = {text = "GPL-3.0-only"}
|
|
||||||
requires-python = ">=3.8"
|
|
||||||
|
|
||||||
[project.urls]
|
|
||||||
homepage = "https://github.com/dw-0/simple-config-parser"
|
|
||||||
repository = "https://github.com/dw-0/simple-config-parser"
|
|
||||||
documentation = "https://github.com/dw-0/simple-config-parser"
|
|
||||||
|
|
||||||
[project.optional-dependencies]
|
|
||||||
dev=["ruff"]
|
|
||||||
|
|
||||||
[tool.ruff]
|
|
||||||
required-version = ">=0.3.4"
|
|
||||||
respect-gitignore = true
|
|
||||||
exclude = [".git",".github", "./docs"]
|
|
||||||
line-length = 88
|
|
||||||
indent-width = 4
|
|
||||||
output-format = "full"
|
|
||||||
|
|
||||||
[tool.ruff.format]
|
|
||||||
indent-style = "space"
|
|
||||||
line-ending = "lf"
|
|
||||||
quote-style = "double"
|
|
||||||
|
|
||||||
[tool.ruff.lint]
|
|
||||||
extend-select = ["I"]
|
|
||||||
|
|
||||||
[tool.pytest.ini_options]
|
|
||||||
minversion = "8.2.1"
|
|
||||||
testpaths = ["tests/**/*.py"]
|
|
||||||
addopts = "--cov --cov-config=pyproject.toml --cov-report=html"
|
|
||||||
|
|
||||||
[tool.coverage.run]
|
|
||||||
branch = true
|
|
||||||
source = ["src.simple_config_parser"]
|
|
||||||
|
|
||||||
[tool.coverage.report]
|
|
||||||
# Regexes for lines to exclude from consideration
|
|
||||||
exclude_also = [
|
|
||||||
# Don't complain about missing debug-only code:
|
|
||||||
"def __repr__",
|
|
||||||
"if self\\.debug",
|
|
||||||
|
|
||||||
# Don't complain if tests don't hit defensive assertion code:
|
|
||||||
"raise AssertionError",
|
|
||||||
"raise NotImplementedError",
|
|
||||||
|
|
||||||
# Don't complain if non-runnable code isn't run:
|
|
||||||
"if 0:",
|
|
||||||
"if __name__ == .__main__.:",
|
|
||||||
|
|
||||||
# Don't complain about abstract methods, they aren't run:
|
|
||||||
"@(abc\\.)?abstractmethod",
|
|
||||||
]
|
|
||||||
|
|
||||||
[tool.coverage.html]
|
|
||||||
title = "SimpleConfigParser Coverage Report"
|
|
||||||
directory = "htmlcov"
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
ruff >= 0.3.4
|
|
||||||
pytest >= 8.2.1
|
|
||||||
pytest-cov >= 5.0.0
|
|
||||||
@@ -1,62 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# Copyright (C) 2024 Dominik Willner <th33xitus@gmail.com> #
|
|
||||||
# #
|
|
||||||
# https://github.com/dw-0/simple-config-parser #
|
|
||||||
# #
|
|
||||||
# This file may be distributed under the terms of the GNU GPLv3 license #
|
|
||||||
# ======================================================================= #
|
|
||||||
import re
|
|
||||||
|
|
||||||
# definition of section line:
|
|
||||||
# - then line MUST start with an opening square bracket - it is the first section marker
|
|
||||||
# - the section marker MUST be followed by at least one character - it is the section name
|
|
||||||
# - the section name MUST be followed by a closing square bracket - it is the second section marker
|
|
||||||
# - the second section marker MAY be followed by any amount of whitespace characters
|
|
||||||
# - the second section marker MAY be followed by a # or ; - it is the comment marker
|
|
||||||
# - the inline comment MAY be of any length and character
|
|
||||||
SECTION_RE = re.compile(r"^\[(\S.*\S|\S)]\s*([#;].*)?$")
|
|
||||||
|
|
||||||
# definition of option line:
|
|
||||||
# - the line MUST start with a word - it is the option name
|
|
||||||
# - the option name MUST be followed by a colon or an equal sign - it is the separator
|
|
||||||
# - the separator MUST be followed by a value
|
|
||||||
# - the separator MAY have any amount of leading or trailing whitespaces
|
|
||||||
# - the separator MUST NOT be directly followed by a colon or equal sign
|
|
||||||
# - the value MAY be of any length and character
|
|
||||||
# - the value MAY contain any amount of trailing whitespaces
|
|
||||||
# - the value MAY be followed by a # or ; - it is the comment marker
|
|
||||||
# - the inline comment MAY be of any length and character
|
|
||||||
OPTION_RE = re.compile(r"^([^;#:=\s]+)\s?[:=]\s*([^;#:=\s][^;#]*?)\s*([#;].*)?$")
|
|
||||||
# definition of options block start line:
|
|
||||||
# - the line MUST start with a word - it is the option name
|
|
||||||
# - the option name MUST be followed by a colon or an equal sign - it is the separator
|
|
||||||
# - the separator MUST NOT be followed by a value
|
|
||||||
# - the separator MAY have any amount of leading or trailing whitespaces
|
|
||||||
# - the separator MUST NOT be directly followed by a colon or equal sign
|
|
||||||
# - the separator MAY be followed by a # or ; - it is the comment marker
|
|
||||||
# - the inline comment MAY be of any length and character
|
|
||||||
OPTIONS_BLOCK_START_RE = re.compile(r"^([^;#:=\s]+)\s*[:=]\s*([#;].*)?$")
|
|
||||||
|
|
||||||
# definition of comment line:
|
|
||||||
# - the line MAY start with any amount of whitespace characters
|
|
||||||
# - the line MUST contain a # or ; - it is the comment marker
|
|
||||||
# - the comment marker MAY be followed by any amount of whitespace characters
|
|
||||||
# - the comment MAY be of any length and character
|
|
||||||
LINE_COMMENT_RE = re.compile(r"^\s*[#;].*")
|
|
||||||
|
|
||||||
# definition of empty line:
|
|
||||||
# - the line MUST contain only whitespace characters
|
|
||||||
EMPTY_LINE_RE = re.compile(r"^\s*$")
|
|
||||||
|
|
||||||
BOOLEAN_STATES = {
|
|
||||||
"1": True,
|
|
||||||
"yes": True,
|
|
||||||
"true": True,
|
|
||||||
"on": True,
|
|
||||||
"0": False,
|
|
||||||
"no": False,
|
|
||||||
"false": False,
|
|
||||||
"off": False,
|
|
||||||
}
|
|
||||||
|
|
||||||
HEADER_IDENT = "#_header"
|
|
||||||
@@ -1,325 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# Copyright (C) 2024 Dominik Willner <th33xitus@gmail.com> #
|
|
||||||
# #
|
|
||||||
# https://github.com/dw-0/simple-config-parser #
|
|
||||||
# #
|
|
||||||
# This file may be distributed under the terms of the GNU GPLv3 license #
|
|
||||||
# ======================================================================= #
|
|
||||||
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import secrets
|
|
||||||
import string
|
|
||||||
from pathlib import Path
|
|
||||||
from typing import Callable, Dict, List
|
|
||||||
|
|
||||||
from ..simple_config_parser.constants import (
|
|
||||||
BOOLEAN_STATES,
|
|
||||||
EMPTY_LINE_RE,
|
|
||||||
HEADER_IDENT,
|
|
||||||
LINE_COMMENT_RE,
|
|
||||||
OPTION_RE,
|
|
||||||
OPTIONS_BLOCK_START_RE,
|
|
||||||
SECTION_RE,
|
|
||||||
)
|
|
||||||
|
|
||||||
_UNSET = object()
|
|
||||||
|
|
||||||
|
|
||||||
class NoSectionError(Exception):
|
|
||||||
"""Raised when a section is not defined"""
|
|
||||||
|
|
||||||
def __init__(self, section: str):
|
|
||||||
msg = f"Section '{section}' is not defined"
|
|
||||||
super().__init__(msg)
|
|
||||||
|
|
||||||
|
|
||||||
class DuplicateSectionError(Exception):
|
|
||||||
"""Raised when a section is defined more than once"""
|
|
||||||
|
|
||||||
def __init__(self, section: str):
|
|
||||||
msg = f"Section '{section}' is defined more than once"
|
|
||||||
super().__init__(msg)
|
|
||||||
|
|
||||||
|
|
||||||
class NoOptionError(Exception):
|
|
||||||
"""Raised when an option is not defined in a section"""
|
|
||||||
|
|
||||||
def __init__(self, option: str, section: str):
|
|
||||||
msg = f"Option '{option}' in section '{section}' is not defined"
|
|
||||||
super().__init__(msg)
|
|
||||||
|
|
||||||
|
|
||||||
# noinspection PyMethodMayBeStatic
|
|
||||||
class SimpleConfigParser:
|
|
||||||
"""A customized config parser targeted at handling Klipper style config files"""
|
|
||||||
|
|
||||||
def __init__(self) -> None:
|
|
||||||
self.header: List[str] = []
|
|
||||||
self.config: Dict = {}
|
|
||||||
self.current_section: str | None = None
|
|
||||||
self.current_opt_block: str | None = None
|
|
||||||
self.current_collector: str | None = None
|
|
||||||
self.in_option_block: bool = False
|
|
||||||
|
|
||||||
def _match_section(self, line: str) -> bool:
|
|
||||||
"""Wheter or not the given line matches the definition of a section"""
|
|
||||||
return SECTION_RE.match(line) is not None
|
|
||||||
|
|
||||||
def _match_option(self, line: str) -> bool:
|
|
||||||
"""Wheter or not the given line matches the definition of an option"""
|
|
||||||
return OPTION_RE.match(line) is not None
|
|
||||||
|
|
||||||
def _match_options_block_start(self, line: str) -> bool:
|
|
||||||
"""Wheter or not the given line matches the definition of a multiline option"""
|
|
||||||
return OPTIONS_BLOCK_START_RE.match(line) is not None
|
|
||||||
|
|
||||||
def _match_line_comment(self, line: str) -> bool:
|
|
||||||
"""Wheter or not the given line matches the definition of a comment"""
|
|
||||||
return LINE_COMMENT_RE.match(line) is not None
|
|
||||||
|
|
||||||
def _match_empty_line(self, line: str) -> bool:
|
|
||||||
"""Wheter or not the given line matches the definition of an empty line"""
|
|
||||||
return EMPTY_LINE_RE.match(line) is not None
|
|
||||||
|
|
||||||
def _parse_line(self, line: str) -> None:
|
|
||||||
"""Parses a line and determines its type"""
|
|
||||||
if self._match_section(line):
|
|
||||||
self.current_collector = None
|
|
||||||
self.current_opt_block = None
|
|
||||||
self.current_section = SECTION_RE.match(line).group(1)
|
|
||||||
self.config[self.current_section] = {"_raw": line}
|
|
||||||
|
|
||||||
elif self._match_option(line):
|
|
||||||
self.current_collector = None
|
|
||||||
self.current_opt_block = None
|
|
||||||
option = OPTION_RE.match(line).group(1)
|
|
||||||
value = OPTION_RE.match(line).group(2)
|
|
||||||
self.config[self.current_section][option] = {"_raw": line, "value": value}
|
|
||||||
|
|
||||||
elif self._match_options_block_start(line):
|
|
||||||
self.current_collector = None
|
|
||||||
option = OPTIONS_BLOCK_START_RE.match(line).group(1)
|
|
||||||
self.current_opt_block = option
|
|
||||||
self.config[self.current_section][option] = {"_raw": line, "value": []}
|
|
||||||
|
|
||||||
elif self.current_opt_block is not None:
|
|
||||||
self.config[self.current_section][self.current_opt_block]["value"].append(
|
|
||||||
line
|
|
||||||
)
|
|
||||||
|
|
||||||
elif self._match_empty_line(line) or self._match_line_comment(line):
|
|
||||||
self.current_opt_block = None
|
|
||||||
|
|
||||||
# if current_section is None, we are at the beginning of the file,
|
|
||||||
# so we consider the part up to the first section as the file header
|
|
||||||
if not self.current_section:
|
|
||||||
self.config.setdefault(HEADER_IDENT, []).append(line)
|
|
||||||
else:
|
|
||||||
section = self.config[self.current_section]
|
|
||||||
|
|
||||||
# set the current collector to a new value, so that continuous
|
|
||||||
# empty lines or comments are collected into the same collector
|
|
||||||
if not self.current_collector:
|
|
||||||
self.current_collector = self._generate_rand_id()
|
|
||||||
section[self.current_collector] = []
|
|
||||||
|
|
||||||
section[self.current_collector].append(line)
|
|
||||||
|
|
||||||
def read_file(self, file: Path) -> None:
|
|
||||||
"""Read and parse a config file"""
|
|
||||||
with open(file, "r") as file:
|
|
||||||
for line in file:
|
|
||||||
self._parse_line(line)
|
|
||||||
|
|
||||||
# print(json.dumps(self.config, indent=4))
|
|
||||||
|
|
||||||
def write_file(self, file: Path) -> None:
|
|
||||||
"""Write the current config to the config file"""
|
|
||||||
if not file:
|
|
||||||
raise ValueError("No config file specified")
|
|
||||||
|
|
||||||
with open(file, "w") as file:
|
|
||||||
self._write_header(file)
|
|
||||||
self._write_sections(file)
|
|
||||||
|
|
||||||
def _write_header(self, file) -> None:
|
|
||||||
"""Write the header to the config file"""
|
|
||||||
for line in self.config.get(HEADER_IDENT, []):
|
|
||||||
file.write(line)
|
|
||||||
|
|
||||||
def _write_sections(self, file) -> None:
|
|
||||||
"""Write the sections to the config file"""
|
|
||||||
for section in self.get_sections():
|
|
||||||
for key, value in self.config[section].items():
|
|
||||||
self._write_section_content(file, key, value)
|
|
||||||
|
|
||||||
def _write_section_content(self, file, key, value) -> None:
|
|
||||||
"""Write the content of a section to the config file"""
|
|
||||||
if key == "_raw":
|
|
||||||
file.write(value)
|
|
||||||
elif key.startswith("#_"):
|
|
||||||
for line in value:
|
|
||||||
file.write(line)
|
|
||||||
elif isinstance(value["value"], list):
|
|
||||||
file.write(value["_raw"])
|
|
||||||
for line in value["value"]:
|
|
||||||
file.write(line)
|
|
||||||
else:
|
|
||||||
file.write(value["_raw"])
|
|
||||||
|
|
||||||
def get_sections(self) -> List[str]:
|
|
||||||
"""Return a list of all section names, but exclude any section starting with '#_'"""
|
|
||||||
return list(
|
|
||||||
filter(
|
|
||||||
lambda section: not section.startswith("#_"),
|
|
||||||
self.config.keys(),
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
def has_section(self, section: str) -> bool:
|
|
||||||
"""Check if a section exists"""
|
|
||||||
return section in self.get_sections()
|
|
||||||
|
|
||||||
def add_section(self, section: str) -> None:
|
|
||||||
"""Add a new section to the config"""
|
|
||||||
if section in self.get_sections():
|
|
||||||
raise DuplicateSectionError(section)
|
|
||||||
|
|
||||||
if len(self.get_sections()) >= 1:
|
|
||||||
self._check_set_section_spacing()
|
|
||||||
|
|
||||||
self.config[section] = {"_raw": f"[{section}]\n"}
|
|
||||||
|
|
||||||
def _check_set_section_spacing(self):
|
|
||||||
prev_section_name: str = self.get_sections()[-1]
|
|
||||||
prev_section_content: Dict = self.config[prev_section_name]
|
|
||||||
last_option_name: str = list(prev_section_content.keys())[-1]
|
|
||||||
|
|
||||||
if last_option_name.startswith("#_"):
|
|
||||||
last_elem_value: str = prev_section_content[last_option_name][-1]
|
|
||||||
|
|
||||||
# if the last section is a collector, we first check if the last element
|
|
||||||
# in the collector ends with a newline. if it does not, we append a newline.
|
|
||||||
# this can happen if the config file does not end with a newline.
|
|
||||||
if not last_elem_value.endswith("\n"):
|
|
||||||
prev_section_content[last_option_name][-1] = f"{last_elem_value}\n"
|
|
||||||
|
|
||||||
# if the last item in a collector is not a newline, we append a newline, so
|
|
||||||
# that the new section is seperated from the options of the previous section
|
|
||||||
# by a newline
|
|
||||||
if last_elem_value != "\n":
|
|
||||||
prev_section_content[last_option_name].append("\n")
|
|
||||||
else:
|
|
||||||
prev_section_content[self._generate_rand_id()] = ["\n"]
|
|
||||||
|
|
||||||
def remove_section(self, section: str) -> None:
|
|
||||||
"""Remove a section from the config"""
|
|
||||||
self.config.pop(section, None)
|
|
||||||
|
|
||||||
def get_options(self, section: str) -> List[str]:
|
|
||||||
"""Return a list of all option names for a given section"""
|
|
||||||
return list(
|
|
||||||
filter(
|
|
||||||
lambda option: option != "_raw" and not option.startswith("#_"),
|
|
||||||
self.config[section].keys(),
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
def has_option(self, section: str, option: str) -> bool:
|
|
||||||
"""Check if an option exists in a section"""
|
|
||||||
return self.has_section(section) and option in self.get_options(section)
|
|
||||||
|
|
||||||
def set_option(self, section: str, option: str, value: str | List[str]) -> None:
|
|
||||||
"""
|
|
||||||
Set the value of an option in a section. If the section does not exist,
|
|
||||||
it is created. If the option does not exist, it is created.
|
|
||||||
"""
|
|
||||||
if not self.has_section(section):
|
|
||||||
self.add_section(section)
|
|
||||||
|
|
||||||
if not self.has_option(section, option):
|
|
||||||
self.config[section][option] = {
|
|
||||||
"_raw": f"{option}:\n"
|
|
||||||
if isinstance(value, list)
|
|
||||||
else f"{option}: {value}\n",
|
|
||||||
"value": value,
|
|
||||||
}
|
|
||||||
else:
|
|
||||||
opt = self.config[section][option]
|
|
||||||
if not isinstance(value, list):
|
|
||||||
opt["_raw"] = opt["_raw"].replace(opt["value"], value)
|
|
||||||
opt["value"] = value
|
|
||||||
|
|
||||||
def remove_option(self, section: str, option: str) -> None:
|
|
||||||
"""Remove an option from a section"""
|
|
||||||
self.config[section].pop(option, None)
|
|
||||||
|
|
||||||
def getval(
|
|
||||||
self, section: str, option: str, fallback: str | _UNSET = _UNSET
|
|
||||||
) -> str | List[str]:
|
|
||||||
"""
|
|
||||||
Return the value of the given option in the given section
|
|
||||||
|
|
||||||
If the key is not found and 'fallback' is provided, it is used as
|
|
||||||
a fallback value.
|
|
||||||
"""
|
|
||||||
try:
|
|
||||||
if section not in self.get_sections():
|
|
||||||
raise NoSectionError(section)
|
|
||||||
if option not in self.get_options(section):
|
|
||||||
raise NoOptionError(option, section)
|
|
||||||
return self.config[section][option]["value"]
|
|
||||||
except (NoSectionError, NoOptionError):
|
|
||||||
if fallback is _UNSET:
|
|
||||||
raise
|
|
||||||
return fallback
|
|
||||||
|
|
||||||
def getint(self, section: str, option: str, fallback: int | _UNSET = _UNSET) -> int:
|
|
||||||
"""Return the value of the given option in the given section as an int"""
|
|
||||||
return self._get_conv(section, option, int, fallback=fallback)
|
|
||||||
|
|
||||||
def getfloat(
|
|
||||||
self, section: str, option: str, fallback: float | _UNSET = _UNSET
|
|
||||||
) -> float:
|
|
||||||
"""Return the value of the given option in the given section as a float"""
|
|
||||||
return self._get_conv(section, option, float, fallback=fallback)
|
|
||||||
|
|
||||||
def getboolean(
|
|
||||||
self, section: str, option: str, fallback: bool | _UNSET = _UNSET
|
|
||||||
) -> bool:
|
|
||||||
"""Return the value of the given option in the given section as a boolean"""
|
|
||||||
return self._get_conv(
|
|
||||||
section, option, self._convert_to_boolean, fallback=fallback
|
|
||||||
)
|
|
||||||
|
|
||||||
def _convert_to_boolean(self, value: str) -> bool:
|
|
||||||
"""Convert a string to a boolean"""
|
|
||||||
if isinstance(value, bool):
|
|
||||||
return value
|
|
||||||
if value.lower() not in BOOLEAN_STATES:
|
|
||||||
raise ValueError("Not a boolean: %s" % value)
|
|
||||||
return BOOLEAN_STATES[value.lower()]
|
|
||||||
|
|
||||||
def _get_conv(
|
|
||||||
self,
|
|
||||||
section: str,
|
|
||||||
option: str,
|
|
||||||
conv: Callable[[str], int | float | bool],
|
|
||||||
fallback: _UNSET = _UNSET,
|
|
||||||
) -> int | float | bool:
|
|
||||||
"""Return the value of the given option in the given section as a converted value"""
|
|
||||||
try:
|
|
||||||
return conv(self.getval(section, option, fallback))
|
|
||||||
except (ValueError, TypeError, AttributeError) as e:
|
|
||||||
if fallback is not _UNSET:
|
|
||||||
return fallback
|
|
||||||
raise ValueError(
|
|
||||||
f"Cannot convert {self.getval(section, option)} to {conv.__name__}"
|
|
||||||
) from e
|
|
||||||
|
|
||||||
def _generate_rand_id(self) -> str:
|
|
||||||
"""Generate a random id with 6 characters"""
|
|
||||||
chars = string.ascii_letters + string.digits
|
|
||||||
rand_string = "".join(secrets.choice(chars) for _ in range(12))
|
|
||||||
return f"#_{rand_string}"
|
|
||||||
File diff suppressed because it is too large
Load Diff
@@ -1,32 +0,0 @@
|
|||||||
# a comment at the very top
|
|
||||||
# should be treated as the file header
|
|
||||||
|
|
||||||
# up to the first section, including all blank lines
|
|
||||||
|
|
||||||
[section_1]
|
|
||||||
option_1: value_1
|
|
||||||
option_1_1: True # this is a boolean
|
|
||||||
option_1_2: 5 ; this is an integer
|
|
||||||
option_1_3: 1.123 #;this is a float
|
|
||||||
|
|
||||||
[section_2] ; comment
|
|
||||||
option_2: value_2
|
|
||||||
|
|
||||||
; comment
|
|
||||||
|
|
||||||
[section_3]
|
|
||||||
option_3: value_3 # comment
|
|
||||||
|
|
||||||
[section_4]
|
|
||||||
# comment
|
|
||||||
option_4: value_4
|
|
||||||
|
|
||||||
[section number 5]
|
|
||||||
#option_5: value_5
|
|
||||||
option_5 = this.is.value-5
|
|
||||||
multi_option:
|
|
||||||
# these are multi-line values
|
|
||||||
value_5_1
|
|
||||||
value_5_2 ; here is a comment
|
|
||||||
value_5_3
|
|
||||||
option_5_1: value_5_1
|
|
||||||
@@ -1,33 +0,0 @@
|
|||||||
# a comment at the very top
|
|
||||||
# should be treated as the file header
|
|
||||||
|
|
||||||
# up to the first section, including all blank lines
|
|
||||||
|
|
||||||
[section_1]
|
|
||||||
option_1: value_1
|
|
||||||
option_1_1: True # this is a boolean
|
|
||||||
option_1_2: 5 ; this is an integer
|
|
||||||
option_1_3: 1.123 #;this is a float
|
|
||||||
|
|
||||||
[section_2] ; comment
|
|
||||||
option_2: value_2
|
|
||||||
|
|
||||||
; comment
|
|
||||||
|
|
||||||
[section_3]
|
|
||||||
option_3: value_3 # comment
|
|
||||||
|
|
||||||
[section_4]
|
|
||||||
# comment
|
|
||||||
option_4: value_4
|
|
||||||
|
|
||||||
[section number 5]
|
|
||||||
#option_5: value_5
|
|
||||||
option_5 = this.is.value-5
|
|
||||||
multi_option:
|
|
||||||
# these are multi-line values
|
|
||||||
value_5_1
|
|
||||||
value_5_2 ; here is a comment
|
|
||||||
value_5_3
|
|
||||||
option_5_1: value_5_1
|
|
||||||
# config ending with a comment
|
|
||||||
@@ -1,94 +0,0 @@
|
|||||||
# a comment at the very top
|
|
||||||
# should be treated as the file header
|
|
||||||
|
|
||||||
# up to the first section, including all blank lines
|
|
||||||
|
|
||||||
[section_1]
|
|
||||||
option_1: value_1
|
|
||||||
option_1_1: True # this is a boolean
|
|
||||||
option_1_2: 5 ; this is an integer
|
|
||||||
option_1_3: 1.123 #;this is a float
|
|
||||||
|
|
||||||
[section_2] ; comment
|
|
||||||
option_2: value_2
|
|
||||||
|
|
||||||
; comment
|
|
||||||
|
|
||||||
[section_3]
|
|
||||||
option_3: value_3 # comment
|
|
||||||
|
|
||||||
[section_4]
|
|
||||||
# comment
|
|
||||||
option_4: value_4
|
|
||||||
|
|
||||||
[section number 5]
|
|
||||||
#option_5: value_5
|
|
||||||
option_5 = this.is.value-5
|
|
||||||
multi_option:
|
|
||||||
# these are multi-line values
|
|
||||||
value_5_1
|
|
||||||
value_5_2 ; here is a comment
|
|
||||||
value_5_3
|
|
||||||
option_5_1: value_5_1
|
|
||||||
|
|
||||||
[gcode_macro M117]
|
|
||||||
rename_existing: M117.1
|
|
||||||
gcode:
|
|
||||||
{% if rawparams %}
|
|
||||||
{% set escaped_msg = rawparams.split(';', 1)[0].split('\x23', 1)[0]|replace('"', '\\"') %}
|
|
||||||
SET_DISPLAY_TEXT MSG="{escaped_msg}"
|
|
||||||
RESPOND TYPE=command MSG="{escaped_msg}"
|
|
||||||
{% else %}
|
|
||||||
SET_DISPLAY_TEXT
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
# SDCard 'looping' (aka Marlin M808 commands) support
|
|
||||||
#
|
|
||||||
# Support SDCard looping
|
|
||||||
[sdcard_loop]
|
|
||||||
[gcode_macro M486]
|
|
||||||
gcode:
|
|
||||||
# Parameters known to M486 are as follows:
|
|
||||||
# [C<flag>] Cancel the current object
|
|
||||||
# [P<index>] Cancel the object with the given index
|
|
||||||
# [S<index>] Set the index of the current object.
|
|
||||||
# If the object with the given index has been canceled, this will cause
|
|
||||||
# the firmware to skip to the next object. The value -1 is used to
|
|
||||||
# indicate something that isn’t an object and shouldn’t be skipped.
|
|
||||||
# [T<count>] Reset the state and set the number of objects
|
|
||||||
# [U<index>] Un-cancel the object with the given index. This command will be
|
|
||||||
# ignored if the object has already been skipped
|
|
||||||
|
|
||||||
{% if 'exclude_object' not in printer %}
|
|
||||||
{action_raise_error("[exclude_object] is not enabled")}
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
{% if 'T' in params %}
|
|
||||||
EXCLUDE_OBJECT RESET=1
|
|
||||||
|
|
||||||
{% for i in range(params.T | int) %}
|
|
||||||
EXCLUDE_OBJECT_DEFINE NAME={i}
|
|
||||||
{% endfor %}
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
{% if 'C' in params %}
|
|
||||||
EXCLUDE_OBJECT CURRENT=1
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
{% if 'P' in params %}
|
|
||||||
EXCLUDE_OBJECT NAME={params.P}
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
{% if 'S' in params %}
|
|
||||||
{% if params.S == '-1' %}
|
|
||||||
{% if printer.exclude_object.current_object %}
|
|
||||||
EXCLUDE_OBJECT_END NAME={printer.exclude_object.current_object}
|
|
||||||
{% endif %}
|
|
||||||
{% else %}
|
|
||||||
EXCLUDE_OBJECT_START NAME={params.S}
|
|
||||||
{% endif %}
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
{% if 'U' in params %}
|
|
||||||
EXCLUDE_OBJECT RESET=1 NAME={params.U}
|
|
||||||
{% endif %}
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -1,7 +0,0 @@
|
|||||||
not_empty
|
|
||||||
[also_not_empty]
|
|
||||||
#
|
|
||||||
;
|
|
||||||
;
|
|
||||||
#
|
|
||||||
option: value
|
|
||||||
@@ -1,39 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# Copyright (C) 2024 Dominik Willner <th33xitus@gmail.com> #
|
|
||||||
# #
|
|
||||||
# https://github.com/dw-0/simple-config-parser #
|
|
||||||
# #
|
|
||||||
# This file may be distributed under the terms of the GNU GPLv3 license #
|
|
||||||
# ======================================================================= #
|
|
||||||
|
|
||||||
from pathlib import Path
|
|
||||||
|
|
||||||
import pytest
|
|
||||||
|
|
||||||
from src.simple_config_parser.simple_config_parser import SimpleConfigParser
|
|
||||||
from tests.utils import load_testdata_from_file
|
|
||||||
|
|
||||||
BASE_DIR = Path(__file__).parent.joinpath("test_data")
|
|
||||||
MATCHING_TEST_DATA_PATH = BASE_DIR.joinpath("matching_data.txt")
|
|
||||||
NON_MATCHING_TEST_DATA_PATH = BASE_DIR.joinpath("non_matching_data.txt")
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
|
||||||
def parser():
|
|
||||||
return SimpleConfigParser()
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize("line", load_testdata_from_file(MATCHING_TEST_DATA_PATH))
|
|
||||||
def test_match_line_comment(parser, line):
|
|
||||||
"""Test that a line matches the definition of a line comment"""
|
|
||||||
assert (
|
|
||||||
parser._match_empty_line(line) is True
|
|
||||||
), f"Expected line '{line}' to match line comment definition!"
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize("line", load_testdata_from_file(NON_MATCHING_TEST_DATA_PATH))
|
|
||||||
def test_non_matching_line_comment(parser, line):
|
|
||||||
"""Test that a line does not match the definition of a line comment"""
|
|
||||||
assert (
|
|
||||||
parser._match_empty_line(line) is False
|
|
||||||
), f"Expected line '{line}' to not match line comment definition!"
|
|
||||||
@@ -1,28 +0,0 @@
|
|||||||
;[example_section]
|
|
||||||
#[example_section]
|
|
||||||
# [example_section]
|
|
||||||
; [example_section]
|
|
||||||
;[gcode_macro CANCEL_PRINT]
|
|
||||||
#[gcode_macro CANCEL_PRINT]
|
|
||||||
# [gcode_macro CANCEL_PRINT]
|
|
||||||
; [gcode_macro CANCEL_PRINT]
|
|
||||||
;[gcode_macro SET_PAUSE_NEXT_LAYER]
|
|
||||||
#[gcode_macro SET_PAUSE_NEXT_LAYER]
|
|
||||||
# [gcode_macro SET_PAUSE_NEXT_LAYER]
|
|
||||||
; [gcode_macro SET_PAUSE_NEXT_LAYER]
|
|
||||||
;[gcode_macro _TOOLHEAD_PARK_PAUSE_CANCEL]
|
|
||||||
#[gcode_macro _TOOLHEAD_PARK_PAUSE_CANCEL]
|
|
||||||
# [gcode_macro _TOOLHEAD_PARK_PAUSE_CANCEL]
|
|
||||||
; [gcode_macro _TOOLHEAD_PARK_PAUSE_CANCEL]
|
|
||||||
;[gcode_macro _TOOLHEAD_PARK_PAUSE_CANCEL]
|
|
||||||
#[gcode_macro _TOOLHEAD_PARK_PAUSE_CANCEL]
|
|
||||||
# [gcode_macro _TOOLHEAD_PARK_PAUSE_CANCEL]
|
|
||||||
; [gcode_macro _TOOLHEAD_PARK_PAUSE_CANCEL]
|
|
||||||
;[gcode_macro _TOOLHEAD_PARK_PAUSE_CANCEL]
|
|
||||||
#[gcode_macro _TOOLHEAD_PARK_PAUSE_CANCEL]
|
|
||||||
# [gcode_macro _TOOLHEAD_PARK_PAUSE_CANCEL]
|
|
||||||
; [gcode_macro _TOOLHEAD_PARK_PAUSE_CANCEL]
|
|
||||||
;[gcode_macro _TOOLHEAD_PARK_PAUSE_CANCEL]
|
|
||||||
#[gcode_macro _TOOLHEAD_PARK_PAUSE_CANCEL]
|
|
||||||
# [gcode_macro _TOOLHEAD_PARK_PAUSE_CANCEL]
|
|
||||||
; [gcode_macro _TOOLHEAD_PARK_PAUSE_CANCEL]
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
not_a_comment: nono
|
|
||||||
|
|
||||||
[also not a comment]
|
|
||||||
not_a_comment: ; comment
|
|
||||||
not_a_comment: # comment
|
|
||||||
@@ -1,39 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# Copyright (C) 2024 Dominik Willner <th33xitus@gmail.com> #
|
|
||||||
# #
|
|
||||||
# https://github.com/dw-0/simple-config-parser #
|
|
||||||
# #
|
|
||||||
# This file may be distributed under the terms of the GNU GPLv3 license #
|
|
||||||
# ======================================================================= #
|
|
||||||
|
|
||||||
from pathlib import Path
|
|
||||||
|
|
||||||
import pytest
|
|
||||||
|
|
||||||
from src.simple_config_parser.simple_config_parser import SimpleConfigParser
|
|
||||||
from tests.utils import load_testdata_from_file
|
|
||||||
|
|
||||||
BASE_DIR = Path(__file__).parent.joinpath("test_data")
|
|
||||||
MATCHING_TEST_DATA_PATH = BASE_DIR.joinpath("matching_data.txt")
|
|
||||||
NON_MATCHING_TEST_DATA_PATH = BASE_DIR.joinpath("non_matching_data.txt")
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
|
||||||
def parser():
|
|
||||||
return SimpleConfigParser()
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize("line", load_testdata_from_file(MATCHING_TEST_DATA_PATH))
|
|
||||||
def test_match_line_comment(parser, line):
|
|
||||||
"""Test that a line matches the definition of a line comment"""
|
|
||||||
assert (
|
|
||||||
parser._match_line_comment(line) is True
|
|
||||||
), f"Expected line '{line}' to match line comment definition!"
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize("line", load_testdata_from_file(NON_MATCHING_TEST_DATA_PATH))
|
|
||||||
def test_non_matching_line_comment(parser, line):
|
|
||||||
"""Test that a line does not match the definition of a line comment"""
|
|
||||||
assert (
|
|
||||||
parser._match_line_comment(line) is False
|
|
||||||
), f"Expected line '{line}' to not match line comment definition!"
|
|
||||||
@@ -1,461 +0,0 @@
|
|||||||
baud: 250000
|
|
||||||
minimum_cruise_ratio: 0.5
|
|
||||||
square_corner_velocity: 5.0
|
|
||||||
full_steps_per_rotation: 200
|
|
||||||
position_min: 0
|
|
||||||
homing_speed: 5.0
|
|
||||||
homing_retract_dist: 5.0
|
|
||||||
kinematics: cartesian
|
|
||||||
kinematics: delta
|
|
||||||
minimum_z_position: 0
|
|
||||||
speed: 50
|
|
||||||
horizontal_move_z: 5
|
|
||||||
kinematics: deltesian
|
|
||||||
minimum_z_position: 0
|
|
||||||
min_angle: 5
|
|
||||||
slow_ratio: 3
|
|
||||||
kinematics: corexy
|
|
||||||
kinematics: corexz
|
|
||||||
kinematics: hybrid_corexy
|
|
||||||
kinematics: hybrid_corexz
|
|
||||||
kinematics: polar
|
|
||||||
kinematics: rotary_delta
|
|
||||||
minimum_z_position: 0
|
|
||||||
speed: 50
|
|
||||||
horizontal_move_z: 5
|
|
||||||
kinematics: winch
|
|
||||||
kinematics: none
|
|
||||||
max_velocity: 1
|
|
||||||
max_accel: 1
|
|
||||||
instantaneous_corner_velocity: 1.000
|
|
||||||
max_extrude_only_distance: 50.0
|
|
||||||
pressure_advance: 0.0
|
|
||||||
pressure_advance_smooth_time: 0.040
|
|
||||||
max_power: 1.0
|
|
||||||
pullup_resistor: 4700
|
|
||||||
smooth_time: 1.0
|
|
||||||
max_delta: 2.0
|
|
||||||
pwm_cycle_time: 0.100
|
|
||||||
min_extrude_temp: 170
|
|
||||||
speed: 50
|
|
||||||
horizontal_move_z: 5
|
|
||||||
probe_count: 3, 3
|
|
||||||
round_probe_count: 5
|
|
||||||
fade_start: 1.0
|
|
||||||
fade_end: 0.0
|
|
||||||
split_delta_z: .025
|
|
||||||
move_check_distance: 5.0
|
|
||||||
mesh_pps: 2, 2
|
|
||||||
algorithm: lagrange
|
|
||||||
bicubic_tension: .2
|
|
||||||
x_adjust: 0
|
|
||||||
y_adjust: 0
|
|
||||||
z_adjust: 0
|
|
||||||
speed: 50
|
|
||||||
horizontal_move_z: 5
|
|
||||||
horizontal_move_z: 5
|
|
||||||
probe_height: 0
|
|
||||||
speed: 50
|
|
||||||
probe_speed: 5
|
|
||||||
speed: 50
|
|
||||||
horizontal_move_z: 5
|
|
||||||
screw_thread: CW-M3
|
|
||||||
speed: 50
|
|
||||||
horizontal_move_z: 5
|
|
||||||
retries: 0
|
|
||||||
retry_tolerance: 0
|
|
||||||
speed: 50
|
|
||||||
horizontal_move_z: 5
|
|
||||||
max_adjust: 4
|
|
||||||
retries: 0
|
|
||||||
retry_tolerance: 0
|
|
||||||
speed: 50.0
|
|
||||||
z_hop_speed: 15.0
|
|
||||||
move_to_previous: False
|
|
||||||
axes: xyz
|
|
||||||
endstop_align_zero: False
|
|
||||||
description: G-Code macro
|
|
||||||
initial_duration: 0.0
|
|
||||||
timeout: 600
|
|
||||||
enable_force_move: False
|
|
||||||
recover_velocity: 50.
|
|
||||||
retract_length: 0
|
|
||||||
retract_speed: 20
|
|
||||||
unretract_extra_length: 0
|
|
||||||
unretract_speed: 10
|
|
||||||
resolution: 1.0
|
|
||||||
default_type: echo
|
|
||||||
default_prefix: echo:
|
|
||||||
shaper_freq_x: 0
|
|
||||||
shaper_freq_y: 0
|
|
||||||
shaper_type: mzv
|
|
||||||
damping_ratio_x: 0.1
|
|
||||||
damping_ratio_y: 0.1
|
|
||||||
spi_speed: 5000000
|
|
||||||
axes_map: x, y, z
|
|
||||||
rate: 3200
|
|
||||||
spi_speed: 5000000
|
|
||||||
axes_map: x, y, z
|
|
||||||
i2c_speed: 400000
|
|
||||||
axes_map: x, y, z
|
|
||||||
min_freq: 5
|
|
||||||
max_freq: 133.33
|
|
||||||
accel_per_hz: 75
|
|
||||||
hz_per_sec: 1
|
|
||||||
mcu: mcu
|
|
||||||
deactivate_on_each_sample: True
|
|
||||||
x_offset: 0.0
|
|
||||||
y_offset: 0.0
|
|
||||||
speed: 5.0
|
|
||||||
samples: 1
|
|
||||||
sampleretract_dist: 2.0
|
|
||||||
samples_result: average
|
|
||||||
samples_tolerance: 0.100
|
|
||||||
samples_toleranceretries: 0
|
|
||||||
pin_move_time: 0.680
|
|
||||||
stow_on_each_sample: True
|
|
||||||
probe_with_touch_mode: False
|
|
||||||
pin_up_reports_not_triggered: True
|
|
||||||
pin_up_touch_modereports_triggered: True
|
|
||||||
recovery_time: 0.4
|
|
||||||
sensor_type: ldc1612
|
|
||||||
speed: 50
|
|
||||||
horizontal_move_z: 5
|
|
||||||
calibrate_start_x: 20
|
|
||||||
calibrate_end_x: 200
|
|
||||||
calibrate_y: 112.5
|
|
||||||
max_error: 120
|
|
||||||
hysteresis: 5
|
|
||||||
heating_gain: 2
|
|
||||||
extruder_heating_z: 50.
|
|
||||||
max_validation_temp: 60.
|
|
||||||
pullup_resistor: 4700
|
|
||||||
inlineresistor: 0
|
|
||||||
adc_voltage: 5.0
|
|
||||||
voltage_offset: 0
|
|
||||||
sensor_type: PT1000
|
|
||||||
pullup_resistor: 4700
|
|
||||||
spi_speed: 4000000
|
|
||||||
tc_type: K
|
|
||||||
tc_use_50Hz_filter: False
|
|
||||||
tc_averaging_count: 1
|
|
||||||
rtd_nominal_r: 100
|
|
||||||
rtd_referencer: 430
|
|
||||||
rtd_num_of_wires: 2
|
|
||||||
rtd_use_50Hz_filter: False
|
|
||||||
sensor_type: BME280
|
|
||||||
sensor_type: AHT10
|
|
||||||
sensor_type: temperature_mcu
|
|
||||||
sensor_mcu: mcu
|
|
||||||
sensor_type: temperature_host
|
|
||||||
sensor_type: DS18B20
|
|
||||||
sensor_type: temperature_combined
|
|
||||||
max_power: 1.0
|
|
||||||
shutdown_speed: 0
|
|
||||||
cycle_time: 0.010
|
|
||||||
hardware_pwm: False
|
|
||||||
kick_start_time: 0.100
|
|
||||||
off_below: 0.0
|
|
||||||
tachometer_ppr: 2
|
|
||||||
tachometer_poll_interval: 0.0015
|
|
||||||
heater: extruder
|
|
||||||
heater_temp: 50.0
|
|
||||||
fan_speed: 1.0
|
|
||||||
fan_speed: 1.0
|
|
||||||
pid_deriv_time: 2.0
|
|
||||||
target_temp: 40.0
|
|
||||||
max_speed: 1.0
|
|
||||||
min_speed: 0.3
|
|
||||||
cycle_time: 0.010
|
|
||||||
hardware_pwm: False
|
|
||||||
initial_RED: 0.0
|
|
||||||
initial_GREEN: 0.0
|
|
||||||
initial_BLUE: 0.0
|
|
||||||
initial_WHITE: 0.0
|
|
||||||
color_order: GRB
|
|
||||||
initial_RED: 0.0
|
|
||||||
initial_GREEN: 0.0
|
|
||||||
initial_BLUE: 0.0
|
|
||||||
initial_WHITE: 0.0
|
|
||||||
initial_RED: 0.0
|
|
||||||
initial_GREEN: 0.0
|
|
||||||
initial_BLUE: 0.0
|
|
||||||
i2c_address: 98
|
|
||||||
initial_RED: 0.0
|
|
||||||
initial_GREEN: 0.0
|
|
||||||
initial_BLUE: 0.0
|
|
||||||
initial_WHITE: 0.0
|
|
||||||
i2c_address: 98
|
|
||||||
color_order: RGBW
|
|
||||||
initial_RED: 0.0
|
|
||||||
initial_GREEN: 0.0
|
|
||||||
initial_BLUE: 0.0
|
|
||||||
initial_WHITE: 0.0
|
|
||||||
maximum_servo_angle: 180
|
|
||||||
minimum_pulse_width: 0.001
|
|
||||||
maximum_pulse_width: 0.002
|
|
||||||
pwm: False
|
|
||||||
cycle_time: 0.100
|
|
||||||
hardware_pwm: False
|
|
||||||
cycle_time: 0.100
|
|
||||||
hardware_pwm: False
|
|
||||||
cycle_time: 0.100
|
|
||||||
interpolate: True
|
|
||||||
senseresistor: 0.110
|
|
||||||
stealthchop_threshold: 0
|
|
||||||
driver_MSLUT0: 2863314260
|
|
||||||
driver_MSLUT1: 1251300522
|
|
||||||
driver_MSLUT2: 608774441
|
|
||||||
driver_MSLUT3: 269500962
|
|
||||||
driver_MSLUT4: 4227858431
|
|
||||||
driver_MSLUT5: 3048961917
|
|
||||||
driver_MSLUT6: 1227445590
|
|
||||||
driver_MSLUT7: 4211234
|
|
||||||
driver_W0: 2
|
|
||||||
driver_W1: 1
|
|
||||||
driver_W2: 1
|
|
||||||
driver_W3: 1
|
|
||||||
driver_X1: 128
|
|
||||||
driver_X2: 255
|
|
||||||
driver_X3: 255
|
|
||||||
driver_START_SIN: 0
|
|
||||||
driver_START_SIN90: 247
|
|
||||||
driver_IHOLDDELAY: 8
|
|
||||||
driver_TPOWERDOWN: 0
|
|
||||||
driver_TBL: 1
|
|
||||||
driver_TOFF: 4
|
|
||||||
driver_HEND: 7
|
|
||||||
driver_HSTRT: 0
|
|
||||||
driver_VHIGHFS: 0
|
|
||||||
driver_VHIGHCHM: 0
|
|
||||||
driver_PWM_AUTOSCALE: True
|
|
||||||
driver_PWM_FREQ: 1
|
|
||||||
driver_PWM_GRAD: 4
|
|
||||||
driver_PWM_AMPL: 128
|
|
||||||
driver_SGT: 0
|
|
||||||
driver_SEMIN: 0
|
|
||||||
driver_SEUP: 0
|
|
||||||
driver_SEMAX: 0
|
|
||||||
driver_SEDN: 0
|
|
||||||
driver_SEIMIN: 0
|
|
||||||
driver_SFILT: 0
|
|
||||||
interpolate: True
|
|
||||||
sense_resistor: 0.110
|
|
||||||
stealthchop_threshold: 0
|
|
||||||
driver_MULTISTEP_FILT: True
|
|
||||||
driver_IHOLDDELAY: 8
|
|
||||||
driver_TPOWERDOWN: 20
|
|
||||||
driver_TBL: 2
|
|
||||||
driver_TOFF: 3
|
|
||||||
driver_HEND: 0
|
|
||||||
driver_HSTRT: 5
|
|
||||||
driver_PWM_AUTOGRAD: True
|
|
||||||
driver_PWM_AUTOSCALE: True
|
|
||||||
driver_PWM_LIM: 12
|
|
||||||
driver_PWM_REG: 8
|
|
||||||
driver_PWM_FREQ: 1
|
|
||||||
driver_PWM_GRAD: 14
|
|
||||||
driver_PWM_OFS: 36
|
|
||||||
interpolate: True
|
|
||||||
sense_resistor: 0.110
|
|
||||||
stealthchop_threshold: 0
|
|
||||||
driver_MULTISTEP_FILT: True
|
|
||||||
driver_IHOLDDELAY: 8
|
|
||||||
driver_TPOWERDOWN: 20
|
|
||||||
driver_TBL: 2
|
|
||||||
driver_TOFF: 3
|
|
||||||
driver_HEND: 0
|
|
||||||
driver_HSTRT: 5
|
|
||||||
driver_PWM_AUTOGRAD: True
|
|
||||||
driver_PWM_AUTOSCALE: True
|
|
||||||
driver_PWM_LIM: 12
|
|
||||||
driver_PWM_REG: 8
|
|
||||||
driver_PWM_FREQ: 1
|
|
||||||
driver_PWM_GRAD: 14
|
|
||||||
driver_PWM_OFS: 36
|
|
||||||
driver_SGTHRS: 0
|
|
||||||
driver_SEMIN: 0
|
|
||||||
driver_SEUP: 0
|
|
||||||
driver_SEMAX: 0
|
|
||||||
driver_SEDN: 0
|
|
||||||
driver_SEIMIN: 0
|
|
||||||
spi_speed: 4000000
|
|
||||||
interpolate: True
|
|
||||||
idle_current_percent: 100
|
|
||||||
driver_TBL: 2
|
|
||||||
driver_RNDTF: 0
|
|
||||||
driver_HDEC: 0
|
|
||||||
driver_CHM: 0
|
|
||||||
driver_HEND: 3
|
|
||||||
driver_HSTRT: 3
|
|
||||||
driver_TOFF: 4
|
|
||||||
driver_SEIMIN: 0
|
|
||||||
driver_SEDN: 0
|
|
||||||
driver_SEMAX: 0
|
|
||||||
driver_SEUP: 0
|
|
||||||
driver_SEMIN: 0
|
|
||||||
driver_SFILT: 0
|
|
||||||
driver_SGT: 0
|
|
||||||
driver_SLPH: 0
|
|
||||||
driver_SLPL: 0
|
|
||||||
driver_DISS2G: 0
|
|
||||||
driver_TS2G: 3
|
|
||||||
interpolate: True
|
|
||||||
rref: 12000
|
|
||||||
stealthchop_threshold: 0
|
|
||||||
driver_MSLUT0: 2863314260
|
|
||||||
driver_MSLUT1: 1251300522
|
|
||||||
driver_MSLUT2: 608774441
|
|
||||||
driver_MSLUT3: 269500962
|
|
||||||
driver_MSLUT4: 4227858431
|
|
||||||
driver_MSLUT5: 3048961917
|
|
||||||
driver_MSLUT6: 1227445590
|
|
||||||
driver_MSLUT7: 4211234
|
|
||||||
driver_W0: 2
|
|
||||||
driver_W1: 1
|
|
||||||
driver_W2: 1
|
|
||||||
driver_W3: 1
|
|
||||||
driver_X1: 128
|
|
||||||
driver_X2: 255
|
|
||||||
driver_X3: 255
|
|
||||||
driver_START_SIN: 0
|
|
||||||
driver_START_SIN90: 247
|
|
||||||
driver_OFFSET_SIN90: 0
|
|
||||||
driver_MULTISTEP_FILT: True
|
|
||||||
driver_IHOLDDELAY: 6
|
|
||||||
driver_IRUNDELAY: 4
|
|
||||||
driver_TPOWERDOWN: 10
|
|
||||||
driver_TBL: 2
|
|
||||||
driver_TOFF: 3
|
|
||||||
driver_HEND: 2
|
|
||||||
driver_HSTRT: 5
|
|
||||||
driver_FD3: 0
|
|
||||||
driver_TPFD: 4
|
|
||||||
driver_CHM: 0
|
|
||||||
driver_VHIGHFS: 0
|
|
||||||
driver_VHIGHCHM: 0
|
|
||||||
driver_DISS2G: 0
|
|
||||||
driver_DISS2VS: 0
|
|
||||||
driver_PWM_AUTOSCALE: True
|
|
||||||
driver_PWM_AUTOGRAD: True
|
|
||||||
driver_PWM_FREQ: 0
|
|
||||||
driver_FREEWHEEL: 0
|
|
||||||
driver_PWM_GRAD: 0
|
|
||||||
driver_PWM_OFS: 29
|
|
||||||
driver_PWM_REG: 4
|
|
||||||
driver_PWM_LIM: 12
|
|
||||||
driver_SGT: 0
|
|
||||||
driver_SEMIN: 0
|
|
||||||
driver_SEUP: 0
|
|
||||||
driver_SEMAX: 0
|
|
||||||
driver_SEDN: 0
|
|
||||||
driver_SEIMIN: 0
|
|
||||||
driver_SFILT: 0
|
|
||||||
driver_SG4_ANGLE_OFFSET: 1
|
|
||||||
interpolate: True
|
|
||||||
sense_resistor: 0.075
|
|
||||||
stealthchop_threshold: 0
|
|
||||||
driver_MSLUT0: 2863314260
|
|
||||||
driver_MSLUT1: 1251300522
|
|
||||||
driver_MSLUT2: 608774441
|
|
||||||
driver_MSLUT3: 269500962
|
|
||||||
driver_MSLUT4: 4227858431
|
|
||||||
driver_MSLUT5: 3048961917
|
|
||||||
driver_MSLUT6: 1227445590
|
|
||||||
driver_MSLUT7: 4211234
|
|
||||||
driver_W0: 2
|
|
||||||
driver_W1: 1
|
|
||||||
driver_W2: 1
|
|
||||||
driver_W3: 1
|
|
||||||
driver_X1: 128
|
|
||||||
driver_X2: 255
|
|
||||||
driver_X3: 255
|
|
||||||
driver_START_SIN: 0
|
|
||||||
driver_START_SIN90: 247
|
|
||||||
driver_MULTISTEP_FILT: True
|
|
||||||
driver_IHOLDDELAY: 6
|
|
||||||
driver_TPOWERDOWN: 10
|
|
||||||
driver_TBL: 2
|
|
||||||
driver_TOFF: 3
|
|
||||||
driver_HEND: 2
|
|
||||||
driver_HSTRT: 5
|
|
||||||
driver_FD3: 0
|
|
||||||
driver_TPFD: 4
|
|
||||||
driver_CHM: 0
|
|
||||||
driver_VHIGHFS: 0
|
|
||||||
driver_VHIGHCHM: 0
|
|
||||||
driver_DISS2G: 0
|
|
||||||
driver_DISS2VS: 0
|
|
||||||
driver_PWM_AUTOSCALE: True
|
|
||||||
driver_PWM_AUTOGRAD: True
|
|
||||||
driver_PWM_FREQ: 0
|
|
||||||
driver_FREEWHEEL: 0
|
|
||||||
driver_PWM_GRAD: 0
|
|
||||||
driver_PWM_OFS: 30
|
|
||||||
driver_PWM_REG: 4
|
|
||||||
driver_PWM_LIM: 12
|
|
||||||
driver_SGT: 0
|
|
||||||
driver_SEMIN: 0
|
|
||||||
driver_SEUP: 0
|
|
||||||
driver_SEMAX: 0
|
|
||||||
driver_SEDN: 0
|
|
||||||
driver_SEIMIN: 0
|
|
||||||
driver_SFILT: 0
|
|
||||||
driver_DRVSTRENGTH: 0
|
|
||||||
driver_BBMCLKS: 4
|
|
||||||
driver_BBMTIME: 0
|
|
||||||
driver_FILT_ISENSE: 0
|
|
||||||
i2c_address: 96
|
|
||||||
analog_pullup_resistor: 4700
|
|
||||||
lcd_type: hd44780
|
|
||||||
hd44780_protocol_init: True
|
|
||||||
lcd_type: hd44780_spi
|
|
||||||
hd44780_protocol_init: True
|
|
||||||
lcd_type: st7920
|
|
||||||
lcd_type: emulated_st7920
|
|
||||||
lcd_type: uc1701
|
|
||||||
vcomh: 0
|
|
||||||
invert: False
|
|
||||||
x_offset: 0
|
|
||||||
type: disabled
|
|
||||||
type: list
|
|
||||||
type: command
|
|
||||||
type: input
|
|
||||||
pause_on_runout: True
|
|
||||||
event_delay: 3.0
|
|
||||||
pause_delay: 0.5
|
|
||||||
detection_length: 7.0
|
|
||||||
default_nominal_filament_diameter: 1.75
|
|
||||||
max_difference: 0.2
|
|
||||||
measurement_delay: 100
|
|
||||||
cal_dia1: 1.50
|
|
||||||
cal_dia2: 2.00
|
|
||||||
raw_dia1: 9500
|
|
||||||
raw_dia2: 10500
|
|
||||||
default_nominal_filament_diameter: 1.75
|
|
||||||
max_difference: 0.200
|
|
||||||
measurement_delay: 70
|
|
||||||
enable: False
|
|
||||||
measurement_interval: 10
|
|
||||||
logging: False
|
|
||||||
min_diameter: 1.0
|
|
||||||
use_current_dia_while_delay: False
|
|
||||||
sensor_type: hx711
|
|
||||||
gain: A-128
|
|
||||||
sample_rate: 80
|
|
||||||
sensor_type: hx717
|
|
||||||
gain: A-128
|
|
||||||
sample_rate: 320
|
|
||||||
sensor_type: ads1220
|
|
||||||
spi_speed: 512000
|
|
||||||
gain: 128
|
|
||||||
sample_rate: 660
|
|
||||||
smooth_time: 2.0
|
|
||||||
enable_pin: !gpio0_20
|
|
||||||
standstill_power_down: False
|
|
||||||
baud: 115200
|
|
||||||
feedrate_splice: 0.8
|
|
||||||
feedrate_normal: 1.0
|
|
||||||
auto_load_speed: 2
|
|
||||||
auto_cancel_variation: 0.1
|
|
||||||
sample_period: 0.000400
|
|
||||||
@@ -1,37 +0,0 @@
|
|||||||
[section]
|
|
||||||
[section with spaces]
|
|
||||||
[section with spaces and comments] ; comment 1
|
|
||||||
[section with spaces and comments] # comment 2
|
|
||||||
indented_option: value
|
|
||||||
option_with_no_value:
|
|
||||||
another_option_with_no_value:
|
|
||||||
indented_option_with_no_value:
|
|
||||||
# position_min: 0
|
|
||||||
# homing_speed: 5.0
|
|
||||||
|
|
||||||
### this is a comment
|
|
||||||
; this is also a comment
|
|
||||||
# [section]
|
|
||||||
# [section with spaces]
|
|
||||||
# [section with spaces and comments] ; comment 1
|
|
||||||
;[section]
|
|
||||||
;[section with spaces]
|
|
||||||
;[section with spaces and comments] ; comment 1
|
|
||||||
# commented_option: value
|
|
||||||
#commented_option: value
|
|
||||||
;commented_option: value
|
|
||||||
; commented_option: value
|
|
||||||
#
|
|
||||||
;
|
|
||||||
option_1 :: value
|
|
||||||
option_1:: value
|
|
||||||
option_1 ::value
|
|
||||||
option_2 == value
|
|
||||||
option_2== value
|
|
||||||
option_2 ==value
|
|
||||||
option_1 := value
|
|
||||||
option_1:= value
|
|
||||||
option_1 :=value
|
|
||||||
option_2 := value
|
|
||||||
option_2:= value
|
|
||||||
option_2 :=value
|
|
||||||
@@ -1,39 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# Copyright (C) 2024 Dominik Willner <th33xitus@gmail.com> #
|
|
||||||
# #
|
|
||||||
# https://github.com/dw-0/simple-config-parser #
|
|
||||||
# #
|
|
||||||
# This file may be distributed under the terms of the GNU GPLv3 license #
|
|
||||||
# ======================================================================= #
|
|
||||||
|
|
||||||
from pathlib import Path
|
|
||||||
|
|
||||||
import pytest
|
|
||||||
|
|
||||||
from src.simple_config_parser.simple_config_parser import SimpleConfigParser
|
|
||||||
from tests.utils import load_testdata_from_file
|
|
||||||
|
|
||||||
BASE_DIR = Path(__file__).parent.joinpath("test_data")
|
|
||||||
MATCHING_TEST_DATA_PATH = BASE_DIR.joinpath("matching_data.txt")
|
|
||||||
NON_MATCHING_TEST_DATA_PATH = BASE_DIR.joinpath("non_matching_data.txt")
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
|
||||||
def parser():
|
|
||||||
return SimpleConfigParser()
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize("line", load_testdata_from_file(MATCHING_TEST_DATA_PATH))
|
|
||||||
def test_match_option(parser, line):
|
|
||||||
"""Test that a line matches the definition of an option"""
|
|
||||||
assert (
|
|
||||||
parser._match_option(line) is True
|
|
||||||
), f"Expected line '{line}' to match option definition!"
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize("line", load_testdata_from_file(NON_MATCHING_TEST_DATA_PATH))
|
|
||||||
def test_non_matching_option(parser, line):
|
|
||||||
"""Test that a line does not match the definition of an option"""
|
|
||||||
assert (
|
|
||||||
parser._match_option(line) is False
|
|
||||||
), f"Expected line '{line}' to not match option definition!"
|
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
trusted_clients:
|
|
||||||
gcode:
|
|
||||||
cors_domains:
|
|
||||||
an_options_block_start_with_comment: ; this is a comment
|
|
||||||
an_options_block_start_with_comment: # this is a comment
|
|
||||||
options_block_start_with_comment:;this is a comment
|
|
||||||
options_block_start_with_comment :;this is a comment
|
|
||||||
options_block_start_with_comment:#this is a comment
|
|
||||||
options_block_start_with_comment :#this is a comment
|
|
||||||
parameter_temperature_(°C):
|
|
||||||
parameter_temperature_(°C)=
|
|
||||||
parameter_humidity_(%_RH):
|
|
||||||
parameter_humidity_(%_RH) :
|
|
||||||
parameter_spool_weight_(%):
|
|
||||||
parameter_spool_weight_(%) =
|
|
||||||
@@ -1,31 +0,0 @@
|
|||||||
type: jsonfile
|
|
||||||
path: /dev/shm/drying_box.json
|
|
||||||
baud: 250000
|
|
||||||
minimum_cruise_ratio: 0.5
|
|
||||||
square_corner_velocity: 5.0
|
|
||||||
full_steps_per_rotation: 200
|
|
||||||
position_min: 0
|
|
||||||
homing_speed: 5.0
|
|
||||||
# baud: 250000
|
|
||||||
# minimum_cruise_ratio: 0.5
|
|
||||||
# square_corner_velocity: 5.0
|
|
||||||
# full_steps_per_rotation: 200
|
|
||||||
# position_min: 0
|
|
||||||
# homing_speed: 5.0
|
|
||||||
|
|
||||||
### this is a comment
|
|
||||||
; this is also a comment
|
|
||||||
;
|
|
||||||
#
|
|
||||||
homing_speed::
|
|
||||||
homing_speed::
|
|
||||||
homing_speed ::
|
|
||||||
homing_speed ::
|
|
||||||
homing_speed==
|
|
||||||
homing_speed==
|
|
||||||
homing_speed ==
|
|
||||||
homing_speed ==
|
|
||||||
homing_speed :=
|
|
||||||
homing_speed :=
|
|
||||||
homing_speed =:
|
|
||||||
homing_speed =:
|
|
||||||
@@ -1,39 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# Copyright (C) 2024 Dominik Willner <th33xitus@gmail.com> #
|
|
||||||
# #
|
|
||||||
# https://github.com/dw-0/simple-config-parser #
|
|
||||||
# #
|
|
||||||
# This file may be distributed under the terms of the GNU GPLv3 license #
|
|
||||||
# ======================================================================= #
|
|
||||||
|
|
||||||
from pathlib import Path
|
|
||||||
|
|
||||||
import pytest
|
|
||||||
|
|
||||||
from src.simple_config_parser.simple_config_parser import SimpleConfigParser
|
|
||||||
from tests.utils import load_testdata_from_file
|
|
||||||
|
|
||||||
BASE_DIR = Path(__file__).parent.joinpath("test_data")
|
|
||||||
MATCHING_TEST_DATA_PATH = BASE_DIR.joinpath("matching_data.txt")
|
|
||||||
NON_MATCHING_TEST_DATA_PATH = BASE_DIR.joinpath("non_matching_data.txt")
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
|
||||||
def parser():
|
|
||||||
return SimpleConfigParser()
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize("line", load_testdata_from_file(MATCHING_TEST_DATA_PATH))
|
|
||||||
def test_match_options_block_start(parser, line):
|
|
||||||
"""Test that a line matches the definition of an options block start"""
|
|
||||||
assert (
|
|
||||||
parser._match_options_block_start(line) is True
|
|
||||||
), f"Expected line '{line}' to match options block start definition!"
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize("line", load_testdata_from_file(NON_MATCHING_TEST_DATA_PATH))
|
|
||||||
def test_non_matching_options_block_start(parser, line):
|
|
||||||
"""Test that a line does not match the definition of an options block start"""
|
|
||||||
assert (
|
|
||||||
parser._match_options_block_start(line) is False
|
|
||||||
), f"Expected line '{line}' to not match options block start definition!"
|
|
||||||
@@ -1,127 +0,0 @@
|
|||||||
[example_section]
|
|
||||||
[gcode_macro CANCEL_PRINT]
|
|
||||||
[gcode_macro SET_PAUSE_NEXT_LAYER]
|
|
||||||
[gcode_macro _TOOLHEAD_PARK_PAUSE_CANCEL]
|
|
||||||
[update_manager moonraker-obico]
|
|
||||||
[include moonraker_obico_macros.cfg]
|
|
||||||
[include moonraker-obico-update.cfg]
|
|
||||||
[example_section two]
|
|
||||||
[valid_content]
|
|
||||||
[valid content]
|
|
||||||
[content123]
|
|
||||||
[a]
|
|
||||||
[valid_content] # comment
|
|
||||||
[something];comment
|
|
||||||
[mcu]
|
|
||||||
[printer]
|
|
||||||
[printer]
|
|
||||||
[stepper_x]
|
|
||||||
[stepper_y]
|
|
||||||
[stepper_z]
|
|
||||||
[printer]
|
|
||||||
[stepper_a]
|
|
||||||
[stepper_b]
|
|
||||||
[stepper_c]
|
|
||||||
[delta_calibrate]
|
|
||||||
[printer]
|
|
||||||
[stepper_left]
|
|
||||||
[stepper_right]
|
|
||||||
[stepper_bed]
|
|
||||||
[stepper_arm]
|
|
||||||
[delta_calibrate]
|
|
||||||
[extruder]
|
|
||||||
[heater_bed]
|
|
||||||
[bed_mesh]
|
|
||||||
[bed_tilt]
|
|
||||||
[bed_screws]
|
|
||||||
[screws_tilt_adjust]
|
|
||||||
[z_tilt]
|
|
||||||
[quad_gantry_level]
|
|
||||||
[skew_correction]
|
|
||||||
[z_thermal_adjust]
|
|
||||||
[safe_z_home]
|
|
||||||
[homing_override]
|
|
||||||
[endstop_phase stepper_z]
|
|
||||||
[gcode_macro my_cmd]
|
|
||||||
[delayed_gcode my_delayed_gcode]
|
|
||||||
[save_variables]
|
|
||||||
[idle_timeout]
|
|
||||||
[virtual_sdcard]
|
|
||||||
[sdcard_loop]
|
|
||||||
[force_move]
|
|
||||||
[pause_resume]
|
|
||||||
[firmware_retraction]
|
|
||||||
[gcode_arcs]
|
|
||||||
[respond]
|
|
||||||
[exclude_object]
|
|
||||||
[input_shaper]
|
|
||||||
[adxl345]
|
|
||||||
[lis2dw]
|
|
||||||
[mpu9250 my_accelerometer]
|
|
||||||
[resonance_tester]
|
|
||||||
[board_pins my_aliases]
|
|
||||||
[duplicate_pin_override]
|
|
||||||
[probe]
|
|
||||||
[bltouch]
|
|
||||||
[smart_effector]
|
|
||||||
[probe_eddy_current my_eddy_probe]
|
|
||||||
[axis_twist_compensation]
|
|
||||||
[stepper_z1]
|
|
||||||
[extruder1]
|
|
||||||
[dual_carriage]
|
|
||||||
[extruder_stepper my_extra_stepper]
|
|
||||||
[manual_stepper my_stepper]
|
|
||||||
[verify_heater heater_config_name]
|
|
||||||
[homing_heaters]
|
|
||||||
[thermistor my_thermistor]
|
|
||||||
[adc_temperature my_sensor]
|
|
||||||
[heater_generic my_generic_heater]
|
|
||||||
[temperature_sensor my_sensor]
|
|
||||||
[temperature_probe my_probe]
|
|
||||||
[fan]
|
|
||||||
[heater_fan heatbreak_cooling_fan]
|
|
||||||
[controller_fan my_controller_fan]
|
|
||||||
[temperature_fan my_temp_fan]
|
|
||||||
[fan_generic extruder_partfan]
|
|
||||||
[led my_led]
|
|
||||||
[neopixel my_neopixel]
|
|
||||||
[dotstar my_dotstar]
|
|
||||||
[pca9533 my_pca9533]
|
|
||||||
[pca9632 my_pca9632]
|
|
||||||
[servo my_servo]
|
|
||||||
[gcode_button my_gcode_button]
|
|
||||||
[output_pin my_pin]
|
|
||||||
[pwm_tool my_tool]
|
|
||||||
[pwm_cycle_time my_pin]
|
|
||||||
[static_digital_output my_output_pins]
|
|
||||||
[multi_pin my_multi_pin]
|
|
||||||
[tmc2130 stepper_x]
|
|
||||||
[tmc2208 stepper_x]
|
|
||||||
[tmc2209 stepper_x]
|
|
||||||
[tmc2660 stepper_x]
|
|
||||||
[tmc2240 stepper_x]
|
|
||||||
[tmc5160 stepper_x]
|
|
||||||
[ad5206 my_digipot]
|
|
||||||
[mcp4451 my_digipot]
|
|
||||||
[mcp4728 my_dac]
|
|
||||||
[mcp4018 my_digipot]
|
|
||||||
[display]
|
|
||||||
[display_data my_group_name my_data_name]
|
|
||||||
[display_template my_template_name]
|
|
||||||
[display_glyph my_display_glyph]
|
|
||||||
[menu __some_list __some_name]
|
|
||||||
[menu some_name]
|
|
||||||
[menu some_list]
|
|
||||||
[menu some_list some_command]
|
|
||||||
[menu some_list some_input]
|
|
||||||
[filament_switch_sensor my_sensor]
|
|
||||||
[filament_motion_sensor my_sensor]
|
|
||||||
[tsl1401cl_filament_width_sensor]
|
|
||||||
[hall_filament_width_sensor]
|
|
||||||
[load_cell]
|
|
||||||
[sx1509 my_sx1509]
|
|
||||||
[samd_sercom my_sercom]
|
|
||||||
[adc_scaled my_name]
|
|
||||||
[replicape]
|
|
||||||
[palette2]
|
|
||||||
[angle my_angle_sensor]
|
|
||||||
@@ -1,19 +0,0 @@
|
|||||||
section: invalid
|
|
||||||
not_a_valid_section
|
|
||||||
[missing_square_bracket
|
|
||||||
missing_square_bracket]
|
|
||||||
[]
|
|
||||||
[ ]
|
|
||||||
[indented_section]
|
|
||||||
[indented_section] # comment
|
|
||||||
[indented_section] ; comment
|
|
||||||
;[commented_section]
|
|
||||||
#[another_commented_section]
|
|
||||||
; [commented_section]
|
|
||||||
# [another_commented_section]
|
|
||||||
this_is_an_option: 123
|
|
||||||
this_is_an_indented_option: 123
|
|
||||||
this_is_an_option_block_start:
|
|
||||||
|
|
||||||
#
|
|
||||||
;
|
|
||||||
@@ -1,39 +0,0 @@
|
|||||||
# ======================================================================= #
|
|
||||||
# Copyright (C) 2024 Dominik Willner <th33xitus@gmail.com> #
|
|
||||||
# #
|
|
||||||
# https://github.com/dw-0/simple-config-parser #
|
|
||||||
# #
|
|
||||||
# This file may be distributed under the terms of the GNU GPLv3 license #
|
|
||||||
# ======================================================================= #
|
|
||||||
|
|
||||||
from pathlib import Path
|
|
||||||
|
|
||||||
import pytest
|
|
||||||
|
|
||||||
from src.simple_config_parser.simple_config_parser import SimpleConfigParser
|
|
||||||
from tests.utils import load_testdata_from_file
|
|
||||||
|
|
||||||
BASE_DIR = Path(__file__).parent.joinpath("test_data")
|
|
||||||
MATCHING_TEST_DATA_PATH = BASE_DIR.joinpath("matching_data.txt")
|
|
||||||
NON_MATCHING_TEST_DATA_PATH = BASE_DIR.joinpath("non_matching_data.txt")
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
|
||||||
def parser():
|
|
||||||
return SimpleConfigParser()
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize("line", load_testdata_from_file(MATCHING_TEST_DATA_PATH))
|
|
||||||
def test_match_section(parser, line):
|
|
||||||
"""Test that a line matches the definition of a section"""
|
|
||||||
assert (
|
|
||||||
parser._match_section(line) is True
|
|
||||||
), f"Expected line '{line}' to match section definition!"
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize("line", load_testdata_from_file(NON_MATCHING_TEST_DATA_PATH))
|
|
||||||
def test_non_matching_section(parser, line):
|
|
||||||
"""Test that a line does not match the definition of a section"""
|
|
||||||
assert (
|
|
||||||
parser._match_section(line) is False
|
|
||||||
), f"Expected line '{line}' to not match section definition!"
|
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user