pyhOn/pyhon/connection/handler.py

137 lines
4.8 KiB
Python
Raw Normal View History

2023-04-09 18:50:28 +00:00
import json
2023-04-09 16:13:50 +00:00
from contextlib import asynccontextmanager
import aiohttp
from pyhon import const
from pyhon.connection.auth import HonAuth, _LOGGER
from pyhon.connection.device import HonDevice
from pyhon.exceptions import HonAuthenticationError
2023-04-09 16:13:50 +00:00
class HonBaseConnectionHandler:
_HEADERS = {"user-agent": const.USER_AGENT, "Content-Type": "application/json"}
2023-04-10 04:34:19 +00:00
def __init__(self, session=None):
self._session = session
2023-04-09 16:13:50 +00:00
self._auth = None
async def __aenter__(self):
2023-04-11 23:07:03 +00:00
self._session = aiohttp.ClientSession()
2023-04-09 18:50:28 +00:00
return await self.create()
2023-04-09 16:13:50 +00:00
async def __aexit__(self, exc_type, exc_val, exc_tb):
await self.close()
async def create(self):
2023-04-09 18:50:28 +00:00
return self
2023-04-09 16:13:50 +00:00
@asynccontextmanager
2023-04-10 04:34:19 +00:00
async def _intercept(self, method, *args, loop=0, **kwargs):
2023-04-09 21:47:33 +00:00
raise NotImplementedError
2023-04-09 16:13:50 +00:00
2023-04-10 04:34:19 +00:00
@asynccontextmanager
async def get(self, *args, **kwargs):
async with self._intercept(self._session.get, *args, **kwargs) as response:
yield response
2023-04-09 16:13:50 +00:00
@asynccontextmanager
async def post(self, *args, **kwargs):
2023-04-10 04:34:19 +00:00
async with self._intercept(self._session.post, *args, **kwargs) as response:
yield response
2023-04-09 16:13:50 +00:00
async def close(self):
await self._session.close()
class HonConnectionHandler(HonBaseConnectionHandler):
2023-04-10 04:34:19 +00:00
def __init__(self, email, password, session=None):
super().__init__(session=session)
2023-04-09 16:13:50 +00:00
self._device = HonDevice()
self._email = email
self._password = password
if not self._email:
raise HonAuthenticationError("An email address must be specified")
2023-04-09 16:13:50 +00:00
if not self._password:
raise HonAuthenticationError("A password address must be specified")
2023-04-09 16:13:50 +00:00
self._request_headers = {}
@property
def device(self):
return self._device
async def create(self):
await super().create()
self._auth = HonAuth(self._session, self._email, self._password, self._device)
2023-04-09 18:50:28 +00:00
return self
2023-04-09 16:13:50 +00:00
async def _check_headers(self, headers):
2023-04-09 18:55:36 +00:00
if (
"cognito-token" not in self._request_headers
or "id-token" not in self._request_headers
):
2023-04-09 16:13:50 +00:00
if await self._auth.authorize():
self._request_headers["cognito-token"] = self._auth.cognito_token
self._request_headers["id-token"] = self._auth.id_token
else:
raise HonAuthenticationError("Can't login")
2023-04-11 23:07:03 +00:00
return self._HEADERS | headers | self._request_headers
2023-04-09 16:13:50 +00:00
@asynccontextmanager
2023-04-09 16:43:57 +00:00
async def _intercept(self, method, *args, loop=0, **kwargs):
2023-04-09 16:13:50 +00:00
kwargs["headers"] = await self._check_headers(kwargs.get("headers", {}))
2023-04-09 16:43:57 +00:00
async with method(*args, **kwargs) as response:
2023-04-11 15:09:02 +00:00
if response.status in [401, 403] and loop == 0:
2023-04-09 16:43:57 +00:00
_LOGGER.info("Try refreshing token...")
await self._auth.refresh()
2023-04-11 20:14:36 +00:00
async with self._intercept(
method, *args, loop=loop + 1, **kwargs
) as result:
2023-04-11 15:09:02 +00:00
yield result
elif response.status in [401, 403] and loop == 1:
2023-04-09 18:55:36 +00:00
_LOGGER.warning(
"%s - Error %s - %s",
response.request_info.url,
response.status,
await response.text(),
)
2023-04-11 23:07:03 +00:00
self._request_headers = {}
self._session.cookie_jar.clear_domain(const.AUTH_API.split("/")[-2])
2023-04-09 16:13:50 +00:00
await self.create()
2023-04-11 20:14:36 +00:00
async with self._intercept(
method, *args, loop=loop + 1, **kwargs
) as result:
2023-04-11 15:09:02 +00:00
yield result
2023-04-09 16:13:50 +00:00
elif loop >= 2:
2023-04-09 18:55:36 +00:00
_LOGGER.error(
"%s - Error %s - %s",
response.request_info.url,
response.status,
await response.text(),
)
raise HonAuthenticationError("Login failure")
2023-04-09 16:13:50 +00:00
else:
2023-04-09 18:50:28 +00:00
try:
await response.json()
yield response
except json.JSONDecodeError:
2023-04-09 18:55:36 +00:00
_LOGGER.warning(
"%s - JsonDecodeError %s - %s",
response.request_info.url,
response.status,
await response.text(),
)
2023-04-09 18:50:28 +00:00
yield {}
2023-04-09 16:13:50 +00:00
class HonAnonymousConnectionHandler(HonBaseConnectionHandler):
_HEADERS = HonBaseConnectionHandler._HEADERS | {"x-api-key": const.API_KEY}
@asynccontextmanager
2023-04-10 04:34:19 +00:00
async def _intercept(self, method, *args, loop=0, **kwargs):
kwargs["headers"] = kwargs.pop("headers", {}) | self._HEADERS
async with method(*args, **kwargs) as response:
if response.status == 403:
_LOGGER.error("Can't authenticate anymore")
2023-04-09 16:13:50 +00:00
yield response