modify_yaml.py 3.8 KB

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