ROS - это структура для простого объединения различных библиотек, которые предоставляют интерфейсы, не определенные как файлы заголовков, а как «узлы», называемые «файлами запуска» (сценарии xml). Это подразумевает, что вы хотите одновременно запустить Yolo для канала видео / камеры, но вы хотите, чтобы он взаимодействовал с другими библиотеками или кодом. Если нет, то вам не нужна ROS.
ROS (v1), прямо сейчас, лучше всего работает на Ubuntu. Работает как изначально, так и в virtualbox. ROS2 поддерживает Windows, но если у вас есть проблемы с этим, это другой вопрос.
Чтобы создать узел ROS для вашего кода Python, сначала поместите все это в отдельный класс / модуль;узел ROS должен иметь только интерфейсный шаблон между «реальным кодом» и связью. Предполагая, что вы используете usb_cam_node
для получения канала камеры, данные будут опубликованы в «теме» <camera_name>/image
[sensor_msgs/Image
], где <camera_name>
-usb_cam_node
параметр. Тема похожа на глобальную переменную между всеми узлами ROS, прочитанную с помощью Subscriber
(с обратным вызовом) и опубликованную с помощью Publisher
.
Затем вы должны решить, что из нее публиковать. Поскольку это йоло, возможно, вам нужны ограничивающие рамки для каждого обнаружения. Существует множество предопределенных сообщений ROS (это «статические» и «строго типизированные» «типы» «тем»). Одним из них является geometry_msgs/PolygonStamped
, что позволяет вам указать углы коробки и поставить штамп.
Вот пример кода, взятого из wiki
# yolo_boxes_node.py
import rospy
from std_msgs.msg import Header
from geometry_msgs.msg import PolygonStamped, Point32
from sensor_msgs.msg import Image
# This is your custom yolo code
import my_yolo as yolo # Assuming a method like as follows:
# yolo.evaluate(img_frame) ->
# boxes ([x,y,width,hight] list), confidences (list), classids (list)
# Subscribers
# img_sub (sensor_msgs/Image): "webcam/image" #Comment: we document a name for the sub, the type, and the default topic for it
# Publishers
# boxes_pub (geometry_msgs/PolygonStamped): "webcam/yolo/boxes"
# Publishers
boxes_pub = None
# Parameters
frequency = 100.0 # Hz
# Global Variables
img_frame = None
header = None
def img_callback(data): # data of type Image
global img_frame
global header
img_frame = data.data
header = data.header
def timer_callback(event): # This is to process data at a fixed rate, perhaps different from camera framerate
# Convert img_frame somehow if needed
if img_frame is None or boxes_pub is None:
return
boxes, confidences, classids = yolo.evaluate(img_frame)
for b in boxes:
msg = PolygonStamped()
msg.header = header # You could use the header differently
msg.polygon.points.append(Point32(x=b[0],y=b[1]))
msg.polygon.points.append(Point32(x=b[0]+b[2],y=b[1]))
msg.polygon.points.append(Point32(x=b[0],y=b[1]+b[3]))
msg.polygon.points.append(Point32(x=b[0]+b[2],y=b[1]+b[3]))
boxes_pub.publish(msg)
# In your main function, you subscribe to topics
def yolo_boxes_node():
# Init ROS
rospy.init_node('yolo_boxes_node', anonymous=True)
# Parameters
if rospy.has_param('~frequency'):
frequency = rospy.get_param('~frequency')
# Subscribers
# Each subscriber has the topic, topic type, AND the callback!
rospy.Subscriber('webcam/image', Image, img_callback)
# Rarely/never need to hold onto the object with a variable:
# img_sub = rospy.Subscriber(...)
rospy.Timer(rospy.Duration(1.0/frequency), timer_callback)
# Publishers
boxes_pub = rospy.Publisher('webcam/yolo/boxes', PolygonStamped, queue_size = 100)
# queue_size increases as buffer for msgs; if you have 1000s of boxes, might need bigger
# spin() simply keeps python from exiting until this node is stopped
# This is an infinite loop, the only code that gets ran are callbacks
rospy.spin()
# NO CODE GOES AFTER THIS, NONE! USE TIMER CALLBACKS!
# unless you need to clean up resource allocation, close(), etc when program dies
if __name__ == '__main__':
yolo_boxes_node()
Ergo пример файла запуска XML может быть:
<?xml version="1.0"?>
<!-- my_main_program.launch -->
<launch>
<!--
Pub: <camera_name>/image [sensor_msgs/Image]
-->
<node name="usb_cam_node" type="usb_cam_node" pkg="usb_cam" output="screen" restart="true">
<param name="camera_name" value="webcam"/>
<param name="video_device" value="/dev/video0"/>
</node>
<!--
img_sub: webcam/image [sensor_msgs/Image]
boxes_pub: webcam/yolo/boxes [geometry_msgs/PolygonStamped]
-->
<node name="yolo_boxes_node" type="yolo_boxes_node" pkg="my_pkg" output="screen">
<param name="frequency" value="30.0"/>
</node>
</launch>