< Summary

Class:Agent
Assembly:bamlab.micromissiles
File(s):/github/workspace/Assets/Scripts/Agent.cs
Covered lines:104
Uncovered lines:156
Coverable lines:260
Total lines:465
Line coverage:40% (104 of 260)
Covered branches:0
Total branches:0
Covered methods:21
Total methods:46
Method coverage:45.6% (21 of 46)

Metrics

MethodBranch coverage Crap Score Cyclomatic complexity NPath complexity Sequence coverage
Agent()0%110100%
SetFlightPhase(...)0%20400%
GetFlightPhase()0%2100%
HasLaunched()0%2100%
HasTerminated()0%2100%
SetDynamicAgentConfig(...)0%110100%
SetStaticAgentConfig(...)0%110100%
IsAssignable()0%2100%
AssignTarget(...)0%2100%
GetAssignedTarget()0%2100%
HasAssignedTarget()0%110100%
GetTargetModel()0%2100%
UnassignTarget()0%6200%
IsTerminated()0%110100%
AddInterceptor(...)0%2100%
RemoveInterceptor(...)0%2100%
TerminateAgent()0%12300%
HandleInterceptHit(...)0%20400%
HandleInterceptMiss()0%20400%
HandleTargetIntercepted()0%6200%
HandleThreatHit()0%12300%
HandleHitGround()0%42600%
SetInitialVelocity(...)0%110100%
GetPosition()0%110100%
SetPosition(...)0%110100%
GetSpeed()0%110100%
GetVelocity()0%110100%
SetVelocity(...)0%110100%
GetAcceleration()0%110100%
SetAcceleration(...)0%2100%
GetRelativeTransformation(...)0%110100%
GetRelativeTransformationToWaypoint(...)0%110100%
GetRelativePositionTransformation(...)0%220100%
GetRelativeVelocityTransformation(...)0%550100%
GetRelativeAccelerationTransformation(...)0%110100%
GetDynamicPressure()0%2100%
Awake()0%2100%
Start()0%110100%
FixedUpdate()0%1821300%
AlignWithVelocity()0%6200%
CalculateAcceleration(...)0%2100%
CalculateMaxForwardAcceleration()0%110100%
CalculateMaxNormalAcceleration()0%110100%
CalculateDrag()0%2100%
CalculateLiftInducedDrag(...)0%2100%

File(s)

/github/workspace/Assets/Scripts/Agent.cs

#LineLine coverage
 1using System.Collections;
 2using System.Collections.Generic;
 3using UnityEngine;
 4
 5public abstract class Agent : MonoBehaviour {
 6  // In the initialized phase, the agent is subject to no forces.
 7  // In the ready phase, the agent is subject to gravity and drag with zero input acceleration.
 8  public enum FlightPhase { INITIALIZED, READY, BOOST, MIDCOURSE, TERMINAL, TERMINATED }
 9
 10  [SerializeField]
 6411  private FlightPhase _flightPhase = FlightPhase.INITIALIZED;
 12
 13  [SerializeField]
 14  protected Vector3 _velocity;
 15
 16  [SerializeField]
 17  protected Vector3 _acceleration;
 18
 19  [SerializeField]
 20  protected Vector3 _dragAcceleration;
 21
 22  [SerializeField]
 23  protected float _speed;
 24
 25  [SerializeField]
 6426  protected List<Agent> _interceptors = new List<Agent>();
 27  [SerializeField]
 28  protected Agent _target;
 29  [SerializeField]
 30  protected Agent _targetModel;
 31
 32  [SerializeField]
 6433  protected double _timeSinceBoost = 0;
 34  [SerializeField]
 6435  protected double _timeInPhase = 0;
 36
 37  public StaticAgentConfig staticAgentConfig;
 38  public DynamicAgentConfig dynamicAgentConfig;
 39
 40  // Define delegates.
 41  public delegate void TerminatedEventHandler(Agent agent);
 42  public event TerminatedEventHandler OnTerminated;
 43
 44  public delegate void InterceptHitEventHandler(Interceptor interceptor, Threat target);
 45  public delegate void InterceptMissEventHandler(Interceptor interceptor, Threat target);
 46  public delegate void ThreatHitEventHandler(Threat threat);
 47  public delegate void ThreatMissEventHandler(Threat threat);
 48
 49  // Define events.
 50  public event InterceptHitEventHandler OnInterceptHit;
 51  public event InterceptMissEventHandler OnInterceptMiss;
 52  public event ThreatHitEventHandler OnThreatHit;
 53  public event ThreatMissEventHandler OnThreatMiss;
 54
 55  private Vector3 _initialVelocity;
 56
 057  public void SetFlightPhase(FlightPhase flightPhase) {
 058    _timeInPhase = 0;
 059    _flightPhase = flightPhase;
 060    if (flightPhase == FlightPhase.INITIALIZED || flightPhase == FlightPhase.TERMINATED) {
 061      GetComponent<Rigidbody>().constraints = RigidbodyConstraints.FreezeAll;
 062    } else {
 063      GetComponent<Rigidbody>().constraints = RigidbodyConstraints.None;
 064    }
 065    if (flightPhase == FlightPhase.BOOST) {
 066      SetVelocity(GetVelocity() + _initialVelocity);
 067    }
 068  }
 69
 070  public FlightPhase GetFlightPhase() {
 071    return _flightPhase;
 072  }
 73
 074  public bool HasLaunched() {
 075    return _flightPhase != FlightPhase.INITIALIZED;
 076  }
 77
 078  public bool HasTerminated() {
 079    return _flightPhase == FlightPhase.TERMINATED;
 080  }
 81
 1582  public virtual void SetDynamicAgentConfig(DynamicAgentConfig config) {
 1583    dynamicAgentConfig = config;
 1584  }
 85
 1586  public virtual void SetStaticAgentConfig(StaticAgentConfig config) {
 1587    staticAgentConfig = config;
 1588    GetComponent<Rigidbody>().mass = staticAgentConfig.bodyConfig.mass;
 1589  }
 90
 091  public virtual bool IsAssignable() {
 092    return true;
 093  }
 94
 095  public virtual void AssignTarget(Agent target) {
 096    _target = target;
 097    _target.AddInterceptor(this);
 098    _targetModel = SimManager.Instance.CreateDummyAgent(target.GetPosition(), target.GetVelocity());
 099  }
 100
 0101  public Agent GetAssignedTarget() {
 0102    return _target;
 0103  }
 104
 14105  public bool HasAssignedTarget() {
 14106    return _target != null;
 14107  }
 108
 0109  public Agent GetTargetModel() {
 0110    return _targetModel;
 0111  }
 112
 0113  public virtual void UnassignTarget() {
 0114    if (HasAssignedTarget()) {
 0115      _target.RemoveInterceptor(this);
 0116    }
 0117    _target = null;
 0118    _targetModel = null;
 0119  }
 120
 24121  public bool IsTerminated() {
 24122    return _flightPhase == FlightPhase.TERMINATED;
 24123  }
 124
 0125  public void AddInterceptor(Agent interceptor) {
 0126    _interceptors.Add(interceptor);
 0127  }
 128
 0129  public void RemoveInterceptor(Agent interceptor) {
 0130    _interceptors.Remove(interceptor);
 0131  }
 132
 133  public IReadOnlyList<Agent> AssignedInterceptors {
 15134    get { return _interceptors; }
 135  }
 136
 0137  public virtual void TerminateAgent() {
 0138    UnassignTarget();
 0139    if (_flightPhase != FlightPhase.TERMINATED) {
 0140      OnTerminated?.Invoke(this);
 0141    }
 0142    _flightPhase = FlightPhase.TERMINATED;
 0143    SetPosition(Vector3.zero);
 0144    gameObject.SetActive(false);
 0145  }
 146
 0147  public void HandleInterceptHit(Agent agent) {
 0148    if (this is Interceptor interceptor && agent is Threat threat) {
 0149      OnInterceptHit?.Invoke(interceptor, threat);
 0150      TerminateAgent();
 0151    }
 0152  }
 153
 0154  public void HandleInterceptMiss() {
 0155    if (this is Interceptor interceptor && _target is Threat threat) {
 0156      OnInterceptMiss?.Invoke(interceptor, threat);
 0157    }
 0158  }
 159
 0160  public void HandleTargetIntercepted() {
 0161    if (this is Threat threat) {
 0162      TerminateAgent();
 0163    }
 0164  }
 165
 0166  public void HandleThreatHit() {
 0167    if (this is Threat threat) {
 0168      OnThreatHit?.Invoke(threat);
 0169      TerminateAgent();
 0170    }
 0171  }
 172
 0173  public void HandleHitGround() {
 0174    if (this is Interceptor interceptor && _target is Threat target) {
 0175      OnInterceptMiss?.Invoke(interceptor, target);
 0176    }
 0177    if (this is Threat threat) {
 0178      OnThreatMiss?.Invoke(threat);
 0179    }
 0180    TerminateAgent();
 0181  }
 182
 15183  public void SetInitialVelocity(Vector3 velocity) {
 15184    _initialVelocity = velocity;
 15185  }
 186
 67187  public Vector3 GetPosition() {
 67188    return transform.position;
 67189  }
 190
 4191  public void SetPosition(Vector3 position) {
 4192    transform.position = position;
 4193  }
 194
 30195  public double GetSpeed() {
 30196    return GetComponent<Rigidbody>().linearVelocity.magnitude;
 30197  }
 198
 67199  public Vector3 GetVelocity() {
 67200    return GetComponent<Rigidbody>().linearVelocity;
 67201  }
 202
 1203  public void SetVelocity(Vector3 velocity) {
 1204    GetComponent<Rigidbody>().linearVelocity = velocity;
 1205  }
 206
 11207  public Vector3 GetAcceleration() {
 11208    return _acceleration;
 11209  }
 210
 0211  public void SetAcceleration(Vector3 acceleration) {
 0212    _acceleration = acceleration;
 0213  }
 214
 3215  public Transformation GetRelativeTransformation(Agent target) {
 3216    Transformation transformation = new Transformation();
 217
 218    // Get the relative position transformation.
 3219    transformation.position =
 220        GetRelativePositionTransformation(target.GetPosition() - GetPosition());
 221
 222    // Get the relative velocity transformation.
 3223    transformation.velocity = GetRelativeVelocityTransformation(
 224        target.GetPosition() - GetPosition(), target.GetVelocity() - GetVelocity());
 225
 226    // Get the relative acceleration transformation.
 3227    transformation.acceleration = GetRelativeAccelerationTransformation(target);
 3228    return transformation;
 3229  }
 230
 5231  public Transformation GetRelativeTransformationToWaypoint(Vector3 waypoint) {
 5232    Transformation transformation = new Transformation();
 233
 234    // Get the relative position transformation.
 5235    transformation.position = GetRelativePositionTransformation(waypoint - GetPosition());
 236
 237    // Get the relative velocity transformation.
 5238    transformation.velocity =
 239        GetRelativeVelocityTransformation(waypoint - GetPosition(), -GetVelocity());
 240
 5241    return transformation;
 5242  }
 243
 8244  private PositionTransformation GetRelativePositionTransformation(Vector3 relativePosition) {
 8245    PositionTransformation positionTransformation = new PositionTransformation();
 246
 247    // Set the relative position in Cartesian coordinates.
 8248    positionTransformation.cartesian = relativePosition;
 249
 250    // Calculate the distance (range) to the target.
 8251    positionTransformation.range = relativePosition.magnitude;
 252
 8253    Vector3 flatRelativePosition = Vector3.ProjectOnPlane(relativePosition, transform.up);
 8254    Vector3 verticalRelativePosition = relativePosition - flatRelativePosition;
 255
 256    // Calculate the elevation (vertical angle relative to forward).
 8257    positionTransformation.elevation =
 258        Mathf.Atan(verticalRelativePosition.magnitude / flatRelativePosition.magnitude);
 259
 260    // Calculate the azimuth (horizontal angle relative to forward).
 9261    if (flatRelativePosition.magnitude == 0) {
 1262      positionTransformation.azimuth = 0;
 8263    } else {
 7264      positionTransformation.azimuth =
 265          Vector3.SignedAngle(transform.forward, flatRelativePosition, transform.up) * Mathf.PI /
 266          180;
 7267    }
 268
 8269    return positionTransformation;
 8270  }
 271
 272  private VelocityTransformation GetRelativeVelocityTransformation(Vector3 relativePosition,
 8273                                                                   Vector3 relativeVelocity) {
 8274    VelocityTransformation velocityTransformation = new VelocityTransformation();
 275
 276    // Set the relative velocity in Cartesian coordinates.
 8277    velocityTransformation.cartesian = relativeVelocity;
 278
 279    // Calculate range rate (radial velocity).
 8280    velocityTransformation.range = Vector3.Dot(relativeVelocity, relativePosition.normalized);
 281
 282    // Project relative velocity onto the sphere passing through the target.
 8283    Vector3 tangentialVelocity = Vector3.ProjectOnPlane(relativeVelocity, relativePosition);
 284
 285    // The target azimuth vector is orthogonal to the relative position vector and
 286    // points to the starboard of the target along the azimuth-elevation sphere.
 8287    Vector3 targetAzimuth = Vector3.Cross(transform.up, relativePosition);
 288    // The target elevation vector is orthogonal to the relative position vector
 289    // and points upwards from the target along the azimuth-elevation sphere.
 8290    Vector3 targetElevation = Vector3.Cross(relativePosition, transform.right);
 291    // If the relative position vector is parallel to the yaw or pitch axis, the
 292    // target azimuth vector or the target elevation vector will be undefined.
 9293    if (targetAzimuth.magnitude == 0) {
 1294      targetAzimuth = Vector3.Cross(targetElevation, relativePosition);
 10295    } else if (targetElevation.magnitude == 0) {
 2296      targetElevation = Vector3.Cross(relativePosition, targetAzimuth);
 2297    }
 298
 299    // Project the relative velocity vector on the azimuth-elevation sphere onto
 300    // the target azimuth vector.
 8301    Vector3 tangentialVelocityOnAzimuth = Vector3.Project(tangentialVelocity, targetAzimuth);
 302
 303    // Calculate the time derivative of the azimuth to the target.
 8304    velocityTransformation.azimuth =
 305        tangentialVelocityOnAzimuth.magnitude / relativePosition.magnitude;
 9306    if (Vector3.Dot(tangentialVelocityOnAzimuth, targetAzimuth) < 0) {
 1307      velocityTransformation.azimuth *= -1;
 1308    }
 309
 310    // Project the velocity vector on the azimuth-elevation sphere onto the target
 311    // elevation vector.
 8312    Vector3 tangentialVelocityOnElevation = Vector3.Project(tangentialVelocity, targetElevation);
 313
 314    // Calculate the time derivative of the elevation to the target.
 8315    velocityTransformation.elevation =
 316        tangentialVelocityOnElevation.magnitude / relativePosition.magnitude;
 9317    if (Vector3.Dot(tangentialVelocityOnElevation, targetElevation) < 0) {
 1318      velocityTransformation.elevation *= -1;
 1319    }
 320
 8321    return velocityTransformation;
 8322  }
 323
 3324  private AccelerationTransformation GetRelativeAccelerationTransformation(Agent agent) {
 325    // Since the agent's acceleration is an input, the relative acceleration is just the agent's
 326    // acceleration.
 3327    AccelerationTransformation accelerationTransformation = new AccelerationTransformation();
 3328    accelerationTransformation.cartesian = agent.GetAcceleration();
 3329    return accelerationTransformation;
 3330  }
 331
 0332  public double GetDynamicPressure() {
 0333    var airDensity = Constants.CalculateAirDensityAtAltitude(transform.position.y);
 0334    var flowSpeed = GetSpeed();
 0335    return 0.5 * airDensity * (flowSpeed * flowSpeed);
 0336  }
 337
 338  protected abstract void UpdateReady(double deltaTime);
 339  protected abstract void UpdateBoost(double deltaTime);
 340  protected abstract void UpdateMidCourse(double deltaTime);
 341
 0342  protected virtual void Awake() {}
 343
 344  // Start is called before the first frame update
 38345  protected virtual void Start() {}
 346
 347  // Update is called once per frame
 0348  protected virtual void FixedUpdate() {
 0349    _speed = (float)GetSpeed();
 0350    if (_flightPhase != FlightPhase.INITIALIZED && _flightPhase != FlightPhase.READY) {
 0351      _timeSinceBoost += Time.fixedDeltaTime;
 0352    }
 0353    _timeInPhase += Time.fixedDeltaTime;
 354
 0355    var boost_time = staticAgentConfig.boostConfig.boostTime;
 0356    double elapsedSimulationTime = SimManager.Instance.GetElapsedSimulationTime();
 357
 0358    if (_flightPhase == FlightPhase.TERMINATED) {
 0359      return;
 360    }
 0361    if (_flightPhase == FlightPhase.INITIALIZED || _flightPhase == FlightPhase.READY) {
 0362      SetFlightPhase(FlightPhase.BOOST);
 0363    }
 0364    if (_timeSinceBoost > boost_time && _flightPhase == FlightPhase.BOOST) {
 0365      SetFlightPhase(FlightPhase.MIDCOURSE);
 0366    }
 0367    AlignWithVelocity();
 0368    switch (_flightPhase) {
 369      case FlightPhase.INITIALIZED:
 0370        break;
 371      case FlightPhase.READY:
 0372        UpdateReady(Time.fixedDeltaTime);
 0373        break;
 374      case FlightPhase.BOOST:
 0375        UpdateBoost(Time.fixedDeltaTime);
 0376        break;
 377      case FlightPhase.MIDCOURSE:
 378      case FlightPhase.TERMINAL:
 0379        UpdateMidCourse(Time.fixedDeltaTime);
 0380        break;
 381      case FlightPhase.TERMINATED:
 0382        break;
 383    }
 384
 0385    _velocity = GetVelocity();
 386    // Store the acceleration because it is set to zero after each simulation step
 0387    _acceleration =
 388        GetComponent<Rigidbody>().GetAccumulatedForce() / GetComponent<Rigidbody>().mass;
 0389  }
 390
 0391  protected virtual void AlignWithVelocity() {
 0392    Vector3 velocity = GetVelocity();
 0393    if (velocity.magnitude > 0.1f)  // Only align if we have significant velocity
 0394    {
 395      // Create a rotation with forward along velocity and up along world up
 0396      Quaternion targetRotation = Quaternion.LookRotation(velocity, Vector3.up);
 397
 398      // Smoothly rotate towards the target rotation
 0399      transform.rotation = Quaternion.RotateTowards(transform.rotation, targetRotation,
 400                                                    10000f * Time.fixedDeltaTime);
 0401    }
 0402  }
 403
 0404  protected Vector3 CalculateAcceleration(Vector3 accelerationInput) {
 0405    Vector3 gravity = Physics.gravity;
 0406    float airDrag = CalculateDrag();
 0407    float liftInducedDrag = CalculateLiftInducedDrag(accelerationInput + gravity);
 0408    float dragAcceleration = -(airDrag + liftInducedDrag);
 409
 410    // Project the drag acceleration onto the forward direction
 0411    Vector3 dragAccelerationAlongRoll = dragAcceleration * transform.forward;
 0412    _dragAcceleration = dragAccelerationAlongRoll;
 413
 0414    return accelerationInput + gravity + dragAccelerationAlongRoll;
 0415  }
 416
 5417  public float CalculateMaxForwardAcceleration() {
 5418    return staticAgentConfig.accelerationConfig.maxForwardAcceleration;
 5419  }
 420
 15421  public float CalculateMaxNormalAcceleration() {
 15422    float maxReferenceNormalAcceleration =
 423        (float)(staticAgentConfig.accelerationConfig.maxReferenceNormalAcceleration *
 424                Constants.kGravity);
 15425    float referenceSpeed = staticAgentConfig.accelerationConfig.referenceSpeed;
 15426    return Mathf.Pow((float)GetSpeed() / referenceSpeed, 2) * maxReferenceNormalAcceleration;
 15427  }
 428
 0429  private float CalculateDrag() {
 0430    float dragCoefficient = staticAgentConfig.liftDragConfig.dragCoefficient;
 0431    float crossSectionalArea = staticAgentConfig.bodyConfig.crossSectionalArea;
 0432    float mass = staticAgentConfig.bodyConfig.mass;
 0433    float dynamicPressure = (float)GetDynamicPressure();
 0434    float dragForce = dragCoefficient * dynamicPressure * crossSectionalArea;
 0435    return dragForce / mass;
 0436  }
 437
 0438  private float CalculateLiftInducedDrag(Vector3 accelerationInput) {
 0439    float liftAcceleration = Vector3.ProjectOnPlane(accelerationInput, transform.up).magnitude;
 0440    float liftDragRatio = staticAgentConfig.liftDragConfig.liftDragRatio;
 0441    return Mathf.Abs(liftAcceleration / liftDragRatio);
 0442  }
 443}
 444
 445public class DummyAgent : Agent {
 446  protected override void Start() {
 447    base.Start();
 448  }
 449
 450  protected override void FixedUpdate() {
 451    GetComponent<Rigidbody>().AddForce(_acceleration, ForceMode.Acceleration);
 452  }
 453
 454  protected override void UpdateReady(double deltaTime) {
 455    // Do nothing.
 456  }
 457
 458  protected override void UpdateBoost(double deltaTime) {
 459    // Do nothing.
 460  }
 461
 462  protected override void UpdateMidCourse(double deltaTime) {
 463    // Do nothing.
 464  }
 465}