__init__.py 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294
  1. import functools
  2. import operator
  3. import os
  4. import sys
  5. import unittest
  6. import uuid
  7. import tenacity
  8. from cloudbridge.base import helpers as cb_helpers
  9. from cloudbridge.factory import CloudProviderFactory
  10. from cloudbridge.interfaces import CloudProvider
  11. from cloudbridge.interfaces import InstanceState
  12. from cloudbridge.interfaces import TestMockHelperMixin
  13. from cloudbridge.interfaces.resources import FloatingIpState
  14. from cloudbridge.interfaces.resources import NetworkState
  15. from cloudbridge.interfaces.resources import SubnetState
  16. def parse_bool(val):
  17. if val:
  18. return str(val).upper() in ['TRUE', 'YES']
  19. else:
  20. return False
  21. def skipIfNoService(services):
  22. """
  23. A decorator for skipping tests if the provider
  24. does not implement a given service.
  25. """
  26. def wrap(func):
  27. """
  28. The actual wrapper
  29. """
  30. @functools.wraps(func)
  31. def wrapper(self, *args, **kwargs):
  32. provider = getattr(self, 'provider')
  33. if provider:
  34. for service in services:
  35. if not provider.has_service(service):
  36. self.skipTest("Skipping test because '%s' service is"
  37. " not implemented" % (service,))
  38. func(self, *args, **kwargs)
  39. return wrapper
  40. return wrap
  41. def skipIfPython(op, major, minor):
  42. """
  43. A decorator for skipping tests if the python
  44. version doesn't match
  45. """
  46. def stringToOperator(op):
  47. op_map = {
  48. "=": operator.eq,
  49. "==": operator.eq,
  50. "<": operator.lt,
  51. "<=": operator.le,
  52. ">": operator.gt,
  53. ">=": operator.ge,
  54. }
  55. return op_map.get(op)
  56. def wrap(func):
  57. """
  58. The actual wrapper
  59. """
  60. @functools.wraps(func)
  61. def wrapper(self, *args, **kwargs):
  62. op_func = stringToOperator(op)
  63. if op_func(sys.version_info, (major, minor)):
  64. self.skipTest(
  65. "Skipping test because python version {0} is {1} expected"
  66. " version {2}".format(sys.version_info[:2],
  67. op, (major, minor)))
  68. func(self, *args, **kwargs)
  69. return wrapper
  70. return wrap
  71. TEST_DATA_CONFIG = {
  72. "AWSCloudProvider": {
  73. # Match the ami value with entry in custom_amis.json for use with moto
  74. "image": cb_helpers.get_env('CB_IMAGE_AWS', 'ami-aa2ea6d0'),
  75. "vm_type": cb_helpers.get_env('CB_VM_TYPE_AWS', 't2.nano'),
  76. "placement": cb_helpers.get_env('CB_PLACEMENT_AWS', 'us-east-1a'),
  77. "placement_cfg_key": "aws_zone_name"
  78. },
  79. 'OpenStackCloudProvider': {
  80. 'image': cb_helpers.get_env('CB_IMAGE_OS',
  81. 'c66bdfa1-62b1-43be-8964-e9ce208ac6a5'),
  82. "vm_type": cb_helpers.get_env('CB_VM_TYPE_OS', 'm1.tiny'),
  83. "placement": cb_helpers.get_env('CB_PLACEMENT_OS', 'nova'),
  84. "placement_cfg_key": "os_zone_name"
  85. },
  86. 'GCPCloudProvider': {
  87. 'image': cb_helpers.get_env(
  88. 'CB_IMAGE_GCP',
  89. 'https://www.googleapis.com/compute/v1/projects/ubuntu-os-cloud/'
  90. 'global/images/ubuntu-1804-bionic-v20200908'),
  91. 'vm_type': cb_helpers.get_env('CB_VM_TYPE_GCP', 'f1-micro'),
  92. 'placement': cb_helpers.get_env('GCP_ZONE_NAME', 'us-central1-a'),
  93. "placement_cfg_key": "gcp_zone_name"
  94. },
  95. "AzureCloudProvider": {
  96. "image":
  97. cb_helpers.get_env('CB_IMAGE_AZURE',
  98. 'Canonical:0001-com-ubuntu-minimal-jammy:minimal-22_04-lts-gen2:latest'),
  99. "vm_type": cb_helpers.get_env('CB_VM_TYPE_AZURE', 'Standard_DC1ds_v3'),
  100. "placement": cb_helpers.get_env('CB_PLACEMENT_AZURE', 'eastus'),
  101. "placement_cfg_key": "azure_zone_name"
  102. }
  103. }
  104. def get_provider_test_data(provider, key):
  105. provider_id = (provider.PROVIDER_ID if isinstance(provider, CloudProvider)
  106. else provider)
  107. if "aws" == provider_id:
  108. return TEST_DATA_CONFIG.get("AWSCloudProvider").get(key)
  109. if "mock" == provider_id:
  110. return TEST_DATA_CONFIG.get("AWSCloudProvider").get(key)
  111. elif "openstack" == provider_id:
  112. return TEST_DATA_CONFIG.get("OpenStackCloudProvider").get(key)
  113. elif "gcp" == provider_id:
  114. return TEST_DATA_CONFIG.get("GCPCloudProvider").get(key)
  115. elif "azure" == provider_id:
  116. return TEST_DATA_CONFIG.get("AzureCloudProvider").get(key)
  117. return None
  118. @tenacity.retry(
  119. stop=tenacity.stop_after_attempt(10),
  120. wait=tenacity.wait_fixed(5),
  121. reraise=True,
  122. )
  123. def get_or_create_default_subnet(provider):
  124. """
  125. Return the default subnet to be used for tests.
  126. Multiple pytest-xdist workers can call this concurrently at cold-start
  127. and race on the singleton default vnet/subnet (on Azure ARM that shows
  128. up as AnotherOperationInProgress / cancelled-and-superseded / subnet
  129. IP-range overlap). Retry with a short backoff so the loser re-runs the
  130. find() and inherits whatever the winner just created.
  131. """
  132. return provider.networking.subnets.get_or_create_default()
  133. def cleanup_subnet(subnet):
  134. if subnet:
  135. subnet.delete()
  136. subnet.wait_for([SubnetState.UNKNOWN],
  137. terminal_states=[SubnetState.ERROR])
  138. def cleanup_network(network):
  139. """
  140. Delete the supplied network, first deleting any contained subnets.
  141. """
  142. if network:
  143. try:
  144. for sn in network.subnets:
  145. with cb_helpers.cleanup_action(lambda: cleanup_subnet(sn)):
  146. pass
  147. finally:
  148. network.delete()
  149. network.wait_for([NetworkState.UNKNOWN],
  150. terminal_states=[NetworkState.ERROR])
  151. def cleanup_fip(fip):
  152. if fip:
  153. fip.delete()
  154. fip.wait_for([FloatingIpState.UNKNOWN],
  155. terminal_states=[FloatingIpState.ERROR])
  156. def get_test_gateway(provider):
  157. """
  158. Get an internet gateway for testing.
  159. This includes creating a network for the gateway, which is also returned.
  160. """
  161. sn = get_or_create_default_subnet(provider)
  162. net = sn.network
  163. return net.gateways.get_or_create()
  164. def cleanup_gateway(gateway):
  165. """
  166. Delete the supplied network and gateway.
  167. """
  168. with cb_helpers.cleanup_action(lambda: gateway.delete()):
  169. pass
  170. def create_test_instance(
  171. provider, instance_label, subnet, launch_config=None,
  172. key_pair=None, vm_firewalls=None, user_data=None):
  173. instance = provider.compute.instances.create(
  174. instance_label, get_provider_test_data(provider, 'image'),
  175. get_provider_test_data(provider, 'vm_type'),
  176. subnet=subnet,
  177. key_pair=key_pair,
  178. vm_firewalls=vm_firewalls,
  179. launch_config=launch_config,
  180. user_data=user_data)
  181. return instance
  182. def get_test_instance(provider, label, key_pair=None, vm_firewalls=None,
  183. subnet=None, user_data=None):
  184. launch_config = None
  185. instance = create_test_instance(
  186. provider,
  187. label,
  188. subnet=subnet,
  189. key_pair=key_pair,
  190. vm_firewalls=vm_firewalls,
  191. launch_config=launch_config,
  192. user_data=user_data)
  193. instance.wait_till_ready()
  194. return instance
  195. def get_test_fixtures_folder():
  196. return os.path.join(os.path.dirname(__file__), '../fixtures/')
  197. def delete_instance(instance):
  198. if instance:
  199. instance.delete()
  200. instance.wait_for([InstanceState.DELETED, InstanceState.UNKNOWN],
  201. terminal_states=[InstanceState.ERROR])
  202. def cleanup_test_resources(instance=None, vm_firewall=None,
  203. key_pair=None, network=None):
  204. """Clean up any combination of supplied resources."""
  205. with cb_helpers.cleanup_action(
  206. lambda: cleanup_network(network) if network else None):
  207. with cb_helpers.cleanup_action(
  208. lambda: key_pair.delete() if key_pair else None):
  209. with cb_helpers.cleanup_action(
  210. lambda: vm_firewall.delete() if vm_firewall else None):
  211. delete_instance(instance)
  212. def get_uuid():
  213. return str(uuid.uuid4())[:6]
  214. class ProviderTestBase(unittest.TestCase):
  215. _provider = None
  216. def setUp(self):
  217. if isinstance(self.provider, TestMockHelperMixin):
  218. self.provider.setUpMock()
  219. def tearDown(self):
  220. if isinstance(self.provider, TestMockHelperMixin):
  221. self.provider.tearDownMock()
  222. self._provider = None
  223. def get_provider_wait_interval(self, provider_class):
  224. if issubclass(provider_class, TestMockHelperMixin):
  225. return 0
  226. else:
  227. return 1
  228. def create_provider_instance(self):
  229. provider_name = cb_helpers.get_env("CB_TEST_PROVIDER", "aws")
  230. zone_cfg_key = get_provider_test_data(provider_name,
  231. 'placement_cfg_key')
  232. factory = CloudProviderFactory()
  233. provider_class = factory.get_provider_class(provider_name)
  234. config = {
  235. 'default_wait_interval': self.get_provider_wait_interval(
  236. provider_class),
  237. 'default_result_limit': 5,
  238. zone_cfg_key: get_provider_test_data(provider_name, 'placement')
  239. }
  240. return provider_class(config)
  241. @property
  242. def provider(self):
  243. if not self._provider:
  244. self._provider = self.create_provider_instance()
  245. return self._provider