Handle non-zero success codes from Salt
Salt does not always return 0 on success so verify actual task results on non-zero return codes. Also add test for salt config hook. Change-Id: Ib848ef27ae29417ca38874d8310bdcd2bc35e2c3 Co-Authored-By: Chris Hultin <Chris.Hultin@rackspace.com> Closes-Bug: #1483336
This commit is contained in:
parent
f9936ee29a
commit
10fd46579a
@ -17,7 +17,7 @@ import logging
|
|||||||
import os
|
import os
|
||||||
import sys
|
import sys
|
||||||
|
|
||||||
import salt.cli
|
import salt.cli.caller
|
||||||
import salt.config
|
import salt.config
|
||||||
from salt.exceptions import SaltInvocationError
|
from salt.exceptions import SaltInvocationError
|
||||||
import yaml
|
import yaml
|
||||||
@ -72,11 +72,12 @@ def main(argv=sys.argv):
|
|||||||
with os.fdopen(os.open(fn, os.O_CREAT | os.O_WRONLY, 0o700), 'w') as f:
|
with os.fdopen(os.open(fn, os.O_CREAT | os.O_WRONLY, 0o700), 'w') as f:
|
||||||
f.write(yaml_config.encode('utf-8'))
|
f.write(yaml_config.encode('utf-8'))
|
||||||
|
|
||||||
caller = salt.cli.caller.Caller(opts)
|
caller = salt.cli.caller.Caller.factory(opts)
|
||||||
|
|
||||||
log.debug('Applying Salt state %s' % state_file)
|
log.debug('Applying Salt state %s' % state_file)
|
||||||
|
|
||||||
stdout, stderr = None, None
|
stdout, stderr = None, None
|
||||||
|
ret = {}
|
||||||
|
|
||||||
try:
|
try:
|
||||||
ret = caller.call()
|
ret = caller.call()
|
||||||
@ -84,22 +85,37 @@ def main(argv=sys.argv):
|
|||||||
log.error(
|
log.error(
|
||||||
'Salt invocation error while applying Salt sate %s' % state_file)
|
'Salt invocation error while applying Salt sate %s' % state_file)
|
||||||
stderr = err
|
stderr = err
|
||||||
log.info('Return code %s' % ret['retcode'])
|
|
||||||
|
if ret:
|
||||||
|
|
||||||
|
log.info('Results: %s' % ret)
|
||||||
|
output = yaml.safe_dump(ret['return'])
|
||||||
|
|
||||||
# returncode of 0 means there were successfull changes
|
# returncode of 0 means there were successfull changes
|
||||||
if ret['retcode'] == 0:
|
if ret['retcode'] == 0:
|
||||||
log.info('Completed applying salt state %s' % state_file)
|
log.info('Completed applying salt state %s' % state_file)
|
||||||
stdout = ret
|
stdout = output
|
||||||
else:
|
else:
|
||||||
|
# Salt doesn't always return sane return codes so we have to check
|
||||||
|
# individual results
|
||||||
|
runfailed = False
|
||||||
|
for state, data in ret['return'].items():
|
||||||
|
if not data['result']:
|
||||||
|
runfailed = True
|
||||||
|
break
|
||||||
|
if runfailed:
|
||||||
log.error('Error applying Salt state %s. [%s]\n'
|
log.error('Error applying Salt state %s. [%s]\n'
|
||||||
% (state_file, ret['retcode']))
|
% (state_file, ret['retcode']))
|
||||||
stderr = ret
|
stderr = output
|
||||||
|
else:
|
||||||
|
ret['retcode'] = 0
|
||||||
|
stdout = output
|
||||||
|
|
||||||
response = {}
|
response = {}
|
||||||
|
|
||||||
for output in c.get('outputs') or []:
|
for output in c.get('outputs', []):
|
||||||
output_name = output['name']
|
output_name = output['name']
|
||||||
response[output_name] = ret[output_name]
|
response[output_name] = ret.get(output_name)
|
||||||
|
|
||||||
response.update({
|
response.update({
|
||||||
'deploy_stdout': stdout,
|
'deploy_stdout': stdout,
|
||||||
|
@ -7,6 +7,7 @@ hacking>=0.10.0,<0.11
|
|||||||
mock>=1.0
|
mock>=1.0
|
||||||
requests>=1.2.1,!=2.4.0
|
requests>=1.2.1,!=2.4.0
|
||||||
requests-mock>=0.4.0 # Apache-2.0
|
requests-mock>=0.4.0 # Apache-2.0
|
||||||
|
salt
|
||||||
testrepository>=0.0.18
|
testrepository>=0.0.18
|
||||||
testscenarios>=0.4
|
testscenarios>=0.4
|
||||||
testtools>=0.9.34
|
testtools>=0.9.34
|
||||||
|
135
tests/software_config/test_hook_salt.py
Normal file
135
tests/software_config/test_hook_salt.py
Normal file
@ -0,0 +1,135 @@
|
|||||||
|
#
|
||||||
|
# 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 yaml
|
||||||
|
|
||||||
|
from tests.software_config import common
|
||||||
|
|
||||||
|
log = logging.getLogger('test_hook_salt')
|
||||||
|
|
||||||
|
slsok = """
|
||||||
|
testit:
|
||||||
|
environ.setenv:
|
||||||
|
- name: does_not_matter
|
||||||
|
- value:
|
||||||
|
foo: {{ opts['fooval'] }}
|
||||||
|
bar: {{ opts['barval'] }}
|
||||||
|
"""
|
||||||
|
|
||||||
|
slsfail = """
|
||||||
|
failure:
|
||||||
|
test.echo:
|
||||||
|
- text: I don't work
|
||||||
|
"""
|
||||||
|
|
||||||
|
slsnotallowed = """
|
||||||
|
install_service:
|
||||||
|
pkg.installed:
|
||||||
|
- name: {{ opts['fooval'] }}
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
class HookSaltTest(common.RunScriptTest):
|
||||||
|
|
||||||
|
data = {
|
||||||
|
'id': 'fake_stack',
|
||||||
|
'name': 'fake_resource_name',
|
||||||
|
'group': 'salt',
|
||||||
|
'inputs': [
|
||||||
|
{'name': 'fooval', 'value': 'bar'},
|
||||||
|
{'name': 'barval', 'value': 'foo'}
|
||||||
|
],
|
||||||
|
'outputs': [
|
||||||
|
{'name': 'first_output'},
|
||||||
|
{'name': 'second_output'}
|
||||||
|
],
|
||||||
|
'config': None
|
||||||
|
}
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(HookSaltTest, self).setUp()
|
||||||
|
self.hook_path = self.relative_path(
|
||||||
|
__file__,
|
||||||
|
'../..',
|
||||||
|
'hot/software-config/elements',
|
||||||
|
'heat-config-salt/install.d/hook-salt.py')
|
||||||
|
|
||||||
|
self.working_dir = self.useFixture(fixtures.TempDir())
|
||||||
|
self.minion_config_dir = self.useFixture(fixtures.TempDir())
|
||||||
|
self.minion_cach_dir = self.useFixture(fixtures.TempDir())
|
||||||
|
|
||||||
|
self.minion_conf = self.minion_config_dir.join("minion")
|
||||||
|
|
||||||
|
self.env = os.environ.copy()
|
||||||
|
self.env.update({
|
||||||
|
'HEAT_SALT_WORKING': self.working_dir.join(),
|
||||||
|
'SALT_MINION_CONFIG': self.minion_conf
|
||||||
|
})
|
||||||
|
|
||||||
|
with open(self.minion_conf, "w+") as conf_file:
|
||||||
|
conf_file.write("cachedir: %s\n" % self.minion_cach_dir.join())
|
||||||
|
conf_file.write("log_level: DEBUG\n")
|
||||||
|
|
||||||
|
def test_hook(self):
|
||||||
|
|
||||||
|
self.data['config'] = slsok
|
||||||
|
|
||||||
|
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.assertEqual(0, ret['deploy_status_code'])
|
||||||
|
self.assertIsNone(ret['deploy_stderr'])
|
||||||
|
self.assertIsNotNone(ret['deploy_stdout'])
|
||||||
|
resp = yaml.safe_load(ret['deploy_stdout'])
|
||||||
|
self.assertTrue(resp.values()[0]['result'])
|
||||||
|
self.assertEqual({'bar': 'foo', 'foo': 'bar'},
|
||||||
|
resp.values()[0]['changes'])
|
||||||
|
|
||||||
|
def test_hook_salt_failed(self):
|
||||||
|
|
||||||
|
self.data['config'] = slsfail
|
||||||
|
|
||||||
|
returncode, stdout, stderr = self.run_cmd(
|
||||||
|
[self.hook_path], self.env, json.dumps(self.data))
|
||||||
|
|
||||||
|
self.assertEqual(0, returncode)
|
||||||
|
self.assertIsNotNone(stderr)
|
||||||
|
self.assertIsNotNone(stdout)
|
||||||
|
jsonout = json.loads(stdout)
|
||||||
|
self.assertIsNone(jsonout.get("deploy_stdout"),
|
||||||
|
jsonout.get("deploy_stdout"))
|
||||||
|
self.assertEqual(2, jsonout.get("deploy_status_code"))
|
||||||
|
self.assertIsNotNone(jsonout.get("deploy_stderr"))
|
||||||
|
self.assertIn("was not found in SLS", jsonout.get("deploy_stderr"))
|
||||||
|
|
||||||
|
def test_hook_salt_retcode(self):
|
||||||
|
|
||||||
|
self.data['config'] = slsnotallowed
|
||||||
|
|
||||||
|
returncode, stdout, stderr = self.run_cmd(
|
||||||
|
[self.hook_path], self.env, json.dumps(self.data))
|
||||||
|
|
||||||
|
self.assertEqual(0, returncode, stderr)
|
||||||
|
self.assertIsNotNone(stdout)
|
||||||
|
self.assertIsNotNone(stderr)
|
||||||
|
ret = json.loads(stdout)
|
||||||
|
self.assertIsNone(ret['deploy_stdout'])
|
||||||
|
self.assertIsNotNone(ret['deploy_stderr'])
|
||||||
|
resp = yaml.safe_load(ret['deploy_stderr']).values()[0]
|
||||||
|
self.assertFalse(resp['result'])
|
Loading…
Reference in New Issue
Block a user