/****************************************************************************** * 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 #include #include "i2-icinga.h" #ifndef _WIN32 # include "icinga-version.h" # define ICINGA_VERSION GIT_MESSAGE #endif /* _WIN32 */ using namespace icinga; /** * The entry point for the Icinga application. * * @param args Command-line arguments. * @returns An exit status. */ int IcingaApplication::Main(const vector& args) { StreamLogger::Ptr consoleLogger = boost::make_shared(&std::cout, LogInformation); Logger::RegisterLogger(consoleLogger); #ifdef _WIN32 Logger::Write(LogInformation, "icinga", "Icinga component loader"); #else /* _WIN32 */ Logger::Write(LogInformation, "icinga", "Icinga component loader (version: " ICINGA_VERSION ")"); #endif /* _WIN32 */ time(&m_StartTime); if (args.size() < 2) { stringstream msgbuf; msgbuf << "Syntax: " << args[0] << " [-S] [-L logfile] [-d] [--] "; Logger::Write(LogInformation, "icinga", msgbuf.str()); return EXIT_FAILURE; } bool enableSyslog = false; bool daemonize = false; bool parseOpts = true; string configFile; /* TODO: clean up this mess; for now it will just have to do */ vector::const_iterator it; for (it = args.begin() + 1 ; it != args.end(); it++) { string arg = *it; /* ignore empty arguments */ if (arg.empty()) continue; if (arg == "--") { parseOpts = false; continue; } if (parseOpts && arg[0] == '-') { if (arg == "-S") { enableSyslog = true; continue; } else if (arg == "-L") { if (it + 1 == args.end()) throw invalid_argument("Option -L requires a parameter"); StreamLogger::Ptr fileLogger = boost::make_shared(LogInformation); fileLogger->OpenFile(*(it + 1)); Logger::RegisterLogger(fileLogger); it++; continue; } else { throw invalid_argument("Unknown option: " + arg); } } configFile = arg; if (it + 1 != args.end()) throw invalid_argument("Trailing command line arguments after config filename."); } if (configFile.empty()) throw invalid_argument("No config file was specified on the command line."); if (enableSyslog) { SyslogLogger::Ptr syslogLogger = boost::make_shared(LogInformation); Logger::RegisterLogger(syslogLogger); } string componentDirectory = Utility::DirName(GetExePath()) + "/../lib/icinga2"; AddComponentSearchDir(componentDirectory); /* register handler for 'component' config objects */ static ConfigObject::Set::Ptr componentObjects = boost::make_shared(ConfigObject::GetAllObjects(), ConfigObject::MakeTypePredicate("component")); componentObjects->OnObjectAdded.connect(boost::bind(&IcingaApplication::NewComponentHandler, this, _2)); componentObjects->OnObjectCommitted.connect(boost::bind(&IcingaApplication::NewComponentHandler, this, _2)); componentObjects->OnObjectRemoved.connect(boost::bind(&IcingaApplication::DeletedComponentHandler, this, _2)); componentObjects->Start(); /* load convenience config component */ ConfigItemBuilder::Ptr convenienceComponentConfig = boost::make_shared(); convenienceComponentConfig->SetType("component"); convenienceComponentConfig->SetName("convenience"); convenienceComponentConfig->SetLocal(true); convenienceComponentConfig->Compile()->Commit(); /* load config file */ ConfigItemBuilder::Ptr fileComponentConfig = boost::make_shared(); fileComponentConfig->SetType("component"); fileComponentConfig->SetName("configfile"); fileComponentConfig->SetLocal(true); fileComponentConfig->AddExpression("configFilename", OperatorSet, configFile); fileComponentConfig->Compile()->Commit(); ConfigObject::Ptr icingaConfig = ConfigObject::GetObject("application", "icinga"); if (!icingaConfig) throw runtime_error("Configuration must contain an 'application' object named 'icinga'."); if (!icingaConfig->IsLocal()) throw runtime_error("'icinga' application object must be 'local'."); icingaConfig->GetProperty("cert", &m_CertificateFile); icingaConfig->GetProperty("ca", &m_CAFile); icingaConfig->GetProperty("node", &m_Node); icingaConfig->GetProperty("service", &m_Service); if (!GetCertificateFile().empty() && !GetCAFile().empty()) { /* set up SSL context */ shared_ptr cert = Utility::GetX509Certificate(GetCertificateFile()); string identity = Utility::GetCertificateCN(cert); Logger::Write(LogInformation, "icinga", "My identity: " + identity); EndpointManager::GetInstance()->SetIdentity(identity); shared_ptr sslContext = Utility::MakeSSLContext(GetCertificateFile(), GetCertificateFile(), GetCAFile()); EndpointManager::GetInstance()->SetSSLContext(sslContext); } /* create the primary RPC listener */ string service = GetService(); if (!service.empty()) EndpointManager::GetInstance()->AddListener(service); if (daemonize) { Logger::Write(LogInformation, "icinga", "Daemonizing."); Utility::Daemonize(); Logger::UnregisterLogger(consoleLogger); } RunEventLoop(); return EXIT_SUCCESS; } void IcingaApplication::NewComponentHandler(const ConfigObject::Ptr& object) { /* don't allow replicated config objects */ if (!object->IsLocal()) throw runtime_error("'component' objects must be 'local'"); string path; if (!object->GetProperty("path", &path)) { #ifdef _WIN32 path = object->GetName() + ".dll"; #else /* _WIN32 */ path = object->GetName() + ".la"; #endif /* _WIN32 */ } LoadComponent(path, object); } IcingaApplication::Ptr IcingaApplication::GetInstance(void) { return static_pointer_cast(Application::GetInstance()); } void IcingaApplication::DeletedComponentHandler(const ConfigObject::Ptr& object) { Component::Ptr component = GetComponent(object->GetName()); UnregisterComponent(component); } string IcingaApplication::GetCertificateFile(void) const { return m_CertificateFile; } string IcingaApplication::GetCAFile(void) const { return m_CAFile; } string IcingaApplication::GetNode(void) const { return m_Node; } string IcingaApplication::GetService(void) const { return m_Service; } time_t IcingaApplication::GetStartTime(void) const { return m_StartTime; }