| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052 | #include "StdInc.h"#include "JsonNode.h"#include "HeroBonus.h"#include "Filesystem/CResourceLoader.h"const JsonNode JsonNode::nullNode;JsonNode::JsonNode(JsonType Type):	type(DATA_NULL){	setType(Type);}JsonNode::JsonNode(const char *data, size_t datasize):	type(DATA_NULL){	JsonParser parser(data, datasize, *this);	JsonValidator validator(*this);}JsonNode::JsonNode(ResourceID && fileURI):	type(DATA_NULL){	std::string filename = CResourceHandler::get()->getResourceName(fileURI);	FILE * file = fopen(filename.c_str(), "rb");	if (!file)	{		tlog1 << "Failed to open file " << filename << "\n";		perror("Last system error was ");		return;	}	fseek(file, 0, SEEK_END);	size_t datasize = ftell(file);	fseek(file, 0, SEEK_SET);	char *input = new char[datasize];	datasize = fread((void*)input, 1, datasize, file);	fclose(file);	JsonParser parser(input, datasize, *this);	JsonValidator validator(*this);	delete [] input;}JsonNode::JsonNode(const JsonNode ©):	type(DATA_NULL){	setType(copy.getType());	switch(type)	{		break; case DATA_NULL:		break; case DATA_BOOL:   Bool() =   copy.Bool();		break; case DATA_FLOAT:  Float() =  copy.Float();		break; case DATA_STRING: String() = copy.String();		break; case DATA_VECTOR: Vector() = copy.Vector();		break; case DATA_STRUCT: Struct() = copy.Struct();	}}JsonNode::~JsonNode(){	setType(DATA_NULL);}void JsonNode::swap(JsonNode &b){	using std::swap;	swap(data, b.data);	swap(type, b.type);}JsonNode & JsonNode::operator =(JsonNode node){	swap(node);	return *this;}bool JsonNode::operator == (const JsonNode &other) const{	if (getType() == other.getType())	{		switch(type)		{			break; case DATA_NULL:   return true;			break; case DATA_BOOL:   return Bool() == other.Bool();			break; case DATA_FLOAT:  return Float() == other.Float();			break; case DATA_STRING: return String() == other.String();			break; case DATA_VECTOR: return Vector() == other.Vector();			break; case DATA_STRUCT: return Struct() == other.Struct();		}	}	return false;}bool JsonNode::operator != (const JsonNode &other) const{	return !(*this == other);}void JsonNode::minimize(const JsonNode& schema){	JsonValidator validator(*this, schema, true);}void JsonNode::validate(const JsonNode& schema){	JsonValidator validator(*this, schema, false);}JsonNode::JsonType JsonNode::getType() const{	return type;}void JsonNode::setType(JsonType Type){	if (type == Type)		return;	//Reset node to NULL	if (Type != DATA_NULL)		setType(DATA_NULL);	switch (type)	{		break; case DATA_STRING:  delete data.String;		break; case DATA_VECTOR:  delete data.Vector;		break; case DATA_STRUCT:  delete data.Struct;		break; default:		break;	}	//Set new node type	type = Type;	switch(type)	{		break; case DATA_NULL:		break; case DATA_BOOL:   data.Bool = false;		break; case DATA_FLOAT:  data.Float = 0;		break; case DATA_STRING: data.String = new std::string;		break; case DATA_VECTOR: data.Vector = new JsonVector;		break; case DATA_STRUCT: data.Struct = new JsonMap;	}}bool JsonNode::isNull() const{	return type == DATA_NULL;}bool & JsonNode::Bool(){	setType(DATA_BOOL);	return data.Bool;}double & JsonNode::Float(){	setType(DATA_FLOAT);	return data.Float;}std::string & JsonNode::String(){	setType(DATA_STRING);	return *data.String;}JsonVector & JsonNode::Vector(){	setType(DATA_VECTOR);	return *data.Vector;}JsonMap & JsonNode::Struct(){	setType(DATA_STRUCT);	return *data.Struct;}const bool boolDefault = false;const bool & JsonNode::Bool() const{	if (type == DATA_NULL)		return boolDefault;	assert(type == DATA_BOOL);	return data.Bool;}const double floatDefault = 0;const double & JsonNode::Float() const{	if (type == DATA_NULL)		return floatDefault;	assert(type == DATA_FLOAT);	return data.Float;}const std::string stringDefault = std::string();const std::string & JsonNode::String() const{	if (type == DATA_NULL)		return stringDefault;	assert(type == DATA_STRING);	return *data.String;}const JsonVector vectorDefault = JsonVector();const JsonVector & JsonNode::Vector() const{	if (type == DATA_NULL)		return vectorDefault;	assert(type == DATA_VECTOR);	return *data.Vector;}const JsonMap mapDefault = JsonMap();const JsonMap & JsonNode::Struct() const{	if (type == DATA_NULL)		return mapDefault;	assert(type == DATA_STRUCT);	return *data.Struct;}JsonNode & JsonNode::operator[](std::string child){	return Struct()[child];}const JsonNode & JsonNode::operator[](std::string child) const{	JsonMap::const_iterator it = Struct().find(child);	if (it != Struct().end())		return it->second;	return nullNode;}////////////////////////////////////////////////////////////////////////////////void JsonNode::merge(JsonNode & dest, JsonNode & source){	switch (source.getType())	{		break; case DATA_NULL:   dest.setType(DATA_NULL);		break; case DATA_BOOL:   std::swap(dest.Bool(), source.Bool());		break; case DATA_FLOAT:  std::swap(dest.Float(), source.Float());		break; case DATA_STRING: std::swap(dest.String(), source.String());		break; case DATA_VECTOR:		{			//reserve place and *move* data from source to dest			source.Vector().reserve(source.Vector().size() + dest.Vector().size());			std::move(source.Vector().begin(), source.Vector().end(),			          std::back_inserter(dest.Vector()));		}		break; case DATA_STRUCT:		{			//recursively merge all entries from struct			BOOST_FOREACH(auto & node, source.Struct())				merge(dest[node.first], node.second);		}	}}void JsonNode::mergeCopy(JsonNode & dest, JsonNode source){	// uses copy created in stack to safely merge two nodes	merge(dest, source);}////////////////////////////////////////////////////////////////////////////////template<typename Iterator>void JsonWriter::writeContainer(Iterator begin, Iterator end){	if (begin == end)		return;	prefix += '\t';	end--;	while (begin != end)	{		writeEntry(begin++);		out<<",\n";	}	writeEntry(begin);	out<<"\n";	prefix.resize(prefix.size()-1);}void JsonWriter::writeEntry(JsonMap::const_iterator entry){	out << prefix;	writeString(entry->first);	out << " : ";	writeNode(entry->second);}void JsonWriter::writeEntry(JsonVector::const_iterator entry){	out << prefix;	writeNode(*entry);}void JsonWriter::writeString(const std::string &string){	static const std::string escaped = "\"\\/\b\f\n\r\t";	out <<'\"';	size_t pos=0, start=0;	for (; pos<string.size(); pos++)	{		size_t escapedChar = escaped.find(string[pos]);		if (escapedChar != std::string::npos)		{			out.write(string.data()+start, pos - start);			out << '\\' << escaped[escapedChar];			start = pos;		}	}	out.write(string.data()+start, pos - start);	out <<'\"';}void JsonWriter::writeNode(const JsonNode &node){	switch(node.getType())	{		break; case JsonNode::DATA_NULL:			out << "null";		break; case JsonNode::DATA_BOOL:			if (node.Bool())				out << "true";			else				out << "false";		break; case JsonNode::DATA_FLOAT:			out << node.Float();		break; case JsonNode::DATA_STRING:			writeString(node.String());		break; case JsonNode::DATA_VECTOR:			out << "[" << "\n";			writeContainer(node.Vector().begin(), node.Vector().end());			out << prefix << "]";		break; case JsonNode::DATA_STRUCT:			out << "{" << "\n";			writeContainer(node.Struct().begin(), node.Struct().end());			out << prefix << "}";	}}JsonWriter::JsonWriter(std::ostream &output, const JsonNode &node):	out(output){	writeNode(node);}std::ostream & operator<<(std::ostream &out, const JsonNode &node){	JsonWriter(out, node);	return out << "\n";}////////////////////////////////////////////////////////////////////////////////JsonParser::JsonParser(const char * inputString, size_t stringSize, JsonNode &root):	input(inputString, stringSize),	lineCount(1),	lineStart(0),	pos(0){	extractValue(root);	extractWhitespace(false);	//Warn if there are any non-whitespace symbols left	if (pos < input.size())		error("Not all file was parsed!", true);	//TODO: better way to show errors (like printing file name as well)	tlog3<<errors;}bool JsonParser::extractSeparator(){	if (!extractWhitespace())		return false;	if ( input[pos] !=':')		return error("Separator expected");	pos++;	return true;}bool JsonParser::extractValue(JsonNode &node){	if (!extractWhitespace())		return false;	switch (input[pos])	{		case '\"': return extractString(node);		case 'n' : return extractNull(node);		case 't' : return extractTrue(node);		case 'f' : return extractFalse(node);		case '{' : return extractStruct(node);		case '[' : return extractArray(node);		case '-' : return extractFloat(node);		default:		{			if (input[pos] >= '0' && input[pos] <= '9')				return extractFloat(node);			return error("Value expected!");		}	}}bool JsonParser::extractWhitespace(bool verbose){	while (true)	{		while (pos < input.size() && (ui8)input[pos] <= ' ')		{			if (input[pos] == '\n')			{				lineCount++;				lineStart = pos+1;			}			pos++;		}		if (pos >= input.size() || input[pos] != '/')			break;		pos++;		if (pos == input.size())			break;		if (input[pos] == '/')			pos++;		else			error("Comments must consist from two slashes!", true);		while (pos < input.size() && input[pos] != '\n')			pos++;	}	if (pos >= input.size() && verbose)		return error("Unexpected end of file!");	return true;}bool JsonParser::extractEscaping(std::string &str){	switch(input[pos++])	{		break; case '\"': str += '\"';		break; case '\\': str += '\\';		break; case  '/': str += '/';		break; case '\b': str += '\b';		break; case '\f': str += '\f';		break; case '\n': str += '\n';		break; case '\r': str += '\r';		break; case '\t': str += '\t';		break; default: return error("Unknown escape sequence!", true);	};	return true;}bool JsonParser::extractString(std::string &str){	if (input[pos] != '\"')		return error("String expected!");	pos++;	size_t first = pos;	while (pos != input.size())	{		if (input[pos] == '\"') // Correct end of string		{			str.append( &input[first], pos-first);			pos++;			return true;		}		if (input[pos] == '\\') // Escaping		{			str.append( &input[first], pos-first);			first = pos++;			if (pos == input.size())				break;			extractEscaping(str);		}		if (input[pos] == '\n') // end-of-line		{			str.append( &input[first], pos-first);			return error("Closing quote not found!", true);		}		if ((unsigned char)(input[pos]) < ' ') // control character		{			str.append( &input[first], pos-first);			first = pos+1;			error("Illegal character in the string!", true);		}		pos++;	}	return error("Unterminated string!");}bool JsonParser::extractString(JsonNode &node){	std::string str;	if (!extractString(str))		return false;	node.setType(JsonNode::DATA_STRING);	node.String() = str;	return true;}bool JsonParser::extractLiteral(const std::string &literal){	if (literal.compare(0, literal.size(), &input[pos], literal.size()) != 0)	{		while (pos < input.size() && ((input[pos]>'a' && input[pos]<'z')		                           || (input[pos]>'A' && input[pos]<'Z')))			pos++;		return error("Unknown literal found", true);	}	pos += literal.size();	return true;}bool JsonParser::extractNull(JsonNode &node){	if (!extractLiteral("null"))		return false;	node.setType(JsonNode::DATA_NULL);	return true;}bool JsonParser::extractTrue(JsonNode &node){	if (!extractLiteral("true"))		return false;	node.Bool() = true;	return true;}bool JsonParser::extractFalse(JsonNode &node){	if (!extractLiteral("false"))		return false;	node.Bool() = false;	return true;}bool JsonParser::extractStruct(JsonNode &node){	node.setType(JsonNode::DATA_STRUCT);	pos++;	if (!extractWhitespace())		return false;	//Empty struct found	if (input[pos] == '}')	{		pos++;		return true;	}	while (true)	{		if (!extractWhitespace())			return false;		std::string key;		if (!extractString(key))			return false;		if (node.Struct().find(key) != node.Struct().end())			error("Dublicated element encountered!", true);		if (!extractSeparator())			return false;		if (!extractElement(node.Struct()[key], '}'))			return false;		if (input[pos] == '}')		{			pos++;			return true;		}	}}bool JsonParser::extractArray(JsonNode &node){	pos++;	node.setType(JsonNode::DATA_VECTOR);	if (!extractWhitespace())		return false;	//Empty array found	if (input[pos] == ']')	{		pos++;		return true;	}	while (true)	{		//NOTE: currently 50% of time is this vector resizing.		//May be useful to use list during parsing and then swap() all items to vector		node.Vector().resize(node.Vector().size()+1);		if (!extractElement(node.Vector().back(), ']'))			return false;		if (input[pos] == ']')		{			pos++;			return true;		}	}}bool JsonParser::extractElement(JsonNode &node, char terminator){	if (!extractValue(node))		return false;	if (!extractWhitespace())		return false;	bool comma = (input[pos] == ',');	if (comma )	{		pos++;		if (!extractWhitespace())			return false;	}	if (input[pos] == terminator)		return true;	if (!comma)		error("Comma expected!", true);	return true;}bool JsonParser::extractFloat(JsonNode &node){	assert(input[pos] == '-' || (input[pos] >= '0' && input[pos] <= '9'));	bool negative=false;	double result=0;	if (input[pos] == '-')	{		pos++;		negative = true;	}	if (input[pos] < '0' || input[pos] > '9')		return error("Number expected!");	//Extract integer part	while (input[pos] >= '0' && input[pos] <= '9')	{		result = result*10+(input[pos]-'0');		pos++;	}	if (input[pos] == '.')	{		//extract fractional part		pos++;		double fractMult = 0.1;		if (input[pos] < '0' || input[pos] > '9')			return error("Decimal part expected!");		while (input[pos] >= '0' && input[pos] <= '9')		{			result = result + fractMult*(input[pos]-'0');			fractMult /= 10;			pos++;		}	}	//TODO: exponential part	if (negative)		result = -result;	node.setType(JsonNode::DATA_FLOAT);	node.Float() = result;	return true;}bool JsonParser::error(const std::string &message, bool warning){	std::ostringstream stream;	std::string type(warning?" warning: ":" error: ");	stream << "At line " << lineCount << ", position "<<pos-lineStart	       << type << message <<"\n";	errors += stream.str();	return warning;}static const std::map<std::string, JsonNode::JsonType> stringToType =	boost::assign::map_list_of		("null",   JsonNode::DATA_NULL)   ("bool",   JsonNode::DATA_BOOL)		("number", JsonNode::DATA_FLOAT)  ("string", JsonNode::DATA_STRING)		("array",  JsonNode::DATA_VECTOR) ("object", JsonNode::DATA_STRUCT);//Check current schema entry for validness and converts "type" string to JsonTypebool JsonValidator::validateSchema(JsonNode::JsonType &type, const JsonNode &schema){	if (schema.isNull())		return addMessage("Missing schema for current entry!");	const JsonNode &nodeType = schema["type"];	if (nodeType.isNull())		return addMessage("Entry type is not defined in schema!");	if (nodeType.getType() != JsonNode::DATA_STRING)		return addMessage("Entry type must be string!");	std::map<std::string, JsonNode::JsonType>::const_iterator iter = stringToType.find(nodeType.String());	if (iter == stringToType.end())		return addMessage("Unknown entry type found!");	type = iter->second;	return true;}//Replaces node with default value if needed and calls type-specific validatorsbool JsonValidator::validateType(JsonNode &node, const JsonNode &schema, JsonNode::JsonType type){	if (node.isNull())	{		const JsonNode & defaultValue = schema["default"];		if (defaultValue.isNull())			return addMessage("Null entry without default entry!");		else			node = defaultValue;	}	if (minimize && node == schema["default"])	{		node.setType(JsonNode::DATA_NULL);		return false;	}	if (type != node.getType())	{		node.setType(JsonNode::DATA_NULL);		return addMessage("Type mismatch!");	}	if (type == JsonNode::DATA_VECTOR)		return validateItems(node, schema["items"]);	if (type == JsonNode::DATA_STRUCT)		return validateProperties(node, schema["properties"]);	return true;}// Basic checks common for any nodesbool JsonValidator::validateNode(JsonNode &node, const JsonNode &schema, const std::string &name){	currentPath.push_back(name);	JsonNode::JsonType type = JsonNode::DATA_NULL;	if (!validateSchema(type, schema)	 || !validateType(node, schema, type))	{		node.setType(JsonNode::DATA_NULL);		currentPath.pop_back();		return false;	}	currentPath.pop_back();	return true;}//Checks "items" entry from schema (type-specific check for Vector)bool JsonValidator::validateItems(JsonNode &node, const JsonNode &schema){	JsonNode::JsonType type = JsonNode::DATA_NULL;	if (!validateSchema(type, schema))		return false;	bool result = true;	BOOST_FOREACH(JsonNode &entry, node.Vector())	{		if (!validateType(entry, schema, type))		{			result = false;			entry.setType(JsonNode::DATA_NULL);		}	}	return result;}//Checks "propertries" entry from schema (type-specific check for Struct)//Function is similar to merging of two sorted lists - check every entry that present in one of the input nodesbool JsonValidator::validateProperties(JsonNode &node, const JsonNode &schema){	if (schema.isNull())		return addMessage("Properties entry is missing for struct in schema");	JsonMap::iterator nodeIter = node.Struct().begin();	JsonMap::const_iterator schemaIter = schema.Struct().begin();	while (nodeIter != node.Struct().end() && schemaIter != schema.Struct().end())	{		if (nodeIter->first < schemaIter->first) //No schema for entry		{			validateNode(nodeIter->second, JsonNode::nullNode, nodeIter->first);			JsonMap::iterator toRemove = nodeIter++;			node.Struct().erase(toRemove);		}		else		if (schemaIter->first < nodeIter->first) //No entry		{			if (!validateNode(node[schemaIter->first], schemaIter->second, schemaIter->first))				node.Struct().erase(schemaIter->first);			schemaIter++;		}		else //both entry and schema are present		{			JsonMap::iterator current = nodeIter++;			if (!validateNode(current->second, schemaIter->second, current->first))				node.Struct().erase(current);			schemaIter++;		}	}	while (nodeIter != node.Struct().end())	{		validateNode(nodeIter->second, JsonNode::nullNode, nodeIter->first);		JsonMap::iterator toRemove = nodeIter++;		node.Struct().erase(toRemove);	}	while (schemaIter != schema.Struct().end())	{		if (!validateNode(node[schemaIter->first], schemaIter->second, schemaIter->first))			node.Struct().erase(schemaIter->first);		schemaIter++;	}	return true;}bool JsonValidator::addMessage(const std::string &message){	std::ostringstream stream;	stream << "At ";	BOOST_FOREACH(const std::string &path, currentPath)		stream << path<<"/";	stream << "\t Error: " << message <<"\n";	errors += stream.str();	return false;}JsonValidator::JsonValidator(JsonNode &root, bool Minimize):	minimize(Minimize){	JsonNode schema;	schema.swap(root["schema"]);	root.Struct().erase("schema");	if (!schema.isNull())	{		validateProperties(root, schema);	}	//This message is quite annoying now - most files do not have schemas. May be re-enabled later	//else	//	addMessage("Schema not found!", true);	//TODO: better way to show errors (like printing file name as well)	tlog3<<errors;}JsonValidator::JsonValidator(JsonNode &root, const JsonNode &schema, bool Minimize):	minimize(Minimize){	validateProperties(root, schema);	if (schema.isNull())		addMessage("Schema not found!");	tlog3<<errors;}Bonus * ParseBonus (const JsonVector &ability_vec) //TODO: merge with AddAbility, create universal parser for all bonus properties{	Bonus * b = new Bonus();	std::string type = ability_vec[0].String();	auto it = bonusNameMap.find(type);	if (it == bonusNameMap.end())	{		tlog1 << "Error: invalid ability type " << type << " in creatures.txt" << std::endl;		return b;	}	b->type = it->second;	b->val = ability_vec[1].Float();	b->subtype = ability_vec[2].Float();	b->additionalInfo = ability_vec[3].Float();	b->duration = Bonus::PERMANENT; //TODO: handle flags (as integer)	b->turnsRemain = 0;	return b;}template <typename T>const T & parseByMap(const std::map<std::string, T> & map, const JsonNode * val, std::string err){	static T defaultValue;	if (!val->isNull())	{		std::string type = val->String();		auto it = map.find(type);		if (it == map.end())		{			tlog1 << "Error: invalid " << err << type << std::endl;			return defaultValue;		}		else		{			return it->second;		}	}	else		return defaultValue;};Bonus * ParseBonus (const JsonNode &ability){	Bonus * b = new Bonus();	const JsonNode *value;	std::string type = ability["type"].String();	auto it = bonusNameMap.find(type);	if (it == bonusNameMap.end())	{		tlog1 << "Error: invalid ability type " << type << std::endl;		return b;	}	b->type = it->second;	value = &ability["subtype"];	if (!value->isNull())		b->subtype = value->Float();	value = &ability["val"];	if (!value->isNull())		b->val = value->Float();	value = &ability["valueType"];	if (!value->isNull())		b->valType = parseByMap(bonusValueMap, value, "value type ");	value = &ability["additionalInfo"];	if (!value->isNull())		b->additionalInfo = value->Float();	value = &ability["turns"];	if (!value->isNull())		b->turnsRemain = value->Float();	value = &ability["sourceID"];	if (!value->isNull())		b->sid = value->Float();	value = &ability["description"];	if (!value->isNull())		b->description = value->String();	value = &ability["effectRange"];	if (!value->isNull())		b->valType = parseByMap(bonusLimitEffect, value, "effect range ");	value = &ability["duration"];	if (!value->isNull())		b->valType = parseByMap(bonusDurationMap, value, "duration type ");	value = &ability["source"];	if (!value->isNull())		b->valType = parseByMap(bonusSourceMap, value, "source type ");	value = &ability["limiter"];	if (!value->isNull())		b->limiter = parseByMap(bonusLimiterMap, value, "limiter type ");	value = &ability["propagator"];	if (!value->isNull())		b->propagator = parseByMap(bonusPropagatorMap, value, "propagator type ");	return b;}//returns first Key with value equal to given onetemplate<class Key, class Val>Key reverseMapFirst(const Val & val, const std::map<Key, Val> map){	BOOST_FOREACH(auto it, map)	{		if(it.second == val)		{			return it.first;		}	}	assert(0);	return "";}DLL_LINKAGE void UnparseBonus( JsonNode &node, const Bonus * bonus ){	node["type"].String() = reverseMapFirst<std::string, int>(bonus->type, bonusNameMap);	node["subtype"].Float() = bonus->subtype;	node["val"].Float() = bonus->val;	node["valueType"].String() = reverseMapFirst<std::string, int>(bonus->valType, bonusValueMap);	node["additionalInfo"].Float() = bonus->additionalInfo;	node["turns"].Float() = bonus->turnsRemain;	node["sourceID"].Float() = bonus->source;	node["description"].String() = bonus->description;	node["effectRange"].String() = reverseMapFirst<std::string, int>(bonus->effectRange, bonusLimitEffect);	node["duration"].String() = reverseMapFirst<std::string, int>(bonus->duration, bonusDurationMap);	node["source"].String() = reverseMapFirst<std::string, int>(bonus->source, bonusSourceMap);	if(bonus->limiter != nullptr)	{		node["limiter"].String() = reverseMapFirst<std::string, TLimiterPtr>(bonus->limiter, bonusLimiterMap);	}	if(bonus->propagator != nullptr)	{		node["propagator"].String() = reverseMapFirst<std::string, TPropagatorPtr>(bonus->propagator, bonusPropagatorMap);	}		}
 |