TestCuraContainerRegistry.py 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328
  1. # Copyright (c) 2019 Ultimaker B.V.
  2. # Cura is released under the terms of the LGPLv3 or higher.
  3. import os #To find the directory with test files and find the test files.
  4. import pytest #To parameterize tests.
  5. import unittest.mock #To mock and monkeypatch stuff.
  6. from UM.Settings.DefinitionContainer import DefinitionContainer
  7. from cura.ReaderWriters.ProfileReader import NoProfileException
  8. from cura.Settings.ExtruderStack import ExtruderStack #Testing for returning the correct types of stacks.
  9. from cura.Settings.GlobalStack import GlobalStack #Testing for returning the correct types of stacks.
  10. import UM.Settings.InstanceContainer #Creating instance containers to register.
  11. import UM.Settings.ContainerRegistry #Making empty container stacks.
  12. import UM.Settings.ContainerStack #Setting the container registry here properly.
  13. import cura.CuraApplication
  14. def teardown():
  15. #If the temporary file for the legacy file rename test still exists, remove it.
  16. temporary_file = os.path.join(os.path.dirname(os.path.abspath(__file__)), "stacks", "temporary.stack.cfg")
  17. if os.path.isfile(temporary_file):
  18. os.remove(temporary_file)
  19. def test_createUniqueName(container_registry):
  20. from cura.CuraApplication import CuraApplication
  21. assert container_registry.createUniqueName("user", "test", "test2", "nope") == "test2"
  22. # Make a conflict (so that "test2" will no longer be an unique name)
  23. instance = UM.Settings.InstanceContainer.InstanceContainer(container_id="test2")
  24. instance.setMetaDataEntry("type", "user")
  25. instance.setMetaDataEntry("setting_version", CuraApplication.SettingVersion)
  26. container_registry.addContainer(instance)
  27. # It should add a #2 to test2
  28. assert container_registry.createUniqueName("user", "test", "test2", "nope") == "test2 #2"
  29. # The provided suggestion is already correct, so nothing to do
  30. assert container_registry.createUniqueName("user", "test", "test2 #2", "nope") == "test2 #2"
  31. # In case we don't provide a new name, use the fallback
  32. assert container_registry.createUniqueName("user", "test", "", "nope") == "nope"
  33. ## Tests whether addContainer properly converts to ExtruderStack.
  34. def test_addContainerExtruderStack(container_registry, definition_container, definition_changes_container):
  35. container_registry.addContainer(definition_container)
  36. container_registry.addContainer(definition_changes_container)
  37. container_stack = ExtruderStack("Test Extruder Stack") #A container we're going to convert.
  38. container_stack.setMetaDataEntry("type", "extruder_train") #This is now an extruder train.
  39. container_stack.setDefinition(definition_container) #Add a definition to it so it doesn't complain.
  40. container_stack.setDefinitionChanges(definition_changes_container)
  41. mock_super_add_container = unittest.mock.MagicMock() #Takes the role of the Uranium-ContainerRegistry where the resulting containers get registered.
  42. with unittest.mock.patch("UM.Settings.ContainerRegistry.ContainerRegistry.addContainer", mock_super_add_container):
  43. container_registry.addContainer(container_stack)
  44. assert len(mock_super_add_container.call_args_list) == 1 #Called only once.
  45. assert len(mock_super_add_container.call_args_list[0][0]) == 1 #Called with one parameter.
  46. assert type(mock_super_add_container.call_args_list[0][0][0]) == ExtruderStack
  47. ## Tests whether addContainer properly converts to GlobalStack.
  48. def test_addContainerGlobalStack(container_registry, definition_container, definition_changes_container):
  49. container_registry.addContainer(definition_container)
  50. container_registry.addContainer(definition_changes_container)
  51. container_stack = GlobalStack("Test Global Stack") #A container we're going to convert.
  52. container_stack.setMetaDataEntry("type", "machine") #This is now a global stack.
  53. container_stack.setDefinition(definition_container) #Must have a definition.
  54. container_stack.setDefinitionChanges(definition_changes_container) #Must have a definition changes.
  55. mock_super_add_container = unittest.mock.MagicMock() #Takes the role of the Uranium-ContainerRegistry where the resulting containers get registered.
  56. with unittest.mock.patch("UM.Settings.ContainerRegistry.ContainerRegistry.addContainer", mock_super_add_container):
  57. container_registry.addContainer(container_stack)
  58. assert len(mock_super_add_container.call_args_list) == 1 #Called only once.
  59. assert len(mock_super_add_container.call_args_list[0][0]) == 1 #Called with one parameter.
  60. assert type(mock_super_add_container.call_args_list[0][0][0]) == GlobalStack
  61. def test_addContainerGoodSettingVersion(container_registry, definition_container):
  62. from cura.CuraApplication import CuraApplication
  63. definition_container.getMetaData()["setting_version"] = CuraApplication.SettingVersion
  64. container_registry.addContainer(definition_container)
  65. instance = UM.Settings.InstanceContainer.InstanceContainer(container_id = "Test Instance Right Version")
  66. instance.setMetaDataEntry("setting_version", CuraApplication.SettingVersion)
  67. instance.setDefinition(definition_container.getId())
  68. mock_super_add_container = unittest.mock.MagicMock() #Take the role of the Uranium-ContainerRegistry where the resulting containers get registered.
  69. with unittest.mock.patch("UM.Settings.ContainerRegistry.ContainerRegistry.addContainer", mock_super_add_container):
  70. container_registry.addContainer(instance)
  71. mock_super_add_container.assert_called_once_with(instance) #The instance must have been registered now.
  72. def test_addContainerNoSettingVersion(container_registry, definition_container):
  73. from cura.CuraApplication import CuraApplication
  74. definition_container.getMetaData()["setting_version"] = CuraApplication.SettingVersion
  75. container_registry.addContainer(definition_container)
  76. instance = UM.Settings.InstanceContainer.InstanceContainer(container_id = "Test Instance No Version")
  77. #Don't add setting_version metadata.
  78. instance.setDefinition(definition_container.getId())
  79. mock_super_add_container = unittest.mock.MagicMock() #Take the role of the Uranium-ContainerRegistry where the resulting container should not get registered.
  80. with unittest.mock.patch("UM.Settings.ContainerRegistry.ContainerRegistry.addContainer", mock_super_add_container):
  81. container_registry.addContainer(instance)
  82. mock_super_add_container.assert_not_called() #Should not get passed on to UM.Settings.ContainerRegistry.addContainer, because the setting_version is interpreted as 0!
  83. def test_addContainerBadSettingVersion(container_registry, definition_container):
  84. from cura.CuraApplication import CuraApplication
  85. definition_container.getMetaData()["setting_version"] = CuraApplication.SettingVersion
  86. container_registry.addContainer(definition_container)
  87. instance = UM.Settings.InstanceContainer.InstanceContainer(container_id = "Test Instance Wrong Version")
  88. instance.setMetaDataEntry("setting_version", 9001) #Wrong version!
  89. instance.setDefinition(definition_container.getId())
  90. mock_super_add_container = unittest.mock.MagicMock() #Take the role of the Uranium-ContainerRegistry where the resulting container should not get registered.
  91. with unittest.mock.patch("UM.Settings.ContainerRegistry.ContainerRegistry.addContainer", mock_super_add_container):
  92. container_registry.addContainer(instance)
  93. mock_super_add_container.assert_not_called() #Should not get passed on to UM.Settings.ContainerRegistry.addContainer, because the setting_version doesn't match its definition!
  94. test_loadMetaDataValidation_data = [
  95. {
  96. "id": "valid_container",
  97. "is_valid": True,
  98. "metadata": {
  99. "id": "valid_container",
  100. "setting_version": None, #The tests sets this to the current version so it's always correct.
  101. "foo": "bar"
  102. }
  103. },
  104. {
  105. "id": "wrong_setting_version",
  106. "is_valid": False,
  107. "metadata": {
  108. "id": "wrong_setting_version",
  109. "setting_version": "5",
  110. "foo": "bar"
  111. }
  112. },
  113. {
  114. "id": "missing_setting_version",
  115. "is_valid": False,
  116. "metadata": {
  117. "id": "missing_setting_version",
  118. "foo": "bar"
  119. }
  120. },
  121. {
  122. "id": "unparsable_setting_version",
  123. "is_valid": False,
  124. "metadata": {
  125. "id": "unparsable_setting_version",
  126. "setting_version": "Not an integer!",
  127. "foo": "bar"
  128. }
  129. }
  130. ]
  131. @pytest.mark.parametrize("parameters", test_loadMetaDataValidation_data)
  132. def test_loadMetadataValidation(container_registry, definition_container, parameters):
  133. from cura.CuraApplication import CuraApplication
  134. definition_container.getMetaData()["setting_version"] = CuraApplication.SettingVersion
  135. container_registry.addContainer(definition_container)
  136. if "setting_version" in parameters["metadata"] and parameters["metadata"]["setting_version"] is None: #Signal that the setting_version must be set to the currently correct version.
  137. parameters["metadata"]["setting_version"] = CuraApplication.SettingVersion
  138. mock_provider = unittest.mock.MagicMock()
  139. mock_provider.getAllIds = unittest.mock.MagicMock(return_value = [parameters["id"]])
  140. mock_provider.loadMetadata = unittest.mock.MagicMock(return_value = parameters["metadata"])
  141. container_registry._providers = [mock_provider]
  142. container_registry.loadAllMetadata() #Run the test.
  143. if parameters["is_valid"]:
  144. assert parameters["id"] in container_registry.metadata
  145. assert container_registry.metadata[parameters["id"]] == parameters["metadata"]
  146. else:
  147. assert parameters["id"] not in container_registry.metadata
  148. class TestExportQualityProfile:
  149. # This class is just there to provide some grouping for the tests.
  150. def test_exportQualityProfileInvalidFileType(self, container_registry):
  151. # With an invalid file_type, we should get a false for success.
  152. assert not container_registry.exportQualityProfile([], "zomg", "invalid")
  153. def test_exportQualityProfileFailedWriter(self, container_registry):
  154. # Create a writer that always fails.
  155. mocked_writer = unittest.mock.MagicMock(name = "mocked_writer")
  156. mocked_writer.write = unittest.mock.MagicMock(return_value = False)
  157. container_registry._findProfileWriter = unittest.mock.MagicMock("findProfileWriter", return_value = mocked_writer)
  158. # Ensure that it actually fails if the writer did.
  159. with unittest.mock.patch("UM.Application.Application.getInstance"):
  160. assert not container_registry.exportQualityProfile([], "zomg", "test files (*.tst)")
  161. def test_exportQualityProfileExceptionWriter(self, container_registry):
  162. # Create a writer that always fails.
  163. mocked_writer = unittest.mock.MagicMock(name = "mocked_writer")
  164. mocked_writer.write = unittest.mock.MagicMock(return_value = True, side_effect = Exception("Failed :("))
  165. container_registry._findProfileWriter = unittest.mock.MagicMock("findProfileWriter", return_value = mocked_writer)
  166. # Ensure that it actually fails if the writer did.
  167. with unittest.mock.patch("UM.Application.Application.getInstance"):
  168. assert not container_registry.exportQualityProfile([], "zomg", "test files (*.tst)")
  169. def test_exportQualityProfileSuccessWriter(self, container_registry):
  170. # Create a writer that always fails.
  171. mocked_writer = unittest.mock.MagicMock(name="mocked_writer")
  172. mocked_writer.write = unittest.mock.MagicMock(return_value=True)
  173. container_registry._findProfileWriter = unittest.mock.MagicMock("findProfileWriter", return_value=mocked_writer)
  174. # Ensure that it actually fails if the writer did.
  175. with unittest.mock.patch("UM.Application.Application.getInstance"):
  176. assert container_registry.exportQualityProfile([], "zomg", "test files (*.tst)")
  177. def test__findProfileWriterNoPlugins(container_registry):
  178. # Mock it so that no IO plugins are found.
  179. container_registry._getIOPlugins = unittest.mock.MagicMock(return_value = [])
  180. with unittest.mock.patch("UM.PluginRegistry.PluginRegistry.getInstance"):
  181. # Since there are no writers, don't return any
  182. assert container_registry._findProfileWriter(".zomg", "dunno") is None
  183. def test__findProfileWriter(container_registry):
  184. # Mock it so that no IO plugins are found.
  185. container_registry._getIOPlugins = unittest.mock.MagicMock(return_value = [("writer_id", {"profile_writer": [{"extension": ".zomg", "description": "dunno"}]})])
  186. with unittest.mock.patch("UM.PluginRegistry.PluginRegistry.getInstance"):
  187. # In this case it's getting a mocked object (from the mocked_plugin_registry)
  188. assert container_registry._findProfileWriter(".zomg", "dunno") is not None
  189. def test_importProfileEmptyFileName(container_registry):
  190. result = container_registry.importProfile("")
  191. assert result["status"] == "error"
  192. mocked_application = unittest.mock.MagicMock(name = "application")
  193. mocked_plugin_registry = unittest.mock.MagicMock(name="mocked_plugin_registry")
  194. @unittest.mock.patch("UM.Application.Application.getInstance", unittest.mock.MagicMock(return_value = mocked_application))
  195. @unittest.mock.patch("UM.PluginRegistry.PluginRegistry.getInstance", unittest.mock.MagicMock(return_value = mocked_plugin_registry))
  196. class TestImportProfile:
  197. mocked_global_stack = unittest.mock.MagicMock(name="global stack")
  198. mocked_global_stack.extruders = {0: unittest.mock.MagicMock(name="extruder stack")}
  199. mocked_global_stack.getId = unittest.mock.MagicMock(return_value="blarg")
  200. mocked_profile_reader = unittest.mock.MagicMock()
  201. mocked_plugin_registry.getPluginObject = unittest.mock.MagicMock(return_value=mocked_profile_reader)
  202. def test_importProfileWithoutGlobalStack(self, container_registry):
  203. mocked_application.getGlobalContainerStack = unittest.mock.MagicMock(return_value = None)
  204. result = container_registry.importProfile("non_empty")
  205. assert result["status"] == "error"
  206. def test_importProfileNoProfileException(self, container_registry):
  207. container_registry._getIOPlugins = unittest.mock.MagicMock(return_value=[("reader_id", {"profile_reader": [{"extension": "zomg", "description": "dunno"}]})])
  208. mocked_application.getGlobalContainerStack = unittest.mock.MagicMock(return_value=self.mocked_global_stack)
  209. self.mocked_profile_reader.read = unittest.mock.MagicMock(side_effect = NoProfileException)
  210. result = container_registry.importProfile("test.zomg")
  211. # It's not an error, but we also didn't find any profile to read.
  212. assert result["status"] == "ok"
  213. def test_importProfileGenericException(self, container_registry):
  214. container_registry._getIOPlugins = unittest.mock.MagicMock(return_value=[("reader_id", {"profile_reader": [{"extension": "zomg", "description": "dunno"}]})])
  215. mocked_application.getGlobalContainerStack = unittest.mock.MagicMock(return_value=self.mocked_global_stack)
  216. self.mocked_profile_reader.read = unittest.mock.MagicMock(side_effect = Exception)
  217. result = container_registry.importProfile("test.zomg")
  218. assert result["status"] == "error"
  219. def test_importProfileNoDefinitionFound(self, container_registry):
  220. container_registry._getIOPlugins = unittest.mock.MagicMock(return_value=[("reader_id", {"profile_reader": [{"extension": "zomg", "description": "dunno"}]})])
  221. mocked_application.getGlobalContainerStack = unittest.mock.MagicMock(return_value=self.mocked_global_stack)
  222. container_registry.findDefinitionContainers = unittest.mock.MagicMock(return_value = [])
  223. mocked_profile = unittest.mock.MagicMock(name = "Mocked_global_profile")
  224. self.mocked_profile_reader.read = unittest.mock.MagicMock(return_value = [mocked_profile])
  225. result = container_registry.importProfile("test.zomg")
  226. assert result["status"] == "error"
  227. @pytest.mark.skip
  228. def test_importProfileSuccess(self, container_registry):
  229. container_registry._getIOPlugins = unittest.mock.MagicMock(return_value=[("reader_id", {"profile_reader": [{"extension": "zomg", "description": "dunno"}]})])
  230. mocked_application.getGlobalContainerStack = unittest.mock.MagicMock(return_value=self.mocked_global_stack)
  231. mocked_definition = unittest.mock.MagicMock(name = "definition")
  232. container_registry.findContainers = unittest.mock.MagicMock(return_value=[mocked_definition])
  233. container_registry.findDefinitionContainers = unittest.mock.MagicMock(return_value = [mocked_definition])
  234. mocked_profile = unittest.mock.MagicMock(name = "Mocked_global_profile")
  235. self.mocked_profile_reader.read = unittest.mock.MagicMock(return_value = [mocked_profile])
  236. with unittest.mock.patch.object(container_registry, "createUniqueName", return_value="derp"):
  237. with unittest.mock.patch.object(container_registry, "_configureProfile", return_value=None):
  238. result = container_registry.importProfile("test.zomg")
  239. assert result["status"] == "ok"
  240. @pytest.mark.parametrize("metadata,result", [(None, False),
  241. ({}, False),
  242. ({"setting_version": cura.CuraApplication.CuraApplication.SettingVersion}, True),
  243. ({"setting_version": 0}, False)])
  244. def test_isMetaDataValid(container_registry, metadata, result):
  245. assert container_registry._isMetadataValid(metadata) == result
  246. def test_getIOPlugins(container_registry):
  247. plugin_registry = unittest.mock.MagicMock()
  248. plugin_registry.getActivePlugins = unittest.mock.MagicMock(return_value = ["lizard"])
  249. plugin_registry.getMetaData = unittest.mock.MagicMock(return_value = {"zomg": {"test": "test"}})
  250. with unittest.mock.patch("UM.PluginRegistry.PluginRegistry.getInstance", unittest.mock.MagicMock(return_value = plugin_registry)):
  251. assert container_registry._getIOPlugins("zomg") == [("lizard", {"zomg": {"test": "test"}})]