Skip to content

Commit

Permalink
Merge pull request #85 from natekspencer/dev
Browse files Browse the repository at this point in the history
Add ability to set camera privacy status
  • Loading branch information
natekspencer authored Mar 10, 2023
2 parents 9df1553 + 5ff91a0 commit d7539b4
Show file tree
Hide file tree
Showing 5 changed files with 49 additions and 28 deletions.
2 changes: 1 addition & 1 deletion pyproject.toml
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
[tool.poetry]
name = "vivintpy"
version = "2023.3.2"
version = "2023.3.3"
description = "Python library for interacting with a Vivint security and smart home system."
authors = ["Nathan Spencer <[email protected]>"]
license = "MIT"
Expand Down
2 changes: 1 addition & 1 deletion tests/test_version.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,4 +4,4 @@

def test_version() -> None:
"""Test version."""
assert __version__ == "2023.3.2"
assert __version__ == "2023.3.3"
2 changes: 1 addition & 1 deletion vivintpy/__init__.py
Original file line number Diff line number Diff line change
@@ -1,2 +1,2 @@
"""Provide a package for vivintpy."""
__version__ = "2023.3.2"
__version__ = "2023.3.3"
8 changes: 7 additions & 1 deletion vivintpy/devices/camera.py
Original file line number Diff line number Diff line change
Expand Up @@ -160,11 +160,17 @@ async def get_direct_rtsp_url(
)

async def set_as_doorbell_chime_extender(self, state: bool) -> None:
"""Set camera's use as doorbell chime extender."""
"""Set use as doorbell chime extender."""
await self.vivintskyapi.set_camera_as_doorbell_chime_extender(
self.alarm_panel.id, self.id, state
)

async def set_privacy_mode(self, state: bool) -> None:
"""Set privacy mode."""
await self.vivintskyapi.set_camera_privacy_mode(
self.alarm_panel.id, self.id, state
)

def handle_pubnub_message(self, message: dict) -> None:
"""Handle a pubnub message addressed to this camera."""
super().handle_pubnub_message(message)
Expand Down
63 changes: 39 additions & 24 deletions vivintpy/vivintskyapi.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,8 @@
import logging
import ssl
from collections.abc import Callable
from typing import Any
from http.cookies import Morsel, SimpleCookie
from typing import Any, cast

import aiohttp
import certifi
Expand All @@ -29,9 +30,9 @@

_LOGGER = logging.getLogger(__name__)

VIVINT_API_ENDPOINT = "https://www.vivintsky.com/api"
VIVINT_BEAM_ENDPOINT = "beam.vivintsky.com:443"
VIVINT_MFA_ENDPOINT = (
API_ENDPOINT = "https://www.vivintsky.com/api"
BEAM_ENDPOINT = "beam.vivintsky.com:443"
MFA_ENDPOINT = (
"https://www.vivintsky.com/platform-user-api/v0/platformusers/2fa/validate"
)

Expand All @@ -54,10 +55,14 @@ def __init__(
self.__has_custom_client_session = client_session is not None
self.__mfa_pending = False

def _get_session_cookie(self) -> Morsel | None:
"""Get the session cookie."""
cookie = self.__client_session.cookie_jar.filter_cookies(API_ENDPOINT)
return cast(SimpleCookie, cookie).get("s")

def is_session_valid(self) -> bool:
"""Return the state of the current session."""
cookies = self.__client_session.cookie_jar.filter_cookies(VIVINT_API_ENDPOINT)
return cookies.get("s") is not None
return self._get_session_cookie() is not None

async def connect(self) -> dict:
"""Connect to VivintSky Cloud Service."""
Expand All @@ -78,10 +83,7 @@ async def disconnect(self) -> None:

async def verify_mfa(self, code: str) -> None:
"""Verify multi-factor authentication code."""
resp = await self.__post(
VIVINT_MFA_ENDPOINT,
data=json.dumps({"code": code}),
)
resp = await self.__post(MFA_ENDPOINT, data=json.dumps({"code": code}))
if resp is not None:
self.__mfa_pending = False

Expand Down Expand Up @@ -182,26 +184,39 @@ async def set_camera_as_doorbell_chime_extender(
) -> None:
"""Set the camera to be used as a doorbell chime extender."""
creds = grpc.ssl_channel_credentials()
metad = [
(
"session",
self.__client_session.cookie_jar.filter_cookies(VIVINT_API_ENDPOINT)
.get("s")
.value,
)
]
assert (cookie := self._get_session_cookie())

async with grpc.aio.secure_channel(
VIVINT_BEAM_ENDPOINT, credentials=creds
) as channel:
async with grpc.aio.secure_channel(BEAM_ENDPOINT, credentials=creds) as channel:
stub: beam_pb2_grpc.BeamStub = beam_pb2_grpc.BeamStub(channel) # type: ignore
response: beam_pb2.SetUseAsDoorbellChimeExtenderResponse = await stub.SetUseAsDoorbellChimeExtender(
beam_pb2.SetUseAsDoorbellChimeExtenderRequest( # pylint: disable=no-member
panel_id=panel_id,
device_id=device_id,
use_as_doorbell_chime_extender=state,
),
metadata=metad,
metadata=[("session", cookie.value)],
)

_LOGGER.debug("Response received: %s", str(response))

async def set_camera_privacy_mode(
self, panel_id: int, device_id: int, state: bool
) -> None:
"""Set the camera privacy mode."""
creds = grpc.ssl_channel_credentials()
assert (cookie := self._get_session_cookie())

async with grpc.aio.secure_channel(BEAM_ENDPOINT, credentials=creds) as channel:
stub: beam_pb2_grpc.BeamStub = beam_pb2_grpc.BeamStub(channel) # type: ignore
response: beam_pb2.SetCameraPrivacyModeResponse = (
await stub.SetCameraPrivacyMode(
beam_pb2.SetCameraPrivacyModeRequest( # pylint: disable=no-member
panel_id=panel_id,
device_id=device_id,
privacy_mode=state,
),
metadata=[("session", cookie.value)],
)
)

_LOGGER.debug("Response received: %s", str(response))
Expand Down Expand Up @@ -460,13 +475,13 @@ async def __call(
if self.__client_session.closed:
raise VivintSkyApiError("The client session has been closed")

is_mfa_request = path == VIVINT_MFA_ENDPOINT
is_mfa_request = path == MFA_ENDPOINT

if self.__mfa_pending and not is_mfa_request:
raise VivintSkyApiMfaRequiredError(AuthenticationResponse.MFA_REQUIRED)

resp = await method(
path if is_mfa_request else f"{VIVINT_API_ENDPOINT}/{path}",
path if is_mfa_request else f"{API_ENDPOINT}/{path}",
headers=headers,
params=params,
data=data,
Expand Down

0 comments on commit d7539b4

Please sign in to comment.