69a6acf7a8
This allows you to extend lists in yaml config. This is useful, for example, in prometheus.yml, where it would be nice to be able to extend the scrape_configs to include exporters that aren't packaged with kolla-ansible. This would provide a mechanism to do so. Change-Id: I7a10e363f42e8ffaae3c0d2c2a758853e2cab7e1 Related: blueprint custom-prometheus-targets
186 lines
5.9 KiB
Python
Executable File
186 lines
5.9 KiB
Python
Executable File
#!/usr/bin/env python
|
|
|
|
# Copyright 2015 Sam Yaple
|
|
# Copyright 2016 intel
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
# you may not use this file except in compliance with the License.
|
|
# You may obtain a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
# See the License for the specific language governing permissions and
|
|
# limitations under the License.
|
|
|
|
import os
|
|
import shutil
|
|
import tempfile
|
|
|
|
from yaml import dump
|
|
from yaml import safe_load
|
|
try:
|
|
from yaml import CDumper as Dumper # noqa: F401
|
|
from yaml import CLoader as Loader # noqa: F401
|
|
except ImportError:
|
|
from yaml import Dumper # noqa: F401
|
|
from yaml import Loader # noqa: F401
|
|
|
|
|
|
from ansible import constants
|
|
from ansible import errors as ansible_errors
|
|
from ansible.plugins import action
|
|
|
|
DOCUMENTATION = '''
|
|
---
|
|
module: merge_yaml
|
|
short_description: Merge yaml-style configs
|
|
description:
|
|
- PyYAML is used to merge several yaml files into one
|
|
options:
|
|
dest:
|
|
description:
|
|
- The destination file name
|
|
required: True
|
|
type: str
|
|
sources:
|
|
description:
|
|
- A list of files on the destination node to merge together
|
|
default: None
|
|
required: True
|
|
type: str
|
|
extend_lists:
|
|
description:
|
|
- For a given key referencing a list, this determines whether
|
|
the list items should be combined with the items in another
|
|
document if an equivalent key is found. An equivalent key
|
|
has the same parents and value as the first. The default
|
|
behaviour is to replace existing entries i.e if you have
|
|
two yaml documents that both define a list with an equivalent
|
|
key, the value from the document that appears later in the
|
|
list of sources will replace the value that appeared in the
|
|
earlier one.
|
|
default: False
|
|
required: False
|
|
type: bool
|
|
author: Sean Mooney
|
|
'''
|
|
|
|
EXAMPLES = '''
|
|
Merge multiple yaml files:
|
|
|
|
- hosts: localhost
|
|
tasks:
|
|
- name: Merge yaml files
|
|
merge_yaml:
|
|
sources:
|
|
- "/tmp/default.yml"
|
|
- "/tmp/override.yml"
|
|
dest:
|
|
- "/tmp/out.yml"
|
|
'''
|
|
|
|
|
|
class ActionModule(action.ActionBase):
|
|
|
|
TRANSFERS_FILES = True
|
|
|
|
def read_config(self, source):
|
|
result = None
|
|
# Only use config if present
|
|
if os.access(source, os.R_OK):
|
|
with open(source, 'r') as f:
|
|
template_data = f.read()
|
|
|
|
# set search path to mimic 'template' module behavior
|
|
searchpath = [
|
|
self._loader._basedir,
|
|
os.path.join(self._loader._basedir, 'templates'),
|
|
os.path.dirname(source),
|
|
]
|
|
self._templar.environment.loader.searchpath = searchpath
|
|
|
|
template_data = self._templar.template(template_data)
|
|
result = safe_load(template_data)
|
|
return result or {}
|
|
|
|
def run(self, tmp=None, task_vars=None):
|
|
if task_vars is None:
|
|
task_vars = dict()
|
|
result = super(ActionModule, self).run(tmp, task_vars)
|
|
del tmp # not used
|
|
|
|
# save template args.
|
|
extra_vars = self._task.args.get('vars', list())
|
|
old_vars = self._templar._available_variables
|
|
|
|
temp_vars = task_vars.copy()
|
|
temp_vars.update(extra_vars)
|
|
self._templar.set_available_variables(temp_vars)
|
|
|
|
output = {}
|
|
sources = self._task.args.get('sources', None)
|
|
extend_lists = self._task.args.get('extend_lists', False)
|
|
if not isinstance(sources, list):
|
|
sources = [sources]
|
|
for source in sources:
|
|
Utils.update_nested_conf(
|
|
output, self.read_config(source), extend_lists)
|
|
|
|
# restore original vars
|
|
self._templar.set_available_variables(old_vars)
|
|
|
|
local_tempdir = tempfile.mkdtemp(dir=constants.DEFAULT_LOCAL_TMP)
|
|
|
|
try:
|
|
result_file = os.path.join(local_tempdir, 'source')
|
|
with open(result_file, 'w') as f:
|
|
f.write(dump(output, default_flow_style=False))
|
|
|
|
new_task = self._task.copy()
|
|
new_task.args.pop('sources', None)
|
|
new_task.args.pop('extend_lists', None)
|
|
new_task.args.update(
|
|
dict(
|
|
src=result_file
|
|
)
|
|
)
|
|
|
|
copy_action = self._shared_loader_obj.action_loader.get(
|
|
'copy',
|
|
task=new_task,
|
|
connection=self._connection,
|
|
play_context=self._play_context,
|
|
loader=self._loader,
|
|
templar=self._templar,
|
|
shared_loader_obj=self._shared_loader_obj)
|
|
result.update(copy_action.run(task_vars=task_vars))
|
|
finally:
|
|
shutil.rmtree(local_tempdir)
|
|
return result
|
|
|
|
|
|
class Utils(object):
|
|
@staticmethod
|
|
def update_nested_conf(conf, update, extend_lists=False):
|
|
for k, v in update.items():
|
|
if isinstance(v, dict):
|
|
conf[k] = Utils.update_nested_conf(
|
|
conf.get(k, {}), v, extend_lists)
|
|
elif k in conf and isinstance(conf[k], list) and extend_lists:
|
|
if not isinstance(v, list):
|
|
errmsg = (
|
|
"Failure merging key `%(key)s` in dictionary "
|
|
"`%(dictionary)s`. Expecting a list, but received: "
|
|
"`%(value)s`, which is of type: `%(type)s`" % {
|
|
"key": k, "dictionary": conf,
|
|
"value": v, "type": type(v)}
|
|
)
|
|
raise ansible_errors.AnsibleModuleError(errmsg)
|
|
conf[k].extend(v)
|
|
else:
|
|
conf[k] = v
|
|
return conf
|