2015-01-28 10:57:34 +01:00
|
|
|
/******************************************************************************
|
|
|
|
* Icinga 2 *
|
2017-01-10 15:54:22 +01:00
|
|
|
* Copyright (C) 2012-2017 Icinga Development Team (https://www.icinga.com/) *
|
2015-01-28 10:57:34 +01:00
|
|
|
* *
|
|
|
|
* 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 *
|
|
|
|
* of the License, 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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. *
|
|
|
|
******************************************************************************/
|
|
|
|
|
|
|
|
#include "perfdata/opentsdbwriter.hpp"
|
2015-03-28 11:04:42 +01:00
|
|
|
#include "perfdata/opentsdbwriter.tcpp"
|
2015-01-28 10:57:34 +01:00
|
|
|
#include "icinga/service.hpp"
|
|
|
|
#include "icinga/macroprocessor.hpp"
|
|
|
|
#include "icinga/icingaapplication.hpp"
|
|
|
|
#include "icinga/compatutility.hpp"
|
|
|
|
#include "base/tcpsocket.hpp"
|
2015-08-15 20:28:05 +02:00
|
|
|
#include "base/configtype.hpp"
|
2015-01-28 10:57:34 +01:00
|
|
|
#include "base/objectlock.hpp"
|
|
|
|
#include "base/logger.hpp"
|
|
|
|
#include "base/convert.hpp"
|
|
|
|
#include "base/utility.hpp"
|
2017-05-15 15:51:39 +02:00
|
|
|
#include "base/perfdatavalue.hpp"
|
2015-01-28 10:57:34 +01:00
|
|
|
#include "base/application.hpp"
|
|
|
|
#include "base/stream.hpp"
|
|
|
|
#include "base/networkstream.hpp"
|
|
|
|
#include "base/exception.hpp"
|
|
|
|
#include "base/statsfunction.hpp"
|
|
|
|
#include <boost/algorithm/string.hpp>
|
|
|
|
#include <boost/algorithm/string/classification.hpp>
|
|
|
|
#include <boost/algorithm/string/split.hpp>
|
|
|
|
#include <boost/algorithm/string/replace.hpp>
|
|
|
|
|
|
|
|
using namespace icinga;
|
|
|
|
|
|
|
|
REGISTER_TYPE(OpenTsdbWriter);
|
|
|
|
|
2015-09-21 11:44:58 +02:00
|
|
|
REGISTER_STATSFUNCTION(OpenTsdbWriter, &OpenTsdbWriter::StatsFunc);
|
2015-01-28 10:57:34 +01:00
|
|
|
|
2015-02-07 22:36:17 +01:00
|
|
|
void OpenTsdbWriter::StatsFunc(const Dictionary::Ptr& status, const Array::Ptr&)
|
2015-01-28 10:57:34 +01:00
|
|
|
{
|
|
|
|
Dictionary::Ptr nodes = new Dictionary();
|
|
|
|
|
2016-08-25 06:19:44 +02:00
|
|
|
for (const OpenTsdbWriter::Ptr& opentsdbwriter : ConfigType::GetObjectsByType<OpenTsdbWriter>()) {
|
2015-01-28 10:57:34 +01:00
|
|
|
nodes->Set(opentsdbwriter->GetName(), 1); //add more stats
|
|
|
|
}
|
|
|
|
|
|
|
|
status->Set("opentsdbwriter", nodes);
|
|
|
|
}
|
|
|
|
|
2015-08-20 17:18:48 +02:00
|
|
|
void OpenTsdbWriter::Start(bool runtimeCreated)
|
2015-01-28 10:57:34 +01:00
|
|
|
{
|
2015-08-20 17:18:48 +02:00
|
|
|
ObjectImpl<OpenTsdbWriter>::Start(runtimeCreated);
|
2015-01-28 10:57:34 +01:00
|
|
|
|
2017-02-08 14:53:52 +01:00
|
|
|
Log(LogInformation, "OpentsdbWriter")
|
|
|
|
<< "'" << GetName() << "' started.";
|
|
|
|
|
2015-01-28 10:57:34 +01:00
|
|
|
m_ReconnectTimer = new Timer();
|
|
|
|
m_ReconnectTimer->SetInterval(10);
|
|
|
|
m_ReconnectTimer->OnTimerExpired.connect(boost::bind(&OpenTsdbWriter::ReconnectTimerHandler, this));
|
|
|
|
m_ReconnectTimer->Start();
|
|
|
|
m_ReconnectTimer->Reschedule(0);
|
|
|
|
|
|
|
|
Service::OnNewCheckResult.connect(boost::bind(&OpenTsdbWriter::CheckResultHandler, this, _1, _2));
|
|
|
|
}
|
|
|
|
|
2017-02-08 14:53:52 +01:00
|
|
|
void OpenTsdbWriter::Stop(bool runtimeRemoved)
|
|
|
|
{
|
|
|
|
Log(LogInformation, "OpentsdbWriter")
|
|
|
|
<< "'" << GetName() << "' stopped.";
|
|
|
|
|
|
|
|
ObjectImpl<OpenTsdbWriter>::Stop(runtimeRemoved);
|
|
|
|
}
|
|
|
|
|
2015-01-28 10:57:34 +01:00
|
|
|
void OpenTsdbWriter::ReconnectTimerHandler(void)
|
|
|
|
{
|
|
|
|
if (m_Stream)
|
|
|
|
return;
|
|
|
|
|
|
|
|
TcpSocket::Ptr socket = new TcpSocket();
|
|
|
|
|
|
|
|
Log(LogNotice, "OpenTsdbWriter")
|
|
|
|
<< "Reconnect to OpenTSDB TSD on host '" << GetHost() << "' port '" << GetPort() << "'.";
|
|
|
|
|
|
|
|
try {
|
|
|
|
socket->Connect(GetHost(), GetPort());
|
|
|
|
} catch (std::exception&) {
|
|
|
|
Log(LogCritical, "OpenTsdbWriter")
|
|
|
|
<< "Can't connect to OpenTSDB TSD on host '" << GetHost() << "' port '" << GetPort() << "'.";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_Stream = new NetworkStream(socket);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OpenTsdbWriter::CheckResultHandler(const Checkable::Ptr& checkable, const CheckResult::Ptr& cr)
|
|
|
|
{
|
|
|
|
CONTEXT("Processing check result for '" + checkable->GetName() + "'");
|
|
|
|
|
|
|
|
if (!IcingaApplication::GetInstance()->GetEnablePerfdata() || !checkable->GetEnablePerfdata())
|
|
|
|
return;
|
|
|
|
|
|
|
|
Service::Ptr service = dynamic_pointer_cast<Service>(checkable);
|
|
|
|
Host::Ptr host;
|
|
|
|
|
|
|
|
if (service)
|
|
|
|
host = service->GetHost();
|
|
|
|
else
|
|
|
|
host = static_pointer_cast<Host>(checkable);
|
|
|
|
|
|
|
|
String metric;
|
|
|
|
std::map<String, String> tags;
|
2015-07-17 18:37:46 +02:00
|
|
|
|
|
|
|
String escaped_hostName = EscapeMetric(host->GetName());
|
|
|
|
tags["host"] = escaped_hostName;
|
2015-01-28 10:57:34 +01:00
|
|
|
|
2015-09-08 06:34:33 +02:00
|
|
|
double ts = cr->GetExecutionEnd();
|
|
|
|
|
2015-01-28 10:57:34 +01:00
|
|
|
if (service) {
|
|
|
|
String serviceName = service->GetShortName();
|
2015-07-17 18:37:46 +02:00
|
|
|
String escaped_serviceName = EscapeMetric(serviceName);
|
|
|
|
metric = "icinga.service." + escaped_serviceName;
|
2015-01-28 10:57:34 +01:00
|
|
|
|
2015-09-08 06:34:33 +02:00
|
|
|
SendMetric(metric + ".state", tags, service->GetState(), ts);
|
2015-01-28 10:57:34 +01:00
|
|
|
} else {
|
|
|
|
metric = "icinga.host";
|
2015-09-08 06:34:33 +02:00
|
|
|
SendMetric(metric + ".state", tags, host->GetState(), ts);
|
2015-01-28 10:57:34 +01:00
|
|
|
}
|
|
|
|
|
2015-09-08 06:34:33 +02:00
|
|
|
SendMetric(metric + ".state_type", tags, checkable->GetStateType(), ts);
|
|
|
|
SendMetric(metric + ".reachable", tags, checkable->IsReachable(), ts);
|
|
|
|
SendMetric(metric + ".downtime_depth", tags, checkable->GetDowntimeDepth(), ts);
|
2016-06-23 13:04:23 +02:00
|
|
|
SendMetric(metric + ".acknowledgement", tags, checkable->GetAcknowledgement(), ts);
|
2015-01-28 10:57:34 +01:00
|
|
|
|
2015-09-08 06:34:33 +02:00
|
|
|
SendPerfdata(metric, tags, cr, ts);
|
2015-01-28 10:57:34 +01:00
|
|
|
|
|
|
|
metric = "icinga.check";
|
|
|
|
|
|
|
|
if (service) {
|
|
|
|
tags["type"] = "service";
|
|
|
|
String serviceName = service->GetShortName();
|
2015-07-17 18:37:46 +02:00
|
|
|
String escaped_serviceName = EscapeTag(serviceName);
|
|
|
|
tags["service"] = escaped_serviceName;
|
2015-01-28 10:57:34 +01:00
|
|
|
} else {
|
|
|
|
tags["type"] = "host";
|
|
|
|
}
|
|
|
|
|
2015-09-08 06:34:33 +02:00
|
|
|
SendMetric(metric + ".current_attempt", tags, checkable->GetCheckAttempt(), ts);
|
|
|
|
SendMetric(metric + ".max_check_attempts", tags, checkable->GetMaxCheckAttempts(), ts);
|
2016-05-10 11:44:14 +02:00
|
|
|
SendMetric(metric + ".latency", tags, cr->CalculateLatency(), ts);
|
|
|
|
SendMetric(metric + ".execution_time", tags, cr->CalculateExecutionTime(), ts);
|
2015-01-28 10:57:34 +01:00
|
|
|
}
|
|
|
|
|
2015-09-08 06:34:33 +02:00
|
|
|
void OpenTsdbWriter::SendPerfdata(const String& metric, const std::map<String, String>& tags, const CheckResult::Ptr& cr, double ts)
|
2015-01-28 10:57:34 +01:00
|
|
|
{
|
|
|
|
Array::Ptr perfdata = cr->GetPerformanceData();
|
|
|
|
|
|
|
|
if (!perfdata)
|
|
|
|
return;
|
|
|
|
|
|
|
|
ObjectLock olock(perfdata);
|
2016-08-25 06:19:44 +02:00
|
|
|
for (const Value& val : perfdata) {
|
2015-01-28 10:57:34 +01:00
|
|
|
PerfdataValue::Ptr pdv;
|
|
|
|
|
|
|
|
if (val.IsObjectType<PerfdataValue>())
|
|
|
|
pdv = val;
|
|
|
|
else {
|
|
|
|
try {
|
|
|
|
pdv = PerfdataValue::Parse(val);
|
|
|
|
} catch (const std::exception&) {
|
|
|
|
Log(LogWarning, "OpenTsdbWriter")
|
|
|
|
<< "Ignoring invalid perfdata value: " << val;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-17 18:37:46 +02:00
|
|
|
String escaped_key = EscapeMetric(pdv->GetLabel());
|
2015-01-28 10:57:34 +01:00
|
|
|
boost::algorithm::replace_all(escaped_key, "::", ".");
|
|
|
|
|
2015-09-08 06:34:33 +02:00
|
|
|
SendMetric(metric + "." + escaped_key, tags, pdv->GetValue(), ts);
|
2015-01-28 10:57:34 +01:00
|
|
|
|
|
|
|
if (pdv->GetCrit())
|
2015-09-08 06:34:33 +02:00
|
|
|
SendMetric(metric + "." + escaped_key + "_crit", tags, pdv->GetCrit(), ts);
|
2015-01-28 10:57:34 +01:00
|
|
|
if (pdv->GetWarn())
|
2015-09-08 06:34:33 +02:00
|
|
|
SendMetric(metric + "." + escaped_key + "_warn", tags, pdv->GetWarn(), ts);
|
2015-01-28 10:57:34 +01:00
|
|
|
if (pdv->GetMin())
|
2015-09-08 06:34:33 +02:00
|
|
|
SendMetric(metric + "." + escaped_key + "_min", tags, pdv->GetMin(), ts);
|
2015-01-28 10:57:34 +01:00
|
|
|
if (pdv->GetMax())
|
2015-09-08 06:34:33 +02:00
|
|
|
SendMetric(metric + "." + escaped_key + "_max", tags, pdv->GetMax(), ts);
|
2015-01-28 10:57:34 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-08 06:34:33 +02:00
|
|
|
void OpenTsdbWriter::SendMetric(const String& metric, const std::map<String, String>& tags, double value, double ts)
|
2015-01-28 10:57:34 +01:00
|
|
|
{
|
|
|
|
String tags_string = "";
|
2016-08-25 06:19:44 +02:00
|
|
|
for (const Dictionary::Pair& tag : tags) {
|
2015-01-28 10:57:34 +01:00
|
|
|
tags_string += " " + tag.first + "=" + Convert::ToString(tag.second);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::ostringstream msgbuf;
|
|
|
|
/*
|
|
|
|
* must be (http://opentsdb.net/docs/build/html/user_guide/writing.html)
|
|
|
|
* put <metric> <timestamp> <value> <tagk1=tagv1[ tagk2=tagv2 ...tagkN=tagvN]>
|
|
|
|
* "tags" must include at least one tag, we use "host=HOSTNAME"
|
|
|
|
*/
|
2015-09-08 06:34:33 +02:00
|
|
|
msgbuf << "put " << metric << " " << static_cast<long>(ts) << " " << Convert::ToString(value) << " " << tags_string;
|
2015-01-28 10:57:34 +01:00
|
|
|
|
|
|
|
Log(LogDebug, "OpenTsdbWriter")
|
|
|
|
<< "Add to metric list:'" << msgbuf.str() << "'.";
|
|
|
|
|
|
|
|
/* do not send \n to debug log */
|
|
|
|
msgbuf << "\n";
|
|
|
|
String put = msgbuf.str();
|
|
|
|
|
|
|
|
ObjectLock olock(this);
|
|
|
|
|
|
|
|
if (!m_Stream)
|
|
|
|
return;
|
|
|
|
|
|
|
|
try {
|
|
|
|
m_Stream->Write(put.CStr(), put.GetLength());
|
|
|
|
} catch (const std::exception& ex) {
|
|
|
|
Log(LogCritical, "OpenTsdbWriter")
|
|
|
|
<< "Cannot write to OpenTSDB TSD on host '" << GetHost() << "' port '" << GetPort() + "'.";
|
|
|
|
|
|
|
|
m_Stream.reset();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* for metric and tag name rules, see
|
|
|
|
* http://opentsdb.net/docs/build/html/user_guide/writing.html#metrics-and-tags
|
|
|
|
*/
|
|
|
|
String OpenTsdbWriter::EscapeTag(const String& str)
|
|
|
|
{
|
|
|
|
String result = str;
|
|
|
|
|
|
|
|
boost::replace_all(result, " ", "_");
|
|
|
|
boost::replace_all(result, "\\", "_");
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
String OpenTsdbWriter::EscapeMetric(const String& str)
|
|
|
|
{
|
|
|
|
String result = str;
|
|
|
|
|
|
|
|
boost::replace_all(result, " ", "_");
|
|
|
|
boost::replace_all(result, ".", "_");
|
|
|
|
boost::replace_all(result, "\\", "_");
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|