bc9a1cc9c1
During doc generation, resources have their own templates and need to be separate from non-resource modules. This removes the ``heat`` ``engine`` and ``plugins`` folders and moves all resources into a ``resources`` folder, effectively separating the custom Rackspace Client from custom Rackspace resources. Partial-Bug: #1271226 Change-Id: Idbdc8400d30e2e745a2dc1b2d2fa1b7cff50d57b
555 lines
23 KiB
Python
555 lines
23 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 uuid
|
|
|
|
import mox
|
|
import paramiko
|
|
import novaclient
|
|
|
|
from heat.db import api as db_api
|
|
from heat.tests.v1_1 import fakes
|
|
from heat.common import template_format
|
|
from heat.common import exception
|
|
from heat.engine import parser
|
|
from heat.engine import resource
|
|
from heat.engine import scheduler
|
|
from heat.tests.common import HeatTestCase
|
|
from heat.tests import utils
|
|
|
|
from heat.engine import clients
|
|
from ..resources import cloud_server # noqa
|
|
|
|
|
|
wp_template = '''
|
|
{
|
|
"AWSTemplateFormatVersion" : "2010-09-09",
|
|
"Description" : "WordPress",
|
|
"Parameters" : {
|
|
"flavor" : {
|
|
"Description" : "Rackspace Cloud Server flavor",
|
|
"Type" : "String",
|
|
"Default" : "m1.small",
|
|
"AllowedValues" : ["256 MB Server", "m1.small", "m1.large", "invalid"],
|
|
"ConstraintDescription" : "must be a valid Rackspace Cloud Server flavor"
|
|
},
|
|
},
|
|
"Resources" : {
|
|
"WebServer": {
|
|
"Type": "Rackspace::Cloud::Server",
|
|
"Properties": {
|
|
"image" : "CentOS 5.2",
|
|
"flavor" : "256 MB Server",
|
|
"user_data" : "wordpress"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
'''
|
|
|
|
rsa_key = """-----BEGIN RSA PRIVATE KEY-----
|
|
MIICWwIBAAKBgQDibWGom/83F2xYfVylBZhUbREiVlw42X7afUuHzNJuh/5EyhXQ
|
|
BmBHjVGL1mxZY4GoISrxIkW1jVmTXbm8FknIlS3jxEOC+xF3IkLBtmZEkFVLOUCv
|
|
Fpru1xThFS0L/pRttiTWLm+dsjboCV4qtg/+y30O0RJ5AAFgGkoVs8idrQIDAQAB
|
|
AoGAQU/7037r5yBCiGPgzVkHz5KGVrlCcMOL68ood0uFh4yCs6T3FcJBE2KYGxYG
|
|
uuIRDEZE9LlGElBrfi6S3MYxEbewITK9Li1cr8K0fJlIbg5PI1MxwiTXzG7i0f8Y
|
|
trtZjo/fs8XNSS4xlGWCUgtiNXvLS6wxyDGGbqeh1BmETgECQQDmoPJ3h5kuZguA
|
|
o7B+iTaKXqyWPf0ImsZ0UQYBgnEWTaZEh8W0015jP55mndALWA9pmhHJm+BC/Hfe
|
|
Kp6jtVyxAkEA+1YctDe62u5pXU/GK8UfDJwi4m1VxUfASrlxh+ALag9knwe6Dlev
|
|
EKKIe8R6HZs2zavaJs6dddxHRcIi8rXfvQJAW6octOVwPMDSUY69140x4E1Ay3ZX
|
|
29OojRKnEHKIABVcwGA2dGiOW2Qt0RtoVRnrBk32Q+twdy9hdSv7YZX0AQJAVDaj
|
|
QYNW2Zp+tWRQa0QORkRer+2gioyjEqaWMsfQK0ZjGaIWJk4c+37qKkZIAHmMYFeP
|
|
recW/XHEc8w7t4VXJQJAevSyciBfFcWMZTwlqq8wXNMCRLJt5CxvO4gSO+hPNrDe
|
|
gDZkz7KcZC7TkO0NYVRssA6/84mCqx6QHpKaYNG9kg==
|
|
-----END RSA PRIVATE KEY-----
|
|
"""
|
|
|
|
|
|
class RackspaceCloudServerTest(HeatTestCase):
|
|
def setUp(self):
|
|
super(RackspaceCloudServerTest, self).setUp()
|
|
self.fc = fakes.FakeClient()
|
|
utils.setup_dummy_db()
|
|
# Test environment may not have pyrax client library installed and if
|
|
# pyrax is not installed resource class would not be registered.
|
|
# So register resource provider class explicitly for unit testing.
|
|
resource._register_class("Rackspace::Cloud::Server",
|
|
cloud_server.CloudServer)
|
|
|
|
def _setup_test_stack(self, stack_name):
|
|
t = template_format.parse(wp_template)
|
|
template = parser.Template(t)
|
|
stack = parser.Stack(utils.dummy_context(), stack_name, template,
|
|
{},
|
|
stack_id=str(uuid.uuid4()))
|
|
return (t, stack)
|
|
|
|
def _mock_ssh_sftp(self, exit_code=0):
|
|
# SSH
|
|
self.m.StubOutWithMock(paramiko, "SSHClient")
|
|
self.m.StubOutWithMock(paramiko, "MissingHostKeyPolicy")
|
|
ssh = self.m.CreateMockAnything()
|
|
paramiko.SSHClient().AndReturn(ssh)
|
|
paramiko.MissingHostKeyPolicy()
|
|
ssh.set_missing_host_key_policy(None)
|
|
ssh.connect(mox.IgnoreArg(),
|
|
key_filename=mox.IgnoreArg(),
|
|
username='root')
|
|
fake_chan = self.m.CreateMockAnything()
|
|
self.m.StubOutWithMock(paramiko.SSHClient, "get_transport")
|
|
chan = ssh.get_transport().AndReturn(fake_chan)
|
|
fake_chan_session = self.m.CreateMockAnything()
|
|
chan_session = chan.open_session().AndReturn(fake_chan_session)
|
|
fake_chan_session.settimeout(3600.0)
|
|
chan_session.exec_command(mox.IgnoreArg())
|
|
fake_chan_session.recv(1024)
|
|
chan_session.recv_exit_status().AndReturn(exit_code)
|
|
fake_chan_session.close()
|
|
ssh.close()
|
|
|
|
# SFTP
|
|
self.m.StubOutWithMock(paramiko, "Transport")
|
|
transport = self.m.CreateMockAnything()
|
|
paramiko.Transport((mox.IgnoreArg(), 22)).AndReturn(transport)
|
|
transport.connect(hostkey=None, username="root", pkey=mox.IgnoreArg())
|
|
sftp = self.m.CreateMockAnything()
|
|
self.m.StubOutWithMock(paramiko, "SFTPClient")
|
|
paramiko.SFTPClient.from_transport(transport).AndReturn(sftp)
|
|
sftp_file = self.m.CreateMockAnything()
|
|
sftp.open(mox.IgnoreArg(), 'w').MultipleTimes().AndReturn(sftp_file)
|
|
sftp_file.write(mox.IgnoreArg()).MultipleTimes()
|
|
sftp_file.close().MultipleTimes()
|
|
sftp.close()
|
|
transport.close()
|
|
|
|
def _setup_test_cs(self, return_server, name, exit_code=0):
|
|
stack_name = '%s_stack' % name
|
|
(t, stack) = self._setup_test_stack(stack_name)
|
|
|
|
self.m.StubOutWithMock(cloud_server.CloudServer, "nova")
|
|
cloud_server.CloudServer.nova().MultipleTimes().AndReturn(self.fc)
|
|
|
|
t['Resources']['WebServer']['Properties']['image'] = 'CentOS 5.2'
|
|
t['Resources']['WebServer']['Properties']['flavor'] = '256 MB Server'
|
|
|
|
cs = cloud_server.CloudServer('%s_name' % name,
|
|
t['Resources']['WebServer'], stack)
|
|
cs._private_key = rsa_key
|
|
cs.t = cs.stack.resolve_runtime_data(cs.t)
|
|
|
|
self.m.StubOutWithMock(self.fc.servers, 'create')
|
|
name_limit = cloud_server.CloudServer.physical_resource_name_limit
|
|
server_name = utils.PhysName(stack_name, cs.name, limit=name_limit)
|
|
self.fc.servers.create(server_name, 1, 1,
|
|
files=mox.IgnoreArg()).AndReturn(return_server)
|
|
return_server.adminPass = "foobar"
|
|
|
|
self.m.StubOutWithMock(cloud_server.CloudServer, 'script')
|
|
cloud_server.CloudServer.script = "foobar"
|
|
|
|
self._mock_ssh_sftp(exit_code)
|
|
return cs
|
|
|
|
def _create_test_cs(self, return_server, name, exit_code=0):
|
|
cs = self._setup_test_cs(return_server, name, exit_code)
|
|
|
|
self.m.ReplayAll()
|
|
scheduler.TaskRunner(cs.create)()
|
|
return cs
|
|
|
|
def _update_test_cs(self, return_server, name, exit_code=0):
|
|
self._mock_ssh_sftp(exit_code)
|
|
self.m.StubOutWithMock(clients.OpenStackClients, "nova")
|
|
clients.OpenStackClients.nova(
|
|
mox.IgnoreArg()).MultipleTimes().AndReturn(self.fc)
|
|
|
|
def test_cs_create(self):
|
|
return_server = self.fc.servers.list()[1]
|
|
cs = self._create_test_cs(return_server, 'test_cs_create')
|
|
# this makes sure the auto increment worked on cloud server creation
|
|
self.assertTrue(cs.id > 0)
|
|
|
|
expected_public = return_server.networks['public'][0]
|
|
expected_private = return_server.networks['private'][0]
|
|
self.assertEqual(expected_public, cs.FnGetAtt('PublicIp'))
|
|
self.assertEqual(expected_private, cs.FnGetAtt('PrivateIp'))
|
|
self.assertEqual(expected_public, cs.FnGetAtt('PublicDnsName'))
|
|
self.assertEqual(expected_public, cs.FnGetAtt('PrivateDnsName'))
|
|
|
|
self.m.VerifyAll()
|
|
|
|
def test_cs_create_with_image_name(self):
|
|
return_server = self.fc.servers.list()[1]
|
|
cs = self._setup_test_cs(return_server, 'test_cs_create_image_id')
|
|
|
|
self.m.ReplayAll()
|
|
scheduler.TaskRunner(cs.create)()
|
|
|
|
# this makes sure the auto increment worked on cloud server creation
|
|
self.assertTrue(cs.id > 0)
|
|
|
|
expected_public = return_server.networks['public'][0]
|
|
expected_private = return_server.networks['private'][0]
|
|
self.assertEqual(expected_public, cs.FnGetAtt('PublicIp'))
|
|
self.assertEqual(expected_private, cs.FnGetAtt('PrivateIp'))
|
|
self.assertEqual(expected_public, cs.FnGetAtt('PublicDnsName'))
|
|
self.assertEqual(expected_public, cs.FnGetAtt('PrivateDnsName'))
|
|
self.assertRaises(exception.InvalidTemplateAttribute,
|
|
cs.FnGetAtt, 'foo')
|
|
self.m.VerifyAll()
|
|
|
|
def test_cs_create_image_name_err(self):
|
|
self.m.StubOutWithMock(cloud_server.CloudServer, "nova")
|
|
cloud_server.CloudServer.nova().MultipleTimes().AndReturn(self.fc)
|
|
stack_name = 'test_cs_create_image_name_err_stack'
|
|
(t, stack) = self._setup_test_stack(stack_name)
|
|
|
|
# create a cloud server with non exist image name
|
|
t['Resources']['WebServer']['Properties']['image'] = 'Slackware'
|
|
|
|
# Mock flavors
|
|
cloud_server.CloudServer.script = None
|
|
self.m.ReplayAll()
|
|
|
|
cs = cloud_server.CloudServer('cs_create_image_err',
|
|
t['Resources']['WebServer'], stack)
|
|
|
|
self.assertRaises(exception.ImageNotFound, cs.validate)
|
|
self.m.VerifyAll()
|
|
|
|
def test_cs_create_image_name_okay(self):
|
|
self.m.StubOutWithMock(cloud_server.CloudServer, "nova")
|
|
cloud_server.CloudServer.nova().MultipleTimes().AndReturn(self.fc)
|
|
stack_name = 'test_cs_create_image_name_err_stack'
|
|
(t, stack) = self._setup_test_stack(stack_name)
|
|
|
|
# create a cloud server with non exist image name
|
|
t['Resources']['WebServer']['Properties']['image'] = 'CentOS 5.2'
|
|
t['Resources']['WebServer']['Properties']['user_data'] = ''
|
|
|
|
cloud_server.CloudServer.script = None
|
|
self.m.ReplayAll()
|
|
|
|
cs = cloud_server.CloudServer('cs_create_image_err',
|
|
t['Resources']['WebServer'], stack)
|
|
|
|
self.assertIsNone(cs.validate())
|
|
self.m.VerifyAll()
|
|
|
|
def test_cs_create_heatscript_nonzero_exit_status(self):
|
|
return_server = self.fc.servers.list()[1]
|
|
cs = self._setup_test_cs(return_server, 'test_cs_create_image_id',
|
|
exit_code=1)
|
|
self.m.ReplayAll()
|
|
create = scheduler.TaskRunner(cs.create)
|
|
exc = self.assertRaises(exception.ResourceFailure, create)
|
|
self.assertIn("The heat-script.sh script exited", str(exc))
|
|
self.m.VerifyAll()
|
|
|
|
def test_cs_create_cfnuserdata_nonzero_exit_status(self):
|
|
return_server = self.fc.servers.list()[1]
|
|
cs = self._setup_test_cs(return_server, 'test_cs_create_image_id',
|
|
exit_code=42)
|
|
self.m.ReplayAll()
|
|
create = scheduler.TaskRunner(cs.create)
|
|
exc = self.assertRaises(exception.ResourceFailure, create)
|
|
self.assertIn("The cfn-userdata script exited", str(exc))
|
|
self.m.VerifyAll()
|
|
|
|
def test_cs_update_cfnuserdata_nonzero_exit_status(self):
|
|
return_server = self.fc.servers.list()[1]
|
|
cs = self._create_test_cs(return_server,
|
|
'test_cs_update_cfnuserdata_nonzero_exit')
|
|
self.m.UnsetStubs()
|
|
self._update_test_cs(return_server,
|
|
'test_cs_update_cfnuserdata_nonzero_exit',
|
|
exit_code=1)
|
|
self.m.ReplayAll()
|
|
update_template = copy.deepcopy(cs.t)
|
|
update_template['Metadata'] = {'test': 123}
|
|
update = scheduler.TaskRunner(cs.update, update_template)
|
|
exc = self.assertRaises(exception.ResourceFailure, update)
|
|
self.assertIn("The cfn-userdata script exited", str(exc))
|
|
|
|
def test_cs_create_flavor_err(self):
|
|
"""validate() should throw an if the flavor is invalid."""
|
|
self.m.StubOutWithMock(cloud_server.CloudServer, "nova")
|
|
cloud_server.CloudServer.nova().MultipleTimes().AndReturn(self.fc)
|
|
stack_name = 'test_cs_create_flavor_err_stack'
|
|
(t, stack) = self._setup_test_stack(stack_name)
|
|
|
|
# create a cloud server with non exist image name
|
|
t['Resources']['WebServer']['Properties']['flavor'] = 'invalid'
|
|
|
|
# Mock flavors
|
|
self.m.ReplayAll()
|
|
|
|
cs = cloud_server.CloudServer('cs_create_flavor_err',
|
|
t['Resources']['WebServer'], stack)
|
|
|
|
self.assertRaises(exception.FlavorMissing, cs.validate)
|
|
|
|
self.m.VerifyAll()
|
|
|
|
def test_cs_create_delete(self):
|
|
return_server = self.fc.servers.list()[1]
|
|
cs = self._create_test_cs(return_server,
|
|
'test_cs_create_delete')
|
|
cs.resource_id = 1234
|
|
|
|
# this makes sure the auto-increment worked on cloud server creation
|
|
self.assertTrue(cs.id > 0)
|
|
|
|
self.m.StubOutWithMock(self.fc.client, 'get_servers_1234')
|
|
get = self.fc.client.get_servers_1234
|
|
get().AndRaise(novaclient.exceptions.NotFound(404))
|
|
mox.Replay(get)
|
|
|
|
scheduler.TaskRunner(cs.delete)()
|
|
self.assertIsNone(cs.resource_id)
|
|
self.assertEqual((cs.DELETE, cs.COMPLETE), cs.state)
|
|
self.m.VerifyAll()
|
|
|
|
def test_cs_update_metadata(self):
|
|
return_server = self.fc.servers.list()[1]
|
|
cs = self._create_test_cs(return_server, 'test_cs_metadata_update')
|
|
self.m.UnsetStubs()
|
|
self._update_test_cs(return_server, 'test_cs_metadata_update')
|
|
self.m.ReplayAll()
|
|
update_template = copy.deepcopy(cs.t)
|
|
update_template['Metadata'] = {'test': 123}
|
|
scheduler.TaskRunner(cs.update, update_template)()
|
|
self.assertEqual({'test': 123}, cs.metadata)
|
|
|
|
def test_cs_update_replace(self):
|
|
return_server = self.fc.servers.list()[1]
|
|
cs = self._create_test_cs(return_server, 'test_cs_update')
|
|
|
|
update_template = copy.deepcopy(cs.t)
|
|
update_template['Notallowed'] = {'test': 123}
|
|
updater = scheduler.TaskRunner(cs.update, update_template)
|
|
self.assertRaises(resource.UpdateReplace, updater)
|
|
|
|
def test_cs_update_properties(self):
|
|
return_server = self.fc.servers.list()[1]
|
|
cs = self._create_test_cs(return_server, 'test_cs_update')
|
|
|
|
update_template = copy.deepcopy(cs.t)
|
|
update_template['Properties']['user_data'] = 'mustreplace'
|
|
updater = scheduler.TaskRunner(cs.update, update_template)
|
|
self.assertRaises(resource.UpdateReplace, updater)
|
|
|
|
def test_cs_status_build(self):
|
|
return_server = self.fc.servers.list()[0]
|
|
cs = self._setup_test_cs(return_server, 'test_cs_status_build')
|
|
cs.resource_id = 1234
|
|
|
|
# Bind fake get method which cs.check_create_complete will call
|
|
def activate_status(server):
|
|
server.status = 'ACTIVE'
|
|
return_server.get = activate_status.__get__(return_server)
|
|
self.m.ReplayAll()
|
|
|
|
scheduler.TaskRunner(cs.create)()
|
|
self.assertEqual((cs.CREATE, cs.COMPLETE), cs.state)
|
|
|
|
def test_cs_status_hard_reboot(self):
|
|
self._test_cs_status_not_build_active('HARD_REBOOT')
|
|
|
|
def test_cs_status_password(self):
|
|
self._test_cs_status_not_build_active('PASSWORD')
|
|
|
|
def test_cs_status_reboot(self):
|
|
self._test_cs_status_not_build_active('REBOOT')
|
|
|
|
def test_cs_status_rescue(self):
|
|
self._test_cs_status_not_build_active('RESCUE')
|
|
|
|
def test_cs_status_resize(self):
|
|
self._test_cs_status_not_build_active('RESIZE')
|
|
|
|
def test_cs_status_revert_resize(self):
|
|
self._test_cs_status_not_build_active('REVERT_RESIZE')
|
|
|
|
def test_cs_status_shutoff(self):
|
|
self._test_cs_status_not_build_active('SHUTOFF')
|
|
|
|
def test_cs_status_suspended(self):
|
|
self._test_cs_status_not_build_active('SUSPENDED')
|
|
|
|
def test_cs_status_verify_resize(self):
|
|
self._test_cs_status_not_build_active('VERIFY_RESIZE')
|
|
|
|
def _test_cs_status_not_build_active(self, uncommon_status):
|
|
return_server = self.fc.servers.list()[0]
|
|
cs = self._setup_test_cs(return_server, 'test_cs_status_build')
|
|
cs.resource_id = 1234
|
|
|
|
# Bind fake get method which cs.check_create_complete will call
|
|
def activate_status(server):
|
|
if hasattr(server, '_test_check_iterations'):
|
|
server._test_check_iterations += 1
|
|
else:
|
|
server._test_check_iterations = 1
|
|
if server._test_check_iterations == 1:
|
|
server.status = uncommon_status
|
|
if server._test_check_iterations > 2:
|
|
server.status = 'ACTIVE'
|
|
return_server.get = activate_status.__get__(return_server)
|
|
self.m.ReplayAll()
|
|
|
|
scheduler.TaskRunner(cs.create)()
|
|
self.assertEqual((cs.CREATE, cs.COMPLETE), cs.state)
|
|
|
|
self.m.VerifyAll()
|
|
|
|
def mock_get_ip(self, cs):
|
|
self.m.UnsetStubs()
|
|
self.m.StubOutWithMock(cloud_server.CloudServer, "server")
|
|
cloud_server.CloudServer.server = cs
|
|
self.m.ReplayAll()
|
|
|
|
def test_cs_get_ip(self):
|
|
stack_name = 'test_cs_get_ip_err'
|
|
(t, stack) = self._setup_test_stack(stack_name)
|
|
cs = cloud_server.CloudServer('cs_create_image_err',
|
|
t['Resources']['WebServer'],
|
|
stack)
|
|
cs.addresses = {'public': [{'version': 4, 'addr': '4.5.6.7'},
|
|
{'version': 6, 'addr': 'fake:ip::6'}],
|
|
'private': [{'version': 4, 'addr': '10.13.12.13'}]}
|
|
self.mock_get_ip(cs)
|
|
self.assertEqual('4.5.6.7', cs.public_ip)
|
|
self.mock_get_ip(cs)
|
|
self.assertEqual('10.13.12.13', cs.private_ip)
|
|
|
|
cs.addresses = {'public': [],
|
|
'private': []}
|
|
self.mock_get_ip(cs)
|
|
self.assertRaises(exception.Error, cs._get_ip, 'public')
|
|
|
|
def test_private_key(self):
|
|
stack_name = 'test_private_key'
|
|
(t, stack) = self._setup_test_stack(stack_name)
|
|
cs = cloud_server.CloudServer('cs_private_key',
|
|
t['Resources']['WebServer'],
|
|
stack)
|
|
|
|
# This gives the fake cloud server an id and created_time attribute
|
|
cs._store_or_update(cs.CREATE, cs.IN_PROGRESS, 'test_store')
|
|
|
|
cs.private_key = 'fake private key'
|
|
self.ctx = utils.dummy_context()
|
|
rs = db_api.resource_get_by_name_and_stack(self.ctx,
|
|
'cs_private_key',
|
|
stack.id)
|
|
encrypted_key = rs.data[0]['value']
|
|
self.assertNotEqual(encrypted_key, "fake private key")
|
|
# Test private_key property returns decrypted value
|
|
self.assertEqual("fake private key", cs.private_key)
|
|
|
|
def test_rackconnect_deployed(self):
|
|
return_server = self.fc.servers.list()[1]
|
|
return_server.metadata = {'rackconnect_automation_status': 'DEPLOYED'}
|
|
self.m.StubOutWithMock(return_server, 'get')
|
|
return_server.get()
|
|
cs = self._setup_test_cs(return_server, 'test_rackconnect_deployed')
|
|
cs.context.roles = ['rack_connect']
|
|
self.m.ReplayAll()
|
|
scheduler.TaskRunner(cs.create)()
|
|
self.assertEqual('CREATE', cs.action)
|
|
self.assertEqual('COMPLETE', cs.status)
|
|
self.m.VerifyAll()
|
|
|
|
def test_rackconnect_failed(self):
|
|
return_server = self.fc.servers.list()[1]
|
|
return_server.metadata = {'rackconnect_automation_status': 'FAILED'}
|
|
self.m.StubOutWithMock(return_server, 'get')
|
|
return_server.get()
|
|
cs = self._setup_test_cs(return_server, 'test_rackconnect_failed')
|
|
cs.context.roles = ['rack_connect']
|
|
self.m.ReplayAll()
|
|
create = scheduler.TaskRunner(cs.create)
|
|
exc = self.assertRaises(exception.ResourceFailure, create)
|
|
self.assertEqual('Error: RackConnect automation FAILED', str(exc))
|
|
|
|
def test_rackconnect_unprocessable(self):
|
|
return_server = self.fc.servers.list()[1]
|
|
return_server.metadata = {'rackconnect_automation_status':
|
|
'UNPROCESSABLE',
|
|
'rackconnect_unprocessable_reason':
|
|
'Fake reason'}
|
|
self.m.StubOutWithMock(return_server, 'get')
|
|
return_server.get()
|
|
cs = self._setup_test_cs(return_server,
|
|
'test_rackconnect_unprocessable')
|
|
cs.context.roles = ['rack_connect']
|
|
self.m.ReplayAll()
|
|
scheduler.TaskRunner(cs.create)()
|
|
self.assertEqual('CREATE', cs.action)
|
|
self.assertEqual('COMPLETE', cs.status)
|
|
self.m.VerifyAll()
|
|
|
|
def test_rackconnect_unknown(self):
|
|
return_server = self.fc.servers.list()[1]
|
|
return_server.metadata = {'rackconnect_automation_status': 'FOO'}
|
|
self.m.StubOutWithMock(return_server, 'get')
|
|
return_server.get()
|
|
cs = self._setup_test_cs(return_server, 'test_rackconnect_unknown')
|
|
cs.context.roles = ['rack_connect']
|
|
self.m.ReplayAll()
|
|
create = scheduler.TaskRunner(cs.create)
|
|
exc = self.assertRaises(exception.ResourceFailure, create)
|
|
self.assertEqual('Error: Unknown RackConnect automation status: FOO',
|
|
str(exc))
|
|
|
|
def test_managed_cloud_complete(self):
|
|
return_server = self.fc.servers.list()[1]
|
|
return_server.metadata = {'rax_service_level_automation': 'Complete'}
|
|
self.m.StubOutWithMock(return_server, 'get')
|
|
return_server.get()
|
|
cs = self._setup_test_cs(return_server, 'test_managed_cloud_complete')
|
|
cs.context.roles = ['rax_managed']
|
|
self.m.ReplayAll()
|
|
scheduler.TaskRunner(cs.create)()
|
|
self.assertEqual('CREATE', cs.action)
|
|
self.assertEqual('COMPLETE', cs.status)
|
|
self.m.VerifyAll()
|
|
|
|
def test_managed_cloud_build_error(self):
|
|
return_server = self.fc.servers.list()[1]
|
|
return_server.metadata = {'rax_service_level_automation':
|
|
'Build Error'}
|
|
self.m.StubOutWithMock(return_server, 'get')
|
|
return_server.get()
|
|
cs = self._setup_test_cs(return_server,
|
|
'test_managed_cloud_build_error')
|
|
cs.context.roles = ['rax_managed']
|
|
self.m.ReplayAll()
|
|
create = scheduler.TaskRunner(cs.create)
|
|
exc = self.assertRaises(exception.ResourceFailure, create)
|
|
self.assertEqual('Error: Managed Cloud automation failed', str(exc))
|
|
|
|
def test_managed_cloud_unknown(self):
|
|
return_server = self.fc.servers.list()[1]
|
|
return_server.metadata = {'rax_service_level_automation': 'FOO'}
|
|
self.m.StubOutWithMock(return_server, 'get')
|
|
return_server.get()
|
|
cs = self._setup_test_cs(return_server, 'test_managed_cloud_unknown')
|
|
cs.context.roles = ['rax_managed']
|
|
self.m.ReplayAll()
|
|
create = scheduler.TaskRunner(cs.create)
|
|
exc = self.assertRaises(exception.ResourceFailure, create)
|
|
self.assertEqual('Error: Unknown Managed Cloud automation status: FOO',
|
|
str(exc))
|