Merge pull request #6427 from gunnarbeutner/fix/recursive-serialize

Improve error message for serializing objects with recursive references
This commit is contained in:
Michael Friedrich 2018-08-03 11:03:42 +02:00 committed by GitHub
commit 33492420f3
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 108 additions and 10 deletions

View File

@ -21,10 +21,68 @@
#include "base/type.hpp" #include "base/type.hpp"
#include "base/application.hpp" #include "base/application.hpp"
#include "base/objectlock.hpp" #include "base/objectlock.hpp"
#include "base/convert.hpp"
#include "base/exception.hpp"
#include <boost/algorithm/string/join.hpp>
#include <deque>
using namespace icinga; using namespace icinga;
static Array::Ptr SerializeArray(const Array::Ptr& input, int attributeTypes) struct SerializeStackEntry
{
String Name;
Value Val;
};
CircularReferenceError::CircularReferenceError(String message, std::vector<String> path)
: m_Message(message), m_Path(path)
{ }
const char *CircularReferenceError::what(void) const throw()
{
return m_Message.CStr();
}
std::vector<String> CircularReferenceError::GetPath() const
{
return m_Path;
}
struct SerializeStack
{
std::deque<SerializeStackEntry> Entries;
inline void Push(const String& name, const Value& val)
{
Object::Ptr obj;
if (val.IsObject())
obj = val;
if (obj) {
for (const auto& entry : Entries) {
if (entry.Val == obj) {
std::vector<String> path;
for (const auto& entry : Entries)
path.push_back(entry.Name);
path.push_back(name);
BOOST_THROW_EXCEPTION(CircularReferenceError("Cannot serialize object which recursively refers to itself. Attribute path which leads to the cycle: " + boost::algorithm::join(path, " -> "), path));
}
}
}
Entries.push_back({ name, obj });
}
inline void Pop()
{
Entries.pop_back();
}
};
static Value SerializeInternal(const Value& value, int attributeTypes, SerializeStack& stack);
static Array::Ptr SerializeArray(const Array::Ptr& input, int attributeTypes, SerializeStack& stack)
{ {
ArrayData result; ArrayData result;
@ -32,14 +90,19 @@ static Array::Ptr SerializeArray(const Array::Ptr& input, int attributeTypes)
ObjectLock olock(input); ObjectLock olock(input);
int index = 0;
for (const Value& value : input) { for (const Value& value : input) {
result.emplace_back(Serialize(value, attributeTypes)); stack.Push(Convert::ToString(index), value);
result.emplace_back(SerializeInternal(value, attributeTypes, stack));
stack.Pop();
index++;
} }
return new Array(std::move(result)); return new Array(std::move(result));
} }
static Dictionary::Ptr SerializeDictionary(const Dictionary::Ptr& input, int attributeTypes) static Dictionary::Ptr SerializeDictionary(const Dictionary::Ptr& input, int attributeTypes, SerializeStack& stack)
{ {
DictionaryData result; DictionaryData result;
@ -48,13 +111,15 @@ static Dictionary::Ptr SerializeDictionary(const Dictionary::Ptr& input, int att
ObjectLock olock(input); ObjectLock olock(input);
for (const Dictionary::Pair& kv : input) { for (const Dictionary::Pair& kv : input) {
result.emplace_back(kv.first, Serialize(kv.second, attributeTypes)); stack.Push(kv.first, kv.second);
result.emplace_back(kv.first, SerializeInternal(kv.second, attributeTypes, stack));
stack.Pop();
} }
return new Dictionary(std::move(result)); return new Dictionary(std::move(result));
} }
static Object::Ptr SerializeObject(const Object::Ptr& input, int attributeTypes) static Object::Ptr SerializeObject(const Object::Ptr& input, int attributeTypes, SerializeStack& stack)
{ {
Type::Ptr type = input->GetReflectionType(); Type::Ptr type = input->GetReflectionType();
@ -73,7 +138,10 @@ static Object::Ptr SerializeObject(const Object::Ptr& input, int attributeTypes)
if (strcmp(field.Name, "type") == 0) if (strcmp(field.Name, "type") == 0)
continue; continue;
fields.emplace_back(field.Name, Serialize(input->GetField(i), attributeTypes)); Value value = input->GetField(i);
stack.Push(field.Name, value);
fields.emplace_back(field.Name, SerializeInternal(input->GetField(i), attributeTypes, stack));
stack.Pop();
} }
fields.emplace_back("type", type->GetName()); fields.emplace_back("type", type->GetName());
@ -158,7 +226,7 @@ static Object::Ptr DeserializeObject(const Object::Ptr& object, const Dictionary
return instance; return instance;
} }
Value icinga::Serialize(const Value& value, int attributeTypes) static Value SerializeInternal(const Value& value, int attributeTypes, SerializeStack& stack)
{ {
if (!value.IsObject()) if (!value.IsObject())
return value; return value;
@ -168,14 +236,20 @@ Value icinga::Serialize(const Value& value, int attributeTypes)
Array::Ptr array = dynamic_pointer_cast<Array>(input); Array::Ptr array = dynamic_pointer_cast<Array>(input);
if (array) if (array)
return SerializeArray(array, attributeTypes); return SerializeArray(array, attributeTypes, stack);
Dictionary::Ptr dict = dynamic_pointer_cast<Dictionary>(input); Dictionary::Ptr dict = dynamic_pointer_cast<Dictionary>(input);
if (dict) if (dict)
return SerializeDictionary(dict, attributeTypes); return SerializeDictionary(dict, attributeTypes, stack);
return SerializeObject(input, attributeTypes); return SerializeObject(input, attributeTypes, stack);
}
Value icinga::Serialize(const Value& value, int attributeTypes)
{
SerializeStack stack;
return SerializeInternal(value, attributeTypes, stack);
} }
Value icinga::Deserialize(const Value& value, bool safe_mode, int attributeTypes) Value icinga::Deserialize(const Value& value, bool safe_mode, int attributeTypes)

View File

@ -23,10 +23,25 @@
#include "base/i2-base.hpp" #include "base/i2-base.hpp"
#include "base/type.hpp" #include "base/type.hpp"
#include "base/value.hpp" #include "base/value.hpp"
#include "base/exception.hpp"
namespace icinga namespace icinga
{ {
class CircularReferenceError : virtual public user_error
{
public:
CircularReferenceError(String message, std::vector<String> path);
~CircularReferenceError() throw() = default;
const char *what(void) const throw() final;
std::vector<String> GetPath() const;
private:
String m_Message;
std::vector<String> m_Path;
};
Value Serialize(const Value& value, int attributeTypes = FAState); Value Serialize(const Value& value, int attributeTypes = FAState);
Value Deserialize(const Value& value, bool safe_mode = false, int attributeTypes = FAState); Value Deserialize(const Value& value, bool safe_mode = false, int attributeTypes = FAState);
Value Deserialize(const Object::Ptr& object, const Value& value, bool safe_mode = false, int attributeTypes = FAState); Value Deserialize(const Object::Ptr& object, const Value& value, bool safe_mode = false, int attributeTypes = FAState);

View File

@ -36,6 +36,7 @@
#include "base/json.hpp" #include "base/json.hpp"
#include "base/exception.hpp" #include "base/exception.hpp"
#include "base/function.hpp" #include "base/function.hpp"
#include <boost/algorithm/string/join.hpp>
#include <sstream> #include <sstream>
#include <fstream> #include <fstream>
@ -289,6 +290,14 @@ ConfigObject::Ptr ConfigItem::Commit(bool discard)
throw; throw;
} }
Value serializedObject;
try {
serializedObject = Serialize(dobj, FAConfig);
} catch (const CircularReferenceError& ex) {
BOOST_THROW_EXCEPTION(ValidationError(dobj, ex.GetPath(), "Circular references are not allowed"));
}
Dictionary::Ptr persistentItem = new Dictionary({ Dictionary::Ptr persistentItem = new Dictionary({
{ "type", type->GetName() }, { "type", type->GetName() },
{ "name", GetName() }, { "name", GetName() },