Compare commits
3 commits
fde834c1b1
...
e49a0aec4d
Author | SHA1 | Date | |
---|---|---|---|
e49a0aec4d | |||
9d028af524 | |||
419a0cec61 |
10 changed files with 150 additions and 199 deletions
|
@ -6,14 +6,14 @@ from multiprocessing import Process
|
||||||
|
|
||||||
try:
|
try:
|
||||||
from websockets.exceptions import ConnectionClosed
|
from websockets.exceptions import ConnectionClosed
|
||||||
from websockets import serve as websocket_serve
|
from websockets import serve as websocket_serve # type: ignore
|
||||||
except ImportError:
|
except ImportError:
|
||||||
from websockets import ConnectionClosed, serve as websocket_serve
|
from websockets import ConnectionClosed, serve as websocket_serve # type: ignore
|
||||||
|
|
||||||
from platypush.backend import Backend
|
from platypush.backend import Backend
|
||||||
from platypush.backend.http.app import application
|
from platypush.backend.http.app import application
|
||||||
from platypush.context import get_or_create_event_loop
|
from platypush.context import get_or_create_event_loop
|
||||||
from platypush.utils import get_ssl_server_context, set_thread_name
|
from platypush.utils import get_ssl_server_context
|
||||||
|
|
||||||
|
|
||||||
class HttpBackend(Backend):
|
class HttpBackend(Backend):
|
||||||
|
@ -186,7 +186,7 @@ class HttpBackend(Backend):
|
||||||
maps=None,
|
maps=None,
|
||||||
run_externally=False,
|
run_externally=False,
|
||||||
uwsgi_args=None,
|
uwsgi_args=None,
|
||||||
**kwargs
|
**kwargs,
|
||||||
):
|
):
|
||||||
"""
|
"""
|
||||||
:param port: Listen port for the web server (default: 8008)
|
:param port: Listen port for the web server (default: 8008)
|
||||||
|
@ -283,15 +283,13 @@ class HttpBackend(Backend):
|
||||||
'--enable-threads',
|
'--enable-threads',
|
||||||
]
|
]
|
||||||
|
|
||||||
self.local_base_url = '{proto}://localhost:{port}'.format(
|
protocol = 'https' if ssl_cert else 'http'
|
||||||
proto=('https' if ssl_cert else 'http'), port=self.port
|
self.local_base_url = f'{protocol}://localhost:{self.port}'
|
||||||
)
|
|
||||||
|
|
||||||
self._websocket_lock_timeout = 10
|
self._websocket_lock_timeout = 10
|
||||||
self._websocket_lock = threading.RLock()
|
self._websocket_lock = threading.RLock()
|
||||||
self._websocket_locks = {}
|
self._websocket_locks = {}
|
||||||
|
|
||||||
def send_message(self, msg, **kwargs):
|
def send_message(self, msg, *_, **__):
|
||||||
self.logger.warning('Use cURL or any HTTP client to query the HTTP backend')
|
self.logger.warning('Use cURL or any HTTP client to query the HTTP backend')
|
||||||
|
|
||||||
def on_stop(self):
|
def on_stop(self):
|
||||||
|
@ -351,9 +349,7 @@ class HttpBackend(Backend):
|
||||||
timeout=self._websocket_lock_timeout
|
timeout=self._websocket_lock_timeout
|
||||||
)
|
)
|
||||||
if not acquire_ok:
|
if not acquire_ok:
|
||||||
raise TimeoutError(
|
raise TimeoutError(f'Websocket on address {addr} not ready to receive data')
|
||||||
'Websocket on address {} not ready to receive data'.format(addr)
|
|
||||||
)
|
|
||||||
|
|
||||||
def _release_websocket_lock(self, ws):
|
def _release_websocket_lock(self, ws):
|
||||||
try:
|
try:
|
||||||
|
@ -368,7 +364,7 @@ class HttpBackend(Backend):
|
||||||
self._websocket_locks[addr].release()
|
self._websocket_locks[addr].release()
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.logger.warning(
|
self.logger.warning(
|
||||||
'Unhandled exception while releasing websocket lock: {}'.format(str(e))
|
'Unhandled exception while releasing websocket lock: %s', e
|
||||||
)
|
)
|
||||||
finally:
|
finally:
|
||||||
self._websocket_lock.release()
|
self._websocket_lock.release()
|
||||||
|
@ -381,7 +377,7 @@ class HttpBackend(Backend):
|
||||||
self._acquire_websocket_lock(ws)
|
self._acquire_websocket_lock(ws)
|
||||||
await ws.send(str(event))
|
await ws.send(str(event))
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.logger.warning('Error on websocket send_event: {}'.format(e))
|
self.logger.warning('Error on websocket send_event: %s', e)
|
||||||
finally:
|
finally:
|
||||||
self._release_websocket_lock(ws)
|
self._release_websocket_lock(ws)
|
||||||
|
|
||||||
|
@ -393,7 +389,7 @@ class HttpBackend(Backend):
|
||||||
loop.run_until_complete(send_event(_ws))
|
loop.run_until_complete(send_event(_ws))
|
||||||
except ConnectionClosed:
|
except ConnectionClosed:
|
||||||
self.logger.warning(
|
self.logger.warning(
|
||||||
'Websocket client {} connection lost'.format(_ws.remote_address)
|
'Websocket client %s connection lost', _ws.remote_address
|
||||||
)
|
)
|
||||||
self.active_websockets.remove(_ws)
|
self.active_websockets.remove(_ws)
|
||||||
if _ws.remote_address in self._websocket_locks:
|
if _ws.remote_address in self._websocket_locks:
|
||||||
|
@ -401,7 +397,6 @@ class HttpBackend(Backend):
|
||||||
|
|
||||||
def websocket(self):
|
def websocket(self):
|
||||||
"""Websocket main server"""
|
"""Websocket main server"""
|
||||||
set_thread_name('WebsocketServer')
|
|
||||||
|
|
||||||
async def register_websocket(websocket, path):
|
async def register_websocket(websocket, path):
|
||||||
address = (
|
address = (
|
||||||
|
@ -411,16 +406,14 @@ class HttpBackend(Backend):
|
||||||
)
|
)
|
||||||
|
|
||||||
self.logger.info(
|
self.logger.info(
|
||||||
'New websocket connection from {} on path {}'.format(address, path)
|
'New websocket connection from %s on path %s', address, path
|
||||||
)
|
)
|
||||||
self.active_websockets.add(websocket)
|
self.active_websockets.add(websocket)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
await websocket.recv()
|
await websocket.recv()
|
||||||
except ConnectionClosed:
|
except ConnectionClosed:
|
||||||
self.logger.info(
|
self.logger.info('Websocket client %s closed connection', address)
|
||||||
'Websocket client {} closed connection'.format(address)
|
|
||||||
)
|
|
||||||
self.active_websockets.remove(websocket)
|
self.active_websockets.remove(websocket)
|
||||||
if address in self._websocket_locks:
|
if address in self._websocket_locks:
|
||||||
del self._websocket_locks[address]
|
del self._websocket_locks[address]
|
||||||
|
@ -435,14 +428,14 @@ class HttpBackend(Backend):
|
||||||
register_websocket,
|
register_websocket,
|
||||||
self.bind_address,
|
self.bind_address,
|
||||||
self.websocket_port,
|
self.websocket_port,
|
||||||
**websocket_args
|
**websocket_args,
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
self._websocket_loop.run_forever()
|
self._websocket_loop.run_forever()
|
||||||
|
|
||||||
def _start_web_server(self):
|
def _start_web_server(self):
|
||||||
def proc():
|
def proc():
|
||||||
self.logger.info('Starting local web server on port {}'.format(self.port))
|
self.logger.info('Starting local web server on port %s', self.port)
|
||||||
kwargs = {
|
kwargs = {
|
||||||
'host': self.bind_address,
|
'host': self.bind_address,
|
||||||
'port': self.port,
|
'port': self.port,
|
||||||
|
@ -470,7 +463,10 @@ class HttpBackend(Backend):
|
||||||
|
|
||||||
if not self.disable_websocket:
|
if not self.disable_websocket:
|
||||||
self.logger.info('Initializing websocket interface')
|
self.logger.info('Initializing websocket interface')
|
||||||
self.websocket_thread = threading.Thread(target=self.websocket)
|
self.websocket_thread = threading.Thread(
|
||||||
|
target=self.websocket,
|
||||||
|
name='WebsocketServer',
|
||||||
|
)
|
||||||
self.websocket_thread.start()
|
self.websocket_thread.start()
|
||||||
|
|
||||||
if not self.run_externally:
|
if not self.run_externally:
|
||||||
|
@ -481,13 +477,13 @@ class HttpBackend(Backend):
|
||||||
self.server_proc.join()
|
self.server_proc.join()
|
||||||
elif self.uwsgi_args:
|
elif self.uwsgi_args:
|
||||||
uwsgi_cmd = ['uwsgi'] + self.uwsgi_args
|
uwsgi_cmd = ['uwsgi'] + self.uwsgi_args
|
||||||
self.logger.info('Starting uWSGI with arguments {}'.format(uwsgi_cmd))
|
self.logger.info('Starting uWSGI with arguments %s', uwsgi_cmd)
|
||||||
self.server_proc = subprocess.Popen(uwsgi_cmd)
|
self.server_proc = subprocess.Popen(uwsgi_cmd)
|
||||||
else:
|
else:
|
||||||
self.logger.info(
|
self.logger.info(
|
||||||
'The web server is configured to be launched externally but '
|
'The web server is configured to be launched externally but '
|
||||||
+ 'no uwsgi_args were provided. Make sure that you run another external service'
|
'no uwsgi_args were provided. Make sure that you run another external service'
|
||||||
+ 'for the webserver (e.g. nginx)'
|
'for the web server (e.g. nginx)'
|
||||||
)
|
)
|
||||||
|
|
||||||
self._service_registry_thread = threading.Thread(target=self._register_service)
|
self._service_registry_thread = threading.Thread(target=self._register_service)
|
||||||
|
|
|
@ -13,7 +13,6 @@ from platypush.message import Message
|
||||||
from platypush.message.event.mqtt import MQTTMessageEvent
|
from platypush.message.event.mqtt import MQTTMessageEvent
|
||||||
from platypush.message.request import Request
|
from platypush.message.request import Request
|
||||||
from platypush.plugins.mqtt import MqttPlugin as MQTTPlugin
|
from platypush.plugins.mqtt import MqttPlugin as MQTTPlugin
|
||||||
from platypush.utils import set_thread_name
|
|
||||||
|
|
||||||
|
|
||||||
class MqttClient(mqtt.Client, threading.Thread):
|
class MqttClient(mqtt.Client, threading.Thread):
|
||||||
|
@ -39,6 +38,7 @@ class MqttClient(mqtt.Client, threading.Thread):
|
||||||
mqtt.Client.__init__(self, *args, client_id=client_id, **kwargs)
|
mqtt.Client.__init__(self, *args, client_id=client_id, **kwargs)
|
||||||
threading.Thread.__init__(self)
|
threading.Thread.__init__(self)
|
||||||
|
|
||||||
|
self.name = f'MQTTClient:{client_id}'
|
||||||
self.host = host
|
self.host = host
|
||||||
self.port = port
|
self.port = port
|
||||||
self.topics = set(topics or [])
|
self.topics = set(topics or [])
|
||||||
|
@ -393,7 +393,6 @@ class MqttBackend(Backend):
|
||||||
def handler(_, __, msg):
|
def handler(_, __, msg):
|
||||||
# noinspection PyShadowingNames
|
# noinspection PyShadowingNames
|
||||||
def response_thread(msg):
|
def response_thread(msg):
|
||||||
set_thread_name('MQTTProcessor')
|
|
||||||
response = self.get_message_response(msg)
|
response = self.get_message_response(msg)
|
||||||
if not response:
|
if not response:
|
||||||
return
|
return
|
||||||
|
@ -428,7 +427,9 @@ class MqttBackend(Backend):
|
||||||
|
|
||||||
if isinstance(msg, Request):
|
if isinstance(msg, Request):
|
||||||
threading.Thread(
|
threading.Thread(
|
||||||
target=response_thread, name='MQTTProcessor', args=(msg,)
|
target=response_thread,
|
||||||
|
name='MQTTProcessorResponseThread',
|
||||||
|
args=(msg,),
|
||||||
).start()
|
).start()
|
||||||
|
|
||||||
return handler
|
return handler
|
||||||
|
|
|
@ -35,7 +35,7 @@ class MultiLevelSwitchEntityManager(EntityManager, ABC):
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def set_value( # pylint: disable=redefined-builtin
|
def set_value( # pylint: disable=redefined-builtin
|
||||||
self, *entities, property=None, data=None, **__
|
self, *_, property=None, data=None, **__
|
||||||
):
|
):
|
||||||
"""Set a value"""
|
"""Set a value"""
|
||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
import asyncio
|
import asyncio
|
||||||
import logging
|
import logging
|
||||||
import threading
|
import threading
|
||||||
import time
|
|
||||||
|
|
||||||
from abc import ABC, abstractmethod
|
from abc import ABC, abstractmethod
|
||||||
from functools import wraps
|
from functools import wraps
|
||||||
|
@ -11,7 +10,7 @@ from platypush.bus import Bus
|
||||||
from platypush.common import ExtensionWithManifest
|
from platypush.common import ExtensionWithManifest
|
||||||
from platypush.event import EventGenerator
|
from platypush.event import EventGenerator
|
||||||
from platypush.message.response import Response
|
from platypush.message.response import Response
|
||||||
from platypush.utils import get_decorators, get_plugin_name_by_class, set_thread_name
|
from platypush.utils import get_decorators, get_plugin_name_by_class
|
||||||
|
|
||||||
PLUGIN_STOP_TIMEOUT = 5 # Plugin stop timeout in seconds
|
PLUGIN_STOP_TIMEOUT = 5 # Plugin stop timeout in seconds
|
||||||
|
|
||||||
|
@ -107,25 +106,22 @@ class RunnablePlugin(Plugin):
|
||||||
return self._should_stop.wait(timeout=timeout)
|
return self._should_stop.wait(timeout=timeout)
|
||||||
|
|
||||||
def start(self):
|
def start(self):
|
||||||
set_thread_name(self.__class__.__name__)
|
self._thread = threading.Thread(
|
||||||
self._thread = threading.Thread(target=self._runner)
|
target=self._runner, name=self.__class__.__name__
|
||||||
|
)
|
||||||
self._thread.start()
|
self._thread.start()
|
||||||
|
|
||||||
def stop(self):
|
def stop(self):
|
||||||
self._should_stop.set()
|
self._should_stop.set()
|
||||||
if self._thread and self._thread.is_alive():
|
if self._thread and self._thread.is_alive():
|
||||||
self.logger.info('Waiting for %s to stop', self.__class__.__name__)
|
self.logger.info('Waiting for the plugin to stop')
|
||||||
try:
|
try:
|
||||||
if self._thread:
|
if self._thread:
|
||||||
self._thread.join(timeout=self._stop_timeout)
|
self._thread.join(timeout=self._stop_timeout)
|
||||||
if self._thread and self._thread.is_alive():
|
if self._thread and self._thread.is_alive():
|
||||||
self.logger.warning(
|
self.logger.warning(
|
||||||
'Timeout (seconds={%s}) on exit for the plugin %s',
|
'Timeout (seconds=%s) on exit for the plugin',
|
||||||
self._stop_timeout,
|
self._stop_timeout,
|
||||||
(
|
|
||||||
get_plugin_name_by_class(self.__class__)
|
|
||||||
or self.__class__.__name__
|
|
||||||
),
|
|
||||||
)
|
)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.logger.warning('Could not join thread on stop: %s', e)
|
self.logger.warning('Could not join thread on stop: %s', e)
|
||||||
|
@ -142,7 +138,7 @@ class RunnablePlugin(Plugin):
|
||||||
self.logger.exception(e)
|
self.logger.exception(e)
|
||||||
|
|
||||||
if self.poll_interval:
|
if self.poll_interval:
|
||||||
time.sleep(self.poll_interval)
|
self.wait_stop(self.poll_interval)
|
||||||
|
|
||||||
self._thread = None
|
self._thread = None
|
||||||
|
|
||||||
|
@ -156,18 +152,27 @@ class AsyncRunnablePlugin(RunnablePlugin, ABC):
|
||||||
super().__init__(*args, **kwargs)
|
super().__init__(*args, **kwargs)
|
||||||
|
|
||||||
self._loop: Optional[asyncio.AbstractEventLoop] = None
|
self._loop: Optional[asyncio.AbstractEventLoop] = None
|
||||||
self._loop_runner: Optional[threading.Thread] = None
|
|
||||||
self._task: Optional[asyncio.Task] = None
|
self._task: Optional[asyncio.Task] = None
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def _should_start_runner(self):
|
def _should_start_runner(self):
|
||||||
|
"""
|
||||||
|
This property is used to determine if the runner and the event loop
|
||||||
|
should be started for this plugin.
|
||||||
|
"""
|
||||||
return True
|
return True
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
async def listen(self):
|
async def listen(self):
|
||||||
pass
|
"""
|
||||||
|
Main body of the async plugin. When it's called, the event loop should
|
||||||
|
already be running and available over `self._loop`.
|
||||||
|
"""
|
||||||
|
|
||||||
async def _listen(self):
|
async def _listen(self):
|
||||||
|
"""
|
||||||
|
Wrapper for :meth:`.listen` that catches any exceptions and logs them.
|
||||||
|
"""
|
||||||
try:
|
try:
|
||||||
await self.listen()
|
await self.listen()
|
||||||
except KeyboardInterrupt:
|
except KeyboardInterrupt:
|
||||||
|
@ -179,41 +184,37 @@ class AsyncRunnablePlugin(RunnablePlugin, ABC):
|
||||||
):
|
):
|
||||||
raise e
|
raise e
|
||||||
|
|
||||||
def _start_listener(self):
|
def _run_listener(self):
|
||||||
set_thread_name(self.__class__.__name__ + ':listener')
|
|
||||||
self._loop = asyncio.new_event_loop()
|
self._loop = asyncio.new_event_loop()
|
||||||
asyncio.set_event_loop(self._loop)
|
asyncio.set_event_loop(self._loop)
|
||||||
|
|
||||||
self._task = self._loop.create_task(self._listen())
|
self._task = self._loop.create_task(self._listen())
|
||||||
if hasattr(self._task, 'set_name'):
|
if hasattr(self._task, 'set_name'):
|
||||||
self._task.set_name(self.__class__.__name__ + '.listen')
|
self._task.set_name(self.__class__.__name__ + '.listen')
|
||||||
self._loop.run_forever()
|
try:
|
||||||
|
self._loop.run_until_complete(self._task)
|
||||||
|
except Exception as e:
|
||||||
|
self.logger.info('The loop has terminated with an error: %s', e)
|
||||||
|
|
||||||
|
self._task.cancel()
|
||||||
|
|
||||||
def main(self):
|
def main(self):
|
||||||
if self.should_stop() or (self._loop_runner and self._loop_runner.is_alive()):
|
if self.should_stop():
|
||||||
self.logger.info('The main loop is already being run/stopped')
|
self.logger.info('The plugin is already scheduled to stop')
|
||||||
return
|
return
|
||||||
|
|
||||||
|
self._loop = asyncio.new_event_loop()
|
||||||
|
|
||||||
if self._should_start_runner:
|
if self._should_start_runner:
|
||||||
self._loop_runner = threading.Thread(target=self._start_listener)
|
self._run_listener()
|
||||||
self._loop_runner.start()
|
|
||||||
|
|
||||||
self.wait_stop()
|
self.wait_stop()
|
||||||
|
|
||||||
def stop(self):
|
def stop(self):
|
||||||
if self._task and self._loop and not self._task.done():
|
|
||||||
self._loop.call_soon_threadsafe(self._task.cancel)
|
|
||||||
|
|
||||||
if self._loop and self._loop.is_running():
|
if self._loop and self._loop.is_running():
|
||||||
self._loop.call_soon_threadsafe(self._loop.stop)
|
self._loop.call_soon_threadsafe(self._loop.stop)
|
||||||
self._loop = None
|
self._loop = None
|
||||||
|
|
||||||
if self._loop_runner and self._loop_runner.is_alive():
|
|
||||||
try:
|
|
||||||
self._loop_runner.join(timeout=self._stop_timeout)
|
|
||||||
finally:
|
|
||||||
self._loop_runner = None
|
|
||||||
|
|
||||||
super().stop()
|
super().stop()
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
import asyncio
|
import asyncio
|
||||||
import json
|
import json
|
||||||
import os
|
import os
|
||||||
from typing import Optional, Collection, Mapping
|
from typing import Any, Callable, Dict, Optional, Collection, Mapping
|
||||||
|
|
||||||
import requests
|
import requests
|
||||||
import websockets
|
import websockets
|
||||||
|
@ -53,12 +53,12 @@ class NtfyPlugin(AsyncRunnablePlugin):
|
||||||
reconnect_wait_secs_max = 60
|
reconnect_wait_secs_max = 60
|
||||||
|
|
||||||
while not self.should_stop():
|
while not self.should_stop():
|
||||||
self.logger.debug(f'Connecting to {url}')
|
self.logger.debug('Connecting to %s', url)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
async with websockets.connect(url) as ws: # type: ignore
|
async with websockets.connect(url) as ws: # pylint: disable=no-member
|
||||||
reconnect_wait_secs = 1
|
reconnect_wait_secs = 1
|
||||||
self.logger.info(f'Connected to {url}')
|
self.logger.info('Connected to %s', url)
|
||||||
async for msg in ws:
|
async for msg in ws:
|
||||||
try:
|
try:
|
||||||
msg = json.loads(msg)
|
msg = json.loads(msg)
|
||||||
|
@ -122,7 +122,7 @@ class NtfyPlugin(AsyncRunnablePlugin):
|
||||||
tags: Optional[Collection[str]] = None,
|
tags: Optional[Collection[str]] = None,
|
||||||
schedule: Optional[str] = None,
|
schedule: Optional[str] = None,
|
||||||
):
|
):
|
||||||
"""
|
r"""
|
||||||
Send a message/notification to a topic.
|
Send a message/notification to a topic.
|
||||||
|
|
||||||
:param topic: Topic where the message will be delivered.
|
:param topic: Topic where the message will be delivered.
|
||||||
|
@ -148,36 +148,36 @@ class NtfyPlugin(AsyncRunnablePlugin):
|
||||||
- ``broadcast``: Send an `Android broadcast <https://developer.android.com/guide/components/broadcasts>`
|
- ``broadcast``: Send an `Android broadcast <https://developer.android.com/guide/components/broadcasts>`
|
||||||
intent upon action selection (only available on Android).
|
intent upon action selection (only available on Android).
|
||||||
|
|
||||||
Example:
|
Actions example:
|
||||||
|
|
||||||
.. code-block:: json
|
.. code-block:: json
|
||||||
|
|
||||||
[
|
[
|
||||||
{
|
{
|
||||||
"action": "view",
|
"action": "view",
|
||||||
"label": "Open portal",
|
"label": "Open portal",
|
||||||
"url": "https://home.nest.com/",
|
"url": "https://home.nest.com/",
|
||||||
"clear": true
|
"clear": true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"action": "http",
|
||||||
|
"label": "Turn down",
|
||||||
|
"url": "https://api.nest.com/",
|
||||||
|
"method": "PUT",
|
||||||
|
"headers": {
|
||||||
|
"Authorization": "Bearer abcdef..."
|
||||||
},
|
},
|
||||||
{
|
"body": "{\\"temperature\\": 65}"
|
||||||
"action": "http",
|
},
|
||||||
"label": "Turn down",
|
{
|
||||||
"url": "https://api.nest.com/",
|
"action": "broadcast",
|
||||||
"method": "PUT",
|
"label": "Take picture",
|
||||||
"headers": {
|
"intent": "com.myapp.TAKE_PICTURE_INTENT",
|
||||||
"Authorization": "Bearer abcdef..."
|
"extras": {
|
||||||
},
|
"camera": "front"
|
||||||
"body": "{\\"temperature\\": 65}"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"action": "broadcast",
|
|
||||||
"label": "Take picture",
|
|
||||||
"intent": "com.myapp.TAKE_PICTURE_INTENT",
|
|
||||||
"extras": {
|
|
||||||
"camera": "front"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
]
|
}
|
||||||
|
]
|
||||||
|
|
||||||
:param email: Forward the notification as an email to the specified
|
:param email: Forward the notification as an email to the specified
|
||||||
address.
|
address.
|
||||||
|
@ -196,9 +196,9 @@ class NtfyPlugin(AsyncRunnablePlugin):
|
||||||
``tomorrow, 3pm``)
|
``tomorrow, 3pm``)
|
||||||
|
|
||||||
"""
|
"""
|
||||||
method = requests.post
|
method: Callable[..., requests.Response] = requests.post
|
||||||
url = server_url or self._server_url
|
url = server_url or self._server_url
|
||||||
args = {}
|
args: Dict[str, Any] = {}
|
||||||
if username and password:
|
if username and password:
|
||||||
args['auth'] = (username, password)
|
args['auth'] = (username, password)
|
||||||
|
|
||||||
|
@ -247,8 +247,8 @@ class NtfyPlugin(AsyncRunnablePlugin):
|
||||||
}
|
}
|
||||||
|
|
||||||
rs = method(url, **args)
|
rs = method(url, **args)
|
||||||
assert rs.ok, 'Could not send message to {}: {}'.format(
|
assert rs.ok, f'Could not send message to {topic}: ' + rs.json().get(
|
||||||
topic, rs.json().get('error', f'HTTP error: {rs.status_code}')
|
'error', f'HTTP error: {rs.status_code}'
|
||||||
)
|
)
|
||||||
|
|
||||||
return rs.json()
|
return rs.json()
|
||||||
|
|
|
@ -859,8 +859,9 @@ class SmartthingsPlugin(
|
||||||
return self.status(device)
|
return self.status(device)
|
||||||
|
|
||||||
@action
|
@action
|
||||||
def set_value( # pylint: disable=arguments-differ,redefined-builtin
|
# pylint: disable=redefined-builtin,arguments-differ
|
||||||
self, device: str, property: Optional[str] = None, data=None, **kwargs
|
def set_value(
|
||||||
|
self, device: str, *_, property: Optional[str] = None, data=None, **kwargs
|
||||||
):
|
):
|
||||||
"""
|
"""
|
||||||
Set the value of a device. It is compatible with the generic
|
Set the value of a device. It is compatible with the generic
|
||||||
|
|
|
@ -1,72 +0,0 @@
|
||||||
from abc import ABC, abstractmethod
|
|
||||||
from typing import List, Union
|
|
||||||
|
|
||||||
from platypush.plugins import Plugin, action
|
|
||||||
|
|
||||||
|
|
||||||
# TODO REMOVE ME
|
|
||||||
class SwitchPlugin(Plugin, ABC):
|
|
||||||
"""
|
|
||||||
Abstract class for interacting with switch devices
|
|
||||||
"""
|
|
||||||
|
|
||||||
def __init__(self, **kwargs):
|
|
||||||
super().__init__(**kwargs)
|
|
||||||
|
|
||||||
@action
|
|
||||||
@abstractmethod
|
|
||||||
def on(self, device, *args, **kwargs):
|
|
||||||
"""Turn the device on"""
|
|
||||||
raise NotImplementedError()
|
|
||||||
|
|
||||||
@action
|
|
||||||
@abstractmethod
|
|
||||||
def off(self, device, *args, **kwargs):
|
|
||||||
"""Turn the device off"""
|
|
||||||
raise NotImplementedError()
|
|
||||||
|
|
||||||
@action
|
|
||||||
@abstractmethod
|
|
||||||
def toggle(self, device, *args, **kwargs):
|
|
||||||
"""Toggle the device status (on/off)"""
|
|
||||||
raise NotImplementedError()
|
|
||||||
|
|
||||||
@action
|
|
||||||
def switch_status(self, device=None) -> Union[dict, List[dict]]:
|
|
||||||
"""
|
|
||||||
Get the status of a specified device or of all the configured devices (default).
|
|
||||||
|
|
||||||
:param device: Filter by device name or ID.
|
|
||||||
:return: .. schema:: switch.SwitchStatusSchema(many=True)
|
|
||||||
"""
|
|
||||||
devices = self.switches
|
|
||||||
if device:
|
|
||||||
devices = [
|
|
||||||
d
|
|
||||||
for d in self.switches
|
|
||||||
if d.get('id') == device or d.get('name') == device
|
|
||||||
]
|
|
||||||
return devices[0] if devices else []
|
|
||||||
|
|
||||||
return devices
|
|
||||||
|
|
||||||
@action
|
|
||||||
def status(self, device=None, *_, **__) -> Union[dict, List[dict]]:
|
|
||||||
"""
|
|
||||||
Get the status of all the devices, or filter by device name or ID (alias for :meth:`.switch_status`).
|
|
||||||
|
|
||||||
:param device: Filter by device name or ID.
|
|
||||||
:return: .. schema:: switch.SwitchStatusSchema(many=True)
|
|
||||||
"""
|
|
||||||
return self.switch_status(device)
|
|
||||||
|
|
||||||
@property
|
|
||||||
@abstractmethod
|
|
||||||
def switches(self) -> List[dict]:
|
|
||||||
"""
|
|
||||||
:return: .. schema:: switch.SwitchStatusSchema(many=True)
|
|
||||||
"""
|
|
||||||
raise NotImplementedError()
|
|
||||||
|
|
||||||
|
|
||||||
# vim:sw=4:ts=4:et:
|
|
|
@ -1105,7 +1105,7 @@ class SwitchbotPlugin(
|
||||||
|
|
||||||
@action
|
@action
|
||||||
# pylint: disable=redefined-builtin,arguments-differ
|
# pylint: disable=redefined-builtin,arguments-differ
|
||||||
def set_value(self, device: str, property=None, data=None, **__):
|
def set_value(self, device: str, *_, property=None, data=None, **__):
|
||||||
entity = self._to_entity(device, property)
|
entity = self._to_entity(device, property)
|
||||||
assert entity, f'No such entity: "{device}"'
|
assert entity, f'No such entity: "{device}"'
|
||||||
|
|
||||||
|
|
|
@ -1,16 +1,16 @@
|
||||||
import enum
|
import enum
|
||||||
import time
|
import time
|
||||||
from typing import List, Optional
|
from typing import Any, Collection, Dict, List, Optional
|
||||||
|
|
||||||
|
from platypush.entities import EnumSwitchEntityManager
|
||||||
|
from platypush.entities.switches import EnumSwitch
|
||||||
from platypush.message.response.bluetooth import BluetoothScanResponse
|
from platypush.message.response.bluetooth import BluetoothScanResponse
|
||||||
from platypush.plugins import action
|
from platypush.plugins import action
|
||||||
from platypush.plugins.bluetooth.ble import BluetoothBlePlugin
|
from platypush.plugins.bluetooth.ble import BluetoothBlePlugin
|
||||||
from platypush.plugins.switch import SwitchPlugin
|
|
||||||
|
|
||||||
|
|
||||||
class SwitchbotBluetoothPlugin( # lgtm [py/missing-call-to-init]
|
# pylint: disable=too-many-ancestors
|
||||||
SwitchPlugin, BluetoothBlePlugin
|
class SwitchbotBluetoothPlugin(BluetoothBlePlugin, EnumSwitchEntityManager):
|
||||||
):
|
|
||||||
"""
|
"""
|
||||||
Plugin to interact with a Switchbot (https://www.switch-bot.com/) device and
|
Plugin to interact with a Switchbot (https://www.switch-bot.com/) device and
|
||||||
programmatically control switches over a Bluetooth interface.
|
programmatically control switches over a Bluetooth interface.
|
||||||
|
@ -91,7 +91,7 @@ class SwitchbotBluetoothPlugin( # lgtm [py/missing-call-to-init]
|
||||||
raise e
|
raise e
|
||||||
time.sleep(5)
|
time.sleep(5)
|
||||||
|
|
||||||
return self.status(device) # type: ignore
|
return self.status(device)
|
||||||
|
|
||||||
@action
|
@action
|
||||||
def press(self, device):
|
def press(self, device):
|
||||||
|
@ -127,6 +127,30 @@ class SwitchbotBluetoothPlugin( # lgtm [py/missing-call-to-init]
|
||||||
"""
|
"""
|
||||||
return self._run(device, self.Command.OFF)
|
return self._run(device, self.Command.OFF)
|
||||||
|
|
||||||
|
@action
|
||||||
|
# pylint: disable=arguments-differ
|
||||||
|
def set_value(self, device: str, data: str, *_, **__):
|
||||||
|
"""
|
||||||
|
Entity-compatible ``set_value`` method to send a command to a device.
|
||||||
|
|
||||||
|
:param device: Device name or address
|
||||||
|
:param data: Command to send. Possible values are:
|
||||||
|
|
||||||
|
- ``on``: Press the button and remain in the pressed state.
|
||||||
|
- ``off``: Release a previously pressed button.
|
||||||
|
- ``press``: Press and release the button.
|
||||||
|
|
||||||
|
"""
|
||||||
|
if data == 'on':
|
||||||
|
return self.on(device)
|
||||||
|
if data == 'off':
|
||||||
|
return self.off(device)
|
||||||
|
if data == 'press':
|
||||||
|
return self.press(device)
|
||||||
|
|
||||||
|
self.logger.warning('Unknown command for SwitchBot "%s": "%s"', device, data)
|
||||||
|
return None
|
||||||
|
|
||||||
@action
|
@action
|
||||||
def scan(
|
def scan(
|
||||||
self, interface: Optional[str] = None, duration: int = 10
|
self, interface: Optional[str] = None, duration: int = 10
|
||||||
|
@ -138,14 +162,16 @@ class SwitchbotBluetoothPlugin( # lgtm [py/missing-call-to-init]
|
||||||
:param duration: Scan duration in seconds
|
:param duration: Scan duration in seconds
|
||||||
"""
|
"""
|
||||||
|
|
||||||
devices = super().scan(interface=interface, duration=duration).devices
|
compatible_devices: Dict[str, Any] = {}
|
||||||
compatible_devices = {}
|
devices = (
|
||||||
|
super().scan(interface=interface, duration=duration).devices # type: ignore
|
||||||
|
)
|
||||||
|
|
||||||
for dev in devices:
|
for dev in devices:
|
||||||
try:
|
try:
|
||||||
characteristics = [
|
characteristics = [
|
||||||
chrc
|
chrc
|
||||||
for chrc in self.discover_characteristics(
|
for chrc in self.discover_characteristics( # type: ignore
|
||||||
dev['addr'],
|
dev['addr'],
|
||||||
channel_type='random',
|
channel_type='random',
|
||||||
wait=False,
|
wait=False,
|
||||||
|
@ -157,14 +183,14 @@ class SwitchbotBluetoothPlugin( # lgtm [py/missing-call-to-init]
|
||||||
if characteristics:
|
if characteristics:
|
||||||
compatible_devices[dev['addr']] = None
|
compatible_devices[dev['addr']] = None
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.logger.warning('Device scan error', e)
|
self.logger.warning('Device scan error: %s', e)
|
||||||
|
|
||||||
self.publish_entities(compatible_devices) # type: ignore
|
self.publish_entities(compatible_devices)
|
||||||
return BluetoothScanResponse(devices=compatible_devices)
|
return BluetoothScanResponse(devices=compatible_devices)
|
||||||
|
|
||||||
@property
|
@action
|
||||||
def switches(self) -> List[dict]:
|
def status(self, *_, **__) -> List[dict]:
|
||||||
self.publish_entities(self.configured_devices) # type: ignore
|
self.publish_entities(self.configured_devices)
|
||||||
return [
|
return [
|
||||||
{
|
{
|
||||||
'address': addr,
|
'address': addr,
|
||||||
|
@ -175,20 +201,17 @@ class SwitchbotBluetoothPlugin( # lgtm [py/missing-call-to-init]
|
||||||
for addr, name in self.configured_devices.items()
|
for addr, name in self.configured_devices.items()
|
||||||
]
|
]
|
||||||
|
|
||||||
def transform_entities(self, devices: dict):
|
def transform_entities(self, entities: Collection[dict]) -> Collection[EnumSwitch]:
|
||||||
from platypush.entities.switches import Switch
|
return [
|
||||||
|
EnumSwitch(
|
||||||
return super().transform_entities( # type: ignore
|
id=addr,
|
||||||
[
|
name=name,
|
||||||
Switch(
|
value='on',
|
||||||
id=addr,
|
values=['on', 'off', 'press'],
|
||||||
name=name,
|
is_write_only=True,
|
||||||
state=False,
|
)
|
||||||
is_write_only=True,
|
for addr, name in entities
|
||||||
)
|
]
|
||||||
for addr, name in devices.items()
|
|
||||||
]
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
# vim:sw=4:ts=4:et:
|
# vim:sw=4:ts=4:et:
|
||||||
|
|
|
@ -1072,8 +1072,9 @@ class ZigbeeMqttPlugin(
|
||||||
return properties
|
return properties
|
||||||
|
|
||||||
@action
|
@action
|
||||||
def set_value( # pylint: disable=redefined-builtin,arguments-differ
|
# pylint: disable=redefined-builtin,arguments-differ
|
||||||
self, device: str, property: Optional[str] = None, data=None, **kwargs
|
def set_value(
|
||||||
|
self, device: str, *_, property: Optional[str] = None, data=None, **kwargs
|
||||||
):
|
):
|
||||||
"""
|
"""
|
||||||
Entity-compatible way of setting a value on a node.
|
Entity-compatible way of setting a value on a node.
|
||||||
|
|
Loading…
Reference in a new issue