package PandoraFMS::DataServer; ########################################################################## # Pandora FMS Data Server. # Pandora FMS. the Flexible Monitoring System. http://www.pandorafms.org ########################################################################## # Copyright (c) 2005-2021 Artica Soluciones Tecnologicas S.L # # This program is free software; you can redistribute it and/or # modify it under the terms of the GNU Lesser General Public License # as published by the Free Software Foundation; version 2 # This program 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 this program; if not, write to the Free Software # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. ########################################################################## use strict; use warnings; use threads; use threads::shared; use Thread::Semaphore; use Time::Local; use XML::Parser::Expat; use XML::Simple; eval "use POSIX::strftime::GNU;1" if ($^O =~ /win/i); use POSIX qw(setsid strftime); use IO::Uncompress::Unzip; use JSON qw(decode_json); use MIME::Base64; # Required for file names with accents use Encode qw(decode); use Encode::Locale (); # For Reverse Geocoding use LWP::Simple; # Default lib dir for RPM and DEB packages BEGIN { push @INC, '/usr/lib/perl5'; } use PandoraFMS::Tools; use PandoraFMS::DB; use PandoraFMS::Core; use PandoraFMS::ProducerConsumerServer; use PandoraFMS::GIS; # Inherits from PandoraFMS::ProducerConsumerServer our @ISA = qw(PandoraFMS::ProducerConsumerServer); # Global variables my @TaskQueue :shared; my %PendingTasks :shared; my %Agents :shared; my $Sem :shared; my $TaskSem :shared; my $AgentSem :shared; my $XMLinSem :shared; ######################################################################################## # Data Server class constructor. ######################################################################################## sub new ($$;$) { my ($class, $config, $dbh) = @_; return undef unless $config->{'dataserver'} == 1; # Initialize semaphores and queues @TaskQueue = (); %PendingTasks = (); %Agents = (); $Sem = Thread::Semaphore->new; $TaskSem = Thread::Semaphore->new (0); $AgentSem = Thread::Semaphore->new (1); $XMLinSem = Thread::Semaphore->new (1); # Call the constructor of the parent class my $self; if ($config->{'dataserver_smart_queue'} == 0) { $self = $class->SUPER::new($config, DATASERVER, \&PandoraFMS::DataServer::data_producer, \&PandoraFMS::DataServer::data_consumer, $dbh); } else { logger($config, "Smart queue enabled for the Pandora FMS DataServer.", 3); $self = $class->SUPER::new($config, DATASERVER, \&PandoraFMS::DataServer::data_producer_smart_queue, \&PandoraFMS::DataServer::data_consumer, $dbh); } # Load external .enc files for XML::Parser. if ($config->{'enc_dir'} ne '') { push(@XML::Parser::Expat::Encoding_Path, $config->{'enc_dir'}); if ($XML::Simple::PREFERRED_PARSER eq 'XML::SAX::ExpatXS') { push(@XML::SAX::ExpatXS::Encoding::Encoding_Path, $config->{'enc_dir'}); } } if ($config->{'autocreate_group_name'} ne '') { if (get_group_id($dbh, $config->{'autocreate_group_name'}) == -1) { my $msg = "Group '" . $config->{'autocreate_group_name'} . "' does not exist (check autocreate_group_name config token)."; logger($config, $msg, 3); print_message($config, $msg, 1); pandora_event ($config, $msg, 0, 0, 0, 0, 0, 'error', 0, $dbh); } } elsif ($config->{'autocreate_group'} > 0) { if (!defined(get_group_name ($dbh, $config->{'autocreate_group'}))) { my $msg = "Group id " . $config->{'autocreate_group'} . " does not exist (check autocreate_group config token)."; logger($config, $msg, 3); print_message($config, $msg, 1); pandora_event ($config, $msg, 0, 0, 0, 0, 0, 'error', 0, $dbh); } } bless $self, $class; return $self; } ############################################################################### # Run. ############################################################################### sub run ($) { my $self = shift; my $pa_config = $self->getConfig (); print_message ($pa_config, " [*] Starting " . $pa_config->{'rb_product_name'} . " Data Server.", 1); $self->setNumThreads ($pa_config->{'dataserver_threads'}); $self->SUPER::run (\@TaskQueue, \%PendingTasks, $Sem, $TaskSem); } ############################################################################### # Data producer. ############################################################################### sub data_producer ($) { my $self = shift; my ($pa_config, $dbh) = ($self->getConfig (), $self->getDBH ()); my @tasks; my @files; my @sorted; # Open the incoming directory opendir (DIR, $pa_config->{'incomingdir'}) || die "[FATAL] Cannot open Incoming data directory at " . $pa_config->{'incomingdir'} . ": $!"; # Do not read more than max_queue_files files my $file_count = 0; while (my $file = readdir (DIR)) { $file = Encode::decode( locale_fs => $file ); # Data files must have the extension .data next if ($file !~ /^.*[\._]\d+\.data$/); # Do not queue more than max_queue_files files if ($file_count >= $pa_config->{"max_queue_files"}) { last; } push (@files, $file); $file_count++; } closedir(DIR); # Sort the queue { # Temporarily disable warnings (some files may have been deleted) no warnings; if ($pa_config->{'dataserver_lifo'} == 0) { @sorted = sort { -M $pa_config->{'incomingdir'} . "/$b" <=> -M $pa_config->{'incomingdir'} . "/$a" || $a cmp $b } (@files); } else { @sorted = sort { -M $pa_config->{'incomingdir'} . "/$a" <=> -M $pa_config->{'incomingdir'} . "/$b" || $b cmp $a } (@files); } } # Do not process more than one XML from the same agent at the same time foreach my $file (@sorted) { next if ($file !~ /^(.*)[\._]\d+\.data$/); my $agent_name = $1; next if (agent_lock($pa_config, $dbh, $agent_name) == 0); push (@tasks, $file); } return @tasks; } ############################################################################### # Data producer with smart queuing. ############################################################################### sub data_producer_smart_queue ($) { my $self = shift; my ($pa_config, $dbh) = ($self->getConfig (), $self->getDBH ()); my @tasks; my @files; my @sorted; # Open the incoming directory opendir (DIR, $pa_config->{'incomingdir'}) || die "[FATAL] Cannot open Incoming data directory at " . $pa_config->{'incomingdir'} . ": $!"; # Do not read more than max_queue_files files my $smart_queue = {}; while (my $file = readdir (DIR)) { $file = Encode::decode( locale_fs => $file ); # Data files must have the extension .data next if ($file !~ /^(.*)[\._]\d+\.data$/); my $agent_name = $1; # Queue a new file. if (!defined($smart_queue->{$agent_name})) { $smart_queue->{$agent_name} = $file; } # Or update a file in the queue. else { # Always work in LIFO mode. if (-M $pa_config->{'incomingdir'} . '/' . $file < -M $pa_config->{'incomingdir'} . '/' . $smart_queue->{$agent_name}) { $smart_queue->{$agent_name} = $file; } } } closedir(DIR); # Do not process more than one XML from the same agent at the same time: while (my ($agent_name, $file) = each(%{$smart_queue})) { next if (agent_lock($pa_config, $dbh, $agent_name) == 0); push (@tasks, $file); } return @tasks; } ############################################################################### # Data consumer. ############################################################################### sub data_consumer ($$) { my ($self, $task) = @_; my ($pa_config, $dbh) = ($self->getConfig (), $self->getDBH ()); return unless ($task =~ /^(.*)[\._]\d+\.data$/); my $agent_name = $1; my $file_name = $pa_config->{'incomingdir'}; my $xml_err; my $error; # Fix path $file_name .= "/" unless (substr ($file_name, -1, 1) eq '/'); $file_name .= $task; # Double check that the file exists if (! -f $file_name) { agent_unlock($pa_config, $agent_name); return; } # Try to parse the XML 2 times, with a delay between tries of 2 seconds my $xml_data; for (0..1) { eval { local $SIG{__DIE__}; threads->yield; # XML::SAX::ExpatXS is not thread safe. if ($XML::Simple::PREFERRED_PARSER eq 'XML::SAX::ExpatXS') { $XMLinSem->down(); } $xml_data = XMLin ($file_name, forcearray => 'module'); if ($XML::Simple::PREFERRED_PARSER eq 'XML::SAX::ExpatXS') { $XMLinSem->up(); } }; # Invalid XML if ($@) { $error = 1; if ($XML::Simple::PREFERRED_PARSER eq 'XML::SAX::ExpatXS') { $XMLinSem->up(); } } if ($error || ref($xml_data) ne 'HASH') { if ($@) { $xml_err = $@; } else { $xml_err = "Invalid XML format."; } logger($pa_config, "Failed to parse $file_name $xml_err", 3); sleep (2); next; } # Ignore the timestamp in the XML and use the file timestamp instead # If 1 => uses timestamp from received XML #5763. $xml_data->{'timestamp'} = strftime ("%Y-%m-%d %H:%M:%S", localtime((stat($file_name))[9])) if ($pa_config->{'use_xml_timestamp'} eq '0' || ! defined ($xml_data->{'timestamp'})); # Double check that the file exists if (! -f $file_name) { agent_unlock($pa_config, $agent_name); return; } unlink ($file_name); eval { if (defined($xml_data->{'server_name'})) { process_xml_server ($self->getConfig (), $file_name, $xml_data, $self->getDBH ()); } elsif (defined($xml_data->{'connection_source'})) { enterprise_hook('process_xml_connections', [$self->getConfig (), $file_name, $xml_data, $self->getDBH ()]); } elsif (defined($xml_data->{'ipam_source'})) { enterprise_hook('process_xml_ipam', [$self->getConfig (), $file_name, $xml_data, $self->getDBH ()]); } elsif (defined($xml_data->{'network_matrix'})){ process_xml_matrix_network( $self->getConfig(), $xml_data, $self->getDBH()); } else { process_xml_data ($self->getConfig (), $file_name, $xml_data, $self->getServerID (), $self->getDBH ()); } }; agent_unlock($pa_config, $agent_name); return; } rename($file_name, $file_name . '_BADXML'); pandora_event ($pa_config, "Unable to process XML data file '$file_name': $xml_err", 0, 0, 0, 0, 0, 'error', 0, $dbh); agent_unlock($pa_config, $agent_name); } ############################################################################### # Process XML data coming from an agent. ############################################################################### sub process_xml_data ($$$$$) { my ($pa_config, $file_name, $data, $server_id, $dbh) = @_; my ($agent_name, $agent_version, $timestamp, $interval, $os_version, $timezone_offset, $custom_id, $url_address) = ($data->{'agent_name'}, $data->{'version'}, $data->{'timestamp'}, $data->{'interval'}, $data->{'os_version'}, $data->{'timezone_offset'}, $data->{'custom_id'}, $data->{'url_address'}); # Timezone offset must be an integer beween -12 and +12 if (!defined($timezone_offset) || $timezone_offset !~ /[-+]?\d+/) { $timezone_offset = 0; } # If set by server, do not use offset. if ($pa_config->{'use_xml_timestamp'} eq '0') { $timezone_offset = 0; } # Parent Agent Name my $parent_id = 0; # Default value for unknown parent my $parent_agent_name = $data->{'parent_agent_name'}; if (defined ($parent_agent_name) && $parent_agent_name ne '') { $parent_id = get_agent_id ($dbh, $parent_agent_name); if ($parent_id < 1) { # Unknown parent $parent_id = 0; } } # Get agent mode my $agent_mode = 1; # Default value learning mode $agent_mode = $data->{'agent_mode'} if (defined ($data->{'agent_mode'})); # Unknown agent! if (! defined ($agent_name) || $agent_name eq '') { logger($pa_config, "$file_name has data from an unnamed agent", 3); return; } # Get current datetime from system if value AUTO is coming in the XML if ( $data->{'timestamp'} =~ /AUTO/ ){ $timestamp = strftime ("%Y/%m/%d %H:%M:%S", localtime()); } # Apply an offset to the timestamp elsif ($timezone_offset != 0) { # Modify the timestamp with the timezone_offset logger($pa_config, "Applied a timezone offset of $timestamp to agent " . $data->{'agent_name'}, 10); # Calculate the start date to add the offset my $utimestamp = 0; eval { if ($timestamp =~ /(\d+)[\/|\-](\d+)[\/|\-](\d+) +(\d+):(\d+):(\d+)/) { $utimestamp = strftime("%s", $6, $5, $4, $3, $2 -1 , $1 - 1900); } }; # Apply the offset if there were no errors if (! $@ && $utimestamp != 0) { $timestamp = strftime ("%Y-%m-%d %H:%M:%S", localtime($utimestamp + ($timezone_offset * 3600))); } } # Check some variables $interval = 300 if (! defined ($interval) || $interval eq ''); $os_version = undef if (! defined ($os_version) || $os_version eq ''); # Get agent address from the XML if available my $address = '' ; my @address_list; if (defined ($data->{'address'}) && $data->{'address'} ne '') { @address_list = split (',', $data->{'address'}); # Trim addresses for (my $i = 0; $i <= $#address_list; $i++) { $address_list[$i] =~ s/^\s+|\s+$//g ; } # Save the first address as the main address if (defined($address_list[0])) { $address = $address_list[0]; $address =~ s/^\s+|\s+$//g ; shift (@address_list); } } # A module with No-learn mode (modo = 0) creates its modules on database only when it is created my $new_agent = 0; # Get agent id my $agent_id = get_agent_id ($dbh, $agent_name); my $group_id = 0; if ($agent_id < 1) { if ($pa_config->{'autocreate'} == 0) { logger($pa_config, "ERROR: There is no agent defined with name $agent_name", 3); return; } # Get OS, group and description my $os = pandora_get_os ($dbh, $data->{'os_name'}); $group_id = pandora_get_agent_group($pa_config, $dbh, $agent_name, $data->{'group'}, $data->{'group_password'}); if ($group_id <= 0) { pandora_event ($pa_config, "Unable to create agent '" . safe_output($agent_name) . "': No valid group found.", 0, 0, 0, 0, 0, 'error', 0, $dbh); logger($pa_config, "Unable to create agent '" . safe_output($agent_name) . "': No valid group found.", 3); return; } my $description = ''; $description = $data->{'description'} if (defined ($data->{'description'})); my $alias = (defined ($data->{'agent_alias'}) && $data->{'agent_alias'} ne '') ? $data->{'agent_alias'} : $data->{'agent_name'}; my $location = get_geoip_info($pa_config, $address); $agent_id = pandora_create_agent($pa_config, $pa_config->{'servername'}, $agent_name, $address, $group_id, $parent_id, $os, $description, $interval, $dbh, $timezone_offset, $location->{'longitude'}, $location->{'latitude'}, undef, undef, $custom_id, $url_address, $agent_mode, $alias ); if (! defined ($agent_id)) { return; } # Update the secondary groups enterprise_hook('add_secondary_groups_name', [$pa_config, $dbh, $agent_id, $data->{'secondary_groups'}]); # This agent is new. $new_agent = 1; # Add the main address to the address list if ($address ne '') { pandora_add_agent_address($pa_config, $agent_id, $agent_name, $address, $dbh); } # Process custom fields if(defined($data->{'custom_fields'})) { foreach my $custom_fields (@{$data->{'custom_fields'}}) { foreach my $custom_field (@{$custom_fields->{'field'}}) { my $cf_name = get_tag_value ($custom_field, 'name', ''); logger($pa_config, "Processing custom field '" . $cf_name . "'", 10); # Check if the custom field exists my $custom_field_info = get_db_single_row ($dbh, 'SELECT * FROM tagent_custom_fields WHERE name = ?', safe_input($cf_name)); # If it exists add the value to the agent if (defined ($custom_field_info)) { my $cf_value = safe_input(get_tag_value ($custom_field, 'value', '')); my $field_agent; $field_agent->{'id_agent'} = $agent_id; $field_agent->{'id_field'} = $custom_field_info->{'id_field'}; $field_agent->{'description'} = $cf_value; db_process_insert($dbh, 'id_field', 'tagent_custom_data', $field_agent); } else { logger($pa_config, "The custom field '" . $cf_name . "' does not exist. Discarded from XML", 5); } } } } if (defined($pa_config->{'autoconfigure_agents'}) && $pa_config->{'autoconfigure_agents'} == 1) { # Update agent configuration once, before create agent - MetaConsole port to Node enterprise_hook('autoconfigure_agent', [$pa_config, $agent_name, $agent_id, $data, $dbh]); } } # Return if metaconsole, no further analysis. return if (PandoraFMS::Tools::is_metaconsole($pa_config)); # Get the data of the agent, if fail return my $agent = get_db_single_row ($dbh, 'SELECT * FROM tagente WHERE id_agente = ?', $agent_id); if (!defined ($agent)) { logger($pa_config, "Error retrieving information for agent ID $agent_id",10); return; } # Get the ID of the Satellite Server if available. my $satellite_server_id = 0; if (defined($data->{'satellite_server'})) { $satellite_server_id = get_server_id($dbh, $data->{'satellite_server'}, SATELLITESERVER); if ($satellite_server_id < 0) { logger($pa_config, "Satellite Server '" . $data->{'satellite_server'} . "' does not exist.", 10); $satellite_server_id = 0; } } # Check if agent is disabled and return if it's disabled. Disabled agents doesnt process data # in order to avoid not only events, also possible invalid data coming from agents. # But, if agent is in mode autodisable, put it enable and retrieve all data if ($agent->{'disabled'} == 1) { return unless ($agent->{'modo'} == 2); logger($pa_config, "Autodisable agent ID $agent_id is recovered to enable mode.",10); db_do ($dbh, 'UPDATE tagente SET disabled=0 WHERE id_agente=?', $agent_id); } # Do not overwrite agent parameters if the agent is in normal mode if ($agent->{'modo'} == 0) {; $interval = $agent->{'intervalo'}; $os_version = $agent->{'os_version'}; $agent_version = $agent->{'agent_version'}; $timezone_offset = $agent->{'timezone_offset'}; $parent_id = $agent->{'id_parent'}; } # Learning mode else { # Update the main address if ($address ne '' && $address ne $agent->{'direccion'}) { pandora_update_agent_address ($pa_config, $agent_id, $agent_name, $address, $dbh) unless $agent->{'fixed_ip'} == 1; pandora_add_agent_address($pa_config, $agent_id, $agent_name, $address, $dbh); } # Update additional addresses foreach my $address (@address_list) { pandora_add_agent_address($pa_config, $agent_id, $agent_name, $address, $dbh); } # Update parent if is allowed and is valid if ($pa_config->{'update_parent'} == 1 && $parent_id != 0) { logger($pa_config, "Updating agent $agent_name parent_id: $parent_id", 5); } else { $parent_id = $agent->{'id_parent'}; } # Process custom fields for update if(defined($data->{'custom_fields'})) { foreach my $custom_fields (@{$data->{'custom_fields'}}) { foreach my $custom_field (@{$custom_fields->{'field'}}) { my $cf_name = get_tag_value ($custom_field, 'name', ''); logger($pa_config, "Processing custom field '" . $cf_name . "'", 10); # Check if the custom field exists my $custom_field_info = get_db_single_row ($dbh, 'SELECT * FROM tagent_custom_fields WHERE name = ?', safe_input($cf_name)); # If it exists add the value to the agent if (defined ($custom_field_info)) { my $custom_field_data = get_db_single_row($dbh, 'SELECT * FROM tagent_custom_data WHERE id_field = ? AND id_agent = ?', $custom_field_info->{"id_field"}, $agent->{"id_agente"}); my $cf_value = safe_input(get_tag_value ($custom_field, 'value', '')); #If not defined we must create if defined just updated if(!defined($custom_field_data)) { my $field_agent; $field_agent->{'id_agent'} = $agent_id; $field_agent->{'id_field'} = $custom_field_info->{'id_field'}; $field_agent->{'description'} = $cf_value; db_process_insert($dbh, 'id_field', 'tagent_custom_data', $field_agent); } else { db_update ($dbh, "UPDATE tagent_custom_data SET description = ? WHERE id_field = ? AND id_agent = ?", $cf_value, $custom_field_info->{"id_field"}, $agent->{'id_agente'}); } } else { logger($pa_config, "The custom field '" . $cf_name . "' does not exist. Discarded from XML", 5); } } } } } # Update agent information pandora_update_agent($pa_config, $timestamp, $agent_id, $os_version, $agent_version, $interval, $dbh, $timezone_offset, $parent_id, $satellite_server_id); # Update GIS data if ($pa_config->{'activate_gis'} != 0 && $agent->{'update_gis_data'} == 1) { pandora_update_gis_data ($pa_config, $dbh, $agent_id, $agent_name, $data->{'longitude'}, $data->{'latitude'}, $data->{'altitude'}, $data->{'position_description'}, $timestamp); } # Update keep alive modules pandora_module_keep_alive ($pa_config, $agent_id, $agent_name, $server_id, $dbh); # Process modules foreach my $module_data (@{$data->{'module'}}) { my $module_name = get_tag_value ($module_data, 'name', ''); # Clean module_name because sometimes due to errors or problems # creating XMLs it could contain carriage returns and later they # are a pain when you update module configuration because the name won't # save the carriage return. $module_name =~ s/\r//g; $module_name =~ s/\n//g; # Unnamed module next if ($module_name eq ''); my $module_type = get_tag_value ($module_data, 'type', 'generic_data'); # Single data if (! defined ($module_data->{'datalist'})) { my $data_timestamp = get_tag_value ($module_data, 'timestamp', $timestamp); process_module_data ($pa_config, $module_data, $server_id, $agent, $module_name, $module_type, $interval, $data_timestamp, $dbh, $new_agent); next; } # Data list foreach my $list (@{$module_data->{'datalist'}}) { # Empty list next unless defined ($list->{'data'}); foreach my $data (@{$list->{'data'}}) { # No value next unless defined ($data->{'value'}); $module_data->{'data'} = $data->{'value'}; my $data_timestamp = get_tag_value ($data, 'timestamp', $timestamp); if ($pa_config->{'use_xml_timestamp'} eq '0' && defined($timestamp)) { $data_timestamp = $timestamp; } process_module_data ($pa_config, $module_data, $server_id, $agent, $module_name, $module_type, $interval, $data_timestamp, $dbh, $new_agent); } } } # Link modules foreach my $module_data (@{$data->{'module'}}) { my $module_name = get_tag_value ($module_data, 'name', ''); $module_name =~ s/\r//g; $module_name =~ s/\n//g; # Unnamed module next if ($module_name eq ''); # No parent module defined my $parent_module_name = get_tag_value ($module_data, 'module_parent', undef); my $parent_module_unlink = get_tag_value ($module_data, 'module_parent_unlink', undef); next if ( (! defined ($parent_module_name)) && (! defined($parent_module_unlink))); link_modules($pa_config, $dbh, $agent_id, $module_name, $parent_module_name) if (defined($parent_module_name) && ($parent_module_name ne '')); unlink_modules($pa_config, $dbh, $agent_id, $module_name) if (defined($parent_module_unlink) && ($parent_module_unlink eq '1')); } # Process inventory modules process_inventory_data($pa_config, $data, $server_id, $agent_name, $interval, $timestamp, $dbh); # Process log modules enterprise_hook('process_log_data', [$pa_config, $data, $server_id, $agent_name, $interval, $timestamp, $dbh]); # Process snmptrapd modules enterprise_hook('process_snmptrap_data', [$pa_config, $data, $server_id, $dbh]); # Process events process_events_dataserver($pa_config, $data, $agent_id, $group_id, $dbh); # Process discovery modules enterprise_hook('process_discovery_data', [$pa_config, $data, $server_id, $dbh]); # Process command responses enterprise_hook('process_rcmd_report', [$pa_config, $data, $server_id, $dbh, $agent_id, $timestamp]); } ########################################################################## # Process module data, creating module if necessary. ########################################################################## sub process_module_data ($$$$$$$$$$) { my ($pa_config, $data, $server_id, $agent, $module_name, $module_type, $interval, $timestamp, $dbh, $force_processing) = @_; # Get agent data if (! defined ($agent)) { logger($pa_config, "Invalid agent for module '$module_name'.", 3); return; } my $agent_name = $agent->{'nombre'}; # Get module parameters, matching column names in tagente_modulo my $module_conf; # Extra usable fields but not supported at DB level. my $extra = {}; # Supported tags my $tags = {'name' => 0, 'data' => 0, 'type' => 0, 'description' => 0, 'max' => 0, 'min' => 0, 'descripcion' => 0, 'post_process' => 0, 'module_interval' => 0, 'min_critical' => 0, 'max_critical' => 0, 'min_warning' => 0, 'max_warning' => 0, 'disabled' => 0, 'min_ff_event' => 0, 'datalist' => 0, 'status' => 0, 'unit' => 0, 'timestamp' => 0, 'module_group' => 0, 'custom_id' => '', 'str_warning' => '', 'str_critical' => '', 'critical_instructions' => '', 'warning_instructions' => '', 'unknown_instructions' => '', 'tags' => '', 'critical_inverse' => 0, 'warning_inverse' => 0, 'quiet' => 0, 'module_ff_interval' => 0, 'alert_template' => '', 'crontab' => '', 'min_ff_event_normal' => 0, 'min_ff_event_warning' => 0, 'min_ff_event_critical' => 0, 'ff_timeout' => 0, 'each_ff' => 0, 'module_parent' => 0, 'module_parent_unlink' => 0, 'cron_interval' => 0, 'ff_type' => 0, 'min_warning_forced' => 0, 'max_warning_forced' => 0, 'min_critical_forced' => 0, 'max_critical_forced' => 0, 'str_warning_forced' => 0, 'str_critical_forced' => 0 }; # Other tags will be saved here $module_conf->{'extended_info'} = ''; # Read tags while (my ($tag, $value) = each (%{$data})) { if (defined ($tags->{$tag})) { $module_conf->{$tag} = get_tag_value ($data, $tag, ''); } else { $module_conf->{'extended_info'} .= "$tag: " . get_tag_value ($data, $tag, '') . '
'; } } # Reload alert_template to get all alerts like an array $module_conf->{'alert_template'} = get_tag_value ($data, 'alert_template', '', 1); # Description XML tag and column name don't match $module_conf->{'descripcion'} = $module_conf->{'description'}; $module_conf->{'descripcion'} = '' unless defined ($module_conf->{'descripcion'}); delete $module_conf->{'description'}; # Name XML tag and column name don't match $module_conf->{'nombre'} = safe_input($module_name); # Check if module is 'Transactional subsystem status' my $enable_transactional_subsystem = 0; if ($module_conf->{'name'} eq "Transactional subsystem status") { $enable_transactional_subsystem = 1; } delete $module_conf->{'name'}; # Calculate the module interval in seconds if (defined($module_conf->{'cron_interval'})) { $module_conf->{'module_interval'} = $module_conf->{'cron_interval'}; } elsif (defined ($module_conf->{'module_interval'})) { $module_conf->{'module_interval'} = $interval * $module_conf->{'module_interval'}; } else { $module_conf->{'module_interval'} = $interval; } # Allow , as a decimal separator $module_conf->{'post_process'} =~ s/,/./ if (defined ($module_conf->{'post_process'})); # avoid NULL columns $module_conf->{'critical_instructions'} = '' unless defined ($module_conf->{'critical_instructions'}); $module_conf->{'warning_instructions'} = '' unless defined ($module_conf->{'warning_instructions'}); $module_conf->{'unknown_instructions'} = '' unless defined ($module_conf->{'unknown_instructions'}); $module_conf->{'disabled_types_event'} = '' unless defined ($module_conf->{'disabled_types_event'}); $module_conf->{'module_macros'} = '' unless defined ($module_conf->{'module_macros'}); # Extract extra fields. foreach my $pk (keys %{$module_conf}) { if ($pk =~ /_forced$/) { $extra->{$pk} = $module_conf->{$pk}; delete $module_conf->{$pk}; } } # Get module data or create it if it does not exist my $module = get_db_single_row ($dbh, 'SELECT * FROM tagente_modulo WHERE id_agente = ? AND ' . db_text ('nombre') . ' = ?', $agent->{'id_agente'}, safe_input($module_name)); if (! defined ($module)) { # This part has a code commentary because it doesn't allow creating new modules on preexistent agents # Do not auto create modules #if ($pa_config->{'autocreate'} ne '1') { # logger($pa_config, "Module '$module_name' not found for agent '$agent_name' and module auto-creation disabled.", 10); # return; #} # Is the agent not learning? if (($agent->{'modo'} == 0) && !($force_processing)) { logger($pa_config, "Learning mode disabled. Skipping module '$module_name' agent '$agent_name'.", 10); return; } # Get the module type $module_conf->{'id_tipo_modulo'} = get_module_id ($dbh, $module_type); if ($module_conf->{'id_tipo_modulo'} <= 0) { logger($pa_config, "Invalid module type '$module_type' for module '$module_name' agent '$agent_name'.", 3); return; } # The group name has to be translated to a group ID if (defined $module_conf->{'module_group'}) { my $id_group_module = get_module_group_id ($dbh, $module_conf->{'module_group'}); if ( $id_group_module >= 0) { $module_conf->{'id_module_group'} = $id_group_module; } delete $module_conf->{'module_group'}; } if ($enable_transactional_subsystem == 1) { # Defines current agent as transactional agent pandora_mark_transactional_agent($dbh, $agent->{'id_agente'}); } $module_conf->{'id_modulo'} = 1; $module_conf->{'id_agente'} = $agent->{'id_agente'}; my $module_tags = undef; if(defined ($module_conf->{'tags'})) { $module_tags = $module_conf->{'tags'}; delete $module_conf->{'tags'}; } my $initial_alert_template = undef; if(defined ($module_conf->{'alert_template'})) { $initial_alert_template = $module_conf->{'alert_template'}; delete $module_conf->{'alert_template'}; } if(cron_check_syntax ($module_conf->{'crontab'})) { $module_conf->{'cron_interval'} = $module_conf->{'crontab'}; } delete $module_conf->{'crontab'}; # module_parent is a special case. It is not stored in the DB, but we will need it later. my $module_parent = $module_conf->{'module_parent'}; delete $module_conf->{'module_parent'}; # module_parent_unlink is a special case. It is not stored in the DB, but we will need it later. my $module_parent_unlink = $module_conf->{'module_parent_unlink'}; delete $module_conf->{'module_parent_unlink'}; # Create the module my $module_id = pandora_create_module_from_hash ($pa_config, $module_conf, $dbh); # Restore module_parent. $module_conf->{'module_parent'} = $module_parent; # Restore module_parent_unlink. $module_conf->{'module_parent_unlink'} = $module_parent_unlink; $module = get_db_single_row ($dbh, 'SELECT * FROM tagente_modulo WHERE id_agente = ? AND ' . db_text('nombre') . ' = ?', $agent->{'id_agente'}, safe_input($module_name)); if (! defined ($module)) { logger($pa_config, "Could not create module '$module_name' for agent '$agent_name'.", 3); return; } # Assign the tags on module if the specified tags exist if(defined ($module_tags)) { logger($pa_config, "Processing module tags '$module_tags' in module '$module_name' for agent '$agent_name'.", 10); my @module_tags = split(/,/, $module_tags); for(my $i=0;$i<=$#module_tags;$i++) { my $tag_info = get_db_single_row ($dbh, 'SELECT * FROM ttag WHERE name = ?', safe_input($module_tags[$i])); if (defined ($tag_info)) { my $tag_module; $tag_module->{'id_tag'} = $tag_info->{'id_tag'}; $tag_module->{'id_agente_modulo'} = $module->{'id_agente_modulo'}; db_process_insert($dbh, 'id_tag', 'ttag_module', $tag_module); } } } # Assign alert-templates if exist if( $initial_alert_template ) { foreach my $individual_template (@{$initial_alert_template}){ my $id_alert_template = get_db_value ($dbh, 'SELECT id FROM talert_templates WHERE talert_templates.name = ?', safe_input($individual_template) ); if( defined($id_alert_template) ) { pandora_create_template_module ($pa_config, $dbh, $module->{'id_agente_modulo'}, $id_alert_template); } } } } else { # Control NULL columns $module->{'descripcion'} = '' unless defined ($module->{'descripcion'}); $module->{'extended_info'} = '' unless defined ($module->{'extended_info'}); # Set default values $module_conf->{'descripcion'} = $module->{'descripcion'} unless defined ($module_conf->{'descripcion'}); $module_conf->{'extended_info'} = $module->{'extended_info'} unless defined ($module_conf->{'extended_info'}); $module_conf->{'module_interval'} = $module->{'module_interval'} unless defined ($module_conf->{'module_interval'}); } # Check if the module is policy linked to update it or not my $policy_linked = 0; if ($module->{'id_policy_module'} != 0) { if ($module->{'policy_adopted'} == 0 || ($module->{'policy_adopted'} == 1 && $module->{'policy_linked'} == 1)) { $policy_linked = 1; } } # Update module configuration if in learning mode and not a policy module if ((($agent->{'modo'} eq '1') || ($agent->{'modo'} eq '2')) && $policy_linked == 0) { update_module_configuration( $pa_config, $dbh, $module, $module_conf, $extra ); } # Module disabled! if ($module->{'disabled'} eq '1') { logger($pa_config, "Skipping disabled module '$module_name' agent '$agent_name'.", 10); return; } # Parse the timestamp and process the module if ($timestamp !~ /(\d+)\/(\d+)\/(\d+) +(\d+):(\d+):(\d+)/ && $timestamp !~ /(\d+)\-(\d+)\-(\d+) +(\d+):(\d+):(\d+)/) { logger($pa_config, "Invalid timestamp '$timestamp' from module '$module_name' agent '$agent_name'.", 3); return; } my $utimestamp; eval { $utimestamp = strftime("%s", $6, $5, $4, $3, $2 - 1, $1 - 1900); }; if ($@) { logger($pa_config, "Invalid timestamp '$timestamp' from module '$module_name' agent '$agent_name'.", 3); return; } #my $value = get_tag_value ($data, 'data', ''); my $data_object = get_module_data($data, $module_type); my $extra_macros = get_macros_for_data($data, $module_type); # Get module status from XML data file if available $module->{'status'} = get_tag_value ($data, 'status', undef); pandora_process_module ($pa_config, $data_object, $agent, $module, $module_type, $timestamp, $utimestamp, $server_id, $dbh, $extra_macros); } ########################################################################## # Retrieve module data from the XML tree. ########################################################################## sub get_module_data($$){ my ($data, $module_type) = @_; my %data_object; # Log4x modules hava extended information if ($module_type eq 'log4x') { foreach my $attr ('severity','message', 'stacktrace'){ $data_object{$attr} = get_tag_value ($data, $attr, ''); } } else { $data_object{'data'} = get_tag_value ($data, 'data', ''); } return \%data_object; } ########################################################################## # Retrieve module data from the XML tree. ########################################################################## sub get_macros_for_data($$){ my ($data, $module_type) = @_; my %macros; if ($module_type eq 'log4x') { foreach my $attr ('severity','message', 'stacktrace') { $macros{'_' . $attr . '_'} = get_tag_value ($data, $attr, ''); } } return \%macros; } ########################################################################## # Update module configuration in tagente_modulo if necessary. ########################################################################## sub update_module_configuration ($$$$$) { my ($pa_config, $dbh, $module, $module_conf, $extra) = @_; # Update if at least one of the configuration tokens has changed foreach my $conf_token ('descripcion', 'extended_info', 'module_interval') { if ($module->{$conf_token} ne $module_conf->{$conf_token}) { logger ($pa_config, "Updating configuration for module '" . safe_output($module->{'nombre'}) . "'.", 10); db_do ($dbh, 'UPDATE tagente_modulo SET descripcion = ?, extended_info = ?, module_interval = ? WHERE id_agente_modulo = ?', $module_conf->{'descripcion'} eq '' ? $module->{'descripcion'} : $module_conf->{'descripcion'}, $module_conf->{'extended_info'}, $module_conf->{'module_interval'}, $module->{'id_agente_modulo'}); last; } } # Update module hash $module->{'extended_info'} = $module_conf->{'extended_info'} if (defined($module_conf->{'extended_info'})) ; $module->{'descripcion'} = ($module_conf->{'descripcion'} eq '') ? $module->{'descripcion'} : $module_conf->{'descripcion'}; $module->{'module_interval'} = ($module_conf->{'module_interval'} eq '') ? $module->{'module_interval'} : $module_conf->{'module_interval'}; # Enterprise updates. enterprise_hook('update_module_fields', [$dbh, $pa_config, $module, $extra]); } ############################################################################### # Process XML data coming from a server. ############################################################################### sub process_xml_server ($$$$) { my ($pa_config, $file_name, $data, $dbh) = @_; my ($server_name, $server_type, $version, $threads, $modules) = ($data->{'server_name'}, $data->{'server_type'}, $data->{'version'}, $data->{'threads'}, $data->{'modules'}); # Unknown server! if (! defined ($server_name) || $server_name eq '') { logger($pa_config, "$file_name has data from an unnamed server", 3); return; } logger($pa_config, "Processing XML from server: $server_name", 10); # Set some default values $server_type = SATELLITESERVER unless defined($server_type); $modules = 0 unless defined($modules); $threads = 0 unless defined($threads); $version = '' unless defined($version); # Update server information pandora_update_server ($pa_config, $dbh, $data->{'server_name'}, 0, 1, $server_type, $threads, $modules, $version, $data->{'keepalive'}); } ############################################################################### # Link two modules ############################################################################### sub link_modules { my ($pa_config, $dbh, $agent_id, $child_name, $parent_name) = @_; # Get the child module ID. my $child_id = get_agent_module_id ($dbh, $child_name, $agent_id); return unless ($child_id != -1); # Get the parent module ID. my $parent_id = get_agent_module_id ($dbh, $parent_name, $agent_id); return unless ($parent_id != -1); # Link them. logger($pa_config, "Linking module $child_name to module $parent_name for agent ID $agent_id", 10); db_do($dbh, "UPDATE tagente_modulo SET parent_module_id = ? WHERE id_agente_modulo = ?", $parent_id, $child_id); } ############################################################################### # Unlink module from parent ############################################################################### sub unlink_modules { my ($pa_config, $dbh, $agent_id, $child_name) = @_; # Get the child module ID. my $child_id = get_agent_module_id ($dbh, $child_name, $agent_id); return unless ($child_id != -1); # Link them. logger($pa_config, "Unlinking parent from module $child_name agent ID $agent_id", 10); db_do($dbh, "UPDATE tagente_modulo SET parent_module_id = 0 WHERE id_agente_modulo = ?", $child_id); } ########################################################################## # Process events in the XML. ########################################################################## sub process_events_dataserver { my ($pa_config, $data, $agent_id, $group_id, $dbh) = @_; return unless defined($data->{'events'}->[0]->{'event'}); foreach my $event (@{$data->{'events'}->[0]->{'event'}}) { next unless defined($event); # Try to decode the base64 inside my $event_info; eval { $event_info = decode_json(decode_base64($event)); }; if ($@) { logger($pa_config, "Error processing base64 event data '$event'.", 5); next; } next unless defined($event_info->{'data'}); pandora_event( $pa_config, $event_info->{'data'}, $group_id, $agent_id, defined($event_info->{'severity'}) ? $event_info->{'severity'} : 0, 0, 0, 'system', 0, $dbh ); } return; } ########################################################################## # Process events in the XML. ########################################################################## sub process_xml_matrix_network { my ($pa_config, $data, $dbh) = @_; my $utimestamp = $data->{'network_matrix'}->[0]->{'utimestamp'}; my $content = $data->{'network_matrix'}->[0]->{'content'}; return unless defined($utimestamp) && defined($content); # Try to decode the base64 inside my $matrix_info; eval { $matrix_info = decode_json(decode_base64($content)); }; if ($@) { logger($pa_config, "Error processing base64 matrix data '$content'.", 5); return; } foreach my $source (keys %$matrix_info) { foreach my $destination (keys %{$matrix_info->{$source}}) { my $matrix_single_data = $matrix_info->{$source}->{$destination}; $matrix_single_data->{'source'} = $source; $matrix_single_data->{'destination'} = $destination; $matrix_single_data->{'utimestamp'} = $utimestamp; eval { db_process_insert($dbh, 'id', 'tnetwork_matrix', $matrix_single_data); }; if ($@) { logger($pa_config, "Error inserted matrix data. Source: $source, destination: $destination, utimestamp: $utimestamp.", 5); } } } return; } ########################################################################## # Get a lock on the given agent. Return 1 on success, 0 otherwise. ########################################################################## sub agent_lock { my ($pa_config, $dbh, $agent_name) = @_; $AgentSem->down (); if (defined ($Agents{$agent_name})) { $AgentSem->up (); return 0; } $Agents{$agent_name} = 1; $AgentSem->up (); return 1; } ########################################################################## # Remove the lock on the given agent. ########################################################################## sub agent_unlock { my ($pa_config, $agent_name) = @_; $AgentSem->down (); delete ($Agents{$agent_name}); $AgentSem->up (); } 1; __END__