mirror of https://github.com/Icinga/icinga2.git
160 lines
5.4 KiB
C++
160 lines
5.4 KiB
C++
/******************************************************************************
|
|
* Icinga 2 *
|
|
* Copyright (C) 2012-2015 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 "cli/daemonutility.hpp"
|
|
#include "base/utility.hpp"
|
|
#include "base/logger.hpp"
|
|
#include "base/application.hpp"
|
|
#include "config/configcompiler.hpp"
|
|
#include "config/configcompilercontext.hpp"
|
|
#include "config/configitembuilder.hpp"
|
|
|
|
|
|
using namespace icinga;
|
|
|
|
static bool ExecuteExpression(Expression *expression)
|
|
{
|
|
if (!expression)
|
|
return false;
|
|
|
|
try {
|
|
ScriptFrame frame;
|
|
expression->Evaluate(frame);
|
|
} catch (const std::exception& ex) {
|
|
Log(LogCritical, "config", DiagnosticInformation(ex));
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static void IncludeZoneDirRecursive(const String& path, const String& package, bool& success)
|
|
{
|
|
String zoneName = Utility::BaseName(path);
|
|
|
|
/* register this zone path for cluster config sync */
|
|
ConfigCompiler::RegisterZoneDir("_etc", path, zoneName);
|
|
|
|
std::vector<Expression *> expressions;
|
|
Utility::GlobRecursive(path, "*.conf", boost::bind(&ConfigCompiler::CollectIncludes, boost::ref(expressions), _1, zoneName, package), GlobFile);
|
|
DictExpression expr(expressions);
|
|
if (!ExecuteExpression(&expr))
|
|
success = false;
|
|
}
|
|
|
|
static void IncludeNonLocalZone(const String& zonePath, const String& package, bool& success)
|
|
{
|
|
String etcPath = Application::GetZonesDir() + "/" + Utility::BaseName(zonePath);
|
|
|
|
if (Utility::PathExists(etcPath) || Utility::PathExists(zonePath + "/.authoritative"))
|
|
return;
|
|
|
|
IncludeZoneDirRecursive(zonePath, package, success);
|
|
}
|
|
|
|
static void IncludePackage(const String& packagePath, bool& success)
|
|
{
|
|
String packageName = Utility::BaseName(packagePath);
|
|
|
|
if (Utility::PathExists(packagePath + "/include.conf")) {
|
|
Expression *expr = ConfigCompiler::CompileFile(packagePath + "/include.conf",
|
|
String(), packageName);
|
|
|
|
if (!ExecuteExpression(expr))
|
|
success = false;
|
|
|
|
delete expr;
|
|
}
|
|
}
|
|
|
|
bool DaemonUtility::ValidateConfigFiles(const std::vector<std::string>& configs, const String& objectsFile)
|
|
{
|
|
bool success;
|
|
if (!objectsFile.IsEmpty())
|
|
ConfigCompilerContext::GetInstance()->OpenObjectsFile(objectsFile);
|
|
|
|
if (!configs.empty()) {
|
|
BOOST_FOREACH(const String& configPath, configs) {
|
|
Expression *expression = ConfigCompiler::CompileFile(configPath, String(), "_etc");
|
|
success = ExecuteExpression(expression);
|
|
delete expression;
|
|
if (!success)
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/* Load cluster config files - this should probably be in libremote but
|
|
* unfortunately moving it there is somewhat non-trivial. */
|
|
success = true;
|
|
|
|
String zonesEtcDir = Application::GetZonesDir();
|
|
if (!zonesEtcDir.IsEmpty() && Utility::PathExists(zonesEtcDir))
|
|
Utility::Glob(zonesEtcDir + "/*", boost::bind(&IncludeZoneDirRecursive, _1, "_etc", boost::ref(success)), GlobDirectory);
|
|
|
|
if (!success)
|
|
return false;
|
|
|
|
String zonesVarDir = Application::GetLocalStateDir() + "/lib/icinga2/api/zones";
|
|
if (Utility::PathExists(zonesVarDir))
|
|
Utility::Glob(zonesVarDir + "/*", boost::bind(&IncludeNonLocalZone, _1, "_cluster", boost::ref(success)), GlobDirectory);
|
|
|
|
if (!success)
|
|
return false;
|
|
|
|
String packagesVarDir = Application::GetLocalStateDir() + "/lib/icinga2/api/packages";
|
|
if (Utility::PathExists(packagesVarDir))
|
|
Utility::Glob(packagesVarDir + "/*", boost::bind(&IncludePackage, _1, boost::ref(success)), GlobDirectory);
|
|
|
|
if (!success)
|
|
return false;
|
|
|
|
String appType = ScriptGlobal::Get("ApplicationType", &Empty);
|
|
|
|
if (ConfigItem::GetItems(appType).empty()) {
|
|
ConfigItemBuilder::Ptr builder = new ConfigItemBuilder();
|
|
builder->SetType(appType);
|
|
builder->SetName("app");
|
|
ConfigItem::Ptr item = builder->Compile();
|
|
item->Register();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool DaemonUtility::LoadConfigFiles(const std::vector<std::string>& configs,
|
|
std::vector<ConfigItem::Ptr>& newItems,
|
|
const String& objectsFile, const String& varsfile)
|
|
{
|
|
ActivationScope ascope;
|
|
|
|
if (!DaemonUtility::ValidateConfigFiles(configs, objectsFile))
|
|
return false;
|
|
|
|
WorkQueue upq(25000, Application::GetConcurrency());
|
|
bool result = ConfigItem::CommitItems(ascope.GetContext(), upq, newItems);
|
|
|
|
if (!result)
|
|
return false;
|
|
|
|
ConfigCompilerContext::GetInstance()->FinishObjectsFile();
|
|
ScriptGlobal::WriteToFile(varsfile);
|
|
|
|
return true;
|
|
}
|