| # -*- 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()) |
| print(json_res) |
| result["new"] = "Vlan {0} was updated".format(json_res["name"]) |
| |
| return result |
| |
| # END NETWORKING |