From 19375005354f8b4e0ce0b5bd403762d2258881f5 Mon Sep 17 00:00:00 2001 From: avollkopf Date: Fri, 3 Jan 2025 10:32:05 +0100 Subject: [PATCH] remove test plugin from cbpi server --- cbpi/extension/iSpindle/__init__.py | 754 ---------------------------- cbpi/extension/iSpindle/config.yaml | 3 - 2 files changed, 757 deletions(-) delete mode 100644 cbpi/extension/iSpindle/__init__.py delete mode 100644 cbpi/extension/iSpindle/config.yaml diff --git a/cbpi/extension/iSpindle/__init__.py b/cbpi/extension/iSpindle/__init__.py deleted file mode 100644 index 10cb75b..0000000 --- a/cbpi/extension/iSpindle/__init__.py +++ /dev/null @@ -1,754 +0,0 @@ - -# -*- 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 deleted file mode 100644 index 2010a13..0000000 --- a/cbpi/extension/iSpindle/config.yaml +++ /dev/null @@ -1,3 +0,0 @@ -name: cbpi4-iSpindle -version: 4 -active: true \ No newline at end of file