# TODO # 1. Allow user of realm ungleich-admin to perform any action on # any user vm. import json from flask import Flask, request from flask_restful import Resource, Api from uuid import uuid4 from etcd3_wrapper import Etcd3Wrapper from ucloud_common.vm import VmPool, VMStatus from schemas import (CreateVMSchema, VMStatusSchema, CreateImageSchema, VmActionSchema, OTPSchema, CreateHostSchema) app = Flask(__name__) api = Api(app) client = Etcd3Wrapper() vm_pool = VmPool(client, "/v1/vm") class CreateVM(Resource): def post(self): data = request.json validator = CreateVMSchema(data) if validator.is_valid(): vm_key = f"/v1/vm/{uuid4().hex}" vm_entry = { "owner": data["name"], "specs": data["specs"], "hostname": "", "status": "REQUESTED_NEW", "image_uuid": data["image_uuid"], } client.put(vm_key, vm_entry, value_in_json=True) return {"message": "VM Creation Queued"}, 200 else: return validator.get_errors(), 400 class VmStatus(Resource): def get(self): data = request.json validator = VMStatusSchema(data) if validator.is_valid(): vm = vm_pool.get(f"/v1/vm/{data['uuid']}") return str(vm) else: return validator.get_errors(), 400 class CreateImage(Resource): def post(self): data = request.json validator = CreateImageSchema(data) if validator.is_valid(): file_entry = client.get(f"/v1/file/{data['uuid']}") file_entry_value = json.loads(file_entry.value) image_entry_json = { "status": "TO_BE_CREATED", "owner": file_entry_value["owner"], "filename": file_entry_value["filename"], "name": data["name"], "store_name": data["image_store"], "visibility": "public", } client.put(f"/v1/image/{data['uuid']}", json.dumps(image_entry_json)) return {"Message": "Image successfully created"} else: return validator.get_errors(), 400 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 VMAction(Resource): def post(self): data = request.json validator = VmActionSchema(data) action = data["action"] if validator.is_valid(): vm = vm_pool.get(f"/v1/vm/{data['uuid']}") if action == "start": vm.status = VMStatus.requested_start elif action == "suspend": vm.status = VMStatus.requested_suspend elif action == "resume": vm.status = VMStatus.requested_resume elif action == "shutdown": vm.status = VMStatus.requested_shutdown elif action == "delete": vm.status = VMStatus.requested_delete client.put(vm.key, json.dumps(vm.value)) return {"message": f"VM Start Queued"} else: return validator.get_errors(), 400 class ListUserVM(Resource): def get(self): data = request.json validator = OTPSchema(data) if validator.is_valid(): vms = client.get_prefix(f"/v1/vm/", value_in_json=True) if vms: return_vms = [] user_vms = list(filter(lambda v: v.value["owner"] == data["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 validator.get_errors(), 400 class ListUserFiles(Resource): def get(self): data = request.json validator = OTPSchema(data) if validator.is_valid(): files = client.get_prefix(f"/v1/file/", value_in_json=True) if files: return_files = [] user_files = list(filter(lambda f: f.value["owner"] == data["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 validator.get_errors(), 400 class CreateHost(Resource): def post(self): data = request.json validator = CreateHostSchema(data) if validator.is_valid(): host_key = f"/v1/host/{uuid4().hex}" host_entry = { "specs": data["specs"], "hostname": data["hostname"], "status": "DEAD", "last_heartbeat": "", } client.put(host_key, host_entry, value_in_json=True) return {"message": "Host Created"}, 200 else: return validator.get_errors(), 400 api.add_resource(CreateVM, "/vm/create") api.add_resource(VmStatus, "/vm/status") api.add_resource(VMAction, "/vm/action") 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)