/****************************************************************************** * Icinga 2 * * Copyright (C) 2012-2018 Icinga Development Team (https://www.icinga.com/) * * * * 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 "redis/rediswriter.hpp" #include "redis/redisconnection.hpp" #include "icinga/command.hpp" #include "icinga/compatutility.hpp" #include "base/configtype.hpp" #include "base/configobject.hpp" #include "icinga/customvarobject.hpp" #include "icinga/host.hpp" #include "icinga/service.hpp" #include "icinga/hostgroup.hpp" #include "icinga/servicegroup.hpp" #include "icinga/usergroup.hpp" #include "icinga/checkcommand.hpp" #include "icinga/eventcommand.hpp" #include "icinga/notificationcommand.hpp" #include "icinga/timeperiod.hpp" #include "remote/zone.hpp" #include "base/json.hpp" #include "base/logger.hpp" #include "base/serializer.hpp" #include "base/tlsutility.hpp" #include "base/initialize.hpp" #include "base/convert.hpp" #include "base/array.hpp" #include "base/exception.hpp" #include #include using namespace icinga; INITIALIZE_ONCE(&RedisWriter::ConfigStaticInitialize); void RedisWriter::ConfigStaticInitialize() { /* triggered in ProcessCheckResult(), requires UpdateNextCheck() to be called before */ Checkable::OnStateChange.connect(std::bind(&RedisWriter::StateChangeHandler, _1)); /* triggered on create, update and delete objects */ ConfigObject::OnActiveChanged.connect(std::bind(&RedisWriter::VersionChangedHandler, _1)); ConfigObject::OnVersionChanged.connect(std::bind(&RedisWriter::VersionChangedHandler, _1)); /* fixed downtime start */ Downtime::OnDowntimeStarted.connect(std::bind(&RedisWriter::DowntimeChangedHandler, _1)); /* flexible downtime start */ Downtime::OnDowntimeTriggered.connect(std::bind(&RedisWriter::DowntimeChangedHandler, _1)); /* fixed/flexible downtime end */ Downtime::OnDowntimeRemoved.connect(std::bind(&RedisWriter::DowntimeChangedHandler, _1)); } void RedisWriter::UpdateAllConfigObjects() { double startTime = Utility::GetTime(); // Use a Workqueue to pack objects in parallel WorkQueue upq(25000, Configuration::Concurrency); upq.SetName("RedisWriter:ConfigDump"); typedef std::pair TypePair; std::vector types; for (const Type::Ptr& type : Type::GetAllTypes()) { ConfigType *ctype = dynamic_cast(type.get()); if (!ctype) continue; String lcType(type->GetName().ToLower()); if (lcType == "downtime") { types.emplace_back(ctype, "hostdowntime"); types.emplace_back(ctype, "servicedowntime"); } else if (lcType == "comment") { types.emplace_back(ctype, "hostcomment"); types.emplace_back(ctype, "servicecomment"); } else { types.emplace_back(ctype, lcType); } } upq.ParallelFor(types, [this](const TypePair& type) { String lcType = type.second; m_Rcon->ExecuteQuery( {"DEL", m_PrefixConfigCheckSum + lcType, m_PrefixConfigObject + lcType, m_PrefixStateObject + lcType}); size_t bulkCounter = 0; std::vector attributes = {"HMSET", m_PrefixConfigObject + lcType}; std::vector customVars = {"HMSET", m_PrefixConfigCustomVar + lcType}; std::vector checksums = {"HMSET", m_PrefixConfigCheckSum + lcType}; std::vector states = {"HMSET", m_PrefixStateObject + lcType }; bool dumpState = (lcType == "host" || lcType == "service"); for (const ConfigObject::Ptr& object : type.first->GetObjects()) { if (lcType != GetLowerCaseTypeNameDB(object)) continue; CreateConfigUpdate(object, lcType, attributes, customVars, checksums, false); // Write out inital state for checkables if (dumpState) { states.emplace_back(GetObjectIdentifier(object)); states.emplace_back(JsonEncode(SerializeState(dynamic_pointer_cast(object)))); } bulkCounter++; if (!bulkCounter % 100) { if (attributes.size() > 2) { m_Rcon->ExecuteQuery(attributes); attributes.erase(attributes.begin() + 2, attributes.end()); } if (customVars.size() > 2) { m_Rcon->ExecuteQuery(customVars); customVars.erase(customVars.begin() + 2, customVars.end()); } if (checksums.size() > 2) { m_Rcon->ExecuteQuery(checksums); checksums.erase(checksums.begin() + 2, checksums.end()); } if (states.size() > 2) { m_Rcon->ExecuteQuery(states); states.erase(states.begin() + 2, states.end()); } } } if (attributes.size() > 2) m_Rcon->ExecuteQuery(attributes); if (customVars.size() > 2) m_Rcon->ExecuteQuery(customVars); if (checksums.size() > 2) m_Rcon->ExecuteQuery(checksums); if (states.size() > 2) m_Rcon->ExecuteQuery(states); m_Rcon->ExecuteQuery({"PUBLISH", "icinga:config:dump", lcType}); Log(LogNotice, "RedisWriter") << "Dumped " << bulkCounter << " objects of type " << type.second; }); upq.Join(); if (upq.HasExceptions()) { for (auto exc : upq.GetExceptions()) { Log(LogCritical, "RedisWriter") << "Exception during ConfigDump: " << exc; } } Log(LogInformation, "RedisWriter") << "Initial config/status dump finished in " << Utility::GetTime() - startTime << " seconds."; } void RedisWriter::UpdateState(const Checkable::Ptr& checkable) { Dictionary::Ptr stateAttrs = SerializeState(checkable); m_Rcon->ExecuteQuery({"HSET", m_PrefixStateObject + GetLowerCaseTypeNameDB(checkable), GetObjectIdentifier(checkable), JsonEncode(stateAttrs)}); } template static ConfigObject::Ptr GetObjectByName(const String& name) { return ConfigObject::GetObject(name); } // Used to update a single object, used for runtime updates void RedisWriter::SendConfigUpdate(const ConfigObject::Ptr& object, bool runtimeUpdate) { if (!m_Rcon || !m_Rcon->IsConnected()) return; String typeName = GetLowerCaseTypeNameDB(object); std::vector attribute = {"HSET", m_PrefixConfigObject + typeName}; std::vector customVar = {"HSET", m_PrefixConfigCustomVar + typeName}; std::vector checksum = {"HSET", m_PrefixConfigCheckSum + typeName}; std::vector state = {"HSET", m_PrefixStateObject + typeName}; CreateConfigUpdate(object, typeName, attribute, customVar, checksum, runtimeUpdate); Checkable::Ptr checkable = dynamic_pointer_cast(object); if (checkable) { m_Rcon->ExecuteQuery({"HSET", m_PrefixStateObject + typeName, GetObjectIdentifier(checkable), JsonEncode(SerializeState(checkable))}); } m_Rcon->ExecuteQuery(attribute); m_Rcon->ExecuteQuery(checksum); if (customVar.size() > 2) m_Rcon->ExecuteQuery(customVar); } void RedisWriter::MakeTypeChecksums(const ConfigObject::Ptr& object, std::set& propertiesBlacklist, Dictionary::Ptr& checkSums) { Endpoint::Ptr endpoint = dynamic_pointer_cast(object); if (endpoint) { auto endpointZone(endpoint->GetZone()); if (endpointZone) { checkSums->Set("zone_checksum", GetObjectIdentifier(endpointZone)); } return; } else { /* 'zone' is available for all config objects, therefore calculate the checksum. */ auto zone(static_pointer_cast(object->GetZone())); if (zone) checkSums->Set("zone_checksum", GetObjectIdentifier(zone)); } User::Ptr user = dynamic_pointer_cast(object); if (user) { propertiesBlacklist.emplace("groups"); Array::Ptr groups; ConfigObject::Ptr (*getGroup)(const String& name); groups = user->GetGroups(); getGroup = &::GetObjectByName; checkSums->Set("groups_checksum", CalculateCheckSumArray(groups)); Array::Ptr groupChecksums = new Array(); ObjectLock groupsLock(groups); ObjectLock groupChecksumsLock(groupChecksums); for (auto group : groups) { groupChecksums->Add(GetObjectIdentifier((*getGroup)(group.Get()))); } checkSums->Set("group_checksums", groupChecksums); auto period(user->GetPeriod()); if (period) checkSums->Set("period_checksum", GetObjectIdentifier(period)); return; } Notification::Ptr notification = dynamic_pointer_cast(object); if (notification) { Host::Ptr host; Service::Ptr service; auto users(notification->GetUsers()); Array::Ptr userChecksums = new Array(); Array::Ptr userNames = new Array(); auto usergroups(notification->GetUserGroups()); Array::Ptr usergroupChecksums = new Array(); Array::Ptr usergroupNames = new Array(); tie(host, service) = GetHostService(notification->GetCheckable()); checkSums->Set("host_checksum", GetObjectIdentifier(host)); checkSums->Set("command_checksum", GetObjectIdentifier(notification->GetCommand())); if (service) checkSums->Set("service_checksum", GetObjectIdentifier(service)); propertiesBlacklist.emplace("users"); userChecksums->Reserve(users.size()); userNames->Reserve(users.size()); for (auto& user : users) { userChecksums->Add(GetObjectIdentifier(user)); userNames->Add(user->GetName()); } checkSums->Set("user_checksums", userChecksums); checkSums->Set("users_checksum", CalculateCheckSumArray(userNames)); propertiesBlacklist.emplace("user_groups"); usergroupChecksums->Reserve(usergroups.size()); usergroupNames->Reserve(usergroups.size()); for (auto& usergroup : usergroups) { usergroupChecksums->Add(GetObjectIdentifier(usergroup)); usergroupNames->Add(usergroup->GetName()); } checkSums->Set("usergroup_checksums", usergroupChecksums); checkSums->Set("usergroups_checksum", CalculateCheckSumArray(usergroupNames)); return; } /* Calculate checkable checksums. */ Checkable::Ptr checkable = dynamic_pointer_cast(object); if (checkable) { /* groups_checksum, group_checksums */ propertiesBlacklist.emplace("groups"); Host::Ptr host; Service::Ptr service; tie(host, service) = GetHostService(checkable); Array::Ptr groups; ConfigObject::Ptr (*getGroup)(const String& name); if (service) { groups = service->GetGroups(); getGroup = &::GetObjectByName; /* Calculate the host_checksum */ checkSums->Set("host_checksum", GetObjectIdentifier(host)); } else { groups = host->GetGroups(); getGroup = &::GetObjectByName; } checkSums->Set("groups_checksum", CalculateCheckSumArray(groups)); Array::Ptr groupChecksums = new Array(); ObjectLock groupsLock(groups); ObjectLock groupChecksumsLock(groupChecksums); for (auto group : groups) { groupChecksums->Add(GetObjectIdentifier((*getGroup)(group.Get()))); } checkSums->Set("group_checksums", groupChecksums); /* command_endpoint_checksum / node_checksum */ Endpoint::Ptr commandEndpoint = checkable->GetCommandEndpoint(); if (commandEndpoint) checkSums->Set("command_endpoint_checksum", GetObjectIdentifier(commandEndpoint)); /* *_command_checksum */ checkSums->Set("checkcommand_checksum", GetObjectIdentifier(checkable->GetCheckCommand())); EventCommand::Ptr eventCommand = checkable->GetEventCommand(); if (eventCommand) checkSums->Set("eventcommand_checksum", GetObjectIdentifier(eventCommand)); /* *_url_checksum, icon_image_checksum */ String actionUrl = checkable->GetActionUrl(); String notesUrl = checkable->GetNotesUrl(); String iconImage = checkable->GetIconImage(); if (!actionUrl.IsEmpty()) checkSums->Set("action_url_checksum", CalculateCheckSumString(actionUrl)); if (!notesUrl.IsEmpty()) checkSums->Set("notes_url_checksum", CalculateCheckSumString(notesUrl)); if (!iconImage.IsEmpty()) checkSums->Set("icon_image_checksum", CalculateCheckSumString(iconImage)); return; } Zone::Ptr zone = dynamic_pointer_cast(object); if (zone) { propertiesBlacklist.emplace("endpoints"); auto endpointObjects = zone->GetEndpoints(); Array::Ptr endpoints = new Array(); endpoints->Resize(endpointObjects.size()); Array::SizeType i = 0; for (auto& endpointObject : endpointObjects) { endpoints->Set(i++, endpointObject->GetName()); } checkSums->Set("endpoints_checksum", CalculateCheckSumArray(endpoints)); Array::Ptr parents(new Array); for (auto& parent : zone->GetAllParentsRaw()) { parents->Add(GetObjectIdentifier(parent)); } checkSums->Set("all_parents_checksums", parents); checkSums->Set("all_parents_checksum", HashValue(zone->GetAllParents())); return; } /* zone_checksum for endpoints already is calculated above. */ Command::Ptr command = dynamic_pointer_cast(object); if (command) { Dictionary::Ptr arguments = command->GetArguments(); Dictionary::Ptr argumentChecksums = new Dictionary; if (arguments) { ObjectLock argumentsLock(arguments); for (auto& kv : arguments) { argumentChecksums->Set(kv.first, HashValue(kv.second)); } } checkSums->Set("arguments_checksum", HashValue(arguments)); checkSums->Set("argument_checksums", argumentChecksums); propertiesBlacklist.emplace("arguments"); Dictionary::Ptr envvars = command->GetEnv(); Dictionary::Ptr envvarChecksums = new Dictionary; if (envvars) { ObjectLock argumentsLock(envvars); for (auto& kv : envvars) { envvarChecksums->Set(kv.first, HashValue(kv.second)); } } checkSums->Set("envvars_checksum", HashValue(envvars)); checkSums->Set("envvar_checksums", envvarChecksums); propertiesBlacklist.emplace("env"); return; } TimePeriod::Ptr timeperiod = dynamic_pointer_cast(object); if (timeperiod) { Dictionary::Ptr ranges = timeperiod->GetRanges(); checkSums->Set("ranges_checksum", HashValue(ranges)); propertiesBlacklist.emplace("ranges"); // Compute checksums for Includes (like groups) Array::Ptr includes; ConfigObject::Ptr (*getInclude)(const String& name); includes = timeperiod->GetIncludes(); getInclude = &::GetObjectByName; checkSums->Set("includes_checksum", CalculateCheckSumArray(includes)); Array::Ptr includeChecksums = new Array(); ObjectLock includesLock(includes); ObjectLock includeChecksumsLock(includeChecksums); for (auto include : includes) { includeChecksums->Add(GetObjectIdentifier((*getInclude)(include.Get()))); } checkSums->Set("include_checksums", includeChecksums); // Compute checksums for Excludes (like groups) Array::Ptr excludes; ConfigObject::Ptr (*getExclude)(const String& name); excludes = timeperiod->GetExcludes(); getExclude = &::GetObjectByName; checkSums->Set("excludes_checksum", CalculateCheckSumArray(excludes)); Array::Ptr excludeChecksums = new Array(); ObjectLock excludesLock(excludes); ObjectLock excludeChecksumsLock(excludeChecksums); for (auto exclude : excludes) { excludeChecksums->Add(GetObjectIdentifier((*getExclude)(exclude.Get()))); } checkSums->Set("exclude_checksums", excludeChecksums); return; } Comment::Ptr comment = dynamic_pointer_cast(object); if (comment) { propertiesBlacklist.emplace("name"); propertiesBlacklist.emplace("host_name"); Host::Ptr host; Service::Ptr service; tie(host, service) = GetHostService(comment->GetCheckable()); if (service) { propertiesBlacklist.emplace("service_name"); checkSums->Set("service_checksum", GetObjectIdentifier(service)); } else checkSums->Set("host_checksum", GetObjectIdentifier(host)); return; } Downtime::Ptr downtime = dynamic_pointer_cast(object); if (downtime) { propertiesBlacklist.emplace("name"); propertiesBlacklist.emplace("host_name"); Host::Ptr host; Service::Ptr service; tie(host, service) = GetHostService(downtime->GetCheckable()); if (service) { propertiesBlacklist.emplace("service_name"); checkSums->Set("service_checksum", GetObjectIdentifier(service)); } else checkSums->Set("host_checksum", GetObjectIdentifier(host)); return; } } /* Creates a config update with computed checksums etc. * Writes attributes, customVars and checksums into the respective supplied vectors. Adds two values to each vector * (if applicable), first the key then the value. To use in a Redis command the command (e.g. HSET) and the key (e.g. * icinga:config:object:downtime) need to be prepended. There is nothing to indicate success or failure. */ void RedisWriter::CreateConfigUpdate(const ConfigObject::Ptr& object, const String typeName, std::vector& attributes, std::vector& customVars, std::vector& checksums, bool runtimeUpdate) { /* TODO: This isn't essentially correct as we don't keep track of config objects ourselves. This would avoid duplicated config updates at startup. if (!runtimeUpdate && m_ConfigDumpInProgress) return; */ if (m_Rcon == nullptr) return; String objectKey = GetObjectIdentifier(object); std::set propertiesBlacklist({"name", "__name", "package", "source_location", "templates"}); Dictionary::Ptr checkSums = new Dictionary(); checkSums->Set("name_checksum", CalculateCheckSumString(object->GetShortName())); checkSums->Set("environment_checksum", CalculateCheckSumString(GetEnvironment())); MakeTypeChecksums(object, propertiesBlacklist, checkSums); /* Send all object attributes to Redis, no extra checksums involved here. */ auto tempAttrs = (UpdateObjectAttrs(object, FAConfig, typeName)); attributes.insert(attributes.end(), std::begin(tempAttrs), std::end(tempAttrs)); /* Custom var checksums. */ CustomVarObject::Ptr customVarObject = dynamic_pointer_cast(object); if (customVarObject) { propertiesBlacklist.emplace("vars"); checkSums->Set("vars_checksum", CalculateCheckSumVars(customVarObject)); auto vars(SerializeVars(customVarObject)); if (vars) { auto varsJson(JsonEncode(vars)); customVars.emplace_back(objectKey); customVars.emplace_back(varsJson); } } checkSums->Set("metadata_checksum", CalculateCheckSumMetadata(object)); /* TODO: Problem: This does not account for `is_in_effect`, `trigger_time` of downtimes. */ checkSums->Set("properties_checksum", CalculateCheckSumProperties(object, propertiesBlacklist)); String checkSumsBody = JsonEncode(checkSums); checksums.emplace_back(objectKey); checksums.emplace_back(checkSumsBody); /* Send an update event to subscribers. */ if (runtimeUpdate) { m_Rcon->ExecuteQuery({"PUBLISH", "icinga:config:update", typeName + ":" + objectKey}); } } void RedisWriter::SendConfigDelete(const ConfigObject::Ptr& object) { String typeName = object->GetReflectionType()->GetName().ToLower(); String objectKey = GetObjectIdentifier(object); m_Rcon->ExecuteQueries({ {"HDEL", m_PrefixConfigObject + typeName, objectKey}, {"DEL", m_PrefixStateObject + typeName + ":" + objectKey}, {"PUBLISH", "icinga:config:delete", typeName + ":" + objectKey} }); } void RedisWriter::SendStatusUpdate(const ConfigObject::Ptr& object) { Checkable::Ptr checkable = dynamic_pointer_cast(object); if (!checkable) return; Host::Ptr host; Service::Ptr service; tie(host, service) = GetHostService(checkable); String streamname; if (service) streamname = "icinga:state:stream:service"; else streamname = "icinga:state:stream:host"; Dictionary::Ptr objectAttrs = SerializeState(checkable); std::vector streamadd({"XADD", streamname, "*"}); ObjectLock olock(objectAttrs); for (const Dictionary::Pair& kv : objectAttrs) { streamadd.emplace_back(kv.first); streamadd.emplace_back(kv.second); } m_Rcon->ExecuteQuery(streamadd); } Dictionary::Ptr RedisWriter::SerializeState(const Checkable::Ptr& checkable) { Dictionary::Ptr attrs = new Dictionary(); Host::Ptr host; Service::Ptr service; tie(host, service) = GetHostService(checkable); attrs->Set("id", GetObjectIdentifier(checkable));; attrs->Set("env_id", CalculateCheckSumString(GetEnvironment())); attrs->Set("state_type", checkable->GetStateType()); // TODO: last_hard/soft_state should be "previous". if (service) { attrs->Set("state", service->GetState()); attrs->Set("last_soft_state", service->GetState()); attrs->Set("last_hard_state", service->GetLastHardState()); } else { attrs->Set("state", host->GetState()); attrs->Set("last_soft_state", host->GetState()); attrs->Set("last_hard_state", host->GetLastHardState()); } attrs->Set("check_attempt", checkable->GetCheckAttempt()); //attrs->Set("severity") //attrs->Set(checkable->GetSeverity()); attrs->Set("is_active", checkable->IsActive()); CheckResult::Ptr cr = checkable->GetLastCheckResult(); if (cr) { attrs->Set("output", CompatUtility::GetCheckResultOutput(cr)); attrs->Set("long_output", CompatUtility::GetCheckResultLongOutput(cr)); attrs->Set("performance_data", JsonEncode(cr->GetPerformanceData())); attrs->Set("command", JsonEncode(cr->GetCommand())); attrs->Set("execution_time", cr->CalculateExecutionTime()); attrs->Set("latency", cr->CalculateLatency()); } bool isProblem = !checkable->IsStateOK(checkable->GetStateRaw()); attrs->Set("is_problem", isProblem); attrs->Set("is_handled", isProblem && (checkable->IsInDowntime() || checkable->IsAcknowledged())); attrs->Set("is_reachable", checkable->IsReachable()); attrs->Set("is_flapping", checkable->IsFlapping()); attrs->Set("is_acknowledged", checkable->IsAcknowledged()); if (checkable->IsAcknowledged()) { Timestamp entry = 0; Comment::Ptr AckComment; for (const Comment::Ptr& c : checkable->GetComments()) { if (c->GetEntryType() == CommentAcknowledgement) { if (c->GetEntryTime() > entry) { entry = c->GetEntryTime(); AckComment = c; } } } if (AckComment != nullptr) { attrs->Set("acknowledgement_comment_id", GetObjectIdentifier(AckComment)); } } attrs->Set("in_downtime", checkable->IsInDowntime()); if (checkable->GetCheckTimeout().IsEmpty()) attrs->Set("check_timeout",checkable->GetCheckCommand()->GetTimeout()); else attrs->Set("check_timeout", checkable->GetCheckTimeout()); attrs->Set("last_update", Utility::GetTime()); attrs->Set("last_state_change", checkable->GetLastStateChange()); attrs->Set("next_check", checkable->GetNextCheck()); return attrs; } std::vector RedisWriter::UpdateObjectAttrs(const ConfigObject::Ptr& object, int fieldType, const String& typeNameOverride) { Type::Ptr type = object->GetReflectionType(); Dictionary::Ptr attrs(new Dictionary); for (int fid = 0; fid < type->GetFieldCount(); fid++) { Field field = type->GetFieldInfo(fid); if ((field.Attributes & fieldType) == 0) continue; Value val = object->GetField(fid); /* hide attributes which shouldn't be user-visible */ if (field.Attributes & FANoUserView) continue; /* hide internal navigation fields */ if (field.Attributes & FANavigation && !(field.Attributes & (FAConfig | FAState))) continue; attrs->Set(field.Name, Serialize(val)); } /* Downtimes require in_effect, which is not an attribute */ Downtime::Ptr downtime = dynamic_pointer_cast(object); if (downtime) { attrs->Set("in_effect", Serialize(downtime->IsInEffect())); attrs->Set("trigger_time", Serialize(downtime->GetTriggerTime())); } /* Use the name checksum as unique key. */ String typeName = type->GetName().ToLower(); if (!typeNameOverride.IsEmpty()) typeName = typeNameOverride.ToLower(); return {GetObjectIdentifier(object), JsonEncode(attrs)}; //m_Rcon->ExecuteQuery({"HSET", keyPrefix + typeName, GetObjectIdentifier(object), JsonEncode(attrs)}); } void RedisWriter::StateChangeHandler(const ConfigObject::Ptr &object) { Type::Ptr type = object->GetReflectionType(); for (const RedisWriter::Ptr& rw : ConfigType::GetObjectsByType()) { rw->m_WorkQueue.Enqueue(std::bind(&RedisWriter::SendStatusUpdate, rw, object)); } } void RedisWriter::VersionChangedHandler(const ConfigObject::Ptr& object) { Type::Ptr type = object->GetReflectionType(); if (object->IsActive()) { // Create or update the object config for (const RedisWriter::Ptr& rw : ConfigType::GetObjectsByType()) { if (rw) rw->m_WorkQueue.Enqueue(std::bind(&RedisWriter::SendConfigUpdate, rw, object, true)); } } else if (!object->IsActive() && object->GetExtension("ConfigObjectDeleted")) { // same as in apilistener-configsync.cpp // Delete object config for (const RedisWriter::Ptr& rw : ConfigType::GetObjectsByType()) { if (rw) rw->m_WorkQueue.Enqueue(std::bind(&RedisWriter::SendConfigDelete, rw, object)); } } } void RedisWriter::DowntimeChangedHandler(const Downtime::Ptr& downtime) { StateChangeHandler(downtime->GetCheckable()); }