craftbeerpi4-pione/cbpi/controller/actor_controller.py

151 lines
4.6 KiB
Python
Raw Normal View History

2018-12-29 00:27:19 +01:00
import logging
2018-12-13 21:45:33 +01:00
from asyncio import Future
2019-01-05 20:43:48 +01:00
from cbpi.api import *
2019-01-04 09:29:09 +01:00
from voluptuous import Schema
2019-01-05 20:43:48 +01:00
from cbpi.controller.crud_controller import CRUDController
from cbpi.database.model import ActorModel
2019-01-02 00:48:36 +01:00
2018-11-01 19:50:04 +01:00
2019-01-02 21:20:44 +01:00
class ActorController(CRUDController):
2018-11-16 20:35:59 +01:00
'''
The main actor controller
'''
2018-11-01 21:25:42 +01:00
model = ActorModel
2018-11-01 19:50:04 +01:00
def __init__(self, cbpi):
super(ActorController, self).__init__(cbpi)
self.cbpi = cbpi
self.state = False;
2018-12-29 00:27:19 +01:00
self.logger = logging.getLogger(__name__)
2019-01-02 21:20:44 +01:00
self.cbpi.register(self)
2018-11-01 19:50:04 +01:00
self.types = {}
self.actors = {}
2018-11-16 20:35:59 +01:00
async def init(self):
2019-01-02 21:20:44 +01:00
"""
2018-11-16 20:35:59 +01:00
This method initializes all actors during startup. It creates actor instances
:return:
2019-01-02 21:20:44 +01:00
"""
2018-11-01 19:50:04 +01:00
await super(ActorController, self).init()
for id, value in self.cache.items():
2018-12-13 21:45:33 +01:00
await self._init_actor(value)
2018-11-01 19:50:04 +01:00
2019-01-04 09:29:09 +01:00
def get_state(self):
return dict(items=self.cache,types=self.types)
2018-12-13 21:45:33 +01:00
async def _init_actor(self, actor):
2019-01-21 22:33:29 +01:00
2019-01-04 09:29:09 +01:00
try:
if actor.type in self.types:
cfg = actor.config.copy()
cfg.update(dict(cbpi=self.cbpi, id=id, name=actor.name))
clazz = self.types[actor.type]["class"];
self.cache[actor.id].instance = clazz(**cfg)
self.cache[actor.id].instance.init()
2019-01-17 22:11:55 +01:00
print(actor.id, self.cache[actor.id].instance)
2019-01-04 09:29:09 +01:00
await self.cbpi.bus.fire(topic="actor/%s/initialized" % actor.id, id=actor.id)
else:
self.logger.error("Actor type '%s' not found (Available Actor Types: %s)" % (actor.type, ', '.join(self.types.keys())))
except Exception as e:
self.logger.error("Failed to init actor %s - Reason %s" % (actor.id, str(e)))
2018-12-13 21:45:33 +01:00
async def _stop_actor(self, actor):
actor.instance.stop()
await self.cbpi.bus.fire(topic="actor/%s/stopped" % actor.id, id=actor.id)
2018-11-29 21:59:08 +01:00
2018-12-03 22:16:03 +01:00
@on_event(topic="actor/+/switch/on")
2019-01-14 07:33:59 +01:00
async def on(self, actor_id, power=100, **kwargs) -> None:
2019-01-02 21:20:44 +01:00
"""
2018-11-18 15:40:10 +01:00
Method to switch an actor on.
Supporting Event Topic "actor/+/on"
2018-11-29 21:59:08 +01:00
:param actor_id: the actor id
2019-01-02 21:20:44 +01:00
:param future
2018-11-18 15:40:10 +01:00
:param power: as integer value between 1 and 100
:param kwargs:
:return:
2019-01-02 21:20:44 +01:00
"""
2018-11-18 15:40:10 +01:00
2019-01-02 21:20:44 +01:00
actor_id = int(actor_id)
if actor_id in self.cache:
self.logger.debug("ON %s" % actor_id)
actor = self.cache[actor_id].instance
2018-11-16 20:35:59 +01:00
actor.on(power)
2019-01-14 07:33:59 +01:00
await self.cbpi.bus.fire("actor/%s/on/ok" % actor_id)
2018-11-16 20:35:59 +01:00
2018-12-13 21:45:33 +01:00
2018-11-16 20:35:59 +01:00
@on_event(topic="actor/+/toggle")
2019-01-02 21:20:44 +01:00
async def toggle(self, actor_id, power=100, **kwargs) -> None:
"""
2018-11-18 15:40:10 +01:00
Method to toggle an actor on or off
Supporting Event Topic "actor/+/toggle"
2018-11-16 20:35:59 +01:00
2019-01-02 21:20:44 +01:00
:param actor_id: the actor actor_id
:param power: the power as integer between 0 and 100
:return:
"""
self.logger.debug("TOGGLE %s" % actor_id)
actor_id = int(actor_id)
if actor_id in self.cache:
actor = self.cache[actor_id].instance
2018-11-16 20:35:59 +01:00
if actor.state is True:
2019-01-14 07:33:59 +01:00
await self.off(actor_id=actor_id)
2018-11-16 20:35:59 +01:00
else:
2019-01-14 07:33:59 +01:00
await self.on(actor_id=actor_id)
2018-11-01 19:50:04 +01:00
2019-01-04 09:29:09 +01:00
2018-11-01 21:25:42 +01:00
@on_event(topic="actor/+/off")
2019-01-02 21:20:44 +01:00
async def off(self, actor_id, **kwargs) -> None:
2018-11-01 21:25:42 +01:00
"""
2018-11-18 15:40:10 +01:00
Method to switch and actor off
Supporting Event Topic "actor/+/off"
2019-01-02 21:20:44 +01:00
:param actor_id: the actor actor_id
2018-11-01 21:25:42 +01:00
:param kwargs:
"""
2019-01-02 21:20:44 +01:00
self.logger.debug("OFF %s" % actor_id)
actor_id = int(actor_id)
2018-11-16 20:35:59 +01:00
2019-01-02 21:20:44 +01:00
if actor_id in self.cache:
actor = self.cache[actor_id].instance
2018-11-16 20:35:59 +01:00
actor.off()
2019-01-14 07:33:59 +01:00
await self.cbpi.bus.fire("actor/%s/off/ok" % actor_id)
2018-12-13 21:45:33 +01:00
2019-01-04 09:29:09 +01:00
@on_event(topic="actor/+/action")
async def call_action(self, actor_id, data, **kwargs) -> None:
schema = Schema({"name":str, "parameter":dict})
schema(data)
name = data.get("name")
parameter = data.get("parameter")
actor = self.cache[actor_id].instance.__getattribute__(name)(**parameter)
2018-12-13 21:45:33 +01:00
async def _post_add_callback(self, m):
'''
:param m:
:return:
'''
2019-01-05 22:44:54 +01:00
print("INIT ACTION")
2018-12-13 21:45:33 +01:00
await self._init_actor(m)
pass
async def _pre_delete_callback(self, actor_id):
2019-01-02 21:20:44 +01:00
#if int(actor_id) not in self.cache:
# return
2018-12-13 21:45:33 +01:00
if self.cache[int(actor_id)].instance is not None:
await self._stop_actor(self.cache[int(actor_id)])
async def _pre_update_callback(self, actor):
if actor.instance is not None:
await self._stop_actor(actor)
async def _post_update_callback(self, actor):
2019-01-17 22:11:55 +01:00
await self._init_actor(actor)