oscillation_ctrl/src/LinkState.py

237 lines
7.2 KiB
Python
Raw Normal View History

2022-03-01 14:09:13 -04:00
#!/usr/bin/env python2.7
### Cesar Rodriguez June 2021
### Script to determine payload and drone state from Gazebo
import rospy, tf
import rosservice
import time
import math
from tf.transformations import *
from oscillation_ctrl.msg import TetheredStatus, LoadAngles
2022-03-01 14:09:13 -04:00
from geometry_msgs.msg import Pose
from gazebo_msgs.srv import GetLinkState
from std_msgs.msg import Bool
class Main:
def __init__(self):
# rate(s)
2022-03-25 16:31:55 -03:00
rate = 60 # rate for the publisher method, specified in Hz -- 20 Hz
2022-08-18 15:29:26 -03:00
self.dt = 1.0/rate
2022-03-01 14:09:13 -04:00
2022-03-18 13:13:22 -03:00
rospy.sleep(5) # Sleep for 1 sec. Need to give time to Gazebo to ru
2022-03-01 14:09:13 -04:00
# Variables needed for testing start
self.tstart = rospy.get_time() # Keep track of the start time
while self.tstart == 0.0: # Need to make sure get_rostime works
self.tstart = rospy.get_time()
# How long should we wait before before starting test
self.param_exists = False
while self.param_exists == False:
2022-08-18 15:29:26 -03:00
if rospy.has_param('wait_time'):
2022-03-01 14:09:13 -04:00
self.wait = rospy.get_param('sim/wait') # wait time
self.param_exists = True
elif rospy.get_time() - self.tstart >= 3.0:
2022-04-05 16:08:28 -03:00
self.wait = 0.0
2022-03-01 14:09:13 -04:00
break
# Will be set to true when test should start
self.bool = False
2022-04-05 16:08:28 -03:00
# variables for message gen
self.status = TetheredStatus()
self.drone_id = 'spiri_with_tether::spiri::base'
self.pload_id = 'spiri_with_tether::mass::payload'
2022-04-05 16:08:28 -03:00
self.loadAngles = LoadAngles()
2022-03-01 14:09:13 -04:00
# initialize variables
self.tetherL = 0.0 # Tether length
self.has_run = 0 # Boolean to keep track of first run instance
self.phibuf = 0.0 # Need buffers to determine their rates
2022-03-01 14:09:13 -04:00
self.thetabuf = 0.0 #
self.pload = True # Check if payload exists
2022-04-05 16:08:28 -03:00
2022-03-01 14:09:13 -04:00
# Max dot values to prevent 'blowup'
self.phidot_max = 3.0
self.thetadot_max = 3.0
# service(s)
self.service1 = '/gazebo/get_link_state'
# wait for service to exist
2022-03-18 13:13:22 -03:00
rospy.wait_for_service(self.service1,timeout=10)
2022-03-01 14:09:13 -04:00
# publisher(s)
self.twobody_pub = rospy.Publisher('/status/twoBody_status', TetheredStatus, queue_size=1)
2022-04-05 16:08:28 -03:00
self.loadAng_pub = rospy.Publisher('/status/load_angles', LoadAngles, queue_size=1)
2022-03-01 14:09:13 -04:00
self.pub_wd = rospy.Publisher('/status/path_follow', Bool, queue_size=1)
self.pub_timer = rospy.Timer(rospy.Duration(1.0/rate), self.link_state)
self.path_timer = rospy.Timer(rospy.Duration(40.0/rate), self.path_follow)
self.gui_timer = rospy.Timer(rospy.Duration(1/10.0), self.user_feedback)
2022-03-01 14:09:13 -04:00
def cutoff(self,value,ceiling):
"""
Takes in value and returns ceiling
if value > ceiling. Otherwise, it returns
value back
"""
# initilize sign
sign = 1
# check if value is negative
if value < 0.0:
sign = -1
# Cutoff value at ceiling
if (value > ceiling or value < -ceiling):
output = sign*ceiling
else:
output = value
return output
2022-03-18 11:50:09 -03:00
def euler_array(self,orientation):
2022-03-01 14:09:13 -04:00
"""
Takes in pose msg object and outputs array of euler angs:
2022-03-18 11:50:09 -03:00
eul[0] = Roll
eul[1] = Pitch
eul[2] = Yaw
2022-03-01 14:09:13 -04:00
"""
2022-03-18 11:50:09 -03:00
eul = euler_from_quaternion([orientation.x,
orientation.y,
orientation.z,
orientation.w])
2022-03-18 11:50:09 -03:00
return eul
2022-03-01 14:09:13 -04:00
# Get link states (drone and pload) and determine angle between them
def link_state(self,pub_timer):
try:
# State which service we are querying
get_P = rospy.ServiceProxy(self.service1,GetLinkState)
# Set reference frame
reference = ''
# Establish links needed --> Spiri base and payload
# P = Position vector
2022-04-05 16:08:28 -03:00
drone_P = get_P(self.drone_id,reference)
2022-03-01 14:09:13 -04:00
2022-03-18 11:50:09 -03:00
# Get orientation of drone in euler angles to determine yaw offset
self.drone_Eul = self.euler_array(drone_P.link_state.pose.orientation)
2022-03-18 11:50:09 -03:00
2022-03-01 14:09:13 -04:00
# Check if payload is part of simulation
if not drone_P.success:
2022-04-05 16:08:28 -03:00
self.drone_id = 'spiri::base'
drone_P = get_P(self.drone_id,reference) # i.e. no payload
2022-03-01 14:09:13 -04:00
self.pload = False
2022-04-05 16:08:28 -03:00
pload_P = get_P(self.pload_id,reference)
2022-03-01 14:09:13 -04:00
if not self.has_run == 1:
if self.pload == True:
2022-03-18 11:50:09 -03:00
# Determine yaw offset
self.yaw_offset = self.drone_Eul[2]
2022-03-18 13:13:22 -03:00
2022-03-18 11:50:09 -03:00
2022-03-01 14:09:13 -04:00
# Get tether length based off initial displacement
self.tetherL = math.sqrt((drone_P.link_state.pose.position.x -
pload_P.link_state.pose.position.x)**2 +
(drone_P.link_state.pose.position.y -
pload_P.link_state.pose.position.y)**2 +
(drone_P.link_state.pose.position.z -
pload_P.link_state.pose.position.z)**2)
2022-08-18 15:29:26 -03:00
rospy.set_param('status/tether_length',self.tetherL)
2022-03-01 14:09:13 -04:00
else:
self.tetherL = 0
self.has_run = 1
# Need to detemine their location to get angle of deflection
# Drone
self.drone_Px = drone_P.link_state.pose.position.x
self.drone_Py = drone_P.link_state.pose.position.y
self.drone_Pz = drone_P.link_state.pose.position.z
2022-03-18 11:50:09 -03:00
2022-03-01 14:09:13 -04:00
if self.pload == True: # If there is payload, determine the variables
# Pload
pload_Px = pload_P.link_state.pose.position.x
pload_Py = pload_P.link_state.pose.position.y
# Determine theta (pitch)
x_sep = pload_Px - self.drone_Px
2022-03-01 14:09:13 -04:00
if math.fabs(x_sep) >= self.tetherL or x_sep == 0:
2022-04-05 16:08:28 -03:00
self.loadAngles.theta = 0
2022-03-01 14:09:13 -04:00
else:
2022-04-05 16:08:28 -03:00
self.loadAngles.theta = math.asin(x_sep/self.tetherL)
2022-03-01 14:09:13 -04:00
# Determine thetadot
2022-04-05 16:08:28 -03:00
self.loadAngles.thetadot = (self.loadAngles.theta - self.thetabuf)/self.dt
self.loadAngles.thetadot = self.cutoff(self.loadAngles.thetadot,self.thetadot_max)
self.thetabuf = self.loadAngles.theta
2022-03-01 14:09:13 -04:00
# Determine phi (roll)
y_sep = pload_Py - self.drone_Py
2022-03-01 14:09:13 -04:00
if math.fabs(y_sep) >= self.tetherL or y_sep == 0:
2022-04-05 16:08:28 -03:00
self.loadAngles.phi = 0
2022-03-01 14:09:13 -04:00
else:
self.loadAngles.phi = -math.asin(y_sep/self.tetherL)
2022-03-01 14:09:13 -04:00
# Determine phidot
2022-04-05 16:08:28 -03:00
self.loadAngles.phidot = (self.loadAngles.phi - self.phibuf)/self.dt
self.loadAngles.phidot = self.cutoff(self.loadAngles.phidot,self.phidot_max)
self.phibuf = self.loadAngles.phi # Update buffer
2022-03-01 14:09:13 -04:00
else: # Otherwise, vars = 0
2022-04-05 16:08:28 -03:00
x_sep = self.loadAngles.phi = self.loadAngles.phidot = self.loadAngles.theta = self.thetadot = 0
2022-03-01 14:09:13 -04:00
# Populate message
2022-03-25 16:31:55 -03:00
self.status.header.stamp = rospy.Time.now()
2022-03-01 14:09:13 -04:00
self.status.drone_pos = drone_P.link_state.pose
self.status.pload_pos = pload_P.link_state.pose
2022-04-05 16:08:28 -03:00
self.loadAngles.header.stamp = rospy.Time.now()
2022-03-01 14:09:13 -04:00
# Publish message
2022-04-05 16:08:28 -03:00
self.twobody_pub.publish(self.status)
self.loadAng_pub.publish(self.loadAngles)
2022-03-01 14:09:13 -04:00
except rospy.ServiceException as e:
rospy.loginfo("Get Link State call failed: {0}".format(e))
def user_feedback(self,gui_timer):
# Print and save results
2022-04-18 10:10:17 -03:00
#print "\n"
rospy.loginfo("")
print"Roll: "+str(round(self.drone_Eul[0]*180/3.14,2))
print"Pitch: " +str(round(self.drone_Eul[1]*180/3.14,2))
print"Yaw: " +str(round(self.drone_Eul[2]*180/3.14,2))
print "drone pos.x: " + str(round(self.drone_Px,2))
print "drone pos.y: " + str(round(self.drone_Py,2))
print "drone pos.z: " + str(round(self.drone_Pz,2))
print "phi: " + str(round(self.loadAngles.phi*180/3.14,3))
print "theta: " + str(round(self.loadAngles.theta*180/3.14,3))
2022-03-01 14:09:13 -04:00
def path_follow(self,path_timer):
now = rospy.get_time()
if now - self.tstart < self.wait:
self.bool = False
else:
self.bool = True
self.pub_wd.publish(self.bool)
if __name__=="__main__":
# Initiate ROS node
rospy.init_node('linkStates_node',anonymous=False)
try:
Main() # create class object
rospy.spin() # loop until shutdown signal
except rospy.ROSInterruptException:
pass