mirror of https://github.com/Icinga/icinga2.git
454 lines
12 KiB
C++
454 lines
12 KiB
C++
/* Icinga 2 | (c) 2012 Icinga GmbH | GPLv2+ */
|
|
|
|
#include "db_ido/dbobject.hpp"
|
|
#include "db_ido/dbtype.hpp"
|
|
#include "db_ido/dbvalue.hpp"
|
|
#include "icinga/customvarobject.hpp"
|
|
#include "icinga/service.hpp"
|
|
#include "icinga/compatutility.hpp"
|
|
#include "icinga/checkcommand.hpp"
|
|
#include "icinga/eventcommand.hpp"
|
|
#include "icinga/notificationcommand.hpp"
|
|
#include "remote/endpoint.hpp"
|
|
#include "base/configobject.hpp"
|
|
#include "base/configtype.hpp"
|
|
#include "base/json.hpp"
|
|
#include "base/serializer.hpp"
|
|
#include "base/json.hpp"
|
|
#include "base/convert.hpp"
|
|
#include "base/objectlock.hpp"
|
|
#include "base/utility.hpp"
|
|
#include "base/initialize.hpp"
|
|
#include "base/logger.hpp"
|
|
|
|
using namespace icinga;
|
|
|
|
boost::signals2::signal<void (const DbQuery&)> DbObject::OnQuery;
|
|
boost::signals2::signal<void (const std::vector<DbQuery>&)> DbObject::OnMultipleQueries;
|
|
boost::signals2::signal<void (const std::function<void (const DbObject::QueryCallbacks&)>&)> DbObject::OnMakeQueries;
|
|
|
|
INITIALIZE_ONCE(&DbObject::StaticInitialize);
|
|
|
|
DbObject::DbObject(intrusive_ptr<DbType> type, String name1, String name2)
|
|
: m_Name1(std::move(name1)), m_Name2(std::move(name2)), m_Type(std::move(type)), m_LastConfigUpdate(0), m_LastStatusUpdate(0)
|
|
{ }
|
|
|
|
void DbObject::StaticInitialize()
|
|
{
|
|
/* triggered in ProcessCheckResult(), requires UpdateNextCheck() to be called before */
|
|
ConfigObject::OnStateChanged.connect([](const ConfigObject::Ptr& object) { StateChangedHandler(object); });
|
|
CustomVarObject::OnVarsChanged.connect([](const CustomVarObject::Ptr& customVar, const Value&) { VarsChangedHandler(customVar); });
|
|
|
|
/* triggered on create, update and delete objects */
|
|
ConfigObject::OnVersionChanged.connect([](const ConfigObject::Ptr& object, const Value&) { VersionChangedHandler(object); });
|
|
}
|
|
|
|
void DbObject::SetObject(const ConfigObject::Ptr& object)
|
|
{
|
|
m_Object = object;
|
|
}
|
|
|
|
ConfigObject::Ptr DbObject::GetObject() const
|
|
{
|
|
return m_Object;
|
|
}
|
|
|
|
String DbObject::GetName1() const
|
|
{
|
|
return m_Name1;
|
|
}
|
|
|
|
String DbObject::GetName2() const
|
|
{
|
|
return m_Name2;
|
|
}
|
|
|
|
DbType::Ptr DbObject::GetType() const
|
|
{
|
|
return m_Type;
|
|
}
|
|
|
|
String DbObject::CalculateConfigHash(const Dictionary::Ptr& configFields) const
|
|
{
|
|
Dictionary::Ptr configFieldsDup = configFields->ShallowClone();
|
|
|
|
{
|
|
ObjectLock olock(configFieldsDup);
|
|
|
|
for (const Dictionary::Pair& kv : configFieldsDup) {
|
|
if (kv.second.IsObjectType<ConfigObject>()) {
|
|
ConfigObject::Ptr obj = kv.second;
|
|
configFieldsDup->Set(kv.first, obj->GetName());
|
|
}
|
|
}
|
|
}
|
|
|
|
Array::Ptr data = new Array();
|
|
data->Add(configFieldsDup);
|
|
|
|
CustomVarObject::Ptr custom_var_object = dynamic_pointer_cast<CustomVarObject>(GetObject());
|
|
|
|
if (custom_var_object)
|
|
data->Add(custom_var_object->GetVars());
|
|
|
|
return HashValue(data);
|
|
}
|
|
|
|
String DbObject::HashValue(const Value& value)
|
|
{
|
|
Value temp;
|
|
|
|
Type::Ptr type = value.GetReflectionType();
|
|
|
|
if (ConfigObject::TypeInstance->IsAssignableFrom(type))
|
|
temp = Serialize(value, FAConfig);
|
|
else
|
|
temp = value;
|
|
|
|
return SHA256(JsonEncode(temp));
|
|
}
|
|
|
|
void DbObject::SendConfigUpdateHeavy(const Dictionary::Ptr& configFields)
|
|
{
|
|
/* update custom var config and status */
|
|
SendVarsConfigUpdateHeavy();
|
|
|
|
/* config attributes */
|
|
if (!configFields)
|
|
return;
|
|
|
|
ASSERT(configFields->Contains("config_hash"));
|
|
|
|
Value configHash = configFields->Get("config_hash");
|
|
// Since all the child tables are relying on the inserted parent ID, we first need to insert/update the
|
|
// configuration fields of the current object without the actual config_hash value. Having the config hash
|
|
// set only after all relation queries eliminates some rare race conditions where e.g. host group members
|
|
// are not written to the database because Icinga 2 / the DBMS was unexpectedly stopped/reloaded shortly
|
|
// after the config_hash column was updated.
|
|
configFields->Set("config_hash", Empty);
|
|
|
|
ConfigObject::Ptr object = GetObject();
|
|
|
|
DbQuery query;
|
|
query.Table = GetType()->GetTable() + "s";
|
|
query.Type = DbQueryInsert | DbQueryUpdate;
|
|
query.Category = DbCatConfig;
|
|
query.Fields = configFields;
|
|
query.Fields->Set(GetType()->GetIDColumn(), object);
|
|
query.Fields->Set("instance_id", 0); /* DbConnection class fills in real ID */
|
|
query.Fields->Set("config_type", 1);
|
|
query.WhereCriteria = new Dictionary({
|
|
{ GetType()->GetIDColumn(), object }
|
|
});
|
|
query.Object = this;
|
|
query.ConfigUpdate = true;
|
|
OnQuery(query);
|
|
|
|
m_LastConfigUpdate = Utility::GetTime();
|
|
|
|
// Trigger config heavy udpates of the child classes.
|
|
OnConfigUpdateHeavy();
|
|
|
|
// Now update the config hash attribute of the current object.
|
|
DbQuery configHashQuery;
|
|
configHashQuery.Table = GetType()->GetTable() + "s";
|
|
configHashQuery.Type = DbQueryUpdate;
|
|
configHashQuery.Category = DbCatConfig;
|
|
configHashQuery.Fields = new Dictionary({{"config_hash", configHash}});
|
|
configHashQuery.Object = this;
|
|
configHashQuery.ConfigUpdate = true;
|
|
configHashQuery.WhereCriteria = new Dictionary({{GetType()->GetIDColumn(), object}});
|
|
OnQuery(configHashQuery);
|
|
|
|
// Lastly, update some common configs that do not affect the config_hash column.
|
|
OnConfigUpdateLight();
|
|
}
|
|
|
|
void DbObject::SendConfigUpdateLight()
|
|
{
|
|
OnConfigUpdateLight();
|
|
}
|
|
|
|
void DbObject::SendStatusUpdate()
|
|
{
|
|
/* status attributes */
|
|
Dictionary::Ptr fields = GetStatusFields();
|
|
|
|
if (!fields)
|
|
return;
|
|
|
|
DbQuery query;
|
|
query.Table = GetType()->GetTable() + "status";
|
|
query.Type = DbQueryInsert | DbQueryUpdate;
|
|
query.Category = DbCatState;
|
|
query.Fields = fields;
|
|
query.Fields->Set(GetType()->GetIDColumn(), GetObject());
|
|
|
|
/* do not override endpoint_object_id for endpoints & zones */
|
|
if (query.Table != "endpointstatus" && query.Table != "zonestatus") {
|
|
String node = IcingaApplication::GetInstance()->GetNodeName();
|
|
|
|
Endpoint::Ptr endpoint = Endpoint::GetByName(node);
|
|
if (endpoint)
|
|
query.Fields->Set("endpoint_object_id", endpoint);
|
|
}
|
|
|
|
query.Fields->Set("instance_id", 0); /* DbConnection class fills in real ID */
|
|
|
|
query.Fields->Set("status_update_time", DbValue::FromTimestamp(Utility::GetTime()));
|
|
query.WhereCriteria = new Dictionary({
|
|
{ GetType()->GetIDColumn(), GetObject() }
|
|
});
|
|
query.Object = this;
|
|
query.StatusUpdate = true;
|
|
OnQuery(query);
|
|
|
|
m_LastStatusUpdate = Utility::GetTime();
|
|
|
|
OnStatusUpdate();
|
|
}
|
|
|
|
void DbObject::SendVarsConfigUpdateHeavy()
|
|
{
|
|
ConfigObject::Ptr obj = GetObject();
|
|
|
|
CustomVarObject::Ptr custom_var_object = dynamic_pointer_cast<CustomVarObject>(obj);
|
|
|
|
if (!custom_var_object)
|
|
return;
|
|
|
|
std::vector<DbQuery> queries;
|
|
|
|
DbQuery query1;
|
|
query1.Table = "customvariables";
|
|
query1.Type = DbQueryDelete;
|
|
query1.Category = DbCatConfig;
|
|
query1.WhereCriteria = new Dictionary({
|
|
{ "object_id", obj }
|
|
});
|
|
queries.emplace_back(std::move(query1));
|
|
|
|
DbQuery query2;
|
|
query2.Table = "customvariablestatus";
|
|
query2.Type = DbQueryDelete;
|
|
query2.Category = DbCatConfig;
|
|
query2.WhereCriteria = new Dictionary({
|
|
{ "object_id", obj }
|
|
});
|
|
queries.emplace_back(std::move(query2));
|
|
|
|
Dictionary::Ptr vars = custom_var_object->GetVars();
|
|
|
|
if (vars) {
|
|
ObjectLock olock (vars);
|
|
|
|
for (const Dictionary::Pair& kv : vars) {
|
|
if (kv.first.IsEmpty())
|
|
continue;
|
|
|
|
String value;
|
|
int is_json = 0;
|
|
|
|
if (kv.second.IsObjectType<Array>() || kv.second.IsObjectType<Dictionary>()) {
|
|
value = JsonEncode(kv.second);
|
|
is_json = 1;
|
|
} else
|
|
value = kv.second;
|
|
|
|
DbQuery query3;
|
|
query3.Table = "customvariables";
|
|
query3.Type = DbQueryInsert;
|
|
query3.Category = DbCatConfig;
|
|
query3.Fields = new Dictionary({
|
|
{ "varname", kv.first },
|
|
{ "varvalue", value },
|
|
{ "is_json", is_json },
|
|
{ "config_type", 1 },
|
|
{ "object_id", obj },
|
|
{ "instance_id", 0 } /* DbConnection class fills in real ID */
|
|
});
|
|
queries.emplace_back(std::move(query3));
|
|
|
|
DbQuery query4;
|
|
query4.Table = "customvariablestatus";
|
|
query4.Type = DbQueryInsert;
|
|
query4.Category = DbCatState;
|
|
|
|
query4.Fields = new Dictionary({
|
|
{ "varname", kv.first },
|
|
{ "varvalue", value },
|
|
{ "is_json", is_json },
|
|
{ "status_update_time", DbValue::FromTimestamp(Utility::GetTime()) },
|
|
{ "object_id", obj },
|
|
{ "instance_id", 0 } /* DbConnection class fills in real ID */
|
|
});
|
|
|
|
queries.emplace_back(std::move(query4));
|
|
}
|
|
}
|
|
|
|
OnMultipleQueries(queries);
|
|
}
|
|
|
|
void DbObject::SendVarsStatusUpdate()
|
|
{
|
|
ConfigObject::Ptr obj = GetObject();
|
|
|
|
CustomVarObject::Ptr custom_var_object = dynamic_pointer_cast<CustomVarObject>(obj);
|
|
|
|
if (!custom_var_object)
|
|
return;
|
|
|
|
Dictionary::Ptr vars = custom_var_object->GetVars();
|
|
|
|
if (vars) {
|
|
std::vector<DbQuery> queries;
|
|
ObjectLock olock (vars);
|
|
|
|
for (const Dictionary::Pair& kv : vars) {
|
|
if (kv.first.IsEmpty())
|
|
continue;
|
|
|
|
String value;
|
|
int is_json = 0;
|
|
|
|
if (kv.second.IsObjectType<Array>() || kv.second.IsObjectType<Dictionary>()) {
|
|
value = JsonEncode(kv.second);
|
|
is_json = 1;
|
|
} else
|
|
value = kv.second;
|
|
|
|
DbQuery query;
|
|
query.Table = "customvariablestatus";
|
|
query.Type = DbQueryInsert | DbQueryUpdate;
|
|
query.Category = DbCatState;
|
|
|
|
query.Fields = new Dictionary({
|
|
{ "varname", kv.first },
|
|
{ "varvalue", value },
|
|
{ "is_json", is_json },
|
|
{ "status_update_time", DbValue::FromTimestamp(Utility::GetTime()) },
|
|
{ "object_id", obj },
|
|
{ "instance_id", 0 } /* DbConnection class fills in real ID */
|
|
});
|
|
|
|
query.WhereCriteria = new Dictionary({
|
|
{ "object_id", obj },
|
|
{ "varname", kv.first }
|
|
});
|
|
|
|
queries.emplace_back(std::move(query));
|
|
}
|
|
|
|
OnMultipleQueries(queries);
|
|
}
|
|
}
|
|
|
|
double DbObject::GetLastConfigUpdate() const
|
|
{
|
|
return m_LastConfigUpdate;
|
|
}
|
|
|
|
double DbObject::GetLastStatusUpdate() const
|
|
{
|
|
return m_LastStatusUpdate;
|
|
}
|
|
|
|
void DbObject::OnConfigUpdateHeavy()
|
|
{
|
|
/* Default handler does nothing. */
|
|
}
|
|
|
|
void DbObject::OnConfigUpdateLight()
|
|
{
|
|
/* Default handler does nothing. */
|
|
}
|
|
|
|
void DbObject::OnStatusUpdate()
|
|
{
|
|
/* Default handler does nothing. */
|
|
}
|
|
|
|
DbObject::Ptr DbObject::GetOrCreateByObject(const ConfigObject::Ptr& object)
|
|
{
|
|
std::unique_lock<std::mutex> lock(GetStaticMutex());
|
|
|
|
DbObject::Ptr dbobj = object->GetExtension("DbObject");
|
|
|
|
if (dbobj)
|
|
return dbobj;
|
|
|
|
DbType::Ptr dbtype = DbType::GetByName(object->GetReflectionType()->GetName());
|
|
|
|
if (!dbtype)
|
|
return nullptr;
|
|
|
|
Service::Ptr service;
|
|
String name1, name2;
|
|
|
|
service = dynamic_pointer_cast<Service>(object);
|
|
|
|
if (service) {
|
|
Host::Ptr host = service->GetHost();
|
|
|
|
name1 = service->GetHost()->GetName();
|
|
name2 = service->GetShortName();
|
|
} else {
|
|
if (object->GetReflectionType() == CheckCommand::TypeInstance ||
|
|
object->GetReflectionType() == EventCommand::TypeInstance ||
|
|
object->GetReflectionType() == NotificationCommand::TypeInstance) {
|
|
Command::Ptr command = dynamic_pointer_cast<Command>(object);
|
|
name1 = CompatUtility::GetCommandName(command);
|
|
}
|
|
else
|
|
name1 = object->GetName();
|
|
}
|
|
|
|
dbobj = dbtype->GetOrCreateObjectByName(name1, name2);
|
|
|
|
dbobj->SetObject(object);
|
|
object->SetExtension("DbObject", dbobj);
|
|
|
|
return dbobj;
|
|
}
|
|
|
|
void DbObject::StateChangedHandler(const ConfigObject::Ptr& object)
|
|
{
|
|
DbObject::Ptr dbobj = GetOrCreateByObject(object);
|
|
|
|
if (!dbobj)
|
|
return;
|
|
|
|
dbobj->SendStatusUpdate();
|
|
}
|
|
|
|
void DbObject::VarsChangedHandler(const CustomVarObject::Ptr& object)
|
|
{
|
|
DbObject::Ptr dbobj = GetOrCreateByObject(object);
|
|
|
|
if (!dbobj)
|
|
return;
|
|
|
|
dbobj->SendVarsStatusUpdate();
|
|
}
|
|
|
|
void DbObject::VersionChangedHandler(const ConfigObject::Ptr& object)
|
|
{
|
|
DbObject::Ptr dbobj = DbObject::GetOrCreateByObject(object);
|
|
|
|
if (dbobj) {
|
|
Dictionary::Ptr configFields = dbobj->GetConfigFields();
|
|
String configHash = dbobj->CalculateConfigHash(configFields);
|
|
configFields->Set("config_hash", configHash);
|
|
|
|
dbobj->SendConfigUpdateHeavy(configFields);
|
|
dbobj->SendStatusUpdate();
|
|
}
|
|
}
|
|
|
|
std::mutex& DbObject::GetStaticMutex()
|
|
{
|
|
static std::mutex mutex;
|
|
return mutex;
|
|
}
|