2018-11-01 19:50:04 +01:00
|
|
|
import logging
|
2019-01-01 15:35:35 +01:00
|
|
|
from os import urandom
|
2018-11-01 19:50:04 +01:00
|
|
|
|
|
|
|
from aiohttp import web
|
|
|
|
from aiohttp_auth import auth
|
|
|
|
from aiohttp_session import session_middleware
|
|
|
|
from aiohttp_session.cookie_storage import EncryptedCookieStorage
|
|
|
|
from aiohttp_swagger import setup_swagger
|
2019-01-05 20:43:48 +01:00
|
|
|
from cbpi.api.exceptions import CBPiException
|
2019-01-02 21:20:44 +01:00
|
|
|
from voluptuous import MultipleInvalid
|
2018-11-18 15:40:10 +01:00
|
|
|
|
2019-01-05 20:43:48 +01:00
|
|
|
from cbpi.controller.dashboard_controller import DashboardController
|
|
|
|
from cbpi.controller.job_controller import JobController
|
|
|
|
from cbpi.controller.actor_controller import ActorController
|
|
|
|
from cbpi.controller.config_controller import ConfigController
|
|
|
|
from cbpi.controller.kettle_controller import KettleController
|
|
|
|
from cbpi.controller.notification_controller import NotificationController
|
|
|
|
from cbpi.controller.plugin_controller import PluginController
|
|
|
|
from cbpi.controller.sensor_controller import SensorController
|
|
|
|
from cbpi.controller.step_controller import StepController
|
|
|
|
from cbpi.controller.system_controller import SystemController
|
|
|
|
from cbpi.database.model import DBModel
|
|
|
|
from cbpi.eventbus import CBPiEventBus
|
|
|
|
from cbpi.http_endpoints.http_login import Login
|
|
|
|
from cbpi.utils import *
|
|
|
|
from cbpi.websocket import CBPiWebSocket
|
|
|
|
from cbpi.http_endpoints.http_actor import ActorHttpEndpoints
|
|
|
|
from cbpi.http_endpoints.http_config import ConfigHttpEndpoints
|
|
|
|
from cbpi.http_endpoints.http_dashboard import DashBoardHttpEndpoints
|
|
|
|
from cbpi.http_endpoints.http_kettle import KettleHttpEndpoints
|
|
|
|
from cbpi.http_endpoints.http_sensor import SensorHttpEndpoints
|
|
|
|
from cbpi.http_endpoints.http_step import StepHttpEndpoints
|
2019-01-07 22:05:52 +01:00
|
|
|
from controller.translation_controller import TranslationController
|
|
|
|
from http_endpoints.http_translation import TranslationHttpEndpoint
|
2018-12-29 00:27:19 +01:00
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
2019-01-05 20:43:48 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
2018-11-01 19:50:04 +01:00
|
|
|
|
2019-01-02 00:48:36 +01:00
|
|
|
@web.middleware
|
|
|
|
async def error_middleware(request, handler):
|
|
|
|
try:
|
|
|
|
response = await handler(request)
|
|
|
|
if response.status != 404:
|
|
|
|
return response
|
|
|
|
message = response.message
|
|
|
|
except web.HTTPException as ex:
|
|
|
|
if ex.status != 404:
|
|
|
|
raise
|
|
|
|
message = ex.reason
|
|
|
|
except CBPiException as ex:
|
|
|
|
message = str(ex)
|
|
|
|
return web.json_response(status=500, data={'error': message})
|
2019-01-02 21:20:44 +01:00
|
|
|
except MultipleInvalid as ex:
|
|
|
|
return web.json_response(status=500, data={'error': str(ex)})
|
2019-01-02 00:48:36 +01:00
|
|
|
return web.json_response({'error': message})
|
|
|
|
|
2018-11-01 19:50:04 +01:00
|
|
|
class CraftBeerPi():
|
2018-11-04 00:47:26 +01:00
|
|
|
|
2019-01-01 15:35:35 +01:00
|
|
|
def __init__(self):
|
2019-01-05 20:43:48 +01:00
|
|
|
|
|
|
|
|
|
|
|
self.static_config = load_config("./config/config.yaml")
|
2019-01-02 00:48:36 +01:00
|
|
|
self.database_file = "./craftbeerpi.db"
|
2018-11-01 19:50:04 +01:00
|
|
|
logger.info("Init CraftBeerPI")
|
2019-01-01 15:35:35 +01:00
|
|
|
|
2018-11-01 19:50:04 +01:00
|
|
|
policy = auth.SessionTktAuthentication(urandom(32), 60, include_ip=True)
|
2019-01-02 00:48:36 +01:00
|
|
|
middlewares = [web.normalize_path_middleware(), session_middleware(EncryptedCookieStorage(urandom(32))), auth.auth_middleware(policy), error_middleware]
|
2018-11-01 19:50:04 +01:00
|
|
|
self.app = web.Application(middlewares=middlewares)
|
2018-12-16 21:42:47 +01:00
|
|
|
|
2019-01-01 15:35:35 +01:00
|
|
|
self._setup_shutdownhook()
|
|
|
|
self.initializer = []
|
2018-12-31 00:22:00 +01:00
|
|
|
self.bus = CBPiEventBus(self.app.loop, self)
|
2019-01-01 15:35:35 +01:00
|
|
|
self.ws = CBPiWebSocket(self)
|
|
|
|
self.job = JobController(self)
|
2019-01-07 22:05:52 +01:00
|
|
|
self.translation = TranslationController(self)
|
2018-11-01 19:50:04 +01:00
|
|
|
self.actor = ActorController(self)
|
|
|
|
self.sensor = SensorController(self)
|
2018-11-04 00:47:26 +01:00
|
|
|
self.plugin = PluginController(self)
|
2018-11-01 19:50:04 +01:00
|
|
|
self.system = SystemController(self)
|
2019-01-01 15:35:35 +01:00
|
|
|
self.config = ConfigController(self)
|
2018-11-18 15:40:10 +01:00
|
|
|
self.kettle = KettleController(self)
|
2018-12-03 22:16:03 +01:00
|
|
|
self.step = StepController(self)
|
2019-01-02 21:20:44 +01:00
|
|
|
self.dashboard = DashboardController(self)
|
|
|
|
|
|
|
|
self.http_step = StepHttpEndpoints(self)
|
|
|
|
self.http_sensor = SensorHttpEndpoints(self)
|
|
|
|
self.http_config = ConfigHttpEndpoints(self)
|
|
|
|
self.http_actor = ActorHttpEndpoints(self)
|
|
|
|
self.http_kettle = KettleHttpEndpoints(self)
|
|
|
|
self.http_dashboard = DashBoardHttpEndpoints(self)
|
2019-01-07 22:05:52 +01:00
|
|
|
self.http_translation = TranslationHttpEndpoint(self)
|
2019-01-02 21:20:44 +01:00
|
|
|
|
2018-11-16 20:35:59 +01:00
|
|
|
self.notification = NotificationController(self)
|
2018-11-01 19:50:04 +01:00
|
|
|
self.login = Login(self)
|
2018-11-04 01:55:54 +01:00
|
|
|
|
2019-01-01 15:35:35 +01:00
|
|
|
def _setup_shutdownhook(self):
|
|
|
|
self.shutdown = False
|
|
|
|
async def on_cleanup(app):
|
|
|
|
self.shutdown = True
|
2018-11-01 19:50:04 +01:00
|
|
|
|
2019-01-01 15:35:35 +01:00
|
|
|
self.app.on_cleanup.append(on_cleanup)
|
2018-11-01 19:50:04 +01:00
|
|
|
|
2018-11-04 00:47:26 +01:00
|
|
|
|
|
|
|
def register_on_startup(self, obj):
|
|
|
|
|
|
|
|
for method in [getattr(obj, f) for f in dir(obj) if callable(getattr(obj, f)) and hasattr(getattr(obj, f), "on_startup")]:
|
|
|
|
name = method.__getattribute__("name")
|
|
|
|
order = method.__getattribute__("order")
|
|
|
|
self.initializer.append(dict(name=name, method=method, order=order))
|
|
|
|
|
2018-12-07 00:18:35 +01:00
|
|
|
def register(self, obj, url_prefix=None, static=None):
|
2018-11-16 20:35:59 +01:00
|
|
|
|
|
|
|
'''
|
|
|
|
This method parses the provided object
|
|
|
|
|
|
|
|
:param obj: the object wich will be parsed for registration
|
|
|
|
:param url_prefix: that prefix for HTTP Endpoints
|
|
|
|
:return: None
|
|
|
|
'''
|
2018-12-07 00:18:35 +01:00
|
|
|
self.register_http_endpoints(obj, url_prefix, static)
|
2019-01-01 15:35:35 +01:00
|
|
|
self.bus.register_object(obj)
|
2019-01-04 09:29:09 +01:00
|
|
|
#self.ws.register_object(obj)
|
2019-01-01 15:35:35 +01:00
|
|
|
self.job.register_background_task(obj)
|
2018-11-04 00:47:26 +01:00
|
|
|
self.register_on_startup(obj)
|
2018-11-01 19:50:04 +01:00
|
|
|
|
2018-12-07 00:18:35 +01:00
|
|
|
def register_http_endpoints(self, obj, url_prefix=None, static=None):
|
2019-01-02 21:20:44 +01:00
|
|
|
|
|
|
|
if url_prefix is None:
|
|
|
|
logger.debug("URL Prefix is None for %s. No endpoints will be registered. Please set / explicit if you want to add it to the root path" % obj)
|
|
|
|
return
|
|
|
|
|
2018-11-01 19:50:04 +01:00
|
|
|
routes = []
|
|
|
|
for method in [getattr(obj, f) for f in dir(obj) if callable(getattr(obj, f)) and hasattr(getattr(obj, f), "route")]:
|
|
|
|
http_method = method.__getattribute__("method")
|
|
|
|
path = method.__getattribute__("path")
|
2018-11-01 21:25:42 +01:00
|
|
|
class_name = method.__self__.__class__.__name__
|
2018-11-16 20:35:59 +01:00
|
|
|
logger.info("Register Endpoint : %s.%s %s %s%s " % (class_name, method.__name__, http_method, url_prefix, path))
|
2018-11-01 19:50:04 +01:00
|
|
|
|
|
|
|
def add_post():
|
|
|
|
routes.append(web.post(method.__getattribute__("path"), method))
|
|
|
|
|
|
|
|
def add_get():
|
2019-01-02 00:48:36 +01:00
|
|
|
routes.append(web.get(method.__getattribute__("path"), method, allow_head=False))
|
2018-11-01 19:50:04 +01:00
|
|
|
|
|
|
|
def add_delete():
|
|
|
|
routes.append(web.delete(path, method))
|
|
|
|
|
|
|
|
def add_put():
|
|
|
|
routes.append(web.put(path, method))
|
2018-11-01 21:25:42 +01:00
|
|
|
|
2018-11-01 19:50:04 +01:00
|
|
|
switcher = {
|
|
|
|
"POST": add_post,
|
|
|
|
"GET": add_get,
|
|
|
|
"DELETE": add_delete,
|
|
|
|
"PUT": add_put
|
|
|
|
}
|
|
|
|
switcher[http_method]()
|
|
|
|
|
2019-01-02 21:20:44 +01:00
|
|
|
if url_prefix != "/":
|
2018-12-29 00:27:19 +01:00
|
|
|
logger.debug("URL Prefix: %s " % (url_prefix,))
|
2018-11-01 19:50:04 +01:00
|
|
|
sub = web.Application()
|
|
|
|
sub.add_routes(routes)
|
2018-12-07 00:18:35 +01:00
|
|
|
if static is not None:
|
2018-12-31 00:22:00 +01:00
|
|
|
sub.add_routes([web.static('/static', static, show_index=False)])
|
2018-11-16 20:35:59 +01:00
|
|
|
self.app.add_subapp(url_prefix, sub)
|
2018-11-01 19:50:04 +01:00
|
|
|
else:
|
|
|
|
self.app.add_routes(routes)
|
|
|
|
|
2018-11-04 00:47:26 +01:00
|
|
|
def _swagger_setup(self):
|
2018-11-16 20:35:59 +01:00
|
|
|
'''
|
|
|
|
Internatl method to expose REST API documentation by swagger
|
|
|
|
|
|
|
|
:return:
|
|
|
|
'''
|
2018-11-04 00:47:26 +01:00
|
|
|
long_description = """
|
2018-11-18 15:40:10 +01:00
|
|
|
This is the api for CraftBeerPi
|
2018-11-04 00:47:26 +01:00
|
|
|
"""
|
|
|
|
|
|
|
|
setup_swagger(self.app,
|
|
|
|
description=long_description,
|
2019-01-01 15:35:35 +01:00
|
|
|
title=self.static_config.get("name", "CraftBeerPi"),
|
|
|
|
api_version=self.static_config.get("version", ""),
|
2018-11-04 00:47:26 +01:00
|
|
|
contact="info@craftbeerpi.com")
|
|
|
|
|
2018-11-16 20:35:59 +01:00
|
|
|
def notify(self, key, message, type="info"):
|
|
|
|
'''
|
|
|
|
This is a convinience method to send notification to the client
|
|
|
|
|
|
|
|
:param key: notification key
|
|
|
|
:param message: notification message
|
|
|
|
:param type: notification type (info,warning,danger,successs)
|
|
|
|
:return:
|
|
|
|
'''
|
2018-12-13 21:45:33 +01:00
|
|
|
self.bus.sync_fire(topic="notification/%s" % key, key=key, message=message, type=type)
|
2018-11-01 19:50:04 +01:00
|
|
|
|
2018-12-31 00:22:00 +01:00
|
|
|
async def call_initializer(self, app):
|
|
|
|
self.initializer = sorted(self.initializer, key=lambda k: k['order'])
|
|
|
|
for i in self.initializer:
|
|
|
|
logger.info("CALL INITIALIZER %s - %s " % (i["name"], i["method"].__name__))
|
|
|
|
await i["method"]()
|
2018-11-04 00:47:26 +01:00
|
|
|
|
2018-12-31 00:22:00 +01:00
|
|
|
def _print_logo(self):
|
|
|
|
from pyfiglet import Figlet
|
|
|
|
f = Figlet(font='big')
|
2019-01-01 15:35:35 +01:00
|
|
|
logger.info("\n%s" % f.renderText("%s %s" % (self.static_config.get("name"), self.static_config.get("version"))))
|
2018-11-04 00:47:26 +01:00
|
|
|
|
2019-01-02 00:48:36 +01:00
|
|
|
|
|
|
|
def _setup_http_index(self):
|
|
|
|
async def http_index(request):
|
|
|
|
url = self.config.static.get("index_url")
|
|
|
|
if url is not None:
|
|
|
|
raise web.HTTPFound(url)
|
|
|
|
else:
|
|
|
|
return web.Response(text="Hello, world")
|
|
|
|
|
|
|
|
self.app.add_routes([web.get('/', http_index)])
|
|
|
|
|
2018-12-31 00:22:00 +01:00
|
|
|
async def init_serivces(self):
|
2018-11-04 00:47:26 +01:00
|
|
|
|
2018-12-31 00:22:00 +01:00
|
|
|
self._print_logo()
|
2019-01-01 15:35:35 +01:00
|
|
|
|
|
|
|
await self.job.init()
|
|
|
|
await DBModel.setup()
|
|
|
|
await self.config.init()
|
2019-01-07 22:05:52 +01:00
|
|
|
await self.translation.init()
|
2019-01-02 00:48:36 +01:00
|
|
|
self._setup_http_index()
|
2018-12-31 00:22:00 +01:00
|
|
|
self.plugin.load_plugins()
|
|
|
|
self.plugin.load_plugins_from_evn()
|
|
|
|
await self.sensor.init()
|
|
|
|
await self.step.init()
|
|
|
|
await self.actor.init()
|
|
|
|
await self.kettle.init()
|
|
|
|
await self.call_initializer(self.app)
|
2019-01-02 21:20:44 +01:00
|
|
|
await self.dashboard.init()
|
2018-11-04 00:47:26 +01:00
|
|
|
self._swagger_setup()
|
2018-11-16 20:35:59 +01:00
|
|
|
|
2018-12-31 00:22:00 +01:00
|
|
|
return self.app
|
|
|
|
|
2019-01-01 15:35:35 +01:00
|
|
|
|
2018-11-16 20:35:59 +01:00
|
|
|
def start(self):
|
2019-01-01 15:35:35 +01:00
|
|
|
web.run_app(self.init_serivces(), port=self.static_config.get("port", 8080))
|