mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2024-09-14 20:13:21 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			242 lines
		
	
	
	
		
			6.8 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			242 lines
		
	
	
	
		
			6.8 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| #!/usr/bin/python
 | |
| # -*- coding: utf-8 -*-
 | |
| 
 | |
| # 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 <http://www.gnu.org/licenses/>.
 | |
| DOCUMENTATION = '''
 | |
| ---
 | |
| module: digital_ocean_domain
 | |
| short_description: Create/delete a DNS record in DigitalOcean
 | |
| description:
 | |
|      - Create/delete a DNS record in DigitalOcean.
 | |
| version_added: "1.6"
 | |
| options:
 | |
|   state:
 | |
|     description:
 | |
|      - Indicate desired state of the target.
 | |
|     default: present
 | |
|     choices: ['present', 'active', 'absent', 'deleted']
 | |
|   client_id:
 | |
|      description:
 | |
|      - Digital Ocean manager id.
 | |
|   api_key:
 | |
|     description:
 | |
|      - Digital Ocean api key.
 | |
|   id:
 | |
|     description:
 | |
|      - Numeric, the droplet id you want to operate on.
 | |
|   name:
 | |
|     description:
 | |
|      - String, this is the name of the droplet - must be formatted by hostname rules, or the name of a SSH key, or the name of a domain.
 | |
|   ip:
 | |
|     description:
 | |
|      - The IP address to point a domain at.
 | |
| 
 | |
| notes:
 | |
|   - Two environment variables can be used, DO_CLIENT_ID and DO_API_KEY.
 | |
| '''
 | |
| 
 | |
| 
 | |
| EXAMPLES = '''
 | |
| # Create a domain record
 | |
| 
 | |
| - digital_ocean_domain: >
 | |
|       state=present
 | |
|       name=my.digitalocean.domain
 | |
|       ip=127.0.0.1
 | |
| 
 | |
| # Create a droplet and a corresponding domain record
 | |
| 
 | |
| - digital_cean_droplet: >
 | |
|       state=present
 | |
|       name=test_droplet
 | |
|       size_id=1
 | |
|       region_id=2
 | |
|       image_id=3
 | |
|   register: test_droplet
 | |
| 
 | |
| - digital_ocean_domain: >
 | |
|       state=present
 | |
|       name={{ test_droplet.name }}.my.domain
 | |
|       ip={{ test_droplet.ip_address }}
 | |
| '''
 | |
| 
 | |
| import sys
 | |
| import os
 | |
| import time
 | |
| 
 | |
| try:
 | |
|     from dopy.manager import DoError, DoManager
 | |
| except ImportError as e:
 | |
|     print "failed=True msg='dopy required for this module'"
 | |
|     sys.exit(1)
 | |
| 
 | |
| class TimeoutError(DoError):
 | |
|     def __init__(self, msg, id):
 | |
|         super(TimeoutError, self).__init__(msg)
 | |
|         self.id = id
 | |
| 
 | |
| class JsonfyMixIn(object):
 | |
|     def to_json(self):
 | |
|         return self.__dict__
 | |
| 
 | |
| class DomainRecord(JsonfyMixIn):
 | |
|     manager = None
 | |
| 
 | |
|     def __init__(self, json):
 | |
|         self.__dict__.update(json)
 | |
|     update_attr = __init__
 | |
| 
 | |
|     def update(self, data = None, record_type = None):
 | |
|         json = self.manager.edit_domain_record(self.domain_id,
 | |
|                                                self.id,
 | |
|                                                record_type if record_type is not None else self.record_type,
 | |
|                                                data if data is not None else self.data)
 | |
|         self.__dict__.update(json)
 | |
|         return self
 | |
| 
 | |
|     def destroy(self):
 | |
|         json = self.manager.destroy_domain_record(self.domain_id, self.id)
 | |
|         return json
 | |
| 
 | |
| class Domain(JsonfyMixIn):
 | |
|     manager = None
 | |
| 
 | |
|     def __init__(self, domain_json):
 | |
|         self.__dict__.update(domain_json)
 | |
| 
 | |
|     def destroy(self):
 | |
|         self.manager.destroy_domain(self.id)
 | |
| 
 | |
|     def records(self):
 | |
|         json = self.manager.all_domain_records(self.id)
 | |
|         return map(DomainRecord, json)
 | |
| 
 | |
|     @classmethod
 | |
|     def add(cls, name, ip):
 | |
|         json = cls.manager.new_domain(name, ip)
 | |
|         return cls(json)
 | |
| 
 | |
|     @classmethod
 | |
|     def setup(cls, client_id, api_key):
 | |
|         cls.manager = DoManager(client_id, api_key)
 | |
|         DomainRecord.manager = cls.manager
 | |
| 
 | |
|     @classmethod
 | |
|     def list_all(cls):
 | |
|         domains = cls.manager.all_domains()
 | |
|         return map(cls, domains)
 | |
| 
 | |
|     @classmethod
 | |
|     def find(cls, name=None, id=None):
 | |
|         if name is None and id is None:
 | |
|             return False
 | |
| 
 | |
|         domains = Domain.list_all()
 | |
|         
 | |
|         if id is not None:
 | |
|             for domain in domains:
 | |
|                 if domain.id == id:
 | |
|                     return domain
 | |
| 
 | |
|         if name is not None:
 | |
|             for domain in domains:
 | |
|                 if domain.name == name:
 | |
|                     return domain
 | |
| 
 | |
|         return False
 | |
| 
 | |
| def core(module):
 | |
|     def getkeyordie(k):
 | |
|         v = module.params[k]
 | |
|         if v is None:
 | |
|             module.fail_json(msg='Unable to load %s' % k)
 | |
|         return v
 | |
| 
 | |
|     try:
 | |
|         # params['client_id'] will be None even if client_id is not passed in
 | |
|         client_id = module.params['client_id'] or os.environ['DO_CLIENT_ID']
 | |
|         api_key = module.params['api_key'] or os.environ['DO_API_KEY']
 | |
|     except KeyError, e:
 | |
|         module.fail_json(msg='Unable to load %s' % e.message)
 | |
| 
 | |
|     changed = True
 | |
|     state = module.params['state']
 | |
| 
 | |
|     Domain.setup(client_id, api_key)
 | |
|     if state in ('present'):
 | |
|         domain = Domain.find(id=module.params["id"])
 | |
| 
 | |
|         if not domain:
 | |
|             domain = Domain.find(name=getkeyordie("name"))
 | |
|             
 | |
|         if not domain:
 | |
|             domain = Domain.add(getkeyordie("name"),
 | |
|                                 getkeyordie("ip"))
 | |
|             module.exit_json(changed=True, domain=domain.to_json())
 | |
|         else:
 | |
|             records = domain.records()
 | |
|             at_record = None
 | |
|             for record in records:
 | |
|                 if record.name == "@":
 | |
|                     at_record = record
 | |
| 
 | |
|             if not at_record.data == getkeyordie("ip"):
 | |
|                 record.update(data=getkeyordie("ip"), record_type='A')
 | |
|                 module.exit_json(changed=True, domain=Domain.find(id=record.domain_id).to_json())
 | |
| 
 | |
|         module.exit_json(changed=False, domain=domain.to_json())
 | |
| 
 | |
|     elif state in ('absent'):
 | |
|         domain = None
 | |
|         if "id" in module.params:
 | |
|             domain = Domain.find(id=module.params["id"])
 | |
|                 
 | |
|         if not domain and "name" in module.params:
 | |
|             domain = Domain.find(name=module.params["name"])
 | |
|                 
 | |
|         if not domain:
 | |
|             module.exit_json(changed=False, msg="Domain not found.")
 | |
| 
 | |
|         event_json = domain.destroy()
 | |
|         module.exit_json(changed=True, event=event_json)
 | |
| 
 | |
| 
 | |
| def main():
 | |
|     module = AnsibleModule(
 | |
|         argument_spec = dict(
 | |
|             state = dict(choices=['active', 'present', 'absent', 'deleted'], default='present'),
 | |
|             client_id = dict(aliases=['CLIENT_ID'], no_log=True),
 | |
|             api_key = dict(aliases=['API_KEY'], no_log=True),
 | |
|             name = dict(type='str'),
 | |
|             id = dict(aliases=['droplet_id'], type='int'),
 | |
|             ip = dict(type='str'),
 | |
|         ),
 | |
|         required_one_of = (
 | |
|             ['id', 'name'],
 | |
|         ),
 | |
|     )
 | |
| 
 | |
|     try:
 | |
|         core(module)
 | |
|     except TimeoutError as e:
 | |
|         module.fail_json(msg=str(e), id=e.id)
 | |
|     except (DoError, Exception) as e:
 | |
|         module.fail_json(msg=str(e))
 | |
| 
 | |
| # import module snippets
 | |
| from ansible.module_utils.basic import *
 | |
| 
 | |
| main()
 |