From: Michael Friedrich Date: Fri, 18 Sep 2015 13:01:44 +0000 (+0200) Subject: Implement support for restoring modified attributes X-Git-Tag: v2.4.0~308 X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=95bcbec5b511a3a067ad5a47936bea2a004c5251;p=icinga2 Implement support for restoring modified attributes Also fix a glitch with ModifyAttribute and original_attributes storage. fixes #9935 --- diff --git a/lib/base/configobject.cpp b/lib/base/configobject.cpp index 6d3dff544..147c1956d 100644 --- a/lib/base/configobject.cpp +++ b/lib/base/configobject.cpp @@ -138,9 +138,9 @@ void ConfigObject::ModifyAttribute(const String& attr, const Value& value, bool SetOriginalAttributes(original_attributes, true); } - if (!original_attributes->Contains(attr)) { + if (!original_attributes->Contains(fieldName)) { updated_original_attributes = true; - original_attributes->Set(attr, oldValue); + original_attributes->Set(fieldName, oldValue); } } @@ -194,16 +194,78 @@ void ConfigObject::ModifyAttribute(const String& attr, const Value& value, bool void ConfigObject::RestoreAttribute(const String& attr) { - //TODO-MA: vars.os + Type::Ptr type = GetReflectionType(); + + std::vector tokens; + boost::algorithm::split(tokens, attr, boost::is_any_of(".")); + + String fieldName = tokens[0]; + + int fid = type->GetFieldId(fieldName); + Field field = type->GetFieldInfo(fid); + + Value currentValue = GetField(fid); + Dictionary::Ptr original_attributes = GetOriginalAttributes(); - if (!original_attributes || !original_attributes->Contains(attr)) + if (!original_attributes || !original_attributes->Contains(fieldName)) return; - Value attrVal = original_attributes->Get(attr); + Value newValue; + + if (tokens.size() > 1) { + newValue = currentValue.Clone(); + Value current = newValue; + + if (current.IsEmpty()) + BOOST_THROW_EXCEPTION(std::invalid_argument("Cannot restore non-existing object attribute")); + + Value old = original_attributes->Get(fieldName); + + for (std::vector::size_type i = 1; i < tokens.size() - 1; i++) { + if (!current.IsObjectType() || !old.IsObjectType()) + BOOST_THROW_EXCEPTION(std::invalid_argument("Value must be a dictionary.")); + + Dictionary::Ptr currentDict = current; + Dictionary::Ptr oldDict = old; + + const String& key = tokens[i]; + + if (!currentDict->Contains(key)) + BOOST_THROW_EXCEPTION(std::invalid_argument("Cannot restore non-existing object attribute")); + + /* silently ignore missing old values */ + if (!oldDict->Contains(key)) + return; + + current = currentDict->Get(key); + old = oldDict->Get(key); + } + + if (!current.IsObjectType() || !old.IsObjectType()) + BOOST_THROW_EXCEPTION(std::invalid_argument("Value must be a dictionary.")); + + Dictionary::Ptr currentDict = current; + Dictionary::Ptr oldDict = old; + + const String& key = tokens[tokens.size() - 1]; + + Value oldValue = oldDict->Get(key).Clone(); + currentDict->Set(key, oldValue); + + oldDict->Remove(key); + } else { + newValue = original_attributes->Get(fieldName); + original_attributes->Remove(fieldName); + } + + SetField(fid, newValue); - SetField(GetReflectionType()->GetFieldId(attr), attrVal); - original_attributes->Remove(attr); + /* increment the version. although restoring would mean + * decrementing the version, but we cannot notify other + * cluster nodes without increment. + */ + SetVersion(GetVersion() + 1); } bool ConfigObject::IsAttributeModified(const String& attr) const