Merge branch 'development' of https://github.com/PiBrewing/craftbeerpi4 into development

This commit is contained in:
avollkopf 2025-01-03 10:30:45 +01:00
commit 54caf00c83
9 changed files with 796 additions and 7 deletions

View file

@ -1,3 +1,3 @@
__version__ = "4.4.9" __version__ = "4.5.0.a2"
__codename__ = "Yeast Starter" __codename__ = "Cross Country"

View file

@ -133,7 +133,8 @@ class Step:
def to_dict(self): def to_dict(self):
msg = self.instance.summary if self.instance is not None else "" 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 @dataclass
class Fermenter: class Fermenter:

View file

@ -46,6 +46,7 @@ class CBPiStep(CBPiBase):
self.props = props self.props = props
self.cancel_reason: StepResult = None self.cancel_reason: StepResult = None
self.summary = "" self.summary = ""
self.summary2 = None
self.task = None self.task = None
self.running: bool = False self.running: bool = False
self.logger = logging.getLogger(__name__) self.logger = logging.getLogger(__name__)

View file

@ -72,7 +72,6 @@ class LogController:
# remove duplicates # remove duplicates
names = set(names) names = set(names)
result = None result = None
def dateparse(time_in_secs): def dateparse(time_in_secs):
@ -120,7 +119,7 @@ class LogController:
data[name] = result.interpolate().tolist() data[name] = result.interpolate().tolist()
logging.info("Send Log for {}".format(names)) logging.info("Send Log for {}".format(names))
return data return data
async def get_data2(self, ids) -> dict: async def get_data2(self, ids) -> dict:

View 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

View file

@ -0,0 +1,3 @@
name: cbpi4-iSpindle
version: 4
active: true

View file

@ -384,7 +384,8 @@ class BoilStep(CBPiStep):
#self.dwelltime=int(self.props.get("DwellTime", 0))*60 #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.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 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)) self.kettle=self.get_kettle(self.props.get("Kettle", None))
if self.kettle is not None: if self.kettle is not None:
@ -405,6 +406,24 @@ class BoilStep(CBPiStep):
await self.setAutoMode(True) await self.setAutoMode(True)
await self.push_update() 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): async def check_hop_timer(self, number, value, text):
if value is not None and self.hops_added[number-1] is not True: 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): 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): async def on_stop(self):
await self.timer.stop() await self.timer.stop()
self.summary = "" self.summary = ""
self.summary2 = None
self.kettle.target_temp = 0 self.kettle.target_temp = 0
if self.AutoMode == True: if self.AutoMode == True:
await self.setAutoMode(False) 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) 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) self.cbpi.notify(self.name, 'Timer started. Estimated completion: {}'.format(estimated_completion_time.strftime("%H:%M")), NotificationType.INFO)
else: 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)) await self.check_hop_timer(x, self.props.get("Hop_%s" % x, None), self.props.get("Hop_%s_text" % x, None))
return StepResult.DONE return StepResult.DONE

View file

@ -2,6 +2,7 @@ from cbpi.controller.step_controller import StepController
from cbpi.api.dataclasses import Props, Step from cbpi.api.dataclasses import Props, Step
from aiohttp import web from aiohttp import web
from cbpi.api import * from cbpi.api import *
import logging
class StepHttpEndpoints(): class StepHttpEndpoints():
@ -10,6 +11,7 @@ class StepHttpEndpoints():
self.controller : StepController = cbpi.step self.controller : StepController = cbpi.step
self.cbpi.register(self, "/step2") self.cbpi.register(self, "/step2")
# Check if this is still needed
def create_dict(self, data): 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()) return dict(name=data["name"], id=data["id"], type=data.get("type"), status=data["status"],props=data["props"], state_text=data["instance"].get_state())

View file

@ -150,6 +150,13 @@ class UploadHttpEndpoints():
description: Get recipe list from Brewfather App description: Get recipe list from Brewfather App
tags: tags:
- Upload - Upload
parameters:
- name: "offset"
in: "path"
description: "offset for recipe list"
required: true
type: "integer"
format: "int64"
responses: responses:
"200": "200":
description: successful operation description: successful operation