#
# ISC License
#
# Copyright (c) 2023, Autonomous Vehicle Systems Lab, University of Colorado at Boulder
#
# Permission to use, copy, modify, and/or distribute this software for any
# purpose with or without fee is hereby granted, provided that the above
# copyright notice and this permission notice appear in all copies.
#
# THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
# WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
# MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
# ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
# WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
# ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
# OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
#
#
# Unit Test Script
# Module Name: solarArrayReference
# Author: Riccardo Calaon
# Creation Date: January 21, 2023
#
import pytest
import math
import numpy as np
# Import all of the modules that we are going to be called in this simulation
from Basilisk.utilities import SimulationBaseClass
from Basilisk.utilities import unitTestSupport # general support file with common unit test functions
from Basilisk.fswAlgorithms import solarArrayReference # import the module that is to be tested
from Basilisk.utilities import macros
from Basilisk.utilities import RigidBodyKinematics as rbk
from Basilisk.architecture import messaging # import the message definitions
from Basilisk.architecture import bskLogging
# this python function computes the same reference angle as the tested module
def computeSunReferenceAngle(sigma_RN, rHat_SB_N, a1Hat_B, a2Hat_B, theta0):
RN = rbk.MRP2C(sigma_RN)
rS_R = np.matmul(RN, rHat_SB_N)
a2_R = []
dotP = np.dot(rS_R, a1Hat_B)
for n in range(3):
a2_R.append(rS_R[n] - dotP * a1Hat_B[n])
a2_R = np.array(a2_R)
a2_R_norm = np.linalg.norm(a2_R)
if a2_R_norm > 1e-6:
a2_R = a2_R / a2_R_norm
theta = np.arccos(min(max(np.dot(a2Hat_B, a2_R),-1),1))
if np.dot(a1Hat_B, np.cross(a2Hat_B, a2_R)) < 0:
theta = -theta
else:
theta = theta0
if theta-theta0 > np.pi:
theta -= np.pi
elif theta-theta0 < -np.pi:
theta += np.pi
return theta
# Uncomment this line is this test is to be skipped in the global unit test run, adjust message as needed.
# @pytest.mark.skipif(conditionstring)
# Uncomment this line if this test has an expected failure, adjust message as needed.
# @pytest.mark.xfail(conditionstring)
# Provide a unique test method name, starting with 'test_'.
# The following 'parametrize' function decorator provides the parameters and expected results for each
# of the multiple test runs for this test. Note that the order in that you add the parametrize method
# matters for the documentation in that it impacts the order in which the test arguments are shown.
# The first parametrize arguments are shown last in the pytest argument list
[docs]
@pytest.mark.parametrize("rHat_SB_N", [[1, 0, 0],
[0, 0, 1]])
@pytest.mark.parametrize("sigma_BN", [[0.1, 0.2, 0.3],
[0.5, 0.4, 0.3]])
@pytest.mark.parametrize("sigma_RN", [[0.3, 0.2, 0.1],
[0.9, 0.7, 0.8]])
@pytest.mark.parametrize("bodyFrame", [0, 1])
@pytest.mark.parametrize("pointingMode", [0, 1])
@pytest.mark.parametrize("accuracy", [1e-12])
def test_solarArrayRotation(show_plots, rHat_SB_N, sigma_BN, sigma_RN, bodyFrame, pointingMode, accuracy):
r"""
**Validation Test Description**
This unit test verifies the correctness of the output reference angle computed by the :ref:`solarArrayReference`.
The inputs provided are the inertial Sun direction, current attitude of the hub, and reference frame. Based on
current attitude, the sun direction vector is mapped into body frame coordinates and passed into the Attitude
Navigation Message.
**Test Parameters**
Args:
rHat_SB_N[3] (double): Sun direction vector, in inertial frame components;
sigma_BN[3] (double): spacecraft hub attitude with respect to the inertial frame, in MRP;
sigma_RN[3] (double): reference frame attitude with respect to the inertial frame, in MRP;
bodyFrame (int): 0 to calculate reference rotation angle w.r.t. reference frame, 1 to calculate it w.r.t the current spacecraft attitude;
pointingMode (int): 0 to compute reference angle to maximize power generation, 1 for maximum momentum dumping
accuracy (float): absolute accuracy value used in the validation tests.
**Description of Variables Being Tested**
This unit test checks the correctness of the output attitude reference message
- ``hingedRigidBodyRefOutMsg``
in all its parts. The reference angle ``theta`` is checked versus the value computed by a python function that computes the same angle.
The reference angle derivative ``thetaDot`` is checked versus zero, as the module is run for only one Update call.
"""
# each test method requires a single assert method to be called
[testResults, testMessage] = solarArrayRotationTestFunction(show_plots, rHat_SB_N, sigma_BN, sigma_RN, bodyFrame, pointingMode, accuracy)
assert testResults < 1, testMessage
def solarArrayRotationTestFunction(show_plots, rHat_SB_N, sigma_BN, sigma_RN, attitudeFrame, pointingMode, accuracy):
a1Hat_B = np.array([1, 0, 0])
a2Hat_B = np.array([0, 1, 0])
r_AB_B = np.array([4.5, 0, 0.5])
BN = rbk.MRP2C(sigma_BN)
rHat_SB_B = np.matmul(BN, rHat_SB_N)
thetaC = 0
thetaDotC = 0
testFailCount = 0 # zero unit test result counter
testMessages = [] # create empty array to store test log messages
unitTaskName = "unitTask" # arbitrary name (don't change)
unitProcessName = "TestProcess" # arbitrary name (don't change)
bskLogging.setDefaultLogLevel(bskLogging.BSK_WARNING)
# Create a sim module as an empty container
unitTestSim = SimulationBaseClass.SimBaseClass()
# Create test thread
testProcessRate = macros.sec2nano(1) # update process rate update time
testProc = unitTestSim.CreateNewProcess(unitProcessName)
testProc.addTask(unitTestSim.CreateNewTask(unitTaskName, testProcessRate))
# Construct tested module and associated C container
solarArray = solarArrayReference.solarArrayReference()
solarArray.ModelTag = "solarArrayReference"
# Add test module to runtime call list
unitTestSim.AddModelToTask(unitTaskName, solarArray)
# Initialize the test module configuration data
solarArray.a1Hat_B = a1Hat_B
solarArray.a2Hat_B = a2Hat_B
solarArray.r_AB_B = r_AB_B
solarArray.pointingMode = pointingMode
solarArray.attitudeFrame = attitudeFrame
solarArray.ThetaMax = np.pi/2
solarArray.sigma = 1
solarArray.n = 1
# Create input attitude navigation message
natAttInMsgData = messaging.NavAttMsgPayload()
natAttInMsgData.sigma_BN = sigma_BN
natAttInMsgData.vehSunPntBdy = rHat_SB_B
natAttInMsg = messaging.NavAttMsg().write(natAttInMsgData)
solarArray.attNavInMsg.subscribeTo(natAttInMsg)
# Create input attitude reference message
attRefInMsgData = messaging.AttRefMsgPayload()
attRefInMsgData.sigma_RN = sigma_RN
attRefInMsg = messaging.AttRefMsg().write(attRefInMsgData)
solarArray.attRefInMsg.subscribeTo(attRefInMsg)
# Create input hinged rigid body message
hingedRigidBodyInMsgData = messaging.HingedRigidBodyMsgPayload()
hingedRigidBodyInMsgData.theta = thetaC
hingedRigidBodyInMsgData.thetaDot = thetaDotC
hingedRigidBodyInMsg = messaging.HingedRigidBodyMsg().write(hingedRigidBodyInMsgData)
solarArray.hingedRigidBodyInMsg.subscribeTo(hingedRigidBodyInMsg)
if pointingMode == 1:
# Create input vehicle config msg
vehConfigInMsgData = messaging.VehicleConfigMsgPayload()
vehConfigInMsgData.CoM_B = [0.0, 0.0, 0.0]
vehConfigInMsg = messaging.VehicleConfigMsg().write(vehConfigInMsgData)
solarArray.vehConfigInMsg.subscribeTo(vehConfigInMsg)
# Create RW configuration message
rwArrayConfigInMsgData = messaging.RWArrayConfigMsgPayload()
rwArrayConfigInMsgData.numRW = 3
rwArrayConfigInMsgData.GsMatrix_B = [1.0, 0.0, 0.0, 0.0, 1.0, 0.0, 0.0, 0.0, 1.0]
rwArrayConfigInMsgData.JsList = [0.15, 0.15, 0.15]
rwArrayConfigInMsg = messaging.RWArrayConfigMsg().write(rwArrayConfigInMsgData)
solarArray.rwConfigDataInMsg.subscribeTo(rwArrayConfigInMsg)
# Create RW speeds message
rwSpeedsInMsgData = messaging.RWSpeedMsgPayload()
rwSpeedsInMsgData.wheelSpeeds = macros.RPM * 1000.0 * np.array([1.0, 1.0, 1.0])
rwSpeedsInMsg = messaging.RWSpeedMsg().write(rwSpeedsInMsgData)
solarArray.rwSpeedsInMsg.subscribeTo(rwSpeedsInMsg)
# Setup logging on the test module output message so that we get all the writes to it
dataLog = solarArray.hingedRigidBodyRefOutMsg.recorder()
unitTestSim.AddModelToTask(unitTaskName, dataLog)
# Need to call the self-init and cross-init methods
unitTestSim.InitializeSimulation()
# Set the simulation time.
# NOTE: the total simulation time may be longer than this value. The
# simulation is stopped at the next logging event on or after the
# simulation end time.
unitTestSim.ConfigureStopTime(macros.sec2nano(0.5)) # seconds to stop simulation
# Begin the simulation time run set above
unitTestSim.ExecuteSimulation()
if attitudeFrame == 0:
thetaSunR = computeSunReferenceAngle(sigma_RN, rHat_SB_N, a1Hat_B, a2Hat_B, thetaC)
else:
thetaSunR = computeSunReferenceAngle(sigma_BN, rHat_SB_N, a1Hat_B, a2Hat_B, thetaC)
if pointingMode == 0:
thetaR = thetaSunR
elif pointingMode == 1:
H_B = np.array([0.0, 0.0, 0.0])
for i in range(rwArrayConfigInMsgData.numRW):
for j in range(3):
H_B[j] = (H_B[j] + rwArrayConfigInMsgData.GsMatrix_B[3*i+j]
* rwArrayConfigInMsgData.JsList[i] * rwSpeedsInMsgData.wheelSpeeds[i])
if attitudeFrame == 0:
BN = rbk.MRP2C(sigma_BN)
RN = rbk.MRP2C(sigma_RN)
RB = np.matmul(RN, BN.transpose())
sHat = np.matmul(RB, rHat_SB_B)
H_R = np.matmul(RB, H_B)
else:
sHat = rHat_SB_B
H_R = H_B
F_R = np.cross(sHat, r_AB_B)
f = np.dot(F_R, H_R)
if f > 0:
thetaR = thetaSunR + 2 * solarArray.ThetaMax / np.pi * np.arctan(solarArray.sigma * f**(solarArray.n))
else:
thetaR = thetaSunR
# If the angle difference is greater than PI, adjust it to the range [-PI, PI]
thetaDiff = math.fmod(thetaR-thetaC, 2 * np.pi)
if thetaDiff > np.pi:
thetaR -= 2*np.pi
elif thetaDiff < -np.pi:
thetaR += 2*np.pi
# compare the module results to the truth values
if not unitTestSupport.isDoubleEqual(dataLog.theta[0], thetaR, accuracy):
testFailCount += 1
testMessages.append("FAILED: "
+ solarArray.ModelTag
+ "solarArrayRotation module failed unit test on thetaR for sigma_BN = [{},{},{}], "
"sigma_RN = [{},{},{}] and attitudeFrame = {} \n".format(
sigma_BN[0], sigma_BN[1], sigma_BN[2], sigma_RN[0], sigma_RN[1], sigma_RN[2], attitudeFrame))
if not unitTestSupport.isDoubleEqual(dataLog.thetaDot[0], 0, accuracy):
testFailCount += 1
testMessages.append("FAILED: "
+ solarArray.ModelTag
+ "solarArrayRotation module failed unit test on thetaDotR for sigma_BN = [{},{},{}], "
"sigma_RN = [{},{},{}] and attitudeFrame = {} \n".format(
sigma_BN[0], sigma_BN[1], sigma_BN[2], sigma_RN[0], sigma_RN[1], sigma_RN[2], attitudeFrame))
# each test method requires a single assert method to be called
# this check below just makes sure no sub-test failures were found
return [testFailCount, ''.join(testMessages)]
#
# This statement below ensures that the unitTestScript can be run as a
# stand-along python script
#
if __name__ == "__main__":
test_solarArrayRotation(
False,
np.array([0, 0, 1]),
np.array([0.5, 0.4, 0.3]),
np.array([0.9, 0.7, 0.8]),
1,
0,
1e-12,
)