
contrib/ directory does not follow any standards and will make it harder to programmatically install plugins on devstack. The current structure can also cause namespace collisions when loading them. This moves all plugin modules to a folder with the same name under their directory (e.g. contrib/rackspace/rackspace), allowing requirements, readmes, and other files to be separate from the actual code. This also helps to avoid namespace collisions when loading all the plugins, since all plugins will be under their namespace in the heat package. Example: heat.engine.plugins heat.engine.plugins.docker heat.engine.plugins.marconi heat.engine.plugins.rackspace Since plugin packages are now in a folder one level under contrib/, testr.conf is also updated so that all plugins are discovered correctly. Co-Authored-By: Anderson Mesquita <andersonvom@gmail.com> Related-Bug: #1271226 Change-Id: Ifc8e3ca388253be82471651737da5d399c7cfb98
120 lines
4.4 KiB
Python
120 lines
4.4 KiB
Python
# vim: tabstop=4 shiftwidth=4 softtabstop=4
|
|
#
|
|
# Copyright (c) 2013 Docker, Inc.
|
|
# All Rights Reserved.
|
|
#
|
|
# 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.
|
|
|
|
from heat.common import exception
|
|
from heat.common import template_format
|
|
from heat.engine import resource
|
|
from heat.engine import scheduler
|
|
from heat.tests.common import HeatTestCase
|
|
from heat.tests import utils
|
|
|
|
from .fake_docker_client import FakeDockerClient # noqa
|
|
from ..resources import docker_container # noqa
|
|
|
|
|
|
template = '''
|
|
{
|
|
"AWSTemplateFormatVersion": "2010-09-09",
|
|
"Description": "Test template",
|
|
"Parameters": {},
|
|
"Resources": {
|
|
"Blog": {
|
|
"Type": "DockerInc::Docker::Container",
|
|
"Properties": {
|
|
"image": "samalba/wordpress",
|
|
"env": [
|
|
"FOO=bar"
|
|
]
|
|
}
|
|
}
|
|
}
|
|
}
|
|
'''
|
|
|
|
|
|
class DockerContainerTest(HeatTestCase):
|
|
|
|
def setUp(self):
|
|
super(DockerContainerTest, self).setUp()
|
|
utils.setup_dummy_db()
|
|
for res_name, res_class in docker_container.resource_mapping().items():
|
|
resource._register_class(res_name, res_class)
|
|
|
|
def create_container(self, resource_name):
|
|
t = template_format.parse(template)
|
|
stack = utils.parse_stack(t)
|
|
resource = docker_container.DockerContainer(
|
|
resource_name, t['Resources'][resource_name], stack)
|
|
self.m.StubOutWithMock(resource, 'get_client')
|
|
resource.get_client().MultipleTimes().AndReturn(FakeDockerClient())
|
|
self.assertIsNone(resource.validate())
|
|
self.m.ReplayAll()
|
|
scheduler.TaskRunner(resource.create)()
|
|
self.assertEqual((resource.CREATE, resource.COMPLETE),
|
|
resource.state)
|
|
return resource
|
|
|
|
def get_container_state(self, resource):
|
|
client = resource.get_client()
|
|
return client.inspect_container(resource.resource_id)['State']
|
|
|
|
def test_resource_create(self):
|
|
container = self.create_container('Blog')
|
|
self.assertTrue(container.resource_id)
|
|
running = self.get_container_state(container)['Running']
|
|
self.assertIs(True, running)
|
|
self.m.VerifyAll()
|
|
|
|
def test_resource_attributes(self):
|
|
container = self.create_container('Blog')
|
|
# Test network info attributes
|
|
self.assertEqual('172.17.42.1', container.FnGetAtt('network_ip'))
|
|
self.assertEqual('1080', container.FnGetAtt('network_tcp_ports'))
|
|
self.assertEqual('', container.FnGetAtt('network_udp_ports'))
|
|
# Test logs attributes
|
|
self.assertEqual('---logs_begin---', container.FnGetAtt('logs_head'))
|
|
self.assertEqual('---logs_end---', container.FnGetAtt('logs_tail'))
|
|
# Test a non existing attribute
|
|
self.assertRaises(exception.InvalidTemplateAttribute,
|
|
container.FnGetAtt, 'invalid_attribute')
|
|
self.m.VerifyAll()
|
|
|
|
def test_resource_delete(self):
|
|
container = self.create_container('Blog')
|
|
scheduler.TaskRunner(container.delete)()
|
|
self.assertEqual((container.DELETE, container.COMPLETE),
|
|
container.state)
|
|
running = self.get_container_state(container)['Running']
|
|
self.assertIs(False, running)
|
|
self.m.VerifyAll()
|
|
|
|
def test_resource_suspend_resume(self):
|
|
container = self.create_container('Blog')
|
|
# Test suspend
|
|
scheduler.TaskRunner(container.suspend)()
|
|
self.assertEqual((container.SUSPEND, container.COMPLETE),
|
|
container.state)
|
|
running = self.get_container_state(container)['Running']
|
|
self.assertIs(False, running)
|
|
# Test resume
|
|
scheduler.TaskRunner(container.resume)()
|
|
self.assertEqual((container.RESUME, container.COMPLETE),
|
|
container.state)
|
|
running = self.get_container_state(container)['Running']
|
|
self.assertIs(True, running)
|
|
self.m.VerifyAll()
|