Merge pull request #345 from rad4day/master

[modules/spaceapi] rewrite
This commit is contained in:
tobi-wan-kenobi 2019-01-07 20:23:17 +01:00 committed by GitHub
commit aa8167ed25
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23

View file

@ -1,17 +1,29 @@
#!/usr/bin/env python
# -*- coding: utf-8 -*-
# pylint: disable=C0111,R0903 # pylint: disable=C0111,R0903
"""Displays the state of a spaceapi endpoint """Displays the state of a Space API endpoint
Space API is an API for hackspaces based on JSON. See spaceapi.io for
an example.
Requires the following libraries: Requires the following libraries:
* requests * requests
* regex
Parameters: Parameters:
* spaceapi.url: String representation of the api endpoint * spaceapi.url: String representation of the api endpoint
* spaceapi.name: String overwriting the space name * spaceapi.format: Format string for the output
* spaceapi.prefix: Prefix for the space string
* spaceapi.interval: time between updates in minutes Format Strings:
* spaceapi.timeout: Maximum time in seconds to wait for a response from API * Format strings are indicated by double %%
endpoint * They represent a leaf in the JSON tree, layers seperated by "."
* Boolean values can be overwritten by appending "%true%false"
in the format string
* Example: to reference "open" in "{"state":{"open": true}}"
you would write "%%state.open%%", if you also want
to say "Open/Closed" depending on the boolean you
would write "%%state.open%Open%Closed%%"
""" """
import bumblebee.input import bumblebee.input
@ -19,6 +31,39 @@ import bumblebee.output
import bumblebee.engine import bumblebee.engine
import requests import requests
import threading
import re
import json
def formatStringBuilder(s, json):
"""
Parses Format Strings
Parameter:
s -> format string
json -> the spaceapi response object
"""
identifiers = re.findall("%%.*?%%", s)
for i in identifiers:
ic = i[2:-2] # Discard %%
j = ic.split("%")
# Only neither of, or both true AND false may be overwritten
if len(j) != 3 and len(j) != 1:
return "INVALID FORMAT STRING"
arr = j[0].split(".")
repl = json
for a in arr: # Walk the JSON tree to find replacement string
repl = repl[a]
if len(j) == 1: # no overwrite
s = s.replace(i, repl)
elif repl: # overwrite for Trfor True
s = s.replace(i, j[1])
else: # overwrite for False
s = s.replace(i, j[2])
return s
class Module(bumblebee.engine.Module): class Module(bumblebee.engine.Module):
@ -27,55 +72,70 @@ class Module(bumblebee.engine.Module):
engine, config, bumblebee.output.Widget(full_text=self.getState) engine, config, bumblebee.output.Widget(full_text=self.getState)
) )
# Represents the state of the hackerspace engine.input.register_callback(
self._open = False self, button=bumblebee.input.LEFT_MOUSE, cmd=self.__forceReload
# Set to true if there was an error calling the spaceapi )
self._error = False
self._data = {}
self._error = None
self._threadingCount = 0
# The URL representing the api endpoint # The URL representing the api endpoint
self._url = self.parameter("url", self._url = self.parameter("url", default="http://club.entropia.de/spaceapi")
default="http://club.entropia.de/spaceapi") self._format = self.parameter(
# Space Name, can be set manually in case of multiple widgets, "format", default=u"%%space%%: %%state.open%Open%Closed%%"
# so you're able to distinguish )
self._name = self.parameter("name", default="")
# The timeout prevents the statusbar from blocking when the destination
# can't be reached.
self._timeout = self.parameter("timeout", default=2)
# Only execute every 5 minutes by default
self.interval(self.parameter("interval", default=5))
def getState(self, widget):
text = self.parameter("prefix", default="")
text += self._name + ": "
if self._error:
text += "ERROR"
elif self._open:
text += "Open"
else:
text += "Closed"
return text
def state(self, widget): def state(self, widget):
if self._error: try:
if self._error is not None:
return ["critical"] return ["critical"]
elif self._open: elif self._data["state"]["open"]:
return ["warning"] return ["warning"]
else: else:
return [] return []
except KeyError:
return ["critical"]
def update(self, widgets): def update(self, widgets):
if self._threadingCount == 0:
thread = threading.Thread(target=self.get_api_async, args=())
thread.start()
self._threadingCount = (
0 if self._threadingCount > 300 else self._threadingCount + 1
)
def getState(self, widget):
text = self._format
if self._error is not None:
text = self._error
else:
try: try:
with requests.get(self._url, timeout=self.timeout) as u: text = formatStringBuilder(self._format, self._data)
json = u.json() except KeyError:
self._open = json["state"]["open"] text = "KeyError"
self._name = self.parameter("name", default=json["space"]) return text
self._error = False
except Exception: def get_api_async(self):
# Displays ERROR status try:
self._error = True with requests.get(self._url, timeout=10) as request:
# Can't implement error handling for python2.7 if I use
# request.json() as it uses simplejson in newer versions
self._data = json.loads(request.text)
self._error = None
except requests.exceptions.Timeout:
self._error = "Timeout"
except requests.exceptions.HTTPError:
self._error = "HTTP Error"
except ValueError:
self._error = "Not a JSON response"
# left_mouse_button handler
def __forceReload(self, event):
self._threadingCount += 300
self._error = "RELOADING"
# Author: Tobias Manske <tobias.manske@mailbox.org> # Author: Tobias Manske <tobias@chaoswg.xyz>
# vim: tabstop=8 expandtab shiftwidth=4 softtabstop=4 # vim: tabstop=8 expandtab shiftwidth=4 softtabstop=4