ROS2使用usb_cam

发布时间:2024年01月24日

1、安装usb_cam

sudo apt-get install ros-humble-usb-cam

2、启动usb_cam

ros2 launch usb_cam camera.launch.py 

但是这里会出现错误

Caught exception in launch (see debug for traceback): Caught exception when trying to load file of format [py]: No module named 'pydantic'

3、下载pydantic库

pip install pydantic  -i https://pypi.tuna.tsinghua.edu.cn/simple some-package

4、下载完成后继续启动

ros2 launch usb_cam camera.launch.py 

Pydantic 版本要求 @root_validator 使用时必须指定 skip_on_failure=True,因为默认的 pre=False 设定可能会导致验证失败。

5、解决

找到/opt//ros/humble/share/usb_cam/launch下的 camera_config.py,并使用vim打开,并修改里面的配置文件

  • 移除 @root_validator 避免 Pydantic 2.5 版本中 @root_validator 使用上的限制和不兼容问题。

原来的代码

from pathlib import Path
from typing import List, Optional

from ament_index_python.packages import get_package_share_directory
from pydantic import BaseModel, root_validator, validator

USB_CAM_DIR = get_package_share_directory('usb_cam')


class CameraConfig(BaseModel):
    name: str = 'camera1'
    param_path: Path = Path(USB_CAM_DIR, 'config', 'params_1.yaml')
    remappings: Optional[List]
    namespace: Optional[str]

    @validator('param_path')
    def validate_param_path(cls, value):
        if value and not value.exists():
            raise FileNotFoundError(f'Could not find parameter file: {value}')
        return value

    @root_validator
    def validate_root(cls, values):
        name = values.get('name')
        remappings = values.get('remappings')
        if name and not remappings:
            # Automatically set remappings if name is set
            remappings = [
                ('image_raw', f'{name}/image_raw'),
                ('image_raw/compressed', f'{name}/image_compressed'),
                ('image_raw/compressedDepth', f'{name}/compressedDepth'),
                ('image_raw/theora', f'{name}/image_raw/theora'),
                ('camera_info', f'{name}/camera_info'),
            ]
        values['remappings'] = remappings
        return values       

修改后的

# Copyright 2023 usb_cam Authors
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are met:
#
#    * Redistributions of source code must retain the above copyright
#      notice, this list of conditions and the following disclaimer.
#
#    * Redistributions in binary form must reproduce the above copyright
#      notice, this list of conditions and the following disclaimer in the
#      documentation and/or other materials provided with the distribution.
#
#    * Neither the name of the usb_cam Authors nor the names of its
#      contributors may be used to endorse or promote products derived from
#      this software without specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
# POSSIBILITY OF SUCH DAMAGE.


from pathlib import Path
from typing import List, Optional

from ament_index_python.packages import get_package_share_directory
#from pydantic import BaseModel, root_validator, validator
from pydantic import BaseModel, model_validator, validator

USB_CAM_DIR = get_package_share_directory('usb_cam')


class CameraConfig(BaseModel):
    name: str = 'camera1'
    param_path: Path = Path(USB_CAM_DIR, 'config', 'params_1.yaml')
    remappings: Optional[List[str]] = []
    namespace: Optional[str] = None

    @validator('param_path')
    def validate_param_path(cls, value):
        if value and not value.exists():
            raise FileNotFoundError(f'Could not find parameter file: {value}')
        return value

#    @root_validator
#   def validate_root(cls, values):
#        name = values.get('name')
#        remappings = values.get('remappings')
#        if name and not remappings:
#            # Automatically set remappings if name is set
#            remappings = [
#                ('image_raw', f'{name}/image_raw'),
#                ('image_raw/compressed', f'{name}/image_compressed'),
#                ('image_raw/compressedDepth', f'{name}/compressedDepth'),
#               ('image_raw/theora', f'{name}/image_raw/theora'),
#               ('camera_info', f'{name}/camera_info'),
#            ]
#        values['remappings'] = remappings
#        return value
    @validator('name')
    def validate_name(cls, value):
        if not value:
            raise ValueError("Name is required")
        return value

    def __init__(self, **data):
        super().__init__(**data)
        if self.name:
            # Automatically set remappings if name is set
            self.remappings.extend([
                ('image_raw', f'{self.name}/image_raw'),
                ('image_raw/compressed', f'{self.name}/image_compressed'),
                ('image_raw/compressedDepth', f'{self.name}/compressedDepth'),
                ('image_raw/theora', f'{self.name}/image_raw/theora'),
                ('camera_info', f'{self.name}/camera_info'),
            ])

5、继续启动

OK,成功

文章来源:https://blog.csdn.net/weixin_64037619/article/details/135744098
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。