test_network_service.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286
  1. from cloudbridge.cloud.base import helpers as cb_helpers
  2. from cloudbridge.cloud.base.resources import BaseNetwork
  3. from cloudbridge.cloud.interfaces.resources import FloatingIP
  4. from cloudbridge.cloud.interfaces.resources import Network
  5. from cloudbridge.cloud.interfaces.resources import NetworkState
  6. from cloudbridge.cloud.interfaces.resources import RouterState
  7. from cloudbridge.cloud.interfaces.resources import Subnet
  8. from cloudbridge.cloud.interfaces.resources import SubnetState
  9. import test.helpers as helpers
  10. from test.helpers import ProviderTestBase
  11. from test.helpers import get_provider_test_data
  12. from test.helpers import standard_interface_tests as sit
  13. class CloudNetworkServiceTestCase(ProviderTestBase):
  14. _multiprocess_can_split_ = True
  15. @helpers.skipIfNoService(['networking.subnets',
  16. 'networking.networks',
  17. 'networking.routers'])
  18. def test_storage_services_event_pattern(self):
  19. expected_event = ".".join((self.provider.PROVIDER_ID,
  20. "networking.networks"))
  21. # pylint:disable=protected-access
  22. self.assertEqual(
  23. self.provider.networking.networks._service_event_pattern,
  24. expected_event,
  25. "Event pattern for {} service should be '{}', "
  26. "but found '{}'.".format("networks",
  27. expected_event,
  28. self.provider.networking.networks.
  29. _service_event_pattern))
  30. expected_event = ".".join((self.provider.PROVIDER_ID,
  31. "networking.subnets"))
  32. # pylint:disable=protected-access
  33. self.assertEqual(
  34. self.provider.networking.subnets._service_event_pattern,
  35. expected_event,
  36. "Event pattern for {} service should be '{}', "
  37. "but found '{}'.".format("subnets",
  38. expected_event,
  39. self.provider.networking.subnets.
  40. _service_event_pattern))
  41. expected_event = ".".join((self.provider.PROVIDER_ID,
  42. "networking.routers"))
  43. # pylint:disable=protected-access
  44. self.assertEqual(
  45. self.provider.networking.routers._service_event_pattern,
  46. expected_event,
  47. "Event pattern for {} service should be '{}', "
  48. "but found '{}'.".format("routers",
  49. expected_event,
  50. self.provider.networking.routers.
  51. _service_event_pattern))
  52. @helpers.skipIfNoService(['networking.networks'])
  53. def test_crud_network(self):
  54. def create_net(label):
  55. return self.provider.networking.networks.create(
  56. label=label, cidr_block=BaseNetwork.CB_DEFAULT_IPV4RANGE)
  57. def cleanup_net(net):
  58. if net:
  59. net.delete()
  60. net.wait_for([NetworkState.UNKNOWN],
  61. terminal_states=[NetworkState.ERROR])
  62. self.assertTrue(
  63. net.state == NetworkState.UNKNOWN,
  64. "Network.state must be unknown after "
  65. "a delete but got %s"
  66. % net.state)
  67. sit.check_crud(self, self.provider.networking.networks, Network,
  68. "cb-crudnetwork", create_net, cleanup_net)
  69. @helpers.skipIfNoService(['networking.networks'])
  70. def test_network_properties(self):
  71. label = 'cb-propnetwork-{0}'.format(helpers.get_uuid())
  72. subnet_label = 'cb-propsubnet-{0}'.format(helpers.get_uuid())
  73. net = self.provider.networking.networks.create(
  74. label=label, cidr_block=BaseNetwork.CB_DEFAULT_IPV4RANGE)
  75. with cb_helpers.cleanup_action(lambda: helpers.cleanup_network(net)):
  76. net.wait_till_ready()
  77. self.assertEqual(
  78. net.state, 'available',
  79. "Network in state '%s', yet should be 'available'" % net.state)
  80. sit.check_repr(self, net)
  81. self.assertIn(
  82. net.cidr_block, ['', BaseNetwork.CB_DEFAULT_IPV4RANGE],
  83. "Network CIDR %s does not contain the expected value %s."
  84. % (net.cidr_block, BaseNetwork.CB_DEFAULT_IPV4RANGE))
  85. cidr = '10.0.20.0/24'
  86. sn = net.subnets.create(
  87. label=subnet_label, cidr_block=cidr,
  88. zone=helpers.get_provider_test_data(self.provider,
  89. 'placement'))
  90. with cb_helpers.cleanup_action(lambda: helpers.cleanup_subnet(sn)):
  91. self.assertTrue(
  92. sn in net.subnets,
  93. "Subnet ID %s should be listed in network subnets %s."
  94. % (sn.id, net.subnets))
  95. self.assertTrue(
  96. sn in self.provider.networking.subnets.list(network=net),
  97. "Subnet ID %s should be included in the subnets list %s."
  98. % (sn.id, self.provider.networking.subnets.list(net))
  99. )
  100. self.assertListEqual(
  101. list(net.subnets), [sn],
  102. "Network should have exactly one subnet: %s." % sn.id)
  103. self.assertEqual(
  104. net.id, sn.network_id,
  105. "Network ID %s and subnet's network id %s should be"
  106. " equal." % (net.id, sn.network_id))
  107. self.assertEqual(
  108. net, sn.network,
  109. "Network obj %s and subnet's parent net obj %s"
  110. " should be equal." % (net, sn.network))
  111. self.assertEqual(
  112. cidr, sn.cidr_block,
  113. "Should be exact cidr block that was requested")
  114. self.assertTrue(
  115. BaseNetwork.cidr_blocks_overlap(cidr, sn.cidr_block),
  116. "Subnet's CIDR %s should overlap the specified one %s." % (
  117. sn.cidr_block, cidr))
  118. def test_crud_subnet(self):
  119. # Late binding will make sure that create_subnet gets the
  120. # correct value
  121. net = self.provider.networking.networks.create(
  122. label="cb-crudsubnet",
  123. cidr_block=BaseNetwork.CB_DEFAULT_IPV4RANGE)
  124. def create_subnet(label):
  125. return self.provider.networking.subnets.create(
  126. label=label, network=net, cidr_block="10.0.10.0/24",
  127. zone=helpers.get_provider_test_data(
  128. self.provider, 'placement'))
  129. def cleanup_subnet(subnet):
  130. if subnet:
  131. net = subnet.network
  132. subnet.delete()
  133. subnet.wait_for([SubnetState.UNKNOWN],
  134. terminal_states=[SubnetState.ERROR])
  135. self.assertTrue(
  136. subnet.state == SubnetState.UNKNOWN,
  137. "Subnet.state must be unknown after "
  138. "a delete but got %s"
  139. % subnet.state)
  140. net.delete()
  141. net.wait_for([NetworkState.UNKNOWN],
  142. terminal_states=[NetworkState.ERROR])
  143. self.assertTrue(
  144. net.state == NetworkState.UNKNOWN,
  145. "Network.state must be unknown after "
  146. "a delete but got %s"
  147. % net.state)
  148. sit.check_crud(self, self.provider.networking.subnets, Subnet,
  149. "cb-crudsubnet", create_subnet, cleanup_subnet)
  150. def test_crud_floating_ip(self):
  151. gw = helpers.get_test_gateway(
  152. self.provider)
  153. def create_fip(label):
  154. fip = gw.floating_ips.create()
  155. return fip
  156. def cleanup_fip(fip):
  157. if fip:
  158. gw.floating_ips.delete(fip.id)
  159. with cb_helpers.cleanup_action(
  160. lambda: helpers.cleanup_gateway(gw)):
  161. sit.check_crud(self, gw.floating_ips, FloatingIP,
  162. "cb-crudfip", create_fip, cleanup_fip,
  163. skip_name_check=True)
  164. def test_floating_ip_properties(self):
  165. # Check floating IP address
  166. gw = helpers.get_test_gateway(
  167. self.provider)
  168. fip = gw.floating_ips.create()
  169. with cb_helpers.cleanup_action(
  170. lambda: helpers.cleanup_gateway(gw)):
  171. with cb_helpers.cleanup_action(lambda: fip.delete()):
  172. fipl = list(gw.floating_ips)
  173. self.assertIn(fip, fipl)
  174. # 2016-08: address filtering not implemented in moto
  175. # empty_ipl = self.provider.network.floating_ips('dummy-net')
  176. # self.assertFalse(
  177. # empty_ipl,
  178. # "Bogus network should not have any floating IPs: {0}"
  179. # .format(empty_ipl))
  180. self.assertFalse(
  181. fip.private_ip,
  182. "Floating IP should not have a private IP value ({0})."
  183. .format(fip.private_ip))
  184. self.assertFalse(
  185. fip.in_use,
  186. "Newly created floating IP address should not be in use.")
  187. @helpers.skipIfNoService(['networking.routers'])
  188. def test_crud_router(self):
  189. def _cleanup(net, subnet, router, gateway):
  190. with cb_helpers.cleanup_action(
  191. lambda: helpers.cleanup_network(net)):
  192. with cb_helpers.cleanup_action(
  193. lambda: helpers.cleanup_subnet(subnet)):
  194. with cb_helpers.cleanup_action(
  195. lambda: router.delete()):
  196. with cb_helpers.cleanup_action(
  197. lambda: helpers.cleanup_gateway(gateway)):
  198. router.detach_subnet(subnet)
  199. router.detach_gateway(gateway)
  200. label = 'cb-crudrouter-{0}'.format(helpers.get_uuid())
  201. # Declare these variables and late binding will allow
  202. # the cleanup method access to the most current values
  203. net = None
  204. sn = None
  205. router = None
  206. gteway = None
  207. with cb_helpers.cleanup_action(
  208. lambda: _cleanup(net, sn, router, gteway)):
  209. net = self.provider.networking.networks.create(
  210. label=label, cidr_block=BaseNetwork.CB_DEFAULT_IPV4RANGE)
  211. router = self.provider.networking.routers.create(label=label,
  212. network=net)
  213. cidr = '10.0.15.0/24'
  214. sn = net.subnets.create(label=label, cidr_block=cidr,
  215. zone=helpers.get_provider_test_data(
  216. self.provider, 'placement'))
  217. # Check basic router properties
  218. sit.check_standard_behaviour(
  219. self, self.provider.networking.routers, router)
  220. if (self.provider.PROVIDER_ID != 'gcp'):
  221. self.assertEqual(
  222. router.state, RouterState.DETACHED,
  223. "Router {0} state {1} should be {2}.".format(
  224. router.id, router.state, RouterState.DETACHED))
  225. # self.assertEqual(
  226. # router.network_id, net.id, "Router {0} should be assoc."
  227. # " with network {1}, but is associated with {2}"
  228. # .format(router.id, net.id, router.network_id))
  229. self.assertTrue(
  230. len(router.subnets) == 0,
  231. "No subnet should be attached to router {1}".format(
  232. sn, router)
  233. )
  234. router.attach_subnet(sn)
  235. self.assertTrue(
  236. len(router.subnets) == 1,
  237. "Subnet {0} not attached to router {1}".format(sn, router)
  238. )
  239. gteway = net.gateways.get_or_create()
  240. router.attach_gateway(gteway)
  241. # TODO: add a check for routes after that's been implemented
  242. sit.check_delete(self, self.provider.networking.routers, router)
  243. # Also make sure that linked resources were properly cleaned up
  244. sit.check_delete(self, self.provider.networking.subnets, sn)
  245. sit.check_delete(self, self.provider.networking.networks, net)
  246. @helpers.skipIfNoService(['networking.networks'])
  247. def test_default_network(self):
  248. subnet = self.provider.networking.subnets.get_or_create_default(
  249. zone=get_provider_test_data(self.provider, 'placement'))
  250. self.assertIsInstance(subnet, Subnet)