2025-05-07 16:20:36 +03:00
|
|
|
# functions for working with domain-api
|
2025-05-12 17:21:00 +03:00
|
|
|
import requests
|
|
|
|
import secrets #for generating unique names
|
2025-05-21 14:12:37 +03:00
|
|
|
import os
|
|
|
|
from rich.console import Console , Align
|
2025-05-16 17:53:57 +03:00
|
|
|
from rich.columns import Columns
|
|
|
|
from rich.panel import Panel
|
2025-05-21 14:12:37 +03:00
|
|
|
from rich.prompt import Prompt
|
2025-05-12 17:21:00 +03:00
|
|
|
|
2025-05-21 14:12:37 +03:00
|
|
|
console = Console() #necessary for pretty menus & output
|
2025-05-12 17:21:00 +03:00
|
|
|
power_state = ["Unknown" , "Off" , "Suspend" , "On"] #3 - on; 2 - suspend; 1 - off; 0 - unknown
|
|
|
|
|
|
|
|
|
|
|
|
def get_domain_info(base_url , api_key , domain_uuid):
|
|
|
|
url= f"http://{base_url}/api/domains/{domain_uuid}"
|
|
|
|
response = requests.get(url , headers={'Authorization' : api_key})
|
|
|
|
|
|
|
|
if response.status_code == 200: #200 - OK
|
|
|
|
domain_data = response.json()
|
|
|
|
return domain_data #returns as dictionary!
|
|
|
|
else:
|
|
|
|
print(f"Failed to retrieve data {response.status_code}")
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def get_domain_all_content(base_url, api_key, domain_uuid):
|
|
|
|
url= f"http://{base_url}/api/domains/{domain_uuid}/all-content"
|
|
|
|
response = requests.get(url , headers={'Authorization' : api_key})
|
|
|
|
if response.status_code == 200: #200 - OK
|
|
|
|
domain_all_data = response.json()
|
|
|
|
return domain_all_data #returns as dictionary!
|
|
|
|
else:
|
|
|
|
print(f"Failed to retrieve data {response.status_code}")
|
|
|
|
|
|
|
|
|
|
|
|
def get_disk_uuids(base_url , api_key , domain_all_content):
|
|
|
|
#domain_all_content (type - dictionary)
|
|
|
|
#returns VMs vdisk uuids (type - list)
|
|
|
|
try:
|
|
|
|
# check for "vdisks" field in recieved json response
|
|
|
|
if 'vdisks' not in domain_all_content:
|
|
|
|
raise KeyError("No 'vdisks' field in recieved data")
|
|
|
|
# Get list of all vdisks
|
|
|
|
disks = domain_all_content['vdisks']
|
|
|
|
# Extracting UUID for each disk
|
|
|
|
vdisk_uuid = [disk['id'] for disk in disks]
|
|
|
|
vdisk_size = []
|
|
|
|
return vdisk_uuid
|
|
|
|
except KeyError as e:
|
|
|
|
print(f"ERROR: {e}")
|
|
|
|
return []
|
|
|
|
except TypeError:
|
|
|
|
print("ERROR: unexpected data format")
|
|
|
|
return []
|
|
|
|
|
2025-05-16 17:53:57 +03:00
|
|
|
|
2025-05-12 17:21:00 +03:00
|
|
|
def delete_disk(base_url , api_key , vdisk_uuid):
|
|
|
|
url = f"http://{base_url}/api/vdisks/{vdisk_uuid}/remove/"
|
|
|
|
headers={
|
|
|
|
"Authorization" : api_key,
|
|
|
|
"Content-Type" : "application/json",
|
|
|
|
}
|
|
|
|
payload= {
|
|
|
|
"force": False,
|
|
|
|
"guaranteed": False,
|
|
|
|
"clean_type": "zero",
|
|
|
|
"clean_count": 1
|
|
|
|
}
|
|
|
|
response = requests.post(url , headers=headers, json=payload)
|
|
|
|
if response.status_code == 200:
|
|
|
|
print(f"vDisk {vdisk_uuid} successfully deleted")
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
print(f"ERROR deleting disk {vdisk_uuid} :\n {response.status_code} - {response.text}")
|
|
|
|
return False
|
2025-05-16 17:53:57 +03:00
|
|
|
|
|
|
|
|
2025-05-12 17:21:00 +03:00
|
|
|
def get_disk_info(domain_all_content):
|
2025-05-21 14:12:37 +03:00
|
|
|
console = Console()
|
2025-05-12 17:21:00 +03:00
|
|
|
# check for "vdisks" field in recieved json response
|
|
|
|
if 'vdisks' not in domain_all_content:
|
|
|
|
print("No 'vdisks' field in recieved data")
|
|
|
|
return
|
|
|
|
# get vdisk list
|
|
|
|
disks = domain_all_content['vdisks']
|
|
|
|
# check for disks
|
|
|
|
if not disks:
|
2025-05-21 14:12:37 +03:00
|
|
|
console.print("[bold yellow]No 'disks' field in recieved data. \nProbably VM does not have any attached disks?")
|
2025-05-12 17:21:00 +03:00
|
|
|
return
|
2025-05-21 14:12:37 +03:00
|
|
|
|
|
|
|
disk_info_renderables = []
|
2025-05-12 17:21:00 +03:00
|
|
|
# Print info for each disk
|
|
|
|
for disk in disks:
|
2025-05-21 14:12:37 +03:00
|
|
|
# check for required fields
|
2025-05-12 17:21:00 +03:00
|
|
|
if 'id' in disk and 'verbose_name' in disk and 'size' in disk:
|
2025-05-21 14:12:37 +03:00
|
|
|
output_string = (
|
|
|
|
f"[bold]Name:[/] {disk['verbose_name']}\n"
|
|
|
|
f"[bold]UUID:[/] [italic]{disk['id']}[/italic]\n"
|
|
|
|
f"[bold]Size:[/] {disk['size']} GB")
|
|
|
|
disk_info_renderables.append(Panel(output_string, expand=False, border_style="magenta"))
|
2025-05-12 17:21:00 +03:00
|
|
|
else:
|
|
|
|
print("ERROR: failed to retrieve vdisk data.")
|
|
|
|
|
2025-05-21 14:12:37 +03:00
|
|
|
console.print(Columns(disk_info_renderables))
|
2025-05-12 17:21:00 +03:00
|
|
|
|
2025-05-21 14:12:37 +03:00
|
|
|
def vm_info(base_url, api_key, vm_uuids):
|
|
|
|
domain_info = get_domain_info(base_url, api_key, vm_uuids)
|
|
|
|
domain_all_content = get_domain_all_content(base_url, api_key, vm_uuids)
|
2025-05-12 17:21:00 +03:00
|
|
|
if domain_info:
|
2025-05-21 14:12:37 +03:00
|
|
|
console = Console()
|
|
|
|
vm_info_lines = f"[bold]Power State:[/] [bold red]{power_state[domain_info['user_power_state']]}[/bold red] \n[bold]vDisks:[/] {domain_info['vdisks_count']}"
|
|
|
|
vm_info_renderable = Panel(vm_info_lines, title=f"[bold magenta]{domain_info['verbose_name']}" , expand=False , border_style="yellow")
|
|
|
|
vm_info_renderable=Align.center(vm_info_renderable, vertical="middle")
|
|
|
|
print("\n")
|
|
|
|
console.rule(style="yellow")
|
|
|
|
console.print(vm_info_renderable)
|
|
|
|
console.rule(title = "[bold yellow]vDisks Info" , style="grey53" , align="center")
|
2025-05-12 17:21:00 +03:00
|
|
|
get_disk_info(domain_all_content)
|
2025-05-21 14:12:37 +03:00
|
|
|
console.rule(style="yellow")
|
2025-05-12 17:21:00 +03:00
|
|
|
|
|
|
|
|
2025-05-21 14:12:37 +03:00
|
|
|
|
|
|
|
def vm_info_short(base_url, api_key):
|
|
|
|
url = f"http://{base_url}/api/domains/"
|
|
|
|
response = requests.get(url, headers={'Authorization': api_key})
|
2025-05-15 15:41:33 +03:00
|
|
|
if response.status_code == 200:
|
|
|
|
vm_info_short = response.json()
|
|
|
|
results_vm_info_short = vm_info_short['results']
|
2025-05-21 14:12:37 +03:00
|
|
|
os.system('cls' if os.name=='nt' else 'clear')
|
|
|
|
console.print(Align.center(Panel(f"[bold magenta]Short VM overview | Total: {vm_info_short['count']}", expand=True , border_style="yellow") , vertical="middle"))
|
|
|
|
console.rule(style="grey53")
|
|
|
|
output_renderables = []
|
2025-05-15 15:41:33 +03:00
|
|
|
for x in results_vm_info_short:
|
2025-05-21 14:12:37 +03:00
|
|
|
output_string = f"VM: [bold]{x['verbose_name']}" + f"\nUUID: [italic]{x['id']}"
|
|
|
|
output_renderable = Panel(output_string, expand=False, border_style="magenta")
|
|
|
|
output_renderables.append(output_renderable) #adds current renderable
|
|
|
|
console.print(Columns(output_renderables)) #print renderables by columns
|
2025-05-15 15:41:33 +03:00
|
|
|
else:
|
2025-05-21 14:12:37 +03:00
|
|
|
print(f"Failed to retrieve data {response.status_code}")
|
|
|
|
console.rule(style="grey53")
|
|
|
|
Prompt.ask("[green_yellow bold]ENTER - return to Main Menu.... :right_arrow_curving_down:")
|
|
|
|
os.system('cls' if os.name=='nt' else 'clear')
|
2025-05-15 15:41:33 +03:00
|
|
|
|
2025-05-12 17:21:00 +03:00
|
|
|
def create_and_attach_disk(base_url , api_key , vm_id, data_pool_uuid, vdisk_size, preallocation):
|
|
|
|
domain_name=get_domain_info(base_url , api_key , vm_id)
|
|
|
|
disk_name=domain_name["verbose_name"]+"_"+secrets.token_hex(5) #generates unique hex id. this method can generate ~million unique ids
|
|
|
|
url = f"http://{base_url}/api/domains/{vm_id}/create-attach-vdisk/"
|
|
|
|
headers={
|
|
|
|
"Authorization" : api_key,
|
|
|
|
"Content-Type" : "application/json",
|
|
|
|
}
|
|
|
|
payload= {
|
|
|
|
"verbose_name": disk_name,
|
|
|
|
"preallocation": preallocation,
|
|
|
|
"size": vdisk_size,
|
|
|
|
"datapool": data_pool_uuid,
|
|
|
|
"target_bus": "virtio",
|
|
|
|
}
|
|
|
|
response = requests.post(url , headers=headers, json=payload)
|
|
|
|
if response.status_code == 200:
|
2025-05-21 14:12:37 +03:00
|
|
|
print(f"vDisk {disk_name} ({vdisk_size}GB) has been created and attached")
|
2025-05-12 17:21:00 +03:00
|
|
|
return True
|
|
|
|
else:
|
|
|
|
print(f"ERROR creating vDisk :\n {response.status_code} - {response.text}")
|
2025-05-16 11:09:27 +03:00
|
|
|
return False
|
|
|
|
|
|
|
|
#checks for power on.
|
|
|
|
def vm_check_power(base_url , api_key , vm_uuids):
|
|
|
|
domain_info = get_domain_info(base_url , api_key , vm_uuids)
|
|
|
|
|
|
|
|
if domain_info:
|
|
|
|
#3 - on; 2 - suspend; 1 - off; 0 - unknown
|
|
|
|
if domain_info['user_power_state'] == 3 or domain_info['user_power_state'] == 2 : #if ON or SUSPEND
|
|
|
|
raise Exception(f"VM - {vm_uuids} IS POWERED ON! \n Turn it off and relaunch Utility.")
|
|
|
|
if domain_info['user_power_state'] == 0:
|
|
|
|
raise Exception(f"VM - {vm_uuids} is UNAVAILABLE! \n Have fun figuring that out D:")
|
|
|
|
if domain_info['user_power_state'] == 1:
|
|
|
|
print(f"VM - {vm_uuids} Power check passed!")
|