helpers.py 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  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. 'fdd9a6ee-2c9b-490c-91ac-ca9b7c134264'),
  52. "instance_type": os.environ.get('CB_INSTANCE_TYPE_OS', 'm1.nano'),
  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(provider, instance_name, launch_config=None):
  63. return provider.compute.instances.create(
  64. instance_name,
  65. get_provider_test_data(provider, 'image'),
  66. get_provider_test_data(provider, 'instance_type'),
  67. launch_config=launch_config)
  68. def get_provider_wait_interval(provider):
  69. if isinstance(provider, TestMockHelperMixin):
  70. return 0
  71. else:
  72. return 1
  73. def get_test_instance(provider, name):
  74. instance = create_test_instance(provider, name)
  75. instance.wait_till_ready(interval=get_provider_wait_interval(provider))
  76. return instance
  77. class TestMockHelperMixin(object):
  78. """
  79. A helper class that providers mock drivers can use to be notified when a
  80. test setup/teardown occurs. This is useful when activating libraries
  81. like HTTPretty which take over socket communications.
  82. """
  83. def setUpMock(self):
  84. """
  85. Called before a test is started.
  86. """
  87. raise NotImplementedError(
  88. 'TestMockHelperMixin.setUpMock not implemented')
  89. def tearDownMock(self):
  90. """
  91. Called before test teardown.
  92. """
  93. raise NotImplementedError(
  94. 'TestMockHelperMixin.tearDownMock not implemented by this'
  95. ' provider')
  96. class ProviderTestBase(object):
  97. """
  98. A dummy base class for Test Cases. Does not inherit from unittest.TestCase
  99. to avoid confusing test discovery by unittest and nose2. unittest.TestCase
  100. is injected as a base class by the generator, so calling the unittest
  101. constructor works correctly.
  102. """
  103. def __init__(self, methodName, provider):
  104. unittest.TestCase.__init__(self, methodName=methodName)
  105. self.provider = provider
  106. def setUp(self):
  107. if isinstance(self.provider, TestMockHelperMixin):
  108. self.provider.setUpMock()
  109. def tearDown(self):
  110. if isinstance(self.provider, TestMockHelperMixin):
  111. self.provider.tearDownMock()
  112. def get_test_wait_interval(self):
  113. return get_provider_wait_interval(self.provider)
  114. class ProviderTestCaseGenerator():
  115. """
  116. Generates test cases for all provider - testcase combinations.
  117. Detailed docs at test/__init__.py
  118. """
  119. def __init__(self, test_classes):
  120. self.all_test_classes = test_classes
  121. def create_provider_instance(self, provider_class):
  122. """
  123. Instantiate a default provider instance. All required connection
  124. settings are expected to be set as environment variables.
  125. """
  126. return provider_class({})
  127. def generate_new_test_class(self, name, testcase_class):
  128. """
  129. Generates a new type which inherits from the given testcase_class and
  130. unittest.TestCase
  131. """
  132. class_name = "{0}{1}".format(name, testcase_class.__name__)
  133. return type(class_name, (testcase_class, unittest.TestCase), {})
  134. def generate_test_suite_for_provider_testcase(
  135. self, provider_class, testcase_class):
  136. """
  137. Generate and return a suite of tests for a specific provider class and
  138. testcase combination
  139. """
  140. testloader = unittest.TestLoader()
  141. testnames = testloader.getTestCaseNames(testcase_class)
  142. suite = unittest.TestSuite()
  143. for name in testnames:
  144. generated_cls = self.generate_new_test_class(
  145. provider_class.__name__,
  146. testcase_class)
  147. suite.addTest(
  148. generated_cls(
  149. name,
  150. self.create_provider_instance(provider_class)))
  151. return suite
  152. def generate_test_suite_for_provider(self, provider_class):
  153. """
  154. Generate and return a suite of all available tests for a given provider
  155. class
  156. """
  157. suite = unittest.TestSuite()
  158. suites = [
  159. self.generate_test_suite_for_provider_testcase(
  160. provider_class, test_class)
  161. for test_class in self.all_test_classes]
  162. for s in suites:
  163. suite.addTest(s)
  164. return suite
  165. def generate_tests(self):
  166. """
  167. Generate and return a suite of tests for all provider and test class
  168. combinations
  169. """
  170. factory = CloudProviderFactory()
  171. use_mock_drivers = parse_bool(
  172. os.environ.get("CB_USE_MOCK_DRIVERS", True))
  173. provider_name = os.environ.get("CB_TEST_PROVIDER", None)
  174. if provider_name:
  175. provider_classes = [
  176. factory.get_provider_class(
  177. provider_name,
  178. get_mock=use_mock_drivers)]
  179. if not provider_classes[0]:
  180. raise ValueError(
  181. "Could not find specified test provider %s" %
  182. provider_name)
  183. else:
  184. provider_classes = factory.get_all_provider_classes(
  185. get_mock=use_mock_drivers)
  186. suite = unittest.TestSuite()
  187. suites = [
  188. self.generate_test_suite_for_provider(p) for p in provider_classes]
  189. for s in suites:
  190. suite.addTest(s)
  191. return suite