icinga2/icinga/jsonrpcendpoint.cpp

157 lines
4.9 KiB
C++
Raw Normal View History

/******************************************************************************
* 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 *
2012-05-11 13:33:57 +02:00
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. *
******************************************************************************/
#include "i2-icinga.h"
using namespace icinga;
2012-04-16 16:27:41 +02:00
2012-04-23 13:45:41 +02:00
string JsonRpcEndpoint::GetAddress(void) const
{
if (!m_Client)
return "<disconnected endpoint>";
return m_Client->GetPeerAddress();
2012-04-23 13:45:41 +02:00
}
2012-04-16 16:27:41 +02:00
JsonRpcClient::Ptr JsonRpcEndpoint::GetClient(void)
{
return m_Client;
}
2012-05-07 13:48:17 +02:00
void JsonRpcEndpoint::Connect(string node, string service, shared_ptr<SSL_CTX> sslContext)
2012-04-18 15:22:25 +02:00
{
2012-04-24 14:02:15 +02:00
JsonRpcClient::Ptr client = make_shared<JsonRpcClient>(RoleOutbound, sslContext);
2012-04-18 15:22:25 +02:00
SetClient(client);
2012-05-07 13:48:17 +02:00
client->Connect(node, service);
client->Start();
2012-04-18 15:22:25 +02:00
}
2012-04-16 16:27:41 +02:00
void JsonRpcEndpoint::SetClient(JsonRpcClient::Ptr client)
{
m_Client = client;
2012-04-18 15:22:25 +02:00
client->OnNewMessage += bind_weak(&JsonRpcEndpoint::NewMessageHandler, shared_from_this());
client->OnClosed += bind_weak(&JsonRpcEndpoint::ClientClosedHandler, shared_from_this());
client->OnError += bind_weak(&JsonRpcEndpoint::ClientErrorHandler, shared_from_this());
client->OnVerifyCertificate += bind_weak(&JsonRpcEndpoint::VerifyCertificateHandler, shared_from_this());
2012-04-18 15:22:25 +02:00
}
bool JsonRpcEndpoint::IsLocal(void) const
{
return false;
2012-04-16 16:27:41 +02:00
}
bool JsonRpcEndpoint::IsConnected(void) const
{
2012-04-19 12:20:03 +02:00
return (bool)m_Client;
2012-04-16 16:27:41 +02:00
}
void JsonRpcEndpoint::ProcessRequest(Endpoint::Ptr sender, const RequestMessage& message)
2012-04-16 16:27:41 +02:00
{
2012-04-18 15:22:25 +02:00
if (IsConnected()) {
string id;
if (message.GetID(&id))
// TODO: remove calls after a certain timeout (and notify callers?)
m_PendingCalls[id] = sender;
2012-04-16 16:27:41 +02:00
m_Client->SendMessage(message);
} else {
// TODO: persist the event
2012-04-18 15:22:25 +02:00
}
2012-04-16 16:27:41 +02:00
}
void JsonRpcEndpoint::ProcessResponse(Endpoint::Ptr sender, const ResponseMessage& message)
2012-04-16 16:27:41 +02:00
{
if (IsConnected())
m_Client->SendMessage(message);
}
2012-04-18 15:22:25 +02:00
int JsonRpcEndpoint::NewMessageHandler(const NewMessageEventArgs& nmea)
{
2012-05-16 11:30:54 +02:00
const MessagePart& message = nmea.Message;
2012-04-18 15:22:25 +02:00
Endpoint::Ptr sender = static_pointer_cast<Endpoint>(shared_from_this());
string method;
2012-05-16 11:30:54 +02:00
if (message.GetProperty("method", &method)) {
if (!HasPublication(method))
2012-04-23 13:45:41 +02:00
return 0;
RequestMessage request = message;
2012-04-18 15:22:25 +02:00
string id;
if (request.GetID(&id))
GetEndpointManager()->SendAnycastMessage(sender, request);
2012-04-18 15:22:25 +02:00
else
GetEndpointManager()->SendMulticastMessage(sender, request);
2012-04-18 15:22:25 +02:00
} else {
ResponseMessage response = message;
2012-04-18 15:22:25 +02:00
// TODO: deal with response messages
throw NotImplementedException();
}
return 0;
}
2012-05-10 13:46:04 +02:00
int JsonRpcEndpoint::ClientClosedHandler(const EventArgs&)
2012-04-18 15:22:25 +02:00
{
2012-04-27 11:57:14 +02:00
Application::Log("Lost connection to endpoint: identity=" + GetIdentity());
2012-04-18 15:22:25 +02:00
m_PendingCalls.clear();
// TODO: _only_ clear non-persistent publications/subscriptions
// unregister ourselves if no persistent publications/subscriptions are left (use a timer for that, once we have a TTL property for the topics)
ClearSubscriptions();
ClearPublications();
// remove the endpoint if there are no more subscriptions */
if (BeginSubscriptions() == EndSubscriptions())
GetEndpointManager()->UnregisterEndpoint(static_pointer_cast<Endpoint>(shared_from_this()));
2012-04-18 15:22:25 +02:00
m_Client.reset();
// TODO: persist events, etc., for now we just disable the endpoint
return 0;
}
int JsonRpcEndpoint::ClientErrorHandler(const SocketErrorEventArgs& ea)
{
2012-05-26 21:30:04 +02:00
cerr << "Error occured for JSON-RPC socket: Message=" << ea.Exception.what() << endl;
2012-04-18 15:22:25 +02:00
return 0;
}
int JsonRpcEndpoint::VerifyCertificateHandler(const VerifyCertificateEventArgs& ea)
{
2012-04-24 16:27:23 +02:00
if (ea.Certificate && ea.ValidCertificate) {
string identity = Utility::GetCertificateCN(ea.Certificate);
if (GetIdentity().empty() && !identity.empty())
SetIdentity(identity);
}
return 0;
}
void JsonRpcEndpoint::Stop(void)
{
if (m_Client)
m_Client->Close();
}