A hook which invokes os-apply-config.
The intent is for this element (hook script) to be used in place of the one in tripleo-image-elements which relies on an external signal handling shell script at the end of the os-refresh-config run (99-refresh-completed). This version will run os-apply-config and return a signal immediately. Because it uses the heat-hook mechanisms it also supports a broader set of signal handling capabilities... which 99-refresh-completed doesn't fully support. Change-Id: Ic9402ff93cbc840bec1debcd8881de563d03cbf0
This commit is contained in:
parent
f123aa1491
commit
7c5aae183a
@ -0,0 +1,14 @@
|
||||
A hook which invokes os-apply-config.
|
||||
|
||||
The intent is for this element (hook script) to be used in place of the one in
|
||||
tripleo-image-elements which relies on an external signal handling
|
||||
shell script at the end of the os-refresh-config run (99-refresh-completed).
|
||||
This version will run os-apply-config and return a signal immediately. Because
|
||||
it uses the heat-hook mechanisms it also supports a broader set of signal
|
||||
handling capabilities... which 99-refresh-completed doesn't fully support.
|
||||
|
||||
It is worth noting that this hook runs os-apply-config against all the
|
||||
accumulated metadata, not just data supplied to an individual hook.
|
||||
|
||||
To use this hook set group: to 'apply-config' instead of 'os-apply-config'
|
||||
in your Heat software configuration resources.
|
@ -0,0 +1 @@
|
||||
heat-config
|
@ -0,0 +1,6 @@
|
||||
#!/bin/bash
|
||||
set -x
|
||||
|
||||
SCRIPTDIR=$(dirname $0)
|
||||
|
||||
install -D -g root -o root -m 0755 ${SCRIPTDIR}/hook-apply-config.py /var/lib/heat-config/hooks/apply-config
|
@ -0,0 +1,57 @@
|
||||
#!/usr/bin/env python
|
||||
#
|
||||
# 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 json
|
||||
import logging
|
||||
import os
|
||||
import subprocess
|
||||
import sys
|
||||
|
||||
APPLY_CONFIG_CMD = os.environ.get('HEAT_APPLY_CONFIG_CMD', 'os-apply-config')
|
||||
|
||||
|
||||
def main(argv=sys.argv):
|
||||
log = logging.getLogger('heat-config')
|
||||
handler = logging.StreamHandler(sys.stderr)
|
||||
handler.setFormatter(
|
||||
logging.Formatter(
|
||||
'[%(asctime)s] (%(name)s) [%(levelname)s] %(message)s'))
|
||||
log.addHandler(handler)
|
||||
log.setLevel('DEBUG')
|
||||
|
||||
env = os.environ.copy()
|
||||
|
||||
log.debug('Running %s' % APPLY_CONFIG_CMD)
|
||||
subproc = subprocess.Popen([APPLY_CONFIG_CMD], stdout=subprocess.PIPE,
|
||||
stderr=subprocess.PIPE, env=env)
|
||||
stdout, stderr = subproc.communicate()
|
||||
|
||||
log.info(stdout)
|
||||
log.debug(stderr)
|
||||
|
||||
if subproc.returncode:
|
||||
log.error("Error running apply-config: [%s]\n" % subproc.returncode)
|
||||
else:
|
||||
log.info('Completed apply-config.')
|
||||
|
||||
response = {
|
||||
'deploy_stdout': stdout,
|
||||
'deploy_stderr': stderr,
|
||||
'deploy_status_code': subproc.returncode,
|
||||
}
|
||||
|
||||
json.dump(response, sys.stdout)
|
||||
|
||||
if __name__ == '__main__':
|
||||
sys.exit(main(sys.argv))
|
@ -12,6 +12,7 @@ testrepository>=0.0.18
|
||||
testscenarios>=0.4
|
||||
testtools>=0.9.34
|
||||
yamllint>=1.2.0
|
||||
os-apply-config
|
||||
|
||||
python-heatclient>=1.2.0
|
||||
python-keystoneclient>=0.10.0
|
||||
|
@ -25,7 +25,8 @@ from tests.software_config import common
|
||||
|
||||
class HeatConfigTest(common.RunScriptTest):
|
||||
|
||||
fake_hooks = ['cfn-init', 'chef', 'puppet', 'salt', 'script']
|
||||
fake_hooks = ['cfn-init', 'chef', 'puppet', 'salt', 'script',
|
||||
'apply-config']
|
||||
|
||||
data = [
|
||||
{
|
||||
@ -66,9 +67,14 @@ class HeatConfigTest(common.RunScriptTest):
|
||||
'config': 'five'
|
||||
}, {
|
||||
'id': '6666',
|
||||
'group': 'apply-config',
|
||||
'inputs': [{'name': 'foo', 'value': 'bar'}],
|
||||
'config': 'six'
|
||||
}, {
|
||||
'id': '9999',
|
||||
'group': 'no-such-hook',
|
||||
'inputs': [],
|
||||
'config': 'six'
|
||||
'config': 'nine'
|
||||
}]
|
||||
|
||||
outputs = {
|
||||
@ -97,6 +103,11 @@ class HeatConfigTest(common.RunScriptTest):
|
||||
'deploy_status_code': '-1',
|
||||
'deploy_stderr': 'A bad thing happened',
|
||||
'deploy_stdout': 'stdout'
|
||||
},
|
||||
'apply-config': {
|
||||
'deploy_status_code': '0',
|
||||
'deploy_stderr': 'stderr',
|
||||
'deploy_stdout': 'stdout'
|
||||
}
|
||||
}
|
||||
|
||||
|
81
tests/software_config/test_hook_apply_config.py
Normal file
81
tests/software_config/test_hook_apply_config.py
Normal file
@ -0,0 +1,81 @@
|
||||
#
|
||||
# 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 fixtures
|
||||
import json
|
||||
import logging
|
||||
import os
|
||||
import tempfile
|
||||
import yaml
|
||||
|
||||
from tests.software_config import common
|
||||
|
||||
log = logging.getLogger('test_hook_apply_config')
|
||||
|
||||
|
||||
class HookApplyConfigTest(common.RunScriptTest):
|
||||
|
||||
data = {
|
||||
'id': 'test_apply_config',
|
||||
'name': 'fake_resource_name',
|
||||
'group': 'apply-config',
|
||||
'config': {'foo': 'bar'}
|
||||
}
|
||||
|
||||
def setUp(self):
|
||||
super(HookApplyConfigTest, self).setUp()
|
||||
self.hook_path = self.relative_path(
|
||||
__file__,
|
||||
'../..',
|
||||
'hot/software-config/elements',
|
||||
'heat-config-apply-config/install.d/hook-apply-config.py')
|
||||
|
||||
self.metadata_dir = self.useFixture(fixtures.TempDir())
|
||||
self.templates_dir = self.useFixture(fixtures.TempDir())
|
||||
tmp_dir = tempfile.NamedTemporaryFile(mode='w', delete=False).name
|
||||
os.unlink(tmp_dir)
|
||||
self.tmp_file = os.path.basename(tmp_dir)
|
||||
self.out_dir = self.templates_dir.join('tmp')
|
||||
|
||||
self.metadata = self.metadata_dir.join(self.tmp_file)
|
||||
|
||||
self.env = os.environ.copy()
|
||||
self.env.update({
|
||||
'OS_CONFIG_FILES': self.metadata,
|
||||
'OS_CONFIG_APPLIER_TEMPLATES': self.templates_dir.join(),
|
||||
})
|
||||
|
||||
# our fake metadata file
|
||||
with open(self.metadata, "w+") as md:
|
||||
md.write(json.dumps({'foo': 'bar'}))
|
||||
|
||||
# This is our fake template root we use to verify os-apply-config
|
||||
# works as expected
|
||||
os.mkdir(self.out_dir)
|
||||
with open(os.path.join(self.out_dir, self.tmp_file), "w+") as template:
|
||||
template.write("foo={{foo}}")
|
||||
|
||||
def test_hook(self):
|
||||
|
||||
returncode, stdout, stderr = self.run_cmd(
|
||||
[self.hook_path], self.env, json.dumps(self.data))
|
||||
|
||||
self.assertEqual(0, returncode, stderr)
|
||||
ret = yaml.safe_load(stdout)
|
||||
self.assertIsNotNone(ret['deploy_stderr'])
|
||||
self.assertEqual('', ret['deploy_stdout'])
|
||||
self.assertEqual(0, ret['deploy_status_code'])
|
||||
f = os.path.join('/tmp', self.tmp_file)
|
||||
with open(f) as out_file:
|
||||
self.assertEqual('foo=bar', out_file.read())
|
||||
os.unlink(f)
|
Loading…
Reference in New Issue
Block a user