#
# ISC License
#
# Copyright (c) 2016, 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.
#
r"""
Overview
--------
Demonstrates how to convert spacecraft states, stored in a text file from another program, into Basilisk
messages using :ref:`dataFileToViz`. These messages are red by :ref:`vizInterface` to save a :ref:`Vizard <vizard>`
compatible data play for offline playback and analysis. In this simulation a servicer is holding a relative
position with respect to an uncontrolled satellite. Custom spacecraft models are specified for Vizard
in the folder ``dataForExamples``.
The script is found in the folder ``basilisk/examples`` and executed by using::
python3 scenarioDataToViz.py
The simulation layout is shown in the following illustration. A single simulation process is created
which contains both modules.
.. image:: /_images/static/test_scenarioDataToViz.svg
:align: center
When the simulation completes several plots are shown for the MRP norm attitude history and the
inertial relative position vector components. A servicer spacecraft approaches a target and holds a specific
target-frame fixed location even while the target itself is slowly rotating. The servicer and target orientations
are controlled to be the same to prepare for a final docking maneuver. If the data is saved to a Vizard file,
then the visualization should look like:
.. image:: /_images/static/vizard-DataFile.jpg
:align: center
Illustration of Simulation Results
----------------------------------
::
show_plots = True
.. image:: /_images/Scenarios/scenarioDataToViz1.svg
:align: center
.. image:: /_images/Scenarios/scenarioDataToViz2.svg
:align: center
"""
#
# Basilisk Scenario Script and Integrated Test
#
# Purpose: Basic simulation showing a servicer (3-axis attitude controlled) and a tumbling debris object.
# Author: Hanspeter Schaub
# Creation Date: Dec. 29, 2019
#
import os
import matplotlib.pyplot as plt
import numpy as np
from Basilisk.simulation import dataFileToViz
from Basilisk.simulation import spacecraft
from Basilisk.utilities import (SimulationBaseClass, macros, simIncludeGravBody, vizSupport)
from Basilisk.utilities import unitTestSupport
try:
from Basilisk.simulation import vizInterface
except ImportError:
pass
# The path to the location of Basilisk
# Used to get the location of supporting data.
fileName = os.path.basename(os.path.splitext(__file__)[0])
[docs]
def run(show_plots, attType):
"""
The scenarios can be run with the followings setups parameters:
Args:
show_plots (bool): Determines if the script should display plots
attType (int): Specify if MRP (0) or Quaternions (1) are used
"""
path = os.path.dirname(os.path.abspath(__file__))
if attType == 0:
dataFileName = os.path.join(path, "dataForExamples", "scHoldTraj_rotating_MRP.csv")
elif attType == 1:
dataFileName = os.path.join(path, "dataForExamples", "scHoldTraj_rotating_EP.csv")
else:
print("unknown attType variable")
exit()
file1 = open(dataFileName, 'r')
Lines = file1.readlines()
delimiter = ","
t0 = float(Lines[1].split(delimiter)[0])
t1 = float(Lines[2].split(delimiter)[0])
tN = float(Lines[-1].split(delimiter)[0])
timeStepSeconds = t1 - t0
simulationTimeSeconds = tN - t0
# Create simulation variable names
simTaskName = "simTask"
simProcessName = "simProcess"
# Create a sim module as an empty container
scSim = SimulationBaseClass.SimBaseClass()
#
# create the simulation process
#
dynProcess = scSim.CreateNewProcess(simProcessName)
# create the dynamics task and specify the simulation time step information
simulationTimeStep = macros.sec2nano(timeStepSeconds)
simulationTime = macros.sec2nano(simulationTimeSeconds)
dynProcess.addTask(scSim.CreateNewTask(simTaskName, simulationTimeStep))
# setup Earth Gravity Body
gravFactory = simIncludeGravBody.gravBodyFactory()
earth = gravFactory.createEarth()
earth.isCentralBody = True # ensure this is the central gravitational body
# create SC dummy objects to setup basic Vizard settings. Only one has to have the Grav Bodies attached
# to show up in Vizard
scObject1 = spacecraft.Spacecraft()
scObject1.ModelTag = "servicer"
gravFactory.addBodiesTo(scObject1)
scObject2 = spacecraft.Spacecraft()
scObject2.ModelTag = "target"
scList = [scObject1, scObject2]
# setup the module to read in the simulation data
dataModule = dataFileToViz.DataFileToViz()
dataModule.ModelTag = "testModule"
dataModule.setNumOfSatellites(2)
# load the data path from the same folder where this python script is
dataModule.attitudeType = attType
dataModule.dataFileName = dataFileName
dataModule.delimiter = delimiter
scSim.AddModelToTask(simTaskName, dataModule)
#
# Setup data logging before the simulation is initialized
#
numDataPoints = 100
samplingTime = unitTestSupport.samplingTime(simulationTime, simulationTimeStep, numDataPoints)
dataLog = []
for scCounter in range(2):
dataLog.append(dataModule.scStateOutMsgs[scCounter].recorder(samplingTime))
scSim.AddModelToTask(simTaskName, dataLog[-1])
# if this scenario is to interface with the BSK Viz, uncomment the following lines
# to save the BSK data to a file, uncomment the saveFile line below
if vizSupport.vizFound:
viz = vizSupport.enableUnityVisualization(scSim, simTaskName, scList
# , saveFile=fileName
)
viz.settings.trueTrajectoryLinesOn = 2 # relative to chief spacecraft
viz.settings.showSpacecraftLabels = 1
viz.settings.spacecraftShadowBrightness = 0.2
# load CAD for target spacecraft
vizSupport.createCustomModel(viz,
modelPath=os.path.join(path, "dataForExamples", "Aura_27.obj"),
shader=1,
simBodiesToModify=[scList[1].ModelTag],
rotation=[180. * macros.D2R, 0.0 * macros.D2R, -90. * macros.D2R],
scale=[1, 1, 1])
# load CAD for servicer spacecraft
vizSupport.createCustomModel(viz,
modelPath=os.path.join(path, "dataForExamples", "Loral-1300Com-main.obj"),
simBodiesToModify=[scList[0].ModelTag],
rotation=[0. * macros.D2R, -90.0 * macros.D2R, 0. * macros.D2R],
scale=[0.09, 0.09, 0.09])
# over-ride the default to not read the SC states from scObjects, but set them directly
# to read from the dataFileToFiz output message
viz.scData.clear()
for c in range(len(scList)):
scData = vizInterface.VizSpacecraftData()
scData.spacecraftName = scList[c].ModelTag
scData.scStateInMsg.subscribeTo(dataModule.scStateOutMsgs[c])
viz.scData.push_back(scData)
# initialize Simulation
scSim.InitializeSimulation()
# configure a simulation stop time and execute the simulation run
scSim.ConfigureStopTime(simulationTime)
scSim.ExecuteSimulation()
# retrieve logged data
posB1N = dataLog[0].r_BN_N
posB2N = dataLog[1].r_BN_N
sigmaB1N = dataLog[0].sigma_BN
sigmaB2N = dataLog[1].sigma_BN
#
# plot the results
#
timeData = dataLog[0].times() * macros.NANO2HOUR
plt.close("all") # clears out plots from earlier test runs
figureList = {}
plt.figure(1)
s1Data = []
for idx in sigmaB1N:
sNorm = np.linalg.norm(idx)
s1Data.append(sNorm)
plt.plot(timeData, s1Data, color=unitTestSupport.getLineColor(1, 3), label=r'$|\sigma_{B1/N}|$')
s2Data = []
for idx in sigmaB2N:
sNorm = np.linalg.norm(idx)
s2Data.append(sNorm)
plt.plot(timeData, s2Data, color=unitTestSupport.getLineColor(2, 3), label=r'$|\sigma_{B2/N}|$')
plt.xlabel('Time [h]')
plt.ylabel(r'MRP Norm')
plt.legend(loc='lower right')
pltName = fileName + "1"
figureList[pltName] = plt.figure(1)
plt.figure(2)
rhoData = []
for r1, r2 in zip(posB1N, posB2N):
rhoData.append(r2 - r1)
rhoData = np.array(rhoData)
for idx in range(3):
plt.plot(timeData, rhoData[:, idx],
color=unitTestSupport.getLineColor(idx, 3),
label=r'$\rho_{' + str(idx+1) + '}$')
plt.legend(loc='lower right')
plt.xlabel('Time [h]')
plt.ylabel(r'$\rho_{S/T}$ (Inertial) [m]')
plt.legend(loc='lower right')
pltName = fileName + "2"
figureList[pltName] = plt.figure(2)
if show_plots:
plt.show()
# close the plots being saved off to avoid over-writing old and new figures
plt.close("all")
return figureList
#
# This statement below ensures that the unit test scrip can be run as a
# stand-along python script
#
if __name__ == "__main__":
run(
True, # show_plots
0 # attitude coordinate type, 0 - MRP, 1 - quaternions
)