platypush/platypush/backend/http/request/__init__.py

122 lines
3.9 KiB
Python
Raw Normal View History

2018-06-08 16:55:06 +02:00
import logging
2018-01-09 18:44:45 +01:00
import re
import requests
import time
from frozendict import frozendict
from threading import Thread
2018-01-10 03:14:27 +01:00
from platypush.message.event.http import HttpEvent
from platypush.utils import set_thread_name
2018-01-09 18:44:45 +01:00
2020-09-27 01:33:38 +02:00
2018-01-10 03:14:27 +01:00
class HttpRequest(object):
2018-01-09 18:44:45 +01:00
poll_seconds = 60
timeout = 5
class HttpRequestArguments(object):
def __init__(self, url, method='get', *args, **kwargs):
self.method = method.lower()
self.url = url
self.args = args
self.kwargs = kwargs
def __init__(self, args, bus=None, poll_seconds=None, timeout=None,
skip_first_call=True, **kwargs):
2018-01-09 18:44:45 +01:00
super().__init__()
self.poll_seconds = poll_seconds or self.poll_seconds
self.timeout = timeout or self.timeout
2018-01-10 03:14:27 +01:00
self.bus = bus
self.skip_first_call = skip_first_call
2018-01-10 03:14:27 +01:00
self.last_request_timestamp = 0
2020-09-27 01:33:38 +02:00
self.logger = logging.getLogger('platypush')
2018-01-09 18:44:45 +01:00
if isinstance(args, self.HttpRequestArguments):
self.args = args
elif isinstance(args, dict):
self.args = self.HttpRequestArguments(**args)
else:
raise RuntimeError('{} is neither a dictionary nor an HttpRequest')
2018-01-11 19:31:44 +01:00
if 'timeout' not in self.args.kwargs:
self.args.kwargs['timeout'] = self.timeout
2018-01-10 03:14:27 +01:00
self.request_args = {
'method': self.args.method, 'url': self.args.url, **self.args.kwargs
}
2018-01-09 18:44:45 +01:00
def execute(self):
2018-01-10 03:14:27 +01:00
def _thread_func():
set_thread_name('HttpPoll')
2018-01-10 03:14:27 +01:00
is_first_call = self.last_request_timestamp == 0
self.last_request_timestamp = time.time()
try:
method = getattr(requests, self.args.method.lower())
response = method(self.args.url, *self.args.args, **self.args.kwargs)
new_items = self.get_new_items(response)
if isinstance(new_items, HttpEvent):
event = new_items
new_items = event.args['response']
else:
event = HttpEvent(dict(self), new_items)
if new_items and self.bus:
if not self.skip_first_call or (
self.skip_first_call and not is_first_call):
self.bus.post(event)
response.raise_for_status()
except Exception as e:
self.logger.exception(e)
self.logger.warning('Encountered an error while retrieving {}: {}'.
format(self.args.url, str(e)))
2018-01-10 03:14:27 +01:00
Thread(target=_thread_func, name='HttpPoll').start()
2018-01-09 18:44:45 +01:00
2018-01-10 03:14:27 +01:00
def get_new_items(self, response):
""" Gets new items out of a response """
2021-04-05 00:58:44 +02:00
raise NotImplementedError("get_new_items must be implemented in a derived class")
2018-01-10 03:14:27 +01:00
def __iter__(self):
for (key, value) in self.request_args.items():
2021-04-05 00:58:44 +02:00
yield key, value
2018-01-09 18:44:45 +01:00
class JsonHttpRequest(HttpRequest):
def __init__(self, path=None, *args, **kwargs):
super().__init__(*args, **kwargs)
self.path = path
self.seen_entries = set()
2018-01-10 03:14:27 +01:00
def get_new_items(self, response):
response = response.json()
2018-01-09 18:44:45 +01:00
new_entries = []
if self.path:
2021-04-05 00:58:44 +02:00
m = re.match(r'\${\s*(.*)\s*}', self.path)
2018-01-09 18:44:45 +01:00
response = eval(m.group(1))
for entry in response:
flattened_entry = deep_freeze(entry)
if flattened_entry not in self.seen_entries:
new_entries.append(entry)
self.seen_entries.add(flattened_entry)
return new_entries
def deep_freeze(x):
2020-09-27 01:33:38 +02:00
if isinstance(x, str) or not hasattr(x, "__len__"):
2018-01-09 18:44:45 +01:00
return x
2020-09-27 01:33:38 +02:00
if hasattr(x, "keys") and hasattr(x, "values"):
return frozendict({deep_freeze(k): deep_freeze(v) for k, v in x.items()})
if hasattr(x, "__getitem__"):
2018-01-09 18:44:45 +01:00
return tuple(map(deep_freeze, x))
2020-09-27 01:33:38 +02:00
return frozenset(map(deep_freeze, x))
2018-01-09 18:44:45 +01:00
# vim:sw=4:ts=4:et: