modify_yaml.py 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117
  1. #!/usr/bin/python
  2. # -*- coding: utf-8 -*-
  3. ''' modify_yaml ansible module '''
  4. import yaml
  5. # ignore pylint errors related to the module_utils import
  6. # pylint: disable=redefined-builtin, unused-wildcard-import, wildcard-import
  7. from ansible.module_utils.basic import * # noqa: F402,F403
  8. DOCUMENTATION = '''
  9. ---
  10. module: modify_yaml
  11. short_description: Modify yaml key value pairs
  12. author: Andrew Butcher
  13. requirements: [ ]
  14. '''
  15. EXAMPLES = '''
  16. - modify_yaml:
  17. dest: /etc/origin/master/master-config.yaml
  18. yaml_key: 'kubernetesMasterConfig.masterCount'
  19. yaml_value: 2
  20. '''
  21. def set_key(yaml_data, yaml_key, yaml_value):
  22. ''' Updates a parsed yaml structure setting a key to a value.
  23. :param yaml_data: yaml structure to modify.
  24. :type yaml_data: dict
  25. :param yaml_key: Key to modify.
  26. :type yaml_key: mixed
  27. :param yaml_value: Value use for yaml_key.
  28. :type yaml_value: mixed
  29. :returns: Changes to the yaml_data structure
  30. :rtype: dict(tuple())
  31. '''
  32. changes = []
  33. ptr = yaml_data
  34. final_key = yaml_key.split('.')[-1]
  35. for key in yaml_key.split('.'):
  36. # Key isn't present and we're not on the final key. Set to empty dictionary.
  37. if key not in ptr and key != final_key:
  38. ptr[key] = {}
  39. ptr = ptr[key]
  40. # Current key is the final key. Update value.
  41. elif key == final_key:
  42. if (key in ptr and module.safe_eval(ptr[key]) != yaml_value) or (key not in ptr): # noqa: F405
  43. ptr[key] = yaml_value
  44. changes.append((yaml_key, yaml_value))
  45. else:
  46. # Next value is None and we're not on the final key.
  47. # Turn value into an empty dictionary.
  48. if ptr[key] is None and key != final_key:
  49. ptr[key] = {}
  50. ptr = ptr[key]
  51. return changes
  52. def main():
  53. ''' Modify key (supplied in jinja2 dot notation) in yaml file, setting
  54. the key to the desired value.
  55. '''
  56. # disabling pylint errors for global-variable-undefined and invalid-name
  57. # for 'global module' usage, since it is required to use ansible_facts
  58. # pylint: disable=global-variable-undefined, invalid-name,
  59. # redefined-outer-name
  60. global module
  61. module = AnsibleModule( # noqa: F405
  62. argument_spec=dict(
  63. dest=dict(required=True),
  64. yaml_key=dict(required=True),
  65. yaml_value=dict(required=True),
  66. backup=dict(required=False, default=True, type='bool'),
  67. ),
  68. supports_check_mode=True,
  69. )
  70. dest = module.params['dest']
  71. yaml_key = module.params['yaml_key']
  72. yaml_value = module.safe_eval(module.params['yaml_value'])
  73. backup = module.params['backup']
  74. # Represent null values as an empty string.
  75. # pylint: disable=missing-docstring, unused-argument
  76. def none_representer(dumper, data):
  77. return yaml.ScalarNode(tag=u'tag:yaml.org,2002:null', value=u'')
  78. yaml.add_representer(type(None), none_representer)
  79. try:
  80. with open(dest) as yaml_file:
  81. yaml_data = yaml.safe_load(yaml_file.read())
  82. changes = set_key(yaml_data, yaml_key, yaml_value)
  83. if len(changes) > 0:
  84. if backup:
  85. module.backup_local(dest)
  86. with open(dest, 'w') as yaml_file:
  87. yaml_string = yaml.dump(yaml_data, default_flow_style=False)
  88. yaml_string = yaml_string.replace('\'\'', '""')
  89. yaml_file.write(yaml_string)
  90. return module.exit_json(changed=(len(changes) > 0), changes=changes)
  91. # ignore broad-except error to avoid stack trace to ansible user
  92. # pylint: disable=broad-except
  93. except Exception as error:
  94. return module.fail_json(msg=str(error))
  95. if __name__ == '__main__':
  96. main()