__init__.py 11 KB

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