mirror of
https://github.com/python-kasa/python-kasa.git
synced 2025-01-23 13:17:06 +00:00
b5f49a3c8a
Some checks are pending
CI / Perform linting checks (3.13) (push) Waiting to run
CI / Python ${{ matrix.python-version}} on ${{ matrix.os }}${{ fromJSON('[" (extras)", ""]')[matrix.extras == ''] }} (false, macos-latest, 3.11) (push) Blocked by required conditions
CI / Python ${{ matrix.python-version}} on ${{ matrix.os }}${{ fromJSON('[" (extras)", ""]')[matrix.extras == ''] }} (false, macos-latest, 3.12) (push) Blocked by required conditions
CI / Python ${{ matrix.python-version}} on ${{ matrix.os }}${{ fromJSON('[" (extras)", ""]')[matrix.extras == ''] }} (false, macos-latest, 3.13) (push) Blocked by required conditions
CI / Python ${{ matrix.python-version}} on ${{ matrix.os }}${{ fromJSON('[" (extras)", ""]')[matrix.extras == ''] }} (false, ubuntu-latest, 3.11) (push) Blocked by required conditions
CI / Python ${{ matrix.python-version}} on ${{ matrix.os }}${{ fromJSON('[" (extras)", ""]')[matrix.extras == ''] }} (false, ubuntu-latest, 3.12) (push) Blocked by required conditions
CI / Python ${{ matrix.python-version}} on ${{ matrix.os }}${{ fromJSON('[" (extras)", ""]')[matrix.extras == ''] }} (false, ubuntu-latest, 3.13) (push) Blocked by required conditions
CI / Python ${{ matrix.python-version}} on ${{ matrix.os }}${{ fromJSON('[" (extras)", ""]')[matrix.extras == ''] }} (false, windows-latest, 3.11) (push) Blocked by required conditions
CI / Python ${{ matrix.python-version}} on ${{ matrix.os }}${{ fromJSON('[" (extras)", ""]')[matrix.extras == ''] }} (false, windows-latest, 3.12) (push) Blocked by required conditions
CI / Python ${{ matrix.python-version}} on ${{ matrix.os }}${{ fromJSON('[" (extras)", ""]')[matrix.extras == ''] }} (false, windows-latest, 3.13) (push) Blocked by required conditions
CI / Python ${{ matrix.python-version}} on ${{ matrix.os }}${{ fromJSON('[" (extras)", ""]')[matrix.extras == ''] }} (true, ubuntu-latest, 3.11) (push) Blocked by required conditions
CI / Python ${{ matrix.python-version}} on ${{ matrix.os }}${{ fromJSON('[" (extras)", ""]')[matrix.extras == ''] }} (true, ubuntu-latest, 3.12) (push) Blocked by required conditions
CI / Python ${{ matrix.python-version}} on ${{ matrix.os }}${{ fromJSON('[" (extras)", ""]')[matrix.extras == ''] }} (true, ubuntu-latest, 3.13) (push) Blocked by required conditions
CodeQL checks / Analyze (python) (push) Waiting to run
Fixes a few issues with the lens mask since migrating it into its own module: - The module didn't provide itself as the container and hence the feature was accessing the same properties on the device. - `enabled` getter on the module incorrect but not picked up due to the previous issue. - No `REQUIRED_COMPONENT` set to ensure the module only created if available. Also changes attribute names to `enabled` from `state` to avoid confusion with device states.
127 lines
4.1 KiB
Python
127 lines
4.1 KiB
Python
"""Implementation of camera module."""
|
|
|
|
from __future__ import annotations
|
|
|
|
import base64
|
|
import logging
|
|
from enum import StrEnum
|
|
from typing import Annotated
|
|
from urllib.parse import quote_plus
|
|
|
|
from ...credentials import Credentials
|
|
from ...device_type import DeviceType
|
|
from ...feature import Feature
|
|
from ...json import loads as json_loads
|
|
from ...module import FeatureAttribute, Module
|
|
from ..smartcammodule import SmartCamModule
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
LOCAL_STREAMING_PORT = 554
|
|
ONVIF_PORT = 2020
|
|
|
|
|
|
class StreamResolution(StrEnum):
|
|
"""Class for stream resolution."""
|
|
|
|
HD = "HD"
|
|
SD = "SD"
|
|
|
|
|
|
class Camera(SmartCamModule):
|
|
"""Implementation of device module."""
|
|
|
|
def _initialize_features(self) -> None:
|
|
"""Initialize features after the initial update."""
|
|
if Module.LensMask in self._device.modules:
|
|
self._add_feature(
|
|
Feature(
|
|
self._device,
|
|
id="state",
|
|
name="State",
|
|
container=self,
|
|
attribute_getter="is_on",
|
|
attribute_setter="set_state",
|
|
type=Feature.Type.Switch,
|
|
category=Feature.Category.Primary,
|
|
)
|
|
)
|
|
|
|
@property
|
|
def is_on(self) -> bool:
|
|
"""Return the device on state."""
|
|
if lens_mask := self._device.modules.get(Module.LensMask):
|
|
return not lens_mask.enabled
|
|
return True
|
|
|
|
async def set_state(self, on: bool) -> Annotated[dict, FeatureAttribute()]:
|
|
"""Set the device on state.
|
|
|
|
If the device does not support setting state will do nothing.
|
|
"""
|
|
if lens_mask := self._device.modules.get(Module.LensMask):
|
|
# Turning off enables the privacy mask which is why value is reversed.
|
|
return await lens_mask.set_enabled(not on)
|
|
return {}
|
|
|
|
def _get_credentials(self) -> Credentials | None:
|
|
"""Get credentials from ."""
|
|
config = self._device.config
|
|
if credentials := config.credentials:
|
|
return credentials
|
|
|
|
if credentials_hash := config.credentials_hash:
|
|
try:
|
|
decoded = json_loads(
|
|
base64.b64decode(credentials_hash.encode()).decode()
|
|
)
|
|
except Exception:
|
|
_LOGGER.warning(
|
|
"Unable to deserialize credentials_hash: %s", credentials_hash
|
|
)
|
|
return None
|
|
if (username := decoded.get("un")) and (password := decoded.get("pwd")):
|
|
return Credentials(username, password)
|
|
|
|
return None
|
|
|
|
def stream_rtsp_url(
|
|
self,
|
|
credentials: Credentials | None = None,
|
|
*,
|
|
stream_resolution: StreamResolution = StreamResolution.HD,
|
|
) -> str | None:
|
|
"""Return the local rtsp streaming url.
|
|
|
|
:param credentials: Credentials for camera account.
|
|
These could be different credentials to tplink cloud credentials.
|
|
If not provided will use tplink credentials if available
|
|
:return: rtsp url with escaped credentials or None if no credentials or
|
|
camera is off.
|
|
"""
|
|
streams = {
|
|
StreamResolution.HD: "stream1",
|
|
StreamResolution.SD: "stream2",
|
|
}
|
|
if (stream := streams.get(stream_resolution)) is None:
|
|
return None
|
|
|
|
if not credentials:
|
|
credentials = self._get_credentials()
|
|
|
|
if not credentials or not credentials.username or not credentials.password:
|
|
return None
|
|
|
|
username = quote_plus(credentials.username)
|
|
password = quote_plus(credentials.password)
|
|
|
|
return f"rtsp://{username}:{password}@{self._device.host}:{LOCAL_STREAMING_PORT}/{stream}"
|
|
|
|
def onvif_url(self) -> str | None:
|
|
"""Return the onvif url."""
|
|
return f"http://{self._device.host}:{ONVIF_PORT}/onvif/device_service"
|
|
|
|
async def _check_supported(self) -> bool:
|
|
"""Additional check to see if the module is supported by the device."""
|
|
return self._device.device_type is DeviceType.Camera
|