Skip to content

Commit

Permalink
finially the ROS2 version bodypix all good!
Browse files Browse the repository at this point in the history
  • Loading branch information
Sveali41 committed Nov 19, 2024
1 parent 04ce569 commit ca73bce
Show file tree
Hide file tree
Showing 12 changed files with 303 additions and 143 deletions.
4 changes: 4 additions & 0 deletions common/vision/lasr_vision_bodypix/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,10 @@ install(PROGRAMS
DESTINATION lib/${PROJECT_NAME}
)

install(PROGRAMS
examples/mask_relay.py
DESTINATION lib/${PROJECT_NAME}
)

# Install launch files
install(DIRECTORY launch
Expand Down
13 changes: 9 additions & 4 deletions common/vision/lasr_vision_bodypix/examples/keypoint_relay.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@
import threading

from sensor_msgs.msg import Image
from lasr_vision_msgs.srv import BodyPixKeypointDetection, BodyPixKeypointDetectionRequest
from lasr_vision_msgs.srv import BodyPixKeypointDetection

class KeypointRelay(Node):
def __init__(self, listen_topic, model):
Expand All @@ -31,9 +31,8 @@ def __init__(self, listen_topic, model):
def detect(self, image):
self.processing = True
self.get_logger().info("Received image message")

# Create a request for the service
req = BodyPixKeypointDetectionRequest()
req = BodyPixKeypointDetection.Request()
req.image_raw = image
req.dataset = self.model
req.confidence = 0.7
Expand All @@ -60,17 +59,23 @@ def image_callback(self, image):


def main(args=None):
print("start keypoint_relay")
# Check if command-line arguments are sufficient
if len(sys.argv) < 2:
print("Usage: ros2 run lasr_vision_bodypix keypoint_relay.py <source_topic> [resnet50|mobilenet50|...]")
sys.exit(1)

# Parse the command-line arguments
listen_topic = sys.argv[1]

listen_topic = '/image_raw'
if isinstance(sys.argv[1], list):
listen_topic = sys.argv[1][0]

model = sys.argv[2] if len(sys.argv) >= 3 else "resnet50"

rclpy.init(args=args)
keypoint_relay_node = KeypointRelay(listen_topic, model)
keypoint_relay_node.get_logger().info("Keypoint relay node started")

try:
rclpy.spin(keypoint_relay_node)
Expand Down
97 changes: 97 additions & 0 deletions common/vision/lasr_vision_bodypix/examples/mask_relay.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,97 @@
#!/usr/bin/env python3

import sys
import threading
import rclpy
from rclpy.node import Node
from sensor_msgs.msg import Image
from lasr_vision_msgs.srv import BodyPixMaskDetection


class ImageListener(Node):
def __init__(self, listen_topic, model):
super().__init__('image_listener')
self.listen_topic = listen_topic
self.model = model
self.processing = False

# Set up the service client
self.detect_service_client = self.create_client(BodyPixMaskDetection, '/bodypix/mask_detection')
while not self.detect_service_client.wait_for_service(timeout_sec=1.0):
self.get_logger().info('Service not available, waiting...')

# Set up the subscriber
self.subscription = self.create_subscription(
Image,
self.listen_topic,
self.image_callback,
10 # QoS profile
)
self.get_logger().info(f'Started listening on topic: {self.listen_topic} with model: {self.model}')

def detect(self, image):
self.processing = True
self.get_logger().info("Received image message")
# Create a request for the service
req = BodyPixMaskDetection.Request()
req.image_raw = image
req.dataset = self.model
req.confidence = 0.7
req.parts = ["left_face", "right_face"]

# Call the service asynchronously
future = self.detect_service_client.call_async(req)
future.add_done_callback(self.detect_callback)

def detect_callback(self, future):
try:
response = future.result()
if response is not None:
# Modify masks for demonstration purposes
for mask in response.masks:
mask.mask = [True, False, True, False]
self.get_logger().info(f"Detection response received: {response}")
else:
self.get_logger().error("Service call returned no response")
except Exception as e:
self.get_logger().error(f"Service call failed: {e}")
finally:
self.processing = False

def image_callback(self, image):
if self.processing:
return

# Start a new thread for detection to avoid blocking
threading.Thread(target=self.detect, args=(image,)).start()


def main(args=None):
print("Starting mask_relay node")
# Check if command-line arguments are sufficient
if len(sys.argv) < 2:
print("Usage: ros2 run lasr_vision_bodypix mask_relay.py <source_topic> [resnet50|mobilenet50|...]")
sys.exit(1)

# Parse the command-line arguments
listen_topic = '/image_raw'
if isinstance(sys.argv[1], list):
listen_topic = sys.argv[1][0]

model = sys.argv[2] if len(sys.argv) >= 3 else "resnet50"

rclpy.init(args=args)
mask_relay_node = ImageListener(listen_topic, model)
mask_relay_node.get_logger().info("Mask relay node started")

try:
rclpy.spin(mask_relay_node)
except KeyboardInterrupt:
pass
finally:
mask_relay_node.destroy_node()
rclpy.shutdown()


if __name__ == "__main__":
main()
9 changes: 6 additions & 3 deletions common/vision/lasr_vision_bodypix/launch/bodypix_launch.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,11 +4,14 @@
from launch.substitutions import LaunchConfiguration

def generate_launch_description():
# Declare preload argument with default value
# Declare preload argument with default value as a YAML list
preload_arg = DeclareLaunchArgument(
'preload', default_value="['resnet50']", description='Array of models to preload when starting the service'
'preload',
default_value=['resnet50'],
# default_value="['resnet50', 'mobilenet50']",
description='Array of models to preload when starting the service'
)

# Create the BodyPix service node
bodypix_node = Node(
package='lasr_vision_bodypix',
Expand Down
35 changes: 30 additions & 5 deletions common/vision/lasr_vision_bodypix/launch/camera_keypoint_launch.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,16 +2,24 @@
from launch.actions import DeclareLaunchArgument, IncludeLaunchDescription
from launch_ros.actions import Node
from launch.launch_description_sources import PythonLaunchDescriptionSource
from launch.substitutions import LaunchConfiguration, TextSubstitution
from launch.substitutions import LaunchConfiguration, TextSubstitution, PythonExpression

import os
from ament_index_python.packages import get_package_share_directory

def generate_launch_description():
# Declare launch arguments
model_arg = DeclareLaunchArgument(
'model', default_value='resnet50', description='Model to use for the demo'
'model', default_value="['resnet50']", description='Model to use for the demo'
)

image_topic_arg = DeclareLaunchArgument(
'image_topic',
default_value='/image_raw', # Default input image topic
description='Input image topic for keypoint relay'
)


# Path to the BodyPix launch file
bodypix_launch_file = os.path.join(
get_package_share_directory('lasr_vision_bodypix'), 'launch', 'bodypix_launch.py'
Expand Down Expand Up @@ -45,8 +53,15 @@ def generate_launch_description():
name='image_view',
output='screen',
arguments=[
TextSubstitution(text='/bodypix/debug/'),
LaunchConfiguration('model')

# [TextSubstitution(text='/bodypix/debug/'), LaunchConfiguration('model')]# Topic to visualize
PythonExpression([
"'/bodypix/debug/' + ", # Static string
"''.join(", # Convert list to string
LaunchConfiguration('model'),
")"
])
# '/bodypix/debug/{}'.format(LaunchConfiguration('resnet50'))
], # Constructs the topic path dynamically
),

Expand All @@ -56,7 +71,17 @@ def generate_launch_description():
executable='keypoint_relay.py', # Specifying the subdirectory
name='keypoint_relay',
output='screen',
arguments=[LaunchConfiguration('model')],
arguments=[
# TextSubstitution(text='/bodypix/debug/'),
# LaunchConfiguration('model')
# '/camera/image_raw /{}'.format(LaunchConfiguration('model'))
'/image_raw ',
PythonExpression([
"''.join(", # Convert model list to a single string
LaunchConfiguration('model'),
")"
])
], # Constructs the topic path dynamically],
),


Expand Down
35 changes: 27 additions & 8 deletions common/vision/lasr_vision_bodypix/launch/camera_mask_launch.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,14 +2,20 @@
from launch.actions import DeclareLaunchArgument, IncludeLaunchDescription
from launch_ros.actions import Node
from launch.launch_description_sources import PythonLaunchDescriptionSource
from launch.substitutions import LaunchConfiguration, TextSubstitution
from launch.substitutions import LaunchConfiguration, TextSubstitution, PythonExpression
import os
from ament_index_python.packages import get_package_share_directory

def generate_launch_description():
# Declare launch arguments
model_arg = DeclareLaunchArgument(
'model', default_value='resnet50', description='Model to use for the demo'
'model', default_value="['resnet50']", description='Model to use for the demo'
)

image_topic_arg = DeclareLaunchArgument(
'image_topic',
default_value='/image_raw', # Default input image topic
description='Input image topic for keypoint relay'
)

# Path to the BodyPix launch file
Expand All @@ -19,7 +25,7 @@ def generate_launch_description():

# Path to the v4l2_camera launch file (replacement for video_stream_opencv)
v4l2_camera_launch_file = os.path.join(
get_package_share_directory('v4l2_camera'), 'launch', 'v4l2_camera_node.launch.py'
get_package_share_directory('lasr_vision_bodypix'), 'launch', 'v4l2_camera_launch.py'
)

return LaunchDescription([
Expand All @@ -41,18 +47,31 @@ def generate_launch_description():
name='image_view',
output='screen',
arguments=[
TextSubstitution(text='/bodypix/debug/'),
LaunchConfiguration('model')
# [TextSubstitution(text='/bodypix/debug/'), LaunchConfiguration('model')]# Topic to visualize
PythonExpression([
"'/bodypix/debug/' + ", # Static string
"''.join(", # Convert list to string
LaunchConfiguration('model'),
")"
])
# '/bodypix/debug/{}'.format(LaunchConfiguration('resnet50'))
], # Constructs the topic path dynamically
),

# Start the keypoint relay service
Node(
package='lasr_vision_bodypix',
executable='keypoint_relay.py', # Removed .py extension, assuming installed without it
name='keypoint_relay',
executable='mask_relay.py', # Removed .py extension, assuming installed without it
name='mask_relay',
output='screen',
arguments=[LaunchConfiguration('model')],
arguments=[
'/image_raw ',
PythonExpression([
"''.join(", # Convert model list to a single string
LaunchConfiguration('model'),
")"
])
], # Constructs the topic path dynamically],],
),

# Include the v4l2_camera launch file
Expand Down
10 changes: 5 additions & 5 deletions common/vision/lasr_vision_bodypix/launch/v4l2_camera_launch.py
Original file line number Diff line number Diff line change
Expand Up @@ -9,11 +9,11 @@ def generate_launch_description():
name='v4l2_camera_node',
output='screen',
parameters=[{
'video_device': '/dev/video0', # 摄像头设备路径
'image_width': 640, # 图像宽度
'image_height': 480, # 图像高度
'pixel_format': 'YUYV', # 像素格式
'frame_rate': 30, # 帧率
'video_device': '/dev/video0', # the video device to open
'image_width': 640,
'image_height': 480,
'pixel_format': 'YUYV', # the pixel format of the image
'frame_rate': 30,
}],
),
])
6 changes: 5 additions & 1 deletion common/vision/lasr_vision_bodypix/nodes/bodypix_services.py
Original file line number Diff line number Diff line change
Expand Up @@ -15,15 +15,18 @@
import numpy as np
from cv2_pcl import pcl_to_img_msg
from typing import Union
from rcl_interfaces.msg import ParameterDescriptor, ParameterType

class BodyPixServiceNode(Node):
def __init__(self):
super().__init__("bodypix_service_node")

# Declare and load parameters
self.declare_parameter('preload', [])
# self.declare_parameter('preload', [])
self.declare_parameter('preload', [''], ParameterDescriptor(type=ParameterType.PARAMETER_STRING_ARRAY))
preload_models = self.get_parameter('preload').get_parameter_value().string_array_value


# Preload models
for model in preload_models:
bodypix.load_model_cached(model)
Expand All @@ -35,6 +38,7 @@ def __init__(self):
self.keypoint_service = self.create_service(
BodyPixKeypointDetection, '/bodypix/keypoint_detection', self.detect_keypoints
)
self.get_logger().info("Keypoint detection service registered.")
self.detect_wave_service = self.create_service(
DetectWave, '/bodypix/detect_wave', self.detect_wave
)
Expand Down
1 change: 1 addition & 0 deletions common/vision/lasr_vision_bodypix/setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@
'bodypix_service_node = lasr_vision_bodypix.nodes.bodypix_services:main',
'bodypix_node = lasr_vision_bodypix.bodypix:main',
'keypoint_relay = lasr_vision_bodypix.examples.keypoint_relay:main',
'mask_relay = lasr_vision_bodypix.examples.mask_relay:main'
],
}
)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,5 +2,7 @@
snake_to_camel,
camel_to_snake,
load_model_cached,
BodyPixNode,
run_inference,
detect_masks,
detect_keypoints,
)
Loading

0 comments on commit ca73bce

Please sign in to comment.