uncloud-mravi/uncloud/api/main.py

489 lines
17 KiB
Python
Raw Normal View History

import json
import pynetbox
import logging
2020-01-03 18:38:59 +05:00
import argparse
from uuid import uuid4
from os.path import join as join_path
from flask import Flask, request
from flask_restful import Resource, Api
from werkzeug.exceptions import HTTPException
from uncloud.common.shared import shared
from uncloud.common import counters
from uncloud.common.vm import VMStatus
2020-01-10 00:03:10 +05:00
from uncloud.common.host import HostStatus
from uncloud.common.request import RequestEntry, RequestType
2020-01-06 12:25:59 +05:00
from uncloud.common.settings import settings
2020-01-10 00:03:10 +05:00
from uncloud.api import schemas
from uncloud.api.schemas import ValidationException
from uncloud.common.network import generate_mac, mac2ipv6
from uncloud.api.helper import make_return_message
from uncloud import UncloudException
logger = logging.getLogger(__name__)
app = Flask(__name__)
api = Api(app)
app.logger.handlers.clear()
2020-01-03 18:38:59 +05:00
arg_parser = argparse.ArgumentParser('api', add_help=False)
arg_parser.add_argument('--port', '-p')
@app.errorhandler(Exception)
def handle_exception(e):
app.logger.error(e)
2020-01-10 00:03:10 +05:00
# pass through HTTP errors
if isinstance(e, HTTPException):
return e
# now you're handling non-HTTP exceptions only
2020-01-03 18:38:59 +05:00
return {'message': 'Server Error'}, 500
class CreateVM(Resource):
@staticmethod
def post():
data = request.json
2020-01-10 00:03:10 +05:00
try:
validator = schemas.CreateVMSchema(data)
validator.is_valid()
except ValidationException as err:
return make_return_message(err, 400)
else:
vm_uuid = uuid4().hex
2020-01-03 18:38:59 +05:00
vm_key = join_path(settings['etcd']['vm_prefix'], vm_uuid)
specs = {
2020-01-03 18:38:59 +05:00
'cpu': validator.specs['cpu'],
'ram': validator.specs['ram'],
'os-ssd': validator.specs['os-ssd'],
'hdd': validator.specs['hdd'],
}
2020-01-10 00:03:10 +05:00
macs = [generate_mac() for _ in range(len(validator.network.value))]
tap_ids = [
2020-01-10 00:03:10 +05:00
counters.increment_etcd_counter(settings['etcd']['tap_counter'])
for _ in range(len(validator.network.value))
]
vm_entry = {
2020-01-10 00:03:10 +05:00
'name': validator.vm_name.value,
'owner': validator.name.value,
'owner_realm': validator.realm.value,
2020-01-03 18:38:59 +05:00
'specs': specs,
'hostname': '',
'status': VMStatus.stopped,
'image_uuid': validator.image_uuid,
'log': [],
'vnc_socket': '',
2020-01-10 00:03:10 +05:00
'network': list(zip(validator.network.value, macs, tap_ids)),
2020-01-03 18:38:59 +05:00
'metadata': {'ssh-keys': []},
'in_migration': False,
}
shared.etcd_client.put(vm_key, vm_entry, value_in_json=True)
# Create ScheduleVM Request
r = RequestEntry.from_scratch(
type=RequestType.ScheduleVM,
uuid=vm_uuid,
2020-01-03 18:38:59 +05:00
request_prefix=settings['etcd']['request_prefix'],
)
shared.request_pool.put(r)
2020-01-10 00:03:10 +05:00
return make_return_message('VM Creation Queued')
class VmStatus(Resource):
@staticmethod
2019-12-31 20:33:55 +05:00
def post():
data = request.json
2020-01-10 00:03:10 +05:00
try:
validator = schemas.VMStatusSchema(data)
validator.is_valid()
except (ValidationException, Exception) as err:
return make_return_message(err, 400)
else:
vm = shared.vm_pool.get(join_path(settings['etcd']['vm_prefix'], data['uuid']))
vm_value = vm.value.copy()
2020-01-03 18:38:59 +05:00
vm_value['ip'] = []
for network_mac_and_tap in vm.network:
network_name, mac, tap = network_mac_and_tap
network = shared.etcd_client.get(
2020-01-10 00:03:10 +05:00
join_path(settings['etcd']['network_prefix'], data['name'], network_name),
value_in_json=True,
)
2020-01-10 00:03:10 +05:00
ipv6_addr = (network.value.get('ipv6').split('::')[0] + '::')
2020-01-03 18:38:59 +05:00
vm_value['ip'].append(mac2ipv6(mac, ipv6_addr))
vm.value = vm_value
2020-01-10 00:03:10 +05:00
return vm.value, 200
class CreateImage(Resource):
@staticmethod
def post():
data = request.json
2020-01-10 00:03:10 +05:00
try:
validator = schemas.CreateImageSchema(data)
validator.is_valid()
except ValidationException as err:
return make_return_message(err, 400)
else:
try:
file_entry = shared.etcd_client.get(
join_path(settings['etcd']['file_prefix'], data['uuid']), value_in_json=True
)
except KeyError:
# TODO: Add some message
pass
else:
image_entry_json = {
'status': 'TO_BE_CREATED',
'owner': file_entry.value['owner'],
'filename': file_entry.value['filename'],
'name': validator.name,
'store_name': validator.image_store,
'visibility': 'public',
}
shared.etcd_client.put(
join_path(settings['etcd']['image_prefix'], data['uuid']),
json.dumps(image_entry_json),
)
2020-01-10 00:03:10 +05:00
return {'message': 'Image queued for creation.'}, 200
class ListPublicImages(Resource):
@staticmethod
def get():
2020-01-10 00:03:10 +05:00
images = shared.etcd_client.get_prefix(settings['etcd']['image_prefix'], value_in_json=True)
2020-01-03 18:38:59 +05:00
r = {'images': []}
for image in images:
2020-01-10 00:03:10 +05:00
image_key = '{}:{}'.format(image.value['store_name'], image.value['name'])
r['images'].append({'name': image_key, 'status': image.value['status']})
return r, 200
class VMAction(Resource):
@staticmethod
def post():
data = request.json
2020-01-10 00:03:10 +05:00
try:
validator = schemas.VmActionSchema(data)
validator.is_valid()
except ValidationException as err:
return make_return_message(err, 400)
else:
vm_entry = shared.vm_pool.get(join_path(settings['etcd']['vm_prefix'], data['uuid']))
action = validator.action.value
2020-01-03 18:38:59 +05:00
if action == 'start':
action = 'schedule'
2020-01-03 18:38:59 +05:00
if action == 'delete' and vm_entry.hostname == '':
2020-01-10 00:03:10 +05:00
if shared.storage_handler.is_vm_image_exists(vm_entry.uuid):
r_status = shared.storage_handler.delete_vm_image(vm_entry.uuid)
if r_status:
shared.etcd_client.client.delete(vm_entry.key)
2020-01-10 00:03:10 +05:00
return make_return_message('VM successfully deleted')
else:
2020-01-10 00:03:10 +05:00
logger.error('Some Error Occurred while deleting VM')
return make_return_message('VM deletion unsuccessfull')
else:
shared.etcd_client.client.delete(vm_entry.key)
2020-01-10 00:03:10 +05:00
return make_return_message('VM successfully deleted')
r = RequestEntry.from_scratch(
2020-01-03 18:38:59 +05:00
type='{}VM'.format(action.title()),
uuid=data['uuid'],
hostname=vm_entry.hostname,
2020-01-03 18:38:59 +05:00
request_prefix=settings['etcd']['request_prefix'],
)
shared.request_pool.put(r)
2020-01-10 00:03:10 +05:00
return make_return_message('VM {} Queued'.format(action.title()))
class VMMigration(Resource):
@staticmethod
def post():
data = request.json
2020-01-10 00:03:10 +05:00
try:
validator = schemas.VmMigrationSchema(data)
validator.is_valid()
except ValidationException as err:
return make_return_message(err), 400
else:
vm = shared.vm_pool.get(validator.uuid.value)
r = RequestEntry.from_scratch(
type=RequestType.InitVMMigration,
uuid=vm.uuid,
hostname=join_path(
2020-01-03 18:38:59 +05:00
settings['etcd']['host_prefix'],
validator.destination.value,
),
2020-01-03 18:38:59 +05:00
request_prefix=settings['etcd']['request_prefix'],
)
shared.request_pool.put(r)
2020-01-10 00:03:10 +05:00
return make_return_message('VM Migration Initialization Queued')
class ListUserVM(Resource):
@staticmethod
2019-12-31 20:33:55 +05:00
def post():
data = request.json
2020-01-10 00:03:10 +05:00
try:
validator = schemas.OTPSchema(data)
validator.is_valid()
except ValidationException as err:
return make_return_message(err, 400)
else:
vms = shared.etcd_client.get_prefix(settings['etcd']['vm_prefix'], value_in_json=True)
return_vms = []
2020-01-10 00:03:10 +05:00
user_vms = filter(lambda v: v.value['owner'] == validator.name.value, vms)
for vm in user_vms:
return_vms.append(
{
2020-01-03 18:38:59 +05:00
'name': vm.value['name'],
'vm_uuid': vm.key.split('/')[-1],
'specs': vm.value['specs'],
'status': vm.value['status'],
'hostname': vm.value['hostname'],
'vnc_socket': vm.value.get('vnc_socket', None),
}
)
2020-01-10 00:03:10 +05:00
return make_return_message(return_vms)
class ListUserFiles(Resource):
@staticmethod
2019-12-31 20:33:55 +05:00
def post():
data = request.json
2020-01-10 00:03:10 +05:00
try:
validator = schemas.OTPSchema(data)
validator.is_valid()
except ValidationException as err:
return make_return_message(err, 400)
else:
files = shared.etcd_client.get_prefix(settings['etcd']['file_prefix'], value_in_json=True)
return_files = []
user_files = [f for f in files if f.value['owner'] == data['name']]
for file in user_files:
file_uuid = file.key.split('/')[-1]
file = file.value
file['uuid'] = file_uuid
file.pop('sha512sum', None)
file.pop('owner', None)
return_files.append(file)
2020-01-10 00:03:10 +05:00
return make_return_message(return_files)
class CreateHost(Resource):
@staticmethod
def post():
data = request.json
2020-01-10 00:03:10 +05:00
try:
validator = schemas.CreateHostSchema(data)
validator.is_valid()
except ValidationException as err:
return make_return_message(err, 400)
else:
host_key = join_path(settings['etcd']['host_prefix'], uuid4().hex)
host_entry = {
2020-01-10 00:03:10 +05:00
'specs': validator.specs.value,
'hostname': validator.hostname.value,
'status': HostStatus.dead,
2020-01-03 18:38:59 +05:00
'last_heartbeat': '',
}
2020-01-10 00:03:10 +05:00
shared.etcd_client.put(host_key, host_entry, value_in_json=True)
return make_return_message('Host Created.')
class ListHost(Resource):
@staticmethod
def get():
hosts = shared.host_pool.hosts
r = {
host.key: {
2020-01-03 18:38:59 +05:00
'status': host.status,
'specs': host.specs,
'hostname': host.hostname,
}
for host in hosts
}
return r, 200
class GetSSHKeys(Resource):
@staticmethod
2019-12-31 20:33:55 +05:00
def post():
data = request.json
2020-01-10 00:03:10 +05:00
try:
validator = schemas.GetSSHSchema(data)
except ValidationException as err:
return make_return_message(err, 400)
else:
etcd_key = join_path(settings['etcd']['user_prefix'], validator.realm.value,
validator.name.value, 'key')
if not validator.key_name.value:
2020-01-10 00:03:10 +05:00
etcd_entry = shared.etcd_client.get_prefix(etcd_key, value_in_json=True)
keys = {
2020-01-03 18:38:59 +05:00
key.key.split('/')[-1]: key.value
for key in etcd_entry
}
2020-01-03 18:38:59 +05:00
return {'keys': keys}
else:
2020-01-10 00:03:10 +05:00
etcd_key = join_path(validator.key_name.value)
try:
etcd_entry = shared.etcd_client.get(etcd_key, value_in_json=True)
except KeyError:
return make_return_message('No such key found.', 400)
else:
return {
2020-01-10 00:03:10 +05:00
'keys': {etcd_entry.key.split('/')[-1]: etcd_entry.value}
}
class AddSSHKey(Resource):
@staticmethod
def post():
data = request.json
2020-01-10 00:03:10 +05:00
try:
validator = schemas.AddSSHSchema(data)
validator.is_valid()
except ValidationException as err:
return make_return_message(err, 400)
else:
# {user_prefix}/{realm}/{name}/key/{key_name}
etcd_key = join_path(
2020-01-10 00:03:10 +05:00
settings['etcd']['user_prefix'], validator.realm.value,
validator.name.value, 'key', validator.key_name.value
)
2020-01-10 00:03:10 +05:00
try:
shared.etcd_client.get(etcd_key, value_in_json=True)
except KeyError:
# Key Not Found. It implies user' haven't added any key yet.
2020-01-10 00:03:10 +05:00
shared.etcd_client.put(etcd_key, validator.key.value, value_in_json=True)
return make_return_message('Key added successfully')
else:
return make_return_message('Key "{}" already exists'.format(validator.key_name.value))
class RemoveSSHKey(Resource):
@staticmethod
2019-12-31 20:33:55 +05:00
def post():
data = request.json
2020-01-10 00:03:10 +05:00
try:
validator = schemas.RemoveSSHSchema(data)
validator.is_valid()
except ValidationException as err:
return make_return_message(err, 400)
else:
# {user_prefix}/{realm}/{name}/key/{key_name}
2020-01-10 00:03:10 +05:00
etcd_key = join_path(settings['etcd']['user_prefix'], validator.realm.value,
validator.name.value, 'key', validator.key_name.value)
try:
etcd_entry = shared.etcd_client.get(etcd_key, value_in_json=True)
except KeyError:
return make_return_message('No Key "{}" exists.'.format(validator.key_name.value))
if etcd_entry:
shared.etcd_client.client.delete(etcd_key)
2020-01-03 18:38:59 +05:00
return {'message': 'Key successfully removed.'}
class CreateNetwork(Resource):
@staticmethod
def post():
data = request.json
2020-01-10 00:03:10 +05:00
try:
validator = schemas.CreateNetwork(data)
validator.is_valid()
except ValidationException as err:
return make_return_message(err, 400)
else:
network_entry = {
2020-01-10 00:03:10 +05:00
'id': counters.increment_etcd_counter(settings['etcd']['vxlan_counter']),
'type': validator.type.value,
}
2019-11-15 21:11:45 +05:00
if validator.user.value:
try:
2020-01-10 00:03:10 +05:00
nb = pynetbox.api(url=settings['netbox']['url'], token=settings['netbox']['token'])
nb_prefix = nb.ipam.prefixes.get(prefix=settings['network']['prefix'])
prefix = nb_prefix.available_prefixes.create(
data={
2020-01-10 00:03:10 +05:00
'prefix_length': int(settings['network']['prefix_length']),
2020-01-03 18:38:59 +05:00
'description': '{}\'s network "{}"'.format(
2020-01-10 00:03:10 +05:00
validator.name.value,
validator.network_name.value
),
2020-01-03 18:38:59 +05:00
'is_pool': True,
}
)
except Exception as err:
app.logger.error(err)
2020-01-10 00:03:10 +05:00
return make_return_message('Error occured while creating network.', 400)
else:
2020-01-03 18:38:59 +05:00
network_entry['ipv6'] = prefix['prefix']
2019-11-15 21:11:45 +05:00
else:
2020-01-03 18:38:59 +05:00
network_entry['ipv6'] = 'fd00::/64'
2020-01-10 00:03:10 +05:00
network_key = join_path(settings['etcd']['network_prefix'], validator.name.value,
validator.network_name.value)
shared.etcd_client.put(network_key, network_entry, value_in_json=True)
return make_return_message('Network successfully added.')
class ListUserNetwork(Resource):
@staticmethod
2019-12-31 20:33:55 +05:00
def post():
data = request.json
2020-01-10 00:03:10 +05:00
try:
validator = schemas.OTPSchema(data)
validator.is_valid()
except ValidationException as err:
return make_return_message(err, 400)
else:
prefix = join_path(settings['etcd']['network_prefix'], data['name'])
networks = shared.etcd_client.get_prefix(prefix, value_in_json=True)
user_networks = []
for net in networks:
2020-01-03 18:38:59 +05:00
net.value['name'] = net.key.split('/')[-1]
user_networks.append(net.value)
2020-01-03 18:38:59 +05:00
return {'networks': user_networks}, 200
2020-01-03 18:38:59 +05:00
api.add_resource(CreateVM, '/vm/create')
api.add_resource(VmStatus, '/vm/status')
2020-01-03 18:38:59 +05:00
api.add_resource(VMAction, '/vm/action')
api.add_resource(VMMigration, '/vm/migrate')
2020-01-03 18:38:59 +05:00
api.add_resource(CreateImage, '/image/create')
api.add_resource(ListPublicImages, '/image/list-public')
2020-01-03 18:38:59 +05:00
api.add_resource(ListUserVM, '/user/vms')
api.add_resource(ListUserFiles, '/user/files')
api.add_resource(ListUserNetwork, '/user/networks')
2020-01-03 18:38:59 +05:00
api.add_resource(AddSSHKey, '/user/add-ssh')
api.add_resource(RemoveSSHKey, '/user/remove-ssh')
api.add_resource(GetSSHKeys, '/user/get-ssh')
2020-01-03 18:38:59 +05:00
api.add_resource(CreateHost, '/host/create')
api.add_resource(ListHost, '/host/list')
2020-01-03 18:38:59 +05:00
api.add_resource(CreateNetwork, '/network/create')
def main(debug=False, port=None):
try:
2020-01-10 00:03:10 +05:00
app.run(host='::', port=port, debug=debug)
except OSError as e:
2020-01-03 18:38:59 +05:00
raise UncloudException('Failed to start Flask: {}'.format(e))
2020-01-03 18:38:59 +05:00
if __name__ == '__main__':
main()