From 15cff3362cbec77da576dd8d313ef9dec8f44d42 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bernd=20Sch=C3=B6rgers?= Date: Tue, 20 Oct 2020 14:28:46 +0200 Subject: [PATCH] Migrate to aiohttp, async (#6) * Migrate to aiohttp, async * Add error handling * Remove unused import --- fullykiosk/__init__.py | 154 ++++++++++++++++++++++++--------------- fullykiosk/exceptions.py | 10 +++ requirements.txt | 2 +- setup.py | 2 +- 4 files changed, 107 insertions(+), 61 deletions(-) create mode 100644 fullykiosk/exceptions.py diff --git a/fullykiosk/__init__.py b/fullykiosk/__init__.py index 25c8a0b..4c00c6c 100644 --- a/fullykiosk/__init__.py +++ b/fullykiosk/__init__.py @@ -1,30 +1,32 @@ +import aiohttp import json import logging -import requests +from .exceptions import FullyKioskError -class FullyKiosk: - def __init__(self, host, port, password): - self.host = host - self.port = port - self.password = password +_LOGGER = logging.getLogger(__name__) - self._deviceInfo = None +RESPONSE_STATUS = "status" +RESPONSE_STATUSTEXT = "statustext" +RESPONSE_ERRORSTATUS = "Error" - def sendCommand(self, cmd, **kwargs): - url = f"http://{self.host}:{self.port}/?cmd={cmd}&password={self.password}&type=json" - for key, value in kwargs.items(): - if value is not None: - url = url + f"&{key}={value}" - try: - result = json.loads(requests.get(url, timeout=10).content) - return result - except requests.exceptions.Timeout: - print("Timeout error") +class FullyKiosk: + def __init__(self, session, host, port, password): + self._rh = _RequestsHandler(session, host, port) + self._password = password + self._deviceInfo = None + + async def sendCommand(self, cmd, **kwargs): + data = await self._rh.get( + cmd=cmd, password=self._password, type="json", **kwargs + ) + if RESPONSE_STATUS in data and data[RESPONSE_STATUS] == RESPONSE_ERRORSTATUS: + raise FullyKioskError(RESPONSE_ERRORSTATUS, data[RESPONSE_STATUSTEXT]) + return data - def getDeviceInfo(self): - result = self.sendCommand("deviceInfo") + async def getDeviceInfo(self): + result = await self.sendCommand("deviceInfo") self._deviceInfo = result return self._deviceInfo @@ -32,64 +34,98 @@ def getDeviceInfo(self): def deviceInfo(self): return self._deviceInfo - def startScreensaver(self): - return self.sendCommand("startScreensaver") + async def startScreensaver(self): + await self.sendCommand("startScreensaver") - def stopScreensaver(self): - return self.sendCommand("stopScreensaver") + async def stopScreensaver(self): + await self.sendCommand("stopScreensaver") - def screenOn(self): - return self.sendCommand("screenOn") + async def screenOn(self): + await self.sendCommand("screenOn") - def screenOff(self): - return self.sendCommand("screenOff") + async def screenOff(self): + await self.sendCommand("screenOff") - def setScreenBrightness(self, brightness): - return self.sendCommand( + async def setScreenBrightness(self, brightness): + await self.sendCommand( "setStringSetting", key="screenBrightness", value=brightness ) - def setAudioVolume(self, volume, stream=None): - return self.sendCommand("setAudioVolume", volume=volume, stream=stream) + async def setAudioVolume(self, volume, stream=None): + await self.sendCommand("setAudioVolume", level=volume, stream=stream) - def restartApp(self): - return self.sendCommand("restartApp") + async def restartApp(self): + await self.sendCommand("restartApp") - def loadStartUrl(self): - return self.sendCommand("loadStartUrl") + async def loadStartUrl(self): + await self.sendCommand("loadStartUrl") - def loadUrl(self, url): - return self.sendCommand("loadUrl", url=url) + async def loadUrl(self, url): + await self.sendCommand("loadUrl", url=url) - def playSound(self, url, stream=None): - return self.sendCommand("playSound", url=url, stream=stream) + async def playSound(self, url, stream=None): + await self.sendCommand("playSound", url=url, stream=stream) - def stopSound(self): - return self.sendCommand("stopSound") + async def stopSound(self): + await self.sendCommand("stopSound") - def toForeground(self): - return self.sendCommand("toForeground") + async def toForeground(self): + await self.sendCommand("toForeground") - def startApplication(self, application): - return self.sendCommand("startApplication", package=application) + async def startApplication(self, application): + await self.sendCommand("startApplication", package=application) - def setConfigurationString(self, setting, stringValue): - return self.sendCommand("setStringSetting", key=setting, value=stringValue) + async def setConfigurationString(self, setting, stringValue): + await self.sendCommand("setStringSetting", key=setting, value=stringValue) - def setConfigurationBool(self, setting, boolValue): - return self.sendCommand("setBooleanSetting", key=setting, value=boolValue) + async def setConfigurationBool(self, setting, boolValue): + await self.sendCommand("setBooleanSetting", key=setting, value=boolValue) - def enableLockedMode(self): - return self.sendCommand("enableLockedMode") + async def enableLockedMode(self): + await self.sendCommand("enableLockedMode") - def disableLockedMode(self): - return self.sendCommand("disableLockedMode") + async def disableLockedMode(self): + await self.sendCommand("disableLockedMode") - def lockKiosk(self): - return self.sendCommand("lockKiosk") + async def lockKiosk(self): + await self.sendCommand("lockKiosk") - def unlockKiosk(self): - return self.sendCommand("unlockKiosk") + async def unlockKiosk(self): + await self.sendCommand("unlockKiosk") - def rebootDevice(self): - return self.sendCommand("rebootDevice") + async def rebootDevice(self): + await self.sendCommand("rebootDevice") + + +class _RequestsHandler: + """Internal class to create FullyKiosk requests""" + + def __init__(self, session: aiohttp.ClientSession, host, port): + self.headers = {"Accept": "application/json"} + + self.session = session + self.host = host + self.port = port + + async def get(self, **kwargs): + url = f"http://{self.host}:{self.port}" + params = [] + + for key, value in kwargs.items(): + if value is not None: + params.append((key, str(value))) + + _LOGGER.debug("Sending request to: %s", url) + _LOGGER.debug("Parameters: %s", params) + async with self.session.get( + url, headers=self.headers, params=params + ) as response: + if response.status != 200: + _LOGGER.warning( + "Invalid response from Fully Kiosk Browser API: %s", response.status + ) + raise FullyKioskError(response.status, await response.text()) + + data = await response.json(content_type="text/html") + _LOGGER.debug(json.dumps(data)) + return data diff --git a/fullykiosk/exceptions.py b/fullykiosk/exceptions.py new file mode 100644 index 0000000..c3236c5 --- /dev/null +++ b/fullykiosk/exceptions.py @@ -0,0 +1,10 @@ +class FullyKioskError(Exception): + """Raised when Fully Kiosk Browser API request ended in error. + Attributes: + status_code - error code returned by Fully Kiosk Browser + status - more detailed description + """ + + def __init__(self, status_code, status): + self.status_code = status_code + self.status = status diff --git a/requirements.txt b/requirements.txt index b7ec5aa..2b9a7c8 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1 +1 @@ -requests>=2.23.0 \ No newline at end of file +aiohttp>=3.6.3 diff --git a/setup.py b/setup.py index 1ccec82..36f1636 100644 --- a/setup.py +++ b/setup.py @@ -5,7 +5,7 @@ PROJECT_DIR = Path(__file__).parent.resolve() README_FILE = PROJECT_DIR / "README.md" -VERSION = "0.0.7" +VERSION = "0.0.8" setup(