forked from OSLL/template-ros-core
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
1 changed file
with
180 additions
and
0 deletions.
There are no files selected for viewing
180 changes: 180 additions & 0 deletions
180
packages/dt-core/packages/apriltag/src/apriltag_postprocessing_node.py
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,180 @@ | ||
#!/usr/bin/env python3 | ||
import rospkg | ||
import rospy | ||
import yaml | ||
from duckietown_msgs.msg import \ | ||
AprilTagsWithInfos, \ | ||
TagInfo, \ | ||
BoolStamped, \ | ||
AprilTagDetection, \ | ||
AprilTagDetectionArray | ||
import numpy as np | ||
import tf.transformations as tr | ||
from geometry_msgs.msg import PoseStamped, Pose | ||
from std_msgs.msg import Header | ||
|
||
|
||
class AprilPostPros(object): | ||
""" """ | ||
def __init__(self): | ||
""" """ | ||
self.node_name = "apriltag_postprocessing_node" | ||
|
||
# Load parameters | ||
self.camera_x = self.setupParam("~camera_x", 0.065) | ||
self.camera_y = self.setupParam("~camera_y", 0.0) | ||
self.camera_z = self.setupParam("~camera_z", 0.11) | ||
self.camera_theta = self.setupParam("~camera_theta", 19.0) | ||
self.scale_x = self.setupParam("~scale_x", 1) | ||
self.scale_y = self.setupParam("~scale_y", 1) | ||
self.scale_z = self.setupParam("~scale_z", 1) | ||
|
||
# -------- Start adding back the tag info stuff | ||
|
||
tags_filepath = self.setupParam("~tags_file") | ||
|
||
self.loc = self.setupParam("~loc", -1) # -1 if no location is given | ||
tags_file = open(tags_filepath, 'r') | ||
self.tags_dict = yaml.safe_load(tags_file) | ||
tags_file.close() | ||
self.info = TagInfo() | ||
|
||
self.sign_types = {"StreetName": self.info.S_NAME, | ||
"TrafficSign": self.info.SIGN, | ||
"Light": self.info.LIGHT, | ||
"Localization": self.info.LOCALIZE, | ||
"Vehicle": self.info.VEHICLE} | ||
self.traffic_sign_types = {"stop": self.info.STOP, | ||
"yield": self.info.YIELD, | ||
"no-right-turn": self.info.NO_RIGHT_TURN, | ||
"no-left-turn": self.info.NO_LEFT_TURN, | ||
"oneway-right": self.info.ONEWAY_RIGHT, | ||
"oneway-left": self.info.ONEWAY_LEFT, | ||
"4-way-intersect": self.info.FOUR_WAY, | ||
"right-T-intersect": self.info.RIGHT_T_INTERSECT, | ||
"left-T-intersect": self.info.LEFT_T_INTERSECT, | ||
"T-intersection": self.info.T_INTERSECTION, | ||
"do-not-enter": self.info.DO_NOT_ENTER, | ||
"pedestrian": self.info.PEDESTRIAN, | ||
"t-light-ahead": self.info.T_LIGHT_AHEAD, | ||
"duck-crossing": self.info.DUCK_CROSSING, | ||
"parking": self.info.PARKING} | ||
|
||
|
||
# ---- end tag info stuff | ||
|
||
|
||
|
||
self.sub_prePros = rospy.Subscriber("~detections", AprilTagDetectionArray, self.callback, queue_size=1) | ||
self.pub_postPros = rospy.Publisher("~apriltags_out", AprilTagsWithInfos, queue_size=1) | ||
self.pub_visualize = rospy.Publisher("~tag_pose", PoseStamped, queue_size=1) | ||
|
||
# topics for state machine | ||
self.pub_postPros_parking = rospy.Publisher("~apriltags_parking", BoolStamped, queue_size=1) | ||
self.pub_postPros_intersection = rospy.Publisher("~apriltags_intersection", BoolStamped, queue_size=1) | ||
|
||
rospy.loginfo("[%s] has started", self.node_name) | ||
|
||
def setupParam(self, param_name, default_value=rospy.client._Unspecified): | ||
value = rospy.get_param(param_name, default_value) | ||
rospy.set_param(param_name, value) | ||
return value | ||
|
||
def callback(self, msg): | ||
|
||
tag_infos = [] | ||
|
||
new_tag_data = AprilTagsWithInfos() | ||
|
||
# Load tag detections message | ||
for detection in msg.detections: | ||
|
||
# ------ start tag info processing | ||
rospy.loginfo("detection.tag_id = %s", str(detection.tag_id)) | ||
new_info = TagInfo() | ||
#Can use id 1 as long as no bundles are used | ||
new_info.id = int(detection.tag_id) | ||
id_info = self.tags_dict[new_info.id] | ||
#rospy.loginfo("[%s] report detected id=[%s] for april tag!",self.node_name,new_info.id) | ||
# Check yaml file to fill in ID-specific information | ||
new_info.tag_type = self.sign_types[id_info['tag_type']] | ||
if new_info.tag_type == self.info.S_NAME: | ||
new_info.street_name = id_info['street_name'] | ||
elif new_info.tag_type == self.info.SIGN: | ||
new_info.traffic_sign_type = self.traffic_sign_types[id_info['traffic_sign_type']] | ||
|
||
# publish for FSM | ||
# parking apriltag event | ||
msg_parking = BoolStamped() | ||
msg_parking.header.stamp = rospy.Time(0) | ||
if new_info.traffic_sign_type == TagInfo.PARKING: | ||
msg_parking.data = True | ||
else: | ||
msg_parking.data = False | ||
self.pub_postPros_parking.publish(msg_parking) | ||
|
||
# intersection apriltag event | ||
msg_intersection = BoolStamped() | ||
msg_intersection.header.stamp = rospy.Time(0) | ||
if (new_info.traffic_sign_type == TagInfo.FOUR_WAY) or (new_info.traffic_sign_type == TagInfo.RIGHT_T_INTERSECT) or (new_info.traffic_sign_type == TagInfo.LEFT_T_INTERSECT) or (new_info.traffic_sign_type == TagInfo.T_INTERSECTION): | ||
msg_intersection.data = True | ||
else: | ||
msg_intersection.data = False | ||
self.pub_postPros_intersection.publish(msg_intersection) | ||
|
||
elif new_info.tag_type == self.info.VEHICLE: | ||
new_info.vehicle_name = id_info['vehicle_name'] | ||
|
||
# TODO: Implement location more than just a float like it is now. | ||
# location is now 0.0 if no location is set which is probably not that smart | ||
if self.loc == 226: | ||
l = (id_info['location_226']) | ||
if l is not None: | ||
new_info.location = l | ||
elif self.loc == 316: | ||
l = (id_info['location_316']) | ||
if l is not None: | ||
new_info.location = l | ||
|
||
tag_infos.append(new_info) | ||
# --- end tag info processing | ||
|
||
|
||
# Define the transforms | ||
veh_t_camxout = tr.translation_matrix((self.camera_x, self.camera_y, self.camera_z)) | ||
veh_R_camxout = tr.euler_matrix(0, self.camera_theta*np.pi/180, 0, 'rxyz') | ||
veh_T_camxout = tr.concatenate_matrices(veh_t_camxout, veh_R_camxout) # 4x4 Homogeneous Transform Matrix | ||
|
||
camxout_T_camzout = tr.euler_matrix(-np.pi/2,0,-np.pi/2,'rzyx') | ||
veh_T_camzout = tr.concatenate_matrices(veh_T_camxout, camxout_T_camzout) | ||
|
||
tagzout_T_tagxout = tr.euler_matrix(-np.pi/2, 0, np.pi/2, 'rxyz') | ||
|
||
#Load translation | ||
trans = detection.transform.translation | ||
rot = detection.transform.rotation | ||
header = Header() | ||
header.stamp = rospy.Time.now() | ||
camzout_t_tagzout = tr.translation_matrix((trans.x*self.scale_x, trans.y*self.scale_y, trans.z*self.scale_z)) | ||
camzout_R_tagzout = tr.quaternion_matrix((rot.x, rot.y, rot.z, rot.w)) | ||
camzout_T_tagzout = tr.concatenate_matrices(camzout_t_tagzout, camzout_R_tagzout) | ||
|
||
veh_T_tagxout = tr.concatenate_matrices(veh_T_camzout, camzout_T_tagzout, tagzout_T_tagxout) | ||
|
||
# Overwrite transformed value | ||
(trans.x, trans.y, trans.z) = tr.translation_from_matrix(veh_T_tagxout) | ||
(rot.x, rot.y, rot.z, rot.w) = tr.quaternion_from_matrix(veh_T_tagxout) | ||
|
||
new_tag_data.detections.append( | ||
detection | ||
) | ||
|
||
new_tag_data.infos = tag_infos | ||
# Publish Message | ||
self.pub_postPros.publish(new_tag_data) | ||
|
||
|
||
if __name__ == '__main__': | ||
rospy.init_node('AprilPostPros', anonymous=False) | ||
node = AprilPostPros() | ||
rospy.spin() |