community.general/lib/ansible/modules/network/f5/bigip_node.py

1195 lines
40 KiB
Python
Raw Normal View History

2014-09-26 01:01:01 +00:00
#!/usr/bin/python
# -*- coding: utf-8 -*-
#
# Copyright: (c) 2016, F5 Networks Inc.
# GNU General Public License v3.0 (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
2014-09-26 01:01:01 +00:00
from __future__ import absolute_import, division, print_function
__metaclass__ = type
2017-08-16 03:16:38 +00:00
ANSIBLE_METADATA = {'metadata_version': '1.1',
'status': ['preview'],
'supported_by': 'certified'}
DOCUMENTATION = r'''
2014-09-26 01:01:01 +00:00
---
module: bigip_node
short_description: Manages F5 BIG-IP LTM nodes
2014-09-26 01:01:01 +00:00
description:
- Manages F5 BIG-IP LTM nodes.
version_added: 1.4
2014-09-26 01:01:01 +00:00
options:
state:
description:
- Specifies the current state of the node. C(enabled) (All traffic
allowed), specifies that system sends traffic to this node regardless
of the node's state. C(disabled) (Only persistent or active connections
allowed), Specifies that the node can handle only persistent or
active connections. C(offline) (Only active connections allowed),
Specifies that the node can handle only active connections. In all
cases except C(absent), the node will be created if it does not yet
exist.
- Be particularly careful about changing the status of a node whose FQDN
cannot be resolved. These situations disable your ability to change their
C(state) to C(disabled) or C(offline). They will remain in an
*Unavailable - Enabled* state.
type: str
choices:
- present
- absent
- enabled
- disabled
- offline
default: present
name:
description:
- Specifies the name of the node.
type: str
required: True
monitor_type:
description:
- Monitor rule type when C(monitors) is specified. When creating a new
pool, if this value is not specified, the default of 'and_list' will
be used.
- Both C(single) and C(and_list) are functionally identical since BIG-IP
considers all monitors as "a list". BIG=IP either has a list of many,
or it has a list of one. Where they differ is in the extra guards that
C(single) provides; namely that it only allows a single monitor.
version_added: "1.3"
type: str
choices:
- and_list
- m_of_n
- single
quorum:
description:
- Monitor quorum value when C(monitor_type) is C(m_of_n).
type: int
version_added: 2.2
monitors:
description:
- Specifies the health monitors that the system currently uses to
monitor this node.
type: list
version_added: 2.2
address:
description:
- IP address of the node. This can be either IPv4 or IPv6. When creating a
new node, one of either C(address) or C(fqdn) must be provided. This
parameter cannot be updated after it is set.
type: str
aliases:
- ip
- host
version_added: 2.2
fqdn:
description:
- FQDN name of the node. This can be any name that is a valid RFC 1123 DNS
name. Therefore, the only characters that can be used are "A" to "Z",
"a" to "z", "0" to "9", the hyphen ("-") and the period (".").
- FQDN names must include at lease one period; delineating the host from
the domain. ex. C(host.domain).
- FQDN names must end with a letter or a number.
- When creating a new node, one of either C(address) or C(fqdn) must be
provided. This parameter cannot be updated after it is set.
type: str
aliases:
- hostname
version_added: 2.5
fqdn_address_type:
description:
- Specifies whether the FQDN of the node resolves to an IPv4 or IPv6 address.
- When creating a new node, if this parameter is not specified and C(fqdn) is
specified, this parameter will default to C(ipv4).
- This parameter cannot be changed after it has been set.
type: str
choices:
- ipv4
- ipv6
- all
version_added: 2.6
fqdn_auto_populate:
description:
- Specifies whether the system automatically creates ephemeral nodes using
the IP addresses returned by the resolution of a DNS query for a node defined
by an FQDN.
- When C(yes), the system generates an ephemeral node for each IP address
returned in response to a DNS query for the FQDN of the node. Additionally,
when a DNS response indicates the IP address of an ephemeral node no longer
exists, the system deletes the ephemeral node.
- When C(no), the system resolves a DNS query for the FQDN of the node with the
single IP address associated with the FQDN.
- When creating a new node, if this parameter is not specified and C(fqdn) is
specified, this parameter will default to C(yes).
- This parameter cannot be changed after it has been set.
type: bool
version_added: 2.6
fqdn_up_interval:
description:
- Specifies the interval in which a query occurs, when the DNS server is up.
The associated monitor attempts to probe three times, and marks the server
down if it there is no response within the span of three times the interval
value, in seconds.
- This parameter accepts a value of C(ttl) to query based off of the TTL of
the FQDN. The default TTL interval is akin to specifying C(3600).
- When creating a new node, if this parameter is not specified and C(fqdn) is
specified, this parameter will default to C(3600).
type: str
version_added: 2.6
fqdn_down_interval:
description:
- Specifies the interval in which a query occurs, when the DNS server is down.
The associated monitor continues polling as long as the DNS server is down.
- When creating a new node, if this parameter is not specified and C(fqdn) is
specified, this parameter will default to C(5).
type: int
version_added: 2.6
description:
description:
- Specifies descriptive text that identifies the node.
- You can remove a description by either specifying an empty string, or by
specifying the special value C(none).
type: str
connection_limit:
description:
- Node connection limit. Setting this to 0 disables the limit.
type: int
version_added: 2.7
rate_limit:
description:
- Node rate limit (connections-per-second). Setting this to 0 disables the limit.
type: int
version_added: 2.7
ratio:
description:
- Node ratio weight. Valid values range from 1 through 100.
- When creating a new node, if this parameter is not specified, the default of
C(1) will be used.
type: int
version_added: 2.7
dynamic_ratio:
description:
- The dynamic ratio number for the node. Used for dynamic ratio load balancing.
- When creating a new node, if this parameter is not specified, the default of
C(1) will be used.
type: int
version_added: 2.7
availability_requirements:
description:
- Specifies, if you activate more than one health monitor, the number of health
monitors that must receive successful responses in order for the link to be
considered available.
suboptions:
type:
description:
- Monitor rule type when C(monitors) is specified.
- When creating a new pool, if this value is not specified, the default of
'all' will be used.
choices:
- all
- at_least
at_least:
description:
- Specifies the minimum number of active health monitors that must be successful
before the link is considered up.
- This parameter is only relevant when a C(type) of C(at_least) is used.
- This parameter will be ignored if a type of C(all) is used.
type: int
type: dict
version_added: 2.8
partition:
description:
- Device partition to manage resources on.
type: str
default: Common
version_added: 2.5
extends_documentation_fragment: f5
author:
- Tim Rupp (@caphrim007)
2019-02-02 05:32:02 +00:00
- Wojciech Wypior (@wojtek0806)
2014-09-26 01:01:01 +00:00
'''
EXAMPLES = r'''
- name: Add node
bigip_node:
host: 10.20.30.40
name: 10.20.30.40
provider:
server: lb.mydomain.com
user: admin
password: secret
delegate_to: localhost
2014-09-26 01:01:01 +00:00
- name: Add node with a single 'ping' monitor
bigip_node:
host: 10.20.30.40
name: mytestserver
monitors:
- /Common/icmp
provider:
server: lb.mydomain.com
user: admin
password: secret
delegate_to: localhost
- name: Modify node description
bigip_node:
name: 10.20.30.40
description: Our best server yet
provider:
server: lb.mydomain.com
user: admin
password: secret
delegate_to: localhost
- name: Delete node
bigip_node:
state: absent
name: 10.20.30.40
provider:
server: lb.mydomain.com
user: admin
password: secret
delegate_to: localhost
- name: Force node offline
bigip_node:
state: disabled
name: 10.20.30.40
provider:
server: lb.mydomain.com
user: admin
password: secret
delegate_to: localhost
- name: Add node by their FQDN
bigip_node:
fqdn: foo.bar.com
2019-02-02 05:32:02 +00:00
name: foobar.net
provider:
server: lb.mydomain.com
user: admin
password: secret
delegate_to: localhost
2014-09-26 01:01:01 +00:00
'''
RETURN = r'''
monitor_type:
description:
- Changed value for the monitor_type of the node.
returned: changed and success
type: str
sample: m_of_n
quorum:
description:
- Changed value for the quorum of the node.
returned: changed and success
type: int
sample: 1
monitors:
description:
- Changed list of monitors for the node.
returned: changed and success
type: list
sample: ['icmp', 'tcp_echo']
description:
description:
- Changed value for the description of the node.
returned: changed and success
type: str
sample: E-Commerce webserver in ORD
session:
description:
- Changed value for the internal session of the node.
returned: changed and success
type: str
sample: user-disabled
state:
description:
- Changed value for the internal state of the node.
returned: changed and success
type: str
sample: m_of_n
'''
import re
import time
from ansible.module_utils.basic import AnsibleModule
from ansible.module_utils.basic import env_fallback
from ansible.module_utils.parsing.convert_bool import BOOLEANS_FALSE
from ansible.module_utils.parsing.convert_bool import BOOLEANS_TRUE
try:
from library.module_utils.network.f5.bigip import F5RestClient
from library.module_utils.network.f5.common import F5ModuleError
from library.module_utils.network.f5.common import AnsibleF5Parameters
from library.module_utils.network.f5.common import fq_name
from library.module_utils.network.f5.common import f5_argument_spec
from library.module_utils.network.f5.common import transform_name
from library.module_utils.compat.ipaddress import ip_address
except ImportError:
from ansible.module_utils.network.f5.bigip import F5RestClient
from ansible.module_utils.network.f5.common import F5ModuleError
from ansible.module_utils.network.f5.common import AnsibleF5Parameters
from ansible.module_utils.network.f5.common import fq_name
from ansible.module_utils.network.f5.common import f5_argument_spec
from ansible.module_utils.network.f5.common import transform_name
from ansible.module_utils.compat.ipaddress import ip_address
class Parameters(AnsibleF5Parameters):
api_map = {
'monitor': 'monitors',
'connectionLimit': 'connection_limit',
'rateLimit': 'rate_limit'
}
api_attributes = [
'description',
'address',
'fqdn',
'ratio',
'connectionLimit',
'rateLimit',
'monitor',
# Used for changing state
#
# user-enabled (enabled)
# user-disabled (disabled)
# user-disabled (offline)
'session',
# Used for changing state
# user-down (offline)
'state'
]
returnables = [
'monitors',
'description',
'fqdn',
'address',
'session',
'state',
'fqdn_auto_populate',
'fqdn_address_type',
'fqdn_up_interval',
'fqdn_down_interval',
'fqdn_name',
'connection_limit',
'ratio',
'rate_limit',
'availability_requirements'
]
updatables = [
'monitors',
'description',
'state',
'fqdn_up_interval',
'fqdn_down_interval',
'tmName',
'fqdn_auto_populate',
'fqdn_address_type',
'connection_limit',
'ratio',
'rate_limit',
]
def to_return(self):
result = {}
try:
for returnable in self.returnables:
result[returnable] = getattr(self, returnable)
result = self._filter_params(result)
return result
except Exception:
return result
@property
def rate_limit(self):
if self._values['rate_limit'] is None:
return None
if self._values['rate_limit'] == 'disabled':
return 0
return int(self._values['rate_limit'])
class Changes(Parameters):
pass
class UsableChanges(Changes):
@property
def fqdn(self):
result = dict()
if self._values['fqdn_up_interval'] is not None:
result['interval'] = self._values['fqdn_up_interval']
if self._values['fqdn_down_interval'] is not None:
result['downInterval'] = self._values['fqdn_down_interval']
if self._values['fqdn_auto_populate'] is not None:
result['autopopulate'] = self._values['fqdn_auto_populate']
if self._values['fqdn_name'] is not None:
result['tmName'] = self._values['fqdn_name']
if self._values['fqdn_address_type'] is not None:
result['addressFamily'] = self._values['fqdn_address_type']
if not result:
return None
return result
@property
def monitors(self):
monitor_string = self._values['monitors']
if monitor_string is None:
return None
if '{' in monitor_string and '}':
tmp = monitor_string.strip('}').split('{')
monitor = ''.join(tmp).rstrip()
return monitor
return monitor_string
class ReportableChanges(Changes):
@property
def monitors(self):
if self._values['monitors'] is None:
return []
try:
result = re.findall(r'/\w+/[^\s}]+', self._values['monitors'])
result.sort()
return result
except Exception:
return self._values['monitors']
@property
def availability_requirement_type(self):
if self._values['monitors'] is None:
return None
if 'min ' in self._values['monitors']:
return 'at_least'
else:
return 'all'
@property
def at_least(self):
"""Returns the 'at least' value from the monitor string.
The monitor string for a Require monitor looks like this.
min 1 of { /Common/gateway_icmp }
This method parses out the first of the numeric values. This values represents
the "at_least" value that can be updated in the module.
Returns:
int: The at_least value if found. None otherwise.
"""
if self._values['monitors'] is None:
return None
pattern = r'min\s+(?P<least>\d+)\s+of\s+'
matches = re.search(pattern, self._values['monitors'])
if matches is None:
return None
return int(matches.group('least'))
@property
def availability_requirements(self):
if self._values['monitors'] is None:
return None
result = dict()
result['type'] = self.availability_requirement_type
result['at_least'] = self.at_least
return result
class ModuleParameters(Parameters):
def _get_availability_value(self, type):
if self._values['availability_requirements'] is None:
return None
if self._values['availability_requirements'][type] is None:
return None
return int(self._values['availability_requirements'][type])
@property
def monitors_list(self):
if self._values['monitors'] is None:
return []
try:
result = re.findall(r'/\w+/[^\s}]+', self._values['monitors'])
except Exception:
result = self._values['monitors']
result.sort()
return result
@property
def monitors(self):
if self._values['monitors'] is None:
return None
if len(self._values['monitors']) == 1 and self._values['monitors'][0] == '':
return '/Common/none'
monitors = [fq_name(self.partition, x) for x in self.monitors_list]
if self.availability_requirement_type == 'at_least':
if self.at_least > len(self.monitors_list):
raise F5ModuleError(
"The 'at_least' value must not exceed the number of 'monitors'."
)
monitors = ' '.join(monitors)
result = 'min {0} of {{ {1} }}'.format(self.at_least, monitors)
else:
result = ' and '.join(monitors).strip()
return result
@property
def availability_requirement_type(self):
if self._values['monitor_type']:
if self._values['monitor_type'] in ['single', 'and_list']:
result = 'all'
else:
result = 'at_least'
self._values['availability_requirements'] = dict(type=None)
self._values['availability_requirements']['type'] = result
if self._values['availability_requirements'] is None:
return None
return self._values['availability_requirements']['type']
@property
def at_least(self):
if self._values['quorum']:
self._values['availability_requirements'] = dict(at_least=None)
self._values['availability_requirements']['at_least'] = self._values['quorum']
return self._get_availability_value('at_least')
@property
def fqdn_up_interval(self):
if self._values['fqdn_up_interval'] is None:
return None
return str(self._values['fqdn_up_interval'])
@property
def fqdn_down_interval(self):
if self._values['fqdn_down_interval'] is None:
return None
return str(self._values['fqdn_down_interval'])
@property
def fqdn_auto_populate(self):
auto_populate = self._values.get('fqdn_auto_populate', None)
if auto_populate in BOOLEANS_TRUE:
return 'enabled'
elif auto_populate in BOOLEANS_FALSE:
return 'disabled'
@property
def fqdn_name(self):
return self._values.get('fqdn', None)
@property
def fqdn(self):
if self._values['fqdn'] is None:
return None
result = dict(
addressFamily=self._values.get('fqdn_address_type', None),
downInterval=self._values.get('fqdn_down_interval', None),
interval=self._values.get('fqdn_up_interval', None),
autopopulate=None,
tmName=self._values.get('fqdn', None)
)
auto_populate = self._values.get('fqdn_auto_populate', None)
if auto_populate in BOOLEANS_TRUE:
result['autopopulate'] = 'enabled'
elif auto_populate in BOOLEANS_FALSE:
result['autopopulate'] = 'disabled'
return result
@property
def description(self):
if self._values['description'] is None:
return None
elif self._values['description'] in ['none', '']:
return ''
return self._values['description']
class ApiParameters(Parameters):
@property
def fqdn_up_interval(self):
if self._values['fqdn'] is None:
return None
if 'interval' in self._values['fqdn']:
return str(self._values['fqdn']['interval'])
@property
def fqdn_down_interval(self):
if self._values['fqdn'] is None:
return None
if 'downInterval' in self._values['fqdn']:
return str(self._values['fqdn']['downInterval'])
@property
def fqdn_address_type(self):
if self._values['fqdn'] is None:
return None
if 'addressFamily' in self._values['fqdn']:
return str(self._values['fqdn']['addressFamily'])
@property
def fqdn_auto_populate(self):
if self._values['fqdn'] is None:
return None
if 'autopopulate' in self._values['fqdn']:
return str(self._values['fqdn']['autopopulate'])
@property
def description(self):
if self._values['description'] in [None, 'none']:
return None
return self._values['description']
@property
def availability_requirement_type(self):
if self._values['monitors'] is None:
return None
if 'min ' in self._values['monitors']:
return 'at_least'
else:
return 'all'
@property
def monitors_list(self):
if self._values['monitors'] is None:
return []
try:
result = re.findall(r'/\w+/[^\s}]+', self._values['monitors'])
except Exception:
result = self._values['monitors']
result.sort()
return result
@property
def monitors(self):
if self._values['monitors'] is None:
return None
if self._values['monitors'] == 'default':
return 'default'
monitors = [fq_name(self.partition, x) for x in self.monitors_list]
if self.availability_requirement_type == 'at_least':
monitors = ' '.join(monitors)
result = 'min {0} of {{ {1} }}'.format(self.at_least, monitors)
else:
result = ' and '.join(monitors).strip()
return result
@property
def at_least(self):
"""Returns the 'at least' value from the monitor string.
The monitor string for a Require monitor looks like this.
min 1 of { /Common/gateway_icmp }
This method parses out the first of the numeric values. This values represents
the "at_least" value that can be updated in the module.
Returns:
int: The at_least value if found. None otherwise.
"""
if self._values['monitors'] is None:
return None
pattern = r'min\s+(?P<least>\d+)\s+of\s+'
matches = re.search(pattern, self._values['monitors'])
if matches is None:
return None
return matches.group('least')
class Difference(object):
def __init__(self, want, have=None):
self.want = want
self.have = have
def compare(self, param):
try:
result = getattr(self, param)
return result
except AttributeError:
return self.__default(param)
def __default(self, param):
attr1 = getattr(self.want, param)
try:
attr2 = getattr(self.have, param)
if attr1 != attr2:
return attr1
except AttributeError:
return attr1
@property
def monitors(self):
if self.want.monitor_type == 'single':
if len(self.want.monitors_list) > 1:
raise F5ModuleError(
"When using a 'monitor_type' of 'single', only one monitor may be provided."
)
elif len(self.have.monitors_list) > 1 and len(self.want.monitors_list) == 0:
# Handle instances where there already exists many monitors, and the
# user runs the module again specifying that the monitor_type should be
# changed to 'single'
raise F5ModuleError(
"A single monitor must be specified if more than one monitor currently exists on your pool."
)
if self.want.monitors is None:
return None
if self.want.monitors == 'default' and self.have.monitors == 'default':
return None
if self.want.monitors == 'default' and self.have.monitors is None:
return None
if self.want.monitors == '/Common/none' and self.have.monitors == '/Common/none':
return None
if self.want.monitors == 'default' and len(self.have.monitors) > 0:
return 'default'
if self.have.monitors is None:
return self.want.monitors
if self.have.monitors != self.want.monitors:
return self.want.monitors
@property
def state(self):
result = None
if self.want.state in ['present', 'enabled']:
if self.have.session not in ['user-enabled', 'monitor-enabled']:
result = dict(
session='user-enabled',
state='user-up',
)
elif self.want.state == 'disabled':
if self.have.session != 'user-disabled' or self.have.state == 'user-down':
result = dict(
session='user-disabled',
state='user-up'
)
elif self.want.state == 'offline':
if self.have.state != 'user-down':
result = dict(
session='user-disabled',
state='user-down'
)
return result
@property
def description(self):
if self.want.description is None:
return None
if self.have.description is None and self.want.description == '':
return None
if self.want.description != self.have.description:
return self.want.description
class ModuleManager(object):
def __init__(self, *args, **kwargs):
self.module = kwargs.get('module', None)
self.client = F5RestClient(**self.module.params)
self.have = None
self.want = ModuleParameters(params=self.module.params)
self.changes = UsableChanges()
def _set_changed_options(self):
changed = {}
for key in Parameters.returnables:
if getattr(self.want, key) is not None:
changed[key] = getattr(self.want, key)
if changed:
self.changes = UsableChanges(params=changed)
def _update_changed_options(self):
diff = Difference(self.want, self.have)
updatables = Parameters.updatables
changed = dict()
for k in updatables:
change = diff.compare(k)
if change is None:
continue
else:
if isinstance(change, dict):
changed.update(change)
else:
changed[k] = change
if changed:
self.changes = UsableChanges(params=changed)
return True
return False
def _announce_deprecations(self): # lgtm [py/similar-function]
warnings = []
if self.want:
warnings += self.want._values.get('__warnings', [])
if self.have:
warnings += self.have._values.get('__warnings', [])
for warning in warnings:
self.module.deprecate(
msg=warning['msg'],
version=warning['version']
)
def exec_module(self):
changed = False
result = dict()
state = self.want.state
try:
if state in ['present', 'enabled', 'disabled', 'offline']:
changed = self.present()
elif state == "absent":
changed = self.absent()
except IOError as e:
raise F5ModuleError(str(e))
changes = self.changes.to_return()
result.update(**changes)
result.update(dict(changed=changed))
self._announce_deprecations()
return result
def present(self):
if self.exists():
return self.update()
else:
return self.create()
def _check_required_creation_vars(self):
if self.want.address is None and self.want.fqdn is None:
raise F5ModuleError(
"At least one of 'address' or 'fqdn' is required when creating a node"
)
elif self.want.address is not None and self.want.fqdn is not None:
raise F5ModuleError(
"Only one of 'address' or 'fqdn' can be provided when creating a node"
)
elif self.want.fqdn is not None:
self.want.update(dict(address='any6'))
def _munge_creation_state_for_device(self):
# Modifying the state before sending to BIG-IP
#
# The 'state' must be set to None to exclude the values (accepted by this
# module) from being sent to the BIG-IP because for specific Ansible states,
# BIG-IP will consider those state values invalid.
if self.want.state in ['present', 'enabled']:
self.want.update(dict(
session='user-enabled',
state='user-up',
))
elif self.want.state in 'disabled':
self.want.update(dict(
session='user-disabled',
state='user-up'
))
else:
# State 'offline'
# Offline state will result in the monitors stopping for the node
self.want.update(dict(
session='user-disabled',
# only a valid state can be specified. The module's value is "offline",
# but this is an invalid value for the BIG-IP. Therefore set it to user-down.
state='user-down',
# Even user-down wil not work when _creating_ a node, so we register another
# want value (that is not sent to the API). This is checked for later to
# determine if we have to PATCH the node to be offline.
is_offline=True
))
def create(self):
self._check_required_creation_vars()
self._munge_creation_state_for_device()
if self.want.fqdn_auto_populate is None:
self.want.update({'fqdn_auto_populate': True})
if self.want.fqdn_address_type is None:
self.want.update({'fqdn_address_type': 'ipv4'})
if self.want.fqdn_up_interval is None:
self.want.update({'fqdn_up_interval': 3600})
if self.want.fqdn_down_interval is None:
self.want.update({'fqdn_down_interval': 5})
if self.want.ratio is None:
self.want.update({'ratio': 1})
if self.want.dynamic_ratio is None:
self.want.update({'dynamic_ratio': 1})
self._set_changed_options()
if self.module.check_mode:
return True
# These are being set here because the ``create_on_device`` method
# uses ``self.changes`` (to get formatting of parameters correct)
# but these two parameters here cannot be changed and also it is
# not easy to get the current versions of them for comparison.
if self.want.address:
self.changes.update({'address': self.want.address})
if self.want.fqdn_up_interval is not None:
self.changes.update({'fqdn_up_interval': self.want.fqdn_up_interval})
if self.want.fqdn_down_interval is not None:
self.changes.update({'fqdn_down_interval': self.want.fqdn_down_interval})
if self.want.fqdn_auto_populate is not None:
self.changes.update({'fqdn_auto_populate': self.want.fqdn_auto_populate})
if self.want.fqdn_name is not None:
self.changes.update({'fqdn_name': self.want.fqdn_name})
if self.want.fqdn_address_type is not None:
self.changes.update({'fqdn_address_type': self.want.fqdn_address_type})
self.create_on_device()
if not self.exists():
raise F5ModuleError("Failed to create the node")
# It appears that you cannot create a node in an 'offline' state, so instead
# we update its status to offline after we create it.
if self.want.is_offline:
self.update_node_offline_on_device()
return True
def should_update(self):
result = self._update_changed_options()
if result:
return True
return False
def update(self):
self.have = self.read_current_from_device()
if not self.should_update():
return False
if self.want.fqdn_auto_populate is not None:
if self.want.fqdn_auto_populate != self.have.fqdn_auto_populate:
raise F5ModuleError(
"The 'fqdn_auto_populate' parameter cannot be changed."
)
if self.want.fqdn_address_type is not None:
if self.want.fqdn_address_type != self.have.fqdn_address_type:
raise F5ModuleError(
"The 'fqdn_address_type' parameter cannot be changed."
)
if self.module.check_mode:
return True
self.update_on_device()
if self.want.state == 'offline':
self.update_node_offline_on_device()
return True
def absent(self):
if self.exists():
return self.remove()
return False
def remove(self):
if self.module.check_mode:
return True
self.remove_from_device()
if self.exists():
raise F5ModuleError("Failed to delete the node.")
return True
def read_current_from_device(self):
uri = "https://{0}:{1}/mgmt/tm/ltm/node/{2}".format(
self.client.provider['server'],
self.client.provider['server_port'],
transform_name(self.want.partition, self.want.name)
)
resp = self.client.api.get(uri)
try:
response = resp.json()
except ValueError as ex:
raise F5ModuleError(str(ex))
if 'code' in response and response['code'] == 400:
if 'message' in response:
raise F5ModuleError(response['message'])
else:
raise F5ModuleError(resp.content)
return ApiParameters(params=response)
def exists(self): # lgtm [py/similar-function]
uri = "https://{0}:{1}/mgmt/tm/ltm/node/{2}".format(
self.client.provider['server'],
self.client.provider['server_port'],
transform_name(self.want.partition, self.want.name)
)
resp = self.client.api.get(uri)
try:
response = resp.json()
except ValueError:
return False
if resp.status == 404 or 'code' in response and response['code'] == 404:
return False
return True
def update_node_offline_on_device(self):
params = dict(
session="user-disabled",
state="user-down"
)
uri = "https://{0}:{1}/mgmt/tm/ltm/node/{2}".format(
self.client.provider['server'],
self.client.provider['server_port'],
transform_name(self.want.partition, self.want.name)
)
resp = self.client.api.patch(uri, json=params)
try:
response = resp.json()
except ValueError as ex:
raise F5ModuleError(str(ex))
if 'code' in response and response['code'] == 400:
if 'message' in response:
raise F5ModuleError(response['message'])
else:
raise F5ModuleError(resp.content)
def update_on_device(self):
params = self.changes.api_params()
uri = "https://{0}:{1}/mgmt/tm/ltm/node/{2}".format(
self.client.provider['server'],
self.client.provider['server_port'],
transform_name(self.want.partition, self.want.name)
)
if params:
resp = self.client.api.patch(uri, json=params)
try:
response = resp.json()
except ValueError as ex:
raise F5ModuleError(str(ex))
if 'code' in response and response['code'] == 400:
if 'message' in response:
raise F5ModuleError(response['message'])
else:
raise F5ModuleError(resp.content)
def create_on_device(self):
params = self.changes.api_params()
params['name'] = self.want.name
params['partition'] = self.want.partition
uri = "https://{0}:{1}/mgmt/tm/ltm/node/".format(
self.client.provider['server'],
self.client.provider['server_port']
)
resp = self.client.api.post(uri, json=params)
try:
response = resp.json()
except ValueError as ex:
raise F5ModuleError(str(ex))
if 'code' in response and response['code'] in [400, 403]:
if 'message' in response:
raise F5ModuleError(response['message'])
else:
raise F5ModuleError(resp.content)
self._wait_for_fqdn_checks()
def _wait_for_fqdn_checks(self):
while True:
have = self.read_current_from_device()
if have.state == 'fqdn-checking':
time.sleep(1)
else:
break
def remove_from_device(self):
uri = "https://{0}:{1}/mgmt/tm/ltm/node/{2}".format(
self.client.provider['server'],
self.client.provider['server_port'],
transform_name(self.want.partition, self.want.name)
)
resp = self.client.api.delete(uri)
if resp.status == 200:
return True
class ArgumentSpec(object):
def __init__(self):
self.supports_check_mode = True
argument_spec = dict(
name=dict(required=True),
address=dict(
aliases=['host', 'ip']
),
fqdn=dict(
aliases=['hostname']
),
description=dict(),
state=dict(
choices=['absent', 'present', 'enabled', 'disabled', 'offline'],
default='present'
),
partition=dict(
default='Common',
fallback=(env_fallback, ['F5_PARTITION'])
),
fqdn_address_type=dict(
choices=['ipv4', 'ipv6', 'all']
),
fqdn_auto_populate=dict(type='bool'),
fqdn_up_interval=dict(),
fqdn_down_interval=dict(type='int'),
connection_limit=dict(type='int'),
rate_limit=dict(type='int'),
ratio=dict(type='int'),
dynamic_ratio=dict(type='int'),
availability_requirements=dict(
type='dict',
options=dict(
type=dict(
choices=['all', 'at_least'],
required=True
),
at_least=dict(type='int'),
),
required_if=[
['type', 'at_least', ['at_least']],
]
),
monitors=dict(type='list'),
# Deprecated parameters
monitor_type=dict(
choices=[
'and_list', 'm_of_n', 'single'
],
removed_in_version=2.12,
),
quorum=dict(
type='int',
removed_in_version=2.12,
),
)
self.argument_spec = {}
self.argument_spec.update(f5_argument_spec)
self.argument_spec.update(argument_spec)
self.mutually_exclusive = [
['monitor_type', 'quorum', 'availability_requirements']
]
2014-09-26 01:01:01 +00:00
def main():
spec = ArgumentSpec()
2015-06-03 06:22:18 +00:00
module = AnsibleModule(
argument_spec=spec.argument_spec,
supports_check_mode=spec.supports_check_mode,
mutually_exclusive=spec.mutually_exclusive
2014-09-26 01:01:01 +00:00
)
try:
mm = ModuleManager(module=module)
results = mm.exec_module()
module.exit_json(**results)
except F5ModuleError as ex:
module.fail_json(msg=str(ex))
2014-09-26 01:01:01 +00:00
if __name__ == '__main__':
main()