mirror of
https://github.com/PiBrewing/craftbeerpi4.git
synced 2025-01-25 05:44:30 +01:00
Merge branch 'development' of https://github.com/PiBrewing/craftbeerpi4 into development
This commit is contained in:
commit
54caf00c83
9 changed files with 796 additions and 7 deletions
|
@ -1,3 +1,3 @@
|
|||
__version__ = "4.4.9"
|
||||
__codename__ = "Yeast Starter"
|
||||
__version__ = "4.5.0.a2"
|
||||
__codename__ = "Cross Country"
|
||||
|
||||
|
|
|
@ -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:
|
||||
|
|
|
@ -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__)
|
||||
|
|
|
@ -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:
|
||||
|
|
754
cbpi/extension/iSpindle/__init__.py
Normal file
754
cbpi/extension/iSpindle/__init__.py
Normal file
|
@ -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
|
3
cbpi/extension/iSpindle/config.yaml
Normal file
3
cbpi/extension/iSpindle/config.yaml
Normal file
|
@ -0,0 +1,3 @@
|
|||
name: cbpi4-iSpindle
|
||||
version: 4
|
||||
active: true
|
|
@ -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
|
||||
|
|
|
@ -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())
|
||||
|
||||
|
|
|
@ -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
|
||||
|
|
Loading…
Add table
Reference in a new issue