Dre4m Shell
Server IP : 85.214.239.14  /  Your IP : 3.141.4.12
Web Server : Apache/2.4.62 (Debian)
System : Linux h2886529.stratoserver.net 4.9.0 #1 SMP Mon Sep 30 15:36:27 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/flasharray/plugins/modules/

Upload File :
current_dir [ Writeable ] document_root [ Writeable ]

 

Command :


[ HOME SHELL ]     

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

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

from __future__ import absolute_import, division, print_function

__metaclass__ = type

ANSIBLE_METADATA = {
    "metadata_version": "1.1",
    "status": ["preview"],
    "supported_by": "community",
}

DOCUMENTATION = r"""
---
module: purefa_pod
short_description:  Manage AC pods in Pure Storage FlashArrays
version_added: '1.0.0'
description:
- Manage AC pods in a Pure Storage FlashArray.
author:
- Pure Storage Ansible Team (@sdodsley) <pure-ansible-team@purestorage.com>
options:
  name:
    description:
    - The name of the pod.
    type: str
    required: true
  stretch:
    description:
    - The name of the array to stretch to/unstretch from. Must be synchromously replicated.
    - To unstretch an array use state I(absent)
    - You can only specify a remote array, ie you cannot unstretch a pod from the
      current array and then restretch back to the current array.
    - To restretch a pod you must perform this from the remaining array the pod
      resides on.
    type: str
  failover:
    description:
    - The name of the array given priority to stay online if arrays loose
      contact with eachother.
    - Oprions are either array in the cluster, or I(auto)
    type: list
    elements: str
  state:
    description:
    - Define whether the pod should exist or not.
    default: present
    choices: [ absent, present ]
    type: str
  eradicate:
    description:
    - Define whether to eradicate the pod on delete or leave in trash.
    type: bool
    default: false
  target:
    description:
    - Name of clone target pod.
    type: str
  mediator:
    description:
    - Name of the mediator to use for a pod
    type: str
    default: purestorage
  promote:
    description:
      - Promote/demote any pod not in a stretched relationship. .
      - Demoting a pod will render it read-only.
    required: false
    type: bool
  quiesce:
    description:
      - Quiesce/Skip quiesce when I(promote) is false and demoting an ActiveDR pod.
      - Quiesce will ensure all local data has been replicated before demotion.
      - Skipping quiesce looses all pending data to be replicated to the remote pod.
      - Can only demote the pod if it is in a Acrive DR replica link relationship.
      - This will default to True
    required: false
    type: bool
  undo:
    description:
      - Use the I(undo-remote) pod when I(promote) is true and promoting an ActiveDR pod.
      - This will default to True
    required: false
    type: bool
  quota:
    description:
      - Logical quota limit of the pod in K, M, G, T or P units, or bytes.
    type: str
    version_added: '1.18.0'
  ignore_usage:
    description:
    -  Flag used to override checks for quota management
       operations.
    - If set to true, pod usage is not checked against the
      quota_limits that are set.
    - If set to false, the actual logical bytes in use are prevented
      from exceeding the limits set on the pod.
    - Client operations might be impacted.
    - If the limit exceeds the quota, the operation is not allowed.
    default: false
    type: bool
    version_added: '1.18.0'

extends_documentation_fragment:
- purestorage.flasharray.purestorage.fa
"""

EXAMPLES = r"""
- name: Create new pod named foo
  purestorage.flasharray.purefa_pod:
    name: foo
    fa_url: 10.10.10.2
    api_token: e31060a7-21fc-e277-6240-25983c6c4592
    state: present

- name: Delete and eradicate pod named foo
  purestorage.flasharray.purefa_pod:
    name: foo
    eradicate: true
    fa_url: 10.10.10.2
    api_token: e31060a7-21fc-e277-6240-25983c6c4592
    state: absent

- name: Set failover array for pod named foo
  purestorage.flasharray.purefa_pod:
    name: foo
    failover:
    - array1
    fa_url: 10.10.10.2
    api_token: e31060a7-21fc-e277-6240-25983c6c4592

- name: Set mediator for pod named foo
  purestorage.flasharray.purefa_pod:
    name: foo
    mediator: bar
    fa_url: 10.10.10.2
    api_token: e31060a7-21fc-e277-6240-25983c6c4592

- name: Stretch a pod named foo to array2
  purestorage.flasharray.purefa_pod:
    name: foo
    stretch: array2
    fa_url: 10.10.10.2
    api_token: e31060a7-21fc-e277-6240-25983c6c4592

- name: Unstretch a pod named foo from array2
  purestorage.flasharray.purefa_pod:
    name: foo
    stretch: array2
    state: absent
    fa_url: 10.10.10.2
    api_token: e31060a7-21fc-e277-6240-25983c6c4592

- name: Create clone of pod foo named bar
  purestorage.flasharray.purefa_pod:
    name: foo
    target: bar
    fa_url: 10.10.10.2
    api_token: e31060a7-21fc-e277-6240-25983c6c4592
    state: present
"""

RETURN = r"""
"""

HAS_PURESTORAGE = True
try:
    from pypureclient import flasharray
except ImportError:
    HAS_PURESTORAGE = False

from ansible.module_utils.basic import AnsibleModule
from ansible_collections.purestorage.flasharray.plugins.module_utils.purefa import (
    get_system,
    get_array,
    purefa_argument_spec,
)


POD_API_VERSION = "1.13"
POD_QUOTA_VERSION = "2.23"


def human_to_bytes(size):
    """Given a human-readable byte string (e.g. 2G, 30M),
    return the number of bytes.  Will return 0 if the argument has
    unexpected form.
    """
    bytes = size[:-1]
    unit = size[-1].upper()
    if bytes.isdigit():
        bytes = int(bytes)
        if unit == "P":
            bytes *= 1125899906842624
        elif unit == "T":
            bytes *= 1099511627776
        elif unit == "G":
            bytes *= 1073741824
        elif unit == "M":
            bytes *= 1048576
        elif unit == "K":
            bytes *= 1024
        else:
            bytes = 0
    else:
        bytes = 0
    return bytes


def get_pod(module, array):
    """Return Pod or None"""
    try:
        return array.get_pod(module.params["name"])
    except Exception:
        return None


def get_undo_pod(module, array):
    """Return Undo Pod or None"""
    try:
        return array.get_pod(module.params["name"] + ".undo-demote", pending_only=True)
    except Exception:
        return None


def get_target(module, array):
    """Return Pod or None"""
    try:
        return array.get_pod(module.params["target"])
    except Exception:
        return None


def get_destroyed_pod(module, array):
    """Return Destroyed Volume or None"""
    try:
        return bool(
            array.get_pod(module.params["name"], pending=True)["time_remaining"] != ""
        )
    except Exception:
        return False


def get_destroyed_target(module, array):
    """Return Destroyed Volume or None"""
    try:
        return bool(
            array.get_pod(module.params["target"], pending=True)["time_remaining"] != ""
        )
    except Exception:
        return False


def check_arrays(module, array):
    """Check if array name provided are sync-replicated"""
    good_arrays = []
    good_arrays.append(array.get()["array_name"])
    connected_arrays = array.list_array_connections()
    for arr in range(0, len(connected_arrays)):
        if connected_arrays[arr]["type"] == "sync-replication":
            good_arrays.append(connected_arrays[arr]["array_name"])
    if module.params["failover"] is not None:
        if module.params["failover"] == ["auto"]:
            failover_array = []
        else:
            failover_array = module.params["failover"]
        if failover_array != []:
            for arr in range(0, len(failover_array)):
                if failover_array[arr] not in good_arrays:
                    module.fail_json(
                        msg="Failover array {0} is not valid.".format(
                            failover_array[arr]
                        )
                    )
    if module.params["stretch"] is not None:
        if module.params["stretch"] not in good_arrays:
            module.fail_json(
                msg="Stretch: Array {0} is not connected.".format(
                    module.params["stretch"]
                )
            )
    return None


def create_pod(module, array):
    """Create Pod"""
    changed = True
    if module.params["target"]:
        module.fail_json(msg="Cannot clone non-existant pod.")
    if not module.check_mode:
        try:
            if module.params["failover"]:
                array.create_pod(
                    module.params["name"], failover_list=module.params["failover"]
                )
            else:
                array.create_pod(module.params["name"])
        except Exception:
            module.fail_json(
                msg="Pod {0} creation failed.".format(module.params["name"])
            )
        if module.params["mediator"] != "purestorage":
            try:
                array.set_pod(module.params["name"], mediator=module.params["mediator"])
            except Exception:
                module.warn(
                    "Failed to communicate with mediator {0}, using default value".format(
                        module.params["mediator"]
                    )
                )
        if module.params["stretch"]:
            current_array = array.get()["array_name"]
            if module.params["stretch"] != current_array:
                try:
                    array.add_pod(module.params["name"], module.params["rrays"])
                except Exception:
                    module.fail_json(
                        msg="Failed to stretch pod {0} to array {1}.".format(
                            module.params["name"], module.params["stretch"]
                        )
                    )
        if module.params["quota"]:
            arrayv6 = get_array(module)
            res = arrayv6.patch_pods(
                names=[module.params["name"]],
                pod=flasharray.PodPatch(
                    quota_limit=human_to_bytes(module.params["quota"])
                ),
            )
            if res.status_code != 200:
                module.fail_json(
                    msg="Failed to apply quota to pod {0}. Error: {1}".format(
                        module.params["name"], res.errors[0].message
                    )
                )
    module.exit_json(changed=changed)


def clone_pod(module, array):
    """Create Pod Clone"""
    changed = False
    if get_target(module, array) is None:
        if not get_destroyed_target(module, array):
            changed = True
            if not module.check_mode:
                try:
                    array.clone_pod(module.params["name"], module.params["target"])
                except Exception:
                    module.fail_json(
                        msg="Clone pod {0} to pod {1} failed.".format(
                            module.params["name"], module.params["target"]
                        )
                    )
        else:
            module.fail_json(
                msg="Target pod {0} already exists but deleted.".format(
                    module.params["target"]
                )
            )

    module.exit_json(changed=changed)


def update_pod(module, array):
    """Update Pod configuration"""
    changed = False
    current_config = array.get_pod(module.params["name"], failover_preference=True)
    if module.params["failover"]:
        current_failover = current_config["failover_preference"]
        if current_failover == [] or sorted(module.params["failover"]) != sorted(
            current_failover
        ):
            changed = True
            if not module.check_mode:
                try:
                    if module.params["failover"] == ["auto"]:
                        if current_failover != []:
                            array.set_pod(module.params["name"], failover_preference=[])
                    else:
                        array.set_pod(
                            module.params["name"],
                            failover_preference=module.params["failover"],
                        )
                except Exception:
                    module.fail_json(
                        msg="Failed to set failover preference for pod {0}.".format(
                            module.params["name"]
                        )
                    )
    current_config = array.get_pod(module.params["name"], mediator=True)
    if current_config["mediator"] != module.params["mediator"]:
        changed = True
        if not module.check_mode:
            try:
                array.set_pod(module.params["name"], mediator=module.params["mediator"])
            except Exception:
                module.warn(
                    "Failed to communicate with mediator {0}. Setting unchanged.".format(
                        module.params["mediator"]
                    )
                )
    if module.params["promote"] is not None:
        if len(current_config["arrays"]) > 1:
            module.fail_json(
                msg="Promotion/Demotion not permitted. Pod {0} is stretched".format(
                    module.params["name"]
                )
            )
        else:
            if (
                current_config["promotion_status"] == "demoted"
                and module.params["promote"]
            ):
                try:
                    if module.params["undo"] is None:
                        module.params["undo"] = True
                    if current_config["promotion_status"] == "quiescing":
                        module.fail_json(
                            msg="Cannot promote pod {0} as it is still quiesing".format(
                                module.params["name"]
                            )
                        )
                    elif module.params["undo"]:
                        changed = True
                        if not module.check_mode:
                            if get_undo_pod(module, array):
                                array.promote_pod(
                                    module.params["name"],
                                    promote_from=module.params["name"] + ".undo-demote",
                                )
                            else:
                                array.promote_pod(module.params["name"])
                                module.warn(
                                    "undo-demote pod remaining for {0}. Consider eradicating this.".format(
                                        module.params["name"]
                                    )
                                )
                    else:
                        changed = True
                        if not module.check_mode:
                            array.promote_pod(module.params["name"])
                except Exception:
                    module.fail_json(
                        msg="Failed to promote pod {0}.".format(module.params["name"])
                    )
            elif (
                current_config["promotion_status"] != "demoted"
                and not module.params["promote"]
            ):
                try:
                    if get_undo_pod(module, array):
                        module.fail_json(
                            msg="Cannot demote pod {0} due to associated undo-demote pod not being eradicated".format(
                                module.params["name"]
                            )
                        )
                    if module.params["quiesce"] is None:
                        module.params["quiesce"] = True
                    if current_config["link_target_count"] == 0:
                        changed = True
                        if not module.check_mode:
                            array.demote_pod(module.params["name"])
                    elif not module.params["quiesce"]:
                        changed = True
                        if not module.check_mode:
                            array.demote_pod(module.params["name"], skip_quiesce=True)
                    else:
                        changed = True
                        if not module.check_mode:
                            array.demote_pod(module.params["name"], quiesce=True)
                except Exception:
                    module.fail_json(
                        msg="Failed to demote pod {0}.".format(module.params["name"])
                    )
    if module.params["quota"]:
        arrayv6 = get_array(module)
        current_pod = list(arrayv6.get_pods(names=[module.params["name"]]).items)[0]
        quota = human_to_bytes(module.params["quota"])
        if current_pod.quota_limit != quota:
            changed = True
            if not module.check_mode:
                res = arrayv6.patch_pods(
                    names=[module.params["name"]],
                    pod=flasharray.PodPatch(
                        quota_limit=quota, ignore_usage=module.params["ignore_usage"]
                    ),
                )
                if res.status_code != 200:
                    module.fail_json(
                        msg="Failed to update quota on pod {0}. Error: {1}".format(
                            module.params["name"], res.errors[0].message
                        )
                    )

    module.exit_json(changed=changed)


def stretch_pod(module, array):
    """Stretch/unstretch Pod configuration"""
    changed = False
    current_config = array.get_pod(module.params["name"], failover_preference=True)
    if module.params["stretch"]:
        current_arrays = []
        for arr in range(0, len(current_config["arrays"])):
            current_arrays.append(current_config["arrays"][arr]["name"])
        if (
            module.params["stretch"] not in current_arrays
            and module.params["state"] == "present"
        ):
            changed = True
            if not module.check_mode:
                try:
                    array.add_pod(module.params["name"], module.params["stretch"])
                except Exception:
                    module.fail_json(
                        msg="Failed to stretch pod {0} to array {1}.".format(
                            module.params["name"], module.params["stretch"]
                        )
                    )

        if (
            module.params["stretch"] in current_arrays
            and module.params["state"] == "absent"
        ):
            changed = True
            if not module.check_mode:
                try:
                    array.remove_pod(module.params["name"], module.params["stretch"])
                except Exception:
                    module.fail_json(
                        msg="Failed to unstretch pod {0} from array {1}.".format(
                            module.params["name"], module.params["stretch"]
                        )
                    )

    module.exit_json(changed=changed)


def delete_pod(module, array):
    """Delete Pod"""
    changed = True
    if not module.check_mode:
        try:
            array.destroy_pod(module.params["name"])
            if module.params["eradicate"]:
                try:
                    array.eradicate_pod(module.params["name"])
                except Exception:
                    module.fail_json(
                        msg="Eradicate pod {0} failed.".format(module.params["name"])
                    )
        except Exception:
            module.fail_json(msg="Delete pod {0} failed.".format(module.params["name"]))
    module.exit_json(changed=changed)


def eradicate_pod(module, array):
    """Eradicate Deleted Pod"""
    changed = True
    if not module.check_mode:
        if module.params["eradicate"]:
            try:
                array.eradicate_pod(module.params["name"])
            except Exception:
                module.fail_json(
                    msg="Eradication of pod {0} failed".format(module.params["name"])
                )
    module.exit_json(changed=changed)


def recover_pod(module, array):
    """Recover Deleted Pod"""
    changed = True
    if not module.check_mode:
        try:
            array.recover_pod(module.params["name"])
        except Exception:
            module.fail_json(
                msg="Recovery of pod {0} failed".format(module.params["name"])
            )
    module.exit_json(changed=changed)


def main():
    argument_spec = purefa_argument_spec()
    argument_spec.update(
        dict(
            name=dict(type="str", required=True),
            stretch=dict(type="str"),
            target=dict(type="str"),
            mediator=dict(type="str", default="purestorage"),
            failover=dict(type="list", elements="str"),
            promote=dict(type="bool"),
            undo=dict(type="bool"),
            quiesce=dict(type="bool"),
            eradicate=dict(type="bool", default=False),
            state=dict(type="str", default="present", choices=["absent", "present"]),
            quota=dict(type="str"),
            ignore_usage=dict(type="bool", default=False),
        )
    )

    mutually_exclusive = [
        ["stretch", "failover"],
        ["stretch", "eradicate"],
        ["stretch", "mediator"],
        ["target", "mediator"],
        ["target", "stretch"],
        ["target", "failover"],
        ["target", "eradicate"],
    ]

    module = AnsibleModule(
        argument_spec, mutually_exclusive=mutually_exclusive, supports_check_mode=True
    )

    state = module.params["state"]
    array = get_system(module)

    api_version = array._list_available_rest_versions()
    if POD_API_VERSION not in api_version:
        module.fail_json(
            msg="FlashArray REST version not supported. "
            "Minimum version required: {0}".format(POD_API_VERSION)
        )

    if module.params["quota"] and POD_QUOTA_VERSION in api_version:
        if not HAS_PURESTORAGE:
            module.fail_json(
                msg="py-pure-client sdk is required to support 'count' parameter"
            )
    pod = get_pod(module, array)
    destroyed = ""
    if not pod:
        destroyed = get_destroyed_pod(module, array)
    if module.params["failover"] or module.params["failover"] != "auto":
        check_arrays(module, array)

    if state == "present" and not pod:
        create_pod(module, array)
    elif pod and module.params["stretch"]:
        stretch_pod(module, array)
    elif state == "present" and pod and module.params["target"]:
        clone_pod(module, array)
    elif state == "present" and pod and module.params["target"]:
        clone_pod(module, array)
    elif state == "present" and pod:
        update_pod(module, array)
    elif state == "absent" and pod and not module.params["stretch"]:
        delete_pod(module, array)
    elif state == "present" and destroyed:
        recover_pod(module, array)
    elif state == "absent" and destroyed:
        eradicate_pod(module, array)
    elif state == "absent" and not pod:
        module.exit_json(changed=False)


if __name__ == "__main__":
    main()

Anon7 - 2022
AnonSec Team