# $Id$ ############################################################################## # # 46_TRX_SECURITY.pm # FHEM module for X10, KD101, Visonic # # Copyright (C) 2012 Willi Herzig # # This file is part of fhem. # # Fhem is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 2 of the License, or # (at your option) any later version. # # Fhem is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with fhem. If not, see . # ############################################################################## # # # values for "set global verbose" # 4: log unknown protocols # 5: log decoding hexlines for debugging # package main; use strict; use warnings; # Debug this module? YES = 1, NO = 0 my $TRX_SECURITY_debug = 0; my $time_old = 0; my $TRX_SECURITY_type_default = "ds10a"; my $DOT = q{_}; my %security_device_codes = ( # HEXSTRING => "NAME", "name of reading", # 0x20: X10, KD101, Visonic, Meiantech 0x2000 => [ "DS10A", "Window" ], 0x2001 => [ "MS10A", "motion" ], 0x2002 => [ "KR18", "key" ], 0x2003 => [ "KD101", "smoke" ], 0x2004 => [ "VISONIC_WINDOW", "window" ], 0x2005 => [ "VISONIC_REMOTE", "key" ], 0x2006 => [ "VISONIC_WINDOW", "window" ], 0x2007 => [ "Meiantech", "alarm" ], ); my %security_device_commands = ( # HEXSTRING => commands # 0x20: X10, KD101, Visonic, Meiantech 0x2000 => [ "", "", "", "", "", "", ""], # DS10A 0x2003 => [ "", "", "", "", "", "", "alert", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "pair",], # KD101 ); my %security_device_c2b; # DEVICE_TYPE->hash (reverse of security_device_codes) ##################################### sub TRX_SECURITY_Initialize($) { my ($hash) = @_; foreach my $k (keys %security_device_codes) { $security_device_c2b{$security_device_codes{$k}->[0]} = $k; } $hash->{Match} = "^..(20).*"; $hash->{SetFn} = "TRX_SECURITY_Set"; $hash->{DefFn} = "TRX_SECURITY_Define"; $hash->{UndefFn} = "TRX_SECURITY_Undef"; $hash->{ParseFn} = "TRX_SECURITY_Parse"; $hash->{AttrList} = "IODev ignore:1,0 event-on-update-reading event-on-change-reading do_not_notify:1,0 loglevel:0,1,2,3,4,5,6"; } ################################### sub TRX_SECURITY_Set($@) { my ($hash, @a) = @_; my $ret = undef; my $na = int(@a); return "no set value specified" if($na < 2 || $na > 3); # look for device_type my $name = $a[0]; my $command = $a[1]; my $level; if ($na == 3) { $level = $a[2]; } else { $level = 0; } my $device_type = $hash->{TRX_SECURITY_type}; my $deviceid = $hash->{TRX_SECURITY_deviceid}; if ($device_type ne "KD101") { return "No set implemented for $device_type"; } my $device_type_num = $security_device_c2b{$device_type}; if(!defined($device_type_num)) { return "Unknown device_type, choose one of " . join(" ", sort keys %security_device_c2b); } my $protocol_type = $device_type_num >> 8; # high bytes # Now check if the command is valid and retrieve the command id: my $rec = $security_device_commands{$device_type_num}; my $i; for ($i=0; $i <= $#$rec && ($rec->[$i] ne $command); $i++) { ;} if($i > $#$rec) { my $l = join(" ", sort @$rec); if ($device_type eq "AC" || $device_type eq "HOMEEASY" || $device_type eq "ANSLUT") { $l =~ s/ level / level:slider,0,1,15 /; } my $error = "Unknown command $command, choose one of $l"; Log 4, $error; return $error; } if ($na == 4 && $command ne "level") { my $error = "Error: level not possible for command $command"; } my $seqnr = 0; my $cmnd = $i; my $hex_prefix; my $hex_command; if ($protocol_type == 0x20) { my $id1; my $id2; my $id3; if ($deviceid =~ /(..)(..)(..)/ ) { $id1 = $1; $id2 = $2; $id3 = $3; } else { Log 4,"TRX_SECURITY_Set lightning1 wrong deviceid: name=$name device_type=$device_type, deviceid=$deviceid"; return "error set name=$name deviceid=$deviceid"; } # lightning1 $hex_prefix = sprintf "0820"; $hex_command = sprintf "%02x%02x%02s%02s%02s%02x00", $device_type_num & 0xff, $seqnr, $id1, $id2, $id3, $cmnd; Log 1,"TRX_SECURITY_Set name=$name device_type=$device_type, deviceid=$deviceid id1=$id1, id1=$id2, id1=$id3, command=$command" if ($TRX_SECURITY_debug == 1); Log 1,"TRX_SECURITY_Set hexline=$hex_prefix$hex_command" if ($TRX_SECURITY_debug == 1); } else { return "No set implemented for $device_type . Unknown protocol type"; } IOWrite($hash, $hex_prefix, $hex_command); my $tn = TimeNow(); $hash->{CHANGED}[0] = $command; $hash->{STATE} = $command; $hash->{READINGS}{state}{TIME} = $tn; $hash->{READINGS}{state}{VAL} = $command; return $ret; } ##################################### sub TRX_SECURITY_Define($$) { my ($hash, $def) = @_; my @a = split("[ \t][ \t]*", $def); my $a = int(@a); if(int(@a) != 5 && int(@a) != 7) { Log 1,"TRX_SECURITY wrong syntax '@a'. \nCorrect syntax is 'define TRX_SECURITY [ ]'"; return "wrong syntax: define TRX_SECURITY [ ]"; } my $name = $a[0]; my $type = lc($a[2]); my $deviceid = $a[3]; my $devicelog = $a[4]; $type = uc($type); my $device_name = "TRX".$DOT.$type.$DOT.$deviceid; if ($type ne "DS10A" && $type ne "SD90" && $type ne "MS10A" && $type ne "MS14A" && $type ne "KR18" && $type ne "KD101" && $type ne "VISONIC_WINDOW" & $type ne "VISONIC_MOTION" & $type ne "VISONIC_REMOTE") { Log 1,"TRX_SECURITY define: wrong type: $type"; return "TRX_SECURITY: wrong type: $type"; } $hash->{TRX_SECURITY_deviceid} = $deviceid; $hash->{TRX_SECURITY_devicelog} = $devicelog; $hash->{TRX_SECURITY_type} = $type; #$hash->{TRX_SECURITY_CODE} = $deviceid; $modules{TRX_SECURITY}{defptr}{$device_name} = $hash; if (int(@a) == 7) { # there is a second deviceid: # my $deviceid2 = $a[5]; my $devicelog2 = $a[6]; my $device_name2 = "TRX_SECURITY".$DOT.$deviceid2; $hash->{TRX_SECURITY_deviceid2} = $deviceid2; $hash->{TRX_SECURITY_devicelog2} = $devicelog2; #$hash->{TRX_SECURITY_CODE2} = $deviceid2; $modules{TRX_SECURITY}{defptr2}{$device_name2} = $hash; } AssignIoPort($hash); return undef; } ##################################### sub TRX_SECURITY_Undef($$) { my ($hash, $name) = @_; delete($modules{TRX_SECURITY}{defptr}{$name}); return undef; } ##################################### sub TRX_SECURITY_parse_X10Sec { my $bytes = shift; my $error; my $subtype = $bytes->[1]; my $device; if ($subtype >= 3) { $device = sprintf '%02x%02x%02x', $bytes->[3], $bytes->[4], $bytes->[5]; } else { # that's how we do it on 43_RFXX10REC.pm $device = sprintf '%02x%02x', $bytes->[5], $bytes->[3]; } my %security_devtype = ( # HEXSTRING => 0x00 => [ "DS10A", "Window" ], # X10 security door/window sensor 0x01 => [ "MS10A", "motion" ], # X10 security motion sensor 0x02 => [ "KR18", "key" ], # X10 security remote (no alive packets) 0x03 => [ "KD101", "smoke" ], # KD101 (no alive packets) 0x04 => [ "VISONIC_WINDOW", "window" ], # Visonic PowerCode door/window sensor – primary contact (with alive packets) 0x05 => [ "VISONIC_MOTION", "motion" ], # Visonic PowerCode motion sensor (with alive packets) 0x06 => [ "VISONIC_REMOTE", "key" ], # Visonic CodeSecure (no alive packets) 0x07 => [ "VISONIC_WINDOW", "window" ], # Visonic PowerCode door/window sensor – auxiliary contact (no alive packets) ); my $dev_type; my $dev_reading; if (exists $security_devtype{$subtype}) { my $rec = $security_devtype{$subtype}; if (ref $rec) { ($dev_type, $dev_reading ) = @$rec; } else { $error = "TRX_SECURITY: x10_devtype wrong for subtype=$subtype"; Log 1, $error; return ""; } } else { $error = "TRX_SECURITY: error undefined subtype=$subtype"; Log 1, $error; return ""; } #-------------- my $device_name = "TRX".$DOT.$dev_type.$DOT.$device; Log 4, "device_name=$device_name"; my $firstdevice = 1; my $def = $modules{TRX_SECURITY}{defptr}{$device_name}; if(!$def) { $firstdevice = 0; $def = $modules{TRX_SECURITY}{defptr2}{$device_name}; if (!$def) { Log 1, "UNDEFINED $device_name TRX_SECURITY $dev_type $device $dev_reading"; Log 3, "TRX_SECURITY: TRX_SECURITY Unknown device $device_name, please define it"; return "UNDEFINED $device_name TRX_SECURITY $dev_type $device $dev_reading"; } } # Use $def->{NAME}, because the device may be renamed: my $name = $def->{NAME}; return "" if(IsIgnored($name)); my $data = $bytes->[6]; my $hexdata = sprintf '%02x', $data; my %x10_security = ( 0x00 => ['X10Sec', 'normal', 'min_delay', ''], 0x01 => ['X10Sec', 'normal', 'max_delay', ''], 0x02 => ['X10Sec', 'alert', 'min_delay', ''], 0x03 => ['X10Sec', 'alert', 'max_delay', ''], 0x04 => ['X10Sec', 'alert', '', ''], 0x05 => ['X10Sec', 'normal', '', ''], 0x06 => ['X10Sec', 'alert', '', ''], 0x07 => ['X10Sec', 'normal', '', ''], 0x08 => ['X10Sec', 'tamper', '', ''], 0x09 => ['X10Sec', 'Security-Arm_Away', 'min_delay', ''], # kr18 0x0a => ['X10Sec', 'Security-Arm_Away', 'max_delay', ''], # kr18 0x0b => ['X10Sec', 'Security-Arm_Home', 'min_delay', ''], # kr18 0x0c => ['X10Sec', 'Security-Arm_Home', 'max_delay', ''], # kr18 0x0d => ['X10Sec', 'Security-Disarm', 'min_delay', ''], # kr18 0x10 => ['X10Sec', 'ButtonA-on', '', ''], # kr18 0x11 => ['X10Sec', 'ButtonA-off', '', ''], # kr18 0x12 => ['X10Sec', 'ButtonB-on', '', ''], # kr18 0x13 => ['X10Sec', 'ButtonB-off', '', ''], # kr18 0x14 => ['X10Sec', 'dark', '', ''], 0x15 => ['X10Sec', 'light', '', ''], 0x16 => ['X10Sec', 'normal', '', 'batt_low'], 0x17 => ['X10Sec', 'pair KD101', '', ''], ); my $command = ""; my $type = ""; my $delay = ""; my $battery = ""; my @res; if (exists $x10_security{$data}) { my $rec = $x10_security{$data}; if (ref $rec) { ($type, $command, $delay, $battery) = @$rec; } else { $command = $rec; } } else { Log 1, "TRX_SECURITY: undefined command cmd=$data device-nr=$device, hex=$hexdata"; return ""; } my $battery_level = $bytes->[7] & 0x0f; if (($battery eq "") && ($dev_type ne "KD101")) { if ($battery_level == 0x9) { $battery = 'batt_ok'} elsif ($battery_level == 0x0) { $battery = 'batt_low'} else { Log 1,"TRX_SECURITY: X10Sec unkown battery_level=$battery_level"; } } my $current = ""; Log 1, "TRX_SECURITY: $name devn=$device_name first=$firstdevice subtype=$subtype command=$command, delay=$delay, batt=$battery cmd=$hexdata" if ($TRX_SECURITY_debug == 1); my $n = 0; my $tm = TimeNow(); my $val = ""; my $device_type = uc($def->{TRX_SECURITY_type}); my $sensor = ""; if ($device_type eq "SD90") { $sensor = $firstdevice == 1 ? $def->{TRX_SECURITY_devicelog} : $def->{TRX_SECURITY_devicelog2}; } else { $sensor = $def->{TRX_SECURITY_devicelog}; } $current =$command; if (($device_type eq "DS10A") || ($device_type eq "VISONIC_WINDOW")) { $current = "Error"; $current = "Open" if ($command eq "alert"); $current = "Closed" if ($command eq "normal"); } readingsBeginUpdate($def); if (($device_type ne "KR18") || ($device_type ne "VISONIC_REMOTE")) { if ($firstdevice == 1) { $val .= $current; } readingsBulkUpdate($def, $sensor, $current); # KD101 does not show normal, so statechange does not make sense if (($def->{STATE} ne $val) && ($device_type ne "KD101")) { $sensor = "statechange"; readingsBulkUpdate($def, $sensor, $current); } } else { # kr18 remote control or VISONIC_REMOTE $current = $command; #$sensor = $def->{TRX_SECURITY_devicelog}; $val = $current; readingsBulkUpdate($def, $sensor, $current); my @cmd_split = split(/-/, $command); $sensor = $cmd_split[0]; $current = $cmd_split[1]; readingsBulkUpdate($def, $sensor, $current); } if ($battery ne "") { $sensor = "battery"; $current = "Error"; $current = "ok" if ($battery eq "batt_ok"); $current = "low" if ($battery eq "batt_low"); readingsBulkUpdate($def, $sensor, $current); } if ($delay ne '') { $sensor = "delay"; $current = "Error"; $current = "min" if ($delay eq "min_delay"); $current = "max" if ($delay eq "max_delay"); readingsBulkUpdate($def, $sensor, $current); } if (($firstdevice == 1) && $val) { $def->{STATE} = $val; readingsBulkUpdate($def, "state", $val); } readingsEndUpdate($def, 1); return $name; } sub TRX_SECURITY_Parse($$) { my ($iohash, $hexline) = @_; my $time = time(); # convert to binary my $msg = pack('H*', $hexline); if ($time_old ==0) { Log 5, "TRX_SECURITY: decoding delay=0 hex=$hexline"; } else { my $time_diff = $time - $time_old ; Log 5, "TRX_SECURITY: decoding delay=$time_diff hex=$hexline"; } $time_old = $time; # convert string to array of bytes. Skip length byte my @rfxcom_data_array = (); foreach (split(//, substr($msg,1))) { push (@rfxcom_data_array, ord($_) ); } my $num_bytes = ord($msg); if ($num_bytes < 3) { return ""; } my $type = $rfxcom_data_array[0]; #Log 1, "TRX_SECURITY: X10Sec num_bytes=$num_bytes hex=$hexline type=$type" if ($TRX_SECURITY_debug == 1); my $res = ""; if ($type == 0x20) { Log 1, "TRX_SECURITY: X10Sec num_bytes=$num_bytes hex=$hexline" if ($TRX_SECURITY_debug == 1); $res = TRX_SECURITY_parse_X10Sec(\@rfxcom_data_array); Log 1, "TRX_SECURITY: unsupported hex=$hexline" if ($res eq ""); return $res; } else { Log 0, "TRX_SECURITY: not implemented num_bytes=$num_bytes hex=$hexline"; } return ""; } 1; =pod =begin html

TRX_SECURITY

    The TRX_SECURITY module interprets X10, KD101 and Visonic security sensors received by a RFXCOM RFXtrx433 RF receiver. You need to define an RFXtrx433 receiver first. See TRX.

    Define
      define <name> TRX_SECURITY <type> <deviceid> <devicelog> [<deviceid2> <devicelog2>]

      <type>
        specifies one of the following security devices:
        • DS10A (X10 security ds10a Door/Window Sensor or compatible devices. This device type reports the status of the switch [Open/Closed], status of the delay switch [min|max]], and battery status [ok|low].)
        • MS10A (X10 security ms10a motion sensor. This device type reports the status of motion sensor [normal|alert] and battery status [ok|low].))
        • SD90 (Marmitek sd90 smoke detector. This device type reports the status of the smoke detector [normal|alert] and battery status [ok|low].)
        • KR18 (X10 security remote control. Report the Reading "Security" with values [Arm|Disarm], "ButtonA" and "ButtonB" with values [on|off] )
        • KD101 (KD101 smoke sensor. Report the Reading "smoke" with values [normal|alert])
        • VISONIC_WINDOW (VISONIC security Door/Window Sensor or compatible devices. This device type reports the status of the switch [Open/Closed] and battery status [ok|low].)
        • VISONIC_MOTION (VISONIC security motion sensor. This device type reports the status of motion sensor [normal|alert] and battery status [ok|low].))

      <deviceid>
        specifies the first device id of the device. X10 security (DS10A, MS10A) and SD90 have a a 16 bit device id which has to be written as a hex-string (example "5a54"). All other devices have a 24 bit device id.

      <devicelog>
        is the name of the Reading used to report. Suggested: "Window" or "Door" for ds10a, "motion" for motion sensors, "smoke" for sd90.

      <deviceid2>
        is optional and specifies the second device id of the device if it exists. For example sd90 smoke sensors can be configured to report two device ids.

      <devicelog2>
        is optional for the name used for the Reading of <deviceid2>.

      Example:
      define livingroom_window TRX_SECURITY ds10a 72cd Window
      define motion_sensor1 TRX_SECURITY ms10a 55c6 motion
      define smoke_sensor1 TRX_SECURITY sd90 54d3 Smoke 54d3 Smoketest

    Set
      set <name> <value>

      where value is one of:
          alert              # only for KD101
          pair               # only for KD101
          
      Example:
      set TRX_KD101_a5ca00 alert

    Get
      N/A

    Attributes
=end html =cut