helpers.py 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229
  1. from contextlib import contextmanager
  2. import os
  3. import sys
  4. import unittest
  5. from six import reraise
  6. from cloudbridge.cloud.factory import CloudProviderFactory
  7. def parse_bool(val):
  8. if val:
  9. return str(val).upper() in ['TRUE', 'YES']
  10. else:
  11. return False
  12. @contextmanager
  13. def cleanup_action(cleanup_func):
  14. """
  15. Context manager to carry out a given
  16. cleanup action after carrying out a set
  17. of tasks, or when an exception occurs.
  18. If any errors occur during the cleanup
  19. action, those are ignored, and the original
  20. traceback is preserved.
  21. :params func: This function is called if
  22. an exception occurs or at the end of the
  23. context block. If any exceptions raised
  24. by func are ignored.
  25. Usage:
  26. with cleanup_action(lambda e: print("Oops!")):
  27. do_something()
  28. """
  29. try:
  30. yield
  31. except Exception:
  32. ex_class, ex_val, ex_traceback = sys.exc_info()
  33. try:
  34. cleanup_func()
  35. except Exception as e:
  36. print("Error during exception cleanup: {0}".format(e))
  37. reraise(ex_class, ex_val, ex_traceback)
  38. try:
  39. cleanup_func()
  40. except Exception as e:
  41. print("Error during normal cleanup: {0}".format(e))
  42. TEST_DATA_CONFIG = {
  43. "AWSCloudProvider": {
  44. "image": os.environ.get('CB_IMAGE_AWS', 'ami-d85e75b0'),
  45. "instance_type": os.environ.get('CB_INSTANCE_TYPE_AWS',
  46. 't1.micro'),
  47. "placement": os.environ.get('CB_PLACEMENT_AWS', 'us-east-1a'),
  48. },
  49. "OpenStackCloudProvider": {
  50. "image": os.environ.get('CB_IMAGE_OS',
  51. 'eaf2fa40-25a3-44cd-b2f1-40de42a62154'),
  52. "instance_type": os.environ.get('CB_INSTANCE_TYPE_OS', 'm1.tiny'),
  53. "placement": os.environ.get('CB_PLACEMENT_OS', 'nova'),
  54. }
  55. }
  56. def get_provider_test_data(provider, key):
  57. if "AWSCloudProvider" in provider.name:
  58. return TEST_DATA_CONFIG.get("AWSCloudProvider").get(key)
  59. elif "OpenStackCloudProvider" in provider.name:
  60. return TEST_DATA_CONFIG.get("OpenStackCloudProvider").get(key)
  61. return None
  62. def create_test_instance(
  63. provider, instance_name, zone=None, launch_config=None):
  64. return provider.compute.instances.create(
  65. instance_name,
  66. get_provider_test_data(provider, 'image'),
  67. get_provider_test_data(provider, 'instance_type'),
  68. zone=zone,
  69. launch_config=launch_config)
  70. def get_provider_wait_interval(provider):
  71. if isinstance(provider, TestMockHelperMixin):
  72. return 0
  73. else:
  74. return 1
  75. def get_test_instance(provider, name):
  76. instance = create_test_instance(provider, name)
  77. instance.wait_till_ready(interval=get_provider_wait_interval(provider))
  78. return instance
  79. class TestMockHelperMixin(object):
  80. """
  81. A helper class that providers mock drivers can use to be notified when a
  82. test setup/teardown occurs. This is useful when activating libraries
  83. like HTTPretty which take over socket communications.
  84. """
  85. def setUpMock(self):
  86. """
  87. Called before a test is started.
  88. """
  89. raise NotImplementedError(
  90. 'TestMockHelperMixin.setUpMock not implemented')
  91. def tearDownMock(self):
  92. """
  93. Called before test teardown.
  94. """
  95. raise NotImplementedError(
  96. 'TestMockHelperMixin.tearDownMock not implemented by this'
  97. ' provider')
  98. class ProviderTestBase(object):
  99. """
  100. A dummy base class for Test Cases. Does not inherit from unittest.TestCase
  101. to avoid confusing test discovery by unittest and nose2. unittest.TestCase
  102. is injected as a base class by the generator, so calling the unittest
  103. constructor works correctly.
  104. """
  105. def __init__(self, methodName, provider):
  106. unittest.TestCase.__init__(self, methodName=methodName)
  107. self.provider = provider
  108. def setUp(self):
  109. if isinstance(self.provider, TestMockHelperMixin):
  110. self.provider.setUpMock()
  111. def tearDown(self):
  112. if isinstance(self.provider, TestMockHelperMixin):
  113. self.provider.tearDownMock()
  114. def get_test_wait_interval(self):
  115. return get_provider_wait_interval(self.provider)
  116. class ProviderTestCaseGenerator():
  117. """
  118. Generates test cases for all provider - testcase combinations.
  119. Detailed docs at test/__init__.py
  120. """
  121. def __init__(self, test_classes):
  122. self.all_test_classes = test_classes
  123. def create_provider_instance(self, provider_class):
  124. """
  125. Instantiate a default provider instance. All required connection
  126. settings are expected to be set as environment variables.
  127. """
  128. return provider_class({})
  129. def generate_new_test_class(self, name, testcase_class):
  130. """
  131. Generates a new type which inherits from the given testcase_class and
  132. unittest.TestCase
  133. """
  134. class_name = "{0}{1}".format(name, testcase_class.__name__)
  135. return type(class_name, (testcase_class, unittest.TestCase), {})
  136. def generate_test_suite_for_provider_testcase(
  137. self, provider_class, testcase_class):
  138. """
  139. Generate and return a suite of tests for a specific provider class and
  140. testcase combination
  141. """
  142. testloader = unittest.TestLoader()
  143. testnames = testloader.getTestCaseNames(testcase_class)
  144. suite = unittest.TestSuite()
  145. for name in testnames:
  146. generated_cls = self.generate_new_test_class(
  147. provider_class.__name__,
  148. testcase_class)
  149. suite.addTest(
  150. generated_cls(
  151. name,
  152. self.create_provider_instance(provider_class)))
  153. return suite
  154. def generate_test_suite_for_provider(self, provider_class):
  155. """
  156. Generate and return a suite of all available tests for a given provider
  157. class
  158. """
  159. suite = unittest.TestSuite()
  160. suites = [
  161. self.generate_test_suite_for_provider_testcase(
  162. provider_class, test_class)
  163. for test_class in self.all_test_classes]
  164. for s in suites:
  165. suite.addTest(s)
  166. return suite
  167. def generate_tests(self):
  168. """
  169. Generate and return a suite of tests for all provider and test class
  170. combinations
  171. """
  172. factory = CloudProviderFactory()
  173. use_mock_drivers = parse_bool(
  174. os.environ.get("CB_USE_MOCK_DRIVERS", True))
  175. provider_name = os.environ.get("CB_TEST_PROVIDER", None)
  176. if provider_name:
  177. provider_classes = [
  178. factory.get_provider_class(
  179. provider_name,
  180. get_mock=use_mock_drivers)]
  181. if not provider_classes[0]:
  182. raise ValueError(
  183. "Could not find specified test provider %s" %
  184. provider_name)
  185. else:
  186. provider_classes = factory.get_all_provider_classes(
  187. get_mock=use_mock_drivers)
  188. suite = unittest.TestSuite()
  189. suites = [
  190. self.generate_test_suite_for_provider(p) for p in provider_classes]
  191. for s in suites:
  192. suite.addTest(s)
  193. return suite