Improve debugging ability for functional tests

Use proper assert*() functions so that we can get some information out of
any failed tests.

Change-Id: I92d937015562371c2b39bbcf5b9cbd2b6ca19d52
Signed-off-by: Zane Bitter <zbitter@redhat.com>
This commit is contained in:
Zane Bitter 2012-09-20 11:14:53 +02:00
parent 4dab682364
commit a974326cd7
18 changed files with 156 additions and 145 deletions

View File

@ -31,9 +31,9 @@ class AutoScalingMultiAZSampleFunctionalTest(unittest.TestCase):
'DBUsername=dbuser', 'DBUsername=dbuser',
'DBPassword=' + os.environ['OS_PASSWORD']]) 'DBPassword=' + os.environ['OS_PASSWORD']])
self.stack = util.Stack(template, 'F17', 'x86_64', 'cfntools', self.stack = util.Stack(self, template, 'F17', 'x86_64', 'cfntools',
stack_paramstr) stack_paramstr)
self.WebServerGroup0 = util.Instance('WebServerGroup-0') self.WebServerGroup0 = util.Instance(self, 'WebServerGroup-0')
def tearDown(self): def tearDown(self):
pass pass
@ -54,7 +54,7 @@ class AutoScalingMultiAZSampleFunctionalTest(unittest.TestCase):
# Give the load balancer 2 minutes to react # Give the load balancer 2 minutes to react
sleep(2 * 60) sleep(2 * 60)
self.WebServerGroup1 = util.Instance('WebServerGroup-1') self.WebServerGroup1 = util.Instance(self, 'WebServerGroup-1')
# Verify the second instance gets launched # Verify the second instance gets launched
self.assertTrue(self.WebServerGroup1.exists()) self.assertTrue(self.WebServerGroup1.exists())
self.WebServerGroup1.wait_for_boot() self.WebServerGroup1.wait_for_boot()

View File

@ -41,7 +41,7 @@ class CfnApiFunctionalTest(unittest.TestCase):
Contrary to the nose docs, the class can be a unittest.TestCase subclass Contrary to the nose docs, the class can be a unittest.TestCase subclass
''' '''
@classmethod @classmethod
def setupAll(self): def setupAll(cls):
print "SETUPALL" print "SETUPALL"
template = 'WordPress_Single_Instance.template' template = 'WordPress_Single_Instance.template'
@ -49,37 +49,51 @@ class CfnApiFunctionalTest(unittest.TestCase):
'DBUsername=dbuser', 'DBUsername=dbuser',
'DBPassword=' + os.environ['OS_PASSWORD']]) 'DBPassword=' + os.environ['OS_PASSWORD']])
self.logical_resource_name = 'WikiDatabase' cls.logical_resource_name = 'WikiDatabase'
self.logical_resource_type = 'AWS::EC2::Instance' cls.logical_resource_type = 'AWS::EC2::Instance'
self.stack = util.Stack(template, 'F17', 'x86_64', 'cfntools',
# Just to get the assert*() methods
class CfnApiFunctions(unittest.TestCase):
@unittest.skip('Not a real test case')
def runTest(self):
pass
inst = CfnApiFunctions()
cls.stack = util.Stack(inst, template, 'F17', 'x86_64', 'cfntools',
stack_paramstr) stack_paramstr)
self.WikiDatabase = util.Instance(self.logical_resource_name) cls.WikiDatabase = util.Instance(inst, cls.logical_resource_name)
self.stack.create()
self.WikiDatabase.wait_for_boot()
self.WikiDatabase.check_cfntools()
self.WikiDatabase.wait_for_provisioning()
self.logical_resource_status = "CREATE_COMPLETE" try:
cls.stack.create()
cls.WikiDatabase.wait_for_boot()
cls.WikiDatabase.check_cfntools()
cls.WikiDatabase.wait_for_provisioning()
# Save some compiled regexes and strings for response validation cls.logical_resource_status = "CREATE_COMPLETE"
self.stack_id_re = re.compile("^arn:openstack:heat::admin:stacks/"
+ self.stack.stackname)
self.time_re = re.compile(
"[0-9]{4}-[0-9]{2}-[0-9]{2}T[0-9]{2}:[0-9]{2}:[0-9]{2}Z$")
self.description_re = re.compile("^AWS CloudFormation Sample Template")
self.stack_status = "CREATE_COMPLETE"
self.stack_status_reason = "Stack successfully created"
self.stack_timeout = str(60)
self.stack_disable_rollback = "True"
# Match the expected format for physical resource ID for an instance # Save some compiled regexes and strings for response validation
self.phys_res_id_re = re.compile( cls.stack_id_re = re.compile("^arn:openstack:heat::admin:stacks/"
"^[0-9a-z]*-[0-9a-z]*-[0-9a-z]*-[0-9a-z]*-[0-9a-z]*$") + cls.stack.stackname)
cls.time_re = re.compile(
"[0-9]{4}-[0-9]{2}-[0-9]{2}T[0-9]{2}:[0-9]{2}:[0-9]{2}Z$")
cls.description_re = re.compile(
"^AWS CloudFormation Sample Template")
cls.stack_status = "CREATE_COMPLETE"
cls.stack_status_reason = "Stack successfully created"
cls.stack_timeout = str(60)
cls.stack_disable_rollback = "True"
# Match the expected format for an instance's physical resource ID
cls.phys_res_id_re = re.compile(
"^[0-9a-z]*-[0-9a-z]*-[0-9a-z]*-[0-9a-z]*-[0-9a-z]*$")
except:
cls.stack.cleanup()
raise
@classmethod @classmethod
def teardownAll(self): def teardownAll(cls):
print "TEARDOWNALL" print "TEARDOWNALL"
self.stack.cleanup() cls.stack.cleanup()
def test_instance(self): def test_instance(self):
# ensure wordpress was installed by checking for expected # ensure wordpress was installed by checking for expected

View File

@ -33,9 +33,11 @@ class HAProxyFunctionalTest(unittest.TestCase):
'DBUsername=dbuser', 'DBUsername=dbuser',
'DBPassword=' + os.environ['OS_PASSWORD']]) 'DBPassword=' + os.environ['OS_PASSWORD']])
self.stack = util.Stack(wp_template, 'F17', 'x86_64', 'cfntools', self.stack = util.Stack(self, wp_template, 'F17', 'x86_64', 'cfntools',
wp_paramstr) wp_paramstr)
self.WikiDatabase = util.Instance('WikiDatabase') self.WikiDatabase = util.Instance(self, 'WikiDatabase')
self.hap_stack = None
def tearDown(self): def tearDown(self):
self.stack.cleanup() self.stack.cleanup()
@ -63,14 +65,16 @@ class HAProxyFunctionalTest(unittest.TestCase):
# So wordpress instance is up, we now launch the HAProxy instance # So wordpress instance is up, we now launch the HAProxy instance
# and prove wordpress is accessable via the proxy instance IP # and prove wordpress is accessable via the proxy instance IP
hap_stackname = 'hap_teststack'
hap_template = 'HAProxy_Single_Instance.template' hap_template = 'HAProxy_Single_Instance.template'
hap_paramstr = ';'.join(['InstanceType=m1.xlarge', hap_paramstr = ';'.join(['InstanceType=m1.xlarge',
"Server1=%s:80" % self.WikiDatabase.ip]) "Server1=%s:80" % self.WikiDatabase.ip])
self.hap_stack = util.Stack(hap_template, 'F17', 'x86_64', 'cfntools', self.hap_stack = util.Stack(self, hap_template,
hap_paramstr, stackname='hap_teststack') 'F17', 'x86_64', 'cfntools',
self.LoadBalancerInstance = util.Instance('LoadBalancerInstance', hap_paramstr, stackname=hap_stackname)
self.hap_stack.stackname) self.LoadBalancerInstance = util.Instance(self, 'LoadBalancerInstance',
hap_stackname)
self.hap_stack.create() self.hap_stack.create()
self.LoadBalancerInstance.wait_for_boot() self.LoadBalancerInstance.wait_for_boot()
self.LoadBalancerInstance.check_cfntools() self.LoadBalancerInstance.check_cfntools()

View File

@ -32,11 +32,11 @@ class OpenShiftFunctionalTest(unittest.TestCase):
template = 'OpenShift_Prebuilt_JEOS.template' template = 'OpenShift_Prebuilt_JEOS.template'
stack_paramstr = ';'.join(['InstanceType=m1.xlarge']) stack_paramstr = ';'.join(['InstanceType=m1.xlarge'])
self.stack = util.Stack(template, 'F16', 'x86_64', self.stack = util.Stack(self, template, 'F16', 'x86_64',
'cfntools-openshift', stack_paramstr) 'cfntools-openshift', stack_paramstr)
self.Node = util.Instance('OpenShiftNodeServer') self.Node = util.Instance(self, 'OpenShiftNodeServer')
self.Broker = util.Instance('OpenShiftBrokerServer') self.Broker = util.Instance(self, 'OpenShiftBrokerServer')
def test_instance(self): def test_instance(self):
self.stack.create() self.stack.create()

View File

@ -34,10 +34,10 @@ class WordPress2Instances(unittest.TestCase):
'DBUsername=dbuser', 'DBUsername=dbuser',
'DBPassword=' + os.environ['OS_PASSWORD']]) 'DBPassword=' + os.environ['OS_PASSWORD']])
self.stack = util.Stack(template, 'F17', 'x86_64', 'cfntools', self.stack = util.Stack(self, template, 'F17', 'x86_64', 'cfntools',
stack_paramstr) stack_paramstr)
self.DatabaseServer = util.Instance('DatabaseServer') self.DatabaseServer = util.Instance(self, 'DatabaseServer')
self.WebServer = util.Instance('WebServer') self.WebServer = util.Instance(self, 'WebServer')
def tearDown(self): def tearDown(self):
self.stack.cleanup() self.stack.cleanup()

View File

@ -34,11 +34,11 @@ class WordPress2InstancesWithEBS(unittest.TestCase):
'DBUsername=dbuser', 'DBUsername=dbuser',
'DBPassword=' + os.environ['OS_PASSWORD']]) 'DBPassword=' + os.environ['OS_PASSWORD']])
self.stack = util.Stack(template, 'F17', 'x86_64', 'cfntools', self.stack = util.Stack(self, template, 'F17', 'x86_64', 'cfntools',
stack_paramstr) stack_paramstr)
self.WikiDatabase = util.Instance('WikiDatabase') self.WikiDatabase = util.Instance(self, 'WikiDatabase')
self.WebServer = util.Instance('WebServer') self.WebServer = util.Instance(self, 'WebServer')
def tearDown(self): def tearDown(self):
self.stack.cleanup() self.stack.cleanup()

View File

@ -30,12 +30,13 @@ class WordPress2EBSEIPFunctionalTest(unittest.TestCase):
'DBUsername=dbuser', 'DBUsername=dbuser',
'DBPassword=' + os.environ['OS_PASSWORD']]) 'DBPassword=' + os.environ['OS_PASSWORD']])
self.stack = util.Stack(self.template, 'F17', 'x86_64', 'cfntools', self.stack = util.Stack(self, self.template,
stack_paramstr) 'F17', 'x86_64', 'cfntools',
stack_paramstr)
self.webserver = util.Instance('WebServer') self.webserver = util.Instance(self, 'WebServer')
self.database = util.Instance('WikiDatabase') self.database = util.Instance(self, 'WikiDatabase')
def tearDown(self): def tearDown(self):
self.stack.cleanup() self.stack.cleanup()

View File

@ -31,13 +31,13 @@ class WordPressComposedInstancesFunctionalTest(unittest.TestCase):
'DBUsername=dbuser', 'DBUsername=dbuser',
'DBPassword=' + os.environ['OS_PASSWORD']]) 'DBPassword=' + os.environ['OS_PASSWORD']])
self.stack = util.Stack(template, 'F17', 'x86_64', 'cfntools', self.stack = util.Stack(self, template, 'F17', 'x86_64', 'cfntools',
stack_paramstr) stack_paramstr)
self.WebServer = util.Instance('WebServer') self.WebServer = util.Instance(self, 'WebServer')
self.MySqlDatabaseServer = \ self.MySqlDatabaseServer = util.Instance(self,
util.Instance('DatabaseTemplate.MySqlDatabaseServer') 'DatabaseTemplate.MySqlDatabaseServer')
def tearDown(self): def tearDown(self):
self.stack.cleanup() self.stack.cleanup()

View File

@ -30,9 +30,9 @@ class WordPressFunctionalTest(unittest.TestCase):
'DBUsername=dbuser', 'DBUsername=dbuser',
'DBPassword=' + os.environ['OS_PASSWORD']]) 'DBPassword=' + os.environ['OS_PASSWORD']])
self.stack = util.Stack(template, 'F17', 'x86_64', 'cfntools', self.stack = util.Stack(self, template, 'F17', 'x86_64', 'cfntools',
stack_paramstr) stack_paramstr)
self.WikiDatabase = util.Instance('WikiDatabase') self.WikiDatabase = util.Instance(self, 'WikiDatabase')
def tearDown(self): def tearDown(self):
self.stack.cleanup() self.stack.cleanup()

View File

@ -31,9 +31,10 @@ class WordPressBotoFunctionalTest(unittest.TestCase):
'DBUsername=dbuser', 'DBUsername=dbuser',
'DBPassword=' + os.environ['OS_PASSWORD']]) 'DBPassword=' + os.environ['OS_PASSWORD']])
self.stack = util.StackBoto(template, 'F17', 'x86_64', 'cfntools', self.stack = util.StackBoto(self, template,
stack_paramstr) 'F17', 'x86_64', 'cfntools',
self.WikiDatabase = util.Instance('WikiDatabase') stack_paramstr)
self.WikiDatabase = util.Instance(self, 'WikiDatabase')
def tearDown(self): def tearDown(self):
self.stack.cleanup() self.stack.cleanup()

View File

@ -34,9 +34,9 @@ class WordPressSingleEBSFunctionalTest(unittest.TestCase):
'DBUsername=dbuser', 'DBUsername=dbuser',
'DBPassword=' + os.environ['OS_PASSWORD']]) 'DBPassword=' + os.environ['OS_PASSWORD']])
self.stack = util.Stack(template, 'F17', 'x86_64', 'cfntools', self.stack = util.Stack(self, template, 'F17', 'x86_64', 'cfntools',
stack_paramstr) stack_paramstr)
self.WikiDatabase = util.Instance('WikiDatabase') self.WikiDatabase = util.Instance(self, 'WikiDatabase')
def tearDown(self): def tearDown(self):
self.stack.cleanup() self.stack.cleanup()

View File

@ -34,9 +34,9 @@ class WordPressEBSEIPFunctionalTest(unittest.TestCase):
'DBUsername=dbuser', 'DBUsername=dbuser',
'DBPassword=' + os.environ['OS_PASSWORD']]) 'DBPassword=' + os.environ['OS_PASSWORD']])
self.stack = util.Stack(template, 'F17', 'x86_64', 'cfntools', self.stack = util.Stack(self, template, 'F17', 'x86_64', 'cfntools',
stack_paramstr) stack_paramstr)
self.WikiServer = util.Instance('WikiServer') self.WikiServer = util.Instance(self, 'WikiServer')
def tearDown(self): def tearDown(self):
self.stack.cleanup() self.stack.cleanup()

View File

@ -34,9 +34,9 @@ class WordPressEIPFunctionalTest(unittest.TestCase):
'DBUsername=dbuser', 'DBUsername=dbuser',
'DBPassword=' + os.environ['OS_PASSWORD']]) 'DBPassword=' + os.environ['OS_PASSWORD']])
self.stack = util.Stack(template, 'F17', 'x86_64', 'cfntools', self.stack = util.Stack(self, template, 'F17', 'x86_64', 'cfntools',
stack_paramstr) stack_paramstr)
self.WebServer = util.Instance('WebServer') self.WebServer = util.Instance(self, 'WebServer')
def tearDown(self): def tearDown(self):
self.stack.cleanup() self.stack.cleanup()

View File

@ -30,9 +30,9 @@ class HaFunctionalTest(unittest.TestCase):
'DBUsername=dbuser', 'DBUsername=dbuser',
'DBPassword=' + os.environ['OS_PASSWORD']]) 'DBPassword=' + os.environ['OS_PASSWORD']])
self.stack = util.Stack(template, 'F17', 'x86_64', 'cfntools', self.stack = util.Stack(self, template, 'F17', 'x86_64', 'cfntools',
stack_paramstr) stack_paramstr)
self.WikiDatabase = util.Instance('WikiDatabase') self.WikiDatabase = util.Instance(self, 'WikiDatabase')
def tearDown(self): def tearDown(self):
self.stack.cleanup() self.stack.cleanup()

View File

@ -31,9 +31,9 @@ class WordPressIHAFunctionalTest(unittest.TestCase):
'DBUsername=dbuser', 'DBUsername=dbuser',
'DBPassword=' + os.environ['OS_PASSWORD']]) 'DBPassword=' + os.environ['OS_PASSWORD']])
self.stack = util.Stack(template, 'F17', 'x86_64', 'cfntools', self.stack = util.Stack(self, template, 'F17', 'x86_64', 'cfntools',
stack_paramstr) stack_paramstr)
self.WikiDatabase = util.Instance('WikiDatabase') self.WikiDatabase = util.Instance(self, 'WikiDatabase')
def tearDown(self): def tearDown(self):
self.stack.cleanup() self.stack.cleanup()
@ -85,7 +85,7 @@ class WordPressIHAFunctionalTest(unittest.TestCase):
self.assertTrue(tries < 500) self.assertTrue(tries < 500)
# Create a new Instance object and wait for boot # Create a new Instance object and wait for boot
self.WikiDatabaseNew = util.Instance('WikiDatabase') self.WikiDatabaseNew = util.Instance(self, 'WikiDatabase')
self.WikiDatabaseNew.wait_for_boot() self.WikiDatabaseNew.wait_for_boot()
self.WikiDatabaseNew.check_cfntools() self.WikiDatabaseNew.check_cfntools()
self.WikiDatabaseNew.wait_for_provisioning() self.WikiDatabaseNew.wait_for_provisioning()

View File

@ -30,12 +30,12 @@ class WordPressWithLBFunctionalTest(unittest.TestCase):
'DBUsername=dbuser', 'DBUsername=dbuser',
'DBPassword=' + os.environ['OS_PASSWORD']]) 'DBPassword=' + os.environ['OS_PASSWORD']])
self.stack = util.Stack(template, 'F17', 'x86_64', 'cfntools', self.stack = util.Stack(self, template, 'F17', 'x86_64', 'cfntools',
stack_paramstr) stack_paramstr)
self.WikiServerOne = util.Instance('WikiServerOne') self.WikiServerOne = util.Instance(self, 'WikiServerOne')
self.LBInstance = util.Instance('LoadBalancer.LB_instance') self.LBInstance = util.Instance(self, 'LoadBalancer.LB_instance')
self.MySqlDatabaseServer = util.Instance( self.MySqlDatabaseServer = util.Instance(self,
'DatabaseServer.MySqlDatabaseServer') 'DatabaseServer.MySqlDatabaseServer')
def tearDown(self): def tearDown(self):

View File

@ -30,9 +30,9 @@ class WordPressRDSFunctionalTest(unittest.TestCase):
'DBUsername=dbuser', 'DBUsername=dbuser',
'DBPassword=' + os.environ['OS_PASSWORD']]) 'DBPassword=' + os.environ['OS_PASSWORD']])
self.stack = util.Stack(template, 'F17', 'x86_64', 'cfntools', self.stack = util.Stack(self, template, 'F17', 'x86_64', 'cfntools',
stack_paramstr) stack_paramstr)
self.WebServer = util.Instance('WebServer') self.WebServer = util.Instance(self, 'WebServer')
def tearDown(self): def tearDown(self):
self.stack.cleanup() self.stack.cleanup()

View File

@ -35,6 +35,7 @@ from nose.exc import SkipTest
from glance import client as glance_client from glance import client as glance_client
from novaclient.v1_1 import client as nova_client from novaclient.v1_1 import client as nova_client
import heat
from heat import utils from heat import utils
from heat.engine import parser from heat.engine import parser
from heat import client as heat_client from heat import client as heat_client
@ -44,8 +45,13 @@ from keystoneclient.v2_0 import client
DEFAULT_STACKNAME = 'teststack' DEFAULT_STACKNAME = 'teststack'
# this test is in heat/tests/functional, so go up 3 dirs
basepath = os.path.join(heat.__path__[0], os.path.pardir)
class Instance(object): class Instance(object):
def __init__(self, instance_name, stackname=DEFAULT_STACKNAME): def __init__(self, testcase, instance_name, stackname=DEFAULT_STACKNAME):
self.testcase = testcase
self.name = '%s.%s' % (stackname, instance_name) self.name = '%s.%s' % (stackname, instance_name)
# during nose test execution this file will be imported even if # during nose test execution this file will be imported even if
@ -55,9 +61,9 @@ class Instance(object):
except KeyError: except KeyError:
raise SkipTest('OS_AUTH_STRATEGY unset, skipping functional test') raise SkipTest('OS_AUTH_STRATEGY unset, skipping functional test')
if os.environ['OS_AUTH_STRATEGY'] != 'keystone': self.testcase.assertEqual(os.environ['OS_AUTH_STRATEGY'],
print 'keystone authentication required' 'keystone',
assert False 'keystone authentication required')
self.creds = dict(username=os.environ['OS_USERNAME'], self.creds = dict(username=os.environ['OS_USERNAME'],
password=os.environ['OS_PASSWORD'], password=os.environ['OS_PASSWORD'],
@ -66,10 +72,6 @@ class Instance(object):
strategy=os.environ['OS_AUTH_STRATEGY']) strategy=os.environ['OS_AUTH_STRATEGY'])
dbusername = 'testuser' dbusername = 'testuser'
# this test is in heat/tests/functional, so go up 3 dirs
basepath = os.path.abspath(
os.path.dirname(os.path.realpath(__file__)) + '/../../..')
self.novaclient = nova_client.Client(self.creds['username'], self.novaclient = nova_client.Client(self.creds['username'],
self.creds['password'], self.creds['tenant'], self.creds['password'], self.creds['tenant'],
self.creds['auth_url'], service_type='compute') self.creds['auth_url'], service_type='compute')
@ -91,7 +93,7 @@ class Instance(object):
self.ip = address.items()[0][1][0]['addr'] self.ip = address.items()[0][1][0]['addr']
time.sleep(10) time.sleep(10)
tries += 1 tries += 1
assert tries < 500 self.testcase.assertTrue(tries < 500, 'Timed out')
print 'Instance (%s) ip (%s) status (%s)' % (self.name, self.ip, print 'Instance (%s) ip (%s) status (%s)' % (self.name, self.ip,
server.status) server.status)
@ -104,7 +106,7 @@ class Instance(object):
(self.name, self.ip)) (self.name, self.ip))
time.sleep(10) time.sleep(10)
tries += 1 tries += 1
assert tries < 50 self.testcase.assertTrue(tries < 50, 'Timed out')
else: else:
print 'Instance (%s) ip (%s) SSH detected.' % (self.name, print 'Instance (%s) ip (%s) SSH detected.' % (self.name,
self.ip) self.ip)
@ -114,7 +116,7 @@ class Instance(object):
while True: while True:
try: try:
tries += 1 tries += 1
assert tries < 50 self.testcase.assertTrue(tries < 50, 'Timed out')
self.ssh.connect(self.ip, username='ec2-user', self.ssh.connect(self.ip, username='ec2-user',
allow_agent=True, look_for_keys=True, password='password') allow_agent=True, look_for_keys=True, password='password')
except paramiko.AuthenticationException: except paramiko.AuthenticationException:
@ -139,7 +141,7 @@ class Instance(object):
except IOError, e: except IOError, e:
tries += 1 tries += 1
if e.errno == errno.ENOENT: if e.errno == errno.ENOENT:
assert tries < 50 self.testcase.assertTrue(tries < 50, 'Timed out')
print("Instance (%s) ip (%s) not booted, waiting..." % print("Instance (%s) ip (%s) not booted, waiting..." %
(self.name, self.ip)) (self.name, self.ip))
time.sleep(15) time.sleep(15)
@ -165,7 +167,7 @@ class Instance(object):
print "Verifying file '%s' exists" % path print "Verifying file '%s' exists" % path
stdin, stdout, sterr = self.ssh.exec_command('ls "%s"' % path) stdin, stdout, sterr = self.ssh.exec_command('ls "%s"' % path)
lines = stdout.readlines() lines = stdout.readlines()
assert len(lines) == 1 self.testcase.assertEqual(len(lines), 1)
result = lines.pop().rstrip() result = lines.pop().rstrip()
return result == path return result == path
@ -195,7 +197,7 @@ class Instance(object):
data = files.pop().split(' ') data = files.pop().split(' ')
cur_file = data[1].rstrip() cur_file = data[1].rstrip()
if cur_file in cfn_tools_files: if cur_file in cfn_tools_files:
assert data[0] == cfntools[cur_file] self.testcase.assertEqual(data[0], cfntools[cur_file])
print 'Instance (%s) cfntools integrity verified.' % self.name print 'Instance (%s) cfntools integrity verified.' % self.name
def wait_for_provisioning(self): def wait_for_provisioning(self):
@ -207,7 +209,7 @@ class Instance(object):
except IOError, e: except IOError, e:
tries += 1 tries += 1
if e.errno == errno.ENOENT: if e.errno == errno.ENOENT:
assert tries < 500 self.testcase.assertTrue(tries < 500, 'Timed out')
print("Instance (%s) provisioning incomplete, waiting..." % print("Instance (%s) provisioning incomplete, waiting..." %
self.name) self.name)
time.sleep(15) time.sleep(15)
@ -229,7 +231,7 @@ class Instance(object):
# sudo chmod 777 /var/lib/cloud/instance/user-data.txt.i\n') # sudo chmod 777 /var/lib/cloud/instance/user-data.txt.i\n')
# time.sleep(1) # necessary for sendall to complete # time.sleep(1) # necessary for sendall to complete
f = open(self.basepath + '/templates/' + template_file) f = open(basepath + '/templates/' + template_file)
t = json.loads(f.read()) t = json.loads(f.read())
f.close() f.close()
@ -255,15 +257,15 @@ class Instance(object):
t_data_list.insert(len(t_data_list) - 1, t_data_list.insert(len(t_data_list) - 1,
u'touch /var/lib/cloud/instance/provision-finished') u'touch /var/lib/cloud/instance/provision-finished')
assert t_data_list == remote_file_list_u self.testcase.assertEqual(t_data_list, remote_file_list_u)
remote_file = self.sftp.open('/var/lib/cloud/instance/user-data.txt.i') remote_file = self.sftp.open('/var/lib/cloud/instance/user-data.txt.i')
msg = email.message_from_file(remote_file) msg = email.message_from_file(remote_file)
remote_file.close() remote_file.close()
filepaths = { filepaths = {
'cloud-config': self.basepath + '/heat/cloudinit/config', 'cloud-config': basepath + '/heat/cloudinit/config',
'part-handler.py': self.basepath + 'part-handler.py': basepath +
'/heat/cloudinit/part-handler.py' '/heat/cloudinit/part-handler.py'
} }
@ -278,7 +280,7 @@ class Instance(object):
if file in filepaths.keys(): if file in filepaths.keys():
with open(filepaths[file]) as f: with open(filepaths[file]) as f:
assert data == f.read() self.testcase.assertEqual(data, f.read())
def get_ssh_client(self): def get_ssh_client(self):
if self.ssh.get_transport() != None: if self.ssh.get_transport() != None:
@ -295,13 +297,27 @@ class Instance(object):
class Stack(object): class Stack(object):
def __init__(self, template_file, distribution, arch, jeos_type,
def __init__(self, testcase, template_file, distribution, arch, jeos_type,
stack_paramstr, stackname=DEFAULT_STACKNAME): stack_paramstr, stackname=DEFAULT_STACKNAME):
self.testcase = testcase
self.stackname = stackname self.stackname = stackname
self.template_file = template_file self.template_file = template_file
self.distribution = distribution self.distribution = distribution
self.stack_paramstr = stack_paramstr self.stack_paramstr = stack_paramstr
self.creds = dict(username=os.environ['OS_USERNAME'],
password=os.environ['OS_PASSWORD'],
tenant=os.environ['OS_TENANT_NAME'],
auth_url=os.environ['OS_AUTH_URL'],
strategy=os.environ['OS_AUTH_STRATEGY'])
self.dbusername = 'testuser'
self.testcase.assertEqual(os.environ['OS_AUTH_STRATEGY'],
'keystone',
'keystone authentication required')
self.prepare_jeos(distribution, arch, jeos_type) self.prepare_jeos(distribution, arch, jeos_type)
self.novaclient = nova_client.Client(self.creds['username'], self.novaclient = nova_client.Client(self.creds['username'],
@ -313,7 +329,7 @@ class Stack(object):
def create(self): def create(self):
self.keyname = self.novaclient.keypairs.list().pop().name self.keyname = self.novaclient.keypairs.list().pop().name
assert self.heatclient self.testcase.assertTrue(self.heatclient)
full_paramstr = ';'.join([self.stack_paramstr, full_paramstr = ';'.join([self.stack_paramstr,
'KeyName=' + self.keyname, 'KeyName=' + self.keyname,
@ -323,7 +339,9 @@ class Stack(object):
# Format parameters and create the stack # Format parameters and create the stack
parameters = {} parameters = {}
parameters['StackName'] = self.stackname parameters['StackName'] = self.stackname
template_path = self.basepath + '/templates/' + self.template_file template_path = os.path.join(basepath,
'templates',
self.template_file)
parameters['TemplateBody'] = open(template_path).read() parameters['TemplateBody'] = open(template_path).read()
parameters.update(self.heatclient.format_parameters(template_params)) parameters.update(self.heatclient.format_parameters(template_params))
result = self.heatclient.create_stack(**parameters) result = self.heatclient.create_stack(**parameters)
@ -334,25 +352,25 @@ class Stack(object):
tries = 0 tries = 0
print 'Waiting for stack creation to be completed' print 'Waiting for stack creation to be completed'
while self.in_state('CREATE_IN_PROGRESS'): while self.get_state() == 'CREATE_IN_PROGRESS':
tries += 1 tries += 1
assert tries < 500 self.testcase.assertTrue(tries < 500, 'Timed out')
time.sleep(10) time.sleep(10)
assert self.in_state('CREATE_COMPLETE') self.testcase.assertEqual(self.get_state(), 'CREATE_COMPLETE')
def _check_create_result(self, result): def _check_create_result(self, result):
# Check result looks OK # Check result looks OK
root = etree.fromstring(result) root = etree.fromstring(result)
create_list = root.xpath('/CreateStackResponse/CreateStackResult') create_list = root.xpath('/CreateStackResponse/CreateStackResult')
assert create_list self.testcase.assertTrue(create_list)
assert len(create_list) == 1 self.testcase.assertEqual(len(create_list), 1)
# Extract StackId from the result, and check the StackName part # Extract StackId from the result, and check the StackName part
stackid = create_list[0].findtext('StackId') stackid = create_list[0].findtext('StackId')
idname = stackid.split('/')[1] idname = stackid.split('/')[1]
print "Checking %s contains name %s" % (stackid, self.stackname) print "Checking %s contains name %s" % (stackid, self.stackname)
assert idname == self.stackname self.testcase.assertEqual(idname, self.stackname)
def _create_heat_client(self): def _create_heat_client(self):
return heat_client.get_client('0.0.0.0', 8000, return heat_client.get_client('0.0.0.0', 8000,
@ -360,32 +378,6 @@ class Stack(object):
self.creds['tenant'], self.creds['auth_url'], self.creds['tenant'], self.creds['auth_url'],
self.creds['strategy'], None, None, False) self.creds['strategy'], None, None, False)
# during nose test execution this file will be imported even if
# the unit tag was specified
try:
os.environ['OS_AUTH_STRATEGY']
except KeyError:
raise SkipTest('OS_AUTH_STRATEGY not set, skipping functional test')
if os.environ['OS_AUTH_STRATEGY'] != 'keystone':
print 'keystone authentication required'
assert False
creds = dict(username=os.environ['OS_USERNAME'],
password=os.environ['OS_PASSWORD'],
tenant=os.environ['OS_TENANT_NAME'],
auth_url=os.environ['OS_AUTH_URL'],
strategy=os.environ['OS_AUTH_STRATEGY'])
dbusername = 'testuser'
# this test is in heat/tests/functional, so go up 3 dirs
basepath = os.path.abspath(
os.path.dirname(os.path.realpath(__file__)) + '/../../..')
novaclient = None
glanceclient = None
heatclient = None
def get_state(self): def get_state(self):
stack_list = self.heatclient.list_stacks(StackName=self.stackname) stack_list = self.heatclient.list_stacks(StackName=self.stackname)
root = etree.fromstring(stack_list) root = etree.fromstring(stack_list)
@ -395,11 +387,10 @@ class Stack(object):
if len(alist): if len(alist):
item = alist.pop() item = alist.pop()
result = item.findtext("StackStatus") result = item.findtext("StackStatus")
if result and result.find('FAILED') >= 0:
print stack_list
return result return result
def in_state(self, state):
return state == self.get_state()
def cleanup(self): def cleanup(self):
parameters = {'StackName': self.stackname} parameters = {'StackName': self.stackname}
c = self.get_heat_client() c = self.get_heat_client()
@ -407,16 +398,17 @@ class Stack(object):
print 'Waiting for stack deletion to be completed' print 'Waiting for stack deletion to be completed'
tries = 0 tries = 0
while self.in_state('DELETE_IN_PROGRESS'): while self.get_state() == 'DELETE_IN_PROGRESS':
tries += 1 tries += 1
assert tries < 50 self.testcase.assertTrue(tries < 50, 'Timed out')
time.sleep(10) time.sleep(10)
# final state for all stacks is DELETE_COMPLETE, but then they # final state for all stacks is DELETE_COMPLETE, but then they
# dissappear hence no result from list_stacks/get_state # dissappear hence no result from list_stacks/get_state
# depending on timing, we could get either result here # depending on timing, we could get either result here
end_state = self.get_state() end_state = self.get_state()
assert (end_state == 'DELETE_COMPLETE' or end_state == None) if end_state is not None:
self.testcase.assertEqual(end_state, 'DELETE_COMPLETE')
def get_nova_client(self): def get_nova_client(self):
if self.novaclient != None: if self.novaclient != None:
@ -441,9 +433,8 @@ class Stack(object):
# skip creating jeos if image already available # skip creating jeos if image already available
if not self.poll_glance(self.glanceclient, imagename, False): if not self.poll_glance(self.glanceclient, imagename, False):
if os.geteuid() != 0: self.testcase.assertEqual(os.geteuid(), 0,
print 'test must be run as root to create jeos' 'No JEOS found - run as root to create')
assert False
# -d: debug, -G: register with glance # -d: debug, -G: register with glance
subprocess.call(['heat-jeos', '-d', '-G', 'create', imagename]) subprocess.call(['heat-jeos', '-d', '-G', 'create', imagename])
@ -457,7 +448,7 @@ class Stack(object):
tries = 0 tries = 0
while imagelistname != imagename: while imagelistname != imagename:
tries += 1 tries += 1
assert tries < 50 self.testcase.assertTrue(tries < 50, 'Timed out')
if block: if block:
time.sleep(15) time.sleep(15)
print "Checking glance for image registration" print "Checking glance for image registration"
@ -509,8 +500,8 @@ class Stack(object):
''' '''
root = etree.fromstring(response) root = etree.fromstring(response)
output_list = root.xpath(prefix) output_list = root.xpath(prefix)
assert output_list self.testcase.assertTrue(output_list)
assert len(output_list) == 1 self.testcase.assertEqual(len(output_list), 1)
output = output_list.pop() output = output_list.pop()
value = output.findtext(key) value = output.findtext(key)
return value return value
@ -534,12 +525,12 @@ class StackBoto(Stack):
auth_url=self.creds['auth_url']) auth_url=self.creds['auth_url'])
ksusers = keystone.users.list() ksusers = keystone.users.list()
ksuid = [u.id for u in ksusers if u.name == self.creds['username']] ksuid = [u.id for u in ksusers if u.name == self.creds['username']]
assert len(ksuid) == 1 self.testcase.assertEqual(len(ksuid), 1)
ec2creds = keystone.ec2.list(ksuid[0]) ec2creds = keystone.ec2.list(ksuid[0])
assert len(ec2creds) == 1 self.testcase.assertEqual(len(ec2creds), 1)
assert ec2creds[0].access self.testcase.assertTrue(ec2creds[0].access)
assert ec2creds[0].secret self.testcase.assertTrue(ec2creds[0].secret)
print "Got EC2 credentials from keystone" print "Got EC2 credentials from keystone"
# most of the arguments passed to heat_client_boto are for # most of the arguments passed to heat_client_boto are for
@ -571,7 +562,7 @@ class StackBoto(Stack):
return [e.physical_resource_id for e in events if match(e)] return [e.physical_resource_id for e in events if match(e)]
def _find_stack_output(self, result, output_key): def _find_stack_output(self, result, output_key):
assert len(result) == 1 self.testcase.assertEqual(len(result), 1)
for o in result[0].outputs: for o in result[0].outputs:
if o.key == output_key: if o.key == output_key: