/****************************************************************************** * 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 "i2-delegation.h" #include using namespace icinga; string DelegationComponent::GetName(void) const { return "delegation"; } void DelegationComponent::Start(void) { m_AllServices = boost::make_shared(ConfigObject::GetAllObjects(), ConfigObject::MakeTypePredicate("service")); m_AllServices->Start(); m_DelegationTimer = boost::make_shared(); m_DelegationTimer->SetInterval(30); m_DelegationTimer->OnTimerExpired.connect(boost::bind(&DelegationComponent::DelegationTimerHandler, this)); m_DelegationTimer->Start(); m_DelegationTimer->Reschedule(0); m_DelegationEndpoint = boost::make_shared(); m_DelegationEndpoint->RegisterPublication("checker::AssignService"); m_DelegationEndpoint->RegisterPublication("checker::ClearServices"); GetEndpointManager()->RegisterEndpoint(m_DelegationEndpoint); GetEndpointManager()->OnNewEndpoint.connect(bind(&DelegationComponent::NewEndpointHandler, this, _2)); } void DelegationComponent::Stop(void) { EndpointManager::Ptr mgr = GetEndpointManager(); if (mgr) mgr->UnregisterEndpoint(m_DelegationEndpoint); } void DelegationComponent::AssignService(const Endpoint::Ptr& checker, const Service& service) { RequestMessage request; request.SetMethod("checker::AssignService"); MessagePart params; params.SetProperty("service", service.GetConfigObject()->GetProperties()); request.SetParams(params); Application::Log(LogDebug, "delegation", "Trying to delegate service '" + service.GetName() + "'"); GetEndpointManager()->SendAPIMessage(m_DelegationEndpoint, checker, request, boost::bind(&DelegationComponent::AssignServiceResponseHandler, this, service, _2, _5)); } void DelegationComponent::AssignServiceResponseHandler(Service& service, const Endpoint::Ptr& sender, bool timedOut) { /* ignore the message if it's not from the designated checker for this service */ if (!sender || service.GetChecker() != sender->GetIdentity()) return; if (timedOut) { Application::Log(LogInformation, "delegation", "Service delegation for service '" + service.GetName() + "' timed out."); service.SetChecker(""); } } void DelegationComponent::ClearServices(const Endpoint::Ptr& checker) { RequestMessage request; request.SetMethod("checker::ClearServices"); MessagePart params; request.SetParams(params); GetEndpointManager()->SendUnicastMessage(m_DelegationEndpoint, checker, request); } vector DelegationComponent::GetCheckerCandidates(const Service& service) const { vector candidates; EndpointManager::Iterator it; for (it = GetEndpointManager()->Begin(); it != GetEndpointManager()->End(); it++) { Endpoint::Ptr endpoint = it->second; if (!endpoint->HasSubscription("checker::AssignService")) continue; candidates.push_back(endpoint); } return candidates; } void DelegationComponent::NewEndpointHandler(const Endpoint::Ptr& endpoint) { endpoint->OnSessionEstablished.connect(bind(&DelegationComponent::SessionEstablishedHandler, this, _1)); } void DelegationComponent::SessionEstablishedHandler(const Endpoint::Ptr& endpoint) { stringstream msgbuf; msgbuf << "Clearing assigned services for endpoint '" << endpoint->GetIdentity() << "'"; Application::Log(LogInformation, "delegation", msgbuf.str()); /* locally clear checker for all services that previously belonged to this endpoint */ ConfigObject::Set::Iterator it; for (it = m_AllServices->Begin(); it != m_AllServices->End(); it++) { Service service = *it; if (service.GetChecker() == endpoint->GetIdentity()) service.SetChecker(""); } /* remotely clear services for this endpoint */ ClearServices(endpoint); } void DelegationComponent::DelegationTimerHandler(void) { map histogram; EndpointManager::Iterator eit; for (eit = GetEndpointManager()->Begin(); eit != GetEndpointManager()->End(); eit++) histogram[eit->second] = 0; vector services; /* build "checker -> service count" histogram */ ConfigObject::Set::Iterator it; for (it = m_AllServices->Begin(); it != m_AllServices->End(); it++) { Service service = *it; services.push_back(service); string checker = service.GetChecker(); if (checker.empty()) continue; Endpoint::Ptr endpoint = GetEndpointManager()->GetEndpointByIdentity(checker); if (!endpoint) continue; histogram[endpoint]++; } std::random_shuffle(services.begin(), services.end()); bool need_clear = false; int delegated = 0; /* re-assign services */ vector::iterator sit; for (sit = services.begin(); sit != services.end(); sit++) { Service service = *sit; string checker = service.GetChecker(); Endpoint::Ptr oldEndpoint; if (!checker.empty()) oldEndpoint = GetEndpointManager()->GetEndpointByIdentity(checker); vector candidates = GetCheckerCandidates(service); int avg_services = 0, overflow_tolerance = 0; vector::iterator cit; if (candidates.size() > 0) { std::random_shuffle(candidates.begin(), candidates.end()); stringstream msgbuf; msgbuf << "Service: " << service.GetName() << ", candidates: " << candidates.size(); Application::Log(LogDebug, "delegation", msgbuf.str()); for (cit = candidates.begin(); cit != candidates.end(); cit++) avg_services += histogram[*cit]; avg_services /= candidates.size(); overflow_tolerance = candidates.size() * 2; } /* don't re-assign service if the checker is still valid * and doesn't have too many services */ if (oldEndpoint && find(candidates.begin(), candidates.end(), oldEndpoint) != candidates.end() && histogram[oldEndpoint] <= avg_services + overflow_tolerance) continue; /* clear the service's current checker */ if (!checker.empty()) { need_clear = true; service.SetChecker(""); if (oldEndpoint) histogram[oldEndpoint]--; } /* find a new checker for the service */ for (cit = candidates.begin(); cit != candidates.end(); cit++) { Endpoint::Ptr newEndpoint = *cit; /* does this checker already have too many services */ if (histogram[newEndpoint] > avg_services) continue; service.SetChecker(newEndpoint->GetIdentity()); histogram[newEndpoint]++; delegated++; break; } assert(candidates.size() == 0 || !service.GetChecker().empty()); } if (delegated > 0) { if (need_clear) { map::iterator hit; for (hit = histogram.begin(); hit != histogram.end(); hit++) { ClearServices(hit->first); } } for (sit = services.begin(); sit != services.end(); sit++) { string checker = sit->GetChecker(); Endpoint::Ptr endpoint = GetEndpointManager()->GetEndpointByIdentity(checker); if (!endpoint) continue; AssignService(endpoint, *sit); } } stringstream msgbuf; msgbuf << "Updated delegations for " << delegated << " services"; Application::Log(LogInformation, "delegation", msgbuf.str()); } EXPORT_COMPONENT(delegation, DelegationComponent);