2020-04-25 16:28:04 +02:00
|
|
|
#!/usr/bin/env python
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
|
|
|
# pylint: disable=C0111,R0903
|
|
|
|
|
|
|
|
"""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:
|
|
|
|
* requests
|
|
|
|
* regex
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
* spaceapi.url: String representation of the api endpoint
|
|
|
|
* spaceapi.format: Format string for the output
|
|
|
|
|
|
|
|
Format Strings:
|
|
|
|
* Format strings are indicated by double %%
|
2020-04-25 16:28:19 +02:00
|
|
|
* They represent a leaf in the JSON tree, layers seperated by '.'
|
|
|
|
* Boolean values can be overwritten by appending '%true%false'
|
2020-04-25 16:28:04 +02:00
|
|
|
in the format string
|
2020-04-25 16:28:19 +02:00
|
|
|
* 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%%'
|
2020-04-25 16:28:04 +02:00
|
|
|
"""
|
|
|
|
|
|
|
|
import bumblebee.input
|
|
|
|
import bumblebee.output
|
|
|
|
import bumblebee.engine
|
|
|
|
|
|
|
|
import requests
|
|
|
|
import threading
|
|
|
|
import re
|
|
|
|
import json
|
|
|
|
|
|
|
|
|
|
|
|
def formatStringBuilder(s, json):
|
|
|
|
"""
|
|
|
|
Parses Format Strings
|
|
|
|
Parameter:
|
|
|
|
s -> format string
|
|
|
|
json -> the spaceapi response object
|
|
|
|
"""
|
2020-04-25 16:28:19 +02:00
|
|
|
identifiers = re.findall('%%.*?%%', s)
|
2020-04-25 16:28:04 +02:00
|
|
|
for i in identifiers:
|
|
|
|
ic = i[2:-2] # Discard %%
|
2020-04-25 16:28:19 +02:00
|
|
|
j = ic.split('%')
|
2020-04-25 16:28:04 +02:00
|
|
|
|
|
|
|
# Only neither of, or both true AND false may be overwritten
|
|
|
|
if len(j) != 3 and len(j) != 1:
|
2020-04-25 16:28:19 +02:00
|
|
|
return 'INVALID FORMAT STRING'
|
2020-04-25 16:28:04 +02:00
|
|
|
|
|
|
|
if len(j) == 1: # no overwrite
|
|
|
|
s = s.replace(i, json[j[0]])
|
|
|
|
elif json[j[0]]: # overwrite for True
|
|
|
|
s = s.replace(i, j[1])
|
|
|
|
else: # overwrite for False
|
|
|
|
s = s.replace(i, j[2])
|
|
|
|
return s
|
|
|
|
|
|
|
|
|
|
|
|
class Module(bumblebee.engine.Module):
|
|
|
|
def __init__(self, engine, config):
|
|
|
|
super(Module, self).__init__(
|
|
|
|
engine, config, bumblebee.output.Widget(full_text=self.getState)
|
|
|
|
)
|
|
|
|
|
|
|
|
engine.input.register_callback(
|
|
|
|
self, button=bumblebee.input.LEFT_MOUSE, cmd=self.__forceReload
|
|
|
|
)
|
|
|
|
|
|
|
|
self._data = {}
|
|
|
|
self._error = None
|
|
|
|
|
|
|
|
self._threadingCount = 0
|
|
|
|
|
|
|
|
# The URL representing the api endpoint
|
2020-04-25 16:28:19 +02:00
|
|
|
self._url = self.parameter('url', default='http://club.entropia.de/spaceapi')
|
2020-04-25 16:28:04 +02:00
|
|
|
self._format = self.parameter(
|
2020-04-25 16:28:19 +02:00
|
|
|
'format', default=u' %%space%%: %%state.open%Open%Closed%%'
|
2020-04-25 16:28:04 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
def state(self, widget):
|
|
|
|
try:
|
|
|
|
if self._error is not None:
|
2020-04-25 16:28:19 +02:00
|
|
|
return ['critical']
|
|
|
|
elif self._data['state.open']:
|
|
|
|
return ['warning']
|
2020-04-25 16:28:04 +02:00
|
|
|
else:
|
|
|
|
return []
|
|
|
|
except KeyError:
|
2020-04-25 16:28:19 +02:00
|
|
|
return ['critical']
|
2020-04-25 16:28:04 +02:00
|
|
|
|
|
|
|
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:
|
|
|
|
text = formatStringBuilder(self._format, self._data)
|
|
|
|
except KeyError:
|
2020-04-25 16:28:19 +02:00
|
|
|
text = 'KeyError'
|
2020-04-25 16:28:04 +02:00
|
|
|
return text
|
|
|
|
|
|
|
|
def get_api_async(self):
|
|
|
|
try:
|
|
|
|
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 = self.__flatten(json.loads(request.text))
|
|
|
|
self._error = None
|
|
|
|
except requests.exceptions.Timeout:
|
2020-04-25 16:28:19 +02:00
|
|
|
self._error = 'Timeout'
|
2020-04-25 16:28:04 +02:00
|
|
|
except requests.exceptions.HTTPError:
|
2020-04-25 16:28:19 +02:00
|
|
|
self._error = 'HTTP Error'
|
2020-04-25 16:28:04 +02:00
|
|
|
except ValueError:
|
2020-04-25 16:28:19 +02:00
|
|
|
self._error = 'Not a JSON response'
|
2020-04-25 16:28:04 +02:00
|
|
|
|
|
|
|
# left_mouse_button handler
|
|
|
|
def __forceReload(self, event):
|
|
|
|
self._threadingCount += 300
|
2020-04-25 16:28:19 +02:00
|
|
|
self._error = 'RELOADING'
|
2020-04-25 16:28:04 +02:00
|
|
|
|
2020-04-25 16:28:19 +02:00
|
|
|
# Flattens the JSON structure recursively, e.g. ['space']['open']
|
|
|
|
# becomes ['space.open']
|
2020-04-25 16:28:04 +02:00
|
|
|
def __flatten(self, json):
|
|
|
|
out = {}
|
|
|
|
for key in json:
|
|
|
|
value = json[key]
|
|
|
|
if type(value) is dict:
|
|
|
|
flattened_key = self.__flatten(value)
|
|
|
|
for fk in flattened_key:
|
2020-04-25 16:28:19 +02:00
|
|
|
out[key + '.' + fk] = flattened_key[fk]
|
2020-04-25 16:28:04 +02:00
|
|
|
else:
|
|
|
|
out[key] = value
|
|
|
|
return out
|
|
|
|
|
|
|
|
|
|
|
|
# Author: Tobias Manske <tobias@chaoswg.xyz>
|
|
|
|
# vim: tabstop=8 expandtab shiftwidth=4 softtabstop=4
|