From c1d02ab5a2594d945e3f7aed18a1c18f296d65e2 Mon Sep 17 00:00:00 2001 From: zsdc Date: Thu, 19 Jan 2023 20:18:42 +0200 Subject: image: T4516: Added system image tools This commit adds the whole set of system image tools written from the scratch in Python that allows performing all the operations on images: * check information * perform installation and deletion * versions management Also, it contains a new service that will update the GRUB menu and keep tracking its version in the future. WARNING: The commit contains non-reversible changes. Because of boot menu changes, it will not be possible to manage images from older VyOS versions after an update. (cherry picked from commit 8f94262e8fa2477700c50303ea6e2c6ddad72adb) --- op-mode-definitions/add-system-image.xml.in | 62 --------- op-mode-definitions/system-image.xml.in | 189 ++++++++++++++++++++++++++++ 2 files changed, 189 insertions(+), 62 deletions(-) delete mode 100644 op-mode-definitions/add-system-image.xml.in create mode 100644 op-mode-definitions/system-image.xml.in (limited to 'op-mode-definitions') diff --git a/op-mode-definitions/add-system-image.xml.in b/op-mode-definitions/add-system-image.xml.in deleted file mode 100644 index 67d8aa3b4..000000000 --- a/op-mode-definitions/add-system-image.xml.in +++ /dev/null @@ -1,62 +0,0 @@ - - - - - - - Add item to a system facility - - - - - Add a new image to the system - - /path/to/vyos-image.iso "http://example.com/vyos-image.iso" - - - sudo ${vyatta_sbindir}/install-image --url "${4}" - - - - Download image via specified VRF - - vrf name - - - sudo ${vyatta_sbindir}/install-image --url "${4}" --vrf "${6}" - - - - Username for authentication - - - - - Password to use with authentication - - sudo ${vyatta_sbindir}/install-image --url "${4}" --vrf "${6}" --username "${8}" --password "${10}" - - - - - - - - Username for authentication - - - - - Password to use with authentication - - sudo ${vyatta_sbindir}/install-image --url "${4}" --username "${6}" --password "${8}" - - - - - - - - - - diff --git a/op-mode-definitions/system-image.xml.in b/op-mode-definitions/system-image.xml.in new file mode 100644 index 000000000..57aeb7bb4 --- /dev/null +++ b/op-mode-definitions/system-image.xml.in @@ -0,0 +1,189 @@ + + + + + Add an object + + + + + Add item to a system facility + + + + + Add a new image to the system + + /path/to/vyos-image.iso "http://example.com/vyos-image.iso" + + + sudo ${vyos_op_scripts_dir}/image_installer.py --action add --image_path "${4}" + + + + Download image via specified VRF + + vrf name + + + sudo ${vyos_op_scripts_dir}/image_installer.py --action add --image_path "${4}" --vrf "${6}" + + + + Username for authentication + + + + + Password to use with authentication + + sudo ${vyos_op_scripts_dir}/image_installer.py --action add --image_path "${4}" --vrf "${6}" --username "${8}" --password "${10}" + + + + + + + + Username for authentication + + + + + Password to use with authentication + + sudo ${vyos_op_scripts_dir}/image_installer.py --action add --image_path "${4}" --username "${6}" --password "${8}" + + + + + + + + + + + + Install a new system + + + + + Set system operational parameters + + + + + Set system image parameters + + + + + Set default image to boot. + + + + + sudo ${vyos_op_scripts_dir}/image_manager.py --action set --image_name "${5}" + + + + + + + + + + Install a new system + + + + + Install new system image to hard drive + + sudo ${vyos_op_scripts_dir}/image_installer.py --action install + + + + + + Delete an object + + + + + Delete system objects + + + + + Remove an installed image from the system + + + + + sudo ${vyos_op_scripts_dir}/image_manager.py --action delete --image_name "${4}" + + + + + + + + Rename an object + + + + + Rename a system object + + + + + System image to rename + + + + + + + + A new name for an image + + sudo ${vyos_op_scripts_dir}/image_manager.py --action rename --image_name "${4}" --image_new_name "${6}" + + + + + + + + + + Rename an object + + + + + Show system information + + + + + Show installed VyOS images + + sudo ${vyos_op_scripts_dir}/image_info.py show_images_summary + + + + Show details about installed VyOS images + + sudo ${vyos_op_scripts_dir}/image_info.py show_images_details + + + + + + + + -- cgit v1.2.3 From e812d494d16a29783254fb1edc0894e054184dcc Mon Sep 17 00:00:00 2001 From: John Estabrook Date: Tue, 31 Oct 2023 12:53:52 -0500 Subject: image: T4516: restore select entry to set/delete image (cherry picked from commit 9ffa3e82d951756696367578dd5e82ef0f690065) --- op-mode-definitions/system-image.xml.in | 12 ++++++++++ python/vyos/utils/io.py | 19 +++++++++++++++ src/op_mode/image_manager.py | 42 ++++++++++++++++++++++++--------- 3 files changed, 62 insertions(+), 11 deletions(-) (limited to 'op-mode-definitions') diff --git a/op-mode-definitions/system-image.xml.in b/op-mode-definitions/system-image.xml.in index 57aeb7bb4..463b985d6 100644 --- a/op-mode-definitions/system-image.xml.in +++ b/op-mode-definitions/system-image.xml.in @@ -77,6 +77,12 @@ Set system image parameters + + + Set default image to boot. + + sudo ${vyos_op_scripts_dir}/image_manager.py --action set + Set default image to boot. @@ -115,6 +121,12 @@ Delete system objects + + + Remove an installed image from the system + + sudo ${vyos_op_scripts_dir}/image_manager.py --action delete + Remove an installed image from the system diff --git a/python/vyos/utils/io.py b/python/vyos/utils/io.py index 8790cbaac..e34a1ba32 100644 --- a/python/vyos/utils/io.py +++ b/python/vyos/utils/io.py @@ -72,3 +72,22 @@ def is_dumb_terminal(): """Check if the current TTY is dumb, so that we can disable advanced terminal features.""" import os return os.getenv('TERM') in ['vt100', 'dumb'] + +def select_entry(l: list, list_msg: str = '', prompt_msg: str = '') -> str: + """Select an entry from a list + + Args: + l (list): a list of entries + list_msg (str): a message to print before listing the entries + prompt_msg (str): a message to print as prompt for selection + + Returns: + str: a selected entry + """ + en = list(enumerate(l, 1)) + print(list_msg) + for i, e in en: + print(f'\t{i}: {e}') + select = ask_input(prompt_msg, numeric_only=True, + valid_responses=range(1, len(l)+1)) + return next(filter(lambda x: x[0] == select, en))[1] diff --git a/src/op_mode/image_manager.py b/src/op_mode/image_manager.py index 55fd5c07d..de53c4cf0 100755 --- a/src/op_mode/image_manager.py +++ b/src/op_mode/image_manager.py @@ -21,23 +21,39 @@ from argparse import ArgumentParser, Namespace from pathlib import Path from shutil import rmtree from sys import exit +from typing import Optional from vyos.system import disk, grub, image, compat -from vyos.utils.io import ask_yes_no +from vyos.utils.io import ask_yes_no, select_entry + +SET_IMAGE_LIST_MSG: str = 'The following images are available:' +SET_IMAGE_PROMPT_MSG: str = 'Select an image to set as default:' +DELETE_IMAGE_LIST_MSG: str = 'The following images are installed:' +DELETE_IMAGE_PROMPT_MSG: str = 'Select an image to delete:' +MSG_DELETE_IMAGE_RUNNING: str = 'Currently running image cannot be deleted; reboot into another image first' +MSG_DELETE_IMAGE_DEFAULT: str = 'Default image cannot be deleted; set another image as default first' @compat.grub_cfg_update -def delete_image(image_name: str) -> None: +def delete_image(image_name: Optional[str] = None, + prompt: bool = True) -> None: """Remove installed image files and boot entry Args: image_name (str): a name of image to delete """ + available_images: list[str] = grub.version_list() + if image_name is None: + if not prompt: + exit('An image name is required for delete action') + else: + image_name = select_entry(available_images, + DELETE_IMAGE_LIST_MSG, + DELETE_IMAGE_PROMPT_MSG) if image_name == image.get_running_image(): - exit('Currently running image cannot be deleted') + exit(MSG_DELETE_IMAGE_RUNNING) if image_name == image.get_default_image(): - exit('Default image cannot be deleted') - available_images: list[str] = grub.version_list() + exit(MSG_DELETE_IMAGE_DEFAULT) if image_name not in available_images: exit(f'The image "{image_name}" cannot be found') presistence_storage: str = disk.find_persistence() @@ -59,15 +75,23 @@ def delete_image(image_name: str) -> None: @compat.grub_cfg_update -def set_image(image_name: str) -> None: +def set_image(image_name: Optional[str] = None, + prompt: bool = True) -> None: """Set default boot image Args: image_name (str): an image name """ + available_images: list[str] = grub.version_list() + if image_name is None: + if not prompt: + exit('An image name is required for set action') + else: + image_name = select_entry(available_images, + SET_IMAGE_LIST_MSG, + SET_IMAGE_PROMPT_MSG) if image_name == image.get_default_image(): exit(f'The image "{image_name}" already configured as default') - available_images: list[str] = grub.version_list() if image_name not in available_images: exit(f'The image "{image_name}" cannot be found') presistence_storage: str = disk.find_persistence() @@ -154,10 +178,6 @@ def parse_arguments() -> Namespace: parser.add_argument('--image_new_name', help='a new name for image') args: Namespace = parser.parse_args() # Validate arguments - if args.action == 'delete' and not args.image_name: - exit('An image name is required for delete action') - if args.action == 'set' and not args.image_name: - exit('An image name is required for set action') if args.action == 'rename' and (not args.image_name or not args.image_new_name): exit('Both old and new image names are required for rename action') -- cgit v1.2.3 From bbbe388b437b03427e5f80a0f955b6f2759ea946 Mon Sep 17 00:00:00 2001 From: John Estabrook Date: Mon, 27 Nov 2023 21:13:22 -0600 Subject: image-tools: T5751: normalize args using hyphen instead of underscore (cherry picked from commit bb578a1cab177e8cee6e4d02144d21387ba13a93) --- op-mode-definitions/system-image.xml.in | 14 +++++++------- src/op_mode/image_installer.py | 2 +- src/op_mode/image_manager.py | 4 ++-- 3 files changed, 10 insertions(+), 10 deletions(-) (limited to 'op-mode-definitions') diff --git a/op-mode-definitions/system-image.xml.in b/op-mode-definitions/system-image.xml.in index 463b985d6..c131087be 100644 --- a/op-mode-definitions/system-image.xml.in +++ b/op-mode-definitions/system-image.xml.in @@ -17,7 +17,7 @@ /path/to/vyos-image.iso "http://example.com/vyos-image.iso" - sudo ${vyos_op_scripts_dir}/image_installer.py --action add --image_path "${4}" + sudo ${vyos_op_scripts_dir}/image_installer.py --action add --image-path "${4}" @@ -26,7 +26,7 @@ vrf name - sudo ${vyos_op_scripts_dir}/image_installer.py --action add --image_path "${4}" --vrf "${6}" + sudo ${vyos_op_scripts_dir}/image_installer.py --action add --image-path "${4}" --vrf "${6}" @@ -37,7 +37,7 @@ Password to use with authentication - sudo ${vyos_op_scripts_dir}/image_installer.py --action add --image_path "${4}" --vrf "${6}" --username "${8}" --password "${10}" + sudo ${vyos_op_scripts_dir}/image_installer.py --action add --image-path "${4}" --vrf "${6}" --username "${8}" --password "${10}" @@ -52,7 +52,7 @@ Password to use with authentication - sudo ${vyos_op_scripts_dir}/image_installer.py --action add --image_path "${4}" --username "${6}" --password "${8}" + sudo ${vyos_op_scripts_dir}/image_installer.py --action add --image-path "${4}" --username "${6}" --password "${8}" @@ -90,7 +90,7 @@ - sudo ${vyos_op_scripts_dir}/image_manager.py --action set --image_name "${5}" + sudo ${vyos_op_scripts_dir}/image_manager.py --action set --image-name "${5}" @@ -134,7 +134,7 @@ - sudo ${vyos_op_scripts_dir}/image_manager.py --action delete --image_name "${4}" + sudo ${vyos_op_scripts_dir}/image_manager.py --action delete --image-name "${4}" @@ -162,7 +162,7 @@ A new name for an image - sudo ${vyos_op_scripts_dir}/image_manager.py --action rename --image_name "${4}" --image_new_name "${6}" + sudo ${vyos_op_scripts_dir}/image_manager.py --action rename --image-name "${4}" --image-new-name "${6}" diff --git a/src/op_mode/image_installer.py b/src/op_mode/image_installer.py index aa4cf301b..380393a34 100755 --- a/src/op_mode/image_installer.py +++ b/src/op_mode/image_installer.py @@ -728,7 +728,7 @@ def parse_arguments() -> Namespace: required=True, help='action to perform with an image') parser.add_argument( - '--image_path', + '--image-path', help='a path (HTTP or local file) to an image that needs to be installed' ) # parser.add_argument('--image_new_name', help='a new name for image') diff --git a/src/op_mode/image_manager.py b/src/op_mode/image_manager.py index e4b2f4833..dcbd290c9 100755 --- a/src/op_mode/image_manager.py +++ b/src/op_mode/image_manager.py @@ -172,10 +172,10 @@ def parse_arguments() -> Namespace: required=True, help='action to perform with an image') parser.add_argument( - '--image_name', + '--image-name', help= 'a name of an image to add, delete, install, rename, or set as default') - parser.add_argument('--image_new_name', help='a new name for image') + parser.add_argument('--image-new-name', help='a new name for image') args: Namespace = parser.parse_args() # Validate arguments if args.action == 'rename' and (not args.image_name or -- cgit v1.2.3