icinga2/icinga/endpointmanager.cpp

272 lines
7.4 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;
/**
* Sets the identity of the endpoint manager. This identity is used when
* connecting to remote peers.
*
* @param identity The new identity.
*/
2012-04-27 13:12:06 +02:00
void EndpointManager::SetIdentity(string identity)
2012-04-24 14:02:15 +02:00
{
m_Identity = identity;
2012-04-24 14:02:15 +02:00
}
/**
* Retrieves the identity for the endpoint manager.
*
* @returns The identity.
*/
string EndpointManager::GetIdentity(void) const
{
return m_Identity;
}
/**
* Sets the SSL context that is used for remote connections.
*
* @param sslContext The new SSL context.
*/
2012-04-27 13:12:06 +02:00
void EndpointManager::SetSSLContext(shared_ptr<SSL_CTX> sslContext)
{
m_SSLContext = sslContext;
}
/**
* Retrieves the SSL context that is used for remote connections.
*
* @returns The SSL context.
*/
2012-04-27 13:12:06 +02:00
shared_ptr<SSL_CTX> EndpointManager::GetSSLContext(void) const
{
return m_SSLContext;
}
/**
* Creates a new JSON-RPC listener on the specified port.
*
2012-05-18 23:25:06 +02:00
* @param service The port to listen on.
*/
2012-05-07 13:48:17 +02:00
void EndpointManager::AddListener(string service)
{
if (!GetSSLContext())
throw InvalidArgumentException("SSL context is required for AddListener()");
2012-04-26 21:33:23 +02:00
stringstream s;
2012-05-07 13:48:17 +02:00
s << "Adding new listener: port " << service;
2012-04-26 21:33:23 +02:00
Application::Log(s.str());
2012-04-24 14:02:15 +02:00
JsonRpcServer::Ptr server = make_shared<JsonRpcServer>(m_SSLContext);
RegisterServer(server);
2012-05-07 13:48:17 +02:00
server->Bind(service, AF_INET6);
server->Listen();
server->Start();
}
/**
* Creates a new JSON-RPC client and connects to the specified host and port.
*
2012-05-18 23:25:06 +02:00
* @param node The remote host.
* @param service The remote port.
*/
2012-05-07 13:48:17 +02:00
void EndpointManager::AddConnection(string node, string service)
{
stringstream s;
2012-05-07 13:48:17 +02:00
s << "Adding new endpoint: [" << node << "]:" << service;
Application::Log(s.str());
2012-04-18 15:22:25 +02:00
JsonRpcEndpoint::Ptr endpoint = make_shared<JsonRpcEndpoint>();
RegisterEndpoint(endpoint);
2012-05-07 13:48:17 +02:00
endpoint->Connect(node, service, m_SSLContext);
}
/**
* Registers a new JSON-RPC server with this endpoint manager.
*
* @param server The JSON-RPC server.
*/
void EndpointManager::RegisterServer(JsonRpcServer::Ptr server)
{
m_Servers.push_back(server);
2012-05-16 11:30:54 +02:00
server->OnNewClient += bind_weak(&EndpointManager::NewClientHandler,
shared_from_this());
}
/**
* Processes a new client connection.
*
* @param ncea Event arguments.
*/
2012-04-18 15:22:25 +02:00
int EndpointManager::NewClientHandler(const NewClientEventArgs& ncea)
{
string address = ncea.Client->GetPeerAddress();
Application::Log("Accepted new client from " + address);
2012-04-18 15:22:25 +02:00
JsonRpcEndpoint::Ptr endpoint = make_shared<JsonRpcEndpoint>();
endpoint->SetClient(static_pointer_cast<JsonRpcClient>(ncea.Client));
RegisterEndpoint(endpoint);
return 0;
}
/**
* Unregisters a JSON-RPC server.
*
* @param server The JSON-RPC server.
*/
2012-04-18 15:22:25 +02:00
void EndpointManager::UnregisterServer(JsonRpcServer::Ptr server)
{
m_Servers.erase(
remove(m_Servers.begin(), m_Servers.end(), server),
m_Servers.end());
2012-04-18 15:22:25 +02:00
// TODO: unbind event
}
/**
* Registers a new endpoint with this endpoint manager.
*
* @param endpoint The new endpoint.
*/
void EndpointManager::RegisterEndpoint(Endpoint::Ptr endpoint)
{
2012-04-27 13:12:06 +02:00
if (!endpoint->IsLocal() && endpoint->GetIdentity() != "")
throw InvalidArgumentException("Identity must be empty.");
2012-04-18 15:22:25 +02:00
endpoint->SetEndpointManager(static_pointer_cast<EndpointManager>(shared_from_this()));
m_Endpoints.push_back(endpoint);
2012-04-18 15:22:25 +02:00
NewEndpointEventArgs neea;
neea.Source = shared_from_this();
neea.Endpoint = endpoint;
OnNewEndpoint(neea);
}
/**
* Unregisters an endpoint.
*
* @param endpoint The endpoint.
*/
void EndpointManager::UnregisterEndpoint(Endpoint::Ptr endpoint)
{
m_Endpoints.erase(
remove(m_Endpoints.begin(), m_Endpoints.end(), endpoint),
m_Endpoints.end());
}
/**
* Sends a unicast message to the specified recipient.
*
* @param sender The sender of the message.
* @param recipient The recipient of the message.
* @param message The request.
*/
2012-05-16 11:30:54 +02:00
void EndpointManager::SendUnicastMessage(Endpoint::Ptr sender,
Endpoint::Ptr recipient, const MessagePart& message)
{
/* don't forward messages back to the sender */
if (sender == recipient)
return;
/* don't forward messages between non-local endpoints */
if (!sender->IsLocal() && !recipient->IsLocal())
return;
recipient->ProcessRequest(sender, message);
}
/**
* Sends a message to exactly one recipient out of all recipients who have a
* subscription for the message's topic.
*
* @param sender The sender of the message.
* @param message The message.
*/
2012-05-16 11:30:54 +02:00
void EndpointManager::SendAnycastMessage(Endpoint::Ptr sender,
const RequestMessage& message)
{
2012-04-16 16:27:41 +02:00
throw NotImplementedException();
}
/**
* Sends a message to all recipients who have a subscription for the
* message's topic.
*
* @param sender The sender of the message.
* @param message The message.
*/
2012-05-16 11:30:54 +02:00
void EndpointManager::SendMulticastMessage(Endpoint::Ptr sender,
const RequestMessage& message)
2012-04-16 16:27:41 +02:00
{
string id;
if (message.GetID(&id))
2012-04-16 16:27:41 +02:00
throw InvalidArgumentException("Multicast requests must not have an ID.");
string method;
if (!message.GetMethod(&method))
throw InvalidArgumentException("Message is missing the 'method' property.");
2012-04-16 16:27:41 +02:00
for (vector<Endpoint::Ptr>::iterator i = m_Endpoints.begin(); i != m_Endpoints.end(); i++)
2012-04-16 16:27:41 +02:00
{
Endpoint::Ptr recipient = *i;
if (recipient->HasSubscription(method))
SendUnicastMessage(sender, recipient, message);
2012-04-18 15:22:25 +02:00
}
}
/**
* Calls the specified callback function for each registered endpoint.
*
* @param callback The callback function.
*/
2012-05-08 09:20:42 +02:00
void EndpointManager::ForEachEndpoint(function<int (const NewEndpointEventArgs&)> callback)
2012-04-18 15:22:25 +02:00
{
NewEndpointEventArgs neea;
neea.Source = shared_from_this();
vector<Endpoint::Ptr>::iterator prev, i;
for (i = m_Endpoints.begin(); i != m_Endpoints.end(); ) {
prev = i;
i++;
neea.Endpoint = *prev;
callback(neea);
}
}
2012-05-07 13:48:17 +02:00
/**
* Retrieves an endpoint that has the specified identity.
*
* @param identity The identity of the endpoint.
*/
Endpoint::Ptr EndpointManager::GetEndpointByIdentity(string identity) const
2012-05-07 13:48:17 +02:00
{
vector<Endpoint::Ptr>::const_iterator i;
2012-05-07 13:48:17 +02:00
for (i = m_Endpoints.begin(); i != m_Endpoints.end(); i++) {
if ((*i)->GetIdentity() == identity)
return *i;
2012-05-07 13:48:17 +02:00
}
return Endpoint::Ptr();
2012-05-07 13:48:17 +02:00
}