diff --git a/comfoair/__init__.py b/comfoair/__init__.py index 1bfcf970f..382495e72 100755 --- a/comfoair/__init__.py +++ b/comfoair/__init__.py @@ -1,4 +1,4 @@ -#!/usr/bin/env python +# !/usr/bin/env python ######################################################################### # Copyright 2013 Stefan Kals ######################################################################### @@ -18,11 +18,9 @@ # along with this plugin. If not, see . ######################################################################### -import logging import socket import time import serial -import re import threading from . import commands @@ -32,33 +30,29 @@ class ComfoAir(SmartPlugin): ALLOW_MULTIINSTANCE = False - PLUGIN_VERSION = '1.3.0' + PLUGIN_VERSION = '1.3.1' - def __init__(self, smarthome, host=None, port=0, serialport=None, kwltype='comfoair350'): - self.logger = logging.getLogger('ComfoAir') + def __init__(self, **kwargs): self.connected = False - self._sh = smarthome self._params = {} self._init_cmds = [] self._cyclic_cmds = {} self._lock = threading.Lock() - self._host = host - self._port = int(port) - self._serialport = serialport + self._host = self.get_parameter_value('host') + self._port = self.get_parameter_value('port') + self._serialport = self.get_parameter_value('serialport') + self._kwltype = self.get_parameter_value('kwltype') self._connection_attempts = 0 self._connection_errorlog = 60 self._initread = False - # automatically (re)connect - smarthome.connections.monitor(self) - # Load controlset and commandset - if kwltype in commands.controlset and kwltype in commands.commandset: - self._controlset = commands.controlset[kwltype] - self._commandset = commands.commandset[kwltype] - self.log_info('Loaded commands for KWL type \'{}\''.format(kwltype)) + if self._kwltype in commands.controlset and self._kwltype in commands.commandset: + self._controlset = commands.controlset[self._kwltype] + self._commandset = commands.commandset[self._kwltype] + self.log_info('Loaded commands for KWL type \'{}\''.format(self._kwltype)) else: - self.log_err('Commands for KWL type \'{}\' could not be found!'.format(kwltype)) + self.log_err('Commands for KWL type \'{}\' could not be found!'.format(self._kwltype)) return None # Remember packet config @@ -69,13 +63,13 @@ def __init__(self, smarthome, host=None, port=0, serialport=None, kwltype='comfo self._reponsecommandinc = self._controlset['ResponseCommandIncrement'] self._commandlength = 2 self._checksumlength = 1 - + def connect(self): - if self._serialport is not None: + if self._serialport: self.connect_serial() else: self.connect_tcp() - + def connect_tcp(self): self._lock.acquire() try: @@ -94,12 +88,12 @@ def connect_tcp(self): self.log_info('connected to {}:{}'.format(self._host, self._port)) self._connection_attempts = 0 self._lock.release() - + def connect_serial(self): self._lock.acquire() try: self._serialconnection = serial.Serial( - self._serialport, 9600, serial.EIGHTBITS, serial.PARITY_NONE, serial.STOPBITS_ONE, timeout=2) + self._serialport, 9600, serial.EIGHTBITS, serial.PARITY_NONE, serial.STOPBITS_ONE, timeout=2) except Exception as e: self._connection_attempts -= 1 if self._connection_attempts <= 0: @@ -118,8 +112,8 @@ def disconnect(self): self.disconnect_serial() else: self.disconnect_tcp() - - def disconnect_tcp(self): + + def disconnect_tcp(self): self.connected = False try: self._sock.shutdown(socket.SHUT_RDWR) @@ -130,54 +124,54 @@ def disconnect_tcp(self): except: pass - def disconnect_serial(self): + def disconnect_serial(self): self.connected = False try: self._serialconnection.close() self._serialconnection = None except: pass - + def send_bytes(self, packet): if self._serialport is not None: self.send_bytes_serial(packet) else: self.send_bytes_tcp(packet) - + def send_bytes_tcp(self, packet): self._sock.sendall(packet) def send_bytes_serial(self, packet): self._serialconnection.write(packet) - + def read_bytes(self, length): if self._serialport is not None: return self.read_bytes_serial(length) else: return self.read_bytes_tcp(length) - + def read_bytes_tcp(self, length): return self._sock.recv(length) def read_bytes_serial(self, length): return self._serialconnection.read(length) - + def parse_item(self, item): # Process the read config if self.has_iattr(item.conf, 'comfoair_read'): commandname = self.get_iattr_value(item.conf, 'comfoair_read') - if (commandname == None or commandname not in self._commandset): + if (commandname is None or commandname not in self._commandset): self.log_err('Item {} contains invalid read command \'{}\'!'.format(item, commandname)) return None - + # Remember the read config to later update this item if the configured response comes in self.log_info('Item {} reads by using command \'{}\'.'.format(item, commandname)) commandconf = self._commandset[commandname] commandcode = commandconf['Command'] - if not commandcode in self._params: + if commandcode not in self._params: self._params[commandcode] = {'commandname': [commandname], 'items': [item]} - elif not item in self._params[commandcode]['items']: + elif item not in self._params[commandcode]['items']: self._params[commandcode]['commandname'].append(commandname) self._params[commandcode]['items'].append(item) @@ -185,7 +179,7 @@ def parse_item(self, item): if (self.has_iattr(item.conf, 'comfoair_init') and self.get_iattr_value(item.conf, 'comfoair_init') == 'true'): self.log_info('Item {} is initialized on startup.'.format(item)) # Only add the item to the initial commands if it is not cyclic. Cyclic commands get called on init because this is the first cycle... - if not commandcode in self._init_cmds and not self.has_iattr(item.conf, 'comfoair_read_cycle'): + if commandcode not in self._init_cmds and not self.has_iattr(item.conf, 'comfoair_read_cycle'): self._init_cmds.append(commandcode) # Allow items to be cyclically updated @@ -193,7 +187,7 @@ def parse_item(self, item): cycle = int(self.get_iattr_value(item.conf, 'comfoair_read_cycle')) self.log_info('Item {} should read cyclic every {} seconds.'.format(item, cycle)) - if not commandcode in self._cyclic_cmds: + if commandcode not in self._cyclic_cmds: self._cyclic_cmds[commandcode] = {'cycle': cycle, 'nexttime': 0} else: # If another item requested this command already with a longer cycle, use the shorter cycle now @@ -203,12 +197,12 @@ def parse_item(self, item): # Process the send config if self.has_iattr(item.conf, 'comfoair_send'): commandname = self.get_iattr_value(item.conf, 'comfoair_send') - if commandname == None: + if commandname is None: return None elif commandname not in self._commandset: self.log_err('Item {} contains invalid write command \'{}\'!'.format(item, commandname)) return None - + self.log_info('Item {} writes by using command \'{}\''.format(item, commandname)) return self.update_item else: @@ -221,7 +215,7 @@ def update_item(self, item, caller=None, source=None, dest=None): if caller != 'ComfoAir' and self.has_iattr(item.conf, 'comfoair_send'): commandname = self.get_iattr_value(item.conf, 'comfoair_send') - if type(item) != int: + if type(item) is not int: value = int(item()) else: value = item() @@ -238,18 +232,18 @@ def update_item(self, item, caller=None, source=None, dest=None): aw = float(readafterwrite) time.sleep(aw) self.send_command(readcommandname) - - # If commands should be triggered after this write + + # If commands should be triggered after this write if self.has_iattr(item.conf, 'comfoair_trigger'): trigger = self.get_iattr_value(item.conf, 'comfoair_trigger') - if trigger == None: + if trigger is None: self.log_err('Item {} contains invalid trigger command list \'{}\'!'.format(item, trigger)) else: - tdelay = 5 # default delay + tdelay = 5 # default delay if self.has_iattr(item.conf, 'comfoair_trigger_afterwrite'): tdelay = float(self.get_iattr_value(item.conf, 'comfoair_trigger_afterwrite')) - if type(trigger) != list: - trigger = [trigger] + if type(trigger) is not list: + trigger = [trigger] for triggername in trigger: triggername = triggername.strip() if triggername is not None and readafterwrite is not None: @@ -268,32 +262,32 @@ def handle_cyclic_cmds(self): self.log_debug('Triggering cyclic read command: {}'.format(commandname)) self.send_command(commandname) entry['nexttime'] = currenttime + entry['cycle'] - + def send_command(self, commandname, value=None): try: - #self.log_debug('Got a new send job: Command {} with value {}'.format(commandname, value)) - + # self.log_debug('Got a new send job: Command {} with value {}'.format(commandname, value)) + # Get command config commandconf = self._commandset[commandname] commandcode = int(commandconf['Command']) commandcodebytecount = commandconf['CommandBytes'] commandtype = commandconf['Type'] commandvaluebytes = commandconf['ValueBytes'] - #self.log_debug('Command config: {}'.format(commandconf)) - + # self.log_debug('Command config: {}'.format(commandconf)) + # Transform value for write commands - #self.log_debug('Got value: {}'.format(value)) + # self.log_debug('Got value: {}'.format(value)) if 'ValueTransform' in commandconf and value is not None and value != '' and commandtype == 'Write': commandtransform = commandconf['ValueTransform'] value = self.value_transform(value, commandtype, commandtransform) - #self.log_debug('Transformed value using method {} to {}'.format(commandtransform, value)) - + # self.log_debug('Transformed value using method {} to {}'.format(commandtransform, value)) + # Build value byte array valuebytes = bytearray() if value is not None and commandvaluebytes > 0: valuebytes = self.int2bytes(value, commandvaluebytes) - #self.log_debug('Created value bytes: {}'.format(valuebytes)) - + # self.log_debug('Created value bytes: {}'.format(valuebytes)) + # Calculate the checksum commandbytes = self.int2bytes(commandcode, commandcodebytecount) payload = bytearray() @@ -301,7 +295,7 @@ def send_command(self, commandname, value=None): if len(valuebytes) > 0: payload.extend(valuebytes) checksum = self.calc_checksum(payload) - + # Build packet packet = bytearray() packet.extend(self.int2bytes(self._controlset['PacketStart'], 2)) @@ -311,22 +305,22 @@ def send_command(self, commandname, value=None): packet.extend(self.int2bytes(checksum, 1)) packet.extend(self.int2bytes(self._controlset['PacketEnd'], 2)) self.log_debug('Preparing command {} with value {} (transformed to value byte \'{}\') to be sent.'.format(commandname, value, self.bytes2hexstring(valuebytes))) - + # Use a lock to allow only one sender at a time self._lock.acquire() if not self.connected: raise Exception("No connection to ComfoAir.") - + try: self.send_bytes(packet) self.log_debug('Successfully sent packet: {}'.format(self.bytes2hexstring(packet))) except Exception as e: raise Exception('Exception while sending: {}'.format(e)) - + if commandtype == 'Read': packet = bytearray() - + # Try to receive a packet start, a command and a data length byte firstpartlen = len(self._packetstart) + self._commandlength + 1 while self.alive and len(packet) < firstpartlen: @@ -335,9 +329,9 @@ def send_command(self, commandname, value=None): self.log_debug('Trying to receive {} bytes for the first part of the response.'.format(bytestoreceive)) chunk = self.read_bytes(bytestoreceive) self.log_debug('Received {} bytes chunk of response part 1: {}'.format(len(chunk), self.bytes2hexstring(chunk))) - if len(chunk) == 0: + if len(chunk) == 0: raise Exception('Received 0 bytes chunk - ignoring packet!') - + # Cut away old ACK (but only if the telegram wasn't started already) if len(packet) == 0: chunk = self.remove_ack_begin(chunk) @@ -346,11 +340,11 @@ def send_command(self, commandname, value=None): raise Exception("error receiving first part of packet: timeout") except Exception as e: raise Exception("error receiving first part of packet: {}".format(e)) - + datalen = packet[firstpartlen - 1] - #self.log_info('Got a data length of: {}'.format(datalen)) + # self.log_info('Got a data length of: {}'.format(datalen)) packetlen = firstpartlen + datalen + self._checksumlength + len(self._packetend) - + # Try to receive the second part of the packet while self.alive and len(packet) < packetlen or packet[-2:] != self._packetend: try: @@ -358,42 +352,42 @@ def send_command(self, commandname, value=None): if len(packet) >= packetlen and packet[-2:] != self._packetend: packetlen = len(packet) + 1 self.log_debug('Extended packet length because of encoded characters.'.format(self.bytes2hexstring(chunk))) - + # Receive next chunk bytestoreceive = packetlen - len(packet) self.log_debug('Trying to receive {} bytes for the second part of the response.'.format(bytestoreceive)) chunk = self.read_bytes(bytestoreceive) self.log_debug('Received {} bytes chunk of response part 2: {}'.format(len(chunk), self.bytes2hexstring(chunk))) packet.extend(chunk) - if len(chunk) == 0: + if len(chunk) == 0: raise Exception('Received 0 bytes chunk - ignoring packet!') except socket.timeout: raise Exception("error receiving second part of packet: timeout") except Exception as e: raise Exception("error receiving second part of packet: {}".format(e)) - + # Send ACK self.send_bytes(self._acknowledge) - + # Parse response self.parse_response(packet) - + except Exception as e: self.disconnect() self.log_err("send_command failed: {}".format(e)) - finally: + finally: # At the end, release the lock self._lock.release() def parse_response(self, response): - #resph = self.bytes2int(response) + # resph = self.bytes2int(response) self.log_debug('Successfully received response: {}'.format(self.bytes2hexstring(response))) # A telegram looks like this: start sequence (2 bytes), command (2 bytes), data length (1 byte), data, checksum (1 byte), end sequence (2 bytes, already cut away) commandcodebytes = response[2:4] - commandcodebytes[1] -= self._reponsecommandinc # The request command of this response is -1 (for comfoair 350) - commandcodebytes.append(0) # Add a data length byte of 0 (always true for read commands) + commandcodebytes[1] -= self._reponsecommandinc # The request command of this response is -1 (for comfoair 350) + commandcodebytes.append(0) # Add a data length byte of 0 (always true for read commands) commandcode = self.bytes2int(commandcodebytes) # Remove begin and checksum to get the data @@ -405,8 +399,8 @@ def parse_response(self, response): # Validate checksum packetpart = bytearray() - packetpart.extend(response[2:5]) # Command and data length - packetpart.extend(databytes) # Decoded data bytes + packetpart.extend(response[2:5]) # Command and data length + packetpart.extend(databytes) # Decoded data bytes checksum = self.calc_checksum(packetpart) receivedchecksum = response[len(response) - len(self._packetend) - 1] if (receivedchecksum != checksum): @@ -437,7 +431,7 @@ def parse_response(self, response): # Extract value valuebytes = databytes[index:index + commandvaluebytes] rawvalue = self.bytes2int(valuebytes) - + # Tranform value value = self.value_transform(rawvalue, commandtype, commandtransform) self.log_debug('Matched command {} and read transformed value {} (raw value was {}) from byte position {} and byte length {}.'.format(commandname, value, rawvalue, commandresppos, commandvaluebytes)) @@ -448,20 +442,24 @@ def parse_response(self, response): self.log_err('Telegram did not contain enough data bytes for the configured command {} to extract a value!'.format(commandname)) def run(self): + # automatically (re)connect + self._sh.connections.monitor(self) + self.alive = True - self._sh.scheduler.add('ComfoAir-init', self.send_init_commands, prio=5, cycle=600, offset=2) + self.scheduler_add('ComfoAir-init', self.send_init_commands, prio=5, cycle=600, offset=2) maxloops = 20 - loops = 0 + loops = 0 while self.alive and not self._initread and loops < maxloops: # wait for init read to finish time.sleep(0.5) loops += 1 - self._sh.scheduler.remove('ComfoAir-init') - + self.scheduler_remove('ComfoAir-init') + def stop(self): - self._sh.scheduler.remove('ComfoAir-cyclic') self.alive = False + self._sh.connections.remove(self) + self.scheduler_remove('ComfoAir-cyclic') self.disconnect() - + def send_init_commands(self): try: # Do the init read commands @@ -471,19 +469,19 @@ def send_init_commands(self): for commandcode in self._init_cmds: commandname = self.commandname_by_commandcode(commandcode) self.send_command(commandname) - + # Find the shortest cycle shortestcycle = -1 for commandname in list(self._cyclic_cmds.keys()): entry = self._cyclic_cmds[commandname] if shortestcycle == -1 or entry['cycle'] < shortestcycle: shortestcycle = entry['cycle'] - + # Start the worker thread if shortestcycle != -1: # Balance unnecessary calls and precision workercycle = int(shortestcycle / 2) - self._sh.scheduler.add('ComfoAir-cyclic', self.handle_cyclic_cmds, cycle=workercycle, prio=5, offset=0) + self.scheduler_add('ComfoAir-cyclic', self.handle_cyclic_cmds, cycle=workercycle, prio=5, offset=0) self.log_info('Added cyclic worker thread ({} sec cycle). Shortest item update cycle found: {} sec.'.format(workercycle, shortestcycle)) finally: self._initread = True @@ -494,30 +492,30 @@ def remove_ack_begin(self, packet): while len(packet) >= acklen and packet[0:acklen] == self._acknowledge: packet = packet[acklen:] return packet - + def calc_checksum(self, packetpart): return (sum(packetpart) + 173) % 256 - - def log_debug(self, text): + + def log_debug(self, text): self.logger.debug('ComfoAir: {}'.format(text)) - def log_info(self, text): + def log_info(self, text): self.logger.info('ComfoAir: {}'.format(text)) - def log_err(self, text): + def log_err(self, text): self.logger.error('ComfoAir: {}'.format(text)) - + def int2bytes(self, value, length): # Limit value to the passed byte length value = value % (2 ** (length * 8)) return value.to_bytes(length, byteorder='big') - + def bytes2int(self, bytesvalue): return int.from_bytes(bytesvalue, byteorder='big', signed=False) - + def bytes2hexstring(self, bytesvalue): return ":".join("{:02x}".format(c) for c in bytesvalue) - + def encode_specialchars(self, packet): specialchar = self._controlset['SpecialCharacter'] encodedpacket = bytearray() @@ -528,9 +526,9 @@ def encode_specialchars(self, packet): # Encoding works by doubling the special char self.log_debug('Encoded special char at position {} of data bytes {}.'.format(count, self.bytes2hexstring(packet))) encodedpacket.append(char) - #self.log_debug('Encoded data bytes: {}.'.format(encodedpacket)) + # self.log_debug('Encoded data bytes: {}.'.format(encodedpacket)) return encodedpacket - + def decode_specialchars(self, packet): specialchar = self._controlset['SpecialCharacter'] decodedpacket = bytearray() @@ -549,7 +547,7 @@ def decode_specialchars(self, packet): # Reset dropping marker specialcharremoved = 0 return decodedpacket - + def value_transform(self, value, commandtype, transformmethod): if transformmethod == 'Temperature': if commandtype == 'Read': @@ -562,7 +560,7 @@ def value_transform(self, value, commandtype, transformmethod): elif commandtype == 'Write': return int(1875000 / value) return value - + def commandname_by_commandcode(self, commandcode): for commandname in self._commandset.keys(): if self._commandset[commandname]['Command'] == commandcode: diff --git a/comfoair/plugin.yaml b/comfoair/plugin.yaml index 806000e79..7438bad8e 100755 --- a/comfoair/plugin.yaml +++ b/comfoair/plugin.yaml @@ -12,11 +12,11 @@ plugin: documentation: https://github.com/smarthomeNG/smarthome/wiki/Comfoair-Plugin # url of documentation (wiki) page support: https://knx-user-forum.de/forum/supportforen/smarthome-py/31291-neues-plugin-comfoair-kwl-wohnraumlüftung-zehnder-paul-wernig - version: 1.3.0 # Plugin version + version: 1.3.1 # Plugin version sh_minversion: 1.3 # minimum shNG version to use this plugin # sh_maxversion: # maximum shNG version to use this plugin (leave empty if latest) - multi_instance: False - restartable: unknown + multi_instance: false + restartable: true classname: ComfoAir # class containing the plugin parameters: @@ -30,16 +30,19 @@ parameters: valid_list: - comfoair350 - comfoair500 + host: type: ip description: de: 'Netzwerverbindung: Hostname/IP des KWL Systems' en: 'Network connection: Hostname/IP of KWL system' + port: type: int description: de: 'Netzwerkverbindung: Port des KWL Systems' en: 'Network connection: Port of KWL system' + serialport: type: str description: @@ -61,22 +64,22 @@ item_attributes: comfoair_read_afterwrite: type: num description: - de: 'Konfiguriert eine Verzögerung in Sekunden nachdem ein Lesekommando nach einem Schreibkommando an das KWL System geschickt wird.' + de: 'Konfiguriert eine Verzögerung in Sekunden, nachdem ein Lesekommando nach einem Schreibkommando an das KWL System geschickt wird.' en: 'Configures delay in seconds to issue a read command after write command.' comfoair_read_cycle: type: num description: - de: 'Konfiguriert ein Interval in Sekunden für das Lesekommando.' + de: 'Konfiguriert ein Intervall in Sekunden für das Lesekommando.' en: 'Configures a interval in seconds for the read command.' comfoair_init: type: bool description: - de: 'Konfiguriert ob der Wert aus dem KWL System initialisiert werden soll.' + de: 'Konfiguriert, ob der Wert aus dem KWL System initialisiert werden soll.' en: 'Configures to initialize the item value with the value from the KWL system.' comfoair_trigger: type: list(str) description: - de: 'Konfiguriert Lesekommandos die nach einem Schreibvorgang auf das Item aufgerufen werden.' + de: 'Konfiguriert Lesekommandos, die nach einem Schreibvorgang auf das Item aufgerufen werden.' en: 'Configures read commands after an update to the item.' comfoair_trigger_afterwrite: type: num diff --git a/helios/__init__.py b/helios/__init__.py index ff7e14bd7..b08b0f9f2 100755 --- a/helios/__init__.py +++ b/helios/__init__.py @@ -1,4 +1,4 @@ -#!/usr/bin/env python +# !/usr/bin/env python ######################################################################### # Copyright 2014 Marcel Tiews marcel.tiews@gmail.com # Modified 2014-2017 by René Jahncke aka Tom-Bom-badil @ github.com @@ -22,11 +22,8 @@ import sys import serial import logging -import socket import threading -import struct import time -import datetime import array from lib.model.smartplugin import SmartPlugin @@ -34,10 +31,10 @@ # old / removed: logger = logging.getLogger("") # Old CONST's - previous definitions -#CONST_BUSMEMBER__MAINBOARD = 0x11 -#CONST_BUSMEMBER__SLAVEBOARDS = 0x10 -#CONST_BUSMEMBER__CONTROLBOARDS = 0x20 -#CONST_BUSMEMBER__ME = 0x2F +# CONST_BUSMEMBER__MAINBOARD = 0x11 +# CONST_BUSMEMBER__SLAVEBOARDS = 0x10 +# CONST_BUSMEMBER__CONTROLBOARDS = 0x20 +# CONST_BUSMEMBER__ME = 0x2F # Broadcast addresses - no way to address slave boards in the units directly (according to Vallox) @@ -45,10 +42,10 @@ CONST_BUS_ALL_REMOTES = 0x20 # Individual addresses -CONST_BUS_MAINBOARD1 = 0x11 # 1st of max 15 ventilation units (mainboards 1-F) -CONST_BUS_REMOTE1 = 0x21 # 1st of max 15 remote controls (remotes 1-F, default jumper = 1) -CONST_BUS_LON = 0x28 # default for LON bus module (just for information --> expensive) -CONST_BUS_ME = 0x2F # stealth mode - we are behaving like a regular remote control +CONST_BUS_MAINBOARD1 = 0x11 # 1st of max 15 ventilation units (mainboards 1-F) +CONST_BUS_REMOTE1 = 0x21 # 1st of max 15 remote controls (remotes 1-F, default jumper = 1) +CONST_BUS_LON = 0x28 # default for LON bus module (just for information --> expensive) +CONST_BUS_ME = 0x2F # stealth mode - we are behaving like a regular remote control CONST_MAP_VARIABLES_TO_ID = { "power_state" : {"varid" : 0xA3, 'type': 'bit', 'bitposition': 0, 'read': True, 'write': True }, @@ -70,9 +67,9 @@ "boost_status" : {"varid" : 0x71, 'type': 'bit', 'bitposition': 6, 'read': True, 'write': False }, "boost_remaining" : {"varid" : 0x79, 'type': 'dec', 'bitposition': -1, 'read': True, 'write': False }, "fan_in_on_off" : {"varid" : 0x08, 'type': 'bit', 'bitposition': 3, 'read': True, 'write': True }, - "fan_in_percent" : {"varid" : 0xB0, 'type': 'dec', 'bitposition': -1, 'read': True, 'write': True }, + "fan_in_percent" : {"varid" : 0xB0, 'type': 'dec', 'bitposition': -1, 'read': True, 'write': True }, "fan_out_on_off" : {"varid" : 0x08, 'type': 'bit', 'bitposition': 5, 'read': True, 'write': True }, - "fan_out_percent" : {"varid" : 0xB1, 'type': 'dec', 'bitposition': -1, 'read': True, 'write': True }, + "fan_out_percent" : {"varid" : 0xB1, 'type': 'dec', 'bitposition': -1, 'read': True, 'write': True }, "clean_filter" : {"varid" : 0xAB, 'type': 'dec', 'bitposition': -1, 'read': True, 'write': True }, "device_error" : {"varid" : 0x36, 'type': 'dec', 'bitposition': -1, 'read': True, 'write': False } } @@ -96,54 +93,56 @@ class HeliosException(Exception): class HeliosBase(SmartPlugin): - PLUGIN_VERSION = "1.4.2" + PLUGIN_VERSION = "1.4.3" ALLOW_MULTIINSTANCE = False - def __init__(self, tty='/dev/ttyUSB0'): + def __init__(self, **kwargs): self.logger = logging.getLogger(__name__) - self._tty = tty self._is_connected = False - self._port = False self._lock = threading.Lock() - + if 'tty' in kwargs: + self._tty = kwargs['tty'] + if 'port' in kwargs: + self._port = kwargs['port'] + def connect(self): if self._is_connected and self._port: return True - + try: self.logger.debug("Helios: Connecting...") self._port = serial.Serial( - self._tty, - baudrate=9600, - bytesize=serial.EIGHTBITS, - parity=serial.PARITY_NONE, - stopbits=serial.STOPBITS_ONE, + self._tty, + baudrate=9600, + bytesize=serial.EIGHTBITS, + parity=serial.PARITY_NONE, + stopbits=serial.STOPBITS_ONE, timeout=1) self._is_connected = True return True except: self.logger.error("Helios: Could not open {0}.".format(self._tty)) return False - + def disconnect(self): if self._is_connected and self._port: self.logger.debug("HeliosBase: Disconnecting...") self._port.close() self._is_connected = False - + def _createTelegram(self, sender, receiver, function, value): - telegram = [1,sender,receiver,function,value,0] + telegram = [1, sender, receiver, function, value, 0] telegram[5] = self._calculateCRC(telegram) return telegram - + def _waitForSilence(self): # Modbus RTU only allows one master (client which controls communication). # So lets try to wait a bit and jump in when nobody's speaking. # Modbus defines a waittime of 3,5 Characters between telegrams: - # (1/9600baud * (1 Start bit + 8 Data bits + 1 Parity bit + 1 Stop bit) + # (1/9600baud * (1 Start bit + 8 Data bits + 1 Parity bit + 1 Stop bit) # => about 4ms # Lets go with 7ms! ;O) - + gotSlot = False backupTimeout = self._port.timeout end = time.time() + 3 @@ -155,128 +154,127 @@ def _waitForSilence(self): gotSlot = True break self._port.timeout = backupTimeout - return gotSlot + return gotSlot def _sendTelegram(self, telegram): if not self._is_connected: return False - + self.logger.debug("Helios: Sending telegram '{0}'".format(self._telegramToString(telegram))) self._port.write(bytearray(telegram)) return True - + def _readTelegram(self, sender, receiver, datapoint): # sometimes a lot of garbage is received...so lets get a bit robust # and read a bit of this junk and see whether we are getting something # useful out of it! # How long does it take until something useful is received??? timeout = time.time() + 1 - telegram = [0,0,0,0,0,0] + telegram = [0, 0, 0, 0, 0, 0] while self._is_connected and timeout > time.time(): char = self._port.read(1) - if(len(char) > 0): + if (len(char) > 0): byte = bytearray(char)[0] telegram.pop(0) telegram.append(byte) # Telegrams always start with a 0x01, is the CRC valid?, ... - if (telegram[0] == 0x01 and - telegram[1] == sender and - telegram[2] == receiver and - telegram[3] == datapoint and - telegram[5] == self._calculateCRC(telegram)): + if (telegram[0] == 0x01 and + telegram[1] == sender and + telegram[2] == receiver and + telegram[3] == datapoint and + telegram[5] == self._calculateCRC(telegram)): self.logger.debug("Telegram received '{0}'".format(self._telegramToString(telegram))) return telegram[4] - return None - + def _calculateCRC(self, telegram): sum = 0 for c in telegram[:-1]: sum = sum + c return sum % 256 - + def _telegramToString(self, telegram): str = "" for c in telegram: # str = str + hex(c) + " " 0x01 was showing as 0x1, 0x1A was showing as 0x1a - str = str + '0x%0*X' % (2,c) + " " - str = str[:-1] # remove trailing space + str = str + '0x%0*X' % (2, c) + " " + str = str[:-1] # remove trailing space return str - + def _convertFromRawValue(self, varname, rawvalue): value = None vardef = CONST_MAP_VARIABLES_TO_ID[varname] - + if vardef["type"] == "temperature": value = CONST_TEMPERATURE[rawvalue] elif vardef["type"] == "fanspeed": if rawvalue == 0x01: value = 1 - elif rawvalue == 0x03: + elif rawvalue == 0x03: value = 2 - elif rawvalue == 0x07: + elif rawvalue == 0x07: value = 3 - elif rawvalue == 0x0F: + elif rawvalue == 0x0F: value = 4 - elif rawvalue == 0x1F: + elif rawvalue == 0x1F: value = 5 - elif rawvalue == 0x3F: + elif rawvalue == 0x3F: value = 6 - elif rawvalue == 0x7F: + elif rawvalue == 0x7F: value = 7 - elif rawvalue == 0xFF: + elif rawvalue == 0xFF: value = 8 else: value = None elif vardef["type"] == "bit": value = rawvalue >> vardef["bitposition"] & 0x01 - elif vardef["type"] == "dec": # decimal value + elif vardef["type"] == "dec": # decimal value value = rawvalue - - return value + + return value def _convertFromValue(self, varname, value, prevvalue): rawvalue = None vardef = CONST_MAP_VARIABLES_TO_ID[varname] - + if vardef['type'] == "temperature": rawvalue = CONST_TEMPERATURE.index(int(value)) elif vardef["type"] == "fanspeed": value = int(value) if value == 1: rawvalue = 0x01 - elif value == 2: + elif value == 2: rawvalue = 0x03 - elif value == 3: + elif value == 3: rawvalue = 0x07 - elif value == 4: + elif value == 4: rawvalue = 0x0F - elif value == 5: + elif value == 5: rawvalue = 0x1F - elif value == 6: + elif value == 6: rawvalue = 0x3F - elif value == 7: + elif value == 7: rawvalue = 0x7F - elif value == 8: + elif value == 8: rawvalue = 0xFF else: rawvalue = None elif vardef["type"] == "bit": # for bits we have to keep the other bits of the byte (previous value) - if value in (True,1,"true","True","1","On","on"): + if value in (True, 1, "true", "True", "1", "On", "on"): rawvalue = prevvalue | (1 << vardef["bitposition"]) else: rawvalue = prevvalue & ~(1 << vardef["bitposition"]) - elif vardef["type"] == "dec": # decimal value + elif vardef["type"] == "dec": # decimal value rawvalue = int(value) - - return rawvalue - - def writeValue(self,varname, value): - if CONST_MAP_VARIABLES_TO_ID[varname]["write"] != True: + + return rawvalue + + def writeValue(self, varname, value): + if CONST_MAP_VARIABLES_TO_ID[varname]["write"] is not True: self.logger.error("Helios: Variable {0} may not be written!".format(varname)) - return False + return False success = False - + self._lock.acquire() try: # if we have got to write a single bit, we need the current (byte) value to @@ -287,43 +285,42 @@ def writeValue(self,varname, value): # Send poll request telegram = self._createTelegram( CONST_BUS_ME, - CONST_BUS_MAINBOARD1, - 0, + CONST_BUS_MAINBOARD1, + 0, CONST_MAP_VARIABLES_TO_ID[varname]["varid"] ) self._sendTelegram(telegram) # Read response currentval = self._readTelegram( - CONST_BUS_MAINBOARD1, - CONST_BUS_ME, + CONST_BUS_MAINBOARD1, + CONST_BUS_ME, CONST_MAP_VARIABLES_TO_ID[varname]["varid"] ) - if currentval == None: - self.logger.error("Helios: Sending value to ventilation system failed. Can not read current variable value '{0}'." - .format(varname)) + if currentval is None: + self.logger.error("Helios: Sending value to ventilation system failed. Can not read current variable value '{0}'.".format(varname)) return False rawvalue = self._convertFromValue(varname, value, currentval) - else: + else: rawvalue = self._convertFromValue(varname, value, None) - - # send the new value + + # send the new value if self._waitForSilence(): - if rawvalue != None: + if rawvalue is not None: # Broadcasting value to all remote control boards telegram = self._createTelegram( CONST_BUS_ME, - CONST_BUS_ALL_REMOTES, - CONST_MAP_VARIABLES_TO_ID[varname]["varid"], + CONST_BUS_ALL_REMOTES, + CONST_MAP_VARIABLES_TO_ID[varname]["varid"], rawvalue ) self._sendTelegram(telegram) - + # Broadcasting value to all mainboards telegram = self._createTelegram( CONST_BUS_ME, - CONST_BUS_ALL_MAINBOARDS, - CONST_MAP_VARIABLES_TO_ID[varname]["varid"], + CONST_BUS_ALL_MAINBOARDS, + CONST_MAP_VARIABLES_TO_ID[varname]["varid"], rawvalue ) self._sendTelegram(telegram) @@ -331,75 +328,75 @@ def writeValue(self,varname, value): # Writing value to 1st mainboard telegram = self._createTelegram( CONST_BUS_ME, - CONST_BUS_MAINBOARD1, - CONST_MAP_VARIABLES_TO_ID[varname]["varid"], - rawvalue + CONST_BUS_MAINBOARD1, + CONST_MAP_VARIABLES_TO_ID[varname]["varid"], + rawvalue ) self._sendTelegram(telegram) - + # Send checksum a second time self._sendTelegram([telegram[5]]) -#################### Special treatment to switch on remote controls after off state: + # Special treatment to switch on remote controls after off state: + # TODO: doesn't work so far if CONST_MAP_VARIABLES_TO_ID[varname]["varid"] == 0xA3 and CONST_MAP_VARIABLES_TO_ID[varname]["bitposition"] == 0: - self.logger.debug("On/off command - special treatment for the remote controls") + self.logger.debug("On/off command - special treatment for the remote controls") telegram = self._createTelegram( CONST_BUS_ME, - CONST_BUS_ALL_REMOTES, - CONST_MAP_VARIABLES_TO_ID[varname]["varid"], - rawvalue + CONST_BUS_ALL_REMOTES, + CONST_MAP_VARIABLES_TO_ID[varname]["varid"], + rawvalue ) self._sendTelegram(telegram) telegram = self._createTelegram( CONST_BUS_ME, - CONST_BUS_REMOTE1, - CONST_MAP_VARIABLES_TO_ID[varname]["varid"], - rawvalue + CONST_BUS_REMOTE1, + CONST_MAP_VARIABLES_TO_ID[varname]["varid"], + rawvalue ) self._sendTelegram(telegram) self._sendTelegram([telegram[5]]) -#################### Doesn't work so far + # Doesn't work so far success = True - + else: - self.logger.error("Helios: Sending value to ventilation system failed. Can not convert value '{0}' for variable '{1}'." - .format(value,varname)) + self.logger.error("Helios: Sending value to ventilation system failed. Can not convert value '{0}' for variable '{1}'.".format(value,varname)) success = False else: self.logger.error("Helios: Sending value to ventilation system failed. No free slot for sending telegrams available.") success = False except Exception as e: - self.logger.error("Helios: Exception in writeValue() occurred: {0}".format(e)) + self.logger.error("Helios: Exception in writeValue() occurred: {0}".format(e)) finally: self._lock.release() - + return success - - def readValue(self,varname): - if CONST_MAP_VARIABLES_TO_ID[varname]["read"] != True: + + def readValue(self, varname): + if CONST_MAP_VARIABLES_TO_ID[varname]["read"] is not True: self.logger.error("Variable {0} may not be read!".format(varname)) return False value = None - + self._lock.acquire() try: - self.logger.debug("Helios: Reading value: {0}".format(varname)) + self.logger.debug("Helios: Reading value: {0}".format(varname)) if self._waitForSilence(): # Send poll request telegram = self._createTelegram( CONST_BUS_ME, - CONST_BUS_MAINBOARD1, - 0, + CONST_BUS_MAINBOARD1, + 0, CONST_MAP_VARIABLES_TO_ID[varname]["varid"] ) self._sendTelegram(telegram) # Read response value = self._readTelegram( - CONST_BUS_MAINBOARD1, - CONST_BUS_ME, + CONST_BUS_MAINBOARD1, + CONST_BUS_ME, CONST_MAP_VARIABLES_TO_ID[varname]["varid"] ) if value is not None: @@ -408,36 +405,37 @@ def readValue(self,varname): self.logger.debug("Value for {0} ({1}) received: {2}|{3}|{4} --> converted = {5}" .format(varname, '0x%0*X' % (2, CONST_MAP_VARIABLES_TO_ID[varname]["varid"]), '0x%0*X' % (2,raw_value), "{0:08b}".format(raw_value), raw_value, value) - ) + ) else: # logging as info only, so we stop spamming log file as some noise on the bus seems to be normal - self.logger.info("Helios: No valid value for '{0}' from ventilation system received." - .format(varname) - ) + self.logger.info("Helios: No valid value for '{0}' from ventilation system received.".format(varname) + ) else: self.logger.warning("Helios: Reading value from ventilation system failed. No free slot to send poll request available.") except Exception as e: - self.logger.error("Helios: Exception in readValue() occurred: {0}".format(e)) + self.logger.error("Helios: Exception in readValue() occurred: {0}".format(e)) finally: self._lock.release() - + return value - -class Helios(HeliosBase): + +class Helios(HeliosBase): _items = {} - - def __init__(self, smarthome, tty, cycle=300): - HeliosBase.__init__(self, tty) - self._sh = smarthome - self._cycle = int(cycle) + + def __init__(self): + self._tty = self.get_parameter_value('tty') + self._cycle = self.get_parameter_value('cycle') + self._port = None + super().__init__() self._alive = False - + def run(self): self.connect() self._alive = True - self._sh.scheduler.add('Helios', self._update, cycle=self._cycle) + self.scheduler_add('Helios', self._update, cycle=self._cycle) def stop(self): + self.scheduler_remove('Helios') self.disconnect() self._alive = False @@ -449,62 +447,65 @@ def parse_item(self, item): return self.update_item else: self.logger.warning("Helios: Ignoring unknown variable '{0}'".format(varname)) - + def update_item(self, item, caller=None, source=None, dest=None): if caller != 'Helios': - self.writeValue(item.conf['helios_var'], item()) - + self.writeValue(item.conf['helios_var'], item()) + def _update(self): self.logger.debug("Helios: Updating values") for var in self._items.keys(): val = self.readValue(var) - if val != None: - self._items[var](val,"Helios") + if val is not None: + self._items[var](val, "Helios") + - def main(): - import argparse - + import argparse + parser = argparse.ArgumentParser( - description="Helios ventilation system commandline interface.", - epilog="Without arguments all readable values using default tty will be retrieved.", - argument_default=argparse.SUPPRESS) - parser.add_argument("-t", "--tty", dest="port", default="/dev/ttyUSB0", help="Serial device to use") + description="Helios ventilation system commandline interface.", + epilog="Without arguments all readable values using default tty will be retrieved.", + argument_default=argparse.SUPPRESS + ) + parser.add_argument("-t", "--tty", dest="tty", default="/dev/ttyUSB0", help="Serial device to use") parser.add_argument("-r", "--read", dest="read_var", help="Read variables from ventilation system") parser.add_argument("-w", "--write", dest="write_var", help="Write variable to ventilation system") parser.add_argument("-v", "--value", dest="value", help="Value to write (required with option -v)") parser.add_argument("-d", "--debug", dest="enable_debug", action="store_true", help="Prints debug statements.") args = vars(parser.parse_args()) - + if "write_var" in args.keys() and "value" not in args.keys(): parser.print_usage() return + logger = logging.getLogger() logger.setLevel(logging.DEBUG) # old log version -# ch = logging.StreamHandler() -# if "enable_debug" in args.keys(): -# ch.setLevel(logging.DEBUG) -# else: -# ch.setLevel(logging.INFO) -# formatter = logging.Formatter("%(asctime)s - %(name)s - %(levelname)s - %(message)s") -# ch.setFormatter(formatter) -# logger.addHandler(ch) +# ch = logging.StreamHandler() +# if "enable_debug" in args.keys(): +# ch.setLevel(logging.DEBUG) +# else: +# ch.setLevel(logging.INFO) +# formatter = logging.Formatter("%(asctime)s - %(name)s - %(levelname)s - %(message)s") +# ch.setFormatter(formatter) +# logger.addHandler(ch) + helios = None try: - helios = HeliosBase(args["port"]) + helios = HeliosBase(tty=args["tty"]) helios.connect() if not helios._is_connected: raise Exception("Not connected") - + if "read_var" in args.keys(): - print("{0} = {1}".format(args["read_var"],helios.readValue(args["read_var"]))) + print("{0} = {1}".format(args["read_var"], helios.readValue(args["read_var"]))) elif "write_var" in args.keys(): helios.writeValue(args["write_var"],args["value"]) else: for var in CONST_MAP_VARIABLES_TO_ID.keys(): - print("{0} = {1}".format(var,helios.readValue(var))) + print("{0} = {1}".format(var, helios.readValue(var))) except Exception as e: print("Exception: {0}".format(e)) return 1 @@ -512,5 +513,6 @@ def main(): if helios: helios.disconnect() + if __name__ == "__main__": - sys.exit(main()) + sys.exit(main()) diff --git a/helios/plugin.yaml b/helios/plugin.yaml index 457c49c97..2a643b9ed 100755 --- a/helios/plugin.yaml +++ b/helios/plugin.yaml @@ -1,3 +1,6 @@ +%YAML 1.1 +--- + plugin: type: interface description: @@ -9,10 +12,10 @@ plugin: keywords: 'helios vallox ventilation' documentation: https://github.com/Tom-Bom-badil/helios/wiki support: https://knx-user-forum.de/forum/supportforen/smarthome-py/40092-erweiterung-helios-vallox-plugin - version: 1.4.2 - sh_minversion: 1.1 - multi_instance: False - restartable: unknown + version: 1.4.3 + sh_minversion: 1.6 + multi_instance: false + restartable: true classname: 'Helios' parameters: diff --git a/intercom_2n/__init__.py b/intercom_2n/__init__.py index 37fba4268..5956279c8 100755 --- a/intercom_2n/__init__.py +++ b/intercom_2n/__init__.py @@ -20,7 +20,6 @@ ######################################################################### import json -import logging import os from time import sleep import requests @@ -31,17 +30,21 @@ requests.packages.urllib3.disable_warnings(InsecureRequestWarning) + class Intercom2n(SmartPlugin): - PLUGIN_VERSION = "1.3.0.1" + PLUGIN_VERSION = "1.3.1" ALLOW_MULTIINSTANCE = False - def __init__(self, sh, intercom_ip, ssl=False, auth_type=0, username=None, password=None): - self._sh = sh + def __init__(self, **kwargs): + self._intercom_ip = self.get_parameter_value('intercom_ip') + self._ssl = self.get_parameter_value('ssl') + self._auth_type = self.get_parameter_value('auth_type') + self._username = self.get_parameter_value('username') + self._password = self.get_parameter_value('password') self.is_stopped = False self.sid = None - self._logger = logging.getLogger(__name__) self.event_timeout = 30 - self.ip_cam = IPCam(intercom_ip, ssl=ssl, auth_type=auth_type, user=username, password=password) + self.ip_cam = IPCam(self._intercom_ip, ssl=self._ssl, auth_type=self._auth_type, user=self._username, password=self._password) # item dictionaries for events self.possible_events = [ @@ -104,7 +107,7 @@ def get_events(self): def parse_event_data(self, raw_data): try: - raw_data =json.loads(raw_data) + raw_data = json.loads(raw_data) except Exception: self._logger.warning("Unknown 2n_event: '{event}' not in dictionary format.".format(event=raw_data)) return @@ -231,8 +234,8 @@ def update_item(self, item, caller=None, source=None, dest=None): elif command == 'firmware_upload': # check for child item firmware_filepath child_items = parent_item.return_children() - for child_item in child_items: - path = child_item._name.replace(parent_item._name,'').lstrip('.') + for child_item in child_items: + path = child_item._name.replace(parent_item._name, '').lstrip('.') if path == "firmware_file": if os.path.exists(child_item()): parent_item(self.ip_cam.commands.firmware_upload(child_item())) @@ -241,16 +244,16 @@ def update_item(self, item, caller=None, source=None, dest=None): elif command == 'config_get': # check for child item config_file child_items = parent_item.return_children() - for child_item in child_items: - path = child_item._name.replace(parent_item._name,'').lstrip('.') + for child_item in child_items: + path = child_item._name.replace(parent_item._name, '').lstrip('.') if path == "config_file": parent_item(self.ip_cam.commands.config_get(filename=child_item())) break elif command == 'config_upload': # check for child item firmware_filepath child_items = parent_item.return_children() - for child_item in child_items: - path = child_item._name.replace(parent_item._name,'').lstrip('.') + for child_item in child_items: + path = child_item._name.replace(parent_item._name, '').lstrip('.') if path == "config_file": if os.path.exists(child_item()): parent_item(self.ip_cam.commands.config_upload(child_item())) @@ -261,8 +264,8 @@ def update_item(self, item, caller=None, source=None, dest=None): elif command == 'switch_status': switch = None child_items = parent_item.return_children() - for child_item in child_items: - path = child_item._name.replace(parent_item._name,'').lstrip('.') + for child_item in child_items: + path = child_item._name.replace(parent_item._name, '').lstrip('.') if path == "switch": switch = child_item() break @@ -272,8 +275,8 @@ def update_item(self, item, caller=None, source=None, dest=None): action = None response = None child_items = parent_item.return_children() - for child_item in child_items: - path = child_item._name.replace(parent_item._name,'').lstrip('.') + for child_item in child_items: + path = child_item._name.replace(parent_item._name, '').lstrip('.') if path == "switch": switch = child_item() elif path == "action": @@ -286,8 +289,8 @@ def update_item(self, item, caller=None, source=None, dest=None): elif command == 'io_caps': port = None child_items = parent_item.return_children() - for child_item in child_items: - path = child_item._name.replace(parent_item._name,'').lstrip('.') + for child_item in child_items: + path = child_item._name.replace(parent_item._name, '').lstrip('.') if path == "port": port = child_item() break @@ -295,8 +298,8 @@ def update_item(self, item, caller=None, source=None, dest=None): elif command == 'io_status': port = None child_items = parent_item.return_children() - for child_item in child_items: - path = child_item._name.replace(parent_item._name,'').lstrip('.') + for child_item in child_items: + path = child_item._name.replace(parent_item._name, '').lstrip('.') if path == "port": port = child_item() break @@ -306,8 +309,8 @@ def update_item(self, item, caller=None, source=None, dest=None): action = None response = None child_items = parent_item.return_children() - for child_item in child_items: - path = child_item._name.replace(parent_item._name,'').lstrip('.') + for child_item in child_items: + path = child_item._name.replace(parent_item._name, '').lstrip('.') if path == "port": port = child_item() elif path == "action": @@ -320,50 +323,50 @@ def update_item(self, item, caller=None, source=None, dest=None): elif command == 'phone_status': account = None child_items = parent_item.return_children() - for child_item in child_items: - path = child_item._name.replace(parent_item._name,'').lstrip('.') + for child_item in child_items: + path = child_item._name.replace(parent_item._name, '').lstrip('.') if path == "account": account = child_item() break parent_item(self.ip_cam.commands.phone_status(account)) elif command == 'call_status': - session = None - child_items = parent_item.return_children() - for child_item in child_items: - path = child_item._name.replace(parent_item._name,'').lstrip('.') - if path == "session": - session = child_item() - break - parent_item(self.ip_cam.commands.call_status(session)) + session = None + child_items = parent_item.return_children() + for child_item in child_items: + path = child_item._name.replace(parent_item._name, '').lstrip('.') + if path == "session": + session = child_item() + break + parent_item(self.ip_cam.commands.call_status(session)) elif command == 'call_dial': - number = None - child_items = parent_item.return_children() - for child_item in child_items: - path = child_item._name.replace(parent_item._name,'').lstrip('.') - if path == "number": - number = child_item() - break - parent_item(self.ip_cam.commands.call_dial(number)) + number = None + child_items = parent_item.return_children() + for child_item in child_items: + path = child_item._name.replace(parent_item._name, '').lstrip('.') + if path == "number": + number = child_item() + break + parent_item(self.ip_cam.commands.call_dial(number)) elif command == 'call_answer': - session = None - child_items = parent_item.return_children() - for child_item in child_items: - path = child_item._name.replace(parent_item._name,'').lstrip('.') - if path == "session": - session = child_item() - break - parent_item(self.ip_cam.commands.call_answer(session)) + session = None + child_items = parent_item.return_children() + for child_item in child_items: + path = child_item._name.replace(parent_item._name, '').lstrip('.') + if path == "session": + session = child_item() + break + parent_item(self.ip_cam.commands.call_answer(session)) elif command == 'call_hangup': - session = None - reason = None - child_items = parent_item.return_children() - for child_item in child_items: - path = child_item._name.replace(parent_item._name,'').lstrip('.') - if path == "session": - session = child_item() - if path == "reason": - reason = child_item() - parent_item(self.ip_cam.commands.call_hangup(session, reason)) + session = None + reason = None + child_items = parent_item.return_children() + for child_item in child_items: + path = child_item._name.replace(parent_item._name, '').lstrip('.') + if path == "session": + session = child_item() + if path == "reason": + reason = child_item() + parent_item(self.ip_cam.commands.call_hangup(session, reason)) elif command == 'camera_caps': parent_item(self.ip_cam.commands.camera_caps()) elif command == 'camera_snapshot': @@ -373,8 +376,8 @@ def update_item(self, item, caller=None, source=None, dest=None): source = None time = None child_items = parent_item.return_children() - for child_item in child_items: - path = child_item._name.replace(parent_item._name,'').lstrip('.') + for child_item in child_items: + path = child_item._name.replace(parent_item._name, '').lstrip('.') if path == "snapshot_file": snapshot_file = child_item() if path == "width": @@ -384,7 +387,7 @@ def update_item(self, item, caller=None, source=None, dest=None): if path == "source": source = child_item() if path == "time": - time == child_item() + time = child_item() parent_item(self.ip_cam.commands.camera_snapshot(width, height, snapshot_file, source, time)) elif command == 'display_caps': parent_item(self.ip_cam.commands.display_caps()) @@ -392,22 +395,22 @@ def update_item(self, item, caller=None, source=None, dest=None): gif_file = None display = None child_items = parent_item.return_children() - for child_item in child_items: - path = child_item._name.replace(parent_item._name,'').lstrip('.') + for child_item in child_items: + path = child_item._name.replace(parent_item._name, '').lstrip('.') if path == "gif_file": gif_file = child_item() if path == "display": display = child_item() parent_item(self.ip_cam.commands.display_upload_image(display, gif_file)) elif command == 'display_delete_image': - display = None - child_items = parent_item.return_children() - for child_item in child_items: - path = child_item._name.replace(parent_item._name,'').lstrip('.') - if path == "display": - display = child_item() - break - parent_item(self.ip_cam.commands.display_delete_image(display)) + display = None + child_items = parent_item.return_children() + for child_item in child_items: + path = child_item._name.replace(parent_item._name, '').lstrip('.') + if path == "display": + display = child_item() + break + parent_item(self.ip_cam.commands.display_delete_image(display)) elif command == 'log_caps': parent_item(self.ip_cam.commands.log_caps()) elif command == 'audio_test': @@ -421,8 +424,8 @@ def update_item(self, item, caller=None, source=None, dest=None): picture_count = None timespan = None child_items = parent_item.return_children() - for child_item in child_items: - path = child_item._name.replace(parent_item._name,'').lstrip('.') + for child_item in child_items: + path = child_item._name.replace(parent_item._name, '').lstrip('.') if path == "to": to = child_item() if path == "width": @@ -441,8 +444,8 @@ def update_item(self, item, caller=None, source=None, dest=None): elif command == 'pcap': pcap_file = None child_items = parent_item.return_children() - for child_item in child_items: - path = child_item._name.replace(parent_item._name,'').lstrip('.') + for child_item in child_items: + path = child_item._name.replace(parent_item._name, '').lstrip('.') if path == "pcap_file": pcap_file = child_item() break diff --git a/intercom_2n/plugin.yaml b/intercom_2n/plugin.yaml index 469911e88..996c54c90 100755 --- a/intercom_2n/plugin.yaml +++ b/intercom_2n/plugin.yaml @@ -12,27 +12,28 @@ plugin: # documentation: https://github.com/smarthomeNG/smarthome/wiki/CLI-Plugin # url of documentation (wiki) page support: https://knx-user-forum.de/forum/supportforen/smarthome-py/1030539-plugin-2n-intercom - version: 1.3.0.1 # Plugin version - sh_minversion: 1.3 # minimum shNG version to use this plugin + version: 1.3.1 # Plugin version + sh_minversion: 1.6 # minimum shNG version to use this plugin # sh_maxversion: # maximum shNG version to use this plugin (leave empty if latest) - multi_instance: False # plugin supports multi instance - restartable: unknown + multi_instance: false # plugin supports multi instance + restartable: false classname: Intercom2n # class containing the plugin parameters: # Definition of parameters to be configured in etc/plugin.yaml -# def __init__(self, sh, intercom_ip, ssl=False, auth_type=0, username=None, password=None): intercom_ip: type: ip description: de: "Intercom IP Adresse" en: "IP address of Intercom" + ssl: type: bool default: False description: de: "ssl Verschlüsselung verwenden" en: "Use ssl encryption" + auth_type: type: int default: 0 @@ -41,12 +42,14 @@ parameters: description: de: "Zu verwendender auth_type (0: no authentication, 1: Basic Authentication, 2: Digest Authentication)" en: "auth_type to use (0: no authentication, 1: Basic Authentication, 2: Digest Authentication)" + username: type: str default: None* description: de: "Benutzername, falls auth_type > 0" en: "Usernam, falls auth_type > 0" + password: type: str default: None* diff --git a/jvcproj/__init__.py b/jvcproj/__init__.py index f7f3d5268..649cea506 100755 --- a/jvcproj/__init__.py +++ b/jvcproj/__init__.py @@ -97,10 +97,9 @@ class JVC_DILA_Control(SmartPlugin): the update functions for the items """ ALLOW_MULTIINSTANCE = False - PLUGIN_VERSION='1.0.1' + PLUGIN_VERSION = '1.0.2' - - def __init__(self, smarthome, host='0.0.0.0', gammaconf_dir='/usr/local/smarthome/etc/jvcproj/'): + def __init__(self, **kwargs): """ Initalizes the plugin. The parameters describe for this method are pulled from the entry in plugin.conf. :param host: JVC DILA Projectors IP address @@ -108,10 +107,9 @@ def __init__(self, smarthome, host='0.0.0.0', gammaconf_dir='/usr/local/smarthom :port is fixed to 20554 """ super().__init__() - self.logger = logging.getLogger(__name__) - self._sh=smarthome - self.host_port = (host, 20554) - self.gammaconf_dir = gammaconf_dir + self.host = self.get_parameter_value('host') + self.gammaconf_dir = self.get_parameter_value('gammaconf_dir') + self.host_port = (self.host, 20554) self.logger.debug("Plugin '{}': configured for host: '{}'".format(self.get_fullname(), self.host_port)) def run(self): @@ -164,7 +162,7 @@ def update_item(self, item, caller=None, source=None, dest=None): :param source: if given it represents the source :param dest: if given it represents the dest """ - if item(): + if item() and self.alive: if self.has_iattr(item.conf, 'jvcproj_cmd'): if self.get_iattr_value(item.conf, 'jvcproj_cmd') == 'None': self.logger.debug("Plugin '{}': no command given for update_item '{}'. Please check jvcproj_cmd!" @@ -184,17 +182,16 @@ def update_item(self, item, caller=None, source=None, dest=None): .format(self.get_fullname(), item, caller, source, dest)) self.check_gamma_cmd(item) - def check_gamma_cmd(self, item): """check gamma options to import new gammatable""" self.logger.debug("Plugin '{}': checking for gamma.conf an correct gamma input (must a custom gammatable) in '{}' : '{}'." .format(self.get_fullname(), item, self.get_iattr_value(item.conf, 'jvcproj_gamma'))) - _checklist = (self.get_iattr_value(item.conf, 'jvcproj_gamma').replace(' ','')).split('|') + _checklist = (self.get_iattr_value(item.conf, 'jvcproj_gamma').replace(' ', '')).split('|') if len(_checklist) != 2: self.logger.debug("Plugin '{}': ERROR! Item:'{}': exactly two arguments (file and custom gamma table) must be given!" .format(self.get_fullname(), item)) return - _cmdlist=[] + _cmdlist = [] if self.gammaconf_dir[-1] != '/': self.gammaconf_dir = self.gammaconf_dir + '/' if os_path.isfile(self.gammaconf_dir + _checklist[0]) is False: @@ -243,7 +240,7 @@ def load_table(self, data): def check_gammadata(self, table): """Check gamma data from file""" gammadata = table - if gammadata is None : + if gammadata is None: return None if len(gammadata) == 256: gammadata = [gammadata, gammadata, gammadata] @@ -268,12 +265,11 @@ def le16_split(self, colortable): yield val % 256 yield int(val / 256) - def check_cmd(self, item): """create command list and execute low level string validation for each command""" self.logger.debug("Plugin '{}': create commandlist for item '{}' : '{}' and check command(s)." .format(self.get_fullname(), item, self.get_iattr_value(item.conf, 'jvcproj_cmd'))) - _checklist = (self.get_iattr_value(item.conf, 'jvcproj_cmd').replace(' ','')).split('|') + _checklist = (self.get_iattr_value(item.conf, 'jvcproj_cmd').replace(' ', '')).split('|') _cmdlist = [] for _cmd in _checklist: if _cmd.upper()[2:6] == UNIT_ID and _cmd.upper()[-2:] == END: @@ -297,7 +293,7 @@ def handleconn_op(self, cmdlist): """handle connection and sending commands for jvcproj_cmd""" self.connect() for cmd in cmdlist: - if cmd[:2] == REQ: ##maybe in the future?? + if cmd[:2] == REQ: # maybe in the future?? self.logger.debug("Plugin '{}': WARNING! A request is not yet supported!" .format(self.get_fullname())) elif cmd[:2] == OPE: @@ -308,7 +304,6 @@ def handleconn_op(self, cmdlist): .format(self.get_fullname(), cmd)) self.disconnect('finished! Now disconnecting!') - def connect(self): """Open network connection to projector and perform handshake""" self.socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) @@ -330,13 +325,13 @@ def set(self, cmd): self.send(binascii.a2b_hex(cmd)) self.expect(binascii.a2b_hex(ACK + cmd[2:10] + END)) except Timeout: - self.disconnect('ERROR! Command not acknowledged! Aborting!') - raise CommandNack('Command not acknowledged', cmd) + self.disconnect('ERROR! Command not acknowledged! Aborting!') + raise CommandNack('Command not acknowledged', cmd) def get(self): pass - def disconnect(self, message ='disconnecting...'): + def disconnect(self, message='disconnecting...'): """Close socket""" self.logger.debug("Plugin '{}': {}" .format(self.get_fullname(), message)) @@ -344,6 +339,8 @@ def disconnect(self, message ='disconnecting...'): def send(self, data): """Send data with optional""" + if not self.alive: + return try: self.socket.send(data) except ConnectionAbortedError as err: diff --git a/jvcproj/plugin.yaml b/jvcproj/plugin.yaml index b584191e5..9305f9815 100755 --- a/jvcproj/plugin.yaml +++ b/jvcproj/plugin.yaml @@ -10,11 +10,11 @@ plugin: state: ready support: https://knx-user-forum.de/forum/supportforen/smarthome-py/1188479-plugin-steuerung-von-jvc-d-ila-projektoren - version: 1.0.1 # Plugin version - sh_minversion: 1.3 # minimum shNG version to use this plugin + version: 1.0.2 # Plugin version + sh_minversion: 1.6 # minimum shNG version to use this plugin # sh_maxversion: # maximum shNG version to use this plugin (leave empty if latest) - multi_instance: False # plugin supports multi instance - restartable: unknown + multi_instance: false # plugin supports multi instance + restartable: true classname: JVC_DILA_Control # class containing the plugin parameters: diff --git a/kostal/__init__.py b/kostal/__init__.py index b2d00cfcc..ffc2ab3bf 100755 --- a/kostal/__init__.py +++ b/kostal/__init__.py @@ -21,13 +21,13 @@ # ######################################################################### -import logging from lib.model.smartplugin import SmartPlugin -from lib.utils import Utils -import urllib.request, json +import urllib.request +import json import time import re + class Kostal(SmartPlugin): """ Since UI-version 6 the inverter can answere requests with json. @@ -37,36 +37,36 @@ class Kostal(SmartPlugin): """ ALLOW_MULTIINSTANCE = True - PLUGIN_VERSION = "1.3.2" + PLUGIN_VERSION = "1.3.3" _key2json = { - 'operation_status' : 16780032, - 'dctot_w' : 33556736, - 'dc1_v' : 33555202, - 'dc1_a' : 33555201, - 'dc1_w' : 33555203, - 'dc2_v' : 33555458, - 'dc2_a' : 33555457, - 'dc2_w' : 33555459, - 'dc3_v' : 33555714, - 'dc3_a' : 33555713, - 'dc3_w' : 33555715, - 'actot_w' : 67109120, - 'actot_Hz' : 67110400, - 'actot_cos' : 67110656, - 'actot_limitation' : 67110144, - 'ac1_v' : 67109378, - 'ac1_a' : 67109377, - 'ac1_w' : 67109379, - 'ac2_v' : 67109634, - 'ac2_a' : 67109633, - 'ac2_w' : 67109635, - 'ac3_v' : 67109890, - 'ac3_a' : 67109889, - 'ac3_w' : 67109891, - 'yield_day_kwh' : 251658754, - 'yield_tot_kwh' : 251658753, - 'operationtime_h' : 251658496 + 'operation_status': 16780032, + 'dctot_w': 33556736, + 'dc1_v': 33555202, + 'dc1_a': 33555201, + 'dc1_w': 33555203, + 'dc2_v': 33555458, + 'dc2_a': 33555457, + 'dc2_w': 33555459, + 'dc3_v': 33555714, + 'dc3_a': 33555713, + 'dc3_w': 33555715, + 'actot_w': 67109120, + 'actot_Hz': 67110400, + 'actot_cos': 67110656, + 'actot_limitation': 67110144, + 'ac1_v': 67109378, + 'ac1_a': 67109377, + 'ac1_w': 67109379, + 'ac2_v': 67109634, + 'ac2_a': 67109633, + 'ac2_w': 67109635, + 'ac3_v': 67109890, + 'ac3_a': 67109889, + 'ac3_w': 67109891, + 'yield_day_kwh': 251658754, + 'yield_tot_kwh': 251658753, + 'operationtime_h': 251658496 } _key2td = { 'actot_w': 9, @@ -105,25 +105,19 @@ class Kostal(SmartPlugin): 'l3_watt': 'ac3_w' } - def __init__(self, sh, ip, user="pvserver", passwd="pvwr",cycle=300, datastructure="html"): - self._sh = sh - self.logger = logging.getLogger(__name__) + def __init__(self, **kwargs): + self.ip = self.get_parameter_value('ip') + self.user = self.get_parameter_value('user') + self.passwd = self.get_parameter_value('passwd') + self.cycle = self.get_parameter_value('cycle') + self.datastructure_param = self.get_parameter_value('datastructure') self.logger.info('Init Kostal plugin') - self.user = user - self.passwd = passwd - self.cycle = int(cycle) self._items = {} - if Utils.is_ip(ip): - self.ip = ip - else: - self.logger.error(str(ip) + " is not a valid IP") - if datastructure == "html": + if self.datastructure_param == "html": self._keytable = self._key2td - #self.datastructure = "html" self.datastructure = self._html else: self._keytable = self._key2json - #self.datastructure = "json" self.datastructure = self._json def run(self): @@ -132,13 +126,14 @@ def run(self): """ self.logger.debug("run method Kostal called") self.alive = True - self._sh.scheduler.add('Kostal', self._refresh, cycle=self.cycle) + self.scheduler_add('Kostal', self._refresh, cycle=self.cycle) def stop(self): """ Stop method for the plugin """ self.logger.debug("stop method Kostal called") + self.scheduler_remove('Kostal') self.alive = False def parse_item(self, item): @@ -155,21 +150,8 @@ def parse_item(self, item): self._items[setting] = item return self.update_item - def parse_logic(self, logic): - pass - - def update_item(self, item, caller=None, source=None, dest=None): - """ - Write items values - :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 - """ - pass - def _html(self): - #HTML-OLD-Coding + # HTML-OLD-Coding try: data = self._sh.tools.fetch_url( 'http://' + self.ip + '/', self.user, self.passwd, timeout=2).decode() @@ -189,18 +171,18 @@ def _html(self): return def _json(self): - #NEW-JSON-Coding + # NEW-JSON-Coding try: # generate url; fetching only needed elements kostalurl = 'http://' + self.ip + '/api/dxs.json?sessionid=SmartHomeNG' for item in self._items: value = self._keytable[item] - kostalurl +='&dxsEntries=' + str(value) + kostalurl += '&dxsEntries=' + str(value) with urllib.request.urlopen(kostalurl) as url: data = json.loads(url.read().decode()) for values in data['dxsEntries']: kostal_key = str(list(self._keytable.keys())[list(self._keytable.values()).index(values['dxsId'])]) - value=values['value'] + value = values['value'] if kostal_key == "operation_status": self.logger.debug("operation_status" + str(value)) if str(value) == "0": @@ -214,16 +196,18 @@ def _json(self): else: value = "unknown" if kostal_key == "yield_day_kwh": - value = value / 1000 + value = float(value) / 1000 if kostal_key in self._items: self._items[kostal_key](value) - self.logger.debug("items[" + str(kostal_key) +"] = " +str(value)) + self.logger.debug("items[" + str(kostal_key) + "] = " + str(value)) except Exception as e: self.logger.error( 'could not retrieve data from {0}: {1}'.format(self.ip, e)) return def _refresh(self): + if not self.alive: + return start = time.time() # run the working methods self.datastructure() diff --git a/kostal/plugin.yaml b/kostal/plugin.yaml index 24d43000e..4e0d5ea70 100755 --- a/kostal/plugin.yaml +++ b/kostal/plugin.yaml @@ -27,11 +27,11 @@ plugin: # documentation: https://github.com/smarthomeNG/smarthome/wiki/CLI-Plugin # url of documentation (wiki) page support: https://knx-user-forum.de/forum/supportforen/smarthome-py/1109697-kostal-plugin-piko-wechselrichter - version: 1.3.2 # Plugin version - sh_minversion: 1.3 # minimum shNG version to use this plugin + version: 1.3.3 # Plugin version + sh_minversion: 1.6 # minimum shNG version to use this plugin # sh_maxversion: # maximum shNG version to use this plugin (leave empty if latest) - multi_instance: True # plugin supports multi instance - restartable: unknown + multi_instance: true # plugin supports multi instance + restartable: true classname: Kostal # class containing the plugin parameters: @@ -41,7 +41,7 @@ parameters: ip: type: ip - mandatory: True + mandatory: true description: de: "IP Adresse des Konverters" en: "IP address of converter" diff --git a/luxtronic2/__init__.py b/luxtronic2/__init__.py index 13cf7b5e5..1d0dee433 100755 --- a/luxtronic2/__init__.py +++ b/luxtronic2/__init__.py @@ -30,6 +30,25 @@ from lib.model.smartplugin import SmartPlugin +MODES = { + 0: 'Heizbetrieb', + 1: 'Keine Anforderung', + 2: 'Netz- Einschaltverzoegerung', + 3: 'SSP Zeit', + 4: 'Sperrzeit', + 5: 'Brauchwasser', + 6: 'Estrich Programm', + 7: 'Abtauen', + 8: 'Pumpenvorlauf', + 9: 'Thermische Desinfektion', + 10: 'Kuehlbetrieb', + 12: 'Schwimmbad', + 13: 'Heizen Ext.', + 14: 'Brauchwasser Ext.', + 16: 'Durchflussueberwachung', + 17: 'ZWE Betrieb' +} + class luxex(Exception): pass @@ -39,7 +58,7 @@ class LuxBase(SmartPlugin): # ATTENTION: This is NOT the SmartPlugin class of the plugin!!! - def __init__(self, host, port=8888): + def __init__(self, host, port=8888, **kwargs): self.logger = logging.getLogger(__name__) self.host = host self.port = int(port) @@ -233,7 +252,7 @@ def refresh_calculated(self): class Luxtronic2(LuxBase): ALLOW_MULTIINSTANCE = False - PLUGIN_VERSION = '1.3.2' + PLUGIN_VERSION = '1.3.3' _parameter = {} _attribute = {} @@ -241,21 +260,22 @@ class Luxtronic2(LuxBase): _decoded = {} alive = True - def __init__(self, smarthome, host, port=8888, cycle=300): - LuxBase.__init__(self, host, port) - self._sh = smarthome - self._cycle = int(cycle) + def __init__(self, **kwargs): + self._is_connected = False + self._cycle = self.get_parameter_value('cycle') + LuxBase.__init__(self, self.get_parameter_value('host'), self.get_parameter_value('port')) self.connect() def run(self): self.alive = True - self._sh.scheduler.add('Luxtronic2', self._refresh, cycle=self._cycle) + self.scheduler_add('Luxtronic2', self._refresh, cycle=self._cycle) def stop(self): self.alive = False + self.scheduler_remove('Luxtronic2') def _refresh(self): - if not self.is_connected: + if not self.is_connected or not self.alive: return start = time.time() if len(self._parameter) > 0: @@ -285,54 +305,8 @@ def _refresh(self): def _decode(self, identifier, value): if identifier == 119: - if value == 0: - return 'Heizbetrieb' - if value == 1: - return 'Keine Anforderung' - if value == 2: - return 'Netz- Einschaltverzoegerung' - if value == 3: - return 'SSP Zeit' - if value == 4: - return 'Sperrzeit' - if value == 5: - return 'Brauchwasser' - if value == 6: - return 'Estrich Programm' - if value == 7: - return 'Abtauen' - if value == 8: - return 'Pumpenvorlauf' - if value == 9: - return 'Thermische Desinfektion' - if value == 10: - return 'Kuehlbetrieb' - if value == 12: - return 'Schwimmbad' - if value == 13: - return 'Heizen Ext.' - if value == 14: - return 'Brauchwasser Ext.' - if value == 16: - return 'Durchflussueberwachung' - if value == 17: - return 'ZWE Betrieb' - return '???' - if identifier == 10: - return float(value) / 10 - if identifier == 11: - return float(value) / 10 - if identifier == 12: - return float(value) / 10 - if identifier == 15: - return float(value) / 10 - if identifier == 19: - return float(value) / 10 - if identifier == 20: - return float(value) / 10 - if identifier == 151: - return float(value) / 10 - if identifier == 152: + return MODES.get(value, '???') + if identifier in (10, 11, 12, 15, 19, 20, 151, 152): return float(value) / 10 return value @@ -356,11 +330,12 @@ def parse_item(self, item): return self.update_item def update_item(self, item, caller=None, source=None, dest=None): - if caller != 'Luxtronic2': + if caller != 'Luxtronic2' and self.alive: self.set_param(self.get_iattr_value(item.conf, 'lux2_p'), item()) def main(): + lux = None try: lux = LuxBase('192.168.178.25') lux.connect() @@ -389,5 +364,6 @@ def main(): if lux: lux.close() + if __name__ == "__main__": sys.exit(main()) diff --git a/luxtronic2/plugin.yaml b/luxtronic2/plugin.yaml index 1a4349c3d..e98af2d62 100755 --- a/luxtronic2/plugin.yaml +++ b/luxtronic2/plugin.yaml @@ -11,11 +11,11 @@ plugin: # documentation: https://github.com/smarthomeNG/plugins/blob/develop/mqtt/README.md # url of documentation (wiki) page # support: https://knx-user-forum.de/forum/supportforen/smarthome-py - version: 1.3.2 # Plugin version + version: 1.3.3 # Plugin version sh_minversion: 1.3 # minimum shNG version to use this plugin # sh_maxversion: # maximum shNG version to use this plugin (leave empty if latest) - multi_instance: False - restartable: unknown + multi_instance: false + restartable: true classname: Luxtronic2 # class containing the plugin parameters: @@ -32,6 +32,13 @@ parameters: de: 'Gibt den Port des Gerätes an' en: 'Specifies the port of the devices' + cycle: + type: int + default: 300 + description: + de: 'Zeitintervall zur Datenabfrage' + en: 'Interval for retrieving data' + item_attributes: # Definition of item attributes defined by this plugin lux2: diff --git a/plex/__init__.py b/plex/__init__.py index 37bce2f96..b21159f38 100755 --- a/plex/__init__.py +++ b/plex/__init__.py @@ -19,7 +19,6 @@ # along with SmartHomeNG. If not, see . ######################################################################### -import logging import json import requests import random @@ -27,41 +26,40 @@ class Plex(SmartPlugin): - PLUGIN_VERSION = "1.0.0" + PLUGIN_VERSION = "1.0.1" ALLOW_MULTIINSTANCE = False - def __init__(self, smarthome, displaytime=6000): - self.logger = logging.getLogger(__name__) + def __init__(self, **kwargs): + self._displayTime = self.get_parameter_value('displaytime') self.logger.info("Init Plex notifications") - self._sh = smarthome - self._displayTime = int(displaytime) self._images = ["info", "error", "warning"] self._clients = [] def run(self): - pass + self.alive = True def stop(self): - pass + self.alive = False def _push(self, host, data): - try: - res = requests.post(host, - headers={ - "User-Agent": "sh.py", - "Content-Type": "application/json"}, - timeout=4, - data=json.dumps(data), - ) - self.logger.debug(res) - response = res.text - del res - self.logger.debug(response) - except Exception as e: - self.logger.exception(e) + if self.alive: + try: + res = requests.post(host, + headers={ + "User-Agent": "sh.py", + "Content-Type": "application/json"}, + timeout=4, + data=json.dumps(data), + ) + self.logger.debug(res) + response = res.text + del res + self.logger.debug(response) + except Exception as e: + self.logger.exception(e) def notify(self, title, message, image="info"): - if not image in self._images: + if image not in self._images: self.logger.warn("Plex image must be: {}".format(", ".join(self._images))) else: data = {"jsonrpc": "2.0", diff --git a/plex/plugin.yaml b/plex/plugin.yaml index 7882ce0c8..302dd0894 100755 --- a/plex/plugin.yaml +++ b/plex/plugin.yaml @@ -12,11 +12,11 @@ plugin: # documentation: https://github.com/smarthomeNG/smarthome/wiki/CLI-Plugin # url of documentation (wiki) page # support: https://knx-user-forum.de/forum/supportforen/smarthome-py - version: 1.0.0 # Plugin version - sh_minversion: 1.1 # minimum shNG version to use this plugin + version: 1.0.1 # Plugin version + sh_minversion: 1.6 # minimum shNG version to use this plugin # sh_maxversion: # maximum shNG version to use this plugin (leave empty if latest) - multi_instance: False # plugin supports multi instance - restartable: unknown + multi_instance: false # plugin supports multi instance + restartable: true classname: Plex # class containing the plugin parameters: diff --git a/rcswitch/__init__.py b/rcswitch/__init__.py index 0226ecc3c..8446f8303 100755 --- a/rcswitch/__init__.py +++ b/rcswitch/__init__.py @@ -1,7 +1,7 @@ #!/usr/bin/env python3 # vim: set encoding=utf-8 tabstop=4 softtabstop=4 shiftwidth=4 expandtab ######################################################################### -# Copyright 2017 Daniel Frank knx-user-forum.de:dafra +# Copyright 2017 Daniel Frank knx-user-forum.de:dafra ######################################################################### # This file is part of SmartHomeNG. https://github.com/smarthomeNG// # @@ -19,9 +19,7 @@ # along with SmartHomeNG. If not, see . ######################################################################### -import logging from lib.model.smartplugin import SmartPlugin -from datetime import datetime, timedelta from socket import gethostname import time import threading @@ -30,105 +28,104 @@ from subprocess import DEVNULL import shlex + class RCswitch(SmartPlugin): - ALLOW_MULTIINSTANCE = False - PLUGIN_VERSION = "1.2.2" + ALLOW_MULTIINSTANCE = False + PLUGIN_VERSION = "1.2.2" + + def __init__(self, **kwargs): + self.setupOK = True + self.mapping = {'a': 1, 'A': 1, 'b': 2, 'B': 2, 'c': 3, 'C': 3, 'd': 4, 'D': 4, 'e': 5, 'E': 5} + self.rcswitch_dir = self.get_parameter_value('rcswitch_dir') + self.rcswitch_sendDuration = self.get_parameter_value('rcswitch_sendDuration') + self.rcswitch_host = self.get_parameter_value('rcswitch_host') + self.rcswitch_user = self.get_parameter_value('rcswitch_user') + self.rcswitch_password = self.get_parameter_value('rcswitch_password') + + # format path: cut possible '/' at end of rcswitch_dir parameter + if self.rcswitch_dir.endswith('/'): + self.rcswitch_dir = self.rcswitch_dir[:-1] + + # Handle host, check if anything is defined in self.rcswitch_host parameter and if is valid hostname or valid IPv4 adress + if self.rcswitch_host: + # then check if user defined its own local host -> error + if ((self.rcswitch_host == gethostname()) or (self.rcswitch_host == '127.0.0.1')): + self.logger.error('RCswitch: rcswitch_host is defined as your own machine, not the remote address! Please check the parameter rcswitch_host, >>{}<< it seems to be not correct!'.format(self.rcswitch_host)) + + # check connection to remote host and accept fingerprint + user = None + try: + # following line shall raise an error in case connection is not possible. + user = subprocess.check_output(shlex.split('sshpass -p {} ssh -o StrictHostKeyChecking=no {}@{} grep {} /etc/passwd'.format(self.rcswitch_password, self.rcswitch_user, self.rcswitch_host, self.rcswitch_user)), stderr=DEVNULL).decode('utf8')[:len(self.rcswitch_user)] + # check if rc switch is installed at the specified path on remote host + self.fileStat = subprocess.check_output(shlex.split('sshpass -p {} ssh {}@{} stat -c %a {}'.format(self.rcswitch_password, self.rcswitch_user, self.rcswitch_host, self.rcswitch_dir)), stderr=DEVNULL).decode('utf8') + self.rcswitch_dir = 'sshpass -p {} ssh {}@{} {}'.format(self.rcswitch_password, self.rcswitch_user, self.rcswitch_host, self.rcswitch_dir) + self.logger.info('RCswitch: Using {} as host.'.format(self.rcswitch_host)) + except subprocess.CalledProcessError as e: + self.setupOK = False + # give user hint where the problem is located + try: + if (user == self.rcswitch_user): + self.logger.error('RCswitch: send file of RCswitch not found at {} on {}. Check if RCswitch is installed correctly at specifed path on {}. System returned: {}'.format(self.rcswitch_dir, self.rcswitch_host, self.rcswitch_host, e)) + else: + self.logger.error('RCswitch: send file of RCswitch not found at {} on {}. Additional problem with user authentication possible. System returned: {}'.format(self.rcswitch_dir, self.rcswitch_host, e)) + except UnboundLocalError as e: + self.logger.error('RCswitch: Cannot connect to {}. Check rcswitch_host, rcswitch_user and rcswitch_password are set (correctly). Ensure SSH server is running on {}. System returned: {}'.format(self.rcswitch_host, self.rcswitch_host, e)) + else: + # check if rc switch is installed at the specified path + if not os.path.isfile('{}/send'.format(self.rcswitch_dir)): + self.logger.error('RCswitch: send file of RCswitch not found at {} on localhost. Check path, if RCswitch is installed correctly on target, and correct format of v4 IP adress (in case rcswitch_host is defined).'.format(self.rcswitch_dir)) + self.setupOK = False + else: + self.logger.info('RCswitch: setup on localhost OK') - def __init__(self, smarthome, rcswitch_dir='/usr/local/bin/rcswitch-pi', rcswitch_sendDuration='0.5', rcswitch_host=None, rcswitch_user=None, rcswitch_password=None): - self.logger = logging.getLogger(__name__) - self.setupOK = True - self.mapping = {'a':1,'A':1,'b':2,'B':2,'c':3,'C':3,'d':4,'D':4,'e':5,'E':5} - self._sh = smarthome - - # format path: cut possible '/' at end of rcswitch_dir parameter - if rcswitch_dir[len(rcswitch_dir)-1] == '/': - self.rcswitch_dir = rcswitch_dir[0:len(rcswitch_dir)-1] - else: - self.rcswitch_dir = rcswitch_dir + # setup semaphore + self.lock = threading.Lock() - # Check optional Parameters: check sendDuration - try: - self.sendDuration = float(rcswitch_sendDuration) - except Exception as e: - self.sendDuration = float(0.5) - self.logger.warning('RCswitch: Argument {} for rcswitch_sendDuration is not a valid number. Using default value instead.'.format(rcswitch_sendDuration)) - - # Handle host, check if anything is defined in rcswitch_host parameter and if is valid hostname or valid IPv4 adress - if (rcswitch_host and (self.is_hostname(rcswitch_host) or self.is_ipv4(rcswitch_host))): - # then check if user defined its own local host -> error - if ((rcswitch_host == gethostname()) or (rcswitch_host == '127.0.0.1')): - self.logger.error('RCswitch: rcswitch_host is defined as your own machine, not the remote address! Please check the parameter rcswitch_host, >>{}<< it seems to be not correct!'.format(rcswitch_host)) + # don't load plugin if init didn't work + if not self.setupOK: + self._init_complete = False - #check connection to remote host and accept fingerprint - try: - # following line shall raise an error in case connection is not possible. - self.user = subprocess.check_output(shlex.split('sshpass -p {} ssh -o StrictHostKeyChecking=no {}@{} grep {} /etc/passwd'.format(rcswitch_password, rcswitch_user, rcswitch_host, rcswitch_user)), stderr=DEVNULL).decode('utf8')[0:len(rcswitch_user)] - # check if rc switch is installed at the specified path on remote host - self.fileStat = subprocess.check_output(shlex.split('sshpass -p {} ssh {}@{} stat -c %a {}'.format(rcswitch_password, rcswitch_user, rcswitch_host, self.rcswitch_dir)), stderr=DEVNULL).decode('utf8') - self.rcswitch_dir = ('sshpass -p {} ssh {}@{} {}'.format(rcswitch_password, rcswitch_user, rcswitch_host, self.rcswitch_dir)) - self.logger.info('RCswitch: Using {} as host.'.format(rcswitch_host)) - except subprocess.CalledProcessError as e: - self.setupOK = False - # give user hint where the problem is located - try: - if (user == rcswitch_user): - self.logger.error('RCswitch: send file of RCswitch not found at {} on {}. Check if RCswitch is installed correctly at specifed path on {}. System returned: {}'.format(self.rcswitch_dir, rcswitch_host, rcswitch_host, e)) - else: - self.logger.error('RCswitch: send file of RCswitch not found at {} on {}. Additional problem with user authentication possible. System returned: {}'.format(self.rcswitch_dir, rcswitch_host, e)) - except UnboundLocalError as e: - self.logger.error('RCswitch: Cannot connect to {}. Check rcswitch_host, rcswitch_user and rcswitch_password are set (correctly). Ensure SSH server is running on {}. System returned: {}'.format(rcswitch_host, rcswitch_host, e)) - else: - # check if rc switch is installed at the specified path - if not os.path.isfile('{}/send'.format(self.rcswitch_dir)): - self.logger.error('RCswitch: send file of RCswitch not found at {} on localhost. Check path, if RCswitch is installed correctly on target, and correct format of v4 IP adress (in case rcswitch_host is defined).'.format(self.rcswitch_dir)) - self.setupOK = False - else: - self.logger.info('RCswitch: setup on localhost OK') - - # setup semaphore - self.lock = threading.Lock() - - def run(self): - self.alive = True + def run(self): + self.alive = True - def stop(self): - self.alive = False + def stop(self): + self.alive = False - def parse_item(self, item): - # generate warnings for incomplete configured itemns - if self.has_iattr(item.conf, 'rc_device'): - if self.has_iattr(item.conf, 'rc_code'): - return self.update_item - else: - self.logger.warning('RC Switch: attribute rc_code for {} missing. Item will be ignored by RCswitch plugin'.format(item)) - return None - elif self.has_iattr(item.conf, 'rc_code'): - self.logger.warning('RC Switch: attribute rc_device for {} missing. Item will be ignored by RCswitch plugin'.format(item)) - return None - else: - return None + def parse_item(self, item): + # generate warnings for incomplete configured itemns + if self.has_iattr(item.conf, 'rc_device'): + if self.has_iattr(item.conf, 'rc_code'): + return self.update_item + else: + self.logger.warning('RC Switch: attribute rc_code for {} missing. Item will be ignored by RCswitch plugin'.format(item)) + return None + elif self.has_iattr(item.conf, 'rc_code'): + self.logger.warning('RC Switch: attribute rc_device for {} missing. Item will be ignored by RCswitch plugin'.format(item)) + return None + else: + return None + def update_item(self, item, caller=None, source=None, dest=None): + # send commands to devices + if self.has_iattr(item.conf, 'rc_code') and self.has_iattr(item.conf, 'rc_device') and self.alive: + # prepare parameters + value = item() + rcCode = self.get_iattr_value(item.conf, 'rc_code') + rcDevice = self.get_iattr_value(item.conf, 'rc_device') - def update_item(self, item, caller=None, source=None, dest=None): - # send commands to devices - if self.has_iattr(item.conf, 'rc_code') and self.has_iattr(item.conf, 'rc_device') and self.setupOK: #if 'rc_device' in item.conf and 'rc_code' in item.conf and self.setupOK: - # prepare parameters - value = item() - rcCode = self.get_iattr_value(item.conf, 'rc_code') - rcDevice = self.get_iattr_value(item.conf, 'rc_device') - - # avoid parallel access by use of semaphore - self.lock.acquire() - # sending commands - if(rcDevice in self.mapping):#handling of device encoded with a,A,b,... - subprocess.call(shlex.split('{}/send {} {} {}'.format(self.rcswitch_dir, rcCode, self.mapping[rcDevice], int(value))), stdout=DEVNULL, stderr=DEVNULL) - self.logger.info('RC Switch: setting device {} with system code {} to {}'.format(rcDevice, rcCode, value)) - else: - try:#handling of devices encoded with 1,2,3 - subprocess.call(shlex.split('{}/send {} {} {}'.format(self.rcswitch_dir, rcCode, int(rcDevice), int(value))), stdout=DEVNULL, stderr=DEVNULL) - self.logger.info('RC Switch: setting device {} with system code {} to {}'.format(rcDevice, rcCode, value)) - except Exception as e:#invalid encoding of device - self.logger.error('RC Switch: requesting invalid device {} with system code {} '.format(rcDevice, rcCode)) - time.sleep(min(self.sendDuration,10))# give the transmitter time to complete sending of the command (but not more than 10s) - self.lock.release() \ No newline at end of file + # avoid parallel access by use of semaphore + self.lock.acquire() + # sending commands + if (rcDevice in self.mapping): # handling of device encoded with a,A,b,... + subprocess.call(shlex.split('{}/send {} {} {}'.format(self.rcswitch_dir, rcCode, self.mapping[rcDevice], int(value))), stdout=DEVNULL, stderr=DEVNULL) + self.logger.info('RC Switch: setting device {} with system code {} to {}'.format(rcDevice, rcCode, value)) + else: + try: # handling of devices encoded with 1,2,3 + subprocess.call(shlex.split('{}/send {} {} {}'.format(self.rcswitch_dir, rcCode, int(rcDevice), int(value))), stdout=DEVNULL, stderr=DEVNULL) + self.logger.info('RC Switch: setting device {} with system code {} to {}'.format(rcDevice, rcCode, value)) + except Exception: # invalid encoding of device + self.logger.error('RC Switch: requesting invalid device {} with system code {} '.format(rcDevice, rcCode)) + time.sleep(min(self.rcswitch_sendDuration, 10)) # give the transmitter time to complete sending of the command (but not more than 10s) + self.lock.release() diff --git a/rcswitch/plugin.yaml b/rcswitch/plugin.yaml index 65fe7b22b..42b301887 100755 --- a/rcswitch/plugin.yaml +++ b/rcswitch/plugin.yaml @@ -13,10 +13,10 @@ plugin: support: https://knx-user-forum.de/forum/supportforen/smarthome-py/39094-logic-und-howto-für-433mhz-steckdosen version: 1.2.2 # Plugin version - sh_minversion: 1.2 # minimum shNG version to use this plugin + sh_minversion: 1.6 # minimum shNG version to use this plugin # sh_maxversion: # maximum shNG version to use this plugin (leave empty if latest) - multi_instance: False # plugin supports multi instance - restartable: unknown + multi_instance: false # plugin supports multi instance + restartable: true classname: RCswitch # class containing the plugin parameters: diff --git a/roomba_980/__init__.py b/roomba_980/__init__.py index ac2ab99e3..b43738a7f 100755 --- a/roomba_980/__init__.py +++ b/roomba_980/__init__.py @@ -19,23 +19,23 @@ # along with SmartHome.py. If not, see . ######################################################################### -import logging +# TODO: das Modul ist im Sourcetree nicht vorhanden... woher soll das kommen? from plugins.roomba_980.roomba import Roomba from lib.model.smartplugin import SmartPlugin -from lib.item import Items + class ROOMBA_980(SmartPlugin): ALLOW_MULTIINSTANCE = False - PLUGIN_VERSION = "1.0.1" + PLUGIN_VERSION = "1.0.2" myroomba = None - def __init__(self, sh, adress=None, blid=None, roombaPassword=None, cycle=900): - self._address = adress - self._blid = blid - self._roombaPassword = roombaPassword - self._cycle = cycle + def __init__(self, **kwargs): + self._address = self.get_parameter_value('adress') + self._blid = self.get_parameter_value('blid') + self._roombaPassword = self.get_parameter_value('roombaPassword') + self._cycle = self.get_parameter_value('cycle') self._status_batterie = None self._status_items = {} @@ -61,7 +61,7 @@ def run(self): self.alive = True def stop(self): - self.scheduler.remove('get_status') + self.scheduler_remove('get_status') self.myroomba.disconnect() self.alive = False @@ -69,36 +69,35 @@ def __call__(self): pass def update_item(self, item, caller=None, source=None, dest=None): - if caller != __name__: + if caller != __name__ and self.alive: self.logger.debug('item_update {} '.format(item)) if self.get_iattr_value(item.conf, 'roomba_980') == "start": - if item() == True: - self.send_command("start") + if item() is True: + self.send_command("start") elif self.get_iattr_value(item.conf, 'roomba_980') == "stop": - if item() == True: - self.send_command("stop") + if item() is True: + self.send_command("stop") elif self.get_iattr_value(item.conf, 'roomba_980') == "dock": - if item() == True: - self.send_command("dock") + if item() is True: + self.send_command("dock") def get_status(self): status = self.myroomba.master_state for status_item in self._status_items: - if status_item == "status_batterie": - self._status_items[status_item](status['state']['reported']['batPct'],__name__) - elif status_item == "status_bin_full": - self._status_items[status_item](status['state']['reported']['bin']['full'],__name__) - elif status_item == "status_cleanMissionStatus_phase": - self._status_items[status_item](status['state']['reported']['cleanMissionStatus']['phase'],__name__) - elif status_item == "status_cleanMissionStatus_error": - self._status_items[status_item](status['state']['reported']['cleanMissionStatus']['error'],__name__) + if status_item == "status_batterie": + self._status_items[status_item](status['state']['reported']['batPct'], __name__) + elif status_item == "status_bin_full": + self._status_items[status_item](status['state']['reported']['bin']['full'], __name__) + elif status_item == "status_cleanMissionStatus_phase": + self._status_items[status_item](status['state']['reported']['cleanMissionStatus']['phase'], __name__) + elif status_item == "status_cleanMissionStatus_error": + self._status_items[status_item](status['state']['reported']['cleanMissionStatus']['error'], __name__) self.logger.debug('Status update') def send_command(self, command): - if self.myroomba != None: - self.myroomba.send_command(command) - self.logger.debug('send command: {} to Roomba'.format(command)) - + if self.myroomba is not None: + self.myroomba.send_command(command) + self.logger.debug('send command: {} to Roomba'.format(command)) diff --git a/roomba_980/plugin.yaml b/roomba_980/plugin.yaml index 8b9ea3e8e..1da819cd6 100755 --- a/roomba_980/plugin.yaml +++ b/roomba_980/plugin.yaml @@ -7,17 +7,17 @@ plugin: en: 'integration of the iRobot Roomba vacuum cleaner series 900' maintainer: 'Zapfen83' tester: '?' - state: ready + state: development keywords: irobot roomba # keywords, where applicable # documentation: https://github.com/smarthomeNG/plugins/blob/develop/mqtt/README.md # url of documentation (wiki) page # support: https://knx-user-forum.de/forum/supportforen/smarthome-py # Following entries are for Smart-Plugins: - version: 1.0.1 # Plugin version - sh_minversion: 1.5 # minimum shNG version to use this plugin + version: 1.0.2 # Plugin version + sh_minversion: 1.6 # minimum shNG version to use this plugin # sh_maxversion: # maximum shNG version to use this plugin (leave empty if latest) - multi_instance: False - restartable: unknown + multi_instance: false + restartable: true classname: ROOMBA_980 # class containing the plugin parameters: @@ -25,28 +25,28 @@ parameters: adress: type: str - default: True + default: true description: de: "Die IP Adresse des roomba Staubsaugers" en: "The IP address of the roomba vacuum cleaner" blid: type: str - default: True + default: true description: de: "Die blid des roomba Staubsaugers -> kann mit der getpassword.py ausgelesen werden" en: "The blid of the roomba vacuum cleaner -> use getpassword.py to get it" roombaPassword: type: str - default: True + default: true description: de: "Das Passwort des roomba Staubsaugers -> kann mit der getpassword.py ausgelesen werden" en: "The password of the roomba vacuum cleaner -> use getpassword.py to get it" cycle: type: num - default: False + default: false description: de: "update des items alle x Sekunden, default wert 900" en: "update the state item every x secounds, default is 900" diff --git a/sma/__init__.py b/sma/__init__.py index 2a453c865..7b6cd32d6 100755 --- a/sma/__init__.py +++ b/sma/__init__.py @@ -11,38 +11,36 @@ # # SMA-Plugin for SmartHomeNG. https://github.com/smarthomeNG// # -# License: Attribution-NonCommercial-ShareAlike 3.0 Unported (CC BY-NC-SA 3.0) -# http://creativecommons.org/licenses/by-nc-sa/3.0/ +# License: Attribution-NonCommercial-ShareAlike 3.0 Unported (CC BY-NC-SA 3.0) +# http://creativecommons.org/licenses/by-nc-sa/3.0/ # -# You are free: -# to Share — to copy, distribute and transmit the work -# to Remix — to adapt the work -# Under the following conditions: -# Attribution: -# You must attribute the work in the manner specified by the author or licensor -# (but not in any way that suggests that they endorse you or your use of the work). -# Noncommercial: -# You may not use this work for commercial purposes. -# Share Alike: -# If you alter, transform, or build upon this work, you may distribute the resulting work -# only under the same or similar license to this one. +# You are free: +# to Share — to copy, distribute and transmit the work +# to Remix — to adapt the work +# Under the following conditions: +# Attribution: +# You must attribute the work in the manner specified by the author or licensor +# (but not in any way that suggests that they endorse you or your use of the work). +# Noncommercial: +# You may not use this work for commercial purposes. +# Share Alike: +# If you alter, transform, or build upon this work, you may distribute the resulting work +# only under the same or similar license to this one. # # DISCLAIMER: -# A user of this plugin acknowledges that he or she is receiving this -# software on an "as is" basis and the user is not relying on the accuracy -# or functionality of the software for any purpose. The user further -# acknowledges that any use of this software will be at his own risk -# and the copyright owner accepts no responsibility whatsoever arising from -# the use or application of the software. +# A user of this plugin acknowledges that he or she is receiving this +# software on an "as is" basis and the user is not relying on the accuracy +# or functionality of the software for any purpose. The user further +# acknowledges that any use of this software will be at his own risk +# and the copyright owner accepts no responsibility whatsoever arising from +# the use or application of the software. ######################################################################### -import logging import threading import time import socket from datetime import datetime from dateutil import tz -import itertools from lib.model.smartplugin import SmartPlugin @@ -185,19 +183,18 @@ class SMA(SmartPlugin): ALLOW_MULTIINSTANCE = False - PLUGIN_VERSION = "1.3.1" - - def __init__(self, smarthome, bt_addr, password="0000", update_cycle="60", allowed_timedelta="10"): - self.logger = logging.getLogger(__name__) - self._sh = smarthome - self._update_cycle = int(update_cycle) + PLUGIN_VERSION = "1.3.2" + + def __init__(self, **kwargs): + # TODO: self._own_bt_addr setzen + self._inv_bt_addr = self.get_parameter_value('bt_addr') + self._inv_password = self.get_parameter_value('password') + self._update_cycle = self.get_parameter_value('update_cycle') + self._allowed_timedelta = self.get_parameter_value('allowed_timedelta') self._fields = {} self._requests = [] self._cmd_lock = threading.Lock() self._reply_lock = threading.Condition() - self._inv_bt_addr = bt_addr - self._inv_password = password - self._allowed_timedelta = int(allowed_timedelta) self._inv_last_read_timestamp_utc = 0 self._inv_serial = 0 self._own_bt_addr_le = bytearray(BCAST_ADDR) @@ -209,7 +206,7 @@ def __init__(self, smarthome, bt_addr, password="0000", update_cycle="60", allow raise Exception("Python socket module does not support Bluetooth - see README.md how to install") def _update_values(self): - #logger.warning("sma: signal strength = {}%%".format(self._inv_get_bt_signal_strength())) + # logger.warning("sma: signal strength = {}%%".format(self._inv_get_bt_signal_strength())) self._cmd_lock.acquire() try: for request in self._requests: @@ -224,7 +221,7 @@ def _update_values(self): self._reply_lock.release() if ('LAST_UPDATE' in self._fields) and not (self._inv_last_read_timestamp_utc == 0): self._inv_last_read_datetime = datetime.fromtimestamp(self._inv_last_read_timestamp_utc, tz.tzlocal()) - #self._inv_last_read_str = self._inv_last_read_datetime.strftime("%d.%m.%Y %H:%M:%S") + # self._inv_last_read_str = self._inv_last_read_datetime.strftime("%d.%m.%Y %H:%M:%S") self._inv_last_read_str = self._inv_last_read_datetime.strftime("%d.%m. %H:%M ") for item in self._fields['LAST_UPDATE']['items']: item(self._inv_last_read_str, 'SMA', self._inv_serial) @@ -241,7 +238,7 @@ def run(self): self._plugin_active = self._plugin_active_item() # "or self._is_connected" ensures the connection will be closed before terminating while self.alive or self._is_connected: - #self.logger.warning("sma: state self._is_connected = {} / self._plugin_active = {} / self.alive = {}".format(self._is_connected, self._plugin_active, self.alive)) + # self.logger.warning("sma: state self._is_connected = {} / self._plugin_active = {} / self.alive = {}".format(self._is_connected, self._plugin_active, self.alive)) # connect to inverter if active but not connected if self._plugin_active and not self._is_connected: @@ -318,7 +315,7 @@ def run(self): if not self.alive: break if msg is None: - #self.logger.debug("sma: no msg...") + # self.logger.debug("sma: no msg...") continue if len(msg) >= 60: i = 41 @@ -331,16 +328,16 @@ def run(self): if lri not in lris: self.logger.info("sma: unknown lri={:#06x} / cls={:#02x} / dataType={:#02x} - trying to continue".format(lri, cls, dataType)) if (dataType == 0x00) or (dataType == 0x40): - i += 28 + i += 28 elif (dataType == 0x08) or (dataType == 0x10): - i += 40 + i += 40 else: self.logger.error("sma: rx - unknown datatype {:#02x}".format(dataType)) raise continue else: timestamp_utc = int.from_bytes(msg[i + 4:i + 8], byteorder='little') - value = eval(lri_evals[lris[lri][0]], dict(msg=msg,i=i,attribute_to_text=attribute_to_text)) + value = eval(lri_evals[lris[lri][0]], dict(msg=msg, i=i, attribute_to_text=attribute_to_text)) i += lris[lri][1] self.logger.debug("sma: lri={:#06x} / cls={:#02x} / timestamp={} / value={}".format(lri, cls, timestamp_utc, value)) if full_id in self._fields: @@ -477,6 +474,7 @@ def _recv_smanet2_msg(self, no_timeout_warning=False): # remove escape characters i = 0 while True: + # TODO: if this works - fine, seems not to be standard Python 3? if smanet2_msg[i] == 0x7d: smanet2_msg[i + 1] ^= 0x20 del(smanet2_msg[i]) @@ -493,6 +491,7 @@ def _recv_smanet2_msg(self, no_timeout_warning=False): def _recv_smanet1_msg_with_cmdcode(self, cmdcodes_expected=[0x0001]): retries = 3 + msg = None while self.alive: retries -= 1 if retries == 0: @@ -522,7 +521,7 @@ def _send_msg(self, msg): # set length fields msg[1:3] = len(msg).to_bytes(2, byteorder='little') msg[3] = msg[1] ^ msg[2] ^ 0x7e - #print("tx: len={} / data=[{}]".format(len(msg), ' '.join(['0x%02x' % b for b in msg]))) + # print("tx: len={} / data=[{}]".format(len(msg), ' '.join(['0x%02x' % b for b in msg]))) self._btsocket.send(msg) def _calc_crc16(self, msg): @@ -530,7 +529,7 @@ def _calc_crc16(self, msg): for i in msg: crc = (crc >> 8) ^ FCSTAB[(crc ^ i) & 0xFF] crc ^= 0xFFFF - #print("crc16 = {:x}".format(crc)) + # print("crc16 = {:x}".format(crc)) return crc def _inv_connect(self): @@ -653,7 +652,7 @@ def _inv_send_request(self, request_set): msg += SMANET2_HDR + bytes([0x09, 0xA0]) + BCAST_ADDR + bytes([0x00, 0x00]) + self._inv_bt_addr_le + bytes([0x00] + [0x00] + [0, 0, 0, 0]) + (self._send_count | 0x8000).to_bytes(2, byteorder='little') msg += request_set[0].to_bytes(4, byteorder='little') + request_set[1].to_bytes(4, byteorder='little') + request_set[2].to_bytes(4, byteorder='little') # send msg to inverter - #self.logger.debug("sma: requesting {:#06x}-{:#06x}...".format(request_set[1], request_set[2])) + # self.logger.debug("sma: requesting {:#06x}-{:#06x}...".format(request_set[1], request_set[2])) self._send_msg(msg) def _inv_set_time(self): @@ -666,7 +665,7 @@ def _inv_set_time(self): msg += SMANET2_HDR + bytes([0x10, 0xA0]) + BCAST_ADDR + bytes([0x00, 0x00]) + self._inv_bt_addr_le + bytes([0x00] + [0x00] + [0, 0, 0, 0]) + (self._send_count | 0x8000).to_bytes(2, byteorder='little') msg += int(0xF000020A).to_bytes(4, byteorder='little') + int(0x00236D00).to_bytes(4, byteorder='little') + int(0x00236D00).to_bytes(4, byteorder='little') + int(0x00236D00).to_bytes(4, byteorder='little') local_time = int(time.time()).to_bytes(4, byteorder='little') - msg += local_time + local_time + local_time + round((datetime.now()-datetime.utcnow()).total_seconds()).to_bytes(4, byteorder='little') + local_time + bytes([0x01, 0x00, 0x00, 0x00]) + msg += local_time + local_time + local_time + round((datetime.now() - datetime.utcnow()).total_seconds()).to_bytes(4, byteorder='little') + local_time + bytes([0x01, 0x00, 0x00, 0x00]) # msg += local_time + local_time + local_time + time.localtime().tm_gmtoff.to_bytes(4, byteorder='little') + local_time + bytes([0x01, 0x00, 0x00, 0x00]) # send msg to inverter - self._send_msg(msg) \ No newline at end of file + self._send_msg(msg) diff --git a/sma/plugin.yaml b/sma/plugin.yaml index 3bc6bc912..f18e18f74 100755 --- a/sma/plugin.yaml +++ b/sma/plugin.yaml @@ -12,18 +12,18 @@ plugin: documentation: https://smarthomeng.de/user/plugins_doc/config/sma.html # url of documentation (wiki) page support: https://knx-user-forum.de/forum/supportforen/smarthome-py/27997-beitrag-plugin-zum-lesen-von-sma-wechselrichtern-sunnyboy-5000tl-21-getestet - version: 1.3.1 # Plugin version - sh_minversion: 1.3 # minimum shNG version to use this plugin + version: 1.3.2 # Plugin version + sh_minversion: 1.6 # minimum shNG version to use this plugin # sh_maxversion: # maximum shNG version to use this plugin (leave empty if latest) - multi_instance: False # plugin supports multi instance - restartable: unknown + multi_instance: false # plugin supports multi instance + restartable: true classname: SMA # class containing the plugin parameters: # Definition of parameters to be configured in etc/plugin.yaml bt_addr: type: str - mandatory: True + mandatory: true description: de: 'Bluetooth-Adresse des SMA Wechselrichters (mit "hcitool scan" herauszufinden).' en: 'Bluetooth address of SMA inverter (find out with "hcitool scan").' @@ -31,14 +31,14 @@ parameters: cycle: type: int default: 60 - mandatory: False + mandatory: false description: de: '(optional) Dieser Parameter muss normalerweise nicht angegeben werden. Er erlaubt es die Update-Frquenz anzupassen (Standard: alle 60 Sekunden).' en: "(optional) This parameter usually doesn't have to be specified. It allows to change the update frequency (cycle every 60 seconds)." password: type: str - mandatory: False + mandatory: false description: de: '(optional) Passwort des Wechselrichters im User-Mode. Default: 0000.' en: '(optional) Password of the inverter in user mode. Default: 0000.' @@ -46,7 +46,7 @@ parameters: allowed_timedelta: type: int default: 10 - mandatory: False + mandatory: false description: de: '(optional) Erlaubter Zeitunterschied zwischen Systemzeit und Zeit des Wechselrichters. Falls der Inverter auf Systemzeit steht, kann die Prüfung mit -1 deaktiviert werden. Default: 10.' en: '(optional) Allowed difference of inverter to system time - if above, inverter is set to system time - set to -1 to disable. Default: 10.' @@ -55,7 +55,7 @@ item_attributes: # Definition of item attributes defined by this plugin sma: type: str - mandatory: True + mandatory: true description: de: 'Auszulesender Wert, möglich sind: PLUGIN_ACTIVE, AC_P_TOTAL, E_DAY, E_TOTAL, INV_SERIAL, INV_ADDRESS, LAST_UPDATE, DC_STRING1_P, DC_STRING2_P, DC_STRING1_U, DC_STRING2_U, DC_STRING1_I, DC_STRING2_I, OPERATING_TIME, FEEDING_TIME, GRID_FREQUENCY, STATUS, GRID_RELAY, SW_VERSION.' en: 'Value to read. Possible: PLUGIN_ACTIVE, AC_P_TOTAL, E_DAY, E_TOTAL, INV_SERIAL, INV_ADDRESS, LAST_UPDATE, DC_STRING1_P, DC_STRING2_P, DC_STRING1_U, DC_STRING2_U, DC_STRING1_I, DC_STRING2_I, OPERATING_TIME, FEEDING_TIME, GRID_FREQUENCY, STATUS, GRID_RELAY, SW_VERSION.' diff --git a/smarttv/__init__.py b/smarttv/__init__.py index 07dde2fc5..8b3afd9c9 100755 --- a/smarttv/__init__.py +++ b/smarttv/__init__.py @@ -19,7 +19,6 @@ # along with SmartHomeNG. If not, see . # -import logging import socket import time import base64 @@ -27,21 +26,22 @@ from lib.model.smartplugin import SmartPlugin from uuid import getnode as getmac + class SmartTV(SmartPlugin): ALLOW_MULTIINSTANCE = True - PLUGIN_VERSION = "1.3.2" - - def __init__(self, smarthome, host, port=55000, tv_version='classic', delay=1): - self.logger = logging.getLogger(__name__) - self._sh = smarthome - self._host = host - self._port = int(port) - self._delay = delay - if tv_version not in ['samsung_m_series', 'classic']: + PLUGIN_VERSION = "1.3.3" + + def __init__(self, **kwargs): + self._tv_version = self.get_parameter_value('tv_version') + self._host = self.get_parameter_value('host') + self._port = self.get_parameter_value('port') + self._delay = self.get_parameter_value('delay') + if self._tv_version not in ['samsung_m_series', 'classic']: self.logger.error('No valid tv_version attribute specified to plugin') - self._tv_version = tv_version - self.logger.debug("Smart TV plugin for {0} SmartTV device initalized".format(tv_version)) + self._init_complete = False + else: + self.logger.debug("Smart TV plugin for {0} SmartTV device initalized".format(self._tv_version)) def push_samsung_m_series(self, key): """ @@ -68,7 +68,7 @@ def push_classic(self, key): self.logger.debug("Connected to {0}:{1}".format(self._host, self._port)) except Exception: self.logger.warning("Could not connect to %s:%s, to send key: %s." % - (self._host, self._port, key)) + (self._host, self._port, key)) return src = s.getsockname()[0] # ip of remote @@ -140,6 +140,8 @@ def parse_item(self, item): return None def update_item(self, item, caller=None, source=None, dest=None): + if not self.alive: + return val = item() if isinstance(val, str): if val.startswith('KEY_'): diff --git a/smarttv/plugin.yaml b/smarttv/plugin.yaml index ef43b482d..8127d8b16 100755 --- a/smarttv/plugin.yaml +++ b/smarttv/plugin.yaml @@ -12,25 +12,25 @@ plugin: # documentation: https://github.com/smarthomeNG/smarthome/wiki/CLI-Plugin # url of documentation (wiki) page # support: https://knx-user-forum.de/forum/supportforen/smarthome-py - version: 1.3.2 # Plugin version - sh_minversion: 1.3 # minimum shNG version to use this plugin + version: 1.3.3 # Plugin version + sh_minversion: 1.6 # minimum shNG version to use this plugin # sh_maxversion: # maximum shNG version to use this plugin (leave empty if latest) - multi_instance: True # plugin supports multi instance - restartable: unknown + multi_instance: true # plugin supports multi instance + restartable: true classname: SmartTV # class containing the plugin parameters: # Definition of parameters to be configured in etc/plugin.yaml host: type: str - mandatory: True + mandatory: true description: de: 'IP Adresse des SmartTV Geräts.' en: 'Specifies the ip address of your SmartTV device.' port: type: int - mandatory: False + mandatory: false default: 55000 description: de: '(optional) Im Fall eines Ports zu setzen, der nicht dem Standard entspricht (Standardports siehe tv_version). Default: 55000.' @@ -38,7 +38,7 @@ parameters: tv_version: type: str - mandatory: False + mandatory: false default: classic description: de: '(optional) Erlaubte Werte sind "classic" (Port 55000) oder "samsung_m_series" (Standardport: 8001). Default: "classic".' @@ -46,7 +46,7 @@ parameters: delay: type: int - mandatory: False + mandatory: false default: 1 description: de: '(optional) Verzögerung in Sekunden, falls mehr als eine Taste gesendet wird. Reduziert Probleme, wenn bspw. TV-Programme wie "135" aufgerufen werden. Default: 1' @@ -57,7 +57,7 @@ item_attributes: # Definition of item attributes defined by this plugin smarttv: type: str - mandatory: True + mandatory: true description: de: 'Es gibt zwei Wege das Attribut zu nutzen. * Auf einem Item des Typs `str` mit dem Wert "true": Jeder String der in das Item geschrieben wird, wird an den SmartTV gesendet. diff --git a/sml/__init__.py b/sml/__init__.py index aabb788d9..bf08ad475 100755 --- a/sml/__init__.py +++ b/sml/__init__.py @@ -20,7 +20,6 @@ # along with SmartHomeNG. If not, see . ######################################################################### -import logging import time import re import serial @@ -32,31 +31,32 @@ from lib.model.smartplugin import SmartPlugin + class Sml(SmartPlugin): ALLOW_MULTIINSTANCE = True - PLUGIN_VERSION = '1.0.0' + PLUGIN_VERSION = '1.0.1' _v1_start = b'\x1b\x1b\x1b\x1b\x01\x01\x01\x01' _v1_end = b'\x1b\x1b\x1b\x1b\x1a' _units = { # Blue book @ http://www.dlms.com/documentation/overviewexcerptsofthedlmsuacolouredbooks/index.html - 1 : 'a', 2 : 'mo', 3 : 'wk', 4 : 'd', 5 : 'h', 6 : 'min.', 7 : 's', 8 : '°', 9 : '°C', 10 : 'currency', - 11 : 'm', 12 : 'm/s', 13 : 'm³', 14 : 'm³', 15 : 'm³/h', 16 : 'm³/h', 17 : 'm³/d', 18 : 'm³/d', 19 : 'l', 20 : 'kg', - 21 : 'N', 22 : 'Nm', 23 : 'Pa', 24 : 'bar', 25 : 'J', 26 : 'J/h', 27 : 'W', 28 : 'VA', 29 : 'var', 30 : 'Wh', - 31 : 'WAh', 32 : 'varh', 33 : 'A', 34 : 'C', 35 : 'V', 36 : 'V/m', 37 : 'F', 38 : 'Ω', 39 : 'Ωm²/h', 40 : 'Wb', - 41 : 'T', 42 : 'A/m', 43 : 'H', 44 : 'Hz', 45 : 'Rac', 46 : 'Rre', 47 : 'Rap', 48 : 'V²h', 49 : 'A²h', 50 : 'kg/s', - 51 : 'Smho' + 1: 'a', 2: 'mo', 3: 'wk', 4: 'd', 5: 'h', 6: 'min.', 7: 's', 8: '°', 9: '°C', 10: 'currency', + 11: 'm', 12: 'm/s', 13: 'm³', 14: 'm³', 15: 'm³/h', 16: 'm³/h', 17: 'm³/d', 18: 'm³/d', 19: 'l', 20: 'kg', + 21: 'N', 22: 'Nm', 23: 'Pa', 24: 'bar', 25: 'J', 26: 'J/h', 27: 'W', 28: 'VA', 29: 'var', 30: 'Wh', + 31: 'WAh', 32: 'varh', 33: 'A', 34: 'C', 35: 'V', 36: 'V/m', 37: 'F', 38: 'Ω', 39: 'Ωm²/h', 40: 'Wb', + 41: 'T', 42: 'A/m', 43: 'H', 44: 'Hz', 45: 'Rac', 46: 'Rre', 47: 'Rap', 48: 'V²h', 49: 'A²h', 50: 'kg/s', + 51: 'Smho' } _devices = { - 'smart-meter-gateway-com-1' : 'hex' + 'smart-meter-gateway-com-1': 'hex' } - def __init__(self, smarthome, host=None, port=0, serialport=None, device="raw", cycle=300): - self._sh = smarthome - self.host = host - self.port = int(port) - self.serialport = serialport - self.cycle = cycle + def __init__(self, **kwargs): + self.host = self.get_parameter_value('host') + self.port = self.get_parameter_value('port') + self.serialport = self.get_parameter_value('serialport') + self.cycle = self.get_parameter_value('cycle') + self.device = self.get_parameter_value('device') self.connected = False self._serial = None self._sock = None @@ -64,25 +64,25 @@ def __init__(self, smarthome, host=None, port=0, serialport=None, device="raw", self._dataoffset = 0 self._items = {} self._lock = threading.Lock() - self.logger = logging.getLogger(__name__) - if device in self._devices: - device = self._devices[device] + if self._device in self._devices: + self._device = self._devices[self._device] - if device == "hex": + if self._device == "hex": self._prepare = self._prepareHex - elif device == "raw": + elif self._device == "raw": self._prepare = self._prepareRaw else: - self.logger.warning("Device type \"{}\" not supported - defaulting to \"raw\"".format(device)) + self.logger.warning("Device type \"{}\" not supported - defaulting to \"raw\"".format(self._device)) self._prepare = self._prepareRaw def run(self): self.alive = True - self._sh.scheduler.add('Sml', self._refresh, cycle=self.cycle) + self.scheduler_add('Sml', self._refresh, cycle=self.cycle) def stop(self): self.alive = False + self.scheduler_remove('Sml') self.disconnect() def parse_item(self, item): @@ -98,16 +98,8 @@ def parse_item(self, item): return self.update_item return None - def parse_logic(self, logic): - pass - - def update_item(self, item, caller=None, source=None, dest=None): - if caller != 'Sml': - pass - def connect(self): self._lock.acquire() - target = None try: if self.serialport is not None: self._target = 'serial://{}'.format(self.serialport) @@ -164,9 +156,9 @@ def _read(self, length): raise e return b''.join(total) - + def _refresh(self): - if self.connected: + if self.connected and self.alive: start = time.time() retry = 5 data = None @@ -183,8 +175,8 @@ def _refresh(self): if start_pos != -1 and end_pos == -1: data = data[:start_pos] elif start_pos != -1 and end_pos != -1: - chunk = data[start_pos:end_pos+len(self._v1_end)+3] - self.logger.debug('Found chunk at {} - {} ({} bytes):{}'.format(start_pos, end_pos, end_pos-start_pos, ''.join(' {:02x}'.format(x) for x in chunk))) + chunk = data[start_pos:end_pos + len(self._v1_end) + 3] + self.logger.debug('Found chunk at {} - {} ({} bytes):{}'.format(start_pos, end_pos, end_pos - start_pos, ''.join(' {:02x}'.format(x) for x in chunk))) chunk_crc_str = '{:02X}{:02X}'.format(chunk[-2], chunk[-1]) chunk_crc_calc = self._crc16(chunk[:-2]) chunk_crc_calc_str = '{:02X}{:02X}'.format((chunk_crc_calc >> 8) & 0xff, chunk_crc_calc & 0xff) @@ -236,27 +228,27 @@ def _parse(self, data): packetsize = 7 self.logger.debug('Data ({} bytes):{}'.format(len(data), ''.join(' {:02x}'.format(x) for x in data))) self._dataoffset = 0 - while self._dataoffset < builtins.len(data)-packetsize: + while self._dataoffset < builtins.len(data) - packetsize: # Find SML_ListEntry starting with 0x77 0x07 and OBIS code end with 0xFF - if data[self._dataoffset] == 0x77 and data[self._dataoffset+1] == 0x07 and data[self._dataoffset+packetsize] == 0xff: + if data[self._dataoffset] == 0x77 and data[self._dataoffset + 1] == 0x07 and data[self._dataoffset + packetsize] == 0xff: packetstart = self._dataoffset self._dataoffset += 1 try: entry = { - 'objName' : self._read_entity(data), - 'status' : self._read_entity(data), - 'valTime' : self._read_entity(data), - 'unit' : self._read_entity(data), - 'scaler' : self._read_entity(data), - 'value' : self._read_entity(data), - 'signature' : self._read_entity(data) + 'objName': self._read_entity(data), + 'status': self._read_entity(data), + 'valTime': self._read_entity(data), + 'unit': self._read_entity(data), + 'scaler': self._read_entity(data), + 'value': self._read_entity(data), + 'signature': self._read_entity(data) } # add additional calculated fields entry['obis'] = '{}-{}:{}.{}.{}*{}'.format(entry['objName'][0], entry['objName'][1], entry['objName'][2], entry['objName'][3], entry['objName'][4], entry['objName'][5]) entry['valueReal'] = entry['value'] * 10 ** entry['scaler'] if entry['scaler'] is not None else entry['value'] - entry['unitName'] = self._units[entry['unit']] if entry['unit'] != None and entry['unit'] in self._units else None + entry['unitName'] = self._units[entry['unit']] if entry['unit'] is not None and entry['unit'] in self._units else None values[entry['obis']] = entry except Exception as e: @@ -269,8 +261,8 @@ def _parse(self, data): def _read_entity(self, data): upack = { - 5 : { 1 : '>b', 2 : '>h', 4 : '>i', 8 : '>q' }, # int - 6 : { 1 : '>B', 2 : '>H', 4 : '>I', 8 : '>Q' } # uint + 5: {1: '>b', 2: '>h', 4: '>i', 8: '>q'}, # int + 6: {1: '>B', 2: '>H', 4: '>I', 8: '>Q'} # uint } result = None @@ -296,23 +288,23 @@ def _read_entity(self, data): self._parse_error('Tried to read {} bytes, but only have {}', [len, builtins.len(data) - self._dataoffset], data, self._dataoffset, packetstart) elif type == 0: # octet string - result = data[self._dataoffset:self._dataoffset+len] + result = data[self._dataoffset:self._dataoffset + len] elif type == 5 or type == 6: # int or uint - d = data[self._dataoffset:self._dataoffset+len] + d = data[self._dataoffset:self._dataoffset + len] ulen = len if ulen not in upack[type]: # extend to next greather unpack unit - while ulen not in upack[type]: - d = b'\x00' + d - ulen += 1 + while ulen not in upack[type]: + d = b'\x00' + d + ulen += 1 result = struct.unpack(upack[type][ulen], d)[0] elif type == 7: # list result = [] self._dataoffset += 1 - for i in range(0, len + 1): + for _ in range(0, len + 1): result.append(self._read_entity(data)) return result @@ -326,7 +318,7 @@ def _read_entity(self, data): def _parse_error(self, msg, msgargs, data, dataoffset, packetstart): position = dataoffset - packetstart databytes = '' - for i, b in enumerate(data[packetstart:packetstart+64]): + for i, b in enumerate(data[packetstart:packetstart + 64]): databytes = databytes + ' {}{:02x}{}'.format( '' if i != position - 1 else '<', b, @@ -348,24 +340,23 @@ def _prepareHex(self, data): data = re.sub("[^a-f0-9]", " ", data) data = re.sub("( +[a-f0-9]|[a-f0-9] +)", "", data) data = data.encode() - return bytes(''.join(chr(int(data[i:i+2], 16)) for i in range(0, len(data), 2)), "iso8859-1") + return bytes(''.join(chr(int(data[i:i + 2], 16)) for i in range(0, len(data), 2)), "iso8859-1") def _crc16(self, data): - crc = 0xffff - - p = 0; - while p < len(data): - c = 0xff & data[p] - p = p + 1 + crc = 0xffff - for i in range(0, 8): - if ((crc & 0x0001) ^ (c & 0x0001)): - crc = (crc >> 1) ^ 0x8408 - else: - crc = crc >> 1 - c = c >> 1 + p = 0 + while p < len(data): + c = 0xff & data[p] + p = p + 1 - crc = ~crc & 0xffff + for i in range(0, 8): + if ((crc & 0x0001) ^ (c & 0x0001)): + crc = (crc >> 1) ^ 0x8408 + else: + crc = crc >> 1 + c = c >> 1 - return ((crc << 8) | ((crc >> 8) & 0xff)) & 0xffff + crc = ~crc & 0xffff + return ((crc << 8) | ((crc >> 8) & 0xff)) & 0xffff diff --git a/sml/plugin.yaml b/sml/plugin.yaml index 91360c5b2..c031f5e10 100755 --- a/sml/plugin.yaml +++ b/sml/plugin.yaml @@ -12,11 +12,11 @@ plugin: documentation: http://smarthomeng.de/user/plugins_doc/config/sml.html # support: https://knx-user-forum.de/forum/supportforen/smarthome-py - version: 1.0.0 # Plugin version - sh_minversion: 1.1 # minimum shNG version to use this plugin + version: 1.0.1 # Plugin version + sh_minversion: 1.6 # minimum shNG version to use this plugin # sh_maxversion: # maximum shNG version to use this plugin (leave empty if latest) - multi_instance: True # plugin supports multi instance - restartable: unknown + multi_instance: true # plugin supports multi instance + restartable: true classname: Sml # class containing the plugin parameters: diff --git a/systemair/__init__.py b/systemair/__init__.py index dc467c6ee..df353ccea 100755 --- a/systemair/__init__.py +++ b/systemair/__init__.py @@ -18,50 +18,47 @@ # You should have received a copy of the GNU General Public License # along with this plugin. If not, see . ######################################################################### -import logging from time import sleep import minimalmodbus -from serial import SerialException import serial import threading from ctypes import c_short from lib.model.smartplugin import SmartPlugin + class Systemair(SmartPlugin): - PLUGIN_VERSION = "1.3.0.1" + PLUGIN_VERSION = "1.3.1" ALLOW_MULTIINSTANCE = False - def __init__(self, smarthome, serialport, slave_address="1", update_cycle="30"): - self._sh = smarthome - self.logger = logging.getLogger(__name__) + def __init__(self, **kwargs): self.instrument = None - self.slave_address = int(slave_address) + self.serialport = self.get_parameter_value('serialport') + self.slave_address = self.get_parameter_value('slave_address') + self._update_cycle = self.get_parameter_value('update_cycle') self._update_coil = {} - self.serialport = serialport - self.slave_address = slave_address minimalmodbus.TIMEOUT = 3 - minimalmodbus.CLOSE_PORT_AFTER_EACH_CALL=True - self._sh.scheduler.add(__name__, self._read_modbus, prio=5, cycle=int(update_cycle)) + minimalmodbus.CLOSE_PORT_AFTER_EACH_CALL = True + self.scheduler_add(__name__, self._read_modbus, prio=5, cycle=self._update_cycle) self.my_reg_items = [] self.mod_write_repeat = 20 # if port is already open, e.g on auto-update, # repeat mod_write attempt x times a 1 seconds self._lockmb = threading.Lock() # modbus serial port lock self.init_serial_connection(self.serialport, self.slave_address) - self._reg_sets = [{'name':'fan', 'range':range(101, 138+1)}, - {'name':'heater', 'range':range(201, 221+1), 'scaled_signed':range(208, 218+1)}, - {'name':'damper', 'range':range(301, 301+1)}, - {'name':'rotor', 'range':range(351, 352+1)}, - {'name':'week', 'range':range(401, 459+1)}, - {'name':'system', 'range':range(501, 507+1)}, - {'name':'clock', 'range':range(551, 557+1)}, - {'name':'filter', 'range':range(601, 602+1)}, - {'name':'VTC_defr', 'range':range(651, 654+1)}, - {'name':'VTR_defr', 'range':range(671, 672+1)}, - {'name':'dig_in', 'range':range(701, 709+1)}, - {'name':'PCU_PB', 'range':range(751, 751+1)}, - {'name':'alarms', 'range':range(801, 802+1)}, - {'name':'demand', 'range':range(851, 859+1)}, - {'name':'wireless', 'range':range(901, 1020+1)},] + self._reg_sets = [{'name': 'fan', 'range': range(101, 138 + 1)}, + {'name': 'heater', 'range': range(201, 221 + 1), 'scaled_signed': range(208, 218 + 1)}, + {'name': 'damper', 'range': range(301, 301 + 1)}, + {'name': 'rotor', 'range': range(351, 352 + 1)}, + {'name': 'week', 'range': range(401, 459 + 1)}, + {'name': 'system', 'range': range(501, 507 + 1)}, + {'name': 'clock', 'range': range(551, 557 + 1)}, + {'name': 'filter', 'range': range(601, 602 + 1)}, + {'name': 'VTC_defr', 'range': range(651, 654 + 1)}, + {'name': 'VTR_defr', 'range': range(671, 672 + 1)}, + {'name': 'dig_in', 'range': range(701, 709 + 1)}, + {'name': 'PCU_PB', 'range': range(751, 751 + 1)}, + {'name': 'alarms', 'range': range(801, 802 + 1)}, + {'name': 'demand', 'range': range(851, 859 + 1)}, + {'name': 'wireless', 'range': range(901, 1020 + 1)},] def init_serial_connection(self, serialport, slave_address): try: @@ -84,9 +81,9 @@ def _read_modbus(self): for reg_set in self._reg_sets: if 'range_used' in reg_set: read_regs = dict(zip(reg_set['range_used'], self.instrument.read_registers( - reg_set['range_used'].start -1, - reg_set['range_used'].stop - reg_set['range_used'].start, - functioncode = 3))) + reg_set['range_used'].start - 1, + reg_set['range_used'].stop - reg_set['range_used'].start, + functioncode=3))) if 'scaled_signed' in reg_set: for scaled_reg in reg_set['scaled_signed']: read_regs[scaled_reg] = c_short(read_regs[scaled_reg]).value / 10 @@ -100,7 +97,7 @@ def _read_modbus(self): # get coils for coil_addr in self._update_coil: - value = self.instrument.read_bit(coil_addr-1, functioncode=2) + value = self.instrument.read_bit(coil_addr - 1, functioncode=2) if value is not None: for item in self._update_coil[coil_addr]: item(value, 'systemair_value_from_bus', "Coil {}".format(coil_addr)) @@ -123,7 +120,7 @@ def connect(self): def update_item(self, item, caller=None, source=None, dest=None): # ignore values from bus - if caller == 'systemair_value_from_bus': + if caller == 'systemair_value_from_bus' or not self.alive: return if item in self.my_reg_items: if self.has_iattr(item.conf, 'mod_write'): @@ -138,9 +135,9 @@ def parse_item(self, item): for reg_set in self._reg_sets: if modbus_regaddr in reg_set['range']: - if not 'regs_used' in reg_set: + if 'regs_used' not in reg_set: reg_set['regs_used'] = dict() - if not modbus_regaddr in reg_set['regs_used']: + if modbus_regaddr not in reg_set['regs_used']: reg_set['regs_used'][modbus_regaddr] = set() reg_set['regs_used'][modbus_regaddr].add(item) @@ -155,7 +152,7 @@ def parse_item(self, item): if self.has_iattr(item.conf, 'systemair_coiladdr'): modbus_coiladdr = int(self.get_iattr_value(item.conf, 'systemair_coiladdr')) self.logger.debug("systemair_value_from_bus: {0} connected to coil register {1:#04x}".format(item, modbus_coiladdr)) - if not modbus_coiladdr in self._update_coil: + if modbus_coiladdr not in self._update_coil: self._update_coil[modbus_coiladdr] = set() self._update_coil[modbus_coiladdr].add(item) diff --git a/systemair/plugin.yaml b/systemair/plugin.yaml index b1979bc82..98e015e4d 100755 --- a/systemair/plugin.yaml +++ b/systemair/plugin.yaml @@ -12,11 +12,11 @@ plugin: # documentation: https://github.com/smarthomeNG/smarthome/wiki/CLI-Plugin # url of documentation (wiki) page support: https://knx-user-forum.de/forum/supportforen/smarthome-py/939623-systemair-modbus-plugin-zentrale-lüftungsanlage - version: 1.3.0.1 # Plugin version - sh_minversion: 1.3 # minimum shNG version to use this plugin + version: 1.3.1 # Plugin version + sh_minversion: 1.6 # minimum shNG version to use this plugin # sh_maxversion: # maximum shNG version to use this plugin (leave empty if latest) - multi_instance: False # plugin supports multi instance - restartable: unknown + multi_instance: false # plugin supports multi instance + restartable: true classname: Systemair # class containing the plugin parameters: