ucloud-api/main.py

373 lines
12 KiB
Python

# TODO
# 1. Allow user of realm ungleich-admin to perform any action on
# any user vm.
import json
from helper import check_otp, add_otp_args, add_vmid_args
from flask import Flask
from flask_restful import Resource, Api, reqparse
from decouple import config
from uuid import uuid4
from etcd3_wrapper import Etcd3Wrapper
from specs_parser import SpecsParser
from functools import wraps
app = Flask(__name__)
api = Api(app)
client = Etcd3Wrapper()
# CreateVM argparser
createvm_argparser = reqparse.RequestParser()
createvm_argparser.add_argument("specs", type=dict, required=True)
createvm_argparser.add_argument("image_uuid", type=str, required=True)
add_otp_args(createvm_argparser)
# CreateImage argparser
createimage_argparser = reqparse.RequestParser()
createimage_argparser.add_argument("uuid", type=str, required=True)
createimage_argparser.add_argument("name", type=str, required=True)
createimage_argparser.add_argument("image_store", type=str, required=True)
# DeleteVM argparser
deletevm_argparser = reqparse.RequestParser()
add_vmid_args(add_otp_args(deletevm_argparser))
# VMStatus argparser
vmstatus_argparser = reqparse.RequestParser()
add_vmid_args(vmstatus_argparser)
# StartVM argparser
startvm_argparser = reqparse.RequestParser()
add_vmid_args(add_otp_args(startvm_argparser))
# UserVM argparser
uservm_argparser = reqparse.RequestParser()
add_otp_args(uservm_argparser)
# Specs parser
specs_parser = SpecsParser(exceptional_devices=["cpu"])
# AddHost argparser
addhost_argparser = reqparse.RequestParser()
addhost_argparser.add_argument("hostname", type=str, required=True)
addhost_argparser.add_argument("specs", type=dict, required=True)
add_otp_args(addhost_argparser)
def is_image_valid(image_uuid):
images = client.get_prefix("/v1/image/")
return image_uuid in [i.key.split("/")[-1] for i in images]
class CreateVM(Resource):
def post(self):
createvm_args = createvm_argparser.parse_args()
name, realm, token, specs, image_uuid = (
createvm_args.name,
createvm_args.realm,
createvm_args.token,
createvm_args.specs,
createvm_args.image_uuid,
)
if check_otp(name, realm, token) == 200:
# User is good
if is_image_valid(image_uuid):
if not specs_parser.transform_specs(specs):
return (
{
"message": f"""Invalid unit - Please use following units {specs_parser.get_allowed_units()}"""
},
400,
)
print(specs)
vm_key = f"/v1/vm/{uuid4().hex}"
vm_entry = {
"owner": name,
"specs": specs,
"hostname": "",
"status": "REQUESTED_NEW",
"image_uuid": image_uuid,
}
client.put(vm_key, vm_entry, value_in_json=True)
return {"message": "VM Creation Queued"}, 200
else:
return {"message": "Image uuid not valid"}
else:
return {"message": "Invalid Credentials"}, 400
class DeleteVM(Resource):
def post(self):
deletevm_args = deletevm_argparser.parse_args()
name, realm, token, vmid = (
deletevm_args.name,
deletevm_args.realm,
deletevm_args.token,
deletevm_args.vmid,
)
if check_otp(name, realm, token) == 200:
# User is good
vmentry_etcd = client.get(f"/v1/vm/{vmid}").value
if vmentry_etcd:
vmentry_etcd = json.loads(vmentry_etcd)
vmentry_etcd["status"] = "REQUESTED_DELETE"
client.put(f"/v1/vm/{vmid}", vmentry_etcd,
value_in_json=True)
return {"message": "VM Deletion Queued"}, 200
else:
return {"message": "Invalid VM ID"}
else:
return {"message": "Invalid Credentials"}, 400
class VmStatus(Resource):
def get(self):
args = vmstatus_argparser.parse_args()
r = client.get(f"/v1/vm/{args.vmid}").value
print(r)
if r:
r = dict(json.loads(r.decode("utf-8")))
return r
return {"Message": "Not Found"}
class CreateImage(Resource):
def post(self):
image_stores = list(client.get_prefix("/v1/image_store/"))
args = createimage_argparser.parse_args()
image_file_uuid = args.uuid
image_store_name = args.image_store
file_entry = client.get(f"/v1/files/{image_file_uuid}")
if file_entry is None:
return (
{"Message": f"Image File with uuid '{image_file_uuid}' Not Found"},
400,
)
file_entry_value = json.loads(file_entry.value)
image_store = list(
filter(
lambda s: json.loads(s.value)["name"] == image_store_name, image_stores
)
)
if not image_store:
return {"Message": f"Store '{image_store_name}' does not exists"}, 400
image_store = image_store[0]
image_entry_json = {
"status": "TO_BE_CREATED",
"owner": file_entry_value["owner"],
"filename": file_entry_value["filename"],
"name": args.name,
"store_name": image_store_name,
"visibility": "public",
}
client.put(f"/v1/image/{image_file_uuid}", json.dumps(image_entry_json))
return {"Message": "Image successfully created"}
class ListPublicImages(Resource):
def get(self):
images = client.get_prefix("/v1/image/")
r = {}
for image in images:
r[image.key.split("/")[-1]] = json.loads(image.value)
return r, 200
class StartVM(Resource):
def post(self):
args = startvm_argparser.parse_args()
name, realm, token, vm_uuid = args.name, args.realm, args.token, args.vmid
if check_otp(name, realm, token) == 200:
vm = client.get(f"/v1/vm/{vm_uuid}", value_in_json=True)
if vm:
if vm.value["owner"] != name:
return {"message": "Invalid User"}
vm.value["status"] = "REQUESTED_START"
client.put(vm.key, json.dumps(vm.value))
return {"message": f"VM Start Queued"}
else:
return {"message": "No such VM found"}
else:
return {"message": "Invalid Credentials"}, 400
class SuspendVM(Resource):
def post(self):
args = startvm_argparser.parse_args()
name, realm, token, vm_uuid = args.name, args.realm, args.token, args.vmid
if check_otp(name, realm, token) == 200:
vm = client.get(f"/v1/vm/{vm_uuid}", value_in_json=True)
if vm:
if vm.value["owner"] != name:
return {"message": "Invalid User"}
vm.value["status"] = "REQUESTED_SUSPEND"
client.put(vm.key, json.dumps(vm.value))
return {"message": f"VM Suspension Queued"}
else:
return {"message": "No such VM found"}
else:
return {"message": "Invalid Credentials"}, 400
class ResumeVM(Resource):
def post(self):
args = startvm_argparser.parse_args()
name, realm, token, vm_uuid = args.name, args.realm, args.token, args.vmid
if check_otp(name, realm, token) == 200:
vm = client.get(f"/v1/vm/{vm_uuid}", value_in_json=True)
if vm:
if vm.value["owner"] != name:
return {"message": "Invalid User"}
vm.value["status"] = "REQUESTED_RESUME"
client.put(vm.key, json.dumps(vm.value))
return {"message": f"VM Resume Queued"}
else:
return {"message": "No such VM found"}
else:
return {"message": "Invalid Credentials"}, 400
class ShutdownVM(Resource):
def post(self):
args = startvm_argparser.parse_args()
name, realm, token, vm_uuid = args.name, args.realm, args.token, args.vmid
if check_otp(name, realm, token) == 200:
vm = client.get(f"/v1/vm/{vm_uuid}", value_in_json=True)
if vm:
if vm.value["owner"] != name:
return {"message": "Invalid User"}
vm.value["status"] = "REQUESTED_SHUTDOWN"
client.put(vm.key, json.dumps(vm.value))
return {"message": f"VM Shutdown Queued"}
else:
return {"message": "No such VM found"}
else:
return {"message": "Invalid Credentials"}, 400
class ListUserVM(Resource):
def get(self):
args = uservm_argparser.parse_args()
name, realm, token = args.name, args.realm, args.token
if check_otp(name, realm, token) == 200:
vms = client.get_prefix(f"/v1/vm/", value_in_json=True)
if vms:
return_vms = []
user_vms = list(filter(lambda v: v.value["owner"] == name, vms))
for vm in user_vms:
return_vms.append(
{
"vm_uuid": vm.key.split("/")[-1],
"specs": vm.value["specs"],
"status": vm.value["status"],
}
)
return {"message": return_vms}, 200
else:
return {"message": "No VM found"}, 404
else:
return {"message": "Invalid Credentials"}, 400
class ListUserFiles(Resource):
def get(self):
args = uservm_argparser.parse_args()
name, realm, token = args.name, args.realm, args.token
if check_otp(name, realm, token) == 200:
files = client.get_prefix(f"/v1/files/", value_in_json=True)
if files:
return_files = []
user_files = list(filter(lambda f: f.value["owner"] == name, files))
for file in user_files:
return_files.append(
{
"filename": file.value["filename"],
"uuid": file.key.split("/")[-1],
}
)
return {"message": return_files}, 200
else:
return {"message": "No File found"}, 404
else:
return {"message": "Invalid Credentials"}, 400
class CreateHost(Resource):
def post(self):
args = addhost_argparser.parse_args()
name, realm, token, specs, hostname = (
args.name,
args.realm,
args.token,
args.specs,
args.hostname,
)
if realm == "ungleich-admin" and check_otp(name, realm, token) == 200:
# User is good
if not specs_parser.transform_specs(specs):
return (
{
"message": f"""Invalid unit - Please use following units {specs_parser.get_allowed_units()}"""
},
400,
)
print(specs)
host_key = f"/v1/host/{uuid4().hex}"
host_entry = {
"specs": specs,
"hostname": hostname,
"status": "DEAD",
"last_heartbeat": "",
}
client.put(host_key, host_entry, value_in_json=True)
return {"message": "Host Created"}, 200
else:
return {"message": "Invalid Credentials/Insufficient Permission"}, 400
api.add_resource(CreateVM, "/vm/create")
api.add_resource(DeleteVM, "/vm/delete")
api.add_resource(VmStatus, "/vm/status")
api.add_resource(StartVM, "/vm/start")
api.add_resource(SuspendVM, "/vm/suspend")
api.add_resource(ResumeVM, "/vm/resume")
api.add_resource(ShutdownVM, "/vm/shutdown")
api.add_resource(CreateImage, "/image/create")
api.add_resource(ListPublicImages, "/image/list-public")
api.add_resource(ListUserVM, "/user/vms")
api.add_resource(ListUserFiles, "/user/files")
api.add_resource(CreateHost, "/host/create")
if __name__ == "__main__":
app.run(host="::", debug=True)