Compare commits
	
		
			363 Commits
		
	
	
		
			529ecfdcf0
			...
			v0.9.62
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 2c3a3cdf52 | |||
| 0b583bfdaf | |||
| 7c07fbefff | |||
| 202dc3ed59 | |||
| 8667f4d03b | |||
| 4c2d99f8a6 | |||
| bcff6650d0 | |||
| a11206253d | |||
| 7f57c6dbf7 | |||
| 6865979e08 | |||
| 5126bc3272 | |||
| 765f0ef13d | |||
| fe258d9d56 | |||
| 93d89a2414 | |||
| a49f3810d3 | |||
| 45ad3b9a17 | |||
| 07623fad1a | |||
| 8331b7ecd8 | |||
| 94d4ee5b9b | |||
| e773211293 | |||
| 32c36c866b | |||
| dc4e56db4b | |||
| e45b3108a2 | |||
| 118237a53b | |||
| 9805681f94 | |||
| 6c9abb2abe | |||
| a1122c6e71 | |||
| 3696f81597 | |||
| 5ca0d903b6 | |||
| 6ddbde763e | |||
| 626424b74a | |||
| b3d99827f5 | |||
| c9ceb3159b | |||
| 6525a2568b | |||
| 09a005d3d7 | |||
| 96defebd0b | |||
| d00b8aa6cd | |||
| e9aa545e9b | |||
| fb0fcc0597 | |||
| 3009f24910 | |||
| 5ae836f1c5 | |||
| 70ba364f1d | |||
| eda1b95d5f | |||
| 3bd93563e6 | |||
| 1f8561d59a | |||
| a2efc83953 | |||
| f2d2537e1c | |||
| 1093ca6264 | |||
| 15ff729f83 | |||
| 29584e5636 | |||
| f4e8449356 | |||
| 388f6556c0 | |||
| ec79acf061 | |||
| 6c7be492b8 | |||
| 00586074cf | |||
| f4eef30770 | |||
| 8565cf26b3 | |||
| 0ecf219910 | |||
| 0f4edc54d1 | |||
| ca91be51e1 | |||
| e29d0e89eb | |||
| 14d29f2986 | |||
| bc88d764b0 | |||
| a3c31564ca | |||
| b07396c39a | |||
| 71139fa66d | |||
| e6f9e6e0e8 | |||
| 1ea4800212 | |||
| 9c14d84bfc | |||
| d8f346abdd | |||
| 2ee52e44d3 | |||
| 3c742a827b | |||
| aeb238f43c | |||
| 671a907236 | |||
| e945fd8590 | |||
| a49510ecc8 | |||
| 6d7730ab52 | |||
| 8135426973 | |||
| 20d436a745 | |||
| 28f6819726 | |||
| 35c07f0384 | |||
| 6127387be4 | |||
| 343d66875b | |||
| 92feeefd26 | |||
| 38d63d9837 | |||
| 095bcb2373 | |||
| 91e450f399 | |||
| 79eb994a5e | |||
| d65f512897 | |||
| 8af7189dd0 | |||
| ea7a4b2b85 | |||
| 59f97ebbfb | |||
| 072337f1f0 | |||
| c3bc55eff8 | |||
| 6c58d52fa1 | |||
| 666e02fbfd | |||
| 46dde055c4 | |||
| ef437c3dbf | |||
| bd2208e8f6 | |||
| 62d5ff11df | |||
| 0019881cfa | |||
| d46133802b | |||
| fcadde057e | |||
| 2608f38d64 | |||
| 89f05ced3f | |||
| 729481126c | |||
| 41eccb9c7d | |||
| e550e39a5a | |||
| dff156b2b0 | |||
| 1c4fb80d1f | |||
| ec7beb08cc | |||
| 3a180193ee | |||
| e26ff8a975 | |||
| 6276414702 | |||
| a34d64a71b | |||
| 71297e0179 | |||
| 45c9909428 | |||
| 7268592c87 | |||
| 726d0a562b | |||
| 39e1fc50ed | |||
| 7a3870fc44 | |||
| bffab7a5a1 | |||
| 6cbaeb5dc8 | |||
| 58ce133c8d | |||
| 43feb33caa | |||
| 3a5d8c61da | |||
| 1e0b502250 | |||
| fe17d28385 | |||
| 8aaac33056 | |||
| cc7952c232 | |||
| 16915ed507 | |||
| 2c624ceb2c | |||
| da85480488 | |||
| 47b0704555 | |||
| 7c49967586 | |||
| e3f96ac87e | |||
| 4df70cf086 | |||
| f1df1cfe93 | |||
| 5942aa50fc | |||
| 096bcdfd75 | |||
| 239c392892 | |||
| 172d0a86e4 | |||
| d8e57a26c5 | |||
| 9b499b9f48 | |||
| 881550b610 | |||
| 2a21d48128 | |||
| 8d0f26ff7a | |||
| bcabd7d079 | |||
| 05a316cdd6 | |||
| 4b36753f27 | |||
| 171f6ac9ed | |||
| 645b525ad7 | |||
| ec559aec0d | |||
| 71ffd5a191 | |||
| 2739c27299 | |||
| 56129a3636 | |||
| 932b3c55a3 | |||
| 92e2ff7449 | |||
| d8d3feee22 | |||
| b1357cafdb | |||
| f8cdcb30ba | |||
| 51ad2058ed | |||
| c401a1f655 | |||
| 7a40c7a55b | |||
| 8027a6efdc | |||
| 3801fcc07b | |||
| c741900baf | |||
| 464f0e0356 | |||
| cea8832f90 | |||
| 5807351405 | |||
| d6ca74376a | |||
| 413100a147 | |||
| 4d698be34b | |||
| 53aed0a735 | |||
| ea709f573f | |||
| 1142454934 | |||
| bbfad340a1 | |||
| c73939e1c5 | |||
| 25fe45dd28 | |||
| 58d57d7037 | |||
| 00d2c67c41 | |||
| 67131de4f6 | |||
| abc23ebb18 | |||
| 9f122e916f | |||
| 3ce4d90693 | |||
| 6ccd19e636 | |||
| d8689e6eaa | |||
| bc49b5eca2 | |||
| 8470dfaa29 | |||
| f164d898c1 | |||
| 195f31501c | |||
| a8899a1d66 | |||
| 817dffcf30 | |||
| eda2a57a73 | |||
| 135d28e60b | |||
| e7d7378bae | |||
| 799c3e8d5d | |||
| d0ec24f690 | |||
| 6e9fcd38a3 | |||
| f51f9fc4c8 | |||
| a6dcffc737 | |||
| 364c190106 | |||
| ea19af6494 | |||
| 7069d3237c | |||
| 619c3f7ff5 | |||
| 8a75bb3011 | |||
| a817c3e678 | |||
| 0cc3f2deab | |||
| 21b4bbe51a | |||
| 87ec31c023 | |||
| 0d857d5ab8 | |||
| 006f40f195 | |||
| 5f193a6134 | |||
| 78faa90139 | |||
| 23b1501f40 | |||
| 66bfad3109 | |||
| eee5c25d6f | |||
| ff4fc18a60 | |||
| ac885b855a | |||
| b9c30baf80 | |||
| 9b12cc0236 | |||
| c41664d2da | |||
| 3779bc960e | |||
| 5c620262e9 | |||
| 6b88fbd1e3 | |||
| a50c8e6a4d | |||
| 7d6e4353f1 | |||
| bf30b31db6 | |||
| 70bd601dc1 | |||
| 2e7b9b28b3 | |||
| 12eef58d42 | |||
| f2e6892fd2 | |||
| 91fb9e1241 | |||
| d87bea4159 | |||
| 3a6f442856 | |||
| dfca998adf | |||
| 55f397a347 | |||
| dfebb2d3e5 | |||
| e88147db4a | |||
| b8204d89ac | |||
| fe73dfbdc9 | |||
| 8f906c1f81 | |||
| 2d9fb9688d | |||
| fb84685c2a | |||
| 032ba44d9c | |||
| b7761877e7 | |||
| 1fe07640b3 | |||
| b8d843ebe4 | |||
| 95d983ddff | |||
| 4c5da1b6a8 | |||
| be6b1e02e3 | |||
| ec2a72ed4b | |||
| b06e327add | |||
| d1f32d2b9c | |||
| 3f78ca1cc9 | |||
| e866335918 | |||
| 221494ed1b | |||
| f13cc04b89 | |||
| 4ed537ee3b | |||
| 95e01f38d5 | |||
| 3122d73bf5 | |||
| 7ed8ef179c | |||
| caead02b2a | |||
| 87bc5f93e6 | |||
| 203893559e | |||
| 2c51bb0705 | |||
| 46d3daf686 | |||
| e9d05aa24e | |||
| d2c18d7b46 | |||
| 6ce28c43af | |||
| 87cda72ca9 | |||
| 8f71a6d2f6 | |||
| c45f8f5bd5 | |||
| 24de0f4189 | |||
| 3690a2c1e0 | |||
| 50d8aa0586 | |||
| db6e65712d | |||
| cf8e16543c | |||
| 1a4fcdcc2d | |||
| 9a71db0800 | |||
| 6ee4c55071 | |||
| c27359c4bf | |||
| 46078932c3 | |||
| c89699bc6f | |||
| 1b9507e4f5 | |||
| 3db7ac48f4 | |||
| 1830ec6465 | |||
| bdb9db8375 | |||
| c61d7bc313 | |||
| c0f7ba0125 | |||
| 761032b321 | |||
| 3566e13e79 | |||
| 6b324029cf | |||
| 13eeabf44b | |||
| d86768d3d0 | |||
| a167757600 | |||
| a95d9680ac | |||
| 63962f10ba | |||
| a7a681d92a | |||
| da9248cfa2 | |||
| aa035a61a7 | |||
| 7c8ba56561 | |||
| bba73980de | |||
| 32b3af697c | |||
| 7c122ac921 | |||
| 0dbf139706 | |||
| c909beaf6d | |||
| 2da49297d2 | |||
| 0ff9a6b8c4 | |||
| 28377178d2 | |||
| e06b114c48 | |||
| 0058f19d88 | |||
| 056cf3740d | |||
| 58f174b87b | |||
| 37b98fd54f | |||
| f83a345bfe | |||
| ce06e4d81b | |||
| 23977b04fc | |||
| bb1cca522f | |||
| 9a4dce4e4c | |||
| f6f6f07488 | |||
| 142c999ce8 | |||
| 1de069298c | |||
| 55221b3d97 | |||
| 0d72798814 | |||
| 3638efc77e | |||
| c2c888d684 | |||
| febef2e406 | |||
| 2a4f38e933 | |||
| 3b805cdc34 | |||
| 06f0f7ed91 | |||
| fd040ab45a | |||
| e23e2dd9bf | |||
| ee4266f8ca | |||
| 0f02c5eaef | |||
| 075abec5fe | |||
| 3a1cbf8d01 | |||
| a438a4155a | |||
| 65df807b09 | |||
| d0f3e9e285 | |||
| adc8a5a3bc | |||
| df277edf1c | |||
| 772807deb3 | |||
| 58db537093 | |||
| e71a6c90bf | |||
| a8e9a56924 | |||
| f3fb492633 | |||
| e962743e51 | |||
| 46f1d761f6 | |||
| be954c1625 | |||
| fb46f5f9e9 | |||
| 694b8e85a0 | |||
| eb321497ee | |||
| 5b81e59481 | |||
| a4c0e0befd | |||
| a18cef5f25 | |||
| f6c5aa9992 | |||
| ffa3dd5edb | |||
| afb0359c20 | |||
| afdf254297 | |||
| 42e776fac1 | |||
| dae67a1b7b | |||
| b86f8c1e09 | 
							
								
								
									
										4
									
								
								.flake8
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								.flake8
									
									
									
									
									
								
							| @@ -3,9 +3,7 @@ | ||||
| #   * W503 (line break before binary operator): Black moves these to new lines | ||||
| #   * E501 (line too long): Long lines are a fact of life in comment blocks; Black handles active instances of this | ||||
| #   * E203 (whitespace before ':'): Black recommends this as disabled | ||||
| #   * F403 (import * used; unable to detect undefined names): We use a wildcard for helpers | ||||
| #   * F405 (possibly undefined name): We use a wildcard for helpers | ||||
| ignore = W503, E501, F403, F405 | ||||
| ignore = W503, E501 | ||||
| extend-ignore = E203 | ||||
| # We exclude the Debian, migrations, and provisioner examples | ||||
| exclude = debian,api-daemon/migrations/versions,api-daemon/provisioner/examples,node-daemon/monitoring | ||||
|   | ||||
							
								
								
									
										30
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										30
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -1,35 +1,5 @@ | ||||
| ## PVC Changelog | ||||
|  | ||||
| ###### [v0.9.64](https://github.com/parallelvirtualcluster/pvc/releases/tag/v0.9.64) | ||||
|  | ||||
|   **Breaking Change [CLI]**: The CLI client root commands have been reorganized. The following commands have changed: | ||||
|  | ||||
|    * `pvc cluster` -> `pvc connection` (all subcommands) | ||||
|    * `pvc task` -> `pvc cluster` (all subcommands) | ||||
|    * `pvc maintenance` -> `pvc cluster maintenance` | ||||
|    * `pvc status` -> `pvc cluster status` | ||||
|  | ||||
| Ensure you have updated to the latest version of the PVC Ansible repository before deploying this version or using PVC Ansible oneshot playbooks for management. | ||||
|  | ||||
|   **Breaking Change [CLI]**: The `--restart` option for VM configuration changes now has an explicit `--no-restart` to disable restarting, or a prompt if neither is specified; `--unsafe` no longer bypasses this prompt which was a bug. Applies to most `vm <cmd> set` commands like `vm vcpu set`, `vm memory set`, etc. All instances also feature restart confirmation afterwards, which, if `--restart` is provided, will prompt for confirmation unless `--yes` or `--unsafe` is specified. | ||||
|  | ||||
|   **Breaking Change [CLI]**: The `--long` option previously on some `info` commands no longer exists; use `-f long`/`--format long` instead. | ||||
|  | ||||
|   * [CLI] Significantly refactors the CLI client code for consistency and cleanliness | ||||
|   * [CLI] Implements `-f`/`--format` options for all `list` and `info` commands in a consistent way | ||||
|   * [CLI] Changes the behaviour of VM modification options with "--restart" to provide a "--no-restart"; defaults to a prompt if neither is specified and ignores the "--unsafe" global entirely | ||||
|   * [API] Fixes several bugs in the 3-debootstrap.py provisioner example script | ||||
|   * [Node] Fixes some bugs around VM shutdown on node flush | ||||
|   * [Documentation] Adds mentions of Ganeti and Harvester | ||||
|  | ||||
| ###### [v0.9.63](https://github.com/parallelvirtualcluster/pvc/releases/tag/v0.9.63) | ||||
|  | ||||
|   * Mentions Ganeti in the docs | ||||
|   * Increases API timeout back to 2s | ||||
|   * Adds .update-* configs to dpkg plugin | ||||
|   * Adds full/nearfull OSD warnings | ||||
|   * Improves size value handling for volumes | ||||
|  | ||||
| ###### [v0.9.62](https://github.com/parallelvirtualcluster/pvc/releases/tag/v0.9.62) | ||||
|  | ||||
|   * [all] Adds an enhanced health checking, monitoring, and reporting system for nodes and clusters | ||||
|   | ||||
| @@ -9,7 +9,7 @@ | ||||
|  | ||||
| ## What is PVC? | ||||
|  | ||||
| PVC is a Linux KVM-based hyperconverged infrastructure (HCI) virtualization cluster solution that is fully Free Software, scalable, redundant, self-healing, self-managing, and designed for administrator simplicity. It is an alternative to other HCI solutions such as Ganeti, Harvester, Nutanix, and VMWare, as well as to other common virtualization stacks such as ProxMox and OpenStack. | ||||
| PVC is a Linux KVM-based hyperconverged infrastructure (HCI) virtualization cluster solution that is fully Free Software, scalable, redundant, self-healing, self-managing, and designed for administrator simplicity. It is an alternative to other HCI solutions such as Harvester, Nutanix, and VMWare, as well as to other common virtualization stacks such as ProxMox and OpenStack. | ||||
|  | ||||
| PVC is a complete HCI solution, built from well-known and well-trusted Free Software tools, to assist an administrator in creating and managing a cluster of servers to run virtual machines, as well as self-managing several important aspects including storage failover, node failure and recovery, virtual machine failure and recovery, and network plumbing. It is designed to act consistently, reliably, and unobtrusively, letting the administrator concentrate on more important things. | ||||
|  | ||||
|   | ||||
| @@ -441,7 +441,7 @@ class VMBuilderScript(VMBuilder): | ||||
|  | ||||
|         # The directory we mounted things on earlier during prepare(); this could very well | ||||
|         # be exposed as a module-level variable if you so choose | ||||
|         temp_dir = "/tmp/target" | ||||
|         temporary_directory = "/tmp/target" | ||||
|  | ||||
|         # Use these convenient aliases for later (avoiding lots of "self.vm_data" everywhere) | ||||
|         vm_name = self.vm_name | ||||
| @@ -469,8 +469,6 @@ class VMBuilderScript(VMBuilder): | ||||
|                 "grub-pc", | ||||
|                 "cloud-init", | ||||
|                 "python3-cffi-backend", | ||||
|                 "acpid", | ||||
|                 "acpi-support-base", | ||||
|                 "wget", | ||||
|             ] | ||||
|  | ||||
| @@ -484,17 +482,17 @@ class VMBuilderScript(VMBuilder): | ||||
|  | ||||
|         # Perform a debootstrap installation | ||||
|         print( | ||||
|             f"Installing system with debootstrap: debootstrap --include={','.join(deb_packages)} {deb_release} {temp_dir} {deb_mirror}" | ||||
|             f"Installing system with debootstrap: debootstrap --include={','.join(deb_packages)} {deb_release} {temporary_directory} {deb_mirror}" | ||||
|         ) | ||||
|         os.system( | ||||
|             f"debootstrap --include={','.join(deb_packages)} {deb_release} {temp_dir} {deb_mirror}" | ||||
|             f"debootstrap --include={','.join(deb_packages)} {deb_release} {temporary_directory} {deb_mirror}" | ||||
|         ) | ||||
|  | ||||
|         # Bind mount the devfs so we can grub-install later | ||||
|         os.system("mount --bind /dev {}/dev".format(temp_dir)) | ||||
|         os.system("mount --bind /dev {}/dev".format(temporary_directory)) | ||||
|  | ||||
|         # Create an fstab entry for each volume | ||||
|         fstab_file = "{}/etc/fstab".format(temp_dir) | ||||
|         fstab_file = "{}/etc/fstab".format(temporary_directory) | ||||
|         # The volume ID starts at zero and increments by one for each volume in the fixed-order | ||||
|         # volume list. This lets us work around the insanity of Libvirt IDs not matching guest IDs, | ||||
|         # while still letting us have some semblance of control here without enforcing things | ||||
| @@ -539,13 +537,13 @@ class VMBuilderScript(VMBuilder): | ||||
|             volume_id += 1 | ||||
|  | ||||
|         # Write the hostname; you could also take an FQDN argument for this as an example | ||||
|         hostname_file = "{}/etc/hostname".format(temp_dir) | ||||
|         hostname_file = "{}/etc/hostname".format(temporary_directory) | ||||
|         with open(hostname_file, "w") as fh: | ||||
|             fh.write("{}".format(vm_name)) | ||||
|  | ||||
|         # Fix the cloud-init.target since it's broken by default in Debian 11 | ||||
|         cloudinit_target_file = "{}/etc/systemd/system/cloud-init.target".format( | ||||
|             temp_dir | ||||
|             temporary_directory | ||||
|         ) | ||||
|         with open(cloudinit_target_file, "w") as fh: | ||||
|             # We lose our indent on these raw blocks to preserve the apperance of the files | ||||
| @@ -559,7 +557,7 @@ After=multi-user.target | ||||
|             fh.write(data) | ||||
|  | ||||
|         # Write the cloud-init configuration | ||||
|         ci_cfg_file = "{}/etc/cloud/cloud.cfg".format(temp_dir) | ||||
|         ci_cfg_file = "{}/etc/cloud/cloud.cfg".format(temporary_directory) | ||||
|         with open(ci_cfg_file, "w") as fh: | ||||
|             fh.write( | ||||
|                 """ | ||||
| @@ -620,15 +618,15 @@ After=multi-user.target | ||||
|                      - arches: [default] | ||||
|                        failsafe: | ||||
|                          primary: {deb_mirror} | ||||
|                 """.format( | ||||
|                     deb_mirror=deb_mirror | ||||
|                 ) | ||||
|             ) | ||||
|                 """ | ||||
|             ).format(deb_mirror=deb_mirror) | ||||
|  | ||||
|         # Due to device ordering within the Libvirt XML configuration, the first Ethernet interface | ||||
|         # will always be on PCI bus ID 2, hence the name "ens2". | ||||
|         # Write a DHCP stanza for ens2 | ||||
|         ens2_network_file = "{}/etc/network/interfaces.d/ens2".format(temp_dir) | ||||
|         ens2_network_file = "{}/etc/network/interfaces.d/ens2".format( | ||||
|             temporary_directory | ||||
|         ) | ||||
|         with open(ens2_network_file, "w") as fh: | ||||
|             data = """auto ens2 | ||||
| iface ens2 inet dhcp | ||||
| @@ -636,7 +634,7 @@ iface ens2 inet dhcp | ||||
|             fh.write(data) | ||||
|  | ||||
|         # Write the DHCP config for ens2 | ||||
|         dhclient_file = "{}/etc/dhcp/dhclient.conf".format(temp_dir) | ||||
|         dhclient_file = "{}/etc/dhcp/dhclient.conf".format(temporary_directory) | ||||
|         with open(dhclient_file, "w") as fh: | ||||
|             # We can use fstrings too, since PVC will always have Python 3.6+, though | ||||
|             # using format() might be preferable for clarity in some situations | ||||
| @@ -656,7 +654,7 @@ interface "ens2" {{ | ||||
|             fh.write(data) | ||||
|  | ||||
|         # Write the GRUB configuration | ||||
|         grubcfg_file = "{}/etc/default/grub".format(temp_dir) | ||||
|         grubcfg_file = "{}/etc/default/grub".format(temporary_directory) | ||||
|         with open(grubcfg_file, "w") as fh: | ||||
|             data = """# Written by the PVC provisioner | ||||
| GRUB_DEFAULT=0 | ||||
| @@ -673,7 +671,7 @@ GRUB_DISABLE_LINUX_UUID=false | ||||
|             fh.write(data) | ||||
|  | ||||
|         # Do some tasks inside the chroot using the provided context manager | ||||
|         with chroot(temp_dir): | ||||
|         with chroot(temporary_directory): | ||||
|             # Install and update GRUB | ||||
|             os.system( | ||||
|                 "grub-install --force /dev/rbd/{}/{}_{}".format( | ||||
| @@ -706,17 +704,16 @@ GRUB_DISABLE_LINUX_UUID=false | ||||
|         """ | ||||
|  | ||||
|         # Run any imports first | ||||
|         import os | ||||
|         from pvcapid.vmbuilder import open_zk | ||||
|         from pvcapid.Daemon import config | ||||
|         import daemon_lib.common as pvc_common | ||||
|         import daemon_lib.ceph as pvc_ceph | ||||
|  | ||||
|         # Set the temp_dir we used in the prepare() and install() steps | ||||
|         # Set the tempdir we used in the prepare() and install() steps | ||||
|         temp_dir = "/tmp/target" | ||||
|  | ||||
|         # Unmount the bound devfs | ||||
|         os.system("umount {}/dev".format(temp_dir)) | ||||
|         os.system("umount {}/dev".format(temporary_directory)) | ||||
|  | ||||
|         # Use this construct for reversing the list, as the normal reverse() messes with the list | ||||
|         for volume in list(reversed(self.vm_data["volumes"])): | ||||
|   | ||||
| @@ -27,7 +27,7 @@ from ssl import SSLContext, TLSVersion | ||||
| from distutils.util import strtobool as dustrtobool | ||||
|  | ||||
| # Daemon version | ||||
| version = "0.9.64" | ||||
| version = "0.9.62" | ||||
|  | ||||
| # API version | ||||
| API_VERSION = 1.0 | ||||
|   | ||||
| @@ -1,33 +0,0 @@ | ||||
| #!/usr/bin/env python3 | ||||
|  | ||||
| # pvc.py - PVC client command-line interface (stub testing interface) | ||||
| # Part of the Parallel Virtual Cluster (PVC) system | ||||
| # | ||||
| #    Copyright (C) 2018-2022 Joshua M. Boniface <joshua@boniface.me> | ||||
| # | ||||
| #    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, version 3. | ||||
| # | ||||
| #    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/>. | ||||
| # | ||||
| ############################################################################### | ||||
|  | ||||
| import pvc.pvc | ||||
|  | ||||
|  | ||||
| # | ||||
| # Main entry point | ||||
| # | ||||
| def main(): | ||||
|     return pvc.pvc.cli(obj={}) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
| @@ -1,201 +0,0 @@ | ||||
| #!/usr/bin/env python3 | ||||
|  | ||||
| # common.py - PVC CLI client function library, Common functions | ||||
| # Part of the Parallel Virtual Cluster (PVC) system | ||||
| # | ||||
| #    Copyright (C) 2018-2022 Joshua M. Boniface <joshua@boniface.me> | ||||
| # | ||||
| #    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, version 3. | ||||
| # | ||||
| #    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/>. | ||||
| # | ||||
| ############################################################################### | ||||
|  | ||||
| import os | ||||
| import math | ||||
| import time | ||||
| import requests | ||||
| import click | ||||
| from urllib3 import disable_warnings | ||||
|  | ||||
|  | ||||
| def format_bytes(size_bytes): | ||||
|     byte_unit_matrix = { | ||||
|         "B": 1, | ||||
|         "K": 1024, | ||||
|         "M": 1024 * 1024, | ||||
|         "G": 1024 * 1024 * 1024, | ||||
|         "T": 1024 * 1024 * 1024 * 1024, | ||||
|         "P": 1024 * 1024 * 1024 * 1024 * 1024, | ||||
|     } | ||||
|     human_bytes = "0B" | ||||
|     for unit in sorted(byte_unit_matrix, key=byte_unit_matrix.get): | ||||
|         formatted_bytes = int(math.ceil(size_bytes / byte_unit_matrix[unit])) | ||||
|         if formatted_bytes < 10000: | ||||
|             human_bytes = "{}{}".format(formatted_bytes, unit) | ||||
|             break | ||||
|     return human_bytes | ||||
|  | ||||
|  | ||||
| def format_metric(integer): | ||||
|     integer_unit_matrix = { | ||||
|         "": 1, | ||||
|         "K": 1000, | ||||
|         "M": 1000 * 1000, | ||||
|         "B": 1000 * 1000 * 1000, | ||||
|         "T": 1000 * 1000 * 1000 * 1000, | ||||
|         "Q": 1000 * 1000 * 1000 * 1000 * 1000, | ||||
|     } | ||||
|     human_integer = "0" | ||||
|     for unit in sorted(integer_unit_matrix, key=integer_unit_matrix.get): | ||||
|         formatted_integer = int(math.ceil(integer / integer_unit_matrix[unit])) | ||||
|         if formatted_integer < 10000: | ||||
|             human_integer = "{}{}".format(formatted_integer, unit) | ||||
|             break | ||||
|     return human_integer | ||||
|  | ||||
|  | ||||
| class UploadProgressBar(object): | ||||
|     def __init__(self, filename, end_message="", end_nl=True): | ||||
|         file_size = os.path.getsize(filename) | ||||
|         file_size_human = format_bytes(file_size) | ||||
|         click.echo("Uploading file (total size {})...".format(file_size_human)) | ||||
|  | ||||
|         self.length = file_size | ||||
|         self.time_last = int(round(time.time() * 1000)) - 1000 | ||||
|         self.bytes_last = 0 | ||||
|         self.bytes_diff = 0 | ||||
|         self.is_end = False | ||||
|  | ||||
|         self.end_message = end_message | ||||
|         self.end_nl = end_nl | ||||
|         if not self.end_nl: | ||||
|             self.end_suffix = " " | ||||
|         else: | ||||
|             self.end_suffix = "" | ||||
|  | ||||
|         self.bar = click.progressbar(length=self.length, show_eta=True) | ||||
|  | ||||
|     def update(self, monitor): | ||||
|         bytes_cur = monitor.bytes_read | ||||
|         self.bytes_diff += bytes_cur - self.bytes_last | ||||
|         if self.bytes_last == bytes_cur: | ||||
|             self.is_end = True | ||||
|         self.bytes_last = bytes_cur | ||||
|  | ||||
|         time_cur = int(round(time.time() * 1000)) | ||||
|         if (time_cur - 1000) > self.time_last: | ||||
|             self.time_last = time_cur | ||||
|             self.bar.update(self.bytes_diff) | ||||
|             self.bytes_diff = 0 | ||||
|  | ||||
|         if self.is_end: | ||||
|             self.bar.update(self.bytes_diff) | ||||
|             self.bytes_diff = 0 | ||||
|             click.echo() | ||||
|             click.echo() | ||||
|             if self.end_message: | ||||
|                 click.echo(self.end_message + self.end_suffix, nl=self.end_nl) | ||||
|  | ||||
|  | ||||
| class ErrorResponse(requests.Response): | ||||
|     def __init__(self, json_data, status_code): | ||||
|         self.json_data = json_data | ||||
|         self.status_code = status_code | ||||
|  | ||||
|     def json(self): | ||||
|         return self.json_data | ||||
|  | ||||
|  | ||||
| def call_api( | ||||
|     config, | ||||
|     operation, | ||||
|     request_uri, | ||||
|     headers={}, | ||||
|     params=None, | ||||
|     data=None, | ||||
|     files=None, | ||||
| ): | ||||
|     # Set the connect timeout to 2 seconds but extremely long (48 hour) data timeout | ||||
|     timeout = (2.05, 172800) | ||||
|  | ||||
|     # Craft the URI | ||||
|     uri = "{}://{}{}{}".format( | ||||
|         config["api_scheme"], config["api_host"], config["api_prefix"], request_uri | ||||
|     ) | ||||
|  | ||||
|     # Craft the authentication header if required | ||||
|     if config["api_key"]: | ||||
|         headers["X-Api-Key"] = config["api_key"] | ||||
|  | ||||
|     # Determine the request type and hit the API | ||||
|     disable_warnings() | ||||
|     try: | ||||
|         if operation == "get": | ||||
|             response = requests.get( | ||||
|                 uri, | ||||
|                 timeout=timeout, | ||||
|                 headers=headers, | ||||
|                 params=params, | ||||
|                 data=data, | ||||
|                 verify=config["verify_ssl"], | ||||
|             ) | ||||
|         if operation == "post": | ||||
|             response = requests.post( | ||||
|                 uri, | ||||
|                 timeout=timeout, | ||||
|                 headers=headers, | ||||
|                 params=params, | ||||
|                 data=data, | ||||
|                 files=files, | ||||
|                 verify=config["verify_ssl"], | ||||
|             ) | ||||
|         if operation == "put": | ||||
|             response = requests.put( | ||||
|                 uri, | ||||
|                 timeout=timeout, | ||||
|                 headers=headers, | ||||
|                 params=params, | ||||
|                 data=data, | ||||
|                 files=files, | ||||
|                 verify=config["verify_ssl"], | ||||
|             ) | ||||
|         if operation == "patch": | ||||
|             response = requests.patch( | ||||
|                 uri, | ||||
|                 timeout=timeout, | ||||
|                 headers=headers, | ||||
|                 params=params, | ||||
|                 data=data, | ||||
|                 verify=config["verify_ssl"], | ||||
|             ) | ||||
|         if operation == "delete": | ||||
|             response = requests.delete( | ||||
|                 uri, | ||||
|                 timeout=timeout, | ||||
|                 headers=headers, | ||||
|                 params=params, | ||||
|                 data=data, | ||||
|                 verify=config["verify_ssl"], | ||||
|             ) | ||||
|     except Exception as e: | ||||
|         message = "Failed to connect to the API: {}".format(e) | ||||
|         response = ErrorResponse({"message": message}, 500) | ||||
|  | ||||
|     # Display debug output | ||||
|     if config["debug"]: | ||||
|         click.echo("API endpoint: {}".format(uri), err=True) | ||||
|         click.echo("Response code: {}".format(response.status_code), err=True) | ||||
|         click.echo("Response headers: {}".format(response.headers), err=True) | ||||
|         click.echo(err=True) | ||||
|  | ||||
|     # Return the response object | ||||
|     return response | ||||
| @@ -1,20 +0,0 @@ | ||||
| from setuptools import setup | ||||
|  | ||||
| setup( | ||||
|     name="pvc", | ||||
|     version="0.9.63", | ||||
|     packages=["pvc", "pvc.lib"], | ||||
|     install_requires=[ | ||||
|         "Click", | ||||
|         "PyYAML", | ||||
|         "lxml", | ||||
|         "colorama", | ||||
|         "requests", | ||||
|         "requests-toolbelt", | ||||
|     ], | ||||
|     entry_points={ | ||||
|         "console_scripts": [ | ||||
|             "pvc = pvc.pvc:cli", | ||||
|         ], | ||||
|     }, | ||||
| ) | ||||
| @@ -1,33 +0,0 @@ | ||||
| #!/usr/bin/env python3 | ||||
|  | ||||
| # pvc.py - PVC client command-line interface (stub testing interface) | ||||
| # Part of the Parallel Virtual Cluster (PVC) system | ||||
| # | ||||
| #    Copyright (C) 2018-2022 Joshua M. Boniface <joshua@boniface.me> | ||||
| # | ||||
| #    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, version 3. | ||||
| # | ||||
| #    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/>. | ||||
| # | ||||
| ############################################################################### | ||||
|  | ||||
| from pvc.cli.cli import cli | ||||
|  | ||||
|  | ||||
| # | ||||
| # Main entry point | ||||
| # | ||||
| def main(): | ||||
|     return cli(obj={}) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,734 +0,0 @@ | ||||
| #!/usr/bin/env python3 | ||||
|  | ||||
| # formatters.py - PVC Click CLI output formatters library | ||||
| # Part of the Parallel Virtual Cluster (PVC) system | ||||
| # | ||||
| #    Copyright (C) 2018-2023 Joshua M. Boniface <joshua@boniface.me> | ||||
| # | ||||
| #    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, version 3. | ||||
| # | ||||
| #    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/>. | ||||
| # | ||||
| ############################################################################### | ||||
|  | ||||
| from pvc.lib.node import format_info as node_format_info | ||||
| from pvc.lib.node import format_list as node_format_list | ||||
| from pvc.lib.vm import format_vm_tags as vm_format_tags | ||||
| from pvc.lib.vm import format_vm_vcpus as vm_format_vcpus | ||||
| from pvc.lib.vm import format_vm_memory as vm_format_memory | ||||
| from pvc.lib.vm import format_vm_networks as vm_format_networks | ||||
| from pvc.lib.vm import format_vm_volumes as vm_format_volumes | ||||
| from pvc.lib.vm import format_info as vm_format_info | ||||
| from pvc.lib.vm import format_list as vm_format_list | ||||
| from pvc.lib.network import format_info as network_format_info | ||||
| from pvc.lib.network import format_list as network_format_list | ||||
| from pvc.lib.network import format_list_dhcp as network_format_dhcp_list | ||||
| from pvc.lib.network import format_list_acl as network_format_acl_list | ||||
| from pvc.lib.network import format_list_sriov_pf as network_format_sriov_pf_list | ||||
| from pvc.lib.network import format_info_sriov_vf as network_format_sriov_vf_info | ||||
| from pvc.lib.network import format_list_sriov_vf as network_format_sriov_vf_list | ||||
| from pvc.lib.storage import format_raw_output as storage_format_raw | ||||
| from pvc.lib.storage import format_info_benchmark as storage_format_benchmark_info | ||||
| from pvc.lib.storage import format_list_benchmark as storage_format_benchmark_list | ||||
| from pvc.lib.storage import format_list_osd as storage_format_osd_list | ||||
| from pvc.lib.storage import format_list_pool as storage_format_pool_list | ||||
| from pvc.lib.storage import format_list_volume as storage_format_volume_list | ||||
| from pvc.lib.storage import format_list_snapshot as storage_format_snapshot_list | ||||
| from pvc.lib.provisioner import format_list_template as provisioner_format_template_list | ||||
| from pvc.lib.provisioner import format_list_userdata as provisioner_format_userdata_list | ||||
| from pvc.lib.provisioner import format_list_script as provisioner_format_script_list | ||||
| from pvc.lib.provisioner import format_list_ova as provisioner_format_ova_list | ||||
| from pvc.lib.provisioner import format_list_profile as provisioner_format_profile_list | ||||
| from pvc.lib.provisioner import format_list_task as provisioner_format_task_status | ||||
|  | ||||
|  | ||||
| # Define colour values for use in formatters | ||||
| ansii = { | ||||
|     "red": "\033[91m", | ||||
|     "blue": "\033[94m", | ||||
|     "cyan": "\033[96m", | ||||
|     "green": "\033[92m", | ||||
|     "yellow": "\033[93m", | ||||
|     "purple": "\033[95m", | ||||
|     "bold": "\033[1m", | ||||
|     "end": "\033[0m", | ||||
| } | ||||
|  | ||||
|  | ||||
| def cli_cluster_status_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the full output of cli_cluster_status | ||||
|     """ | ||||
|  | ||||
|     # Normalize data to local variables | ||||
|     health = data.get("cluster_health", {}).get("health", -1) | ||||
|     messages = data.get("cluster_health", {}).get("messages", None) | ||||
|     maintenance = data.get("maintenance", "N/A") | ||||
|     primary_node = data.get("primary_node", "N/A") | ||||
|     pvc_version = data.get("pvc_version", "N/A") | ||||
|     upstream_ip = data.get("upstream_ip", "N/A") | ||||
|     total_nodes = data.get("nodes", {}).get("total", 0) | ||||
|     total_vms = data.get("vms", {}).get("total", 0) | ||||
|     total_networks = data.get("networks", 0) | ||||
|     total_osds = data.get("osds", {}).get("total", 0) | ||||
|     total_pools = data.get("pools", 0) | ||||
|     total_volumes = data.get("volumes", 0) | ||||
|     total_snapshots = data.get("snapshots", 0) | ||||
|  | ||||
|     if maintenance == "true" or health == -1: | ||||
|         health_colour = ansii["blue"] | ||||
|     elif health > 90: | ||||
|         health_colour = ansii["green"] | ||||
|     elif health > 50: | ||||
|         health_colour = ansii["yellow"] | ||||
|     else: | ||||
|         health_colour = ansii["red"] | ||||
|  | ||||
|     output = list() | ||||
|  | ||||
|     output.append(f"{ansii['bold']}PVC cluster status:{ansii['end']}") | ||||
|     output.append("") | ||||
|  | ||||
|     if health != "-1": | ||||
|         health = f"{health}%" | ||||
|     else: | ||||
|         health = "N/A" | ||||
|  | ||||
|     if maintenance == "true": | ||||
|         health = f"{health} (maintenance on)" | ||||
|  | ||||
|     output.append( | ||||
|         f"{ansii['purple']}Cluster health:{ansii['end']}   {health_colour}{health}{ansii['end']}" | ||||
|     ) | ||||
|  | ||||
|     if messages is not None and len(messages) > 0: | ||||
|         messages = "\n                  ".join(sorted(messages)) | ||||
|         output.append(f"{ansii['purple']}Health messages:{ansii['end']}  {messages}") | ||||
|  | ||||
|     output.append("") | ||||
|  | ||||
|     output.append(f"{ansii['purple']}Primary node:{ansii['end']}     {primary_node}") | ||||
|     output.append(f"{ansii['purple']}PVC version:{ansii['end']}      {pvc_version}") | ||||
|     output.append(f"{ansii['purple']}Upstream IP:{ansii['end']}      {upstream_ip}") | ||||
|     output.append("") | ||||
|  | ||||
|     node_states = ["run,ready"] | ||||
|     node_states.extend( | ||||
|         [ | ||||
|             state | ||||
|             for state in data.get("nodes", {}).keys() | ||||
|             if state not in ["total", "run,ready"] | ||||
|         ] | ||||
|     ) | ||||
|  | ||||
|     nodes_strings = list() | ||||
|     for state in node_states: | ||||
|         if state in ["run,ready"]: | ||||
|             state_colour = ansii["green"] | ||||
|         elif state in ["run,flush", "run,unflush", "run,flushed"]: | ||||
|             state_colour = ansii["blue"] | ||||
|         elif "dead" in state or "stop" in state: | ||||
|             state_colour = ansii["red"] | ||||
|         else: | ||||
|             state_colour = ansii["yellow"] | ||||
|  | ||||
|         nodes_strings.append( | ||||
|             f"{data.get('nodes', {}).get(state)}/{total_nodes} {state_colour}{state}{ansii['end']}" | ||||
|         ) | ||||
|  | ||||
|     nodes_string = ", ".join(nodes_strings) | ||||
|  | ||||
|     output.append(f"{ansii['purple']}Nodes:{ansii['end']}            {nodes_string}") | ||||
|  | ||||
|     vm_states = ["start", "disable"] | ||||
|     vm_states.extend( | ||||
|         [ | ||||
|             state | ||||
|             for state in data.get("vms", {}).keys() | ||||
|             if state not in ["total", "start", "disable"] | ||||
|         ] | ||||
|     ) | ||||
|  | ||||
|     vms_strings = list() | ||||
|     for state in vm_states: | ||||
|         if data.get("vms", {}).get(state) is None: | ||||
|             continue | ||||
|         if state in ["start"]: | ||||
|             state_colour = ansii["green"] | ||||
|         elif state in ["migrate", "disable"]: | ||||
|             state_colour = ansii["blue"] | ||||
|         elif state in ["stop", "fail"]: | ||||
|             state_colour = ansii["red"] | ||||
|         else: | ||||
|             state_colour = ansii["yellow"] | ||||
|  | ||||
|         vms_strings.append( | ||||
|             f"{data.get('vms', {}).get(state)}/{total_vms} {state_colour}{state}{ansii['end']}" | ||||
|         ) | ||||
|  | ||||
|     vms_string = ", ".join(vms_strings) | ||||
|  | ||||
|     output.append(f"{ansii['purple']}VMs:{ansii['end']}              {vms_string}") | ||||
|  | ||||
|     osd_states = ["up,in"] | ||||
|     osd_states.extend( | ||||
|         [ | ||||
|             state | ||||
|             for state in data.get("osds", {}).keys() | ||||
|             if state not in ["total", "up,in"] | ||||
|         ] | ||||
|     ) | ||||
|  | ||||
|     osds_strings = list() | ||||
|     for state in osd_states: | ||||
|         if state in ["up,in"]: | ||||
|             state_colour = ansii["green"] | ||||
|         elif state in ["down,out"]: | ||||
|             state_colour = ansii["red"] | ||||
|         else: | ||||
|             state_colour = ansii["yellow"] | ||||
|  | ||||
|         osds_strings.append( | ||||
|             f"{data.get('osds', {}).get(state)}/{total_osds} {state_colour}{state}{ansii['end']}" | ||||
|         ) | ||||
|  | ||||
|     osds_string = " ".join(osds_strings) | ||||
|  | ||||
|     output.append(f"{ansii['purple']}OSDs:{ansii['end']}             {osds_string}") | ||||
|  | ||||
|     output.append(f"{ansii['purple']}Pools:{ansii['end']}            {total_pools}") | ||||
|  | ||||
|     output.append(f"{ansii['purple']}Volumes:{ansii['end']}          {total_volumes}") | ||||
|  | ||||
|     output.append(f"{ansii['purple']}Snapshots:{ansii['end']}        {total_snapshots}") | ||||
|  | ||||
|     output.append(f"{ansii['purple']}Networks:{ansii['end']}         {total_networks}") | ||||
|  | ||||
|     output.append("") | ||||
|  | ||||
|     return "\n".join(output) | ||||
|  | ||||
|  | ||||
| def cli_cluster_status_format_short(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the health-only output of cli_cluster_status | ||||
|     """ | ||||
|  | ||||
|     # Normalize data to local variables | ||||
|     health = data.get("cluster_health", {}).get("health", -1) | ||||
|     messages = data.get("cluster_health", {}).get("messages", None) | ||||
|     maintenance = data.get("maintenance", "N/A") | ||||
|  | ||||
|     if maintenance == "true" or health == -1: | ||||
|         health_colour = ansii["blue"] | ||||
|     elif health > 90: | ||||
|         health_colour = ansii["green"] | ||||
|     elif health > 50: | ||||
|         health_colour = ansii["yellow"] | ||||
|     else: | ||||
|         health_colour = ansii["red"] | ||||
|  | ||||
|     output = list() | ||||
|  | ||||
|     output.append(f"{ansii['bold']}PVC cluster status:{ansii['end']}") | ||||
|     output.append("") | ||||
|  | ||||
|     if health != "-1": | ||||
|         health = f"{health}%" | ||||
|     else: | ||||
|         health = "N/A" | ||||
|  | ||||
|     if maintenance == "true": | ||||
|         health = f"{health} (maintenance on)" | ||||
|  | ||||
|     output.append( | ||||
|         f"{ansii['purple']}Cluster health:{ansii['end']}   {health_colour}{health}{ansii['end']}" | ||||
|     ) | ||||
|  | ||||
|     if messages is not None and len(messages) > 0: | ||||
|         messages = "\n                  ".join(sorted(messages)) | ||||
|         output.append(f"{ansii['purple']}Health messages:{ansii['end']}  {messages}") | ||||
|  | ||||
|     output.append("") | ||||
|  | ||||
|     return "\n".join(output) | ||||
|  | ||||
|  | ||||
| def cli_connection_list_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_connection_list | ||||
|     """ | ||||
|  | ||||
|     # Set the fields data | ||||
|     fields = { | ||||
|         "name": {"header": "Name", "length": len("Name") + 1}, | ||||
|         "description": {"header": "Description", "length": len("Description") + 1}, | ||||
|         "address": {"header": "Address", "length": len("Address") + 1}, | ||||
|         "port": {"header": "Port", "length": len("Port") + 1}, | ||||
|         "scheme": {"header": "Scheme", "length": len("Scheme") + 1}, | ||||
|         "api_key": {"header": "API Key", "length": len("API Key") + 1}, | ||||
|     } | ||||
|  | ||||
|     # Parse each connection and adjust field lengths | ||||
|     for connection in data: | ||||
|         for field, length in [(f, fields[f]["length"]) for f in fields]: | ||||
|             _length = len(str(connection[field])) | ||||
|             if _length > length: | ||||
|                 length = len(str(connection[field])) + 1 | ||||
|  | ||||
|             fields[field]["length"] = length | ||||
|  | ||||
|     # Create the output object and define the line format | ||||
|     output = list() | ||||
|     line = "{bold}{name: <{lname}} {desc: <{ldesc}} {addr: <{laddr}} {port: <{lport}} {schm: <{lschm}} {akey: <{lakey}}{end}" | ||||
|  | ||||
|     # Add the header line | ||||
|     output.append( | ||||
|         line.format( | ||||
|             bold=ansii["bold"], | ||||
|             end=ansii["end"], | ||||
|             name=fields["name"]["header"], | ||||
|             lname=fields["name"]["length"], | ||||
|             desc=fields["description"]["header"], | ||||
|             ldesc=fields["description"]["length"], | ||||
|             addr=fields["address"]["header"], | ||||
|             laddr=fields["address"]["length"], | ||||
|             port=fields["port"]["header"], | ||||
|             lport=fields["port"]["length"], | ||||
|             schm=fields["scheme"]["header"], | ||||
|             lschm=fields["scheme"]["length"], | ||||
|             akey=fields["api_key"]["header"], | ||||
|             lakey=fields["api_key"]["length"], | ||||
|         ) | ||||
|     ) | ||||
|  | ||||
|     # Add a line per connection | ||||
|     for connection in data: | ||||
|         output.append( | ||||
|             line.format( | ||||
|                 bold="", | ||||
|                 end="", | ||||
|                 name=connection["name"], | ||||
|                 lname=fields["name"]["length"], | ||||
|                 desc=connection["description"], | ||||
|                 ldesc=fields["description"]["length"], | ||||
|                 addr=connection["address"], | ||||
|                 laddr=fields["address"]["length"], | ||||
|                 port=connection["port"], | ||||
|                 lport=fields["port"]["length"], | ||||
|                 schm=connection["scheme"], | ||||
|                 lschm=fields["scheme"]["length"], | ||||
|                 akey=connection["api_key"], | ||||
|                 lakey=fields["api_key"]["length"], | ||||
|             ) | ||||
|         ) | ||||
|  | ||||
|     return "\n".join(output) | ||||
|  | ||||
|  | ||||
| def cli_connection_detail_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_connection_detail | ||||
|     """ | ||||
|  | ||||
|     # Set the fields data | ||||
|     fields = { | ||||
|         "name": {"header": "Name", "length": len("Name") + 1}, | ||||
|         "description": {"header": "Description", "length": len("Description") + 1}, | ||||
|         "health": {"header": "Health", "length": len("Health") + 1}, | ||||
|         "primary_node": {"header": "Primary", "length": len("Primary") + 1}, | ||||
|         "pvc_version": {"header": "Version", "length": len("Version") + 1}, | ||||
|         "nodes": {"header": "Nodes", "length": len("Nodes") + 1}, | ||||
|         "vms": {"header": "VMs", "length": len("VMs") + 1}, | ||||
|         "networks": {"header": "Networks", "length": len("Networks") + 1}, | ||||
|         "osds": {"header": "OSDs", "length": len("OSDs") + 1}, | ||||
|         "pools": {"header": "Pools", "length": len("Pools") + 1}, | ||||
|         "volumes": {"header": "Volumes", "length": len("Volumes") + 1}, | ||||
|         "snapshots": {"header": "Snapshots", "length": len("Snapshots") + 1}, | ||||
|     } | ||||
|  | ||||
|     # Parse each connection and adjust field lengths | ||||
|     for connection in data: | ||||
|         for field, length in [(f, fields[f]["length"]) for f in fields]: | ||||
|             _length = len(str(connection[field])) | ||||
|             if _length > length: | ||||
|                 length = len(str(connection[field])) + 1 | ||||
|  | ||||
|             fields[field]["length"] = length | ||||
|  | ||||
|     # Create the output object and define the line format | ||||
|     output = list() | ||||
|     line = "{bold}{name: <{lname}} {desc: <{ldesc}} {chlth}{hlth: <{lhlth}}{endc} {prin: <{lprin}} {vers: <{lvers}} {nods: <{lnods}} {vms: <{lvms}} {nets: <{lnets}} {osds: <{losds}} {pols: <{lpols}} {vols: <{lvols}} {snts: <{lsnts}}{end}" | ||||
|  | ||||
|     # Add the header line | ||||
|     output.append( | ||||
|         line.format( | ||||
|             bold=ansii["bold"], | ||||
|             end=ansii["end"], | ||||
|             chlth="", | ||||
|             endc="", | ||||
|             name=fields["name"]["header"], | ||||
|             lname=fields["name"]["length"], | ||||
|             desc=fields["description"]["header"], | ||||
|             ldesc=fields["description"]["length"], | ||||
|             hlth=fields["health"]["header"], | ||||
|             lhlth=fields["health"]["length"], | ||||
|             prin=fields["primary_node"]["header"], | ||||
|             lprin=fields["primary_node"]["length"], | ||||
|             vers=fields["pvc_version"]["header"], | ||||
|             lvers=fields["pvc_version"]["length"], | ||||
|             nods=fields["nodes"]["header"], | ||||
|             lnods=fields["nodes"]["length"], | ||||
|             vms=fields["vms"]["header"], | ||||
|             lvms=fields["vms"]["length"], | ||||
|             nets=fields["networks"]["header"], | ||||
|             lnets=fields["networks"]["length"], | ||||
|             osds=fields["osds"]["header"], | ||||
|             losds=fields["osds"]["length"], | ||||
|             pols=fields["pools"]["header"], | ||||
|             lpols=fields["pools"]["length"], | ||||
|             vols=fields["volumes"]["header"], | ||||
|             lvols=fields["volumes"]["length"], | ||||
|             snts=fields["snapshots"]["header"], | ||||
|             lsnts=fields["snapshots"]["length"], | ||||
|         ) | ||||
|     ) | ||||
|  | ||||
|     # Add a line per connection | ||||
|     for connection in data: | ||||
|         if connection["health"] == "N/A": | ||||
|             health_value = "N/A" | ||||
|             health_colour = ansii["purple"] | ||||
|         else: | ||||
|             health_value = f"{connection['health']}%" | ||||
|             if connection["maintenance"] == "true": | ||||
|                 health_colour = ansii["blue"] | ||||
|             elif connection["health"] > 90: | ||||
|                 health_colour = ansii["green"] | ||||
|             elif connection["health"] > 50: | ||||
|                 health_colour = ansii["yellow"] | ||||
|             else: | ||||
|                 health_colour = ansii["red"] | ||||
|  | ||||
|         output.append( | ||||
|             line.format( | ||||
|                 bold="", | ||||
|                 end="", | ||||
|                 chlth=health_colour, | ||||
|                 endc=ansii["end"], | ||||
|                 name=connection["name"], | ||||
|                 lname=fields["name"]["length"], | ||||
|                 desc=connection["description"], | ||||
|                 ldesc=fields["description"]["length"], | ||||
|                 hlth=health_value, | ||||
|                 lhlth=fields["health"]["length"], | ||||
|                 prin=connection["primary_node"], | ||||
|                 lprin=fields["primary_node"]["length"], | ||||
|                 vers=connection["pvc_version"], | ||||
|                 lvers=fields["pvc_version"]["length"], | ||||
|                 nods=connection["nodes"], | ||||
|                 lnods=fields["nodes"]["length"], | ||||
|                 vms=connection["vms"], | ||||
|                 lvms=fields["vms"]["length"], | ||||
|                 nets=connection["networks"], | ||||
|                 lnets=fields["networks"]["length"], | ||||
|                 osds=connection["osds"], | ||||
|                 losds=fields["osds"]["length"], | ||||
|                 pols=connection["pools"], | ||||
|                 lpols=fields["pools"]["length"], | ||||
|                 vols=connection["volumes"], | ||||
|                 lvols=fields["volumes"]["length"], | ||||
|                 snts=connection["snapshots"], | ||||
|                 lsnts=fields["snapshots"]["length"], | ||||
|             ) | ||||
|         ) | ||||
|  | ||||
|     return "\n".join(output) | ||||
|  | ||||
|  | ||||
| def cli_node_info_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the basic output of cli_node_info | ||||
|     """ | ||||
|  | ||||
|     return node_format_info(CLI_CONFIG, data, long_output=False) | ||||
|  | ||||
|  | ||||
| def cli_node_info_format_long(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the full output of cli_node_info | ||||
|     """ | ||||
|  | ||||
|     return node_format_info(CLI_CONFIG, data, long_output=True) | ||||
|  | ||||
|  | ||||
| def cli_node_list_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_node_list | ||||
|     """ | ||||
|  | ||||
|     return node_format_list(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_vm_tag_get_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_vm_tag_get | ||||
|     """ | ||||
|  | ||||
|     return vm_format_tags(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_vm_vcpu_get_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_vm_vcpu_get | ||||
|     """ | ||||
|  | ||||
|     return vm_format_vcpus(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_vm_memory_get_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_vm_memory_get | ||||
|     """ | ||||
|  | ||||
|     return vm_format_memory(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_vm_network_get_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_vm_network_get | ||||
|     """ | ||||
|  | ||||
|     return vm_format_networks(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_vm_volume_get_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_vm_volume_get | ||||
|     """ | ||||
|  | ||||
|     return vm_format_volumes(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_vm_info_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the basic output of cli_vm_info | ||||
|     """ | ||||
|  | ||||
|     return vm_format_info(CLI_CONFIG, data, long_output=False) | ||||
|  | ||||
|  | ||||
| def cli_vm_info_format_long(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the full output of cli_vm_info | ||||
|     """ | ||||
|  | ||||
|     return vm_format_info(CLI_CONFIG, data, long_output=True) | ||||
|  | ||||
|  | ||||
| def cli_vm_list_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_vm_list | ||||
|     """ | ||||
|  | ||||
|     return vm_format_list(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_network_info_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the full output of cli_network_info | ||||
|     """ | ||||
|  | ||||
|     return network_format_info(CLI_CONFIG, data, long_output=True) | ||||
|  | ||||
|  | ||||
| def cli_network_info_format_long(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the full output of cli_network_info | ||||
|     """ | ||||
|  | ||||
|     return network_format_info(CLI_CONFIG, data, long_output=True) | ||||
|  | ||||
|  | ||||
| def cli_network_list_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_network_list | ||||
|     """ | ||||
|  | ||||
|     return network_format_list(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_network_dhcp_list_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_network_dhcp_list | ||||
|     """ | ||||
|  | ||||
|     return network_format_dhcp_list(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_network_acl_list_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_network_acl_list | ||||
|     """ | ||||
|  | ||||
|     return network_format_acl_list(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_network_sriov_pf_list_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_network_sriov_pf_list | ||||
|     """ | ||||
|  | ||||
|     return network_format_sriov_pf_list(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_network_sriov_vf_info_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_network_sriov_vf_info | ||||
|     """ | ||||
|  | ||||
|     return network_format_sriov_vf_info(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_network_sriov_vf_list_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_network_sriov_vf_list | ||||
|     """ | ||||
|  | ||||
|     return network_format_sriov_vf_list(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_storage_status_format_raw(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Direct format the output of cli_storage_status | ||||
|     """ | ||||
|  | ||||
|     return storage_format_raw(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_storage_util_format_raw(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Direct format the output of cli_storage_util | ||||
|     """ | ||||
|  | ||||
|     return storage_format_raw(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_storage_benchmark_info_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_storage_benchmark_info | ||||
|     """ | ||||
|  | ||||
|     return storage_format_benchmark_info(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_storage_benchmark_list_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_storage_benchmark_list | ||||
|     """ | ||||
|  | ||||
|     return storage_format_benchmark_list(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_storage_osd_list_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_storage_osd_list | ||||
|     """ | ||||
|  | ||||
|     return storage_format_osd_list(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_storage_pool_list_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_storage_pool_list | ||||
|     """ | ||||
|  | ||||
|     return storage_format_pool_list(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_storage_volume_list_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_storage_volume_list | ||||
|     """ | ||||
|  | ||||
|     return storage_format_volume_list(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_storage_snapshot_list_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_storage_snapshot_list | ||||
|     """ | ||||
|  | ||||
|     return storage_format_snapshot_list(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_provisioner_template_system_list_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_provisioner_template_system_list | ||||
|     """ | ||||
|  | ||||
|     return provisioner_format_template_list(CLI_CONFIG, data, template_type="system") | ||||
|  | ||||
|  | ||||
| def cli_provisioner_template_network_list_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_provisioner_template_network_list | ||||
|     """ | ||||
|  | ||||
|     return provisioner_format_template_list(CLI_CONFIG, data, template_type="network") | ||||
|  | ||||
|  | ||||
| def cli_provisioner_template_storage_list_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_provisioner_template_storage_list | ||||
|     """ | ||||
|  | ||||
|     return provisioner_format_template_list(CLI_CONFIG, data, template_type="storage") | ||||
|  | ||||
|  | ||||
| def cli_provisioner_userdata_list_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_provisioner_userdata_list | ||||
|     """ | ||||
|  | ||||
|     return provisioner_format_userdata_list(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_provisioner_script_list_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_provisioner_script_list | ||||
|     """ | ||||
|  | ||||
|     return provisioner_format_script_list(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_provisioner_ova_list_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_provisioner_ova_list | ||||
|     """ | ||||
|  | ||||
|     return provisioner_format_ova_list(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_provisioner_profile_list_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_provisioner_profile_list | ||||
|     """ | ||||
|  | ||||
|     return provisioner_format_profile_list(CLI_CONFIG, data) | ||||
|  | ||||
|  | ||||
| def cli_provisioner_status_format_pretty(CLI_CONFIG, data): | ||||
|     """ | ||||
|     Pretty format the output of cli_provisioner_status | ||||
|     """ | ||||
|  | ||||
|     return provisioner_format_task_status(CLI_CONFIG, data) | ||||
| @@ -1,241 +0,0 @@ | ||||
| #!/usr/bin/env python3 | ||||
|  | ||||
| # helpers.py - PVC Click CLI helper function library | ||||
| # Part of the Parallel Virtual Cluster (PVC) system | ||||
| # | ||||
| #    Copyright (C) 2018-2023 Joshua M. Boniface <joshua@boniface.me> | ||||
| # | ||||
| #    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, version 3. | ||||
| # | ||||
| #    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/>. | ||||
| # | ||||
| ############################################################################### | ||||
|  | ||||
| from click import echo as click_echo | ||||
| from click import progressbar | ||||
| from distutils.util import strtobool | ||||
| from json import load as jload | ||||
| from json import dump as jdump | ||||
| from os import chmod, environ, getpid, path | ||||
| from socket import gethostname | ||||
| from sys import argv | ||||
| from syslog import syslog, openlog, closelog, LOG_AUTH | ||||
| from time import sleep | ||||
| from yaml import load as yload | ||||
| from yaml import BaseLoader | ||||
|  | ||||
| import pvc.lib.provisioner | ||||
|  | ||||
|  | ||||
| DEFAULT_STORE_DATA = {"cfgfile": "/etc/pvc/pvcapid.yaml"} | ||||
| DEFAULT_STORE_FILENAME = "pvc.json" | ||||
| DEFAULT_API_PREFIX = "/api/v1" | ||||
| DEFAULT_NODE_HOSTNAME = gethostname().split(".")[0] | ||||
|  | ||||
|  | ||||
| def echo(config, message, newline=True, stderr=False): | ||||
|     """ | ||||
|     Output a message with click.echo respecting our configuration | ||||
|     """ | ||||
|  | ||||
|     if config.get("colour", False): | ||||
|         colour = True | ||||
|     else: | ||||
|         colour = None | ||||
|  | ||||
|     if config.get("silent", False): | ||||
|         pass | ||||
|     elif config.get("quiet", False) and stderr: | ||||
|         pass | ||||
|     else: | ||||
|         click_echo(message=message, color=colour, nl=newline, err=stderr) | ||||
|  | ||||
|  | ||||
| def audit(): | ||||
|     """ | ||||
|     Log an audit message to the local syslog AUTH facility | ||||
|     """ | ||||
|  | ||||
|     args = argv | ||||
|     args[0] = "pvc" | ||||
|     pid = getpid() | ||||
|  | ||||
|     openlog(facility=LOG_AUTH, ident=f"{args[0]}[{pid}]") | ||||
|     syslog( | ||||
|         f"""client audit: command "{' '.join(args)}" by user {environ.get('USER', None)}""" | ||||
|     ) | ||||
|     closelog() | ||||
|  | ||||
|  | ||||
| def read_config_from_yaml(cfgfile): | ||||
|     """ | ||||
|     Read the PVC API configuration from the local API configuration file | ||||
|     """ | ||||
|  | ||||
|     try: | ||||
|         with open(cfgfile) as fh: | ||||
|             api_config = yload(fh, Loader=BaseLoader)["pvc"]["api"] | ||||
|  | ||||
|         host = api_config["listen_address"] | ||||
|         port = api_config["listen_port"] | ||||
|         scheme = "https" if strtobool(api_config["ssl"]["enabled"]) else "http" | ||||
|         api_key = ( | ||||
|             api_config["authentication"]["tokens"][0]["token"] | ||||
|             if strtobool(api_config["authentication"]["enabled"]) | ||||
|             else None | ||||
|         ) | ||||
|     except KeyError: | ||||
|         host = None | ||||
|         port = None | ||||
|         scheme = None | ||||
|         api_key = None | ||||
|  | ||||
|     return cfgfile, host, port, scheme, api_key | ||||
|  | ||||
|  | ||||
| def get_config(store_data, connection=None): | ||||
|     """ | ||||
|     Load CLI configuration from store data | ||||
|     """ | ||||
|  | ||||
|     if store_data is None: | ||||
|         return {"badcfg": True} | ||||
|  | ||||
|     connection_details = store_data.get(connection, None) | ||||
|  | ||||
|     if not connection_details: | ||||
|         connection = "local" | ||||
|         connection_details = DEFAULT_STORE_DATA | ||||
|  | ||||
|     if connection_details.get("cfgfile", None) is not None: | ||||
|         if path.isfile(connection_details.get("cfgfile", None)): | ||||
|             description, host, port, scheme, api_key = read_config_from_yaml( | ||||
|                 connection_details.get("cfgfile", None) | ||||
|             ) | ||||
|             if None in [description, host, port, scheme]: | ||||
|                 return {"badcfg": True} | ||||
|         else: | ||||
|             return {"badcfg": True} | ||||
|         # Rewrite a wildcard listener to use localhost instead | ||||
|         if host == "0.0.0.0": | ||||
|             host = "127.0.0.1" | ||||
|     else: | ||||
|         # This is a static configuration, get the details directly | ||||
|         description = connection_details["description"] | ||||
|         host = connection_details["host"] | ||||
|         port = connection_details["port"] | ||||
|         scheme = connection_details["scheme"] | ||||
|         api_key = connection_details["api_key"] | ||||
|  | ||||
|     config = dict() | ||||
|     config["debug"] = False | ||||
|     config["connection"] = connection | ||||
|     config["description"] = description | ||||
|     config["api_host"] = f"{host}:{port}" | ||||
|     config["api_scheme"] = scheme | ||||
|     config["api_key"] = api_key | ||||
|     config["api_prefix"] = DEFAULT_API_PREFIX | ||||
|     if connection == "local": | ||||
|         config["verify_ssl"] = False | ||||
|     else: | ||||
|         config["verify_ssl"] = bool( | ||||
|             strtobool(environ.get("PVC_CLIENT_VERIFY_SSL", "True")) | ||||
|         ) | ||||
|  | ||||
|     return config | ||||
|  | ||||
|  | ||||
| def get_store(store_path): | ||||
|     """ | ||||
|     Load store information from the store path | ||||
|     """ | ||||
|  | ||||
|     store_file = f"{store_path}/{DEFAULT_STORE_FILENAME}" | ||||
|  | ||||
|     with open(store_file) as fh: | ||||
|         try: | ||||
|             store_data = jload(fh) | ||||
|             return store_data | ||||
|         except Exception: | ||||
|             return dict() | ||||
|  | ||||
|  | ||||
| def update_store(store_path, store_data): | ||||
|     """ | ||||
|     Update store information to the store path, creating it (with sensible permissions) if needed | ||||
|     """ | ||||
|  | ||||
|     store_file = f"{store_path}/{DEFAULT_STORE_FILENAME}" | ||||
|  | ||||
|     if not path.exists(store_file): | ||||
|         with open(store_file, "w") as fh: | ||||
|             fh.write("") | ||||
|         chmod(store_file, int(environ.get("PVC_CLIENT_DB_PERMS", "600"), 8)) | ||||
|  | ||||
|     with open(store_file, "w") as fh: | ||||
|         jdump(store_data, fh, sort_keys=True, indent=4) | ||||
|  | ||||
|  | ||||
| def wait_for_provisioner(CLI_CONFIG, task_id): | ||||
|     """ | ||||
|     Wait for a provisioner task to complete | ||||
|     """ | ||||
|  | ||||
|     echo(CLI_CONFIG, f"Task ID: {task_id}") | ||||
|     echo(CLI_CONFIG, "") | ||||
|  | ||||
|     # Wait for the task to start | ||||
|     echo(CLI_CONFIG, "Waiting for task to start...", newline=False) | ||||
|     while True: | ||||
|         sleep(1) | ||||
|         task_status = pvc.lib.provisioner.task_status( | ||||
|             CLI_CONFIG, task_id, is_watching=True | ||||
|         ) | ||||
|         if task_status.get("state") != "PENDING": | ||||
|             break | ||||
|         echo(".", newline=False) | ||||
|     echo(CLI_CONFIG, " done.") | ||||
|     echo(CLI_CONFIG, "") | ||||
|  | ||||
|     # Start following the task state, updating progress as we go | ||||
|     total_task = task_status.get("total") | ||||
|     with progressbar(length=total_task, show_eta=False) as bar: | ||||
|         last_task = 0 | ||||
|         maxlen = 0 | ||||
|         while True: | ||||
|             sleep(1) | ||||
|             if task_status.get("state") != "RUNNING": | ||||
|                 break | ||||
|             if task_status.get("current") > last_task: | ||||
|                 current_task = int(task_status.get("current")) | ||||
|                 bar.update(current_task - last_task) | ||||
|                 last_task = current_task | ||||
|                 # The extensive spaces at the end cause this to overwrite longer previous messages | ||||
|                 curlen = len(str(task_status.get("status"))) | ||||
|                 if curlen > maxlen: | ||||
|                     maxlen = curlen | ||||
|                 lendiff = maxlen - curlen | ||||
|                 overwrite_whitespace = " " * lendiff | ||||
|                 echo( | ||||
|                     CLI_CONFIG, | ||||
|                     "  " + task_status.get("status") + overwrite_whitespace, | ||||
|                     newline=False, | ||||
|                 ) | ||||
|             task_status = pvc.lib.provisioner.task_status( | ||||
|                 CLI_CONFIG, task_id, is_watching=True | ||||
|             ) | ||||
|         if task_status.get("state") == "SUCCESS": | ||||
|             bar.update(total_task - last_task) | ||||
|  | ||||
|     echo(CLI_CONFIG, "") | ||||
|     retdata = task_status.get("state") + ": " + task_status.get("status") | ||||
|  | ||||
|     return retdata | ||||
| @@ -1,124 +0,0 @@ | ||||
| #!/usr/bin/env python3 | ||||
|  | ||||
| # parsers.py - PVC Click CLI data parser function library | ||||
| # Part of the Parallel Virtual Cluster (PVC) system | ||||
| # | ||||
| #    Copyright (C) 2018-2023 Joshua M. Boniface <joshua@boniface.me> | ||||
| # | ||||
| #    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, version 3. | ||||
| # | ||||
| #    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/>. | ||||
| # | ||||
| ############################################################################### | ||||
|  | ||||
| from os import path | ||||
| from re import sub | ||||
|  | ||||
| from pvc.cli.helpers import read_config_from_yaml, get_config | ||||
|  | ||||
| import pvc.lib.cluster | ||||
|  | ||||
|  | ||||
| def cli_connection_list_parser(connections_config, show_keys_flag): | ||||
|     """ | ||||
|     Parse connections_config into formatable data for cli_connection_list | ||||
|     """ | ||||
|  | ||||
|     connections_data = list() | ||||
|  | ||||
|     for connection, details in connections_config.items(): | ||||
|         if details.get("cfgfile", None) is not None: | ||||
|             if path.isfile(details.get("cfgfile")): | ||||
|                 description, address, port, scheme, api_key = read_config_from_yaml( | ||||
|                     details.get("cfgfile") | ||||
|                 ) | ||||
|             else: | ||||
|                 continue | ||||
|             if not show_keys_flag and api_key is not None: | ||||
|                 api_key = sub(r"[a-z0-9]", "x", api_key) | ||||
|             connections_data.append( | ||||
|                 { | ||||
|                     "name": connection, | ||||
|                     "description": description, | ||||
|                     "address": address, | ||||
|                     "port": port, | ||||
|                     "scheme": scheme, | ||||
|                     "api_key": api_key, | ||||
|                 } | ||||
|             ) | ||||
|         else: | ||||
|             if not show_keys_flag: | ||||
|                 details["api_key"] = sub(r"[a-z0-9]", "x", details["api_key"]) | ||||
|             connections_data.append( | ||||
|                 { | ||||
|                     "name": connection, | ||||
|                     "description": details["description"], | ||||
|                     "address": details["host"], | ||||
|                     "port": details["port"], | ||||
|                     "scheme": details["scheme"], | ||||
|                     "api_key": details["api_key"], | ||||
|                 } | ||||
|             ) | ||||
|  | ||||
|     return connections_data | ||||
|  | ||||
|  | ||||
| def cli_connection_detail_parser(connections_config): | ||||
|     """ | ||||
|     Parse connections_config into formatable data for cli_connection_detail | ||||
|     """ | ||||
|     connections_data = list() | ||||
|     for connection, details in connections_config.items(): | ||||
|         cluster_config = get_config(connections_config, connection=connection) | ||||
|         if cluster_config.get("badcfg", False): | ||||
|             continue | ||||
|         # Connect to each API and gather cluster status | ||||
|         retcode, retdata = pvc.lib.cluster.get_info(cluster_config) | ||||
|         if retcode == 0: | ||||
|             # Create dummy data of N/A for all fields | ||||
|             connections_data.append( | ||||
|                 { | ||||
|                     "name": cluster_config["connection"], | ||||
|                     "description": cluster_config["description"], | ||||
|                     "health": "N/A", | ||||
|                     "maintenance": "N/A", | ||||
|                     "primary_node": "N/A", | ||||
|                     "pvc_version": "N/A", | ||||
|                     "nodes": "N/A", | ||||
|                     "vms": "N/A", | ||||
|                     "networks": "N/A", | ||||
|                     "osds": "N/A", | ||||
|                     "pools": "N/A", | ||||
|                     "volumes": "N/A", | ||||
|                     "snapshots": "N/A", | ||||
|                 } | ||||
|             ) | ||||
|         else: | ||||
|             # Normalize data into nice formattable version | ||||
|             connections_data.append( | ||||
|                 { | ||||
|                     "name": cluster_config["connection"], | ||||
|                     "description": cluster_config["description"], | ||||
|                     "health": retdata.get("cluster_health", {}).get("health", "N/A"), | ||||
|                     "maintenance": retdata.get("maintenance", "N/A"), | ||||
|                     "primary_node": retdata.get("primary_node", "N/A"), | ||||
|                     "pvc_version": retdata.get("pvc_version", "N/A"), | ||||
|                     "nodes": retdata.get("nodes", {}).get("total", "N/A"), | ||||
|                     "vms": retdata.get("vms", {}).get("total", "N/A"), | ||||
|                     "networks": retdata.get("networks", "N/A"), | ||||
|                     "osds": retdata.get("osds", {}).get("total", "N/A"), | ||||
|                     "pools": retdata.get("pools", "N/A"), | ||||
|                     "volumes": retdata.get("volumes", "N/A"), | ||||
|                     "snapshots": retdata.get("snapshots", "N/A"), | ||||
|                 } | ||||
|             ) | ||||
|  | ||||
|     return connections_data | ||||
| @@ -1,64 +0,0 @@ | ||||
| #!/usr/bin/env python3 | ||||
|  | ||||
| # waiters.py - PVC Click CLI output waiters library | ||||
| # Part of the Parallel Virtual Cluster (PVC) system | ||||
| # | ||||
| #    Copyright (C) 2018-2023 Joshua M. Boniface <joshua@boniface.me> | ||||
| # | ||||
| #    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, version 3. | ||||
| # | ||||
| #    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/>. | ||||
| # | ||||
| ############################################################################### | ||||
|  | ||||
| from time import sleep, time | ||||
|  | ||||
| from pvc.cli.helpers import echo | ||||
|  | ||||
| import pvc.lib.node | ||||
|  | ||||
|  | ||||
| def cli_node_waiter(config, node, state_field, state_value): | ||||
|     """ | ||||
|     Wait for state transitions for cli_node tasks | ||||
|  | ||||
|     {node} is the name of the node | ||||
|     {state_field} is the node_info field to check for {state_value} | ||||
|     {state_value} is the TRANSITIONAL value that, when no longer set, will terminate waiting | ||||
|     """ | ||||
|  | ||||
|     # Sleep for this long between API polls | ||||
|     sleep_time = 1 | ||||
|  | ||||
|     # Print a dot after this many {sleep_time}s | ||||
|     dot_time = 5 | ||||
|  | ||||
|     t_start = time() | ||||
|  | ||||
|     echo(config, "Waiting...", newline=False) | ||||
|     sleep(sleep_time) | ||||
|  | ||||
|     count = 0 | ||||
|     while True: | ||||
|         count += 1 | ||||
|         try: | ||||
|             _retcode, _retdata = pvc.lib.node.node_info(config, node) | ||||
|             if _retdata[state_field] != state_value: | ||||
|                 break | ||||
|             else: | ||||
|                 raise ValueError | ||||
|         except Exception: | ||||
|             sleep(sleep_time) | ||||
|             if count % dot_time == 0: | ||||
|                 echo(config, ".", newline=False) | ||||
|  | ||||
|     t_end = time() | ||||
|     echo(config, f" done. [{int(t_end - t_start)}s]") | ||||
| @@ -27,8 +27,8 @@ from requests_toolbelt.multipart.encoder import ( | ||||
|     MultipartEncoderMonitor, | ||||
| ) | ||||
| 
 | ||||
| import pvc.lib.ansiprint as ansiprint | ||||
| from pvc.lib.common import UploadProgressBar, call_api | ||||
| import pvc.cli_lib.ansiprint as ansiprint | ||||
| from pvc.cli_lib.common import UploadProgressBar, call_api | ||||
| 
 | ||||
| # | ||||
| # Supplemental functions | ||||
| @@ -21,8 +21,8 @@ | ||||
| 
 | ||||
| import json | ||||
| 
 | ||||
| import pvc.lib.ansiprint as ansiprint | ||||
| from pvc.lib.common import call_api | ||||
| import pvc.cli_lib.ansiprint as ansiprint | ||||
| from pvc.cli_lib.common import call_api | ||||
| 
 | ||||
| 
 | ||||
| def initialize(config, overwrite=False): | ||||
| @@ -124,8 +124,8 @@ def call_api( | ||||
|     data=None, | ||||
|     files=None, | ||||
| ): | ||||
|     # Set the connect timeout to 2 seconds but extremely long (48 hour) data timeout | ||||
|     timeout = (2.05, 172800) | ||||
|     # Set the connect timeout to 1 seconds but extremely long (48 hour) data timeout | ||||
|     timeout = (1.05, 172800) | ||||
| 
 | ||||
|     # Craft the URI | ||||
|     uri = "{}://{}{}{}".format( | ||||
| @@ -20,8 +20,8 @@ | ||||
| ############################################################################### | ||||
| 
 | ||||
| import re | ||||
| import pvc.lib.ansiprint as ansiprint | ||||
| from pvc.lib.common import call_api | ||||
| import pvc.cli_lib.ansiprint as ansiprint | ||||
| from pvc.cli_lib.common import call_api | ||||
| 
 | ||||
| 
 | ||||
| def isValidMAC(macaddr): | ||||
| @@ -21,8 +21,8 @@ | ||||
| 
 | ||||
| import time | ||||
| 
 | ||||
| import pvc.lib.ansiprint as ansiprint | ||||
| from pvc.lib.common import call_api | ||||
| import pvc.cli_lib.ansiprint as ansiprint | ||||
| from pvc.cli_lib.common import call_api | ||||
| 
 | ||||
| 
 | ||||
| # | ||||
| @@ -24,8 +24,8 @@ from requests_toolbelt.multipart.encoder import ( | ||||
|     MultipartEncoderMonitor, | ||||
| ) | ||||
| 
 | ||||
| import pvc.lib.ansiprint as ansiprint | ||||
| from pvc.lib.common import UploadProgressBar, call_api | ||||
| import pvc.cli_lib.ansiprint as ansiprint | ||||
| from pvc.cli_lib.common import UploadProgressBar, call_api | ||||
| from ast import literal_eval | ||||
| 
 | ||||
| 
 | ||||
| @@ -22,8 +22,8 @@ | ||||
| import time | ||||
| import re | ||||
| 
 | ||||
| import pvc.lib.ansiprint as ansiprint | ||||
| from pvc.lib.common import call_api, format_bytes, format_metric | ||||
| import pvc.cli_lib.ansiprint as ansiprint | ||||
| from pvc.cli_lib.common import call_api, format_bytes, format_metric | ||||
| 
 | ||||
| 
 | ||||
| # | ||||
| @@ -677,7 +677,7 @@ def vm_networks_add( | ||||
|     from lxml.objectify import fromstring | ||||
|     from lxml.etree import tostring | ||||
|     from random import randint | ||||
|     import pvc.lib.network as pvc_network | ||||
|     import pvc.cli_lib.network as pvc_network | ||||
| 
 | ||||
|     network_exists, _ = pvc_network.net_info(config, network) | ||||
|     if not network_exists: | ||||
| @@ -1046,7 +1046,7 @@ def vm_volumes_add(config, vm, volume, disk_id, bus, disk_type, live, restart): | ||||
|     from lxml.objectify import fromstring | ||||
|     from lxml.etree import tostring | ||||
|     from copy import deepcopy | ||||
|     import pvc.lib.ceph as pvc_ceph | ||||
|     import pvc.cli_lib.ceph as pvc_ceph | ||||
| 
 | ||||
|     if disk_type == "rbd": | ||||
|         # Verify that the provided volume is valid | ||||
| @@ -1,97 +0,0 @@ | ||||
| #!/usr/bin/env python3 | ||||
|  | ||||
| # ansiprint.py - Printing function for formatted messages | ||||
| # Part of the Parallel Virtual Cluster (PVC) system | ||||
| # | ||||
| #    Copyright (C) 2018-2022 Joshua M. Boniface <joshua@boniface.me> | ||||
| # | ||||
| #    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, version 3. | ||||
| # | ||||
| #    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/>. | ||||
| # | ||||
| ############################################################################### | ||||
|  | ||||
| import datetime | ||||
|  | ||||
|  | ||||
| # ANSII colours for output | ||||
| def red(): | ||||
|     return "\033[91m" | ||||
|  | ||||
|  | ||||
| def blue(): | ||||
|     return "\033[94m" | ||||
|  | ||||
|  | ||||
| def cyan(): | ||||
|     return "\033[96m" | ||||
|  | ||||
|  | ||||
| def green(): | ||||
|     return "\033[92m" | ||||
|  | ||||
|  | ||||
| def yellow(): | ||||
|     return "\033[93m" | ||||
|  | ||||
|  | ||||
| def purple(): | ||||
|     return "\033[95m" | ||||
|  | ||||
|  | ||||
| def bold(): | ||||
|     return "\033[1m" | ||||
|  | ||||
|  | ||||
| def end(): | ||||
|     return "\033[0m" | ||||
|  | ||||
|  | ||||
| # Print function | ||||
| def echo(message, prefix, state): | ||||
|     # Get the date | ||||
|     date = "{} - ".format(datetime.datetime.now().strftime("%Y/%m/%d %H:%M:%S.%f")) | ||||
|     endc = end() | ||||
|  | ||||
|     # Continuation | ||||
|     if state == "c": | ||||
|         date = "" | ||||
|         colour = "" | ||||
|         prompt = "    " | ||||
|     # OK | ||||
|     elif state == "o": | ||||
|         colour = green() | ||||
|         prompt = ">>> " | ||||
|     # Error | ||||
|     elif state == "e": | ||||
|         colour = red() | ||||
|         prompt = ">>> " | ||||
|     # Warning | ||||
|     elif state == "w": | ||||
|         colour = yellow() | ||||
|         prompt = ">>> " | ||||
|     # Tick | ||||
|     elif state == "t": | ||||
|         colour = purple() | ||||
|         prompt = ">>> " | ||||
|     # Information | ||||
|     elif state == "i": | ||||
|         colour = blue() | ||||
|         prompt = ">>> " | ||||
|     else: | ||||
|         colour = bold() | ||||
|         prompt = ">>> " | ||||
|  | ||||
|     # Append space to prefix | ||||
|     if prefix != "": | ||||
|         prefix = prefix + " " | ||||
|  | ||||
|     print(colour + prompt + endc + date + prefix + message) | ||||
| @@ -1,116 +0,0 @@ | ||||
| #!/usr/bin/env python3 | ||||
|  | ||||
| # cluster.py - PVC CLI client function library, cluster management | ||||
| # Part of the Parallel Virtual Cluster (PVC) system | ||||
| # | ||||
| #    Copyright (C) 2018-2022 Joshua M. Boniface <joshua@boniface.me> | ||||
| # | ||||
| #    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, version 3. | ||||
| # | ||||
| #    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/>. | ||||
| # | ||||
| ############################################################################### | ||||
|  | ||||
| import json | ||||
|  | ||||
| from pvc.lib.common import call_api | ||||
|  | ||||
|  | ||||
| def initialize(config, overwrite=False): | ||||
|     """ | ||||
|     Initialize the PVC cluster | ||||
|  | ||||
|     API endpoint: GET /api/v1/initialize | ||||
|     API arguments: overwrite, yes-i-really-mean-it | ||||
|     API schema: {json_data_object} | ||||
|     """ | ||||
|     params = {"yes-i-really-mean-it": "yes", "overwrite": overwrite} | ||||
|     response = call_api(config, "post", "/initialize", params=params) | ||||
|  | ||||
|     if response.status_code == 200: | ||||
|         retstatus = True | ||||
|     else: | ||||
|         retstatus = False | ||||
|  | ||||
|     return retstatus, response.json().get("message", "") | ||||
|  | ||||
|  | ||||
| def backup(config): | ||||
|     """ | ||||
|     Get a JSON backup of the cluster | ||||
|  | ||||
|     API endpoint: GET /api/v1/backup | ||||
|     API arguments: | ||||
|     API schema: {json_data_object} | ||||
|     """ | ||||
|     response = call_api(config, "get", "/backup") | ||||
|  | ||||
|     if response.status_code == 200: | ||||
|         return True, response.json() | ||||
|     else: | ||||
|         return False, response.json().get("message", "") | ||||
|  | ||||
|  | ||||
| def restore(config, cluster_data): | ||||
|     """ | ||||
|     Restore a JSON backup to the cluster | ||||
|  | ||||
|     API endpoint: POST /api/v1/restore | ||||
|     API arguments: yes-i-really-mean-it | ||||
|     API schema: {json_data_object} | ||||
|     """ | ||||
|     cluster_data_json = json.dumps(cluster_data) | ||||
|  | ||||
|     params = {"yes-i-really-mean-it": "yes"} | ||||
|     data = {"cluster_data": cluster_data_json} | ||||
|     response = call_api(config, "post", "/restore", params=params, data=data) | ||||
|  | ||||
|     if response.status_code == 200: | ||||
|         retstatus = True | ||||
|     else: | ||||
|         retstatus = False | ||||
|  | ||||
|     return retstatus, response.json().get("message", "") | ||||
|  | ||||
|  | ||||
| def maintenance_mode(config, state): | ||||
|     """ | ||||
|     Enable or disable PVC cluster maintenance mode | ||||
|  | ||||
|     API endpoint: POST /api/v1/status | ||||
|     API arguments: {state}={state} | ||||
|     API schema: {json_data_object} | ||||
|     """ | ||||
|     params = {"state": state} | ||||
|     response = call_api(config, "post", "/status", params=params) | ||||
|  | ||||
|     if response.status_code == 200: | ||||
|         retstatus = True | ||||
|     else: | ||||
|         retstatus = False | ||||
|  | ||||
|     return retstatus, response.json().get("message", "") | ||||
|  | ||||
|  | ||||
| def get_info(config): | ||||
|     """ | ||||
|     Get status of the PVC cluster | ||||
|  | ||||
|     API endpoint: GET /api/v1/status | ||||
|     API arguments: | ||||
|     API schema: {json_data_object} | ||||
|     """ | ||||
|     response = call_api(config, "get", "/status") | ||||
|  | ||||
|     if response.status_code == 200: | ||||
|         return True, response.json() | ||||
|     else: | ||||
|         return False, response.json().get("message", "") | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,706 +0,0 @@ | ||||
| #!/usr/bin/env python3 | ||||
|  | ||||
| # node.py - PVC CLI client function library, node management | ||||
| # Part of the Parallel Virtual Cluster (PVC) system | ||||
| # | ||||
| #    Copyright (C) 2018-2022 Joshua M. Boniface <joshua@boniface.me> | ||||
| # | ||||
| #    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, version 3. | ||||
| # | ||||
| #    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/>. | ||||
| # | ||||
| ############################################################################### | ||||
|  | ||||
| import time | ||||
|  | ||||
| import pvc.lib.ansiprint as ansiprint | ||||
| from pvc.lib.common import call_api | ||||
|  | ||||
|  | ||||
| # | ||||
| # Primary functions | ||||
| # | ||||
| def node_coordinator_state(config, node, action): | ||||
|     """ | ||||
|     Set node coordinator state state (primary/secondary) | ||||
|  | ||||
|     API endpoint: POST /api/v1/node/{node}/coordinator-state | ||||
|     API arguments: action={action} | ||||
|     API schema: {"message": "{data}"} | ||||
|     """ | ||||
|     params = {"state": action} | ||||
|     response = call_api( | ||||
|         config, | ||||
|         "post", | ||||
|         "/node/{node}/coordinator-state".format(node=node), | ||||
|         params=params, | ||||
|     ) | ||||
|  | ||||
|     if response.status_code == 200: | ||||
|         retstatus = True | ||||
|     else: | ||||
|         retstatus = False | ||||
|  | ||||
|     return retstatus, response.json().get("message", "") | ||||
|  | ||||
|  | ||||
| def node_domain_state(config, node, action): | ||||
|     """ | ||||
|     Set node domain state state (flush/ready) | ||||
|  | ||||
|     API endpoint: POST /api/v1/node/{node}/domain-state | ||||
|     API arguments: action={action}, wait={wait} | ||||
|     API schema: {"message": "{data}"} | ||||
|     """ | ||||
|     params = {"state": action} | ||||
|     response = call_api( | ||||
|         config, "post", "/node/{node}/domain-state".format(node=node), params=params | ||||
|     ) | ||||
|  | ||||
|     if response.status_code == 200: | ||||
|         retstatus = True | ||||
|     else: | ||||
|         retstatus = False | ||||
|  | ||||
|     return retstatus, response.json().get("message", "") | ||||
|  | ||||
|  | ||||
| def view_node_log(config, node, lines=100): | ||||
|     """ | ||||
|     Return node log lines from the API (and display them in a pager in the main CLI) | ||||
|  | ||||
|     API endpoint: GET /node/{node}/log | ||||
|     API arguments: lines={lines} | ||||
|     API schema: {"name":"{node}","data":"{node_log}"} | ||||
|     """ | ||||
|     params = {"lines": lines} | ||||
|     response = call_api( | ||||
|         config, "get", "/node/{node}/log".format(node=node), params=params | ||||
|     ) | ||||
|  | ||||
|     if response.status_code != 200: | ||||
|         return False, response.json().get("message", "") | ||||
|  | ||||
|     node_log = response.json()["data"] | ||||
|  | ||||
|     # Shrink the log buffer to length lines | ||||
|     shrunk_log = node_log.split("\n")[-lines:] | ||||
|     loglines = "\n".join(shrunk_log) | ||||
|  | ||||
|     return True, loglines | ||||
|  | ||||
|  | ||||
| def follow_node_log(config, node, lines=10): | ||||
|     """ | ||||
|     Return and follow node log lines from the API | ||||
|  | ||||
|     API endpoint: GET /node/{node}/log | ||||
|     API arguments: lines={lines} | ||||
|     API schema: {"name":"{nodename}","data":"{node_log}"} | ||||
|     """ | ||||
|     # We always grab 200 to match the follow call, but only _show_ `lines` number | ||||
|     params = {"lines": 200} | ||||
|     response = call_api( | ||||
|         config, "get", "/node/{node}/log".format(node=node), params=params | ||||
|     ) | ||||
|  | ||||
|     if response.status_code != 200: | ||||
|         return False, response.json().get("message", "") | ||||
|  | ||||
|     # Shrink the log buffer to length lines | ||||
|     node_log = response.json()["data"] | ||||
|     shrunk_log = node_log.split("\n")[-int(lines) :] | ||||
|     loglines = "\n".join(shrunk_log) | ||||
|  | ||||
|     # Print the initial data and begin following | ||||
|     print(loglines, end="") | ||||
|     print("\n", end="") | ||||
|  | ||||
|     while True: | ||||
|         # Grab the next line set (200 is a reasonable number of lines per half-second; any more are skipped) | ||||
|         try: | ||||
|             params = {"lines": 200} | ||||
|             response = call_api( | ||||
|                 config, "get", "/node/{node}/log".format(node=node), params=params | ||||
|             ) | ||||
|             new_node_log = response.json()["data"] | ||||
|         except Exception: | ||||
|             break | ||||
|         # Split the new and old log strings into constitutent lines | ||||
|         old_node_loglines = node_log.split("\n") | ||||
|         new_node_loglines = new_node_log.split("\n") | ||||
|  | ||||
|         # Set the node log to the new log value for the next iteration | ||||
|         node_log = new_node_log | ||||
|  | ||||
|         # Get the difference between the two sets of lines | ||||
|         old_node_loglines_set = set(old_node_loglines) | ||||
|         diff_node_loglines = [ | ||||
|             x for x in new_node_loglines if x not in old_node_loglines_set | ||||
|         ] | ||||
|  | ||||
|         # If there's a difference, print it out | ||||
|         if len(diff_node_loglines) > 0: | ||||
|             print("\n".join(diff_node_loglines), end="") | ||||
|             print("\n", end="") | ||||
|  | ||||
|         # Wait half a second | ||||
|         time.sleep(0.5) | ||||
|  | ||||
|     return True, "" | ||||
|  | ||||
|  | ||||
| def node_info(config, node): | ||||
|     """ | ||||
|     Get information about node | ||||
|  | ||||
|     API endpoint: GET /api/v1/node/{node} | ||||
|     API arguments: | ||||
|     API schema: {json_data_object} | ||||
|     """ | ||||
|     response = call_api(config, "get", "/node/{node}".format(node=node)) | ||||
|  | ||||
|     if response.status_code == 200: | ||||
|         if isinstance(response.json(), list) and len(response.json()) != 1: | ||||
|             # No exact match, return not found | ||||
|             return False, "Node not found." | ||||
|         else: | ||||
|             # Return a single instance if the response is a list | ||||
|             if isinstance(response.json(), list): | ||||
|                 return True, response.json()[0] | ||||
|             # This shouldn't happen, but is here just in case | ||||
|             else: | ||||
|                 return True, response.json() | ||||
|     else: | ||||
|         return False, response.json().get("message", "") | ||||
|  | ||||
|  | ||||
| def node_list( | ||||
|     config, limit, target_daemon_state, target_coordinator_state, target_domain_state | ||||
| ): | ||||
|     """ | ||||
|     Get list information about nodes (limited by {limit}) | ||||
|  | ||||
|     API endpoint: GET /api/v1/node | ||||
|     API arguments: limit={limit} | ||||
|     API schema: [{json_data_object},{json_data_object},etc.] | ||||
|     """ | ||||
|     params = dict() | ||||
|     if limit: | ||||
|         params["limit"] = limit | ||||
|     if target_daemon_state: | ||||
|         params["daemon_state"] = target_daemon_state | ||||
|     if target_coordinator_state: | ||||
|         params["coordinator_state"] = target_coordinator_state | ||||
|     if target_domain_state: | ||||
|         params["domain_state"] = target_domain_state | ||||
|  | ||||
|     response = call_api(config, "get", "/node", params=params) | ||||
|  | ||||
|     if response.status_code == 200: | ||||
|         return True, response.json() | ||||
|     else: | ||||
|         return False, response.json().get("message", "") | ||||
|  | ||||
|  | ||||
| # | ||||
| # Output display functions | ||||
| # | ||||
| def getOutputColours(node_information): | ||||
|     node_health = node_information.get("health", "N/A") | ||||
|     if isinstance(node_health, int): | ||||
|         if node_health <= 50: | ||||
|             health_colour = ansiprint.red() | ||||
|         elif node_health <= 90: | ||||
|             health_colour = ansiprint.yellow() | ||||
|         elif node_health <= 100: | ||||
|             health_colour = ansiprint.green() | ||||
|         else: | ||||
|             health_colour = ansiprint.blue() | ||||
|     else: | ||||
|         health_colour = ansiprint.blue() | ||||
|  | ||||
|     if node_information["daemon_state"] == "run": | ||||
|         daemon_state_colour = ansiprint.green() | ||||
|     elif node_information["daemon_state"] == "stop": | ||||
|         daemon_state_colour = ansiprint.red() | ||||
|     elif node_information["daemon_state"] == "shutdown": | ||||
|         daemon_state_colour = ansiprint.yellow() | ||||
|     elif node_information["daemon_state"] == "init": | ||||
|         daemon_state_colour = ansiprint.yellow() | ||||
|     elif node_information["daemon_state"] == "dead": | ||||
|         daemon_state_colour = ansiprint.red() + ansiprint.bold() | ||||
|     else: | ||||
|         daemon_state_colour = ansiprint.blue() | ||||
|  | ||||
|     if node_information["coordinator_state"] == "primary": | ||||
|         coordinator_state_colour = ansiprint.green() | ||||
|     elif node_information["coordinator_state"] == "secondary": | ||||
|         coordinator_state_colour = ansiprint.blue() | ||||
|     else: | ||||
|         coordinator_state_colour = ansiprint.cyan() | ||||
|  | ||||
|     if node_information["domain_state"] == "ready": | ||||
|         domain_state_colour = ansiprint.green() | ||||
|     else: | ||||
|         domain_state_colour = ansiprint.blue() | ||||
|  | ||||
|     if node_information["memory"]["allocated"] > node_information["memory"]["total"]: | ||||
|         mem_allocated_colour = ansiprint.yellow() | ||||
|     else: | ||||
|         mem_allocated_colour = "" | ||||
|  | ||||
|     if node_information["memory"]["provisioned"] > node_information["memory"]["total"]: | ||||
|         mem_provisioned_colour = ansiprint.yellow() | ||||
|     else: | ||||
|         mem_provisioned_colour = "" | ||||
|  | ||||
|     return ( | ||||
|         health_colour, | ||||
|         daemon_state_colour, | ||||
|         coordinator_state_colour, | ||||
|         domain_state_colour, | ||||
|         mem_allocated_colour, | ||||
|         mem_provisioned_colour, | ||||
|     ) | ||||
|  | ||||
|  | ||||
| def format_info(config, node_information, long_output): | ||||
|     ( | ||||
|         health_colour, | ||||
|         daemon_state_colour, | ||||
|         coordinator_state_colour, | ||||
|         domain_state_colour, | ||||
|         mem_allocated_colour, | ||||
|         mem_provisioned_colour, | ||||
|     ) = getOutputColours(node_information) | ||||
|  | ||||
|     # Format a nice output; do this line-by-line then concat the elements at the end | ||||
|     ainformation = [] | ||||
|     # Basic information | ||||
|     ainformation.append( | ||||
|         "{}Name:{}                  {}".format( | ||||
|             ansiprint.purple(), | ||||
|             ansiprint.end(), | ||||
|             node_information["name"], | ||||
|         ) | ||||
|     ) | ||||
|     ainformation.append( | ||||
|         "{}PVC Version:{}           {}".format( | ||||
|             ansiprint.purple(), | ||||
|             ansiprint.end(), | ||||
|             node_information["pvc_version"], | ||||
|         ) | ||||
|     ) | ||||
|  | ||||
|     node_health = node_information.get("health", "N/A") | ||||
|     if isinstance(node_health, int): | ||||
|         node_health_text = f"{node_health}%" | ||||
|     else: | ||||
|         node_health_text = node_health | ||||
|     ainformation.append( | ||||
|         "{}Health:{}                {}{}{}".format( | ||||
|             ansiprint.purple(), | ||||
|             ansiprint.end(), | ||||
|             health_colour, | ||||
|             node_health_text, | ||||
|             ansiprint.end(), | ||||
|         ) | ||||
|     ) | ||||
|  | ||||
|     node_health_details = node_information.get("health_details", []) | ||||
|     if long_output: | ||||
|         node_health_messages = "\n                       ".join( | ||||
|             [f"{plugin['name']}: {plugin['message']}" for plugin in node_health_details] | ||||
|         ) | ||||
|     else: | ||||
|         node_health_messages = "\n                       ".join( | ||||
|             [ | ||||
|                 f"{plugin['name']}: {plugin['message']}" | ||||
|                 for plugin in node_health_details | ||||
|                 if int(plugin.get("health_delta", 0)) > 0 | ||||
|             ] | ||||
|         ) | ||||
|  | ||||
|     if len(node_health_messages) > 0: | ||||
|         ainformation.append( | ||||
|             "{}Health Plugin Details:{} {}".format( | ||||
|                 ansiprint.purple(), ansiprint.end(), node_health_messages | ||||
|             ) | ||||
|         ) | ||||
|     ainformation.append("") | ||||
|  | ||||
|     ainformation.append( | ||||
|         "{}Daemon State:{}          {}{}{}".format( | ||||
|             ansiprint.purple(), | ||||
|             ansiprint.end(), | ||||
|             daemon_state_colour, | ||||
|             node_information["daemon_state"], | ||||
|             ansiprint.end(), | ||||
|         ) | ||||
|     ) | ||||
|     ainformation.append( | ||||
|         "{}Coordinator State:{}     {}{}{}".format( | ||||
|             ansiprint.purple(), | ||||
|             ansiprint.end(), | ||||
|             coordinator_state_colour, | ||||
|             node_information["coordinator_state"], | ||||
|             ansiprint.end(), | ||||
|         ) | ||||
|     ) | ||||
|     ainformation.append( | ||||
|         "{}Domain State:{}          {}{}{}".format( | ||||
|             ansiprint.purple(), | ||||
|             ansiprint.end(), | ||||
|             domain_state_colour, | ||||
|             node_information["domain_state"], | ||||
|             ansiprint.end(), | ||||
|         ) | ||||
|     ) | ||||
|     if long_output: | ||||
|         ainformation.append("") | ||||
|         ainformation.append( | ||||
|             "{}Architecture:{}          {}".format( | ||||
|                 ansiprint.purple(), ansiprint.end(), node_information["arch"] | ||||
|             ) | ||||
|         ) | ||||
|         ainformation.append( | ||||
|             "{}Operating System:{}      {}".format( | ||||
|                 ansiprint.purple(), ansiprint.end(), node_information["os"] | ||||
|             ) | ||||
|         ) | ||||
|         ainformation.append( | ||||
|             "{}Kernel Version:{}        {}".format( | ||||
|                 ansiprint.purple(), ansiprint.end(), node_information["kernel"] | ||||
|             ) | ||||
|         ) | ||||
|     ainformation.append("") | ||||
|     ainformation.append( | ||||
|         "{}Active VM Count:{}       {}".format( | ||||
|             ansiprint.purple(), ansiprint.end(), node_information["domains_count"] | ||||
|         ) | ||||
|     ) | ||||
|     ainformation.append( | ||||
|         "{}Host CPUs:{}             {}".format( | ||||
|             ansiprint.purple(), ansiprint.end(), node_information["vcpu"]["total"] | ||||
|         ) | ||||
|     ) | ||||
|     ainformation.append( | ||||
|         "{}vCPUs:{}                 {}".format( | ||||
|             ansiprint.purple(), ansiprint.end(), node_information["vcpu"]["allocated"] | ||||
|         ) | ||||
|     ) | ||||
|     ainformation.append( | ||||
|         "{}Load:{}                  {}".format( | ||||
|             ansiprint.purple(), ansiprint.end(), node_information["load"] | ||||
|         ) | ||||
|     ) | ||||
|     ainformation.append( | ||||
|         "{}Total RAM (MiB):{}       {}".format( | ||||
|             ansiprint.purple(), ansiprint.end(), node_information["memory"]["total"] | ||||
|         ) | ||||
|     ) | ||||
|     ainformation.append( | ||||
|         "{}Used RAM (MiB):{}        {}".format( | ||||
|             ansiprint.purple(), ansiprint.end(), node_information["memory"]["used"] | ||||
|         ) | ||||
|     ) | ||||
|     ainformation.append( | ||||
|         "{}Free RAM (MiB):{}        {}".format( | ||||
|             ansiprint.purple(), ansiprint.end(), node_information["memory"]["free"] | ||||
|         ) | ||||
|     ) | ||||
|     ainformation.append( | ||||
|         "{}Allocated RAM (MiB):{}   {}{}{}".format( | ||||
|             ansiprint.purple(), | ||||
|             ansiprint.end(), | ||||
|             mem_allocated_colour, | ||||
|             node_information["memory"]["allocated"], | ||||
|             ansiprint.end(), | ||||
|         ) | ||||
|     ) | ||||
|     ainformation.append( | ||||
|         "{}Provisioned RAM (MiB):{} {}{}{}".format( | ||||
|             ansiprint.purple(), | ||||
|             ansiprint.end(), | ||||
|             mem_provisioned_colour, | ||||
|             node_information["memory"]["provisioned"], | ||||
|             ansiprint.end(), | ||||
|         ) | ||||
|     ) | ||||
|  | ||||
|     # Join it all together | ||||
|     ainformation.append("") | ||||
|     return "\n".join(ainformation) | ||||
|  | ||||
|  | ||||
| def format_list(config, node_list): | ||||
|     if node_list == "Node not found.": | ||||
|         return node_list | ||||
|  | ||||
|     node_list_output = [] | ||||
|  | ||||
|     # Determine optimal column widths | ||||
|     node_name_length = 5 | ||||
|     pvc_version_length = 8 | ||||
|     health_length = 7 | ||||
|     daemon_state_length = 7 | ||||
|     coordinator_state_length = 12 | ||||
|     domain_state_length = 7 | ||||
|     domains_count_length = 4 | ||||
|     cpu_count_length = 6 | ||||
|     load_length = 5 | ||||
|     mem_total_length = 6 | ||||
|     mem_used_length = 5 | ||||
|     mem_free_length = 5 | ||||
|     mem_alloc_length = 6 | ||||
|     mem_prov_length = 5 | ||||
|     for node_information in node_list: | ||||
|         # node_name column | ||||
|         _node_name_length = len(node_information["name"]) + 1 | ||||
|         if _node_name_length > node_name_length: | ||||
|             node_name_length = _node_name_length | ||||
|         # node_pvc_version column | ||||
|         _pvc_version_length = len(node_information.get("pvc_version", "N/A")) + 1 | ||||
|         if _pvc_version_length > pvc_version_length: | ||||
|             pvc_version_length = _pvc_version_length | ||||
|         # node_health column | ||||
|         node_health = node_information.get("health", "N/A") | ||||
|         if isinstance(node_health, int): | ||||
|             node_health_text = f"{node_health}%" | ||||
|         else: | ||||
|             node_health_text = node_health | ||||
|         _health_length = len(node_health_text) + 1 | ||||
|         if _health_length > health_length: | ||||
|             health_length = _health_length | ||||
|         # daemon_state column | ||||
|         _daemon_state_length = len(node_information["daemon_state"]) + 1 | ||||
|         if _daemon_state_length > daemon_state_length: | ||||
|             daemon_state_length = _daemon_state_length | ||||
|         # coordinator_state column | ||||
|         _coordinator_state_length = len(node_information["coordinator_state"]) + 1 | ||||
|         if _coordinator_state_length > coordinator_state_length: | ||||
|             coordinator_state_length = _coordinator_state_length | ||||
|         # domain_state column | ||||
|         _domain_state_length = len(node_information["domain_state"]) + 1 | ||||
|         if _domain_state_length > domain_state_length: | ||||
|             domain_state_length = _domain_state_length | ||||
|         # domains_count column | ||||
|         _domains_count_length = len(str(node_information["domains_count"])) + 1 | ||||
|         if _domains_count_length > domains_count_length: | ||||
|             domains_count_length = _domains_count_length | ||||
|         # cpu_count column | ||||
|         _cpu_count_length = len(str(node_information["cpu_count"])) + 1 | ||||
|         if _cpu_count_length > cpu_count_length: | ||||
|             cpu_count_length = _cpu_count_length | ||||
|         # load column | ||||
|         _load_length = len(str(node_information["load"])) + 1 | ||||
|         if _load_length > load_length: | ||||
|             load_length = _load_length | ||||
|         # mem_total column | ||||
|         _mem_total_length = len(str(node_information["memory"]["total"])) + 1 | ||||
|         if _mem_total_length > mem_total_length: | ||||
|             mem_total_length = _mem_total_length | ||||
|         # mem_used column | ||||
|         _mem_used_length = len(str(node_information["memory"]["used"])) + 1 | ||||
|         if _mem_used_length > mem_used_length: | ||||
|             mem_used_length = _mem_used_length | ||||
|         # mem_free column | ||||
|         _mem_free_length = len(str(node_information["memory"]["free"])) + 1 | ||||
|         if _mem_free_length > mem_free_length: | ||||
|             mem_free_length = _mem_free_length | ||||
|         # mem_alloc column | ||||
|         _mem_alloc_length = len(str(node_information["memory"]["allocated"])) + 1 | ||||
|         if _mem_alloc_length > mem_alloc_length: | ||||
|             mem_alloc_length = _mem_alloc_length | ||||
|  | ||||
|         # mem_prov column | ||||
|         _mem_prov_length = len(str(node_information["memory"]["provisioned"])) + 1 | ||||
|         if _mem_prov_length > mem_prov_length: | ||||
|             mem_prov_length = _mem_prov_length | ||||
|  | ||||
|     # Format the string (header) | ||||
|     node_list_output.append( | ||||
|         "{bold}{node_header: <{node_header_length}} {state_header: <{state_header_length}} {resource_header: <{resource_header_length}} {memory_header: <{memory_header_length}}{end_bold}".format( | ||||
|             node_header_length=node_name_length | ||||
|             + pvc_version_length | ||||
|             + health_length | ||||
|             + 2, | ||||
|             state_header_length=daemon_state_length | ||||
|             + coordinator_state_length | ||||
|             + domain_state_length | ||||
|             + 2, | ||||
|             resource_header_length=domains_count_length | ||||
|             + cpu_count_length | ||||
|             + load_length | ||||
|             + 2, | ||||
|             memory_header_length=mem_total_length | ||||
|             + mem_used_length | ||||
|             + mem_free_length | ||||
|             + mem_alloc_length | ||||
|             + mem_prov_length | ||||
|             + 4, | ||||
|             bold=ansiprint.bold(), | ||||
|             end_bold=ansiprint.end(), | ||||
|             node_header="Nodes " | ||||
|             + "".join( | ||||
|                 [ | ||||
|                     "-" | ||||
|                     for _ in range( | ||||
|                         6, node_name_length + pvc_version_length + health_length + 1 | ||||
|                     ) | ||||
|                 ] | ||||
|             ), | ||||
|             state_header="States " | ||||
|             + "".join( | ||||
|                 [ | ||||
|                     "-" | ||||
|                     for _ in range( | ||||
|                         7, | ||||
|                         daemon_state_length | ||||
|                         + coordinator_state_length | ||||
|                         + domain_state_length | ||||
|                         + 1, | ||||
|                     ) | ||||
|                 ] | ||||
|             ), | ||||
|             resource_header="Resources " | ||||
|             + "".join( | ||||
|                 [ | ||||
|                     "-" | ||||
|                     for _ in range( | ||||
|                         10, domains_count_length + cpu_count_length + load_length + 1 | ||||
|                     ) | ||||
|                 ] | ||||
|             ), | ||||
|             memory_header="Memory (M) " | ||||
|             + "".join( | ||||
|                 [ | ||||
|                     "-" | ||||
|                     for _ in range( | ||||
|                         11, | ||||
|                         mem_total_length | ||||
|                         + mem_used_length | ||||
|                         + mem_free_length | ||||
|                         + mem_alloc_length | ||||
|                         + mem_prov_length | ||||
|                         + 3, | ||||
|                     ) | ||||
|                 ] | ||||
|             ), | ||||
|         ) | ||||
|     ) | ||||
|  | ||||
|     node_list_output.append( | ||||
|         "{bold}{node_name: <{node_name_length}} {node_pvc_version: <{pvc_version_length}} {node_health: <{health_length}} \ | ||||
| {daemon_state_colour}{node_daemon_state: <{daemon_state_length}}{end_colour} {coordinator_state_colour}{node_coordinator_state: <{coordinator_state_length}}{end_colour} {domain_state_colour}{node_domain_state: <{domain_state_length}}{end_colour} \ | ||||
| {node_domains_count: <{domains_count_length}} {node_cpu_count: <{cpu_count_length}} {node_load: <{load_length}} \ | ||||
| {node_mem_total: <{mem_total_length}} {node_mem_used: <{mem_used_length}} {node_mem_free: <{mem_free_length}} {node_mem_allocated: <{mem_alloc_length}} {node_mem_provisioned: <{mem_prov_length}}{end_bold}".format( | ||||
|             node_name_length=node_name_length, | ||||
|             pvc_version_length=pvc_version_length, | ||||
|             health_length=health_length, | ||||
|             daemon_state_length=daemon_state_length, | ||||
|             coordinator_state_length=coordinator_state_length, | ||||
|             domain_state_length=domain_state_length, | ||||
|             domains_count_length=domains_count_length, | ||||
|             cpu_count_length=cpu_count_length, | ||||
|             load_length=load_length, | ||||
|             mem_total_length=mem_total_length, | ||||
|             mem_used_length=mem_used_length, | ||||
|             mem_free_length=mem_free_length, | ||||
|             mem_alloc_length=mem_alloc_length, | ||||
|             mem_prov_length=mem_prov_length, | ||||
|             bold=ansiprint.bold(), | ||||
|             end_bold=ansiprint.end(), | ||||
|             daemon_state_colour="", | ||||
|             coordinator_state_colour="", | ||||
|             domain_state_colour="", | ||||
|             end_colour="", | ||||
|             node_name="Name", | ||||
|             node_pvc_version="Version", | ||||
|             node_health="Health", | ||||
|             node_daemon_state="Daemon", | ||||
|             node_coordinator_state="Coordinator", | ||||
|             node_domain_state="Domain", | ||||
|             node_domains_count="VMs", | ||||
|             node_cpu_count="vCPUs", | ||||
|             node_load="Load", | ||||
|             node_mem_total="Total", | ||||
|             node_mem_used="Used", | ||||
|             node_mem_free="Free", | ||||
|             node_mem_allocated="Alloc", | ||||
|             node_mem_provisioned="Prov", | ||||
|         ) | ||||
|     ) | ||||
|  | ||||
|     # Format the string (elements) | ||||
|     for node_information in sorted(node_list, key=lambda n: n["name"]): | ||||
|         ( | ||||
|             health_colour, | ||||
|             daemon_state_colour, | ||||
|             coordinator_state_colour, | ||||
|             domain_state_colour, | ||||
|             mem_allocated_colour, | ||||
|             mem_provisioned_colour, | ||||
|         ) = getOutputColours(node_information) | ||||
|  | ||||
|         node_health = node_information.get("health", "N/A") | ||||
|         if isinstance(node_health, int): | ||||
|             node_health_text = f"{node_health}%" | ||||
|         else: | ||||
|             node_health_text = node_health | ||||
|  | ||||
|         node_list_output.append( | ||||
|             "{bold}{node_name: <{node_name_length}} {node_pvc_version: <{pvc_version_length}} {health_colour}{node_health: <{health_length}}{end_colour} \ | ||||
| {daemon_state_colour}{node_daemon_state: <{daemon_state_length}}{end_colour} {coordinator_state_colour}{node_coordinator_state: <{coordinator_state_length}}{end_colour} {domain_state_colour}{node_domain_state: <{domain_state_length}}{end_colour} \ | ||||
| {node_domains_count: <{domains_count_length}} {node_cpu_count: <{cpu_count_length}} {node_load: <{load_length}} \ | ||||
| {node_mem_total: <{mem_total_length}} {node_mem_used: <{mem_used_length}} {node_mem_free: <{mem_free_length}} {mem_allocated_colour}{node_mem_allocated: <{mem_alloc_length}}{end_colour} {mem_provisioned_colour}{node_mem_provisioned: <{mem_prov_length}}{end_colour}{end_bold}".format( | ||||
|                 node_name_length=node_name_length, | ||||
|                 pvc_version_length=pvc_version_length, | ||||
|                 health_length=health_length, | ||||
|                 daemon_state_length=daemon_state_length, | ||||
|                 coordinator_state_length=coordinator_state_length, | ||||
|                 domain_state_length=domain_state_length, | ||||
|                 domains_count_length=domains_count_length, | ||||
|                 cpu_count_length=cpu_count_length, | ||||
|                 load_length=load_length, | ||||
|                 mem_total_length=mem_total_length, | ||||
|                 mem_used_length=mem_used_length, | ||||
|                 mem_free_length=mem_free_length, | ||||
|                 mem_alloc_length=mem_alloc_length, | ||||
|                 mem_prov_length=mem_prov_length, | ||||
|                 bold="", | ||||
|                 end_bold="", | ||||
|                 health_colour=health_colour, | ||||
|                 daemon_state_colour=daemon_state_colour, | ||||
|                 coordinator_state_colour=coordinator_state_colour, | ||||
|                 domain_state_colour=domain_state_colour, | ||||
|                 mem_allocated_colour=mem_allocated_colour, | ||||
|                 mem_provisioned_colour=mem_allocated_colour, | ||||
|                 end_colour=ansiprint.end(), | ||||
|                 node_name=node_information["name"], | ||||
|                 node_pvc_version=node_information.get("pvc_version", "N/A"), | ||||
|                 node_health=node_health_text, | ||||
|                 node_daemon_state=node_information["daemon_state"], | ||||
|                 node_coordinator_state=node_information["coordinator_state"], | ||||
|                 node_domain_state=node_information["domain_state"], | ||||
|                 node_domains_count=node_information["domains_count"], | ||||
|                 node_cpu_count=node_information["vcpu"]["allocated"], | ||||
|                 node_load=node_information["load"], | ||||
|                 node_mem_total=node_information["memory"]["total"], | ||||
|                 node_mem_used=node_information["memory"]["used"], | ||||
|                 node_mem_free=node_information["memory"]["free"], | ||||
|                 node_mem_allocated=node_information["memory"]["allocated"], | ||||
|                 node_mem_provisioned=node_information["memory"]["provisioned"], | ||||
|             ) | ||||
|         ) | ||||
|  | ||||
|     return "\n".join(node_list_output) | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,102 +0,0 @@ | ||||
| #!/usr/bin/env python3 | ||||
|  | ||||
| # zkhandler.py - Secure versioned ZooKeeper updates | ||||
| # Part of the Parallel Virtual Cluster (PVC) system | ||||
| # | ||||
| #    Copyright (C) 2018-2022 Joshua M. Boniface <joshua@boniface.me> | ||||
| # | ||||
| #    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, version 3. | ||||
| # | ||||
| #    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/>. | ||||
| # | ||||
| ############################################################################### | ||||
|  | ||||
| import uuid | ||||
|  | ||||
|  | ||||
| # Exists function | ||||
| def exists(zk_conn, key): | ||||
|     stat = zk_conn.exists(key) | ||||
|     if stat: | ||||
|         return True | ||||
|     else: | ||||
|         return False | ||||
|  | ||||
|  | ||||
| # Child list function | ||||
| def listchildren(zk_conn, key): | ||||
|     children = zk_conn.get_children(key) | ||||
|     return children | ||||
|  | ||||
|  | ||||
| # Delete key function | ||||
| def deletekey(zk_conn, key, recursive=True): | ||||
|     zk_conn.delete(key, recursive=recursive) | ||||
|  | ||||
|  | ||||
| # Data read function | ||||
| def readdata(zk_conn, key): | ||||
|     data_raw = zk_conn.get(key) | ||||
|     data = data_raw[0].decode("utf8") | ||||
|     return data | ||||
|  | ||||
|  | ||||
| # Data write function | ||||
| def writedata(zk_conn, kv): | ||||
|     # Start up a transaction | ||||
|     zk_transaction = zk_conn.transaction() | ||||
|  | ||||
|     # Proceed one KV pair at a time | ||||
|     for key in sorted(kv): | ||||
|         data = kv[key] | ||||
|  | ||||
|         # Check if this key already exists or not | ||||
|         if not zk_conn.exists(key): | ||||
|             # We're creating a new key | ||||
|             zk_transaction.create(key, str(data).encode("utf8")) | ||||
|         else: | ||||
|             # We're updating a key with version validation | ||||
|             orig_data = zk_conn.get(key) | ||||
|             version = orig_data[1].version | ||||
|  | ||||
|             # Set what we expect the new version to be | ||||
|             new_version = version + 1 | ||||
|  | ||||
|             # Update the data | ||||
|             zk_transaction.set_data(key, str(data).encode("utf8")) | ||||
|  | ||||
|             # Set up the check | ||||
|             try: | ||||
|                 zk_transaction.check(key, new_version) | ||||
|             except TypeError: | ||||
|                 print('Zookeeper key "{}" does not match expected version'.format(key)) | ||||
|                 return False | ||||
|  | ||||
|     # Commit the transaction | ||||
|     try: | ||||
|         zk_transaction.commit() | ||||
|         return True | ||||
|     except Exception: | ||||
|         return False | ||||
|  | ||||
|  | ||||
| # Write lock function | ||||
| def writelock(zk_conn, key): | ||||
|     lock_id = str(uuid.uuid1()) | ||||
|     lock = zk_conn.WriteLock("{}".format(key), lock_id) | ||||
|     return lock | ||||
|  | ||||
|  | ||||
| # Read lock function | ||||
| def readlock(zk_conn, key): | ||||
|     lock_id = str(uuid.uuid1()) | ||||
|     lock = zk_conn.ReadLock("{}".format(key), lock_id) | ||||
|     return lock | ||||
| @@ -37,13 +37,13 @@ from distutils.util import strtobool | ||||
| 
 | ||||
| from functools import wraps | ||||
| 
 | ||||
| import pvc.lib.ansiprint as ansiprint | ||||
| import pvc.lib.cluster as pvc_cluster | ||||
| import pvc.lib.node as pvc_node | ||||
| import pvc.lib.vm as pvc_vm | ||||
| import pvc.lib.network as pvc_network | ||||
| import pvc.lib.ceph as pvc_ceph | ||||
| import pvc.lib.provisioner as pvc_provisioner | ||||
| import pvc.cli_lib.ansiprint as ansiprint | ||||
| import pvc.cli_lib.cluster as pvc_cluster | ||||
| import pvc.cli_lib.node as pvc_node | ||||
| import pvc.cli_lib.vm as pvc_vm | ||||
| import pvc.cli_lib.network as pvc_network | ||||
| import pvc.cli_lib.ceph as pvc_ceph | ||||
| import pvc.cli_lib.provisioner as pvc_provisioner | ||||
| 
 | ||||
| 
 | ||||
| myhostname = socket.gethostname().split(".")[0] | ||||
| @@ -2,8 +2,8 @@ from setuptools import setup | ||||
|  | ||||
| setup( | ||||
|     name="pvc", | ||||
|     version="0.9.64", | ||||
|     packages=["pvc.cli", "pvc.lib"], | ||||
|     version="0.9.62", | ||||
|     packages=["pvc", "pvc.cli_lib"], | ||||
|     install_requires=[ | ||||
|         "Click", | ||||
|         "PyYAML", | ||||
| @@ -14,7 +14,7 @@ setup( | ||||
|     ], | ||||
|     entry_points={ | ||||
|         "console_scripts": [ | ||||
|             "pvc = pvc.cli.cli:cli", | ||||
|             "pvc = pvc.pvc:cli", | ||||
|         ], | ||||
|     }, | ||||
| ) | ||||
|   | ||||
| @@ -73,11 +73,6 @@ byte_unit_matrix = { | ||||
|     "G": 1024 * 1024 * 1024, | ||||
|     "T": 1024 * 1024 * 1024 * 1024, | ||||
|     "P": 1024 * 1024 * 1024 * 1024 * 1024, | ||||
|     "E": 1024 * 1024 * 1024 * 1024 * 1024 * 1024, | ||||
|     "Z": 1024 * 1024 * 1024 * 1024 * 1024 * 1024 * 1024, | ||||
|     "Y": 1024 * 1024 * 1024 * 1024 * 1024 * 1024 * 1024 * 1024, | ||||
|     "R": 1024 * 1024 * 1024 * 1024 * 1024 * 1024 * 1024 * 1024 * 1024, | ||||
|     "Q": 1024 * 1024 * 1024 * 1024 * 1024 * 1024 * 1024 * 1024 * 1024 * 1024, | ||||
| } | ||||
|  | ||||
| # Matrix of human-to-metric values | ||||
| @@ -88,11 +83,6 @@ ops_unit_matrix = { | ||||
|     "G": 1000 * 1000 * 1000, | ||||
|     "T": 1000 * 1000 * 1000 * 1000, | ||||
|     "P": 1000 * 1000 * 1000 * 1000 * 1000, | ||||
|     "E": 1000 * 1000 * 1000 * 1000 * 1000 * 1000, | ||||
|     "Z": 1000 * 1000 * 1000 * 1000 * 1000 * 1000 * 1000, | ||||
|     "Y": 1000 * 1000 * 1000 * 1000 * 1000 * 1000 * 1000 * 1000, | ||||
|     "R": 1000 * 1000 * 1000 * 1000 * 1000 * 1000 * 1000 * 1000 * 1000, | ||||
|     "Q": 1000 * 1000 * 1000 * 1000 * 1000 * 1000 * 1000 * 1000 * 1000 * 1000, | ||||
| } | ||||
|  | ||||
|  | ||||
| @@ -113,18 +103,14 @@ def format_bytes_tohuman(databytes): | ||||
|  | ||||
|  | ||||
| def format_bytes_fromhuman(datahuman): | ||||
|     if not re.search(r"[A-Za-z]+", datahuman): | ||||
|     # Trim off human-readable character | ||||
|     dataunit = str(datahuman)[-1] | ||||
|     datasize = int(str(datahuman)[:-1]) | ||||
|     if not re.match(r"[A-Z]", dataunit): | ||||
|         dataunit = "B" | ||||
|         datasize = int(datahuman) | ||||
|     else: | ||||
|         dataunit = str(re.match(r"[0-9]+([A-Za-z])[iBb]*", datahuman).group(1)) | ||||
|         datasize = int(re.match(r"([0-9]+)[A-Za-z]+", datahuman).group(1)) | ||||
|  | ||||
|     if byte_unit_matrix.get(dataunit): | ||||
|         databytes = datasize * byte_unit_matrix[dataunit] | ||||
|         return databytes | ||||
|     else: | ||||
|         return None | ||||
|     databytes = datasize * byte_unit_matrix[dataunit] | ||||
|     return databytes | ||||
|  | ||||
|  | ||||
| # Format ops sizes to/from human-readable units | ||||
| @@ -745,26 +731,22 @@ def getVolumeInformation(zkhandler, pool, volume): | ||||
|  | ||||
|  | ||||
| def add_volume(zkhandler, pool, name, size): | ||||
|     # Add 'B' if the volume is in bytes | ||||
|     if re.match(r"^[0-9]+$", size): | ||||
|         size = "{}B".format(size) | ||||
|  | ||||
|     # 1. Verify the size of the volume | ||||
|     pool_information = getPoolInformation(zkhandler, pool) | ||||
|     size_bytes = format_bytes_fromhuman(size) | ||||
|     if size_bytes is None: | ||||
|         return ( | ||||
|             False, | ||||
|             f"ERROR: Requested volume size '{size}' does not have a valid SI unit", | ||||
|         ) | ||||
|  | ||||
|     if size_bytes >= int(pool_information["stats"]["free_bytes"]): | ||||
|         return ( | ||||
|             False, | ||||
|             f"ERROR: Requested volume size '{format_bytes_tohuman(size_bytes)}' is greater than the available free space in the pool ('{format_bytes_tohuman(pool_information['stats']['free_bytes'])}')", | ||||
|             "ERROR: Requested volume size is greater than the available free space in the pool", | ||||
|         ) | ||||
|  | ||||
|     # 2. Create the volume | ||||
|     retcode, stdout, stderr = common.run_os_command( | ||||
|         "rbd create --size {} {}/{}".format( | ||||
|             format_bytes_tohuman(size_bytes), pool, name | ||||
|         ) | ||||
|         "rbd create --size {} {}/{}".format(size, pool, name) | ||||
|     ) | ||||
|     if retcode: | ||||
|         return False, 'ERROR: Failed to create RBD volume "{}": {}'.format(name, stderr) | ||||
| @@ -784,9 +766,7 @@ def add_volume(zkhandler, pool, name, size): | ||||
|         ] | ||||
|     ) | ||||
|  | ||||
|     return True, 'Created RBD volume "{}" of size "{}" in pool "{}".'.format( | ||||
|         name, format_bytes_tohuman(size_bytes), pool | ||||
|     ) | ||||
|     return True, 'Created RBD volume "{}/{}" ({}).'.format(pool, name, size) | ||||
|  | ||||
|  | ||||
| def clone_volume(zkhandler, pool, name_src, name_new): | ||||
| @@ -833,32 +813,28 @@ def resize_volume(zkhandler, pool, name, size): | ||||
|             name, pool | ||||
|         ) | ||||
|  | ||||
|     # Add 'B' if the volume is in bytes | ||||
|     if re.match(r"^[0-9]+$", size): | ||||
|         size = "{}B".format(size) | ||||
|  | ||||
|     # 1. Verify the size of the volume | ||||
|     pool_information = getPoolInformation(zkhandler, pool) | ||||
|     size_bytes = format_bytes_fromhuman(size) | ||||
|     if size_bytes is None: | ||||
|         return ( | ||||
|             False, | ||||
|             f"ERROR: Requested volume size '{size}' does not have a valid SI unit", | ||||
|         ) | ||||
|  | ||||
|     if size_bytes >= int(pool_information["stats"]["free_bytes"]): | ||||
|         return ( | ||||
|             False, | ||||
|             f"ERROR: Requested volume size '{format_bytes_tohuman(size_bytes)}' is greater than the available free space in the pool ('{format_bytes_tohuman(pool_information['stats']['free_bytes'])}')", | ||||
|             "ERROR: Requested volume size is greater than the available free space in the pool", | ||||
|         ) | ||||
|  | ||||
|     # 2. Resize the volume | ||||
|     retcode, stdout, stderr = common.run_os_command( | ||||
|         "rbd resize --size {} {}/{}".format( | ||||
|             format_bytes_tohuman(size_bytes), pool, name | ||||
|         ) | ||||
|         "rbd resize --size {} {}/{}".format(size, pool, name) | ||||
|     ) | ||||
|     if retcode: | ||||
|         return ( | ||||
|             False, | ||||
|             'ERROR: Failed to resize RBD volume "{}" to size "{}" in pool "{}": {}'.format( | ||||
|                 name, format_bytes_tohuman(size_bytes), pool, stderr | ||||
|                 name, size, pool, stderr | ||||
|             ), | ||||
|         ) | ||||
|  | ||||
| @@ -884,7 +860,7 @@ def resize_volume(zkhandler, pool, name, size): | ||||
|             if target_vm_conn: | ||||
|                 target_vm_conn.blockResize( | ||||
|                     volume_id, | ||||
|                     size_bytes, | ||||
|                     format_bytes_fromhuman(size), | ||||
|                     libvirt.VIR_DOMAIN_BLOCK_RESIZE_BYTES, | ||||
|                 ) | ||||
|             target_lv_conn.close() | ||||
| @@ -907,7 +883,7 @@ def resize_volume(zkhandler, pool, name, size): | ||||
|     ) | ||||
|  | ||||
|     return True, 'Resized RBD volume "{}" to size "{}" in pool "{}".'.format( | ||||
|         name, format_bytes_tohuman(size_bytes), pool | ||||
|         name, size, pool | ||||
|     ) | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -51,8 +51,6 @@ def getClusterHealth(zkhandler, node_list, vm_list, ceph_osd_list): | ||||
|         "vm_stopped": 10, | ||||
|         "osd_out": 50, | ||||
|         "osd_down": 10, | ||||
|         "osd_full": 50, | ||||
|         "osd_nearfull": 10, | ||||
|         "memory_overprovisioned": 50, | ||||
|         "ceph_err": 50, | ||||
|         "ceph_warn": 10, | ||||
| @@ -112,18 +110,6 @@ def getClusterHealth(zkhandler, node_list, vm_list, ceph_osd_list): | ||||
|                 f"cluster: Ceph OSD {ceph_osd['id']} in {up_texts[ceph_osd['stats']['up']].upper()} state" | ||||
|             ) | ||||
|  | ||||
|         # Handle full or nearfull OSDs (>85%) | ||||
|         if ceph_osd["stats"]["utilization"] >= 90: | ||||
|             cluster_health_value -= health_delta_map["osd_full"] | ||||
|             cluster_health_messages.append( | ||||
|                 f"cluster: Ceph OSD {ceph_osd['id']} is FULL ({ceph_osd['stats']['utilization']:.1f}% > 90%)" | ||||
|             ) | ||||
|         elif ceph_osd["stats"]["utilization"] >= 85: | ||||
|             cluster_health_value -= health_delta_map["osd_nearfull"] | ||||
|             cluster_health_messages.append( | ||||
|                 f"cluster: Ceph OSD {ceph_osd['id']} is NEARFULL ({ceph_osd['stats']['utilization']:.1f}% > 85%)" | ||||
|             ) | ||||
|  | ||||
|     # Check for (n-1) overprovisioning | ||||
|     #   Assume X nodes. If the total VM memory allocation (counting only running VMss) is greater than | ||||
|     #   the total memory of the (n-1) smallest nodes, trigger this warning. | ||||
|   | ||||
| @@ -127,14 +127,16 @@ def getNodeInformation(zkhandler, node_name): | ||||
| def secondary_node(zkhandler, node): | ||||
|     # Verify node is valid | ||||
|     if not common.verifyNode(zkhandler, node): | ||||
|         return False, "ERROR: No node named {} is present in the cluster.".format(node) | ||||
|         return False, 'ERROR: No node named "{}" is present in the cluster.'.format( | ||||
|             node | ||||
|         ) | ||||
|  | ||||
|     # Ensure node is a coordinator | ||||
|     daemon_mode = zkhandler.read(("node.mode", node)) | ||||
|     if daemon_mode == "hypervisor": | ||||
|         return ( | ||||
|             False, | ||||
|             "ERROR: Cannot change coordinator state on non-coordinator node {}".format( | ||||
|             'ERROR: Cannot change coordinator mode on non-coordinator node "{}"'.format( | ||||
|                 node | ||||
|             ), | ||||
|         ) | ||||
| @@ -142,14 +144,14 @@ def secondary_node(zkhandler, node): | ||||
|     # Ensure node is in run daemonstate | ||||
|     daemon_state = zkhandler.read(("node.state.daemon", node)) | ||||
|     if daemon_state != "run": | ||||
|         return False, "ERROR: Node {} is not active".format(node) | ||||
|         return False, 'ERROR: Node "{}" is not active'.format(node) | ||||
|  | ||||
|     # Get current state | ||||
|     current_state = zkhandler.read(("node.state.router", node)) | ||||
|     if current_state == "secondary": | ||||
|         return True, "Node {} is already in secondary coordinator state.".format(node) | ||||
|         return True, 'Node "{}" is already in secondary coordinator mode.'.format(node) | ||||
|  | ||||
|     retmsg = "Setting node {} in secondary coordinator state.".format(node) | ||||
|     retmsg = "Setting node {} in secondary coordinator mode.".format(node) | ||||
|     zkhandler.write([("base.config.primary_node", "none")]) | ||||
|  | ||||
|     return True, retmsg | ||||
| @@ -158,14 +160,16 @@ def secondary_node(zkhandler, node): | ||||
| def primary_node(zkhandler, node): | ||||
|     # Verify node is valid | ||||
|     if not common.verifyNode(zkhandler, node): | ||||
|         return False, "ERROR: No node named {} is present in the cluster.".format(node) | ||||
|         return False, 'ERROR: No node named "{}" is present in the cluster.'.format( | ||||
|             node | ||||
|         ) | ||||
|  | ||||
|     # Ensure node is a coordinator | ||||
|     daemon_mode = zkhandler.read(("node.mode", node)) | ||||
|     if daemon_mode == "hypervisor": | ||||
|         return ( | ||||
|             False, | ||||
|             "ERROR: Cannot change coordinator state on non-coordinator node {}".format( | ||||
|             'ERROR: Cannot change coordinator mode on non-coordinator node "{}"'.format( | ||||
|                 node | ||||
|             ), | ||||
|         ) | ||||
| @@ -173,14 +177,14 @@ def primary_node(zkhandler, node): | ||||
|     # Ensure node is in run daemonstate | ||||
|     daemon_state = zkhandler.read(("node.state.daemon", node)) | ||||
|     if daemon_state != "run": | ||||
|         return False, "ERROR: Node {} is not active".format(node) | ||||
|         return False, 'ERROR: Node "{}" is not active'.format(node) | ||||
|  | ||||
|     # Get current state | ||||
|     current_state = zkhandler.read(("node.state.router", node)) | ||||
|     if current_state == "primary": | ||||
|         return True, "Node {} is already in primary coordinator state.".format(node) | ||||
|         return True, 'Node "{}" is already in primary coordinator mode.'.format(node) | ||||
|  | ||||
|     retmsg = "Setting node {} in primary coordinator state.".format(node) | ||||
|     retmsg = "Setting node {} in primary coordinator mode.".format(node) | ||||
|     zkhandler.write([("base.config.primary_node", node)]) | ||||
|  | ||||
|     return True, retmsg | ||||
| @@ -189,12 +193,14 @@ def primary_node(zkhandler, node): | ||||
| def flush_node(zkhandler, node, wait=False): | ||||
|     # Verify node is valid | ||||
|     if not common.verifyNode(zkhandler, node): | ||||
|         return False, "ERROR: No node named {} is present in the cluster.".format(node) | ||||
|         return False, 'ERROR: No node named "{}" is present in the cluster.'.format( | ||||
|             node | ||||
|         ) | ||||
|  | ||||
|     if zkhandler.read(("node.state.domain", node)) == "flushed": | ||||
|         return True, "Node {} is already flushed.".format(node) | ||||
|         return True, "Hypervisor {} is already flushed.".format(node) | ||||
|  | ||||
|     retmsg = "Removing node {} from active service.".format(node) | ||||
|     retmsg = "Flushing hypervisor {} of running VMs.".format(node) | ||||
|  | ||||
|     # Add the new domain to Zookeeper | ||||
|     zkhandler.write([(("node.state.domain", node), "flush")]) | ||||
| @@ -202,7 +208,7 @@ def flush_node(zkhandler, node, wait=False): | ||||
|     if wait: | ||||
|         while zkhandler.read(("node.state.domain", node)) == "flush": | ||||
|             time.sleep(1) | ||||
|         retmsg = "Removed node {} from active service.".format(node) | ||||
|         retmsg = "Flushed hypervisor {} of running VMs.".format(node) | ||||
|  | ||||
|     return True, retmsg | ||||
|  | ||||
| @@ -210,12 +216,14 @@ def flush_node(zkhandler, node, wait=False): | ||||
| def ready_node(zkhandler, node, wait=False): | ||||
|     # Verify node is valid | ||||
|     if not common.verifyNode(zkhandler, node): | ||||
|         return False, "ERROR: No node named {} is present in the cluster.".format(node) | ||||
|         return False, 'ERROR: No node named "{}" is present in the cluster.'.format( | ||||
|             node | ||||
|         ) | ||||
|  | ||||
|     if zkhandler.read(("node.state.domain", node)) == "ready": | ||||
|         return True, "Node {} is already ready.".format(node) | ||||
|         return True, "Hypervisor {} is already ready.".format(node) | ||||
|  | ||||
|     retmsg = "Restoring node {} to active service.".format(node) | ||||
|     retmsg = "Restoring hypervisor {} to active service.".format(node) | ||||
|  | ||||
|     # Add the new domain to Zookeeper | ||||
|     zkhandler.write([(("node.state.domain", node), "unflush")]) | ||||
| @@ -223,7 +231,7 @@ def ready_node(zkhandler, node, wait=False): | ||||
|     if wait: | ||||
|         while zkhandler.read(("node.state.domain", node)) == "unflush": | ||||
|             time.sleep(1) | ||||
|         retmsg = "Restored node {} to active service.".format(node) | ||||
|         retmsg = "Restored hypervisor {} to active service.".format(node) | ||||
|  | ||||
|     return True, retmsg | ||||
|  | ||||
| @@ -231,7 +239,9 @@ def ready_node(zkhandler, node, wait=False): | ||||
| def get_node_log(zkhandler, node, lines=2000): | ||||
|     # Verify node is valid | ||||
|     if not common.verifyNode(zkhandler, node): | ||||
|         return False, "ERROR: No node named {} is present in the cluster.".format(node) | ||||
|         return False, 'ERROR: No node named "{}" is present in the cluster.'.format( | ||||
|             node | ||||
|         ) | ||||
|  | ||||
|     # Get the data from ZK | ||||
|     node_log = zkhandler.read(("logs.messages", node)) | ||||
| @@ -249,12 +259,14 @@ def get_node_log(zkhandler, node, lines=2000): | ||||
| def get_info(zkhandler, node): | ||||
|     # Verify node is valid | ||||
|     if not common.verifyNode(zkhandler, node): | ||||
|         return False, "ERROR: No node named {} is present in the cluster.".format(node) | ||||
|         return False, 'ERROR: No node named "{}" is present in the cluster.'.format( | ||||
|             node | ||||
|         ) | ||||
|  | ||||
|     # Get information about node in a pretty format | ||||
|     node_information = getNodeInformation(zkhandler, node) | ||||
|     if not node_information: | ||||
|         return False, "ERROR: Could not get information about node {}.".format(node) | ||||
|         return False, 'ERROR: Could not get information about node "{}".'.format(node) | ||||
|  | ||||
|     return True, node_information | ||||
|  | ||||
|   | ||||
							
								
								
									
										34
									
								
								debian/changelog
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										34
									
								
								debian/changelog
									
									
									
									
										vendored
									
									
								
							| @@ -1,37 +1,3 @@ | ||||
| pvc (0.9.64-0) unstable; urgency=high | ||||
|  | ||||
|   **Breaking Change [CLI]**: The CLI client root commands have been reorganized. The following commands have changed: | ||||
|  | ||||
|    * `pvc cluster` -> `pvc connection` (all subcommands) | ||||
|    * `pvc task` -> `pvc cluster` (all subcommands) | ||||
|    * `pvc maintenance` -> `pvc cluster maintenance` | ||||
|    * `pvc status` -> `pvc cluster status` | ||||
|  | ||||
| Ensure you have updated to the latest version of the PVC Ansible repository before deploying this version or using PVC Ansible oneshot playbooks for management. | ||||
|  | ||||
|   **Breaking Change [CLI]**: The `--restart` option for VM configuration changes now has an explicit `--no-restart` to disable restarting, or a prompt if neither is specified; `--unsafe` no longer bypasses this prompt which was a bug. Applies to most `vm <cmd> set` commands like `vm vcpu set`, `vm memory set`, etc. All instances also feature restart confirmation afterwards, which, if `--restart` is provided, will prompt for confirmation unless `--yes` or `--unsafe` is specified. | ||||
|  | ||||
|   **Breaking Change [CLI]**: The `--long` option previously on some `info` commands no longer exists; use `-f long`/`--format long` instead. | ||||
|  | ||||
|   * [CLI] Significantly refactors the CLI client code for consistency and cleanliness | ||||
|   * [CLI] Implements `-f`/`--format` options for all `list` and `info` commands in a consistent way | ||||
|   * [CLI] Changes the behaviour of VM modification options with "--restart" to provide a "--no-restart"; defaults to a prompt if neither is specified and ignores the "--unsafe" global entirely | ||||
|   * [API] Fixes several bugs in the 3-debootstrap.py provisioner example script | ||||
|   * [Node] Fixes some bugs around VM shutdown on node flush | ||||
|   * [Documentation] Adds mentions of Ganeti and Harvester | ||||
|  | ||||
|  -- Joshua M. Boniface <joshua@boniface.me>  Fri, 18 Aug 2023 12:20:43 -0400 | ||||
|  | ||||
| pvc (0.9.63-0) unstable; urgency=high | ||||
|  | ||||
|   * Mentions Ganeti in the docs | ||||
|   * Increases API timeout back to 2s | ||||
|   * Adds .update-* configs to dpkg plugin | ||||
|   * Adds full/nearfull OSD warnings | ||||
|   * Improves size value handling for volumes | ||||
|  | ||||
|  -- Joshua M. Boniface <joshua@boniface.me>  Fri, 28 Apr 2023 14:47:04 -0400 | ||||
|  | ||||
| pvc (0.9.62-0) unstable; urgency=high | ||||
|  | ||||
|   * [all] Adds an enhanced health checking, monitoring, and reporting system for nodes and clusters | ||||
|   | ||||
| @@ -25,13 +25,13 @@ As part of these trends, Infrastructure-as-a-Service (IaaS) has become a critica | ||||
|  | ||||
| However, the current state of the free and open source virtualization ecosystem is lacking. | ||||
|  | ||||
| At the lower- to middle-end, projects like ProxMox provide an easy way to administer small virtualization clusters, but these projects tend to lack advanced redundancy facilities that are built-in by default. Ganeti, a former Google tool, was long-dead when PVC was initially conceived, but has recently been given new life by the FLOSS community, and was the inspiration for much of PVC's functionality. Harvester is also a newer player in the space, created by Rancher Labs after PVC was established, but its use of custom solutions for everything, especially the storage backend, gives us some pause. | ||||
| At the lower end, projects like ProxMox provide an easy way to administer small virtualization clusters, but these projects tend to lack advanced redundancy facilities that are built-in by default. While there are some new contenders in this space, such as Harvester, the options are limited and their feature-sets and tool stacks can be cumbersome or unproven. | ||||
|  | ||||
| At the high-end, very large projects like OpenStack and CloudStack provide very advanced functionality, but these project are sprawling and complicated for Administrators to use, and are very focused on large enterprise deployments, not suitable for smaller clusters and teams. | ||||
| At the higher end, very large projects like OpenStack and CloudStack provide very advanced functionality, but these project are sprawling and complicated for Administrators to use, and are very focused on large enterprise deployments, not suitable for smaller clusters and teams. | ||||
|  | ||||
| Finally, proprietary solutions dominate this space. VMWare and Nutanix are the two largest names, with these products providing functionality for both small and large clusters, but proprietary software limits both flexibility and freedom, and the costs associated with these solutions is immense. | ||||
|  | ||||
| PVC aims to bridge the gaps between these categories. Like the larger FLOSS and proprietary projects, PVC can scale up to very large cluster sizes, while remaining usable even for small clusters as well. Like the smaller FLOSS and proprietary projects, PVC aims to be very simple to use, with a fully programmable API, allowing administrators to get on with more important things. Like the other FLOSS solutions, PVC is free, both as in beer and as in speech, allowing the administrator to inspect, modify, and tailor it to their needs. And finally, PVC is built from the ground-up to support host-level redundancy at every layer, rather than this being an expensive, optional, or tacked on feature, using standard, well-tested and well-supported components. | ||||
| PVC aims to bridge the gaps between these three categories. Like the larger FLOSS and proprietary projects, PVC can scale up to very large cluster sizes, while remaining usable even for small clusters as well. Like the smaller FLOSS and proprietary projects, PVC aims to be very simple to use, with a fully programmable API, allowing administrators to get on with more important things. Like the other FLOSS solutions, PVC is free, both as in beer and as in speech, allowing the administrator to inspect, modify, and tailor it to their needs. And finally, PVC is built from the ground-up to support host-level redundancy at every layer, rather than this being an expensive, optional, or tacked on feature. | ||||
|  | ||||
| In short, it is a Free Software, scalable, redundant, self-healing, and self-managing private cloud solution designed with administrator simplicity in mind.  | ||||
|  | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
|  | ||||
| ## What is PVC? | ||||
|  | ||||
| PVC is a Linux KVM-based hyperconverged infrastructure (HCI) virtualization cluster solution that is fully Free Software, scalable, redundant, self-healing, self-managing, and designed for administrator simplicity. It is an alternative to other HCI solutions such as Ganeti, Harvester, Nutanix, and VMWare, as well as to other common virtualization stacks such as ProxMox and OpenStack. | ||||
| PVC is a Linux KVM-based hyperconverged infrastructure (HCI) virtualization cluster solution that is fully Free Software, scalable, redundant, self-healing, self-managing, and designed for administrator simplicity. It is an alternative to other HCI solutions such as Harvester, Nutanix, and VMWare, as well as to other common virtualization stacks such as ProxMox and OpenStack. | ||||
|  | ||||
| PVC is a complete HCI solution, built from well-known and well-trusted Free Software tools, to assist an administrator in creating and managing a cluster of servers to run virtual machines, as well as self-managing several important aspects including storage failover, node failure and recovery, virtual machine failure and recovery, and network plumbing. It is designed to act consistently, reliably, and unobtrusively, letting the administrator concentrate on more important things. | ||||
|  | ||||
|   | ||||
| @@ -110,8 +110,8 @@ class MonitoringPluginScript(MonitoringPlugin): | ||||
|  | ||||
|         count_upgradable = len(list_upgradable) | ||||
|  | ||||
|         # Get obsolete config files (dpkg-*, ucf-*, or update-* under /etc) | ||||
|         retcode, stdout, stderr = pvc_common.run_os_command("/usr/bin/find /etc -type f -a \( -name '*.dpkg-*' -o -name '*.ucf-*' -o -name '*.update-*' \)") | ||||
|         # Get obsolete config files (dpkg-* or ucf-* under /etc) | ||||
|         retcode, stdout, stderr = pvc_common.run_os_command("/usr/bin/find /etc -type f -a \( -name '*.dpkg-*' -o -name '*.ucf-*' \)") | ||||
|  | ||||
|         obsolete_conffiles = list() | ||||
|         for conffile_line in stdout.split('\n'): | ||||
|   | ||||
| @@ -49,7 +49,7 @@ import re | ||||
| import json | ||||
|  | ||||
| # Daemon version | ||||
| version = "0.9.64" | ||||
| version = "0.9.62" | ||||
|  | ||||
|  | ||||
| ########################################################## | ||||
|   | ||||
| @@ -790,19 +790,6 @@ class NodeInstance(object): | ||||
|                 self.flush_stopper = False | ||||
|                 return | ||||
|  | ||||
|             # Wait for a VM in "restart" or "shutdown" state to complete transition | ||||
|             while self.zkhandler.read(("domain.state", dom_uuid)) in [ | ||||
|                 "restart", | ||||
|                 "shutdown", | ||||
|             ]: | ||||
|                 self.logger.out( | ||||
|                     'Waiting 2s for VM state change completion for VM "{}"'.format( | ||||
|                         dom_uuid | ||||
|                     ), | ||||
|                     state="i", | ||||
|                 ) | ||||
|                 time.sleep(2) | ||||
|  | ||||
|             self.logger.out( | ||||
|                 'Selecting target to migrate VM "{}"'.format(dom_uuid), state="i" | ||||
|             ) | ||||
| @@ -819,19 +806,17 @@ class NodeInstance(object): | ||||
|  | ||||
|             if target_node is None: | ||||
|                 self.logger.out( | ||||
|                     'Failed to find migration target for running VM "{}"; shutting down and setting autostart flag'.format( | ||||
|                     'Failed to find migration target for VM "{}"; shutting down and setting autostart flag'.format( | ||||
|                         dom_uuid | ||||
|                     ), | ||||
|                     state="e", | ||||
|                 ) | ||||
|  | ||||
|                 if self.zkhandler.read(("domain.state", dom_uuid)) in ["start"]: | ||||
|                     self.zkhandler.write( | ||||
|                         [ | ||||
|                             (("domain.state", dom_uuid), "shutdown"), | ||||
|                             (("domain.meta.autostart", dom_uuid), "True"), | ||||
|                         ] | ||||
|                     ) | ||||
|                 self.zkhandler.write( | ||||
|                     [ | ||||
|                         (("domain.state", dom_uuid), "shutdown"), | ||||
|                         (("domain.meta.autostart", dom_uuid), "True"), | ||||
|                     ] | ||||
|                 ) | ||||
|             else: | ||||
|                 self.logger.out( | ||||
|                     'Migrating VM "{}" to node "{}"'.format(dom_uuid, target_node), | ||||
|   | ||||
| @@ -1,54 +1,31 @@ | ||||
| #!/usr/bin/env bash | ||||
|  | ||||
| set -o errexit | ||||
|  | ||||
| if [[ -z ${1} ]]; then | ||||
|     echo "Please specify a cluster to run tests against." | ||||
|     exit 1 | ||||
| fi | ||||
| test_cluster="${1}" | ||||
| shift | ||||
|  | ||||
| if [[ ${1} == "--test-dangerously" ]]; then | ||||
|     test_dangerously="y" | ||||
| else | ||||
|     test_dangerously="" | ||||
| fi | ||||
|  | ||||
| _pvc() { | ||||
|     echo "> pvc --connection ${test_cluster} $@" | ||||
|     pvc --quiet --connection ${test_cluster} "$@" | ||||
|     echo "> pvc --cluster ${test_cluster} $@" | ||||
|     pvc --quiet --cluster ${test_cluster} "$@" | ||||
|     sleep 1 | ||||
| } | ||||
|  | ||||
| time_start=$(date +%s) | ||||
|  | ||||
| set -o errexit | ||||
|  | ||||
| pushd $( git rev-parse --show-toplevel ) &>/dev/null | ||||
|  | ||||
| # Cluster tests | ||||
| _pvc connection list | ||||
| _pvc connection detail | ||||
|  | ||||
| _pvc cluster maintenance on | ||||
| _pvc cluster maintenance off | ||||
| _pvc cluster status | ||||
| _pvc maintenance on | ||||
| _pvc maintenance off | ||||
| backup_tmp=$(mktemp) | ||||
| _pvc cluster backup --file ${backup_tmp} | ||||
| if [[ -n ${test_dangerously} ]]; then | ||||
|     # This is dangerous, so don't test it unless option given | ||||
|     _pvc cluster restore --yes --file ${backup_tmp} | ||||
| fi | ||||
| _pvc task backup --file ${backup_tmp} | ||||
| _pvc task restore --yes --file ${backup_tmp} | ||||
| rm ${backup_tmp} || true | ||||
|  | ||||
| # Provisioner tests | ||||
| _pvc provisioner profile list test || true | ||||
| _pvc provisioner template system add --vcpus 1 --vram 1024 --serial --vnc --vnc-bind 0.0.0.0 --node-limit hv1 --node-selector mem --node-autostart --migration-method live system-test || true | ||||
| _pvc provisioner template network add network-test || true | ||||
| _pvc provisioner template network vni add network-test 10000 || true | ||||
| _pvc provisioner template storage add storage-test || true | ||||
| _pvc provisioner template storage disk add --pool vms --size 8 --filesystem ext4 --mountpoint / storage-test sda || true | ||||
| _pvc provisioner script add script-test $( find . -name "3-debootstrap.py" ) || true | ||||
| _pvc provisioner profile add --profile-type provisioner --system-template system-test --network-template network-test --storage-template storage-test --userdata empty --script script-test --script-arg deb_release=bullseye test || true | ||||
| _pvc provisioner profile list test | ||||
| _pvc provisioner create --wait testx test | ||||
| sleep 30 | ||||
|  | ||||
| @@ -59,7 +36,7 @@ _pvc vm shutdown --yes --wait testx | ||||
| _pvc vm start testx | ||||
| sleep 30 | ||||
| _pvc vm stop --yes testx | ||||
| _pvc vm disable --yes testx | ||||
| _pvc vm disable testx | ||||
| _pvc vm undefine --yes testx | ||||
| _pvc vm define --target hv3 --tag pvc-test ${vm_tmp} | ||||
| _pvc vm start testx | ||||
| @@ -72,21 +49,21 @@ _pvc vm unmigrate --wait testx | ||||
| sleep 5 | ||||
| _pvc vm move --wait --target hv1 testx | ||||
| sleep 5 | ||||
| _pvc vm meta testx --limit hv1 --node-selector vms --method live --profile test --no-autostart | ||||
| _pvc vm meta testx --limit hv1 --selector vms --method live --profile test --no-autostart | ||||
| _pvc vm tag add testx mytag | ||||
| _pvc vm tag get testx | ||||
| _pvc vm list --tag mytag | ||||
| _pvc vm tag remove testx mytag | ||||
| _pvc vm network get testx | ||||
| _pvc vm vcpu set --no-restart testx 4 | ||||
| _pvc vm vcpu set testx 4 | ||||
| _pvc vm vcpu get testx | ||||
| _pvc vm memory set --no-restart testx 4096 | ||||
| _pvc vm memory set testx 4096 | ||||
| _pvc vm memory get testx | ||||
| _pvc vm vcpu set --no-restart testx 2 | ||||
| _pvc vm vcpu set testx 2 | ||||
| _pvc vm memory set testx 2048 --restart --yes | ||||
| sleep 15 | ||||
| sleep 5 | ||||
| _pvc vm list testx | ||||
| _pvc vm info --format long testx | ||||
| _pvc vm info --long testx | ||||
| rm ${vm_tmp} || true | ||||
|  | ||||
| # Node tests | ||||
| @@ -100,7 +77,6 @@ _pvc node flush --wait hv1 | ||||
| _pvc node ready --wait hv1 | ||||
| _pvc node list hv1 | ||||
| _pvc node info hv1 | ||||
| sleep 15 | ||||
|  | ||||
| # Network tests | ||||
| _pvc network add 10001 --description testing --type managed --domain testing.local --ipnet 10.100.100.0/24 --gateway 10.100.100.1 --dhcp --dhcp-start 10.100.100.100 --dhcp-end 10.100.100.199 | ||||
| @@ -108,7 +84,7 @@ sleep 5 | ||||
| _pvc vm network add --restart --yes testx 10001 | ||||
| sleep 30 | ||||
| _pvc vm network remove --restart --yes testx 10001 | ||||
| sleep 15 | ||||
| sleep 5 | ||||
|  | ||||
| _pvc network acl add 10001 --in --description test-acl --order 0 --rule "'ip daddr 10.0.0.0/8 counter'" | ||||
| _pvc network acl list 10001 | ||||
| @@ -119,34 +95,31 @@ _pvc network dhcp remove --yes 10001 12:34:56:78:90:ab | ||||
|  | ||||
| _pvc network modify --domain test10001.local 10001 | ||||
| _pvc network list | ||||
| _pvc network info --format long 10001 | ||||
| _pvc network info --long 10001 | ||||
|  | ||||
| # Network-VM interaction tests | ||||
| _pvc vm network add testx 10001 --model virtio --restart --yes | ||||
| sleep 30 | ||||
| _pvc vm network get testx | ||||
| _pvc vm network remove testx 10001 --restart --yes | ||||
| sleep 15 | ||||
| sleep 5 | ||||
|  | ||||
| _pvc network remove --yes 10001 | ||||
|  | ||||
| # Storage tests | ||||
| _pvc storage status | ||||
| _pvc storage util | ||||
| if [[ -n ${test_dangerously} ]]; then | ||||
|     # This is dangerous, so don't test it unless option given | ||||
|     _pvc storage osd set noout | ||||
|     _pvc storage osd out 0 | ||||
|     _pvc storage osd in 0 | ||||
|     _pvc storage osd unset noout | ||||
| fi | ||||
| _pvc storage osd set noout | ||||
| _pvc storage osd out 0 | ||||
| _pvc storage osd in 0 | ||||
| _pvc storage osd unset noout | ||||
| _pvc storage osd list | ||||
| _pvc storage pool add testing 64 --replcfg "copies=3,mincopies=2" | ||||
| sleep 5 | ||||
| _pvc storage pool list | ||||
| _pvc storage volume add testing testx 1G | ||||
| _pvc storage volume resize --yes testing testx 2G | ||||
| _pvc storage volume rename --yes testing testx testerX | ||||
| _pvc storage volume resize testing testx 2G | ||||
| _pvc storage volume rename testing testx testerX | ||||
| _pvc storage volume clone testing testerX testerY | ||||
| _pvc storage volume list --pool testing | ||||
| _pvc storage volume snapshot add testing testerX asnapshotX | ||||
| @@ -159,7 +132,7 @@ _pvc vm volume add testx --type rbd --disk-id sdh --bus scsi testing/testerY --r | ||||
| sleep 30 | ||||
| _pvc vm volume get testx | ||||
| _pvc vm volume remove testx testing/testerY --restart --yes | ||||
| sleep 15 | ||||
| sleep 5 | ||||
|  | ||||
| _pvc storage volume remove --yes testing testerY | ||||
| _pvc storage volume remove --yes testing testerX | ||||
| @@ -169,14 +142,6 @@ _pvc storage pool remove --yes testing | ||||
| _pvc vm stop --yes testx | ||||
| _pvc vm remove --yes testx | ||||
|  | ||||
| _pvc provisioner profile remove --yes test | ||||
| _pvc provisioner script remove --yes script-test | ||||
| _pvc provisioner template system remove --yes system-test | ||||
| _pvc provisioner template network remove --yes network-test | ||||
| _pvc provisioner template storage remove --yes storage-test | ||||
|  | ||||
| popd | ||||
|  | ||||
| time_end=$(date +%s) | ||||
|  | ||||
| echo | ||||
|   | ||||
		Reference in New Issue
	
	Block a user