#!/usr/bin/python # This file is part of Ansible # # Ansible is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # Ansible is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with Ansible. If not, see . import sys try: import ovh import ovh.exceptions from ovh.exceptions import APIError HAS_OVH = True except ImportError: HAS_OVH = False # import module snippets from ansible.module_utils.basic import * DOCUMENTATION = ''' --- module: ovh_ip_loadbalancing_backend short_description: Manage OVH IP LoadBalancing backends description: - Manage OVH (French European hosting provider) LoadBalancing IP backends version_added: "2.1" author: Pascal HERAUD @pascalheraud notes: - Uses the python OVH Api U(https://github.com/ovh/python-ovh). \ You have to create an application (a key and secret) with a consummer \ key as described into U(https://eu.api.ovh.com/g934.first_step_with_api) requirements: - ovh > 0.3.5 options: name: required: true description: - Name of the LoadBalancing internal name (ip-X.X.X.X) backend: required: true description: - The IP address of the backend to update / modify / delete state: required: false default: present choices: ['present', 'absent'] description: - Determines wether the backend is to be created/modified \ or deleted probe: required: false default: none choices: ['none', 'http', 'icmp' , 'oco'] description: - Determines the type of probe to use for this backend weight: required: false default: 8 description: - Determines the weight for this backend endpoint: required: true description: - The endpoint to use ( for instance ovh-eu) application_key: required: true description: - The applicationKey to use application_secret: required: true description: - The application secret to use consumer_key: required: true description: - The consumer key to use timeout: required: false type: "int" default: 120 description: - The timeout in seconds used to wait for a task to be \ completed. Default is 120 seconds. ''' EXAMPLES = ''' # Adds or modify the backend '212.1.1.1' to a \ loadbalancing 'ip-1.1.1.1' - ovh_ip_loadbalancing name=ip-1.1.1.1 backend=212.1.1.1 \ state=present probe=none weight=8 \ endpoint=ovh-eu application_key=yourkey \ application_secret=yoursecret consumer_key=yourconsumerkey # Removes a backend '212.1.1.1' from a loadbalancing \ 'ip-1.1.1.1' - ovh_ip_loadbalancing name=ip-1.1.1.1 backend=212.1.1.1 state=absent endpoint=ovh-eu application_key=yourkey \ application_secret=yoursecret consumer_key=yourconsumerkey ''' RETURN = ''' ''' def getOvhClient(ansibleModule): endpoint = ansibleModule.params.get('endpoint') application_key = ansibleModule.params.get('application_key') application_secret = ansibleModule.params.get('application_secret') consumer_key = ansibleModule.params.get('consumer_key') return ovh.Client( endpoint=endpoint, application_key=application_key, application_secret=application_secret, consumer_key=consumer_key ) def waitForNoTask(client, name, timeout): currentTimeout = timeout while len(client.get('/ip/loadBalancing/{}/task'.format(name))) > 0: time.sleep(1) # Delay for 1 sec currentTimeout -= 1 if currentTimeout < 0: return False return True def main(): module = AnsibleModule( argument_spec=dict( name=dict(required=True), backend=dict(required=True), weight=dict(default=8, type='int'), probe=dict(default='none', choices=['none', 'http', 'icmp', 'oco']), state=dict(default='present', choices=['present', 'absent']), endpoint=dict(required=True), application_key=dict(required=True, no_log=True), application_secret=dict(required=True, no_log=True), consumer_key=dict(required=True, no_log=True), timeout=dict(default=120, type='int') ) ) if not HAS_OVH: module.fail_json(msg='ovh-api python module\ is required to run this module ') # Get parameters name = module.params.get('name') state = module.params.get('state') backend = module.params.get('backend') weight = long(module.params.get('weight')) probe = module.params.get('probe') timeout = module.params.get('timeout') # Connect to OVH API client = getOvhClient(module) # Check that the load balancing exists try: loadBalancings = client.get('/ip/loadBalancing') except APIError as apiError: module.fail_json( msg='Unable to call OVH api for getting the list of loadBalancing, \ check application key, secret, consumerkey and parameters. \ Error returned by OVH api was : {}'.format(apiError)) if name not in loadBalancings: module.fail_json(msg='IP LoadBalancing {} does not exist'.format(name)) # Check that no task is pending before going on try: if not waitForNoTask(client, name, timeout): module.fail_json( msg='Timeout of {} seconds while waiting for no pending \ tasks before executing the module '.format(timeout)) except APIError as apiError: module.fail_json( msg='Unable to call OVH api for getting the list of pending tasks \ of the loadBalancing, check application key, secret, consumerkey \ and parameters. Error returned by OVH api was : {}\ '.format(apiError)) try: backends = client.get('/ip/loadBalancing/{}/backend'.format(name)) except APIError as apiError: module.fail_json( msg='Unable to call OVH api for getting the list of backends \ of the loadBalancing, check application key, secret, consumerkey \ and parameters. Error returned by OVH api was : {}\ '.format(apiError)) backendExists = backend in backends moduleChanged = False if state == "absent": if backendExists: # Remove backend try: client.delete( '/ip/loadBalancing/{}/backend/{}'.format(name, backend)) if not waitForNoTask(client, name, timeout): module.fail_json( msg='Timeout of {} seconds while waiting for completion \ of removing backend task'.format(timeout)) except APIError as apiError: module.fail_json( msg='Unable to call OVH api for deleting the backend, \ check application key, secret, consumerkey and \ parameters. Error returned by OVH api was : {}\ '.format(apiError)) moduleChanged = True else: if backendExists: # Get properties try: backendProperties = client.get( '/ip/loadBalancing/{}/backend/{}'.format(name, backend)) except APIError as apiError: module.fail_json( msg='Unable to call OVH api for getting the backend properties, \ check application key, secret, consumerkey and \ parameters. Error returned by OVH api was : {}\ '.format(apiError)) if (backendProperties['weight'] != weight): # Change weight try: client.post( '/ip/loadBalancing/{}/backend/{}/setWeight\ '.format(name, backend), weight=weight) if not waitForNoTask(client, name, timeout): module.fail_json( msg='Timeout of {} seconds while waiting for completion \ of setWeight to backend task'.format(timeout)) except APIError as apiError: module.fail_json( msg='Unable to call OVH api for updating the weight of the \ backend, check application key, secret, consumerkey \ and parameters. Error returned by OVH api was : {}\ '.format(apiError)) moduleChanged = True if (backendProperties['probe'] != probe): # Change probe backendProperties['probe'] = probe try: client.put( '/ip/loadBalancing/{}/backend/{}\ '.format(name, backend), probe=probe) if not waitForNoTask(client, name, timeout): module.fail_json( msg='Timeout of {} seconds while waiting for completion of \ setProbe to backend task'.format(timeout)) except APIError as apiError: module.fail_json( msg='Unable to call OVH api for updating the propbe of \ the backend, check application key, secret, \ consumerkey and parameters. Error returned by OVH api \ was : {}\ '.format(apiError)) moduleChanged = True else: # Creates backend try: try: client.post('/ip/loadBalancing/{}/backend'.format(name), ipBackend=backend, probe=probe, weight=weight) except APIError as apiError: module.fail_json( msg='Unable to call OVH api for creating the backend, check \ application key, secret, consumerkey and parameters. \ Error returned by OVH api was : {}'.format(apiError)) if not waitForNoTask(client, name, timeout): module.fail_json( msg='Timeout of {} seconds while waiting for completion of \ backend creation task'.format(timeout)) except APIError as apiError: module.fail_json( msg='Unable to call OVH api for creating the backend, check \ application key, secret, consumerkey and parameters. \ Error returned by OVH api was : {}'.format(apiError)) moduleChanged = True module.exit_json(changed=moduleChanged) # We should never reach here module.fail_json(msg='Internal ovh_ip_loadbalancing_backend module error') if __name__ == '__main__': main()