2016-04-19 13:54:41 +02:00
/******************************************************************************
* Icinga 2 *
* Copyright ( C ) 2012 - 2016 Icinga Development Team ( https : //www.icinga.org/) *
* *
* 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/influxdbwriter.hpp"
# include "perfdata/influxdbwriter.tcpp"
# include "remote/url.hpp"
# include "remote/httprequest.hpp"
# include "remote/httpresponse.hpp"
# include "icinga/service.hpp"
# include "icinga/macroprocessor.hpp"
# include "icinga/icingaapplication.hpp"
# include "icinga/compatutility.hpp"
# include "icinga/perfdatavalue.hpp"
# include "icinga/checkcommand.hpp"
# include "base/tcpsocket.hpp"
# include "base/configtype.hpp"
# include "base/objectlock.hpp"
# include "base/logger.hpp"
# include "base/convert.hpp"
# include "base/utility.hpp"
# include "base/application.hpp"
# include "base/stream.hpp"
# include "base/networkstream.hpp"
# include "base/exception.hpp"
# include "base/statsfunction.hpp"
# include "base/tlsutility.hpp"
# include <boost/algorithm/string.hpp>
# include <boost/algorithm/string/classification.hpp>
# include <boost/foreach.hpp>
# include <boost/algorithm/string/split.hpp>
# include <boost/algorithm/string/replace.hpp>
2016-06-07 14:35:16 +02:00
# include <boost/regex.hpp>
2016-04-19 13:54:41 +02:00
using namespace icinga ;
REGISTER_TYPE ( InfluxdbWriter ) ;
REGISTER_STATSFUNCTION ( InfluxdbWriter , & InfluxdbWriter : : StatsFunc ) ;
void InfluxdbWriter : : StatsFunc ( const Dictionary : : Ptr & status , const Array : : Ptr & )
{
Dictionary : : Ptr nodes = new Dictionary ( ) ;
BOOST_FOREACH ( const InfluxdbWriter : : Ptr & influxdbwriter , ConfigType : : GetObjectsByType < InfluxdbWriter > ( ) ) {
nodes - > Set ( influxdbwriter - > GetName ( ) , 1 ) ; //add more stats
}
status - > Set ( " influxdbwriter " , nodes ) ;
}
void InfluxdbWriter : : Start ( bool runtimeCreated )
{
m_DataBuffer = new Array ( ) ;
ObjectImpl < InfluxdbWriter > : : Start ( runtimeCreated ) ;
m_FlushTimer = new Timer ( ) ;
m_FlushTimer - > SetInterval ( GetFlushInterval ( ) ) ;
m_FlushTimer - > OnTimerExpired . connect ( boost : : bind ( & InfluxdbWriter : : FlushTimeout , this ) ) ;
m_FlushTimer - > Start ( ) ;
m_FlushTimer - > Reschedule ( 0 ) ;
Service : : OnNewCheckResult . connect ( boost : : bind ( & InfluxdbWriter : : CheckResultHandler , this , _1 , _2 ) ) ;
}
Stream : : Ptr InfluxdbWriter : : Connect ( void )
{
TcpSocket : : Ptr socket = new TcpSocket ( ) ;
Log ( LogNotice , " InfluxdbWriter " )
< < " Reconnecting to InfluxDB on host ' " < < GetHost ( ) < < " ' port ' " < < GetPort ( ) < < " '. " ;
try {
socket - > Connect ( GetHost ( ) , GetPort ( ) ) ;
} catch ( std : : exception & ) {
Log ( LogWarning , " InfluxdbWriter " )
< < " Can't connect to InfluxDB on host ' " < < GetHost ( ) < < " ' port ' " < < GetPort ( ) < < " '. " ;
return Stream : : Ptr ( ) ;
}
if ( GetSslEnable ( ) ) {
boost : : shared_ptr < SSL_CTX > ssl_context ;
try {
ssl_context = MakeSSLContext ( GetSslCert ( ) , GetSslKey ( ) , GetSslCaCert ( ) ) ;
} catch ( std : : exception & ) {
Log ( LogWarning , " InfluxdbWriter " )
< < " Unable to create SSL context. " ;
return Stream : : Ptr ( ) ;
}
TlsStream : : Ptr tls_stream = new TlsStream ( socket , GetHost ( ) , RoleClient , ssl_context ) ;
try {
tls_stream - > Handshake ( ) ;
} catch ( std : : exception & ) {
Log ( LogWarning , " InfluxdbWriter " )
< < " TLS handshake with host ' " < < GetHost ( ) < < " ' failed. " ;
return Stream : : Ptr ( ) ;
}
return tls_stream ;
} else {
return new NetworkStream ( socket ) ;
}
}
void InfluxdbWriter : : CheckResultHandler ( const Checkable : : Ptr & checkable , const CheckResult : : Ptr & cr )
{
CONTEXT ( " Processing check result for ' " + checkable - > GetName ( ) + " ' " ) ;
if ( ! IcingaApplication : : GetInstance ( ) - > GetEnablePerfdata ( ) | | ! checkable - > GetEnablePerfdata ( ) )
return ;
Host : : Ptr host ;
Service : : Ptr service ;
boost : : tie ( host , service ) = GetHostService ( checkable ) ;
MacroProcessor : : ResolverList resolvers ;
if ( service )
resolvers . push_back ( std : : make_pair ( " service " , service ) ) ;
resolvers . push_back ( std : : make_pair ( " host " , host ) ) ;
resolvers . push_back ( std : : make_pair ( " icinga " , IcingaApplication : : GetInstance ( ) ) ) ;
String prefix ;
double ts = cr - > GetExecutionEnd ( ) ;
// Clone the template and perform an in-place macro expansion of measurement and tag values
Dictionary : : Ptr tmpl_clean = service ? GetServiceTemplate ( ) : GetHostTemplate ( ) ;
Dictionary : : Ptr tmpl = static_pointer_cast < Dictionary > ( tmpl_clean - > Clone ( ) ) ;
tmpl - > Set ( " measurement " , MacroProcessor : : ResolveMacros ( tmpl - > Get ( " measurement " ) , resolvers , cr ) ) ;
Dictionary : : Ptr tags = tmpl - > Get ( " tags " ) ;
if ( tags ) {
ObjectLock olock ( tags ) ;
retry :
BOOST_FOREACH ( const Dictionary : : Pair & pair , tags ) {
// Prevent missing macros from warning; will return an empty value
// which will be filtered out in SendMetric()
String missing_macro ;
tags - > Set ( pair . first , MacroProcessor : : ResolveMacros ( pair . second , resolvers , cr , & missing_macro ) ) ;
}
}
2016-06-08 12:09:21 +02:00
SendPerfdata ( tmpl , checkable , cr , ts ) ;
2016-04-19 13:54:41 +02:00
}
2016-06-08 12:09:21 +02:00
String InfluxdbWriter : : FormatInteger ( const int val )
{
return Convert : : ToString ( val ) + " i " ;
}
String InfluxdbWriter : : FormatBoolean ( const bool val )
{
return val ? " true " : " false " ;
}
void InfluxdbWriter : : SendPerfdata ( const Dictionary : : Ptr & tmpl , const Checkable : : Ptr & checkable , const CheckResult : : Ptr & cr , double ts )
2016-04-19 13:54:41 +02:00
{
Array : : Ptr perfdata = cr - > GetPerformanceData ( ) ;
if ( ! perfdata )
return ;
ObjectLock olock ( perfdata ) ;
BOOST_FOREACH ( const Value & val , perfdata ) {
PerfdataValue : : Ptr pdv ;
if ( val . IsObjectType < PerfdataValue > ( ) )
pdv = val ;
else {
try {
pdv = PerfdataValue : : Parse ( val ) ;
} catch ( const std : : exception & ) {
Log ( LogWarning , " InfluxdbWriter " )
< < " Ignoring invalid perfdata value: " < < val ;
continue ;
}
}
2016-06-07 14:35:16 +02:00
Dictionary : : Ptr fields = new Dictionary ( ) ;
fields - > Set ( String ( " value " ) , pdv - > GetValue ( ) ) ;
2016-06-08 12:09:21 +02:00
2016-04-19 13:54:41 +02:00
if ( GetEnableSendThresholds ( ) ) {
if ( pdv - > GetCrit ( ) )
2016-06-07 14:35:16 +02:00
fields - > Set ( String ( " crit " ) , pdv - > GetCrit ( ) ) ;
2016-04-19 13:54:41 +02:00
if ( pdv - > GetWarn ( ) )
2016-06-07 14:35:16 +02:00
fields - > Set ( String ( " warn " ) , pdv - > GetWarn ( ) ) ;
2016-04-19 13:54:41 +02:00
if ( pdv - > GetMin ( ) )
2016-06-07 14:35:16 +02:00
fields - > Set ( String ( " min " ) , pdv - > GetMin ( ) ) ;
2016-04-19 13:54:41 +02:00
if ( pdv - > GetMax ( ) )
2016-06-07 14:35:16 +02:00
fields - > Set ( String ( " max " ) , pdv - > GetMax ( ) ) ;
2016-04-19 13:54:41 +02:00
}
2016-06-07 14:35:16 +02:00
2016-06-08 12:09:21 +02:00
if ( GetEnableSendMetadata ( ) ) {
Host : : Ptr host ;
Service : : Ptr service ;
boost : : tie ( host , service ) = GetHostService ( checkable ) ;
if ( service )
fields - > Set ( String ( " state " ) , FormatInteger ( service - > GetState ( ) ) ) ;
else
fields - > Set ( String ( " state " ) , FormatInteger ( host - > GetState ( ) ) ) ;
fields - > Set ( String ( " current_attempt " ) , FormatInteger ( checkable - > GetCheckAttempt ( ) ) ) ;
fields - > Set ( String ( " max_check_attempts " ) , FormatInteger ( checkable - > GetMaxCheckAttempts ( ) ) ) ;
fields - > Set ( String ( " state_type " ) , FormatInteger ( checkable - > GetStateType ( ) ) ) ;
fields - > Set ( String ( " reachable " ) , FormatBoolean ( checkable - > IsReachable ( ) ) ) ;
fields - > Set ( String ( " downtime_depth " ) , FormatInteger ( checkable - > GetDowntimeDepth ( ) ) ) ;
2016-06-23 13:04:23 +02:00
fields - > Set ( String ( " acknowledgement " ) , FormatInteger ( checkable - > GetAcknowledgement ( ) ) ) ;
2016-06-08 12:09:21 +02:00
fields - > Set ( String ( " latency " ) , cr - > CalculateLatency ( ) ) ;
fields - > Set ( String ( " execution_time " ) , cr - > CalculateExecutionTime ( ) ) ;
}
2016-06-07 14:35:16 +02:00
SendMetric ( tmpl , pdv - > GetLabel ( ) , fields , ts ) ;
2016-04-19 13:54:41 +02:00
}
}
2016-06-07 14:35:16 +02:00
String InfluxdbWriter : : EscapeKey ( const String & str )
{
// Iterate over the key name and escape commas and spaces with a backslash
String result = str ;
boost : : algorithm : : replace_all ( result , " , " , " \\ , " ) ;
boost : : algorithm : : replace_all ( result , " " , " \\ " ) ;
2016-07-28 11:28:53 +02:00
// InfluxDB 'feature': although backslashes are allowed in keys they also act
// as escape sequences when followed by ',' or ' '. When your tag is like
// 'metric=C:\' bad things happen. Backslashes themselves cannot be escaped
// and through experimentation they also escape '='. To be safe we replace
// trailing backslashes with and underscore.
size_t length = result . GetLength ( ) ;
if ( result [ length - 1 ] = = ' \\ ' )
result [ length - 1 ] = ' _ ' ;
2016-06-27 11:38:07 +02:00
return result ;
2016-06-07 14:35:16 +02:00
}
String InfluxdbWriter : : EscapeField ( const String & str )
{
2016-06-08 12:09:21 +02:00
// Handle integers
boost : : regex integer ( " -? \\ d+i " ) ;
if ( boost : : regex_match ( str . GetData ( ) , integer ) ) {
return str ;
}
2016-06-07 14:35:16 +02:00
// Handle numerics
boost : : regex numeric ( " -? \\ d+( \\ . \\ d+)?((e|E)[+-]? \\ d+)? " ) ;
if ( boost : : regex_match ( str . GetData ( ) , numeric ) ) {
return str ;
}
// Handle booleans
boost : : regex boolean_true ( " t|true " , boost : : regex : : icase ) ;
if ( boost : : regex_match ( str . GetData ( ) , boolean_true ) )
return " true " ;
boost : : regex boolean_false ( " f|false " , boost : : regex : : icase ) ;
if ( boost : : regex_match ( str . GetData ( ) , boolean_false ) )
return " false " ;
// Otherwise it's a string and needs escaping and quoting
String result = str ;
boost : : algorithm : : replace_all ( result , " \" " , " \\ \" " ) ;
return " \" " + result + " \" " ;
}
void InfluxdbWriter : : SendMetric ( const Dictionary : : Ptr & tmpl , const String & label , const Dictionary : : Ptr & fields , double ts )
2016-04-19 13:54:41 +02:00
{
std : : ostringstream msgbuf ;
2016-06-07 14:35:16 +02:00
msgbuf < < EscapeKey ( tmpl - > Get ( " measurement " ) ) ;
2016-04-19 13:54:41 +02:00
Dictionary : : Ptr tags = tmpl - > Get ( " tags " ) ;
if ( tags ) {
ObjectLock olock ( tags ) ;
BOOST_FOREACH ( const Dictionary : : Pair & pair , tags ) {
// Empty macro expansion, no tag
2016-06-07 14:35:16 +02:00
if ( ! pair . second . IsEmpty ( ) ) {
msgbuf < < " , " < < EscapeKey ( pair . first ) < < " = " < < EscapeKey ( pair . second ) ;
}
2016-04-19 13:54:41 +02:00
}
}
2016-07-28 11:28:53 +02:00
msgbuf < < " ,metric= " < < EscapeKey ( label ) < < " " ;
2016-06-07 14:35:16 +02:00
bool first = true ;
ObjectLock fieldLock ( fields ) ;
BOOST_FOREACH ( const Dictionary : : Pair & pair , fields ) {
if ( first )
first = false ;
else
msgbuf < < " , " ;
msgbuf < < EscapeKey ( pair . first ) < < " = " < < EscapeField ( pair . second ) ;
}
msgbuf < < " " < < static_cast < unsigned long > ( ts ) ;
2016-04-19 13:54:41 +02:00
Log ( LogDebug , " InfluxdbWriter " )
< < " Add to metric list:' " < < msgbuf . str ( ) < < " '. " ;
// Atomically buffer the data point
ObjectLock olock ( m_DataBuffer ) ;
m_DataBuffer - > Add ( String ( msgbuf . str ( ) ) ) ;
// Flush if we've buffered too much to prevent excessive memory use
if ( m_DataBuffer - > GetLength ( ) > = GetFlushThreshold ( ) ) {
Log ( LogDebug , " InfluxdbWriter " )
< < " Data buffer overflow writing " < < m_DataBuffer - > GetLength ( ) < < " data points " ;
Flush ( ) ;
}
}
void InfluxdbWriter : : FlushTimeout ( void )
{
// Prevent new data points from being added to the array, there is a
// race condition where they could disappear
ObjectLock olock ( m_DataBuffer ) ;
// Flush if there are any data available
if ( m_DataBuffer - > GetLength ( ) > 0 ) {
Log ( LogDebug , " InfluxdbWriter " )
< < " Timer expired writing " < < m_DataBuffer - > GetLength ( ) < < " data points " ;
Flush ( ) ;
}
}
void InfluxdbWriter : : Flush ( void )
{
Stream : : Ptr stream = Connect ( ) ;
// Unable to connect, play it safe and lose the data points
// to avoid a memory leak
if ( ! stream . get ( ) ) {
m_DataBuffer - > Clear ( ) ;
return ;
}
Url : : Ptr url = new Url ( ) ;
url - > SetScheme ( GetSslEnable ( ) ? " https " : " http " ) ;
url - > SetHost ( GetHost ( ) ) ;
url - > SetPort ( GetPort ( ) ) ;
std : : vector < String > path ;
path . push_back ( " write " ) ;
url - > SetPath ( path ) ;
url - > AddQueryElement ( " db " , GetDatabase ( ) ) ;
url - > AddQueryElement ( " precision " , " s " ) ;
if ( ! GetUsername ( ) . IsEmpty ( ) )
url - > AddQueryElement ( " u " , GetUsername ( ) ) ;
if ( ! GetPassword ( ) . IsEmpty ( ) )
url - > AddQueryElement ( " p " , GetPassword ( ) ) ;
// Ensure you hold a lock against m_DataBuffer so that things
// don't go missing after creating the body and clearing the buffer
String body = Utility : : Join ( m_DataBuffer , ' \n ' ) ;
m_DataBuffer - > Clear ( ) ;
HttpRequest req ( stream ) ;
req . RequestMethod = " POST " ;
req . RequestUrl = url ;
try {
req . WriteBody ( body . CStr ( ) , body . GetLength ( ) ) ;
req . Finish ( ) ;
} catch ( const std : : exception & ) {
Log ( LogWarning , " InfluxdbWriter " )
< < " Cannot write to TCP socket on host ' " < < GetHost ( ) < < " ' port ' " < < GetPort ( ) < < " '. " ;
2016-06-08 12:09:21 +02:00
return ;
2016-04-19 13:54:41 +02:00
}
HttpResponse resp ( stream , req ) ;
StreamReadContext context ;
try {
resp . Parse ( context , true ) ;
2016-06-08 12:09:21 +02:00
} catch ( const std : : exception & ) {
2016-04-19 13:54:41 +02:00
Log ( LogWarning , " InfluxdbWriter " )
< < " Cannot read from TCP socket from host ' " < < GetHost ( ) < < " ' port ' " < < GetPort ( ) < < " '. " ;
2016-06-08 12:09:21 +02:00
return ;
2016-04-19 13:54:41 +02:00
}
if ( resp . StatusCode ! = 204 ) {
Log ( LogWarning , " InfluxdbWriter " )
< < " Unexpected response code " < < resp . StatusCode ;
}
}
void InfluxdbWriter : : ValidateHostTemplate ( const Dictionary : : Ptr & value , const ValidationUtils & utils )
{
ObjectImpl < InfluxdbWriter > : : ValidateHostTemplate ( value , utils ) ;
String measurement = value - > Get ( " measurement " ) ;
if ( ! MacroProcessor : : ValidateMacroString ( measurement ) )
BOOST_THROW_EXCEPTION ( ValidationError ( this , boost : : assign : : list_of ( " host_template " ) ( " measurement " ) , " Closing $ not found in macro format string ' " + measurement + " '. " ) ) ;
Dictionary : : Ptr tags = value - > Get ( " tags " ) ;
if ( tags ) {
ObjectLock olock ( tags ) ;
BOOST_FOREACH ( const Dictionary : : Pair & pair , tags ) {
if ( ! MacroProcessor : : ValidateMacroString ( pair . second ) )
BOOST_THROW_EXCEPTION ( ValidationError ( this , boost : : assign : : list_of < String > ( " host_template " ) ( " tags " ) ( pair . first ) , " Closing $ not found in macro format string ' " + pair . second ) ) ;
}
}
}
void InfluxdbWriter : : ValidateServiceTemplate ( const Dictionary : : Ptr & value , const ValidationUtils & utils )
{
ObjectImpl < InfluxdbWriter > : : ValidateServiceTemplate ( value , utils ) ;
String measurement = value - > Get ( " measurement " ) ;
if ( ! MacroProcessor : : ValidateMacroString ( measurement ) )
BOOST_THROW_EXCEPTION ( ValidationError ( this , boost : : assign : : list_of ( " service_template " ) ( " measurement " ) , " Closing $ not found in macro format string ' " + measurement + " '. " ) ) ;
Dictionary : : Ptr tags = value - > Get ( " tags " ) ;
if ( tags ) {
ObjectLock olock ( tags ) ;
BOOST_FOREACH ( const Dictionary : : Pair & pair , tags ) {
if ( ! MacroProcessor : : ValidateMacroString ( pair . second ) )
BOOST_THROW_EXCEPTION ( ValidationError ( this , boost : : assign : : list_of < String > ( " service_template " ) ( " tags " ) ( pair . first ) , " Closing $ not found in macro format string ' " + pair . second ) ) ;
}
}
}