/****************************************************************************** * Icinga 2 * * Copyright (C) 2012-2013 Icinga Development Team (http://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 "livestatus/listener.h" #include "config/configcompilercontext.h" #include "base/objectlock.h" #include "base/dynamictype.h" #include "base/logger_fwd.h" #include "base/exception.h" #include "base/tcpsocket.h" #include "base/unixsocket.h" #include "base/networkstream.h" #include "base/application.h" #include "base/scriptfunction.h" #include "base/convert.h" using namespace icinga; REGISTER_TYPE(LivestatusListener); REGISTER_SCRIPTFUNCTION(ValidateSocketType, &LivestatusListener::ValidateSocketType); static int l_ClientsConnected = 0; static int l_Connections = 0; static boost::mutex l_ComponentMutex; /** * Starts the component. */ void LivestatusListener::Start(void) { DynamicObject::Start(); if (GetSocketType() == "tcp") { TcpSocket::Ptr socket = make_shared(); socket->Bind(GetBindHost(), GetBindPort(), AF_INET); boost::thread thread(boost::bind(&LivestatusListener::ServerThreadProc, this, socket)); thread.detach(); Log(LogInformation, "livestatus", "Created tcp socket listening on host '" + GetBindHost() + "' port '" + GetBindPort() + "'."); } else if (GetSocketType() == "unix") { #ifndef _WIN32 UnixSocket::Ptr socket = make_shared(); socket->Bind(GetSocketPath()); /* group must be able to write */ mode_t mode = S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP; if (chmod(GetSocketPath().CStr(), mode) < 0) { BOOST_THROW_EXCEPTION(posix_error() << boost::errinfo_api_function("chmod") << boost::errinfo_errno(errno) << boost::errinfo_file_name(GetSocketPath())); } boost::thread thread(boost::bind(&LivestatusListener::ServerThreadProc, this, socket)); thread.detach(); Log(LogInformation, "livestatus", "Created unix socket in '" + GetSocketPath() + "'."); #else /* no unix sockets on windows */ Log(LogCritical, "livestatus", "Unix sockets are not supported on Windows."); return; #endif } } int LivestatusListener::GetClientsConnected(void) { boost::mutex::scoped_lock lock(l_ComponentMutex); return l_ClientsConnected; } int LivestatusListener::GetConnections(void) { boost::mutex::scoped_lock lock(l_ComponentMutex); return l_Connections; } void LivestatusListener::ServerThreadProc(const Socket::Ptr& server) { server->Listen(); for (;;) { Socket::Ptr client = server->Accept(); Log(LogInformation, "livestatus", "Client connected"); boost::thread thread(boost::bind(&LivestatusListener::ClientThreadProc, this, client)); thread.detach(); } } void LivestatusListener::ClientThreadProc(const Socket::Ptr& client) { { boost::mutex::scoped_lock lock(l_ComponentMutex); l_ClientsConnected++; l_Connections++; } Stream::Ptr stream = make_shared(client); for (;;) { String line; ReadLineContext context; std::vector lines; while (stream->ReadLine(&line, context)) { if (line.GetLength() > 0) lines.push_back(line); else break; } try { Query::Ptr query = make_shared(lines, GetCompatLogPath()); if (!query->Execute(stream)) break; } catch (const std::exception& ex) { std::ostringstream info; info << "Exception thrown while running livestatus query: " << std::endl << DiagnosticInformation(ex); Log(LogCritical, "livestatus", info.str()); return; } } { boost::mutex::scoped_lock lock(l_ComponentMutex); l_ClientsConnected--; } } void LivestatusListener::ValidateSocketType(const String& location, const Dictionary::Ptr& attrs) { Value socket_type = attrs->Get("socket_type"); if (!socket_type.IsEmpty() && socket_type != "unix" && socket_type != "tcp") { ConfigCompilerContext::GetInstance()->AddMessage(true, "Validation failed for " + location + ": Socket type '" + socket_type + "' is invalid."); } }