forked from uncloud/uncloud
605 lines
19 KiB
Python
605 lines
19 KiB
Python
import json
|
|
import pynetbox
|
|
import logging
|
|
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 import counters
|
|
from uncloud.common.vm import VMStatus
|
|
from uncloud.common.request import RequestEntry, RequestType
|
|
from uncloud.settings import settings
|
|
from uncloud.shared import shared
|
|
from . import schemas
|
|
from .helper import generate_mac, mac2ipv6
|
|
from uncloud import UncloudException
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
app = Flask(__name__)
|
|
api = Api(app)
|
|
app.logger.handlers.clear()
|
|
|
|
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)
|
|
# pass through HTTP errors
|
|
if isinstance(e, HTTPException):
|
|
return e
|
|
|
|
# now you're handling non-HTTP exceptions only
|
|
return {'message': 'Server Error'}, 500
|
|
|
|
|
|
class CreateVM(Resource):
|
|
'''API Request to Handle Creation of VM'''
|
|
|
|
@staticmethod
|
|
def post():
|
|
data = request.json
|
|
validator = schemas.CreateVMSchema(data)
|
|
if validator.is_valid():
|
|
vm_uuid = uuid4().hex
|
|
vm_key = join_path(settings['etcd']['vm_prefix'], vm_uuid)
|
|
specs = {
|
|
'cpu': validator.specs['cpu'],
|
|
'ram': validator.specs['ram'],
|
|
'os-ssd': validator.specs['os-ssd'],
|
|
'hdd': validator.specs['hdd'],
|
|
}
|
|
macs = [generate_mac() for _ in range(len(data['network']))]
|
|
tap_ids = [
|
|
counters.increment_etcd_counter(
|
|
shared.etcd_client, '/v1/counter/tap'
|
|
)
|
|
for _ in range(len(data['network']))
|
|
]
|
|
vm_entry = {
|
|
'name': data['vm_name'],
|
|
'owner': data['name'],
|
|
'owner_realm': data['realm'],
|
|
'specs': specs,
|
|
'hostname': '',
|
|
'status': VMStatus.stopped,
|
|
'image_uuid': validator.image_uuid,
|
|
'log': [],
|
|
'vnc_socket': '',
|
|
'network': list(zip(data['network'], macs, tap_ids)),
|
|
'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,
|
|
request_prefix=settings['etcd']['request_prefix'],
|
|
)
|
|
shared.request_pool.put(r)
|
|
|
|
return {'message': 'VM Creation Queued'}, 200
|
|
return validator.get_errors(), 400
|
|
|
|
|
|
class VmStatus(Resource):
|
|
@staticmethod
|
|
def post():
|
|
data = request.json
|
|
validator = schemas.VMStatusSchema(data)
|
|
if validator.is_valid():
|
|
vm = shared.vm_pool.get(
|
|
join_path(settings['etcd']['vm_prefix'], data['uuid'])
|
|
)
|
|
vm_value = vm.value.copy()
|
|
vm_value['ip'] = []
|
|
for network_mac_and_tap in vm.network:
|
|
network_name, mac, tap = network_mac_and_tap
|
|
network = shared.etcd_client.get(
|
|
join_path(
|
|
settings['etcd']['network_prefix'],
|
|
data['name'],
|
|
network_name,
|
|
),
|
|
value_in_json=True,
|
|
)
|
|
ipv6_addr = (
|
|
network.value.get('ipv6').split('::')[0] + '::'
|
|
)
|
|
vm_value['ip'].append(mac2ipv6(mac, ipv6_addr))
|
|
vm.value = vm_value
|
|
return vm.value
|
|
else:
|
|
return validator.get_errors(), 400
|
|
|
|
|
|
class CreateImage(Resource):
|
|
@staticmethod
|
|
def post():
|
|
data = request.json
|
|
validator = schemas.CreateImageSchema(data)
|
|
if validator.is_valid():
|
|
file_entry = shared.etcd_client.get(
|
|
join_path(settings['etcd']['file_prefix'], 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',
|
|
}
|
|
shared.etcd_client.put(
|
|
join_path(
|
|
settings['etcd']['image_prefix'], data['uuid']
|
|
),
|
|
json.dumps(image_entry_json),
|
|
)
|
|
|
|
return {'message': 'Image queued for creation.'}
|
|
return validator.get_errors(), 400
|
|
|
|
|
|
class ListPublicImages(Resource):
|
|
@staticmethod
|
|
def get():
|
|
images = shared.etcd_client.get_prefix(
|
|
settings['etcd']['image_prefix'], value_in_json=True
|
|
)
|
|
r = {'images': []}
|
|
for image in images:
|
|
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
|
|
validator = schemas.VmActionSchema(data)
|
|
|
|
if validator.is_valid():
|
|
vm_entry = shared.vm_pool.get(
|
|
join_path(settings['etcd']['vm_prefix'], data['uuid'])
|
|
)
|
|
action = data['action']
|
|
|
|
if action == 'start':
|
|
action = 'schedule'
|
|
|
|
if action == 'delete' and vm_entry.hostname == '':
|
|
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)
|
|
return {'message': 'VM successfully deleted'}
|
|
else:
|
|
logger.error(
|
|
'Some Error Occurred while deleting VM'
|
|
)
|
|
return {'message': 'VM deletion unsuccessfull'}
|
|
else:
|
|
shared.etcd_client.client.delete(vm_entry.key)
|
|
return {'message': 'VM successfully deleted'}
|
|
|
|
r = RequestEntry.from_scratch(
|
|
type='{}VM'.format(action.title()),
|
|
uuid=data['uuid'],
|
|
hostname=vm_entry.hostname,
|
|
request_prefix=settings['etcd']['request_prefix'],
|
|
)
|
|
shared.request_pool.put(r)
|
|
return (
|
|
{'message': 'VM {} Queued'.format(action.title())},
|
|
200,
|
|
)
|
|
else:
|
|
return validator.get_errors(), 400
|
|
|
|
|
|
class VMMigration(Resource):
|
|
@staticmethod
|
|
def post():
|
|
data = request.json
|
|
validator = schemas.VmMigrationSchema(data)
|
|
|
|
if validator.is_valid():
|
|
vm = shared.vm_pool.get(data['uuid'])
|
|
r = RequestEntry.from_scratch(
|
|
type=RequestType.InitVMMigration,
|
|
uuid=vm.uuid,
|
|
hostname=join_path(
|
|
settings['etcd']['host_prefix'],
|
|
validator.destination.value,
|
|
),
|
|
request_prefix=settings['etcd']['request_prefix'],
|
|
)
|
|
|
|
shared.request_pool.put(r)
|
|
return (
|
|
{'message': 'VM Migration Initialization Queued'},
|
|
200,
|
|
)
|
|
else:
|
|
return validator.get_errors(), 400
|
|
|
|
|
|
class ListUserVM(Resource):
|
|
@staticmethod
|
|
def post():
|
|
data = request.json
|
|
validator = schemas.OTPSchema(data)
|
|
|
|
if validator.is_valid():
|
|
vms = shared.etcd_client.get_prefix(
|
|
settings['etcd']['vm_prefix'], value_in_json=True
|
|
)
|
|
return_vms = []
|
|
user_vms = filter(
|
|
lambda v: v.value['owner'] == data['name'], vms
|
|
)
|
|
for vm in user_vms:
|
|
return_vms.append(
|
|
{
|
|
'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),
|
|
}
|
|
)
|
|
if return_vms:
|
|
return {'message': return_vms}, 200
|
|
return {'message': 'No VM found'}, 404
|
|
|
|
else:
|
|
return validator.get_errors(), 400
|
|
|
|
|
|
class ListUserFiles(Resource):
|
|
@staticmethod
|
|
def post():
|
|
data = request.json
|
|
validator = schemas.OTPSchema(data)
|
|
|
|
if validator.is_valid():
|
|
files = shared.etcd_client.get_prefix(
|
|
settings['etcd']['file_prefix'], value_in_json=True
|
|
)
|
|
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 validator.get_errors(), 400
|
|
|
|
|
|
class CreateHost(Resource):
|
|
@staticmethod
|
|
def post():
|
|
data = request.json
|
|
validator = schemas.CreateHostSchema(data)
|
|
if validator.is_valid():
|
|
host_key = join_path(
|
|
settings['etcd']['host_prefix'], uuid4().hex
|
|
)
|
|
host_entry = {
|
|
'specs': data['specs'],
|
|
'hostname': data['hostname'],
|
|
'status': 'DEAD',
|
|
'last_heartbeat': '',
|
|
}
|
|
shared.etcd_client.put(
|
|
host_key, host_entry, value_in_json=True
|
|
)
|
|
|
|
return {'message': 'Host Created'}, 200
|
|
|
|
return validator.get_errors(), 400
|
|
|
|
|
|
class ListHost(Resource):
|
|
@staticmethod
|
|
def get():
|
|
hosts = shared.host_pool.hosts
|
|
r = {
|
|
host.key: {
|
|
'status': host.status,
|
|
'specs': host.specs,
|
|
'hostname': host.hostname,
|
|
}
|
|
for host in hosts
|
|
}
|
|
return r, 200
|
|
|
|
|
|
class GetSSHKeys(Resource):
|
|
@staticmethod
|
|
def post():
|
|
data = request.json
|
|
validator = schemas.GetSSHSchema(data)
|
|
if validator.is_valid():
|
|
if not validator.key_name.value:
|
|
|
|
# {user_prefix}/{realm}/{name}/key/
|
|
etcd_key = join_path(
|
|
settings['etcd']['user_prefix'],
|
|
data['realm'],
|
|
data['name'],
|
|
'key',
|
|
)
|
|
etcd_entry = shared.etcd_client.get_prefix(
|
|
etcd_key, value_in_json=True
|
|
)
|
|
|
|
keys = {
|
|
key.key.split('/')[-1]: key.value
|
|
for key in etcd_entry
|
|
}
|
|
return {'keys': keys}
|
|
else:
|
|
|
|
# {user_prefix}/{realm}/{name}/key/{key_name}
|
|
etcd_key = join_path(
|
|
settings['etcd']['user_prefix'],
|
|
data['realm'],
|
|
data['name'],
|
|
'key',
|
|
data['key_name'],
|
|
)
|
|
etcd_entry = shared.etcd_client.get(
|
|
etcd_key, value_in_json=True
|
|
)
|
|
|
|
if etcd_entry:
|
|
return {
|
|
'keys': {
|
|
etcd_entry.key.split('/')[
|
|
-1
|
|
]: etcd_entry.value
|
|
}
|
|
}
|
|
else:
|
|
return {'keys': {}}
|
|
else:
|
|
return validator.get_errors(), 400
|
|
|
|
|
|
class AddSSHKey(Resource):
|
|
@staticmethod
|
|
def post():
|
|
data = request.json
|
|
validator = schemas.AddSSHSchema(data)
|
|
if validator.is_valid():
|
|
|
|
# {user_prefix}/{realm}/{name}/key/{key_name}
|
|
etcd_key = join_path(
|
|
settings['etcd']['user_prefix'],
|
|
data['realm'],
|
|
data['name'],
|
|
'key',
|
|
data['key_name'],
|
|
)
|
|
etcd_entry = shared.etcd_client.get(
|
|
etcd_key, value_in_json=True
|
|
)
|
|
if etcd_entry:
|
|
return {
|
|
'message': 'Key with name "{}" already exists'.format(
|
|
data['key_name']
|
|
)
|
|
}
|
|
else:
|
|
# Key Not Found. It implies user' haven't added any key yet.
|
|
shared.etcd_client.put(
|
|
etcd_key, data['key'], value_in_json=True
|
|
)
|
|
return {'message': 'Key added successfully'}
|
|
else:
|
|
return validator.get_errors(), 400
|
|
|
|
|
|
class RemoveSSHKey(Resource):
|
|
@staticmethod
|
|
def post():
|
|
data = request.json
|
|
validator = schemas.RemoveSSHSchema(data)
|
|
if validator.is_valid():
|
|
|
|
# {user_prefix}/{realm}/{name}/key/{key_name}
|
|
etcd_key = join_path(
|
|
settings['etcd']['user_prefix'],
|
|
data['realm'],
|
|
data['name'],
|
|
'key',
|
|
data['key_name'],
|
|
)
|
|
etcd_entry = shared.etcd_client.get(
|
|
etcd_key, value_in_json=True
|
|
)
|
|
if etcd_entry:
|
|
shared.etcd_client.client.delete(etcd_key)
|
|
return {'message': 'Key successfully removed.'}
|
|
else:
|
|
return {
|
|
'message': 'No Key with name "{}" Exists at all.'.format(
|
|
data['key_name']
|
|
)
|
|
}
|
|
else:
|
|
return validator.get_errors(), 400
|
|
|
|
|
|
class CreateNetwork(Resource):
|
|
@staticmethod
|
|
def post():
|
|
data = request.json
|
|
validator = schemas.CreateNetwork(data)
|
|
|
|
if validator.is_valid():
|
|
|
|
network_entry = {
|
|
'id': counters.increment_etcd_counter(
|
|
shared.etcd_client, '/v1/counter/vxlan'
|
|
),
|
|
'type': data['type'],
|
|
}
|
|
if validator.user.value:
|
|
try:
|
|
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={
|
|
'prefix_length': int(
|
|
settings['network']['prefix_length']
|
|
),
|
|
'description': '{}\'s network "{}"'.format(
|
|
data['name'], data['network_name']
|
|
),
|
|
'is_pool': True,
|
|
}
|
|
)
|
|
except Exception as err:
|
|
app.logger.error(err)
|
|
return {
|
|
'message': 'Error occured while creating network.'
|
|
}
|
|
else:
|
|
network_entry['ipv6'] = prefix['prefix']
|
|
else:
|
|
network_entry['ipv6'] = 'fd00::/64'
|
|
|
|
network_key = join_path(
|
|
settings['etcd']['network_prefix'],
|
|
data['name'],
|
|
data['network_name'],
|
|
)
|
|
shared.etcd_client.put(
|
|
network_key, network_entry, value_in_json=True
|
|
)
|
|
return {'message': 'Network successfully added.'}
|
|
else:
|
|
return validator.get_errors(), 400
|
|
|
|
|
|
class ListUserNetwork(Resource):
|
|
@staticmethod
|
|
def post():
|
|
data = request.json
|
|
validator = schemas.OTPSchema(data)
|
|
|
|
if validator.is_valid():
|
|
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:
|
|
net.value['name'] = net.key.split('/')[-1]
|
|
user_networks.append(net.value)
|
|
return {'networks': user_networks}, 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(VMMigration, '/vm/migrate')
|
|
|
|
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(ListUserNetwork, '/user/networks')
|
|
|
|
api.add_resource(AddSSHKey, '/user/add-ssh')
|
|
api.add_resource(RemoveSSHKey, '/user/remove-ssh')
|
|
api.add_resource(GetSSHKeys, '/user/get-ssh')
|
|
|
|
api.add_resource(CreateHost, '/host/create')
|
|
api.add_resource(ListHost, '/host/list')
|
|
|
|
api.add_resource(CreateNetwork, '/network/create')
|
|
|
|
|
|
def main(debug=False, port=None):
|
|
try:
|
|
image_stores = list(
|
|
shared.etcd_client.get_prefix(
|
|
settings['etcd']['image_store_prefix'], value_in_json=True
|
|
)
|
|
)
|
|
except KeyError:
|
|
image_stores = False
|
|
|
|
# Do not inject default values that might be very wrong
|
|
# fail when required, not before
|
|
#
|
|
# if not image_stores:
|
|
# data = {
|
|
# 'is_public': True,
|
|
# 'type': 'ceph',
|
|
# 'name': 'images',
|
|
# 'description': 'first ever public image-store',
|
|
# 'attributes': {'list': [], 'key': [], 'pool': 'images'},
|
|
# }
|
|
|
|
# shared.etcd_client.put(
|
|
# join_path(
|
|
# settings['etcd']['image_store_prefix'], uuid4().hex
|
|
# ),
|
|
# json.dumps(data),
|
|
# )
|
|
|
|
try:
|
|
app.run(host='::',
|
|
port=port,
|
|
debug=debug)
|
|
except OSError as e:
|
|
raise UncloudException('Failed to start Flask: {}'.format(e))
|
|
|
|
|
|
if __name__ == '__main__':
|
|
main()
|