Add working --last-used-mac

{'create_vm': False, 'last_used_mac': True, 'get_new_mac': False, 'debug': False, 'conf_dir': '/home/nico/uncloud', 'etcd_host': 'etcd1.ungleich.ch', 'etcd_port': None, 'etcd_ca_cert': '/home/nico/vcs/ungleich-dot-cdist/files/etcd/ca.pem', 'etcd_cert_cert': '/home/nico/vcs/ungleich-dot-cdist/files/etcd/nico.pem', 'etcd_cert_key': '/home/nico/vcs/ungleich-dot-cdist/files/etcd/nico-key.pem'}
00:20:00:00:00:00
(venv) [19:02] diamond:uncloud% ./bin/uncloud-run-reinstall hack  --etcd-host etcd1.ungleich.ch --etcd-ca-cert /home/nico/vcs/ungleich-dot-cdist/files/etcd/ca.pem --etcd-cert-cert /home/nico/vcs/ungleich-dot-cdist/files/etcd/nico.pem --etcd-cert-key /home/nico/vcs/ungleich-dot-cdist/files/etcd/nico-key.pem --last-used-mac
This commit is contained in:
Nico Schottelius 2020-01-14 19:02:15 +01:00
parent 1b36c2f96f
commit 8078ffae5a
4 changed files with 132 additions and 102 deletions

View file

@ -22,30 +22,75 @@
import etcd3 import etcd3
import json import json
import logging
from functools import wraps
from uncloud import UncloudException
log = logging.getLogger(__name__)
def readable_errors(func):
@wraps(func)
def wrapper(*args, **kwargs):
try:
return func(*args, **kwargs)
except etcd3.exceptions.ConnectionFailedError as e:
raise UncloudException('Cannot connect to etcd: is etcd running and reachable? {}'.format(e))
except etcd3.exceptions.ConnectionTimeoutError as e:
raise UncloudException('etcd connection timeout. {}'.format(e))
return wrapper
class DB(object): class DB(object):
def __init__(self, config): def __init__(self, config, prefix="/"):
self.config = config self.config = config
self.prefix= '/nicohack/'
# Root for everything
self.base_prefix= '/nicohack'
# Can be set from outside
self.prefix = prefix
self.connect() self.connect()
@readable_errors
def connect(self): def connect(self):
self._db_clients = [] self._db_clients = []
for endpoint in self.config.etcd_hosts: for endpoint in self.config.etcd_hosts:
client = etcd3.client(host=endpoint, **self.config.etcd_args) client = etcd3.client(host=endpoint, **self.config.etcd_args)
self._db_clients.append(client) self._db_clients.append(client)
def get_value(self, key): def realkey(self, key):
pass return "{}{}/{}".format(self.base_prefix,
self.prefix,
key)
def set(self, key, value, store_as_json=False, **kwargs): @readable_errors
if store_as_json: def get(self, key, as_json=False, **kwargs):
value, _ = self._db_clients[0].get(self.realkey(key), **kwargs)
if as_json:
value = json.loads(value)
return value
@readable_errors
def set(self, key, value, as_json=False, **kwargs):
if as_json:
value = json.dumps(value) value = json.dumps(value)
key = "{}/{}".format(self.prefix, key)
# FIXME: iterate over clients in case of failure ? # FIXME: iterate over clients in case of failure ?
return self._db_clients[0].put(key, value, **kwargs) return self._db_clients[0].put(self.realkey(key), value, **kwargs)
@readable_errors
def increment(key, **kwargs):
with self._db_clients[0].lock(key) as lock:
value = int(self.get(self.realkey(key), **kwargs))
self.set(self.realkey(key), str(value + 1), **kwargs)
if __name__ == '__main__': if __name__ == '__main__':
endpoints = [ "https://etcd1.ungleich.ch:2379", endpoints = [ "https://etcd1.ungleich.ch:2379",

View file

@ -25,93 +25,65 @@ import logging
import os.path import os.path
import os import os
import re import re
import json
from uncloud import UncloudException
from uncloud.hack.db import DB
log = logging.getLogger(__name__) log = logging.getLogger(__name__)
class Error(Exception):
pass
class MAC(object):
class Mac(object): def __init__(self, config):
def __init__(self): self.config = config
self.base_dir = "." self.db = DB(config, prefix="/mac")
self.prefix = 0x002000000000 self.prefix = 0x002000000000
#self.prefix = "{:012x}".format(self._prefix)
self.free = self.read_file("mac-free")
self.last = self.read_file("mac-last")
def read_file(self, filename):
fname = os.path.join(self.base_dir, filename)
ret = []
try:
with open(fname, "r") as fd:
ret = fd.readlines()
except Exception as e:
pass
return ret
def append_to_file(self, text, filename):
fname = os.path.join(self.base_dir, filename)
with open(fname, "a+") as fd:
fd.write("{}\n".format(text))
@staticmethod @staticmethod
def validate_mac(mac): def validate_mac(mac):
if not re.match(r'([0-9A-F]{2}[-:]){5}[0-9A-F]{2}$', mac, re.I): if not re.match(r'([0-9A-F]{2}[-:]){5}[0-9A-F]{2}$', mac, re.I):
raise Error("Not a valid mac address: %s" % mac) raise Error("Not a valid mac address: %s" % mac)
def free_append(self, mac): def last_used_index(self):
if mac in self.free: value = self.db.get("last_used_index")
raise Error("Mac already in free database: %s" % mac) if not value:
return 0
self.append_to_file(mac, "mac-free") return int(value)
self.free = self.read_file("mac-free")
def last_used_mac(self):
return self.int_to_mac(self.prefix + self.last_used_index())
@staticmethod @staticmethod
def int_to_mac(number): def int_to_mac(number):
b = number.to_bytes(6, byteorder="big") b = number.to_bytes(6, byteorder="big")
return ':'.join(format(s, '02x') for s in b) return ':'.join(format(s, '02x') for s in b)
def getnext(self): def get_next(self, vmuuid=None, as_int=False):
# if self.free: # if self.free:
# return self.free.pop() # return self.free.pop()
# if not self.prefix: last_number = self.last_used_index()
# raise Error("Cannot generate address without prefix - use prefix-set")
if self.last: # FIXME: compare to 48bit minus prefix length
last_number = int(self.last[0], 16) if last_number == int('0xffffff', 16):
raise UncloudException("Exhausted all possible mac addresses - try to free some")
if last_number == int('0xffffff', 16):
raise Error("Exhausted all possible mac addresses - try to free some")
next_number = last_number + 1
else:
next_number = 0
next_number = last_number + 1
next_number_string = "{:012x}".format(next_number) next_number_string = "{:012x}".format(next_number)
next_mac_number = self.prefix + next_number next_mac_number = self.prefix + next_number
next_mac = self.int_to_mac(next_mac_number) next_mac = self.int_to_mac(next_mac_number)
with open(os.path.join(self.base_dir, "mac-last"), "w+") as fd: db_entry = {}
fd.write("{}\n".format(next_number_string)) db_entry['vm_uuid'] = vmuuid
db_entry['index'] = next_number
db_entry['mac_address'] = next_mac
return next_mac self.db.set("used/{}".format(next_mac),
db_entry)
@classmethod if as_int:
def commandline(cls): return next_mac_number
pass else:
return next_mac
if __name__ == '__main__':
m = Mac()
m.commandline()
# print(m.free)
#print(m.last)
print(m.getnext())

View file

@ -2,10 +2,13 @@ import argparse
from uncloud.hack.vm import VM from uncloud.hack.vm import VM
from uncloud.hack.config import Config from uncloud.hack.config import Config
from uncloud.hack.mac import MAC
arg_parser = argparse.ArgumentParser('hack', add_help=False) arg_parser = argparse.ArgumentParser('hack', add_help=False)
#description="Commands that are unfinished - use at own risk") #description="Commands that are unfinished - use at own risk")
arg_parser.add_argument('--create-vm', action='store_true') arg_parser.add_argument('--create-vm', action='store_true')
arg_parser.add_argument('--last-used-mac', action='store_true')
arg_parser.add_argument('--get-new-mac', action='store_true')
def main(arguments): def main(arguments):
@ -16,3 +19,11 @@ def main(arguments):
print("Creating VM") print("Creating VM")
vm = VM(config) vm = VM(config)
vm.create() vm.create()
if arguments['last_used_mac']:
m = MAC(config)
print(m.last_used_mac())
if arguments['get_new_mac']:
m = MAC(config).get_next()
print(m.last_used())

View file

@ -25,48 +25,50 @@ import uuid
import os import os
from uncloud.hack.db import DB from uncloud.hack.db import DB
from uncloud.hack.mac import MAC
class VM(object): class VM(object):
def __init__(self, config): def __init__(self, config):
self.config = config self.config = config
self.db = DB(config) self.db = DB(config, prefix="/vm")
self.hackprefix="/home/nico/vcs/uncloud/uncloud/hack/hackcloud" self.hackprefix="/home/nico/vcs/uncloud/uncloud/hack/hackcloud"
self.qemu="/usr/bin/qemu-system-x86_64" self.qemu="/usr/bin/qemu-system-x86_64"
self.accel="kvm"
self.vm = {} self.vm = {}
self.accel="kvm"
# self.mac=$(./mac-gen.py) self.owner="nico"
self.mac="42:00:00:00:00:42" self.bridge="br100"
self.owner="nico"
self.bridge="br100"
self.ifup = os.path.join(self.hackprefix, "ifup.sh") self.ifup = os.path.join(self.hackprefix, "ifup.sh")
self.ifdown = os.path.join(self.hackprefix, "ifdown.sh") self.ifdown = os.path.join(self.hackprefix, "ifdown.sh")
self.uuid = uuid.uuid4() def create(self):
self.vm['uuid'] = str(self.uuid) self.uuid = uuid.uuid4()
self.vm['memory']=1024 self.vm['uuid'] = str(self.uuid)
self.vm['cores']=2 self.vm['memory'] = 1024
self.vm['os_image'] = os.path.join(self.hackprefix, "alpine-virt-3.11.2-x86_64.iso") self.vm['cores'] = 2
self.vm['os_image'] = os.path.join(self.hackprefix, "alpine-virt-3.11.2-x86_64.iso")
self.vm['commandline' ] = [ "sudo", self.mac=MAC().next()
"{}".format(self.qemu),
"-name", "uncloud-{}".format(self.vm['uuid']),
"-machine", "pc,accel={}".format(self.accel),
"-m", "{}".format(self.vm['memory']),
"-smp", "{}".format(self.vm['cores']),
"-uuid", "{}".format(self.vm['uuid']),
"-drive", "file={},media=cdrom".format(self.vm['os_image']),
"-netdev", "tap,id=netmain,script={},downscript={}".format(self.ifup, self.ifdown),
"-device", "virtio-net-pci,netdev=netmain,id=net0,mac={}".format(self.mac)
]
def create(self): self.vm['commandline' ] = [ "sudo",
self.db.set("vm/{}".format(str(self.vm['uuid'])), "{}".format(self.qemu),
self.vm, store_as_json=True) "-name", "uncloud-{}".format(self.vm['uuid']),
"-machine", "pc,accel={}".format(self.accel),
"-m", "{}".format(self.vm['memory']),
"-smp", "{}".format(self.vm['cores']),
"-uuid", "{}".format(self.vm['uuid']),
"-drive", "file={},media=cdrom".format(self.vm['os_image']),
"-netdev", "tap,id=netmain,script={},downscript={}".format(self.ifup, self.ifdown),
"-device", "virtio-net-pci,netdev=netmain,id=net0,mac={}".format(self.mac)
]
print(" ".join(self.vm['commandline'])) self.db.set(str(self.vm['uuid']),
subprocess.run(self.vm['commandline']) self.vm,
as_json=True)
print(" ".join(self.vm['commandline']))
subprocess.run(self.vm['commandline'])