test_block_store_service.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. import time
  2. import six
  3. from cloudbridge.cloud.factory import ProviderList
  4. from cloudbridge.cloud.interfaces import SnapshotState
  5. from cloudbridge.cloud.interfaces import VolumeState
  6. from cloudbridge.cloud.interfaces.provider import TestMockHelperMixin
  7. from cloudbridge.cloud.interfaces.resources import AttachmentInfo
  8. from cloudbridge.cloud.interfaces.resources import Snapshot
  9. from cloudbridge.cloud.interfaces.resources import Volume
  10. from test import helpers
  11. from test.helpers import ProviderTestBase
  12. from test.helpers import standard_interface_tests as sit
  13. class CloudBlockStoreServiceTestCase(ProviderTestBase):
  14. _multiprocess_can_split_ = True
  15. @helpers.skipIfNoService(['storage.volumes', 'storage.volumes'])
  16. def test_storage_services_event_pattern(self):
  17. # pylint:disable=protected-access
  18. self.assertEqual(
  19. self.provider.storage.volumes._service_event_pattern,
  20. "provider.storage.volumes",
  21. "Event pattern for {} service should be '{}', "
  22. "but found '{}'.".format("volumes",
  23. "provider.storage.volumes",
  24. self.provider.storage.volumes.
  25. _service_event_pattern))
  26. # pylint:disable=protected-access
  27. self.assertEqual(
  28. self.provider.storage.snapshots._service_event_pattern,
  29. "provider.storage.snapshots",
  30. "Event pattern for {} service should be '{}', "
  31. "but found '{}'.".format("snapshots",
  32. "provider.storage.snapshots",
  33. self.provider.storage.snapshots.
  34. _service_event_pattern))
  35. @helpers.skipIfNoService(['storage.volumes'])
  36. def test_crud_volume(self):
  37. def create_vol(label):
  38. return self.provider.storage.volumes.create(
  39. label, 1,
  40. helpers.get_provider_test_data(self.provider, "placement"))
  41. def cleanup_vol(vol):
  42. if vol:
  43. vol.delete()
  44. vol.wait_for([VolumeState.DELETED, VolumeState.UNKNOWN],
  45. terminal_states=[VolumeState.ERROR])
  46. vol.refresh()
  47. self.assertTrue(
  48. vol.state == VolumeState.UNKNOWN,
  49. "Volume.state must be unknown when refreshing after a "
  50. "delete but got %s"
  51. % vol.state)
  52. sit.check_crud(self, self.provider.storage.volumes, Volume,
  53. "cb-createvol", create_vol, cleanup_vol)
  54. @helpers.skipIfNoService(['storage.volumes'])
  55. def test_attach_detach_volume(self):
  56. label = "cb-attachvol-{0}".format(helpers.get_uuid())
  57. # Declare these variables and late binding will allow
  58. # the cleanup method access to the most current values
  59. test_instance = None
  60. with helpers.cleanup_action(lambda: helpers.cleanup_test_resources(
  61. test_instance)):
  62. subnet = helpers.get_or_create_default_subnet(
  63. self.provider)
  64. test_instance = helpers.get_test_instance(
  65. self.provider, label, subnet=subnet)
  66. test_vol = self.provider.storage.volumes.create(
  67. label, 1, test_instance.zone_id)
  68. with helpers.cleanup_action(lambda: test_vol.delete()):
  69. test_vol.wait_till_ready()
  70. test_vol.attach(test_instance, '/dev/sda2')
  71. test_vol.wait_for(
  72. [VolumeState.IN_USE],
  73. terminal_states=[VolumeState.ERROR, VolumeState.DELETED])
  74. test_vol.detach()
  75. test_vol.wait_for(
  76. [VolumeState.AVAILABLE],
  77. terminal_states=[VolumeState.ERROR, VolumeState.DELETED])
  78. @helpers.skipIfNoService(['storage.volumes'])
  79. def test_volume_properties(self):
  80. label = "cb-volprops-{0}".format(helpers.get_uuid())
  81. vol_desc = 'newvoldesc1'
  82. # Declare these variables and late binding will allow
  83. # the cleanup method access to the most current values
  84. test_instance = None
  85. with helpers.cleanup_action(lambda: helpers.cleanup_test_resources(
  86. test_instance)):
  87. subnet = helpers.get_or_create_default_subnet(
  88. self.provider)
  89. test_instance = helpers.get_test_instance(
  90. self.provider, label, subnet=subnet)
  91. test_vol = self.provider.storage.volumes.create(
  92. label, 1, test_instance.zone_id, description=vol_desc)
  93. with helpers.cleanup_action(lambda: test_vol.delete()):
  94. test_vol.wait_till_ready()
  95. self.assertTrue(
  96. isinstance(test_vol.size, six.integer_types) and
  97. test_vol.size >= 0,
  98. "Volume.size must be a positive number, but got %s"
  99. % test_vol.size)
  100. self.assertTrue(
  101. test_vol.description is None or
  102. isinstance(test_vol.description, six.string_types),
  103. "Volume.description must be None or a string. Got: %s"
  104. % test_vol.description)
  105. self.assertIsNone(test_vol.source)
  106. self.assertIsNone(test_vol.source)
  107. self.assertIsNotNone(test_vol.create_time)
  108. self.assertIsNotNone(test_vol.zone_id)
  109. self.assertIsNone(test_vol.attachments)
  110. test_vol.attach(test_instance, '/dev/sda2')
  111. test_vol.wait_for(
  112. [VolumeState.IN_USE],
  113. terminal_states=[VolumeState.ERROR, VolumeState.DELETED])
  114. self.assertIsNotNone(test_vol.attachments)
  115. self.assertIsInstance(test_vol.attachments, AttachmentInfo)
  116. self.assertEqual(test_vol.attachments.volume, test_vol)
  117. self.assertEqual(test_vol.attachments.instance_id,
  118. test_instance.id)
  119. if (self.provider.PROVIDER_ID != 'azure' and
  120. self.provider.PROVIDER_ID != 'gcp'):
  121. self.assertEqual(test_vol.attachments.device,
  122. "/dev/sda2")
  123. test_vol.detach()
  124. test_vol.label = 'newvolname1'
  125. test_vol.wait_for(
  126. [VolumeState.AVAILABLE],
  127. terminal_states=[VolumeState.ERROR, VolumeState.DELETED])
  128. self.assertEqual(test_vol.label, 'newvolname1')
  129. self.assertEqual(test_vol.description, vol_desc)
  130. self.assertIsNone(test_vol.attachments)
  131. test_vol.wait_for(
  132. [VolumeState.AVAILABLE],
  133. terminal_states=[VolumeState.ERROR, VolumeState.DELETED])
  134. @helpers.skipIfNoService(['storage.snapshots'])
  135. def test_crud_snapshot(self):
  136. # Create a new volume, create a snapshot of the volume, and check
  137. # whether list_snapshots properly detects the new snapshot.
  138. # Delete everything afterwards.
  139. label = "cb-crudsnap-{0}".format(helpers.get_uuid())
  140. test_vol = self.provider.storage.volumes.create(
  141. label, 1,
  142. helpers.get_provider_test_data(self.provider, "placement"))
  143. with helpers.cleanup_action(lambda: test_vol.delete()):
  144. test_vol.wait_till_ready()
  145. def create_snap(label):
  146. return test_vol.create_snapshot(label=label,
  147. description=label)
  148. def cleanup_snap(snap):
  149. if snap:
  150. snap.delete()
  151. snap.wait_for([SnapshotState.UNKNOWN],
  152. terminal_states=[SnapshotState.ERROR])
  153. snap.refresh()
  154. self.assertTrue(
  155. snap.state == SnapshotState.UNKNOWN,
  156. "Snapshot.state must be unknown when refreshing after "
  157. "a delete but got %s"
  158. % snap.state)
  159. sit.check_crud(self, self.provider.storage.snapshots, Snapshot,
  160. "cb-snap", create_snap, cleanup_snap)
  161. # Test creation of a snap via SnapshotService
  162. def create_snap2(label):
  163. return self.provider.storage.snapshots.create(
  164. label=label, volume=test_vol, description=label)
  165. if (self.provider.PROVIDER_ID == ProviderList.AWS and
  166. not isinstance(self.provider, TestMockHelperMixin)):
  167. time.sleep(15) # Or get SnapshotCreationPerVolumeRateExceeded
  168. sit.check_crud(self, self.provider.storage.snapshots, Snapshot,
  169. "cb-snaptwo", create_snap2, cleanup_snap)
  170. @helpers.skipIfNoService(['storage.snapshots'])
  171. def test_snapshot_properties(self):
  172. label = "cb-snapprop-{0}".format(helpers.get_uuid())
  173. test_vol = self.provider.storage.volumes.create(
  174. label, 1,
  175. helpers.get_provider_test_data(self.provider, "placement"))
  176. with helpers.cleanup_action(lambda: test_vol.delete()):
  177. test_vol.wait_till_ready()
  178. snap_label = "cb-snap-{0}".format(label)
  179. test_snap = test_vol.create_snapshot(label=snap_label,
  180. description=snap_label)
  181. def cleanup_snap(snap):
  182. if snap:
  183. snap.delete()
  184. snap.wait_for([SnapshotState.UNKNOWN],
  185. terminal_states=[SnapshotState.ERROR])
  186. with helpers.cleanup_action(lambda: cleanup_snap(test_snap)):
  187. test_snap.wait_till_ready()
  188. self.assertTrue(isinstance(test_vol.size, six.integer_types))
  189. self.assertEqual(
  190. test_snap.size, test_vol.size,
  191. "Snapshot.size must match original volume's size: %s"
  192. " but is: %s" % (test_vol.size, test_snap.size))
  193. self.assertTrue(
  194. test_vol.description is None or
  195. isinstance(test_vol.description, six.string_types),
  196. "Snapshot.description must be None or a string. Got: %s"
  197. % test_vol.description)
  198. self.assertEqual(test_vol.id, test_snap.volume_id)
  199. self.assertIsNotNone(test_vol.create_time)
  200. test_snap.label = 'snapnewname1'
  201. test_snap.description = 'snapnewdescription1'
  202. test_snap.refresh()
  203. self.assertEqual(test_snap.label, 'snapnewname1')
  204. self.assertEqual(test_snap.description, 'snapnewdescription1')
  205. # Test volume creation from a snapshot (via VolumeService)
  206. sv_label = "cb-snapvol-{0}".format(test_snap.name)
  207. snap_vol = self.provider.storage.volumes.create(
  208. sv_label, 1,
  209. helpers.get_provider_test_data(self.provider, "placement"),
  210. snapshot=test_snap)
  211. with helpers.cleanup_action(lambda: snap_vol.delete()):
  212. snap_vol.wait_till_ready()
  213. # Test volume creation from a snapshot (via Snapshot)
  214. snap_vol2 = test_snap.create_volume(
  215. helpers.get_provider_test_data(self.provider, "placement"))
  216. with helpers.cleanup_action(lambda: snap_vol2.delete()):
  217. snap_vol2.wait_till_ready()