/* Defines a parent class for a Pandora module. Copyright (c) 2006-2023 Pandora FMS. Written by Esteban Sanchez. This program 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, or (at your option) any later version. 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #include "pandora_module.h" #include "pandora_windows_service.h" #include "../misc/pandora_file.h" #include "../pandora_strutils.h" #include "../pandora.h" #include #include #include #define BUFSIZE 4096 using namespace Pandora; using namespace Pandora_Modules; using namespace Pandora_Strutils; /** * Creates a Pandora_Module. * * Initializes all attributes. The default interval is set to 1 loop. * * @param name Module name. */ Pandora_Module::Pandora_Module (string name) { this->module_name = name; this->executions = 0; this->module_interval = 1; this->module_timeout = 15000; this->max = 0; this->min = 0; this->post_process = ""; this->has_limits = false; this->has_min = false; this->has_max = false; this->async = false; this->data_list = NULL; this->inventory_list = NULL; this->precondition_list = NULL; this->condition_list = NULL; this->cron = NULL; this->min_critical = ""; this->max_critical = ""; this->min_warning = ""; this->max_warning = ""; this->disabled = ""; this->min_ff_event = ""; this->intensive_condition_list = NULL; this->intensive_interval = 1; this->timestamp = 0; this->intensive_match = 0; this->unit = ""; this->module_group = ""; this->custom_id = ""; this->str_warning = ""; this->str_critical = ""; this->critical_instructions = ""; this->warning_instructions = ""; this->unknown_instructions = ""; this->tags = ""; this->critical_inverse = ""; this->warning_inverse = ""; this->quiet = ""; this->module_ff_interval = ""; this->module_ff_type = ""; this->module_alert_template = ""; this->module_crontab = ""; this->module_wait_timeout = 500; } /** * Virtual destructor of Pandora_Module. * * Should be redefined by child classes. */ Pandora_Module::~Pandora_Module () { Condition *cond = NULL; Condition *precond = NULL; list::iterator iter; list::iterator iter_pre; /* Clean data lists */ this->cleanDataList (); /* Clean precondition list */ if (this->precondition_list != NULL && this->precondition_list->size () > 0) { iter_pre = this->precondition_list->begin (); for (iter_pre = this->precondition_list->begin (); iter_pre != this->precondition_list->end (); iter_pre++) { /* Free regular expressions */ precond = *iter_pre; if (precond->string_value != "") { regfree (&(precond->regexp)); } delete (*iter_pre); } delete (this->precondition_list); this->precondition_list = NULL; } /* Clean condition list */ if (this->condition_list != NULL && this->condition_list->size () > 0) { iter = this->condition_list->begin (); for (iter = this->condition_list->begin (); iter != this->condition_list->end (); iter++) { /* Free regular expressions */ cond = *iter; if (cond->string_value != "") { regfree (&(cond->regexp)); } delete (*iter); } delete (this->condition_list); this->condition_list = NULL; } /* Clean intensive_condition list */ if (this->intensive_condition_list != NULL && this->intensive_condition_list->size () > 0) { iter = this->intensive_condition_list->begin (); for (iter = this->intensive_condition_list->begin (); iter != this->intensive_condition_list->end (); iter++) { /* Free regular expressions */ cond = *iter; if (cond->string_value != "") { regfree (&(cond->regexp)); } delete (*iter); } delete (this->intensive_condition_list); this->intensive_condition_list = NULL; } /* Clean the module cron */ if (this->cron != NULL) { delete (this->cron); this->cron = NULL; } } void Pandora_Module::cleanDataList () { Pandora_Data *data; list::iterator iter; if (this->data_list) { if (this->data_list->size () > 0) { iter = this->data_list->begin (); for (iter = this->data_list->begin (); iter != this->data_list->end (); iter++) { data = *iter; delete data; } } delete this->data_list; this->data_list = NULL; } if (this->inventory_list) { if (this->inventory_list->size () > 0) { iter = this->inventory_list->begin (); for (iter = this->inventory_list->begin (); iter != this->inventory_list->end (); iter++) { data = *iter; delete data; } } delete this->inventory_list; this->inventory_list = NULL; } } /** * Get the Module_Type from a string type. * * @param type String type. * * @return The Module_Type which represents the type. */ Module_Type Pandora_Module::parseModuleTypeFromString (string type) { if (type == module_generic_data_str || type == "") { return TYPE_GENERIC_DATA; } else if (type == module_generic_data_inc_str) { return TYPE_GENERIC_DATA_INC; } else if (type == module_generic_data_string_str) { return TYPE_GENERIC_DATA_STRING; } else if (type == module_generic_proc_str) { return TYPE_GENERIC_PROC; } else if (type == module_async_data_str) { return TYPE_ASYNC_DATA; } else if (type == module_async_proc_str) { return TYPE_ASYNC_PROC; } else if (type == module_async_string_str) { return TYPE_ASYNC_STRING; } else if (type == module_log_str) { return TYPE_LOG; } else if (type == module_generic_data_inc_abs_str) { return TYPE_GENERIC_DATA_INC_ABS; } else { return TYPE_0; } } /** * Get the Module_Kind from a string Kind. * * @param kind String Kind. * * @return The Module_Kind which represents the Kind. */ Module_Kind Pandora_Module::parseModuleKindFromString (string kind) { if (kind == module_exec_str) { return MODULE_EXEC; } else if (kind == module_proc_str) { return MODULE_PROC; } else if (kind == module_service_str) { return MODULE_SERVICE; } else if (kind == module_freedisk_str) { return MODULE_FREEDISK; } else if (kind == module_freedisk_percent_str) { return MODULE_FREEDISK_PERCENT; } else if (kind == module_freememory_str) { return MODULE_FREEMEMORY; } else if (kind == module_freememory_percent_str) { return MODULE_FREEMEMORY_PERCENT; } else if (kind == module_cpuusage_str) { return MODULE_CPUUSAGE; } else if (kind == module_inventory_str) { return MODULE_INVENTORY; } else if (kind == module_logevent_str) { return MODULE_LOGEVENT; } else if (kind == module_logchannel_str) { return MODULE_LOGCHANNEL; } else if (kind == module_wmiquery_str) { return MODULE_WMIQUERY; } else if (kind == module_perfcounter_str) { return MODULE_PERFCOUNTER; } else if (kind == module_tcpcheck_str) { return MODULE_TCPCHECK; } else if (kind == module_regexp_str) { return MODULE_REGEXP; } else if (kind == module_plugin_str) { return MODULE_PLUGIN; } else if (kind == module_ping_str) { return MODULE_PING; } else if (kind == module_snmpget_str) { return MODULE_SNMPGET; } else { return MODULE_0; } } /** * Get the name of the module. * * @return The name of the module. */ string Pandora_Module::getName () const { return this->module_name; } /** * Get the description of the module. * * @return The module description. */ string Pandora_Module::getDescription () const { return this->module_description; } /** * Get the module type in a human readable string. * * @return The module type.. */ string Pandora_Module::getTypeString () const { return this->module_type_str; } /** * Get the module type in a integer value. * * @return The module type in a integer value. */ Module_Type Pandora_Module::getTypeInt () const { return this->module_type; } /** * Get the kind of the module in a integer_value. * * @return The module kind in a integer value. */ Module_Kind Pandora_Module::getModuleKind () const { return this->module_kind; } /** * Get the output of the module. * * @return The module output in a string value. */ string Pandora_Module::getLatestOutput () const { return this->latest_output; } /** * Get the type of the module in a integer_value. * * @return The module type in a integer value. */ Module_Type Pandora_Module::getModuleType () const { return this->module_type; } /** * Get the module output. * * After running the module, this function will return the output, * based on the module_type and the interval. * * @return The output in a string. * * @exception Output_Error Throwed if the module_type is not correct. * @exception Value_Error Throwed when the output is not in * the interval range. */ string Pandora_Module::getDataOutput (Pandora_Data *data) { double value; if (this->module_type == TYPE_GENERIC_DATA_STRING || this->module_type == TYPE_ASYNC_STRING || this->module_type == TYPE_LOG) { return data->getValue (); } try { value = Pandora_Strutils::strtodouble (data->getValue ()); } catch (Pandora_Strutils::Invalid_Conversion e) { pandoraLog ("Output error on module %s", this->module_name.c_str ()); throw Output_Error (); } if (this->has_limits) { if (value > this->max || value < this->min) { pandoraLog ("The returned value was not in the interval on module %s", this->module_name.c_str ()); throw Value_Error (); } } return trim (data->getValue ()); } /** * Export the module output to en environment variable. */ void Pandora_Module::exportDataOutput () { Pandora_Data *pandora_data = NULL; string putenv_str, module_data; /* putenv expects a string of the form name=value */ putenv_str = this->save + "="; /* No data */ if ( (!this->has_output) || this->data_list == NULL) { putenv (putenv_str.c_str ()); return; } /* Get the module data */ pandora_data = data_list->front (); if (pandora_data == NULL) { putenv (putenv_str.c_str ()); return; } module_data = pandora_data->getValue (); putenv_str += module_data; /* Save it as an environment variable */ putenv (putenv_str.c_str ()); } /** * Set the output of the module. * * If the function is called more than once before calling getXML, the * output will be accumulated and added to a tag. * * @param output Output to add. */ void Pandora_Module::setOutput (string output) { Pandora_Data *data; if (this->data_list == NULL) this->data_list = new list (); data = new Pandora_Data (output, this->module_name); this->data_list->push_back (data); this->latest_output = output; } /** * Set the output of the module. * * If the function is called more than once before calling getXML, the * output will be accumulated and added to a tag. * * @param output Output to add. * @param system_time Timestamp. */ void Pandora_Module::setOutput (string output, SYSTEMTIME *system_time) { Pandora_Data *data; if (this->data_list == NULL) this->data_list = new list (); data = new Pandora_Data (output, system_time, this->module_name); this->data_list->push_back (data); } /** * Set no output for the module. */ void Pandora_Module::setNoOutput () { this->cleanDataList (); this->has_output = false; } /** * Run the module and generates the output. * * It is used by the child classes to check the execution interval * value and increment the executions variable. * * @exception Interval_Not_Fulfilled Throwed when the execution * interval value indicates that the module doesn't have * to execute. */ void Pandora_Module::run () { // Run once. if (this->intensive_interval == 0) { if (this->executions == 0) { has_output = false; throw Interval_Not_Fulfilled (); } } // Run periodically. else if (++this->executions < this->intensive_interval) { pandoraDebug ("%s: Interval is not fulfilled", this->module_name.c_str ()); has_output = false; throw Interval_Not_Fulfilled (); } // Reset the execution counter. this->executions = 0; has_output = true; // Update the execution timestamp. this->updateTimestampFile(); } /** * Get the XML output of the value. * * The output is a element of the TinyXML library. A sample output of * a module is: * @verbatim Conexiones abiertas generic_data 5 Conexiones abiertas @endverbatim * * @return A pointer to the TiXmlElement if successful which has to be * freed by the caller. NULL if the XML could not be created. */ string Pandora_Module::getXml () { ostringstream module_interval, min, max; string module_xml, data_clean, interval_str; Pandora_Data *data; pandoraDebug ("%s getXML begin", module_name.c_str ()); /* No data */ if (!this->has_output || this->data_list == NULL) { return ""; } /* Log module */ if (this->module_type == TYPE_LOG) { module_xml = "\n\tmodule_name; module_xml += "]]>\n\tbase64\n\tdata_list && this->data_list->size () > 1) { list::iterator iter; iter = this->data_list->begin (); for (iter = this->data_list->begin (); iter != this->data_list->end (); iter++) { data = *iter; try { data_clean += this->getDataOutput(data); } catch (Module_Exception e) { continue; } } } else { data = data_list->front (); try { data_clean = this->getDataOutput (data); } catch (Module_Exception e) { } } module_xml += base64Encode(data_clean); module_xml += "]]>"; /* Clean up */ this->cleanDataList (); pandoraDebug ("%s getXML end", module_name.c_str ()); return module_xml; } /* Compose the module XML */ module_xml = "\n\tmodule_name; module_xml += "]]>\n\tmodule_type_str; module_xml += "]]>\n"; /* Description */ if (this->module_description != "") { module_xml += "\tmodule_description; module_xml += "]]>\n"; } /* Interval */ module_interval << this->module_interval; module_xml += "\t\n"; /* Min */ if (this->has_min) { min << this->min; module_xml += "\t\n"; } /* Max */ if (this->has_max) { max << this->max; module_xml += "\t\n"; } /* Post process */ if (this->post_process != "") { module_xml += "\tpost_process; module_xml += "]]>\n"; } /* Min critical */ if (this->min_critical != "") { module_xml += "\tmin_critical; module_xml += "]]>\n"; } /* Max critical */ if (this->max_critical != "") { module_xml += "\tmax_critical; module_xml += "]]>\n"; } /* Min warning */ if (this->min_warning != "") { module_xml += "\tmin_warning; module_xml += "]]>\n"; } /* Max warning */ if (this->max_warning != "") { module_xml += "\tmax_warning; module_xml += "]]>\n"; } /* Disabled */ if (this->disabled != "") { module_xml += "\tdisabled; module_xml += "]]>\n"; } /* Min ff event */ if (this->min_ff_event != "") { module_xml += "\tmin_ff_event; module_xml += "]]>\n"; } /* Unit */ if (this->unit != "") { module_xml += "\tunit; module_xml += "]]>\n"; } /* Module group */ if (this->module_group != "") { module_xml += "\t"; module_xml += this->module_group; module_xml += "\n"; } /* Custom ID */ if (this->custom_id != "") { module_xml += "\t"; module_xml += this->custom_id; module_xml += "\n"; } /* Str warning */ if (this->str_warning != "") { module_xml += "\t"; module_xml += this->str_warning; module_xml += "\n"; } /* Str critical */ if (this->str_critical != "") { module_xml += "\t"; module_xml += this->str_critical; module_xml += "\n"; } /* Critical instructions */ if (this->critical_instructions != "") { module_xml += "\t"; module_xml += this->critical_instructions; module_xml += "\n"; } /* Warning instructions */ if (this->warning_instructions != "") { module_xml += "\t"; module_xml += this->warning_instructions; module_xml += "\n"; } /* Unknown instructions */ if (this->unknown_instructions != "") { module_xml += "\t"; module_xml += this->unknown_instructions; module_xml += "\n"; } /* Tags */ if (this->tags != "") { module_xml += "\t"; module_xml += this->tags; module_xml += "\n"; } /* Critical inverse */ if (this->critical_inverse != "") { module_xml += "\t"; module_xml += this->critical_inverse; module_xml += "\n"; } /* Warning inverse */ if (this->warning_inverse != "") { module_xml += "\t"; module_xml += this->warning_inverse; module_xml += "\n"; } /* Quiet */ if (this->quiet != "") { module_xml += "\t"; module_xml += this->quiet; module_xml += "\n"; } /* Module FF interval */ if (this->module_ff_interval != "") { module_xml += "\t"; module_xml += this->module_ff_interval; module_xml += "\n"; } /* Module FF type */ if (this->module_ff_type != "") { module_xml += "\t"; module_xml += this->module_ff_type; module_xml += "\n"; } /* Module Alert template */ if (this->module_alert_template != "") { module_xml += "\t"; module_xml += this->module_alert_template; module_xml += "\n"; } /* Module Crontab */ if (this->cron->getIsSet()) { module_xml += "\t"; module_xml += this->cron->getCronString(); module_xml += "\n"; module_xml += "\tcron->getCronIntervalStr(); module_xml += "]]>\n"; } /* Write module data */ if (this->data_list && this->data_list->size () > 1) { list::iterator iter; module_xml += "\t\n"; iter = this->data_list->begin (); for (iter = this->data_list->begin (); iter != this->data_list->end (); iter++) { data = *iter; try { data_clean = strreplace (this->getDataOutput (data), "%", "%%" ); } catch (Module_Exception e) { continue; } module_xml += "\t\t\n\t\t\t\n\t\t\tgetTimestamp (); module_xml += "]]>\n\t\t\n"; } module_xml += "\t\n"; } else { data = data_list->front (); try { data_clean = strreplace (this->getDataOutput (data), "%", "%%" ); module_xml += "\t\n"; } catch (Module_Exception e) { } } /* Close the module tag */ module_xml += "\n"; /* Clean up */ this->cleanDataList (); pandoraDebug ("%s getXML end", module_name.c_str ()); return module_xml; } /** * Set the max value the module can have. * * The range is closed, so the value is included. * * @param value Max value to set. */ void Pandora_Module::setMax (int value) { this->has_limits = true; this->has_max = true; this->max = value; } /** * Set the min value the module can have. * * The range is closed, so the value is included. * * @param value Min value to set. */ void Pandora_Module::setMin (int value) { this->has_limits = true; this->has_min = true; this->min = value; } /** * Set the post process value for the module. * * @param value Post process value to set. */ void Pandora_Module::setPostProcess (string value) { this->post_process = value; } /** * Set the min critical value for the module. * * @param value Min critical value to set. */ void Pandora_Module::setMinCritical (string value) { this->min_critical = value; } /** * Set the max critical value for the module. * * @param value Max critical value to set. */ void Pandora_Module::setMaxCritical (string value) { this->max_critical = value; } /** * Set the min warning value for the module. * * @param value Min warning value to set. */ void Pandora_Module::setMinWarning (string value) { this->min_warning = value; } /** * Set the max warning value for the module. * * @param value Max warning value to set. */ void Pandora_Module::setMaxWarning (string value) { this->max_warning = value; } /** * Set the disabled value for the module. * * @param value Disabled value to set. */ void Pandora_Module::setDisabled (string value) { this->disabled = value; } /** * Set the min ff event value for the module. * * @param value Min ff event value to set. */ void Pandora_Module::setMinFFEvent (string value) { this->min_ff_event = value; } /** * Set the unit value for the module. * * @param value unit value to set. */ void Pandora_Module::setUnit (string value) { this->unit = value; } /** * Set the module group for the module. * * @param value module group value to set. */ void Pandora_Module::setModuleGroup (string value) { this->module_group = value; } /** * Set the custom id for the module. * * @param value custom id value to set. */ void Pandora_Module::setCustomId (string value) { this->custom_id = value; } /** * Set the str warning for the module. * * @param value str warning value to set. */ void Pandora_Module::setStrWarning (string value) { this->str_warning = value; } /** * Set the str critical for the module. * * @param value str critical value to set. */ void Pandora_Module::setStrCritical (string value) { this->str_critical = value; } /** * Set the critical instructions for the module. * * @param value critical instructions value to set. */ void Pandora_Module::setCriticalInstructions (string value) { this->critical_instructions = value; } /** * Set the warning instructions for the module. * * @param value warning instructions value to set. */ void Pandora_Module::setWarningInstructions (string value) { this->warning_instructions = value; } /** * Set the unknown instructions for the module. * * @param value unknown instructions value to set. */ void Pandora_Module::setUnknownInstructions (string value) { this->unknown_instructions = value; } /** * Set the tags for the module. * * @param value tags value to set. */ void Pandora_Module::setTags (string value) { this->tags = value; } /** * Set the critical inverse for the module. * * @param value critical inverse value to set. */ void Pandora_Module::setCriticalInverse (string value) { this->critical_inverse = value; } /** * Set the warning inverse for the module. * * @param value warning inverse value to set. */ void Pandora_Module::setWarningInverse (string value) { this->warning_inverse = value; } /** * Set the quiet for the module. * * @param value quiet value to set. */ void Pandora_Module::setQuiet (string value) { this->quiet = value; } /** * Set the module FF interval for the module. * * @param value module FF interval value to set. */ void Pandora_Module::setModuleFFInterval (string value) { this->module_ff_interval = value; } /** * Set the module FF type for the module. * * @param value module FF type value to set. */ void Pandora_Module::setModuleFFType (string value) { this->module_ff_type = value; } /** * Set the module Alert template for the module. * * @param value module Alert template value to set. */ void Pandora_Module::setModuleAlertTemplate (string value) { this->module_alert_template = value; } /** * Set the module Crontab for the module. * * @param value module Crontab value to set. */ void Pandora_Module::setModuleCrontab (string value) { this->module_crontab = value; } /** * Set the async flag to the module. * * If a module is set to be async, it would try to works only when the * events happen. Note that not all the modules can work in async mode. * * @param async Flag to set. */ void Pandora_Module::setAsync (bool async) { this->async = async; } /** * Set the module type from a string type. * * @param type String type. */ void Pandora_Module::setType (string type) { this->module_type_str = type; this->module_type = parseModuleTypeFromString (type); } /** * Set the module kind from a string kind. * * @param kind String kind. */ void Pandora_Module::setKind (string kind) { this->module_kind_str = kind; this->module_kind = parseModuleKindFromString (kind); } /** * Set the interval execution. * * @param interval Interval between executions. */ void Pandora_Module::setInterval (int interval) { this->module_interval = interval; } /** * Set the intensive interval. * * @param intensive_interval Intensive interval. */ void Pandora_Module::setIntensiveInterval (int intensive_interval) { this->intensive_interval = intensive_interval; } /** * Set the execution timeout. * * @param timeout Execution timeout. */ void Pandora_Module::setTimeout (int timeout) { if (timeout < 0) { return; } /* WaitForSingleObject expects milliseconds */ this->module_timeout = timeout * 1000; } /** * Get the execution interval. * * @return The execution interval. */ int Pandora_Module::getInterval () { return this->module_interval; } /** * Get the intensive interval. * * @return The intensive interval. */ int Pandora_Module::getIntensiveInterval () { return this->intensive_interval; } /** * Get the execution timeout. * * @return The execution timeout. */ int Pandora_Module::getTimeout () { return this->module_timeout; } /** * Set the module description. * * @param description Description of the module. */ void Pandora_Module::setDescription (string description) { this->module_description = description; } /** * Set the name of the environment variable where the module data will be saved. * * @param save Name of the environment variable. */ void Pandora_Module::setSave (string save) { this->save = save; } /** * Get the name of the environment variable where the module data will be saved. * * @return The name of the environment variable. */ string Pandora_Module::getSave () { return this->save; } /** * Adds a new condition to a condition list. * * @param condition Condition string. * @param condition_list Pointer to the condition list. */ void Pandora_Module::addGenericCondition (string condition, list **condition_list) { Condition *cond; char operation[256], string_value[1024], command[1024]; /* Create the condition list if it does not exist */ if (*condition_list == NULL) { *condition_list = new list (); } /* Create the new condition */ cond = new Condition; if (cond == NULL) { return; } cond->value_1 = 0; cond->value_2 = 0; /* Numeric comparison */ if (sscanf (condition.c_str (), "%255s %lf %1023[^\n]s", operation, &(cond->value_1), command) == 3) { cond->operation = operation; cond->command = command; cond->command = command; cond->command = "cmd.exe /c \"" + cond->command + "\""; (*condition_list)->push_back (cond); /* Regular expression */ } else if (sscanf (condition.c_str (), "=~ %1023s %1023[^\n]s", string_value, command) == 2) { cond->operation = "=~"; cond->string_value = string_value; cond->command = command; cond->command = "cmd.exe /c \"" + cond->command + "\""; if (regcomp (&(cond->regexp), string_value, REG_EXTENDED) != 0) { pandoraDebug ("Invalid regular expression %s", string_value); delete (cond); return; } (*condition_list)->push_back (cond); /* Interval */ } else if (sscanf (condition.c_str (), "(%lf , %lf) %1023[^\n]s", &(cond->value_1), &(cond->value_2), command) == 3) { cond->operation = "()"; cond->command = command; cond->command = "cmd.exe /c \"" + cond->command + "\""; (*condition_list)->push_back (cond); } else { pandoraLog ("Invalid condition: %s", condition.c_str ()); delete (cond); return; } return; } /** * Adds a new module condition. * * @param condition Condition string. */ void Pandora_Module::addCondition (string condition) { addGenericCondition (condition, &(this->condition_list)); } /** * Adds a new module pre-condition. * * @param condition Condition string. */ void Pandora_Module::addPreCondition (string condition) { addGenericCondition (condition, &(this->precondition_list)); } /** * Adds a new module intensive condition. * * @param condition Condition string. */ void Pandora_Module::addIntensiveCondition (string condition) { Condition *cond; char operation[256], string_value[1024], command[1024]; /* Create the condition list if it does not exist */ if (this->intensive_condition_list == NULL) { this->intensive_condition_list = new list (); } /* Create the new condition */ cond = new Condition; if (cond == NULL) { return; } cond->value_1 = 0; cond->value_2 = 0; /* Numeric comparison */ if (sscanf (condition.c_str (), "%255s %lf", operation, &(cond->value_1)) == 2) { cond->operation = operation; (this->intensive_condition_list)->push_back (cond); /* Regular expression */ } else if (sscanf (condition.c_str (), "=~ %1023s", string_value) == 1) { cond->operation = "=~"; cond->string_value = string_value; if (regcomp (&(cond->regexp), string_value, REG_EXTENDED) != 0) { pandoraDebug ("Invalid regular expression %s", string_value); delete (cond); return; } (this->intensive_condition_list)->push_back (cond); /* Interval */ } else if (sscanf (condition.c_str (), "(%lf , %lf)", &(cond->value_1), &(cond->value_2)) == 2) { cond->operation = "()"; (this->intensive_condition_list)->push_back (cond); } else { pandoraDebug ("Invalid intensive condition: %s", condition.c_str ()); delete (cond); return; } } /** * Evaluates and executes module preconditions. */ int Pandora_Module::evaluatePreconditions () { STARTUPINFO si; PROCESS_INFORMATION pi; DWORD retval, dwRet; SECURITY_ATTRIBUTES attributes; HANDLE out, new_stdout, out_read, job; string working_dir; Condition *precond = NULL; double double_output; list::iterator iter; unsigned char run; string output; if (this->precondition_list != NULL && this->precondition_list->size () > 0) { iter = this->precondition_list->begin (); for (iter = this->precondition_list->begin (); iter != this->precondition_list->end (); iter++) { precond = *iter; run = 0; /* Set the bInheritHandle flag so pipe handles are inherited. */ attributes.nLength = sizeof (SECURITY_ATTRIBUTES); attributes.bInheritHandle = TRUE; attributes.lpSecurityDescriptor = NULL; /* Create a job to kill the child tree if it become zombie */ /* CAUTION: In order to compile this, WINVER should be defined to 0x0500. This may need no change, since it was redefined by the program, but if needed, the macro is defined in */ job = CreateJobObject (&attributes, this->module_name.c_str ()); if (job == NULL) { pandoraLog ("evaluatePreconditions: CreateJobObject failed. Err: %d", GetLastError ()); return 0; } /* Get the handle to the current STDOUT. */ out = GetStdHandle (STD_OUTPUT_HANDLE); if (! CreatePipe (&out_read, &new_stdout, &attributes, 0)) { pandoraLog ("evaluatePreconditions: CreatePipe failed. Err: %d", GetLastError ()); return 0; } /* Ensure the read handle to the pipe for STDOUT is not inherited */ SetHandleInformation (out_read, HANDLE_FLAG_INHERIT, 0); /* Set up members of the STARTUPINFO structure. */ ZeroMemory (&si, sizeof (si)); GetStartupInfo (&si); si.cb = sizeof (si); si.dwFlags = STARTF_USESTDHANDLES | STARTF_USESHOWWINDOW; si.wShowWindow = SW_HIDE; si.hStdError = new_stdout; si.hStdOutput = new_stdout; /* Set up members of the PROCESS_INFORMATION structure. */ ZeroMemory (&pi, sizeof (pi)); pandoraDebug ("Executing pre-condition: %s", precond->command.c_str ()); /* Set the working directory of the process. It's "utils" directory to find the GNU W32 tools */ working_dir = getPandoraInstallDir () + "util\\"; /* Create the child process. */ if (! CreateProcess (NULL, (CHAR *) precond->command.c_str (), NULL, NULL, TRUE, CREATE_SUSPENDED | CREATE_NO_WINDOW, NULL, working_dir.c_str (), &si, &pi)) { pandoraLog ("evaluatePreconditions: %s CreateProcess failed. Err: %d", this->module_name.c_str (), GetLastError ()); } else { char buffer[BUFSIZE + 1]; unsigned long read, avail; if (! AssignProcessToJobObject (job, pi.hProcess)) { pandoraLog ("evaluatePreconditions: could not assign proccess to job (error %d)", GetLastError ()); } ResumeThread (pi.hThread); /*string output;*/ output = ""; int tickbase = GetTickCount(); while ( (dwRet = WaitForSingleObject (pi.hProcess, 500)) != WAIT_ABANDONED ) { PeekNamedPipe (out_read, buffer, BUFSIZE, &read, &avail, NULL); if (avail > 0) { ReadFile (out_read, buffer, BUFSIZE, &read, NULL); buffer[read] = '\0'; output += (char *) buffer; } try { double_output = Pandora_Strutils::strtodouble (output); } catch (Pandora_Strutils::Invalid_Conversion e) { double_output = 0; } if (dwRet == WAIT_OBJECT_0) { break; } else if(this->getTimeout() < GetTickCount() - tickbase) { /* STILL_ACTIVE */ TerminateProcess(pi.hThread, STILL_ACTIVE); pandoraLog ("evaluatePreconditions: %s timed out (retcode: %d)", this->module_name.c_str (), STILL_ACTIVE); break; } } GetExitCodeProcess (pi.hProcess, &retval); if (retval != 0) { if (! TerminateJobObject (job, 0)) { pandoraLog ("evaluatePreconditions: TerminateJobObject failed. (error %d)", GetLastError ()); } if (retval != STILL_ACTIVE) { pandoraLog ("evaluatePreconditions: %s did not executed well (retcode: %d)", this->module_name.c_str (), retval); } /* Close job, process and thread handles. */ CloseHandle (job); CloseHandle (pi.hProcess); CloseHandle (pi.hThread); CloseHandle (new_stdout); CloseHandle (out_read); return 0; } /* Close job, process and thread handles. */ CloseHandle (job); CloseHandle (pi.hProcess); CloseHandle (pi.hThread); } CloseHandle (new_stdout); CloseHandle (out_read); if (evaluateCondition (output, double_output, precond) == 0) { return 0; } } } return 1; } /** * Evaluates and executes module conditions. */ void Pandora_Module::evaluateConditions () { unsigned char run; double double_value; string string_value; Condition *cond = NULL; list::iterator iter; PROCESS_INFORMATION pi; STARTUPINFO si; Pandora_Data *pandora_data = NULL; regex_t regex; /* No data */ if ( (!this->has_output) || this->data_list == NULL) { return; } /* Get the module data */ pandora_data = data_list->front (); /* Get the string value of the data */ string_value = pandora_data->getValue (); /* Get the double value of the data */ try { double_value = Pandora_Strutils::strtodouble (string_value); } catch (Pandora_Strutils::Invalid_Conversion e) { double_value = 0; } if (this->condition_list != NULL && this->condition_list->size () > 0) { iter = this->condition_list->begin (); for (iter = this->condition_list->begin (); iter != this->condition_list->end (); iter++) { cond = *iter; run = 0; if (evaluateCondition (string_value, double_value, cond) == 1) { /* Run the condition command */ ZeroMemory (&si, sizeof (si)); ZeroMemory (&pi, sizeof (pi)); if (CreateProcess (NULL , (CHAR *)cond->command.c_str (), NULL, NULL, FALSE, CREATE_NO_WINDOW, NULL, NULL, &si, &pi) == 0) { return; } WaitForSingleObject(pi.hProcess, this->module_timeout); CloseHandle (pi.hProcess); CloseHandle (pi.hThread); } } } } /** * Evaluates and executes intensive module conditions. */ int Pandora_Module::evaluateIntensiveConditions () { double double_value; string string_value; Condition *cond = NULL; list::iterator iter; PROCESS_INFORMATION pi; STARTUPINFO si; Pandora_Data *pandora_data = NULL; regex_t regex; /* Not an intensive module */ if (this->intensive_condition_list == NULL || this->intensive_condition_list->size () <= 0) { return 1; } /* No data */ if ( (!this->has_output) || this->data_list == NULL) { return 0; } /* Get the module data */ pandora_data = data_list->front (); /* Get the string value of the data */ string_value = pandora_data->getValue (); /* Get the double value of the data */ try { double_value = Pandora_Strutils::strtodouble (string_value); } catch (Pandora_Strutils::Invalid_Conversion e) { double_value = 0; } iter = this->intensive_condition_list->begin (); for (iter = this->intensive_condition_list->begin (); iter != this->intensive_condition_list->end (); iter++) { cond = *iter; if (evaluateCondition (string_value, double_value, cond) == 0) { return 0; } } return 1; } /** * Checks the module cron. Returns 1 if the module should run, 0 if not. * * @return 1 if the module should run, 0 if not. */ bool Pandora_Module::checkCron (int interval) { // Execute always if cron is not configured if (!this->cron->getIsSet()) return true; time_t now = time(NULL); if (!this->cron->shouldExecuteAt(now)) return false; // Check if should execute this module at first before update cron params bool execute = this->cron->shouldExecuteAtFirst(now); this->cron->update(now, interval); return execute; } /** * Sets the module cron from a string. */ void Pandora_Module::setCron (string cron_string) { this->cron = new Cron(cron_string); } /** * Evaluate a single condition. Returns 1 if the condition matches, 0 * otherwise. * * @param string_value String value. * @param double_value Double value. * @param condition Pointer to the condition. */ int Pandora_Module::evaluateCondition (string string_value, double double_value, Condition *condition) { if ((condition->operation == ">" && double_value > condition->value_1) || (condition->operation == "<" && double_value < condition->value_1) || (condition->operation == "=" && double_value == condition->value_1) || (condition->operation == "!=" && double_value != condition->value_1) || (condition->operation == "=~" && regexec (&(condition->regexp), string_value.c_str(), 0, NULL, 0) == 0) || (condition->operation == "()" && double_value > condition->value_1 && double_value < condition->value_2)) { return 1; } return 0; } /** * Checks if a module has data. * * @return 1 if the module has data, 0 otherwise. */ int Pandora_Module::hasOutput () { if (this->has_output == 1) { return 1; } return 0; } /** * Sets the module timestamp. * * @param timestamp Module timestamp in seconds. */ void Pandora_Module::setTimestamp (time_t timestamp) { this->timestamp = timestamp; } /** * Gets the module timestamp. * * @return Module timestamp in seconds. */ time_t Pandora_Module::getTimestamp () { return this->timestamp; } /** * Sets the module timestamp file. * * @param file_name The name of the timestamp file. */ void Pandora_Module::setTimestampFile (string file_name) { this->timestamp_file = file_name; } /** * Gets the module timestamp file. * * @return The name of the timestamp file. */ string Pandora_Module::getTimestampFile () { return this->timestamp_file; } /** * Update the timestamp file with the current time. * */ void Pandora_Module::updateTimestampFile () { try { Pandora_File::writeFile(this->timestamp_file, std::to_string(std::time(NULL))); } catch (...) { /* Ignore errors. */ } } /** * Initialize the module's internal execution counter. * */ void Pandora_Module::initExecutions () { string timestamp; try { if (this->timestamp_file != "" && Pandora_File::readFile(this->timestamp_file, timestamp) != FILE_NOT_FOUND) { // If the interval is 0, setting executions to 0 will prevent the module from running. this->executions = this->intensive_interval == 0 ? 0 : floor((1000.0 * (std::time(NULL) - strtoint(timestamp))) / Pandora_Windows_Service::getInstance()->getInterval()); return; } } catch (...) { // Ignore errors. } // If the interval is 0, setting executions to any value != 0 will make the module run. this->executions = this->intensive_interval == 0 ? 1 : this->intensive_interval; } /** * Sets the value of intensive_match. * * @param intensive_match 0 or 1. */ void Pandora_Module::setIntensiveMatch (unsigned char intensive_match) { this->intensive_match = intensive_match; } /** * Gets the value of intensive_match. * * @return The value of intensive match. */ unsigned char Pandora_Module::getIntensiveMatch () { return this->intensive_match; } bool Pandora_Module::getAsync () { return this->async; } /** * Get current exections */ long Pandora_Module::getExecutions () { return this->executions; } /** * Set current execution (global) used for brokers. */ void Pandora_Module::setExecutions (long executions) { this->executions = executions; } /** * Checks if the module has intensive conditions. * * @return true if the module's intensive condition list is not empty, false if it is. */ bool Pandora_Module::isIntensive () { if (this->intensive_condition_list == NULL || this->intensive_condition_list->size () <= 0) { return false; } return true; } /** * Set the WaitForSingleObject timeout. * * @param timeout Timeout in milliseconds. */ void Pandora_Module::setWaitTimeout (int timeout) { if (timeout < 0) { return; } this->module_wait_timeout = timeout; }