Add Octoprint camera entity (#79689)
parent
f73fc9e355
commit
c197e1765a
|
@ -56,7 +56,7 @@ def ensure_valid_path(value):
|
|||
return value
|
||||
|
||||
|
||||
PLATFORMS = [Platform.BINARY_SENSOR, Platform.BUTTON, Platform.SENSOR]
|
||||
PLATFORMS = [Platform.BINARY_SENSOR, Platform.BUTTON, Platform.CAMERA, Platform.SENSOR]
|
||||
DEFAULT_NAME = "OctoPrint"
|
||||
CONF_NUMBER_OF_TOOLS = "number_of_tools"
|
||||
CONF_BED = "bed"
|
||||
|
|
|
@ -0,0 +1,59 @@
|
|||
"""Support for OctoPrint binary camera."""
|
||||
from __future__ import annotations
|
||||
|
||||
from pyoctoprintapi import OctoprintClient, WebcamSettings
|
||||
|
||||
from homeassistant.components.mjpeg.camera import MjpegCamera
|
||||
from homeassistant.config_entries import ConfigEntry
|
||||
from homeassistant.core import HomeAssistant
|
||||
from homeassistant.helpers.entity import DeviceInfo
|
||||
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
||||
|
||||
from . import OctoprintDataUpdateCoordinator
|
||||
from .const import DOMAIN
|
||||
|
||||
|
||||
async def async_setup_entry(
|
||||
hass: HomeAssistant,
|
||||
config_entry: ConfigEntry,
|
||||
async_add_entities: AddEntitiesCallback,
|
||||
) -> None:
|
||||
"""Set up the available OctoPrint camera."""
|
||||
coordinator: OctoprintDataUpdateCoordinator = hass.data[DOMAIN][
|
||||
config_entry.entry_id
|
||||
]["coordinator"]
|
||||
client: OctoprintClient = hass.data[DOMAIN][config_entry.entry_id]["client"]
|
||||
device_id = config_entry.unique_id
|
||||
|
||||
assert device_id is not None
|
||||
|
||||
camera_info = await client.get_webcam_info()
|
||||
|
||||
if not camera_info or not camera_info.enabled:
|
||||
return
|
||||
|
||||
async_add_entities(
|
||||
[
|
||||
OctoprintCamera(
|
||||
camera_info,
|
||||
coordinator.device_info,
|
||||
device_id,
|
||||
)
|
||||
]
|
||||
)
|
||||
|
||||
|
||||
class OctoprintCamera(MjpegCamera):
|
||||
"""Representation of an OctoPrint Camera Stream."""
|
||||
|
||||
def __init__(
|
||||
self, camera_settings: WebcamSettings, device_info: DeviceInfo, device_id: str
|
||||
) -> None:
|
||||
"""Initialize as a subclass of MjpegCamera."""
|
||||
super().__init__(
|
||||
device_info=device_info,
|
||||
mjpeg_url=camera_settings.stream_url,
|
||||
name="OctoPrint Camera",
|
||||
still_image_url=camera_settings.external_snapshot_url,
|
||||
unique_id=f"camera-{device_id}",
|
||||
)
|
|
@ -0,0 +1,67 @@
|
|||
"""The tests for Octoptint camera module."""
|
||||
|
||||
from unittest.mock import patch
|
||||
|
||||
from pyoctoprintapi import WebcamSettings
|
||||
|
||||
from homeassistant.components.camera import DOMAIN as CAMERA_DOMAIN
|
||||
from homeassistant.helpers import entity_registry as er
|
||||
|
||||
from . import init_integration
|
||||
|
||||
|
||||
async def test_camera(hass):
|
||||
"""Test the underlying camera."""
|
||||
with patch(
|
||||
"pyoctoprintapi.OctoprintClient.get_webcam_info",
|
||||
return_value=WebcamSettings(
|
||||
base_url="http://fake-octoprint/",
|
||||
raw={
|
||||
"streamUrl": "/webcam/?action=stream",
|
||||
"snapshotUrl": "http://127.0.0.1:8080/?action=snapshot",
|
||||
"webcamEnabled": True,
|
||||
},
|
||||
),
|
||||
):
|
||||
await init_integration(hass, CAMERA_DOMAIN)
|
||||
|
||||
entity_registry = er.async_get(hass)
|
||||
|
||||
entry = entity_registry.async_get("camera.octoprint_camera")
|
||||
assert entry is not None
|
||||
assert entry.unique_id == "camera-uuid"
|
||||
|
||||
|
||||
async def test_camera_disabled(hass):
|
||||
"""Test that the camera does not load if there is not one configured."""
|
||||
with patch(
|
||||
"pyoctoprintapi.OctoprintClient.get_webcam_info",
|
||||
return_value=WebcamSettings(
|
||||
base_url="http://fake-octoprint/",
|
||||
raw={
|
||||
"streamUrl": "/webcam/?action=stream",
|
||||
"snapshotUrl": "http://127.0.0.1:8080/?action=snapshot",
|
||||
"webcamEnabled": False,
|
||||
},
|
||||
),
|
||||
):
|
||||
await init_integration(hass, CAMERA_DOMAIN)
|
||||
|
||||
entity_registry = er.async_get(hass)
|
||||
|
||||
entry = entity_registry.async_get("camera.octoprint_camera")
|
||||
assert entry is None
|
||||
|
||||
|
||||
async def test_no_supported_camera(hass):
|
||||
"""Test that the camera does not load if there is not one configured."""
|
||||
with patch(
|
||||
"pyoctoprintapi.OctoprintClient.get_webcam_info",
|
||||
return_value=None,
|
||||
):
|
||||
await init_integration(hass, CAMERA_DOMAIN)
|
||||
|
||||
entity_registry = er.async_get(hass)
|
||||
|
||||
entry = entity_registry.async_get("camera.octoprint_camera")
|
||||
assert entry is None
|
Loading…
Reference in New Issue