2021-01-02 23:10:01 +00:00
import subprocess
import re
import sys
import requests
import json
import asyncio
from typing import List , Tuple
import aiohttp
from flask import current_app
2021-01-04 21:02:56 +00:00
from flask import session
2021-01-02 23:10:01 +00:00
from time import sleep
from os . path import join
from subprocess import run
2021-01-04 19:32:52 +00:00
from capsulflask . db import get_model
2021-01-03 21:19:29 +00:00
from capsulflask . http_client import HTTPResult
2021-01-04 19:32:52 +00:00
from capsulflask . shared import VirtualizationInterface , VirtualMachine , OnlineHost , validate_capsul_id , my_exec_info_message
2021-01-02 23:10:01 +00:00
2021-01-03 21:19:29 +00:00
class MockHub ( VirtualizationInterface ) :
2021-01-02 23:10:01 +00:00
def capacity_avaliable ( self , additional_ram_bytes ) :
return True
2021-02-16 00:16:15 +00:00
def get ( self , id , get_ssh_host_keys ) :
2021-01-02 23:10:01 +00:00
validate_capsul_id ( id )
2021-02-16 00:16:15 +00:00
if get_ssh_host_keys :
ssh_host_keys = json . loads ( """ [
{ " key_type " : " ED25519 " , " content " : " ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIN8cna0zeKSKl/r8whdn/KmDWhdzuWRVV0GaKIM+eshh " , " sha256 " : " V4X2apAF6btGAfS45gmpldknoDX0ipJ5c6DLfZR2ttQ " } ,
{ " key_type " : " RSA " , " content " : " ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAABAQCvotgzgEP65JUQ8S8OoNKy1uEEPEAcFetSp7QpONe6hj4wPgyFNgVtdoWdNcU19dX3hpdse0G8OlaMUTnNVuRlbIZXuifXQ2jTtCFUA2mmJ5bF+XjGm3TXKMNGh9PN+wEPUeWd14vZL+QPUMev5LmA8cawPiU5+vVMLid93HRBj118aCJFQxLgrdP48VPfKHFRfCR6TIjg1ii3dH4acdJAvlmJ3GFB6ICT42EmBqskz2MPe0rIFxH8YohCBbAbrbWYcptHt4e48h4UdpZdYOhEdv89GrT8BF2C5cbQ5i9qVpI57bXKrj8hPZU5of48UHLSpXG8mbH0YDiOQOfKX/Mt " , " sha256 " : " ghee6KzRnBJhND2kEUZSaouk7CD6o6z2aAc8GPkV+GQ " } ,
{ " key_type " : " ECDSA " , " content " : " ecdsa-sha2-nistp256 AAAAE2VjZHNhLXNoYTItbmlzdHAyNTYAAAAIbmlzdHAyNTYAAABBBLLgOoATz9R4aS2kk7vWoxX+lshK63t9+5BIHdzZeFE1o+shlcf0Wji8cN/L1+m3bi0uSETZDOAWMP3rHLJj9Hk= " , " sha256 " : " aCYG1aD8cv/TjzJL0bi9jdabMGksdkfa7R8dCGm1yYs " }
] """ )
return VirtualMachine ( id , current_app . config [ " SPOKE_HOST_ID " ] , ipv4 = " 1.1.1.1 " , ssh_host_keys = ssh_host_keys )
2021-01-02 23:10:01 +00:00
return VirtualMachine ( id , current_app . config [ " SPOKE_HOST_ID " ] , ipv4 = " 1.1.1.1 " )
def list_ids ( self ) - > list :
return get_model ( ) . all_non_deleted_vm_ids ( )
2021-02-16 01:44:26 +00:00
def create ( self , email : str , id : str , template_image_file_name : str , vcpus : int , memory_mb : int , ssh_authorized_keys : list ) :
2021-01-02 23:10:01 +00:00
validate_capsul_id ( id )
current_app . logger . info ( f " mock create: { id } for { email } " )
sleep ( 1 )
def destroy ( self , email : str , id : str ) :
current_app . logger . info ( f " mock destroy: { id } for { email } " )
2021-02-17 03:13:51 +00:00
def vm_state_command ( self , email : str , id : str , command : str ) :
current_app . logger . info ( f " mock { command } : { id } for { email } " )
2021-01-02 23:10:01 +00:00
2021-01-03 21:19:29 +00:00
class CapsulFlaskHub ( VirtualizationInterface ) :
2021-01-04 19:32:52 +00:00
2021-01-04 21:02:56 +00:00
def synchronous_operation ( self , hosts : List [ OnlineHost ] , payload : str ) - > List [ HTTPResult ] :
return self . generic_operation ( hosts , payload , True ) [ 1 ]
def asynchronous_operation ( self , hosts : List [ OnlineHost ] , payload : str ) - > Tuple [ int , List [ HTTPResult ] ] :
return self . generic_operation ( hosts , payload , False )
2021-01-03 21:19:29 +00:00
def generic_operation ( self , hosts : List [ OnlineHost ] , payload : str , immediate_mode : bool ) - > Tuple [ int , List [ HTTPResult ] ] :
2021-01-04 21:02:56 +00:00
email = session [ " account " ]
if not email or email == " " :
raise ValueError ( " generic_operation was called but user was not logged in " )
url_path = " /spoke/operation "
operation_id = None
if not immediate_mode :
operation_id = get_model ( ) . create_operation ( hosts , email , payload )
url_path = f " /spoke/operation/ { operation_id } "
2021-01-04 01:17:30 +00:00
authorization_header = f " Bearer { current_app . config [ ' HUB_TOKEN ' ] } "
2021-02-16 05:51:59 +00:00
results = current_app . config [ " HTTP_CLIENT " ] . do_multi_http_sync ( hosts , url_path , payload , authorization_header = authorization_header )
2021-01-04 21:02:56 +00:00
2021-01-02 23:10:01 +00:00
for i in range ( len ( hosts ) ) :
host = hosts [ i ]
result = results [ i ]
assignment_status = " pending "
if result . status_code == - 1 :
assignment_status = " no_response_from_host "
if result . status_code != 200 :
assignment_status = " error_response_from_host "
else :
valid_statuses = {
" assigned " : True ,
" not_applicable " : True ,
" assigned_to_other_host " : True ,
}
result_is_json = False
result_is_dict = False
result_has_status = False
result_has_valid_status = False
assignment_status = " invalid_response_from_host "
2021-01-03 21:19:29 +00:00
error_message = " "
2021-01-02 23:10:01 +00:00
try :
result_body = json . loads ( result . body )
result_is_json = True
result_is_dict = isinstance ( result_body , dict )
result_has_status = result_is_dict and ' assignment_status ' in result_body
result_has_valid_status = result_has_status and result_body [ ' assignment_status ' ] in valid_statuses
if result_has_valid_status :
assignment_status = result_body [ ' assignment_status ' ]
2021-01-03 21:19:29 +00:00
if result_is_dict and " error_message " in result_body :
error_message = result_body [ ' error_message ' ]
2021-01-02 23:10:01 +00:00
except :
pass
if not result_has_valid_status :
2021-01-04 21:27:18 +00:00
operation_desc = " "
if operation_id :
operation_desc = f " for operation { operation_id } "
current_app . logger . error ( f """ error reading assignment_status { operation_desc } from host { host . id } :
2021-01-02 23:10:01 +00:00
result_is_json : { result_is_json }
result_is_dict : { result_is_dict }
result_has_status : { result_has_status }
result_has_valid_status : { result_has_valid_status }
2021-01-03 21:19:29 +00:00
error_message : { error_message }
2021-01-02 23:10:01 +00:00
"""
)
2021-01-04 21:02:56 +00:00
if not immediate_mode :
get_model ( ) . update_host_operation ( host . id , operation_id , assignment_status , None )
2021-01-02 23:10:01 +00:00
2021-01-04 21:02:56 +00:00
return ( operation_id , results )
2021-01-02 23:10:01 +00:00
2021-01-03 21:19:29 +00:00
def capacity_avaliable ( self , additional_ram_bytes ) :
2021-01-02 23:10:01 +00:00
online_hosts = get_model ( ) . get_online_hosts ( )
payload = json . dumps ( dict ( type = " capacity_avaliable " , additional_ram_bytes = additional_ram_bytes ) )
2021-01-04 21:02:56 +00:00
results = self . synchronous_operation ( online_hosts , payload )
2021-01-02 23:10:01 +00:00
for result in results :
try :
result_body = json . loads ( result . body )
if isinstance ( result_body , dict ) and ' capacity_avaliable ' in result_body and result_body [ ' capacity_avaliable ' ] == True :
return True
except :
pass
return False
2021-02-16 00:16:15 +00:00
def get ( self , id , get_ssh_host_keys ) - > VirtualMachine :
2021-01-02 23:10:01 +00:00
validate_capsul_id ( id )
host = get_model ( ) . host_of_capsul ( id )
if host is not None :
2021-02-16 00:16:15 +00:00
payload = json . dumps ( dict ( type = " get " , id = id , get_ssh_host_keys = get_ssh_host_keys ) )
2021-01-04 21:02:56 +00:00
results = self . synchronous_operation ( [ host ] , payload )
2021-01-02 23:10:01 +00:00
for result in results :
try :
result_body = json . loads ( result . body )
if isinstance ( result_body , dict ) and ( ' ipv4 ' in result_body or ' ipv6 ' in result_body ) :
2021-02-16 00:16:15 +00:00
return VirtualMachine ( id , host = host , ipv4 = result_body [ ' ipv4 ' ] , ipv6 = result_body [ ' ipv6 ' ] , ssh_host_keys = result_body [ ' ssh_host_keys ' ] )
2021-01-02 23:10:01 +00:00
except :
pass
return None
def list_ids ( self ) - > list :
online_hosts = get_model ( ) . get_online_hosts ( )
payload = json . dumps ( dict ( type = " list_ids " ) )
2021-01-04 21:02:56 +00:00
results = self . synchronous_operation ( online_hosts , payload )
2021-01-02 23:10:01 +00:00
to_return = [ ]
for i in range ( len ( results ) ) :
host = online_hosts [ i ]
result = results [ i ]
try :
result_body = json . loads ( result . body )
if isinstance ( result_body , dict ) and ' ids ' in result_body and isinstance ( result_body [ ' ids ' ] , list ) :
all_valid = True
for id in result_body [ ' ids ' ] :
try :
validate_capsul_id ( id )
to_return . append ( id )
except :
all_valid = False
2021-01-04 21:02:56 +00:00
if not all_valid :
2021-01-04 21:18:50 +00:00
current_app . logger . error ( f """ error reading ids for list_ids operation, host { host . id } """ )
2021-01-02 23:10:01 +00:00
else :
result_json_string = json . dumps ( { " error_message " : " invalid response, missing ' ids ' list " } )
2021-01-04 21:18:50 +00:00
current_app . logger . error ( f """ missing ' ids ' list for list_ids operation, host { host . id } """ )
2021-01-02 23:10:01 +00:00
except :
# no need to do anything here since if it cant be parsed then generic_operation will handle it.
pass
return to_return
2021-02-16 01:44:26 +00:00
def create ( self , email : str , id : str , template_image_file_name : str , vcpus : int , memory_mb : int , ssh_authorized_keys : list ) :
2021-01-02 23:10:01 +00:00
validate_capsul_id ( id )
online_hosts = get_model ( ) . get_online_hosts ( )
2021-02-16 03:00:34 +00:00
#current_app.logger.debug(f"hub_model.create(): ${len(online_hosts)} hosts")
2021-01-02 23:10:01 +00:00
payload = json . dumps ( dict (
type = " create " ,
email = email ,
id = id ,
template_image_file_name = template_image_file_name ,
vcpus = vcpus ,
memory_mb = memory_mb ,
2021-02-16 01:44:26 +00:00
ssh_authorized_keys = ssh_authorized_keys ,
2021-01-02 23:10:01 +00:00
) )
2021-01-04 21:02:56 +00:00
op = self . asynchronous_operation ( online_hosts , payload )
2021-01-02 23:10:01 +00:00
operation_id = op [ 0 ]
results = op [ 1 ]
number_of_assigned = 0
2021-01-03 20:44:56 +00:00
error_message = " "
2021-01-02 23:10:01 +00:00
assigned_hosts = [ ]
for i in range ( len ( results ) ) :
host = online_hosts [ i ]
result = results [ i ]
try :
result_body = json . loads ( result . body )
if isinstance ( result_body , dict ) and ' assignment_status ' in result_body and result_body [ ' assignment_status ' ] == " assigned " :
number_of_assigned + = 1
assigned_hosts . append ( host . id )
2021-01-03 20:44:56 +00:00
if isinstance ( result_body , dict ) and ' error_message ' in result_body :
error_message = result_body [ ' error_message ' ]
2021-01-02 23:10:01 +00:00
except :
# no need to do anything here since if it cant be parsed then generic_operation will handle it.
pass
if number_of_assigned != 1 :
assigned_hosts_string = " , " . join ( assigned_hosts )
raise ValueError ( f " expected create capsul operation { operation_id } to be assigned to one host, it was assigned to { number_of_assigned } ( { assigned_hosts_string } ) " )
2021-01-03 20:44:56 +00:00
if error_message != " " :
raise ValueError ( f " create capsul operation { operation_id } on { assigned_hosts_string } failed with { error_message } " )
2021-01-02 23:10:01 +00:00
def destroy ( self , email : str , id : str ) :
validate_capsul_id ( id )
result_status = None
host = get_model ( ) . host_of_capsul ( id )
if host is not None :
2021-01-04 21:27:18 +00:00
payload = json . dumps ( dict ( type = " destroy " , email = email , id = id ) )
2021-01-04 21:02:56 +00:00
results = self . synchronous_operation ( [ host ] , payload )
2021-01-02 23:10:01 +00:00
result_json_string = " <no response from host> "
for result in results :
try :
result_json_string = result . body
result_body = json . loads ( result_json_string )
if isinstance ( result_body , dict ) and ' status ' in result_body :
result_status = result_body [ ' status ' ]
except :
pass
if not result_status == " success " :
2021-01-04 21:27:18 +00:00
raise ValueError ( f """ failed to destroy vm " { id } " on host " { host . id } " for { email } : { result_json_string } """ )
2021-02-17 03:13:51 +00:00
def vm_state_command ( self , email : str , id : str , command : str ) :
validate_capsul_id ( id )
result_status = None
host = get_model ( ) . host_of_capsul ( id )
if host is not None :
payload = json . dumps ( dict ( type = " vm_state_command " , email = email , id = id , command = command ) )
results = self . synchronous_operation ( [ host ] , payload )
result_json_string = " <no response from host> "
for result in results :
try :
result_json_string = result . body
result_body = json . loads ( result_json_string )
if isinstance ( result_body , dict ) and ' status ' in result_body :
result_status = result_body [ ' status ' ]
except :
pass
if not result_status == " success " :
raise ValueError ( f """ failed to { command } vm " { id } " on host " { host . id } " for { email } : { result_json_string } """ )