赞
踩
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,成功
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。