Merge branch 'master' of code.ungleich.ch:uncloud/uncloud

This commit is contained in:
Nico Schottelius 2020-01-11 00:06:29 +01:00
commit 7c9e3d747a
32 changed files with 431 additions and 396 deletions

View file

@ -1,6 +1,7 @@
[etcd] [etcd]
url = localhost url = localhost
port = 2379 port = 2379
base_prefix = /
ca_cert ca_cert
cert_cert cert_cert
cert_key cert_key
@ -9,3 +10,4 @@ cert_key
name = replace_me name = replace_me
realm = replace_me realm = replace_me
seed = replace_me seed = replace_me
api_server = http://localhost:5000

View file

@ -3,10 +3,14 @@ import logging
import sys import sys
import importlib import importlib
import argparse import argparse
import multiprocessing as mp
from uncloud import UncloudException from uncloud import UncloudException
from contextlib import suppress
# the components that use etcd
ETCD_COMPONENTS = ['api', 'scheduler', 'host', 'filescanner', 'imagescanner', 'metadata', 'configure']
ALL_COMPONENTS = ETCD_COMPONENTS.copy()
ALL_COMPONENTS.append('cli')
def exception_hook(exc_type, exc_value, exc_traceback): def exception_hook(exc_type, exc_value, exc_traceback):
@ -27,32 +31,48 @@ if __name__ == '__main__':
subparsers = arg_parser.add_subparsers(dest='command') subparsers = arg_parser.add_subparsers(dest='command')
parent_parser = argparse.ArgumentParser(add_help=False) parent_parser = argparse.ArgumentParser(add_help=False)
parent_parser.add_argument('--debug', '-d', action='store_true', default=False, parent_parser.add_argument('--debug', '-d',
action='store_true',
default=False,
help='More verbose logging') help='More verbose logging')
parent_parser.add_argument('--conf-dir', '-c',
help='Configuration directory')
for component in ['api', 'scheduler', 'host', 'filescanner', 'imagescanner', etcd_parser = argparse.ArgumentParser(add_help=False)
'metadata', 'configure', 'cli']: etcd_parser.add_argument('--etcd-host')
etcd_parser.add_argument('--etcd-port')
etcd_parser.add_argument('--etcd-ca-cert', help='CA that signed the etcd certificate')
etcd_parser.add_argument('--etcd-cert-cert', help='Path to client certificate')
etcd_parser.add_argument('--etcd-cert-key', help='Path to client certificate key')
for component in ALL_COMPONENTS:
mod = importlib.import_module('uncloud.{}.main'.format(component)) mod = importlib.import_module('uncloud.{}.main'.format(component))
parser = getattr(mod, 'arg_parser') parser = getattr(mod, 'arg_parser')
subparsers.add_parser(name=parser.prog, parents=[parser, parent_parser])
if component in ETCD_COMPONENTS:
subparsers.add_parser(name=parser.prog, parents=[parser, parent_parser, etcd_parser])
else:
subparsers.add_parser(name=parser.prog, parents=[parser, parent_parser])
args = arg_parser.parse_args() args = arg_parser.parse_args()
if not args.command: if not args.command:
arg_parser.print_help() arg_parser.print_help()
else: else:
# if we start etcd in seperate process with default settings
# i.e inheriting few things from parent process etcd3 module
# errors out, so the following command configure multiprocessing
# module to not inherit anything from parent.
mp.set_start_method('spawn')
arguments = vars(args) arguments = vars(args)
name = arguments.pop('command')
mod = importlib.import_module('uncloud.{}.main'.format(name))
main = getattr(mod, 'main')
# If the component requires etcd3, we import it and catch the
# etcd3.exceptions.ConnectionFailedError
if name in ETCD_COMPONENTS:
import etcd3
try: try:
name = arguments.pop('command') main(arguments)
mod = importlib.import_module('uncloud.{}.main'.format(name))
main = getattr(mod, 'main')
main(**arguments)
except UncloudException as err: except UncloudException as err:
logger.error(err) logger.error(err)
sys.exit(1)
except Exception as err: except Exception as err:
logger.exception(err) logger.exception(err)
sys.exit(1)

View file

@ -1,7 +1,7 @@
import os import os
from uncloud.shared import shared from uncloud.common.shared import shared
from uncloud.settings import settings from uncloud.common.settings import settings
class Optional: class Optional:

View file

@ -3,18 +3,18 @@ import os
from uuid import uuid4 from uuid import uuid4
from uncloud.shared import shared from uncloud.common.shared import shared
from uncloud.settings import settings from uncloud.common.settings import settings
data = { data = {
"is_public": True, 'is_public': True,
"type": "ceph", 'type': 'ceph',
"name": "images", 'name': 'images',
"description": "first ever public image-store", 'description': 'first ever public image-store',
"attributes": {"list": [], "key": [], "pool": "images"}, 'attributes': {'list': [], 'key': [], 'pool': 'images'},
} }
shared.etcd_client.put( shared.etcd_client.put(
os.path.join(settings["etcd"]["image_store_prefix"], uuid4().hex), os.path.join(settings['etcd']['image_store_prefix'], uuid4().hex),
json.dumps(data), json.dumps(data),
) )

View file

@ -1,14 +1,13 @@
import binascii import binascii
import ipaddress import ipaddress
import random import random
import subprocess as sp
import logging import logging
import requests import requests
from pyotp import TOTP from pyotp import TOTP
from uncloud.shared import shared from uncloud.common.shared import shared
from uncloud.settings import settings from uncloud.common.settings import settings
logger = logging.getLogger(__name__) logger = logging.getLogger(__name__)

View file

@ -10,11 +10,12 @@ from flask import Flask, request
from flask_restful import Resource, Api from flask_restful import Resource, Api
from werkzeug.exceptions import HTTPException from werkzeug.exceptions import HTTPException
from uncloud.common.shared import shared
from uncloud.common import counters from uncloud.common import counters
from uncloud.common.vm import VMStatus from uncloud.common.vm import VMStatus
from uncloud.common.request import RequestEntry, RequestType from uncloud.common.request import RequestEntry, RequestType
from uncloud.settings import settings from uncloud.common.settings import settings
from uncloud.shared import shared
from . import schemas from . import schemas
from .helper import generate_mac, mac2ipv6 from .helper import generate_mac, mac2ipv6
from uncloud import UncloudException from uncloud import UncloudException
@ -41,7 +42,7 @@ def handle_exception(e):
class CreateVM(Resource): class CreateVM(Resource):
'''API Request to Handle Creation of VM''' """API Request to Handle Creation of VM"""
@staticmethod @staticmethod
def post(): def post():
@ -59,7 +60,7 @@ class CreateVM(Resource):
macs = [generate_mac() for _ in range(len(data['network']))] macs = [generate_mac() for _ in range(len(data['network']))]
tap_ids = [ tap_ids = [
counters.increment_etcd_counter( counters.increment_etcd_counter(
shared.etcd_client, '/v1/counter/tap' shared.etcd_client, settings['etcd']['tap_counter']
) )
for _ in range(len(data['network'])) for _ in range(len(data['network']))
] ]
@ -289,18 +290,16 @@ class ListUserFiles(Resource):
settings['etcd']['file_prefix'], value_in_json=True settings['etcd']['file_prefix'], value_in_json=True
) )
return_files = [] return_files = []
user_files = list( user_files = [f for f in files if f.value['owner'] == data['name']]
filter(
lambda f: f.value['owner'] == data['name'], files
)
)
for file in user_files: for file in user_files:
return_files.append( file_uuid = file.key.split('/')[-1]
{ file = file.value
'filename': file.value['filename'], file['uuid'] = file_uuid
'uuid': file.key.split('/')[-1],
} file.pop('sha512sum', None)
) file.pop('owner', None)
return_files.append(file)
return {'message': return_files}, 200 return {'message': return_files}, 200
else: else:
return validator.get_errors(), 400 return validator.get_errors(), 400
@ -472,7 +471,7 @@ class CreateNetwork(Resource):
network_entry = { network_entry = {
'id': counters.increment_etcd_counter( 'id': counters.increment_etcd_counter(
shared.etcd_client, '/v1/counter/vxlan' shared.etcd_client, settings['etcd']['vxlan_counter']
), ),
'type': data['type'], 'type': data['type'],
} }
@ -564,7 +563,10 @@ api.add_resource(ListHost, '/host/list')
api.add_resource(CreateNetwork, '/network/create') api.add_resource(CreateNetwork, '/network/create')
def main(debug=False, port=None): def main(arguments):
debug = arguments['debug']
port = arguments['port']
try: try:
image_stores = list( image_stores = list(
shared.etcd_client.get_prefix( shared.etcd_client.get_prefix(
@ -594,12 +596,6 @@ def main(debug=False, port=None):
# ) # )
try: try:
app.run(host='::', app.run(host='::', port=port, debug=debug)
port=port,
debug=debug)
except OSError as e: except OSError as e:
raise UncloudException('Failed to start Flask: {}'.format(e)) raise UncloudException('Failed to start Flask: {}'.format(e))
if __name__ == '__main__':
main()

View file

@ -21,8 +21,8 @@ import bitmath
from uncloud.common.host import HostStatus from uncloud.common.host import HostStatus
from uncloud.common.vm import VMStatus from uncloud.common.vm import VMStatus
from uncloud.shared import shared from uncloud.common.shared import shared
from uncloud.settings import settings from uncloud.common.settings import settings
from . import helper, logger from . import helper, logger
from .common_fields import Field, VmUUIDField from .common_fields import Field, VmUUIDField
from .helper import check_otp, resolve_vm_name from .helper import check_otp, resolve_vm_name

View file

@ -5,15 +5,24 @@ import binascii
from pyotp import TOTP from pyotp import TOTP
from os.path import join as join_path from os.path import join as join_path
from uncloud.settings import settings from uncloud.common.settings import settings
def get_otp_parser(): def get_otp_parser():
otp_parser = argparse.ArgumentParser('otp') otp_parser = argparse.ArgumentParser('otp')
otp_parser.add_argument('--name', default=settings['client']['name']) try:
otp_parser.add_argument('--realm', default=settings['client']['realm']) name = settings['client']['name']
otp_parser.add_argument('--seed', type=get_token, default=settings['client']['seed'], realm = settings['client']['realm']
dest='token', metavar='SEED') seed = settings['client']['seed']
except Exception:
otp_parser.add_argument('--name', required=True)
otp_parser.add_argument('--realm', required=True)
otp_parser.add_argument('--seed', required=True, type=get_token, dest='token', metavar='SEED')
else:
otp_parser.add_argument('--name', default=name)
otp_parser.add_argument('--realm', default=realm)
otp_parser.add_argument('--seed', default=seed, type=get_token, dest='token', metavar='SEED')
return otp_parser return otp_parser

View file

@ -12,7 +12,7 @@ class ImageParser(BaseParser):
p = self.subparser.add_parser('create', **kwargs) p = self.subparser.add_parser('create', **kwargs)
p.add_argument('--name', required=True) p.add_argument('--name', required=True)
p.add_argument('--uuid', required=True) p.add_argument('--uuid', required=True)
p.add_argument('--image-store-name', default='image_store') p.add_argument('--image-store', required=True, dest='image_store')
def list(self, **kwargs): def list(self, **kwargs):
self.subparser.add_parser('list', **kwargs) self.subparser.add_parser('list', **kwargs)

View file

@ -12,12 +12,12 @@ for component in ['user', 'host', 'image', 'network', 'vm']:
subparser.add_parser(name=parser.prog, parents=[parser]) subparser.add_parser(name=parser.prog, parents=[parser])
def main(**kwargs): def main(arguments):
if not kwargs['subcommand']: if not arguments['subcommand']:
arg_parser.print_help() arg_parser.print_help()
else: else:
name = kwargs.pop('subcommand') name = arguments.pop('subcommand')
kwargs.pop('debug') arguments.pop('debug')
mod = importlib.import_module('uncloud.cli.{}'.format(name)) mod = importlib.import_module('uncloud.cli.{}'.format(name))
_main = getattr(mod, 'main') _main = getattr(mod, 'main')
_main(**kwargs) _main(**arguments)

View file

@ -1,24 +1,21 @@
import etcd3 import etcd3
import json import json
import queue
import copy
from uncloud import UncloudException
from collections import namedtuple
from functools import wraps from functools import wraps
from . import logger from uncloud import UncloudException
from uncloud.common import logger
PseudoEtcdMeta = namedtuple("PseudoEtcdMeta", ["key"])
class EtcdEntry: class EtcdEntry:
# key: str def __init__(self, meta_or_key, value, value_in_json=False):
# value: str if hasattr(meta_or_key, 'key'):
# if meta has attr 'key' then get it
def __init__(self, meta, value, value_in_json=False): self.key = meta_or_key.key.decode('utf-8')
self.key = meta.key.decode("utf-8") else:
self.value = value.decode("utf-8") # otherwise meta is the 'key'
self.key = meta_or_key
self.value = value.decode('utf-8')
if value_in_json: if value_in_json:
self.value = json.loads(self.value) self.value = json.loads(self.value)
@ -29,18 +26,12 @@ def readable_errors(func):
def wrapper(*args, **kwargs): def wrapper(*args, **kwargs):
try: try:
return func(*args, **kwargs) return func(*args, **kwargs)
except etcd3.exceptions.ConnectionFailedError as err: except etcd3.exceptions.ConnectionFailedError:
raise UncloudException( raise UncloudException('Cannot connect to etcd: is etcd running as configured in uncloud.conf?')
"Cannot connect to etcd: is etcd running as configured in uncloud.conf?"
)
except etcd3.exceptions.ConnectionTimeoutError as err: except etcd3.exceptions.ConnectionTimeoutError as err:
raise etcd3.exceptions.ConnectionTimeoutError( raise etcd3.exceptions.ConnectionTimeoutError('etcd connection timeout.') from err
"etcd connection timeout."
) from err
except Exception: except Exception:
logger.exception( logger.exception('Some etcd error occured. See syslog for details.')
"Some etcd error occured. See syslog for details."
)
return wrapper return wrapper
@ -64,55 +55,21 @@ class Etcd3Wrapper:
_value = json.dumps(_value) _value = json.dumps(_value)
if not isinstance(_key, str): if not isinstance(_key, str):
_key = _key.decode("utf-8") _key = _key.decode('utf-8')
return self.client.put(_key, _value, **kwargs) return self.client.put(_key, _value, **kwargs)
@readable_errors @readable_errors
def get_prefix(self, *args, value_in_json=False, **kwargs): def get_prefix(self, *args, value_in_json=False, raise_exception=True, **kwargs):
r = self.client.get_prefix(*args, **kwargs) event_iterator = self.client.get_prefix(*args, **kwargs)
for entry in r: for e in event_iterator:
e = EtcdEntry(*entry[::-1], value_in_json=value_in_json) yield EtcdEntry(*e[::-1], value_in_json=value_in_json)
if e.value:
yield e
@readable_errors @readable_errors
def watch_prefix(self, key, timeout=0, value_in_json=False): def watch_prefix(self, key, raise_exception=True, value_in_json=False):
timeout_event = EtcdEntry( event_iterator, cancel = self.client.watch_prefix(key)
PseudoEtcdMeta(key=b"TIMEOUT"), for e in event_iterator:
value=str.encode( if hasattr(e, '_event'):
json.dumps({"status": "TIMEOUT", "type": "TIMEOUT"}) e = e._event
), if e.type == e.PUT:
value_in_json=value_in_json, yield EtcdEntry(e.kv.key, e.kv.value, value_in_json=value_in_json)
)
event_queue = queue.Queue()
def add_event_to_queue(event):
if hasattr(event, "events"):
for e in event.events:
if e.value:
event_queue.put(
EtcdEntry(
e, e.value, value_in_json=value_in_json
)
)
self.client.add_watch_prefix_callback(key, add_event_to_queue)
while True:
try:
while True:
v = event_queue.get(timeout=timeout)
yield v
except queue.Empty:
event_queue.put(copy.deepcopy(timeout_event))
class PsuedoEtcdEntry(EtcdEntry):
def __init__(self, key, value, value_in_json=False):
super().__init__(
PseudoEtcdMeta(key=key.encode("utf-8")),
value,
value_in_json=value_in_json,
)

View file

@ -1,8 +1,6 @@
import subprocess as sp import subprocess as sp
import random import random
import logging import logging
import socket
from contextlib import closing
logger = logging.getLogger(__name__) logger = logging.getLogger(__name__)

View file

@ -2,8 +2,8 @@ import json
from os.path import join from os.path import join
from uuid import uuid4 from uuid import uuid4
from .etcd_wrapper import PsuedoEtcdEntry from uncloud.common.etcd_wrapper import EtcdEntry
from .classes import SpecificEtcdEntryBase from uncloud.common.classes import SpecificEtcdEntryBase
class RequestType: class RequestType:
@ -29,11 +29,8 @@ class RequestEntry(SpecificEtcdEntryBase):
@classmethod @classmethod
def from_scratch(cls, request_prefix, **kwargs): def from_scratch(cls, request_prefix, **kwargs):
e = PsuedoEtcdEntry( e = EtcdEntry(meta_or_key=join(request_prefix, uuid4().hex),
join(request_prefix, uuid4().hex), value=json.dumps(kwargs).encode('utf-8'), value_in_json=True)
value=json.dumps(kwargs).encode("utf-8"),
value_in_json=True,
)
return cls(e) return cls(e)

View file

@ -4,8 +4,8 @@ import sys
import os import os
from datetime import datetime from datetime import datetime
from uncloud.common.etcd_wrapper import Etcd3Wrapper from uncloud.common.etcd_wrapper import Etcd3Wrapper
from os.path import join as join_path
logger = logging.getLogger(__name__) logger = logging.getLogger(__name__)
@ -16,7 +16,7 @@ class CustomConfigParser(configparser.RawConfigParser):
result = super().__getitem__(key) result = super().__getitem__(key)
except KeyError as err: except KeyError as err:
raise KeyError( raise KeyError(
"Key '{}' not found in configuration. Make sure you configure uncloud.".format( 'Key \'{}\' not found in configuration. Make sure you configure uncloud.'.format(
key key
) )
) from err ) from err
@ -25,40 +25,41 @@ class CustomConfigParser(configparser.RawConfigParser):
class Settings(object): class Settings(object):
def __init__(self, config_key="/uncloud/config/"): def __init__(self):
conf_name = "uncloud.conf" conf_name = 'uncloud.conf'
conf_dir = os.environ.get( conf_dir = os.environ.get('UCLOUD_CONF_DIR', os.path.expanduser('~/uncloud/'))
"UCLOUD_CONF_DIR", os.path.expanduser("~/uncloud/") self.config_file = join_path(conf_dir, conf_name)
)
self.config_file = os.path.join(conf_dir, conf_name)
self.config_parser = CustomConfigParser(allow_no_value=True)
self.config_key = config_key
# this is used to cache config from etcd for 1 minutes. Without this we # this is used to cache config from etcd for 1 minutes. Without this we
# would make a lot of requests to etcd which slows down everything. # would make a lot of requests to etcd which slows down everything.
self.last_config_update = datetime.fromtimestamp(0) self.last_config_update = datetime.fromtimestamp(0)
self.read_internal_values() self.config_parser = CustomConfigParser(allow_no_value=True)
self.config_parser.add_section('etcd')
self.config_parser.set('etcd', 'base_prefix', '/')
try: try:
self.config_parser.read(self.config_file) self.config_parser.read(self.config_file)
except Exception as err: except Exception as err:
logger.error("%s", err) logger.error('%s', err)
self.config_key = join_path(self['etcd']['base_prefix'] + 'uncloud/config/')
self.read_internal_values()
def get_etcd_client(self): def get_etcd_client(self):
args = tuple() args = tuple()
try: try:
kwargs = { kwargs = {
"host": self.config_parser.get("etcd", "url"), 'host': self.config_parser.get('etcd', 'url'),
"port": self.config_parser.get("etcd", "port"), 'port': self.config_parser.get('etcd', 'port'),
"ca_cert": self.config_parser.get("etcd", "ca_cert"), 'ca_cert': self.config_parser.get('etcd', 'ca_cert'),
"cert_cert": self.config_parser.get( 'cert_cert': self.config_parser.get('etcd', 'cert_cert'),
"etcd", "cert_cert" 'cert_key': self.config_parser.get('etcd', 'cert_key'),
),
"cert_key": self.config_parser.get("etcd", "cert_key"),
} }
except configparser.Error as err: except configparser.Error as err:
raise configparser.Error( raise configparser.Error(
"{} in config file {}".format( '{} in config file {}'.format(
err.message, self.config_file err.message, self.config_file
) )
) from err ) from err
@ -67,8 +68,8 @@ class Settings(object):
wrapper = Etcd3Wrapper(*args, **kwargs) wrapper = Etcd3Wrapper(*args, **kwargs)
except Exception as err: except Exception as err:
logger.error( logger.error(
"etcd connection not successfull. Please check your config file." 'etcd connection not successfull. Please check your config file.'
"\nDetails: %s\netcd connection parameters: %s", '\nDetails: %s\netcd connection parameters: %s',
err, err,
kwargs, kwargs,
) )
@ -77,17 +78,20 @@ class Settings(object):
return wrapper return wrapper
def read_internal_values(self): def read_internal_values(self):
base_prefix = self['etcd']['base_prefix']
self.config_parser.read_dict( self.config_parser.read_dict(
{ {
"etcd": { 'etcd': {
"file_prefix": "/files/", 'file_prefix': join_path(base_prefix, 'files/'),
"host_prefix": "/hosts/", 'host_prefix': join_path(base_prefix, 'hosts/'),
"image_prefix": "/images/", 'image_prefix': join_path(base_prefix, 'images/'),
"image_store_prefix": "/imagestore/", 'image_store_prefix': join_path(base_prefix, 'imagestore/'),
"network_prefix": "/networks/", 'network_prefix': join_path(base_prefix, 'networks/'),
"request_prefix": "/requests/", 'request_prefix': join_path(base_prefix, 'requests/'),
"user_prefix": "/users/", 'user_prefix': join_path(base_prefix, 'users/'),
"vm_prefix": "/vms/", 'vm_prefix': join_path(base_prefix, 'vms/'),
'vxlan_counter': join_path(base_prefix, 'counters/vxlan'),
'tap_counter': join_path(base_prefix, 'counters/tap')
} }
} }
) )
@ -95,15 +99,13 @@ class Settings(object):
def read_config_file_values(self, config_file): def read_config_file_values(self, config_file):
try: try:
# Trying to read configuration file # Trying to read configuration file
with open(config_file, "r") as config_file_handle: with open(config_file) as config_file_handle:
self.config_parser.read_file(config_file_handle) self.config_parser.read_file(config_file_handle)
except FileNotFoundError: except FileNotFoundError:
sys.exit( sys.exit('Configuration file {} not found!'.format(config_file))
"Configuration file {} not found!".format(config_file)
)
except Exception as err: except Exception as err:
logger.exception(err) logger.exception(err)
sys.exit("Error occurred while reading configuration file") sys.exit('Error occurred while reading configuration file')
def read_values_from_etcd(self): def read_values_from_etcd(self):
etcd_client = self.get_etcd_client() etcd_client = self.get_etcd_client()
@ -113,7 +115,7 @@ class Settings(object):
self.config_parser.read_dict(config_from_etcd.value) self.config_parser.read_dict(config_from_etcd.value)
self.last_config_update = datetime.utcnow() self.last_config_update = datetime.utcnow()
else: else:
raise KeyError("Key '{}' not found in etcd. Please configure uncloud.".format(self.config_key)) raise KeyError('Key \'{}\' not found in etcd. Please configure uncloud.'.format(self.config_key))
def __getitem__(self, key): def __getitem__(self, key):
# Allow failing to read from etcd if we have # Allow failing to read from etcd if we have
@ -121,9 +123,8 @@ class Settings(object):
if key not in self.config_parser.sections(): if key not in self.config_parser.sections():
try: try:
self.read_values_from_etcd() self.read_values_from_etcd()
except KeyError as e: except KeyError:
pass pass
return self.config_parser[key] return self.config_parser[key]

View file

@ -1,4 +1,4 @@
from uncloud.settings import settings from uncloud.common.settings import settings
from uncloud.common.vm import VmPool from uncloud.common.vm import VmPool
from uncloud.common.host import HostPool from uncloud.common.host import HostPool
from uncloud.common.request import RequestPool from uncloud.common.request import RequestPool

View file

@ -7,7 +7,7 @@ from abc import ABC
from . import logger from . import logger
from os.path import join as join_path from os.path import join as join_path
from uncloud.settings import settings as config from uncloud.common.settings import settings as config
class ImageStorageHandler(ABC): class ImageStorageHandler(ABC):

View file

@ -1,8 +1,8 @@
import os import os
import argparse import argparse
from uncloud.settings import settings from uncloud.common.settings import settings
from uncloud.shared import shared from uncloud.common.shared import shared
arg_parser = argparse.ArgumentParser('configure', add_help=False) arg_parser = argparse.ArgumentParser('configure', add_help=False)
configure_subparsers = arg_parser.add_subparsers(dest='subcommand') configure_subparsers = arg_parser.add_subparsers(dest='subcommand')
@ -40,18 +40,14 @@ ceph_storage_parser.add_argument('--ceph-image-pool', required=True)
def update_config(section, kwargs): def update_config(section, kwargs):
uncloud_config = shared.etcd_client.get( uncloud_config = shared.etcd_client.get(settings.config_key, value_in_json=True)
settings.config_key, value_in_json=True
)
if not uncloud_config: if not uncloud_config:
uncloud_config = {} uncloud_config = {}
else: else:
uncloud_config = uncloud_config.value uncloud_config = uncloud_config.value
uncloud_config[section] = kwargs uncloud_config[section] = kwargs
shared.etcd_client.put( shared.etcd_client.put(settings.config_key, uncloud_config, value_in_json=True)
settings.config_key, uncloud_config, value_in_json=True
)
def main(**kwargs): def main(**kwargs):

12
uncloud/docs/README.md Normal file
View file

@ -0,0 +1,12 @@
# uncloud docs
## Requirements
1. Python3
2. Sphinx
## Usage
Run `make build` to build docs.
Run `make clean` to remove build directory.
Run `make publish` to push build dir to https://ungleich.ch/ucloud/

View file

@ -4,15 +4,16 @@ import pathlib
import subprocess as sp import subprocess as sp
import time import time
import argparse import argparse
import bitmath
from uuid import uuid4 from uuid import uuid4
from . import logger from . import logger
from uncloud.settings import settings from uncloud.common.settings import settings
from uncloud.shared import shared from uncloud.common.shared import shared
arg_parser = argparse.ArgumentParser('filescanner', add_help=False) arg_parser = argparse.ArgumentParser('filescanner', add_help=False)
arg_parser.add_argument('--hostname', required=True)
def sha512sum(file: str): def sha512sum(file: str):
@ -28,66 +29,58 @@ def sha512sum(file: str):
if not isinstance(file, str): if not isinstance(file, str):
raise TypeError raise TypeError
try: try:
output = sp.check_output(["sha512sum", file], stderr=sp.PIPE) output = sp.check_output(['sha512sum', file], stderr=sp.PIPE)
except sp.CalledProcessError as e: except sp.CalledProcessError as e:
error = e.stderr.decode("utf-8") error = e.stderr.decode('utf-8')
if "No such file or directory" in error: if 'No such file or directory' in error:
raise FileNotFoundError from None raise FileNotFoundError from None
else: else:
output = output.decode("utf-8").strip() output = output.decode('utf-8').strip()
output = output.split(" ") output = output.split(' ')
return output[0] return output[0]
return None return None
def track_file(file, base_dir): def track_file(file, base_dir, host):
file_id = uuid4() file_path = file.relative_to(base_dir)
file_str = str(file)
# Get Username # Get Username
owner = pathlib.Path(file).parts[len(pathlib.Path(base_dir).parts)] try:
owner = file_path.parts[0]
except IndexError:
pass
else:
file_path = file_path.relative_to(owner)
creation_date = time.ctime(os.stat(file_str).st_ctime)
# Get Creation Date of File entry_key = os.path.join(settings['etcd']['file_prefix'], str(uuid4()))
# Here, we are assuming that ctime is creation time entry_value = {
# which is mostly not true. 'filename': str(file_path),
creation_date = time.ctime(os.stat(file).st_ctime) 'owner': owner,
'sha512sum': sha512sum(file_str),
'creation_date': creation_date,
'size': str(bitmath.Byte(os.path.getsize(file_str)).to_MB()),
'host': host
}
file_path = pathlib.Path(file).parts[-1] logger.info('Tracking %s', file_str)
# Create Entry shared.etcd_client.put(entry_key, entry_value, value_in_json=True)
entry_key = os.path.join(
settings["etcd"]["file_prefix"], str(file_id)
)
entry_value = {
"filename": file_path,
"owner": owner,
"sha512sum": sha512sum(file),
"creation_date": creation_date,
"size": os.path.getsize(file),
}
logger.info("Tracking %s", file)
shared.etcd_client.put(entry_key, entry_value, value_in_json=True)
os.setxattr(file, "user.utracked", b"True")
def main(debug=False): def main(arguments):
base_dir = settings["storage"]["file_dir"] hostname = arguments['hostname']
base_dir = settings['storage']['file_dir']
# Recursively Get All Files and Folder below BASE_DIR # Recursively Get All Files and Folder below BASE_DIR
files = glob.glob("{}/**".format(base_dir), recursive=True) files = glob.glob('{}/**'.format(base_dir), recursive=True)
files = [pathlib.Path(f) for f in files if pathlib.Path(f).is_file()]
# Retain only Files # Files that are already tracked
files = [file for file in files if os.path.isfile(file)] tracked_files = [
pathlib.Path(os.path.join(base_dir, f.value['owner'], f.value['filename']))
untracked_files = [] for f in shared.etcd_client.get_prefix(settings['etcd']['file_prefix'], value_in_json=True)
for file in files: if f.value['host'] == hostname
try: ]
os.getxattr(file, "user.utracked") untracked_files = set(files) - set(tracked_files)
except OSError: for file in untracked_files:
track_file(file, base_dir) track_file(file, base_dir, hostname)
untracked_files.append(file)
if __name__ == "__main__":
main()

3
uncloud/hack/hackcloud/.gitignore vendored Normal file
View file

@ -0,0 +1,3 @@
*.iso
radvdpid
foo

9
uncloud/hack/hackcloud/ifup.sh Executable file
View file

@ -0,0 +1,9 @@
#!/bin/sh
echo $@ >> foo
dev=$1; shift
# bridge is setup from outside
ip link set dev "$dev" master ${bridge}
ip link set dev "$dev" up

24
uncloud/hack/hackcloud/net.sh Executable file
View file

@ -0,0 +1,24 @@
#!/bin/sh
netid=100
dev=wlp2s0
dev=wlp0s20f3
dev=wlan0
ip=2a0a:e5c1:111:888::42/64
vxlandev=vxlan${netid}
bridgedev=br${netid}
ip -6 link add ${vxlandev} type vxlan \
id ${netid} \
dstport 4789 \
group ff05::${netid} \
dev ${dev} \
ttl 5
ip link set ${vxlandev} up
ip link add ${bridgedev} type bridge
ip link set ${bridgedev} up
ip addr add ${ip} dev ${bridgedev}

View file

@ -0,0 +1,13 @@
interface br100
{
AdvSendAdvert on;
MinRtrAdvInterval 3;
MaxRtrAdvInterval 5;
AdvDefaultLifetime 3600;
prefix 2a0a:e5c1:111:888::/64 {
};
RDNSS 2a0a:e5c0::3 2a0a:e5c0::4 { AdvRDNSSLifetime 6000; };
DNSSL place7.ungleich.ch { AdvDNSSLLifetime 6000; } ;
};

View file

@ -0,0 +1,3 @@
#!/bin/sh
radvd -C ./radvd.conf -n -p ./radvdpid

48
uncloud/hack/hackcloud/vm.sh Executable file
View file

@ -0,0 +1,48 @@
#!/bin/sh
vmid=$1; shift
qemu=/usr/bin/qemu-system-x86_64
accel=kvm
accel=tcg
memory=1024
cores=2
uuid=732e08c7-84f8-4d43-9571-263db4f80080
export bridge=br100
$qemu -name uc${vmid} \
-machine pc,accel=${accel} \
-m ${memory} \
-smp ${cores} \
-uuid ${uuid} \
-drive file=alpine-virt-3.11.2-x86_64.iso,media=cdrom \
-netdev tap,id=netmain,script=./ifup.sh \
-device virtio-net-pci,netdev=netmain,id=net0,mac=02:00:f0:a9:c4:4e
exit 0
-S -object secret,id=masterKey0,format=raw,file=/var/lib/libvirt/qemu/domain-17-one-24992/master-key.aes
-machine pc-i440fx-2.8,accel=kvm,usb=off,dump-guest-core=off
-m 2048
-realtime mlock=off
-smp 1,sockets=1,cores=1,threads=1
-uuid 732e08c7-84f8-4d43-9571-263db4f80080 -no-user-config \
-nodefaults
-chardev socket,id=charmonitor,path=/var/lib/libvirt/qemu/domain-17-one-24992/monitor.sock,server,nowait
-mon chardev=charmonitor,id=monitor,mode=control
-rtc base=utc -no-shutdown
-boot strict=on
-device piix3-usb-uhci,id=usb,bus=pci.0,addr=0x1.0x2
-drive file=rbd:ssd/one-292-24992-0:id=libvirt:auth_supported=cephx\;none:mon_host=ceph1\:6789\;ceph2\:6789\;ceph3\:6789,format=raw,if=none,id=drive-virtio-disk0,cache=none
-device virtio-blk-pci,scsi=off,bus=pci.0,addr=0x4,drive=drive-virtio-disk0,id=virtio-disk0,bootindex=1
-drive file=/var/lib/one//datastores/104/24992/disk.1,format=raw,if=none,id=drive-ide0-0-0,readonly=on
-device ide-cd,bus=ide.0,unit=0,drive=drive-ide0-0-0,id=ide0-0-0
-netdev tap,fd=36,id=hostnet0,vhost=on,vhostfd=38
-device virtio-net-pci,netdev=hostnet0,id=net0,mac=02:00:f0:a9:c4:4e,bus=pci.0,addr=0x3
-vnc [::]:4414 -device cirrus-vga,id=video0,bus=pci.0,addr=0x2 -device virtio-balloon-pci,id=balloon0,bus=pci.0,addr=0x5 -msg timestamp=on

View file

@ -5,8 +5,8 @@ import time
from uuid import uuid4 from uuid import uuid4
from uncloud.common.request import RequestEntry, RequestType from uncloud.common.request import RequestEntry, RequestType
from uncloud.shared import shared from uncloud.common.shared import shared
from uncloud.settings import settings from uncloud.common.settings import settings
from uncloud.common.vm import VMStatus from uncloud.common.vm import VMStatus
from uncloud.vmm import VMM from uncloud.vmm import VMM
from os.path import join as join_path from os.path import join as join_path
@ -33,10 +33,10 @@ def maintenance(host):
vmm = VMM() vmm = VMM()
running_vms = vmm.discover() running_vms = vmm.discover()
for vm_uuid in running_vms: for vm_uuid in running_vms:
if vmm.is_running(vm_uuid) and vmm.get_status(vm_uuid) == "running": if vmm.is_running(vm_uuid) and vmm.get_status(vm_uuid) == 'running':
logger.debug('VM {} is running on {}'.format(vm_uuid, host)) logger.debug('VM {} is running on {}'.format(vm_uuid, host))
vm = shared.vm_pool.get( vm = shared.vm_pool.get(
join_path(settings["etcd"]["vm_prefix"], vm_uuid) join_path(settings['etcd']['vm_prefix'], vm_uuid)
) )
vm.status = VMStatus.running vm.status = VMStatus.running
vm.vnc_socket = vmm.get_vnc(vm_uuid) vm.vnc_socket = vmm.get_vnc(vm_uuid)
@ -44,20 +44,21 @@ def maintenance(host):
shared.vm_pool.put(vm) shared.vm_pool.put(vm)
def main(hostname, debug=False): def main(arguments):
hostname = arguments['hostname']
host_pool = shared.host_pool host_pool = shared.host_pool
host = next(filter(lambda h: h.hostname == hostname, host_pool.hosts), None) host = next(filter(lambda h: h.hostname == hostname, host_pool.hosts), None)
# Does not yet exist, create it # Does not yet exist, create it
if not host: if not host:
host_key = join_path( host_key = join_path(
settings["etcd"]["host_prefix"], uuid4().hex settings['etcd']['host_prefix'], uuid4().hex
) )
host_entry = { host_entry = {
"specs": "", 'specs': '',
"hostname": hostname, 'hostname': hostname,
"status": "DEAD", 'status': 'DEAD',
"last_heartbeat": "", 'last_heartbeat': '',
} }
shared.etcd_client.put( shared.etcd_client.put(
host_key, host_entry, value_in_json=True host_key, host_entry, value_in_json=True
@ -70,54 +71,54 @@ def main(hostname, debug=False):
heartbeat_updating_process = mp.Process(target=update_heartbeat, args=(hostname,)) heartbeat_updating_process = mp.Process(target=update_heartbeat, args=(hostname,))
heartbeat_updating_process.start() heartbeat_updating_process.start()
except Exception as e: except Exception as e:
raise Exception("uncloud-host heartbeat updating mechanism is not working") from e raise Exception('uncloud-host heartbeat updating mechanism is not working') from e
for events_iterator in [ # The below while True is neccessary for gracefully handling leadership transfer and temporary
shared.etcd_client.get_prefix(settings["etcd"]["request_prefix"], value_in_json=True), # unavailability in etcd. Why does it work? It works because the get_prefix,watch_prefix return
shared.etcd_client.watch_prefix(settings["etcd"]["request_prefix"], timeout=10, value_in_json=True) # iter([]) that is iterator of empty list on exception (that occur due to above mentioned reasons)
]: # which ends the loop immediately. So, having it inside infinite loop we try again and again to
for request_event in events_iterator: # get prefix until either success or deamon death comes.
request_event = RequestEntry(request_event) while True:
for events_iterator in [
shared.etcd_client.get_prefix(settings['etcd']['request_prefix'], value_in_json=True,
raise_exception=False),
shared.etcd_client.watch_prefix(settings['etcd']['request_prefix'], value_in_json=True,
raise_exception=False)
]:
for request_event in events_iterator:
request_event = RequestEntry(request_event)
if request_event.type == "TIMEOUT":
maintenance(host.key) maintenance(host.key)
elif request_event.hostname == host.key: if request_event.hostname == host.key:
logger.debug("VM Request: %s on Host %s", request_event, host.hostname) logger.debug('VM Request: %s on Host %s', request_event, host.hostname)
shared.request_pool.client.client.delete(request_event.key)
vm_entry = shared.etcd_client.get(
join_path(settings["etcd"]["vm_prefix"], request_event.uuid)
)
logger.debug("VM hostname: {}".format(vm_entry.value))
vm = virtualmachine.VM(vm_entry)
if request_event.type == RequestType.StartVM:
vm.start()
elif request_event.type == RequestType.StopVM: shared.request_pool.client.client.delete(request_event.key)
vm.stop() vm_entry = shared.etcd_client.get(
join_path(settings['etcd']['vm_prefix'], request_event.uuid)
)
elif request_event.type == RequestType.DeleteVM: logger.debug('VM hostname: {}'.format(vm_entry.value))
vm.delete()
elif request_event.type == RequestType.InitVMMigration: vm = virtualmachine.VM(vm_entry)
vm.start(destination_host_key=host.key) if request_event.type == RequestType.StartVM:
vm.start()
elif request_event.type == RequestType.TransferVM: elif request_event.type == RequestType.StopVM:
destination_host = host_pool.get(request_event.destination_host_key) vm.stop()
if destination_host:
vm.migrate(
destination_host=destination_host.hostname,
destination_sock_path=request_event.destination_sock_path,
)
else:
logger.error("Host %s not found!", request_event.destination_host_key)
elif request_event.type == RequestType.DeleteVM:
vm.delete()
if __name__ == "__main__": elif request_event.type == RequestType.InitVMMigration:
argparser = argparse.ArgumentParser() vm.start(destination_host_key=host.key)
argparser.add_argument(
"hostname", help="Name of this host. e.g uncloud1.ungleich.ch" elif request_event.type == RequestType.TransferVM:
) destination_host = host_pool.get(request_event.destination_host_key)
args = argparser.parse_args() if destination_host:
mp.set_start_method("spawn") vm.migrate(
main(args.hostname) destination_host=destination_host.hostname,
destination_sock_path=request_event.destination_sock_path,
)
else:
logger.error('Host %s not found!', request_event.destination_host_key)

View file

@ -16,8 +16,8 @@ from uncloud.common.vm import VMStatus, declare_stopped
from uncloud.common.network import create_dev, delete_network_interface from uncloud.common.network import create_dev, delete_network_interface
from uncloud.common.schemas import VMSchema, NetworkSchema from uncloud.common.schemas import VMSchema, NetworkSchema
from uncloud.host import logger from uncloud.host import logger
from uncloud.shared import shared from uncloud.common.shared import shared
from uncloud.settings import settings from uncloud.common.settings import settings
from uncloud.vmm import VMM from uncloud.vmm import VMM
from marshmallow import ValidationError from marshmallow import ValidationError
@ -42,7 +42,7 @@ class VM:
def get_qemu_args(self): def get_qemu_args(self):
command = ( command = (
"-drive file={file},format=raw,if=virtio,cache=none" "-drive file={file},format=raw,if=virtio"
" -device virtio-rng-pci" " -device virtio-rng-pci"
" -m {memory} -smp cores={cores},threads={threads}" " -m {memory} -smp cores={cores},threads={threads}"
" -name {owner}_{name}" " -name {owner}_{name}"
@ -153,7 +153,10 @@ class VM:
) )
) )
return command.split(" ") if command:
command = command.split(' ')
return command
def delete_network_dev(self): def delete_network_dev(self):
try: try:

View file

@ -4,8 +4,8 @@ import argparse
import subprocess as sp import subprocess as sp
from os.path import join as join_path from os.path import join as join_path
from uncloud.settings import settings from uncloud.common.settings import settings
from uncloud.shared import shared from uncloud.common.shared import shared
from uncloud.imagescanner import logger from uncloud.imagescanner import logger
@ -30,7 +30,7 @@ def qemu_img_type(path):
return qemu_img_info["format"] return qemu_img_info["format"]
def main(debug=False): def main(arguments):
# We want to get images entries that requests images to be created # We want to get images entries that requests images to be created
images = shared.etcd_client.get_prefix( images = shared.etcd_client.get_prefix(
settings["etcd"]["image_prefix"], value_in_json=True settings["etcd"]["image_prefix"], value_in_json=True

View file

@ -5,8 +5,8 @@ from flask import Flask, request
from flask_restful import Resource, Api from flask_restful import Resource, Api
from werkzeug.exceptions import HTTPException from werkzeug.exceptions import HTTPException
from uncloud.settings import settings from uncloud.common.settings import settings
from uncloud.shared import shared from uncloud.common.shared import shared
app = Flask(__name__) app = Flask(__name__)
api = Api(app) api = Api(app)
@ -84,40 +84,11 @@ class Root(Resource):
data.value["metadata"]["ssh-keys"] += user_personal_ssh_keys data.value["metadata"]["ssh-keys"] += user_personal_ssh_keys
return data.value["metadata"], 200 return data.value["metadata"], 200
@staticmethod
def post():
return {"message": "Previous Implementation is deprecated."}
# data = etcd_client.get("/v1/metadata/{}".format(request.remote_addr), value_in_json=True)
# print(data)
# if data:
# for k in request.json:
# if k not in data.value:
# data.value[k] = request.json[k]
# if k.endswith("-list"):
# data.value[k] = [request.json[k]]
# else:
# if k.endswith("-list"):
# data.value[k].append(request.json[k])
# else:
# data.value[k] = request.json[k]
# etcd_client.put("/v1/metadata/{}".format(request.remote_addr),
# data.value, value_in_json=True)
# else:
# data = {}
# for k in request.json:
# data[k] = request.json[k]
# if k.endswith("-list"):
# data[k] = [request.json[k]]
# etcd_client.put("/v1/metadata/{}".format(request.remote_addr),
# data, value_in_json=True)
api.add_resource(Root, "/") api.add_resource(Root, "/")
def main(port=None, debug=False): def main(arguments):
port = arguments['port']
debug = arguments['debug']
app.run(debug=debug, host="::", port=port) app.run(debug=debug, host="::", port=port)
if __name__ == "__main__":
main()

View file

@ -6,8 +6,8 @@ import bitmath
from uncloud.common.host import HostStatus from uncloud.common.host import HostStatus
from uncloud.common.request import RequestEntry, RequestType from uncloud.common.request import RequestEntry, RequestType
from uncloud.common.vm import VMStatus from uncloud.common.vm import VMStatus
from uncloud.shared import shared from uncloud.common.shared import shared
from uncloud.settings import settings from uncloud.common.settings import settings
def accumulated_specs(vms_specs): def accumulated_specs(vms_specs):

View file

@ -6,59 +6,47 @@
import argparse import argparse
from uncloud.common.settings import settings
from uncloud.common.request import RequestEntry, RequestType from uncloud.common.request import RequestEntry, RequestType
from uncloud.shared import shared from uncloud.common.shared import shared
from uncloud.settings import settings from uncloud.scheduler import logger
from .helper import (dead_host_mitigation, dead_host_detection, assign_host, NoSuitableHostFound) from uncloud.scheduler.helper import (dead_host_mitigation, dead_host_detection,
from . import logger assign_host, NoSuitableHostFound)
arg_parser = argparse.ArgumentParser('scheduler', add_help=False) arg_parser = argparse.ArgumentParser('scheduler', add_help=False)
def main(debug=False): def main(arguments):
for request_iterator in [ # The below while True is neccessary for gracefully handling leadership transfer and temporary
shared.etcd_client.get_prefix( # unavailability in etcd. Why does it work? It works because the get_prefix,watch_prefix return
settings["etcd"]["request_prefix"], value_in_json=True # iter([]) that is iterator of empty list on exception (that occur due to above mentioned reasons)
), # which ends the loop immediately. So, having it inside infinite loop we try again and again to
shared.etcd_client.watch_prefix( # get prefix until either success or deamon death comes.
settings["etcd"]["request_prefix"], while True:
timeout=5, for request_iterator in [
value_in_json=True, shared.etcd_client.get_prefix(settings['etcd']['request_prefix'], value_in_json=True,
), raise_exception=False),
]: shared.etcd_client.watch_prefix(settings['etcd']['request_prefix'], value_in_json=True,
for request_event in request_iterator: raise_exception=False),
request_entry = RequestEntry(request_event) ]:
# Never Run time critical mechanism inside timeout for request_event in request_iterator:
# mechanism because timeout mechanism only comes dead_host_mitigation(dead_host_detection())
# when no other event is happening. It means under request_entry = RequestEntry(request_event)
# heavy load there would not be a timeout event.
if request_entry.type == "TIMEOUT":
# Detect hosts that are dead and set their status if request_entry.type == RequestType.ScheduleVM:
# to "DEAD", and their VMs' status to "KILLED" logger.debug('%s, %s', request_entry.key, request_entry.value)
dead_hosts = dead_host_detection()
if dead_hosts:
logger.debug("Dead hosts: %s", dead_hosts)
dead_host_mitigation(dead_hosts)
elif request_entry.type == RequestType.ScheduleVM: vm_entry = shared.vm_pool.get(request_entry.uuid)
logger.debug("%s, %s", request_entry.key, request_entry.value) if vm_entry is None:
logger.info('Trying to act on {} but it is deleted'.format(request_entry.uuid))
continue
vm_entry = shared.vm_pool.get(request_entry.uuid) shared.etcd_client.client.delete(request_entry.key) # consume Request
if vm_entry is None:
logger.info("Trying to act on {} but it is deleted".format(request_entry.uuid))
continue
shared.etcd_client.client.delete(request_entry.key) # consume Request try:
assign_host(vm_entry)
except NoSuitableHostFound:
vm_entry.add_log('Can\'t schedule VM. No Resource Left.')
shared.vm_pool.put(vm_entry)
try: logger.info('No Resource Left. Emailing admin....')
assign_host(vm_entry)
except NoSuitableHostFound:
vm_entry.add_log("Can't schedule VM. No Resource Left.")
shared.vm_pool.put(vm_entry)
logger.info("No Resource Left. Emailing admin....")
if __name__ == "__main__":
main()

View file

@ -190,18 +190,10 @@ class VMM:
err.stderr.decode("utf-8"), err.stderr.decode("utf-8"),
) )
else: else:
with suppress(sp.CalledProcessError): sp.check_output(
sp.check_output( ["sudo", "-p", "Enter password to correct permission for uncloud-vmm's directory",
[ "chmod", "-R", "o=rwx,g=rwx", self.vmm_backend]
"sudo", )
"-p",
"Enter password to correct permission for uncloud-vmm's directory",
"chmod",
"-R",
"o=rwx,g=rwx",
self.vmm_backend,
]
)
# TODO: Find some good way to check whether the virtual machine is up and # TODO: Find some good way to check whether the virtual machine is up and
# running without relying on non-guarenteed ways. # running without relying on non-guarenteed ways.