TestCuraContainerRegistry.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239
  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.Settings.ExtruderStack import ExtruderStack #Testing for returning the correct types of stacks.
  8. from cura.Settings.GlobalStack import GlobalStack #Testing for returning the correct types of stacks.
  9. import UM.Settings.InstanceContainer #Creating instance containers to register.
  10. import UM.Settings.ContainerRegistry #Making empty container stacks.
  11. import UM.Settings.ContainerStack #Setting the container registry here properly.
  12. def teardown():
  13. #If the temporary file for the legacy file rename test still exists, remove it.
  14. temporary_file = os.path.join(os.path.dirname(os.path.abspath(__file__)), "stacks", "temporary.stack.cfg")
  15. if os.path.isfile(temporary_file):
  16. os.remove(temporary_file)
  17. def test_createUniqueName(container_registry):
  18. from cura.CuraApplication import CuraApplication
  19. assert container_registry.createUniqueName("user", "test", "test2", "nope") == "test2"
  20. # Make a conflict (so that "test2" will no longer be an unique name)
  21. instance = UM.Settings.InstanceContainer.InstanceContainer(container_id="test2")
  22. instance.setMetaDataEntry("type", "user")
  23. instance.setMetaDataEntry("setting_version", CuraApplication.SettingVersion)
  24. container_registry.addContainer(instance)
  25. # It should add a #2 to test2
  26. assert container_registry.createUniqueName("user", "test", "test2", "nope") == "test2 #2"
  27. ## Tests whether addContainer properly converts to ExtruderStack.
  28. def test_addContainerExtruderStack(container_registry, definition_container, definition_changes_container):
  29. container_registry.addContainer(definition_container)
  30. container_registry.addContainer(definition_changes_container)
  31. container_stack = UM.Settings.ContainerStack.ContainerStack(stack_id = "Test Extruder Stack") #A container we're going to convert.
  32. container_stack.setMetaDataEntry("type", "extruder_train") #This is now an extruder train.
  33. container_stack.insertContainer(0, definition_container) #Add a definition to it so it doesn't complain.
  34. container_stack.insertContainer(1, definition_changes_container)
  35. mock_super_add_container = unittest.mock.MagicMock() #Takes the role of the Uranium-ContainerRegistry where the resulting containers get registered.
  36. with unittest.mock.patch("UM.Settings.ContainerRegistry.ContainerRegistry.addContainer", mock_super_add_container):
  37. container_registry.addContainer(container_stack)
  38. assert len(mock_super_add_container.call_args_list) == 1 #Called only once.
  39. assert len(mock_super_add_container.call_args_list[0][0]) == 1 #Called with one parameter.
  40. assert type(mock_super_add_container.call_args_list[0][0][0]) == ExtruderStack
  41. ## Tests whether addContainer properly converts to GlobalStack.
  42. def test_addContainerGlobalStack(container_registry, definition_container, definition_changes_container):
  43. container_registry.addContainer(definition_container)
  44. container_registry.addContainer(definition_changes_container)
  45. container_stack = UM.Settings.ContainerStack.ContainerStack(stack_id = "Test Global Stack") #A container we're going to convert.
  46. container_stack.setMetaDataEntry("type", "machine") #This is now a global stack.
  47. container_stack.insertContainer(0, definition_container) #Must have a definition.
  48. container_stack.insertContainer(1, definition_changes_container) #Must have a definition changes.
  49. mock_super_add_container = unittest.mock.MagicMock() #Takes the role of the Uranium-ContainerRegistry where the resulting containers get registered.
  50. with unittest.mock.patch("UM.Settings.ContainerRegistry.ContainerRegistry.addContainer", mock_super_add_container):
  51. container_registry.addContainer(container_stack)
  52. assert len(mock_super_add_container.call_args_list) == 1 #Called only once.
  53. assert len(mock_super_add_container.call_args_list[0][0]) == 1 #Called with one parameter.
  54. assert type(mock_super_add_container.call_args_list[0][0][0]) == GlobalStack
  55. def test_addContainerGoodSettingVersion(container_registry, definition_container):
  56. from cura.CuraApplication import CuraApplication
  57. definition_container.getMetaData()["setting_version"] = CuraApplication.SettingVersion
  58. container_registry.addContainer(definition_container)
  59. instance = UM.Settings.InstanceContainer.InstanceContainer(container_id = "Test Instance Right Version")
  60. instance.setMetaDataEntry("setting_version", CuraApplication.SettingVersion)
  61. instance.setDefinition(definition_container.getId())
  62. mock_super_add_container = unittest.mock.MagicMock() #Take the role of the Uranium-ContainerRegistry where the resulting containers get registered.
  63. with unittest.mock.patch("UM.Settings.ContainerRegistry.ContainerRegistry.addContainer", mock_super_add_container):
  64. container_registry.addContainer(instance)
  65. mock_super_add_container.assert_called_once_with(instance) #The instance must have been registered now.
  66. def test_addContainerNoSettingVersion(container_registry, definition_container):
  67. from cura.CuraApplication import CuraApplication
  68. definition_container.getMetaData()["setting_version"] = CuraApplication.SettingVersion
  69. container_registry.addContainer(definition_container)
  70. instance = UM.Settings.InstanceContainer.InstanceContainer(container_id = "Test Instance No Version")
  71. #Don't add setting_version metadata.
  72. instance.setDefinition(definition_container.getId())
  73. mock_super_add_container = unittest.mock.MagicMock() #Take the role of the Uranium-ContainerRegistry where the resulting container should not get registered.
  74. with unittest.mock.patch("UM.Settings.ContainerRegistry.ContainerRegistry.addContainer", mock_super_add_container):
  75. container_registry.addContainer(instance)
  76. 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!
  77. def test_addContainerBadSettingVersion(container_registry, definition_container):
  78. from cura.CuraApplication import CuraApplication
  79. definition_container.getMetaData()["setting_version"] = CuraApplication.SettingVersion
  80. container_registry.addContainer(definition_container)
  81. instance = UM.Settings.InstanceContainer.InstanceContainer(container_id = "Test Instance Wrong Version")
  82. instance.setMetaDataEntry("setting_version", 9001) #Wrong version!
  83. instance.setDefinition(definition_container.getId())
  84. mock_super_add_container = unittest.mock.MagicMock() #Take the role of the Uranium-ContainerRegistry where the resulting container should not get registered.
  85. with unittest.mock.patch("UM.Settings.ContainerRegistry.ContainerRegistry.addContainer", mock_super_add_container):
  86. container_registry.addContainer(instance)
  87. 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!
  88. test_loadMetaDataValidation_data = [
  89. {
  90. "id": "valid_container",
  91. "is_valid": True,
  92. "metadata": {
  93. "id": "valid_container",
  94. "setting_version": None, #The tests sets this to the current version so it's always correct.
  95. "foo": "bar"
  96. }
  97. },
  98. {
  99. "id": "wrong_setting_version",
  100. "is_valid": False,
  101. "metadata": {
  102. "id": "wrong_setting_version",
  103. "setting_version": "5",
  104. "foo": "bar"
  105. }
  106. },
  107. {
  108. "id": "missing_setting_version",
  109. "is_valid": False,
  110. "metadata": {
  111. "id": "missing_setting_version",
  112. "foo": "bar"
  113. }
  114. },
  115. {
  116. "id": "unparsable_setting_version",
  117. "is_valid": False,
  118. "metadata": {
  119. "id": "unparsable_setting_version",
  120. "setting_version": "Not an integer!",
  121. "foo": "bar"
  122. }
  123. }
  124. ]
  125. @pytest.mark.parametrize("parameters", test_loadMetaDataValidation_data)
  126. def test_loadMetadataValidation(container_registry, definition_container, parameters):
  127. from cura.CuraApplication import CuraApplication
  128. definition_container.getMetaData()["setting_version"] = CuraApplication.SettingVersion
  129. container_registry.addContainer(definition_container)
  130. 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.
  131. parameters["metadata"]["setting_version"] = CuraApplication.SettingVersion
  132. mock_provider = unittest.mock.MagicMock()
  133. mock_provider.getAllIds = unittest.mock.MagicMock(return_value = [parameters["id"]])
  134. mock_provider.loadMetadata = unittest.mock.MagicMock(return_value = parameters["metadata"])
  135. container_registry._providers = [mock_provider]
  136. container_registry.loadAllMetadata() #Run the test.
  137. if parameters["is_valid"]:
  138. assert parameters["id"] in container_registry.metadata
  139. assert container_registry.metadata[parameters["id"]] == parameters["metadata"]
  140. else:
  141. assert parameters["id"] not in container_registry.metadata
  142. class TestExportQualityProfile:
  143. # This class is just there to provide some grouping for the tests.
  144. def test_exportQualityProfileInvalidFileType(self, container_registry):
  145. # With an invalid file_type, we should get a false for success.
  146. assert not container_registry.exportQualityProfile([], "zomg", "invalid")
  147. def test_exportQualityProfileFailedWriter(self, container_registry):
  148. # Create a writer that always fails.
  149. mocked_writer = unittest.mock.MagicMock(name = "mocked_writer")
  150. mocked_writer.write = unittest.mock.MagicMock(return_value = False)
  151. container_registry._findProfileWriter = unittest.mock.MagicMock("findProfileWriter", return_value = mocked_writer)
  152. # Ensure that it actually fails if the writer did.
  153. with unittest.mock.patch("UM.Application.Application.getInstance"):
  154. assert not container_registry.exportQualityProfile([], "zomg", "test files (*.tst)")
  155. def test_exportQualityProfileExceptionWriter(self, container_registry):
  156. # Create a writer that always fails.
  157. mocked_writer = unittest.mock.MagicMock(name = "mocked_writer")
  158. mocked_writer.write = unittest.mock.MagicMock(return_value = True, side_effect = Exception("Failed :("))
  159. container_registry._findProfileWriter = unittest.mock.MagicMock("findProfileWriter", return_value = mocked_writer)
  160. # Ensure that it actually fails if the writer did.
  161. with unittest.mock.patch("UM.Application.Application.getInstance"):
  162. assert not container_registry.exportQualityProfile([], "zomg", "test files (*.tst)")
  163. def test_exportQualityProfileSuccessWriter(self, container_registry):
  164. # Create a writer that always fails.
  165. mocked_writer = unittest.mock.MagicMock(name="mocked_writer")
  166. mocked_writer.write = unittest.mock.MagicMock(return_value=True)
  167. container_registry._findProfileWriter = unittest.mock.MagicMock("findProfileWriter", return_value=mocked_writer)
  168. # Ensure that it actually fails if the writer did.
  169. with unittest.mock.patch("UM.Application.Application.getInstance"):
  170. assert container_registry.exportQualityProfile([], "zomg", "test files (*.tst)")
  171. def test__findProfileWriterNoPlugins(container_registry):
  172. # Mock it so that no IO plugins are found.
  173. container_registry._getIOPlugins = unittest.mock.MagicMock(return_value = [])
  174. mocked_plugin_registry = unittest.mock.MagicMock(name = "plugin registry")
  175. with unittest.mock.patch("UM.PluginRegistry.PluginRegistry.getInstance", mocked_plugin_registry):
  176. # Since there are no writers, don't return any
  177. assert container_registry._findProfileWriter(".zomg", "dunno") is None
  178. def test__findProfileWriter(container_registry):
  179. # Mock it so that no IO plugins are found.
  180. container_registry._getIOPlugins = unittest.mock.MagicMock(return_value = [("writer_id", {"profile_writer": [{"extension": ".zomg", "description": "dunno"}]})])
  181. mocked_plugin_registry = unittest.mock.MagicMock(name = "plugin registry")
  182. with unittest.mock.patch("UM.PluginRegistry.PluginRegistry.getInstance", mocked_plugin_registry):
  183. # In this case it's getting a mocked object (from the mocked_plugin_registry)
  184. assert container_registry._findProfileWriter(".zomg", "dunno") is not None