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
|
testscenarios>=0.4
|
||||||
testtools>=0.9.34
|
testtools>=0.9.34
|
||||||
yamllint>=1.2.0
|
yamllint>=1.2.0
|
||||||
|
os-apply-config
|
||||||
|
|
||||||
python-heatclient>=1.2.0
|
python-heatclient>=1.2.0
|
||||||
python-keystoneclient>=0.10.0
|
python-keystoneclient>=0.10.0
|
||||||
|
@ -25,7 +25,8 @@ from tests.software_config import common
|
|||||||
|
|
||||||
class HeatConfigTest(common.RunScriptTest):
|
class HeatConfigTest(common.RunScriptTest):
|
||||||
|
|
||||||
fake_hooks = ['cfn-init', 'chef', 'puppet', 'salt', 'script']
|
fake_hooks = ['cfn-init', 'chef', 'puppet', 'salt', 'script',
|
||||||
|
'apply-config']
|
||||||
|
|
||||||
data = [
|
data = [
|
||||||
{
|
{
|
||||||
@ -66,9 +67,14 @@ class HeatConfigTest(common.RunScriptTest):
|
|||||||
'config': 'five'
|
'config': 'five'
|
||||||
}, {
|
}, {
|
||||||
'id': '6666',
|
'id': '6666',
|
||||||
|
'group': 'apply-config',
|
||||||
|
'inputs': [{'name': 'foo', 'value': 'bar'}],
|
||||||
|
'config': 'six'
|
||||||
|
}, {
|
||||||
|
'id': '9999',
|
||||||
'group': 'no-such-hook',
|
'group': 'no-such-hook',
|
||||||
'inputs': [],
|
'inputs': [],
|
||||||
'config': 'six'
|
'config': 'nine'
|
||||||
}]
|
}]
|
||||||
|
|
||||||
outputs = {
|
outputs = {
|
||||||
@ -97,6 +103,11 @@ class HeatConfigTest(common.RunScriptTest):
|
|||||||
'deploy_status_code': '-1',
|
'deploy_status_code': '-1',
|
||||||
'deploy_stderr': 'A bad thing happened',
|
'deploy_stderr': 'A bad thing happened',
|
||||||
'deploy_stdout': 'stdout'
|
'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