< Summary

Class:ConstrainedKMeansClustererTest
Assembly:bamlab.test.editmode
File(s):/github/workspace/Assets/Tests/EditMode/KMeansClustererTest.cs
Covered lines:35
Uncovered lines:0
Coverable lines:35
Total lines:150
Line coverage:100% (35 of 35)
Covered branches:0
Total branches:0
Covered methods:6
Total methods:6
Method coverage:100% (6 of 6)

Metrics

MethodBranch coverage Crap Score Cyclomatic complexity NPath complexity Sequence coverage
ConstrainedKMeansClustererTest()0%110100%
GenerateObject(...)0%110100%
TestSingleCluster()0%110100%
TestMaxSizeOne()0%330100%
TestZeroRadius()0%330100%
TestSmallRadius()0%110100%

File(s)

/github/workspace/Assets/Tests/EditMode/KMeansClustererTest.cs

#LineLine coverage
 1using NUnit.Framework;
 2using System.Collections;
 3using System.Collections.Generic;
 4using System.Linq;
 5using UnityEngine;
 6using UnityEngine.TestTools;
 7
 8public class KMeansClustererTest {
 9  public static readonly List<GameObject> Objects = new List<GameObject> {
 10    GenerateObject(new Vector3(0, 0, 0)),
 11    GenerateObject(new Vector3(0, 1, 0)),
 12    GenerateObject(new Vector3(0, 1.5f, 0)),
 13    GenerateObject(new Vector3(0, 2.5f, 0)),
 14  };
 15
 16  public static GameObject GenerateObject(in Vector3 position) {
 17    GameObject obj = new GameObject();
 18    obj.transform.position = position;
 19    return obj;
 20  }
 21
 22  [Test]
 23  public void TestSingleCluster() {
 24    KMeansClusterer clusterer = new KMeansClusterer(Objects, k: 1);
 25    clusterer.Cluster();
 26    Cluster cluster = clusterer.Clusters[0];
 27    Assert.AreEqual(Objects.Count, cluster.Size());
 28    Assert.AreEqual(new Vector3(0, 1.25f, 0), cluster.Coordinates);
 29    Assert.AreEqual(new Vector3(0, 1.25f, 0), cluster.Centroid());
 30  }
 31
 32  // Test to reveal improper clearing of cluster memberships.
 33  [Test]
 34  public void TestTwoDistinctClustersWithResetNeeded() {
 35    // Group A: points near (0, 0, 0).
 36    var groupA = new List<GameObject> {
 37      GenerateObject(new Vector3(0, 0, 0)),
 38      GenerateObject(new Vector3(1, 0, 0)),
 39      GenerateObject(new Vector3(0, 1, 0)),
 40      GenerateObject(new Vector3(1, 1, 0)),
 41    };
 42
 43    // Group B: points near (10, 10, 10).
 44    var groupB = new List<GameObject> {
 45      GenerateObject(new Vector3(10, 10, 10)),
 46      GenerateObject(new Vector3(11, 10, 10)),
 47      GenerateObject(new Vector3(10, 11, 10)),
 48      GenerateObject(new Vector3(11, 11, 10)),
 49    };
 50
 51    // Combine them.
 52    var objects = new List<GameObject>();
 53    objects.AddRange(groupA);
 54    objects.AddRange(groupB);
 55
 56    // Create clusterer with k = 2.
 57    KMeansClusterer clusterer = new KMeansClusterer(objects, k: 2);
 58    clusterer.Cluster();
 59
 60    // We expect exactly 2 clusters.
 61    Assert.AreEqual(2, clusterer.Clusters.Count);
 62
 63    // Retrieve the clusters.
 64    Cluster c0 = clusterer.Clusters[0];
 65    Cluster c1 = clusterer.Clusters[1];
 66
 67    // Because the clusters are well-separated, each cluster should contain all points from one
 68    // group or the other, not a mixture. Check via centroids.
 69    var centroid0 = c0.Centroid();
 70    var centroid1 = c1.Centroid();
 71
 72    // One centroid should be near (0.5, 0.5, 0), the other near (10.5, 10.5, 10).
 73    var expectedCentroid0 = new Vector3(0.5f, 0.5f, 0);
 74    var expectedCentroid1 = new Vector3(10.5f, 10.5f, 10);
 75    bool correctPlacement = (centroid0 == expectedCentroid0 && centroid1 == expectedCentroid1) ||
 76                            (centroid0 == expectedCentroid1 && centroid1 == expectedCentroid0);
 77    Assert.IsTrue(
 78        correctPlacement,
 79        "Centroids not close to the expected group centers. Possible leftover membership from a previous iteration if cl
 80
 81    // Additionally, we can count membership to confirm that each cluster got exactly four points
 82    // for a more direct check.
 83    int cluster0Count = c0.Size();
 84    int cluster1Count = c1.Size();
 85    Assert.AreEqual(8, cluster0Count + cluster1Count,
 86                    "Total membership across clusters does not match the total number of objects.");
 87
 88    // Even if the clusters swapped roles, each cluster should have 4 points if membership was
 89    // properly reset and re-assigned.
 90    bool clusterCountsValid = cluster0Count == 4 && cluster1Count == 4;
 91    Assert.IsTrue(clusterCountsValid,
 92                  $"Cluster sizes not as expected. c0={cluster0Count}, c1={cluster1Count}.");
 93  }
 94}
 95
 96public class ConstrainedKMeansClustererTest {
 197  public static readonly List<GameObject> Objects = new List<GameObject> {
 98    GenerateObject(new Vector3(0, 0, 0)),
 99    GenerateObject(new Vector3(0, 1, 0)),
 100    GenerateObject(new Vector3(0, 1.5f, 0)),
 101    GenerateObject(new Vector3(0, 2.5f, 0)),
 102  };
 103
 4104  public static GameObject GenerateObject(in Vector3 position) {
 4105    GameObject obj = new GameObject();
 4106    obj.transform.position = position;
 4107    return obj;
 4108  }
 109
 110  [Test]
 1111  public void TestSingleCluster() {
 1112    ConstrainedKMeansClusterer clusterer =
 113        new ConstrainedKMeansClusterer(Objects, maxSize: Objects.Count, maxRadius: Mathf.Infinity);
 1114    clusterer.Cluster();
 1115    Assert.AreEqual(1, clusterer.Clusters.Count);
 1116    Cluster cluster = clusterer.Clusters[0];
 1117    Assert.AreEqual(Objects.Count, cluster.Size());
 1118    Assert.AreEqual(new Vector3(0, 1.25f, 0), cluster.Centroid());
 1119  }
 120
 121  [Test]
 1122  public void TestMaxSizeOne() {
 1123    ConstrainedKMeansClusterer clusterer =
 124        new ConstrainedKMeansClusterer(Objects, maxSize: 1, maxRadius: Mathf.Infinity);
 1125    clusterer.Cluster();
 1126    Assert.AreEqual(Objects.Count, clusterer.Clusters.Count);
 15127    foreach (var cluster in clusterer.Clusters) {
 4128      Assert.AreEqual(1, cluster.Size());
 4129    }
 1130  }
 131
 132  [Test]
 1133  public void TestZeroRadius() {
 1134    ConstrainedKMeansClusterer clusterer =
 135        new ConstrainedKMeansClusterer(Objects, maxSize: Objects.Count, maxRadius: 0);
 1136    clusterer.Cluster();
 1137    Assert.AreEqual(Objects.Count, clusterer.Clusters.Count);
 15138    foreach (var cluster in clusterer.Clusters) {
 4139      Assert.AreEqual(1, cluster.Size());
 4140    }
 1141  }
 142
 143  [Test]
 1144  public void TestSmallRadius() {
 1145    ConstrainedKMeansClusterer clusterer =
 146        new ConstrainedKMeansClusterer(Objects, maxSize: Objects.Count, maxRadius: 1);
 1147    clusterer.Cluster();
 1148    Assert.AreEqual(2, clusterer.Clusters.Count);
 1149  }
 150}