diff --git a/Bridge/README.md b/Bridge/README.md index c0631845..1a1588b1 100644 --- a/Bridge/README.md +++ b/Bridge/README.md @@ -1,7 +1,103 @@ -# IPS-Z2MBridge - Anbindung von www.zigbee2mqtt.io an IP-Symcon. - Modul für die Zigbee2MQTT Bridge +# Bridge + Modul für alle Systemweiten Funktionen von Zigbee2MQTT -Die Bridge wird aktuell NICHT benötigt. Also auch nicht anlegen! -Zukünftig wird es für die Bridge neue Funktionalitäten geben. Diese werden in der BEta ausgetestet und dann in die Master übertragen. \ No newline at end of file +## Inhaltverzeichnis +- [1. Gruppen in Z2M](#1-gruppen-in-z2m) +- [2. Konfiguration](#2-konfiguration) +- [3. Funktionen](#3-funktionen) + + | +## 3. Instanz-Funktionen + +```php +bool Z2M_InstallSymconExtension(int $InstanzID); +``` +Die aktuelle Symcon Erweiterung wird in Z2M installiert. + +-- + +```php +bool Z2M_SetLastSeen(int $InstanzID); +``` +Die Konfiguration der `last_seen` Einstellung in Z2M wird auf `epoch` verändert, damit die Instanzen in Symcon den Wert korrekt darstellen können. + +-- +```php +bool Z2M_SetPermitJoin(int $InstanzID, bool $PermitJoin); +``` + +-- +```php +bool Z2M_SetLogLevel(int $InstanzID, string $LogLevel); +``` + +-- +```php +bool Z2M_Restart(int $InstanzID); +``` + +-- +```php +bool Z2M_CreateGroup(int $InstanzID, string $GroupName); +``` + +-- +```php +bool Z2M_DeleteGroup(int $InstanzID, string $GroupName); +``` + +-- +```php +bool Z2M_RenameGroup(int $InstanzID, string $OldName, string $NewName); +``` + +-- +```php +bool Z2M_AddDeviceToGroup(int $InstanzID, string $GroupName, string $DeviceName); +``` + +-- +```php +bool Z2M_RemoveDeviceFromGroup(int $InstanzID, string $GroupName, string $DeviceName); +``` + +-- +```php +bool Z2M_RemoveAllDevicesFromGroup(int $InstanzID, string $GroupName); +``` + +-- +```php +bool Z2M_Bind(int $InstanzID, string $SourceDevice, string $TargetDevice); +``` + +-- +```php +bool Z2M_Unbind(int $InstanzID, string $SourceDevice, string $TargetDevice); +``` + +-- +```php +bool Z2M_RequestNetworkmap(int $InstanzID); +``` + +-- +```php +bool Z2M_RenameDevice(int $InstanzID, string $OldDeviceName, string $NewDeviceName); +``` + +-- +```php +bool Z2M_RemoveDevice(int $InstanzID, string $DeviceName); +``` + +-- +```php +bool Z2M_CheckOTAUpdate(int $InstanzID, string $DeviceName); +``` + +-- +```php +bool Z2M_PerformOTAUpdate(int $InstanzID, string $DeviceName); +``` diff --git a/Bridge/form.json b/Bridge/form.json index d8d7838e..b1a56d79 100644 --- a/Bridge/form.json +++ b/Bridge/form.json @@ -2,8 +2,50 @@ "elements": [ { "type": "ValidationTextBox", - "name": "MQTTTopic", - "caption": "MQTT Topic" + "name": "MQTTBaseTopic", + "caption": "MQTT Base Topic" + } + ], + "actions": [ + { + "type": "Button", + "label": "Install or upgrade Symcon-Extension", + "onClick": "Z2M_InstallSymconExtension($id);" + }, + { + "type": "Button", + "label": "Set last_seen setting to epoch", + "onClick": "Z2M_SetLastSeen($id);" + }, + { + "type": "TestCenter" + }, + { + "type": "Label", + "caption": "Spenden / Schenkung" + }, + { + "type": "Label", + "caption": "Dieses Modul ist für die nicht kommerzielle Nutzung kostenlos, Schenkungen als Unterstützung für den Autor werden hier akzeptiert: " + }, + { + "type": "RowLayout", + "items": [ + { + "type": "Image", + "onClick": "echo 'https://www.paypal.com/donate?hosted_button_id=EK4JRP87XLSHW';", + "image": "" + }, + { + "type": "Label", + "caption": "" + }, + { + "type": "Image", + "onClick": "echo 'https://www.amazon.de/hz/wishlist/ls/3JVWED9SZMDPK?ref_=wl_share';", + "image": "" + } + ] } ] } \ No newline at end of file diff --git a/Bridge/locale.json b/Bridge/locale.json index 2005ec11..776ef6d1 100644 --- a/Bridge/locale.json +++ b/Bridge/locale.json @@ -1,7 +1,25 @@ { "translations": { "de": { - "State": "Status" + "Install or upgrade Symcon-Extension": "Symcon Erweiterung installieren oder updaten", + "Symcon-Extension is up-to-date": "Symcon Erweiterung ist aktuell", + "Set last_seen setting to epoch": "Setze die last_seen Einstellung auf epoch", + "last_seen setting is correct": "last_seen Einstellung ist korrekt", + "Error": "Fehler", + "Warning": "Warnungen", + "State": "Status", + "Log Level": "Protokollierung", + "Network Map": "Netzwerkkarte", + "Allow joining the network": "Beitritt zum Netzwerk zulassen", + "Restart Required": "Neustart erforderlich", + "Perform a restart": "Neustart durchführen", + "Network Channel": "Netzwerkkanal", + "Extension Version": "Erweiterung Version", + "Extension is up to date": "Erweiterung ist aktuell", + "Wrong last_seen setting in Zigbee2MQTT. Please set last_seen to epoch.": "Falsche Einstellung für last_seen in Zigbee2MQTT. Bitte last_seen auf epoch einstellen.", + "Symcon Extension in Zigbee2MQTT is outdated. Please update the extension.": "Symcon Erweiterung in Zigbee2MQTT ist veraltet. Bitte Erweiterung updaten.", + "No Symcon Extension in Zigbee2MQTT installed. Please install the extension.": "Symcon Erweiterung in Zigebee2MQTT nicht installiert. Bitte die Erweiterung installieren.", + "Extension Loaded": "Erweiterung geladen" } } } \ No newline at end of file diff --git a/Bridge/module.php b/Bridge/module.php index e24caec4..55c7839c 100644 --- a/Bridge/module.php +++ b/Bridge/module.php @@ -1,131 +1,502 @@ '{043EA491-0325-4ADD-8FC2-A30C8EEB4D3F}', + 'PacketType' => 3, + 'QualityOfService' => 0, + 'Retain' => false, + 'Topic' => '', + 'Payload' => '' + ]; public function Create() { //Never delete this line! parent::Create(); $this->ConnectParent('{C6D2AEB3-6E1F-4B2E-8E69-3A1A00246850}'); - $this->RegisterPropertyString('MQTTTopic', 'bridge'); + $this->RegisterPropertyString('MQTTBaseTopic', 'zigbee2mqtt'); + $Version = 'unknown'; + $File = file(dirname(__DIR__) . '/libs/IPSymconExtension.js'); + $Start = strpos($File[2], 'Version: '); + if ($Start) { + $Version = trim(substr($File[2], $Start + strlen('Version: '))); + } + $this->actualExtensionVersion = $Version; + $this->ExtensionFilename = ''; + $this->ConfigLastSeen = 'epoch'; + $this->TransactionData = []; } public function ApplyChanges() { + $this->TransactionData = []; //Never delete this line! parent::ApplyChanges(); $this->ConnectParent('{C6D2AEB3-6E1F-4B2E-8E69-3A1A00246850}'); - //Setze Filter für ReceiveData - $MQTTTopic = $this->ReadPropertyString('MQTTTopic'); - $this->SetReceiveDataFilter('.*' . $MQTTTopic . '.*'); + $BaseTopic = $this->ReadPropertyString('MQTTBaseTopic'); + if (empty($BaseTopic)) { + $this->SetStatus(IS_INACTIVE); + $this->SetReceiveDataFilter('NOTHING_TO_RECEIVE'); //block all + } else { + $this->SetStatus(IS_ACTIVE); + //Setze Filter für ReceiveData + $this->SetReceiveDataFilter('.*"Topic":"' . $this->ReadPropertyString('MQTTBaseTopic') . '/bridge/.*'); + } + $this->RegisterProfileIntegerEx('Z2M.bridge.restart', '', '', '', [ + [0, $this->Translate('Restart'), '', 0xFF0000], + ]); + $this->RegisterProfileStringEx('Z2M.brigde.loglevel', '', '', '', [ + ['error', $this->Translate('Error'), '', 0x00FF00], + ['warning', $this->Translate('Warning'), '', 0x00FF00], + ['info', $this->Translate('Information'), '', 0x00FF00], + ['debug', $this->Translate('Debug'), '', 0x00FF00], + ]); } public function ReceiveData($JSONString) { - $this->SendDebug('JSON', $JSONString, 0); - if (!empty($this->ReadPropertyString('MQTTTopic'))) { - $Buffer = json_decode($JSONString); - // Buffer decodieren und in eine Variable schreiben - $this->SendDebug('MQTT Topic', $Buffer->Topic, 0); - $this->SendDebug('MQTT Payload', $Buffer->Payload, 0); - if (property_exists($Buffer, 'Topic')) { - $Payload = json_decode($Buffer->Payload); - if (fnmatch('*state*', $Buffer->Topic)) { - $this->RegisterVariableBoolean('Z2M_State', $this->Translate('State'), ''); - switch ($Buffer->Payload) { - case 'online': - SetValue($this->GetIDForIdent('Z2M_State'), true); - break; - case 'offline': - SetValue($this->GetIDForIdent('Z2M_State'), false); - break; - default: - $this->SendDebug('Bridge State', 'Invalid Payload' . $Buffer->Payload, 0); - } + $BaseTopic = $this->ReadPropertyString('MQTTBaseTopic'); + if (empty($BaseTopic)) { + return ''; + } + $this->SendDebug('ReceiveData', $JSONString, 0); + + $Buffer = json_decode($JSONString, true); + if (!isset($Buffer['Topic'])) { + return ''; + } + $ReceiveTopic = $Buffer['Topic']; + $this->SendDebug('MQTT FullTopic', $ReceiveTopic, 0); + $Topic = substr($ReceiveTopic, strlen($BaseTopic . '/bridge/')); + $Topics = explode('/', $Topic); + $Topic = array_shift($Topics); + $this->SendDebug('MQTT Topic', $Topic, 0); + $this->SendDebug('MQTT Payload', $Buffer['Payload'], 0); + $Payload = json_decode($Buffer['Payload'], true); + switch ($Topic) { + case 'request': //nothing todo + break; + case 'response': //response from request + if (isset($Payload['transaction'])) { + $this->UpdateTransaction($Payload); + break; } - if (fnmatch('*config*', $Buffer->Topic)) { - if (property_exists($Payload, 'log_level')) { - $this->RegisterVariableString('Z2M_Log_Level', $this->Translate('Log Level'), ''); - SetValue($this->GetIDForIdent('Z2M_Log_Level'), $Payload->log_level); - } - if (property_exists($Payload, 'permit_join')) { - $this->RegisterVariableString('Z2M_Permit_Join', $this->Translate('Permit Join'), ''); - SetValue($this->GetIDForIdent('Z2M_Permit_Join'), $Payload->permit_join); + if (count($Topics)) { + if ($Topics[0] == 'networkmap') { + if ($Payload['status'] == 'ok') { + $this->RegisterVariableString($Payload['data']['type'], $this->Translate('Network Map')); + $this->SetValue($Payload['data']['type'], $Payload['data']['value']); + } } } - if (fnmatch('*event*', $Buffer->Topic)) { - if (property_exists($Payload, 'type')) { - switch ($Payload->type) { - case 'device_announce': - $this->RegisterVariableString('Z2M_Device_Announce', $this->Translate('Announce'), ''); - SetValue($this->GetIDForIdent('Z2M_Device_Announce'), $Buffer->Payload); - break; - } + break; + case 'state': + $this->RegisterVariableBoolean('state', $this->Translate('State')); + $this->SetValue('state', $Payload['state'] == 'online'); + break; + case 'info': + if (isset($Payload['log_level'])) { + $this->RegisterVariableString('log_level', $this->Translate('Log Level'), 'Z2M.brigde.loglevel'); + $this->EnableAction('log_level'); + $this->SetValue('log_level', $Payload['log_level']); + } + if (isset($Payload['permit_join'])) { + $this->RegisterVariableBoolean('permit_join', $this->Translate('Allow joining the network'), '~Switch'); + $this->EnableAction('permit_join'); + $this->SetValue('permit_join', $Payload['permit_join']); + } + if (isset($Payload['restart_required'])) { + $this->RegisterVariableBoolean('restart_required', $this->Translate('Restart Required')); + $this->SetValue('restart_required', $Payload['restart_required']); + $this->RegisterVariableInteger('restart_request', $this->Translate('Perform a restart'), 'Z2M.bridge.restart'); + $this->EnableAction('restart_request'); + } + if (isset($Payload['version'])) { + $this->RegisterVariableString('version', $this->Translate('Version')); + $this->SetValue('version', $Payload['version']); + } + if (isset($Payload['config']['advanced']['last_seen'])) { + $this->SendDebug('last_seen', $Payload['config']['advanced']['last_seen'], 0); + $this->ConfigLastSeen = $Payload['config']['advanced']['last_seen']; + if ($Payload['config']['advanced']['last_seen'] != 'epoch') { + $this->LogMessage($this->Translate('Wrong last_seen setting in Zigbee2MQTT. Please set last_seen to epoch.'), KL_ERROR); } } - if (fnmatch('*log*', $Buffer->Topic)) { - if (property_exists($Payload, 'type')) { - switch ($Payload->type) { - case 'pairing': - $this->RegisterVariableString('Z2M_Pairing', $this->Translate('Pairing'), ''); - SetValue($this->GetIDForIdent('Z2M_Pairing'), $Buffer->Payload); - break; - case 'device_connected': - $this->RegisterVariableString('Z2M_Device_Connected', $this->Translate('Device Connected'), ''); - SetValue($this->GetIDForIdent('Z2M_Device_Connected'), $Buffer->Payload); - break; - case 'device removed': - $this->RegisterVariableString('Z2M_Device_Removed', $this->Translate('Device Removed'), ''); - SetValue($this->GetIDForIdent('Z2M_Device_Removed'), $Buffer->Payload); - break; - case 'device_banned': - $this->RegisterVariableString('Z2M_Device_Banned', $this->Translate('Device Banned'), ''); - SetValue($this->GetIDForIdent('Z2M_Device_Banned'), $Buffer->Payload); - break; - case 'device_renamed': - $this->RegisterVariableString('Z2M_Device_Renamed', $this->Translate('Device Renamed'), ''); - SetValue($this->GetIDForIdent('Z2M_Device_Renamed'), $Payload->message->from . ' >> ' . $Payload->message->to); - break; - case 'device_bind': - $this->RegisterVariableString('Z2M_Device_bind', $this->Translate('Device Bind'), ''); - SetValue($this->GetIDForIdent('Z2M_Device_bind'), $Buffer->Payload); - break; - case 'device_unbind': - $this->RegisterVariableString('Z2M_Device_Unbind', $this->Translate('Device Unbind'), ''); - SetValue($this->GetIDForIdent('Z2M_Device_Unbind'), $Buffer->Payload); - break; - case 'device_group_add': - $this->RegisterVariableString('Z2M_Device_Group_Add', $this->Translate('Device Group Add'), ''); - SetValue($this->GetIDForIdent('Z2M_Device_Group_Add'), $Buffer->Payload); - break; - case 'device_group_remove': - $this->RegisterVariableString('Z2M_Device_Group_Remove', $this->Translate('Device Group Remove'), ''); - SetValue($this->GetIDForIdent('Z2M_Device_Group_Remove'), $Buffer->Payload); - break; - case 'device_group_remove_all': - $this->RegisterVariableString('Z2M_Device_Group_Remove_All', $this->Translate('Device Group Remove All'), ''); - SetValue($this->GetIDForIdent('Z2M_Device_Group_Remove_All'), $Buffer->Payload); - break; - case 'devices': - $this->RegisterVariableString('Z2M_Devices', $this->Translate('Devices'), ''); - SetValue($this->GetIDForIdent('Z2M_Devices'), $Buffer->Payload); - break; - case 'device_publish_error': - $this->RegisterVariableString('Z2M_Device_Publish_Error', $this->Translate('Device Publish Error'), ''); - SetValue($this->GetIDForIdent('Z2M_Device_Publish_Error'), $Payload->device_publish_error); - break; + if (isset($Payload['network'])) { + $this->RegisterVariableInteger('network_channel', $this->Translate('Network Channel')); + $this->SetValue('network_channel', $Payload['network']['channel']); + } + break; + case 'extensions': + $foundExtension = false; + foreach ($Payload as $Extension) { + if (strpos($Extension['code'], 'class IPSymconExtension')) { + $foundExtension = true; + $this->ExtensionName = $Extension['name']; + $Version = 'unknown'; + $Lines = explode("\n", $Extension['code']); + $Start = strpos($Lines[2], 'Version: '); + if ($Start) { + $Version = trim(substr($Lines[2], $Start + strlen('Version: '))); } + $this->RegisterVariableString('extension_version', $this->Translate('Extension Version')); + $this->SetValue('extension_version', $Version); + $this->RegisterVariableBoolean('extension_is_current', $this->Translate('Extension is up to date')); + $this->SetValue('extension_is_current', $this->actualExtensionVersion == $Version); + if ($this->actualExtensionVersion != $Version) { + $this->LogMessage($this->Translate('Symcon Extension in Zigbee2MQTT is outdated. Please update the extension.'), KL_ERROR); + } + break; } } + $this->RegisterVariableBoolean('extension_loaded', $this->Translate('Extension Loaded')); + $this->SetValue('extension_loaded', $foundExtension); + if (!$foundExtension) { + $this->LogMessage($this->Translate('No Symcon Extension in Zigbee2MQTT installed. Please install the extension.'), KL_ERROR); + } + break; + } + return ''; + } + + public function RequestAction($Ident, $Value) + { + switch ($Ident) { + case 'permit_join': + $this->SetPermitJoin((bool) $Value); + break; + case 'log_level': + $this->SetLogLevel((string) $Value); + break; + case'restart_request': + $this->Restart(); + break; + } + } + + public function GetConfigurationForm() + { + $Form = json_decode(file_get_contents(__DIR__ . '/form.json'), true); + if ($this->GetValue('extension_loaded') && $this->GetValue('extension_is_current')) { + $Form['actions'][0]['enabled'] = false; + $Form['actions'][0]['label'] = $this->Translate('Symcon-Extension is up-to-date'); + } + if ($this->ConfigLastSeen == 'epoch') { + $Form['actions'][1]['enabled'] = false; + $Form['actions'][1]['label'] = $this->Translate('last_seen setting is correct'); + } + return json_encode($Form); + } + + public function InstallSymconExtension() + { + if (empty($this->ExtensionName)) { + $ExtensionName = 'IPSymconExtension.js'; + } + $Topic = '/bridge/request/extension/save'; + $Payload = ['name'=>$ExtensionName, 'code'=>file_get_contents(dirname(__DIR__) . '/libs/IPSymconExtension.js')]; + $Result = $this->SendData($Topic, $Payload); + if ($Result) { //todo check the Response + return true; + } + return false; + } + + public function SetLastSeen() + { + $Topic = '/bridge/request/options'; + $Payload = [ + 'options'=> [ + 'advanced'=> [ + 'last_seen'=> 'epoch' + ] + ] + ]; + $Result = $this->SendData($Topic, $Payload); + if ($Result) { //todo check the Response + return true; + } + return false; + } + + public function SetPermitJoin(bool $PermitJoin) + { + $Topic = '/bridge/request/permit_join'; + $Payload = ['value'=>$PermitJoin]; + $Result = $this->SendData($Topic, $Payload); + if ($Result) { //todo check the Response + return true; + } + return false; + } + + public function SetLogLevel(string $LogLevel) + { + $Topic = '/bridge/request/options'; + $Payload = ['options' =>['advanced' => ['log_level'=> $LogLevel]]]; + $Result = $this->SendData($Topic, $Payload); + if ($Result) { //todo check the Response + return true; + } + return false; + } + + public function Restart() + { + $Topic = '/bridge/request/restart'; + $Result = $this->SendData($Topic); + if ($Result) { //todo check the Response + return true; + } + return false; + } + + public function CreateGroup(string $GroupName) + { + $Topic = '/bridge/request/group/add'; + $Payload = ['friendly_name' => $GroupName]; + $Result = $this->SendData($Topic, $Payload); + if ($Result) { //todo check the Response + return true; + } + return false; + } + + public function DeleteGroup(string $GroupName) + { + $Topic = '/bridge/request/group/remove'; + $Payload = ['id' => $GroupName]; + $Result = $this->SendData($Topic, $Payload); + if ($Result) { //todo check the Response + return true; + } + return false; + } + + public function RenameGroup(string $OldName, string $NewName) + { + $Topic = '/bridge/request/group/rename'; + $Payload = ['from' => $OldName, 'to' => $NewName]; + $Result = $this->SendData($Topic, $Payload); + if ($Result) { //todo check the Response + return true; + } + return false; + } + + public function AddDeviceToGroup(string $GroupName, string $DeviceName) + { + $Topic = '/bridge/request/group/members/add'; + $Payload = ['group'=>$GroupName, 'device' => $DeviceName]; + $Result = $this->SendData($Topic, $Payload); + if ($Result) { //todo check the Response + return true; + } + return false; + } + + public function RemoveDeviceFromGroup(string $GroupName, string $DeviceName) + { + $Topic = '/bridge/request/group/members/remove'; + $Payload = ['group'=>$GroupName, 'device' => $DeviceName, 'skip_disable_reporting'=>true]; + $Result = $this->SendData($Topic, $Payload); + if ($Result) { //todo check the Response + return true; + } + return false; + } + + public function RemoveAllDevicesFromGroup(string $GroupName) + { + $Topic = '/bridge/request/group/members/remove_all'; + $Payload = ['group'=>$GroupName]; + $Result = $this->SendData($Topic, $Payload); + if ($Result) { //todo check the Response + return true; + } + return false; + } + + public function Bind(string $SourceDevice, string $TargetDevice) + { + $Topic = '/bridge/request/device/bind'; + $Payload = ['from' => $SourceDevice, 'to' => $TargetDevice]; + $Result = $this->SendData($Topic, $Payload); + if ($Result) { //todo check the Response + return true; + } + return false; + } + + public function Unbind(string $SourceDevice, string $TargetDevice) + { + $Topic = '/bridge/request/device/unbind'; + $Payload = ['from' => $SourceDevice, 'to' => $TargetDevice]; + $Result = $this->SendData($Topic, $Payload); + if ($Result) { //todo check the Response + return true; + } + return false; + } + + public function RequestNetworkmap() + { + $Topic = '/bridge/request/networkmap'; + $Payload = ['type' => 'graphviz', 'routes' => true]; + return $this->SendData($Topic, $Payload, 0); + } + + public function RenameDevice(string $OldDeviceName, string $NewDeviceName) + { + $Topic = '/bridge/request/device/rename'; + $Payload = ['from' => $OldDeviceName, 'to' => $NewDeviceName]; + $Result = $this->SendData($Topic, $Payload); + if ($Result) { //todo check the Response + return true; + } + return false; + } + + public function RemoveDevice(string $DeviceName) + { + $Topic = '/bridge/request/device/remove'; + $Payload = ['id'=>$DeviceName]; + $Result = $this->SendData($Topic, $Payload); + if ($Result) { //todo check the Response + return true; + } + return false; + } + + public function CheckOTAUpdate(string $DeviceName) + { + $Topic = '/bridge/request/device/ota_update/check'; + $Payload = ['id'=>$DeviceName]; + $Result = $this->SendData($Topic, $Payload); + if ($Result) { //todo check the Response + return true; + } + return false; + } + + public function PerformOTAUpdate(string $DeviceName) + { + $Topic = '/bridge/request/device/ota_update/update'; + $Payload = ['id'=>$DeviceName]; + $Result = $this->SendData($Topic, $Payload); + if ($Result) { //todo check the Response + return true; + } + return false; + } + private function SendData(string $Topic, array $Payload = [], int $Timeout = 5000) + { + if ($Timeout) { + $TransactionId = $this->AddTransaction($Payload); + } + $this->SendDebug(__FUNCTION__ . ':Topic', $Topic, 0); + $this->SendDebug(__FUNCTION__ . ':Payload', json_encode($Payload), 0); + $DataJSON = self::BuildRequest($this->ReadPropertyString('MQTTBaseTopic') . $Topic, $Payload); + $this->SendDataToParent($DataJSON); + if ($Timeout) { + $Result = $this->WaitForTransactionEnd($TransactionId, $Timeout); + if ($Result === false) { + trigger_error($this->Translate('Zigbee2MQTT did not response.'), E_USER_NOTICE); + return false; } + return $Result; + } + return true; + } + + private function WaitForTransactionEnd(int $TransactionId, int $Timeout) + { + $Sleep = intdiv($Timeout, 1000); + for ($i = 0; $i < 1000; $i++) { + $Buffer = $this->TransactionData; + if (!array_key_exists($TransactionId, $Buffer)) { + return false; + } + if (count($Buffer[$TransactionId])) { + $this->RemoveTransaction($TransactionId); + return $Buffer[$TransactionId]; + } + IPS_Sleep($Sleep); + } + $this->RemoveTransaction($TransactionId); + return false; + } + //################# SENDQUEUE + + private function AddTransaction(array &$Payload) + { + if (!$this->lock('TransactionData')) { + throw new Exception($this->Translate('TransactionData is locked'), E_USER_NOTICE); + } + $TransactionId = mt_rand(1, 10000); + $Payload['transaction'] = $TransactionId; + $TransactionData = $this->TransactionData; + $TransactionData[$TransactionId] = []; + $this->TransactionData = $TransactionData; + $this->unlock('TransactionData'); + return $TransactionId; + } + + private function UpdateTransaction(array $Data) + { + if (!$this->lock('TransactionData')) { + throw new Exception($this->Translate('TransactionData is locked'), E_USER_NOTICE); + } + $TransactionData = $this->TransactionData; + if (array_key_exists($Data['transaction'], $TransactionData)) { + $TransactionData[$Data['transaction']] = $Data; + $this->TransactionData = $TransactionData; + $this->unlock('TransactionData'); + return; } + $this->unlock('TransactionData'); + return; + } + + private function RemoveTransaction(int $TransactionId) + { + if (!$this->lock('TransactionData')) { + throw new Exception($this->Translate('TransactionData is locked'), E_USER_NOTICE); + } + $TransactionData = $this->TransactionData; + unset($TransactionData[$TransactionId]); + $this->TransactionData = $TransactionData; + $this->unlock('TransactionData'); + } + + private static function BuildRequest(string $Topic, array $Payload) + { + return json_encode( + array_merge( + self::$MQTTDataArray, + [ + 'Topic' => $Topic, + 'Payload'=> json_encode($Payload) + ] + ), + JSON_UNESCAPED_SLASHES + ); } } diff --git a/libs/BufferHelper.php b/libs/BufferHelper.php new file mode 100644 index 00000000..f7fc049e --- /dev/null +++ b/libs/BufferHelper.php @@ -0,0 +1,72 @@ + + * @copyright 2018 Michael Tröger + * @license https://creativecommons.org/licenses/by-nc-sa/4.0/ CC BY-NC-SA 4.0 + * @version 5.0 + */ + +/** + * Trait welcher Objekt-Eigenschaften in den Instance-Buffer schreiben und lesen kann. + */ +trait BufferHelper +{ + /** + * Wert einer Eigenschaft aus den InstanceBuffer lesen. + * + * @access public + * @param string $name PropertyName + * @return mixed Value of Name + */ + public function __get($name) + { + if (strpos($name, 'Multi_') === 0) { + $Lines = ''; + foreach ($this->{'BufferListe_' . $name} as $BufferIndex) { + $Lines .= $this->{'Part_' . $name . $BufferIndex}; + } + return unserialize($Lines); + } + return unserialize($this->GetBuffer($name)); + } + + /** + * Wert einer Eigenschaft in den InstanceBuffer schreiben. + * + * @access public + * @param string $name PropertyName + * @param mixed Value of Name + */ + public function __set($name, $value) + { + $Data = serialize($value); + if (strpos($name, 'Multi_') === 0) { + $OldBuffers = $this->{'BufferListe_' . $name}; + if ($OldBuffers == false) { + $OldBuffers = []; + } + $Lines = str_split($Data, 8000); + foreach ($Lines as $BufferIndex => $BufferLine) { + $this->{'Part_' . $name . $BufferIndex} = $BufferLine; + } + $NewBuffers = array_keys($Lines); + $this->{'BufferListe_' . $name} = $NewBuffers; + $DelBuffers = array_diff_key($OldBuffers, $NewBuffers); + foreach ($DelBuffers as $DelBuffer) { + $this->{'Part_' . $name . $DelBuffer} = ''; + } + return; + } + $this->SetBuffer($name, $Data); + } +} diff --git a/libs/IPSymconExtension.js b/libs/IPSymconExtension.js index 5854733e..c4954f32 100644 --- a/libs/IPSymconExtension.js +++ b/libs/IPSymconExtension.js @@ -1,5 +1,9 @@ -const ZigbeeHerdsmanConverters = require('zigbee-herdsman-converters'); +/* + IPSymconExtension + Version: 4.4.0 +*/ +const ZigbeeHerdsmanConverters = require('zigbee-herdsman-converters'); class IPSymconExtension { constructor(zigbee, mqtt, state, publishEntityState, eventBus, settings, logger) { this.zigbee = zigbee; @@ -64,9 +68,9 @@ class IPSymconExtension { #createDevicePayload(device, boolExposes) { const definition = this.zigbeeHerdsmanConverters.findByDevice(device.zh); let exposes; - if (boolExposes) { - exposes = device.exposes(); - } + if (boolExposes) { + exposes = device.exposes(); + } return { ieeeAddr: device.ieeeAddr, diff --git a/libs/SemaphoreHelper.php b/libs/SemaphoreHelper.php new file mode 100644 index 00000000..e68b5bf2 --- /dev/null +++ b/libs/SemaphoreHelper.php @@ -0,0 +1,51 @@ + + * @copyright 2018 Michael Tröger + * @license https://creativecommons.org/licenses/by-nc-sa/4.0/ CC BY-NC-SA 4.0 + * @version 5.0 + */ + +/** + * Biete Funktionen um auf Objekte Thread-Safe zuzugreifen. + */ +trait Semaphore +{ + /** + * Versucht eine Semaphore zu setzen und wiederholt dies bei Misserfolg bis zu 100 mal. + * @param string $ident Ein String der den Lock bezeichnet. + * @return boolean TRUE bei Erfolg, FALSE bei Misserfolg. + */ + private function lock($ident) + { + for ($i = 0; $i < 100; $i++) { + if (IPS_SemaphoreEnter(__CLASS__ . '.' . (string) $this->InstanceID . (string) $ident, 1)) { + return true; + } else { + IPS_Sleep(mt_rand(1, 5)); + } + } + return false; + } + + /** + * Löscht eine Semaphore. + * @param string $ident Ein String der den Lock bezeichnet. + */ + private function unlock($ident) + { + IPS_SemaphoreLeave(__CLASS__ . '.' . (string) $this->InstanceID . (string) $ident); + } +} + +/* @} */ diff --git a/libs/Zigbee2MQTTBridgeHelper.php b/libs/Zigbee2MQTTBridgeHelper.php deleted file mode 100644 index 75dd0ccd..00000000 --- a/libs/Zigbee2MQTTBridgeHelper.php +++ /dev/null @@ -1,152 +0,0 @@ -ReadPropertyString('MQTTTopic') . '/group/' . $group_name . '/add'; - $Data['Payload'] = $friendly_name; - $DataJSON = json_encode($Data, JSON_UNESCAPED_SLASHES); - $this->SendDebug(__FUNCTION__ . 'Add Group Topic', $Data['Topic'], 0); - $this->SendDebug(__FUNCTION__, $DataJSON, 0); - $this->SendDataToParent($DataJSON); - } - - public function RemoveGroup(string $group_name, string $friendly_name) - { - $Data['DataID'] = '{043EA491-0325-4ADD-8FC2-A30C8EEB4D3F}'; - $Data['PacketType'] = 3; - $Data['QualityOfService'] = 0; - $Data['Retain'] = false; - $Data['Topic'] = MQTT_GROUP_TOPIC . '/' . $this->ReadPropertyString('MQTTTopic') . '/group/' . $group_name . '/remove'; - $Data['Payload'] = $friendly_name; - $DataJSON = json_encode($Data, JSON_UNESCAPED_SLASHES); - $this->SendDebug(__FUNCTION__ . 'Remove Group Topic', $Data['Topic'], 0); - $this->SendDebug(__FUNCTION__, $DataJSON, 0); - $this->SendDataToParent($DataJSON); - } - - public function RemoveAllGroup(string $group_name, string $friendly_name) - { - $Data['DataID'] = '{043EA491-0325-4ADD-8FC2-A30C8EEB4D3F}'; - $Data['PacketType'] = 3; - $Data['QualityOfService'] = 0; - $Data['Retain'] = false; - $Data['Topic'] = MQTT_GROUP_TOPIC . '/' . $this->ReadPropertyString('MQTTTopic') . '/group/' . $group_name . '/remove_all'; - $Data['Payload'] = $friendly_name; - $DataJSON = json_encode($Data, JSON_UNESCAPED_SLASHES); - $this->SendDebug(__FUNCTION__ . 'Publish Topic', $Data['Topic'], 0); - $this->SendDebug(__FUNCTION__, $DataJSON, 0); - $this->SendDataToParent($DataJSON); - } - - public function Bind(string $source_device, string $target_device) - { - $Data['DataID'] = '{043EA491-0325-4ADD-8FC2-A30C8EEB4D3F}'; - $Data['PacketType'] = 3; - $Data['QualityOfService'] = 0; - $Data['Retain'] = false; - $Data['Topic'] = MQTT_GROUP_TOPIC . '/' . $this->ReadPropertyString('MQTTTopic') . '/bind/' . $source_device; - $Data['Payload'] = $target_device; - $DataJSON = json_encode($Data, JSON_UNESCAPED_SLASHES); - $this->SendDebug(__FUNCTION__ . 'Bind Topic', $Data['Topic'], 0); - $this->SendDebug(__FUNCTION__, $DataJSON, 0); - $this->SendDataToParent($DataJSON); - } - - public function Unbind(string $source_device, string $target_device) - { - $Data['DataID'] = '{043EA491-0325-4ADD-8FC2-A30C8EEB4D3F}'; - $Data['PacketType'] = 3; - $Data['QualityOfService'] = 0; - $Data['Retain'] = false; - $Data['Topic'] = MQTT_GROUP_TOPIC . '/' . $this->ReadPropertyString('MQTTTopic') . '/unbind/' . $source_device; - $Data['Payload'] = $target_device; - $DataJSON = json_encode($Data, JSON_UNESCAPED_SLASHES); - $this->SendDebug(__FUNCTION__ . 'Unbind Topic', $Data['Topic'], 0); - $this->SendDebug(__FUNCTION__, $DataJSON, 0); - $this->SendDataToParent($DataJSON); - } - - public function getGroupMembership(string $friendly_name) - { - $Data['DataID'] = '{043EA491-0325-4ADD-8FC2-A30C8EEB4D3F}'; - $Data['PacketType'] = 3; - $Data['QualityOfService'] = 0; - $Data['Retain'] = false; - $Data['Topic'] = MQTT_GROUP_TOPIC . '/' . $this->ReadPropertyString('MQTTTopic') . '/device/' . $friendly_name . '/get_group_membership'; - $Data['Payload'] = ''; - $DataJSON = json_encode($Data, JSON_UNESCAPED_SLASHES); - $this->SendDebug(__FUNCTION__ . 'getGroupMembership Topic', $Data['Topic'], 0); - $this->SendDebug(__FUNCTION__, $DataJSON, 0); - $this->SendDataToParent($DataJSON); - } - - public function Networkmap() - { - $Data['DataID'] = '{043EA491-0325-4ADD-8FC2-A30C8EEB4D3F}'; - $Data['PacketType'] = 3; - $Data['QualityOfService'] = 0; - $Data['Retain'] = false; - $Data['Topic'] = MQTT_GROUP_TOPIC . '/' . $this->ReadPropertyString('MQTTTopic') . '/networkmap'; - $Data['Payload'] = 'graphviz'; - $DataJSON = json_encode($Data, JSON_UNESCAPED_SLASHES); - $this->SendDebug(__FUNCTION__ . 'Bind Topic', $Data['Topic'], 0); - $this->SendDebug(__FUNCTION__, $DataJSON, 0); - $this->SendDataToParent($DataJSON); - } - - public function RenameDevice(string $old_friendly_name, string $new_friendly_name) - { - $Data['DataID'] = '{043EA491-0325-4ADD-8FC2-A30C8EEB4D3F}'; - $Data['PacketType'] = 3; - $Data['QualityOfService'] = 0; - $Data['Retain'] = false; - $Data['Topic'] = MQTT_GROUP_TOPIC . '/' . $this->ReadPropertyString('MQTTTopic') . '/config/rename'; - - $Payload['old'] = $old_friendly_name; - $Payload['new'] = $new_friendly_name; - $Data['Payload'] = json_encode($Payload); - - $DataJSON = json_encode($Data, JSON_UNESCAPED_SLASHES); - $this->SendDebug(__FUNCTION__ . 'RenameDevice Topic', $Data['Topic'], 0); - $this->SendDebug(__FUNCTION__, $DataJSON, 0); - $this->SendDataToParent($DataJSON); - } - - public function BanDevice(string $friendly_name) - { - $Data['DataID'] = '{043EA491-0325-4ADD-8FC2-A30C8EEB4D3F}'; - $Data['PacketType'] = 3; - $Data['QualityOfService'] = 0; - $Data['Retain'] = false; - $Data['Topic'] = MQTT_GROUP_TOPIC . '/' . $this->ReadPropertyString('MQTTTopic') . '/config/ban'; - $Data['Payload'] = $friendly_name; - $DataJSON = json_encode($Data, JSON_UNESCAPED_SLASHES); - $this->SendDebug(__FUNCTION__ . 'Bind Topic', $Data['Topic'], 0); - $this->SendDebug(__FUNCTION__, $DataJSON, 0); - $this->SendDataToParent($DataJSON); - } - - public function RemoveDevice(string $friendly_name) - { - $Data['DataID'] = '{043EA491-0325-4ADD-8FC2-A30C8EEB4D3F}'; - $Data['PacketType'] = 3; - $Data['QualityOfService'] = 0; - $Data['Retain'] = false; - $Data['Topic'] = MQTT_GROUP_TOPIC . '/' . $this->ReadPropertyString('MQTTTopic') . '/config/remove'; - $Data['Payload'] = $friendly_name; - $DataJSON = json_encode($Data, JSON_UNESCAPED_SLASHES); - $this->SendDebug(__FUNCTION__ . 'Bind Topic', $Data['Topic'], 0); - $this->SendDebug(__FUNCTION__, $DataJSON, 0); - $this->SendDataToParent($DataJSON); - } -}