mirror of https://github.com/Icinga/icinga2.git
306 lines
8.3 KiB
C++
306 lines
8.3 KiB
C++
/* Icinga 2 | (c) 2012 Icinga GmbH | GPLv2+ */
|
|
|
|
#include "remote/configobjectutility.hpp"
|
|
#include "remote/configpackageutility.hpp"
|
|
#include "remote/apilistener.hpp"
|
|
#include "config/configcompiler.hpp"
|
|
#include "config/configitem.hpp"
|
|
#include "base/configwriter.hpp"
|
|
#include "base/exception.hpp"
|
|
#include "base/dependencygraph.hpp"
|
|
#include "base/utility.hpp"
|
|
#include <boost/algorithm/string/case_conv.hpp>
|
|
#include <boost/filesystem.hpp>
|
|
#include <boost/system/error_code.hpp>
|
|
#include <fstream>
|
|
|
|
using namespace icinga;
|
|
|
|
String ConfigObjectUtility::GetConfigDir()
|
|
{
|
|
String prefix = ConfigPackageUtility::GetPackageDir() + "/_api/";
|
|
String activeStage = ConfigPackageUtility::GetActiveStage("_api");
|
|
|
|
if (activeStage.IsEmpty())
|
|
RepairPackage("_api");
|
|
|
|
return prefix + activeStage;
|
|
}
|
|
|
|
String ConfigObjectUtility::GetObjectConfigPath(const Type::Ptr& type, const String& fullName)
|
|
{
|
|
String typeDir = type->GetPluralName();
|
|
boost::algorithm::to_lower(typeDir);
|
|
|
|
/* This may throw an exception the caller above must handle. */
|
|
String prefix = GetConfigDir();
|
|
|
|
return prefix + "/conf.d/" + typeDir +
|
|
"/" + EscapeName(fullName) + ".conf";
|
|
}
|
|
|
|
void ConfigObjectUtility::RepairPackage(const String& package)
|
|
{
|
|
/* Try to fix the active stage, whenever we find a directory in there.
|
|
* This automatically heals packages < 2.11 which remained broken.
|
|
*/
|
|
namespace fs = boost::filesystem;
|
|
|
|
fs::path path(ConfigPackageUtility::GetPackageDir() + "/" + package + "/");
|
|
|
|
fs::recursive_directory_iterator end;
|
|
|
|
String foundActiveStage;
|
|
|
|
for (fs::recursive_directory_iterator it(path); it != end; it++) {
|
|
boost::system::error_code ec;
|
|
|
|
const fs::path d = *it;
|
|
if (fs::is_directory(d, ec)) {
|
|
/* Extract the relative directory name. */
|
|
foundActiveStage = d.stem().string();
|
|
|
|
break; // Use the first found directory.
|
|
}
|
|
}
|
|
|
|
if (!foundActiveStage.IsEmpty()) {
|
|
Log(LogInformation, "ConfigObjectUtility")
|
|
<< "Repairing config package '" << package << "' with stage '" << foundActiveStage << "'.";
|
|
|
|
ConfigPackageUtility::ActivateStage(package, foundActiveStage);
|
|
} else {
|
|
BOOST_THROW_EXCEPTION(std::invalid_argument("Cannot repair package '" + package + "', please check the troubleshooting docs."));
|
|
}
|
|
}
|
|
|
|
void ConfigObjectUtility::CreateStorage()
|
|
{
|
|
boost::mutex::scoped_lock lock(ConfigPackageUtility::GetStaticPackageMutex());
|
|
|
|
/* For now, we only use _api as our creation target. */
|
|
String package = "_api";
|
|
|
|
if (!ConfigPackageUtility::PackageExists(package)) {
|
|
Log(LogNotice, "ConfigObjectUtility")
|
|
<< "Package " << package << " doesn't exist yet, creating it.";
|
|
|
|
ConfigPackageUtility::CreatePackage(package);
|
|
|
|
String stage = ConfigPackageUtility::CreateStage(package);
|
|
ConfigPackageUtility::ActivateStage(package, stage);
|
|
}
|
|
}
|
|
|
|
String ConfigObjectUtility::EscapeName(const String& name)
|
|
{
|
|
return Utility::EscapeString(name, "<>:\"/\\|?*", true);
|
|
}
|
|
|
|
String ConfigObjectUtility::CreateObjectConfig(const Type::Ptr& type, const String& fullName,
|
|
bool ignoreOnError, const Array::Ptr& templates, const Dictionary::Ptr& attrs)
|
|
{
|
|
auto *nc = dynamic_cast<NameComposer *>(type.get());
|
|
Dictionary::Ptr nameParts;
|
|
String name;
|
|
|
|
if (nc) {
|
|
nameParts = nc->ParseName(fullName);
|
|
name = nameParts->Get("name");
|
|
} else
|
|
name = fullName;
|
|
|
|
Dictionary::Ptr allAttrs = new Dictionary();
|
|
|
|
if (attrs) {
|
|
attrs->CopyTo(allAttrs);
|
|
|
|
ObjectLock olock(attrs);
|
|
for (const Dictionary::Pair& kv : attrs) {
|
|
int fid = type->GetFieldId(kv.first.SubStr(0, kv.first.FindFirstOf(".")));
|
|
|
|
if (fid < 0)
|
|
BOOST_THROW_EXCEPTION(ScriptError("Invalid attribute specified: " + kv.first));
|
|
|
|
Field field = type->GetFieldInfo(fid);
|
|
|
|
if (!(field.Attributes & FAConfig) || kv.first == "name")
|
|
BOOST_THROW_EXCEPTION(ScriptError("Attribute is marked for internal use only and may not be set: " + kv.first));
|
|
}
|
|
}
|
|
|
|
if (nameParts)
|
|
nameParts->CopyTo(allAttrs);
|
|
|
|
allAttrs->Remove("name");
|
|
|
|
/* update the version for config sync */
|
|
allAttrs->Set("version", Utility::GetTime());
|
|
|
|
std::ostringstream config;
|
|
ConfigWriter::EmitConfigItem(config, type->GetName(), name, false, ignoreOnError, templates, allAttrs);
|
|
ConfigWriter::EmitRaw(config, "\n");
|
|
|
|
return config.str();
|
|
}
|
|
|
|
bool ConfigObjectUtility::CreateObject(const Type::Ptr& type, const String& fullName,
|
|
const String& config, const Array::Ptr& errors, const Array::Ptr& diagnosticInformation)
|
|
{
|
|
CreateStorage();
|
|
|
|
ConfigItem::Ptr item = ConfigItem::GetByTypeAndName(type, fullName);
|
|
|
|
if (item) {
|
|
errors->Add("Object '" + fullName + "' already exists.");
|
|
return false;
|
|
}
|
|
|
|
String path;
|
|
|
|
try {
|
|
path = GetObjectConfigPath(type, fullName);
|
|
} catch (const std::exception& ex) {
|
|
errors->Add("Config package broken: " + DiagnosticInformation(ex, false));
|
|
return false;
|
|
}
|
|
|
|
Utility::MkDirP(Utility::DirName(path), 0700);
|
|
|
|
std::ofstream fp(path.CStr(), std::ofstream::out | std::ostream::trunc);
|
|
fp << config;
|
|
fp.close();
|
|
|
|
std::unique_ptr<Expression> expr = ConfigCompiler::CompileFile(path, String(), "_api");
|
|
|
|
try {
|
|
ActivationScope ascope;
|
|
|
|
ScriptFrame frame(true);
|
|
expr->Evaluate(frame);
|
|
expr.reset();
|
|
|
|
WorkQueue upq;
|
|
upq.SetName("ConfigObjectUtility::CreateObject");
|
|
|
|
std::vector<ConfigItem::Ptr> newItems;
|
|
|
|
/* Disable logging for object creation, but do so ourselves later on. */
|
|
if (!ConfigItem::CommitItems(ascope.GetContext(), upq, newItems, true) || !ConfigItem::ActivateItems(upq, newItems, true, true)) {
|
|
if (errors) {
|
|
Utility::Remove(path);
|
|
|
|
for (const boost::exception_ptr& ex : upq.GetExceptions()) {
|
|
errors->Add(DiagnosticInformation(ex, false));
|
|
|
|
if (diagnosticInformation)
|
|
diagnosticInformation->Add(DiagnosticInformation(ex));
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/* if (type != Comment::TypeInstance && type != Downtime::TypeInstance)
|
|
* Does not work since this would require libicinga, which has a dependency on libremote
|
|
* Would work if these libs were static.
|
|
*/
|
|
if (type->GetName() != "Comment" && type->GetName() != "Downtime")
|
|
ApiListener::UpdateObjectAuthority();
|
|
|
|
|
|
Log(LogInformation, "ConfigObjectUtility")
|
|
<< "Created and activated object '" << fullName << "' of type '" << type->GetName() << "'.";
|
|
|
|
} catch (const std::exception& ex) {
|
|
Utility::Remove(path);
|
|
|
|
if (errors)
|
|
errors->Add(DiagnosticInformation(ex, false));
|
|
|
|
if (diagnosticInformation)
|
|
diagnosticInformation->Add(DiagnosticInformation(ex));
|
|
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool ConfigObjectUtility::DeleteObjectHelper(const ConfigObject::Ptr& object, bool cascade,
|
|
const Array::Ptr& errors, const Array::Ptr& diagnosticInformation)
|
|
{
|
|
std::vector<Object::Ptr> parents = DependencyGraph::GetParents(object);
|
|
|
|
Type::Ptr type = object->GetReflectionType();
|
|
|
|
String name = object->GetName();
|
|
|
|
if (!parents.empty() && !cascade) {
|
|
if (errors) {
|
|
errors->Add("Object '" + name + "' of type '" + type->GetName() +
|
|
"' cannot be deleted because other objects depend on it. "
|
|
"Use cascading delete to delete it anyway.");
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
for (const Object::Ptr& pobj : parents) {
|
|
ConfigObject::Ptr parentObj = dynamic_pointer_cast<ConfigObject>(pobj);
|
|
|
|
if (!parentObj)
|
|
continue;
|
|
|
|
DeleteObjectHelper(parentObj, cascade, errors, diagnosticInformation);
|
|
}
|
|
|
|
ConfigItem::Ptr item = ConfigItem::GetByTypeAndName(type, name);
|
|
|
|
try {
|
|
/* mark this object for cluster delete event */
|
|
object->SetExtension("ConfigObjectDeleted", true);
|
|
/* triggers signal for DB IDO and other interfaces */
|
|
object->Deactivate(true);
|
|
|
|
if (item)
|
|
item->Unregister();
|
|
else
|
|
object->Unregister();
|
|
|
|
} catch (const std::exception& ex) {
|
|
if (errors)
|
|
errors->Add(DiagnosticInformation(ex, false));
|
|
|
|
if (diagnosticInformation)
|
|
diagnosticInformation->Add(DiagnosticInformation(ex));
|
|
|
|
return false;
|
|
}
|
|
|
|
String path;
|
|
|
|
try {
|
|
path = GetObjectConfigPath(object->GetReflectionType(), name);
|
|
} catch (const std::exception& ex) {
|
|
errors->Add("Config package broken: " + DiagnosticInformation(ex, false));
|
|
return false;
|
|
}
|
|
|
|
Utility::Remove(path);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool ConfigObjectUtility::DeleteObject(const ConfigObject::Ptr& object, bool cascade, const Array::Ptr& errors, const Array::Ptr& diagnosticInformation)
|
|
{
|
|
if (object->GetPackage() != "_api") {
|
|
if (errors)
|
|
errors->Add("Object cannot be deleted because it was not created using the API.");
|
|
|
|
return false;
|
|
}
|
|
|
|
return DeleteObjectHelper(object, cascade, errors, diagnosticInformation);
|
|
}
|