/****************************************************************************** * Icinga 2 * * Copyright (C) 2012 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/component.h" #include "base/objectlock.h" #include "base/dynamictype.h" #include "base/logger_fwd.h" #include "base/tcpsocket.h" #include "base/unixsocket.h" #include "base/networkstream.h" #include "base/application.h" #include using namespace icinga; using namespace livestatus; REGISTER_TYPE(LivestatusComponent); static int l_ClientsConnected = 0; static int l_Connections = 0; static boost::mutex l_ComponentMutex; /** * Starts the component. */ void LivestatusComponent::Start(void) { DynamicObject::Start(); if (GetSocketType() == "tcp") { TcpSocket::Ptr socket = boost::make_shared(); socket->Bind(GetHost(), GetPort(), AF_INET); boost::thread thread(boost::bind(&LivestatusComponent::ServerThreadProc, this, socket)); thread.detach(); } else if (GetSocketType() == "unix") { #ifndef _WIN32 UnixSocket::Ptr socket = boost::make_shared(); socket->Bind(GetSocketPath()); boost::thread thread(boost::bind(&LivestatusComponent::ServerThreadProc, this, socket)); thread.detach(); #else /* no unix sockets on windows */ Log(LogCritical, "livestatus", "Unix sockets are not supported on Windows."); return; #endif } } String LivestatusComponent::GetSocketType(void) const { Value socketType = m_SocketType; if (socketType.IsEmpty()) return "unix"; else return socketType; } String LivestatusComponent::GetSocketPath(void) const { Value socketPath = m_SocketPath; if (socketPath.IsEmpty()) return Application::GetLocalStateDir() + "/run/icinga2/livestatus"; else return socketPath; } String LivestatusComponent::GetHost(void) const { Value node = m_Host; if (node.IsEmpty()) return "127.0.0.1"; else return node; } String LivestatusComponent::GetPort(void) const { Value service = m_Port; if (service.IsEmpty()) return "6558"; else return service; } int LivestatusComponent::GetClientsConnected(void) { boost::mutex::scoped_lock lock(l_ComponentMutex); return l_ClientsConnected; } int LivestatusComponent::GetConnections(void) { boost::mutex::scoped_lock lock(l_ComponentMutex); return l_Connections; } void LivestatusComponent::ServerThreadProc(const Socket::Ptr& server) { server->Listen(); for (;;) { Socket::Ptr client = server->Accept(); Log(LogInformation, "livestatus", "Client connected"); boost::thread thread(boost::bind(&LivestatusComponent::ClientThreadProc, this, client)); thread.detach(); } } void LivestatusComponent::ClientThreadProc(const Socket::Ptr& client) { { boost::mutex::scoped_lock lock(l_ComponentMutex); l_ClientsConnected++; l_Connections++; } Stream::Ptr stream = boost::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; } Query::Ptr query = boost::make_shared(lines); if (!query->Execute(stream)) break; } { boost::mutex::scoped_lock lock(l_ComponentMutex); l_ClientsConnected--; } } void LivestatusComponent::InternalSerialize(const Dictionary::Ptr& bag, int attributeTypes) const { DynamicObject::InternalSerialize(bag, attributeTypes); if (attributeTypes & Attribute_Config) { bag->Set("socket_type", m_SocketType); bag->Set("socket_path", m_SocketPath); bag->Set("host", m_Host); bag->Set("port", m_Port); } } void LivestatusComponent::InternalDeserialize(const Dictionary::Ptr& bag, int attributeTypes) { DynamicObject::InternalDeserialize(bag, attributeTypes); if (attributeTypes & Attribute_Config) { m_SocketType = bag->Get("socket_type"); m_SocketPath = bag->Get("socket_path"); m_Host = bag->Get("host"); m_Port = bag->Get("port"); } }