mirror of
https://github.com/huggingface/lerobot.git
synced 2026-05-22 12:09:42 +00:00
bc38261321
* feat(robots): use read_latest() camera * fix(test): add read_latest reachy cam mock
593 lines
23 KiB
Python
593 lines
23 KiB
Python
# Copyright 2024 The HuggingFace Inc. team. All rights reserved.
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
# you may not use this file except in compliance with the License.
|
|
# You may obtain a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
# See the License for the specific language governing permissions and
|
|
# limitations under the License.
|
|
|
|
"""
|
|
Provides the OpenCVCamera class for capturing frames from cameras using OpenCV.
|
|
"""
|
|
|
|
import logging
|
|
import math
|
|
import os
|
|
import platform
|
|
import time
|
|
from pathlib import Path
|
|
from threading import Event, Lock, Thread
|
|
from typing import Any
|
|
|
|
from numpy.typing import NDArray # type: ignore # TODO: add type stubs for numpy.typing
|
|
|
|
# Fix MSMF hardware transform compatibility for Windows before importing cv2
|
|
if platform.system() == "Windows" and "OPENCV_VIDEOIO_MSMF_ENABLE_HW_TRANSFORMS" not in os.environ:
|
|
os.environ["OPENCV_VIDEOIO_MSMF_ENABLE_HW_TRANSFORMS"] = "0"
|
|
import cv2 # type: ignore # TODO: add type stubs for OpenCV
|
|
|
|
from lerobot.utils.decorators import check_if_already_connected, check_if_not_connected
|
|
from lerobot.utils.errors import DeviceNotConnectedError
|
|
|
|
from ..camera import Camera
|
|
from ..utils import get_cv2_rotation
|
|
from .configuration_opencv import ColorMode, OpenCVCameraConfig
|
|
|
|
# NOTE(Steven): The maximum opencv device index depends on your operating system. For instance,
|
|
# if you have 3 cameras, they should be associated to index 0, 1, and 2. This is the case
|
|
# on MacOS. However, on Ubuntu, the indices are different like 6, 16, 23.
|
|
# When you change the USB port or reboot the computer, the operating system might
|
|
# treat the same cameras as new devices. Thus we select a higher bound to search indices.
|
|
MAX_OPENCV_INDEX = 60
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
class OpenCVCamera(Camera):
|
|
"""
|
|
Manages camera interactions using OpenCV for efficient frame recording.
|
|
|
|
This class provides a high-level interface to connect to, configure, and read
|
|
frames from cameras compatible with OpenCV's VideoCapture. It supports both
|
|
synchronous and asynchronous frame reading.
|
|
|
|
An OpenCVCamera instance requires a camera index (e.g., 0) or a device path
|
|
(e.g., '/dev/video0' on Linux). Camera indices can be unstable across reboots
|
|
or port changes, especially on Linux. Use the provided utility script to find
|
|
available camera indices or paths:
|
|
```bash
|
|
lerobot-find-cameras opencv
|
|
```
|
|
|
|
The camera's default settings (FPS, resolution, color mode) are used unless
|
|
overridden in the configuration.
|
|
|
|
Example:
|
|
```python
|
|
from lerobot.cameras.opencv import OpenCVCamera
|
|
from lerobot.cameras.configuration_opencv import OpenCVCameraConfig
|
|
|
|
# Basic usage with camera index 0
|
|
config = OpenCVCameraConfig(index_or_path=0)
|
|
camera = OpenCVCamera(config)
|
|
camera.connect()
|
|
|
|
# Read 1 frame synchronously (blocking)
|
|
color_image = camera.read()
|
|
|
|
# Read 1 frame asynchronously (waits for new frame with a timeout)
|
|
async_image = camera.async_read()
|
|
|
|
# Get the latest frame immediately (no wait, returns timestamp)
|
|
latest_image, timestamp = camera.read_latest()
|
|
|
|
# When done, properly disconnect the camera using
|
|
camera.disconnect()
|
|
```
|
|
"""
|
|
|
|
def __init__(self, config: OpenCVCameraConfig):
|
|
"""
|
|
Initializes the OpenCVCamera instance.
|
|
|
|
Args:
|
|
config: The configuration settings for the camera.
|
|
"""
|
|
super().__init__(config)
|
|
|
|
self.config = config
|
|
self.index_or_path = config.index_or_path
|
|
|
|
self.fps = config.fps
|
|
self.color_mode = config.color_mode
|
|
self.warmup_s = config.warmup_s
|
|
|
|
self.videocapture: cv2.VideoCapture | None = None
|
|
|
|
self.thread: Thread | None = None
|
|
self.stop_event: Event | None = None
|
|
self.frame_lock: Lock = Lock()
|
|
self.latest_frame: NDArray[Any] | None = None
|
|
self.latest_timestamp: float | None = None
|
|
self.new_frame_event: Event = Event()
|
|
|
|
self.rotation: int | None = get_cv2_rotation(config.rotation)
|
|
self.backend: int = config.backend
|
|
|
|
if self.height and self.width:
|
|
self.capture_width, self.capture_height = self.width, self.height
|
|
if self.rotation in [cv2.ROTATE_90_CLOCKWISE, cv2.ROTATE_90_COUNTERCLOCKWISE]:
|
|
self.capture_width, self.capture_height = self.height, self.width
|
|
|
|
def __str__(self) -> str:
|
|
return f"{self.__class__.__name__}({self.index_or_path})"
|
|
|
|
@property
|
|
def is_connected(self) -> bool:
|
|
"""Checks if the camera is currently connected and opened."""
|
|
return isinstance(self.videocapture, cv2.VideoCapture) and self.videocapture.isOpened()
|
|
|
|
@check_if_already_connected
|
|
def connect(self, warmup: bool = True) -> None:
|
|
"""
|
|
Connects to the OpenCV camera specified in the configuration.
|
|
|
|
Initializes the OpenCV VideoCapture object, sets desired camera properties
|
|
(FPS, width, height), starts the background reading thread and performs initial checks.
|
|
|
|
Args:
|
|
warmup (bool): If True, waits at connect() time until at least one valid frame
|
|
has been captured by the background thread. Defaults to True.
|
|
|
|
Raises:
|
|
DeviceAlreadyConnectedError: If the camera is already connected.
|
|
ConnectionError: If the specified camera index/path is not found or fails to open.
|
|
RuntimeError: If the camera opens but fails to apply requested settings.
|
|
"""
|
|
|
|
# Use 1 thread for OpenCV operations to avoid potential conflicts or
|
|
# blocking in multi-threaded applications, especially during data collection.
|
|
cv2.setNumThreads(1)
|
|
|
|
self.videocapture = cv2.VideoCapture(self.index_or_path, self.backend)
|
|
|
|
if not self.videocapture.isOpened():
|
|
self.videocapture.release()
|
|
self.videocapture = None
|
|
raise ConnectionError(
|
|
f"Failed to open {self}.Run `lerobot-find-cameras opencv` to find available cameras."
|
|
)
|
|
|
|
self._configure_capture_settings()
|
|
self._start_read_thread()
|
|
|
|
if warmup and self.warmup_s > 0:
|
|
start_time = time.time()
|
|
while time.time() - start_time < self.warmup_s:
|
|
self.async_read(timeout_ms=self.warmup_s * 1000)
|
|
time.sleep(0.1)
|
|
with self.frame_lock:
|
|
if self.latest_frame is None:
|
|
raise ConnectionError(f"{self} failed to capture frames during warmup.")
|
|
|
|
logger.info(f"{self} connected.")
|
|
|
|
@check_if_not_connected
|
|
def _configure_capture_settings(self) -> None:
|
|
"""
|
|
Applies the specified FOURCC, FPS, width, and height settings to the connected camera.
|
|
|
|
This method attempts to set the camera properties via OpenCV. It checks if
|
|
the camera successfully applied the settings and raises an error if not.
|
|
FOURCC is set first (if specified) as it can affect the available FPS and resolution options.
|
|
|
|
Args:
|
|
fourcc: The desired FOURCC code (e.g., "MJPG", "YUYV"). If None, auto-detect.
|
|
fps: The desired frames per second. If None, the setting is skipped.
|
|
width: The desired capture width. If None, the setting is skipped.
|
|
height: The desired capture height. If None, the setting is skipped.
|
|
|
|
Raises:
|
|
RuntimeError: If the camera fails to set any of the specified properties
|
|
to the requested value.
|
|
DeviceNotConnectedError: If the camera is not connected.
|
|
"""
|
|
|
|
# Set FOURCC first (if specified) as it can affect available FPS/resolution options
|
|
if self.config.fourcc is not None:
|
|
self._validate_fourcc()
|
|
if self.videocapture is None:
|
|
raise DeviceNotConnectedError(f"{self} videocapture is not initialized")
|
|
|
|
default_width = int(round(self.videocapture.get(cv2.CAP_PROP_FRAME_WIDTH)))
|
|
default_height = int(round(self.videocapture.get(cv2.CAP_PROP_FRAME_HEIGHT)))
|
|
|
|
if self.width is None or self.height is None:
|
|
self.width, self.height = default_width, default_height
|
|
self.capture_width, self.capture_height = default_width, default_height
|
|
if self.rotation in [cv2.ROTATE_90_CLOCKWISE, cv2.ROTATE_90_COUNTERCLOCKWISE]:
|
|
self.width, self.height = default_height, default_width
|
|
self.capture_width, self.capture_height = default_width, default_height
|
|
else:
|
|
self._validate_width_and_height()
|
|
|
|
if self.fps is None:
|
|
self.fps = self.videocapture.get(cv2.CAP_PROP_FPS)
|
|
else:
|
|
self._validate_fps()
|
|
|
|
def _validate_fps(self) -> None:
|
|
"""Validates and sets the camera's frames per second (FPS)."""
|
|
|
|
if self.videocapture is None:
|
|
raise DeviceNotConnectedError(f"{self} videocapture is not initialized")
|
|
|
|
if self.fps is None:
|
|
raise ValueError(f"{self} FPS is not set")
|
|
|
|
success = self.videocapture.set(cv2.CAP_PROP_FPS, float(self.fps))
|
|
actual_fps = self.videocapture.get(cv2.CAP_PROP_FPS)
|
|
# Use math.isclose for robust float comparison
|
|
if not success or not math.isclose(self.fps, actual_fps, rel_tol=1e-3):
|
|
raise RuntimeError(f"{self} failed to set fps={self.fps} ({actual_fps=}).")
|
|
|
|
def _validate_fourcc(self) -> None:
|
|
"""Validates and sets the camera's FOURCC code."""
|
|
|
|
fourcc_code = cv2.VideoWriter_fourcc(*self.config.fourcc)
|
|
|
|
if self.videocapture is None:
|
|
raise DeviceNotConnectedError(f"{self} videocapture is not initialized")
|
|
|
|
success = self.videocapture.set(cv2.CAP_PROP_FOURCC, fourcc_code)
|
|
actual_fourcc_code = self.videocapture.get(cv2.CAP_PROP_FOURCC)
|
|
|
|
# Convert actual FOURCC code back to string for comparison
|
|
actual_fourcc_code_int = int(actual_fourcc_code)
|
|
actual_fourcc = "".join([chr((actual_fourcc_code_int >> 8 * i) & 0xFF) for i in range(4)])
|
|
|
|
if not success or actual_fourcc != self.config.fourcc:
|
|
logger.warning(
|
|
f"{self} failed to set fourcc={self.config.fourcc} (actual={actual_fourcc}, success={success}). "
|
|
f"Continuing with default format."
|
|
)
|
|
|
|
def _validate_width_and_height(self) -> None:
|
|
"""Validates and sets the camera's frame capture width and height."""
|
|
|
|
if self.videocapture is None:
|
|
raise DeviceNotConnectedError(f"{self} videocapture is not initialized")
|
|
|
|
if self.capture_width is None or self.capture_height is None:
|
|
raise ValueError(f"{self} capture_width or capture_height is not set")
|
|
|
|
width_success = self.videocapture.set(cv2.CAP_PROP_FRAME_WIDTH, float(self.capture_width))
|
|
height_success = self.videocapture.set(cv2.CAP_PROP_FRAME_HEIGHT, float(self.capture_height))
|
|
|
|
actual_width = int(round(self.videocapture.get(cv2.CAP_PROP_FRAME_WIDTH)))
|
|
if not width_success or self.capture_width != actual_width:
|
|
raise RuntimeError(
|
|
f"{self} failed to set capture_width={self.capture_width} ({actual_width=}, {width_success=})."
|
|
)
|
|
|
|
actual_height = int(round(self.videocapture.get(cv2.CAP_PROP_FRAME_HEIGHT)))
|
|
if not height_success or self.capture_height != actual_height:
|
|
raise RuntimeError(
|
|
f"{self} failed to set capture_height={self.capture_height} ({actual_height=}, {height_success=})."
|
|
)
|
|
|
|
@staticmethod
|
|
def find_cameras() -> list[dict[str, Any]]:
|
|
"""
|
|
Detects available OpenCV cameras connected to the system.
|
|
|
|
On Linux, it scans '/dev/video*' paths. On other systems (like macOS, Windows),
|
|
it checks indices from 0 up to `MAX_OPENCV_INDEX`.
|
|
|
|
Returns:
|
|
List[Dict[str, Any]]: A list of dictionaries,
|
|
where each dictionary contains 'type', 'id' (port index or path),
|
|
and the default profile properties (width, height, fps, format).
|
|
"""
|
|
found_cameras_info = []
|
|
|
|
targets_to_scan: list[str | int]
|
|
if platform.system() == "Linux":
|
|
possible_paths = sorted(Path("/dev").glob("video*"), key=lambda p: p.name)
|
|
targets_to_scan = [str(p) for p in possible_paths]
|
|
else:
|
|
targets_to_scan = [int(i) for i in range(MAX_OPENCV_INDEX)]
|
|
|
|
for target in targets_to_scan:
|
|
camera = cv2.VideoCapture(target)
|
|
if camera.isOpened():
|
|
default_width = int(camera.get(cv2.CAP_PROP_FRAME_WIDTH))
|
|
default_height = int(camera.get(cv2.CAP_PROP_FRAME_HEIGHT))
|
|
default_fps = camera.get(cv2.CAP_PROP_FPS)
|
|
default_format = camera.get(cv2.CAP_PROP_FORMAT)
|
|
|
|
# Get FOURCC code and convert to string
|
|
default_fourcc_code = camera.get(cv2.CAP_PROP_FOURCC)
|
|
default_fourcc_code_int = int(default_fourcc_code)
|
|
default_fourcc = "".join([chr((default_fourcc_code_int >> 8 * i) & 0xFF) for i in range(4)])
|
|
|
|
camera_info = {
|
|
"name": f"OpenCV Camera @ {target}",
|
|
"type": "OpenCV",
|
|
"id": target,
|
|
"backend_api": camera.getBackendName(),
|
|
"default_stream_profile": {
|
|
"format": default_format,
|
|
"fourcc": default_fourcc,
|
|
"width": default_width,
|
|
"height": default_height,
|
|
"fps": default_fps,
|
|
},
|
|
}
|
|
|
|
found_cameras_info.append(camera_info)
|
|
camera.release()
|
|
|
|
return found_cameras_info
|
|
|
|
def _read_from_hardware(self) -> NDArray[Any]:
|
|
if self.videocapture is None:
|
|
raise DeviceNotConnectedError(f"{self} videocapture is not initialized")
|
|
|
|
ret, frame = self.videocapture.read()
|
|
|
|
if not ret:
|
|
raise RuntimeError(f"{self} read failed (status={ret}).")
|
|
|
|
return frame
|
|
|
|
@check_if_not_connected
|
|
def read(self, color_mode: ColorMode | None = None) -> NDArray[Any]:
|
|
"""
|
|
Reads a single frame synchronously from the camera.
|
|
|
|
This is a blocking call. It waits for the next available frame from the
|
|
camera hardware via OpenCV.
|
|
|
|
Returns:
|
|
np.ndarray: The captured frame as a NumPy array in the format
|
|
(height, width, channels), using the specified or default
|
|
color mode and applying any configured rotation.
|
|
|
|
Raises:
|
|
DeviceNotConnectedError: If the camera is not connected.
|
|
RuntimeError: If reading the frame from the camera fails or if the
|
|
received frame dimensions don't match expectations before rotation.
|
|
ValueError: If an invalid `color_mode` is requested.
|
|
"""
|
|
|
|
start_time = time.perf_counter()
|
|
|
|
if color_mode is not None:
|
|
logger.warning(
|
|
f"{self} read() color_mode parameter is deprecated and will be removed in future versions."
|
|
)
|
|
|
|
if self.thread is None or not self.thread.is_alive():
|
|
raise RuntimeError(f"{self} read thread is not running.")
|
|
|
|
self.new_frame_event.clear()
|
|
frame = self.async_read(timeout_ms=10000)
|
|
|
|
read_duration_ms = (time.perf_counter() - start_time) * 1e3
|
|
logger.debug(f"{self} read took: {read_duration_ms:.1f}ms")
|
|
|
|
return frame
|
|
|
|
def _postprocess_image(self, image: NDArray[Any]) -> NDArray[Any]:
|
|
"""
|
|
Applies color conversion, dimension validation, and rotation to a raw frame.
|
|
|
|
Args:
|
|
image (np.ndarray): The raw image frame (expected BGR format from OpenCV).
|
|
|
|
Returns:
|
|
np.ndarray: The processed image frame.
|
|
|
|
Raises:
|
|
ValueError: If the requested `color_mode` is invalid.
|
|
RuntimeError: If the raw frame dimensions do not match the configured
|
|
`width` and `height`.
|
|
"""
|
|
|
|
if self.color_mode not in (ColorMode.RGB, ColorMode.BGR):
|
|
raise ValueError(
|
|
f"Invalid color mode '{self.color_mode}'. Expected {ColorMode.RGB} or {ColorMode.BGR}."
|
|
)
|
|
|
|
h, w, c = image.shape
|
|
|
|
if h != self.capture_height or w != self.capture_width:
|
|
raise RuntimeError(
|
|
f"{self} frame width={w} or height={h} do not match configured width={self.capture_width} or height={self.capture_height}."
|
|
)
|
|
|
|
if c != 3:
|
|
raise RuntimeError(f"{self} frame channels={c} do not match expected 3 channels (RGB/BGR).")
|
|
|
|
processed_image = image
|
|
if self.color_mode == ColorMode.RGB:
|
|
processed_image = cv2.cvtColor(image, cv2.COLOR_BGR2RGB)
|
|
|
|
if self.rotation in [cv2.ROTATE_90_CLOCKWISE, cv2.ROTATE_90_COUNTERCLOCKWISE, cv2.ROTATE_180]:
|
|
processed_image = cv2.rotate(processed_image, self.rotation)
|
|
|
|
return processed_image
|
|
|
|
def _read_loop(self) -> None:
|
|
"""
|
|
Internal loop run by the background thread for asynchronous reading.
|
|
|
|
On each iteration:
|
|
1. Reads a color frame
|
|
2. Stores result in latest_frame and updates timestamp (thread-safe)
|
|
3. Sets new_frame_event to notify listeners
|
|
|
|
Stops on DeviceNotConnectedError, logs other errors and continues.
|
|
"""
|
|
if self.stop_event is None:
|
|
raise RuntimeError(f"{self}: stop_event is not initialized before starting read loop.")
|
|
|
|
failure_count = 0
|
|
while not self.stop_event.is_set():
|
|
try:
|
|
raw_frame = self._read_from_hardware()
|
|
processed_frame = self._postprocess_image(raw_frame)
|
|
capture_time = time.perf_counter()
|
|
|
|
with self.frame_lock:
|
|
self.latest_frame = processed_frame
|
|
self.latest_timestamp = capture_time
|
|
self.new_frame_event.set()
|
|
failure_count = 0
|
|
|
|
except DeviceNotConnectedError:
|
|
break
|
|
except Exception as e:
|
|
if failure_count <= 10:
|
|
failure_count += 1
|
|
logger.warning(f"Error reading frame in background thread for {self}: {e}")
|
|
else:
|
|
raise RuntimeError(f"{self} exceeded maximum consecutive read failures.") from e
|
|
|
|
def _start_read_thread(self) -> None:
|
|
"""Starts or restarts the background read thread if it's not running."""
|
|
self._stop_read_thread()
|
|
|
|
self.stop_event = Event()
|
|
self.thread = Thread(target=self._read_loop, args=(), name=f"{self}_read_loop")
|
|
self.thread.daemon = True
|
|
self.thread.start()
|
|
time.sleep(0.1)
|
|
|
|
def _stop_read_thread(self) -> None:
|
|
"""Signals the background read thread to stop and waits for it to join."""
|
|
if self.stop_event is not None:
|
|
self.stop_event.set()
|
|
|
|
if self.thread is not None and self.thread.is_alive():
|
|
self.thread.join(timeout=2.0)
|
|
|
|
self.thread = None
|
|
self.stop_event = None
|
|
|
|
with self.frame_lock:
|
|
self.latest_frame = None
|
|
self.latest_timestamp = None
|
|
self.new_frame_event.clear()
|
|
|
|
@check_if_not_connected
|
|
def async_read(self, timeout_ms: float = 200) -> NDArray[Any]:
|
|
"""
|
|
Reads the latest available frame asynchronously.
|
|
|
|
This method retrieves the most recent frame captured by the background
|
|
read thread. It does not block waiting for the camera hardware directly,
|
|
but may wait up to timeout_ms for the background thread to provide a frame.
|
|
It is “best effort” under high FPS.
|
|
|
|
Args:
|
|
timeout_ms (float): Maximum time in milliseconds to wait for a frame
|
|
to become available. Defaults to 200ms (0.2 seconds).
|
|
|
|
Returns:
|
|
np.ndarray: The latest captured frame as a NumPy array in the format
|
|
(height, width, channels), processed according to configuration.
|
|
|
|
Raises:
|
|
DeviceNotConnectedError: If the camera is not connected.
|
|
TimeoutError: If no frame becomes available within the specified timeout.
|
|
RuntimeError: If an unexpected error occurs.
|
|
"""
|
|
|
|
if self.thread is None or not self.thread.is_alive():
|
|
raise RuntimeError(f"{self} read thread is not running.")
|
|
|
|
if not self.new_frame_event.wait(timeout=timeout_ms / 1000.0):
|
|
raise TimeoutError(
|
|
f"Timed out waiting for frame from camera {self} after {timeout_ms} ms. "
|
|
f"Read thread alive: {self.thread.is_alive()}."
|
|
)
|
|
|
|
with self.frame_lock:
|
|
frame = self.latest_frame
|
|
self.new_frame_event.clear()
|
|
|
|
if frame is None:
|
|
raise RuntimeError(f"Internal error: Event set but no frame available for {self}.")
|
|
|
|
return frame
|
|
|
|
@check_if_not_connected
|
|
def read_latest(self, max_age_ms: int = 500) -> NDArray[Any]:
|
|
"""Return the most recent frame captured immediately (Peeking).
|
|
|
|
This method is non-blocking and returns whatever is currently in the
|
|
memory buffer. The frame may be stale,
|
|
meaning it could have been captured a while ago (hanging camera scenario e.g.).
|
|
|
|
Returns:
|
|
NDArray[Any]: The frame image (numpy array).
|
|
|
|
Raises:
|
|
TimeoutError: If the latest frame is older than `max_age_ms`.
|
|
DeviceNotConnectedError: If the camera is not connected.
|
|
RuntimeError: If the camera is connected but has not captured any frames yet.
|
|
"""
|
|
|
|
if self.thread is None or not self.thread.is_alive():
|
|
raise RuntimeError(f"{self} read thread is not running.")
|
|
|
|
with self.frame_lock:
|
|
frame = self.latest_frame
|
|
timestamp = self.latest_timestamp
|
|
|
|
if frame is None or timestamp is None:
|
|
raise RuntimeError(f"{self} has not captured any frames yet.")
|
|
|
|
age_ms = (time.perf_counter() - timestamp) * 1e3
|
|
if age_ms > max_age_ms:
|
|
raise TimeoutError(
|
|
f"{self} latest frame is too old: {age_ms:.1f} ms (max allowed: {max_age_ms} ms)."
|
|
)
|
|
|
|
return frame
|
|
|
|
def disconnect(self) -> None:
|
|
"""
|
|
Disconnects from the camera and cleans up resources.
|
|
|
|
Stops the background read thread (if running) and releases the OpenCV
|
|
VideoCapture object.
|
|
|
|
Raises:
|
|
DeviceNotConnectedError: If the camera is already disconnected.
|
|
"""
|
|
if not self.is_connected and self.thread is None:
|
|
raise DeviceNotConnectedError(f"{self} not connected.")
|
|
|
|
if self.thread is not None:
|
|
self._stop_read_thread()
|
|
|
|
if self.videocapture is not None:
|
|
self.videocapture.release()
|
|
self.videocapture = None
|
|
|
|
with self.frame_lock:
|
|
self.latest_frame = None
|
|
self.latest_timestamp = None
|
|
self.new_frame_event.clear()
|
|
|
|
logger.info(f"{self} disconnected.")
|