2019-02-25 14:48:22 +01:00
|
|
|
/* Icinga 2 | (c) 2012 Icinga GmbH | GPLv2+ */
|
2014-05-13 13:18:27 +02:00
|
|
|
|
2014-05-25 16:23:35 +02:00
|
|
|
#include "remote/apilistener.hpp"
|
|
|
|
#include "remote/apifunction.hpp"
|
2015-07-21 09:32:17 +02:00
|
|
|
#include "config/configcompiler.hpp"
|
2015-08-15 20:28:05 +02:00
|
|
|
#include "base/configtype.hpp"
|
2014-10-19 14:21:12 +02:00
|
|
|
#include "base/logger.hpp"
|
2014-05-25 16:23:35 +02:00
|
|
|
#include "base/convert.hpp"
|
2014-12-15 10:16:06 +01:00
|
|
|
#include "base/exception.hpp"
|
2014-05-13 13:18:27 +02:00
|
|
|
#include <fstream>
|
2015-02-09 15:22:55 +01:00
|
|
|
#include <iomanip>
|
2014-05-13 13:18:27 +02:00
|
|
|
|
|
|
|
using namespace icinga;
|
|
|
|
|
2014-05-13 15:57:02 +02:00
|
|
|
REGISTER_APIFUNCTION(Update, config, &ApiListener::ConfigUpdateHandler);
|
|
|
|
|
2016-01-26 10:46:27 +01:00
|
|
|
void ApiListener::ConfigGlobHandler(ConfigDirInformation& config, const String& path, const String& file)
|
2014-05-13 13:18:27 +02:00
|
|
|
{
|
|
|
|
CONTEXT("Creating config update for file '" + file + "'");
|
|
|
|
|
2014-10-20 10:09:57 +02:00
|
|
|
Log(LogNotice, "ApiListener")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "Creating config update for file '" << file << "'.";
|
2014-06-12 22:51:48 +02:00
|
|
|
|
2015-02-09 07:44:45 +01:00
|
|
|
std::ifstream fp(file.CStr(), std::ifstream::binary);
|
2014-05-13 13:18:27 +02:00
|
|
|
if (!fp)
|
|
|
|
return;
|
|
|
|
|
|
|
|
String content((std::istreambuf_iterator<char>(fp)), std::istreambuf_iterator<char>());
|
2016-01-26 10:46:27 +01:00
|
|
|
|
|
|
|
Dictionary::Ptr update;
|
|
|
|
|
|
|
|
if (Utility::Match("*.conf", file))
|
|
|
|
update = config.UpdateV1;
|
|
|
|
else
|
|
|
|
update = config.UpdateV2;
|
|
|
|
|
|
|
|
update->Set(file.SubStr(path.GetLength()), content);
|
|
|
|
}
|
|
|
|
|
|
|
|
Dictionary::Ptr ApiListener::MergeConfigUpdate(const ConfigDirInformation& config)
|
|
|
|
{
|
|
|
|
Dictionary::Ptr result = new Dictionary();
|
|
|
|
|
|
|
|
if (config.UpdateV1)
|
|
|
|
config.UpdateV1->CopyTo(result);
|
|
|
|
|
|
|
|
if (config.UpdateV2)
|
|
|
|
config.UpdateV2->CopyTo(result);
|
|
|
|
|
|
|
|
return result;
|
2014-05-13 13:18:27 +02:00
|
|
|
}
|
|
|
|
|
2016-01-26 10:46:27 +01:00
|
|
|
ConfigDirInformation ApiListener::LoadConfigDir(const String& dir)
|
2014-05-13 13:18:27 +02:00
|
|
|
{
|
2016-01-26 10:46:27 +01:00
|
|
|
ConfigDirInformation config;
|
|
|
|
config.UpdateV1 = new Dictionary();
|
|
|
|
config.UpdateV2 = new Dictionary();
|
2017-11-23 06:51:48 +01:00
|
|
|
Utility::GlobRecursive(dir, "*", std::bind(&ApiListener::ConfigGlobHandler, std::ref(config), dir, _1), GlobFile);
|
2014-05-13 15:57:02 +02:00
|
|
|
return config;
|
|
|
|
}
|
2014-05-13 13:18:27 +02:00
|
|
|
|
2016-01-26 10:46:27 +01:00
|
|
|
bool ApiListener::UpdateConfigDir(const ConfigDirInformation& oldConfigInfo, const ConfigDirInformation& newConfigInfo, const String& configDir, bool authoritative)
|
2014-05-13 15:57:02 +02:00
|
|
|
{
|
|
|
|
bool configChange = false;
|
2014-05-13 13:18:27 +02:00
|
|
|
|
2016-01-26 10:46:27 +01:00
|
|
|
Dictionary::Ptr oldConfig = MergeConfigUpdate(oldConfigInfo);
|
|
|
|
Dictionary::Ptr newConfig = MergeConfigUpdate(newConfigInfo);
|
|
|
|
|
2016-01-25 15:21:25 +01:00
|
|
|
double oldTimestamp;
|
|
|
|
|
2017-04-28 16:50:36 +02:00
|
|
|
if (!oldConfig->Contains("/.timestamp"))
|
2016-01-25 15:21:25 +01:00
|
|
|
oldTimestamp = 0;
|
|
|
|
else
|
2017-04-28 16:50:36 +02:00
|
|
|
oldTimestamp = oldConfig->Get("/.timestamp");
|
2016-01-25 15:21:25 +01:00
|
|
|
|
|
|
|
double newTimestamp;
|
2014-05-15 14:27:50 +02:00
|
|
|
|
2017-04-28 16:50:36 +02:00
|
|
|
if (!newConfig->Contains("/.timestamp"))
|
2016-01-25 15:21:25 +01:00
|
|
|
newTimestamp = Utility::GetTime();
|
|
|
|
else
|
2017-04-28 16:50:36 +02:00
|
|
|
newTimestamp = newConfig->Get("/.timestamp");
|
2016-01-25 14:25:37 +01:00
|
|
|
|
2017-05-04 15:20:32 +02:00
|
|
|
/* skip update if our configuration files are more recent */
|
2017-04-28 16:51:23 +02:00
|
|
|
if (oldTimestamp >= newTimestamp) {
|
2017-09-20 13:30:02 +02:00
|
|
|
Log(LogNotice, "ApiListener")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "Our configuration is more recent than the received configuration update."
|
|
|
|
<< " Ignoring configuration file update for path '" << configDir << "'. Current timestamp '"
|
|
|
|
<< Utility::FormatDateTime("%Y-%m-%d %H:%M:%S %z", oldTimestamp) << "' ("
|
|
|
|
<< std::fixed << std::setprecision(6) << oldTimestamp
|
|
|
|
<< ") >= received timestamp '"
|
|
|
|
<< Utility::FormatDateTime("%Y-%m-%d %H:%M:%S %z", newTimestamp) << "' ("
|
|
|
|
<< newTimestamp << ").";
|
2016-01-25 14:25:37 +01:00
|
|
|
return false;
|
2017-04-28 16:51:23 +02:00
|
|
|
}
|
|
|
|
|
2017-08-21 11:08:13 +02:00
|
|
|
size_t numBytes = 0;
|
2014-05-13 13:18:27 +02:00
|
|
|
|
2015-02-24 16:03:32 +01:00
|
|
|
{
|
|
|
|
ObjectLock olock(newConfig);
|
2016-08-25 06:19:44 +02:00
|
|
|
for (const Dictionary::Pair& kv : newConfig) {
|
2015-02-24 16:03:32 +01:00
|
|
|
if (oldConfig->Get(kv.first) != kv.second) {
|
2016-01-26 15:42:54 +01:00
|
|
|
if (!Utility::Match("*/.timestamp", kv.first))
|
|
|
|
configChange = true;
|
2015-02-24 16:03:32 +01:00
|
|
|
|
|
|
|
String path = configDir + "/" + kv.first;
|
|
|
|
Log(LogInformation, "ApiListener")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "Updating configuration file: " << path;
|
2015-02-24 16:03:32 +01:00
|
|
|
|
2017-08-21 11:08:13 +02:00
|
|
|
/* Sync string content only. */
|
|
|
|
String content = kv.second;
|
|
|
|
|
|
|
|
/* Generate a directory tree (zones/1/2/3 might not exist yet). */
|
2015-02-24 16:03:32 +01:00
|
|
|
Utility::MkDirP(Utility::DirName(path), 0755);
|
|
|
|
std::ofstream fp(path.CStr(), std::ofstream::out | std::ostream::binary | std::ostream::trunc);
|
2017-08-21 11:08:13 +02:00
|
|
|
fp << content;
|
2015-02-24 16:03:32 +01:00
|
|
|
fp.close();
|
2017-08-21 11:08:13 +02:00
|
|
|
|
|
|
|
numBytes += content.GetLength();
|
2015-02-24 16:03:32 +01:00
|
|
|
}
|
2014-05-13 13:18:27 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-21 11:08:13 +02:00
|
|
|
Log(LogInformation, "ApiListener")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "Applying configuration file update for path '" << configDir << "' (" << numBytes << " Bytes). Received timestamp '"
|
|
|
|
<< Utility::FormatDateTime("%Y-%m-%d %H:%M:%S %z", newTimestamp) << "' ("
|
|
|
|
<< std::fixed << std::setprecision(6) << newTimestamp
|
|
|
|
<< "), Current timestamp '"
|
|
|
|
<< Utility::FormatDateTime("%Y-%m-%d %H:%M:%S %z", oldTimestamp) << "' ("
|
|
|
|
<< oldTimestamp << ").";
|
2017-08-21 11:08:13 +02:00
|
|
|
|
2015-02-24 16:03:32 +01:00
|
|
|
ObjectLock xlock(oldConfig);
|
2016-08-25 06:19:44 +02:00
|
|
|
for (const Dictionary::Pair& kv : oldConfig) {
|
2014-05-13 15:57:02 +02:00
|
|
|
if (!newConfig->Contains(kv.first)) {
|
|
|
|
configChange = true;
|
|
|
|
|
|
|
|
String path = configDir + "/" + kv.first;
|
2014-05-13 13:18:27 +02:00
|
|
|
(void) unlink(path.CStr());
|
|
|
|
}
|
|
|
|
}
|
2014-05-13 15:57:02 +02:00
|
|
|
|
2014-05-15 14:27:50 +02:00
|
|
|
String tsPath = configDir + "/.timestamp";
|
|
|
|
if (!Utility::PathExists(tsPath)) {
|
|
|
|
std::ofstream fp(tsPath.CStr(), std::ofstream::out | std::ostream::trunc);
|
2016-01-25 14:25:37 +01:00
|
|
|
fp << std::fixed << newTimestamp;
|
2014-05-15 14:27:50 +02:00
|
|
|
fp.close();
|
|
|
|
}
|
|
|
|
|
2015-01-20 13:18:40 +01:00
|
|
|
if (authoritative) {
|
|
|
|
String authPath = configDir + "/.authoritative";
|
2015-02-07 23:41:43 +01:00
|
|
|
if (!Utility::PathExists(authPath)) {
|
|
|
|
std::ofstream fp(authPath.CStr(), std::ofstream::out | std::ostream::trunc);
|
2015-01-20 13:18:40 +01:00
|
|
|
fp.close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-13 15:57:02 +02:00
|
|
|
return configChange;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ApiListener::SyncZoneDir(const Zone::Ptr& zone) const
|
|
|
|
{
|
2016-01-26 10:46:27 +01:00
|
|
|
ConfigDirInformation newConfigInfo;
|
|
|
|
newConfigInfo.UpdateV1 = new Dictionary();
|
|
|
|
newConfigInfo.UpdateV2 = new Dictionary();
|
|
|
|
|
2016-08-25 06:19:44 +02:00
|
|
|
for (const ZoneFragment& zf : ConfigCompiler::GetZoneDirs(zone->GetName())) {
|
2016-01-26 10:46:27 +01:00
|
|
|
ConfigDirInformation newConfigPart = LoadConfigDir(zf.Path);
|
2015-07-21 09:32:17 +02:00
|
|
|
|
2016-01-26 10:46:27 +01:00
|
|
|
{
|
|
|
|
ObjectLock olock(newConfigPart.UpdateV1);
|
2016-08-25 06:19:44 +02:00
|
|
|
for (const Dictionary::Pair& kv : newConfigPart.UpdateV1) {
|
2016-01-26 10:46:27 +01:00
|
|
|
newConfigInfo.UpdateV1->Set("/" + zf.Tag + kv.first, kv.second);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
ObjectLock olock(newConfigPart.UpdateV2);
|
2016-08-25 06:19:44 +02:00
|
|
|
for (const Dictionary::Pair& kv : newConfigPart.UpdateV2) {
|
2016-01-26 10:46:27 +01:00
|
|
|
newConfigInfo.UpdateV2->Set("/" + zf.Tag + kv.first, kv.second);
|
|
|
|
}
|
2015-07-21 09:32:17 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-01-26 10:46:27 +01:00
|
|
|
int sumUpdates = newConfigInfo.UpdateV1->GetLength() + newConfigInfo.UpdateV2->GetLength();
|
|
|
|
|
|
|
|
if (sumUpdates == 0)
|
2015-12-11 19:54:17 +01:00
|
|
|
return;
|
|
|
|
|
2018-08-09 15:37:23 +02:00
|
|
|
String oldDir = Configuration::DataDir + "/api/zones/" + zone->GetName();
|
2014-05-13 15:57:02 +02:00
|
|
|
|
2014-10-20 10:09:57 +02:00
|
|
|
Log(LogInformation, "ApiListener")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "Copying " << sumUpdates << " zone configuration files for zone '" << zone->GetName() << "' to '" << oldDir << "'.";
|
2014-08-20 14:07:23 +02:00
|
|
|
|
2016-01-26 10:45:03 +01:00
|
|
|
Utility::MkDirP(oldDir, 0700);
|
2014-05-13 15:57:02 +02:00
|
|
|
|
2016-01-26 10:46:27 +01:00
|
|
|
ConfigDirInformation oldConfigInfo = LoadConfigDir(oldDir);
|
2014-05-13 15:57:02 +02:00
|
|
|
|
2016-01-26 10:46:27 +01:00
|
|
|
UpdateConfigDir(oldConfigInfo, newConfigInfo, oldDir, true);
|
2014-05-13 13:18:27 +02:00
|
|
|
}
|
|
|
|
|
2018-01-04 04:25:35 +01:00
|
|
|
void ApiListener::SyncZoneDirs() const
|
2014-05-13 13:18:27 +02:00
|
|
|
{
|
2016-08-25 06:19:44 +02:00
|
|
|
for (const Zone::Ptr& zone : ConfigType::GetObjectsByType<Zone>()) {
|
2014-06-13 09:23:05 +02:00
|
|
|
try {
|
|
|
|
SyncZoneDir(zone);
|
2014-08-25 08:35:35 +02:00
|
|
|
} catch (const std::exception&) {
|
2014-06-13 09:23:05 +02:00
|
|
|
continue;
|
|
|
|
}
|
2014-05-13 13:18:27 +02:00
|
|
|
}
|
2014-05-13 15:57:02 +02:00
|
|
|
}
|
|
|
|
|
2015-06-22 11:11:21 +02:00
|
|
|
void ApiListener::SendConfigUpdate(const JsonRpcConnection::Ptr& aclient)
|
2014-05-13 15:57:02 +02:00
|
|
|
{
|
|
|
|
Endpoint::Ptr endpoint = aclient->GetEndpoint();
|
|
|
|
ASSERT(endpoint);
|
|
|
|
|
|
|
|
Zone::Ptr azone = endpoint->GetZone();
|
|
|
|
Zone::Ptr lzone = Zone::GetLocalZone();
|
|
|
|
|
|
|
|
/* don't try to send config updates to our master */
|
2014-05-23 18:35:43 +02:00
|
|
|
if (!azone->IsChildOf(lzone))
|
2014-05-13 15:57:02 +02:00
|
|
|
return;
|
|
|
|
|
2016-01-26 10:46:27 +01:00
|
|
|
Dictionary::Ptr configUpdateV1 = new Dictionary();
|
|
|
|
Dictionary::Ptr configUpdateV2 = new Dictionary();
|
2014-05-13 15:57:02 +02:00
|
|
|
|
2018-08-09 15:37:23 +02:00
|
|
|
String zonesDir = Configuration::DataDir + "/api/zones";
|
2014-05-13 15:57:02 +02:00
|
|
|
|
2016-08-25 06:19:44 +02:00
|
|
|
for (const Zone::Ptr& zone : ConfigType::GetObjectsByType<Zone>()) {
|
2014-05-13 15:57:02 +02:00
|
|
|
String zoneDir = zonesDir + "/" + zone->GetName();
|
|
|
|
|
2016-01-21 13:02:53 +01:00
|
|
|
if (!zone->IsChildOf(azone) && !zone->IsGlobal())
|
2014-05-13 15:57:02 +02:00
|
|
|
continue;
|
2016-01-21 13:02:53 +01:00
|
|
|
|
|
|
|
if (!Utility::PathExists(zoneDir))
|
2014-06-12 14:31:07 +02:00
|
|
|
continue;
|
|
|
|
|
2015-09-15 16:09:56 +02:00
|
|
|
Log(LogInformation, "ApiListener")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "Syncing configuration files for " << (zone->IsGlobal() ? "global " : "")
|
|
|
|
<< "zone '" << zone->GetName() << "' to endpoint '" << endpoint->GetName() << "'.";
|
2014-05-13 15:57:02 +02:00
|
|
|
|
2016-01-26 10:46:27 +01:00
|
|
|
ConfigDirInformation config = LoadConfigDir(zonesDir + "/" + zone->GetName());
|
|
|
|
configUpdateV1->Set(zone->GetName(), config.UpdateV1);
|
|
|
|
configUpdateV2->Set(zone->GetName(), config.UpdateV2);
|
2014-05-13 15:57:02 +02:00
|
|
|
}
|
|
|
|
|
2018-01-11 11:17:38 +01:00
|
|
|
Dictionary::Ptr message = new Dictionary({
|
|
|
|
{ "jsonrpc", "2.0" },
|
|
|
|
{ "method", "config::Update" },
|
|
|
|
{ "params", new Dictionary({
|
|
|
|
{ "update", configUpdateV1 },
|
|
|
|
{ "update_v2", configUpdateV2 }
|
|
|
|
}) }
|
|
|
|
});
|
2014-05-13 15:57:02 +02:00
|
|
|
|
|
|
|
aclient->SendMessage(message);
|
|
|
|
}
|
|
|
|
|
2015-08-04 14:47:44 +02:00
|
|
|
Value ApiListener::ConfigUpdateHandler(const MessageOrigin::Ptr& origin, const Dictionary::Ptr& params)
|
2014-05-13 15:57:02 +02:00
|
|
|
{
|
2015-08-04 14:47:44 +02:00
|
|
|
if (!origin->FromClient->GetEndpoint() || (origin->FromZone && !Zone::GetLocalZone()->IsChildOf(origin->FromZone)))
|
2014-05-13 15:57:02 +02:00
|
|
|
return Empty;
|
|
|
|
|
2014-05-15 10:13:32 +02:00
|
|
|
ApiListener::Ptr listener = ApiListener::GetInstance();
|
|
|
|
|
2014-06-12 14:31:07 +02:00
|
|
|
if (!listener) {
|
|
|
|
Log(LogCritical, "ApiListener", "No instance available.");
|
2014-05-15 10:13:32 +02:00
|
|
|
return Empty;
|
2014-06-12 14:31:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!listener->GetAcceptConfig()) {
|
2014-10-20 10:09:57 +02:00
|
|
|
Log(LogWarning, "ApiListener")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "Ignoring config update. '" << listener->GetName() << "' does not accept config.";
|
2014-06-12 14:31:07 +02:00
|
|
|
return Empty;
|
|
|
|
}
|
2014-05-15 10:13:32 +02:00
|
|
|
|
2017-04-28 16:51:23 +02:00
|
|
|
Log(LogInformation, "ApiListener")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "Applying config update from endpoint '" << origin->FromClient->GetEndpoint()->GetName()
|
|
|
|
<< "' of zone '" << GetFromZoneName(origin->FromZone) << "'.";
|
2017-04-28 16:51:23 +02:00
|
|
|
|
2016-01-26 10:46:27 +01:00
|
|
|
Dictionary::Ptr updateV1 = params->Get("update");
|
|
|
|
Dictionary::Ptr updateV2 = params->Get("update_v2");
|
2014-05-13 13:18:27 +02:00
|
|
|
|
|
|
|
bool configChange = false;
|
|
|
|
|
2016-01-26 10:46:27 +01:00
|
|
|
ObjectLock olock(updateV1);
|
2016-08-25 06:19:44 +02:00
|
|
|
for (const Dictionary::Pair& kv : updateV1) {
|
2014-05-13 15:57:02 +02:00
|
|
|
Zone::Ptr zone = Zone::GetByName(kv.first);
|
|
|
|
|
|
|
|
if (!zone) {
|
2014-10-20 10:09:57 +02:00
|
|
|
Log(LogWarning, "ApiListener")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "Ignoring config update for unknown zone '" << kv.first << "'.";
|
2015-03-09 08:43:46 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2015-12-11 19:54:17 +01:00
|
|
|
if (ConfigCompiler::HasZoneConfigAuthority(kv.first)) {
|
2015-03-09 08:43:46 +01:00
|
|
|
Log(LogWarning, "ApiListener")
|
2017-12-19 15:50:05 +01:00
|
|
|
<< "Ignoring config update for zone '" << kv.first << "' because we have an authoritative version of the zone's config.";
|
2014-05-13 15:57:02 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-08-09 15:37:23 +02:00
|
|
|
String oldDir = Configuration::DataDir + "/api/zones/" + zone->GetName();
|
2014-05-13 15:57:02 +02:00
|
|
|
|
2016-01-26 10:45:03 +01:00
|
|
|
Utility::MkDirP(oldDir, 0700);
|
2014-05-13 15:57:02 +02:00
|
|
|
|
2016-01-26 10:46:27 +01:00
|
|
|
ConfigDirInformation newConfigInfo;
|
|
|
|
newConfigInfo.UpdateV1 = kv.second;
|
|
|
|
|
|
|
|
if (updateV2)
|
|
|
|
newConfigInfo.UpdateV2 = updateV2->Get(kv.first);
|
|
|
|
|
2014-05-13 15:57:02 +02:00
|
|
|
Dictionary::Ptr newConfig = kv.second;
|
2016-01-26 10:46:27 +01:00
|
|
|
ConfigDirInformation oldConfigInfo = LoadConfigDir(oldDir);
|
2014-05-13 15:57:02 +02:00
|
|
|
|
2016-01-26 10:46:27 +01:00
|
|
|
if (UpdateConfigDir(oldConfigInfo, newConfigInfo, oldDir, false))
|
2014-05-13 15:57:02 +02:00
|
|
|
configChange = true;
|
|
|
|
}
|
2014-05-13 13:18:27 +02:00
|
|
|
|
|
|
|
if (configChange) {
|
2014-05-28 13:58:56 +02:00
|
|
|
Log(LogInformation, "ApiListener", "Restarting after configuration change.");
|
2014-05-13 13:18:27 +02:00
|
|
|
Application::RequestRestart();
|
|
|
|
}
|
2014-05-13 15:57:02 +02:00
|
|
|
|
|
|
|
return Empty;
|
2014-05-13 13:18:27 +02:00
|
|
|
}
|