Skip to content

Commit 15cff33

Browse files
authored
Migrate to aiohttp, async (#6)
* Migrate to aiohttp, async * Add error handling * Remove unused import
1 parent d5034de commit 15cff33

4 files changed

Lines changed: 107 additions & 61 deletions

File tree

fullykiosk/__init__.py

Lines changed: 95 additions & 59 deletions
Original file line numberDiff line numberDiff line change
@@ -1,95 +1,131 @@
1+
import aiohttp
12
import json
23
import logging
3-
import requests
44

5+
from .exceptions import FullyKioskError
56

6-
class FullyKiosk:
7-
def __init__(self, host, port, password):
8-
self.host = host
9-
self.port = port
10-
self.password = password
7+
_LOGGER = logging.getLogger(__name__)
118

12-
self._deviceInfo = None
9+
RESPONSE_STATUS = "status"
10+
RESPONSE_STATUSTEXT = "statustext"
11+
RESPONSE_ERRORSTATUS = "Error"
1312

14-
def sendCommand(self, cmd, **kwargs):
15-
url = f"http://{self.host}:{self.port}/?cmd={cmd}&password={self.password}&type=json"
16-
for key, value in kwargs.items():
17-
if value is not None:
18-
url = url + f"&{key}={value}"
1913

20-
try:
21-
result = json.loads(requests.get(url, timeout=10).content)
22-
return result
23-
except requests.exceptions.Timeout:
24-
print("Timeout error")
14+
class FullyKiosk:
15+
def __init__(self, session, host, port, password):
16+
self._rh = _RequestsHandler(session, host, port)
17+
self._password = password
18+
self._deviceInfo = None
19+
20+
async def sendCommand(self, cmd, **kwargs):
21+
data = await self._rh.get(
22+
cmd=cmd, password=self._password, type="json", **kwargs
23+
)
24+
if RESPONSE_STATUS in data and data[RESPONSE_STATUS] == RESPONSE_ERRORSTATUS:
25+
raise FullyKioskError(RESPONSE_ERRORSTATUS, data[RESPONSE_STATUSTEXT])
26+
return data
2527

26-
def getDeviceInfo(self):
27-
result = self.sendCommand("deviceInfo")
28+
async def getDeviceInfo(self):
29+
result = await self.sendCommand("deviceInfo")
2830
self._deviceInfo = result
2931
return self._deviceInfo
3032

3133
@property
3234
def deviceInfo(self):
3335
return self._deviceInfo
3436

35-
def startScreensaver(self):
36-
return self.sendCommand("startScreensaver")
37+
async def startScreensaver(self):
38+
await self.sendCommand("startScreensaver")
3739

38-
def stopScreensaver(self):
39-
return self.sendCommand("stopScreensaver")
40+
async def stopScreensaver(self):
41+
await self.sendCommand("stopScreensaver")
4042

41-
def screenOn(self):
42-
return self.sendCommand("screenOn")
43+
async def screenOn(self):
44+
await self.sendCommand("screenOn")
4345

44-
def screenOff(self):
45-
return self.sendCommand("screenOff")
46+
async def screenOff(self):
47+
await self.sendCommand("screenOff")
4648

47-
def setScreenBrightness(self, brightness):
48-
return self.sendCommand(
49+
async def setScreenBrightness(self, brightness):
50+
await self.sendCommand(
4951
"setStringSetting", key="screenBrightness", value=brightness
5052
)
5153

52-
def setAudioVolume(self, volume, stream=None):
53-
return self.sendCommand("setAudioVolume", volume=volume, stream=stream)
54+
async def setAudioVolume(self, volume, stream=None):
55+
await self.sendCommand("setAudioVolume", level=volume, stream=stream)
5456

55-
def restartApp(self):
56-
return self.sendCommand("restartApp")
57+
async def restartApp(self):
58+
await self.sendCommand("restartApp")
5759

58-
def loadStartUrl(self):
59-
return self.sendCommand("loadStartUrl")
60+
async def loadStartUrl(self):
61+
await self.sendCommand("loadStartUrl")
6062

61-
def loadUrl(self, url):
62-
return self.sendCommand("loadUrl", url=url)
63+
async def loadUrl(self, url):
64+
await self.sendCommand("loadUrl", url=url)
6365

64-
def playSound(self, url, stream=None):
65-
return self.sendCommand("playSound", url=url, stream=stream)
66+
async def playSound(self, url, stream=None):
67+
await self.sendCommand("playSound", url=url, stream=stream)
6668

67-
def stopSound(self):
68-
return self.sendCommand("stopSound")
69+
async def stopSound(self):
70+
await self.sendCommand("stopSound")
6971

70-
def toForeground(self):
71-
return self.sendCommand("toForeground")
72+
async def toForeground(self):
73+
await self.sendCommand("toForeground")
7274

73-
def startApplication(self, application):
74-
return self.sendCommand("startApplication", package=application)
75+
async def startApplication(self, application):
76+
await self.sendCommand("startApplication", package=application)
7577

76-
def setConfigurationString(self, setting, stringValue):
77-
return self.sendCommand("setStringSetting", key=setting, value=stringValue)
78+
async def setConfigurationString(self, setting, stringValue):
79+
await self.sendCommand("setStringSetting", key=setting, value=stringValue)
7880

79-
def setConfigurationBool(self, setting, boolValue):
80-
return self.sendCommand("setBooleanSetting", key=setting, value=boolValue)
81+
async def setConfigurationBool(self, setting, boolValue):
82+
await self.sendCommand("setBooleanSetting", key=setting, value=boolValue)
8183

82-
def enableLockedMode(self):
83-
return self.sendCommand("enableLockedMode")
84+
async def enableLockedMode(self):
85+
await self.sendCommand("enableLockedMode")
8486

85-
def disableLockedMode(self):
86-
return self.sendCommand("disableLockedMode")
87+
async def disableLockedMode(self):
88+
await self.sendCommand("disableLockedMode")
8789

88-
def lockKiosk(self):
89-
return self.sendCommand("lockKiosk")
90+
async def lockKiosk(self):
91+
await self.sendCommand("lockKiosk")
9092

91-
def unlockKiosk(self):
92-
return self.sendCommand("unlockKiosk")
93+
async def unlockKiosk(self):
94+
await self.sendCommand("unlockKiosk")
9395

94-
def rebootDevice(self):
95-
return self.sendCommand("rebootDevice")
96+
async def rebootDevice(self):
97+
await self.sendCommand("rebootDevice")
98+
99+
100+
class _RequestsHandler:
101+
"""Internal class to create FullyKiosk requests"""
102+
103+
def __init__(self, session: aiohttp.ClientSession, host, port):
104+
self.headers = {"Accept": "application/json"}
105+
106+
self.session = session
107+
self.host = host
108+
self.port = port
109+
110+
async def get(self, **kwargs):
111+
url = f"http://{self.host}:{self.port}"
112+
params = []
113+
114+
for key, value in kwargs.items():
115+
if value is not None:
116+
params.append((key, str(value)))
117+
118+
_LOGGER.debug("Sending request to: %s", url)
119+
_LOGGER.debug("Parameters: %s", params)
120+
async with self.session.get(
121+
url, headers=self.headers, params=params
122+
) as response:
123+
if response.status != 200:
124+
_LOGGER.warning(
125+
"Invalid response from Fully Kiosk Browser API: %s", response.status
126+
)
127+
raise FullyKioskError(response.status, await response.text())
128+
129+
data = await response.json(content_type="text/html")
130+
_LOGGER.debug(json.dumps(data))
131+
return data

fullykiosk/exceptions.py

Lines changed: 10 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
class FullyKioskError(Exception):
2+
"""Raised when Fully Kiosk Browser API request ended in error.
3+
Attributes:
4+
status_code - error code returned by Fully Kiosk Browser
5+
status - more detailed description
6+
"""
7+
8+
def __init__(self, status_code, status):
9+
self.status_code = status_code
10+
self.status = status

requirements.txt

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1 +1 @@
1-
requests>=2.23.0
1+
aiohttp>=3.6.3

setup.py

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@
55

66
PROJECT_DIR = Path(__file__).parent.resolve()
77
README_FILE = PROJECT_DIR / "README.md"
8-
VERSION = "0.0.7"
8+
VERSION = "0.0.8"
99

1010

1111
setup(

0 commit comments

Comments
 (0)