Dre4m Shell
Server IP : 85.214.239.14  /  Your IP : 3.141.202.161
Web Server : Apache/2.4.62 (Debian)
System : Linux h2886529.stratoserver.net 4.9.0 #1 SMP Tue Jan 9 19:45:01 MSK 2024 x86_64
User : www-data ( 33)
PHP Version : 7.4.18
Disable Function : pcntl_alarm,pcntl_fork,pcntl_waitpid,pcntl_wait,pcntl_wifexited,pcntl_wifstopped,pcntl_wifsignaled,pcntl_wifcontinued,pcntl_wexitstatus,pcntl_wtermsig,pcntl_wstopsig,pcntl_signal,pcntl_signal_get_handler,pcntl_signal_dispatch,pcntl_get_last_error,pcntl_strerror,pcntl_sigprocmask,pcntl_sigwaitinfo,pcntl_sigtimedwait,pcntl_exec,pcntl_getpriority,pcntl_setpriority,pcntl_async_signals,pcntl_unshare,
MySQL : OFF  |  cURL : OFF  |  WGET : ON  |  Perl : ON  |  Python : ON  |  Sudo : ON  |  Pkexec : OFF
Directory :  /lib/python3/dist-packages/ansible_collections/purestorage/fusion/plugins/modules/

Upload File :
current_dir [ Writeable ] document_root [ Writeable ]

 

Command :


[ HOME SHELL ]     

Current File : /lib/python3/dist-packages/ansible_collections/purestorage/fusion/plugins/modules//fusion_info.py
#!/usr/bin/python
# -*- coding: utf-8 -*-

# (c) 2021, Simon Dodsley (simon@purestorage.com), Andrej Pajtas (apajtas@purestorage.com)
# GNU General Public License v3.0+ (see COPYING.GPLv3 or https://www.gnu.org/licenses/gpl-3.0.txt)

from __future__ import absolute_import, division, print_function

__metaclass__ = type

DOCUMENTATION = r"""
---
module: fusion_info
version_added: '1.0.0'
short_description: Collect information from Pure Fusion
description:
  - Collect information from a Pure Fusion environment.
  - By default, the module will collect basic
    information including counts for arrays, availability_zones, volumes, snapshots
    . Fleet capacity and data reduction rates are also provided.
  - Additional information can be collected based on the configured set of arguments.
author:
  - Pure Storage ansible Team (@sdodsley) <pure-ansible-team@purestorage.com>
notes:
- Supports C(check mode).
options:
  gather_subset:
    description:
      - When supplied, this argument will define the information to be collected.
        Possible values for this include all, minimum, roles, users, arrays, hardware_types,
        volumes, host_access_policies, storage_classes, protection_policies, placement_groups,
        network_interfaces, availability_zones, network_interface_groups, storage_endpoints,
        snapshots, regions, storage_services, tenants, tenant_spaces, network_interface_groups and api_clients.
    type: list
    elements: str
    required: false
    default: minimum
extends_documentation_fragment:
  - purestorage.fusion.purestorage.fusion
"""

EXAMPLES = r"""
- name: Collect default set of information
  purestorage.fusion.fusion_info:
    issuer_id: key_name
    private_key_file: "az-admin-private-key.pem"
    register: fusion_info

- name: Show default information
  ansible.builtin.debug:
    msg: "{{ fusion_info['fusion_info']['default'] }}"

- name: Collect all information
  purestorage.fusion.fusion_info:
    gather_subset:
      - all
    issuer_id: key_name
    private_key_file: "az-admin-private-key.pem"

- name: Show all information
  ansible.builtin.debug:
    msg: "{{ fusion_info['fusion_info'] }}"
"""

RETURN = r"""
fusion_info:
  description: Returns the information collected from Fusion
  returned: always
  type: dict
"""

try:
    import fusion as purefusion
except ImportError:
    pass

from ansible.module_utils.basic import AnsibleModule
from ansible_collections.purestorage.fusion.plugins.module_utils.fusion import (
    fusion_argument_spec,
)
from ansible_collections.purestorage.fusion.plugins.module_utils.startup import (
    setup_fusion,
)
import time
import http


def _convert_microseconds(micros):
    seconds = (micros / 1000) % 60
    minutes = (micros / (1000 * 60)) % 60
    hours = (micros / (1000 * 60 * 60)) % 24
    return seconds, minutes, hours


def _api_permission_denied_handler(name):
    """Return decorator which catches #403 errors"""

    def inner(func):
        def wrapper(module, fusion, *args, **kwargs):
            try:
                return func(module, fusion, *args, **kwargs)
            except purefusion.rest.ApiException as exc:
                if exc.status == http.HTTPStatus.FORBIDDEN:
                    module.warn(f"Cannot get [{name} dict], reason: Permission denied")
                    return None
                else:
                    # other exceptions will be handled by our exception hook
                    raise exc

        return wrapper

    return inner


def generate_default_dict(module, fusion):
    def warning_api_exception(name):
        module.warn(f"Cannot get {name} in [default dict], reason: Permission denied")

    def warning_argument_none(name, requirement):
        module.warn(
            f"Cannot get {name} in [default dict], reason: Required argument `{requirement}` not available."
        )

    # All values are independent on each other - if getting one value fails, we will show warning and continue.
    # That's also the reason why there's so many nested for loops repeating all over again.
    version = None
    users_num = None
    protection_policies_num = None
    host_access_policies_num = None
    hardware_types_num = None
    storage_services = None
    storage_services_num = None
    tenants = None
    tenants_num = None
    regions = None
    regions_num = None
    roles = None
    roles_num = None
    storage_classes_num = None
    role_assignments_num = None
    tenant_spaces_num = None
    volumes_num = None
    placement_groups_num = None
    snapshots_num = None
    availability_zones_num = None
    arrays_num = None
    network_interfaces_num = None
    network_interface_groups_num = None
    storage_endpoints_num = None

    try:
        version = purefusion.DefaultApi(fusion).get_version().version
    except purefusion.rest.ApiException as exc:
        if exc.status == http.HTTPStatus.FORBIDDEN:
            warning_api_exception("API version")
        else:
            # other exceptions will be handled by our exception hook
            raise exc

    try:
        users_num = len(purefusion.IdentityManagerApi(fusion).list_users())
    except purefusion.rest.ApiException as exc:
        if exc.status == http.HTTPStatus.FORBIDDEN:
            warning_api_exception("Users")
        else:
            # other exceptions will be handled by our exception hook
            raise exc

    try:
        protection_policies_num = len(
            purefusion.ProtectionPoliciesApi(fusion).list_protection_policies().items
        )
    except purefusion.rest.ApiException as exc:
        if exc.status == http.HTTPStatus.FORBIDDEN:
            warning_api_exception("Protection Policies")
        else:
            # other exceptions will be handled by our exception hook
            raise exc

    try:
        host_access_policies_num = len(
            purefusion.HostAccessPoliciesApi(fusion).list_host_access_policies().items
        )
    except purefusion.rest.ApiException as exc:
        if exc.status == http.HTTPStatus.FORBIDDEN:
            warning_api_exception("Host Access Policies")
        else:
            # other exceptions will be handled by our exception hook
            raise exc

    try:
        hardware_types_num = len(
            purefusion.HardwareTypesApi(fusion).list_hardware_types().items
        )
    except purefusion.rest.ApiException as exc:
        if exc.status == http.HTTPStatus.FORBIDDEN:
            warning_api_exception("Hardware Types")
        else:
            # other exceptions will be handled by our exception hook
            raise exc

    try:
        storage_services = purefusion.StorageServicesApi(fusion).list_storage_services()
        storage_services_num = len(storage_services.items)
    except purefusion.rest.ApiException as exc:
        if exc.status == http.HTTPStatus.FORBIDDEN:
            warning_api_exception("Storage Services")
        else:
            # other exceptions will be handled by our exception hook
            raise exc

    try:
        tenants = purefusion.TenantsApi(fusion).list_tenants()
        tenants_num = len(tenants.items)
    except purefusion.rest.ApiException as exc:
        if exc.status == http.HTTPStatus.FORBIDDEN:
            warning_api_exception("Tenants")
        else:
            # other exceptions will be handled by our exception hook
            raise exc

    try:
        regions = purefusion.RegionsApi(fusion).list_regions()
        regions_num = len(regions.items)
    except purefusion.rest.ApiException as exc:
        if exc.status == http.HTTPStatus.FORBIDDEN:
            warning_api_exception("Regions")
        else:
            # other exceptions will be handled by our exception hook
            raise exc

    try:
        roles = purefusion.RolesApi(fusion).list_roles()
        roles_num = len(roles)
    except purefusion.rest.ApiException as exc:
        if exc.status == http.HTTPStatus.FORBIDDEN:
            warning_api_exception("Roles")
        else:
            # other exceptions will be handled by our exception hook
            raise exc

    if storage_services is not None:
        try:
            storage_class_api_instance = purefusion.StorageClassesApi(fusion)
            storage_classes_num = sum(
                len(
                    storage_class_api_instance.list_storage_classes(
                        storage_service_name=storage_service.name
                    ).items
                )
                for storage_service in storage_services.items
            )
        except purefusion.rest.ApiException as exc:
            if exc.status == http.HTTPStatus.FORBIDDEN:
                warning_api_exception("Storage Classes")
            else:
                # other exceptions will be handled by our exception hook
                raise exc
    else:
        warning_argument_none("Storage Classes", "storage_services")

    if roles is not None:
        try:
            role_assign_api_instance = purefusion.RoleAssignmentsApi(fusion)
            role_assignments_num = sum(
                len(role_assign_api_instance.list_role_assignments(role_name=role.name))
                for role in roles
            )
        except purefusion.rest.ApiException as exc:
            if exc.status == http.HTTPStatus.FORBIDDEN:
                warning_api_exception("Role Assignments")
            else:
                # other exceptions will be handled by our exception hook
                raise exc
    else:
        warning_argument_none("Role Assignments", "roles")

    if tenants is not None:
        tenantspace_api_instance = purefusion.TenantSpacesApi(fusion)

        try:
            tenant_spaces_num = sum(
                len(
                    tenantspace_api_instance.list_tenant_spaces(
                        tenant_name=tenant.name
                    ).items
                )
                for tenant in tenants.items
            )
        except purefusion.rest.ApiException as exc:
            if exc.status == http.HTTPStatus.FORBIDDEN:
                warning_api_exception("Tenant Spaces")
            else:
                # other exceptions will be handled by our exception hook
                raise exc

        try:
            vol_api_instance = purefusion.VolumesApi(fusion)
            volumes_num = sum(
                len(
                    vol_api_instance.list_volumes(
                        tenant_name=tenant.name,
                        tenant_space_name=tenant_space.name,
                    ).items
                )
                for tenant in tenants.items
                for tenant_space in tenantspace_api_instance.list_tenant_spaces(
                    tenant_name=tenant.name
                ).items
            )
        except purefusion.rest.ApiException as exc:
            if exc.status == http.HTTPStatus.FORBIDDEN:
                warning_api_exception("Volumes")
            else:
                # other exceptions will be handled by our exception hook
                raise exc

        try:
            plgrp_api_instance = purefusion.PlacementGroupsApi(fusion)
            placement_groups_num = sum(
                len(
                    plgrp_api_instance.list_placement_groups(
                        tenant_name=tenant.name,
                        tenant_space_name=tenant_space.name,
                    ).items
                )
                for tenant in tenants.items
                for tenant_space in tenantspace_api_instance.list_tenant_spaces(
                    tenant_name=tenant.name
                ).items
            )
        except purefusion.rest.ApiException as exc:
            if exc.status == http.HTTPStatus.FORBIDDEN:
                warning_api_exception("Placement Groups")
            else:
                # other exceptions will be handled by our exception hook
                raise exc

        try:
            snapshot_api_instance = purefusion.SnapshotsApi(fusion)
            snapshots_num = sum(
                len(
                    snapshot_api_instance.list_snapshots(
                        tenant_name=tenant.name,
                        tenant_space_name=tenant_space.name,
                    ).items
                )
                for tenant in tenants.items
                for tenant_space in tenantspace_api_instance.list_tenant_spaces(
                    tenant_name=tenant.name
                ).items
            )
        except purefusion.rest.ApiException as exc:
            if exc.status == http.HTTPStatus.FORBIDDEN:
                warning_api_exception("Snapshots")
            else:
                # other exceptions will be handled by our exception hook
                raise exc
    else:
        warning_argument_none("Tenant Spaces", "tenants")
        warning_argument_none("Volumes", "tenants")
        warning_argument_none("Placement Groups", "tenants")
        warning_argument_none("Snapshots", "tenants")

    if regions is not None:
        az_api_instance = purefusion.AvailabilityZonesApi(fusion)

        try:
            availability_zones_num = sum(
                len(
                    az_api_instance.list_availability_zones(
                        region_name=region.name
                    ).items
                )
                for region in regions.items
            )
        except purefusion.rest.ApiException as exc:
            if exc.status == http.HTTPStatus.FORBIDDEN:
                warning_api_exception("Availability Zones")
            else:
                # other exceptions will be handled by our exception hook
                raise exc

        try:
            arrays_api_instance = purefusion.ArraysApi(fusion)
            arrays_num = sum(
                len(
                    arrays_api_instance.list_arrays(
                        availability_zone_name=availability_zone.name,
                        region_name=region.name,
                    ).items
                )
                for region in regions.items
                for availability_zone in az_api_instance.list_availability_zones(
                    region_name=region.name
                ).items
            )
        except purefusion.rest.ApiException as exc:
            if exc.status == http.HTTPStatus.FORBIDDEN:
                warning_api_exception("Arrays")
            else:
                # other exceptions will be handled by our exception hook
                raise exc

        try:
            nig_api_instance = purefusion.NetworkInterfaceGroupsApi(fusion)
            network_interface_groups_num = sum(
                len(
                    nig_api_instance.list_network_interface_groups(
                        availability_zone_name=availability_zone.name,
                        region_name=region.name,
                    ).items
                )
                for region in regions.items
                for availability_zone in az_api_instance.list_availability_zones(
                    region_name=region.name
                ).items
            )
        except purefusion.rest.ApiException as exc:
            if exc.status == http.HTTPStatus.FORBIDDEN:
                warning_api_exception("Network Interface Groups")
            else:
                # other exceptions will be handled by our exception hook
                raise exc

        try:
            send_api_instance = purefusion.StorageEndpointsApi(fusion)
            storage_endpoints_num = sum(
                len(
                    send_api_instance.list_storage_endpoints(
                        availability_zone_name=availability_zone.name,
                        region_name=region.name,
                    ).items
                )
                for region in regions.items
                for availability_zone in az_api_instance.list_availability_zones(
                    region_name=region.name
                ).items
            )
        except purefusion.rest.ApiException as exc:
            if exc.status == http.HTTPStatus.FORBIDDEN:
                warning_api_exception("Storage Endpoints")
            else:
                # other exceptions will be handled by our exception hook
                raise exc

        try:
            nic_api_instance = purefusion.NetworkInterfacesApi(fusion)
            network_interfaces_num = sum(
                len(
                    nic_api_instance.list_network_interfaces(
                        availability_zone_name=availability_zone.name,
                        region_name=region.name,
                        array_name=array_detail.name,
                    ).items
                )
                for region in regions.items
                for availability_zone in az_api_instance.list_availability_zones(
                    region_name=region.name
                ).items
                for array_detail in arrays_api_instance.list_arrays(
                    availability_zone_name=availability_zone.name,
                    region_name=region.name,
                ).items
            )
        except purefusion.rest.ApiException as exc:
            if exc.status == http.HTTPStatus.FORBIDDEN:
                warning_api_exception("Network Interfaces")
            else:
                # other exceptions will be handled by our exception hook
                raise exc
    else:
        warning_argument_none("Availability Zones", "regions")
        warning_argument_none("Network Interfaces", "regions")
        warning_argument_none("Network Interface Groups", "regions")
        warning_argument_none("Storage Endpoints", "regions")
        warning_argument_none("Arrays", "regions")

    return {
        "version": version,
        "users": users_num,
        "protection_policies": protection_policies_num,
        "host_access_policies": host_access_policies_num,
        "hardware_types": hardware_types_num,
        "storage_services": storage_services_num,
        "tenants": tenants_num,
        "regions": regions_num,
        "storage_classes": storage_classes_num,
        "roles": roles_num,
        "role_assignments": role_assignments_num,
        "tenant_spaces": tenant_spaces_num,
        "volumes": volumes_num,
        "placement_groups": placement_groups_num,
        "snapshots": snapshots_num,
        "availability_zones": availability_zones_num,
        "arrays": arrays_num,
        "network_interfaces": network_interfaces_num,
        "network_interface_groups": network_interface_groups_num,
        "storage_endpoints": storage_endpoints_num,
    }


@_api_permission_denied_handler("network_interfaces")
def generate_nics_dict(module, fusion):
    nics_info = {}
    nic_api_instance = purefusion.NetworkInterfacesApi(fusion)
    arrays_api_instance = purefusion.ArraysApi(fusion)
    az_api_instance = purefusion.AvailabilityZonesApi(fusion)
    regions_api_instance = purefusion.RegionsApi(fusion)
    regions = regions_api_instance.list_regions()
    for region in regions.items:
        azs = az_api_instance.list_availability_zones(region_name=region.name)
        for az in azs.items:
            array_details = arrays_api_instance.list_arrays(
                availability_zone_name=az.name,
                region_name=region.name,
            )
            for array_detail in array_details.items:
                array_name = az.name + "/" + array_detail.name
                nics_info[array_name] = {}
                nics = nic_api_instance.list_network_interfaces(
                    availability_zone_name=az.name,
                    region_name=region.name,
                    array_name=array_detail.name,
                )

                for nic in nics.items:
                    nics_info[array_name][nic.name] = {
                        "enabled": nic.enabled,
                        "display_name": nic.display_name,
                        "interface_type": nic.interface_type,
                        "services": nic.services,
                        "max_speed": nic.max_speed,
                        "vlan": nic.eth.vlan,
                        "address": nic.eth.address,
                        "mac_address": nic.eth.mac_address,
                        "gateway": nic.eth.gateway,
                        "mtu": nic.eth.mtu,
                        "network_interface_group": nic.network_interface_group.name,
                        "availability_zone": nic.availability_zone.name,
                    }
    return nics_info


@_api_permission_denied_handler("host_access_policies")
def generate_hap_dict(module, fusion):
    hap_info = {}
    api_instance = purefusion.HostAccessPoliciesApi(fusion)
    hosts = api_instance.list_host_access_policies()
    for host in hosts.items:
        name = host.name
        hap_info[name] = {
            "personality": host.personality,
            "display_name": host.display_name,
            "iqn": host.iqn,
        }
    return hap_info


@_api_permission_denied_handler("arrays")
def generate_array_dict(module, fusion):
    array_info = {}
    array_api_instance = purefusion.ArraysApi(fusion)
    az_api_instance = purefusion.AvailabilityZonesApi(fusion)
    regions_api_instance = purefusion.RegionsApi(fusion)
    regions = regions_api_instance.list_regions()
    for region in regions.items:
        azs = az_api_instance.list_availability_zones(region_name=region.name)
        for az in azs.items:
            arrays = array_api_instance.list_arrays(
                availability_zone_name=az.name,
                region_name=region.name,
            )
            for array in arrays.items:
                array_name = array.name
                array_space = array_api_instance.get_array_space(
                    availability_zone_name=az.name,
                    array_name=array_name,
                    region_name=region.name,
                )
                array_perf = array_api_instance.get_array_performance(
                    availability_zone_name=az.name,
                    array_name=array_name,
                    region_name=region.name,
                )
                array_info[array_name] = {
                    "region": region.name,
                    "availability_zone": az.name,
                    "host_name": array.host_name,
                    "maintenance_mode": array.maintenance_mode,
                    "unavailable_mode": array.unavailable_mode,
                    "display_name": array.display_name,
                    "hardware_type": array.hardware_type.name,
                    "appliance_id": array.appliance_id,
                    "apartment_id": getattr(array, "apartment_id", None),
                    "space": {
                        "total_physical_space": array_space.total_physical_space,
                    },
                    "performance": {
                        "read_bandwidth": array_perf.read_bandwidth,
                        "read_latency_us": array_perf.read_latency_us,
                        "reads_per_sec": array_perf.reads_per_sec,
                        "write_bandwidth": array_perf.write_bandwidth,
                        "write_latency_us": array_perf.write_latency_us,
                        "writes_per_sec": array_perf.writes_per_sec,
                    },
                }
    return array_info


@_api_permission_denied_handler("placement_groups")
def generate_pg_dict(module, fusion):
    pg_info = {}
    tenant_api_instance = purefusion.TenantsApi(fusion)
    tenantspace_api_instance = purefusion.TenantSpacesApi(fusion)
    pg_api_instance = purefusion.PlacementGroupsApi(fusion)
    tenants = tenant_api_instance.list_tenants()
    for tenant in tenants.items:
        tenant_spaces = tenantspace_api_instance.list_tenant_spaces(
            tenant_name=tenant.name
        ).items
        for tenant_space in tenant_spaces:
            groups = pg_api_instance.list_placement_groups(
                tenant_name=tenant.name,
                tenant_space_name=tenant_space.name,
            )
            for group in groups.items:
                group_name = tenant.name + "/" + tenant_space.name + "/" + group.name
                pg_info[group_name] = {
                    "tenant": group.tenant.name,
                    "display_name": group.display_name,
                    "placement_engine": group.placement_engine,
                    "tenant_space": group.tenant_space.name,
                    "az": group.availability_zone.name,
                    "array": getattr(group.array, "name", None),
                }
    return pg_info


@_api_permission_denied_handler("tenant_spaces")
def generate_ts_dict(module, fusion):
    ts_info = {}
    tenant_api_instance = purefusion.TenantsApi(fusion)
    tenantspace_api_instance = purefusion.TenantSpacesApi(fusion)
    tenants = tenant_api_instance.list_tenants()
    for tenant in tenants.items:
        tenant_spaces = tenantspace_api_instance.list_tenant_spaces(
            tenant_name=tenant.name
        ).items
        for tenant_space in tenant_spaces:
            ts_name = tenant.name + "/" + tenant_space.name
            ts_info[ts_name] = {
                "tenant": tenant.name,
                "display_name": tenant_space.display_name,
            }
    return ts_info


@_api_permission_denied_handler("protection_policies")
def generate_pp_dict(module, fusion):
    pp_info = {}
    api_instance = purefusion.ProtectionPoliciesApi(fusion)
    policies = api_instance.list_protection_policies()
    for policy in policies.items:
        policy_name = policy.name
        pp_info[policy_name] = {
            "objectives": policy.objectives,
        }
    return pp_info


@_api_permission_denied_handler("tenants")
def generate_tenant_dict(module, fusion):
    tenants_api_instance = purefusion.TenantsApi(fusion)
    return {
        tenant.name: {
            "display_name": tenant.display_name,
        }
        for tenant in tenants_api_instance.list_tenants().items
    }


@_api_permission_denied_handler("regions")
def generate_regions_dict(module, fusion):
    regions_api_instance = purefusion.RegionsApi(fusion)
    return {
        region.name: {
            "display_name": region.display_name,
        }
        for region in regions_api_instance.list_regions().items
    }


@_api_permission_denied_handler("availability_zones")
def generate_zones_dict(module, fusion):
    zones_info = {}
    az_api_instance = purefusion.AvailabilityZonesApi(fusion)
    regions_api_instance = purefusion.RegionsApi(fusion)
    regions = regions_api_instance.list_regions()
    for region in regions.items:
        zones = az_api_instance.list_availability_zones(region_name=region.name)
        for zone in zones.items:
            az_name = zone.name
            zones_info[az_name] = {
                "display_name": zone.display_name,
                "region": zone.region.name,
            }
    return zones_info


@_api_permission_denied_handler("role_assignments")
def generate_ras_dict(module, fusion):
    ras_info = {}
    ras_api_instance = purefusion.RoleAssignmentsApi(fusion)
    role_api_instance = purefusion.RolesApi(fusion)
    roles = role_api_instance.list_roles()
    for role in roles:
        ras = ras_api_instance.list_role_assignments(role_name=role.name)
        for assignment in ras:
            name = assignment.name
            ras_info[name] = {
                "display_name": assignment.display_name,
                "role": assignment.role.name,
                "scope": assignment.scope.name,
            }
    return ras_info


@_api_permission_denied_handler("roles")
def generate_roles_dict(module, fusion):
    roles_info = {}
    api_instance = purefusion.RolesApi(fusion)
    roles = api_instance.list_roles()
    for role in roles:
        name = role.name
        roles_info[name] = {
            "display_name": role.display_name,
            "scopes": role.assignable_scopes,
        }
    return roles_info


@_api_permission_denied_handler("api_clients")
def generate_api_client_dict(module, fusion):
    client_info = {}
    api_instance = purefusion.IdentityManagerApi(fusion)
    clients = api_instance.list_api_clients()
    for client in clients:
        client_info[client.name] = {
            "display_name": client.display_name,
            "issuer": client.issuer,
            "public_key": client.public_key,
            "creator_id": client.creator_id,
            "last_key_update": time.strftime(
                "%a, %d %b %Y %H:%M:%S %Z",
                time.localtime(client.last_key_update / 1000),
            ),
            "last_used": time.strftime(
                "%a, %d %b %Y %H:%M:%S %Z",
                time.localtime(client.last_used / 1000),
            ),
        }
    return client_info


@_api_permission_denied_handler("users")
def generate_users_dict(module, fusion):
    users_info = {}
    api_instance = purefusion.IdentityManagerApi(fusion)
    users = api_instance.list_users()
    for user in users:
        users_info[user.name] = {
            "display_name": user.display_name,
            "email": user.email,
            "id": user.id,
        }
    return users_info


@_api_permission_denied_handler("hardware_types")
def generate_hardware_types_dict(module, fusion):
    hardware_info = {}
    api_instance = purefusion.HardwareTypesApi(fusion)
    hw_types = api_instance.list_hardware_types()
    for hw_type in hw_types.items:
        hardware_info[hw_type.name] = {
            "array_type": hw_type.array_type,
            "display_name": hw_type.display_name,
            "media_type": hw_type.media_type,
        }
    return hardware_info


@_api_permission_denied_handler("storage_classes")
def generate_sc_dict(module, fusion):
    sc_info = {}
    ss_api_instance = purefusion.StorageServicesApi(fusion)
    sc_api_instance = purefusion.StorageClassesApi(fusion)
    services = ss_api_instance.list_storage_services()
    for service in services.items:
        classes = sc_api_instance.list_storage_classes(
            storage_service_name=service.name,
        )
        for s_class in classes.items:
            sc_info[s_class.name] = {
                "bandwidth_limit": getattr(s_class, "bandwidth_limit", None),
                "iops_limit": getattr(s_class, "iops_limit", None),
                "size_limit": getattr(s_class, "size_limit", None),
                "display_name": s_class.display_name,
                "storage_service": service.name,
            }
    return sc_info


@_api_permission_denied_handler("storage_services")
def generate_storserv_dict(module, fusion):
    ss_dict = {}
    ss_api_instance = purefusion.StorageServicesApi(fusion)
    services = ss_api_instance.list_storage_services()
    for service in services.items:
        ss_dict[service.name] = {
            "display_name": service.display_name,
            "hardware_types": None,
        }
        # can be None if we don't have permission to see this
        if service.hardware_types is not None:
            ss_dict[service.name]["hardware_types"] = []
            for hwtype in service.hardware_types:
                ss_dict[service.name]["hardware_types"].append(hwtype.name)
    return ss_dict


@_api_permission_denied_handler("storage_endpoints")
def generate_se_dict(module, fusion):
    se_dict = {}
    se_api_instance = purefusion.StorageEndpointsApi(fusion)
    az_api_instance = purefusion.AvailabilityZonesApi(fusion)
    regions_api_instance = purefusion.RegionsApi(fusion)
    regions = regions_api_instance.list_regions()
    for region in regions.items:
        azs = az_api_instance.list_availability_zones(region_name=region.name)
        for az in azs.items:
            endpoints = se_api_instance.list_storage_endpoints(
                region_name=region.name,
                availability_zone_name=az.name,
            )
            for endpoint in endpoints.items:
                name = region.name + "/" + az.name + "/" + endpoint.name
                se_dict[name] = {
                    "display_name": endpoint.display_name,
                    "endpoint_type": endpoint.endpoint_type,
                    "iscsi_interfaces": [],
                }
                for iface in endpoint.iscsi.discovery_interfaces:
                    dct = {
                        "address": iface.address,
                        "gateway": iface.gateway,
                        "mtu": iface.mtu,
                        "network_interface_groups": None,
                    }
                    if iface.network_interface_groups is not None:
                        dct["network_interface_groups"] = [
                            nig.name for nig in iface.network_interface_groups
                        ]
                    se_dict[name]["iscsi_interfaces"].append(dct)
    return se_dict


@_api_permission_denied_handler("network_interface_groups")
def generate_nigs_dict(module, fusion):
    nigs_dict = {}
    nig_api_instance = purefusion.NetworkInterfaceGroupsApi(fusion)
    az_api_instance = purefusion.AvailabilityZonesApi(fusion)
    regions_api_instance = purefusion.RegionsApi(fusion)
    regions = regions_api_instance.list_regions()
    for region in regions.items:
        azs = az_api_instance.list_availability_zones(region_name=region.name)
        for az in azs.items:
            nigs = nig_api_instance.list_network_interface_groups(
                region_name=region.name,
                availability_zone_name=az.name,
            )
            for nig in nigs.items:
                name = region.name + "/" + az.name + "/" + nig.name
                nigs_dict[name] = {
                    "display_name": nig.display_name,
                    "gateway": nig.eth.gateway,
                    "prefix": nig.eth.prefix,
                    "mtu": nig.eth.mtu,
                }
    return nigs_dict


@_api_permission_denied_handler("snapshots")
def generate_snap_dicts(module, fusion):
    snap_dict = {}
    vsnap_dict = {}
    tenant_api_instance = purefusion.TenantsApi(fusion)
    tenantspace_api_instance = purefusion.TenantSpacesApi(fusion)
    snap_api_instance = purefusion.SnapshotsApi(fusion)
    vsnap_api_instance = purefusion.VolumeSnapshotsApi(fusion)
    tenants = tenant_api_instance.list_tenants()
    for tenant in tenants.items:
        tenant_spaces = tenantspace_api_instance.list_tenant_spaces(
            tenant_name=tenant.name
        ).items
        for tenant_space in tenant_spaces:
            snaps = snap_api_instance.list_snapshots(
                tenant_name=tenant.name,
                tenant_space_name=tenant_space.name,
            )
            for snap in snaps.items:
                snap_name = tenant.name + "/" + tenant_space.name + "/" + snap.name
                secs, mins, hours = _convert_microseconds(snap.time_remaining)
                snap_dict[snap_name] = {
                    "display_name": snap.display_name,
                    "protection_policy": snap.protection_policy,
                    "time_remaining": "{0} hours, {1} mins, {2} secs".format(
                        int(hours), int(mins), int(secs)
                    ),
                    "volume_snapshots_link": snap.volume_snapshots_link,
                }
                vsnaps = vsnap_api_instance.list_volume_snapshots(
                    tenant_name=tenant.name,
                    tenant_space_name=tenant_space.name,
                    snapshot_name=snap.name,
                )
                for vsnap in vsnaps.items:
                    vsnap_name = (
                        tenant.name
                        + "/"
                        + tenant_space.name
                        + "/"
                        + snap.name
                        + "/"
                        + vsnap.name
                    )
                    secs, mins, hours = _convert_microseconds(vsnap.time_remaining)
                    vsnap_dict[vsnap_name] = {
                        "size": vsnap.size,
                        "display_name": vsnap.display_name,
                        "protection_policy": vsnap.protection_policy,
                        "serial_number": vsnap.serial_number,
                        "created_at": time.strftime(
                            "%a, %d %b %Y %H:%M:%S %Z",
                            time.localtime(vsnap.created_at / 1000),
                        ),
                        "time_remaining": "{0} hours, {1} mins, {2} secs".format(
                            int(hours), int(mins), int(secs)
                        ),
                        "placement_group": vsnap.placement_group.name,
                    }
    return snap_dict, vsnap_dict


@_api_permission_denied_handler("volumes")
def generate_volumes_dict(module, fusion):
    volume_info = {}

    tenant_api_instance = purefusion.TenantsApi(fusion)
    vol_api_instance = purefusion.VolumesApi(fusion)
    tenant_space_api_instance = purefusion.TenantSpacesApi(fusion)

    tenants = tenant_api_instance.list_tenants()
    for tenant in tenants.items:
        tenant_spaces = tenant_space_api_instance.list_tenant_spaces(
            tenant_name=tenant.name
        ).items
        for tenant_space in tenant_spaces:
            volumes = vol_api_instance.list_volumes(
                tenant_name=tenant.name,
                tenant_space_name=tenant_space.name,
            )
            for volume in volumes.items:
                vol_name = tenant.name + "/" + tenant_space.name + "/" + volume.name
                volume_info[vol_name] = {
                    "tenant": tenant.name,
                    "tenant_space": tenant_space.name,
                    "name": volume.name,
                    "size": volume.size,
                    "display_name": volume.display_name,
                    "placement_group": volume.placement_group.name,
                    "source_volume_snapshot": getattr(
                        volume.source_volume_snapshot, "name", None
                    ),
                    "protection_policy": getattr(
                        volume.protection_policy, "name", None
                    ),
                    "storage_class": volume.storage_class.name,
                    "serial_number": volume.serial_number,
                    "target": {},
                    "array": getattr(volume.array, "name", None),
                }

                volume_info[vol_name]["target"] = {
                    "iscsi": {
                        "addresses": volume.target.iscsi.addresses,
                        "iqn": volume.target.iscsi.iqn,
                    },
                    "nvme": {
                        "addresses": None,
                        "nqn": None,
                    },
                    "fc": {
                        "addresses": None,
                        "wwns": None,
                    },
                }
    return volume_info


def main():
    argument_spec = fusion_argument_spec()
    argument_spec.update(
        dict(gather_subset=dict(default="minimum", type="list", elements="str"))
    )

    module = AnsibleModule(argument_spec, supports_check_mode=True)

    # will handle all errors (except #403 which should be handled in code)
    fusion = setup_fusion(module)

    subset = [test.lower() for test in module.params["gather_subset"]]
    valid_subsets = (
        "all",
        "minimum",
        "roles",
        "users",
        "placements",
        "arrays",
        "hardware_types",
        "volumes",
        "hosts",
        "storage_classes",
        "protection_policies",
        "placement_groups",
        "interfaces",
        "zones",
        "nigs",
        "storage_endpoints",
        "snapshots",
        "storage_services",
        "tenants",
        "tenant_spaces",
        "network_interface_groups",
        "api_clients",
        "availability_zones",
        "host_access_policies",
        "network_interfaces",
        "regions",
    )
    for option in subset:
        if option not in valid_subsets:
            module.fail_json(
                msg=f"value gather_subset must be one or more of: {','.join(valid_subsets)}, got: {','.join(subset)}\nvalue {option} is not allowed"
            )

    info = {}

    if "minimum" in subset or "all" in subset:
        info["default"] = generate_default_dict(module, fusion)
    if "hardware_types" in subset or "all" in subset:
        info["hardware_types"] = generate_hardware_types_dict(module, fusion)
    if "users" in subset or "all" in subset:
        info["users"] = generate_users_dict(module, fusion)
    if "regions" in subset or "all" in subset:
        info["regions"] = generate_regions_dict(module, fusion)
    if "availability_zones" in subset or "all" in subset or "zones" in subset:
        info["availability_zones"] = generate_zones_dict(module, fusion)
        if "zones" in subset:
            module.warn(
                "The 'zones' subset is deprecated and will be removed in the version 2.0.0\nUse 'availability_zones' subset instead."
            )
    if "roles" in subset or "all" in subset:
        info["roles"] = generate_roles_dict(module, fusion)
        info["role_assignments"] = generate_ras_dict(module, fusion)
    if "storage_services" in subset or "all" in subset:
        info["storage_services"] = generate_storserv_dict(module, fusion)
    if "volumes" in subset or "all" in subset:
        info["volumes"] = generate_volumes_dict(module, fusion)
    if "protection_policies" in subset or "all" in subset:
        info["protection_policies"] = generate_pp_dict(module, fusion)
    if "placement_groups" in subset or "all" in subset or "placements" in subset:
        info["placement_groups"] = generate_pg_dict(module, fusion)
        if "placements" in subset:
            module.warn(
                "The 'placements' subset is deprecated and will be removed in the version 1.7.0"
            )
    if "storage_classes" in subset or "all" in subset:
        info["storage_classes"] = generate_sc_dict(module, fusion)
    if "network_interfaces" in subset or "all" in subset or "interfaces" in subset:
        info["network_interfaces"] = generate_nics_dict(module, fusion)
        if "interfaces" in subset:
            module.warn(
                "The 'interfaces' subset is deprecated and will be removed in the version 2.0.0\nUse 'network_interfaces' subset instead."
            )
    if "host_access_policies" in subset or "all" in subset or "hosts" in subset:
        info["host_access_policies"] = generate_hap_dict(module, fusion)
        if "hosts" in subset:
            module.warn(
                "The 'hosts' subset is deprecated and will be removed in the version 2.0.0\nUse 'host_access_policies' subset instead."
            )
    if "arrays" in subset or "all" in subset:
        info["arrays"] = generate_array_dict(module, fusion)
    if "tenants" in subset or "all" in subset:
        info["tenants"] = generate_tenant_dict(module, fusion)
    if "tenant_spaces" in subset or "all" in subset:
        info["tenant_spaces"] = generate_ts_dict(module, fusion)
    if "storage_endpoints" in subset or "all" in subset:
        info["storage_endpoints"] = generate_se_dict(module, fusion)
    if "api_clients" in subset or "all" in subset:
        info["api_clients"] = generate_api_client_dict(module, fusion)
    if "network_interface_groups" in subset or "all" in subset or "nigs" in subset:
        info["network_interface_groups"] = generate_nigs_dict(module, fusion)
        if "nigs" in subset:
            module.warn(
                "The 'nigs' subset is deprecated and will be removed in the version 1.7.0"
            )
    if "snapshots" in subset or "all" in subset:
        snap_dicts = generate_snap_dicts(module, fusion)
        if snap_dicts is not None:
            info["snapshots"], info["volume_snapshots"] = snap_dicts
        else:
            info["snapshots"], info["volume_snapshots"] = None, None

    module.exit_json(changed=False, fusion_info=info)


if __name__ == "__main__":
    main()

Anon7 - 2022
AnonSec Team