2019-02-13 20:21:14 +00:00
|
|
|
"""Support for Axis camera streaming."""
|
2021-01-22 23:15:58 +00:00
|
|
|
from urllib.parse import urlencode
|
|
|
|
|
2022-04-05 21:42:05 +00:00
|
|
|
from homeassistant.components.camera import CameraEntityFeature
|
2022-02-18 08:03:41 +00:00
|
|
|
from homeassistant.components.mjpeg import MjpegCamera, filter_urllib3_logging
|
2022-01-03 12:22:41 +00:00
|
|
|
from homeassistant.config_entries import ConfigEntry
|
2022-02-15 14:24:52 +00:00
|
|
|
from homeassistant.const import HTTP_DIGEST_AUTHENTICATION
|
2022-01-03 12:22:41 +00:00
|
|
|
from homeassistant.core import HomeAssistant
|
2019-03-24 15:16:50 +00:00
|
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_connect
|
2022-01-03 12:22:41 +00:00
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
2017-05-12 15:51:54 +00:00
|
|
|
|
2019-05-20 05:45:31 +00:00
|
|
|
from .axis_base import AxisEntityBase
|
2021-01-22 23:15:58 +00:00
|
|
|
from .const import DEFAULT_STREAM_PROFILE, DEFAULT_VIDEO_SOURCE, DOMAIN as AXIS_DOMAIN
|
2020-03-19 23:30:38 +00:00
|
|
|
|
2017-05-12 15:51:54 +00:00
|
|
|
|
2022-01-03 12:22:41 +00:00
|
|
|
async def async_setup_entry(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
config_entry: ConfigEntry,
|
|
|
|
async_add_entities: AddEntitiesCallback,
|
|
|
|
) -> None:
|
2019-03-24 15:16:50 +00:00
|
|
|
"""Set up the Axis camera video stream."""
|
2019-01-03 18:56:36 +00:00
|
|
|
filter_urllib3_logging()
|
|
|
|
|
2020-01-04 07:58:18 +00:00
|
|
|
device = hass.data[AXIS_DOMAIN][config_entry.unique_id]
|
2019-03-24 15:16:50 +00:00
|
|
|
|
2020-06-04 12:25:50 +00:00
|
|
|
if not device.api.vapix.params.image_format:
|
2020-05-15 21:56:09 +00:00
|
|
|
return
|
|
|
|
|
2020-06-01 16:45:38 +00:00
|
|
|
async_add_entities([AxisCamera(device)])
|
2017-06-24 07:14:57 +00:00
|
|
|
|
|
|
|
|
2019-05-20 05:45:31 +00:00
|
|
|
class AxisCamera(AxisEntityBase, MjpegCamera):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Representation of a Axis camera."""
|
2017-06-24 07:14:57 +00:00
|
|
|
|
2022-04-05 21:42:05 +00:00
|
|
|
_attr_supported_features = CameraEntityFeature.STREAM
|
|
|
|
|
2020-06-01 16:45:38 +00:00
|
|
|
def __init__(self, device):
|
2017-06-24 07:14:57 +00:00
|
|
|
"""Initialize Axis Communications camera component."""
|
2019-05-20 05:45:31 +00:00
|
|
|
AxisEntityBase.__init__(self, device)
|
2020-06-01 16:45:38 +00:00
|
|
|
|
2022-02-15 14:24:52 +00:00
|
|
|
MjpegCamera.__init__(
|
|
|
|
self,
|
|
|
|
name=device.name,
|
|
|
|
username=device.username,
|
|
|
|
password=device.password,
|
|
|
|
mjpeg_url=self.mjpeg_source,
|
|
|
|
still_image_url=self.image_source,
|
|
|
|
authentication=HTTP_DIGEST_AUTHENTICATION,
|
|
|
|
)
|
2019-03-24 15:16:50 +00:00
|
|
|
|
2021-06-10 17:15:01 +00:00
|
|
|
self._attr_unique_id = f"{device.unique_id}-camera"
|
|
|
|
|
2019-03-24 15:16:50 +00:00
|
|
|
async def async_added_to_hass(self):
|
|
|
|
"""Subscribe camera events."""
|
2020-04-02 16:25:33 +00:00
|
|
|
self.async_on_remove(
|
2019-07-31 19:25:30 +00:00
|
|
|
async_dispatcher_connect(
|
2020-05-14 08:49:27 +00:00
|
|
|
self.hass, self.device.signal_new_address, self._new_address
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
|
|
|
)
|
2019-03-29 14:20:12 +00:00
|
|
|
|
2019-05-20 05:45:31 +00:00
|
|
|
await super().async_added_to_hass()
|
2019-04-15 22:06:45 +00:00
|
|
|
|
2021-01-22 23:15:58 +00:00
|
|
|
def _new_address(self) -> None:
|
2019-04-02 18:13:11 +00:00
|
|
|
"""Set new device address for video stream."""
|
2020-06-01 16:45:38 +00:00
|
|
|
self._mjpeg_url = self.mjpeg_source
|
|
|
|
self._still_image_url = self.image_source
|
2019-03-27 17:25:01 +00:00
|
|
|
|
2020-06-01 16:45:38 +00:00
|
|
|
@property
|
2021-01-22 23:15:58 +00:00
|
|
|
def image_source(self) -> str:
|
2020-06-01 16:45:38 +00:00
|
|
|
"""Return still image URL for device."""
|
2021-01-22 23:15:58 +00:00
|
|
|
options = self.generate_options(skip_stream_profile=True)
|
|
|
|
return f"http://{self.device.host}:{self.device.port}/axis-cgi/jpg/image.cgi{options}"
|
2020-06-01 16:45:38 +00:00
|
|
|
|
|
|
|
@property
|
2021-01-22 23:15:58 +00:00
|
|
|
def mjpeg_source(self) -> str:
|
2020-06-01 16:45:38 +00:00
|
|
|
"""Return mjpeg URL for device."""
|
2021-01-22 23:15:58 +00:00
|
|
|
options = self.generate_options()
|
2021-01-16 18:54:48 +00:00
|
|
|
return f"http://{self.device.host}:{self.device.port}/axis-cgi/mjpg/video.cgi{options}"
|
2020-06-01 16:45:38 +00:00
|
|
|
|
2021-01-22 23:15:58 +00:00
|
|
|
async def stream_source(self) -> str:
|
2020-06-01 16:45:38 +00:00
|
|
|
"""Return the stream source."""
|
2021-01-22 23:15:58 +00:00
|
|
|
options = self.generate_options(add_video_codec_h264=True)
|
|
|
|
return f"rtsp://{self.device.username}:{self.device.password}@{self.device.host}/axis-media/media.amp{options}"
|
|
|
|
|
|
|
|
def generate_options(
|
|
|
|
self, skip_stream_profile: bool = False, add_video_codec_h264: bool = False
|
|
|
|
) -> str:
|
|
|
|
"""Generate options for video stream."""
|
|
|
|
options_dict = {}
|
|
|
|
|
|
|
|
if add_video_codec_h264:
|
|
|
|
options_dict["videocodec"] = "h264"
|
|
|
|
|
|
|
|
if (
|
|
|
|
not skip_stream_profile
|
|
|
|
and self.device.option_stream_profile != DEFAULT_STREAM_PROFILE
|
|
|
|
):
|
|
|
|
options_dict["streamprofile"] = self.device.option_stream_profile
|
|
|
|
|
|
|
|
if self.device.option_video_source != DEFAULT_VIDEO_SOURCE:
|
|
|
|
options_dict["camera"] = self.device.option_video_source
|
|
|
|
|
|
|
|
if not options_dict:
|
|
|
|
return ""
|
|
|
|
return f"?{urlencode(options_dict)}"
|