From 4f2f3badea5342f38c2e8a9bc7f6ed7a3654fcf8 Mon Sep 17 00:00:00 2001 From: avollkopf Date: Sun, 22 Dec 2024 19:53:49 +0100 Subject: [PATCH 1/3] fix for last change -> GUI Version --- cbpi/__init__.py | 2 +- cbpi/http_endpoints/http_system.py | 7 ------- 2 files changed, 1 insertion(+), 8 deletions(-) diff --git a/cbpi/__init__.py b/cbpi/__init__.py index 8a9c69a..fc28431 100644 --- a/cbpi/__init__.py +++ b/cbpi/__init__.py @@ -1,3 +1,3 @@ -__version__ = "4.4.8" +__version__ = "4.4.9" __codename__ = "Yeast Starter" diff --git a/cbpi/http_endpoints/http_system.py b/cbpi/http_endpoints/http_system.py index 62ccc52..a17eef4 100644 --- a/cbpi/http_endpoints/http_system.py +++ b/cbpi/http_endpoints/http_system.py @@ -40,13 +40,6 @@ class SystemHttpEndpoints: else: spindledata = False - try: - plugin_list = await self.cbpi.plugin.load_plugin_list("cbpi4spindle") - version= plugin_list[0].get("Version", "not detected") - except: - version="not detected" - - return web.json_response(data=dict( actor=self.cbpi.actor.get_state(), fermenter=self.cbpi.fermenter.get_state(), From 839e6caf88704f32d217b1db05d3e932b6000abd Mon Sep 17 00:00:00 2001 From: avollkopf Date: Sat, 28 Dec 2024 15:20:06 +0100 Subject: [PATCH 2/3] add 2nd status text option for steps --- cbpi/__init__.py | 4 ++-- cbpi/api/dataclasses.py | 3 ++- cbpi/api/step.py | 1 + cbpi/extension/mashstep/__init__.py | 26 ++++++++++++++++++++++++-- cbpi/http_endpoints/http_step.py | 2 ++ 5 files changed, 31 insertions(+), 5 deletions(-) diff --git a/cbpi/__init__.py b/cbpi/__init__.py index fc28431..cb221bd 100644 --- a/cbpi/__init__.py +++ b/cbpi/__init__.py @@ -1,3 +1,3 @@ -__version__ = "4.4.9" -__codename__ = "Yeast Starter" +__version__ = "4.5.0.a1" +__codename__ = "Cross Country" diff --git a/cbpi/api/dataclasses.py b/cbpi/api/dataclasses.py index 4515012..0966cf0 100644 --- a/cbpi/api/dataclasses.py +++ b/cbpi/api/dataclasses.py @@ -133,7 +133,8 @@ class Step: def to_dict(self): msg = self.instance.summary if self.instance is not None else "" - return dict(id=self.id, name=self.name, state_text=msg, type=self.type, status=self.status.value, props=self.props.to_dict()) + msg2 = self.instance.summary2 if ((self.instance is not None) and (self.instance.summary2 is not None)) else None + return dict(id=self.id, name=self.name, state_text=msg, state_text2=msg2, type=self.type, status=self.status.value, props=self.props.to_dict()) @dataclass class Fermenter: diff --git a/cbpi/api/step.py b/cbpi/api/step.py index bf30778..3b81017 100644 --- a/cbpi/api/step.py +++ b/cbpi/api/step.py @@ -46,6 +46,7 @@ class CBPiStep(CBPiBase): self.props = props self.cancel_reason: StepResult = None self.summary = "" + self.summary2 = None self.task = None self.running: bool = False self.logger = logging.getLogger(__name__) diff --git a/cbpi/extension/mashstep/__init__.py b/cbpi/extension/mashstep/__init__.py index 1047c80..75e314f 100644 --- a/cbpi/extension/mashstep/__init__.py +++ b/cbpi/extension/mashstep/__init__.py @@ -384,7 +384,8 @@ class BoilStep(CBPiStep): #self.dwelltime=int(self.props.get("DwellTime", 0))*60 self.dwelltime=5*60 #tested with 5 minutes -> not exactly 5 min due to accuracy of asyncio.sleep self.deviationlimit=0.3 # derived from a test - logging.warning(self.AutoTimer) + #logging.warning(self.AutoTimer) + self.summary2=None self.kettle=self.get_kettle(self.props.get("Kettle", None)) if self.kettle is not None: @@ -405,6 +406,24 @@ class BoilStep(CBPiStep): await self.setAutoMode(True) await self.push_update() + async def next_hop_timer(self): + hop_timers = [] + for x in range(1, 6): + try: + hop = int(self.props.get("Hop_%s" % x, None)) * 60 + except: + hop = None + if hop is not None: + hop_left = self.remaining_seconds - hop + if hop_left > 0: + hop_timers.append(hop_left) + + if len(hop_timers) != 0: + next_hop_timer = time.strftime("%H:%M:%S", time.gmtime(min(hop_timers))) + else: + next_hop_timer = None + return next_hop_timer + async def check_hop_timer(self, number, value, text): if value is not None and self.hops_added[number-1] is not True: if self.remaining_seconds != None and self.remaining_seconds <= (int(value) * 60 + 1): @@ -417,6 +436,7 @@ class BoilStep(CBPiStep): async def on_stop(self): await self.timer.stop() self.summary = "" + self.summary2 = None self.kettle.target_temp = 0 if self.AutoMode == True: await self.setAutoMode(False) @@ -456,7 +476,9 @@ class BoilStep(CBPiStep): estimated_completion_time = datetime.fromtimestamp(time.time()+ (int(self.props.get("Timer", 0)))*60) self.cbpi.notify(self.name, 'Timer started. Estimated completion: {}'.format(estimated_completion_time.strftime("%H:%M")), NotificationType.INFO) else: - for x in range(1, 6): + nexthoptimer=await self.next_hop_timer() + self.summary2="Add Hop in: %s" % nexthoptimer if nexthoptimer is not None else None + for x in range(1, 6): await self.check_hop_timer(x, self.props.get("Hop_%s" % x, None), self.props.get("Hop_%s_text" % x, None)) return StepResult.DONE diff --git a/cbpi/http_endpoints/http_step.py b/cbpi/http_endpoints/http_step.py index d7b2242..56a906e 100644 --- a/cbpi/http_endpoints/http_step.py +++ b/cbpi/http_endpoints/http_step.py @@ -2,6 +2,7 @@ from cbpi.controller.step_controller import StepController from cbpi.api.dataclasses import Props, Step from aiohttp import web from cbpi.api import * +import logging class StepHttpEndpoints(): @@ -10,6 +11,7 @@ class StepHttpEndpoints(): self.controller : StepController = cbpi.step self.cbpi.register(self, "/step2") + # Check if this is still needed def create_dict(self, data): return dict(name=data["name"], id=data["id"], type=data.get("type"), status=data["status"],props=data["props"], state_text=data["instance"].get_state()) From a706c6d544658df502791c9cb78c06ef3c3fde11 Mon Sep 17 00:00:00 2001 From: avollkopf Date: Fri, 3 Jan 2025 10:15:14 +0100 Subject: [PATCH 3/3] minor adaptions -> bf upload swagger description --- cbpi/__init__.py | 2 +- cbpi/controller/log_file_controller.py | 3 +- cbpi/extension/iSpindle/__init__.py | 754 +++++++++++++++++++++++++ cbpi/extension/iSpindle/config.yaml | 3 + cbpi/http_endpoints/http_upload.py | 7 + 5 files changed, 766 insertions(+), 3 deletions(-) create mode 100644 cbpi/extension/iSpindle/__init__.py create mode 100644 cbpi/extension/iSpindle/config.yaml diff --git a/cbpi/__init__.py b/cbpi/__init__.py index cb221bd..9018f45 100644 --- a/cbpi/__init__.py +++ b/cbpi/__init__.py @@ -1,3 +1,3 @@ -__version__ = "4.5.0.a1" +__version__ = "4.5.0.a2" __codename__ = "Cross Country" diff --git a/cbpi/controller/log_file_controller.py b/cbpi/controller/log_file_controller.py index fecae47..8e50c0f 100644 --- a/cbpi/controller/log_file_controller.py +++ b/cbpi/controller/log_file_controller.py @@ -72,7 +72,6 @@ class LogController: # remove duplicates names = set(names) - result = None def dateparse(time_in_secs): @@ -120,7 +119,7 @@ class LogController: data[name] = result.interpolate().tolist() logging.info("Send Log for {}".format(names)) - + return data async def get_data2(self, ids) -> dict: diff --git a/cbpi/extension/iSpindle/__init__.py b/cbpi/extension/iSpindle/__init__.py new file mode 100644 index 0000000..10cb75b --- /dev/null +++ b/cbpi/extension/iSpindle/__init__.py @@ -0,0 +1,754 @@ + +# -*- coding: utf-8 -*- +import os +from aiohttp import web +import logging +from unittest.mock import MagicMock, patch +import asyncio +import random +from cbpi.api import * +from aiohttp import web +from cbpi.api import * +import re +import time +import json +from cbpi.api.config import ConfigType +from cbpi.api.dataclasses import DataType +import mysql.connector +import datetime +from json import JSONEncoder +from pandas import DataFrame + +logger = logging.getLogger(__name__) + +cache = {} + +# subclass JSONEncoder +class DateTimeEncoder(JSONEncoder): + #Override the default method + def default(self, obj): + if isinstance(obj, (datetime.date, datetime.datetime)): + return obj.isoformat() + +class iSpindleConfig(CBPiExtension): + + def __init__(self,cbpi): + self.cbpi = cbpi + self._task = asyncio.create_task(self.init_config()) + + async def init_config(self): + global parametercheck, sql_connection + + parametercheck=False + sql_connection=False + + plugin = await self.cbpi.plugin.load_plugin_list("cbpi4-iSpindle") + try: + self.version=plugin[0].get("Version","0.0.0") + self.name=plugin[0].get("Name","cbpi4-iSPindle") + except: + self.version="0.0.0" + self.name="craftbeerpi" + + self.iSpindle_update = self.cbpi.config.get(self.name+"_update", None) + await self.iSpindle_config() + + while parametercheck == False: + await asyncio.sleep(1) + + if spindle_SQL == "Yes": + cnx = mysql.connector.connect( + user=spindle_SQL_USER, port=spindle_SQL_PORT, password=spindle_SQL_PASSWORD, host=spindle_SQL_HOST, database=spindle_SQL_DB) + cur = cnx.cursor() + sqlselect = "SELECT VERSION()" + cur.execute(sqlselect) + results = cur.fetchone() + ver = results[0] + logger.warning("MySQL connection available. MySQL version: %s" % ver) + sql_connection=True + if (ver is None): + logger.error("MySQL connection failed") + + + async def iSpindle_config(self): + global spindledata, spindle_SQL, spindle_SQL_HOST, spindle_SQL_DB, spindle_SQL_TABLE, spindle_SQL_USER, spindle_SQL_PASSWORD, spindle_SQL_PORT, parametercheck + parametercheck=False + spindledata = self.cbpi.config.get("spindledata", None) + if spindledata is None: + logger.warning("INIT Spindledata extra page in UI") + try: + await self.cbpi.config.add("spindledata", "No", type=ConfigType.SELECT, description="Enable extra page for spindledata in ui", + source=self.name, + options= [{"label": "No","value": "No"}, + {"label": "Yes", "value": "Yes"}]) + + spindledata = self.cbpi.config.get("spindledata", "No") + except: + logger.warning('Unable to update database') + else: + if self.iSpindle_update == None or self.iSpindle_update != self.version: + try: + await self.cbpi.config.add("spindledata", spindledata, type=ConfigType.SELECT, description="Enable extra page for spindledata in ui", + source=self.name, + options= [{"label": "No","value": "No"}, + {"label": "Yes", "value": "Yes"}]) + + except: + logger.warning('Unable to update database') + + spindle_SQL = self.cbpi.config.get("spindle_SQL", None) # 1 to enable output to MySQL database + if spindle_SQL is None: + logger.warning("INIT Spindle database select for transfer") + try: + await self.cbpi.config.add("spindle_SQL", "No", type=ConfigType.SELECT, description="Enable transfer of Spindle data to SQL database", + source=self.name, + options= [{"label": "No","value": "No"}, + {"label": "Yes", "value": "Yes"}]) + + spindle_SQL = self.cbpi.config.get("spindle_SQL", "No") + except: + logger.warning('Unable to update database') + else: + if self.iSpindle_update == None or self.iSpindle_update != self.version: + try: + await self.cbpi.config.add("spindle_SQL", spindle_SQL, type=ConfigType.SELECT, description="Enable transfer of Spindle data to SQL database", + source=self.name, + options= [{"label": "No","value": "No"}, + {"label": "Yes", "value": "Yes"}]) + + except: + logger.warning('Unable to update database') + + spindle_SQL_HOST = self.cbpi.config.get("spindle_SQL_HOST", None) # Database host name (default: localhost - 127.0.0.1 loopback interface) + if spindle_SQL_HOST is None: + logger.warning("INIT Spindle database host name") + try: + await self.cbpi.config.add("spindle_SQL_HOST", "", type=ConfigType.STRING, description="SQL database host. e.g: localhost or IP address", + source=self.name) + + spindle_SQL = self.cbpi.config.get("spindle_SQL_HOST", "") + except: + logger.warning('Unable to update database') + else: + if self.iSpindle_update == None or self.iSpindle_update != self.version: + try: + await self.cbpi.config.add("spindle_SQL_HOST", spindle_SQL_HOST, type=ConfigType.STRING, description="SQL database host. e.g: localhost or IP address", + source=self.name) + + except: + logger.warning('Unable to update database') + + spindle_SQL_DB = self.cbpi.config.get("spindle_SQL_DB", None) # Database name + if spindle_SQL_DB is None: + logger.warning("INIT Spindle Database Name") + try: + await self.cbpi.config.add("spindle_SQL_DB", "", type=ConfigType.STRING, description="SQL database name", + source=self.name) + + spindle_SQL = self.cbpi.config.get("spindle_SQL_DB", "") + except: + logger.warning('Unable to update database') + else: + if self.iSpindle_update == None or self.iSpindle_update != self.version: + try: + await self.cbpi.config.add("spindle_SQL_DB", spindle_SQL_DB, type=ConfigType.STRING, description="SQL database name", + source=self.name) + + except: + logger.warning('Unable to update database') + + spindle_SQL_TABLE = self.cbpi.config.get("spindle_SQL_TABLE", None) # Table name + if spindle_SQL_TABLE is None: + logger.warning("INIT Spindle Database table Name") + try: + await self.cbpi.config.add("spindle_SQL_TABLE", "", type=ConfigType.STRING, description="SQL database table name", + source=self.name) + + spindle_SQL = self.cbpi.config.get("spindle_SQL_TABLE", "") + except: + logger.warning('Unable to update database') + else: + if self.iSpindle_update == None or self.iSpindle_update != self.version: + try: + await self.cbpi.config.add("spindle_SQL_TABLE", spindle_SQL_TABLE, type=ConfigType.STRING, description="SQL database table name", + source=self.name) + + except: + logger.warning('Unable to update database') + + spindle_SQL_USER = self.cbpi.config.get("spindle_SQL_USER", None) # DB user + if spindle_SQL_USER is None: + logger.warning("INIT Spindle Database user name") + try: + await self.cbpi.config.add("spindle_SQL_USER", "", type=ConfigType.STRING, description="SQL database user name", + source=self.name) + + spindle_SQL = self.cbpi.config.get("spindle_SQL_USER", "") + except: + logger.warning('Unable to update database') + else: + if self.iSpindle_update == None or self.iSpindle_update != self.version: + try: + await self.cbpi.config.add("spindle_SQL_USER", spindle_SQL_USER, type=ConfigType.STRING, description="SQL database user name", + source=self.name) + + except: + logger.warning('Unable to update database') + + spindle_SQL_PASSWORD = self.cbpi.config.get("spindle_SQL_PASSWORD", None) # DB user's password (change this) + if spindle_SQL_PASSWORD is None: + logger.warning("INIT Spindle Database password") + try: + await self.cbpi.config.add("spindle_SQL_PASSWORD", "", type=ConfigType.STRING, description="SQL database password", + source=self.name) + + spindle_SQL = self.cbpi.config.get("spindle_SQL_PASSWORD", "") + except: + logger.warning('Unable to update database') + else: + if self.iSpindle_update == None or self.iSpindle_update != self.version: + try: + await self.cbpi.config.add("spindle_SQL_PASSWORD", spindle_SQL_PASSWORD, type=ConfigType.STRING, description="SQL database password", + source=self.name) + + except: + logger.warning('Unable to update database') + + spindle_SQL_PORT = self.cbpi.config.get("spindle_SQL_PORT", None) + if spindle_SQL_PORT is None: + logger.warning("INIT Spindle Database port number") + try: + await self.cbpi.config.add("spindle_SQL_PORT", "", type=ConfigType.NUMBER, description="SQL database port number", + source=self.name) + + spindle_SQL = self.cbpi.config.get("spindle_SQL_PORT", "") + except: + logger.warning('Unable to update database') + else: + if self.iSpindle_update == None or self.iSpindle_update != self.version: + try: + await self.cbpi.config.add("spindle_SQL_PORT", spindle_SQL_PORT, type=ConfigType.NUMBER, description="SQL database port number", + source=self.name) + + except: + logger.warning('Unable to update database') + + + if self.iSpindle_update == None or self.iSpindle_update != self.version: + try: + await self.cbpi.config.add(self.name+"_update", self.version,type=ConfigType.STRING, description='iSpindle Version Update', source='hidden') + except Exception as e: + logger.warning('Unable to update database') + logger.warning(e) + + parametercheck=True + + +async def calcGravity(polynom, tilt, unitsGravity): + if unitsGravity == "SG": + rounddec = 3 + else: + rounddec = 2 + + # Calculate gravity from polynomial + tilt = float(tilt) + result = eval(polynom) + result = round(float(result),rounddec) + return result + +@parameters([Property.Text(label="iSpindle", configurable=True, description="Enter the name of your iSpindel"), + Property.Select("Type", options=["Temperature", "Gravity/Angle", "Battery", "RSSI", "DateTime"], description="Select which type of data to register for this sensor. For Angle, Polynomial has to be left empty"), + Property.Text(label="Polynomial", configurable=True, description="Enter your iSpindel polynomial. Use the variable tilt for the angle reading from iSpindel. Does not support ^ character."), + Property.Select("Units", options=["SG", "Brix", "°P"], description="Displays gravity reading with this unit if the Data Type is set to Gravity. Does not convert between units, to do that modify your polynomial."), + Property.Sensor("FermenterTemp",description="Select Fermenter Temp Sensor that you want to provide to TCP Server")]) + +class iSpindle(CBPiSensor): + + def __init__(self, cbpi, id, props): + super(iSpindle, self).__init__(cbpi, id, props) + self.value = 0 + self.key = self.props.get("iSpindle", None) + self.Polynomial = self.props.get("Polynomial", "tilt") + self.temp_sensor_id = self.props.get("FermenterTemp", None) + self.datatype = DataType.DATETIME if self.props.get("Type", None) == "DateTime" else DataType.VALUE + self.time_old = 0 + + def get_unit(self): + if self.props.get("Type") == "Temperature": + return "°C" if self.get_config_value("TEMP_UNIT", "C") == "C" else "°F" + elif self.props.get("Type") == "Gravity/Angle": + return self.props.Units + elif self.props.get("Type") == "Battery": + return "V" + elif self.props.get("Type") == "RSSI": + return "dB" + else: + return " " + + async def run(self): + global cache + global fermenter_temp + Spindle_name = self.props.get("iSpindle") + while self.running == True: + try: + if (float(cache[self.key]['Time']) > float(self.time_old)): + self.time_old = float(cache[self.key]['Time']) + if self.props.get("Type") == "Gravity/Angle": + self.value = await calcGravity(self.Polynomial, cache[self.key]['Angle'], self.props.get("Units")) + elif self.props.get("Type") == "DateTime": + self.value=float(cache[self.key]['Time']) + else: + self.value = float(cache[self.key][self.props.Type]) + self.log_data(self.value) + self.push_update(self.value) + self.push_update(self.value,False) + #self.cbpi.ws.send(dict(topic="sensorstate", id=self.id, value=self.value)) + + except Exception as e: + pass + await asyncio.sleep(2) + + def get_state(self): + return dict(value=self.value) + +class iSpindleEndpoint(CBPiExtension): + + def __init__(self, cbpi): + ''' + Initializer + :param cbpi: + ''' + self.pattern_check = re.compile("^[a-zA-Z0-9,.]{0,10}$") + self.cbpi = cbpi + self.sensor_controller : SensorController = cbpi.sensor + # register component for http, events + # In addtion the sub folder static is exposed to access static content via http + self.cbpi.register(self, "/api/hydrometer/v1/data") + + async def run(self): + await self.get_spindle_sensor() + + @request_mapping(path='', method="POST", auth_required=False) + async def http_new_value3(self, request): + import time + """ + --- + description: Get iSpindle Value + tags: + - iSpindle + parameters: + - name: "data" + in: "body" + description: "Data" + required: "name" + type: "object" + type: string + responses: + "204": + description: successful operation + """ + + global cache + try: + data = await request.json() + except Exception as e: + print(e) + logging.info(data) + time = time.time() + key = data['name'] + temp = round(float(data['temperature']), 2) + angle = data['angle'] + battery = data['battery'] + try: + rssi = data['RSSI'] + except: + rssi = 0 + cache[key] = {'Time': time,'Temperature': temp, 'Angle': angle, 'Battery': battery, 'RSSI': rssi} + + + @request_mapping(path='/gettemp/{SpindleID}', method="POST", auth_required=False) + async def get_fermenter_temp(self, request): + SpindleID = request.match_info['SpindleID'] + sensor_value = await self.get_spindle_sensor(SpindleID) + data = {'Temp': sensor_value} + return web.json_response(data=data) + + async def get_spindle_sensor(self, iSpindleID = None): + self.sensor = self.sensor_controller.get_state() + for id in self.sensor['data']: + if id['type'] == 'iSpindle': + name= id['props']['iSpindle'] + if name == iSpindleID: + try: + sensor= id['props']['FermenterTemp'] + except: + sensor = None + if (sensor is not None) and sensor != "": + sensor_value = self.cbpi.sensor.get_sensor_value(sensor).get('value') + else: + sensor_value = None + return sensor_value + + @request_mapping(path='/getarchive', method="GET", auth_required=False) + async def get_archive_headers(self, request): + """ + --- + description: Get all stored fermentations from database archive + tags: + - iSpindle + responses: + "200": + description: successful operation + """ + + data= await self.get_archive_list() + return web.json_response(data=data) + + + async def get_archive_list(self): + ORDER="DESC" + archive_sql = "SELECT Recipe_ID as value, CONCAT(Batch, ' | ', Name, ' | ',DATE_FORMAT(Start_date, '%Y-%m-%d'),' | ', Recipe, ' (', Recipe_ID,')' ) as 'label' FROM Archive ORDER BY Recipe_ID {}".format(ORDER) + cnx = mysql.connector.connect( + user=spindle_SQL_USER, port=spindle_SQL_PORT, password=spindle_SQL_PASSWORD, host=spindle_SQL_HOST, database=spindle_SQL_DB) + cur = cnx.cursor() + cur.execute(archive_sql) + columns = [column[0] for column in cur.description] + results = [dict(zip(columns, row)) for row in cur.fetchall()] + return results + + @request_mapping(path='/getdiagrams', method="GET", auth_required=False) + async def get_diagrams(self, request): + """ + --- + description: Get available diagrams + tags: + - iSpindle + responses: + "200": + description: successful operation + """ + + data= await self.get_diagram_list() + return web.json_response(data=data) + + async def get_diagram_list(self): + results = [{'value': '0', 'label': 'Gravity and Temperature (RasPySpindle)'}, + {'value': '1', 'label': 'Gravity and Temperature (iSpindle Polynom)'}, + {'value': '2', 'label': 'Tilt and Temperature'}, + {'value': '3', 'label': 'Attenuation'}, + {'value': '4', 'label': 'Battery and Wifi Signal'}] + + return results + + @request_mapping(path='/getarchiveheader/{ArchiveID}/', method="POST", auth_required=False) + async def get_archive_header(self, request): + """ + --- + description: Get Archive header data for specified archive id + tags: + - iSpindle + parameters: + - name: "ArchiveID" + in: "path" + description: "ArchiveID" + required: true + type: "integer" + format: "int64" + responses: + "200": + description: successful operation + """ + ArchiveID = request.match_info['ArchiveID'] + header_data = await self.get_archive_header_data(ArchiveID) + #logger.error(header_data) + return web.json_response(data=header_data) + + async def get_archive_header_data(self, ArchiveID): + result_angle=[] + + cnx = mysql.connector.connect( + user=spindle_SQL_USER, port=spindle_SQL_PORT, password=spindle_SQL_PASSWORD, host=spindle_SQL_HOST, database=spindle_SQL_DB) + cur = cnx.cursor() + + #get other archive data + archive_sql = "Select * FROM Archive WHERE Recipe_ID = {}".format(ArchiveID) + cur.execute(archive_sql) + columns = [column[0] for column in cur.description] + result_archive = [dict(zip(columns, row)) for row in cur.fetchall()] + + Spindle_Name=result_archive[0]['Name'] + Batch=result_archive[0]['Batch'] + Recipe=result_archive[0]['Recipe'] + Start_date=result_archive[0]['Start_date'] + End_date=result_archive[0]['End_date'] + + const0=result_archive[0]['const0'] + const1=result_archive[0]['const1'] + const2=result_archive[0]['const2'] + const3=result_archive[0]['const3'] + + if const0 == 0: + forumla= "{:+.5f} * tilt^2 {:+.5f} * tilt {:+.5f}".format(const1, const2, const3) + else: + forumla= "{:+.5f} * tilt^3 {:+.5f} * tilt^2 {:+.5f} * tilt {:+.5f}".format(const0, const1, const2, const3) + + #if no entry for end date in archive table, get last timestamp of last dataset for selected recipe from data table + if End_date is None: + get_end_date_sql = "SELECT max(Timestamp) FROM Data WHERE Recipe_ID = {}".format(ArchiveID) + cur.execute(get_end_date_sql) + columns = [column[0] for column in cur.description] + result_end_date = [dict(zip(columns, row)) for row in cur.fetchall()] + End_date=result_end_date[0]['max(Timestamp)'] + + # check if end flag is set for the archive + RID_END=False + check_RID_END = "SELECT * FROM Data WHERE Recipe_ID = {} AND Internal = 'RID_END'".format(ArchiveID) + cur.execute(check_RID_END) + columns = [column[0] for column in cur.description] + result_RID_END = [dict(zip(columns, row)) for row in cur.fetchall()] + if len(result_RID_END) > 0: + End_date=result_RID_END[0]['Timestamp'] + RID_END=True + + # Get Angle data for the first hour after reset + where_sql = "WHERE Recipe_ID = {} AND Timestamp > (Select MAX(Data.Timestamp) FROM Data WHERE Data.ResetFlag = true AND Recipe_id = {}) AND Timestamp < DATE_ADD((SELECT MAX(Data.Timestamp) FROM Data WHERE Recipe_ID = {} AND Data.ResetFlag = true), INTERVAL 1 HOUR)".format(ArchiveID, ArchiveID, ArchiveID) + sql_select = "SELECT AVG(Data.Angle) as angle FROM Data {}".format(where_sql) + cur.execute(sql_select) + columns = [column[0] for column in cur.description] + result_angle = [dict(zip(columns, row)) for row in cur.fetchall()] + + if len(result_angle) > 0: + initial_angle=result_angle[0]['angle'] + initial_gravity = round((const0 * initial_angle**3 + const1 * initial_angle**2 + const2 * initial_angle + const3),2) + pass + + # Get Angle data for the last hour before end date -> Final Gravity caculation + where_sql="WHERE Recipe_id = {} and Timestamp < '{}' and Recipe_id = {} AND Timestamp > DATE_SUB('{}', INTERVAL 1 HOUR)".format(ArchiveID, End_date, ArchiveID, End_date) + sql_select="SELECT AVG(Data.Angle) as angle FROM Data {}".format(where_sql) + cur.execute(sql_select) + columns = [column[0] for column in cur.description] + result_angle = [dict(zip(columns, row)) for row in cur.fetchall()] + if len(result_angle) > 0: + final_angle=result_angle[0]['angle'] + final_gravity = round((const0 * final_angle**3 + const1 * final_angle**2 + const2 * final_angle + const3),2) + + attenuation=round((initial_gravity - final_gravity)*100/initial_gravity,1) + real_gravity = 0.1808 * initial_gravity + 0.8192 * final_gravity + alcohol_by_weight = ( 100 * (real_gravity - initial_gravity) / (1.0665 * initial_gravity - 206.65)) + alcohol_by_volume = round((alcohol_by_weight / 0.795),1) + + archive_header=dict.fromkeys(['ArchiveID', 'Spindle_Name', 'Batch', 'Recipe', 'Start_date', 'End_date', 'Const0', 'Const1', 'Const2', 'Const3', 'Formula', 'Initial_Gravity', 'Final_Gravity', 'Attenuation', 'Alcohol_by_volume']) + archive_header['Spindle_Name']=Spindle_Name + archive_header['Batch']=Batch + archive_header['Recipe']=Recipe + archive_header['Start_date']=Start_date.strftime('%Y-%m-%d') + archive_header['End_date']=End_date.strftime('%Y-%m-%d') + archive_header['RID_END']=RID_END + archive_header['Const0']=const0 + archive_header['Const1']=const1 + archive_header['Const2']=const2 + archive_header['Const3']=const3 + archive_header['Formula']=forumla + archive_header['Initial_Gravity']=initial_gravity + archive_header['Final_Gravity']=final_gravity + archive_header['Attenuation']=attenuation + archive_header['Alcohol_by_volume']=alcohol_by_volume + archive_header['ArchiveID']=ArchiveID + + return archive_header + + + @request_mapping(path='/getarchivevalues', method="POST", auth_required=False) + async def get_archive_values(self, request): + """ + --- + description: get archive values for specified archive id + tags: + - iSpindle + parameters: + - in: body + name: body + description: get archive values for specified archive id + required: true + + schema: + type: object + + properties: + name: + type: string + sensor: + type: "integer" + format: "int64" + heater: + type: "integer" + format: "int64" + agitator: + type: "integer" + format: "int64" + target_temp: + type: "integer" + format: "int64" + type: + type: string + example: + name: "Kettle 1" + type: "CustomKettleLogic" + + + responses: + "204": + description: successful operation + """ + data = await request.json() + + result= await self.get_all_archive_values(data) + + #header_data = await self.get_archive_header_data(ArchiveID) + #logger.error(header_data) + #return web.json_response(data=header_data) + return web.json_response(data=result) + + async def get_all_archive_values(self, data): + ArchiveID = data.get('ArchiveID') + Const0 = float(data.get('Const0')) + Const1 = float(data.get('Const1')) + Const2 = float(data.get('Const2')) + Const3 = float(data.get('Const3')) + Initial_Gravity = data.get('Initial_Gravity') + Start = data.get('Start_date') + End = data.get('End_date') + RID_END = data.get('RID_END') + if RID_END == True: + AND_SQL = " AND Timestamp <= (Select max(Timestamp) FROM Data WHERE Recipe_ID='{}' AND Internal = 'RID_END')".format(ArchiveID) + else: + AND_SQL="" + + sql_select = (f"SELECT DATE_FORMAT(Timestamp, '%Y-%m-%d %H:%i:%s') as Timestamp, temperature, angle, \ + ({Const0}*angle*angle*angle+ {Const1}*angle*angle +{Const2}*angle + {Const3}) as Servergravity, gravity, battery, rssi, \ + (({Initial_Gravity} - ({Const0}*angle*angle*angle+ {Const1}*angle*angle +{Const2}*angle + {Const3}))* 100 / {Initial_Gravity}) as attenuation, \ + ((100 * (0.1808 * {Initial_Gravity} + 0.8192 * ({Const0}*angle*angle*angle+ {Const1}*angle*angle +{Const2}*angle + {Const3})- {Initial_Gravity}) / (1.0665 * {Initial_Gravity} - 206.65)) / 0.795) as alcohol \ + FROM Data WHERE Recipe_ID = '{ArchiveID}'{AND_SQL} ORDER BY Timestamp ASC") + + # Get all data for the selected recipe + cnx = mysql.connector.connect( + user=spindle_SQL_USER, port=spindle_SQL_PORT, password=spindle_SQL_PASSWORD, host=spindle_SQL_HOST, database=spindle_SQL_DB) + cur = cnx.cursor() + + cur.execute(sql_select) + df = DataFrame(cur.fetchall()) + columns = [column[0] for column in cur.description] + df.columns = columns + df.set_index('Timestamp', inplace=True) + data = {"time": df.index.tolist()} + for col in df.columns: + data[col] = df[col].tolist() + return data + + @request_mapping(path='/removeridflag/{ArchiveID}/', method="POST", auth_required=False) + async def removeridflag(self, request): + """ + --- + description: Remove end of archive flag for specified archive id + tags: + - iSpindle + parameters: + - name: "ArchiveID" + in: "path" + description: "ArchiveID" + required: true + type: "integer" + format: "int64" + responses: + "200": + description: successful operation + """ + ArchiveID = request.match_info['ArchiveID'] + await self.removearchiveflag(ArchiveID) + #logger.error(header_data) + return web.json_response(status=200) + + async def removearchiveflag(self, ArchiveID): + cnx = mysql.connector.connect( + user=spindle_SQL_USER, port=spindle_SQL_PORT, password=spindle_SQL_PASSWORD, host=spindle_SQL_HOST, database=spindle_SQL_DB) + cur = cnx.cursor() + sql_update = "UPDATE Data SET Internal = NULL WHERE Recipe_ID = '{}' AND Internal = 'RID_END'".format(ArchiveID) + cur.execute(sql_update) + cnx.commit() + + @request_mapping(path='/addridflag/{ArchiveID}/{Timestamp}/', method="POST", auth_required=False) + async def addridflag(self, request): + """ + --- + description: Remove end of archive flag for specified archive id + tags: + - iSpindle + parameters: + - name: "ArchiveID" + in: "path" + description: "ArchiveID" + required: true + type: "integer" + format: "int64" + - name: "Timestamp" + in: "path" + description: "Timestamp" + required: true + type: "Timestamp" + format: "YYYY-MM-DD HH:MM:SS" + responses: + "200": + description: successful operation + """ + ArchiveID = request.match_info['ArchiveID'] + Timestamp = round(int(request.match_info['Timestamp'])/1000) + await self.addarchiveflag(ArchiveID, Timestamp) + #logger.error(header_data) + return web.json_response(status=200) + + async def addarchiveflag(self, ArchiveID, Timestamp): + cnx = mysql.connector.connect( + user=spindle_SQL_USER, port=spindle_SQL_PORT, password=spindle_SQL_PASSWORD, host=spindle_SQL_HOST, database=spindle_SQL_DB) + cur = cnx.cursor() + sql_update = "UPDATE Data SET Internal = 'RID_END' WHERE Recipe_ID = '{}' AND UNIX_TIMESTAMP(Timestamp) = {}".format(ArchiveID, Timestamp) + cur.execute(sql_update) + cnx.commit() + + @request_mapping(path='/deletearchive/{ArchiveID}/', method="POST", auth_required=False) + async def deletearchive(self, request): + """ + --- + description: Delete data from database for specified archive id + tags: + - iSpindle + parameters: + - name: "ArchiveID" + in: "path" + description: "ArchiveID" + required: true + type: "integer" + format: "int64" + responses: + "200": + description: successful operation + """ + ArchiveID = request.match_info['ArchiveID'] + await self.deletearchivefromdatabase(ArchiveID) + #logger.error(header_data) + return web.json_response(status=200) + + async def deletearchivefromdatabase(self, ArchiveID): + cnx = mysql.connector.connect( + user=spindle_SQL_USER, port=spindle_SQL_PORT, password=spindle_SQL_PASSWORD, host=spindle_SQL_HOST, database=spindle_SQL_DB) + cur = cnx.cursor() + sql_delete1 = "DELETE FROM Archive WHERE Recipe_ID = '{}'".format(ArchiveID) + sql_delete2 = "DELETE FROM Data WHERE Recipe_ID = '{}'".format(ArchiveID) + cur.execute(sql_delete1) + cur.execute(sql_delete2) + cnx.commit() + +def setup(cbpi): + cbpi.plugin.register("iSpindle", iSpindle) + cbpi.plugin.register("iSpindleEndpoint", iSpindleEndpoint) + cbpi.plugin.register("iSpindleConfig", iSpindleConfig) + pass diff --git a/cbpi/extension/iSpindle/config.yaml b/cbpi/extension/iSpindle/config.yaml new file mode 100644 index 0000000..2010a13 --- /dev/null +++ b/cbpi/extension/iSpindle/config.yaml @@ -0,0 +1,3 @@ +name: cbpi4-iSpindle +version: 4 +active: true \ No newline at end of file diff --git a/cbpi/http_endpoints/http_upload.py b/cbpi/http_endpoints/http_upload.py index 9bf52d4..701fe83 100644 --- a/cbpi/http_endpoints/http_upload.py +++ b/cbpi/http_endpoints/http_upload.py @@ -150,6 +150,13 @@ class UploadHttpEndpoints(): description: Get recipe list from Brewfather App tags: - Upload + parameters: + - name: "offset" + in: "path" + description: "offset for recipe list" + required: true + type: "integer" + format: "int64" responses: "200": description: successful operation