com.wallstop-studios.unity-helpers 2.0.0-rc58 → 2.0.0-rc60

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (298) hide show
  1. package/.config/dotnet-tools.json +9 -9
  2. package/.editorconfig +184 -184
  3. package/.gitattributes +63 -63
  4. package/.github/workflows/npm-publish.yml +75 -66
  5. package/.pre-commit-config.yaml +21 -21
  6. package/Editor/AnimationCopier.cs +181 -181
  7. package/Editor/AnimationCopier.cs.meta +2 -2
  8. package/Editor/AnimationCreator.cs +253 -253
  9. package/Editor/AnimationCreator.cs.meta +11 -11
  10. package/Editor/AnimationEventEditor.cs +887 -887
  11. package/Editor/AnimatorControllerCopier.cs +162 -162
  12. package/Editor/AnimatorControllerCopier.cs.meta +2 -2
  13. package/Editor/CustomEditors/MatchColliderToSpriteEditor.cs +34 -34
  14. package/Editor/CustomEditors/MatchColliderToSpriteEditor.cs.meta +2 -2
  15. package/Editor/CustomEditors.meta +2 -2
  16. package/Editor/FitTextureSizeWizard.cs +147 -147
  17. package/Editor/FitTextureSizeWizard.cs.meta +2 -2
  18. package/Editor/PrefabCheckWizard.cs +170 -170
  19. package/Editor/PrefabCheckWizard.cs.meta +11 -11
  20. package/Editor/SpriteSettingsApplier.cs +272 -272
  21. package/Editor/SpriteSettingsApplier.cs.meta +2 -2
  22. package/Editor/StringInListeDrawer.cs +56 -56
  23. package/Editor/TextureResizerWizard.cs +181 -181
  24. package/Editor/TextureResizerWizard.cs.meta +2 -2
  25. package/Editor/TextureSettingsApplier.cs +178 -178
  26. package/Editor/TextureSettingsApplier.cs.meta +2 -2
  27. package/Editor/Utils/DxReadOnlyPropertyDrawer.cs +26 -26
  28. package/Editor/Utils/DxReadOnlyPropertyDrawer.cs.meta +11 -11
  29. package/Editor/Utils/EditorUtilities.cs +22 -22
  30. package/Editor/Utils/EditorUtilities.cs.meta +11 -11
  31. package/Editor/Utils.meta +8 -8
  32. package/Editor/WShowIfPropertyDrawer.cs +63 -63
  33. package/Editor/WallstopStudios.UnityHelpers.Editor.asmdef +17 -17
  34. package/Editor/WallstopStudios.UnityHelpers.Editor.asmdef.meta +7 -7
  35. package/LICENSE +21 -21
  36. package/LICENSE.md +6 -6
  37. package/LICENSE.meta +7 -7
  38. package/README.md +177 -177
  39. package/Runtime/Binaries/Microsoft.Bcl.AsyncInterfaces.dll.meta +33 -33
  40. package/Runtime/Binaries/Microsoft.Bcl.AsyncInterfaces.xml +223 -223
  41. package/Runtime/Binaries/Microsoft.Bcl.AsyncInterfaces.xml.meta +7 -7
  42. package/Runtime/Binaries/System.Text.Encodings.Web.dll.meta +33 -33
  43. package/Runtime/Binaries/System.Text.Encodings.Web.xml +935 -935
  44. package/Runtime/Binaries/System.Text.Encodings.Web.xml.meta +7 -7
  45. package/Runtime/Binaries/System.Text.Json.dll.meta +33 -33
  46. package/Runtime/Binaries/System.Text.Json.xml +4829 -4829
  47. package/Runtime/Binaries/System.Text.Json.xml.meta +7 -7
  48. package/Runtime/Binaries.meta +8 -8
  49. package/Runtime/Core/Attributes/AnimationEventAttribute.cs +131 -131
  50. package/Runtime/Core/Attributes/ChildComponentAttribute.cs +213 -209
  51. package/Runtime/Core/Attributes/DxReadOnlyAttribute.cs +6 -6
  52. package/Runtime/Core/Attributes/KSerializableAttribute.cs +19 -19
  53. package/Runtime/Core/Attributes/NotNullAttribute.cs +32 -32
  54. package/Runtime/Core/Attributes/ParentComponent.cs +185 -185
  55. package/Runtime/Core/Attributes/RelationalComponentExtensions.cs +14 -14
  56. package/Runtime/Core/Attributes/SiblingComponentAttribute.cs +117 -117
  57. package/Runtime/Core/Attributes/SiblingComponentAttribute.cs.meta +11 -11
  58. package/Runtime/Core/Attributes/ValidateAssignmentAttribute.cs +101 -101
  59. package/Runtime/Core/Attributes/ValidateAssignmentAttribute.cs.meta +11 -11
  60. package/Runtime/Core/Attributes/WShowIfAttribute.cs +16 -16
  61. package/Runtime/Core/Attributes.meta +8 -8
  62. package/Runtime/Core/DataStructure/Adapters/FastVector2Int.cs +92 -92
  63. package/Runtime/Core/DataStructure/Adapters/FastVector3Int.cs +185 -185
  64. package/Runtime/Core/DataStructure/Adapters/KGuid.cs +305 -305
  65. package/Runtime/Core/DataStructure/Adapters/KVector2.cs +80 -80
  66. package/Runtime/Core/DataStructure/Circle.cs +50 -50
  67. package/Runtime/Core/DataStructure/CyclicBuffer.cs +193 -153
  68. package/Runtime/Core/DataStructure/ISpatialTree.cs +60 -60
  69. package/Runtime/Core/DataStructure/ISpatialTree.cs.meta +11 -11
  70. package/Runtime/Core/DataStructure/KDTree.cs +292 -292
  71. package/Runtime/Core/DataStructure/KDTree.cs.meta +11 -11
  72. package/Runtime/Core/DataStructure/QuadTree.cs +287 -287
  73. package/Runtime/Core/DataStructure/RTree.cs +346 -346
  74. package/Runtime/Core/DataStructure/RTree.cs.meta +11 -11
  75. package/Runtime/Core/DataStructure/StringWrapper.cs +91 -91
  76. package/Runtime/Core/DataStructure/TimedCache.cs +66 -66
  77. package/Runtime/Core/Extension/AnimatorExtensions.cs +25 -25
  78. package/Runtime/Core/Extension/AsyncOperationExtensions.cs +110 -110
  79. package/Runtime/Core/Extension/CircleExtensions.cs +25 -25
  80. package/Runtime/Core/Extension/ColorExtensions.cs +629 -629
  81. package/Runtime/Core/Extension/DictionaryExtensions.cs +279 -279
  82. package/Runtime/Core/Extension/DirectionExtensions.cs +213 -213
  83. package/Runtime/Core/Extension/EnumExtensions.cs +37 -37
  84. package/Runtime/Core/Extension/EnumExtensions.cs.meta +2 -2
  85. package/Runtime/Core/Extension/HashSetExtensions.cs +12 -12
  86. package/Runtime/Core/Extension/IEnumerableExtensions.cs +122 -122
  87. package/Runtime/Core/Extension/IListExtensions.cs +146 -106
  88. package/Runtime/Core/Extension/LoggingExtensions.cs +264 -258
  89. package/Runtime/Core/Extension/RandomExtensions.cs +109 -109
  90. package/Runtime/Core/Extension/SerializedPropertyExtensions.cs +157 -157
  91. package/Runtime/Core/Extension/StringExtensions.cs +180 -151
  92. package/Runtime/Core/Extension/UnityExtensions.cs +1608 -1608
  93. package/Runtime/Core/Helper/ArrayConverter.cs +39 -39
  94. package/Runtime/Core/Helper/ArrayConverter.cs.meta +2 -2
  95. package/Runtime/Core/Helper/AssignUtilities.cs +14 -14
  96. package/Runtime/Core/Helper/AssignUtilities.cs.meta +11 -11
  97. package/Runtime/Core/Helper/DirectoryHelper.cs +132 -0
  98. package/Runtime/Core/Helper/DirectoryHelper.cs.meta +3 -0
  99. package/Runtime/Core/Helper/Enumerables.cs +17 -17
  100. package/Runtime/Core/Helper/FormattingHelpers.cs +32 -32
  101. package/Runtime/Core/Helper/Geometry.cs +43 -43
  102. package/Runtime/Core/Helper/Helpers.cs +722 -722
  103. package/Runtime/Core/Helper/Helpers.cs.meta +11 -11
  104. package/Runtime/Core/Helper/IterationHelpers.cs +32 -32
  105. package/Runtime/Core/Helper/IterationHelpers.cs.meta +11 -11
  106. package/Runtime/Core/Helper/LifetimeHelpers.cs +13 -13
  107. package/Runtime/Core/Helper/Objects.cs +769 -769
  108. package/Runtime/Core/Helper/Partials/LogHelpers.cs +13 -13
  109. package/Runtime/Core/Helper/Partials/LogHelpers.cs.meta +2 -2
  110. package/Runtime/Core/Helper/Partials/MathHelpers.cs +30 -30
  111. package/Runtime/Core/Helper/Partials/MathHelpers.cs.meta +2 -2
  112. package/Runtime/Core/Helper/Partials/ObjectHelpers.cs +388 -388
  113. package/Runtime/Core/Helper/Partials/ObjectHelpers.cs.meta +2 -2
  114. package/Runtime/Core/Helper/Partials/TransformHelpers.cs +189 -189
  115. package/Runtime/Core/Helper/Partials/TransformHelpers.cs.meta +2 -2
  116. package/Runtime/Core/Helper/Partials.meta +2 -2
  117. package/Runtime/Core/Helper/ReflectionHelpers.cs +452 -452
  118. package/Runtime/Core/Helper/ReflectionHelpers.cs.meta +2 -2
  119. package/Runtime/Core/Helper/SceneHelper.cs +209 -209
  120. package/Runtime/Core/Helper/SpriteHelpers.cs +41 -41
  121. package/Runtime/Core/Helper/SpriteHelpers.cs.meta +11 -11
  122. package/Runtime/Core/Helper/StringInList.cs +31 -31
  123. package/Runtime/Core/Helper/StringInList.cs.meta +11 -11
  124. package/Runtime/Core/Helper/UnityMainThreadDispatcher.cs +82 -82
  125. package/Runtime/Core/Helper/WallMath.cs +166 -166
  126. package/Runtime/Core/Math/Line.cs +55 -55
  127. package/Runtime/Core/Math/Parabola.cs +47 -47
  128. package/Runtime/Core/Math/PointPolygonCheck.cs +36 -36
  129. package/Runtime/Core/Math/PointPolygonCheck.cs.meta +11 -11
  130. package/Runtime/Core/Math/Range.cs +92 -92
  131. package/Runtime/Core/Math/XXHash.cs +310 -310
  132. package/Runtime/Core/Math/XXHash.cs.meta +11 -11
  133. package/Runtime/Core/Model/Direction.cs +43 -43
  134. package/Runtime/Core/OneOf/FastOneOf.cs +152 -152
  135. package/Runtime/Core/OneOf/None.cs +4 -4
  136. package/Runtime/Core/Random/AbstractRandom.cs +585 -585
  137. package/Runtime/Core/Random/DotNetRandom.cs +54 -54
  138. package/Runtime/Core/Random/DotNetRandom.cs.meta +2 -2
  139. package/Runtime/Core/Random/IRandom.cs +161 -161
  140. package/Runtime/Core/Random/LinearCongruentialGenerator.cs +49 -49
  141. package/Runtime/Core/Random/NativePcgRandom.cs +97 -97
  142. package/Runtime/Core/Random/PRNG.cs +7 -7
  143. package/Runtime/Core/Random/PRNG.cs.meta +2 -2
  144. package/Runtime/Core/Random/PcgRandom.cs +149 -149
  145. package/Runtime/Core/Random/PerlinNoise.cs +369 -369
  146. package/Runtime/Core/Random/PerlinNoise.cs.meta +2 -2
  147. package/Runtime/Core/Random/RandomState.cs +131 -131
  148. package/Runtime/Core/Random/RandomUtilities.cs +26 -26
  149. package/Runtime/Core/Random/RandomUtilities.cs.meta +11 -11
  150. package/Runtime/Core/Random/RomuDuo.cs +116 -116
  151. package/Runtime/Core/Random/RomuDuo.cs.meta +2 -2
  152. package/Runtime/Core/Random/SplitMix64.cs +94 -94
  153. package/Runtime/Core/Random/SplitMix64.cs.meta +2 -2
  154. package/Runtime/Core/Random/SquirrelRandom.cs +84 -84
  155. package/Runtime/Core/Random/SystemRandom.cs +162 -162
  156. package/Runtime/Core/Random/ThreadLocalRandom.cs +12 -12
  157. package/Runtime/Core/Random/UnityRandom.cs +57 -57
  158. package/Runtime/Core/Random/UnityRandom.cs.meta +11 -11
  159. package/Runtime/Core/Random/WyRandom.cs +121 -121
  160. package/Runtime/Core/Random/WyRandom.cs.meta +2 -2
  161. package/Runtime/Core/Random/XorShiftRandom.cs +52 -52
  162. package/Runtime/Core/Random/XorShiftRandom.cs.meta +11 -11
  163. package/Runtime/Core/Random/XorShiroRandom.cs +119 -119
  164. package/Runtime/Core/Random/XorShiroRandom.cs.meta +2 -2
  165. package/Runtime/Core/Serialization/JsonConverters/ColorConverter.cs +88 -88
  166. package/Runtime/Core/Serialization/JsonConverters/GameObjectConverter.cs +37 -37
  167. package/Runtime/Core/Serialization/JsonConverters/Matrix4x4Converter.cs +218 -218
  168. package/Runtime/Core/Serialization/JsonConverters/TypeConverter.cs +28 -28
  169. package/Runtime/Core/Serialization/JsonConverters/Vector2Converter.cs +74 -74
  170. package/Runtime/Core/Serialization/JsonConverters/Vector3Converter.cs +81 -81
  171. package/Runtime/Core/Serialization/JsonConverters/Vector4Converter.cs +88 -88
  172. package/Runtime/Core/Serialization/Serializer.cs +195 -195
  173. package/Runtime/Core/Threading/SingleThreadedThreadPool.cs +113 -113
  174. package/Runtime/Protobuf-Net/System.Buffers.dll.meta +33 -33
  175. package/Runtime/Protobuf-Net/System.Collections.Immutable.dll.meta +33 -33
  176. package/Runtime/Protobuf-Net/System.Collections.Immutable.xml +5379 -5379
  177. package/Runtime/Protobuf-Net/System.Collections.Immutable.xml.meta +7 -7
  178. package/Runtime/Protobuf-Net/System.Numerics.Vectors.dll.meta +33 -33
  179. package/Runtime/Protobuf-Net/System.Runtime.CompilerServices.Unsafe.dll.meta +33 -33
  180. package/Runtime/Protobuf-Net/System.Runtime.CompilerServices.Unsafe.xml +290 -290
  181. package/Runtime/Protobuf-Net/System.Runtime.CompilerServices.Unsafe.xml.meta +7 -7
  182. package/Runtime/Protobuf-Net/protobuf-net.Core.dll.meta +33 -33
  183. package/Runtime/Protobuf-Net/protobuf-net.dll.meta +33 -33
  184. package/Runtime/UI/LayeredImage.cs +364 -364
  185. package/Runtime/UI/LayeredImage.cs.meta +2 -2
  186. package/Runtime/UI.meta +2 -2
  187. package/Runtime/Utils/AnimationEventEqualityComparer.cs +161 -161
  188. package/Runtime/Utils/AnimatorEnumStateMachine.cs +88 -88
  189. package/Runtime/Utils/Buffers.cs +33 -33
  190. package/Runtime/Utils/CenterPointOffset.cs +30 -30
  191. package/Runtime/Utils/CenterPointOffset.cs.meta +2 -2
  192. package/Runtime/Utils/CircleLineRenderer.cs +134 -134
  193. package/Runtime/Utils/CoroutineHandler.cs +4 -4
  194. package/Runtime/Utils/CoroutineHandler.cs.meta +2 -2
  195. package/Runtime/Utils/DeferredDisposalResult.cs +23 -23
  196. package/Runtime/Utils/MatchColliderToSprite.cs +94 -94
  197. package/Runtime/Utils/MatchColliderToSprite.cs.meta +2 -2
  198. package/Runtime/Utils/Oscillator.cs +27 -27
  199. package/Runtime/Utils/RuntimeSingleton.cs +71 -69
  200. package/Runtime/Utils/RuntimeSingleton.cs.meta +11 -11
  201. package/Runtime/Utils/SetTextureImportData.cs +69 -69
  202. package/Runtime/Utils/SpriteRendererMetadata.cs +374 -374
  203. package/Runtime/Utils/SpriteRendererMetadata.cs.meta +2 -2
  204. package/Runtime/Utils/SpriteRendererSyncer.cs +100 -100
  205. package/Runtime/Utils/SpriteRendererSyncer.cs.meta +2 -2
  206. package/Runtime/Utils/StartTracker.cs +15 -0
  207. package/Runtime/Utils/StartTracker.cs.meta +3 -0
  208. package/Runtime/Utils/TextureScale.cs +179 -179
  209. package/Runtime/Utils/TextureScale.cs.meta +2 -2
  210. package/Runtime/Utils/UnityObjectNameComparer.cs +33 -0
  211. package/Runtime/Utils/UnityObjectNameComparer.cs.meta +3 -0
  212. package/Runtime/WallstopStudios.UnityHelpers.asmdef +13 -13
  213. package/Tests/Runtime/Attributes/ChildComponentTests.cs +81 -81
  214. package/Tests/Runtime/Attributes/Components/ExpectChildSpriteRenderers.cs +28 -28
  215. package/Tests/Runtime/Attributes/Components/ExpectParentSpriteRenderers.cs +28 -28
  216. package/Tests/Runtime/Attributes/ParentComponentTests.cs +68 -68
  217. package/Tests/Runtime/Components/RelationalComponentTesterComplex.cs +34 -34
  218. package/Tests/Runtime/Components/RelationalComponentTesterComplex.cs.meta +2 -2
  219. package/Tests/Runtime/Components/RelationalComponentsTesterSimple.cs +40 -40
  220. package/Tests/Runtime/Components.meta +2 -2
  221. package/Tests/Runtime/DataStructures/BalancedKDTreeTests.cs +14 -14
  222. package/Tests/Runtime/DataStructures/BalancedKDTreeTests.cs.meta +11 -11
  223. package/Tests/Runtime/DataStructures/CyclicBufferTests.cs +324 -324
  224. package/Tests/Runtime/DataStructures/QuadTreeTests.cs +14 -14
  225. package/Tests/Runtime/DataStructures/QuadTreeTests.cs.meta +11 -11
  226. package/Tests/Runtime/DataStructures/SpatialTreeTests.cs +130 -130
  227. package/Tests/Runtime/DataStructures/SpatialTreeTests.cs.meta +11 -11
  228. package/Tests/Runtime/DataStructures/UnbalancedKDTreeTests.cs +14 -14
  229. package/Tests/Runtime/DataStructures/UnbalancedKDTreeTests.cs.meta +11 -11
  230. package/Tests/Runtime/DataStructures.meta +8 -8
  231. package/Tests/Runtime/Extensions/DictionaryExtensionTests.cs +439 -439
  232. package/Tests/Runtime/Extensions/DictionaryExtensionTests.cs.meta +2 -2
  233. package/Tests/Runtime/Extensions/EnumExtensionTests.cs +128 -128
  234. package/Tests/Runtime/Extensions/EnumExtensionTests.cs.meta +2 -2
  235. package/Tests/Runtime/Extensions/IListExtensionTests.cs +104 -104
  236. package/Tests/Runtime/Extensions/RandomExtensionTests.cs +27 -27
  237. package/Tests/Runtime/Extensions/RandomExtensionTests.cs.meta +2 -2
  238. package/Tests/Runtime/Extensions/StringExtensionTests.cs +31 -31
  239. package/Tests/Runtime/Extensions/StringExtensionTests.cs.meta +2 -2
  240. package/Tests/Runtime/Extensions.meta +2 -2
  241. package/Tests/Runtime/Helper/ArrayConverterTests.cs +19 -19
  242. package/Tests/Runtime/Helper/ArrayConverterTests.cs.meta +2 -2
  243. package/Tests/Runtime/Helper/FormattingHelperTests.cs +129 -129
  244. package/Tests/Runtime/Helper/FormattingHelperTests.cs.meta +2 -2
  245. package/Tests/Runtime/Helper/ObjectHelperTests.cs +402 -402
  246. package/Tests/Runtime/Helper/ObjectHelperTests.cs.meta +2 -2
  247. package/Tests/Runtime/Helper/ReflectionHelperTests.cs +536 -536
  248. package/Tests/Runtime/Helper/SceneHelperTests.cs +94 -94
  249. package/Tests/Runtime/Helper/WallMathTests.cs +233 -233
  250. package/Tests/Runtime/Helper/WallMathTests.cs.meta +2 -2
  251. package/Tests/Runtime/Helper.meta +2 -2
  252. package/Tests/Runtime/Performance/KDTreePerformanceTests.cs +14 -14
  253. package/Tests/Runtime/Performance/KDTreePerformanceTests.cs.meta +11 -11
  254. package/Tests/Runtime/Performance/QuadTreePerformanceTests.cs +14 -14
  255. package/Tests/Runtime/Performance/QuadTreePerformanceTests.cs.meta +11 -11
  256. package/Tests/Runtime/Performance/RandomPerformanceTests.cs +180 -157
  257. package/Tests/Runtime/Performance/RandomPerformanceTests.cs.meta +11 -11
  258. package/Tests/Runtime/Performance/RelationComponentPerformanceTests.cs +61 -61
  259. package/Tests/Runtime/Performance/RelationComponentPerformanceTests.cs.meta +2 -2
  260. package/Tests/Runtime/Performance/SpatialTreePerformanceTest.cs +154 -154
  261. package/Tests/Runtime/Performance/SpatialTreePerformanceTest.cs.meta +11 -11
  262. package/Tests/Runtime/Performance/UnbalancedKDTreeTests.cs +14 -14
  263. package/Tests/Runtime/Performance/UnbalancedKDTreeTests.cs.meta +11 -11
  264. package/Tests/Runtime/Performance.meta +8 -8
  265. package/Tests/Runtime/Random/DotNetRandomTests.cs +9 -9
  266. package/Tests/Runtime/Random/DotNetRandomTests.cs.meta +2 -2
  267. package/Tests/Runtime/Random/LinearCongruentialGeneratorTests.cs +12 -12
  268. package/Tests/Runtime/Random/PcgRandomTests.cs +9 -9
  269. package/Tests/Runtime/Random/PcgRandomTests.cs.meta +11 -11
  270. package/Tests/Runtime/Random/RandomTestBase.cs +787 -787
  271. package/Tests/Runtime/Random/RandomTestBase.cs.meta +11 -11
  272. package/Tests/Runtime/Random/RomuDuoRandomTests.cs +9 -9
  273. package/Tests/Runtime/Random/RomuDuoRandomTests.cs.meta +2 -2
  274. package/Tests/Runtime/Random/SplitMix64RandomTests.cs +9 -9
  275. package/Tests/Runtime/Random/SplitMix64RandomTests.cs.meta +2 -2
  276. package/Tests/Runtime/Random/SquirrelRandomTests.cs +14 -14
  277. package/Tests/Runtime/Random/SquirrelRandomTests.cs.meta +11 -11
  278. package/Tests/Runtime/Random/SystemRandomTests.cs +10 -10
  279. package/Tests/Runtime/Random/SystemRandomTests.cs.meta +11 -11
  280. package/Tests/Runtime/Random/UnityRandomTests.cs +9 -9
  281. package/Tests/Runtime/Random/UnityRandomTests.cs.meta +11 -11
  282. package/Tests/Runtime/Random/WyRandomTests.cs +9 -9
  283. package/Tests/Runtime/Random/WyRandomTests.cs.meta +2 -2
  284. package/Tests/Runtime/Random/XorShiftRandomTests.cs +9 -9
  285. package/Tests/Runtime/Random/XorShiftRandomTests.cs.meta +11 -11
  286. package/Tests/Runtime/Random/XorShiroRandomTests.cs +9 -9
  287. package/Tests/Runtime/Random/XorShiroRandomTests.cs.meta +2 -2
  288. package/Tests/Runtime/Random.meta +8 -8
  289. package/Tests/Runtime/Serialization/JsonSerializationTest.cs +156 -156
  290. package/Tests/Runtime/Serialization/JsonSerializationTest.cs.meta +2 -2
  291. package/Tests/Runtime/Serialization.meta +2 -2
  292. package/Tests/Runtime/Utils/SpriteRendererMetadataTests.cs +399 -399
  293. package/Tests/Runtime/Utils/SpriteRendererMetadataTests.cs.meta +2 -2
  294. package/Tests/Runtime/Utils.meta +2 -2
  295. package/Tests/Runtime/WallstopStudios.UnityHelpers.Tests.Runtime.asmdef +22 -22
  296. package/Tests/Runtime/WallstopStudios.UnityHelpers.Tests.Runtime.asmdef.meta +7 -7
  297. package/Tests/Runtime.meta +8 -8
  298. package/package.json +38 -38
@@ -1,787 +1,787 @@
1
- namespace UnityHelpers.Tests.Random
2
- {
3
- using System;
4
- using System.Collections.Generic;
5
- using System.Linq;
6
- using System.Runtime.CompilerServices;
7
- using Core.DataStructure.Adapters;
8
- using Core.Extension;
9
- using Core.Serialization;
10
- using NUnit.Framework;
11
- using UnityHelpers.Core.Random;
12
-
13
- public enum TestValues
14
- {
15
- Value0,
16
- Value1,
17
- Value2,
18
- Value3,
19
- Value4,
20
- Value5,
21
- Value6,
22
- Value7,
23
- Value8,
24
- Value9,
25
- }
26
-
27
- public abstract class RandomTestBase
28
- {
29
- private const int NumGeneratorChecks = 1_000;
30
- private const int NormalIterations = 1_000;
31
- private const int SampleCount = 12_750_000;
32
-
33
- private readonly int[] _samples = new int[1_000];
34
-
35
- protected abstract IRandom NewRandom();
36
-
37
- [SetUp]
38
- public virtual void Setup()
39
- {
40
- Array.Clear(_samples, 0, _samples.Length);
41
- }
42
-
43
- [TearDown]
44
- public virtual void Teardown()
45
- {
46
- // No-op in base
47
- }
48
-
49
- [Test]
50
- [Parallelizable]
51
- public void Bool()
52
- {
53
- TestAndVerify(
54
- random => Math.Min(_samples.Length - 1, Convert.ToInt32(random.NextBool())),
55
- maxLength: Math.Min(2, _samples.Length)
56
- );
57
- }
58
-
59
- [Test]
60
- [Parallelizable]
61
- public void Int()
62
- {
63
- TestAndVerify(random => random.Next(0, _samples.Length));
64
- }
65
-
66
- [Test]
67
- [Parallelizable]
68
- public void IntRaw()
69
- {
70
- int sampleLength = GetSampleLength();
71
- TestAndVerify(random =>
72
- (int)((1.0 * random.Next()) / (1.0 * int.MaxValue) * sampleLength)
73
- );
74
- }
75
-
76
- [Test]
77
- [Parallelizable]
78
- public void IntRange()
79
- {
80
- TestAndVerify(random => random.Next(_samples.Length));
81
- }
82
-
83
- [Test]
84
- [Parallelizable]
85
- public void IntDistribution()
86
- {
87
- TestAndVerify(random =>
88
- (int)(random.Next() / ((1.0 * int.MaxValue) / _samples.Length))
89
- );
90
- }
91
-
92
- [Test]
93
- [Parallelizable]
94
- public void IntMaxRange()
95
- {
96
- TestAndVerify(random =>
97
- (int)(
98
- (random.Next(int.MinValue, int.MaxValue) + (-1.0 * int.MinValue))
99
- / (1.0 * int.MaxValue - int.MinValue)
100
- * _samples.Length
101
- )
102
- );
103
- }
104
-
105
- [Test]
106
- [Parallelizable]
107
- public void Uint()
108
- {
109
- TestAndVerify(random => (int)random.NextUint(0, (uint)_samples.Length));
110
- }
111
-
112
- [Test]
113
- [Parallelizable]
114
- public void UintRange()
115
- {
116
- TestAndVerify(random => (int)random.NextUint((uint)_samples.Length));
117
- }
118
-
119
- [Test]
120
- [Parallelizable]
121
- public void UintDistribution()
122
- {
123
- TestAndVerify(random =>
124
- (int)(random.NextUint() / ((1.0 * uint.MaxValue) / _samples.Length))
125
- );
126
- }
127
-
128
- [Test]
129
- [Parallelizable]
130
- public void UintMaxRange()
131
- {
132
- TestAndVerify(random =>
133
- (int)(
134
- (random.NextUint(uint.MinValue, uint.MaxValue) + (1.0 * uint.MinValue))
135
- / (1.0 * uint.MaxValue)
136
- * _samples.Length
137
- )
138
- );
139
- }
140
-
141
- [Test]
142
- [Parallelizable]
143
- public void Short()
144
- {
145
- TestAndVerify(
146
- random => random.NextShort(0, (short)_samples.Length),
147
- maxLength: (short.MaxValue - short.MinValue)
148
- );
149
- }
150
-
151
- [Test]
152
- [Parallelizable]
153
- public void ShortRaw()
154
- {
155
- int sampleLength = GetSampleLength(short.MaxValue);
156
- TestAndVerify(
157
- random => (int)((1.0 * random.NextShort()) / (1.0 * short.MaxValue) * sampleLength),
158
- maxLength: short.MaxValue
159
- );
160
- }
161
-
162
- [Test]
163
- [Parallelizable]
164
- public void ShortRange()
165
- {
166
- TestAndVerify(
167
- random => random.NextShort((short)_samples.Length),
168
- maxLength: (short.MaxValue - short.MinValue)
169
- );
170
- }
171
-
172
- [Test]
173
- [Parallelizable]
174
- public void ShortMaxRange()
175
- {
176
- TestAndVerify(random =>
177
- (int)(
178
- (random.NextShort(short.MinValue, short.MaxValue) + (-1.0 * short.MinValue))
179
- / (1.0 * short.MaxValue - short.MinValue)
180
- * _samples.Length
181
- )
182
- );
183
- }
184
-
185
- [Test]
186
- [Parallelizable]
187
- public void Byte()
188
- {
189
- TestAndVerify(
190
- random =>
191
- random.NextByte(
192
- 0,
193
- (byte)(_samples.Length < byte.MaxValue ? _samples.Length : byte.MaxValue)
194
- ),
195
- byte.MaxValue
196
- );
197
- }
198
-
199
- [Test]
200
- [Parallelizable]
201
- public void ByteRaw()
202
- {
203
- int sampleLength = GetSampleLength(byte.MaxValue);
204
- TestAndVerify(
205
- random => (int)((1.0 * random.NextByte()) / (1.0 * byte.MaxValue) * sampleLength),
206
- maxLength: byte.MaxValue
207
- );
208
- }
209
-
210
- [Test]
211
- [Parallelizable]
212
- public void ByteRange()
213
- {
214
- TestAndVerify(
215
- random =>
216
- random.NextByte(
217
- (byte)(_samples.Length < byte.MaxValue ? _samples.Length : byte.MaxValue)
218
- ),
219
- byte.MaxValue
220
- );
221
- }
222
-
223
- [Test]
224
- [Parallelizable]
225
- public void ByteMaxRange()
226
- {
227
- int sampleCount = Math.Min((byte.MaxValue - byte.MinValue), _samples.Length);
228
- TestAndVerify(
229
- random =>
230
- Math.Clamp(
231
- (int)(
232
- (random.NextByte(byte.MinValue, byte.MaxValue) + (-1.0 * byte.MinValue))
233
- / (1.0 * byte.MaxValue - byte.MinValue)
234
- * sampleCount
235
- ),
236
- 0,
237
- sampleCount - 1
238
- ),
239
- maxLength: sampleCount
240
- );
241
- }
242
-
243
- [Test]
244
- [Parallelizable]
245
- public void Float()
246
- {
247
- TestAndVerify(random =>
248
- Math.Clamp(
249
- (int)Math.Floor(random.NextFloat(0, _samples.Length)),
250
- 0,
251
- _samples.Length - 1
252
- )
253
- );
254
- }
255
-
256
- [Test]
257
- [Parallelizable]
258
- public void FloatRange()
259
- {
260
- TestAndVerify(random =>
261
- Math.Clamp(
262
- (int)Math.Floor(random.NextFloat(_samples.Length)),
263
- 0,
264
- _samples.Length - 1
265
- )
266
- );
267
- }
268
-
269
- [Test]
270
- [Parallelizable]
271
- public void FloatDistribution()
272
- {
273
- TestAndVerify(random => (int)(random.NextFloat() * _samples.Length));
274
- }
275
-
276
- [Test]
277
- [Parallelizable]
278
- public void FloatMaxRange()
279
- {
280
- TestAndVerify(random =>
281
- Math.Clamp(
282
- (int)(
283
- (random.NextFloat(float.MinValue, float.MaxValue) + (-1.0 * float.MinValue))
284
- / (1.0 * float.MaxValue - float.MinValue)
285
- * _samples.Length
286
- ),
287
- 0,
288
- _samples.Length - 1
289
- )
290
- );
291
- }
292
-
293
- [Test]
294
- [Parallelizable]
295
- public void Double()
296
- {
297
- TestAndVerify(random =>
298
- Math.Clamp(
299
- (int)Math.Floor(random.NextDouble(0, _samples.Length)),
300
- 0,
301
- _samples.Length - 1
302
- )
303
- );
304
- }
305
-
306
- [Test]
307
- [Parallelizable]
308
- public void DoubleRange()
309
- {
310
- TestAndVerify(random =>
311
- Math.Clamp(
312
- (int)Math.Floor(random.NextDouble(_samples.Length)),
313
- 0,
314
- _samples.Length - 1
315
- )
316
- );
317
- }
318
-
319
- [Test]
320
- [Parallelizable]
321
- public void DoubleDistribution()
322
- {
323
- TestAndVerify(random => (int)(random.NextDouble() * _samples.Length));
324
- }
325
-
326
- [Test]
327
- [Parallelizable]
328
- public void DoubleMaxRange()
329
- {
330
- IRandom random = NewRandom();
331
- for (int i = 0; i < SampleCount; ++i)
332
- {
333
- double value = random.NextDouble(double.MinValue, double.MaxValue);
334
- Assert.IsFalse(double.IsNaN(value));
335
- Assert.IsFalse(double.IsInfinity(value));
336
- }
337
- }
338
-
339
- [Test]
340
- [Parallelizable]
341
- public void Long()
342
- {
343
- TestAndVerify(random => (int)random.NextLong(0, _samples.Length));
344
- }
345
-
346
- [Test]
347
- [Parallelizable]
348
- public void LongRaw()
349
- {
350
- int sampleLength = GetSampleLength();
351
- TestAndVerify(random =>
352
- (int)(1.0 * random.NextLong() / (1.0 * long.MaxValue) * sampleLength)
353
- );
354
- }
355
-
356
- [Test]
357
- [Parallelizable]
358
- public void LongRange()
359
- {
360
- TestAndVerify(random => (int)random.NextLong(_samples.Length));
361
- }
362
-
363
- [Test]
364
- [Parallelizable]
365
- public void LongMaxRange()
366
- {
367
- TestAndVerify(random =>
368
- (int)(
369
- (random.NextLong(long.MinValue, long.MaxValue) + (-1.0 * long.MinValue))
370
- / (1.0 * long.MaxValue - long.MinValue)
371
- * _samples.Length
372
- )
373
- );
374
- }
375
-
376
- [Test]
377
- [Parallelizable]
378
- public void Ulong()
379
- {
380
- TestAndVerify(random => (int)random.NextUlong(0, (ulong)_samples.Length));
381
- }
382
-
383
- [Test]
384
- [Parallelizable]
385
- public void UlongRaw()
386
- {
387
- int sampleLength = GetSampleLength();
388
- TestAndVerify(random =>
389
- (int)(1.0 * random.NextUlong() / (1.0 * ulong.MaxValue) * sampleLength)
390
- );
391
- }
392
-
393
- [Test]
394
- [Parallelizable]
395
- public void UlongRange()
396
- {
397
- TestAndVerify(random => (int)random.NextUlong((ulong)_samples.Length));
398
- }
399
-
400
- [Test]
401
- [Parallelizable]
402
- public void UlongMaxRange()
403
- {
404
- TestAndVerify(random =>
405
- (int)(
406
- (random.NextUlong(ulong.MinValue, ulong.MaxValue) + (-1.0 * ulong.MinValue))
407
- / (1.0 * ulong.MaxValue - ulong.MinValue)
408
- * _samples.Length
409
- )
410
- );
411
- }
412
-
413
- [Test]
414
- [Parallelizable]
415
- public void NextBytes()
416
- {
417
- IRandom random = NewRandom();
418
- HashSet<byte> seen = new();
419
- int total = 0;
420
- for (int i = 0; i < NormalIterations; ++i)
421
- {
422
- int size = random.Next(1, 40);
423
- total += size;
424
- byte[] buffer = new byte[size];
425
- random.NextBytes(buffer);
426
- foreach (byte value in buffer)
427
- {
428
- seen.Add(value);
429
- }
430
- }
431
-
432
- double seenTargetMin = Math.Min(total, byte.MaxValue) * 0.9;
433
- double seenTargetMax = Math.Min(total, byte.MaxValue) * 1.1;
434
- Assert.LessOrEqual(seenTargetMin, seen.Count);
435
- Assert.GreaterOrEqual(seenTargetMax, seen.Count);
436
- }
437
-
438
- [Test]
439
- [Parallelizable]
440
- public void NextGaussian()
441
- {
442
- IRandom random = NewRandom();
443
- List<double> values = new();
444
- for (int i = 0; i < NormalIterations; ++i)
445
- {
446
- double value = random.NextGaussian();
447
- values.Add(value);
448
- }
449
-
450
- Assert.IsTrue(CheckApproximateNormality(values));
451
- }
452
-
453
- [Test]
454
- [Parallelizable]
455
- public void NextGuid()
456
- {
457
- IRandom random = NewRandom();
458
-
459
- HashSet<Guid> seen = new();
460
- for (int i = 0; i < NormalIterations; ++i)
461
- {
462
- Guid value = random.NextGuid();
463
- seen.Add(value);
464
- }
465
-
466
- Assert.LessOrEqual(NormalIterations * 0.9, seen.Count);
467
- }
468
-
469
- [Test]
470
- [Parallelizable]
471
- public void NextKGuid()
472
- {
473
- IRandom random = NewRandom();
474
-
475
- HashSet<KGuid> seen = new();
476
- for (int i = 0; i < NormalIterations; ++i)
477
- {
478
- KGuid value = random.NextKGuid();
479
- seen.Add(value);
480
- }
481
-
482
- Assert.LessOrEqual(NormalIterations * 0.9, seen.Count);
483
- }
484
-
485
- [Test]
486
- [Parallelizable]
487
- public void Copy()
488
- {
489
- IRandom random1 = NewRandom();
490
- IRandom random2 = random1.Copy();
491
- Assert.AreEqual(random1.InternalState, random2.InternalState);
492
- // UnityRandom has shared state, the below test is not possible for it. We did all we could.
493
- if (NewRandom() is not UnityRandom)
494
- {
495
- for (int i = 0; i < NumGeneratorChecks; ++i)
496
- {
497
- Assert.AreEqual(random1.Next(), random2.Next());
498
- Assert.AreEqual(random1.InternalState, random2.InternalState);
499
- }
500
- }
501
-
502
- Assert.AreEqual(random1.InternalState, random2.InternalState);
503
- IRandom random3 = random1.Copy();
504
- Assert.AreEqual(random1.InternalState, random3.InternalState);
505
- if (NewRandom() is not UnityRandom)
506
- {
507
- for (int i = 0; i < NumGeneratorChecks; ++i)
508
- {
509
- Assert.AreEqual(random1.Next(), random3.Next());
510
- Assert.AreEqual(random1.InternalState, random3.InternalState);
511
- }
512
- }
513
- }
514
-
515
- [Test]
516
- [Parallelizable]
517
- public void Json()
518
- {
519
- IRandom random = NewRandom();
520
- string json = random.ToJson();
521
- IRandom deserialized = Serializer.JsonDeserialize<IRandom>(json, random.GetType());
522
- Assert.AreEqual(random.InternalState, deserialized.InternalState);
523
-
524
- if (NewRandom() is not UnityRandom)
525
- {
526
- for (int i = 0; i < NumGeneratorChecks; ++i)
527
- {
528
- Assert.AreEqual(random.Next(), deserialized.Next());
529
- Assert.AreEqual(random.InternalState, deserialized.InternalState);
530
- }
531
- }
532
- }
533
-
534
- [Test]
535
- [Parallelizable]
536
- public void NextEnumerable()
537
- {
538
- IRandom random = NewRandom();
539
- for (int i = 0; i < NormalIterations; ++i)
540
- {
541
- HashSet<TestValues> selected = Enum.GetValues(typeof(TestValues))
542
- .OfType<TestValues>()
543
- .Shuffled(random)
544
- .Skip(3)
545
- .ToHashSet();
546
-
547
- TestValues value = random.NextOf(selected.Shuffled(random));
548
- Assert.IsTrue(selected.Contains(value));
549
- }
550
- }
551
-
552
- [Test]
553
- [Parallelizable]
554
- public void NextEnumerableExcept()
555
- {
556
- IRandom random = NewRandom();
557
- for (int i = 0; i < NormalIterations; ++i)
558
- {
559
- TestValues exception = random.NextEnum<TestValues>();
560
- HashSet<TestValues> selected = Enum.GetValues(typeof(TestValues))
561
- .OfType<TestValues>()
562
- .Shuffled(random)
563
- .Skip(3)
564
- .ToHashSet();
565
-
566
- TestValues value = random.NextOfExcept(selected.Shuffled(random), exception);
567
- Assert.IsTrue(selected.Contains(value));
568
- Assert.AreNotEqual(value, exception);
569
- }
570
- }
571
-
572
- [Test]
573
- [Parallelizable]
574
- public void NextArray()
575
- {
576
- IRandom random = NewRandom();
577
- for (int i = 0; i < NormalIterations; ++i)
578
- {
579
- TestValues[] values = Enum.GetValues(typeof(TestValues))
580
- .OfType<TestValues>()
581
- .Shuffled(random)
582
- .Skip(3)
583
- .ToArray();
584
-
585
- TestValues value = random.NextOf(values);
586
- Assert.IsTrue(values.Contains(value));
587
- }
588
- }
589
-
590
- [Test]
591
- [Parallelizable]
592
- public void NextList()
593
- {
594
- IRandom random = NewRandom();
595
- for (int i = 0; i < NormalIterations; ++i)
596
- {
597
- List<TestValues> values = Enum.GetValues(typeof(TestValues))
598
- .OfType<TestValues>()
599
- .Shuffled(random)
600
- .Skip(3)
601
- .ToList();
602
-
603
- TestValues value = random.NextOf(values);
604
- Assert.IsTrue(values.Contains(value));
605
- }
606
- }
607
-
608
- [Test]
609
- [Parallelizable]
610
- public void NextHashSet()
611
- {
612
- IRandom random = NewRandom();
613
- for (int i = 0; i < NormalIterations; ++i)
614
- {
615
- HashSet<TestValues> values = Enum.GetValues(typeof(TestValues))
616
- .OfType<TestValues>()
617
- .Shuffled(random)
618
- .Skip(3)
619
- .ToHashSet();
620
-
621
- TestValues value = random.NextOf(values);
622
- Assert.IsTrue(values.Contains(value));
623
- }
624
- }
625
-
626
- [Test]
627
- [Parallelizable]
628
- public void NextLinkedList()
629
- {
630
- IRandom random = NewRandom();
631
- for (int i = 0; i < NormalIterations; ++i)
632
- {
633
- LinkedList<TestValues> values = Enum.GetValues(typeof(TestValues))
634
- .OfType<TestValues>()
635
- .Shuffled(random)
636
- .Skip(3)
637
- .ToLinkedList();
638
-
639
- TestValues value = random.NextOf(values);
640
- Assert.IsTrue(values.Contains(value));
641
- }
642
- }
643
-
644
- [Test]
645
- [Parallelizable]
646
- public void NextEnum()
647
- {
648
- IRandom random = NewRandom();
649
- HashSet<TestValues> seenEnums = new();
650
- for (int i = 0; i < NormalIterations; ++i)
651
- {
652
- TestValues value = random.NextEnum<TestValues>();
653
- _ = seenEnums.Add(value);
654
- }
655
-
656
- Assert.AreEqual(Enum.GetValues(typeof(TestValues)).Length, seenEnums.Count);
657
- }
658
-
659
- [Test]
660
- [Parallelizable]
661
- public void NextEnumExcept()
662
- {
663
- IRandom random = NewRandom();
664
- HashSet<TestValues> seenEnums = new();
665
- for (int i = 0; i < NormalIterations; ++i)
666
- {
667
- TestValues value = random.NextEnumExcept(TestValues.Value8, TestValues.Value9);
668
- _ = seenEnums.Add(value);
669
- }
670
-
671
- Assert.AreEqual(Enum.GetValues(typeof(TestValues)).Length - 2, seenEnums.Count);
672
- Assert.IsFalse(seenEnums.Contains(TestValues.Value8));
673
- Assert.IsFalse(seenEnums.Contains(TestValues.Value9));
674
- }
675
-
676
- [Test]
677
- [Parallelizable]
678
- public void NextNoiseMap()
679
- {
680
- IRandom random = NewRandom();
681
- for (int i = 0; i < NormalIterations; ++i)
682
- {
683
- int width = random.Next(1, 75);
684
- int height = random.Next(1, 75);
685
- float[,] noise = random.NextNoiseMap(width, height);
686
- Assert.IsNotNull(noise);
687
- Assert.AreEqual(width * height, noise.Length);
688
- foreach (float value in noise)
689
- {
690
- Assert.LessOrEqual(0f, value);
691
- Assert.GreaterOrEqual(1.1f, value);
692
- }
693
- }
694
- }
695
-
696
- protected virtual double GetDeviationFor(string caller)
697
- {
698
- return 0.0625;
699
- }
700
-
701
- private static bool CheckApproximateNormality(IEnumerable<double> data)
702
- {
703
- IReadOnlyList<double> input = data as IReadOnlyList<double> ?? data.ToArray();
704
- int n = input.Count;
705
- if (n < 3)
706
- {
707
- return true;
708
- }
709
-
710
- double mean = input.Average();
711
- double variance = input.Sum(x => Math.Pow(x - mean, 2)) / n;
712
- double stdDev = Math.Sqrt(variance);
713
-
714
- double skewness = input.Sum(x => Math.Pow((x - mean) / stdDev, 3)) / n;
715
- double kurtosis = input.Sum(x => Math.Pow((x - mean) / stdDev, 4)) / n - 3; // Excess kurtosis
716
-
717
- // Thresholds can be defined based on empirical rules or domain-specific values
718
- const double skewnessThreshold = 0.5;
719
- const double kurtosisThreshold = 1.0;
720
-
721
- return Math.Abs(skewness) < skewnessThreshold && Math.Abs(kurtosis) < kurtosisThreshold;
722
- }
723
-
724
- protected int GetSampleLength(int? sampleLength = null)
725
- {
726
- return Math.Min(_samples.Length, sampleLength ?? _samples.Length);
727
- }
728
-
729
- private void TestAndVerify(
730
- Func<IRandom, int> sample,
731
- int? maxLength = null,
732
- [CallerMemberName] string caller = ""
733
- )
734
- {
735
- IRandom random = NewRandom();
736
- int sampleLength = _samples.Length;
737
- for (int i = 0; i < SampleCount; ++i)
738
- {
739
- int index = sample(random);
740
- if (index < 0 || sampleLength <= index)
741
- {
742
- Assert.Fail("Index {0} out of range", index);
743
- }
744
- else
745
- {
746
- _samples[index]++;
747
- }
748
- }
749
-
750
- sampleLength = GetSampleLength(maxLength);
751
- double average = SampleCount * 1.0 / sampleLength;
752
- double deviationAllowed = average * GetDeviationFor(caller);
753
- List<int> zeroCountIndexes = new();
754
- List<int> outsideRange = new();
755
- for (int i = 0; i < sampleLength; i++)
756
- {
757
- int count = _samples[i];
758
- if (count == 0)
759
- {
760
- zeroCountIndexes.Add(i);
761
- }
762
-
763
- if (deviationAllowed < Math.Abs(count - average))
764
- {
765
- outsideRange.Add(i);
766
- }
767
- }
768
-
769
- Assert.AreEqual(
770
- 0,
771
- zeroCountIndexes.Count,
772
- "No samples at {0} indices: [{1}]",
773
- zeroCountIndexes.Count,
774
- string.Join(",", zeroCountIndexes)
775
- );
776
- Assert.AreEqual(
777
- 0,
778
- outsideRange.Count,
779
- "{0} indexes outside of dev {1:0.00}. Expected: {2:0.00}. Found: [{3}]",
780
- outsideRange.Count,
781
- deviationAllowed,
782
- average,
783
- string.Join(",", outsideRange.Select(index => _samples[index]))
784
- );
785
- }
786
- }
787
- }
1
+ namespace UnityHelpers.Tests.Random
2
+ {
3
+ using System;
4
+ using System.Collections.Generic;
5
+ using System.Linq;
6
+ using System.Runtime.CompilerServices;
7
+ using Core.DataStructure.Adapters;
8
+ using Core.Extension;
9
+ using Core.Serialization;
10
+ using NUnit.Framework;
11
+ using UnityHelpers.Core.Random;
12
+
13
+ public enum TestValues
14
+ {
15
+ Value0,
16
+ Value1,
17
+ Value2,
18
+ Value3,
19
+ Value4,
20
+ Value5,
21
+ Value6,
22
+ Value7,
23
+ Value8,
24
+ Value9,
25
+ }
26
+
27
+ public abstract class RandomTestBase
28
+ {
29
+ private const int NumGeneratorChecks = 1_000;
30
+ private const int NormalIterations = 1_000;
31
+ private const int SampleCount = 12_750_000;
32
+
33
+ private readonly int[] _samples = new int[1_000];
34
+
35
+ protected abstract IRandom NewRandom();
36
+
37
+ [SetUp]
38
+ public virtual void Setup()
39
+ {
40
+ Array.Clear(_samples, 0, _samples.Length);
41
+ }
42
+
43
+ [TearDown]
44
+ public virtual void Teardown()
45
+ {
46
+ // No-op in base
47
+ }
48
+
49
+ [Test]
50
+ [Parallelizable]
51
+ public void Bool()
52
+ {
53
+ TestAndVerify(
54
+ random => Math.Min(_samples.Length - 1, Convert.ToInt32(random.NextBool())),
55
+ maxLength: Math.Min(2, _samples.Length)
56
+ );
57
+ }
58
+
59
+ [Test]
60
+ [Parallelizable]
61
+ public void Int()
62
+ {
63
+ TestAndVerify(random => random.Next(0, _samples.Length));
64
+ }
65
+
66
+ [Test]
67
+ [Parallelizable]
68
+ public void IntRaw()
69
+ {
70
+ int sampleLength = GetSampleLength();
71
+ TestAndVerify(random =>
72
+ (int)((1.0 * random.Next()) / (1.0 * int.MaxValue) * sampleLength)
73
+ );
74
+ }
75
+
76
+ [Test]
77
+ [Parallelizable]
78
+ public void IntRange()
79
+ {
80
+ TestAndVerify(random => random.Next(_samples.Length));
81
+ }
82
+
83
+ [Test]
84
+ [Parallelizable]
85
+ public void IntDistribution()
86
+ {
87
+ TestAndVerify(random =>
88
+ (int)(random.Next() / ((1.0 * int.MaxValue) / _samples.Length))
89
+ );
90
+ }
91
+
92
+ [Test]
93
+ [Parallelizable]
94
+ public void IntMaxRange()
95
+ {
96
+ TestAndVerify(random =>
97
+ (int)(
98
+ (random.Next(int.MinValue, int.MaxValue) + (-1.0 * int.MinValue))
99
+ / (1.0 * int.MaxValue - int.MinValue)
100
+ * _samples.Length
101
+ )
102
+ );
103
+ }
104
+
105
+ [Test]
106
+ [Parallelizable]
107
+ public void Uint()
108
+ {
109
+ TestAndVerify(random => (int)random.NextUint(0, (uint)_samples.Length));
110
+ }
111
+
112
+ [Test]
113
+ [Parallelizable]
114
+ public void UintRange()
115
+ {
116
+ TestAndVerify(random => (int)random.NextUint((uint)_samples.Length));
117
+ }
118
+
119
+ [Test]
120
+ [Parallelizable]
121
+ public void UintDistribution()
122
+ {
123
+ TestAndVerify(random =>
124
+ (int)(random.NextUint() / ((1.0 * uint.MaxValue) / _samples.Length))
125
+ );
126
+ }
127
+
128
+ [Test]
129
+ [Parallelizable]
130
+ public void UintMaxRange()
131
+ {
132
+ TestAndVerify(random =>
133
+ (int)(
134
+ (random.NextUint(uint.MinValue, uint.MaxValue) + (1.0 * uint.MinValue))
135
+ / (1.0 * uint.MaxValue)
136
+ * _samples.Length
137
+ )
138
+ );
139
+ }
140
+
141
+ [Test]
142
+ [Parallelizable]
143
+ public void Short()
144
+ {
145
+ TestAndVerify(
146
+ random => random.NextShort(0, (short)_samples.Length),
147
+ maxLength: (short.MaxValue - short.MinValue)
148
+ );
149
+ }
150
+
151
+ [Test]
152
+ [Parallelizable]
153
+ public void ShortRaw()
154
+ {
155
+ int sampleLength = GetSampleLength(short.MaxValue);
156
+ TestAndVerify(
157
+ random => (int)((1.0 * random.NextShort()) / (1.0 * short.MaxValue) * sampleLength),
158
+ maxLength: short.MaxValue
159
+ );
160
+ }
161
+
162
+ [Test]
163
+ [Parallelizable]
164
+ public void ShortRange()
165
+ {
166
+ TestAndVerify(
167
+ random => random.NextShort((short)_samples.Length),
168
+ maxLength: (short.MaxValue - short.MinValue)
169
+ );
170
+ }
171
+
172
+ [Test]
173
+ [Parallelizable]
174
+ public void ShortMaxRange()
175
+ {
176
+ TestAndVerify(random =>
177
+ (int)(
178
+ (random.NextShort(short.MinValue, short.MaxValue) + (-1.0 * short.MinValue))
179
+ / (1.0 * short.MaxValue - short.MinValue)
180
+ * _samples.Length
181
+ )
182
+ );
183
+ }
184
+
185
+ [Test]
186
+ [Parallelizable]
187
+ public void Byte()
188
+ {
189
+ TestAndVerify(
190
+ random =>
191
+ random.NextByte(
192
+ 0,
193
+ (byte)(_samples.Length < byte.MaxValue ? _samples.Length : byte.MaxValue)
194
+ ),
195
+ byte.MaxValue
196
+ );
197
+ }
198
+
199
+ [Test]
200
+ [Parallelizable]
201
+ public void ByteRaw()
202
+ {
203
+ int sampleLength = GetSampleLength(byte.MaxValue);
204
+ TestAndVerify(
205
+ random => (int)((1.0 * random.NextByte()) / (1.0 * byte.MaxValue) * sampleLength),
206
+ maxLength: byte.MaxValue
207
+ );
208
+ }
209
+
210
+ [Test]
211
+ [Parallelizable]
212
+ public void ByteRange()
213
+ {
214
+ TestAndVerify(
215
+ random =>
216
+ random.NextByte(
217
+ (byte)(_samples.Length < byte.MaxValue ? _samples.Length : byte.MaxValue)
218
+ ),
219
+ byte.MaxValue
220
+ );
221
+ }
222
+
223
+ [Test]
224
+ [Parallelizable]
225
+ public void ByteMaxRange()
226
+ {
227
+ int sampleCount = Math.Min((byte.MaxValue - byte.MinValue), _samples.Length);
228
+ TestAndVerify(
229
+ random =>
230
+ Math.Clamp(
231
+ (int)(
232
+ (random.NextByte(byte.MinValue, byte.MaxValue) + (-1.0 * byte.MinValue))
233
+ / (1.0 * byte.MaxValue - byte.MinValue)
234
+ * sampleCount
235
+ ),
236
+ 0,
237
+ sampleCount - 1
238
+ ),
239
+ maxLength: sampleCount
240
+ );
241
+ }
242
+
243
+ [Test]
244
+ [Parallelizable]
245
+ public void Float()
246
+ {
247
+ TestAndVerify(random =>
248
+ Math.Clamp(
249
+ (int)Math.Floor(random.NextFloat(0, _samples.Length)),
250
+ 0,
251
+ _samples.Length - 1
252
+ )
253
+ );
254
+ }
255
+
256
+ [Test]
257
+ [Parallelizable]
258
+ public void FloatRange()
259
+ {
260
+ TestAndVerify(random =>
261
+ Math.Clamp(
262
+ (int)Math.Floor(random.NextFloat(_samples.Length)),
263
+ 0,
264
+ _samples.Length - 1
265
+ )
266
+ );
267
+ }
268
+
269
+ [Test]
270
+ [Parallelizable]
271
+ public void FloatDistribution()
272
+ {
273
+ TestAndVerify(random => (int)(random.NextFloat() * _samples.Length));
274
+ }
275
+
276
+ [Test]
277
+ [Parallelizable]
278
+ public void FloatMaxRange()
279
+ {
280
+ TestAndVerify(random =>
281
+ Math.Clamp(
282
+ (int)(
283
+ (random.NextFloat(float.MinValue, float.MaxValue) + (-1.0 * float.MinValue))
284
+ / (1.0 * float.MaxValue - float.MinValue)
285
+ * _samples.Length
286
+ ),
287
+ 0,
288
+ _samples.Length - 1
289
+ )
290
+ );
291
+ }
292
+
293
+ [Test]
294
+ [Parallelizable]
295
+ public void Double()
296
+ {
297
+ TestAndVerify(random =>
298
+ Math.Clamp(
299
+ (int)Math.Floor(random.NextDouble(0, _samples.Length)),
300
+ 0,
301
+ _samples.Length - 1
302
+ )
303
+ );
304
+ }
305
+
306
+ [Test]
307
+ [Parallelizable]
308
+ public void DoubleRange()
309
+ {
310
+ TestAndVerify(random =>
311
+ Math.Clamp(
312
+ (int)Math.Floor(random.NextDouble(_samples.Length)),
313
+ 0,
314
+ _samples.Length - 1
315
+ )
316
+ );
317
+ }
318
+
319
+ [Test]
320
+ [Parallelizable]
321
+ public void DoubleDistribution()
322
+ {
323
+ TestAndVerify(random => (int)(random.NextDouble() * _samples.Length));
324
+ }
325
+
326
+ [Test]
327
+ [Parallelizable]
328
+ public void DoubleMaxRange()
329
+ {
330
+ IRandom random = NewRandom();
331
+ for (int i = 0; i < SampleCount; ++i)
332
+ {
333
+ double value = random.NextDouble(double.MinValue, double.MaxValue);
334
+ Assert.IsFalse(double.IsNaN(value));
335
+ Assert.IsFalse(double.IsInfinity(value));
336
+ }
337
+ }
338
+
339
+ [Test]
340
+ [Parallelizable]
341
+ public void Long()
342
+ {
343
+ TestAndVerify(random => (int)random.NextLong(0, _samples.Length));
344
+ }
345
+
346
+ [Test]
347
+ [Parallelizable]
348
+ public void LongRaw()
349
+ {
350
+ int sampleLength = GetSampleLength();
351
+ TestAndVerify(random =>
352
+ (int)(1.0 * random.NextLong() / (1.0 * long.MaxValue) * sampleLength)
353
+ );
354
+ }
355
+
356
+ [Test]
357
+ [Parallelizable]
358
+ public void LongRange()
359
+ {
360
+ TestAndVerify(random => (int)random.NextLong(_samples.Length));
361
+ }
362
+
363
+ [Test]
364
+ [Parallelizable]
365
+ public void LongMaxRange()
366
+ {
367
+ TestAndVerify(random =>
368
+ (int)(
369
+ (random.NextLong(long.MinValue, long.MaxValue) + (-1.0 * long.MinValue))
370
+ / (1.0 * long.MaxValue - long.MinValue)
371
+ * _samples.Length
372
+ )
373
+ );
374
+ }
375
+
376
+ [Test]
377
+ [Parallelizable]
378
+ public void Ulong()
379
+ {
380
+ TestAndVerify(random => (int)random.NextUlong(0, (ulong)_samples.Length));
381
+ }
382
+
383
+ [Test]
384
+ [Parallelizable]
385
+ public void UlongRaw()
386
+ {
387
+ int sampleLength = GetSampleLength();
388
+ TestAndVerify(random =>
389
+ (int)(1.0 * random.NextUlong() / (1.0 * ulong.MaxValue) * sampleLength)
390
+ );
391
+ }
392
+
393
+ [Test]
394
+ [Parallelizable]
395
+ public void UlongRange()
396
+ {
397
+ TestAndVerify(random => (int)random.NextUlong((ulong)_samples.Length));
398
+ }
399
+
400
+ [Test]
401
+ [Parallelizable]
402
+ public void UlongMaxRange()
403
+ {
404
+ TestAndVerify(random =>
405
+ (int)(
406
+ (random.NextUlong(ulong.MinValue, ulong.MaxValue) + (-1.0 * ulong.MinValue))
407
+ / (1.0 * ulong.MaxValue - ulong.MinValue)
408
+ * _samples.Length
409
+ )
410
+ );
411
+ }
412
+
413
+ [Test]
414
+ [Parallelizable]
415
+ public void NextBytes()
416
+ {
417
+ IRandom random = NewRandom();
418
+ HashSet<byte> seen = new();
419
+ int total = 0;
420
+ for (int i = 0; i < NormalIterations; ++i)
421
+ {
422
+ int size = random.Next(1, 40);
423
+ total += size;
424
+ byte[] buffer = new byte[size];
425
+ random.NextBytes(buffer);
426
+ foreach (byte value in buffer)
427
+ {
428
+ seen.Add(value);
429
+ }
430
+ }
431
+
432
+ double seenTargetMin = Math.Min(total, byte.MaxValue) * 0.9;
433
+ double seenTargetMax = Math.Min(total, byte.MaxValue) * 1.1;
434
+ Assert.LessOrEqual(seenTargetMin, seen.Count);
435
+ Assert.GreaterOrEqual(seenTargetMax, seen.Count);
436
+ }
437
+
438
+ [Test]
439
+ [Parallelizable]
440
+ public void NextGaussian()
441
+ {
442
+ IRandom random = NewRandom();
443
+ List<double> values = new();
444
+ for (int i = 0; i < NormalIterations; ++i)
445
+ {
446
+ double value = random.NextGaussian();
447
+ values.Add(value);
448
+ }
449
+
450
+ Assert.IsTrue(CheckApproximateNormality(values));
451
+ }
452
+
453
+ [Test]
454
+ [Parallelizable]
455
+ public void NextGuid()
456
+ {
457
+ IRandom random = NewRandom();
458
+
459
+ HashSet<Guid> seen = new();
460
+ for (int i = 0; i < NormalIterations; ++i)
461
+ {
462
+ Guid value = random.NextGuid();
463
+ seen.Add(value);
464
+ }
465
+
466
+ Assert.LessOrEqual(NormalIterations * 0.9, seen.Count);
467
+ }
468
+
469
+ [Test]
470
+ [Parallelizable]
471
+ public void NextKGuid()
472
+ {
473
+ IRandom random = NewRandom();
474
+
475
+ HashSet<KGuid> seen = new();
476
+ for (int i = 0; i < NormalIterations; ++i)
477
+ {
478
+ KGuid value = random.NextKGuid();
479
+ seen.Add(value);
480
+ }
481
+
482
+ Assert.LessOrEqual(NormalIterations * 0.9, seen.Count);
483
+ }
484
+
485
+ [Test]
486
+ [Parallelizable]
487
+ public void Copy()
488
+ {
489
+ IRandom random1 = NewRandom();
490
+ IRandom random2 = random1.Copy();
491
+ Assert.AreEqual(random1.InternalState, random2.InternalState);
492
+ // UnityRandom has shared state, the below test is not possible for it. We did all we could.
493
+ if (NewRandom() is not UnityRandom)
494
+ {
495
+ for (int i = 0; i < NumGeneratorChecks; ++i)
496
+ {
497
+ Assert.AreEqual(random1.Next(), random2.Next());
498
+ Assert.AreEqual(random1.InternalState, random2.InternalState);
499
+ }
500
+ }
501
+
502
+ Assert.AreEqual(random1.InternalState, random2.InternalState);
503
+ IRandom random3 = random1.Copy();
504
+ Assert.AreEqual(random1.InternalState, random3.InternalState);
505
+ if (NewRandom() is not UnityRandom)
506
+ {
507
+ for (int i = 0; i < NumGeneratorChecks; ++i)
508
+ {
509
+ Assert.AreEqual(random1.Next(), random3.Next());
510
+ Assert.AreEqual(random1.InternalState, random3.InternalState);
511
+ }
512
+ }
513
+ }
514
+
515
+ [Test]
516
+ [Parallelizable]
517
+ public void Json()
518
+ {
519
+ IRandom random = NewRandom();
520
+ string json = random.ToJson();
521
+ IRandom deserialized = Serializer.JsonDeserialize<IRandom>(json, random.GetType());
522
+ Assert.AreEqual(random.InternalState, deserialized.InternalState);
523
+
524
+ if (NewRandom() is not UnityRandom)
525
+ {
526
+ for (int i = 0; i < NumGeneratorChecks; ++i)
527
+ {
528
+ Assert.AreEqual(random.Next(), deserialized.Next());
529
+ Assert.AreEqual(random.InternalState, deserialized.InternalState);
530
+ }
531
+ }
532
+ }
533
+
534
+ [Test]
535
+ [Parallelizable]
536
+ public void NextEnumerable()
537
+ {
538
+ IRandom random = NewRandom();
539
+ for (int i = 0; i < NormalIterations; ++i)
540
+ {
541
+ HashSet<TestValues> selected = Enum.GetValues(typeof(TestValues))
542
+ .OfType<TestValues>()
543
+ .Shuffled(random)
544
+ .Skip(3)
545
+ .ToHashSet();
546
+
547
+ TestValues value = random.NextOf(selected.Shuffled(random));
548
+ Assert.IsTrue(selected.Contains(value));
549
+ }
550
+ }
551
+
552
+ [Test]
553
+ [Parallelizable]
554
+ public void NextEnumerableExcept()
555
+ {
556
+ IRandom random = NewRandom();
557
+ for (int i = 0; i < NormalIterations; ++i)
558
+ {
559
+ TestValues exception = random.NextEnum<TestValues>();
560
+ HashSet<TestValues> selected = Enum.GetValues(typeof(TestValues))
561
+ .OfType<TestValues>()
562
+ .Shuffled(random)
563
+ .Skip(3)
564
+ .ToHashSet();
565
+
566
+ TestValues value = random.NextOfExcept(selected.Shuffled(random), exception);
567
+ Assert.IsTrue(selected.Contains(value));
568
+ Assert.AreNotEqual(value, exception);
569
+ }
570
+ }
571
+
572
+ [Test]
573
+ [Parallelizable]
574
+ public void NextArray()
575
+ {
576
+ IRandom random = NewRandom();
577
+ for (int i = 0; i < NormalIterations; ++i)
578
+ {
579
+ TestValues[] values = Enum.GetValues(typeof(TestValues))
580
+ .OfType<TestValues>()
581
+ .Shuffled(random)
582
+ .Skip(3)
583
+ .ToArray();
584
+
585
+ TestValues value = random.NextOf(values);
586
+ Assert.IsTrue(values.Contains(value));
587
+ }
588
+ }
589
+
590
+ [Test]
591
+ [Parallelizable]
592
+ public void NextList()
593
+ {
594
+ IRandom random = NewRandom();
595
+ for (int i = 0; i < NormalIterations; ++i)
596
+ {
597
+ List<TestValues> values = Enum.GetValues(typeof(TestValues))
598
+ .OfType<TestValues>()
599
+ .Shuffled(random)
600
+ .Skip(3)
601
+ .ToList();
602
+
603
+ TestValues value = random.NextOf(values);
604
+ Assert.IsTrue(values.Contains(value));
605
+ }
606
+ }
607
+
608
+ [Test]
609
+ [Parallelizable]
610
+ public void NextHashSet()
611
+ {
612
+ IRandom random = NewRandom();
613
+ for (int i = 0; i < NormalIterations; ++i)
614
+ {
615
+ HashSet<TestValues> values = Enum.GetValues(typeof(TestValues))
616
+ .OfType<TestValues>()
617
+ .Shuffled(random)
618
+ .Skip(3)
619
+ .ToHashSet();
620
+
621
+ TestValues value = random.NextOf(values);
622
+ Assert.IsTrue(values.Contains(value));
623
+ }
624
+ }
625
+
626
+ [Test]
627
+ [Parallelizable]
628
+ public void NextLinkedList()
629
+ {
630
+ IRandom random = NewRandom();
631
+ for (int i = 0; i < NormalIterations; ++i)
632
+ {
633
+ LinkedList<TestValues> values = Enum.GetValues(typeof(TestValues))
634
+ .OfType<TestValues>()
635
+ .Shuffled(random)
636
+ .Skip(3)
637
+ .ToLinkedList();
638
+
639
+ TestValues value = random.NextOf(values);
640
+ Assert.IsTrue(values.Contains(value));
641
+ }
642
+ }
643
+
644
+ [Test]
645
+ [Parallelizable]
646
+ public void NextEnum()
647
+ {
648
+ IRandom random = NewRandom();
649
+ HashSet<TestValues> seenEnums = new();
650
+ for (int i = 0; i < NormalIterations; ++i)
651
+ {
652
+ TestValues value = random.NextEnum<TestValues>();
653
+ _ = seenEnums.Add(value);
654
+ }
655
+
656
+ Assert.AreEqual(Enum.GetValues(typeof(TestValues)).Length, seenEnums.Count);
657
+ }
658
+
659
+ [Test]
660
+ [Parallelizable]
661
+ public void NextEnumExcept()
662
+ {
663
+ IRandom random = NewRandom();
664
+ HashSet<TestValues> seenEnums = new();
665
+ for (int i = 0; i < NormalIterations; ++i)
666
+ {
667
+ TestValues value = random.NextEnumExcept(TestValues.Value8, TestValues.Value9);
668
+ _ = seenEnums.Add(value);
669
+ }
670
+
671
+ Assert.AreEqual(Enum.GetValues(typeof(TestValues)).Length - 2, seenEnums.Count);
672
+ Assert.IsFalse(seenEnums.Contains(TestValues.Value8));
673
+ Assert.IsFalse(seenEnums.Contains(TestValues.Value9));
674
+ }
675
+
676
+ [Test]
677
+ [Parallelizable]
678
+ public void NextNoiseMap()
679
+ {
680
+ IRandom random = NewRandom();
681
+ for (int i = 0; i < NormalIterations; ++i)
682
+ {
683
+ int width = random.Next(1, 75);
684
+ int height = random.Next(1, 75);
685
+ float[,] noise = random.NextNoiseMap(width, height);
686
+ Assert.IsNotNull(noise);
687
+ Assert.AreEqual(width * height, noise.Length);
688
+ foreach (float value in noise)
689
+ {
690
+ Assert.LessOrEqual(0f, value);
691
+ Assert.GreaterOrEqual(1.1f, value);
692
+ }
693
+ }
694
+ }
695
+
696
+ protected virtual double GetDeviationFor(string caller)
697
+ {
698
+ return 0.0625;
699
+ }
700
+
701
+ private static bool CheckApproximateNormality(IEnumerable<double> data)
702
+ {
703
+ IReadOnlyList<double> input = data as IReadOnlyList<double> ?? data.ToArray();
704
+ int n = input.Count;
705
+ if (n < 3)
706
+ {
707
+ return true;
708
+ }
709
+
710
+ double mean = input.Average();
711
+ double variance = input.Sum(x => Math.Pow(x - mean, 2)) / n;
712
+ double stdDev = Math.Sqrt(variance);
713
+
714
+ double skewness = input.Sum(x => Math.Pow((x - mean) / stdDev, 3)) / n;
715
+ double kurtosis = input.Sum(x => Math.Pow((x - mean) / stdDev, 4)) / n - 3; // Excess kurtosis
716
+
717
+ // Thresholds can be defined based on empirical rules or domain-specific values
718
+ const double skewnessThreshold = 0.5;
719
+ const double kurtosisThreshold = 1.0;
720
+
721
+ return Math.Abs(skewness) < skewnessThreshold && Math.Abs(kurtosis) < kurtosisThreshold;
722
+ }
723
+
724
+ protected int GetSampleLength(int? sampleLength = null)
725
+ {
726
+ return Math.Min(_samples.Length, sampleLength ?? _samples.Length);
727
+ }
728
+
729
+ private void TestAndVerify(
730
+ Func<IRandom, int> sample,
731
+ int? maxLength = null,
732
+ [CallerMemberName] string caller = ""
733
+ )
734
+ {
735
+ IRandom random = NewRandom();
736
+ int sampleLength = _samples.Length;
737
+ for (int i = 0; i < SampleCount; ++i)
738
+ {
739
+ int index = sample(random);
740
+ if (index < 0 || sampleLength <= index)
741
+ {
742
+ Assert.Fail("Index {0} out of range", index);
743
+ }
744
+ else
745
+ {
746
+ _samples[index]++;
747
+ }
748
+ }
749
+
750
+ sampleLength = GetSampleLength(maxLength);
751
+ double average = SampleCount * 1.0 / sampleLength;
752
+ double deviationAllowed = average * GetDeviationFor(caller);
753
+ List<int> zeroCountIndexes = new();
754
+ List<int> outsideRange = new();
755
+ for (int i = 0; i < sampleLength; i++)
756
+ {
757
+ int count = _samples[i];
758
+ if (count == 0)
759
+ {
760
+ zeroCountIndexes.Add(i);
761
+ }
762
+
763
+ if (deviationAllowed < Math.Abs(count - average))
764
+ {
765
+ outsideRange.Add(i);
766
+ }
767
+ }
768
+
769
+ Assert.AreEqual(
770
+ 0,
771
+ zeroCountIndexes.Count,
772
+ "No samples at {0} indices: [{1}]",
773
+ zeroCountIndexes.Count,
774
+ string.Join(",", zeroCountIndexes)
775
+ );
776
+ Assert.AreEqual(
777
+ 0,
778
+ outsideRange.Count,
779
+ "{0} indexes outside of dev {1:0.00}. Expected: {2:0.00}. Found: [{3}]",
780
+ outsideRange.Count,
781
+ deviationAllowed,
782
+ average,
783
+ string.Join(",", outsideRange.Select(index => _samples[index]))
784
+ );
785
+ }
786
+ }
787
+ }