helpers.py 6.9 KB

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