blob: 9977551e8b3d36dc3138291067eaad8472cae909 [file] [log] [blame]
import difflib
import os
import logging
from salt.exceptions import CommandExecutionError
from salt.serializers import yaml
LOG = logging.getLogger(__name__)
def _get_values_from_file(values_file=None):
if values_file:
try:
with open(values_file) as values_stream:
values = yaml.deserialize(values_stream)
return values
except e:
raise CommandExecutionError("encountered error reading from values "
"file (%s): %s" % (values_file, e))
return None
def _get_yaml_diff(new_yaml=None, old_yaml=None):
if not new_yaml and not old_yaml:
return None
old_str = yaml.serialize(old_yaml, default_flow_style=False)
new_str = yaml.serialize(new_yaml, default_flow_style=False)
return difflib.unified_diff(old_str.split('\n'), new_str.split('\n'))
def _failure(name, message, changes={}):
return {
'name': name,
'changes': changes,
'result': False,
'comment': message,
}
def present(name, chart_name, namespace, version=None, values_file=None,
tiller_namespace='kube-system', **kwargs):
'''
Ensure that a release with the supplied name is in the desired state in the
Tiller installation. This state will handle change detection to determine
whether an installation or update needs to be made.
In the event the namespace to which a release is installed changes, the
state will first delete and purge the release and the re-install it into
the new namespace, since Helm does not support updating a release into a
new namespace.
name
The name of the release to ensure is present
chart_name
The name of the chart to install, including the repository name as
applicable (such as `stable/mysql`)
namespace
The namespace to which the release should be (re-)installed
version
The version of the chart to install. Defaults to the latest version
values_file
The path to the a values file containing all the chart values that
should be applied to the release. Note that this should not be passed
if there are not chart value overrides required.
'''
kwargs['tiller_namespace'] = tiller_namespace
if __opts__['test'] == True:
return {
'result': None,
'name': name,
'comment': 'Helm chart "{0}" will be installed'.format(chart_name),
'changes': {
'chart_name': chart_name,
'namespace': namespace,
}
}
old_release = __salt__['k8s_helm.get_release'](name, **kwargs)
if not old_release:
try:
result = __salt__['k8s_helm.release_create'](
name, chart_name, namespace, version, values_file, **kwargs
)
return {
'name': name,
'changes': {
'name': name,
'chart_name': chart_name,
'namespace': namespace,
'version': version,
'values': _get_values_from_file(values_file),
'stdout': result.get('stdout')
},
'result': True,
'comment': ('Release "%s" was created' % name +
'\nExecuted command: %s' % result['cmd'])
}
except CommandExecutionError as e:
msg = (("Failed to create new release: %s" % e.error) +
"\nExecuted command: %s" % e.cmd)
return _failure(name, msg)
changes = {}
warnings = []
if old_release.get('chart') != chart_name.split("/")[1]:
changes['chart'] = { 'old': old_release['chart'], 'new': chart_name }
if old_release.get('version') != version:
changes['version'] = { 'old': old_release['version'], 'new': version }
if old_release.get('namespace') != namespace:
changes['namespace'] = { 'old': old_release['namespace'], 'new': namespace }
if (not values_file and old_release.get("values") or
not old_release.get("values") and values_file):
changes['values'] = { 'old': old_release['values'], 'new': values_file }
values = _get_values_from_file(values_file)
diff = _get_yaml_diff(values, old_release.get('values'))
if diff:
diff_string = '\n'.join(diff)
if diff_string:
changes['values'] = diff_string
if not changes:
return {
'name': name,
'result': True,
'changes': {},
'comment': 'Release "{}" is already in the desired state'.format(name)
}
module_fn = 'k8s_helm.release_upgrade'
if changes.get("namespace"):
LOG.debug("purging old release (%s) due to namespace change" % name)
try:
result = __salt__['k8s_helm.release_delete'](name, **kwargs)
except CommandExecutionError as e:
msg = ("Failed to delete release for namespace change: %s" % e.error +
"\nExecuted command: %s" % e.cmd)
return _failure(name, msg, changes)
module_fn = 'k8s_helm.release_create'
warnings.append('Release (%s) was replaced due to namespace change' % name)
try:
result = __salt__[module_fn](
name, chart_name, namespace, version, values_file, **kwargs
)
changes.update({ 'stdout': result.get('stdout') })
ret = {
'name': name,
'changes': changes,
'result': True,
'comment': 'Release "%s" was updated\nExecuted command: %s' % (name, result['cmd'])
}
if warnings:
ret['warnings'] = warnings
return ret
except CommandExecutionError as e:
msg = ("Failed to delete release for namespace change: %s" % e.error +
"\nExecuted command: %s" % e.cmd)
return _failure(name, msg, changes)
def absent(name, tiller_namespace='kube-system', **kwargs):
'''
Ensure that any release with the supplied release name is absent from the
tiller installation.
name
The name of the release to ensure is absent
'''
kwargs['tiller_namespace'] = tiller_namespace
exists = __salt__['k8s_helm.release_exists'](name, **kwargs)
if not exists:
return {
'name': name,
'changes': {},
'result': True,
'comment': 'Release "%s" doesn\'t exist' % name
}
try:
result = __salt__['k8s_helm.release_delete'](name, **kwargs)
return {
'name': name,
'changes': { name: 'DELETED', 'stdout': result['stdout'] },
'result': True,
'comment': 'Release "%s" was deleted\nExecuted command: %s' % (name, result['cmd'])
}
except CommandExecutionError as e:
return _failure(e.cmd, e.error)