Skip to content

Commit

Permalink
Merge pull request #981 from SergeoLacruz/develop
Browse files Browse the repository at this point in the history
Add plugin for piusv
  • Loading branch information
Morg42 authored Jan 13, 2025
2 parents 7aac02b + 334c192 commit 850341c
Show file tree
Hide file tree
Showing 9 changed files with 771 additions and 0 deletions.
240 changes: 240 additions & 0 deletions piusv/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,240 @@
#!/usr/bin/env python3
# vim: set encoding=utf-8 tabstop=4 softtabstop=4 shiftwidth=4 expandtab
#########################################################################
# Copyright 2020- <AUTHOR> <EMAIL>
#########################################################################
# This file is part of SmartHomeNG.
# https://www.smarthomeNG.de
# https://knx-user-forum.de/forum/supportforen/smarthome-py
#
# Sample plugin for new plugins to run with SmartHomeNG version 1.8 and
# upwards.
#
# SmartHomeNG is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# SmartHomeNG is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with SmartHomeNG. If not, see <http://www.gnu.org/licenses/>.
#
#########################################################################

from lib.model.smartplugin import SmartPlugin
from lib.item import Items

from .webif import WebInterface

import smbus

class piusv(SmartPlugin):
"""
Main class of the Plugin. Does all plugin specific stuff and provides the update functions for the items
"""

PLUGIN_VERSION = '0.1.0'

def __init__(self, sh):
"""
Initalizes the plugin.
If you need the sh object at all, use the method self.get_sh() to get it. There should be almost no need for
a reference to the sh object anymore.
"""

# Call init code of parent class (SmartPlugin)
super().__init__()

self._item_dict = {}
self._cyclic_update_active = False
self.alive = False
self.suspended = False

# check if shNG is running on Raspberry Pi
try:
self.webif_pagelength = self.get_parameter_value('webif_pagelength')
self.poll_cycle = self.get_parameter_value('poll_cycle')
self.i2c_address = self.get_parameter_value('i2c_address')
except KeyError as e:
self.logger.critical("Plugin '{}': Inconsistent plugin (invalid metadata definition: {} not defined)".format(self.get_shortname(), e))
self._init_complete = False
return

self.init_webinterface(WebInterface)
return

####################################################################################
# Die Parameter der Pi USV+ byteweise auslesen
def get_parameter(self, index):
parameters = [0,0,0,0,0,0,0,0,0,0]

try:
self.piusv_handle.write_byte(self.i2c_address, 0x02)
except (IOError):
self.logger.error("get_parameter: error writing to piusv")
return(0)
for i in range(10):
try:
parameters[i] = self.piusv_handle.read_byte(self.i2c_address)
except (IOError):
self.logger.error("get_parameter: error reading to piusv")
return(0)
value = 256*parameters[index] + parameters[index+1]
return value

# Statusbyte auslesen
def get_status(self):

try:
self.piusv_handle.write_byte(self.i2c_address, 0x00)
except (IOError):
self.logger.error("get_status: error writing to piusv")
return(0)
try:
status = self.piusv_handle.read_byte(self.i2c_address)
except (IOError):
self.logger.error("get_status: error reading to piusv")
return(0)
return status

# Firmware auslesen
def get_firmware(self):

version = ''
try:
self.piusv_handle.write_byte(self.i2c_address, 0x01)
except (IOError):
self.logger.error("get_get_firmware: error writing to piusv")
return(0)
for i in range (12):
try:
version = version + chr(self.piusv_handle.read_byte(self.i2c_address))
except (IOError):
self.logger.error("get_firmware: error reading to piusv")
return(0)
return version
###############################################################################
def run(self):
"""
Run method for the plugin
"""
self.logger.debug("Run method called")
# Handle
self.piusv_handle = smbus.SMBus(1)

# setup scheduler for device poll loop (disable the following line, if you don't need to poll the device. Rember to comment the self_cycle statement in __init__ as well)
self.scheduler_add('poll_device', self.poll_device, cycle=self.poll_cycle)
self.alive = True

def stop(self):
"""
Stop method for the plugin
"""
self.logger.debug("Stop method called")
self.scheduler_remove('poll_device')
self.alive = False

def parse_item(self, item):
"""
Default plugin parse_item method. Is called when the plugin is initialized.
The plugin can, corresponding to its attribute keywords, decide what to do with
the item in the future, like adding it to an internal array for future reference
:param item: The item to process.
:return: If the plugin needs to be informed of an items change you should return a call back function
like the function update_item down below. An example when this is needed is the knx plugin
where parse_item returns the update_item function when the attribute knx_send is found.
This means that when the items value is about to be updated, the call back function is called
with the item, caller, source and dest as arguments and in case of the knx plugin the value
can be sent to the knx with a knx write function within the knx plugin.
"""
if self.has_iattr(item.conf, 'piusv_func'):
self.logger.debug(f"parse item: {item}")
self._item_dict[item] = self.get_iattr_value(item.conf, 'piusv_func')

elif self.has_iattr(item.conf, 'piusv_sys'):
return self.update_item

def update_item(self, item, caller=None, source=None, dest=None):
"""
Item has been updated
This method is called, if the value of an item has been updated by SmartHomeNG.
:param item: item to be updated towards the plugin
:param caller: if given it represents the callers name
:param source: if given it represents the source
:param dest: if given it represents the dest
"""
if self.alive and caller != self.get_shortname():
# code to execute if the plugin is not stopped and only, if the item has not been changed by this plugin:
self.logger.info(f"Update item: {item.property.path}, item has been changed outside this plugin")

if self.has_iattr(item.conf, 'piusv_sys'):
self.logger.debug(f"update_item was called with item {item.property.path} from caller {caller}, source {source} and dest {dest}")
if self.get_iattr_value(item.conf, 'piusv_sys') == 'update' and bool(item()):
self.logger.info(f"Update of all items of piusv Plugin requested. ")
self.poll_device()
item(False)
pass

def poll_device(self):
"""
Polls for updates of the device
"""
# check if another cyclic cmd run is still active
if self._cyclic_update_active:
self.logger.warning('Triggered cyclic poll_device, but previous cyclic run is still active. Therefore request will be skipped.')
return
elif self.suspended:
self.logger.warning('Triggered cyclic poll_device, but Plugin in suspended. Therefore request will be skipped.')
return
else:
self.logger.info('Triggering cyclic poll_device')

# set lock
self._cyclic_update_active = True

for item in self._item_dict:
# self.logger.debug(f"poll_device: handle item {item.id()}")
value = eval(f"self.{self.get_iattr_value(item.conf, 'piusv_func')}()")
# self.logger.info(f"poll_device: {value=} for item {item.id()} will be set.")
item(value, self.get_shortname())

# release lock
self._cyclic_update_active = False

pass

def u_batt(self):
return self.get_parameter(0)

def i_rasp(self):
return self.get_parameter(2)

def u_rasp(self):
return self.get_parameter(4)

def u_usb(self):
return self.get_parameter(6)

def u_ext(self):
return self.get_parameter(8)

def piusv_status(self):
return self.get_status()

def piusv_firmware(self):
return self.get_firmware()

@property
def item_list(self):
return list(self._item_dict.keys())

@property
def log_level(self):
return self.logger.getEffectiveLevel()
17 changes: 17 additions & 0 deletions piusv/locale.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
# translations for the web interface
plugin_translations:
'Value':
de: 'Wert'
en: 'Value'
'Attribute':
de: 'Attribut'
en: 'Attribute'
'Last Change':
de: 'Letzte Änderung'
en: 'Last Change'
'Last Update':
de: 'Letztes Update'
en: 'Last Update'
'Type':
de: 'Typ'
en: 'Type'
Loading

0 comments on commit 850341c

Please sign in to comment.