A stable modding interface between Techblox and mods https://mod.exmods.org/
您最多选择25个主题 主题必须以字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符

202 行
11KB

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Reflection;
  5. using DataLoader;
  6. using Svelto.Tasks;
  7. using Unity.Mathematics;
  8. using TechbloxModdingAPI.App;
  9. using TechbloxModdingAPI.Tests;
  10. using TechbloxModdingAPI.Utility;
  11. namespace TechbloxModdingAPI.Blocks
  12. {
  13. #if TEST
  14. /// <summary>
  15. /// Block test cases. Not accessible in release versions.
  16. /// </summary>
  17. [APITestClass]
  18. public static class BlockTests
  19. {
  20. [APITestCase(TestType.Game)] //At least one block must be placed for simulation to work
  21. public static void TestPlaceNew()
  22. {
  23. Block newBlock = Block.PlaceNew(BlockIDs.Cube, float3.zero);
  24. Assert.NotNull(newBlock.Id, "Newly placed block is missing Id. This should be populated when the block is placed.", "Newly placed block Id is not null, block successfully placed.");
  25. }
  26. [APITestCase(TestType.EditMode)]
  27. public static void TestInitProperty()
  28. {
  29. Block newBlock = Block.PlaceNew(BlockIDs.Cube, float3.zero + 2);
  30. if (!Assert.CloseTo(newBlock.Position, (float3.zero + 2), $"Newly placed block at {newBlock.Position} is expected at {Unity.Mathematics.float3.zero + 2}.", "Newly placed block position matches.")) return;
  31. //Assert.Equal(newBlock.Exists, true, "Newly placed block does not exist, possibly because Sync() skipped/missed/failed.", "Newly placed block exists, Sync() successful.");
  32. }
  33. [APITestCase(TestType.EditMode)]
  34. public static void TestBlockIDCoverage()
  35. {
  36. Assert.Equal(
  37. FullGameFields._dataDb.GetValues<CubeListData>().Keys.Select(ushort.Parse).OrderBy(id => id)
  38. .SequenceEqual(Enum.GetValues(typeof(BlockIDs)).Cast<ushort>().OrderBy(id => id)
  39. .Except(new[] {(ushort) BlockIDs.Invalid})), true,
  40. "Block ID enum is different than the known block types, update needed.",
  41. "Block ID enum matches the known block types.");
  42. }
  43. [APITestCase(TestType.EditMode)]
  44. public static void TestBlockIDs()
  45. {
  46. float3 pos = new float3();
  47. foreach (BlockIDs id in Enum.GetValues(typeof(BlockIDs)))
  48. {
  49. if (id == BlockIDs.Invalid) continue;
  50. try
  51. {
  52. Block.PlaceNew(id, pos);
  53. pos += 0.2f;
  54. }
  55. catch (Exception e)
  56. { //Only print failed case
  57. Assert.Fail($"Failed to place block type {id}: {e}");
  58. return;
  59. }
  60. }
  61. Assert.Pass("Placing all possible block types succeeded.");
  62. }
  63. [APITestCase(TestType.EditMode)]
  64. public static IEnumerator<TaskContract> TestBlockProperties()
  65. { //Uses the result of the previous test case
  66. var blocks = Game.CurrentGame().GetBlocksInGame();
  67. for (var index = 0; index < blocks.Length; index++)
  68. {
  69. if (index % 50 == 0) yield return Yield.It; //The material or flipped status can only be changed 130 times per submission
  70. var block = blocks[index];
  71. if (!block.Exists) continue;
  72. foreach (var property in block.GetType().GetProperties())
  73. {
  74. //Includes specialised block properties
  75. if (property.SetMethod == null) continue;
  76. var testValues = new (Type, object, Predicate<object>)[]
  77. {
  78. //(type, default value, predicate or null for equality)
  79. (typeof(long), 3, null),
  80. (typeof(int), 4, null),
  81. (typeof(double), 5.2f, obj => Math.Abs((double) obj - 5.2f) < float.Epsilon),
  82. (typeof(float), 5.2f, obj => Math.Abs((float) obj - 5.2f) < float.Epsilon),
  83. (typeof(bool), true, obj => (bool) obj),
  84. (typeof(string), "Test", obj => (string) obj == "Test"), //String equality check
  85. (typeof(float3), (float3) 2, obj => math.all((float3) obj - 2 < (float3) float.Epsilon)),
  86. (typeof(BlockColor), new BlockColor(BlockColors.Aqua, 2), null),
  87. (typeof(float4), (float4) 5, obj => math.all((float4) obj - 5 < (float4) float.Epsilon))
  88. };
  89. var propType = property.PropertyType;
  90. if (!propType.IsValueType) continue;
  91. (object valueToUse, Predicate<object> predicateToUse) = (null, null);
  92. foreach (var (type, value, predicate) in testValues)
  93. {
  94. if (type.IsAssignableFrom(propType))
  95. {
  96. valueToUse = value;
  97. predicateToUse = predicate ?? (obj => Equals(obj, value));
  98. break;
  99. }
  100. }
  101. if (propType.IsEnum)
  102. {
  103. var values = propType.GetEnumValues();
  104. valueToUse = values.GetValue(values.Length / 2);
  105. predicateToUse = val => Equals(val, valueToUse);
  106. }
  107. if (valueToUse == null)
  108. {
  109. Assert.Fail($"Property {block.GetType().Name}.{property.Name} has an unknown type {propType}, test needs fixing.");
  110. yield break;
  111. }
  112. property.SetValue(block, valueToUse);
  113. object got = property.GetValue(block);
  114. var attr = property.GetCustomAttribute<TestValueAttribute>();
  115. if (!predicateToUse(got) && (attr == null || !Equals(attr.PossibleValue, got)))
  116. {
  117. Assert.Fail($"Property {block.GetType().Name}.{property.Name} value {got} does not equal {valueToUse} for block {block}.");
  118. yield break;
  119. }
  120. }
  121. }
  122. Assert.Pass("Setting all possible properties of all registered API block types succeeded.");
  123. }
  124. [APITestCase(TestType.EditMode)]
  125. public static void TestDampedSpring()
  126. {
  127. Block newBlock = Block.PlaceNew(BlockIDs.DampedSpring, Unity.Mathematics.float3.zero + 1);
  128. DampedSpring b = null; // Note: the assignment operation is a lambda, which slightly confuses the compiler
  129. Assert.Errorless(() => { b = (DampedSpring) newBlock; }, "Casting block to DampedSpring raised an exception: ", "Casting block to DampedSpring completed without issue.");
  130. if (!Assert.CloseTo(b.Stiffness, 30f, $"DampedSpring.Stiffness {b.Stiffness} does not equal default value, possibly because it failed silently.", "DampedSpring.Stiffness is close enough to default.")) return;
  131. if (!Assert.CloseTo(b.Damping, 30f, $"DampedSpring.Damping {b.Damping} does not equal default value, possibly because it failed silently.", "DampedSpring.Damping is close enough to default.")) return;
  132. if (!Assert.CloseTo(b.MaxExtension, 0.3f, $"DampedSpring.MaxExtension {b.MaxExtension} does not equal default value, possibly because it failed silently.", "DampedSpring.MaxExtension is close enough to default.")) return;
  133. }
  134. /*[APITestCase(TestType.Game)]
  135. public static void TestMusicBlock1()
  136. {
  137. Block newBlock = Block.PlaceNew(BlockIDs.MusicBlock, Unity.Mathematics.float3.zero + 2);
  138. MusicBlock b = null; // Note: the assignment operation is a lambda, which slightly confuses the compiler
  139. Assert.Errorless(() => { b = newBlock.Specialise<MusicBlock>(); }, "Block.Specialize<MusicBlock>() raised an exception: ", "Block.Specialize<MusicBlock>() completed without issue.");
  140. if (!Assert.NotNull(b, "Block.Specialize<MusicBlock>() returned null, possibly because it failed silently.", "Specialized MusicBlock is not null.")) return;
  141. if (!Assert.CloseTo(b.Volume, 100f, $"MusicBlock.Volume {b.Volume} does not equal default value, possibly because it failed silently.", "MusicBlock.Volume is close enough to default.")) return;
  142. if (!Assert.Equal(b.TrackIndex, 0, $"MusicBlock.TrackIndex {b.TrackIndex} does not equal default value, possibly because it failed silently.", "MusicBlock.TrackIndex is equal to default.")) return;
  143. _musicBlock = b;
  144. }
  145. private static MusicBlock _musicBlock;
  146. [APITestCase(TestType.EditMode)]
  147. public static void TestMusicBlock2()
  148. {
  149. //Block newBlock = Block.GetLastPlacedBlock();
  150. var b = _musicBlock;
  151. if (!Assert.NotNull(b, "Block.Specialize<MusicBlock>() returned null, possibly because it failed silently.", "Specialized MusicBlock is not null.")) return;
  152. b.IsPlaying = true; // play sfx
  153. if (!Assert.Equal(b.IsPlaying, true, $"MusicBlock.IsPlaying {b.IsPlaying} does not equal true, possibly because it failed silently.", "MusicBlock.IsPlaying is set properly.")) return;
  154. if (!Assert.Equal(b.ChannelType, ChannelType.None, $"MusicBlock.ChannelType {b.ChannelType} does not equal default value, possibly because it failed silently.", "MusicBlock.ChannelType is equal to default.")) return;
  155. //Assert.Log(b.Track.ToString());
  156. if (!Assert.Equal(b.Track.ToString(), new Guid("3237ff8f-f5f2-4f84-8144-496ca280f8c0").ToString(), $"MusicBlock.Track {b.Track} does not equal default value, possibly because it failed silently.", "MusicBlock.Track is equal to default.")) return;
  157. }
  158. [APITestCase(TestType.EditMode)]
  159. public static void TestLogicGate()
  160. {
  161. Block newBlock = Block.PlaceNew(BlockIDs.NOTLogicBlock, Unity.Mathematics.float3.zero + 1);
  162. LogicGate b = null; // Note: the assignment operation is a lambda, which slightly confuses the compiler
  163. Assert.Errorless(() => { b = newBlock.Specialise<LogicGate>(); }, "Block.Specialize<LogicGate>() raised an exception: ", "Block.Specialize<LogicGate>() completed without issue.");
  164. if (!Assert.NotNull(b, "Block.Specialize<LogicGate>() returned null, possibly because it failed silently.", "Specialized LogicGate is not null.")) return;
  165. if (!Assert.Equal(b.InputCount, 1u, $"LogicGate.InputCount {b.InputCount} does not equal default value, possibly because it failed silently.", "LogicGate.InputCount is default.")) return;
  166. if (!Assert.Equal(b.OutputCount, 1u, $"LogicGate.OutputCount {b.OutputCount} does not equal default value, possibly because it failed silently.", "LogicGate.OutputCount is default.")) return;
  167. if (!Assert.NotNull(b, "Block.Specialize<LogicGate>() returned null, possibly because it failed silently.", "Specialized LogicGate is not null.")) return;
  168. //if (!Assert.Equal(b.PortName(0, true), "Input", $"LogicGate.PortName(0, input:true) {b.PortName(0, true)} does not equal default value, possibly because it failed silently.", "LogicGate.PortName(0, input:true) is close enough to default.")) return;
  169. LogicGate target = null;
  170. if (!Assert.Errorless(() => { target = Block.PlaceNew<LogicGate>(BlockIDs.ANDLogicBlock, Unity.Mathematics.float3.zero + 2); })) return;
  171. Wire newWire = null;
  172. if (!Assert.Errorless(() => { newWire = b.Connect(0, target, 0);})) return;
  173. if (!Assert.NotNull(newWire, "SignalingBlock.Connect(...) returned null, possible because it failed silently.", "SignalingBlock.Connect(...) returned a non-null value.")) return;
  174. }*/
  175. /*[APITestCase(TestType.EditMode)]
  176. public static void TestSpecialiseError()
  177. {
  178. Block newBlock = Block.PlaceNew(BlockIDs.Bench, new float3(1, 1, 1));
  179. if (Assert.Errorful<BlockTypeException>(() => newBlock.Specialise<MusicBlock>(), "Block.Specialise<MusicBlock>() was expected to error on a bench block.", "Block.Specialise<MusicBlock>() errored as expected for a bench block.")) return;
  180. }*/
  181. }
  182. #endif
  183. }