dyn_record.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351
  1. #!/usr/bin/python
  2. #
  3. # (c) 2015, Russell Harrison <rharriso@redhat.com>
  4. #
  5. # Licensed under the Apache License, Version 2.0 (the "License");
  6. # you may not use this file except in compliance with the License.
  7. # You may obtain a copy of the License at
  8. #
  9. # http://www.apache.org/licenses/LICENSE-2.0
  10. #
  11. # Unless required by applicable law or agreed to in writing, software
  12. # distributed under the License is distributed on an "AS IS" BASIS,
  13. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. # See the License for the specific language governing permissions and
  15. # limitations under the License.
  16. # pylint: disable=too-many-branches
  17. '''Ansible module to manage records in the Dyn Managed DNS service'''
  18. DOCUMENTATION = '''
  19. ---
  20. module: dyn_record
  21. version_added: "1.9"
  22. short_description: Manage records in the Dyn Managed DNS service.
  23. description:
  24. - "Manages DNS records via the REST API of the Dyn Managed DNS service. It
  25. - "handles records only; there is no manipulation of zones or account support"
  26. - "yet. See: U(https://help.dyn.com/dns-api-knowledge-base/)"
  27. options:
  28. state:
  29. description:
  30. -"Whether the record should be c(present) or c(absent). Optionally the"
  31. - "state c(list) can be used to return the current value of a record."
  32. required: true
  33. choices: [ 'present', 'absent', 'list' ]
  34. default: present
  35. customer_name:
  36. description:
  37. - "The Dyn customer name for your account. If not set the value of the"
  38. - "c(DYNECT_CUSTOMER_NAME) environment variable is used."
  39. required: false
  40. default: nil
  41. user_name:
  42. description:
  43. - "The Dyn user name to log in with. If not set the value of the"
  44. - "c(DYNECT_USER_NAME) environment variable is used."
  45. required: false
  46. default: null
  47. user_password:
  48. description:
  49. - "The Dyn user's password to log in with. If not set the value of the"
  50. - "c(DYNECT_PASSWORD) environment variable is used."
  51. required: false
  52. default: null
  53. zone:
  54. description:
  55. - "The DNS zone in which your record is located."
  56. required: true
  57. default: null
  58. record_fqdn:
  59. description:
  60. - "Fully qualified domain name of the record name to get, create, delete,"
  61. - "or update."
  62. required: true
  63. default: null
  64. record_type:
  65. description:
  66. - "Record type."
  67. required: true
  68. choices: [ 'A', 'AAAA', 'CNAME', 'PTR', 'TXT' ]
  69. default: null
  70. record_value:
  71. description:
  72. - "Record value. If record_value is not specified; no changes will be"
  73. - "made and the module will fail"
  74. required: false
  75. default: null
  76. record_ttl:
  77. description:
  78. - 'Record's "Time to live". Number of seconds the record remains cached'
  79. - 'in DNS servers or c(0) to use the default TTL for the zone.'
  80. - 'This option is mutually exclusive with use_zone_ttl'
  81. required: false
  82. default: 0
  83. use_zone_ttl:
  84. description:
  85. - 'Use the DYN Zone's Default TTL'
  86. - 'This option is mutually exclusive with record_ttl'
  87. required: false
  88. default: false
  89. mutually exclusive with: record_ttl
  90. notes:
  91. - The module makes a broad assumption that there will be only one record per "node" (FQDN).
  92. - This module returns record(s) in the "result" element when 'state' is set to 'present'. This value can be be registered and used in your playbooks.
  93. requirements: [ dyn ]
  94. author: "Russell Harrison"
  95. '''
  96. EXAMPLES = '''
  97. # Attempting to cname www.example.com to web1.example.com
  98. - name: Update CNAME record
  99. dyn_record:
  100. state: present
  101. record_fqdn: www.example.com
  102. zone: example.com
  103. record_type: CNAME
  104. record_value: web1.example.com
  105. record_ttl: 7200
  106. # Use the zones default TTL
  107. - name: Update CNAME record
  108. dyn_record:
  109. state: present
  110. record_fqdn: www.example.com
  111. zone: example.com
  112. record_type: CNAME
  113. record_value: web1.example.com
  114. use_zone_ttl: true
  115. - name: Update A record
  116. dyn_record:
  117. state: present
  118. record_fqdn: web1.example.com
  119. zone: example.com
  120. record_value: 10.0.0.10
  121. record_type: A
  122. '''
  123. try:
  124. IMPORT_ERROR = False
  125. from dyn.tm.session import DynectSession
  126. from dyn.tm.zones import Zone
  127. import dyn.tm.errors
  128. import os
  129. except ImportError as error:
  130. IMPORT_ERROR = str(error)
  131. # Each of the record types use a different method for the value.
  132. RECORD_PARAMS = {
  133. 'A' : {'value_param': 'address'},
  134. 'AAAA' : {'value_param': 'address'},
  135. 'CNAME' : {'value_param': 'cname'},
  136. 'PTR' : {'value_param': 'ptrdname'},
  137. 'TXT' : {'value_param': 'txtdata'}
  138. }
  139. # You'll notice that the value_param doesn't match the key (records_key)
  140. # in the dict returned from Dyn when doing a dyn_node.get_all_records()
  141. # This is a frustrating lookup dict to allow mapping to the RECORD_PARAMS
  142. # dict so we can lookup other values in it efficiently
  143. def get_record_type(record_key):
  144. '''Get the record type represented by the keys returned from get_any_records.'''
  145. return record_key.replace('_records', '').upper()
  146. def get_record_key(record_type):
  147. '''Get the key to look up records in the dictionary returned from get_any_records.
  148. example:
  149. 'cname_records'
  150. '''
  151. return record_type.lower() + '_records'
  152. def get_any_records(module, node):
  153. '''Get any records for a given node'''
  154. # Lets get a list of the A records for the node
  155. try:
  156. records = node.get_any_records()
  157. except dyn.tm.errors.DynectGetError as error:
  158. if 'Not in zone' in str(error):
  159. # The node isn't in the zone so we'll return an empty dictionary
  160. return {}
  161. else:
  162. # An unknown error happened so we'll need to return it.
  163. module.fail_json(msg='Unable to get records',
  164. error=str(error))
  165. # Return a dictionary of the record objects
  166. return records
  167. def get_record_values(records):
  168. '''Get the record values for each record returned by get_any_records.'''
  169. # This simply returns the values from a record
  170. ret_dict = {}
  171. for key in records.keys():
  172. record_type = get_record_type(key)
  173. params = [RECORD_PARAMS[record_type]['value_param'], 'ttl', 'zone', 'fqdn']
  174. ret_dict[key] = []
  175. properties = {}
  176. for elem in records[key]:
  177. for param in params:
  178. properties[param] = getattr(elem, param)
  179. ret_dict[key].append(properties)
  180. return ret_dict
  181. def compare_record_values(record_type_key, user_record_value, dyn_values):
  182. ''' Verify the user record_value exists in dyn'''
  183. rtype = get_record_type(record_type_key)
  184. for record in dyn_values[record_type_key]:
  185. if user_record_value in record[RECORD_PARAMS[rtype]['value_param']]:
  186. return True
  187. return False
  188. def compare_record_ttl(record_type_key, user_record_value, dyn_values, user_param_ttl):
  189. ''' Verify the ttls match for the record'''
  190. rtype = get_record_type(record_type_key)
  191. for record in dyn_values[record_type_key]:
  192. # find the right record
  193. if user_record_value in record[RECORD_PARAMS[rtype]['value_param']]:
  194. # Compare ttls from the records
  195. if int(record['ttl']) == user_param_ttl:
  196. return True
  197. return False
  198. def main():
  199. '''Ansible module for managing Dyn DNS records.'''
  200. module = AnsibleModule(
  201. argument_spec=dict(
  202. state=dict(default='present', choices=['present', 'absent', 'list']),
  203. customer_name=dict(default=os.environ.get('DYNECT_CUSTOMER_NAME', None), type='str'),
  204. user_name=dict(default=os.environ.get('DYNECT_USER_NAME', None), type='str', no_log=True),
  205. user_password=dict(default=os.environ.get('DYNECT_PASSWORD', None), type='str', no_log=True),
  206. zone=dict(required=True, type='str'),
  207. record_fqdn=dict(required=False, type='str'),
  208. record_type=dict(required=False, type='str', choices=[
  209. 'A', 'AAAA', 'CNAME', 'PTR', 'TXT']),
  210. record_value=dict(required=False, type='str'),
  211. record_ttl=dict(required=False, default=None, type='int'),
  212. use_zone_ttl=dict(required=False, default=False),
  213. ),
  214. required_together=(
  215. ['record_fqdn', 'record_value', 'record_ttl', 'record_type']
  216. ),
  217. mutually_exclusive=[('record_ttl', 'use_zone_ttl')]
  218. )
  219. if IMPORT_ERROR:
  220. module.fail_json(msg="Unable to import dyn module: https://pypi.python.org/pypi/dyn", error=IMPORT_ERROR)
  221. if module.params['record_ttl'] != None and int(module.params['record_ttl']) <= 0:
  222. module.fail_json(msg="Invalid Value for record TTL")
  223. # Start the Dyn session
  224. try:
  225. _ = DynectSession(module.params['customer_name'],
  226. module.params['user_name'],
  227. module.params['user_password'])
  228. except dyn.tm.errors.DynectAuthError as error:
  229. module.fail_json(msg='Unable to authenticate with Dyn', error=str(error))
  230. # Retrieve zone object
  231. try:
  232. dyn_zone = Zone(module.params['zone'])
  233. except dyn.tm.errors.DynectGetError as error:
  234. if 'No such zone' in str(error):
  235. module.fail_json(msg="Not a valid zone for this account", zone=module.params['zone'])
  236. else:
  237. module.fail_json(msg="Unable to retrieve zone", error=str(error))
  238. # To retrieve the node object we need to remove the zone name from the FQDN
  239. dyn_node_name = module.params['record_fqdn'].replace('.' + module.params['zone'], '')
  240. # Retrieve the zone object from dyn
  241. dyn_zone = Zone(module.params['zone'])
  242. # Retrieve the node object from dyn
  243. dyn_node = dyn_zone.get_node(node=dyn_node_name)
  244. # All states will need a list of the exiting records for the zone.
  245. dyn_node_records = get_any_records(module, dyn_node)
  246. dyn_values = get_record_values(dyn_node_records)
  247. if module.params['state'] == 'list':
  248. module.exit_json(changed=False, dyn_records=dyn_values)
  249. elif module.params['state'] == 'absent':
  250. # If there are any records present we'll want to delete the node.
  251. if dyn_node_records:
  252. dyn_node.delete()
  253. # Publish the zone since we've modified it.
  254. dyn_zone.publish()
  255. module.exit_json(changed=True, msg="Removed node %s from zone %s" % (dyn_node_name, module.params['zone']))
  256. module.exit_json(changed=False)
  257. elif module.params['state'] == 'present':
  258. # configure the TTL variable:
  259. # if use_zone_ttl, use the default TTL of the account.
  260. # if TTL == None, don't check it, set it as 0 (api default)
  261. # if TTL > 0, ensure this TTL is set
  262. if module.params['use_zone_ttl']:
  263. user_param_ttl = dyn_zone.ttl
  264. elif not module.params['record_ttl']:
  265. user_param_ttl = 0
  266. else:
  267. user_param_ttl = module.params['record_ttl']
  268. # First get a list of existing records for the node
  269. record_type_key = get_record_key(module.params['record_type'])
  270. user_record_value = module.params['record_value']
  271. # Check to see if the record is already in place before doing anything.
  272. if dyn_node_records and compare_record_values(record_type_key, user_record_value, dyn_values):
  273. if user_param_ttl == 0 or \
  274. compare_record_ttl(record_type_key, user_record_value, dyn_values, user_param_ttl):
  275. module.exit_json(changed=False, dyn_record=dyn_values)
  276. # Working on the assumption that there is only one record per
  277. # node we will first delete the node if there are any records before
  278. # creating the correct record
  279. if dyn_node_records:
  280. dyn_node.delete()
  281. # Now lets create the correct node entry.
  282. record = dyn_zone.add_record(dyn_node_name,
  283. module.params['record_type'],
  284. module.params['record_value'],
  285. user_param_ttl
  286. )
  287. # Now publish the zone since we've updated it.
  288. dyn_zone.publish()
  289. rmsg = "Created node [%s] " % dyn_node_name
  290. rmsg += "in zone: [%s]" % module.params['zone']
  291. module.exit_json(changed=True, msg=rmsg, dyn_record=get_record_values({record_type_key: [record]}))
  292. module.fail_json(msg="Unknown state: [%s]" % module.params['state'])
  293. # Ansible tends to need a wild card import so we'll use it here
  294. # pylint: disable=redefined-builtin, unused-wildcard-import, wildcard-import, locally-disabled
  295. from ansible.module_utils.basic import *
  296. if __name__ == '__main__':
  297. main()