< Summary

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

Metrics

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

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 {
 097  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
 0104  public static GameObject GenerateObject(in Vector3 position) {
 0105    GameObject obj = new GameObject();
 0106    obj.transform.position = position;
 0107    return obj;
 0108  }
 109
 110  [Test]
 0111  public void TestSingleCluster() {
 0112    ConstrainedKMeansClusterer clusterer =
 113        new ConstrainedKMeansClusterer(Objects, maxSize: Objects.Count, maxRadius: Mathf.Infinity);
 0114    clusterer.Cluster();
 0115    Assert.AreEqual(1, clusterer.Clusters.Count);
 0116    Cluster cluster = clusterer.Clusters[0];
 0117    Assert.AreEqual(Objects.Count, cluster.Size());
 0118    Assert.AreEqual(new Vector3(0, 1.25f, 0), cluster.Centroid());
 0119  }
 120
 121  [Test]
 0122  public void TestMaxSizeOne() {
 0123    ConstrainedKMeansClusterer clusterer =
 124        new ConstrainedKMeansClusterer(Objects, maxSize: 1, maxRadius: Mathf.Infinity);
 0125    clusterer.Cluster();
 0126    Assert.AreEqual(Objects.Count, clusterer.Clusters.Count);
 0127    foreach (var cluster in clusterer.Clusters) {
 0128      Assert.AreEqual(1, cluster.Size());
 0129    }
 0130  }
 131
 132  [Test]
 0133  public void TestZeroRadius() {
 0134    ConstrainedKMeansClusterer clusterer =
 135        new ConstrainedKMeansClusterer(Objects, maxSize: Objects.Count, maxRadius: 0);
 0136    clusterer.Cluster();
 0137    Assert.AreEqual(Objects.Count, clusterer.Clusters.Count);
 0138    foreach (var cluster in clusterer.Clusters) {
 0139      Assert.AreEqual(1, cluster.Size());
 0140    }
 0141  }
 142
 143  [Test]
 0144  public void TestSmallRadius() {
 0145    ConstrainedKMeansClusterer clusterer =
 146        new ConstrainedKMeansClusterer(Objects, maxSize: Objects.Count, maxRadius: 1);
 0147    clusterer.Cluster();
 0148    Assert.AreEqual(2, clusterer.Clusters.Count);
 0149  }
 150}