< Summary

Class:ThreatAssignment
Assembly:bamlab.micromissiles
File(s):/github/workspace/Assets/Scripts/Assignment/ThreatAssignment.cs
Covered lines:0
Uncovered lines:38
Coverable lines:38
Total lines:73
Line coverage:0% (0 of 38)
Covered branches:0
Total branches:0
Covered methods:0
Total methods:5
Method coverage:0% (0 of 5)

Metrics

MethodBranch coverage Crap Score Cyclomatic complexity NPath complexity Sequence coverage
Assign(...)0%42600%
CalculateThreatLevels(...)0%12300%
ThreatInfo(...)0%2100%

File(s)

/github/workspace/Assets/Scripts/Assignment/ThreatAssignment.cs

#LineLine coverage
 1using System;
 2using System.Collections.Generic;
 3using System.Diagnostics.Contracts;
 4using System.Linq;
 5using UnityEngine;
 6
 7// The threat assignment class assigns interceptors to the targets based
 8// on the threat level of the targets.
 9public class ThreatAssignment : IAssignment {
 10  // Assign a target to each interceptor that has not been assigned a target yet.
 11  [Pure]
 12  public IEnumerable<IAssignment.AssignmentItem> Assign(in IReadOnlyList<Interceptor> interceptors,
 013                                                        in IReadOnlyList<Threat> threats) {
 014    List<IAssignment.AssignmentItem> assignments = new List<IAssignment.AssignmentItem>();
 15
 016    List<Interceptor> assignableInterceptors = IAssignment.GetAssignableInterceptors(interceptors);
 017    if (assignableInterceptors.Count == 0) {
 018      Debug.LogWarning("No assignable interceptors found.");
 019      return assignments;
 20    }
 21
 022    List<Threat> activeThreats = IAssignment.GetActiveThreats(threats);
 023    if (activeThreats.Count == 0) {
 024      Debug.LogWarning("No active threats found.");
 025      return assignments;
 26    }
 27
 028    Vector3 positionToDefend = Vector3.zero;
 029    List<ThreatInfo> threatInfos = CalculateThreatLevels(activeThreats, positionToDefend);
 30
 31    // Sort the threats first by whether an interceptor is assigned to them already and then by
 32    // their threat level in descending order.
 033    threatInfos = threatInfos.OrderBy(threat => threat.Threat.AssignedInterceptors.Count)
 034                      .ThenByDescending(threat => threat.ThreatLevel)
 35                      .ToList();
 36
 037    var assignableInterceptorsEnumerator = assignableInterceptors.GetEnumerator();
 038    int threatIndex = 0;
 039    while (assignableInterceptorsEnumerator.MoveNext()) {
 040      assignments.Add(new IAssignment.AssignmentItem(assignableInterceptorsEnumerator.Current,
 41                                                     threatInfos[threatIndex].Threat));
 042      threatIndex = (threatIndex + 1) % threatInfos.Count;
 043    }
 044    return assignments;
 045  }
 46
 047  private List<ThreatInfo> CalculateThreatLevels(List<Threat> threats, Vector3 defensePosition) {
 048    List<ThreatInfo> threatInfos = new List<ThreatInfo>();
 49
 050    foreach (var threat in threats) {
 051      float distanceToMean = Vector3.Distance(threat.transform.position, defensePosition);
 052      float velocityMagnitude = threat.GetVelocity().magnitude;
 53
 54      // Calculate the threat level based on proximity and velocity.
 055      float threatLevel = (1 / distanceToMean) * velocityMagnitude;
 56
 057      threatInfos.Add(new ThreatInfo(threat, threatLevel));
 058    }
 59
 60    // Sort threats in descending order.
 061    return threatInfos.OrderByDescending(threat => threat.ThreatLevel).ToList();
 062  }
 63
 64  private class ThreatInfo {
 065    public Threat Threat { get; }
 066    public float ThreatLevel { get; }
 67
 068    public ThreatInfo(Threat threat, float threatLevel) {
 069      Threat = threat;
 070      ThreatLevel = threatLevel;
 071    }
 72  }
 73}