From 0d4692827489699a86ee29cbcc85961316751441 Mon Sep 17 00:00:00 2001 From: nasseeder1 Date: Wed, 16 Aug 2017 22:32:44 +0000 Subject: [PATCH] 93_Log2Syslog.pm: new to contrib git-svn-id: https://svn.fhem.de/fhem/trunk@14910 2b470e98-0d58-463d-a4d8-8e2adae1ed80 --- fhem/MAINTAINER.txt | 3 +- fhem/contrib/93_Log2Syslog.pm | 572 ++++++++++++++++++++++++++++++++++ 2 files changed, 574 insertions(+), 1 deletion(-) create mode 100644 fhem/contrib/93_Log2Syslog.pm diff --git a/fhem/MAINTAINER.txt b/fhem/MAINTAINER.txt index c032eedd5..73a7c9165 100644 --- a/fhem/MAINTAINER.txt +++ b/fhem/MAINTAINER.txt @@ -495,7 +495,8 @@ FHEM/GPUtils.pm ntruchsess http://forum.fhem.de FHEM Deve contrib/23_WEBTHERM.pm betateilchen/sachag http://forum.fhem.de Sonstiges contrib/70_ONKYO_AVR_PULL.pm loredo http://forum.fhem.de Multimedia -contrib/92_rsyslog.pm DS_Starter +contrib/92_rsyslog.pm DS_Starter http://forum.fhem.de Automatisierung +contrib/93_Log2Syslog.pm DS_Starter http://forum.fhem.de Automatisierung contrib/98_exportdevice.pm loredo http://forum.fhem.de Sonstiges contrib/betateilchen/* betateilchen http://forum.fhem.de Sonstiges contrib/holiday betateilchen http://forum.fhem.de Sonstiges diff --git a/fhem/contrib/93_Log2Syslog.pm b/fhem/contrib/93_Log2Syslog.pm new file mode 100644 index 000000000..b5e1892b8 --- /dev/null +++ b/fhem/contrib/93_Log2Syslog.pm @@ -0,0 +1,572 @@ +####################################################################################################### +# $Id: $ +####################################################################################################### +# 93_Log2Syslog.pm +# +# (c) 2017 by Heiko Maaz +# e-mail: Heiko dot Maaz at t-online dot de +# +# This script 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 . +# +# The module based on idea and input from betateilchen 92_rsyslog.pm +# Implements the Syslog Protocol of RFC 5424 (https://tools.ietf.org/html/rfc5424) +# +####################################################################################################### +# Versions History: +# +# 2.0.0 16.08.2017 create syslog without SYS::SYSLOG +# 1.1.1 13.08.2017 registrate fhemlog_log to %loginform in case of sending fhem-log +# attribute timeout, commandref revised +# 1.1.0 26.07.2017 add regex search to sub fhemlog_Log +# 1.0.0 25.07.2017 initial version + +package main; + +use strict; +use warnings; +eval "use IO::Socket::INET;1" or my $MissModulSocket = "IO::Socket::INET"; +eval "use Net::Domain qw(hostfqdn);1" or my $MissModulNDom = "Net::Domain"; + +my $Log2SyslogVn = "2.0.0"; + +# Mappinghash BSD-Formatierung Monat +my %Log2Syslog_BSDMonth = ( + "01" => "Jan", + "02" => "Feb", + "03" => "Mar", + "04" => "Apr", + "05" => "May", + "06" => "Jun", + "07" => "Jul", + "08" => "Aug", + "09" => "Sep", + "10" => "Oct", + "11" => "Nov", + "12" => "Dec" +); + +##################################### +sub Log2Syslog_Initialize($) { + my ($hash) = @_; + + $hash->{DefFn} = "Log2Syslog_Define"; + $hash->{UndefFn} = "Log2Syslog_Undef"; + $hash->{DeleteFn} = "Log2Syslog_Delete"; + $hash->{AttrFn} = "Log2Syslog_Attr"; + $hash->{NotifyFn} = "Log2Syslog_Main"; + + $hash->{AttrList} = "addStateEvent:1,0 ". + "disable:1,0 ". + "addTimestamp:0,1 ". + "logFormat:BSD,IETF ". + "type:UDP,TCP ". + "port " + ; +return undef; +} + +############################################################## +sub Log2Syslog_Define($@) { + my ($hash, $def) = @_; + my @a = split("[ \t][ \t]*", $def); + + return "Error: Perl module ".$MissModulSocket." is missing. Install it on Debian with: sudo apt-get install libio-socket-multicast-perl" if($MissModulSocket); + return "Error: Perl module ".$MissModulNDom." is missing." if($MissModulNDom); + + if (int(devspec2array('TYPE=Log2Syslog')) > 1) { + my @ldvs = devspec2array('TYPE=Log2Syslog'); + my $ldvs = shift(@ldvs); + return "Log2Syslog device '$ldvs' is already defined. Only one device can be defined ! "; + } + + # Example: define splunklog Log2Syslog splunk.myds.me event:.* fhem:.* + return "wrong syntax, use: define Log2Syslog [event:] [fhem:] " + if(int(@a)-3 < 0); + + delete($hash->{HELPER}{EVNTLOG}); + delete($hash->{HELPER}{FHEMLOG}); + + if ($a[3]) { + $hash->{HELPER}{EVNTLOG} = (split("event:",$a[3]))[1] if(lc($a[3]) =~ m/^event:.*/); + $hash->{HELPER}{FHEMLOG} = (split("fhem:",$a[3]))[1] if(lc($a[3]) =~ m/^fhem:.*/); + } + if ($a[4]) { + $hash->{HELPER}{EVNTLOG} = (split("event:",$a[4]))[1] if(lc($a[4]) =~ m/^event:.*/); + $hash->{HELPER}{FHEMLOG} = (split("fhem:",$a[4]))[1] if(lc($a[4]) =~ m/^fhem:.*/); + } + return "Bad regexp: starting with *" + if((defined($hash->{HELPER}{EVNTLOG}) && $hash->{HELPER}{EVNTLOG} =~ m/^\*/) || (defined($hash->{HELPER}{FHEMLOG}) && $hash->{HELPER}{FHEMLOG} =~ m/^\*/)); + eval { "Hallo" =~ m/^$hash->{HELPER}{EVNTLOG}$/ } if($hash->{HELPER}{EVNTLOG}); + return "Bad regexp: $@" if($@); + eval { "Hallo" =~ m/^$hash->{HELPER}{FHEMLOG}$/ } if($hash->{HELPER}{FHEMLOG}); + return "Bad regexp: $@" if($@); + + $hash->{PEERHOST} = $a[2]; # Destination Host (Syslog Server) + $hash->{MYHOST} = hostfqdn (); # FQDN eigener Host + $hash->{HELPER}{PID} = $$; # PROCID in IETF + $hash->{VERSION} = $Log2SyslogVn; + $logInform{$hash->{NAME}} = "Log2Syslog_Main"; # Funktion die in hash %loginform für $name eingetragen wird + + readingsSingleUpdate($hash, "state", "initialized", 1); + +return undef; +} + +sub Log2Syslog_Undef($$) { + my ($hash, $name) = @_; + $hash->{HELPER}{SOCK}->close() if($hash->{HELPER}{SOCK}); +return undef; +} + +sub Log2Syslog_Delete($$) { + my ($hash, $arg) = @_; + $hash->{HELPER}{SOCK}->close() if($hash->{HELPER}{SOCK}); + delete $logInform{$hash->{NAME}}; +return undef; +} + +################################################################ +sub Log2Syslog_Attr { + my ($cmd,$name,$aName,$aVal) = @_; + my $hash = $defs{$name}; + my $do; + + # $cmd can be "del" or "set" + # $name is device name + # aName and aVal are Attribute name and value + + if ($aName eq "disable") { + if($cmd eq "set") { + $do = ($aVal) ? 1 : 0; + } + $do = 0 if($cmd eq "del"); + my $val = ($do == 1 ? "disabled" : "active"); + + readingsSingleUpdate($hash, "state", $val, 1); + } + + if ($cmd eq "set" && $aName eq "port") { + if($aVal !~ m/^\d+$/) { return " The Value of \"$aName\" is not valid. Use only figures !";} + } + + if ($aName =~ m/^.*logFilter$/) { + return "Bad regexp: starting with *" if($aVal =~ m/^\*/); + eval { "Hallo" =~ m/^$aVal$/ }; + return "Bad regexp: $@" if($@); + } + +return undef; +} + +################################################################################# +# Main sub +################################################################################# +sub Log2Syslog_Main ($$) { + my ($a,$b) = @_; + my ($hash,$name,$dev,$raw); + my $call = "init"; + my $ret; + + if (ref($a) eq "HASH") { + # call from notifyFn + # $hash is my entry, $dev is the entry of the changed device + $hash = $a; + $name = $hash->{NAME}; + $dev = $b; + $call = "evt"; + } else { + # call from %loginform + $name = $a; + $hash = $defs{$name}; + $raw = $b; + $call = "log"; + } + return if(IsDisabled($name)); + + my $host = $hash->{PEERHOST}; + my $port = AttrVal($name, "port", 514); + my $type = lc(AttrVal($name, "type", "udp")); + my $st = "active"; + + # Create Socket and check if successful + $hash->{HELPER}{SOCK} = new IO::Socket::INET (PeerHost => $host, PeerPort => $port, Proto => $type); + + if ($hash->{HELPER}{SOCK}) { + if($call eq "log" && defined($hash->{HELPER}{FHEMLOG})) { + # FHEM System-Logs + $ret = fhemlog_Log($name,$raw); + } + if ($call eq "evt" && defined($hash->{HELPER}{EVNTLOG})) { + # FHEM Eventlogs + $ret = event_Log($hash,$dev); + } + } else { + $st = "unable for open socket for $host, $type, $port"; + } + readingsSingleUpdate($hash, "state", $st, 1) if($st ne OldValue($name)); + $hash->{HELPER}{SOCK}->close() if($hash->{HELPER}{SOCK}); + delete $hash->{HELPER}{SOCK}; + +return $ret; +} + +################################################################################# +# Eventlogging +################################################################################# +sub event_Log($$) { + # $hash is my entry, $dev is the entry of the changed device + my ($hash,$dev) = @_; + my $name = $hash->{NAME}; + my $rex = $hash->{HELPER}{EVNTLOG}; + my $sock = $hash->{HELPER}{SOCK}; + my ($prival); + + return if(!$rex); + my $events = deviceEvents($dev, AttrVal($name, "addStateEvent", 0)); + return if(!$events); + + my $n = $dev->{NAME}; + my $max = int(@{$events}); + my $tn = $dev->{NTFY_TRIGGERTIME}; + my $ct = $dev->{CHANGETIME}; + + for (my $i = 0; $i < $max; $i++) { + my $s = $events->[$i]; + $s = "" if(!defined($s)); + + my $tim = (($ct && $ct->[$i]) ? $ct->[$i] : $tn); + my ($date,$time) = split(" ",$tim); + + if($n =~ m/^$rex$/ || "$n:$s" =~ m/^$rex$/ || "$tim:$n:$s" =~ m/^$rex$/) { + my $otp = "$n -> $s"; + $otp = "$tim $otp" if AttrVal($name,'addTimestamp',0); + $prival = setprival($s); + + my $data = setpayload($hash,$prival,$date,$time,$otp,"event"); + eval {$sock->send($data);}; + } + } + +return ""; +} + +################################################################################# +# FHEM system logging +################################################################################# +sub fhemlog_Log($$) { + my ($name,$raw) = @_; + my $hash = $defs{$name}; + my $rex = $hash->{HELPER}{FHEMLOG}; + my $sock = $hash->{HELPER}{SOCK}; + my ($prival); + + return if(!$rex); + + my ($date,$time,$vbose,undef,$text) = split(" ",$raw,5); + $date =~ s/\./-/g; + my $tim = $date." ".$time; + if($text =~ m/^$rex$/ || "$vbose: $text" =~ m/^$rex$/) { + my $otp = "$vbose: $text"; + $otp = "$tim $otp" if AttrVal($name,'addTimestamp',0); + $prival = setprival($text,$vbose); + + my $data = setpayload($hash,$prival,$date,$time,$otp,"fhem"); + eval {$sock->send($data);}; + } + +return ""; +} + +############################################################################### +# set PRIVAL (severity & facility) +############################################################################### +sub setprival ($;$$) { + my ($text,$vbose)= @_; + my $prival; + + # Priority = (facility * 8) + severity + # https://tools.ietf.org/html/rfc5424 + + # determine facility + my $fac = 5; # facility by syslogd + + # calculate severity + # mapping verbose level to severity + # 0: Critical -> 2 + # 1: Error -> 3 + # 2: Warning -> 4 + # 3: Notice -> 5 + # 4: Informational -> 6 + # 5: Debug -> 7 + + my $sv = 5; # notice (default) + + if ($vbose) { + # map verbose to severity + $sv = 2 if ($vbose == 0); + $sv = 3 if ($vbose == 1); + $sv = 4 if ($vbose == 2); + $sv = 5 if ($vbose == 3); + $sv = 6 if ($vbose == 4); + $sv = 7 if ($vbose == 5); + } + + $sv = 3 if (lc($text) =~ m/error/); # error condition + $sv = 4 if (lc($text) =~ m/warning/); # warning conditions + + $prival = ($fac*8)+$sv; + +return($prival); +} + +############################################################################### +# create Payload for Syslog +############################################################################### +sub setpayload ($$$$$$) { + my ($hash,$prival,$date,$time,$otp,$lt)= @_; + my $name = $hash->{NAME}; + my $ident = $name."_".$lt; + my $myhost = $hash->{MYHOST}?$hash->{MYHOST}:"0.0.0.0"; + my $lf = AttrVal($name, "logFormat", "IETF"); + my $data; + + my ($year,$month,$day) = split("-",$date); + + if ($lf eq "BSD") { + # BSD Protokollformat https://tools.ietf.org/html/rfc3164 + $time = (split(".",$time))[0]; # msec ist nicht erlaubt + $month = $Log2Syslog_BSDMonth{$month}; # Monatsmapping z.B. 01 -> Jan + $day =~ s/0/ /; # in Tagen < 10 muss 0 durch Space ersetzt werden + no warnings 'uninitialized'; + $data = "<$prival>$month $day $time $myhost $ident :$otp"; + use warnings; + } + + if ($lf eq "IETF") { + # IETF Protokollformat https://tools.ietf.org/html/rfc5424 + my $pid = $hash->{HELPER}{PID}; + my $mid = "FHEM"; # message ID, identify type of message, e.g. for firewalls + my $tim = $date."T".$time."Z"; + no warnings 'uninitialized'; + $data = "<$prival>1 $tim $myhost $ident $pid $mid - :$otp"; + use warnings; + } + +return($data); +} + +1; + +=pod +=item helper +=item summary send FHEM system logs and/or events to a syslog server. +=item summary_DE leitet FHEM Systemlogs und/oder Events an einen Syslog-Server weiter + +=begin html + + +

Log2Syslog

+
    + Send FHEM system log entries and/or FHEM events to an external syslog server.

    + + Prerequisits +
      +
      + The additional perl module "IO::Socket::INET" must be installed on your system.
      + Install this package from cpan or by

      + + apt-get install libio-socket-multicast-perl (only on Debian based installations)
      +
    +
    + + + Define +
      +
      + define <name> Log2Syslog <destination host> [event:] [fhem:]
      +
      + + <destination host> = host where the syslog server is running
      + [event:] = optional regex to filter events for logging
      + [fhem:] = optional regex to filter fhem system log for logging

      + + After definition the new device sends all new appearing fhem systemlog entries and events to the destination host, + port=514/UDP format:IETF, immediately without further settings if the regex for fhem or event were set.
      + Without setting regex no fhem system log or event log will be forwarded.
      + The module uses functions of Perls SYS::SYSLOG module. If you want know more, see for detailed descriptions + on perldoc.
      + +
      + Example to log anything:
      +
      + define splunklog Log2Syslog fhemtest 192.168.2.49 event:.* fhem:.*
      +
      + will produce output like:
      +
      Mar 20 15:25:22 fhem-vm-8 fhem: global: SAVE
      +Mar 20 15:25:44 fhem-vm-8 fhem: global: SHUTDOWN
      +Mar 20 15:25:57 fhem-vm-8 fhem: global: INITIALIZED
      +Mar 20 15:26:05 fhem-vm-8 fhem: PegelCux: Niedrigwasser-1: 20.03.2016 18:03
      +Mar 20 15:26:05 fhem-vm-8 fhem: PegelCux: Hochwasser-1: 20.03.2016 23:45
      +
    +
    + + + + Attributes +
      +
      + +
    • addTimestamp [0|1]
      +
      + If set to 1, fhem timestamps will be logged too.
      + Default behavior is to not log these timestamps, because syslog uses own timestamps.
      + Maybe useful if mseclog is activated in fhem.
      +
      + Example output:
      +
      Mar 20 15:47:42 fhem-vm-8 fhem: 2016-03-20_15:47:42 global: SAVE
      +Mar 20 15:47:46 fhem-vm-8 fhem: 2016-03-20_15:47:46 global: SHUTDOWN
      +Mar 20 15:47:53 fhem-vm-8 fhem: 2016-03-20_15:47:53 global: INITIALIZED
      +

    • + +
    • addStateEvent [0|1]
      +
      + If set to 1, events will be completed with "state" if a state-event appears.
      + Default behavior is without getting "state". +

    • + +
    • logFormat [BSD|IETF]
      +
      + Set the syslog protocol format.
      + Default value is "IETF" if not specified. +

    • + +
    • type [TCP|UDP]
      +
      + Sets the socket type which should be used. You can choose UDP or TCP.
      + Default value is "UDP" if not specified. +

    • + +
    • port
      +
      + The port of the syslog server is listening. Default port is 514 if not specified. +

    • + +
    +
    + +
+ +=end html +=begin html_DE + + +

Log2Syslog

+
    + Sendet FHEM Systemlog Einträge und/oder Events an einen externen Syslog-Server weiter.
    +
    + + Voraussetzungen +
      +
      + Es wird das Perl Modul "IO::Socket::INET" benötigt und muss installiert sein.
      + Das Modul kann über CPAN oder mit

      + + apt-get install libio-socket-multicast-perl (auf Debian Linux Systemen)

      + + installiert werden. +
    +
    + + + Definition +
      +
      + define <name> Log2Syslog <destination host> [event:] [fhem:]
      +
      + + <Zielhost> = Host (Name oder IP-Adresse) auf dem der Syslog-Server läuft
      + [event:] = optionaler regulärer Ausdruck zur Filterung von Events zur Weiterleitung
      + [fhem:] = optionaler regulärer Ausdruck zur Filterung von FHEM Logs zur Weiterleitung

      + + Direkt nach der Definition sendet das neue Device alle neu auftretenden FHEM Systemlog Einträge und Events ohne weitere + Einstellungen an den Zielhost, Port=514/UDP Format:IETF, wenn reguläre Ausdrücke für Events/FHEM angegeben wurden.
      + Wurde kein Regex gesetzt, erfolgt keine Weiterleitung von Events oder FHEM Systemlogs.
      + Das Modul verwendet Funktionen des Perl-Moduls SYS::SYSLOG. Weitere Informationen dazu sind auf + perldoc zu finden.
      + +
      + Beispiel:
      +
      + define splunklog Log2Syslog fhemtest 192.168.2.49
      +
      + Es werden alle Events weitergeleitet:
      +
      Mar 20 15:25:22 fhem-vm-8 fhem: global: SAVE
      +Mar 20 15:25:44 fhem-vm-8 fhem: global: SHUTDOWN
      +Mar 20 15:25:57 fhem-vm-8 fhem: global: INITIALIZED
      +Mar 20 15:26:05 fhem-vm-8 fhem: PegelCux: Niedrigwasser-1: 20.03.2016 18:03
      +Mar 20 15:26:05 fhem-vm-8 fhem: PegelCux: Hochwasser-1: 20.03.2016 23:45
      +
    +
    + + + + Attribute +
      +
      + + +
    • addTimestamp [0|1]
      +
      + Wenn gesetzt, werden FHEM Timestamps im Datensatz mit übertragen.
      + Per default werden die Timestamps nicht mit übertragen, da der Syslog-Server eigene Timestamps verwendet.
      + Die Einstellung kann hilfeich sein wenn mseclog in FHEM aktiviert ist.
      +
      + + Beispielausgabe:
      +
      Mar 20 15:47:42 fhem-vm-8 fhem: 2016-03-20_15:47:42 global: SAVE
      +Mar 20 15:47:46 fhem-vm-8 fhem: 2016-03-20_15:47:46 global: SHUTDOWN
      +Mar 20 15:47:53 fhem-vm-8 fhem: 2016-03-20_15:47:53 global: INITIALIZED
      +

    • + +
    • addStateEvent [0|1]
      +
      + Wenn gesetzt, werden state-events mit dem Reading "state" ergänzt.
      + Die Standardeinstellung ist ohne state-Ergänzung. +

    • + +
    • logFormat [BSD|IETF]
      +
      + Stellt das Protokollformat ein.
      + Der Standardwert ist "IETF". +

    • + +
    • type [TCP|UDP]
      +
      + Setzt den Protokolltyp der verwendet werden soll. Es kann UDP oder TCP gewählt werden.
      + Standard ist "UDP" wenn nichts spezifiziert ist. +

    • + +
    • port
      +
      + Der verwendete Port des Syslog-Servers. Default Port ist 514 wenn nicht gesetzt. +

    • + +
    +
    + +
+=end html_DE +=cut \ No newline at end of file