diff options
| author | Kenny Woodson <kwoodson@redhat.com> | 2016-03-18 13:02:37 -0400 | 
|---|---|---|
| committer | Kenny Woodson <kwoodson@redhat.com> | 2016-03-18 13:02:37 -0400 | 
| commit | cfbf424142e8d8fc5effea09ba58a87a935233a8 (patch) | |
| tree | bed797068e6992954827452ea23b06ceca2cd26e /roles | |
| parent | 03066664a45c12ed01b952da5a965246833cba33 (diff) | |
| parent | 0d288bb5f276c1ffe5c9433aabd052813b18b5b8 (diff) | |
| download | openshift-cfbf424142e8d8fc5effea09ba58a87a935233a8.tar.gz openshift-cfbf424142e8d8fc5effea09ba58a87a935233a8.tar.bz2 openshift-cfbf424142e8d8fc5effea09ba58a87a935233a8.tar.xz openshift-cfbf424142e8d8fc5effea09ba58a87a935233a8.zip | |
Merge pull request #1637 from kwoodson/yaml_edit
Yaml edit ansible module
Diffstat (limited to 'roles')
| -rw-r--r-- | roles/lib_yaml_editor/library/yedit.py | 220 | 
1 files changed, 220 insertions, 0 deletions
| diff --git a/roles/lib_yaml_editor/library/yedit.py b/roles/lib_yaml_editor/library/yedit.py new file mode 100644 index 000000000..b75aff109 --- /dev/null +++ b/roles/lib_yaml_editor/library/yedit.py @@ -0,0 +1,220 @@ +#!/usr/bin/env python +''' +module for openshift cloud secrets +''' + +import os +import yaml + +class YeditException(Exception): +    ''' Exception class for Yedit ''' +    pass + +class Yedit(object): +    ''' Class to modify yaml files ''' + +    def __init__(self, filename): +        self.filename = filename +        self.__yaml_dict = None +        self.get() + +    @property +    def yaml_dict(self): +        ''' getter method for yaml_dict ''' +        return self.__yaml_dict + +    @yaml_dict.setter +    def yaml_dict(self, value): +        ''' setter method for yaml_dict ''' +        self.__yaml_dict = value + +    @staticmethod +    def remove_entry(data, keys): +        ''' remove an item from a dictionary with key notation a.b.c +            d = {'a': {'b': 'c'}}} +            keys = a.b +            item = c +        ''' +        if "." in keys: +            key, rest = keys.split(".", 1) +            if key in data.keys(): +                Yedit.remove_entry(data[key], rest) +        else: +            del data[keys] + +    @staticmethod +    def add_entry(data, keys, item): +        ''' Add an item to a dictionary with key notation a.b.c +            d = {'a': {'b': 'c'}}} +            keys = a.b +            item = c +        ''' +        if "." in keys: +            key, rest = keys.split(".", 1) +            if key not in data: +                data[key] = {} + +            if not isinstance(data, dict): +                raise YeditException('Invalid add_entry called on a [%s] of type [%s].' % (data, type(data))) +            else: +                Yedit.add_entry(data[key], rest, item) + +        else: +            data[keys] = item + + +    @staticmethod +    def get_entry(data, keys): +        ''' Get an item from a dictionary with key notation a.b.c +            d = {'a': {'b': 'c'}}} +            keys = a.b +            return c +        ''' +        if keys and "." in keys: +            key, rest = keys.split(".", 1) +            if not isinstance(data[key], dict): +                raise YeditException('Invalid get_entry called on a [%s] of type [%s].' % (data, type(data))) + +            else: +                return Yedit.get_entry(data[key], rest) + +        else: +            return data.get(keys, None) + + +    def write(self): +        ''' write to file ''' +        with open(self.filename, 'w') as yfd: +            yfd.write(yaml.dump(self.yaml_dict, default_flow_style=False)) + +    def read(self): +        ''' write to file ''' +        # check if it exists +        if not self.exists(): +            return None + +        contents = None +        with open(self.filename) as yfd: +            contents = yfd.read() + +        return contents + +    def exists(self): +        ''' return whether file exists ''' +        if os.path.exists(self.filename): +            return True + +        return False +    def get(self): +        ''' return yaml file ''' +        contents = self.read() + +        if not contents: +            return None + +        # check if it is yaml +        try: +            self.yaml_dict = yaml.load(contents) +        except yaml.YAMLError as _: +            # Error loading yaml +            return None + +        return self.yaml_dict + +    def delete(self, key): +        ''' put key, value into a yaml file ''' +        try: +            entry = Yedit.get_entry(self.yaml_dict, key) +        except KeyError as _: +            entry = None +        if not entry: +            return  (False, self.yaml_dict) + +        Yedit.remove_entry(self.yaml_dict, key) +        self.write() +        return (True, self.get()) + +    def put(self, key, value): +        ''' put key, value into a yaml file ''' +        try: +            entry = Yedit.get_entry(self.yaml_dict, key) +        except KeyError as _: +            entry = None + +        if entry == value: +            return (False, self.yaml_dict) + +        Yedit.add_entry(self.yaml_dict, key, value) +        self.write() +        return (True, self.get()) + +    def create(self, key, value): +        ''' create the file ''' +        if not self.exists(): +            self.yaml_dict = {key: value} +            self.write() +            return (True, self.get()) + +        return (False, self.get()) + + +def main(): +    ''' +    ansible oc module for secrets +    ''' + +    module = AnsibleModule( +        argument_spec=dict( +            state=dict(default='present', type='str', +                       choices=['present', 'absent', 'list']), +            debug=dict(default=False, type='bool'), + +            src=dict(default=None, type='str'), +            key=dict(default=None, type='str'), +            value=dict(default=None, type='str'), +            value_format=dict(default='yaml', choices=['yaml', 'json'], type='str'), +        ), +        mutually_exclusive=[["contents", "files"]], + +        supports_check_mode=True, +    ) +    state = module.params['state'] + +    yamlfile = Yedit(module.params['src']) + +    rval = yamlfile.get() +    if not rval and state != 'present': +        module.fail_json(msg='Error opening file [%s].  Verify that the' + \ +                             ' file exists and that it is has correct permissions.') + +    if state == 'list': +        module.exit_json(changed=False, results=rval, state="list") + +    if state == 'absent': +        rval = yamlfile.delete(module.params['key']) +        module.exit_json(changed=rval[0], results=rval[1], state="absent") + +    if state == 'present': + +        if module.params['value_format'] == 'yaml': +            value = yaml.load(module.params['value']) +        elif module.params['value_format'] == 'json': +            value = json.loads(module.params['value']) + +        if rval: +            rval = yamlfile.put(module.params['key'], value) +            module.exit_json(changed=rval[0], results=rval[1], state="present") + +        rval = yamlfile.create(module.params['key'], value) +        module.exit_json(changed=rval[0], results=rval[1], state="present") + +    module.exit_json(failed=True, +                     changed=False, +                     results='Unknown state passed. %s' % state, +                     state="unknown") + +# pylint: disable=redefined-builtin, unused-wildcard-import, wildcard-import, locally-disabled +# import module snippets.  This are required +from ansible.module_utils.basic import * + +main() | 
