| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281 | /* * JsonDeserializer.cpp, part of VCMI engine * * Authors: listed in file AUTHORS in main folder * * License: GNU General Public License v2.0 or later * Full text of license available in license.txt file, in main folder * */#include "StdInc.h"#include "JsonDeserializer.h"#include "../JsonNode.h"JsonDeserializer::JsonDeserializer(const IInstanceResolver * instanceResolver_, const JsonNode & root_):	JsonTreeSerializer(instanceResolver_, &root_, false){}void JsonDeserializer::serializeInternal(const std::string & fieldName, boost::logic::tribool & value){	const JsonNode & data = currentObject->operator[](fieldName);	if(data.getType() != JsonNode::JsonType::DATA_BOOL)		value = boost::logic::indeterminate;	else		value = data.Bool();}void JsonDeserializer::serializeInternal(const std::string & fieldName, si32 & value, const boost::optional<si32> & defaultValue, const TDecoder & decoder, const TEncoder & encoder){	std::string identifier;	serializeString(fieldName, identifier);	value = defaultValue ? defaultValue.get() : 0;	if(identifier != "")	{		si32 rawId = decoder(identifier);		if(rawId < 0) //may be, user has installed the mod into another directory...		{			auto internalId = vstd::splitStringToPair(identifier, ':').second;			auto currentScope = getCurrent().meta;			auto actualId = currentScope.length() > 0 ? currentScope + ":" + internalId : internalId;			rawId = decoder(actualId);			if(rawId >= 0)				logMod->warn("Identifier %s has been resolved as %s instead of %s", internalId, actualId, identifier);		}		if(rawId >= 0)			value = rawId;	}}void JsonDeserializer::serializeInternal(const std::string & fieldName, std::vector<si32> & value, const TDecoder & decoder, const TEncoder & encoder){	const JsonVector & data = currentObject->operator[](fieldName).Vector();	value.clear();	value.reserve(data.size());	for(const JsonNode elem : data)	{		si32 rawId = decoder(elem.String());		if(rawId >= 0)			value.push_back(rawId);	}}void JsonDeserializer::serializeInternal(const std::string & fieldName, double & value, const boost::optional<double> & defaultValue){	const JsonNode & data = currentObject->operator[](fieldName);	if(!data.isNumber())		value = defaultValue ? defaultValue.get() : 0;//todo: report error on not null?	else		value = data.Float();}void JsonDeserializer::serializeInternal(const std::string & fieldName, si64 & value, const boost::optional<si64> & defaultValue){	const JsonNode & data = currentObject->operator[](fieldName);	if(!data.isNumber())		value = defaultValue ? defaultValue.get() : 0;//todo: report error on not null?	else		value = data.Integer();}void JsonDeserializer::serializeInternal(const std::string & fieldName, si32 & value, const boost::optional<si32> & defaultValue, const std::vector<std::string> & enumMap){	const std::string & valueName = currentObject->operator[](fieldName).String();	const si32 actualOptional = defaultValue ? defaultValue.get() : 0;	si32 rawValue = vstd::find_pos(enumMap, valueName);	if(rawValue < 0)		value = actualOptional;	else		value = rawValue;}void JsonDeserializer::serializeInternal(std::string & value){	value = currentObject->String();}void JsonDeserializer::serializeInternal(int64_t & value){	value = currentObject->Integer();}void JsonDeserializer::serializeLIC(const std::string & fieldName, const TDecoder & decoder, const TEncoder & encoder, const std::vector<bool> & standard, std::vector<bool> & value){	const JsonNode & field = currentObject->operator[](fieldName);	const JsonNode & anyOf = field["anyOf"];	const JsonNode & allOf = field["allOf"];	const JsonNode & noneOf = field["noneOf"];	if(anyOf.Vector().empty() && allOf.Vector().empty())	{		//permissive mode		value = standard;	}	else	{		//restrictive mode		value.clear();		value.resize(standard.size(), false);		readLICPart(anyOf, decoder, true, value);		readLICPart(allOf, decoder, true, value);	}	readLICPart(noneOf, decoder, false, value);}void JsonDeserializer::serializeLIC(const std::string & fieldName, LIC & value){	const JsonNode & field = currentObject->operator[](fieldName);	const JsonNode & anyOf = field["anyOf"];	const JsonNode & allOf = field["allOf"];	const JsonNode & noneOf = field["noneOf"];	if(anyOf.Vector().empty())	{		//permissive mode		value.any = value.standard;	}	else	{		//restrictive mode		value.any.clear();		value.any.resize(value.standard.size(), false);		readLICPart(anyOf, value.decoder, true, value.any);	}	readLICPart(allOf, value.decoder, true, value.all);	readLICPart(noneOf, value.decoder, true, value.none);	//remove any banned from allowed and required	for(si32 idx = 0; idx < value.none.size(); idx++)	{		if(value.none[idx])		{			value.all[idx] = false;			value.any[idx] = false;		}	}	//add all required to allowed	for(si32 idx = 0; idx < value.all.size(); idx++)	{		if(value.all[idx])		{			value.any[idx] = true;		}	}}void JsonDeserializer::serializeLIC(const std::string & fieldName, LICSet & value){	const JsonNode & field = currentObject->operator[](fieldName);	const JsonNode & anyOf = field["anyOf"];	const JsonNode & allOf = field["allOf"];	const JsonNode & noneOf = field["noneOf"];	value.all.clear();	value.none.clear();	if(anyOf.Vector().empty())	{		//permissive mode		value.any = value.standard;	}	else	{		//restrictive mode		value.any.clear();		readLICPart(anyOf, value.decoder, value.any);		for(si32 item : value.standard)			if(!vstd::contains(value.any, item))				value.none.insert(item);	}	readLICPart(allOf, value.decoder, value.all);	readLICPart(noneOf, value.decoder, value.none);	//remove any banned from allowed and required	auto isBanned = [&value](const si32 item)->bool	{		return vstd::contains(value.none, item);	};	vstd::erase_if(value.all, isBanned);	vstd::erase_if(value.any, isBanned);	//add all required to allowed	for(si32 item : value.all)	{		value.any.insert(item);	}}void JsonDeserializer::serializeString(const std::string & fieldName, std::string & value){	value = currentObject->operator[](fieldName).String();}void JsonDeserializer::serializeRaw(const std::string & fieldName, JsonNode & value, const boost::optional<const JsonNode &> defaultValue){	const JsonNode & data = currentObject->operator[](fieldName);	if(data.getType() == JsonNode::JsonType::DATA_NULL)	{		if(defaultValue)			value = defaultValue.get();		else			value.clear();	}	else	{		value = data;	}}void JsonDeserializer::readLICPart(const JsonNode & part, const TDecoder & decoder, const bool val, std::vector<bool> & value){	for(size_t index = 0; index < part.Vector().size(); index++)	{		const std::string & identifier = part.Vector()[index].String();		const si32 rawId = decoder(identifier);		if(rawId >= 0)		{			if(rawId < value.size())				value[rawId] = val;			else				logGlobal->error("JsonDeserializer::serializeLIC: id out of bounds %d", rawId);		}	}}void JsonDeserializer::readLICPart(const JsonNode & part, const TDecoder & decoder, std::set<si32> & value){	for(size_t index = 0; index < part.Vector().size(); index++)	{		const std::string & identifier = part.Vector()[index].String();		const si32 rawId = decoder(identifier);		if(rawId != -1)			value.insert(rawId);	}}
 |