diff --git a/.github/workflows/style.yml b/.github/workflows/style.yml new file mode 100644 index 0000000..23ef350 --- /dev/null +++ b/.github/workflows/style.yml @@ -0,0 +1,16 @@ +name: Check Style + +on: [push, pull_request] + +jobs: + style: + runs-on: ubuntu-latest + steps: + - name: Setup PHP + uses: shivammathur/setup-php@v2 + with: + php-version: 7.4 + - name: Checkout module + uses: actions/checkout@master + - name: Check style + uses: Nall-chan/action-style@master diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml new file mode 100644 index 0000000..e6563d0 --- /dev/null +++ b/.github/workflows/tests.yml @@ -0,0 +1,14 @@ +name: Run Tests + +on: [push, pull_request] + +jobs: + test: + runs-on: ubuntu-latest + steps: + - name: Checkout module + uses: actions/checkout@master + with: + submodules: true + - name: Run tests + uses: symcon/action-tests@master diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..3aba2ef --- /dev/null +++ b/.gitignore @@ -0,0 +1,6 @@ +<<<<<<< HEAD +/docs/ +======= +docs/ +>>>>>>> 4236f2949deef76002a8103f5f36eaf5e39b2d66 +.vscode/settings.json diff --git a/.gitmodules b/.gitmodules new file mode 100644 index 0000000..304d376 --- /dev/null +++ b/.gitmodules @@ -0,0 +1,12 @@ +[submodule "libs/helper"] + path = libs/helper + url = https://github.com/Nall-chan/SymconModulHelper +[submodule ".style"] + path = .style + url = https://github.com/Nall-chan/StylePHP +[submodule "tests/stubs"] + path = tests/stubs + url = https://github.com/Nall-chan/SymconStubs +[submodule ".vscode"] + path = .vscode + url = https://github.com/Nall-chan/SymconVSCTasks.git diff --git a/.style b/.style new file mode 160000 index 0000000..ccb6319 --- /dev/null +++ b/.style @@ -0,0 +1 @@ +Subproject commit ccb631937353df3a0c61fe19de0f60c31b8cb190 diff --git a/.vscode b/.vscode new file mode 160000 index 0000000..672a096 --- /dev/null +++ b/.vscode @@ -0,0 +1 @@ +Subproject commit 672a096abc8fcc6afa43f4aedd47821b5583b4c5 diff --git a/KLF200Configurator/README.md b/KLF200Configurator/README.md new file mode 100644 index 0000000..f984ccd --- /dev/null +++ b/KLF200Configurator/README.md @@ -0,0 +1,127 @@ +[![SDK](https://img.shields.io/badge/Symcon-PHPModul-red.svg?style=flat-square)](https://www.symcon.de/service/dokumentation/entwicklerbereich/sdk-tools/sdk-php/) +[![Version](https://img.shields.io/badge/Modul%20Version-0.80-blue.svg?style=flat-square)](https://community.symcon.de/t/modul-velux-klf200/50429) +[![Version](https://img.shields.io/badge/Symcon%20Version-5.5%20%3E-green.svg?style=flat-square)](https://www.symcon.de/service/dokumentation/installation/migrationen/v54-v55-q4-2020/) +[![License](https://img.shields.io/badge/License-CC%20BY--NC--SA%204.0-green.svg?style=flat-square)](https://creativecommons.org/licenses/by-nc-sa/4.0/) +[![Check Style](https://github.com/Nall-chan/VeluxKLF200/workflows/Check%20Style/badge.svg)](https://github.com/Nall-chan/VeluxKLF200/actions) [![Run Tests](https://github.com/Nall-chan/VeluxKLF200/workflows/Run%20Tests/badge.svg)](https://github.com/Nall-chan/VeluxKLF200/actions) +[![Spenden](https://www.paypalobjects.com/de_DE/DE/i/btn/btn_donate_SM.gif)](#3-spenden) +[![Wunschliste](https://img.shields.io/badge/Wunschliste-Amazon-ff69fb.svg)](#3-spenden) + +# Velux KLF200 Configurator + +## Inhaltsverzeichnis + +- [1. Funktionsumfang](#1-funktionsumfang) +- [2. Voraussetzungen](#2-voraussetzungen) +- [3. Software-Installation](#3-software-installation) +- [4. Einrichten der Instanzen in IP-Symcon](#4-einrichten-der-instanzen-in-ip-symcon) +- [5. Statusvariablen und Profile](#5-statusvariablen-und-profile) +- [6. Gateway Kommandos](#6-gateway-kommandos) +- [7. PHP-Befehlsreferenz](#7-php-befehlsreferenz) +- [8. Aktionen](#8-aktionen) +- [9. Anhang](#9-anhang) + - [1. Changelog](#1-changelog) + - [2. Spenden](#2-spenden) +- [10. Lizenz](#10-lizenz) + + +## 1. Funktionsumfang + + - Auslesen und darstellen aller vom Gateway bekannten Geräte (Nodes). + - Einfaches Anlegen von neuen Instanzen in IPS. + - Anlernen und löschen von Nodes im Gateway. + +## 2. Voraussetzungen + + - IPS ab Version 5.5 + - KLF200 io-homecontrol® Gateway + - KLF muss per LAN angeschlossen sein + - KLF Firmware 2.0.0.71 oder neuer + +## 3. Software-Installation + +* Dieses Modul ist Bestandteil der [VeluxKLF200-Library](../README.md#3-software-installation). + +## 4. Einrichten der Instanzen in IP-Symcon + +Eine einfache Einrichtung ist über diese Instanz möglich. +Bei der installation aus dem Store wird das anlegen der Instanz automatisch angeboten. + +In den sich öffneden Konfigurationsfenstern ist das Passwort und die IP-Adresse bzw. der Hostname einzutragen. +``Das Standardkennwort ist dasselbe wie das auf der Rückseite des KLF200 angegebene WLAN-Kennwort.`` + +Bei der manuellen Einrichtung ist das Modul im Dialog `Instanz hinzufügen` unter den Hersteller `VELUX` zu finden. +![Instanz hinzufügen](../imgs/instanzen.png) + +Alternativ ist es auch in der Liste alle Konfiguratoren aufgeführt. +![Instanz hinzufügen](../imgs/instanzen_configurator.png) + +Es wird automatisch eine `KLF200 Gateway` Instanz erzeugt, wenn noch keine vorhanden ist. +Werden in dem sich öffnenden Konfigurationsformular keine Geräte angezeigt, so ist zuerst die IO-Instanz korrekt zu konfigurieren. +Diese kann über die Schaltfläche `Gateway konfigurieren` und dann `Schnittstelle konfigurieren` erreicht werden. + +Ist die Gateway-Instanz korrekt verbunden, wird beim öffnen des Konfigurator folgendender Dialog angezeigt. +![Konfigurator](../imgs/conf_configurator.png) + +Über das selektieren eines Eintrages in der Tabelle und betätigen des dazugehörigen `Erstellen` Button, +können Instanzen in IPS angelegt werden. + +## 5. Statusvariablen und Profile + +Dieses Modul erstellt keine Statusvariablen und Profile. + +## 6. Gateway Kommandos + +In dem Konfigurator sind über den ersten Eintrag die Gateway Kommandos erreichbar. +Mit den dort vorhandenen Schaltflächen können Geräte (Nodes) an dem Gateway an- un ab gelernt werden, sowie das Gateway selber neugestartet werden. +Durch die Schaltfläche `Suche Gerät`, wird die Gerätesuche gestartet. Alle Geräte welche sich im Anlern-Modus befinden werden hierdurch an das Gateway angelernt. +![Konfigurator](../imgs/conf_configurator1.png) + +Über die Schaltfläche `Entferne Gerät` wird eine Liste mit allen im Gateway vorhandenen Geräten (Nodes) angezeigt. +Durch auswählen eines Eintrages und betätigen von `Entferne Gerät` wird das Gerät aus dem Gateway gelöscht. +![Konfigurator](../imgs/conf_configurator2.png) + +## 7. PHP-Befehlsreferenz + +**Folgende Funktionen liefern `TRUE` bei Erfolg. +Im Fehlerfall wird eine Warnung erzeugt und `FALSE` zurückgegeben.** + +```php +bool KLF200_DiscoveryNodes(int $InstanzeID); +``` +Sucht nach neuen Geräten und lernte Diese am Gateway an. + +```php +bool KLF200_RemoveNode(int $InstanzeID, int $Node); +``` +Entfernt das angelernte Gerät mit der in '$Node' übergebenen NodeId aus dem Gateway. + +```php +bool KLF200_RebootGateway(int $InstanzeID); +``` +Startet das Gateway KLF200 neu. + + +## 8. Aktionen + +Es gibt keine speziellen Aktionen für dieses Modul. + +## 9. Anhang + +### 1. Changelog + +[Changelog der Library](../README.md#2-changelog) + +### 2. Spenden + + Die Library ist für die nicht kommerzielle Nutzung kostenlos, Schenkungen als Unterstützung für den Autor werden hier akzeptiert: + + + +[![Wunschliste](https://img.shields.io/badge/Wunschliste-Amazon-ff69fb.svg)](https://www.amazon.de/hz/wishlist/ls/YU4AI9AQT9F?ref_=wl_share) + + +## 10. Lizenz + + IPS-Modul: + [CC BY-NC-SA 4.0](https://creativecommons.org/licenses/by-nc-sa/4.0/) + \ No newline at end of file diff --git a/KLF200Configurator/form.json b/KLF200Configurator/form.json new file mode 100644 index 0000000..6c03827 --- /dev/null +++ b/KLF200Configurator/form.json @@ -0,0 +1,162 @@ +{ + "actions": [ + { + "type": "ExpansionPanel", + "caption": "Gateway commands", + "items": [ + { + "type": "RowLayout", + "name": "GatewayCommands", + "items": [ + { + "type": "Button", + "label": "Discover device", + "onClick": "KLF200_DiscoveryNodes($id);" + }, + { + "type": "PopupButton", + "caption": "Remove device", + "popup": { + "caption": "Select device to remove", + "items": [ + { + "type": "Button", + "label": "Remove device", + "onClick": "" + }, + { + "type": "List", + "name": "RemoveNode", + "add": false, + "delete": false, + "sort": { + "column": "nodeid", + "direction": "ascending" + }, + "columns": [ + { + "caption": "Node ID", + "name": "nodeid", + "width": "80px" + }, + { + "caption": "Type", + "name": "type", + "width": "200px" + }, + { + "caption": "Name", + "name": "name", + "width": "auto" + } + ], + "values": [] + }, + { + "type": "ProgressBar", + "name": "ProgressRemove", + "current": 100, + "indeterminate": true, + "visible": false, + "caption": "Waiting for finish remove." + } + ] + } + }, + { + "type": "Button", + "label": "Reboot Gateway", + "onClick": "" + }, + { + "type": "Button", + "label": "Refresh Devicelist", + "onClick": "" + } + ] + }, + { + "type": "RowLayout", + "name": "GatewayProgress", + "items": [ + { + "type": "ProgressBar", + "name": "ProgressLearn", + "current": 100, + "indeterminate": true, + "visible": false, + "caption": "Waiting for finish discovery." + } + ] + } + ] + }, + { + "type": "Configurator", + "name": "Config", + "caption": "", + "add": false, + "delete": true, + "sort": { + "column": "nodeid", + "direction": "ascending" + }, + "columns": [ + { + "caption": "Node ID", + "name": "nodeid", + "width": "80px" + }, + { + "caption": "Type", + "name": "type", + "width": "300px" + }, + { + "caption": "Name", + "name": "name", + "width": "300px" + }, + { + "caption": "Location", + "name": "location", + "width": "auto" + } + ], + "values": [] + }, + { + "type": "PopupAlert", + "name": "AlertPopup", + "visible": false, + "popup": { + "items": [ + { + "name": "AlertText", + "type": "Label", + "caption": "" + } + ] + } + }, + { + "type": "Label", + "caption": "This module is free for non-commercial use,\r\nDonations in support of the author are accepted here:" + }, + { + "type": "RowLayout", + "items": [ + { + "type": "Image", + "onClick": "echo 'https://www.paypal.com/donate?hosted_button_id=G2SLW2MEMQZH2';", + "image": "" + }, + { + "type": "Image", + "onClick": "echo 'https://www.amazon.de/hz/wishlist/ls/YU4AI9AQT9F?ref_=wl_share';", + "image": "" + } + ] + } + ] +} \ No newline at end of file diff --git a/KLF200Configurator/locale.json b/KLF200Configurator/locale.json new file mode 100644 index 0000000..a32db96 --- /dev/null +++ b/KLF200Configurator/locale.json @@ -0,0 +1,33 @@ +{ + "translations": { + "de": { + "Gateway commands": "Gateway Kommandos", + "Discover device": "Suche Gerät", + "Remove device": "Entferne Gerät", + "Select device to remove": "Gerät zum Entfernen auswählen", + "Reboot Gateway": "Gateway neustarten", + "Refresh Devicelist": "Aktualisiere Geräteliste", + "Type": "Typ", + "Zone": "Zone", + "Name": "Name", + "Location": "Lokation", + "Instance has no active parent.": "Instanz hat keinen aktiven Parent.", + "Splitter has no IO instance.": "Splitter hat keine IO Instanz.", + "%s out of range.": "%s außerhalb des zulässigen Bereiches.", + "The view will reload after discovery is finished.": "Die Ansicht wird neu geladen, sobald die Suche abgeschlossen wurde.", + "The view will reload after remove is finished.": "Die Ansicht wird neu geladen, nachdem das Entfernen beendet wurde.", + "The KLF200 will now reboot.": "Das KLF200 wird jetzt neustarten.", + "No error.": "Kein Fehler.", + "Timeout.": "Zeitüberschreitung.", + "Waiting for finish remove.": "Warte auf entfernen.", + "Waiting for finish discovery.": "Warte auf Gerätesuche.", + "Not further defined error.": "Nicht weiter definierter Fehler.", + "Unknown command or command is not accepted at this state.": "Unbekanntes Kommando oder Kommando wird zu diesem Zeitpunkt nicht akzeptiert.", + "Error on Frame Structure.": "Fehler in Frame Struktur.", + "Busy. Try again later.": "Beschäftigt. Später nochmal versuchen.", + "Bad system table index.": "Ungültiger Index der Systemtabelle.", + "Not authenticated.": "Nicht berechtigt.", + "This module is free for non-commercial use,\r\nDonations in support of the author are accepted here:": "Dieses Modul ist für die nicht kommerzielle Nutzung kostenlos,\r\nSchenkungen als Unterstützung für den Autor werden hier akzeptiert:" + } + } +} \ No newline at end of file diff --git a/KLF200Configurator/module.json b/KLF200Configurator/module.json new file mode 100644 index 0000000..d13f2c1 --- /dev/null +++ b/KLF200Configurator/module.json @@ -0,0 +1,18 @@ +{ + "id": "{38724E6E-8202-4D37-9FA7-BDD2EDA79520}", + "name": "KLF200 Configurator", + "type": 4, + "vendor": "VELUX", + "aliases": [ + "KLF200 Configurator" + ], + "parentRequirements": [ + "{7B0F87CC-0408-4283-8E0E-2D48141E42E8}" + ], + "childRequirements": [], + "implemented": [ + "{5242DAEF-EEBD-441F-AB0B-E83C01475B65}" + ], + "prefix": "KLF200", + "url": "https://github.com/Nall-chan/VeluxKLF200/blob/master/KLF200Configurator/README.md" +} \ No newline at end of file diff --git a/KLF200Configurator/module.php b/KLF200Configurator/module.php new file mode 100644 index 0000000..ade12d5 --- /dev/null +++ b/KLF200Configurator/module.php @@ -0,0 +1,429 @@ +' . file_get_contents(__DIR__ . '/../libs/helper/BufferHelper.php') . '}'); +eval('declare(strict_types=1);namespace KLF200Configurator {?>' . file_get_contents(__DIR__ . '/../libs/helper/SemaphoreHelper.php') . '}'); +eval('declare(strict_types=1);namespace KLF200Configurator {?>' . file_get_contents(__DIR__ . '/../libs/helper/DebugHelper.php') . '}'); +eval('declare(strict_types=1);namespace KLF200Configurator {?>' . file_get_contents(__DIR__ . '/../libs/helper/ParentIOHelper.php') . '}'); + +/** + * @property array $Nodes + * @property bool $GetNodeInfoIsRunning + */ +class KLF200Configurator extends IPSModule +{ + use \KLF200Configurator\Semaphore, + \KLF200Configurator\BufferHelper, + \KLF200Configurator\DebugHelper, + \KLF200Configurator\InstanceStatus { + \KLF200Configurator\InstanceStatus::MessageSink as IOMessageSink; + \KLF200Configurator\InstanceStatus::RequestAction as IORequestAction; + \KLF200Configurator\DebugHelper::SendDebug as SendDebug2; + } + + /** + * Interne Funktion des SDK. + */ + public function Create() + { + parent::Create(); + $this->ConnectParent('{725D4DF6-C8FC-463C-823A-D3481A3D7003}'); + $this->GetNodeInfoIsRunning = false; + $this->Nodes = []; + $this->ParentID = 0; + } + + /** + * Interne Funktion des SDK. + */ + public function ApplyChanges() + { + $this->RegisterMessage(0, IPS_KERNELSTARTED); + $this->RegisterMessage($this->InstanceID, FM_CONNECT); + $this->RegisterMessage($this->InstanceID, FM_DISCONNECT); + + parent::ApplyChanges(); + $APICommands = [ + \KLF200\APICommand::GET_ALL_GROUPS_INFORMATION_NTF, + \KLF200\APICommand::GET_ALL_GROUPS_INFORMATION_FINISHED_NTF, + \KLF200\APICommand::GET_ALL_NODES_INFORMATION_NTF, + \KLF200\APICommand::GET_ALL_NODES_INFORMATION_FINISHED_NTF, + \KLF200\APICommand::GET_SCENE_INFORMATION_NTF, + \KLF200\APICommand::GET_SCENE_LIST_NTF, + \KLF200\APICommand::CS_DISCOVER_NODES_NTF, + \KLF200\APICommand::CS_SYSTEM_TABLE_UPDATE_NTF + ]; + + if (count($APICommands) > 0) { + foreach ($APICommands as $APICommand) { + $Lines[] = '.*"Command":' . $APICommand . '.*'; + } + $Line = implode('|', $Lines); + $this->SetReceiveDataFilter('(' . $Line . ')'); + $this->SendDebug('FILTER', $Line, 0); + } + if (IPS_GetKernelRunlevel() == KR_READY) { + $this->KernelReady(); + } + + } + + /** + * Interne Funktion des SDK. + */ + public function MessageSink($TimeStamp, $SenderID, $Message, $Data) + { + $this->IOMessageSink($TimeStamp, $SenderID, $Message, $Data); + switch ($Message) { + case IPS_KERNELSTARTED: + $this->KernelReady(); + break; + } + } + + public function RequestAction($Ident, $Value) + { + if ($this->IORequestAction($Ident, $Value)) { + return true; + } + if ($Ident == 'GetAllNodesInformation') { + if ($Value) { + if ($this->GetAllNodesInformation()) { + while ($this->GetNodeInfoIsRunning) { + IPS_Sleep(10); + } + return true; + } + } else { + return $this->GetAllNodesInformation(); + } + } + return false; + } + + public function DiscoveryNodes() + { + $APIData = new \KLF200\APIData(\KLF200\APICommand::CS_DISCOVER_NODES_REQ, "\x00"); + $ResultAPIData = $this->SendAPIData($APIData); + if ($ResultAPIData->isError()) { + trigger_error($this->Translate($ResultAPIData->ErrorToString()), E_USER_NOTICE); + return false; + } + $this->UpdateFormField('GatewayCommands', 'visible', false); + $this->UpdateFormField('Config', 'visible', false); + $this->UpdateFormField('ProgressLearn', 'visible', true); + return true; + } + + public function RebootGateway() + { + $APIData = new \KLF200\APIData(\KLF200\APICommand::REBOOT_REQ); + $ResultAPIData = $this->SendAPIData($APIData); + if ($ResultAPIData->isError()) { + trigger_error($this->Translate($ResultAPIData->ErrorToString()), E_USER_NOTICE); + return false; + } + return true; + } + + public function RemoveNode(int $Node) + { + if (($Node < 0) || ($Node > 199)) { + trigger_error(sprintf($this->Translate('%s out of range.'), 'Node'), E_USER_NOTICE); + return false; + } + $this->UpdateFormField('ProgressRemove', 'visible', true); + $this->UpdateFormField('RemoveNode', 'visible', false); + $Data = "\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"; + $Data[intdiv($Node, 8)] = chr(1 << ($Node % 8)); + $APIData = new \KLF200\APIData(\KLF200\APICommand::CS_REMOVE_NODES_REQ, $Data); + $ResultAPIData = $this->SendAPIData($APIData); + if ($ResultAPIData->isError()) { + trigger_error($this->Translate($ResultAPIData->ErrorToString()), E_USER_NOTICE); + return false; + } + //PopupRemove + //$this->UpdateFormField('ProgressRemove', 'visible', false); + //$this->UpdateFormField('RemoveNode', 'visible', true); + return true; + } + + public function GetConfigurationForm() + { + $Form = json_decode(file_get_contents(__DIR__ . '/form.json'), true); + $NodeValues = []; + if (!$this->HasActiveParent()) { + $Form['actions'][2]['visible'] = true; + $Form['actions'][2]['popup']['items'][0]['caption'] = 'Instance has no active parent.'; + $Form['actions'][0]['items'][0]['visible'] = false; + } else { + $Splitter = IPS_GetInstance($this->InstanceID)['ConnectionID']; + $IO = IPS_GetInstance($Splitter)['ConnectionID']; + if ($IO == 0) { + $Form['actions'][2]['visible'] = true; + $Form['actions'][2]['popup']['items'][0]['caption'] = 'Splitter has no IO instance.'; + } else { + $NodeValues = $this->GetNodeConfigFormValues($Splitter); + } + } + $Form['actions'][1]['values'] = $NodeValues; + + /* $Form['actions'][0]['items'][0]['items'][0]['onClick'] = <<<'EOT' + if(KLF200_DiscoveryNodes($id)){ + echo + EOT . ' "' . $this->Translate('The view will reload after discovery is finished.') . '";}'; + */ + $DeleteNodeValues = $this->GetDeleteNodeConfigFormValues(); + $Form['actions'][0]['items'][0]['items'][1]['popup']['items'][1]['values'] = $DeleteNodeValues; + $Form['actions'][0]['items'][0]['items'][1]['popup']['items'][0]['onClick'] = <<<'EOT' + if (is_int($RemoveNode['nodeid'])){ + KLF200_RemoveNode($id,$RemoveNode['nodeid']); + } else { + EOT . ' echo "' . $this->Translate('Nothing selected.') . '";}'; + + $Form['actions'][0]['items'][0]['items'][2]['onClick'] = <<<'EOT' + if(KLF200_RebootGateway($id)){ + echo + EOT . ' "' . $this->Translate('The KLF200 will now reboot.') . '";}'; + + $Form['actions'][0]['items'][0]['items'][3]['onClick'] = <<<'EOT' + IPS_RequestAction($id,'GetAllNodesInformation',true); + EOT; + + $this->SendDebug('FORM', json_encode($Form), 0); + $this->SendDebug('FORM', json_last_error_msg(), 0); + return json_encode($Form); + } + + public function ReceiveData($JSONString) + { + $APIData = new \KLF200\APIData($JSONString); + $this->SendDebug('Event', $APIData, 1); + $this->ReceiveEvent($APIData); + } + + /** + * Wird ausgeführt wenn der Kernel hochgefahren wurde. + */ + protected function KernelReady() + { + $this->RegisterParent(); + if ($this->HasActiveParent()) { + $this->IOChangeState(IS_ACTIVE); + } + } + + /** + * Wird über den Trait InstanceStatus ausgeführt wenn sich der Status des Parent ändert. + * Oder wenn sich die Zuordnung zum Parent ändert. + * + * @param int $State Der neue Status des Parent. + */ + protected function IOChangeState($State) + { + if ($State == IS_ACTIVE) { + $this->UpdateFormField('GatewayCommands', 'visible', true); + $this->GetAllNodesInformation(); + } else { + $this->Nodes = []; + $NodeValues = []; + $Splitter = IPS_GetInstance($this->InstanceID)['ConnectionID']; + if ($Splitter > 0) { + $NodeValues = $this->GetNodeConfigFormValues($Splitter); + } + $this->UpdateFormField('Config', 'values', json_encode($NodeValues)); + $this->UpdateFormField('RemoveNode', 'values', json_encode([])); + $this->UpdateFormField('GatewayCommands', 'visible', false); + } + } + + protected function UpdateFormField($Name, $Field, $Value) + { + $this->SendDebug('Form: ' . $Name . '.' . $Field, $Value, 0); + parent::UpdateFormField($Name, $Field, $Value); + } + + protected function SendDebug($Message, $Data, $Format) + { + if (is_a($Data, '\\KLF200\\APIData')) { + /* @var $Data \KLF200\APIData */ + $this->SendDebug2($Message . ':Command', \KLF200\APICommand::ToString($Data->Command), 0); + if ($Data->isError()) { + $this->SendDebug2('Error', $Data->ErrorToString(), 0); + } elseif ($Data->Data != '') { + $this->SendDebug2($Message . ':Data', $Data->Data, $Format); + } + } else { + $this->SendDebug2($Message, $Data, $Format); + } + } + + private function ReceiveEvent(\KLF200\APIData $APIData) + { + switch ($APIData->Command) { + case \KLF200\APICommand::CS_DISCOVER_NODES_NTF: + if (!$this->GetNodeInfoIsRunning) { + IPS_RunScriptText('IPS_RequestAction(' . $this->InstanceID . ',"GetAllNodesInformation",false);'); + } + break; + case \KLF200\APICommand::CS_SYSTEM_TABLE_UPDATE_NTF: + sleep(3); + if (!$this->GetNodeInfoIsRunning) { + IPS_RunScriptText('IPS_RequestAction(' . $this->InstanceID . ',"GetAllNodesInformation",false);'); + } + break; + case \KLF200\APICommand::GET_ALL_NODES_INFORMATION_NTF: + $NodeID = ord($APIData->Data[0]); + $Name = trim(substr($APIData->Data, 4, 64)); + $NodeTypeSubType = unpack('n', substr($APIData->Data, 69, 2))[1]; + $this->SendDebug('NodeID', $NodeID, 0); + $this->SendDebug('Name', $Name, 0); + $this->SendDebug('NodeTypeSubType', $NodeTypeSubType, 0); + $this->SendDebug('SerialNumber', substr($APIData->Data, 76, 8), 1); + $this->SendDebug('BuildNumber', ord($APIData->Data[75]), 0); + $Nodes = $this->Nodes; + $Nodes[$NodeID] = [ + 'Name' => $Name, + 'NodeTypeSubType' => $NodeTypeSubType + ]; + $this->Nodes = $Nodes; + break; + case \KLF200\APICommand::GET_ALL_NODES_INFORMATION_FINISHED_NTF: + $Splitter = IPS_GetInstance($this->InstanceID)['ConnectionID']; + $NodeValues = $this->GetNodeConfigFormValues($Splitter); + $this->UpdateFormField('Config', 'values', json_encode($NodeValues)); + $this->UpdateFormField('Config', 'visible', true); + $this->UpdateFormField('GatewayCommands', 'visible', true); + $this->UpdateFormField('ProgressLearn', 'visible', false); + $DeleteNodeValues = $this->GetDeleteNodeConfigFormValues(); + $this->UpdateFormField('RemoveNode', 'values', json_encode($DeleteNodeValues)); + $this->UpdateFormField('RemoveNode', 'visible', true); + $this->UpdateFormField('ProgressRemove', 'visible', false); + $this->GetNodeInfoIsRunning = false; + break; + } + } + + private function GetInstanceList(string $GUID, int $Parent, string $ConfigParam) + { + $InstanceIDList = []; + foreach (IPS_GetInstanceListByModuleID($GUID) as $InstanceID) { + // Fremde Geräte überspringen + if (IPS_GetInstance($InstanceID)['ConnectionID'] == $Parent) { + $InstanceIDList[] = $InstanceID; + } + } + if ($ConfigParam != '') { + $InstanceIDList = array_flip(array_values($InstanceIDList)); + array_walk($InstanceIDList, [$this, 'GetConfigParam'], $ConfigParam); + } + return $InstanceIDList; + } + + private function GetConfigParam(&$item1, $InstanceID, $ConfigParam) + { + $item1 = IPS_GetProperty($InstanceID, $ConfigParam); + } + + private function GetAllNodesInformation() + { + $this->Nodes = []; + + $APIData = new \KLF200\APIData(\KLF200\APICommand::GET_ALL_NODES_INFORMATION_REQ); + $ResultAPIData = $this->SendAPIData($APIData); + if ($ResultAPIData->isError()) { + return false; + } + $this->GetNodeInfoIsRunning = true; + return ord($ResultAPIData->Data[0]) == 1; + } + + /** + * Interne Funktion des SDK. + */ + private function GetNodeConfigFormValues(int $Splitter) + { + $FoundNodes = $this->Nodes; + $this->SendDebug('Found Nodes', $FoundNodes, 0); + $InstanceIDListNodes = $this->GetInstanceList('{4EBD07B1-2962-4531-AC5F-7944789A9CE5}', $Splitter, 'NodeId'); + $this->SendDebug('IPS Nodes', $InstanceIDListNodes, 0); + $NodeValues = []; + foreach ($FoundNodes as $NodeID => $Node) { + $InstanceIDNode = array_search($NodeID, $InstanceIDListNodes); + if ($InstanceIDNode !== false) { + $AddValue = [ + 'instanceID' => $InstanceIDNode, + 'nodeid' => $NodeID, + 'name' => IPS_GetName($InstanceIDNode), + 'type' => \KLF200\Node::$SubType[$Node['NodeTypeSubType']], + 'location' => stristr(IPS_GetLocation($InstanceIDNode), IPS_GetName($InstanceIDNode), true) + ]; + unset($InstanceIDListNodes[$InstanceIDNode]); + } else { + $AddValue = [ + 'instanceID' => 0, + 'nodeid' => $NodeID, + 'name' => $Node['Name'], + 'type' => \KLF200\Node::$SubType[$Node['NodeTypeSubType']], + 'location' => '' + ]; + } + $AddValue['create'] = [ + 'moduleID' => '{4EBD07B1-2962-4531-AC5F-7944789A9CE5}', + 'configuration' => ['NodeId' => $NodeID], + 'location' => ['Velux KLF200'] + ]; + + $NodeValues[] = $AddValue; + } + + foreach ($InstanceIDListNodes as $InstanceIDNode => $Node) { + $NodeValues[] = [ + 'instanceID' => $InstanceIDNode, + 'nodeid' => $Node, + 'name' => IPS_GetName($InstanceIDNode), + 'type' => 'unknown', + 'location' => stristr(IPS_GetLocation($InstanceIDNode), IPS_GetName($InstanceIDNode), true) + ]; + } + return $NodeValues; + } + + private function GetDeleteNodeConfigFormValues() + { + $NodeValues = []; + foreach ($this->Nodes as $NodeID => $Node) { + $AddValue = [ + 'nodeid' => $NodeID, + 'name' => $Node['Name'], + 'type' => \KLF200\Node::$SubType[$Node['NodeTypeSubType']] + ]; + $NodeValues[] = $AddValue; + } + + return $NodeValues; + } + + private function SendAPIData(\KLF200\APIData $APIData) + { + $this->SendDebug('ForwardData', $APIData, 1); + + try { + if (!$this->HasActiveParent()) { + throw new Exception($this->Translate('Instance has no active parent.'), E_USER_NOTICE); + } + /** @var \KLF200\APIData $ResponseAPIData */ + $ret = @$this->SendDataToParent($APIData->ToJSON('{7B0F87CC-0408-4283-8E0E-2D48141E42E8}')); + $ResponseAPIData = @unserialize($ret); + $this->SendDebug('Response', $ResponseAPIData, 1); + return $ResponseAPIData; + } catch (Exception $exc) { + $this->SendDebug('Error', $exc->getMessage(), 0); + return new \KLF200\APIData(\KLF200\APICommand::ERROR_NTF, chr(\KLF200\ErrorNTF::TIMEOUT)); + } + } +} diff --git a/KLF200Gateway/README.md b/KLF200Gateway/README.md new file mode 100644 index 0000000..2df763a --- /dev/null +++ b/KLF200Gateway/README.md @@ -0,0 +1,140 @@ +[![SDK](https://img.shields.io/badge/Symcon-PHPModul-red.svg?style=flat-square)](https://www.symcon.de/service/dokumentation/entwicklerbereich/sdk-tools/sdk-php/) +[![Version](https://img.shields.io/badge/Modul%20Version-0.80-blue.svg?style=flat-square)](https://community.symcon.de/t/modul-velux-klf200/50429) +[![Version](https://img.shields.io/badge/Symcon%20Version-5.5%20%3E-green.svg?style=flat-square)](https://www.symcon.de/service/dokumentation/installation/migrationen/v54-v55-q4-2020/) +[![License](https://img.shields.io/badge/License-CC%20BY--NC--SA%204.0-green.svg?style=flat-square)](https://creativecommons.org/licenses/by-nc-sa/4.0/) +[![Check Style](https://github.com/Nall-chan/VeluxKLF200/workflows/Check%20Style/badge.svg)](https://github.com/Nall-chan/VeluxKLF200/actions) [![Run Tests](https://github.com/Nall-chan/VeluxKLF200/workflows/Run%20Tests/badge.svg)](https://github.com/Nall-chan/VeluxKLF200/actions) +[![Spenden](https://www.paypalobjects.com/de_DE/DE/i/btn/btn_donate_SM.gif)](#3-spenden) +[![Wunschliste](https://img.shields.io/badge/Wunschliste-Amazon-ff69fb.svg)](#3-spenden) + +# Velux KLF200 Gateway + +## Inhaltsverzeichnis + +- [1. Funktionsumfang](#1-funktionsumfang) +- [2. Voraussetzungen](#2-voraussetzungen) +- [3. Software-Installation](#3-software-installation) +- [4. Einrichten der Instanzen in IP-Symcon](#4-einrichten-der-instanzen-in-ip-symcon) +- [5. Statusvariablen und Profile](#5-statusvariablen-und-profile) +- [6. WebFront](#6-webfront) +- [7. PHP-Befehlsreferenz](#7-php-befehlsreferenz) +- [8. Aktionen](#8-aktionen) +- [9. Anhang](#9-anhang) + - [1. Changelog](#1-changelog) + - [2. Spenden](#2-spenden) +- [10. Lizenz](#10-lizenz) + + +## 1. Funktionsumfang + + - Bindeglied zwischen Netzwerk und IPS-Instanzen vom Type Node und Konfigurator. + - Auslesen und darstellen von Zuständes des KLF200. + +## 2. Voraussetzungen + + - IPS ab Version 5.2 + - KLF200 io-homecontrol® Gateway + - KLF muss per LAN angeschlossen sein + - KLF Firmware 2.0.0.71 oder neuer + +## 3. Software-Installation + +* Dieses Modul ist Bestandteil der [VeluxKLF200-Library](../README.md#3-software-installation). + +## 4. Einrichten der Instanzen in IP-Symcon + +Eine einfache Einrichtung ist über die im Objektbaum unter `Konfigurator Instanzen` zu findene Instanz [KLF200 Konfigurator](../KLF200Configurator/README.md) möglich. + +Bei der manuellen Einrichtung ist das Modul im Dialog `Instanz hinzufügen` unter den Hersteller `VELUX` zu finden. +![Instanz hinzufügen](../imgs/instanzen.png) + +Alternativ ist es auch in der Liste alle Splitter aufgeführt. +![Instanz hinzufügen](../imgs/instanzen_splitter.png) + +In dem sich öffnenden Konfigurationsformular ist das Passwort einzutragen. +``Das Standardkennwort ist dasselbe wie das auf der Rückseite des KLF200 angegebene WLAN-Kennwort.`` + +## 5. Statusvariablen und Profile + +![Objektbaum Splitter](../imgs/logbaum_splitter.png) + +**Statusvariablen:** + +| Name | Typ | Ident | Beschreibung | +| :--------------- | :------ | :-------------- | :------------------------------------------------------ | +| Hardware Version | integer | HardwareVersion | Hardware-Version des KLF200 Gateways. | +| Firmware Version | string | FirmwareVersion | Firmware-Version des KLF200 Gateways. | +| Protocol Version | string | ProtocolVersion | Version des aktuell unterstützen Protokolls des KLF200. | + + +**Profile:** + +Dieses Modul erstellt keine Profile. + +## 6. WebFront + +Sollen die vorhandene Statusvariablen im WebFront angezeigt werden, so müssen diese verlinkt werden. + +## 7. PHP-Befehlsreferenz + +**Folgende Funktionen liefern 'TRUE' bei Erfolg. +Im Fehlerfall wird eine Warnung erzeugt und 'FALSE' zurückgegeben.** + +```php +bool KLF200_RequestGatewayVersion(int $InstanzeID); +``` +Liest die Hard- und Firmware Version des Gateways, und speichert das Ergebnis in den Statusvariablen. +Wird automatisch beim Verbinden mit dem Gateway ausgeführt. + +```php +bool KLF200_RequestProtocolVersion(int $InstanzeID); +``` +Liest die Protokoll Version des Gateways, und speichert das Ergebnis in der Statusvariable. +Wird automatisch beim Verbinden mit dem Gateway ausgeführt. + +```php +bool KLF200_SetGatewayTime(int $InstanzeID); +``` +Schreibt die UTC Zeit in das Gateway. +Wird automatisch beim Verbinden mit dem Gateway ausgeführt. + +```php +array KLF200_GetGatewayTime(int $InstanzeID); +``` +Liest die aktuelle Zeit aus dem Gateway, und liefert ein Array mit folgenden Feldern. +Es wird keine lokale Uhrzeit, und keine Sommerzeit unterstützt. Das bool-Feld 'DaylightSavingFlag' wird immer false sein. +```php + 'Timestamp', + 'Second', + 'Minute', + 'Hour', + 'DayOfMonth', + 'Month', + 'Year', + 'WeekDay', + 'DayOfYear', + 'DaylightSavingFlag' +``` + +## 8. Aktionen + +Es gibt keine speziellen Aktionen für dieses Modul. + +## 9. Anhang + +### 1. Changelog + +[Changelog der Library](../README.md#2-changelog) + +### 2. Spenden + + Die Library ist für die nicht kommerzielle Nutzung kostenlos, Schenkungen als Unterstützung für den Autor werden hier akzeptiert: + + + +[![Wunschliste](https://img.shields.io/badge/Wunschliste-Amazon-ff69fb.svg)](https://www.amazon.de/hz/wishlist/ls/YU4AI9AQT9F?ref_=wl_share) + + +## 10. Lizenz + + IPS-Modul: + [CC BY-NC-SA 4.0](https://creativecommons.org/licenses/by-nc-sa/4.0/) \ No newline at end of file diff --git a/KLF200Gateway/form.json b/KLF200Gateway/form.json new file mode 100644 index 0000000..57fe044 --- /dev/null +++ b/KLF200Gateway/form.json @@ -0,0 +1,73 @@ +{ + "elements": [ + { + "name": "Password", + "type": "PasswordTextBox", + "caption": "Password", + "validate": "^\\S{1,31}$" + } + ], + "actions": [ + { + "type": "Button", + "caption": "Request KLF version", + "onClick": "KLF200_RequestGatewayVersion($id);" + }, + { + "type": "Button", + "caption": "Request KLF state", + "onClick": "KLF200_ReadGatewayState($id,'ALL');" + }, + { + "type": "Label", + "caption": "This module is free for non-commercial use,\r\nDonations in support of the author are accepted here:" + }, + { + "type": "RowLayout", + "items": [ + { + "type": "Image", + "onClick": "echo 'https://www.paypal.com/donate?hosted_button_id=G2SLW2MEMQZH2';", + "image": "" + }, + { + "type": "Image", + "onClick": "echo 'https://www.amazon.de/hz/wishlist/ls/YU4AI9AQT9F?ref_=wl_share';", + "image": "" + } + ] + } + ], + "status": [ + { + "code": 102, + "icon": "active", + "caption": "Connected" + }, + { + "code": 104, + "icon": "inactive", + "caption": "Interface closed" + }, + { + "code": 201, + "icon": "error", + "caption": "Authentication error" + }, + { + "code": 202, + "icon": "error", + "caption": "Handshake error" + }, + { + "code": 203, + "icon": "error", + "caption": "Connection lost" + }, + { + "code": 204, + "icon": "error", + "caption": "Configuration error" + } + ] +} \ No newline at end of file diff --git a/KLF200Gateway/locale.json b/KLF200Gateway/locale.json new file mode 100644 index 0000000..5e2d70b --- /dev/null +++ b/KLF200Gateway/locale.json @@ -0,0 +1,38 @@ +{ + "translations": { + "de": { + "Active": "Aktiv", + "Password": "Passwort", + "Connected": "Verbunden", + "Interface closed": "Verbindung geschlossen", + "Connection lost": "Verbindung verloren", + "Handshake error": "Fehler im Handshake", + "Authentication error": "Authentifizierungsfehler", + "Request KLF version": "Lese KLF Version", + "Request KLF state": "Lese KLF Status", + "No error.": "Kein Fehler.", + "Timeout.": "Zeitüberschreitung.", + "Not further defined error.": "Nicht weiter definierter Fehler.", + "Unknown command or command is not accepted at this state.": "Unbekanntes Kommando oder Kommando wird zu diesem Zeitpunkt nicht akzeptiert.", + "Error on Frame Structure.": "Fehler in Frame Struktur.", + "Busy. Try again later.": "Beschäftigt. Später nochmal versuchen.", + "Bad system table index.": "Ungültiger Index der Systemtabelle.", + "Not authenticated.": "Nicht berechtigt.", + "Command Version": "Kommando Version", + "Protocol Version": "Protokoll Version", + "Send is blocked for: ": "Senden ist blockiert für: ", + "Socket not connected": "Socket nicht verbunden", + "Successfully connected to KLF200.": "Erfolgreich mit KLF200 verbunden.", + "Parameter is unable to execute.": "Parameter kann nicht ausgeführt werden.", + "Execution has failed.": "Die Ausführung ist fehlgeschlagen.", + "not used.": "nicht benutzt", + "waiting for power.": "Warten auf Strom.", + "Executing.": "Ausführen.", + "done.": "fertig.", + "unknown.": "unbekannt.", + "Execution is completed with no errors.": "Ausführung wurde ohne Fehler beendet.", + "Execution is still active.": "Die Ausführung ist noch aktiv.", + "This module is free for non-commercial use,\r\nDonations in support of the author are accepted here:": "Dieses Modul ist für die nicht kommerzielle Nutzung kostenlos,\r\nSchenkungen als Unterstützung für den Autor werden hier akzeptiert:" + } + } +} \ No newline at end of file diff --git a/KLF200Gateway/module.json b/KLF200Gateway/module.json new file mode 100644 index 0000000..e6d14f0 --- /dev/null +++ b/KLF200Gateway/module.json @@ -0,0 +1,21 @@ +{ + "id": "{725D4DF6-C8FC-463C-823A-D3481A3D7003}", + "name": "KLF200 Gateway", + "type": 2, + "vendor": "VELUX", + "aliases": [ + "KLF200 Gateway" + ], + "parentRequirements": [ + "{79827379-F36E-4ADA-8A95-5F8D1DC92FA9}" + ], + "childRequirements": [ + "{5242DAEF-EEBD-441F-AB0B-E83C01475B65}" + ], + "implemented": [ + "{018EF6B5-AB94-40C6-AA53-46943E824ACF}", + "{7B0F87CC-0408-4283-8E0E-2D48141E42E8}" + ], + "prefix": "KLF200", + "url": "https://github.com/Nall-chan/VeluxKLF200/blob/master/KLF200Gateway/README.md" +} \ No newline at end of file diff --git a/KLF200Gateway/module.php b/KLF200Gateway/module.php new file mode 100644 index 0000000..2c0e953 --- /dev/null +++ b/KLF200Gateway/module.php @@ -0,0 +1,549 @@ +' . file_get_contents(__DIR__ . '/../libs/helper/BufferHelper.php') . '}'); +eval('declare(strict_types=1);namespace KLF200Gateway {?>' . file_get_contents(__DIR__ . '/../libs/helper/SemaphoreHelper.php') . '}'); +eval('declare(strict_types=1);namespace KLF200Gateway {?>' . file_get_contents(__DIR__ . '/../libs/helper/DebugHelper.php') . '}'); +eval('declare(strict_types=1);namespace KLF200Gateway {?>' . file_get_contents(__DIR__ . '/../libs/helper/ParentIOHelper.php') . '}'); +eval('declare(strict_types=1);namespace KLF200Gateway {?>' . file_get_contents(__DIR__ . '/../libs/helper/VariableHelper.php') . '}'); + +/* + * @addtogroup klf200 + * @{ + * + * @package KLF200 + * @file module.php + * @author Michael Tröger + * @copyright 2020 Michael Tröger + * @license https://creativecommons.org/licenses/by-nc-sa/4.0/ CC BY-NC-SA 4.0 + * @version 1.0 + */ + +/** + * KLF200Gateway Klasse implementiert die KLF 200 API + * Erweitert IPSModule. + * + * @author Michael Tröger + * @copyright 2020 Michael Tröger + * @license https://creativecommons.org/licenses/by-nc-sa/4.0/ CC BY-NC-SA 4.0 + * + * @version 1.0 + * + * @example Ohne + * + * @property string $Host + * @property string $ReceiveBuffer + * @property APIData $ReceiveAPIData + * @property APIData $ReplyAPIData + * @property array $Nodes + * @property int $WaitForNodes + * @property int $SessionId + */ +class KLF200Gateway extends IPSModule +{ + use \KLF200Gateway\Semaphore, + \KLF200Gateway\BufferHelper, + \KLF200Gateway\DebugHelper, + \KLF200Gateway\VariableHelper, + \KLF200Gateway\InstanceStatus { + \KLF200Gateway\InstanceStatus::MessageSink as IOMessageSink; + \KLF200Gateway\InstanceStatus::RegisterParent as IORegisterParent; + \KLF200Gateway\InstanceStatus::RequestAction as IORequestAction; + \KLF200Gateway\DebugHelper::SendDebug as SendDebug2; + } + + /** + * Interne Funktion des SDK. + */ + public function Create() + { + parent::Create(); + $this->RequireParent('{3CFF0FD9-E306-41DB-9B5A-9D06D38576C3}'); + $this->RegisterPropertyString('Password', ''); + $this->RegisterTimer('KeepAlive', 0, 'KLF200_ReadGatewayState($_IPS[\'TARGET\']);'); + $this->Host = ''; + $this->ReceiveBuffer = ''; + $this->ReplyAPIData = null; + $this->Nodes = []; + $this->SessionId = 1; + $this->ParentID = 0; + } + + /** + * Interne Funktion des SDK. + */ + public function MessageSink($TimeStamp, $SenderID, $Message, $Data) + { + $this->IOMessageSink($TimeStamp, $SenderID, $Message, $Data); + switch ($Message) { + case IPS_KERNELSTARTED: + $this->KernelReady(); + break; + // case IPS_KERNELSHUTDOWN: + //$this->SendDisconnect(); + // Todo + // break; + } + } + + public function RequestAction($Ident, $Value) + { + if ($this->IORequestAction($Ident, $Value)) { + return true; + } + return false; + } + + /** + * Interne Funktion des SDK. + */ + public function GetConfigurationForParent() + { + $Config = [ + 'Port' => 51200, + 'UseSSL' => true, + 'VerifyPeer'=> false, + 'VerifyHost'=> true + ]; + return json_encode($Config); + } + + /** + * Interne Funktion des SDK. + */ + public function ApplyChanges() + { + $this->RegisterMessage(0, IPS_KERNELSTARTED); + $this->RegisterMessage(0, IPS_KERNELSHUTDOWN); + + $this->RegisterMessage($this->InstanceID, FM_CONNECT); + $this->RegisterMessage($this->InstanceID, FM_DISCONNECT); + + parent::ApplyChanges(); + $this->RegisterVariableString('FirmwareVersion', $this->Translate('Firmware Version'), '', 0); + $this->RegisterVariableInteger('HardwareVersion', $this->Translate('Hardware Version'), '', 0); + $this->RegisterVariableString('ProtocolVersion', $this->Translate('Protocol Version'), '', 0); + if (IPS_GetKernelRunlevel() == KR_READY) { + $this->KernelReady(); + } + + } + + public function ReadGatewayState() + { + $APIData = new \KLF200\APIData(\KLF200\APICommand::GET_STATE_REQ); + //$APIData = new \KLF200\APIData(\KLF200\APICommand::GET_SCENE_LIST_REQ); + $ResultAPIData = $this->SendAPIData($APIData); + //todo + // brauchen wir state? Oder substate? + /* + Command Data 1 Data 2 Data 3 – 6 + GW_GET_STATE_CFM GatewayState SubState StateData + + GatewayState value Description + 0 Test mode. + 1 Gateway mode, no actuator nodes in the system table. + 2 Gateway mode, with one or more actuator nodes in the system table. + 3 Beacon mode, not configured by a remote controller. + 4 Beacon mode, has been configured by a remote controller. + 5 - 255 Reserved. + + SubState value, when + GatewayState is 1 or 2 Description + 0x00 Idle state. + 0x01 Performing task in Configuration Service handler + 0x02 Performing Scene Configuration + 0x03 Performing Information Service Configuration. + 0x04 Performing Contact input Configuration. + 0x?? In Contact input Learn state. ??? + 0x80 Performing task in Command Handler + 0x81 Performing task in Activate Group Handler + 0x82 Performing task in Activate Scene Handler + */ + } + + public function RequestGatewayVersion() + { + $APIData = new \KLF200\APIData(\KLF200\APICommand::GET_VERSION_REQ); + $ResultAPIData = $this->SendAPIData($APIData); + if ($ResultAPIData->isError()) { + return false; + } + $this->SetValueString('FirmwareVersion', ord($ResultAPIData->Data[1]) . '.' . + ord($ResultAPIData->Data[2]) . '.' . + ord($ResultAPIData->Data[3]) . '.' . + ord($ResultAPIData->Data[4])); + $this->SetValueInteger('HardwareVersion', ord($ResultAPIData->Data[6])); + return true; + } + + public function RequestProtocolVersion() + { + $APIData = new \KLF200\APIData(\KLF200\APICommand::GET_PROTOCOL_VERSION_REQ); + $ResultAPIData = $this->SendAPIData($APIData); + if ($ResultAPIData->isError()) { + return false; + } + $this->SetValueString( + 'ProtocolVersion', + unpack('n', substr($ResultAPIData->Data, 0, 2))[1] . '.' . + unpack('n', substr($ResultAPIData->Data, 2, 2))[1] + ); + return true; + } + + public function SetGatewayTime() + { + $APIData = new \KLF200\APIData(\KLF200\APICommand::SET_UTC_REQ, pack('N', time())); + $ResultAPIData = $this->SendAPIData($APIData); + return !$ResultAPIData->isError(); + } + + public function GetGatewayTime() + { + $APIData = new \KLF200\APIData(\KLF200\APICommand::GET_LOCAL_TIME_REQ); + $ResultAPIData = $this->SendAPIData($APIData); + if ($ResultAPIData->isError()) { + return false; + } + $Result = [ + 'Timestamp' => unpack('N', substr($ResultAPIData->Data, 0, 4))[1], + 'Second' => ord($ResultAPIData->Data[4]), + 'Minute' => ord($ResultAPIData->Data[5]), + 'Hour' => ord($ResultAPIData->Data[6]), + 'DayOfMonth' => ord($ResultAPIData->Data[7]), + 'Month' => 1 + ord($ResultAPIData->Data[8]), + 'Year' => 1900 + unpack('n', substr($ResultAPIData->Data, 9, 2))[1], + 'WeekDay' => ord($ResultAPIData->Data[11]), + 'DayOfYear' => unpack('n', substr($ResultAPIData->Data, 12, 2))[1], + 'DaylightSavingFlag' => unpack('c', $ResultAPIData->Data[14])[1] + ]; + return $Result; + } + + //################# DATAPOINTS CHILDREN + + /** + * Interne Funktion des SDK. Nimmt Daten von Children entgegen und sendet Diese weiter. + * + * @param string $JSONString + * @result bool true wenn Daten gesendet werden konnten, sonst false. + */ + public function ForwardData($JSONString) + { + if ($this->GetStatus() != IS_ACTIVE) { + return serialize(new \KLF200\APIData(\KLF200\APICommand::ERROR_NTF, chr(\KLF200\ErrorNTF::TIMEOUT))); + } + $APIData = new \KLF200\APIData($JSONString); + $result = @$this->SendAPIData($APIData); + return serialize($result); + } + + //################# DATAPOINTS PARENT + + /** + * Empfängt Daten vom Parent. + * + * @param string $JSONString Das empfangene JSON-kodierte Objekt vom Parent. + * @result bool True wenn Daten verarbeitet wurden, sonst false. + */ + public function ReceiveData($JSONString) + { + $data = json_decode($JSONString); + $this->DecodeSLIPData(utf8_decode($data->Buffer)); + } + + public function RebootGateway() + { + $APIData = new \KLF200\APIData(\KLF200\APICommand::REBOOT_REQ); + $ResultAPIData = $this->SendAPIData($APIData); + return !$ResultAPIData->isError(); + } + + /** + * Wird ausgeführt wenn der Kernel hochgefahren wurde. + */ + protected function KernelReady() + { + $this->ReceiveBuffer = ''; + $this->ReplyAPIData = null; + $this->Nodes = []; + $this->RegisterParent(); + if ($this->HasActiveParent()) { + $this->IOChangeState(IS_ACTIVE); + } else { + $this->IOChangeState(IS_INACTIVE); + } + } + + protected function RegisterParent() + { + $IOId = $this->IORegisterParent(); + if ($IOId > 0) { + $this->Host = IPS_GetProperty($this->ParentID, 'Host'); + $this->SetSummary(IPS_GetProperty($IOId, 'Host')); + } else { + $this->Host = ''; + $this->SetSummary(('none')); + } + return $IOId; + } + + /** + * Wird über den Trait InstanceStatus ausgeführt wenn sich der Status des Parent ändert. + * Oder wenn sich die Zuordnung zum Parent ändert. + * + * @param int $State Der neue Status des Parent. + */ + protected function IOChangeState($State) + { + if ($State == IS_ACTIVE) { + if ($this->Connect()) { + $this->SetTimerInterval('KeepAlive', 600000); + $this->LogMessage($this->Translate('Successfully connected to KLF200.'), KL_NOTIFY); + $this->SessionId = 1; + $this->RequestProtocolVersion(); + $this->SetGatewayTime(); + $this->ReadGatewayState(); + $this->RequestGatewayVersion(); + $this->SetHouseStatusMonitor(); + } else { + $this->SetTimerInterval('KeepAlive', 0); + } + } else { + $this->SetTimerInterval('KeepAlive', 0); + $this->SetStatus(IS_INACTIVE); + } + } + + protected function SendDebug($Message, $Data, $Format) + { + if (is_a($Data, '\\KLF200\\APIData')) { + /* @var $Data \KLF200\APIData */ + $this->SendDebug2($Message . ':Command', \KLF200\APICommand::ToString($Data->Command), 0); + if ($Data->isError()) { + $this->SendDebug2('Error', $Data->ErrorToString(), 0); + } elseif ($Data->Data != '') { + $this->SendDebug2($Message . ':Data', $Data->Data, $Format); + } + } else { + $this->SendDebug2($Message, $Data, $Format); + } + } + + /* + public function GetSystemTable() + { + $APIData = new \KLF200\APIData(\KLF200\APICommand::CS_GET_SYSTEMTABLE_DATA_REQ); + $ResultAPIData = $this->SendAPIData($APIData); + //$this->lock('SendAPIData'); + // wait for finish + // 01 00 3A DC 1C 03 C0 1C 01 00 00 00 00 + //$this->unlock('SendAPIData'); + } + + public function GetSceneList() + { + $APIData = new \KLF200\APIData(\KLF200\APICommand::GET_SCENE_LIST_REQ); + $ResultAPIData = $this->SendAPIData($APIData); + } + */ + private function SetHouseStatusMonitor() + { + $APIData = new \KLF200\APIData(\KLF200\APICommand::HOUSE_STATUS_MONITOR_ENABLE_REQ); + $ResultAPIData = $this->SendAPIData($APIData); + return !$ResultAPIData->isError(); + } + + //################# PRIVATE + private function ReceiveEvent(\KLF200\APIData $APIData) + { + $this->SendAPIDataToChildren($APIData); + } + + private function Connect() + { + if (strlen($this->ReadPropertyString('Password')) > 31) { + $this->SetStatus(IS_EBASE + 4); + return false; + } + + $APIData = new \KLF200\APIData(\KLF200\APICommand::PASSWORD_ENTER_REQ, str_pad($this->ReadPropertyString('Password'), 32, "\x00")); + $ResultAPIData = $this->SendAPIData($APIData, false); + if ($ResultAPIData === false) { + $this->SetStatus(IS_EBASE + 2); + return false; + } + if ($ResultAPIData->isError()) { + $this->SetStatus(IS_EBASE + 3); + trigger_error($this->Translate($ResultAPIData->ErrorToString()), E_USER_NOTICE); + return false; + } + if ($ResultAPIData->Data != "\x00") { + $this->SendDebug('Login Error', '', 0); + $this->SetStatus(IS_EBASE + 1); + $this->LogMessage('Access denied', KL_ERROR); + return false; + } + $this->SendDebug('Login successfully', '', 0); + $this->SetStatus(IS_ACTIVE); + return true; + } + + /** + * Sendet die Events an die Children. + * + * @param \KLF200\APIData $APIData + */ + private function SendAPIDataToChildren(\KLF200\APIData $APIData) + { + $this->SendDataToChildren($APIData->ToJSON('{5242DAEF-EEBD-441F-AB0B-E83C01475B65}')); + } + + private function DecodeSLIPData($SLIPData) + { + $SLIPData = $this->ReceiveBuffer . $SLIPData; + $this->SendDebug('Input SLIP Data', $SLIPData, 1); + $Start = strpos($SLIPData, chr(0xc0)); + if ($Start === false) { + $this->SendDebug('ERROR', 'SLIP Start Marker not found', 0); + $this->ReceiveBuffer = ''; + return false; + } + if ($Start != 0) { + $this->SendDebug('WARNING', 'SLIP start is ' . $Start . ' and not 0', 0); + } + $End = strpos($SLIPData, chr(0xc0), 1); + if ($End === false) { + $this->SendDebug('WAITING', 'SLIP End Marker not found', 0); + $this->ReceiveBuffer = $SLIPData; + return false; + } + $TransportData = str_replace( + ["\xDB\xDC", "\xDB\xDD"], + ["\xC0", "\xDB"], + substr($SLIPData, $Start + 1, $End - $Start - 1) + ); + $Tail = substr($SLIPData, $End + 1); + $this->ReceiveBuffer = $Tail; + if (ord($TransportData[0]) != 0) { + $this->SendDebug('ERROR', 'Wrong ProtocolID', 0); + return false; + } + $len = ord($TransportData[1]) + 2; + if (strlen($TransportData) != $len) { + $this->SendDebug('ERROR', 'Wrong frame length', 0); + return false; + } + $Checksum = substr($TransportData, -1); + $ChecksumData = substr($TransportData, 0, -1); + //todo Checksum + $Command = unpack('n', substr($TransportData, 2, 2))[1]; + $Data = substr($TransportData, 4, $len - 5); + $APIData = new \KLF200\APIData($Command, $Data); + if ($APIData->isEvent()) { + $this->SendDebug('Event', $APIData, 1); + $this->ReceiveEvent($APIData); + } else { + $this->ReplyAPIData = $APIData; + } + if (strpos($Tail, chr(0xc0)) !== false) { + $this->SendDebug('Tail hast Start Marker', '', 0); + $this->DecodeSLIPData(''); + } + } + + /** + * Wartet auf eine Antwort einer Anfrage an den LMS. + * + * @param string $APICommand + * @result mixed + */ + private function ReadReplyAPIData() + { + for ($i = 0; $i < 2000; $i++) { + $Buffer = $this->ReplyAPIData; + if (!is_null($Buffer)) { + $this->ReplyAPIData = null; + return $Buffer; + } + usleep(1000); + } + return null; + } + + //################# SENDQUEUE + + private function GetSessionId() + { + $SessionId = ($this->SessionId + 1) & 0xffff; + $this->SessionId = $SessionId; + return pack('n', $SessionId); + } + + private function SendAPIData(\KLF200\APIData $APIData, bool $SetState = true) + { + //Statt SessionId benutzen wir einfach NodeID. + /* if (in_array($APIData->Command, [ + \KLF200\APICommand::COMMAND_SEND_REQ, + \KLF200\APICommand::STATUS_REQUEST_REQ, + \KLF200\APICommand::WINK_SEND_REQ, + \KLF200\APICommand::SET_LIMITATION_REQ, + \KLF200\APICommand::GET_LIMITATION_STATUS_REQ, + \KLF200\APICommand::MODE_SEND_REQ, + \KLF200\APICommand::ACTIVATE_SCENE_REQ, + \KLF200\APICommand::STOP_SCENE_REQ, + \KLF200\APICommand::ACTIVATE_PRODUCTGROUP_REQ + ])) { + $APIData->Data = $this->GetSessionId() . $APIData->Data; + } */ + try { + $this->SendDebug('Wait to send', $APIData, 1); + $time = microtime(true); + while (true) { + if ($this->lock('SendAPIData')) { + break; + } + if (microtime(true) - $time > 5) { + throw new Exception($this->Translate('Send is blocked for: ') . \KLF200\APICommand::ToString($APIData->Command), E_USER_ERROR); + } + } + if (!$this->HasActiveParent()) { + throw new Exception($this->Translate('Socket not connected'), E_USER_NOTICE); + } + $Data = $APIData->GetSLIPData(); + $this->SendDebug('Send', $APIData, 1); + $this->SendDebug('Send SLIP Data', $Data, 1); + $JSON['DataID'] = '{79827379-F36E-4ADA-8A95-5F8D1DC92FA9}'; + $JSON['Buffer'] = utf8_encode($Data); + $JsonString = json_encode($JSON); + $this->ReplyAPIData = null; + parent::SendDataToParent($JsonString); + $ResponseAPIData = $this->ReadReplyAPIData(); + + if ($ResponseAPIData === null) { + throw new Exception($this->Translate('Timeout.'), E_USER_NOTICE); + } + $this->SendDebug('Response', $ResponseAPIData, 1); + $this->unlock('SendAPIData'); + if ($ResponseAPIData->isError()) { + trigger_error($this->Translate($ResponseAPIData->ErrorToString()), E_USER_NOTICE); + } + return $ResponseAPIData; + } catch (Exception $exc) { + $this->SendDebug('Error', $exc->getMessage(), 0); + if ($exc->getCode() != E_USER_ERROR) { + $this->unlock('SendAPIData'); + } + trigger_error($this->Translate($exc->getMessage()), E_USER_NOTICE); + if ($SetState) { + $this->SetStatus(IS_EBASE + 3); + } + return new \KLF200\APIData(\KLF200\APICommand::ERROR_NTF, chr(\KLF200\ErrorNTF::TIMEOUT)); + } + } +} + +/* @} */ diff --git a/KLF200Node/README.md b/KLF200Node/README.md new file mode 100644 index 0000000..5f8c881 --- /dev/null +++ b/KLF200Node/README.md @@ -0,0 +1,260 @@ +[![SDK](https://img.shields.io/badge/Symcon-PHPModul-red.svg?style=flat-square)](https://www.symcon.de/service/dokumentation/entwicklerbereich/sdk-tools/sdk-php/) +[![Version](https://img.shields.io/badge/Modul%20Version-0.80-blue.svg?style=flat-square)](https://community.symcon.de/t/modul-velux-klf200/50429) +[![Version](https://img.shields.io/badge/Symcon%20Version-5.5%20%3E-green.svg?style=flat-square)](https://www.symcon.de/service/dokumentation/installation/migrationen/v54-v55-q4-2020/) +[![License](https://img.shields.io/badge/License-CC%20BY--NC--SA%204.0-green.svg?style=flat-square)](https://creativecommons.org/licenses/by-nc-sa/4.0/) +[![Check Style](https://github.com/Nall-chan/VeluxKLF200/workflows/Check%20Style/badge.svg)](https://github.com/Nall-chan/VeluxKLF200/actions) [![Run Tests](https://github.com/Nall-chan/VeluxKLF200/workflows/Run%20Tests/badge.svg)](https://github.com/Nall-chan/VeluxKLF200/actions) +[![Spenden](https://www.paypalobjects.com/de_DE/DE/i/btn/btn_donate_SM.gif)](#3-spenden) +[![Wunschliste](https://img.shields.io/badge/Wunschliste-Amazon-ff69fb.svg)](#3-spenden) + +# Velux KLF200 Node + +## Inhaltsverzeichnis + +- [1. Funktionsumfang](#1-funktionsumfang) +- [2. Voraussetzungen](#2-voraussetzungen) +- [3. Software-Installation](#3-software-installation) +- [4. Einrichten der Instanzen in IP-Symcon](#4-einrichten-der-instanzen-in-ip-symcon) +- [5. Statusvariablen und Profile](#5-statusvariablen-und-profile) +- [6. WebFront](#6-webfront) +- [7. PHP-Befehlsreferenz](#7-php-befehlsreferenz) + - [Allgemein](#allgemein) + - [Shutter](#shutter) + - [Slats](#slats) + - [Dimmer Light / Heating](#dimmer-light--heating) +- [8. Aktionen](#8-aktionen) +- [9. Anhang](#9-anhang) + - [1. Changelog](#1-changelog) + - [2. Spenden](#2-spenden) +- [10. Lizenz](#10-lizenz) + +## 1. Funktionsumfang + + - Auslesen, darstellen und Steuern von Zuständen eines Gerätes. + - Bereitstellung von PHP-Funktionen. + +## 2. Voraussetzungen + + - IPS ab Version 5.5 + - KLF200 io-homecontrol® Gateway + - KLF muss per LAN angeschlossen sein + - KLF Firmware 2.0.0.71 oder neuer + +## 3. Software-Installation + +* Dieses Modul ist Bestandteil der [VeluxKLF200-Library](../README.md#3-software-installation). + +## 4. Einrichten der Instanzen in IP-Symcon + +Eine einfache Einrichtung ist über die im Objektbaum unter `Konfigurator Instanzen` zu findene Instanz [KLF200 Konfigurator](../KLF200Configurator/README.md) möglich. + +Bei der manuellen Einrichtung ist das Modul im Dialog `Instanz hinzufügen` unter den Hersteller `VELUX` zu finden. +![Instanz hinzufügen](../imgs/instanzen.png) + +In dem sich öffnenden Konfigurationsformular ist die `Node ID` des Gerätes einzutragen. + +## 5. Statusvariablen und Profile + + +Die Statusvariablen werden je nach Geräte-Typ des Nodes angelegt oder auch wieder entfernt. +Es werden folgende Statusvariablen verwendet: 'MAIN', 'FP1', 'FP2' und 'FP3'. +Dabei entsprechen die Werte und Aktionen dem jeweiligen Funktions-Parameter, welche bei jedem Geräte-Typ (NodeSubType) unterschiedlich sind. +Entsprechend sind auch der Name und das Profil beim anlegen der Instanz je nach Geräte-Typ unterschiedlich. + +**Statusvariable MAIN:** + +| Name | Typ | Ident | Beschreibung | +| :-------------- | :------ | :---- | :-------------------------------------------------------- | +| Status | boolean | MAIN | Licht | +| Schalter | boolean | MAIN | Aktoren | +| Schloss | boolean | MAIN | Schlösser | +| Position | integer | MAIN | Alle Geräte welche eine prozentuale Position unterstützen | +| Intensität | integer | MAIN | Dimmbares Licht | +| Doppelrollladen | integer | MAIN | Nur bei Doppelrollladen vorhanden | +| Geschlossen | integer | MAIN | prozentuale Schließung von Lüftungen/Heizungen | + +**Statusvariable FP1:** + +| Name | Typ | Ident | Beschreibung | +| :------------- | :------ | :---- | :---------------------------------- | +| Orientierung | integer | FP1 | Ausrichtung von Lamellen in Prozent | +| Obere Position | integer | FP1 | Obere Position in Prozent | + +**Statusvariable FP2:** + +| Name | Typ | Ident | Beschreibung | +| :-------------- | :------ | :---- | :------------------------- | +| Untere Position | integer | FP2 | Untere Position in Prozent | + +**Statusvariable FP3:** + +| Name | Typ | Ident | Beschreibung | +| :----------- | :------ | :---- | :---------------------------------- | +| Orientierung | integer | FP3 | Ausrichtung von Lamellen in Prozent | + +**Profile**: + +| Name | Typ | verwendet von Statusvariablen (Ident) | +| :-------------------------- | :------ | :------------------------------------- | +| KLF200.Light.Reversed | boolean | MAIN | +| KLF200.Lock | boolean | MAIN | +| KLF200.Blind | integer | MAIN | +| KLF200.RollerShutter | integer | MAIN, FP1, FP2 | +| KLF200.Window | integer | MAIN | +| KLF200.Garage | integer | MAIN | +| KLF200.Light.51200.Reversed | integer | MAIN | +| KLF200.Intensity.51200 | integer | MAIN | +| KLF200.Slats | integer | FP1, FP3 | + + +## 6. WebFront + +Die direkte Darstellung im WebFront ist möglich, es wird aber empfohlen mit Links zu arbeiten. +Alle Statusvariablen sind auch bedienbar. + +## 7. PHP-Befehlsreferenz + +Alle Funktionen liefern `TRUE` wenn das Gateway den Empfang des Befehls bestätigt hat. +Anschließend führt das KLF200 die gewünschte Aktion aus. +Im Fehlerfall, oder wenn das Gateway den Befehl ablehnt, wird eine Warnung erzeugt und `FALSE` zurückgegeben. + +Alle `$Value` Variablen vom Typ integer (int) haben eine Wertebereich von 0 (`0x000`) bis 51200 (`0xC800`) für absolute Werte. + +Es kann bei allen `$Value` Variablen vom Typ integer (int) auch eine Relative Ansteuerung erfolgen. +Hierzu ist der Wertebereich von 51456 (`0xC900`) für -100% bis zu 53456 (`0xD0D0`) für +100% reserviert. + +--- + +### Allgemein + +```php +bool KLF200_RequestStatus(int $InstanzeID) +``` +Stellt eine Anfrage über den aktuellen Status des Node an das Gateway. + +```php +bool KLF200_RequestNodeInformation(int $InstanzeID) +``` +Stellt eine Anfrage über Informationen zum Node an das Gateway. + +```php +bool KLF200_SetMainParameter(int $InstanzeID, int $Value) +``` +Main Parameter auf `$Value` setzen. + +```php +bool KLF200_SetFunctionParameter1(int $InstanzeID, int $Value) +``` +Function Parameter 1 auf `$Value` setzen. + +```php +bool KLF200_SetFunctionParameter2(int $InstanzeID, int $Value) +``` +Function Parameter 2 auf `$Value` setzen. + +```php +bool KLF200_SetFunctionParameter3(int $InstanzeID, int $Value) +``` +Function Parameter 3 auf `$Value` setzen. + +```php +bool KLF200_SwitchMode(int $InstanzeID, bool $Value) +``` +Gerät wird auf den in `$Value` übergebenen Wert ein/ausgeschaltet bzw. komplett geschlossen/geöffnet. + +--- + +### Shutter + +Betrifft alle Gerätetypen, welche eine prozentuale Ansteuerung für Positionen bieten. + +```php +bool KLF200_ShutterMove(int $InstanzeID, int $Value) +``` +Gerät auf in den `$Value` übergebenen Wert setzen. + +```php +bool KLF200_ShutterMoveUp(int $InstanzeID) +``` +Öffnungsbefehl an das Gerät senden. + +```php +bool KLF200_ShutterMoveDown(int $InstanzeID) +``` +Schließbefehl an das Gerät senden. + +```php +bool KLF200_ShutterMoveStop(int $InstanzeID) +``` +Stop Befehl an das Gerät senden. + +--- + +### Slats + +```php +bool KLF200_OrientationSet(int $InstanzeID, int $Value) +``` +Öffnungswinkel auf den in `$Value` übergebenen Wert setzen. + +```php +bool KLF200_OrientationUp(int $InstanzeID) +``` +Öffnungswinkel auf 0 setzen. + +```php +bool KLF200_OrientationDown(int $InstanzeID) +``` +Öffnungswinkel auf maximum setzen. + +```php +bool KLF200_OrientationStop(int $InstanzeID) +``` +Stop Befehl an das Gerät senden. + +--- + +### Dimmer Light / Heating + +```php +bool KLF200_DimSet(int $InstanzeID, int $Value) +``` +Setzten den Dimmwert auf den in `$Value` übergebenen Wert. + +```php +bool KLF200_DimUp(int $InstanzeID) +``` +Setzten den Dimmwert auf das Maximum. + +```php +bool KLF200_DimDown(int $InstanzeID) +``` +Setzten den Dimmwert auf das Minimum. + +```php +bool KLF200_DimStop(int $InstanzeID) +``` +Stop Befehl an das Gerät senden. + +## 8. Aktionen + +Es gibt keine speziellen Aktionen für dieses Modul. + +## 9. Anhang + +### 1. Changelog + +[Changelog der Library](../README.md#2-changelog) + +### 2. Spenden + + Die Library ist für die nicht kommerzielle Nutzung kostenlos, Schenkungen als Unterstützung für den Autor werden hier akzeptiert: + + + +[![Wunschliste](https://img.shields.io/badge/Wunschliste-Amazon-ff69fb.svg)](https://www.amazon.de/hz/wishlist/ls/YU4AI9AQT9F?ref_=wl_share) + + +## 10. Lizenz + + IPS-Modul: + [CC BY-NC-SA 4.0](https://creativecommons.org/licenses/by-nc-sa/4.0/) \ No newline at end of file diff --git a/KLF200Node/form.json b/KLF200Node/form.json new file mode 100644 index 0000000..fac8095 --- /dev/null +++ b/KLF200Node/form.json @@ -0,0 +1,43 @@ +{ + "elements": [ + { + "name": "NodeId", + "type": "NumberSpinner", + "caption": "Node ID" + } + ], + "actions": [ + { + "type": "Button", + "caption": "Request device state", + "onClick": "KLF200_RequestStatus($id);" + }, + { + "type": "Button", + "caption": "Request device information", + "onClick": "KLF200_RequestNodeInformation($id);" + }, + { + "type": "TestCenter" + }, + { + "type": "Label", + "caption": "This module is free for non-commercial use,\r\nDonations in support of the author are accepted here:" + }, + { + "type": "RowLayout", + "items": [ + { + "type": "Image", + "onClick": "echo 'https://www.paypal.com/donate?hosted_button_id=G2SLW2MEMQZH2';", + "image": "" + }, + { + "type": "Image", + "onClick": "echo 'https://www.amazon.de/hz/wishlist/ls/YU4AI9AQT9F?ref_=wl_share';", + "image": "" + } + ] + } + ] +} \ No newline at end of file diff --git a/KLF200Node/locale.json b/KLF200Node/locale.json new file mode 100644 index 0000000..9a71092 --- /dev/null +++ b/KLF200Node/locale.json @@ -0,0 +1,41 @@ +{ + "translations": { + "de": { + "Request device state": "Lese Gerätestatus", + "Request device information": "Lese Geräte-Informationen", + "No error.": "Kein Fehler.", + "Timeout.": "Zeitüberschreitung.", + "Not further defined error.": "Nicht weiter definierter Fehler.", + "Unknown command or command is not accepted at this state.": "Unbekanntes Kommando oder Kommando wird zu diesem Zeitpunkt nicht akzeptiert.", + "Error on Frame Structure.": "Fehler in Frame Struktur.", + "Busy. Try again later.": "Beschäftigt. Später nochmal versuchen.", + "Bad system table index.": "Ungültiger Index der Systemtabelle.", + "Not authenticated.": "Nicht berechtigt.", + "Position": "Position", + "Orientation": "Orientierung", + "Intensity": "Intensität", + "State": "Status", + "Lock": "Schloss", + "Dual Roller Shutter": "Doppelrollladen", + "Switch": "Schalter", + "Upper position": "Obere Position", + "Lower position": "Untere Position", + "Closed": "Geschlossen", + "Instance does not implement this function": "Instanze unterstützt diese Funktion nicht", + "Invalid Ident": "Ungültiger Ident", + "Request rejected": "Anfrage abgelehnt", + "Invalid node index": "Ungültiger Node-Eintrag", + "Instance has no active parent.": "Instanz hat keinen aktiven Parent.", + "Parameter is unable to execute.": "Parameter kann nicht ausgeführt werden.", + "Execution has failed.": "Die Ausführung ist fehlgeschlagen.", + "not used.": "nicht benutzt", + "waiting for power.": "Warten auf Strom.", + "Executing.": "Ausführen.", + "done.": "fertig.", + "unknown.": "unbekannt.", + "Execution is completed with no errors.": "Ausführung wurde ohne Fehler beendet.", + "Execution is still active.": "Die Ausführung ist noch aktiv.", + "This module is free for non-commercial use,\r\nDonations in support of the author are accepted here:": "Dieses Modul ist für die nicht kommerzielle Nutzung kostenlos,\r\nSchenkungen als Unterstützung für den Autor werden hier akzeptiert:" + } + } +} \ No newline at end of file diff --git a/KLF200Node/module.json b/KLF200Node/module.json new file mode 100644 index 0000000..0c6a941 --- /dev/null +++ b/KLF200Node/module.json @@ -0,0 +1,18 @@ +{ + "id": "{4EBD07B1-2962-4531-AC5F-7944789A9CE5}", + "name": "KLF200 Node", + "type": 3, + "vendor": "VELUX", + "aliases": [ + "KLF200 Node" + ], + "parentRequirements": [ + "{7B0F87CC-0408-4283-8E0E-2D48141E42E8}" + ], + "childRequirements": [], + "implemented": [ + "{5242DAEF-EEBD-441F-AB0B-E83C01475B65}" + ], + "prefix": "KLF200", + "url": "https://github.com/Nall-chan/VeluxKLF200/blob/master/KLF200Node/README.md" +} \ No newline at end of file diff --git a/KLF200Node/module.php b/KLF200Node/module.php new file mode 100644 index 0000000..0deb3f8 --- /dev/null +++ b/KLF200Node/module.php @@ -0,0 +1,796 @@ +' . file_get_contents(__DIR__ . '/../libs/helper/BufferHelper.php') . '}'); +eval('declare(strict_types=1);namespace KLF200Node {?>' . file_get_contents(__DIR__ . '/../libs/helper/SemaphoreHelper.php') . '}'); +eval('declare(strict_types=1);namespace KLF200Node {?>' . file_get_contents(__DIR__ . '/../libs/helper/DebugHelper.php') . '}'); +eval('declare(strict_types=1);namespace KLF200Node {?>' . file_get_contents(__DIR__ . '/../libs/helper/VariableHelper.php') . '}'); +eval('declare(strict_types=1);namespace KLF200Node {?>' . file_get_contents(__DIR__ . '/../libs/helper/VariableProfileHelper.php') . '}'); + +/** + * @property char $NodeId + * @property int $SessionId + * @property int $NodeSubType + */ +class KLF200Node extends IPSModule +{ + use \KLF200Node\Semaphore, + \KLF200Node\BufferHelper, + \KLF200Node\VariableHelper, + \KLF200Node\VariableProfileHelper, + \KLF200Node\DebugHelper { + \KLF200Node\DebugHelper::SendDebug as SendDebug2; + } + + /** + * Interne Funktion des SDK. + */ + public function Create() + { + parent::Create(); + $this->ConnectParent('{725D4DF6-C8FC-463C-823A-D3481A3D7003}'); + $this->RegisterPropertyInteger('NodeId', -1); + $this->RegisterAttributeInteger('NodeSubType', -1); + $this->SessionId = 1; + $this->NodeSubType = -1; + } + + /** + * Interne Funktion des SDK. + */ + public function ApplyChanges() + { + parent::ApplyChanges(); + $APICommands = [ + \KLF200\APICommand::GET_NODE_INFORMATION_NTF, + \KLF200\APICommand::NODE_INFORMATION_CHANGED_NTF, + \KLF200\APICommand::NODE_STATE_POSITION_CHANGED_NTF, + \KLF200\APICommand::COMMAND_RUN_STATUS_NTF, + \KLF200\APICommand::COMMAND_REMAINING_TIME_NTF, + \KLF200\APICommand::SESSION_FINISHED_NTF, + \KLF200\APICommand::STATUS_REQUEST_NTF, + \KLF200\APICommand::WINK_SEND_NTF, + \KLF200\APICommand::MODE_SEND_NTF + ]; + $this->SessionId = 1; + $NodeId = $this->ReadPropertyInteger('NodeId'); + $this->NodeId = chr($NodeId); + if (($NodeId < 0) || ($NodeId > 255)) { + $Line = 'NOTHING'; + } else { + $NodeId = preg_quote(substr(json_encode(utf8_encode(chr($this->ReadPropertyInteger('NodeId')))), 0, -1)); + foreach ($APICommands as $APICommand) { + $Lines[] = '.*"Command":' . $APICommand . ',"Data":' . $NodeId . '.*'; + } + $Line = implode('|', $Lines); + } + $this->SetReceiveDataFilter('(' . $Line . ')'); + $this->SendDebug('FILTER', $Line, 0); + $this->NodeSubType = $this->ReadAttributeInteger('NodeSubType'); + $this->SetSummary(sprintf('%04X', $this->NodeSubType)); + $this->RegisterProfileInteger('KLF200.Intensity.51200', '', '', ' %', 0, 0xC800, 1); + $this->RegisterProfileInteger('KLF200.RollerShutter', 'Jalousie', '', ' %', 0, 0xC800, 1); + $this->RegisterProfileInteger('KLF200.Slats', 'Speedo', '', ' %', 0, 0xC800, 1); + $this->RegisterProfileInteger('KLF200.Blind', 'Raffstore', '', ' %', 0, 0xC800, 1); + $this->RegisterProfileInteger('KLF200.Window', 'Window', '', ' %', 0, 0xC800, 1); + $this->RegisterProfileInteger('KLF200.Heating.Reversed', 'Temperature', '', ' %', 0, 0xC800, 1); + $this->RegisterProfileInteger('KLF200.Garage', 'Garage', '', ' %', 0, 0xC800, 1); + $this->RegisterProfileInteger('KLF200.Light.51200.Reversed', 'Light', '', ' %', 0, 0xC800, 1); + $this->RegisterProfileBoolean('KLF200.Light.Reversed', 'Light', '', ''); + $this->RegisterProfileBoolean('KLF200.Lock', 'Lock', '', ''); + if (IPS_GetKernelRunlevel() == KR_READY) { + $this->RequestNodeInformation(); + } + } + + public function SwitchMode(bool $Value) + { + return $this->SetMainParameter($Value ? 0xC800 : 0x0000); + } + + public function ShutterMove(int $Value) + { + switch ($this->NodeSubType) { + case 0x0040: //Interior Venetian Blind + case 0x0080: //Roller Shutter + case 0x0081: //Adjustable slats rolling shutter + case 0x0082: //Roller Shutter With projection + case 0x00C0: //Vertical Exterior Awning + case 0x0100: //Window opener + case 0x0101: //Window opener with integrated rain sensor + case 0x0140: //Garage door opener + case 0x017A: //Garage door opener + case 0x0200: //Rolling Door Opener + case 0x01C0: //Gate opener + case 0x01FA: //Gate opener + case 0x0280: //Vertical Interior Blinds + case 0x0340: //Dual Roller Shutter + case 0x0400: //Horizontal awning + case 0x04C0: //Curtain track + case 0x0600: //Swinging Shutters + case 0x0601: //Swinging Shutter with independent handling of the leaves + case 0x0440: //Exterior Venetian blind + case 0x0480: //Louver blind + case 0x0500: //Ventilation point + case 0x0501: //Air inlet + case 0x0502: //Air transfer + case 0x0503: //Air outlet + return $this->SetMainParameter($Value); + } + trigger_error($this->Translate('Instance does not implement this function'), E_USER_NOTICE); + return false; + } + + public function ShutterMoveUp() + { + return $this->SetMainParameter(0x0000); + } + + public function ShutterMoveDown() + { + return $this->SetMainParameter(0xC800); + } + + public function ShutterMoveStop() + { + return $this->SetMainParameter(0xD200); + } + + public function OrientationSet(int $Value) + { + switch ($this->NodeSubType) { + case 0x0040: + return $this->SetFunctionParameter1($Value); + case 0x0440: + case 0x0081: + case 0x0480: + return $this->SetFunctionParameter3($Value); + } + trigger_error($this->Translate('Instance does not implement this function'), E_USER_NOTICE); + return false; + } + + public function OrientationUp() + { + return $this->SetOrientation(0x0000); + } + + public function OrientationDown() + { + return $this->SetOrientation(0xC800); + } + + public function OrientationStop() + { + return $this->SetOrientation(0xD200); + } + + public function DimSet(int $Value) + { + switch ($this->NodeSubType) { + case 0x0180: //Light + case 0x0540: //Exterior heating + case 0x057A: //Exterior heating + return $this->SetMainParameter($Value); + } + trigger_error($this->Translate('Instance does not implement this function'), E_USER_NOTICE); + return false; + } + + public function DimUp() + { + return $this->DimSet(0x0000); + } + + public function DimDown() + { + return $this->DimSet(0xC800); + } + + public function DimStop() + { + return $this->DimSet(0xD200); + } + + public function RequestAction($Ident, $Value) + { + if (IPS_GetVariable($this->GetIDForIdent($Ident))['VariableType'] == VARIABLETYPE_BOOLEAN) { + $Value = $Value ? 0xC800 : 0x0000; + } + switch ($Ident) { + case 'MAIN': + return $this->SetMainParameter($Value); + case 'FP1': + return $this->SetFunctionParameter1($Value); + case 'FP2': + return $this->SetFunctionParameter2($Value); + case 'FP3': + return $this->SetFunctionParameter3($Value); + } + echo $this->Translate('Invalid Ident'); + return; + } + + public function RequestNodeInformation() + { + $APIData = new \KLF200\APIData(\KLF200\APICommand::GET_NODE_INFORMATION_REQ, $this->NodeId); + $ResultAPIData = $this->SendAPIData($APIData); + if ($ResultAPIData === null) { + return false; + } + $State = ord($ResultAPIData->Data[0]); + switch ($State) { + case 0: + return true; + case 1: + trigger_error($this->Translate('Request rejected'), E_USER_NOTICE); + return false; + case 2: + trigger_error($this->Translate('Invalid node index'), E_USER_NOTICE); + return false; + } + } + + public function RequestStatus() + { + /* + Command Data 1 – 2 Data 3 Data 4 – 23 Data 24 + GW_STATUS_REQUEST_REQ SessionID IndexArrayCount IndexArray StatusType + Data 25 Data 26 + FPI1 FPI2 + StatusType + value |Description + 0 |Request Target position + 1 |Request Current position + 2 |Request Remaining time + 3 |Request Main info. + */ + $Data = $this->NodeId . $this->GetSessionId() . chr(1) . $this->NodeId . "\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"; + $Data .= chr(1) . chr(0b11100000) . chr(0); + $APIData = new \KLF200\APIData(\KLF200\APICommand::STATUS_REQUEST_REQ, $Data); + $ResultAPIData = $this->SendAPIData($APIData); + if ($ResultAPIData === null) { + return false; + } + return ord($ResultAPIData->Data[2]) == 1; + } + + public function SetMainParameter(int $Value) + { + /* + Command Data 1 – 2 Data 3 Data 4 Data 5 + GW_COMMAND_SEND_REQ SessionID CommandOriginator PriorityLevel ParameterActive + Data 6 Data 7 Data 8 - 41 Data 42 Data 43 – 62 Data 63 + FPI1 FPI2 FunctionalParameterValueArray IndexArrayCount IndexArray PriorityLevelLock + Data 64 Data 65 Data 66 + PL_0_3 PL_4_7 LockTime + */ + $Data = $this->NodeId . $this->GetSessionId(); //Data 1-2 + $Data .= chr(1) . chr(3) . chr(0); // Data 3-5 + $Data .= chr(0) . chr(0); // Data 6-7 + $Data .= pack('n', $Value); // Data 8-9 + $Data .= "\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"; // Data 10-25 + $Data .= "\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"; // Data 26-41 + $Data .= chr(1) . $this->NodeId . "\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"; //Data 42-62 + $Data .= chr(0); // Data 63 + $Data .= chr(0) . chr(0) . chr(0); // Data 64-66 + $APIData = new \KLF200\APIData(\KLF200\APICommand::COMMAND_SEND_REQ, $Data); + $ResultAPIData = $this->SendAPIData($APIData); + if ($ResultAPIData === null) { + return false; + } + return ord($ResultAPIData->Data[2]) == 1; + } + + public function SetFunctionParameter1(int $Value) + { + /* + Command Data 1 – 2 Data 3 Data 4 Data 5 + GW_COMMAND_SEND_REQ SessionID CommandOriginator PriorityLevel ParameterActive + Data 6 Data 7 Data 8 - 41 Data 42 Data 43 – 62 Data 63 + FPI1 FPI2 FunctionalParameterValueArray IndexArrayCount IndexArray PriorityLevelLock + Data 64 Data 65 Data 66 + PL_0_3 PL_4_7 LockTime + */ + $Data = $this->NodeId . $this->GetSessionId(); //Data 1-2 + $Data .= chr(1) . chr(3) . chr(1); // Data 3-5 + $Data .= chr(0x80) . chr(0); // Data 6-7 + $Data .= "\xD4\x00"; // Data 8-9 -> ignore + $Data .= pack('n', $Value); // Data 10-11 + $Data .= "\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"; // Data 12-25 + $Data .= "\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"; // Data 26-41 + $Data .= chr(1) . $this->NodeId . "\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"; //Data 42-62 + $Data .= chr(0); // Data 63 + $Data .= chr(0) . chr(0) . chr(0); // Data 64-66 + $APIData = new \KLF200\APIData(\KLF200\APICommand::COMMAND_SEND_REQ, $Data); + $ResultAPIData = $this->SendAPIData($APIData); + if ($ResultAPIData === null) { + return false; + } + return ord($ResultAPIData->Data[2]) == 1; + } + + public function SetFunctionParameter2(int $Value) + { + /* + Command Data 1 – 2 Data 3 Data 4 Data 5 + GW_COMMAND_SEND_REQ SessionID CommandOriginator PriorityLevel ParameterActive + Data 6 Data 7 Data 8 - 41 Data 42 Data 43 – 62 Data 63 + FPI1 FPI2 FunctionalParameterValueArray IndexArrayCount IndexArray PriorityLevelLock + Data 64 Data 65 Data 66 + PL_0_3 PL_4_7 LockTime + */ + $Data = $this->NodeId . $this->GetSessionId(); //Data 1-2 + $Data .= chr(1) . chr(3) . chr(2); // Data 3-5 + $Data .= chr(0x40) . chr(0); // Data 6-7 + $Data .= "\xD4\x00"; // Data 8-9 -> ignore + $Data .= "\x00\x00"; // Data 10-11 + $Data .= pack('n', $Value); // Data 12-13 + $Data .= "\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"; // Data 14-25 + $Data .= "\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"; // Data 26-41 + $Data .= chr(1) . $this->NodeId . "\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"; //Data 42-62 + $Data .= chr(0); // Data 63 + $Data .= chr(0) . chr(0) . chr(0); // Data 64-66 + $APIData = new \KLF200\APIData(\KLF200\APICommand::COMMAND_SEND_REQ, $Data); + $ResultAPIData = $this->SendAPIData($APIData); + if ($ResultAPIData === null) { + return false; + } + return ord($ResultAPIData->Data[2]) == 1; + } + + public function SetFunctionParameter3(int $Value) + { + /* + Command Data 1 – 2 Data 3 Data 4 Data 5 + GW_COMMAND_SEND_REQ SessionID CommandOriginator PriorityLevel ParameterActive + Data 6 Data 7 Data 8 - 41 Data 42 Data 43 – 62 Data 63 + FPI1 FPI2 FunctionalParameterValueArray IndexArrayCount IndexArray PriorityLevelLock + Data 64 Data 65 Data 66 + PL_0_3 PL_4_7 LockTime + */ + $Data = $this->NodeId . $this->GetSessionId(); //Data 1-2 + $Data .= chr(1) . chr(3) . chr(0); // Data 3-5 + $Data .= chr(0x20) . chr(0); // Data 6-7 + $Data .= "\xD4\x00"; // Data 8-9 -> ignore + $Data .= "\x00\x00\x00\x00"; // Data 10-14 + $Data .= pack('n', $Value); // Data 14-15 + $Data .= "\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"; // Data 16-25 + $Data .= "\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"; // Data 26-41 + $Data .= chr(1) . $this->NodeId . "\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"; //Data 42-62 + $Data .= chr(0); // Data 63 + $Data .= chr(0) . chr(0) . chr(0); // Data 64-66 + $APIData = new \KLF200\APIData(\KLF200\APICommand::COMMAND_SEND_REQ, $Data); + $ResultAPIData = $this->SendAPIData($APIData); + if ($ResultAPIData === null) { + return false; + } + return ord($ResultAPIData->Data[2]) == 1; + } + + public function ReceiveData($JSONString) + { + $APIData = new \KLF200\APIData($JSONString); + $this->SendDebug('Event', $APIData, 1); + $this->ReceiveEvent($APIData); + } + + protected function SendDebug($Message, $Data, $Format) + { + if (is_a($Data, '\\KLF200\\APIData')) { + /* @var $Data \KLF200\APIData */ + $this->SendDebug2($Message . ':Command', \KLF200\APICommand::ToString($Data->Command), 0); + if ($Data->isError()) { + $this->SendDebug2('Error', $Data->ErrorToString(), 0); + } elseif ($Data->Data != '') { + $this->SendDebug2($Message . ':Data', $Data->Data, $Format); + } + } else { + $this->SendDebug2($Message, $Data, $Format); + } + } + + private function RegisterNodeVariables(int $NodeTypeSubType) + { + $this->NodeSubType = $NodeTypeSubType; + switch ($NodeTypeSubType) { + case 0x0040: //Interior Venetian Blind + $this->RegisterVariableInteger('MAIN', $this->Translate('Position'), 'KLF200.Blind', 0); + $this->RegisterVariableInteger('FP1', $this->Translate('Orientation'), 'KLF200.Slats', 0); + $this->UnregisterVariable('FP2'); + $this->UnregisterVariable('FP3'); + break; + case 0x0080: //Roller Shutter + case 0x0082: //Roller Shutter With projection + $this->RegisterVariableInteger('MAIN', $this->Translate('Position'), 'KLF200.RollerShutter', 0); + $this->UnregisterVariable('FP1'); + $this->UnregisterVariable('FP2'); + $this->UnregisterVariable('FP3'); + break; + case 0x0081: //Adjustable slats rolling shutter + $this->RegisterVariableInteger('MAIN', $this->Translate('Position'), 'KLF200.RollerShutter', 0); + $this->UnregisterVariable('FP1'); + $this->UnregisterVariable('FP2'); + $this->RegisterVariableInteger('FP3', $this->Translate('Orientation'), 'KLF200.Slats', 0); + break; + case 0x00C0: //Vertical Exterior Awning + $this->RegisterVariableInteger('MAIN', $this->Translate('Position'), 'KLF200.RollerShutter', 0); + $this->UnregisterVariable('FP1'); + $this->UnregisterVariable('FP2'); + $this->UnregisterVariable('FP3'); + break; + case 0x0100: //Window opener + case 0x0101: //Window opener with integrated rain sensor + $this->RegisterVariableInteger('MAIN', $this->Translate('Position'), 'KLF200.Window', 0); + $this->UnregisterVariable('FP1'); + $this->UnregisterVariable('FP2'); + $this->UnregisterVariable('FP3'); + break; + case 0x0140: //Garage door opener + case 0x017A: //Garage door opener + case 0x0200: //Rolling Door Opener + $this->RegisterVariableInteger('MAIN', $this->Translate('Position'), 'KLF200.Garage', 0); + $this->UnregisterVariable('FP1'); + $this->UnregisterVariable('FP2'); + $this->UnregisterVariable('FP3'); + break; + case 0x0180: //Light + $this->RegisterVariableInteger('MAIN', $this->Translate('Intensity'), 'KLF200.Light.51200.Reversed', 0); + $this->UnregisterVariable('FP1'); + $this->UnregisterVariable('FP2'); + $this->UnregisterVariable('FP3'); + break; + case 0x01BA: //Light only supporting on/off + $this->RegisterVariableBoolean('MAIN', $this->Translate('State'), 'KLF200.Light.Reversed', 0); + $this->UnregisterVariable('FP1'); + $this->UnregisterVariable('FP2'); + $this->UnregisterVariable('FP3'); + break; + case 0x01C0: //Gate opener + case 0x01FA: //Gate opener + $this->RegisterVariableInteger('MAIN', $this->Translate('Position'), 'KLF200.Intensity.51200', 0); + $this->UnregisterVariable('FP1'); + $this->UnregisterVariable('FP2'); + $this->UnregisterVariable('FP3'); + break; + case 0x0240: //Door lock + case 0x0241: //Window lock + $this->RegisterVariableBoolean('MAIN', $this->Translate('Lock'), 'KLF200.Lock', 0); + $this->UnregisterVariable('FP1'); + $this->UnregisterVariable('FP2'); + $this->UnregisterVariable('FP3'); + break; + case 0x0280: //Vertical Interior Blinds + $this->RegisterVariableInteger('MAIN', $this->Translate('Position'), 'KLF200.Blind', 0); + $this->UnregisterVariable('FP1'); + $this->UnregisterVariable('FP2'); + $this->UnregisterVariable('FP3'); + break; + case 0x0340: //Dual Roller Shutter + $this->RegisterVariableInteger('MAIN', $this->Translate('Dual Roller Shutter'), 'KLF200.RollerShutter', 0); + $this->RegisterVariableInteger('FP1', $this->Translate('Upper position'), 'KLF200.RollerShutter', 0); + $this->RegisterVariableInteger('FP2', $this->Translate('Lower position'), 'KLF200.RollerShutter', 0); + $this->UnregisterVariable('FP3'); + break; + case 0x03C0: //On/Off switch + $this->RegisterVariableBoolean('MAIN', $this->Translate('Switch'), '~Switch', 0); + $this->UnregisterVariable('FP1'); + $this->UnregisterVariable('FP2'); + $this->UnregisterVariable('FP3'); + break; + case 0x0400: //Horizontal awning + case 0x04C0: //Curtain track + case 0x0600: //Swinging Shutters + case 0x0601: //Swinging Shutter with independent handling of the leaves + $this->RegisterVariableInteger('MAIN', $this->Translate('Position'), 'KLF200.Intensity.51200', 0); + $this->UnregisterVariable('FP1'); + $this->UnregisterVariable('FP2'); + $this->UnregisterVariable('FP3'); + break; + case 0x0440: //Exterior Venetian blind + $this->RegisterVariableInteger('MAIN', $this->Translate('Position'), 'KLF200.Blind', 0); + $this->UnregisterVariable('FP1'); + $this->UnregisterVariable('FP2'); + $this->RegisterVariableInteger('FP3', $this->Translate('Orientation'), 'KLF200.Slats', 0); + break; + case 0x0480: //Louver blind + $this->RegisterVariableInteger('MAIN', $this->Translate('Position'), 'KLF200.Intensity.51200', 0); + $this->UnregisterVariable('FP1'); + $this->UnregisterVariable('FP2'); + $this->RegisterVariableInteger('FP3', $this->Translate('Orientation'), 'KLF200.Slats', 0); + break; + case 0x0500: //Ventilation point + case 0x0501: //Air inlet + case 0x0502: //Air transfer + case 0x0503: //Air outlet + $this->RegisterVariableInteger('MAIN', $this->Translate('Closed'), 'KLF200.Intensity.51200', 0); + $this->UnregisterVariable('FP1'); + $this->UnregisterVariable('FP2'); + $this->UnregisterVariable('FP3'); + break; + case 0x0540: //Exterior heating + case 0x057A: //Exterior heating + $this->RegisterVariableInteger('MAIN', $this->Translate('Closed'), 'KLF200.Intensity.51200', 0); + $this->UnregisterVariable('FP1'); + $this->UnregisterVariable('FP2'); + $this->UnregisterVariable('FP3'); + break; + case 0x0300: //Beacon + case 0x0380: //Heating Temperature Interface + case 0x0580: //Heat pump + case 0x05C0: //Intrusion alarm + default: + $this->UnregisterVariable('MAIN'); + $this->UnregisterVariable('FP1'); + $this->UnregisterVariable('FP2'); + $this->UnregisterVariable('FP3'); + break; + } + if (@$this->GetIDForIdent('MAIN') > 0) { + $this->EnableAction('MAIN'); + } + if (@$this->GetIDForIdent('FP1') > 0) { + $this->EnableAction('FP1'); + } + if (@$this->GetIDForIdent('FP2') > 0) { + $this->EnableAction('FP2'); + } + if (@$this->GetIDForIdent('FP3') > 0) { + $this->EnableAction('FP3'); + } + } + + private function SetValues(int $CurrentPosition, int $FP1CurrentPosition, int $FP2CurrentPosition, int $FP3CurrentPosition) + { + // nur absolute Werte in Variablen schreiben + $Main = @$this->GetIDForIdent('MAIN'); + if (($Main > 0) && ($CurrentPosition <= 0xC800)) { + if (IPS_GetVariable($Main)['VariableType'] == VARIABLETYPE_BOOLEAN) { + $CurrentPosition = ($CurrentPosition == 0xC800); + } + $this->SetValue('MAIN', $CurrentPosition); + } + $FP1 = @$this->GetIDForIdent('FP1'); + if (($FP1 > 0) && ($FP1CurrentPosition <= 0xC800)) { + if (IPS_GetVariable($FP1)['VariableType'] == VARIABLETYPE_BOOLEAN) { + $FP1CurrentPosition = ($FP1CurrentPosition == 0xC800); + } + $this->SetValue('FP1', $FP1CurrentPosition); + } + $FP2 = @$this->GetIDForIdent('FP2'); + if (($FP2 > 0) && ($FP2CurrentPosition <= 0xC800)) { + if (IPS_GetVariable($FP2)['VariableType'] == VARIABLETYPE_BOOLEAN) { + $FP2CurrentPosition = ($FP2CurrentPosition == 0xC800); + } + $this->SetValue('FP2', $FP2CurrentPosition); + } + $FP3 = @$this->GetIDForIdent('FP3'); + if (($FP3 > 0) && ($FP3CurrentPosition <= 0xC800)) { + if (IPS_GetVariable($FP3)['VariableType'] == VARIABLETYPE_BOOLEAN) { + $FP3CurrentPosition = ($FP3CurrentPosition == 0xC800); + } + $this->SetValue('FP3', $FP3CurrentPosition); + } + } + + private function ReceiveEvent(\KLF200\APIData $APIData) + { + switch ($APIData->Command) { + case \KLF200\APICommand::GET_NODE_INFORMATION_NTF: + $NodeID = ord($APIData->Data[0]); + /* Data 1 Data 2 - 3 Data 4 Data 5 - 68 Data 69 + NodeID Order Placement Name Velocity + Data 70 - 71 Data 72 Data 73 Data 74 Data 75 Data 76 + NodeTypeSubType ProductGroup ProductType NodeVariation PowerMode BuildNumber + Data 77 - 84 Data 85 Data 86 - 87 Data 88 - 89 Data 90 - 91 Data 92 - 93 + SerialNumber State CurrentPosition Target FP1CurrentPosition FP2CurrentPosition + Data 94 - 95 Data 96 - 97 Data 98 - 99 Data 100 - 103 Data 104 Data 105 - 125 + FP3CurrentPosition FP4CurrentPosition RemainingTime TimeStamp NbrOfAlias AliasArray + */ + + $Name = trim(substr($APIData->Data, 4, 64)); + $NodeTypeSubType = unpack('n', substr($APIData->Data, 69, 2))[1]; + $this->SendDebug('NodeID', $NodeID, 0); + $this->SendDebug('Name', $Name, 0); + $this->SendDebug('NodeTypeSubType', sprintf('%04X', $NodeTypeSubType), 0); + $this->SendDebug('NodeTypeSubType', \KLF200\Node::$SubType[$NodeTypeSubType], 0); + $this->SendDebug('NodeType', ($NodeTypeSubType >> 6), 0); + $this->SendDebug('SubType', ($NodeTypeSubType & 0x003F), 0); + /* + $this->SendDebug('ProductGroup', ord($APIData->Data[71]), 0); + $this->SendDebug('ProductType', ord($APIData->Data[72]), 0); + $this->SendDebug('NodeVariation', ord($APIData->Data[73]), 0); + $this->SendDebug('PowerMode', ord($APIData->Data[74]), 0); + $this->SendDebug('BuildNumber', ord($APIData->Data[75]), 0); + $this->SendDebug('SerialNumber', substr($APIData->Data, 76, 8), 1); + */ + $State = ord($APIData->Data[84]); + $this->SendDebug('State', \KLF200\State::ToString($State), 0); + $CurrentPosition = unpack('n', substr($APIData->Data, 85, 2))[1]; + $this->SendDebug('CurrentPosition', sprintf('%04X', $CurrentPosition), 0); + /* $Target = unpack('n', substr($APIData->Data, 87, 2))[1]; + $this->SendDebug('Target', sprintf('%04X', $Target), 0); + */ + $FP1CurrentPosition = unpack('n', substr($APIData->Data, 89, 2))[1]; + $this->SendDebug('FP1CurrentPosition', sprintf('%04X', $FP1CurrentPosition), 0); + $FP2CurrentPosition = unpack('n', substr($APIData->Data, 91, 2))[1]; + $this->SendDebug('FP2CurrentPosition', sprintf('%04X', $FP2CurrentPosition), 0); + $FP3CurrentPosition = unpack('n', substr($APIData->Data, 93, 2))[1]; + $this->SendDebug('FP3CurrentPosition', sprintf('%04X', $FP3CurrentPosition), 0); + /* + $FP4CurrentPosition = unpack('n', substr($APIData->Data, 95, 2))[1]; + $this->SendDebug('FP4CurrentPosition', sprintf('%04X', $FP4CurrentPosition), 0); + $RemainingTime = unpack('n', substr($APIData->Data, 97, 2))[1]; + $this->SendDebug('RemainingTime', $RemainingTime, 0); + $TimeStamp = unpack('N', substr($APIData->Data, 99, 4))[1]; + $this->SendDebug('TimeStamp', $TimeStamp, 0); + $this->SendDebug('TimeStamp', strftime('%H:%M:%S %d.%m.%Y', $TimeStamp), 0); + */ + if ($NodeTypeSubType != $this->NodeSubType) { + $this->WriteAttributeInteger('NodeSubType', $NodeTypeSubType); + $this->SetSummary(sprintf('%04X', $NodeTypeSubType)); + $this->RegisterNodeVariables($NodeTypeSubType); + } + $this->SetValues($CurrentPosition, $FP1CurrentPosition, $FP2CurrentPosition, $FP3CurrentPosition); + break; + /* case \KLF200\APICommand::NODE_INFORMATION_CHANGED_NTF: + break; */ + case \KLF200\APICommand::NODE_STATE_POSITION_CHANGED_NTF: + /* + Data 1 Data 2 Data 3 - 4 Data 5 - 6 + NodeID State CurrentPosition Target + Data 7 - 8 Data 9 - 10 Data 11 -12 Data 13 - 14 Data 15 - 16 + FP1CurrentPosition FP2CurrentPosition FP3CurrentPosition FP4CurrentPosition RemainingTime + Data 17 - 20 + TimeStamp + */ + $State = ord($APIData->Data[1]); + $this->SendDebug('State', \KLF200\State::ToString($State), 0); + $CurrentPosition = unpack('n', substr($APIData->Data, 2, 2))[1]; + $this->SendDebug('CurrentPosition', sprintf('%04X', $CurrentPosition), 0); + $Target = unpack('n', substr($APIData->Data, 4, 2))[1]; + $this->SendDebug('Target', sprintf('%04X', $Target), 0); + $FP1CurrentPosition = unpack('n', substr($APIData->Data, 6, 2))[1]; + $this->SendDebug('FP1CurrentPosition', sprintf('%04X', $FP1CurrentPosition), 0); + $FP2CurrentPosition = unpack('n', substr($APIData->Data, 8, 2))[1]; + $this->SendDebug('FP2CurrentPosition', sprintf('%04X', $FP2CurrentPosition), 0); + $FP3CurrentPosition = unpack('n', substr($APIData->Data, 10, 2))[1]; + /* + $this->SendDebug('FP3CurrentPosition', sprintf('%04X', $FP3CurrentPosition), 0); + $FP4CurrentPosition = unpack('n', substr($APIData->Data, 12, 2))[1]; + $this->SendDebug('FP4CurrentPosition', sprintf('%04X', $FP4CurrentPosition), 0); + */ + $RemainingTime = unpack('n', substr($APIData->Data, 14, 2))[1]; + $this->SendDebug('RemainingTime', $RemainingTime, 0); + $TimeStamp = unpack('N', substr($APIData->Data, 16, 4))[1]; + $this->SendDebug('TimeStamp', $TimeStamp, 0); + $this->SendDebug('TimeStamp', strftime('%H:%M:%S %d.%m.%Y', $TimeStamp), 0); + if ($State == \KLF200\State::DONE) { + // Wert $CurrentPosition umrechnen und setzen + $this->SetValues($CurrentPosition, $FP1CurrentPosition, $FP2CurrentPosition, $FP3CurrentPosition); + } + break; + case \KLF200\APICommand::COMMAND_RUN_STATUS_NTF: + // 00 06 01 00 00 FF FF 01 02 0E 00 00 00 + /* + Command Data 1 - 2 Data 3 Data 4 + GW_COMMAND_RUN_STATUS_NTF SessionID StatusID Index + Data 5 Data 6 – 7 + NodeParameter ParameterValue + Data 8 Data 9 Data 10 - 13 + RunStatus StatusReply InformationCode + */ + $NodeParameter = ord($APIData->Data[4]); + $this->SendDebug('NodeParameter', $NodeParameter, 0); + $ParameterValue = unpack('n', substr($APIData->Data, 5, 2))[1]; + $this->SendDebug('ParameterValue', sprintf('%04X', $ParameterValue), 0); + $RunStatus = ord($APIData->Data[7]); + $this->SendDebug('RunStatus', \KLF200\RunStatus::ToString($RunStatus), 0); + $StatusReply = ord($APIData->Data[8]); + $this->SendDebug('StatusReply', \KLF200\StatusReply::ToString($StatusReply), 0); + if ($RunStatus == \KLF200\RunStatus::EXECUTION_FAILED) { + trigger_error($this->Translate(\KLF200\RunStatus::ToString($RunStatus)), E_USER_NOTICE); + return; + } + + break; + case \KLF200\APICommand::COMMAND_REMAINING_TIME_NTF: + break; + case \KLF200\APICommand::SESSION_FINISHED_NTF: + break; + case \KLF200\APICommand::STATUS_REQUEST_NTF: + //00 00 01 00 01 02 FF 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 + //01 05 01 01 00 01 01 + + /* + Command Data 1 – 2 Data 3 Data 4 Data 5 Data 6 + GW_STATUS_REQUEST_NTF SessionID StatusID NodeIndex RunStatus StatusReply + Data 7 + StatusType + * 0 = “Target Position” or + * 1 = “Current Position” or + * 2 = “Remaining Time” + * 01 00 C8 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 + * Data 8 Data 9 - 59 + * StatusCount ParameterData + * + Data 7 + StatusType + * 3 = “Main Info” + * Data 8 - 9 Data 10 - 11 Data 12 - 13 + * TargetPosition CurrentPosition RemainingTime + * Data 14 - 17 Data 18 + * LastMasterExecutionAddress LastCommandOriginator + */ + $StatusID = ord($APIData->Data[2]); + $this->SendDebug('StatusID', $StatusID, 0); + $NodeIndex = ord($APIData->Data[3]); + $this->SendDebug('NodeIndex', $NodeIndex, 0); + $RunStatus = ord($APIData->Data[4]); + $this->SendDebug('RunStatus', \KLF200\RunStatus::ToString($RunStatus), 0); + $StatusReply = ord($APIData->Data[5]); + $this->SendDebug('StatusReply', \KLF200\StatusReply::ToString($StatusReply), 0); + $StatusType = ord($APIData->Data[6]); + $this->SendDebug('StatusType', $StatusType, 0); + if ($StatusType == 0xFF) { + $this->SendDebug('Error', \KLF200\StatusReply::ToString($StatusReply), 0); + trigger_error($this->Translate(\KLF200\StatusReply::ToString($StatusReply)), E_USER_NOTICE); + return; + } + if ($StatusType == 0x01) { + $ParameterCount = ord($APIData->Data[7]); + $this->SendDebug('ParameterCount', $ParameterCount, 0); + $ParameterData = substr($APIData->Data, 8); + $Data = [ + 0 => 0xF7FF, + 1 => 0xF7FF, + 2 => 0xF7FF, + 3 => 0xF7FF + ]; + for ($index = 0; $index < $ParameterCount; $index++) { + $Data[ord($ParameterData[$index * 3])] = unpack('n', substr($ParameterData, ($index * 3) + 1, 2))[1]; + } + $this->SetValues($Data[0], $Data[1], $Data[2], $Data[3]); + } + break; + case \KLF200\APICommand::WINK_SEND_NTF: + break; + case \KLF200\APICommand::MODE_SEND_NTF: + break; + } + } + + private function GetSessionId() + { + $SessionId = ($this->SessionId + 1) & 0xff; + $this->SessionId = $SessionId; + return chr($SessionId); + } + + private function SendAPIData(\KLF200\APIData $APIData) + { + if ($this->NodeId == chr(-1)) { + return null; + } + $this->SendDebug('ForwardData', $APIData, 1); + + try { + if (!$this->HasActiveParent()) { + throw new Exception($this->Translate('Instance has no active parent.'), E_USER_NOTICE); + } + /** @var \KLF200\APIData $ResponseAPIData */ + $ret = @$this->SendDataToParent($APIData->ToJSON('{7B0F87CC-0408-4283-8E0E-2D48141E42E8}')); + $ResponseAPIData = @unserialize($ret); + $this->SendDebug('Response', $ResponseAPIData, 1); + if ($ResponseAPIData->isError()) { + trigger_error($this->Translate($ResponseAPIData->ErrorToString()), E_USER_NOTICE); + return null; + } + return $ResponseAPIData; + } catch (Exception $exc) { + $this->SendDebug('Error', $exc->getMessage(), 0); + return null; + } + } +} diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..63f44cc --- /dev/null +++ b/LICENSE @@ -0,0 +1,360 @@ +Creative Commons Attribution-NonCommercial-ShareAlike 4.0 International +Public License + +By exercising the Licensed Rights (defined below), You accept and agree +to be bound by the terms and conditions of this Creative Commons +Attribution-NonCommercial-ShareAlike 4.0 International Public License +("Public License"). To the extent this Public License may be +interpreted as a contract, You are granted the Licensed Rights in +consideration of Your acceptance of these terms and conditions, and the +Licensor grants You such rights in consideration of benefits the +Licensor receives from making the Licensed Material available under +these terms and conditions. + + +Section 1 -- Definitions. + + a. Adapted Material means material subject to Copyright and Similar + Rights that is derived from or based upon the Licensed Material + and in which the Licensed Material is translated, altered, + arranged, transformed, or otherwise modified in a manner requiring + permission under the Copyright and Similar Rights held by the + Licensor. For purposes of this Public License, where the Licensed + Material is a musical work, performance, or sound recording, + Adapted Material is always produced where the Licensed Material is + synched in timed relation with a moving image. + + b. Adapter's License means the license You apply to Your Copyright + and Similar Rights in Your contributions to Adapted Material in + accordance with the terms and conditions of this Public License. + + c. BY-NC-SA Compatible License means a license listed at + creativecommons.org/compatiblelicenses, approved by Creative + Commons as essentially the equivalent of this Public License. + + d. Copyright and Similar Rights means copyright and/or similar rights + closely related to copyright including, without limitation, + performance, broadcast, sound recording, and Sui Generis Database + Rights, without regard to how the rights are labeled or + categorized. For purposes of this Public License, the rights + specified in Section 2(b)(1)-(2) are not Copyright and Similar + Rights. + + e. Effective Technological Measures means those measures that, in the + absence of proper authority, may not be circumvented under laws + fulfilling obligations under Article 11 of the WIPO Copyright + Treaty adopted on December 20, 1996, and/or similar international + agreements. + + f. Exceptions and Limitations means fair use, fair dealing, and/or + any other exception or limitation to Copyright and Similar Rights + that applies to Your use of the Licensed Material. + + g. License Elements means the license attributes listed in the name + of a Creative Commons Public License. The License Elements of this + Public License are Attribution, NonCommercial, and ShareAlike. + + h. Licensed Material means the artistic or literary work, database, + or other material to which the Licensor applied this Public + License. + + i. Licensed Rights means the rights granted to You subject to the + terms and conditions of this Public License, which are limited to + all Copyright and Similar Rights that apply to Your use of the + Licensed Material and that the Licensor has authority to license. + + j. Licensor means the individual(s) or entity(ies) granting rights + under this Public License. + + k. NonCommercial means not primarily intended for or directed towards + commercial advantage or monetary compensation. For purposes of + this Public License, the exchange of the Licensed Material for + other material subject to Copyright and Similar Rights by digital + file-sharing or similar means is NonCommercial provided there is + no payment of monetary compensation in connection with the + exchange. + + l. Share means to provide material to the public by any means or + process that requires permission under the Licensed Rights, such + as reproduction, public display, public performance, distribution, + dissemination, communication, or importation, and to make material + available to the public including in ways that members of the + public may access the material from a place and at a time + individually chosen by them. + + m. Sui Generis Database Rights means rights other than copyright + resulting from Directive 96/9/EC of the European Parliament and of + the Council of 11 March 1996 on the legal protection of databases, + as amended and/or succeeded, as well as other essentially + equivalent rights anywhere in the world. + + n. You means the individual or entity exercising the Licensed Rights + under this Public License. Your has a corresponding meaning. + + +Section 2 -- Scope. + + a. License grant. + + 1. Subject to the terms and conditions of this Public License, + the Licensor hereby grants You a worldwide, royalty-free, + non-sublicensable, non-exclusive, irrevocable license to + exercise the Licensed Rights in the Licensed Material to: + + a. reproduce and Share the Licensed Material, in whole or + in part, for NonCommercial purposes only; and + + b. produce, reproduce, and Share Adapted Material for + NonCommercial purposes only. + + 2. Exceptions and Limitations. For the avoidance of doubt, where + Exceptions and Limitations apply to Your use, this Public + License does not apply, and You do not need to comply with + its terms and conditions. + + 3. Term. The term of this Public License is specified in Section + 6(a). + + 4. Media and formats; technical modifications allowed. The + Licensor authorizes You to exercise the Licensed Rights in + all media and formats whether now known or hereafter created, + and to make technical modifications necessary to do so. The + Licensor waives and/or agrees not to assert any right or + authority to forbid You from making technical modifications + necessary to exercise the Licensed Rights, including + technical modifications necessary to circumvent Effective + Technological Measures. For purposes of this Public License, + simply making modifications authorized by this Section 2(a) + (4) never produces Adapted Material. + + 5. Downstream recipients. + + a. Offer from the Licensor -- Licensed Material. Every + recipient of the Licensed Material automatically + receives an offer from the Licensor to exercise the + Licensed Rights under the terms and conditions of this + Public License. + + b. Additional offer from the Licensor -- Adapted Material. + Every recipient of Adapted Material from You + automatically receives an offer from the Licensor to + exercise the Licensed Rights in the Adapted Material + under the conditions of the Adapter's License You apply. + + c. No downstream restrictions. You may not offer or impose + any additional or different terms or conditions on, or + apply any Effective Technological Measures to, the + Licensed Material if doing so restricts exercise of the + Licensed Rights by any recipient of the Licensed + Material. + + 6. No endorsement. Nothing in this Public License constitutes or + may be construed as permission to assert or imply that You + are, or that Your use of the Licensed Material is, connected + with, or sponsored, endorsed, or granted official status by, + the Licensor or others designated to receive attribution as + provided in Section 3(a)(1)(A)(i). + + b. Other rights. + + 1. Moral rights, such as the right of integrity, are not + licensed under this Public License, nor are publicity, + privacy, and/or other similar personality rights; however, to + the extent possible, the Licensor waives and/or agrees not to + assert any such rights held by the Licensor to the limited + extent necessary to allow You to exercise the Licensed + Rights, but not otherwise. + + 2. Patent and trademark rights are not licensed under this + Public License. + + 3. To the extent possible, the Licensor waives any right to + collect royalties from You for the exercise of the Licensed + Rights, whether directly or through a collecting society + under any voluntary or waivable statutory or compulsory + licensing scheme. In all other cases the Licensor expressly + reserves any right to collect such royalties, including when + the Licensed Material is used other than for NonCommercial + purposes. + + +Section 3 -- License Conditions. + +Your exercise of the Licensed Rights is expressly made subject to the +following conditions. + + a. Attribution. + + 1. If You Share the Licensed Material (including in modified + form), You must: + + a. retain the following if it is supplied by the Licensor + with the Licensed Material: + + i. identification of the creator(s) of the Licensed + Material and any others designated to receive + attribution, in any reasonable manner requested by + the Licensor (including by pseudonym if + designated); + + ii. a copyright notice; + + iii. a notice that refers to this Public License; + + iv. a notice that refers to the disclaimer of + warranties; + + v. a URI or hyperlink to the Licensed Material to the + extent reasonably practicable; + + b. indicate if You modified the Licensed Material and + retain an indication of any previous modifications; and + + c. indicate the Licensed Material is licensed under this + Public License, and include the text of, or the URI or + hyperlink to, this Public License. + + 2. You may satisfy the conditions in Section 3(a)(1) in any + reasonable manner based on the medium, means, and context in + which You Share the Licensed Material. For example, it may be + reasonable to satisfy the conditions by providing a URI or + hyperlink to a resource that includes the required + information. + 3. If requested by the Licensor, You must remove any of the + information required by Section 3(a)(1)(A) to the extent + reasonably practicable. + + b. ShareAlike. + + In addition to the conditions in Section 3(a), if You Share + Adapted Material You produce, the following conditions also apply. + + 1. The Adapter's License You apply must be a Creative Commons + license with the same License Elements, this version or + later, or a BY-NC-SA Compatible License. + + 2. You must include the text of, or the URI or hyperlink to, the + Adapter's License You apply. You may satisfy this condition + in any reasonable manner based on the medium, means, and + context in which You Share Adapted Material. + + 3. You may not offer or impose any additional or different terms + or conditions on, or apply any Effective Technological + Measures to, Adapted Material that restrict exercise of the + rights granted under the Adapter's License You apply. + + +Section 4 -- Sui Generis Database Rights. + +Where the Licensed Rights include Sui Generis Database Rights that +apply to Your use of the Licensed Material: + + a. for the avoidance of doubt, Section 2(a)(1) grants You the right + to extract, reuse, reproduce, and Share all or a substantial + portion of the contents of the database for NonCommercial purposes + only; + + b. if You include all or a substantial portion of the database + contents in a database in which You have Sui Generis Database + Rights, then the database in which You have Sui Generis Database + Rights (but not its individual contents) is Adapted Material, + including for purposes of Section 3(b); and + + c. You must comply with the conditions in Section 3(a) if You Share + all or a substantial portion of the contents of the database. + +For the avoidance of doubt, this Section 4 supplements and does not +replace Your obligations under this Public License where the Licensed +Rights include other Copyright and Similar Rights. + + +Section 5 -- Disclaimer of Warranties and Limitation of Liability. + + a. UNLESS OTHERWISE SEPARATELY UNDERTAKEN BY THE LICENSOR, TO THE + EXTENT POSSIBLE, THE LICENSOR OFFERS THE LICENSED MATERIAL AS-IS + AND AS-AVAILABLE, AND MAKES NO REPRESENTATIONS OR WARRANTIES OF + ANY KIND CONCERNING THE LICENSED MATERIAL, WHETHER EXPRESS, + IMPLIED, STATUTORY, OR OTHER. THIS INCLUDES, WITHOUT LIMITATION, + WARRANTIES OF TITLE, MERCHANTABILITY, FITNESS FOR A PARTICULAR + PURPOSE, NON-INFRINGEMENT, ABSENCE OF LATENT OR OTHER DEFECTS, + ACCURACY, OR THE PRESENCE OR ABSENCE OF ERRORS, WHETHER OR NOT + KNOWN OR DISCOVERABLE. WHERE DISCLAIMERS OF WARRANTIES ARE NOT + ALLOWED IN FULL OR IN PART, THIS DISCLAIMER MAY NOT APPLY TO YOU. + + b. TO THE EXTENT POSSIBLE, IN NO EVENT WILL THE LICENSOR BE LIABLE + TO YOU ON ANY LEGAL THEORY (INCLUDING, WITHOUT LIMITATION, + NEGLIGENCE) OR OTHERWISE FOR ANY DIRECT, SPECIAL, INDIRECT, + INCIDENTAL, CONSEQUENTIAL, PUNITIVE, EXEMPLARY, OR OTHER LOSSES, + COSTS, EXPENSES, OR DAMAGES ARISING OUT OF THIS PUBLIC LICENSE OR + USE OF THE LICENSED MATERIAL, EVEN IF THE LICENSOR HAS BEEN + ADVISED OF THE POSSIBILITY OF SUCH LOSSES, COSTS, EXPENSES, OR + DAMAGES. WHERE A LIMITATION OF LIABILITY IS NOT ALLOWED IN FULL OR + IN PART, THIS LIMITATION MAY NOT APPLY TO YOU. + + c. The disclaimer of warranties and limitation of liability provided + above shall be interpreted in a manner that, to the extent + possible, most closely approximates an absolute disclaimer and + waiver of all liability. + + +Section 6 -- Term and Termination. + + a. This Public License applies for the term of the Copyright and + Similar Rights licensed here. However, if You fail to comply with + this Public License, then Your rights under this Public License + terminate automatically. + + b. Where Your right to use the Licensed Material has terminated under + Section 6(a), it reinstates: + + 1. automatically as of the date the violation is cured, provided + it is cured within 30 days of Your discovery of the + violation; or + + 2. upon express reinstatement by the Licensor. + + For the avoidance of doubt, this Section 6(b) does not affect any + right the Licensor may have to seek remedies for Your violations + of this Public License. + + c. For the avoidance of doubt, the Licensor may also offer the + Licensed Material under separate terms or conditions or stop + distributing the Licensed Material at any time; however, doing so + will not terminate this Public License. + + d. Sections 1, 5, 6, 7, and 8 survive termination of this Public + License. + + +Section 7 -- Other Terms and Conditions. + + a. The Licensor shall not be bound by any additional or different + terms or conditions communicated by You unless expressly agreed. + + b. Any arrangements, understandings, or agreements regarding the + Licensed Material not stated herein are separate from and + independent of the terms and conditions of this Public License. + + +Section 8 -- Interpretation. + + a. For the avoidance of doubt, this Public License does not, and + shall not be interpreted to, reduce, limit, restrict, or impose + conditions on any use of the Licensed Material that could lawfully + be made without permission under this Public License. + + b. To the extent possible, if any provision of this Public License is + deemed unenforceable, it shall be automatically reformed to the + minimum extent necessary to make it enforceable. If the provision + cannot be reformed, it shall be severed from this Public License + without affecting the enforceability of the remaining terms and + conditions. + + c. No term or condition of this Public License will be waived and no + failure to comply consented to unless expressly agreed to by the + Licensor. + + d. Nothing in this Public License constitutes or may be interpreted + as a limitation upon, or waiver of, any privileges and immunities + that apply to the Licensor or You, including from the legal + processes of any jurisdiction or authority. \ No newline at end of file diff --git a/README.md b/README.md new file mode 100644 index 0000000..48903b9 --- /dev/null +++ b/README.md @@ -0,0 +1,106 @@ +[![SDK](https://img.shields.io/badge/Symcon-PHPModul-red.svg?style=flat-square)](https://www.symcon.de/service/dokumentation/entwicklerbereich/sdk-tools/sdk-php/) +[![Version](https://img.shields.io/badge/Modul%20Version-0.80-blue.svg?style=flat-square)](https://community.symcon.de/t/modul-velux-klf200/50429) +[![Version](https://img.shields.io/badge/Symcon%20Version-5.5%20%3E-green.svg?style=flat-square)](https://www.symcon.de/service/dokumentation/installation/migrationen/v54-v55-q4-2020/) +[![License](https://img.shields.io/badge/License-CC%20BY--NC--SA%204.0-green.svg?style=flat-square)](https://creativecommons.org/licenses/by-nc-sa/4.0/) +[![Check Style](https://github.com/Nall-chan/VeluxKLF200/workflows/Check%20Style/badge.svg)](https://github.com/Nall-chan/VeluxKLF200/actions) [![Run Tests](https://github.com/Nall-chan/VeluxKLF200/workflows/Run%20Tests/badge.svg)](https://github.com/Nall-chan/VeluxKLF200/actions) +[![Spenden](https://www.paypalobjects.com/de_DE/DE/i/btn/btn_donate_SM.gif)](#3-spenden) +[![Wunschliste](https://img.shields.io/badge/Wunschliste-Amazon-ff69fb.svg)](#3-spenden) +# VeluxKLF200 + +Diese Implementierung der API von dem Velux KLF200 Gateway +ermöglicht die Einbindung von allen io-homecontrol® Geräten, welche von diesem Gateway unterstützt werden. + + +## Dokumentation + +**Inhaltsverzeichnis** + +- [1. Funktionsumfang](#1-funktionsumfang) + - [KLF200 Configurator](#klf200-configurator) + - [KLF200 Gateway](#klf200-gateway) + - [KLF200 Node](#klf200-node) +- [2. Voraussetzungen](#2-voraussetzungen) +- [3. Software-Installation](#3-software-installation) +- [4. Einrichten der Instanzen in IP-Symcon](#4-einrichten-der-instanzen-in-ip-symcon) +- [5. Anhang](#5-anhang) + - [1. GUID der Module](#1-guid-der-module) + - [2. Changelog](#2-changelog) + - [3. Spenden](#3-spenden) +- [6. Lizenz](#6-lizenz) + +## 1. Funktionsumfang + +### [KLF200 Configurator](KLF200Configurator/README.md) + - Einfaches Anlegen von Geräte-Instanzen in IPS. + +### [KLF200 Gateway](KLF200Gateway/README.md) + - Bindeglied zwischen dem KLF200 und den Geräte und Konfigurator-Instanzen. + +### [KLF200 Node](KLF200Node/README.md) + - Bildet ein Gerät (Node) des KLF200 in Symcon ab. + +## 2. Voraussetzungen + + - IPS 5.5 oder höher + - KLF200 io-homecontrol® Gateway + - KLF muss per LAN angeschlossen sein + - KLF Firmware 2.0.0.71 oder neuer + +## 3. Software-Installation + + Über den `Module-Store` in IPS das Modul `VELUX KLF200` hinzufügen. + **Bei kommerzieller Nutzung (z.B. als Errichter oder Integrator) wenden Sie sich bitte an den Autor.** + +## 4. Einrichten der Instanzen in IP-Symcon + +Details sind direkt in der Dokumentation der jeweiligen Module beschrieben. +Es wird empfohlen die Einrichtung mit der Konfigurator-Instanz zu starten ([KLF200 Configurator:](KLF200Configurator/README.md)). + +## 5. Anhang + +### 1. GUID der Module + + +| Modul | Typ | Prefix | GUID | +| ------------------- | ------------ | ------ | -------------------------------------- | +| KLF200 Configurator | Configurator | KLF200 | {38724E6E-8202-4D37-9FA7-BDD2EDA79520} | +| KLF200 Gateway | Splitter | KLF200 | {725D4DF6-C8FC-463C-823A-D3481A3D7003} | +| KLF200 Node | Device | KLF200 | {4EBD07B1-2962-4531-AC5F-7944789A9CE5} | + +### 2. Changelog + + Version 0.80: + - Doku aktualisiert. + - + + Version 0.71: + - Fehlermeldung wenn Timeout beim Verbindungsaufbau auftrat. + - Gateway zeigte beim Symcon start verbunden an, auch wenn ClientSocket gestört war. + + Version 0.7: + - PTLS entfernt. + + Version 0.6: + - Button im Splitter für das Lesen der Firmwareversion war ohne Funktion. + - Fehler im Konfigurator, wenn Namen der Geräte einen Umlaut enthielt. + - Der Konfigurator hat eine neuen Button, um im laufenden Betrieb die Geräteliste neu zu laden. + + Version 0.5: + - Öffentliche Betaversion + - Dokumentation erstellt + + Version 0.1: + - Testversion + + +### 3. Spenden + + Die Library ist für die nicht kommerzielle Nutzung kostenlos, Schenkungen als Unterstützung für den Autor werden hier akzeptiert: + + + +[![Wunschliste](https://img.shields.io/badge/Wunschliste-Amazon-ff69fb.svg)](https://www.amazon.de/hz/wishlist/ls/YU4AI9AQT9F?ref_=wl_share) + +## 6. Lizenz + + [CC BY-NC-SA 4.0](https://creativecommons.org/licenses/by-nc-sa/4.0/) \ No newline at end of file diff --git a/imgs/conf_configurator.png b/imgs/conf_configurator.png new file mode 100644 index 0000000..303a6fa Binary files /dev/null and b/imgs/conf_configurator.png differ diff --git a/imgs/conf_configurator1.png b/imgs/conf_configurator1.png new file mode 100644 index 0000000..6c6b500 Binary files /dev/null and b/imgs/conf_configurator1.png differ diff --git a/imgs/conf_configurator2.png b/imgs/conf_configurator2.png new file mode 100644 index 0000000..dc3ba2e Binary files /dev/null and b/imgs/conf_configurator2.png differ diff --git a/imgs/instanzen.png b/imgs/instanzen.png new file mode 100644 index 0000000..7f38588 Binary files /dev/null and b/imgs/instanzen.png differ diff --git a/imgs/instanzen_configurator.png b/imgs/instanzen_configurator.png new file mode 100644 index 0000000..1ac6232 Binary files /dev/null and b/imgs/instanzen_configurator.png differ diff --git a/imgs/instanzen_splitter.png b/imgs/instanzen_splitter.png new file mode 100644 index 0000000..5cd435b Binary files /dev/null and b/imgs/instanzen_splitter.png differ diff --git a/imgs/logbaum_splitter.png b/imgs/logbaum_splitter.png new file mode 100644 index 0000000..1dec525 Binary files /dev/null and b/imgs/logbaum_splitter.png differ diff --git a/library.json b/library.json new file mode 100644 index 0000000..9044175 --- /dev/null +++ b/library.json @@ -0,0 +1,12 @@ +{ + "id": "{7D6B48F7-9957-4E50-B98B-B9831963D802}", + "author": "Michael Tröger", + "name": "Velux Library", + "url": "https://github.com/Nall-chan/VeluxKLF200/", + "compatibility": { + "version": "5.5" + }, + "version": "0.80", + "build": 80, + "date": 1605267164 +} \ No newline at end of file diff --git a/libs/KLF200Class.php b/libs/KLF200Class.php new file mode 100644 index 0000000..dac8b31 --- /dev/null +++ b/libs/KLF200Class.php @@ -0,0 +1,886 @@ + + * @copyright 2020 Michael Tröger + * @license https://creativecommons.org/licenses/by-nc-sa/4.0/ CC BY-NC-SA 4.0 + * @version 1.1 + * @example Ohne + */ + +class APIData +{ + /** + * Alle Kommandos als Array. + * + * @var int + */ + public $Command; + + /** + * Alle Daten des Kommandos. + * + * @var string + */ + public $Data; + public $LastError = \KLF200\ErrorNTF::NO_ERROR; + + public function __construct($Command, $Data = '') + { + if (is_string($Command)) { + $Data = json_decode($Command, true); + $this->Command = $Data['Command']; + $this->Data = utf8_decode($Data['Data']); + return; + } + $this->Command = $Command; + if ($Command == \KLF200\APICommand::ERROR_NTF) { + $this->LastError = ord($Data); + $this->Data = ''; + } else { + $this->Data = $Data; + } + } + + public function ToJSON(string $GUID) + { + $Data['DataID'] = $GUID; + $Data['Command'] = $this->Command; + $Data['Data'] = utf8_encode($this->Data); + return json_encode($Data); + } + + public function GetSLIPData() + { + $TransportData = "\x00" . chr(strlen((string) $this->Data) + 3) . pack('n', $this->Command) . (string) $this->Data; + + $cs = 0; + for ($i = 1; $i < strlen($TransportData); $i++) { + $cs ^= ord($TransportData[$i]); + } + $TransportData .= chr($cs); + return "\xc0" . str_replace(["\xDB", "\xC0"], ["\xDB\xDD", "\xDB\xDC"], $TransportData) . "\xc0"; + } + + public function isEvent() + { + return \KLF200\APICommand::isEvent($this->Command); + } + + public function isError() + { + return $this->LastError != \KLF200\ErrorNTF::NO_ERROR; + } + + public function ErrorToString() + { + return \KLF200\ErrorNTF::ToString($this->LastError); + } +} + +class ErrorNTF +{ + public const NO_ERROR = -1; + public const NOT_DEFINED = 0; // Not further defined error. + public const UNKNOWN_COMMAND = 1; // Unknown Command or command is not accepted at this state. + public const ERROR_ON_FRAME_STRUCTURE = 2; // ERROR on Frame Structure. + public const BUSY = 7; // Busy. Try again later. + public const BAD_SYSTEM_TABLE_INDEX = 8; // Bad system table index. + public const NOT_AUTHENTICATED = 12; // Not authenticated. + public const TIMEOUT = 99; + + public static function ToString(int $Error) + { + switch ($Error) { + case self::NO_ERROR: + return 'No error.'; + case self::TIMEOUT: + return 'Timeout.'; + case self::NOT_DEFINED: + return 'Not further defined error.'; + case self::UNKNOWN_COMMAND: + return 'Unknown command or command is not accepted at this state.'; + case self::ERROR_ON_FRAME_STRUCTURE: + return 'Error on Frame Structure.'; + case self::BUSY: + return 'Busy. Try again later.'; + case self::BAD_SYSTEM_TABLE_INDEX: + return 'Bad system table index.'; + case self::NOT_AUTHENTICATED: + return 'Not authenticated.'; + } + } +} + +class State +{ + public const NON_EXECUTING = 0; + public const ERROR_WHILE_EXECUTION = 1; + public const NOT_USED = 2; + public const WAITING_FOR_POWER = 3; + public const EXECUTING = 4; + public const DONE = 5; + public const UNKNOWN = 255; + + public static function ToString(int $State) + { + switch ($State) { + case self::NON_EXECUTING: + return 'Parameter is unable to execute.'; + case self::ERROR_WHILE_EXECUTION: + return 'Execution has failed.'; + case self::NOT_USED: + return 'not used.'; + case self::WAITING_FOR_POWER: + return 'waiting for power.'; + case self::EXECUTING: + return 'Executing.'; + case self::DONE: + return 'done.'; + case self::UNKNOWN: + return 'unknown.'; + default: + return 'unknown state value: 0x' . sprintf('%02X', $State); + } + } +} + +class RunStatus +{ + public const EXECUTION_COMPLETED = 0; + public const EXECUTION_FAILED = 1; + public const EXECUTION_ACTIVE = 2; + + public static function ToString(int $RunStatus) + { + switch ($RunStatus) { + case self::EXECUTION_COMPLETED: + return 'Execution is completed with no errors.'; + case self::EXECUTION_FAILED: + return 'Execution has failed.'; + case self::EXECUTION_ACTIVE: + return 'Execution is still active.'; + } + } +} + +class StatusReply +{ + public const UNKNOWN_STATUS_REPLY = 0x00; + public const COMMAND_COMPLETED_OK = 0x01; + public const NO_CONTACT = 0x02; + public const MANUALLY_OPERATED = 0x03; + public const BLOCKED = 0x04; + public const WRONG_SYSTEMKEY = 0x05; + public const PRIORITY_LEVEL_LOCKED = 0x06; + public const REACHED_WRONG_POSITION = 0x07; + public const ERROR_DURING_EXECUTION = 0x08; + public const NO_EXECUTION = 0x09; + public const CALIBRATING = 0x0A; + public const POWER_CONSUMPTION_TOO_HIGH = 0x0B; + public const POWER_CONSUMPTION_TOO_LOW = 0x0C; + public const LOCK_POSITION_OPEN = 0x0D; + public const MOTION_TIME_TOO_LONG__COMMUNICATION_ENDED = 0x0E; + public const THERMAL_PROTECTION = 0x0F; + public const PRODUCT_NOT_OPERATIONAL = 0x10; + public const FILTER_MAINTENANCE_NEEDED = 0x11; + public const BATTERY_LEVEL = 0x12; + public const TARGET_MODIFIED = 0x13; + public const MODE_NOT_IMPLEMENTED = 0x14; + public const COMMAND_INCOMPATIBLE_TO_MOVEMENT = 0x15; + public const USER_ACTION = 0x16; + public const DEAD_BOLT_ERROR = 0x17; + public const AUTOMATIC_CYCLE_ENGAGED = 0x18; + public const WRONG_LOAD_CONNECTED = 0x19; + public const COLOUR_NOT_REACHABLE = 0x1A; + public const TARGET_NOT_REACHABLE = 0x1B; + public const BAD_INDEX_RECEIVED = 0x1C; + public const COMMAND_OVERRULED = 0x1D; + public const NODE_WAITING_FOR_POWER = 0x1E; + public const INFORMATION_CODE = 0xDF; + public const PARAMETER_LIMITED = 0xE0; + public const LIMITATION_BY_LOCAL_USER = 0xE1; + public const LIMITATION_BY_USER = 0xE2; + public const LIMITATION_BY_RAIN = 0xE3; + public const LIMITATION_BY_TIMER = 0xE4; + public const LIMITATION_BY_UPS = 0xE6; + public const LIMITATION_BY_UNKNOWN_DEVICE = 0xE7; + public const LIMITATION_BY_SAAC = 0xEA; + public const LIMITATION_BY_WIND = 0xEB; + public const LIMITATION_BY_MYSELF = 0xEC; + public const LIMITATION_BY_AUTOMATIC_CYCLE = 0xED; + public const LIMITATION_BY_EMERGENCY = 0xEE; + + public static function ToString(int $StatusReply) + { + switch ($StatusReply) { + case self::UNKNOWN_STATUS_REPLY: + return 'unknown reply'; + case self::COMMAND_COMPLETED_OK: + return 'no errors detected'; + case self::NO_CONTACT: + return 'no communication to node'; + case self::MANUALLY_OPERATED: + return 'manually operated by a user'; + case self::BLOCKED: + return 'node has been blocked by an object'; + case self::WRONG_SYSTEMKEY: + return 'the node contains a wrong system key'; + case self::PRIORITY_LEVEL_LOCKED: + return 'the node is locked on this priority level'; + case self::REACHED_WRONG_POSITION: + return 'node has stopped in another position than expected'; + case self::ERROR_DURING_EXECUTION: + return 'an error has occurred during execution of command'; + case self::NO_EXECUTION: + return 'no movement of the node parameter'; + case self::CALIBRATING: + return 'the node is calibrating the parameters'; + case self::POWER_CONSUMPTION_TOO_HIGH: + return 'the node power consumption is too high'; + case self::POWER_CONSUMPTION_TOO_LOW: + return 'the node power consumption is too low'; + case self::LOCK_POSITION_OPEN: + return 'door lock errors. (Door open during lock command)'; + case self::MOTION_TIME_TOO_LONG__COMMUNICATION_ENDED: + return 'the target was not reached in time'; + case self::THERMAL_PROTECTION: + return 'the node has gone into thermal protection mode'; + case self::PRODUCT_NOT_OPERATIONAL: + return 'the node is not currently operational'; + case self::FILTER_MAINTENANCE_NEEDED: + return 'the filter needs maintenance'; + case self::BATTERY_LEVEL: + return 'the battery level is low'; + case self::TARGET_MODIFIED: + return 'the node has modified the target value of the command'; + case self::MODE_NOT_IMPLEMENTED: + return 'this node does not support the mode received'; + case self::COMMAND_INCOMPATIBLE_TO_MOVEMENT: + return 'the node is unable to move in the right direction'; + case self::USER_ACTION: + return 'dead bolt is manually locked during unlock command'; + case self::DEAD_BOLT_ERROR: + return 'dead bolt error'; + case self::AUTOMATIC_CYCLE_ENGAGED: + return 'the node has gone into automatic cycle mode'; + case self::WRONG_LOAD_CONNECTED: + return 'wrong load on node'; + case self::COLOUR_NOT_REACHABLE: + return 'that node is unable to reach received colour code'; + case self::TARGET_NOT_REACHABLE: + return 'the node is unable to reach received target position'; + case self::BAD_INDEX_RECEIVED: + return 'io-protocol has received an invalid index'; + case self::COMMAND_OVERRULED: + return 'that the command was overruled by a new command'; + case self::NODE_WAITING_FOR_POWER: + return 'that the node reported waiting for power'; + case self::INFORMATION_CODE: + return 'an unknown error code received'; + case self::PARAMETER_LIMITED: + return 'the parameter was limited by an unknown device'; + case self::LIMITATION_BY_LOCAL_USER: + return 'the parameter was limited by local button'; + case self::LIMITATION_BY_USER: + return 'the parameter was limited by a remote control'; + case self::LIMITATION_BY_RAIN: + return 'the parameter was limited by a rain sensor'; + case self::LIMITATION_BY_TIMER: + return 'the parameter was limited by a timer'; + case self::LIMITATION_BY_UPS: + return 'the parameter was limited by a power supply'; + case self::LIMITATION_BY_UNKNOWN_DEVICE: + return 'the parameter was limited by an unknown device'; + case self::LIMITATION_BY_SAAC: + return 'the parameter was limited by a standalone automatic controller'; + case self::LIMITATION_BY_WIND: + return 'the parameter was limited by a wind sensor'; + case self::LIMITATION_BY_MYSELF: + return 'the parameter was limited by the node itself'; + case self::LIMITATION_BY_AUTOMATIC_CYCLE: + return 'the parameter was limited by an automatic cycle'; + case self::LIMITATION_BY_EMERGENCY: + return 'the parameter was limited by an emergency'; + } + } +} + +class APICommand +{ + public const ERROR_NTF = 0x0000; //Provides information on what triggered the error. + public const REBOOT_REQ = 0x0001; //Request gateway to reboot. + public const REBOOT_CFM = 0x0002; //Acknowledge to REBOOT_REQ command. + public const SET_FACTORY_DEFAULT_REQ = 0x0003; //Request gateway to clear system table, scene table and set Ethernet settings to factory default. Gateway will reboot. + public const SET_FACTORY_DEFAULT_CFM = 0x0004; //Acknowledge to SET_FACTORY_DEFAULT_REQ command. + public const GET_VERSION_REQ = 0x0008; //Request version information. + public const GET_VERSION_CFM = 0x0009; //Acknowledge to GET_VERSION_REQ command. + public const GET_PROTOCOL_VERSION_REQ = 0x000A; //Request KLF 200 API protocol version. + public const GET_PROTOCOL_VERSION_CFM = 0x000B; //Acknowledge to GET_PROTOCOL_VERSION_REQ command. + public const GET_STATE_REQ = 0x000C; //Request the state of the gateway + public const GET_STATE_CFM = 0x000D; //Acknowledge to GET_STATE_REQ command. + public const LEAVE_LEARN_STATE_REQ = 0x000E; //Request gateway to leave learn state. + public const LEAVE_LEARN_STATE_CFM = 0x000F; //Acknowledge to LEAVE_LEARN_STATE_REQ command. + public const GET_NETWORK_SETUP_REQ = 0x00E0; //Request network parameters. + public const GET_NETWORK_SETUP_CFM = 0x00E1; //Acknowledge to GET_NETWORK_SETUP_REQ. + public const SET_NETWORK_SETUP_REQ = 0x00E2; //Set network parameters. + public const SET_NETWORK_SETUP_CFM = 0x00E3; //Acknowledge to SET_NETWORK_SETUP_REQ. + public const CS_GET_SYSTEMTABLE_DATA_REQ = 0x0100; //Request a list of nodes in the gateways system table. + public const CS_GET_SYSTEMTABLE_DATA_CFM = 0x0101; //Acknowledge to CS_GET_SYSTEMTABLE_DATA_REQ + public const CS_GET_SYSTEMTABLE_DATA_NTF = 0x0102; //Acknowledge to CS_GET_SYSTEM_TABLE_DATA_REQList of nodes in the gateways systemtable. + public const CS_DISCOVER_NODES_REQ = 0x0103; //Start CS DiscoverNodes macro in KLF200. + public const CS_DISCOVER_NODES_CFM = 0x0104; //Acknowledge to CS_DISCOVER_NODES_REQ command. + public const CS_DISCOVER_NODES_NTF = 0x0105; //Acknowledge to CS_DISCOVER_NODES_REQ command. + public const CS_REMOVE_NODES_REQ = 0x0106; //Remove one or more nodes in the systemtable. + public const CS_REMOVE_NODES_CFM = 0x0107; //Acknowledge to CS_REMOVE_NODES_REQ. + public const CS_VIRGIN_STATE_REQ = 0x0108; //Clear systemtable and delete system key. + public const CS_VIRGIN_STATE_CFM = 0x0109; //Acknowledge to CS_VIRGIN_STATE_REQ. + public const CS_CONTROLLER_COPY_REQ = 0x010A; //Setup KLF200 to get or give a system to or from another io-homecontrol® remote control. By a system means all nodes in the systemtable and the system key. + public const CS_CONTROLLER_COPY_CFM = 0x010B; //Acknowledge to CS_CONTROLLER_COPY_REQ. + public const CS_CONTROLLER_COPY_NTF = 0x010C; //Acknowledge to CS_CONTROLLER_COPY_REQ. + public const CS_CONTROLLER_COPY_CANCEL_NTF = 0x010D; //Cancellation of system copy to other controllers. + public const CS_RECEIVE_KEY_REQ = 0x010E; //Receive system key from another controller. + public const CS_RECEIVE_KEY_CFM = 0x010F; //Acknowledge to CS_RECEIVE_KEY_REQ. + public const CS_RECEIVE_KEY_NTF = 0x0110; //Acknowledge to CS_RECEIVE_KEY_REQ with status. + public const CS_PGC_JOB_NTF = 0x0111; //Information on Product Generic Configuration job initiated by press on PGC button. + public const CS_SYSTEM_TABLE_UPDATE_NTF = 0x0112; //Broadcasted to all clients and gives information about added and removed actuator nodes in system table. + public const CS_GENERATE_NEW_KEY_REQ = 0x0113; //Generate new system key and update actuators in systemtable. + public const CS_GENERATE_NEW_KEY_CFM = 0x0114; //Acknowledge to CS_GENERATE_NEW_KEY_REQ. + public const CS_GENERATE_NEW_KEY_NTF = 0x0115; //Acknowledge to CS_GENERATE_NEW_KEY_REQ with status. + public const CS_REPAIR_KEY_REQ = 0x0116; //Update key in actuators holding an old key. + public const CS_REPAIR_KEY_CFM = 0x0117; //Acknowledge to CS_REPAIR_KEY_REQ. + public const CS_REPAIR_KEY_NTF = 0x0118; //Acknowledge to CS_REPAIR_KEY_REQ with status. + public const CS_ACTIVATE_CONFIGURATION_MODE_REQ = 0x0119; //Request one or more actuator to open for configuration. + public const CS_ACTIVATE_CONFIGURATION_MODE_CFM = 0x011A; //Acknowledge to CS_ACTIVATE_CONFIGURATION_MODE_REQ. + public const GET_NODE_INFORMATION_REQ = 0x0200; //Request extended information of one specific actuator node. + public const GET_NODE_INFORMATION_CFM = 0x0201; //Acknowledge to GET_NODE_INFORMATION_REQ. + public const GET_NODE_INFORMATION_NTF = 0x0210; //Acknowledge to GET_NODE_INFORMATION_REQ. + public const GET_ALL_NODES_INFORMATION_REQ = 0x0202; //Request extended information of all nodes. + public const GET_ALL_NODES_INFORMATION_CFM = 0x0203; //Acknowledge to GET_ALL_NODES_INFORMATION_REQ + public const GET_ALL_NODES_INFORMATION_NTF = 0x0204; //Acknowledge to GET_ALL_NODES_INFORMATION_REQ. Holds node information + public const GET_ALL_NODES_INFORMATION_FINISHED_NTF = 0x0205; //Acknowledge to GET_ALL_NODES_INFORMATION_REQ. No more nodes. + public const SET_NODE_VARIATION_REQ = 0x0206; //Set node variation. + public const SET_NODE_VARIATION_CFM = 0x0207; //Acknowledge to SET_NODE_VARIATION_REQ. + public const SET_NODE_NAME_REQ = 0x0208; //Set node name. + public const SET_NODE_NAME_CFM = 0x0209; //Acknowledge to SET_NODE_NAME_REQ. + public const SET_NODE_VELOCITY_REQ = 0x020A; //Set node velocity. + public const SET_NODE_VELOCITY_CFM = 0x020B; //Acknowledge to SET_NODE_VELOCITY_REQ. + public const NODE_INFORMATION_CHANGED_NTF = 0x020C; //Information has been updated. + public const NODE_STATE_POSITION_CHANGED_NTF = 0x0211; //Information has been updated. + public const SET_NODE_ORDER_AND_PLACEMENT_REQ = 0x020D; //Set search order and room placement. + public const SET_NODE_ORDER_AND_PLACEMENT_CFM = 0x020E; //Acknowledge to SET_NODE_ORDER_AND_PLACEMENT_REQ. + public const GET_GROUP_INFORMATION_REQ = 0x0220; //Request information about all defined groups. + public const GET_GROUP_INFORMATION_CFM = 0x0221; //Acknowledge to GET_GROUP_INFORMATION_REQ. + public const GET_GROUP_INFORMATION_NTF = 0x0230; //Acknowledge to GET_NODE_INFORMATION_REQ. + public const SET_GROUP_INFORMATION_REQ = 0x0222; //Change an existing group. + public const SET_GROUP_INFORMATION_CFM = 0x0223; //Acknowledge to SET_GROUP_INFORMATION_REQ. + public const GROUP_INFORMATION_CHANGED_NTF = 0x0224; //Broadcast to all, about group information of a group has been changed. + public const DELETE_GROUP_REQ = 0x0225; //Delete a group. + public const DELETE_GROUP_CFM = 0x0226; //Acknowledge to DELETE_GROUP_INFORMATION_REQ. + public const NEW_GROUP_REQ = 0x0227; //Request new group to be created. + public const NEW_GROUP_CFM = 0x0228; + public const GET_ALL_GROUPS_INFORMATION_REQ = 0x0229; //Request information about all defined groups. + public const GET_ALL_GROUPS_INFORMATION_CFM = 0x022A; //Acknowledge to GET_ALL_GROUPS_INFORMATION_REQ. + public const GET_ALL_GROUPS_INFORMATION_NTF = 0x022B; //Acknowledge to GET_ALL_GROUPS_INFORMATION_REQ. + public const GET_ALL_GROUPS_INFORMATION_FINISHED_NTF = 0x022C; //Acknowledge to GET_ALL_GROUPS_INFORMATION_REQ. + public const GROUP_DELETED_NTF = 0x022D; //GROUP_DELETED_NTF is broadcasted to all, when a group has been removed. + public const HOUSE_STATUS_MONITOR_ENABLE_REQ = 0x0240; //Enable house status monitor. + public const HOUSE_STATUS_MONITOR_ENABLE_CFM = 0x0241; //Acknowledge to HOUSE_STATUS_MONITOR_ENABLE_REQ. + public const HOUSE_STATUS_MONITOR_DISABLE_REQ = 0x0242; //Disable house status monitor. + public const HOUSE_STATUS_MONITOR_DISABLE_CFM = 0x0243; //Acknowledge to HOUSE_STATUS_MONITOR_DISABLE_REQ. + public const COMMAND_SEND_REQ = 0x0300; //Send activating command direct to one or more io-homecontrol® nodes. + public const COMMAND_SEND_CFM = 0x0301; //Acknowledge to COMMAND_SEND_REQ. + public const COMMAND_RUN_STATUS_NTF = 0x0302; //Gives run status for io-homecontrol® node. + public const COMMAND_REMAINING_TIME_NTF = 0x0303; //Gives remaining time before io-homecontrol® node enter target position. + public const SESSION_FINISHED_NTF = 0x0304; //Command send, Status request, Wink, Mode or Stop session is finished. + public const STATUS_REQUEST_REQ = 0x0305; //Get status request from one or more io-homecontrol® nodes. + public const STATUS_REQUEST_CFM = 0x0306; //Acknowledge to STATUS_REQUEST_REQ. + public const STATUS_REQUEST_NTF = 0x0307; //Acknowledge to STATUS_REQUEST_REQ. Status request from one or more io-homecontrol® nodes. + public const WINK_SEND_REQ = 0x0308; //Request from one or more io-homecontrol® nodes to Wink. + public const WINK_SEND_CFM = 0x0309; //Acknowledge to WINK_SEND_REQ + public const WINK_SEND_NTF = 0x030A; //Status info for performed wink request. + public const SET_LIMITATION_REQ = 0x0310; //Set a parameter limitation in an actuator. + public const SET_LIMITATION_CFM = 0x0311; //Acknowledge to SET_LIMITATION_REQ. + public const GET_LIMITATION_STATUS_REQ = 0x0312; //Get parameter limitation in an actuator. + public const GET_LIMITATION_STATUS_CFM = 0x0313; //Acknowledge to GET_LIMITATION_STATUS_REQ. + public const LIMITATION_STATUS_NTF = 0x0314; //Hold information about limitation. + public const MODE_SEND_REQ = 0x0320; //Send Activate Mode to one or more io-homecontrol® nodes. + public const MODE_SEND_CFM = 0x0321; //Acknowledge to MODE_SEND_REQ + public const MODE_SEND_NTF = 0x0322; //Notify with Mode activation info. + public const INITIALIZE_SCENE_REQ = 0x0400; //Prepare gateway to record a scene. + public const INITIALIZE_SCENE_CFM = 0x0401; //Acknowledge to INITIALIZE_SCENE_REQ. + public const INITIALIZE_SCENE_NTF = 0x0402; //Acknowledge to INITIALIZE_SCENE_REQ. + public const INITIALIZE_SCENE_CANCEL_REQ = 0x0403; //Cancel record scene process. + public const INITIALIZE_SCENE_CANCEL_CFM = 0x0404; //Acknowledge to INITIALIZE_SCENE_CANCEL_REQ command. + public const RECORD_SCENE_REQ = 0x0405; //Store actuator positions changes since INITIALIZE_SCENE, as a scene. + public const RECORD_SCENE_CFM = 0x0406; //Acknowledge to RECORD_SCENE_REQ. + public const RECORD_SCENE_NTF = 0x0407; //Acknowledge to RECORD_SCENE_REQ. + public const DELETE_SCENE_REQ = 0x0408; //Delete a recorded scene. + public const DELETE_SCENE_CFM = 0x0409; //Acknowledge to DELETE_SCENE_REQ. + public const RENAME_SCENE_REQ = 0x040A; //Request a scene to be renamed. + public const RENAME_SCENE_CFM = 0x040B; //Acknowledge to RENAME_SCENE_REQ. + public const GET_SCENE_LIST_REQ = 0x040C; //Request a list of scenes. + public const GET_SCENE_LIST_CFM = 0x040D; //Acknowledge to GET_SCENE_LIST. + public const GET_SCENE_LIST_NTF = 0x040E; //Acknowledge to GET_SCENE_LIST. + public const GET_SCENE_INFORMATION_REQ = 0x040F; //Request extended information for one given scene. + public const GET_SCENE_INFORMATION_CFM = 0x0410; //Acknowledge to GET_SCENE_INFORMATION_REQ. + public const GET_SCENE_INFORMATION_NTF = 0x0411; //Acknowledge to GET_SCENE_INFORMATION_REQ. + public const ACTIVATE_SCENE_REQ = 0x0412; //Request gateway to enter a scene. + public const ACTIVATE_SCENE_CFM = 0x0413; //Acknowledge to ACTIVATE_SCENE_REQ. + public const STOP_SCENE_REQ = 0x0415; //Request all nodes in a given scene to stop at their current position. + public const STOP_SCENE_CFM = 0x0416; //Acknowledge to STOP_SCENE_REQ. + public const SCENE_INFORMATION_CHANGED_NTF = 0x0419; //A scene has either been changed or removed. + public const ACTIVATE_PRODUCTGROUP_REQ = 0x0447; //Activate a product group in a given direction. + public const ACTIVATE_PRODUCTGROUP_CFM = 0x0448; //Acknowledge to ACTIVATE_PRODUCTGROUP_REQ. + public const ACTIVATE_PRODUCTGROUP_NTF = 0x0449; //Acknowledge to ACTIVATE_PRODUCTGROUP_REQ. + public const GET_CONTACT_INPUT_LINK_LIST_REQ = 0x0460; //Get list of assignments to all Contact Input to scene or product group. + public const GET_CONTACT_INPUT_LINK_LIST_CFM = 0x0461; //Acknowledge to GET_CONTACT_INPUT_LINK_LIST_REQ. + public const SET_CONTACT_INPUT_LINK_REQ = 0x0462; //Set a link from a Contact Input to a scene or product group. + public const SET_CONTACT_INPUT_LINK_CFM = 0x0463; //Acknowledge to SET_CONTACT_INPUT_LINK_REQ. + public const REMOVE_CONTACT_INPUT_LINK_REQ = 0x0464; //Remove a link from a Contact Input to a scene. + public const REMOVE_CONTACT_INPUT_LINK_CFM = 0x0465; //Acknowledge to REMOVE_CONTACT_INPUT_LINK_REQ. + public const GET_ACTIVATION_LOG_HEADER_REQ = 0x0500; //Request header from activation log. + public const GET_ACTIVATION_LOG_HEADER_CFM = 0x0501; //Confirm header from activation log. + public const CLEAR_ACTIVATION_LOG_REQ = 0x0502; //Request clear all data in activation log. + public const CLEAR_ACTIVATION_LOG_CFM = 0x0503; //Confirm clear all data in activation log. + public const GET_ACTIVATION_LOG_LINE_REQ = 0x0504; //Request line from activation log. + public const GET_ACTIVATION_LOG_LINE_CFM = 0x0505; //Confirm line from activation log. + public const ACTIVATION_LOG_UPDATED_NTF = 0x0506; //Confirm line from activation log. + public const GET_MULTIPLE_ACTIVATION_LOG_LINES_REQ = 0x0507; //Request lines from activation log. + public const GET_MULTIPLE_ACTIVATION_LOG_LINES_NTF = 0x0508; //Error log data from activation log. + public const GET_MULTIPLE_ACTIVATION_LOG_LINES_CFM = 0x0509; //Confirm lines from activation log. + public const SET_UTC_REQ = 0x2000; //Request to set UTC time. + public const SET_UTC_CFM = 0x2001; //Acknowledge to SET_UTC_REQ. + public const RTC_SET_TIME_ZONE_REQ = 0x2002; //Set time zone and daylight savings rules. + public const RTC_SET_TIME_ZONE_CFM = 0x2003; //Acknowledge to RTC_SET_TIME_ZONE_REQ. + public const GET_LOCAL_TIME_REQ = 0x2004; //Request the local time based on current time zone and daylight savings rules. + public const GET_LOCAL_TIME_CFM = 0x2005; //Acknowledge to RTC_SET_TIME_ZONE_REQ. + public const IGNORE_2120 = 0x2120; + public const IGNORE_2121 = 0x2121; + public const IGNORE_2122 = 0x2122; + public const IGNORE_2123 = 0x2123; + public const IGNORE_2124 = 0x2124; + public const PASSWORD_ENTER_REQ = 0x3000; //Enter password to authenticate request + public const PASSWORD_ENTER_CFM = 0x3001; //Acknowledge to PASSWORD_ENTER_REQ + public const PASSWORD_CHANGE_REQ = 0x3002; //Request password change. + public const PASSWORD_CHANGE_CFM = 0x3003; //Acknowledge to PASSWORD_CHANGE_REQ. + public const PASSWORD_CHANGE_NTF = 0x3004; //Acknowledge to PASSWORD_CHANGE_REQ. Broadcasted to all connected clients. + + public static $NotifyCommand = [ + self::CS_GET_SYSTEMTABLE_DATA_NTF, + self::CS_DISCOVER_NODES_NTF, + self::CS_CONTROLLER_COPY_NTF, + self::CS_CONTROLLER_COPY_CANCEL_NTF, + self::CS_RECEIVE_KEY_NTF, + self::CS_PGC_JOB_NTF, + self::CS_SYSTEM_TABLE_UPDATE_NTF, + self::CS_GENERATE_NEW_KEY_NTF, + self::CS_REPAIR_KEY_NTF, + self::GET_NODE_INFORMATION_NTF, + self::GET_ALL_NODES_INFORMATION_NTF, + self::GET_ALL_NODES_INFORMATION_FINISHED_NTF, + self::NODE_INFORMATION_CHANGED_NTF, + self::NODE_STATE_POSITION_CHANGED_NTF, + self::GET_GROUP_INFORMATION_NTF, + self::GROUP_INFORMATION_CHANGED_NTF, + self::GET_ALL_GROUPS_INFORMATION_NTF, + self::GET_ALL_GROUPS_INFORMATION_FINISHED_NTF, + self::GROUP_DELETED_NTF, + self::COMMAND_RUN_STATUS_NTF, + self::COMMAND_REMAINING_TIME_NTF, + self::SESSION_FINISHED_NTF, + self::STATUS_REQUEST_NTF, + self::WINK_SEND_NTF, + self::LIMITATION_STATUS_NTF, + self::MODE_SEND_NTF, + self::INITIALIZE_SCENE_NTF, + self::RECORD_SCENE_NTF, + self::GET_SCENE_LIST_NTF, + self::GET_SCENE_INFORMATION_NTF, + self::SCENE_INFORMATION_CHANGED_NTF, + self::ACTIVATE_PRODUCTGROUP_NTF, + self::ACTIVATION_LOG_UPDATED_NTF, + self::GET_MULTIPLE_ACTIVATION_LOG_LINES_NTF, + self::PASSWORD_CHANGE_NTF, + self::IGNORE_2120, + self::IGNORE_2121, + self::IGNORE_2122, + self::IGNORE_2123, + self::IGNORE_2124 + ]; + + public static function isEvent(int $APICommand) + { + return in_array($APICommand, self::$NotifyCommand); + } + + public static function ToString($APICommand) + { + switch ($APICommand) { + case self::ERROR_NTF: + return 'ERROR_NTF'; + case self::REBOOT_REQ: + return 'REBOOT_REQ'; + case self::REBOOT_CFM: + return 'REBOOT_CFM'; + case self::SET_FACTORY_DEFAULT_REQ: + return 'SET_FACTORY_DEFAULT_REQ'; + case self::SET_FACTORY_DEFAULT_CFM: + return 'SET_FACTORY_DEFAULT_CFM'; + case self::GET_VERSION_REQ: + return 'GET_VERSION_REQ'; + case self::GET_VERSION_CFM: + return 'GET_VERSION_CFM'; + case self::GET_PROTOCOL_VERSION_REQ: + return 'GET_PROTOCOL_VERSION_REQ'; + case self::GET_PROTOCOL_VERSION_CFM: + return 'GET_PROTOCOL_VERSION_CFM'; + case self::GET_STATE_REQ: + return 'GET_STATE_REQ'; + case self::GET_STATE_CFM: + return 'GET_STATE_CFM'; + case self::LEAVE_LEARN_STATE_REQ: + return 'LEAVE_LEARN_STATE_REQ'; + case self::LEAVE_LEARN_STATE_CFM: + return 'LEAVE_LEARN_STATE_CFM'; + case self::GET_NETWORK_SETUP_REQ: + return 'GET_NETWORK_SETUP_REQ'; + case self::GET_NETWORK_SETUP_CFM: + return 'GET_NETWORK_SETUP_CFM'; + case self::SET_NETWORK_SETUP_REQ: + return 'SET_NETWORK_SETUP_REQ'; + case self::SET_NETWORK_SETUP_CFM: + return 'SET_NETWORK_SETUP_CFM'; + case self::CS_GET_SYSTEMTABLE_DATA_REQ: + return 'CS_GET_SYSTEMTABLE_DATA_REQ'; + case self::CS_GET_SYSTEMTABLE_DATA_CFM: + return 'CS_GET_SYSTEMTABLE_DATA_CFM'; + case self::CS_GET_SYSTEMTABLE_DATA_NTF: + return 'CS_GET_SYSTEMTABLE_DATA_NTF'; + case self::CS_DISCOVER_NODES_REQ: + return 'CS_DISCOVER_NODES_REQ'; + case self::CS_DISCOVER_NODES_CFM: + return 'CS_DISCOVER_NODES_CFM'; + case self::CS_DISCOVER_NODES_NTF: + return 'CS_DISCOVER_NODES_NTF'; + case self::CS_REMOVE_NODES_REQ: + return 'CS_REMOVE_NODES_REQ'; + case self::CS_REMOVE_NODES_CFM: + return 'CS_REMOVE_NODES_CFM'; + case self::CS_VIRGIN_STATE_REQ: + return 'CS_VIRGIN_STATE_REQ'; + case self::CS_VIRGIN_STATE_CFM: + return 'CS_VIRGIN_STATE_CFM'; + case self::CS_CONTROLLER_COPY_REQ: + return 'CS_CONTROLLER_COPY_REQ'; + case self::CS_CONTROLLER_COPY_CFM: + return 'CS_CONTROLLER_COPY_CFM'; + case self::CS_CONTROLLER_COPY_NTF: + return 'CS_CONTROLLER_COPY_NTF'; + case self::CS_CONTROLLER_COPY_CANCEL_NTF: + return 'CS_CONTROLLER_COPY_CANCEL_NTF'; + case self::CS_RECEIVE_KEY_REQ: + return 'CS_RECEIVE_KEY_REQ'; + case self::CS_RECEIVE_KEY_CFM: + return 'CS_RECEIVE_KEY_CFM'; + case self::CS_RECEIVE_KEY_NTF: + return 'CS_RECEIVE_KEY_NTF'; + case self::CS_PGC_JOB_NTF: + return 'CS_PGC_JOB_NTF'; + case self::CS_SYSTEM_TABLE_UPDATE_NTF: + return 'CS_SYSTEM_TABLE_UPDATE_NTF'; + case self::CS_GENERATE_NEW_KEY_REQ: + return 'CS_GENERATE_NEW_KEY_REQ'; + case self::CS_GENERATE_NEW_KEY_CFM: + return 'CS_GENERATE_NEW_KEY_CFM'; + case self::CS_GENERATE_NEW_KEY_NTF: + return 'CS_GENERATE_NEW_KEY_NTF'; + case self::CS_REPAIR_KEY_REQ: + return 'CS_REPAIR_KEY_REQ'; + case self::CS_REPAIR_KEY_CFM: + return 'CS_REPAIR_KEY_CFM'; + case self::CS_REPAIR_KEY_NTF: + return 'CS_REPAIR_KEY_NTF'; + case self::CS_ACTIVATE_CONFIGURATION_MODE_REQ: + return 'CS_ACTIVATE_CONFIGURATION_MODE_REQ'; + case self::CS_ACTIVATE_CONFIGURATION_MODE_CFM: + return 'CS_ACTIVATE_CONFIGURATION_MODE_CFM'; + case self::GET_NODE_INFORMATION_REQ: + return 'GET_NODE_INFORMATION_REQ'; + case self::GET_NODE_INFORMATION_CFM: + return 'GET_NODE_INFORMATION_CFM'; + case self::GET_NODE_INFORMATION_NTF: + return 'GET_NODE_INFORMATION_NTF'; + case self::GET_ALL_NODES_INFORMATION_REQ: + return 'GET_ALL_NODES_INFORMATION_REQ'; + case self::GET_ALL_NODES_INFORMATION_CFM: + return 'GET_ALL_NODES_INFORMATION_CFM'; + case self::GET_ALL_NODES_INFORMATION_NTF: + return 'GET_ALL_NODES_INFORMATION_NTF'; + case self::GET_ALL_NODES_INFORMATION_FINISHED_NTF: + return 'GET_ALL_NODES_INFORMATION_FINISHED_NTF'; + case self::SET_NODE_VARIATION_REQ: + return 'SET_NODE_VARIATION_REQ'; + case self::SET_NODE_VARIATION_CFM: + return 'SET_NODE_VARIATION_CFM'; + case self::SET_NODE_NAME_REQ: + return 'SET_NODE_NAME_REQ'; + case self::SET_NODE_NAME_CFM: + return 'SET_NODE_NAME_CFM'; + case self::SET_NODE_VELOCITY_REQ: + return 'SET_NODE_VELOCITY_REQ'; + case self::SET_NODE_VELOCITY_CFM: + return 'SET_NODE_VELOCITY_CFM'; + case self::NODE_INFORMATION_CHANGED_NTF: + return 'NODE_INFORMATION_CHANGED_NTF'; + case self::NODE_STATE_POSITION_CHANGED_NTF: + return 'NODE_STATE_POSITION_CHANGED_NTF'; + case self::SET_NODE_ORDER_AND_PLACEMENT_REQ: + return 'SET_NODE_ORDER_AND_PLACEMENT_REQ'; + case self::SET_NODE_ORDER_AND_PLACEMENT_CFM: + return 'SET_NODE_ORDER_AND_PLACEMENT_CFM'; + case self::GET_GROUP_INFORMATION_REQ: + return 'GET_GROUP_INFORMATION_REQ'; + case self::GET_GROUP_INFORMATION_CFM: + return 'GET_GROUP_INFORMATION_CFM'; + case self::GET_GROUP_INFORMATION_NTF: + return 'GET_GROUP_INFORMATION_NTF'; + case self::SET_GROUP_INFORMATION_REQ: + return 'SET_GROUP_INFORMATION_REQ'; + case self::SET_GROUP_INFORMATION_CFM: + return 'SET_GROUP_INFORMATION_CFM'; + case self::GROUP_INFORMATION_CHANGED_NTF: + return 'GROUP_INFORMATION_CHANGED_NTF'; + case self::DELETE_GROUP_REQ: + return 'DELETE_GROUP_REQ'; + case self::DELETE_GROUP_CFM: + return 'DELETE_GROUP_CFM'; + case self::NEW_GROUP_REQ: + return 'NEW_GROUP_REQ'; + case self::NEW_GROUP_CFM: + return 'NEW_GROUP_CFM'; + case self::GET_ALL_GROUPS_INFORMATION_REQ: + return 'GET_ALL_GROUPS_INFORMATION_REQ'; + case self::GET_ALL_GROUPS_INFORMATION_CFM: + return 'GET_ALL_GROUPS_INFORMATION_CFM'; + case self::GET_ALL_GROUPS_INFORMATION_NTF: + return 'GET_ALL_GROUPS_INFORMATION_NTF'; + case self::GET_ALL_GROUPS_INFORMATION_FINISHED_NTF: + return 'GET_ALL_GROUPS_INFORMATION_FINISHED_NTF'; + case self::GROUP_DELETED_NTF: + return 'GROUP_DELETED_NTF'; + case self::HOUSE_STATUS_MONITOR_ENABLE_REQ: + return 'HOUSE_STATUS_MONITOR_ENABLE_REQ'; + case self::HOUSE_STATUS_MONITOR_ENABLE_CFM: + return 'HOUSE_STATUS_MONITOR_ENABLE_CFM'; + case self::HOUSE_STATUS_MONITOR_DISABLE_REQ: + return 'HOUSE_STATUS_MONITOR_DISABLE_REQ'; + case self::HOUSE_STATUS_MONITOR_DISABLE_CFM: + return 'HOUSE_STATUS_MONITOR_DISABLE_CFM'; + case self::COMMAND_SEND_REQ: + return 'COMMAND_SEND_REQ'; + case self::COMMAND_SEND_CFM: + return 'COMMAND_SEND_CFM'; + case self::COMMAND_RUN_STATUS_NTF: + return 'COMMAND_RUN_STATUS_NTF'; + case self::COMMAND_REMAINING_TIME_NTF: + return 'COMMAND_REMAINING_TIME_NTF'; + case self::SESSION_FINISHED_NTF: + return 'SESSION_FINISHED_NTF'; + case self::STATUS_REQUEST_REQ: + return 'STATUS_REQUEST_REQ'; + case self::STATUS_REQUEST_CFM: + return 'STATUS_REQUEST_CFM'; + case self::STATUS_REQUEST_NTF: + return 'STATUS_REQUEST_NTF'; + case self::WINK_SEND_REQ: + return 'WINK_SEND_REQ'; + case self::WINK_SEND_CFM: + return 'WINK_SEND_CFM'; + case self::WINK_SEND_NTF: + return 'WINK_SEND_NTF'; + case self::SET_LIMITATION_REQ: + return 'SET_LIMITATION_REQ'; + case self::SET_LIMITATION_CFM: + return 'SET_LIMITATION_CFM'; + case self::GET_LIMITATION_STATUS_REQ: + return 'GET_LIMITATION_STATUS_REQ'; + case self::GET_LIMITATION_STATUS_CFM: + return 'GET_LIMITATION_STATUS_CFM'; + case self::LIMITATION_STATUS_NTF: + return 'LIMITATION_STATUS_NTF'; + case self::MODE_SEND_REQ: + return 'MODE_SEND_REQ'; + case self::MODE_SEND_CFM: + return 'MODE_SEND_CFM'; + case self::MODE_SEND_NTF: + return 'MODE_SEND_NTF'; + case self::INITIALIZE_SCENE_REQ: + return 'INITIALIZE_SCENE_REQ'; + case self::INITIALIZE_SCENE_CFM: + return 'INITIALIZE_SCENE_CFM'; + case self::INITIALIZE_SCENE_NTF: + return 'INITIALIZE_SCENE_NTF'; + case self::INITIALIZE_SCENE_CANCEL_REQ: + return 'INITIALIZE_SCENE_CANCEL_REQ'; + case self::INITIALIZE_SCENE_CANCEL_CFM: + return 'INITIALIZE_SCENE_CANCEL_CFM'; + case self::RECORD_SCENE_REQ: + return 'RECORD_SCENE_REQ'; + case self::RECORD_SCENE_CFM: + return 'RECORD_SCENE_CFM'; + case self::RECORD_SCENE_NTF: + return 'RECORD_SCENE_NTF'; + case self::DELETE_SCENE_REQ: + return 'DELETE_SCENE_REQ'; + case self::DELETE_SCENE_CFM: + return 'DELETE_SCENE_CFM'; + case self::RENAME_SCENE_REQ: + return 'RENAME_SCENE_REQ'; + case self::RENAME_SCENE_CFM: + return 'RENAME_SCENE_CFM'; + case self::GET_SCENE_LIST_REQ: + return 'GET_SCENE_LIST_REQ'; + case self::GET_SCENE_LIST_CFM: + return 'GET_SCENE_LIST_CFM'; + case self::GET_SCENE_LIST_NTF: + return 'GET_SCENE_LIST_NTF'; + case self::GET_SCENE_INFORMATION_REQ: + return 'GET_SCENE_INFORMATION_REQ'; + case self::GET_SCENE_INFORMATION_CFM: + return 'GET_SCENE_INFORMATION_CFM'; + case self::GET_SCENE_INFORMATION_NTF: + return 'GET_SCENE_INFORMATION_NTF'; + case self::ACTIVATE_SCENE_REQ: + return 'ACTIVATE_SCENE_REQ'; + case self::ACTIVATE_SCENE_CFM: + return 'ACTIVATE_SCENE_CFM'; + case self::STOP_SCENE_REQ: + return 'STOP_SCENE_REQ'; + case self::STOP_SCENE_CFM: + return 'STOP_SCENE_CFM'; + case self::SCENE_INFORMATION_CHANGED_NTF: + return 'SCENE_INFORMATION_CHANGED_NTF'; + case self::ACTIVATE_PRODUCTGROUP_REQ: + return 'ACTIVATE_PRODUCTGROUP_REQ'; + case self::ACTIVATE_PRODUCTGROUP_CFM: + return 'ACTIVATE_PRODUCTGROUP_CFM'; + case self::ACTIVATE_PRODUCTGROUP_NTF: + return 'ACTIVATE_PRODUCTGROUP_NTF'; + case self::GET_CONTACT_INPUT_LINK_LIST_REQ: + return 'GET_CONTACT_INPUT_LINK_LIST_REQ'; + case self::GET_CONTACT_INPUT_LINK_LIST_CFM: + return 'GET_CONTACT_INPUT_LINK_LIST_CFM'; + case self::SET_CONTACT_INPUT_LINK_REQ: + return 'SET_CONTACT_INPUT_LINK_REQ'; + case self::SET_CONTACT_INPUT_LINK_CFM: + return 'SET_CONTACT_INPUT_LINK_CFM'; + case self::REMOVE_CONTACT_INPUT_LINK_REQ: + return 'REMOVE_CONTACT_INPUT_LINK_REQ'; + case self::REMOVE_CONTACT_INPUT_LINK_CFM: + return 'REMOVE_CONTACT_INPUT_LINK_CFM'; + case self::GET_ACTIVATION_LOG_HEADER_REQ: + return 'GET_ACTIVATION_LOG_HEADER_REQ'; + case self::GET_ACTIVATION_LOG_HEADER_CFM: + return 'GET_ACTIVATION_LOG_HEADER_CFM'; + case self::CLEAR_ACTIVATION_LOG_REQ: + return 'CLEAR_ACTIVATION_LOG_REQ'; + case self::CLEAR_ACTIVATION_LOG_CFM: + return 'CLEAR_ACTIVATION_LOG_CFM'; + case self::GET_ACTIVATION_LOG_LINE_REQ: + return 'GET_ACTIVATION_LOG_LINE_REQ'; + case self::GET_ACTIVATION_LOG_LINE_CFM: + return 'GET_ACTIVATION_LOG_LINE_CFM'; + case self::ACTIVATION_LOG_UPDATED_NTF: + return 'ACTIVATION_LOG_UPDATED_NTF'; + case self::GET_MULTIPLE_ACTIVATION_LOG_LINES_REQ: + return 'GET_MULTIPLE_ACTIVATION_LOG_LINES_REQ'; + case self::GET_MULTIPLE_ACTIVATION_LOG_LINES_NTF: + return 'GET_MULTIPLE_ACTIVATION_LOG_LINES_NTF'; + case self::GET_MULTIPLE_ACTIVATION_LOG_LINES_CFM: + return 'GET_MULTIPLE_ACTIVATION_LOG_LINES_CFM'; + case self::SET_UTC_REQ: + return 'SET_UTC_REQ'; + case self::SET_UTC_CFM: + return 'SET_UTC_CFM'; + case self::RTC_SET_TIME_ZONE_REQ: + return 'RTC_SET_TIME_ZONE_REQ'; + case self::RTC_SET_TIME_ZONE_CFM: + return 'RTC_SET_TIME_ZONE_CFM'; + case self::GET_LOCAL_TIME_REQ: + return 'GET_LOCAL_TIME_REQ'; + case self::GET_LOCAL_TIME_CFM: + return 'GET_LOCAL_TIME_CFM'; + case self::PASSWORD_ENTER_REQ: + return 'PASSWORD_ENTER_REQ'; + case self::PASSWORD_ENTER_CFM: + return 'PASSWORD_ENTER_CFM'; + case self::PASSWORD_CHANGE_REQ: + return 'PASSWORD_CHANGE_REQ'; + case self::PASSWORD_CHANGE_CFM: + return 'PASSWORD_CHANGE_CFM'; + case self::PASSWORD_CHANGE_NTF: + return 'PASSWORD_CHANGE_NTF'; + case self::IGNORE_2120: + return 'IGNORE_2120'; + case self::IGNORE_2121: + return 'IGNORE_2121'; + case self::IGNORE_2122: + return 'IGNORE_2122'; + case self::IGNORE_2123: + return 'IGNORE_2123'; + case self::IGNORE_2124: + return 'IGNORE_2124'; + } + } +} + +class Node +{ + public static $SubType = [ + 0x0040 => 'Interior Venetian Blind', + 0x0080 => 'Roller Shutter', + 0x0081 => 'Adjustable slats rolling shutter', + 0x0082 => 'Roller Shutter With projection', + 0x00C0 => 'Vertical Exterior Awning', + 0x0100 => 'Window opener', + 0x0101 => 'Window opener with integrated rain sensor', + 0x0140 => 'Garage door opener', + 0x017A => 'Garage door opener', + 0x0180 => 'Light', + 0x01BA => 'Light only supporting on/off', + 0x01C0 => 'Gate opener', + 0x01FA => 'Gate opener', + 0x0200 => 'Rolling Door Opener', + 0x0240 => 'Door lock', + 0x0241 => 'Window lock', + 0x0280 => 'Vertical Interior Blinds', + 0x0300 => 'Beacon', + 0x0340 => 'Dual Roller Shutter', + 0x0380 => 'Heating Temperature Interface', + 0x03C0 => 'On/Off switch', + 0x0400 => 'Horizontal awning', + 0x0440 => 'Exterior Venetian blind', + 0x0480 => 'Louver blind', + 0x04C0 => 'Curtain track', + 0x0500 => 'Ventilation point', + 0x0501 => 'Air inlet', + 0x0502 => 'Air transfer', + 0x0503 => 'Air outlet', + 0x0540 => 'Exterior heating', + 0x057A => 'Exterior heating', + 0x0580 => 'Heat pump', + 0x05C0 => 'Intrusion alarm', + 0x0600 => 'Swinging Shutters', + 0x0601 => 'Swinging Shutter with independent handling of the leaves' + ]; +} + +/* @} */ diff --git a/libs/helper b/libs/helper new file mode 160000 index 0000000..6cb1dce --- /dev/null +++ b/libs/helper @@ -0,0 +1 @@ +Subproject commit 6cb1dce113b8fe3135d8ccb6aa06b7b89c73d1db diff --git a/tests/LibraryTest.php b/tests/LibraryTest.php new file mode 100644 index 0000000..316da36 --- /dev/null +++ b/tests/LibraryTest.php @@ -0,0 +1,27 @@ +validateLibrary(__DIR__ . '/..'); + } + + public function testValidateKLF200Gateway(): void + { + $this->validateModule(__DIR__ . '/../KLF200Gateway'); + } + + public function testValidateKLF200Configurator(): void + { + $this->validateModule(__DIR__ . '/../KLF200Configurator'); + } + public function testValidateKLF200Node(): void + { + $this->validateModule(__DIR__ . '/../KLF200Node'); + } +} diff --git a/tests/phpunit.xml b/tests/phpunit.xml new file mode 100644 index 0000000..51b6d5b --- /dev/null +++ b/tests/phpunit.xml @@ -0,0 +1,11 @@ + + + + + ../VeluxKLF200 + + + + + + diff --git a/tests/stubs b/tests/stubs new file mode 160000 index 0000000..9ee089a --- /dev/null +++ b/tests/stubs @@ -0,0 +1 @@ +Subproject commit 9ee089af18a9f19b8046b2f0672939121f4f3d42