2019-02-25 14:48:22 +01:00
|
|
|
/* Icinga 2 | (c) 2012 Icinga GmbH | GPLv2+ */
|
2013-07-09 17:46:48 +02:00
|
|
|
|
2014-05-25 16:23:35 +02:00
|
|
|
#include "livestatus/logtable.hpp"
|
2014-05-28 14:25:12 +02:00
|
|
|
#include "livestatus/livestatuslogutility.hpp"
|
2014-05-25 16:23:35 +02:00
|
|
|
#include "livestatus/hoststable.hpp"
|
|
|
|
#include "livestatus/servicestable.hpp"
|
|
|
|
#include "livestatus/contactstable.hpp"
|
|
|
|
#include "livestatus/commandstable.hpp"
|
|
|
|
#include "icinga/icingaapplication.hpp"
|
|
|
|
#include "icinga/cib.hpp"
|
|
|
|
#include "icinga/service.hpp"
|
|
|
|
#include "icinga/host.hpp"
|
|
|
|
#include "icinga/user.hpp"
|
|
|
|
#include "icinga/checkcommand.hpp"
|
|
|
|
#include "icinga/eventcommand.hpp"
|
|
|
|
#include "icinga/notificationcommand.hpp"
|
|
|
|
#include "base/convert.hpp"
|
|
|
|
#include "base/utility.hpp"
|
2014-10-19 14:21:12 +02:00
|
|
|
#include "base/logger.hpp"
|
2014-05-25 16:23:35 +02:00
|
|
|
#include "base/application.hpp"
|
|
|
|
#include "base/objectlock.hpp"
|
2013-10-29 13:44:43 +01:00
|
|
|
#include <boost/tuple/tuple.hpp>
|
|
|
|
#include <boost/algorithm/string.hpp>
|
|
|
|
#include <boost/algorithm/string/replace.hpp>
|
|
|
|
#include <boost/algorithm/string/predicate.hpp>
|
|
|
|
#include <fstream>
|
2013-07-09 17:46:48 +02:00
|
|
|
|
|
|
|
using namespace icinga;
|
|
|
|
|
2013-12-18 16:06:39 +01:00
|
|
|
LogTable::LogTable(const String& compat_log_path, time_t from, time_t until)
|
2013-07-09 17:46:48 +02:00
|
|
|
{
|
2013-12-18 16:28:33 +01:00
|
|
|
/* store attributes for FetchRows */
|
2013-10-29 13:44:43 +01:00
|
|
|
m_TimeFrom = from;
|
|
|
|
m_TimeUntil = until;
|
2013-12-18 16:28:33 +01:00
|
|
|
m_CompatLogPath = compat_log_path;
|
2013-10-29 13:44:43 +01:00
|
|
|
|
2013-07-09 17:46:48 +02:00
|
|
|
AddColumns(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void LogTable::AddColumns(Table *table, const String& prefix,
|
2017-12-19 15:50:05 +01:00
|
|
|
const Column::ObjectAccessor& objectAccessor)
|
2013-07-09 17:46:48 +02:00
|
|
|
{
|
|
|
|
table->AddColumn(prefix + "time", Column(&LogTable::TimeAccessor, objectAccessor));
|
|
|
|
table->AddColumn(prefix + "lineno", Column(&LogTable::LinenoAccessor, objectAccessor));
|
|
|
|
table->AddColumn(prefix + "class", Column(&LogTable::ClassAccessor, objectAccessor));
|
|
|
|
table->AddColumn(prefix + "message", Column(&LogTable::MessageAccessor, objectAccessor));
|
|
|
|
table->AddColumn(prefix + "type", Column(&LogTable::TypeAccessor, objectAccessor));
|
|
|
|
table->AddColumn(prefix + "options", Column(&LogTable::OptionsAccessor, objectAccessor));
|
|
|
|
table->AddColumn(prefix + "comment", Column(&LogTable::CommentAccessor, objectAccessor));
|
|
|
|
table->AddColumn(prefix + "plugin_output", Column(&LogTable::PluginOutputAccessor, objectAccessor));
|
|
|
|
table->AddColumn(prefix + "state", Column(&LogTable::StateAccessor, objectAccessor));
|
|
|
|
table->AddColumn(prefix + "state_type", Column(&LogTable::StateTypeAccessor, objectAccessor));
|
|
|
|
table->AddColumn(prefix + "attempt", Column(&LogTable::AttemptAccessor, objectAccessor));
|
|
|
|
table->AddColumn(prefix + "service_description", Column(&LogTable::ServiceDescriptionAccessor, objectAccessor));
|
|
|
|
table->AddColumn(prefix + "host_name", Column(&LogTable::HostNameAccessor, objectAccessor));
|
|
|
|
table->AddColumn(prefix + "contact_name", Column(&LogTable::ContactNameAccessor, objectAccessor));
|
|
|
|
table->AddColumn(prefix + "command_name", Column(&LogTable::CommandNameAccessor, objectAccessor));
|
|
|
|
|
2017-11-21 11:52:55 +01:00
|
|
|
HostsTable::AddColumns(table, "current_host_", std::bind(&LogTable::HostAccessor, _1, objectAccessor));
|
|
|
|
ServicesTable::AddColumns(table, "current_service_", std::bind(&LogTable::ServiceAccessor, _1, objectAccessor));
|
|
|
|
ContactsTable::AddColumns(table, "current_contact_", std::bind(&LogTable::ContactAccessor, _1, objectAccessor));
|
|
|
|
CommandsTable::AddColumns(table, "current_command_", std::bind(&LogTable::CommandAccessor, _1, objectAccessor));
|
2013-07-09 17:46:48 +02:00
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
String LogTable::GetName() const
|
2013-07-09 17:46:48 +02:00
|
|
|
{
|
2013-07-09 18:05:47 +02:00
|
|
|
return "log";
|
2013-07-09 17:46:48 +02:00
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
String LogTable::GetPrefix() const
|
2014-06-25 11:30:27 +02:00
|
|
|
{
|
|
|
|
return "log";
|
|
|
|
}
|
|
|
|
|
2013-07-09 17:46:48 +02:00
|
|
|
void LogTable::FetchRows(const AddRowFunction& addRowFn)
|
|
|
|
{
|
2014-10-20 10:09:57 +02:00
|
|
|
Log(LogDebug, "LogTable")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "Pre-selecting log file from " << m_TimeFrom << " until " << m_TimeUntil;
|
2013-12-18 16:28:33 +01:00
|
|
|
|
|
|
|
/* create log file index */
|
2014-05-28 14:25:12 +02:00
|
|
|
LivestatusLogUtility::CreateLogIndex(m_CompatLogPath, m_LogFileIndex);
|
2013-12-18 16:28:33 +01:00
|
|
|
|
|
|
|
/* generate log cache */
|
2014-05-28 14:25:12 +02:00
|
|
|
LivestatusLogUtility::CreateLogCache(m_LogFileIndex, this, m_TimeFrom, m_TimeUntil, addRowFn);
|
2013-12-18 17:19:16 +01:00
|
|
|
}
|
2013-12-18 16:28:33 +01:00
|
|
|
|
2014-05-28 14:25:12 +02:00
|
|
|
/* gets called in LivestatusLogUtility::CreateLogCache */
|
2013-12-18 17:19:16 +01:00
|
|
|
void LogTable::UpdateLogEntries(const Dictionary::Ptr& log_entry_attrs, int line_count, int lineno, const AddRowFunction& addRowFn)
|
|
|
|
{
|
|
|
|
/* additional attributes only for log table */
|
|
|
|
log_entry_attrs->Set("lineno", lineno);
|
2013-10-29 13:44:43 +01:00
|
|
|
|
2015-02-13 15:50:20 +01:00
|
|
|
addRowFn(log_entry_attrs, LivestatusGroupByNone, Empty);
|
2013-10-29 13:44:43 +01:00
|
|
|
}
|
|
|
|
|
2014-05-22 09:02:44 +02:00
|
|
|
Object::Ptr LogTable::HostAccessor(const Value& row, const Column::ObjectAccessor&)
|
2013-10-29 13:44:43 +01:00
|
|
|
{
|
|
|
|
String host_name = static_cast<Dictionary::Ptr>(row)->Get("host_name");
|
2013-07-09 17:46:48 +02:00
|
|
|
|
2013-10-29 13:44:43 +01:00
|
|
|
if (host_name.IsEmpty())
|
2017-11-30 08:36:35 +01:00
|
|
|
return nullptr;
|
2013-10-29 13:44:43 +01:00
|
|
|
|
|
|
|
return Host::GetByName(host_name);
|
|
|
|
}
|
|
|
|
|
2014-05-22 09:02:44 +02:00
|
|
|
Object::Ptr LogTable::ServiceAccessor(const Value& row, const Column::ObjectAccessor&)
|
2013-10-29 13:44:43 +01:00
|
|
|
{
|
|
|
|
String host_name = static_cast<Dictionary::Ptr>(row)->Get("host_name");
|
|
|
|
String service_description = static_cast<Dictionary::Ptr>(row)->Get("service_description");
|
|
|
|
|
|
|
|
if (service_description.IsEmpty() || host_name.IsEmpty())
|
2017-11-30 08:36:35 +01:00
|
|
|
return nullptr;
|
2013-10-29 13:44:43 +01:00
|
|
|
|
|
|
|
return Service::GetByNamePair(host_name, service_description);
|
|
|
|
}
|
|
|
|
|
2014-05-22 09:02:44 +02:00
|
|
|
Object::Ptr LogTable::ContactAccessor(const Value& row, const Column::ObjectAccessor&)
|
2013-10-29 13:44:43 +01:00
|
|
|
{
|
|
|
|
String contact_name = static_cast<Dictionary::Ptr>(row)->Get("contact_name");
|
|
|
|
|
|
|
|
if (contact_name.IsEmpty())
|
2017-11-30 08:36:35 +01:00
|
|
|
return nullptr;
|
2013-10-29 13:44:43 +01:00
|
|
|
|
|
|
|
return User::GetByName(contact_name);
|
|
|
|
}
|
|
|
|
|
2014-05-22 09:02:44 +02:00
|
|
|
Object::Ptr LogTable::CommandAccessor(const Value& row, const Column::ObjectAccessor&)
|
2013-10-29 13:44:43 +01:00
|
|
|
{
|
|
|
|
String command_name = static_cast<Dictionary::Ptr>(row)->Get("command_name");
|
|
|
|
|
|
|
|
if (command_name.IsEmpty())
|
2017-11-30 08:36:35 +01:00
|
|
|
return nullptr;
|
2013-10-29 13:44:43 +01:00
|
|
|
|
|
|
|
CheckCommand::Ptr check_command = CheckCommand::GetByName(command_name);
|
|
|
|
if (!check_command) {
|
|
|
|
EventCommand::Ptr event_command = EventCommand::GetByName(command_name);
|
|
|
|
if (!event_command) {
|
|
|
|
NotificationCommand::Ptr notification_command = NotificationCommand::GetByName(command_name);
|
|
|
|
if (!notification_command)
|
2017-11-30 08:36:35 +01:00
|
|
|
return nullptr;
|
2013-10-29 13:44:43 +01:00
|
|
|
else
|
|
|
|
return notification_command;
|
|
|
|
} else
|
|
|
|
return event_command;
|
|
|
|
} else
|
|
|
|
return check_command;
|
2013-07-09 17:46:48 +02:00
|
|
|
}
|
|
|
|
|
2013-07-10 14:06:46 +02:00
|
|
|
Value LogTable::TimeAccessor(const Value& row)
|
2013-07-09 17:46:48 +02:00
|
|
|
{
|
2013-10-29 13:44:43 +01:00
|
|
|
return static_cast<Dictionary::Ptr>(row)->Get("time");
|
2013-07-09 17:46:48 +02:00
|
|
|
}
|
|
|
|
|
2013-07-10 14:06:46 +02:00
|
|
|
Value LogTable::LinenoAccessor(const Value& row)
|
2013-07-09 17:46:48 +02:00
|
|
|
{
|
2013-10-29 13:44:43 +01:00
|
|
|
return static_cast<Dictionary::Ptr>(row)->Get("lineno");
|
2013-07-09 17:46:48 +02:00
|
|
|
}
|
|
|
|
|
2013-07-10 14:06:46 +02:00
|
|
|
Value LogTable::ClassAccessor(const Value& row)
|
2013-07-09 17:46:48 +02:00
|
|
|
{
|
2013-10-29 13:44:43 +01:00
|
|
|
return static_cast<Dictionary::Ptr>(row)->Get("class");
|
2013-07-09 17:46:48 +02:00
|
|
|
}
|
|
|
|
|
2013-07-10 14:06:46 +02:00
|
|
|
Value LogTable::MessageAccessor(const Value& row)
|
2013-07-09 17:46:48 +02:00
|
|
|
{
|
2013-10-29 13:44:43 +01:00
|
|
|
return static_cast<Dictionary::Ptr>(row)->Get("message");
|
2013-07-09 17:46:48 +02:00
|
|
|
}
|
|
|
|
|
2013-07-10 14:06:46 +02:00
|
|
|
Value LogTable::TypeAccessor(const Value& row)
|
2013-07-09 17:46:48 +02:00
|
|
|
{
|
2013-10-29 13:44:43 +01:00
|
|
|
return static_cast<Dictionary::Ptr>(row)->Get("type");
|
2013-07-09 17:46:48 +02:00
|
|
|
}
|
|
|
|
|
2013-07-10 14:06:46 +02:00
|
|
|
Value LogTable::OptionsAccessor(const Value& row)
|
2013-07-09 17:46:48 +02:00
|
|
|
{
|
2013-10-29 13:44:43 +01:00
|
|
|
return static_cast<Dictionary::Ptr>(row)->Get("options");
|
2013-07-09 17:46:48 +02:00
|
|
|
}
|
|
|
|
|
2013-07-10 14:06:46 +02:00
|
|
|
Value LogTable::CommentAccessor(const Value& row)
|
2013-07-09 17:46:48 +02:00
|
|
|
{
|
2013-10-29 13:44:43 +01:00
|
|
|
return static_cast<Dictionary::Ptr>(row)->Get("comment");
|
2013-07-09 17:46:48 +02:00
|
|
|
}
|
|
|
|
|
2013-07-10 14:06:46 +02:00
|
|
|
Value LogTable::PluginOutputAccessor(const Value& row)
|
2013-07-09 17:46:48 +02:00
|
|
|
{
|
2013-10-29 13:44:43 +01:00
|
|
|
return static_cast<Dictionary::Ptr>(row)->Get("plugin_output");
|
2013-07-09 17:46:48 +02:00
|
|
|
}
|
|
|
|
|
2013-07-10 14:06:46 +02:00
|
|
|
Value LogTable::StateAccessor(const Value& row)
|
2013-07-09 17:46:48 +02:00
|
|
|
{
|
2013-10-29 13:44:43 +01:00
|
|
|
return static_cast<Dictionary::Ptr>(row)->Get("state");
|
2013-07-09 17:46:48 +02:00
|
|
|
}
|
|
|
|
|
2013-07-10 14:06:46 +02:00
|
|
|
Value LogTable::StateTypeAccessor(const Value& row)
|
2013-07-09 17:46:48 +02:00
|
|
|
{
|
2013-10-29 13:44:43 +01:00
|
|
|
return static_cast<Dictionary::Ptr>(row)->Get("state_type");
|
2013-07-09 17:46:48 +02:00
|
|
|
}
|
|
|
|
|
2013-07-10 14:06:46 +02:00
|
|
|
Value LogTable::AttemptAccessor(const Value& row)
|
2013-07-09 17:46:48 +02:00
|
|
|
{
|
2013-10-29 13:44:43 +01:00
|
|
|
return static_cast<Dictionary::Ptr>(row)->Get("attempt");
|
2013-07-09 17:46:48 +02:00
|
|
|
}
|
|
|
|
|
2013-07-10 14:06:46 +02:00
|
|
|
Value LogTable::ServiceDescriptionAccessor(const Value& row)
|
2013-07-09 17:46:48 +02:00
|
|
|
{
|
2013-10-29 13:44:43 +01:00
|
|
|
return static_cast<Dictionary::Ptr>(row)->Get("service_description");
|
2013-07-09 17:46:48 +02:00
|
|
|
}
|
|
|
|
|
2013-07-10 14:06:46 +02:00
|
|
|
Value LogTable::HostNameAccessor(const Value& row)
|
2013-07-09 17:46:48 +02:00
|
|
|
{
|
2013-10-29 13:44:43 +01:00
|
|
|
return static_cast<Dictionary::Ptr>(row)->Get("host_name");
|
2013-07-09 17:46:48 +02:00
|
|
|
}
|
|
|
|
|
2013-07-10 14:06:46 +02:00
|
|
|
Value LogTable::ContactNameAccessor(const Value& row)
|
2013-07-09 17:46:48 +02:00
|
|
|
{
|
2013-10-29 13:44:43 +01:00
|
|
|
return static_cast<Dictionary::Ptr>(row)->Get("contact_name");
|
2013-07-09 17:46:48 +02:00
|
|
|
}
|
|
|
|
|
2013-07-10 14:06:46 +02:00
|
|
|
Value LogTable::CommandNameAccessor(const Value& row)
|
2013-07-09 17:46:48 +02:00
|
|
|
{
|
2013-10-29 13:44:43 +01:00
|
|
|
return static_cast<Dictionary::Ptr>(row)->Get("command_name");
|
|
|
|
}
|