BuilderSDKUnitTests.cpp 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200
  1. /*
  2. * Copyright (c) Contributors to the Open 3D Engine Project.
  3. * For complete copyright and license terms please see the LICENSE at the root of this distribution.
  4. *
  5. * SPDX-License-Identifier: Apache-2.0 OR MIT
  6. *
  7. */
  8. #include <AssetBuilderSDK/AssetBuilderSDK.h>
  9. #if !defined(Q_MOC_RUN)
  10. #include <AzCore/UnitTest/TestTypes.h>
  11. #include <native/unittests/UnitTestUtils.h>
  12. #endif
  13. #include <QDir>
  14. #include <QTemporaryDir>
  15. namespace UnitTest
  16. {
  17. // note that this is copied from BuilderSDK.cpp because its intentionally not exposed as publicly available types
  18. // the game would import it from LmbrCentral's various headers such as MaterialAsset.h
  19. // but that would require including lmbrcentral into the unit tests, which is not acceptable.
  20. static AZ::Data::AssetType sliceAssetType("{C62C7A87-9C09-4148-A985-12F2C99C0A45}"); // from SliceAsset.h
  21. static AZ::Data::AssetType fontAssetType("{57767D37-0EBE-43BE-8F60-AB36D2056EF8}"); // form UiAssetTypes.h
  22. class BuilderSDKUnitTests : public LeakDetectionFixture
  23. {
  24. public:
  25. // QTemporaryDir autodeletes on destruct.
  26. BuilderSDKUnitTests()
  27. {
  28. m_folder = QDir(m_tempDir.path());
  29. }
  30. QDir m_folder;
  31. private:
  32. QTemporaryDir m_tempDir;
  33. };
  34. TEST_F(BuilderSDKUnitTests, JobProduct_InferredAssetTypes_MatchExpectedValues)
  35. {
  36. EXPECT_EQ(
  37. AssetBuilderSDK::JobProduct::InferAssetTypeByProductFileName("no_extension"), AZ::Data::AssetType::CreateNull());
  38. }
  39. TEST_F(BuilderSDKUnitTests, XMLParsing_EmptyOrInvalid_MatchesExpectedResult)
  40. {
  41. // this must neither crash nor fail to return the null type.
  42. QString dummyFileName = m_folder.absoluteFilePath("test_empty_xml");
  43. EXPECT_TRUE(UnitTestUtils::CreateDummyFile(dummyFileName, ""));
  44. EXPECT_EQ(
  45. AssetBuilderSDK::JobProduct::InferAssetTypeByProductFileName(dummyFileName.toUtf8().data()), AZ::Uuid::CreateNull());
  46. EXPECT_TRUE(UnitTestUtils::CreateDummyFile(dummyFileName, "dummy"));
  47. EXPECT_EQ(
  48. AssetBuilderSDK::JobProduct::InferAssetTypeByProductFileName(dummyFileName.toUtf8().data()), AZ::Uuid::CreateNull());
  49. EXPECT_TRUE(UnitTestUtils::CreateDummyFile(dummyFileName, "<truncatedfile "));
  50. EXPECT_EQ(
  51. AssetBuilderSDK::JobProduct::InferAssetTypeByProductFileName(dummyFileName.toUtf8().data()), AZ::Uuid::CreateNull());
  52. EXPECT_TRUE(UnitTestUtils::CreateDummyFile(dummyFileName, "<truncated attribute=\"test"));
  53. EXPECT_EQ(
  54. AssetBuilderSDK::JobProduct::InferAssetTypeByProductFileName(dummyFileName.toUtf8().data()), AZ::Uuid::CreateNull());
  55. }
  56. TEST_F(BuilderSDKUnitTests, XMLParsing_BasicTypes_MatchesExpectedResults)
  57. {
  58. QString dummyFileName = m_folder.absoluteFilePath("test_basic_file_xml");
  59. // note that the above file is NOT AN XML FILE so it should not work despite contianing the expected tag.
  60. EXPECT_TRUE(UnitTestUtils::CreateDummyFile(dummyFileName, "<fontshader>stuff</fontshader>"));
  61. EXPECT_EQ(
  62. AssetBuilderSDK::JobProduct::InferAssetTypeByProductFileName(dummyFileName.toUtf8().data()), AZ::Uuid::CreateNull());
  63. dummyFileName = m_folder.absoluteFilePath("test_basic_file.xml");
  64. EXPECT_TRUE(UnitTestUtils::CreateDummyFile(dummyFileName, "<fontshader>stuff</fontshader>"));
  65. EXPECT_EQ(AssetBuilderSDK::JobProduct::InferAssetTypeByProductFileName(dummyFileName.toUtf8().data()), fontAssetType);
  66. EXPECT_TRUE(UnitTestUtils::CreateDummyFile(
  67. dummyFileName,
  68. "<ObjectStream>stuff</ObjectStream>")); // note - objectstream with no data in it should not crash or return anything useful
  69. EXPECT_EQ(
  70. AssetBuilderSDK::JobProduct::InferAssetTypeByProductFileName(dummyFileName.toUtf8().data()), AZ::Uuid::CreateNull());
  71. EXPECT_TRUE(UnitTestUtils::CreateDummyFile(dummyFileName, "<UnknownThing>stuff</UnknownThing>"));
  72. EXPECT_EQ(
  73. AssetBuilderSDK::JobProduct::InferAssetTypeByProductFileName(dummyFileName.toUtf8().data()), AZ::Uuid::CreateNull());
  74. }
  75. TEST_F(BuilderSDKUnitTests, XMLParsing_ObjectStreamTypes_MatchesExpectedResults)
  76. {
  77. // this must neither crash nor fail to return the null type.
  78. QString dummyFileName = m_folder.absoluteFilePath("test_objectstream.xml");
  79. // objectstream missing its 'type' inside the class
  80. EXPECT_TRUE(UnitTestUtils::CreateDummyFile(dummyFileName, "<ObjectStream><Class/></ObjectStream>"));
  81. EXPECT_EQ(
  82. AssetBuilderSDK::JobProduct::InferAssetTypeByProductFileName(dummyFileName.toUtf8().data()), AZ::Uuid::CreateNull());
  83. {
  84. UnitTestUtils::AssertAbsorber absorber;
  85. // objectstream with an empty string 'type' inside the class
  86. EXPECT_TRUE(UnitTestUtils::CreateDummyFile(dummyFileName, "<ObjectStream><Class type=\"\"/></ObjectStream>"));
  87. EXPECT_EQ(
  88. AssetBuilderSDK::JobProduct::InferAssetTypeByProductFileName(dummyFileName.toUtf8().data()), AZ::Uuid::CreateNull());
  89. }
  90. {
  91. UnitTestUtils::AssertAbsorber absorber;
  92. // objectstream with invalid 'type' inside the class (bad guid)
  93. EXPECT_TRUE(UnitTestUtils::CreateDummyFile(dummyFileName, "<ObjectStream><Class type=\"123 NOT A GUID\"/></ObjectStream>"));
  94. EXPECT_EQ(
  95. AssetBuilderSDK::JobProduct::InferAssetTypeByProductFileName(dummyFileName.toUtf8().data()), AZ::Uuid::CreateNull());
  96. EXPECT_GT(absorber.m_numWarningsAbsorbed, 0);
  97. }
  98. // objectstream with an actual 'guid' inside the class (bad guid)
  99. EXPECT_TRUE(UnitTestUtils::CreateDummyFile(
  100. dummyFileName, "<ObjectStream><Class type=\"{49375937-7F37-41B1-96A5-B099A8657DDE}\"/></ObjectStream>"));
  101. EXPECT_EQ(
  102. AssetBuilderSDK::JobProduct::InferAssetTypeByProductFileName(dummyFileName.toUtf8().data()),
  103. AZ::Uuid("{49375937-7F37-41B1-96A5-B099A8657DDE}"));
  104. EXPECT_TRUE(UnitTestUtils::CreateDummyFile(
  105. dummyFileName, "<ObjectStream><Class type=\"{49375937-7F37-41B1-96A5-B099A8657DDE}\"/></ObjectStream>"));
  106. EXPECT_EQ(
  107. AssetBuilderSDK::JobProduct::InferAssetTypeByProductFileName(dummyFileName.toUtf8().data()),
  108. AZ::Uuid("{49375937-7F37-41B1-96A5-B099A8657DDE}"));
  109. // use old format which doesnt use 'class' keyword
  110. EXPECT_TRUE(UnitTestUtils::CreateDummyFile(
  111. dummyFileName, "<ObjectStream><ASDFASDFASDFASDF type=\"{49375937-7F37-41B1-96A5-B099A8657DDE}\"/></ObjectStream>"));
  112. EXPECT_EQ(
  113. AssetBuilderSDK::JobProduct::InferAssetTypeByProductFileName(dummyFileName.toUtf8().data()),
  114. AZ::Uuid("{49375937-7F37-41B1-96A5-B099A8657DDE}"));
  115. // special case - recognize the old UICanvas format :(
  116. const char* canvasFile =
  117. "<ObjectStream version=\"1\">"
  118. " <Entity type=\"{75651658-8663-478D-9090-2432DFCAFA44}\">"
  119. " <uint64 name=\"Id\" value=\"13069065444211002982\" type=\"{D6597933-47CD-4FC8-B911-63F3E2B0993A}\"/>"
  120. " <AZStd::string name=\"Name\" value=\"triggerex01.xml\" type=\"{EF8FF807-DDEE-4EB0-B678-4CA3A2C490A4}\"/>"
  121. " <bool name=\"IsDependencyReady\" value=\"true\" type=\"{A0CA880C-AFE4-43CB-926C-59AC48496112}\"/>"
  122. " <AZStd::vector name=\"Components\" type=\"{2BADE35A-6F1B-4698-B2BC-3373D010020C}\">"
  123. " <UiCanvas name=\"element\" version=\"1\" type=\"{50B8CF6C-B19A-4D86-AFE9-96EFB820D422}\">"
  124. " </UiCanvas>"
  125. " </AZStd::vector>"
  126. " </Entity>"
  127. "</ObjectStream>";
  128. EXPECT_TRUE(UnitTestUtils::CreateDummyFile(dummyFileName, canvasFile));
  129. EXPECT_EQ(
  130. AssetBuilderSDK::JobProduct::InferAssetTypeByProductFileName(dummyFileName.toUtf8().data()),
  131. AZ::Uuid("{E48DDAC8-1F1E-4183-AAAB-37424BCC254B}"));
  132. }
  133. TEST_F(BuilderSDKUnitTests, SubIDMath_MatchesExpectedResults)
  134. {
  135. AZ::u32 tester = AssetBuilderSDK::ConstructSubID(5, 10);
  136. EXPECT_EQ(AssetBuilderSDK::GetSubID_ID(tester), 5);
  137. EXPECT_EQ(AssetBuilderSDK::GetSubID_LOD(tester), 10);
  138. tester |= 0xFFF00000;
  139. for (AZ::u32 idx = 0; idx < 9; ++idx)
  140. {
  141. EXPECT_EQ(AssetBuilderSDK::GetSubID_ID(AssetBuilderSDK::ConstructSubID(0, idx)), 0);
  142. EXPECT_EQ(AssetBuilderSDK::GetSubID_LOD(AssetBuilderSDK::ConstructSubID(0, idx)), idx);
  143. EXPECT_EQ(AssetBuilderSDK::GetSubID_ID(AssetBuilderSDK::ConstructSubID(idx, 0)), idx);
  144. EXPECT_EQ(AssetBuilderSDK::GetSubID_LOD(AssetBuilderSDK::ConstructSubID(idx, 0)), 0);
  145. EXPECT_EQ(AssetBuilderSDK::GetSubID_ID(AssetBuilderSDK::ConstructSubID(9 - idx, idx)), 9 - idx);
  146. EXPECT_EQ(AssetBuilderSDK::GetSubID_LOD(AssetBuilderSDK::ConstructSubID(9 - idx, idx)), idx);
  147. }
  148. // make sure that the flags are not disturbed if you modify the id and lods and pass the previous value in.
  149. // we pass in the existing value as the third param, which should replace the existing ids and lods, but keep the flags.
  150. tester = AssetBuilderSDK::ConstructSubID(512, 12, tester);
  151. EXPECT_EQ((tester & 0xFFF00000), 0xFFF00000);
  152. EXPECT_EQ(AssetBuilderSDK::GetSubID_ID(tester), 512);
  153. EXPECT_EQ(AssetBuilderSDK::GetSubID_LOD(tester), 12);
  154. }
  155. TEST_F(BuilderSDKUnitTests, SubIdGeneration_MatchesExpectedResult)
  156. {
  157. // test subid autogeneration
  158. // files with no UUID and no extension always return null
  159. EXPECT_EQ(AssetBuilderSDK::JobProduct::InferSubIDFromProductFileName(AZ::Data::AssetType::CreateNull(), "blah"), 0);
  160. // files with no UUID and no known extension always return null
  161. EXPECT_EQ(
  162. AssetBuilderSDK::JobProduct::InferSubIDFromProductFileName(AZ::Data::AssetType::CreateNull(), "blah.whatever"), 0);
  163. // ("editor") slices always have subid 1
  164. EXPECT_EQ(AssetBuilderSDK::JobProduct::InferSubIDFromProductFileName(sliceAssetType, "blah.slice"), 1);
  165. }
  166. } // namespace UnitTest