heat-agents/tests/test_hook_docker_cmd.py

1023 lines
31 KiB
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 copy
import json
import os
import tempfile
import fixtures
import six
from testtools import testcase
from tests import common
@testcase.skip('Skipped until story/2004926 fixed')
class HookDockerCmdTest(common.RunScriptTest):
data = {
"name": "abcdef001",
"group": "docker-cmd",
"id": "abc123",
"inputs": [{
"name": "deploy_stack_id",
"value": "the_stack",
}, {
"name": "deploy_resource_name",
"value": "the_deployment",
}],
"config": {
"db": {
"name": "x",
"image": "xxx",
"privileged": False,
"environment": ["foo=bar"],
"env_file": "env.file",
"start_order": 0
},
"web-ls": {
"action": "exec",
"start_order": 2,
"command": ["web", "/bin/ls", "-l"]
},
"web": {
"name": "y",
"start_order": 1,
"image": "yyy",
"net": "host",
"restart": "always",
"privileged": True,
"user": "root",
"command": "/bin/webserver start",
"volumes": [
"/run:/run",
"db:/var/lib/db"
],
"environment": [
"KOLLA_CONFIG_STRATEGY=COPY_ALWAYS",
"FOO=BAR"
],
"env_file": [
"foo.env",
"bar.conf"
]
}
}
}
data_exit_code = {
"name": "abcdef001",
"group": "docker-cmd",
"id": "abc123",
"config": {
"web-ls": {
"action": "exec",
"command": ["web", "/bin/ls", "-l"],
"exit_codes": [0, 1]
}
}
}
def setUp(self):
super(HookDockerCmdTest, self).setUp()
self.hook_path = self.relative_path(
__file__,
'..',
'heat-config-docker-cmd/install.d/hook-docker-cmd.py')
self.cleanup_path = self.relative_path(
__file__,
'..',
'heat-config-docker-cmd/',
'os-refresh-config/configure.d/50-heat-config-docker-cmd')
self.fake_tool_path = six.text_type(self.relative_path(
__file__,
'config-tool-fake.py'))
self.working_dir = self.useFixture(fixtures.TempDir())
self.outputs_dir = self.useFixture(fixtures.TempDir())
self.test_state_path = self.outputs_dir.join('test_state.json')
self.env = os.environ.copy()
self.env.update({
'HEAT_DOCKER_CMD': self.fake_tool_path,
'TEST_STATE_PATH': self.test_state_path,
})
def check_basic_response(self, state):
self.assertEqual([
self.fake_tool_path,
u'inspect',
u'--type',
u'image',
u'--format',
u'exists',
u'xxx'
], state[0]['args'])
self.assertEqual([
self.fake_tool_path,
u'pull',
u'xxx'
], state[1]['args'])
self.assertEqual([
self.fake_tool_path,
u'inspect',
u'--type',
u'image',
u'--format',
u'exists',
u'yyy'
], state[2]['args'])
self.assertEqual([
self.fake_tool_path,
u'pull',
u'yyy'
], state[3]['args'])
self.assertEqual([
self.fake_tool_path,
u'ps',
u'-a',
u'--filter',
u'label=managed_by=docker-cmd',
u'--filter',
u'label=config_id=abc123',
u'--format',
u'{{.Names}} {{.Label "container_name"}}'
], state[4]['args'])
self.assertEqual([
self.fake_tool_path,
u'ps',
u'-a',
u'--filter',
u'label=managed_by=docker-cmd',
u'--format',
u'{{.Names}} {{.Label "container_name"}}'
], state[5]['args'])
self.assertEqual([
self.fake_tool_path,
u'ps',
u'-a',
u'--filter',
u'label=managed_by=docker-cmd',
u'--filter',
u'label=config_id=abc123',
u'--format',
u'{{.Names}} {{.Label "container_name"}}'
], state[6]['args'])
self.assertEqual([
self.fake_tool_path,
u'inspect',
u'--type',
u'container',
u'--format',
u'exists',
u'db'
], state[7]['args'])
def assert_args_and_labels(self, expected_args, expected_labels, observed):
'''Assert the labels arguments separately to other arguments.
Tests that each expected_labels label exists, and remaining
expected arguments match exactly.
This allows paunch to add new label arguments without breaking these
tests.
'''
args = []
labels = []
j = 0
while j < len(observed):
if observed[j] == '--label':
j += 1
labels.append(observed[j])
else:
args.append(observed[j])
j += 1
self.assertEqual(expected_args, args)
for label in expected_labels:
self.assertIn(label, labels)
def test_hook(self):
self.env.update({
'TEST_RESPONSE': json.dumps([
# inspect for image xxx
{},
# poll for image xxx
{},
# inspect for image yyy
{},
# poll for image yyy
{},
# ps for delete missing
{},
# ps for renames
{},
# ps for currently running containers
{},
# inspect for db unique container name
{},
# docker run db
{'stderr': 'Creating db...'},
# inspect for web unique container name
{},
# docker run web
{'stderr': 'Creating web...'},
# inspect
{},
# name lookup for exec web
{'stdout': 'web'},
# docker exec web
{'stderr': 'one.txt\ntwo.txt\nthree.txt'},
])
})
returncode, stdout, stderr = self.run_cmd(
[self.hook_path], self.env, json.dumps(self.data))
self.assertEqual(0, returncode, stderr)
self.assertEqual({
'deploy_stdout': '',
'deploy_stderr': 'Creating db...\n'
'Creating web...\n'
'one.txt\ntwo.txt\nthree.txt',
'deploy_status_code': 0
}, json.loads(stdout.decode('utf-8')))
state = list(self.json_from_files(self.test_state_path, 14))
self.check_basic_response(state)
self.assert_args_and_labels([
self.fake_tool_path,
u'run',
u'--name',
u'db',
u'--detach=true',
u'--env-file=env.file',
u'--env=foo=bar',
u'--privileged=false',
u'xxx'
u''
], [
u'deploy_stack_id=the_stack',
u'deploy_resource_name=the_deployment',
u'config_id=abc123',
u'container_name=db',
u'managed_by=docker-cmd',
], state[8]['args'])
self.assertEqual([
self.fake_tool_path,
u'inspect',
u'--type',
u'container',
u'--format',
u'exists',
u'web',
], state[9]['args'])
self.assert_args_and_labels([
self.fake_tool_path,
u'run',
u'--name',
u'web',
u'--detach=true',
u'--env-file=foo.env',
u'--env-file=bar.conf',
u'--env=KOLLA_CONFIG_STRATEGY=COPY_ALWAYS',
u'--env=FOO=BAR',
u'--net=host',
u'--privileged=true',
u'--restart=always',
u'--user=root',
u'--volume=/run:/run',
u'--volume=db:/var/lib/db',
u'yyy',
u'/bin/webserver',
u'start'
], [
u'deploy_stack_id=the_stack',
u'deploy_resource_name=the_deployment',
u'config_id=abc123',
u'container_name=web',
u'managed_by=docker-cmd',
], state[10]['args'])
self.assertEqual([
self.fake_tool_path,
'inspect',
'--type',
'container',
'--format',
'exists',
'web-ls'
], state[11]['args'])
self.assertEqual([
self.fake_tool_path,
u'ps',
u'-a',
u'--filter',
u'label=container_name=web',
u'--filter',
u'label=config_id=abc123',
u'--format',
u'{{.Names}}',
], state[12]['args'])
self.assertEqual([
self.fake_tool_path,
u'exec',
u'web',
u'/bin/ls',
u'-l'
], state[13]['args'])
def test_hook_exit_codes(self):
self.env.update({
'TEST_RESPONSE': json.dumps([
# ps for delete missing
{},
# ps for renames
{},
# ps for currently running containers
{},
{},
{'stdout': 'web'},
{
'stdout': '',
'stderr': 'Warning: custom exit code',
'returncode': 1
}
])
})
returncode, stdout, stderr = self.run_cmd(
[self.hook_path], self.env, json.dumps(self.data_exit_code))
self.assertEqual({
'deploy_stdout': '',
'deploy_stderr': 'Warning: custom exit code',
'deploy_status_code': 0
}, json.loads(stdout.decode('utf-8')))
state = list(self.json_from_files(self.test_state_path, 6))
self.assertEqual([
self.fake_tool_path,
'ps',
'-a',
'--filter',
'label=managed_by=docker-cmd',
'--filter',
'label=config_id=abc123',
'--format',
'{{.Names}} {{.Label "container_name"}}'
], state[0]['args'])
self.assertEqual([
self.fake_tool_path,
'ps',
'-a',
'--filter',
'label=managed_by=docker-cmd',
'--format',
'{{.Names}} {{.Label "container_name"}}'
], state[1]['args'])
self.assertEqual([
self.fake_tool_path,
'ps',
'-a',
'--filter',
'label=managed_by=docker-cmd',
'--filter',
'label=config_id=abc123',
'--format',
'{{.Names}} {{.Label "container_name"}}'
], state[2]['args'])
self.assertEqual([
self.fake_tool_path,
'inspect',
'--type',
'container',
'--format',
'exists',
'web-ls'
], state[3]['args'])
self.assertEqual([
self.fake_tool_path,
'ps',
'-a',
'--filter',
'label=container_name=web',
'--filter',
'label=config_id=abc123',
'--format',
'{{.Names}}',
], state[4]['args'])
self.assertEqual([
self.fake_tool_path,
'exec',
'web',
'/bin/ls',
'-l'
], state[5]['args'])
def test_hook_failed(self):
self.env.update({
'TEST_RESPONSE': json.dumps([
# inspect for image xxx
{},
# poll for image xxx
{},
# inspect for image yyy
{},
# poll for image yyy
{},
# ps for delete missing
{},
# ps for renames
{},
# ps for currently running containers
{},
# inspect for db unique container name
{},
# docker run db
{'stderr': 'Creating db...'},
# inspect for web unique container name
{},
# docker run web
{'stderr': 'Creating web...'},
# inspect
{},
# name lookup for exec web
{'stdout': 'web'},
# docker exec web fails
{
'stdout': '',
'stderr': 'No such file or directory',
'returncode': 2
}
])
})
returncode, stdout, stderr = self.run_cmd(
[self.hook_path], self.env, json.dumps(self.data))
self.assertEqual({
'deploy_stdout': '',
'deploy_stderr': 'Creating db...\n'
'Creating web...\n'
'No such file or directory',
'deploy_status_code': 2
}, json.loads(stdout.decode('utf-8')))
state = list(self.json_from_files(self.test_state_path, 14))
self.check_basic_response(state)
self.assert_args_and_labels([
self.fake_tool_path,
u'run',
u'--name',
u'db',
u'--detach=true',
u'--env-file=env.file',
u'--env=foo=bar',
u'--privileged=false',
u'xxx'
u''
], [
u'deploy_stack_id=the_stack',
u'deploy_resource_name=the_deployment',
u'config_id=abc123',
u'container_name=db',
u'managed_by=docker-cmd',
], state[8]['args'])
self.assertEqual([
self.fake_tool_path,
u'inspect',
u'--type',
u'container',
u'--format',
u'exists',
u'web',
], state[9]['args'])
self.assert_args_and_labels([
self.fake_tool_path,
u'run',
u'--name',
u'web',
u'--detach=true',
u'--env-file=foo.env',
u'--env-file=bar.conf',
u'--env=KOLLA_CONFIG_STRATEGY=COPY_ALWAYS',
u'--env=FOO=BAR',
u'--net=host',
u'--privileged=true',
u'--restart=always',
u'--user=root',
u'--volume=/run:/run',
u'--volume=db:/var/lib/db',
u'yyy',
u'/bin/webserver',
u'start'
], [
u'deploy_stack_id=the_stack',
u'deploy_resource_name=the_deployment',
u'config_id=abc123',
u'container_name=web',
u'managed_by=docker-cmd',
], state[10]['args'])
self.assertEqual([
self.fake_tool_path,
'inspect',
'--type',
'container',
'--format',
'exists',
'web-ls'
], state[11]['args'])
self.assertEqual([
self.fake_tool_path,
u'ps',
u'-a',
u'--filter',
u'label=container_name=web',
u'--filter',
u'label=config_id=abc123',
u'--format',
u'{{.Names}}',
], state[12]['args'])
self.assertEqual([
self.fake_tool_path,
u'exec',
u'web',
u'/bin/ls',
u'-l'
], state[13]['args'])
def test_hook_unique_names(self):
self.env.update({
'TEST_RESPONSE': json.dumps([
# inspect for image xxx
{},
# poll for image xxx
{},
# inspect for image yyy
{},
# poll for image yyy
{},
# ps for delete missing in this config id
{},
# ps for renames
{'stdout': 'web web\ndb db\n'},
# ps for currently running containers in this config id
{},
# inspect for db unique container name
{'stdout': 'exists'},
{
'stderr': 'Error: No such container: db-blah',
'returncode': 1
},
# docker run db
{'stderr': 'Creating db...'},
# # inspect for web unique container name
{'stdout': 'exists'},
{
'stderr': 'Error: No such container: web-blah',
'returncode': 1
},
# # docker run web
{'stderr': 'Creating web...'},
# name lookup for exec web
{'stdout': 'web-asdf1234'},
# docker exec web-asdf1234
{'stderr': 'one.txt\ntwo.txt\nthree.txt'},
])
})
returncode, stdout, stderr = self.run_cmd(
[self.hook_path], self.env, json.dumps(self.data))
self.assertEqual(0, returncode, stderr)
state = list(self.json_from_files(self.test_state_path, 17))
dd = []
for i in state:
dd.append(i['args'])
db_container_name = state[8]['args'][6]
web_container_name = state[11]['args'][6]
self.assertRegex(db_container_name, 'db-[0-9a-z]{8}')
self.assertRegex(web_container_name, 'web-[0-9a-z]{8}')
self.check_basic_response(state)
self.assertEqual([
self.fake_tool_path,
u'inspect',
u'--type',
u'container',
u'--format',
u'exists',
db_container_name,
], state[8]['args'])
self.assert_args_and_labels([
self.fake_tool_path,
u'run',
u'--name',
db_container_name,
u'--detach=true',
u'--env-file=env.file',
u'--env=foo=bar',
u'--privileged=false',
u'xxx'
], [
u'deploy_stack_id=the_stack',
u'deploy_resource_name=the_deployment',
u'config_id=abc123',
u'container_name=db',
u'managed_by=docker-cmd',
], state[9]['args'])
self.assertEqual([
self.fake_tool_path,
u'inspect',
u'--type',
u'container',
u'--format',
u'exists',
u'web',
], state[10]['args'])
self.assertEqual([
self.fake_tool_path,
u'inspect',
u'--type',
u'container',
u'--format',
u'exists',
web_container_name,
], state[11]['args'])
self.assert_args_and_labels([
self.fake_tool_path,
u'run',
u'--name',
web_container_name,
u'--detach=true',
u'--env-file=foo.env',
u'--env-file=bar.conf',
u'--env=KOLLA_CONFIG_STRATEGY=COPY_ALWAYS',
u'--env=FOO=BAR',
u'--net=host',
u'--privileged=true',
u'--restart=always',
u'--user=root',
u'--volume=/run:/run',
u'--volume=db:/var/lib/db',
u'yyy',
u'/bin/webserver',
u'start'
], [
u'deploy_stack_id=the_stack',
u'deploy_resource_name=the_deployment',
u'config_id=abc123',
u'container_name=web',
u'managed_by=docker-cmd',
], state[12]['args'])
self.assertEqual([
self.fake_tool_path,
'inspect',
'--type',
'container',
'--format',
'exists',
'web-ls'
], state[13]['args'])
self.assertEqual([
self.fake_tool_path,
'inspect',
'--type',
'container',
'--format',
'exists',
state[14]['args'][6]
], state[14]['args'])
self.assertEqual([
self.fake_tool_path,
u'ps',
u'-a',
u'--filter',
u'label=container_name=web',
u'--filter',
u'label=config_id=abc123',
u'--format',
u'{{.Names}}',
], state[15]['args'])
self.assertEqual([
self.fake_tool_path,
u'exec',
u'web',
u'/bin/ls',
u'-l'
], state[16]['args'])
def test_cleanup_deleted(self):
self.env.update({
'TEST_RESPONSE': json.dumps([{
# first run, no running containers
'stdout': '\n'
}, {
# list name and container_name label for all containers
'stdout': '\n'
}])
})
conf_dir = self.useFixture(fixtures.TempDir()).join()
with tempfile.NamedTemporaryFile(dir=conf_dir, delete=False,
mode='w') as f:
f.write(json.dumps([self.data]))
f.flush()
self.env['HEAT_SHELL_CONFIG'] = f.name
returncode, stdout, stderr = self.run_cmd(
[self.cleanup_path], self.env)
# on the first run, no docker rm calls made
state = list(self.json_from_files(self.test_state_path, 2))
self.assertEqual([
self.fake_tool_path,
'ps',
'-a',
'--filter',
'label=managed_by=docker-cmd',
'--format',
'{{.Label "config_id"}}'
], state[0]['args'])
self.assertEqual([
self.fake_tool_path,
'ps',
'-a',
'--filter',
'label=managed_by=docker-cmd',
'--format',
'{{.Names}} {{.Label "container_name"}}'
], state[1]['args'])
self.env.update({
'TEST_RESPONSE': json.dumps([{
# list config_id labels, 3 containers same config
'stdout': 'abc123\nabc123\nabc123\n'
}, {
# list containers with config_id
'stdout': '111\n222\n333\n'
}, {
'stdout': '111 deleted'
}, {
'stdout': '222 deleted'
}, {
'stdout': '333 deleted'
}, {
# list name and container_name label for all containers
'stdout': '\n'
}])
})
# run again with empty config data
with tempfile.NamedTemporaryFile(dir=conf_dir, delete=False,
mode='w') as f:
f.write(json.dumps([]))
f.flush()
self.env['HEAT_SHELL_CONFIG'] = f.name
returncode, stdout, stderr = self.run_cmd(
[self.cleanup_path], self.env)
# on the second run, abc123 is deleted,
# docker rm is run on all containers
state = list(self.json_from_files(self.test_state_path, 6))
self.assertEqual([
self.fake_tool_path,
'ps',
'-a',
'--filter',
'label=managed_by=docker-cmd',
'--format',
'{{.Label "config_id"}}'
], state[0]['args'])
self.assertEqual([
self.fake_tool_path,
'ps',
'-q',
'-a',
'--filter',
'label=managed_by=docker-cmd',
'--filter',
'label=config_id=abc123'
], state[1]['args'])
self.assertEqual([
self.fake_tool_path,
'rm',
'-f',
'111',
], state[2]['args'])
self.assertEqual([
self.fake_tool_path,
'rm',
'-f',
'222',
], state[3]['args'])
self.assertEqual([
self.fake_tool_path,
'rm',
'-f',
'333',
], state[4]['args'])
self.assertEqual([
self.fake_tool_path,
'ps',
'-a',
'--filter',
'label=managed_by=docker-cmd',
'--format',
'{{.Names}} {{.Label "container_name"}}'
], state[5]['args'])
def test_cleanup_changed(self):
self.env.update({
'TEST_RESPONSE': json.dumps([{
# list config_id labels, 3 containers same config
'stdout': 'abc123\nabc123\nabc123\n'
}, {
# list name and container_name label for all containers
'stdout': '111 111\n'
'222 222\n'
'333\n'
}])
})
conf_dir = self.useFixture(fixtures.TempDir()).join()
with tempfile.NamedTemporaryFile(dir=conf_dir, delete=False,
mode='w') as f:
f.write(json.dumps([self.data]))
f.flush()
self.env['HEAT_SHELL_CONFIG'] = f.name
returncode, stdout, stderr = self.run_cmd(
[self.cleanup_path], self.env)
# on the first run, no docker rm calls made
state = list(self.json_from_files(self.test_state_path, 2))
self.assertEqual([
self.fake_tool_path,
'ps',
'-a',
'--filter',
'label=managed_by=docker-cmd',
'--format',
'{{.Label "config_id"}}'
], state[0]['args'])
self.assertEqual([
self.fake_tool_path,
'ps',
'-a',
'--filter',
'label=managed_by=docker-cmd',
'--format',
'{{.Names}} {{.Label "container_name"}}'
], state[1]['args'])
# run again with changed config data
self.env.update({
'TEST_RESPONSE': json.dumps([{
# list config_id labels, 3 containers same config
'stdout': 'abc123\nabc123\nabc123\n'
}, {
# list containers with config_id
'stdout': '111\n222\n333\n'
}, {
'stdout': '111 deleted'
}, {
'stdout': '222 deleted'
}, {
'stdout': '333 deleted'
}, {
# list name and container_name label for all containers
'stdout': 'abc123 abc123\n'
}])
})
new_data = copy.deepcopy(self.data)
new_data['config']['web']['image'] = 'yyy'
new_data['id'] = 'def456'
with tempfile.NamedTemporaryFile(dir=conf_dir, delete=False,
mode='w') as f:
f.write(json.dumps([new_data]))
f.flush()
self.env['HEAT_SHELL_CONFIG'] = f.name
returncode, stdout, stderr = self.run_cmd(
[self.cleanup_path], self.env)
# on the second run, abc123 is deleted,
# docker rm is run on all containers
state = list(self.json_from_files(self.test_state_path, 6))
self.assertEqual([
self.fake_tool_path,
'ps',
'-a',
'--filter',
'label=managed_by=docker-cmd',
'--format',
'{{.Label "config_id"}}'
], state[0]['args'])
self.assertEqual([
self.fake_tool_path,
'ps',
'-q',
'-a',
'--filter',
'label=managed_by=docker-cmd',
'--filter',
'label=config_id=abc123'
], state[1]['args'])
self.assertEqual([
self.fake_tool_path,
'rm',
'-f',
'111',
], state[2]['args'])
self.assertEqual([
self.fake_tool_path,
'rm',
'-f',
'222',
], state[3]['args'])
self.assertEqual([
self.fake_tool_path,
'rm',
'-f',
'333',
], state[4]['args'])
self.assertEqual([
self.fake_tool_path,
'ps',
'-a',
'--filter',
'label=managed_by=docker-cmd',
'--format',
'{{.Names}} {{.Label "container_name"}}'
], state[5]['args'])
def test_cleanup_rename(self):
self.env.update({
'TEST_RESPONSE': json.dumps([{
# list config_id labels, 3 containers same config
'stdout': 'abc123\nabc123\nabc123\n'
}, {
# list name and container_name label for all containers
'stdout': '111 111-s84nf83h\n'
'222 222\n'
'333 333-3nd83nfi\n'
}])
})
conf_dir = self.useFixture(fixtures.TempDir()).join()
with tempfile.NamedTemporaryFile(dir=conf_dir, delete=False,
mode='w') as f:
f.write(json.dumps([self.data]))
f.flush()
self.env['HEAT_SHELL_CONFIG'] = f.name
returncode, stdout, stderr = self.run_cmd(
[self.cleanup_path], self.env)
# on the first run, no docker rm calls made
state = list(self.json_from_files(self.test_state_path, 4))
self.assertEqual([
self.fake_tool_path,
'ps',
'-a',
'--filter',
'label=managed_by=docker-cmd',
'--format',
'{{.Label "config_id"}}'
], state[0]['args'])
self.assertEqual([
self.fake_tool_path,
'ps',
'-a',
'--filter',
'label=managed_by=docker-cmd',
'--format',
'{{.Names}} {{.Label "container_name"}}'
], state[1]['args'])
self.assertEqual([
self.fake_tool_path,
'rename',
'111',
'111-s84nf83h'
], state[2]['args'])
self.assertEqual([
self.fake_tool_path,
'rename',
'333',
'333-3nd83nfi'
], state[3]['args'])