1 /******************************************************************************
3 * Copyright (C) 2012-2015 Icinga Development Team (http://www.icinga.org) *
5 * This program is free software; you can redistribute it and/or *
6 * modify it under the terms of the GNU General Public License *
7 * as published by the Free Software Foundation; either version 2 *
8 * of the License, or (at your option) any later version. *
10 * This program is distributed in the hope that it will be useful, *
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
13 * GNU General Public License for more details. *
15 * You should have received a copy of the GNU General Public License *
16 * along with this program; if not, write to the Free Software Foundation *
17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. *
18 ******************************************************************************/
20 #include "livestatus/logtable.hpp"
21 #include "livestatus/livestatuslogutility.hpp"
22 #include "livestatus/hoststable.hpp"
23 #include "livestatus/servicestable.hpp"
24 #include "livestatus/contactstable.hpp"
25 #include "livestatus/commandstable.hpp"
26 #include "icinga/icingaapplication.hpp"
27 #include "icinga/cib.hpp"
28 #include "icinga/service.hpp"
29 #include "icinga/host.hpp"
30 #include "icinga/user.hpp"
31 #include "icinga/checkcommand.hpp"
32 #include "icinga/eventcommand.hpp"
33 #include "icinga/notificationcommand.hpp"
34 #include "base/convert.hpp"
35 #include "base/utility.hpp"
36 #include "base/logger.hpp"
37 #include "base/application.hpp"
38 #include "base/objectlock.hpp"
39 #include <boost/foreach.hpp>
40 #include <boost/tuple/tuple.hpp>
41 #include <boost/algorithm/string.hpp>
42 #include <boost/algorithm/string/split.hpp>
43 #include <boost/algorithm/string/classification.hpp>
44 #include <boost/algorithm/string/replace.hpp>
45 #include <boost/algorithm/string/predicate.hpp>
48 using namespace icinga;
50 LogTable::LogTable(const String& compat_log_path, time_t from, time_t until)
52 /* store attributes for FetchRows */
55 m_CompatLogPath = compat_log_path;
62 void LogTable::AddColumns(Table *table, const String& prefix,
63 const Column::ObjectAccessor& objectAccessor)
65 table->AddColumn(prefix + "time", Column(&LogTable::TimeAccessor, objectAccessor));
66 table->AddColumn(prefix + "lineno", Column(&LogTable::LinenoAccessor, objectAccessor));
67 table->AddColumn(prefix + "class", Column(&LogTable::ClassAccessor, objectAccessor));
68 table->AddColumn(prefix + "message", Column(&LogTable::MessageAccessor, objectAccessor));
69 table->AddColumn(prefix + "type", Column(&LogTable::TypeAccessor, objectAccessor));
70 table->AddColumn(prefix + "options", Column(&LogTable::OptionsAccessor, objectAccessor));
71 table->AddColumn(prefix + "comment", Column(&LogTable::CommentAccessor, objectAccessor));
72 table->AddColumn(prefix + "plugin_output", Column(&LogTable::PluginOutputAccessor, objectAccessor));
73 table->AddColumn(prefix + "state", Column(&LogTable::StateAccessor, objectAccessor));
74 table->AddColumn(prefix + "state_type", Column(&LogTable::StateTypeAccessor, objectAccessor));
75 table->AddColumn(prefix + "attempt", Column(&LogTable::AttemptAccessor, objectAccessor));
76 table->AddColumn(prefix + "service_description", Column(&LogTable::ServiceDescriptionAccessor, objectAccessor));
77 table->AddColumn(prefix + "host_name", Column(&LogTable::HostNameAccessor, objectAccessor));
78 table->AddColumn(prefix + "contact_name", Column(&LogTable::ContactNameAccessor, objectAccessor));
79 table->AddColumn(prefix + "command_name", Column(&LogTable::CommandNameAccessor, objectAccessor));
81 HostsTable::AddColumns(table, "current_host_", boost::bind(&LogTable::HostAccessor, _1, objectAccessor));
82 ServicesTable::AddColumns(table, "current_service_", boost::bind(&LogTable::ServiceAccessor, _1, objectAccessor));
83 ContactsTable::AddColumns(table, "current_contact_", boost::bind(&LogTable::ContactAccessor, _1, objectAccessor));
84 CommandsTable::AddColumns(table, "current_command_", boost::bind(&LogTable::CommandAccessor, _1, objectAccessor));
87 String LogTable::GetName(void) const
92 String LogTable::GetPrefix(void) const
97 void LogTable::FetchRows(const AddRowFunction& addRowFn)
99 Log(LogDebug, "LogTable")
100 << "Pre-selecting log file from " << m_TimeFrom << " until " << m_TimeUntil;
102 /* create log file index */
103 LivestatusLogUtility::CreateLogIndex(m_CompatLogPath, m_LogFileIndex);
105 /* generate log cache */
106 LivestatusLogUtility::CreateLogCache(m_LogFileIndex, this, m_TimeFrom, m_TimeUntil, addRowFn);
109 /* gets called in LivestatusLogUtility::CreateLogCache */
110 void LogTable::UpdateLogEntries(const Dictionary::Ptr& log_entry_attrs, int line_count, int lineno, const AddRowFunction& addRowFn)
112 /* additional attributes only for log table */
113 log_entry_attrs->Set("lineno", lineno);
115 addRowFn(log_entry_attrs);
118 Object::Ptr LogTable::HostAccessor(const Value& row, const Column::ObjectAccessor&)
120 String host_name = static_cast<Dictionary::Ptr>(row)->Get("host_name");
122 if (host_name.IsEmpty())
123 return Object::Ptr();
125 return Host::GetByName(host_name);
128 Object::Ptr LogTable::ServiceAccessor(const Value& row, const Column::ObjectAccessor&)
130 String host_name = static_cast<Dictionary::Ptr>(row)->Get("host_name");
131 String service_description = static_cast<Dictionary::Ptr>(row)->Get("service_description");
133 if (service_description.IsEmpty() || host_name.IsEmpty())
134 return Object::Ptr();
136 return Service::GetByNamePair(host_name, service_description);
139 Object::Ptr LogTable::ContactAccessor(const Value& row, const Column::ObjectAccessor&)
141 String contact_name = static_cast<Dictionary::Ptr>(row)->Get("contact_name");
143 if (contact_name.IsEmpty())
144 return Object::Ptr();
146 return User::GetByName(contact_name);
149 Object::Ptr LogTable::CommandAccessor(const Value& row, const Column::ObjectAccessor&)
151 String command_name = static_cast<Dictionary::Ptr>(row)->Get("command_name");
153 if (command_name.IsEmpty())
154 return Object::Ptr();
156 CheckCommand::Ptr check_command = CheckCommand::GetByName(command_name);
157 if (!check_command) {
158 EventCommand::Ptr event_command = EventCommand::GetByName(command_name);
159 if (!event_command) {
160 NotificationCommand::Ptr notification_command = NotificationCommand::GetByName(command_name);
161 if (!notification_command)
162 return Object::Ptr();
164 return notification_command;
166 return event_command;
168 return check_command;
170 return Object::Ptr();
173 Value LogTable::TimeAccessor(const Value& row)
175 return static_cast<Dictionary::Ptr>(row)->Get("time");
178 Value LogTable::LinenoAccessor(const Value& row)
180 return static_cast<Dictionary::Ptr>(row)->Get("lineno");
183 Value LogTable::ClassAccessor(const Value& row)
185 return static_cast<Dictionary::Ptr>(row)->Get("class");
188 Value LogTable::MessageAccessor(const Value& row)
190 return static_cast<Dictionary::Ptr>(row)->Get("message");
193 Value LogTable::TypeAccessor(const Value& row)
195 return static_cast<Dictionary::Ptr>(row)->Get("type");
198 Value LogTable::OptionsAccessor(const Value& row)
200 return static_cast<Dictionary::Ptr>(row)->Get("options");
203 Value LogTable::CommentAccessor(const Value& row)
205 return static_cast<Dictionary::Ptr>(row)->Get("comment");
208 Value LogTable::PluginOutputAccessor(const Value& row)
210 return static_cast<Dictionary::Ptr>(row)->Get("plugin_output");
213 Value LogTable::StateAccessor(const Value& row)
215 return static_cast<Dictionary::Ptr>(row)->Get("state");
218 Value LogTable::StateTypeAccessor(const Value& row)
220 return static_cast<Dictionary::Ptr>(row)->Get("state_type");
223 Value LogTable::AttemptAccessor(const Value& row)
225 return static_cast<Dictionary::Ptr>(row)->Get("attempt");
228 Value LogTable::ServiceDescriptionAccessor(const Value& row)
230 return static_cast<Dictionary::Ptr>(row)->Get("service_description");
233 Value LogTable::HostNameAccessor(const Value& row)
235 return static_cast<Dictionary::Ptr>(row)->Get("host_name");
238 Value LogTable::ContactNameAccessor(const Value& row)
240 return static_cast<Dictionary::Ptr>(row)->Get("contact_name");
243 Value LogTable::CommandNameAccessor(const Value& row)
245 return static_cast<Dictionary::Ptr>(row)->Get("command_name");