| # -*- coding: utf-8 -*- |
| """ |
| Module for handling maas calls. |
| |
| :optdepends: pyapi-maas Python adapter |
| :configuration: This module is not usable until the following are specified |
| either in a pillar or in the minion's config file:: |
| |
| maas.url: 'https://maas.domain.com/' |
| maas.token: fdsfdsdsdsfa:fsdfae3fassd:fdsfdsfsafasdfsa |
| |
| """ |
| |
| from __future__ import absolute_import |
| |
| import collections |
| import copy |
| import hashlib |
| import io |
| import json |
| import logging |
| import os.path |
| import time |
| import urllib2 |
| # Salt utils |
| from salt.exceptions import CommandExecutionError, SaltInvocationError |
| |
| LOG = logging.getLogger(__name__) |
| |
| SIZE = { |
| "M": 1000000, |
| "G": 1000000000, |
| "T": 1000000000000, |
| } |
| |
| RAID = { |
| 0: "raid-0", |
| 1: "raid-1", |
| 5: "raid-5", |
| 10: "raid-10", |
| } |
| |
| # Import third party libs |
| HAS_MASS = False |
| try: |
| from maas_client import MAASClient, MAASDispatcher, MAASOAuth |
| HAS_MASS = True |
| except ImportError: |
| LOG.debug('Missing MaaS client module is Missing. Skipping') |
| |
| |
| def __virtual__(): |
| """ |
| Only load this module if maas-client |
| is installed on this minion. |
| """ |
| if HAS_MASS: |
| return 'maasng' |
| return False |
| |
| |
| APIKEY_FILE = '/var/lib/maas/.maas_credentials' |
| |
| |
| def _format_data(data): |
| class Lazy: |
| def __str__(self): |
| return ' '.join(['{0}={1}'.format(k, v) |
| for k, v in data.iteritems()]) |
| return Lazy() |
| |
| |
| def _create_maas_client(): |
| global APIKEY_FILE |
| try: |
| api_token = file(APIKEY_FILE).read().splitlines()[-1].strip()\ |
| .split(':') |
| except: |
| LOG.exception('token') |
| auth = MAASOAuth(*api_token) |
| api_url = 'http://localhost:5240/MAAS' |
| dispatcher = MAASDispatcher() |
| return MAASClient(auth, dispatcher, api_url) |
| |
| |
| def _get_blockdevice_id_by_name(hostname, device): |
| |
| # TODO validation |
| return list_blockdevices(hostname)[device]["id"] |
| |
| |
| def _get_volume_group_id_by_name(hostname, device): |
| |
| # TODO validation |
| return list_volume_groups(hostname)[device]["id"] |
| |
| |
| def _get_volume_id_by_name(hostname, volume_name, volume_group, maas_volname=True): |
| |
| if not maas_volname: |
| # MAAS-like name |
| volume_name = str("%s-%s" % (volume_group,volume_name)) |
| ##TODO validation |
| return get_volumes(hostname, volume_group)[volume_name]["id"] |
| |
| |
| def _get_partition_id_by_name(hostname, device, partition): |
| |
| # TODO validation |
| return list_partitions(hostname, device)[partition]["id"] |
| |
| # MACHINE SECTION |
| |
| |
| def get_machine(hostname): |
| """ |
| Get information aboout specified machine |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt-call maasng.get_machine server_hostname |
| """ |
| try: |
| return list_machines()[hostname] |
| except KeyError: |
| return {"error": "Machine not found on MaaS server"} |
| |
| |
| def list_machines(): |
| """ |
| Get list of all machines from maas server |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'maas-node' maasng.list_machines |
| """ |
| machines = {} |
| maas = _create_maas_client() |
| json_res = json.loads(maas.get(u'api/2.0/machines/').read()) |
| for item in json_res: |
| machines[item["hostname"]] = item |
| return machines |
| |
| |
| def create_machine(): |
| # TODO |
| |
| return False |
| |
| |
| def update_machine(): |
| # TODO |
| |
| return False |
| |
| # END MACHINE SECTION |
| # RAID SECTION |
| |
| |
| def create_raid(hostname, name, level, disks=[], partitions=[], **kwargs): |
| """ |
| Create new raid on machine. |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt-call maasng.create_raid hostname=kvm03 name=md0 level=1 disks=[vdb,vdc] partitions=[vdd-part1,vde-part1] |
| """ |
| |
| result = {} |
| |
| if len(disks) == 0 and len(partitions) == 0: |
| result["error"] = "Disks or partitions need to be provided" |
| |
| disk_ids = [] |
| partition_ids = [] |
| |
| for disk in disks: |
| try: |
| disk_ids.append(str(_get_blockdevice_id_by_name(hostname, disk))) |
| except KeyError: |
| result["error"] = "Device {0} does not exists on machine {1}".format( |
| disk, hostname) |
| return result |
| |
| for partition in partitions: |
| try: |
| device = partition.split("-")[0] |
| device_part = list_partitions(hostname, device) |
| partition_ids.append(str(device_part[partition]["id"])) |
| except KeyError: |
| result["error"] = "Partition {0} does not exists on machine {1}".format( |
| partition, hostname) |
| return result |
| |
| data = { |
| "name": name, |
| "level": RAID[int(level)], |
| "block_devices": disk_ids, |
| "partitions": partition_ids, |
| } |
| |
| maas = _create_maas_client() |
| system_id = get_machine(hostname)["system_id"] |
| LOG.info(system_id) |
| |
| # TODO validation |
| LOG.info(data) |
| json_res = json.loads( |
| maas.post(u"api/2.0/nodes/{0}/raids/".format(system_id), None, **data).read()) |
| LOG.info(json_res) |
| result["new"] = "Raid {0} created".format(name) |
| |
| return result |
| |
| |
| def list_raids(hostname): |
| """ |
| Get list all raids on machine |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt-call maasng.list_raids server_hostname |
| """ |
| |
| raids = {} |
| maas = _create_maas_client() |
| system_id = get_machine(hostname)["system_id"] |
| #TODO validation |
| json_res = json.loads(maas.get(u"api/2.0/nodes/{0}/raids/".format(system_id)).read()) |
| LOG.debug('list_raids:{} {}'.format(system_id, json_res)) |
| for item in json_res: |
| raids[item["name"]] = item |
| return raids |
| |
| |
| def get_raid(hostname, name): |
| """ |
| Get information about specific raid on machine |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt-call maasng.get_raids server_hostname md0 |
| """ |
| |
| return list_raids(hostname)[name] |
| |
| |
| def _get_raid_id_by_name(hostname, raid_name): |
| return get_raid(hostname, raid_name)['id'] |
| |
| |
| def delete_raid(hostname, raid_name): |
| """ |
| Delete RAID on a machine. |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'maas-node' maasng.delete_raid server_hostname raid_name |
| salt-call maasng.delete_raid server_hostname raid_name |
| """ |
| result = {} |
| maas=_create_maas_client() |
| system_id = get_machine(hostname)["system_id"] |
| raid_id = _get_raid_id_by_name(hostname, raid_name) |
| LOG.debug('delete_raid: {} {}'.format(system_id,raid_id)) |
| maas.delete(u"api/2.0/nodes/{0}/raid/{1}/".format(system_id, raid_id)).read() |
| |
| result["new"] = "Raid {0} deleted".format(raid_name) |
| return result |
| |
| # END RAID SECTION |
| # BLOCKDEVICES SECTION |
| |
| |
| def list_blockdevices(hostname): |
| """ |
| Get list of all blockdevices (disks) on machine |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'maas-node' maasng.list_blockdevices server_hostname |
| salt-call maasng.list_blockdevices server_hostname |
| """ |
| ret = {} |
| |
| maas = _create_maas_client() |
| system_id = get_machine(hostname)["system_id"] |
| LOG.info(system_id) |
| |
| # TODO validation if exists |
| |
| json_res = json.loads( |
| maas.get(u"api/2.0/nodes/{0}/blockdevices/".format(system_id)).read()) |
| LOG.info(json_res) |
| for item in json_res: |
| ret[item["name"]] = item |
| |
| return ret |
| |
| |
| def get_blockdevice(hostname, name): |
| """ |
| Get information about blockdevice (disk) on machine |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'maas-node' maasng.get_blockdevice server_hostname sda |
| salt-call maasng.get_blockdevice server_hostname sda |
| """ |
| |
| return list_blockdevices(hostname)[name] |
| |
| # END BLOCKDEVICES SECTION |
| # PARTITIONS |
| |
| |
| def list_partitions(hostname, device): |
| """ |
| Get list of all partitions on specific device located on specific machine |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'maas-node' maasng.list_partitions server_hostname sda |
| salt-call maasng.list_partitions server_hostname sda |
| """ |
| ret = {} |
| maas = _create_maas_client() |
| system_id = get_machine(hostname)["system_id"] |
| LOG.info(system_id) |
| |
| partitions = get_blockdevice(hostname, device)["partitions"] |
| LOG.info(partitions) |
| |
| #json_res = json.loads(maas.get(u"api/2.0/nodes/{0}/blockdevices/{1}/partitions/".format(system_id, device_id)).read()) |
| # LOG.info(json_res) |
| |
| if len(device) > 0: |
| for item in partitions: |
| name = item["path"].split('/')[-1] |
| ret[name] = item |
| |
| return ret |
| |
| |
| def get_partition(hostname, device, partition): |
| """ |
| Get information about specific parition on device located on machine |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'maas-node' maasng.get_partition server_hostname disk_name partition |
| salt-call maasng.get_partition server_hostname disk_name partition |
| |
| root_size = size in GB |
| """ |
| |
| return list_partitions(partition)[name] |
| |
| |
| def create_partition(hostname, disk, size, fs_type=None, mount=None): |
| """ |
| Create new partition on device. |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'maas-node' maasng.create_partition server_hostname disk_name 10 ext4 "/" |
| salt-call maasng.create_partition server_hostname disk_name 10 ext4 "/" |
| """ |
| # TODO validation |
| result = {} |
| maas = _create_maas_client() |
| system_id = get_machine(hostname)["system_id"] |
| LOG.info(system_id) |
| |
| device_id = _get_blockdevice_id_by_name(hostname, disk) |
| LOG.info(device_id) |
| |
| value, unit = size[:-1], size[-1] |
| calc_size = str(int(value) * SIZE[unit]) |
| LOG.info(calc_size) |
| |
| data = { |
| "size": calc_size |
| } |
| |
| # TODO validation |
| partition = json.loads(maas.post( |
| u"api/2.0/nodes/{0}/blockdevices/{1}/partitions/".format(system_id, device_id), None, **data).read()) |
| LOG.info(partition) |
| result["partition"] = "Partition created on {0}".format(disk) |
| |
| if fs_type != None: |
| data_fs_type = { |
| "fstype": fs_type |
| } |
| partition_id = str(partition["id"]) |
| LOG.info("Partition id: " + partition_id) |
| # TODO validation |
| json_res = json.loads(maas.post(u"api/2.0/nodes/{0}/blockdevices/{1}/partition/{2}".format( |
| system_id, device_id, partition_id), "format", **data_fs_type).read()) |
| LOG.info(json_res) |
| result["filesystem"] = "Filesystem {0} created".format(fs_type) |
| |
| if mount != None: |
| data = { |
| "mount_point": mount |
| } |
| |
| # TODO validation |
| json_res = json.loads(maas.post(u"api/2.0/nodes/{0}/blockdevices/{1}/partition/{2}".format( |
| system_id, device_id, str(partition['id'])), "mount", **data).read()) |
| LOG.info(json_res) |
| result["mount"] = "Mount point {0} created".format(mount) |
| |
| return result |
| |
| |
| def delete_partition(hostname, disk, partition_name): |
| """ |
| Delete partition on device. |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'maas-node' maasng.delete_partition server_hostname disk_name partition_name |
| salt-call maasng.delete_partition server_hostname disk_name partition_name |
| |
| root_size = size in GB |
| """ |
| result = {} |
| data = {} |
| maas = _create_maas_client() |
| system_id = get_machine(hostname)["system_id"] |
| LOG.info(system_id) |
| |
| device_id = _get_blockdevice_id_by_name(hostname, disk) |
| LOG.info(device_id) |
| |
| partition_id = _get_partition_id_by_name(hostname, disk, partition_name) |
| |
| maas.delete(u"api/2.0/nodes/{0}/blockdevices/{1}/partition/{2}".format( |
| system_id, device_id, partition_id)).read() |
| result["new"] = "Partition {0} deleted".format(partition_name) |
| return result |
| |
| |
| def delete_partition_by_id(hostname, disk, partition_id): |
| """ |
| Delete partition on device. Partition spefified by id of parition |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'maas-node' maasng.delete_partition_by_id server_hostname disk_name partition_id |
| salt-call maasng.delete_partition_by_id server_hostname disk_name partition_id |
| |
| root_size = size in GB |
| """ |
| result = {} |
| data = {} |
| maas = _create_maas_client() |
| system_id = get_machine(hostname)["system_id"] |
| LOG.info(system_id) |
| |
| device_id = _get_blockdevice_id_by_name(hostname, disk) |
| LOG.info(device_id) |
| |
| maas.delete(u"api/2.0/nodes/{0}/blockdevices/{1}/partition/{2}".format( |
| system_id, device_id, partition_id)).read() |
| result["new"] = "Partition {0} deleted".format(partition_id) |
| return result |
| # END PARTITIONS |
| # DISK LAYOUT |
| |
| |
| def drop_storage_schema(hostname,disk=None): |
| """ |
| #1. Drop lv |
| #2. Drop vg |
| #3. Drop md # need to zero-block? |
| #3. Drop part |
| """ |
| |
| if __opts__['test']: |
| ret['result'] = None |
| ret['comment'] = 'Storage schema on {0} will be removed'.format(hostname) |
| return ret |
| #TODO validation if exists |
| vgs = list_volume_groups(hostname) |
| for vg in vgs: |
| delete_volume_group(hostname, vg) |
| |
| raids = list_raids(hostname) |
| for raid in raids: |
| delete_raid(hostname, raid) |
| |
| blocks = list_blockdevices(hostname) |
| for block_d in blocks: |
| partitions = __salt__['maasng.list_partitions'](hostname, block_d) |
| for partition_name, partition in partitions.iteritems(): |
| LOG.info('delete partition:\n{}'.format(partition)) |
| __salt__['maasng.delete_partition_by_id'](hostname, block_d, partition["id"]) |
| |
| |
| def update_disk_layout(hostname, layout, root_size=None, root_device=None, volume_group=None, volume_name=None, volume_size=None): |
| """ |
| Update disk layout. Flat or LVM layout supported. |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'maas-node' maasng.update_disk_layout server_hostname lvm root_size=None, root_device=None, volume_group=None, volume_name=None, volume_size=None |
| salt-call maasng.update_disk_layout server_hostname lvm root_size=None, root_device=None, volume_group=None, volume_name=None, volume_size=None |
| |
| root_size = size in GB |
| """ |
| result = {} |
| data = { |
| "storage_layout": layout, |
| } |
| |
| maas = _create_maas_client() |
| system_id = get_machine(hostname)["system_id"] |
| LOG.info(system_id) |
| |
| if layout == 'custom': |
| drop_storage_schema(hostname) |
| result["new"] = { |
| "storage_layout": layout, |
| } |
| |
| return result |
| |
| if root_size != None: |
| bit_size = str(root_size * 1073741824) |
| LOG.info(bit_size) |
| data["root_size"] = bit_size |
| |
| if root_device != None: |
| LOG.info(root_device) |
| data["root_device"] = str( |
| _get_blockdevice_id_by_name(hostname, root_device)) |
| |
| if layout == 'lvm': |
| if volume_group != None: |
| LOG.info(volume_group) |
| data["vg_name"] = volume_group |
| if volume_name != None: |
| LOG.info(volume_name) |
| data["lv_name"] = volume_name |
| if volume_size != None: |
| vol_size = str(volume_size * 1073741824) |
| LOG.info(vol_size) |
| data["lv_size"] = vol_size |
| |
| # TODO validation |
| json_res = json.loads(maas.post( |
| u"api/2.0/machines/{0}/".format(system_id), "set_storage_layout", **data).read()) |
| LOG.info(json_res) |
| result["new"] = { |
| "storage_layout": layout, |
| } |
| |
| return result |
| |
| # END DISK LAYOUT |
| # LVM |
| |
| def list_volume_groups(hostname): |
| """ |
| Get list of all volume group on machine. |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'maas-node' maasng.list_volume_groups server_hostname |
| salt-call maasng.list_volume_groups server_hostname |
| """ |
| volume_groups = {} |
| |
| maas = _create_maas_client() |
| system_id = get_machine(hostname)["system_id"] |
| LOG.info(system_id) |
| |
| # TODO validation if exists |
| |
| json_res = json.loads( |
| maas.get(u"api/2.0/nodes/{0}/volume-groups/".format(system_id)).read()) |
| LOG.info(json_res) |
| for item in json_res: |
| volume_groups[item["name"]] = item |
| # return |
| return volume_groups |
| |
| |
| def get_volume_group(hostname, name): |
| """ |
| Get information about specific volume group on machine. |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'maas-node' maasng.list_blockdevices server_hostname |
| salt-call maasng.list_blockdevices server_hostname |
| """ |
| # TODO validation that exists |
| return list_volume_groups(hostname)[name] |
| |
| |
| def create_volume_group(hostname, volume_group_name, disks=[], partitions=[]): |
| """ |
| Create new volume group on machine. Disks or partitions needs to be provided. |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'maas-node' maasng.create_volume_group volume_group_name, disks=[sda,sdb], partitions=[] |
| salt-call maasng.create_volume_group server_hostname |
| """ |
| result = {} |
| |
| data = { |
| "name": volume_group_name, |
| } |
| |
| maas = _create_maas_client() |
| system_id = get_machine(hostname)["system_id"] |
| LOG.info(system_id) |
| |
| disk_ids = [] |
| partition_ids = [] |
| |
| for disk in disks: |
| p_disk = get_blockdevice(hostname, disk) |
| if p_disk["partition_table_type"] == None: |
| disk_ids.append(str(p_disk["id"])) |
| else: |
| result["error"] = "Device {0} on machine {1} cointains partition table".format( |
| disk, hostname) |
| return result |
| |
| for partition in partitions: |
| try: |
| device = partition.split("-")[0] |
| device_part = list_partitions(hostname, device) |
| partition_ids.append(str(device_part[partition]["id"])) |
| except KeyError: |
| result["error"] = "Partition {0} does not exists on machine {1}".format( |
| partition, hostname) |
| return result |
| |
| data["block_devices"] = disk_ids |
| data["partitions"] = partition_ids |
| LOG.info(partition_ids) |
| LOG.info(partitions) |
| |
| # TODO validation |
| json_res = json.loads(maas.post( |
| u"api/2.0/nodes/{0}/volume-groups/".format(system_id), None, **data).read()) |
| LOG.info(json_res) |
| result["new"] = "Volume group {0} created".format(json_res["name"]) |
| |
| return result |
| |
| |
| def delete_volume_group(hostname, name): |
| """ |
| Delete volume group on machine. |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'maas-node' maasng.delete_volume_group server_hostname vg0 |
| salt-call maasng.delete_volume_group server_hostname vg0 |
| """ |
| |
| maas = _create_maas_client() |
| system_id = get_machine(hostname)["system_id"] |
| LOG.debug('delete_volume_group:{}'.format(system_id)) |
| |
| vg_id = str(_get_volume_group_id_by_name(hostname, name)) |
| for vol in get_volumes(hostname, name): |
| delete_volume(hostname,vol,name) |
| |
| #TODO validation |
| json_res = json.loads(maas.delete(u"api/2.0/nodes/{0}/volume-group/{1}/".format(system_id, vg_id)).read() or 'null') |
| LOG.info(json_res) |
| |
| return True |
| |
| |
| def create_volume(hostname, volume_name, volume_group, size, fs_type=None, mount=None): |
| """ |
| Create volume on volume group. |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'maas-node' maasng.create_volume server_hostname volume_name, volume_group, size, fs_type=None, mount=None |
| salt-call maasng.create_volume server_hostname volume_name, volume_group, size, fs_type=None, mount=None |
| """ |
| |
| data = { |
| "name": volume_name, |
| } |
| |
| value, unit = size[:-1], size[-1] |
| bit_size = str(int(value) * SIZE[unit]) |
| LOG.info(bit_size) |
| |
| data["size"] = bit_size |
| |
| maas = _create_maas_client() |
| system_id = get_machine(hostname)["system_id"] |
| LOG.info(system_id) |
| |
| volume_group_id = str(_get_volume_group_id_by_name(hostname, volume_group)) |
| |
| LOG.info(volume_group_id) |
| |
| # TODO validation |
| json_res = json.loads(maas.post(u"api/2.0/nodes/{0}/volume-group/{1}/".format( |
| system_id, volume_group_id), "create_logical_volume", **data).read()) |
| LOG.info(json_res) |
| |
| if fs_type != None or mount != None: |
| ret = create_volume_filesystem(hostname, volume_group + "-" + volume_name, fs_type, mount) |
| |
| return True |
| |
| |
| def delete_volume(hostname, volume_name, volume_group): |
| """ |
| Delete volume from volume group. |
| Tips: maas always use 'volume_group-volume_name' name schema.Example: 'vg0-glusterfs' |
| This function expexts same format. |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'maas-node' maasng.delete_volume server_hostname volume_name volume_group |
| salt 'maas-node' maasng.delete_volume server_hostname vg0-vol0 vg0 |
| salt-call maasng.delete_volume server_hostname volume_name volume_group |
| """ |
| |
| maas=_create_maas_client() |
| system_id = get_machine(hostname)["system_id"] |
| LOG.debug('delete_volume:{}'.format(system_id)) |
| |
| volume_group_id = str(_get_volume_group_id_by_name(hostname, volume_group)) |
| volume_id = str(_get_volume_id_by_name(hostname, volume_name, volume_group)) |
| |
| if None in [volume_group_id, volume_id]: |
| return False |
| |
| data = { |
| "id": volume_id, |
| } |
| |
| #TODO validation |
| json_res = json.loads(maas.post(u"api/2.0/nodes/{0}/volume-group/{1}/".format(system_id, volume_group_id), "delete_logical_volume", **data).read() or 'null') |
| return True |
| |
| |
| def get_volumes(hostname, vg_name): |
| """ |
| Get list of volumes in volume group. |
| """ |
| volumes = {} |
| _volumes = list_volume_groups(hostname)[vg_name].get('logical_volumes', False) |
| if _volumes: |
| for item in _volumes: |
| volumes[item["name"]] = item |
| return volumes |
| |
| # END LVM |
| |
| |
| def create_volume_filesystem(hostname, device, fs_type=None, mount=None): |
| |
| maas = _create_maas_client() |
| system_id = get_machine(hostname)["system_id"] |
| |
| blockdevices_id = _get_blockdevice_id_by_name(hostname, device) |
| data = {} |
| if fs_type != None: |
| data["fstype"] = fs_type |
| # TODO validation |
| json_res = json.loads(maas.post(u"/api/2.0/nodes/{0}/blockdevices/{1}/".format( |
| system_id, blockdevices_id), "format", **data).read()) |
| LOG.info(json_res) |
| |
| if mount != None: |
| data["mount_point"] = mount |
| # TODO validation |
| json_res = json.loads(maas.post(u"/api/2.0/nodes/{0}/blockdevices/{1}/".format( |
| system_id, blockdevices_id), "mount", **data).read()) |
| LOG.info(json_res) |
| |
| return True |
| |
| |
| def set_boot_disk(hostname, name): |
| """ |
| Create volume on volume group. |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'maas-node' maasng.set_boot_disk server_hostname disk_name |
| salt-call maasng.set_boot_disk server_hostname disk_name |
| """ |
| data = {} |
| result = {} |
| maas = _create_maas_client() |
| system_id = get_machine(hostname)["system_id"] |
| blockdevices_id = _get_blockdevice_id_by_name(hostname, name) |
| |
| maas.post(u"/api/2.0/nodes/{0}/blockdevices/{1}/".format( |
| system_id, blockdevices_id), "set_boot_disk", **data).read() |
| # TODO validation for error response (disk does not exists and node does not exists) |
| result["new"] = "Disk {0} was set as bootable".format(name) |
| |
| return result |
| |
| # NETWORKING |
| |
| |
| def list_fabric(): |
| """ |
| Get list of all fabric |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'maas-node' maasng.list_fabric |
| """ |
| fabrics = {} |
| maas = _create_maas_client() |
| json_res = json.loads(maas.get(u'api/2.0/fabrics/').read()) |
| LOG.info(json_res) |
| for item in json_res: |
| fabrics[item["name"]] = item |
| return fabrics |
| |
| |
| def create_fabric(name): |
| """ |
| Create new fabric. |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'maas-node' maasng.create_fabric |
| """ |
| result = {} |
| data = { |
| "name": name, |
| "description": '', |
| "class_type": '', |
| |
| } |
| |
| maas = _create_maas_client() |
| json_res = json.loads(maas.post(u"api/2.0/fabrics/", None, **data).read()) |
| LOG.info(json_res) |
| result["new"] = "Fabrics {0} created".format(json_res["name"]) |
| return result |
| |
| |
| def list_subnet(): |
| """ |
| Get list of all subnets |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'maas-node' maasng.list_subnet |
| """ |
| subnets = {} |
| maas = _create_maas_client() |
| json_res = json.loads(maas.get(u'api/2.0/subnets/').read()) |
| LOG.info(json_res) |
| for item in json_res: |
| subnets[item["name"]] = item |
| return subnets |
| |
| |
| def list_vlans(fabric): |
| """ |
| Get list of all vlans for specific fabric |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'maas-node' maasng.list_vlans |
| """ |
| vlans = {} |
| maas = _create_maas_client() |
| fabric_id = get_fabric(fabric) |
| |
| json_res = json.loads( |
| maas.get(u'api/2.0/fabrics/{0}/vlans/'.format(fabric_id)).read()) |
| LOG.info(json_res) |
| for item in json_res: |
| vlans[item["name"]] = item |
| return vlans |
| |
| |
| def get_fabric(fabric): |
| """ |
| Get id for specific fabric |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt-call maasng.get_fabric fabric_name |
| """ |
| try: |
| return list_fabric()[fabric]['id'] |
| except KeyError: |
| return {"error": "Frabic not found on MaaS server"} |
| |
| |
| def update_vlan(name, fabric, vid, description, primary_rack, dhcp_on=False): |
| """ |
| Update vlan |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'maas-node' maasng.update_vlan name, fabric, vid, description, dhcp_on |
| """ |
| result = {} |
| |
| data = { |
| "name": name, |
| "dhcp_on": str(dhcp_on), |
| "description": description, |
| "primary_rack": primary_rack, |
| } |
| maas = _create_maas_client() |
| fabric_id = get_fabric(fabric) |
| |
| json_res = json.loads(maas.put( |
| u'api/2.0/fabrics/{0}/vlans/{1}/'.format(fabric_id, vid), **data).read()) |
| LOG.debug("update_vlan:{}".format(json_res)) |
| result["new"] = "Vlan {0} was updated".format(json_res["name"]) |
| |
| return result |
| |
| # END NETWORKING |
| |
| # MAAS CONFIG SECTION |
| |
| |
| def _get_boot_source_id_by_url(url): |
| # FIXME: fix ret\validation |
| try: |
| bs_id = get_boot_source(url=url)["id"] |
| except KeyError: |
| return {"error": "boot-source:{0} not exist!".format(url)} |
| return bs_id |
| |
| |
| def get_boot_source(url=None): |
| """ |
| Read a boot source by url. If url not specified - return all. |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'maas-node' maasng.get_boot_source url |
| |
| """ |
| boot_sources = {} |
| maas = _create_maas_client() |
| json_res = json.loads(maas.get(u'api/2.0/boot-sources/').read() or 'null') |
| for item in json_res: |
| boot_sources[str(item["url"])] = item |
| if url: |
| return boot_sources.get(url, {}) |
| return boot_sources |
| |
| |
| def delete_boot_source(url, bs_id=None): |
| """ |
| Delete a boot source by url. |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| sal 'maas-node' maasng.delete url |
| |
| """ |
| result = {} |
| if not bs_id: |
| bs_id = _get_boot_source_id_by_url(url) |
| maas = _create_maas_client() |
| json_res = json.loads(maas.delete( |
| u'/api/2.0/boot-sources/{0}/'.format(bs_id)).read() or 'null') |
| LOG.debug("delete_boot_source:{}".format(json_res)) |
| result["new"] = "Boot-resource {0} deleted".format(url) |
| return result |
| |
| |
| def boot_sources_delete_all_others(except_urls=[]): |
| """ |
| Delete all boot-sources, except defined in 'except_urls' list. |
| """ |
| result = {} |
| maas_boot_sources = get_boot_source() |
| if 0 in [len(except_urls), len(maas_boot_sources)]: |
| result['result'] = None |
| result[ |
| "comment"] = "Exclude or maas sources for delete empty. No changes goinng to be." |
| return result |
| for url in maas_boot_sources.keys(): |
| if url not in except_urls: |
| LOG.info("Removing boot-source:{}".format(url)) |
| boot_resources_import(action='stop_import', wait=True) |
| result["changes"] = delete_boot_source(url) |
| return result |
| |
| |
| def create_boot_source(url, keyring_filename='', keyring_data='', wait=False): |
| """ |
| Create and import maas boot-source: link to maas-ephemeral repo |
| Be aware, those step will import resource to rack ctrl, but you also need to import |
| them into the region! |
| |
| |
| :param url: The URL of the BootSource. |
| :param keyring_filename: The path to the keyring file for this BootSource. |
| :param keyring_data: The GPG keyring for this BootSource, base64-encoded data. |
| |
| """ |
| |
| # TODO: not work with 'update' currently => keyring update may fail. |
| result = {} |
| |
| data = { |
| "url": url, |
| "keyring_filename": keyring_filename, |
| "keyring_data": str(keyring_data), |
| } |
| |
| maas = _create_maas_client() |
| if url in get_boot_source(): |
| result['result'] = None |
| result["comment"] = "boot resource already exist" |
| return result |
| |
| # NOTE: maas.post will return 400, if url already defined. |
| json_res = json.loads( |
| maas.post(u'api/2.0/boot-sources/', None, **data).read()) |
| if wait: |
| LOG.debug( |
| "Sleep for 5s,to get MaaS some time to process previous request") |
| time.sleep(5) |
| ret = boot_resources_is_importing(wait=True) |
| if ret is dict: |
| return ret |
| LOG.debug("create_boot_source:{}".format(json_res)) |
| result["new"] = "boot resource {0} was created".format(json_res["url"]) |
| |
| return result |
| |
| |
| def boot_resources_import(action='import', wait=False): |
| """ |
| import/stop_import the boot resources. |
| |
| :param action: import\stop_import |
| :param wait: True\False. Wait till process finished. |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'maas-node' maasng.boot_resources_import action='import' |
| |
| """ |
| maas = _create_maas_client() |
| # Have no idea why, but usual jsonloads not work here.. |
| imp = maas.post(u'api/2.0/boot-resources/', action) |
| if imp.code == 200: |
| LOG.debug('boot_resources_import:{}'.format(imp.readline())) |
| if wait: |
| boot_resources_is_importing(wait=True) |
| return True |
| else: |
| return False |
| |
| |
| def boot_resources_is_importing(wait=False): |
| maas = _create_maas_client() |
| result = {} |
| if wait: |
| started_at = time.time() |
| poll_time = 5 |
| timeout = 60 * 15 |
| while boot_resources_is_importing(wait=False): |
| c_timeout = timeout - (time.time() - started_at) |
| if c_timeout <= 0: |
| result['result'] = False |
| result["comment"] = "Boot-resources import not finished in time" |
| return result |
| LOG.info( |
| "Waiting boot-resources import done\n" |
| "sleep for:{}s " |
| "Left:{}/{}s".format(poll_time, round(c_timeout), timeout)) |
| time.sleep(poll_time) |
| return json.loads( |
| maas.get(u'api/2.0/boot-resources/', 'is_importing').read()) |
| else: |
| return json.loads( |
| maas.get(u'api/2.0/boot-resources/', 'is_importing').read()) |
| |
| ##### |
| #def boot_sources_selections_delete_all_others(except_urls=[]): |
| # """ |
| # """ |
| # result = {} |
| # return result |
| |
| |
| def is_boot_source_selections_in(dict1, list1): |
| """ |
| Check that requested boot-selection already in maas bs selections, if True- return bss id. |
| # FIXME: those hack check doesn't look good. |
| """ |
| for bs in list1: |
| same = set(dict1.keys()) & set(bs.keys()) |
| if all(elem in same for elem in |
| ['os', 'release', 'arches', 'subarches', 'labels']): |
| LOG.debug( |
| "boot-selection in maas:{0}\nlooks same to requested:{1}".format( |
| bs, dict1)) |
| return bs['id'] |
| return False |
| |
| |
| def get_boot_source_selections(bs_url): |
| """ |
| Get boot-source selections. |
| """ |
| # check for key_error! |
| bs_id = _get_boot_source_id_by_url(bs_url) |
| maas = _create_maas_client() |
| json_res = json.loads( |
| maas.get(u'/api/2.0/boot-sources/{0}/selections/'.format(bs_id)).read()) |
| LOG.debug( |
| "get_boot_source_selections for url:{} \n{}".format(bs_url, json_res)) |
| return json_res |
| |
| |
| def create_boot_source_selections(bs_url, os, release, arches="*", |
| subarches="*", labels="*", wait=True): |
| """ |
| Create a new boot source selection for bs_url. |
| :param os: The OS (e.g. ubuntu, centos) for which to import resources.Required. |
| :param release: The release for which to import resources. Required. |
| :param arches: The architecture list for which to import resources. |
| :param subarches: The subarchitecture list for which to import resources. |
| :param labels: The label lists for which to import resources. |
| """ |
| |
| result = { "result" : True, 'name' : bs_url, 'changes' : None } |
| |
| data = { |
| "os": os, |
| "release": release, |
| "arches": arches, |
| "subarches": subarches, |
| "labels": labels, |
| } |
| |
| maas = _create_maas_client() |
| bs_id = _get_boot_source_id_by_url(bs_url) |
| # TODO add pre-create verify |
| maas_bs_s = get_boot_source_selections(bs_url) |
| if is_boot_source_selections_in(data, maas_bs_s): |
| result["result"] = True |
| result[ |
| "comment"] = 'Requested boot-source selection for {0} already exist.'.format( |
| bs_url) |
| return result |
| |
| # NOTE: maas.post will return 400, if url already defined. |
| # Also, maas need's some time to import info about stream. |
| # unfortunatly, maas don't have any call to check stream-import-info - so, we need to implement |
| # at least simple retry ;( |
| json_res = False |
| poll_time = 5 |
| for i in range(0,5): |
| try: |
| json_res = json.loads( |
| maas.post(u'api/2.0/boot-sources/{0}/selections/'.format(bs_id), None, |
| **data).read()) |
| except Exception as inst: |
| m = inst.readlines() |
| LOG.warning("boot_source_selections catch error during processing. Most-probably, streams not imported yet.Sleep:{}s\nRetry:{}/5".format(poll_time,i)) |
| LOG.warning("Message:{0}".format(m)) |
| time.sleep(poll_time) |
| continue |
| break |
| LOG.debug("create_boot_source_selections:{}".format(json_res)) |
| if not json_res: |
| result["result"] = False |
| result[ |
| "comment"] = 'Failed to create requested boot-source selection for {0}.'.format(bs_url) |
| return result |
| if wait: |
| LOG.debug( |
| "Sleep for 5s,to get MaaS some time to process previous request") |
| time.sleep(5) |
| ret = boot_resources_import(action='import', wait=True) |
| if ret is dict: |
| return ret |
| result["comment"] = "boot-source selection for {0} was created".format(bs_url) |
| result["new"] = data |
| |
| return result |
| |
| # END MAAS CONFIG SECTION |
| |
| # RACK CONTROLLERS SECTION |
| |
| |
| def get_rack(hostname): |
| """ |
| Get information about specified rackd |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt-call maasng.get_rack rack_hostname |
| """ |
| try: |
| return list_racks()[hostname] |
| except KeyError: |
| return {"error": "rack:{} not found on MaaS server".format(hostname)} |
| |
| |
| def list_racks(): |
| """ |
| Get list of all rack controllers from maas server |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt-call maasng.list_racks |
| """ |
| racks = {} |
| maas = _create_maas_client() |
| json_res = json.loads( |
| maas.get(u"/api/2.0/rackcontrollers/").read() or 'null') |
| for item in json_res: |
| racks[item["hostname"]] = item |
| return racks |
| |
| |
| def sync_bs_to_rack(hostname=None): |
| """ |
| Sync RACK boot-sources with REGION. If no hostname probided - sync to all. |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt-call maasng.sync_bs_to_rack rack_hostname |
| """ |
| ret = {} |
| maas = _create_maas_client() |
| if not hostname: |
| LOG.info("boot-sources sync initiated for ALL Rack's") |
| # Convert to json-like format |
| json_res = json.loads('["{0}"]'.format( |
| maas.post(u"/api/2.0/rackcontrollers/", |
| 'import_boot_images').read())) |
| LOG.debug("sync_bs_to_rack:{}".format(json_res)) |
| ret['result'] = True |
| ret['comment'] = "boot-sources sync initiated for ALL Rack's" |
| return ret |
| LOG.info("boot-sources sync initiated for RACK:{0}".format(hostname)) |
| # Convert to json-like format |
| json_res = json.loads('["{0}"]'.format(maas.post( |
| u"/api/2.0/rackcontrollers/{0}/".format( |
| get_rack(hostname)['system_id']), |
| 'import_boot_images').read())) |
| LOG.debug("sync_bs_to_rack:{}".format(json_res)) |
| ret['result'] = True |
| ret['comment'] = "boot-sources sync initiated for {0} Rack's".format( |
| hostname) |
| return |
| |
| |
| def rack_list_boot_imgs(hostname): |
| ret = {} |
| maas = _create_maas_client() |
| LOG.debug("rack_list_boot_imgs:{}".format(hostname)) |
| ret = json.loads(maas.get(u"/api/2.0/rackcontrollers/{0}/".format( |
| get_rack(hostname)['system_id']), 'list_boot_images').read() or 'null') |
| return ret |
| |
| |
| def is_rack_synced(hostname): |
| rez = rack_list_boot_imgs(hostname)['status'] |
| if rez == 'synced': |
| return True |
| return False |
| |
| # TODO do we actually need _exact_ check per-pack? |
| # def wait_for_images_on_rack(hostname): |
| # |
| # """ |
| # WA function, to be able check that RACK actually done SYNC images |
| # for REQUIRED images at least. |
| # Required image to be fetched from |
| # reclass:maas:region:boot_sources_selections:[keys]:os/release formation |
| # |
| # CLI Example: |
| # |
| # .. code-block:: bash |
| # |
| # salt-call maasng.wait_for_sync_bs_to_rack rack_hostname |
| # """ |
| # try: |
| # bss = __salt__['config.get']('maas')['region']['boot_sources_selections'] |
| # except KeyError: |
| # ret['result'] = None |
| # ret['comment'] = "boot_sources_selections definition for sync not found." |
| # return ret |
| # s_names = [] |
| # # Format u'name': u'ubuntu/xenial' |
| # for v in bss.values():s_names.append("{0}/{1}".format(v['os'],v['release'])) |
| # # Each names, should be in rack and whole rack should be in sync-ed state |
| |
| |
| def sync_and_wait_bs_to_all_racks(): |
| """ |
| Sync ALL rack's with regions source images. |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt-call maasng.sync_and_wait_bs_to_all_racks |
| """ |
| sync_bs_to_rack() |
| for rack in list_racks().keys(): |
| wait_for_sync_bs_to_rack(hostname=rack) |
| return True |
| |
| |
| def wait_for_sync_bs_to_rack(hostname=None): |
| """ |
| Wait for boot images sync finished, on exact rack |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt-call maasng.wait_for_sync_bs_to_rack rack_hostname |
| """ |
| ret = {} |
| started_at = time.time() |
| poll_time = 5 |
| timeout = 60 * 15 |
| while not is_rack_synced(hostname): |
| c_timeout = timeout - (time.time() - started_at) |
| if c_timeout <= 0: |
| ret['result'] = False |
| ret[ |
| "comment"] = "Boot-resources sync on rackd:{0}" \ |
| "not finished in time".format( |
| hostname) |
| return ret |
| LOG.info( |
| "Waiting boot-resources sync done to rack:{0}\n" |
| "sleep for:{1}s " |
| "Left:{2}/{3}s".format(hostname, poll_time, round(c_timeout), |
| timeout)) |
| time.sleep(poll_time) |
| ret['result'] = is_rack_synced(hostname) |
| ret["comment"] = "Boot-resources sync on rackd:{0} finished".format( |
| hostname) |
| return ret |
| |
| # END RACK CONTROLLERS SECTION |