// Ryzom - MMORPG Framework <http://dev.ryzom.com/projects/ryzom/>
// Copyright (C) 2010  Winch Gate Property Limited
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as
// published by the Free Software Foundation, either version 3 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 Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program.  If not, see <http://www.gnu.org/licenses/>.

#include "nel/misc/types_nl.h"
#include "nel/misc/debug.h"
#include "nel/misc/string_common.h"
#include "nel/misc/path.h"
#include "nel/misc/sstring.h"
#include "nel/misc/smart_ptr.h"
#include "nel/misc/factory.h"
#include "nel/ligo/primitive.h"
#include "nel/ligo/primitive_utils.h"
#include "nel/ligo/ligo_config.h"
#include "../../../common/src/game_share/string_manager_sender.h"
#include "nel/misc/string_conversion.h"
#include <string>
#include <numeric>

class CMissionData;
class IStep;

const std::string NL("\n\r");

/// Exception thows when the parser meet somethink invalid.
struct EParseException
{
	EParseException(NLLIGO::IPrimitive *prim, const char *why)
		:	Primitive(prim),
		Why(why)
	{
	}
	
	NLLIGO::IPrimitive	*Primitive;
	std::string		Why;
};


// utility to untag a variable (tag are the '$' added before and after the var name)
inline void untagVar(std::string &var)
{
	if (!var.empty())
	{
		if (var[0] != '$')
			var = "";
		else
		{
			var = var.substr(1);
			if (var[var.size()-1] != '$')
				var = "";
			else
				var = var.substr(0, var.size()-1);
		}
	}
}

// utility to 'tabulate' the lines in a string
void tabulateLine(std::string &text, uint nbTabs);


/* Interface class for variables types*/
class IVar
{
public:

	typedef NLLIGO::IPrimitive *	TCtorParam;

	enum TVarType
	{
		vt_integer,
		vt_npc,
		vt_item,
		vt_place,
		vt_text
	};
	
	IVar(TVarType type, NLLIGO::IPrimitive *prim)
	{
		_VarType = type;
		_VarName = getPrimProperty(prim, "var_name");
	}
	
	// force virtual destructor
	virtual ~IVar() {}
	
	/** Return the variable type. Should return a string to 
	 *	limit coupling with implementation ?
	 */
	TVarType getVarType() const
	{
		return _VarType;
	}

	/**	Return the type of the variable as defined in the string manager.
	 *	Can return  NB_ENTITY_TYPES for compiler variable that don't match 
	 *	to a type in the string manager.
	 */
	virtual STRING_MANAGER::TParamType getStringManagerType()=0;

	/// Return the name of the variable.
	const std::string getVarName() const
	{
		return _VarName;
	}
	
	/// Evaluate the content of the variable with an optional sub part.
	virtual std::string evalVar(const std::string &subPart) = 0;
	
	/** Factory method to create new variable. Caller become responsible
	 *	for deleting the allocated variable.
	 */
	static IVar *createVar(CMissionData &md, NLLIGO::IPrimitive *prim);

	/// Generate variable declaration in mission script
	virtual std::string genDecl(CMissionData &md) = 0;

	/// Generate the phrase
	virtual std::string genPhrase()
	{
		static std::string emptyString;
		return emptyString;
	}
	
protected:
	/// Helper function to read a primitive property.
	std::string getPrimProperty(NLLIGO::IPrimitive *prim, const std::string &propName)
	{
		std::string *s;
		if (prim->getPropertyByName(propName.c_str(), s))
			return *s;
		else
			return "";
	}

	/// Helper function to read a primitive array property
	std::vector<std::string> getPrimPropertyArray(NLLIGO::IPrimitive *prim, const std::string &propName)
	{
		std::vector<std::string> *sv;
		if (prim->getPropertyByName(propName.c_str(), sv))
			return *sv;
		else
			return std::vector<std::string>();
	}
protected:
	/// Variable type.
	TVarType			_VarType;
	/// Variable name.
	std::string			_VarName;
};

class CMissionData;


/** Class for text management.
 *	Handle different expression of text such
 *	phrase identifier, reference to text variable
 *	or literal string.
 *	The class also handle the parameter list for
 *	the phrase.
 */
class CPhrase
{
public:
	/// Structure to store phrase parameters infos.
	struct TParamInfo
	{
		/// The name of the parameter
		std::string					ParamName;
		/// Compiler param
		std::string					CompilerParam;
		/// The type of the parameters (as defined in the string manager).
		STRING_MANAGER::TParamType	ParamType;

		TParamInfo()
			: ParamType(STRING_MANAGER::NB_PARAM_TYPES)
		{
		}

		TParamInfo(const std::string &name, STRING_MANAGER::TParamType type, const std::string &compilerParam = "")
			: ParamName(name),
			CompilerParam(compilerParam),
			ParamType(type)
			
		{
		}
	};

	typedef std::vector<std::vector<TParamInfo> >	TPredefParams;

private:

	/** Number of variant for this phrase.
	 *	Zero denote that there is no variant, thus no need to add
	 *	a postfix number after the phrase identifier for literal
	 *	generated phrase.
	 */
	uint32					_NumEntry;

	/** The phrase id. This is the identifier that must be use with the 
	 *	string manager.
	 */
	std::string				_PhraseId;
	/** String literal value. When the user fill a quoted string, then
	 *	this property receive the string value of the quoted string.
	 */
	std::vector<std::string> _PhraseLiterals;
	/** Suffixe for literal string identifier */
	std::string				_Suffixe;
	/// The list of default parameters.
	TPredefParams			_DefaultParams;
	/// The list of additional parameters.
	std::vector<TParamInfo>	_AdditionalParams;
public:
	

	/// init the phrase
	void initPhrase (CMissionData &md,
				NLLIGO::IPrimitive *prim, 
				const std::vector<std::string> &texts, 
				uint32 numEntry = 0, 
				const TPredefParams &predefParams = TPredefParams());

	/// generate the common string script for any script instruction.
	std::string genScript(CMissionData &md);

	/** generate the phrase file content. Return empty string 
	 *	if the phrase is not a literal.
	 */
	std::string genPhrase();

	/// Test if the phrase is empty
	bool isEmpty();

	/// Test if the phrase contains some additionnal parameters
	bool asAdditionnalParams();
};


/* Class for jumps */
struct TJumpInfo
{
	std::string		StepName;
	std::string		JumpName;
	bool			Discardable;

	TJumpInfo(const std::string &stepName, const std::string &jumpName = std::string(), bool discardable = true)
		: StepName(stepName),
		JumpName(jumpName),
		Discardable(discardable)
	{}

	bool operator <(const TJumpInfo &other) const
	{
		return StepName < other.StepName;
	}
};


/* Class for mission information storing */
class CMissionData : public NLMISC::CRefCount
{
public:
	
	CMissionData();
	~CMissionData();


//	void setAlias(const std::string &alias)	{ _Alias = alias; }
//	const std::string &getAlias()			{ return _Alias; }

	void setMissionName(const std::string &missionName);
	const std::string &getMissionName();

	const std::string &getGiverPrimitive()	{ return _GiverPrimitive; }
	void setGiverPrimitive(const std::string &giverPrimitive)	{ _GiverPrimitive = giverPrimitive; }

	const std::string &getGiverName()		{ return _MissionGiver; }
	void setGiverName(const std::string &giverName)	{ _MissionGiver = giverName; }
	
	bool addVariable(NLLIGO::IPrimitive *prim, IVar *var);

	IVar *getVariable(const std::string &varName);
	
	bool addStep(IStep *step);
	void addStepName(const std::string &name, IStep *step) { _StepsByNames[name] = step; }

	IStep *getNextStep(IStep *current);

	IStep *getStepByName(const std::string &stepName);

	std::string generateMissionScript(const std::string &primFileName);

	std::string generatePhraseFile();
	
	std::string generateDotScript();

	void parseMissionHeader(NLLIGO::IPrimitive *prim);
	void initHeaderPhrase(NLLIGO::IPrimitive *prim);
	void parsePrerequisites(NLLIGO::IPrimitive *prim);
	
	std::string replaceVar(NLLIGO::IPrimitive *prim, const std::string &str);
	std::vector<std::string> replaceVar(NLLIGO::IPrimitive *prim, const std::vector<std::string> &strs);

	std::string getProperty(NLLIGO::IPrimitive *prim, const std::string &propertyName, bool replaceVar, bool canFail);
	std::vector<std::string> getPropertyArray(NLLIGO::IPrimitive *prim, const std::string &propertyName, bool replaceVar, bool canFail);


	bool isThereAJumpTo(const std::string &stepName);

	bool isGuildMission() const
	{
		return _Guild;
	}

private:

	std::string genPreRequisites();

	// forbidden copy constructor !
	CMissionData(const CMissionData &other):NLMISC::CRefCount()
	{
		nlstop;
	}

//	std::string		_Alias;	
	std::string		_MissionName;
	std::string		_MissionGiver;
	std::string		_GiverPrimitive;
	bool		_MonoInstance;
	bool		_RunOnce;
	bool		_Replayable;
	bool		_Solo;
	bool		_Guild;
	bool		_NotInJournal;
	bool		_AutoRemoveFromJournal; // When mission ends (fail or success) automatically remove it from the journal
	std::string _MissionCategory;
	uint32		_PlayerReplayTimer;
	uint32		_GlobalReplayTimer;
	bool		_NotProposed;
	bool		_NonAbandonnable;
	bool		_NeedValidation;
	bool		_FailIfInventoryIsFull;
	std::string _MissionIcon;

	std::vector<std::string>	_MissionTitleRaw;
	CPhrase						_MissionTitle;
	std::vector<std::string>	_MissionDescriptionRaw;
	CPhrase						_MissionDescription;

	bool						_MissionAuto;
	std::vector<std::string>	_MissionAutoMenuRaw;
	CPhrase						_MissionAutoMenu;

	////// Pre requisites /////////
	struct TReqSkill
	{
		std::string		Skill;
		std::string		MinLevel;
		std::string		MaxLevel;
	};

	struct TReqFame
	{
		std::string		Faction;
		std::string		Fame;
	};
	std::vector<TReqSkill>		_ReqSkills;
	std::vector<std::string>	_ReqMissionDone;
	std::vector<std::string>	_ReqMissionNotDone;
	std::vector<std::string>	_ReqMissionRunning;
	std::vector<std::string>	_ReqMissionNotRunning;
	std::vector<std::string>	_ReqWearItem;
	std::vector<std::string>	_ReqOwnItem;
	std::string					_ReqTitle;
	std::vector<TReqFame>		_ReqFames;
	bool						_ReqGuild;
	std::string					_ReqGrade;
	std::string					_ReqTeamSize;
	std::vector<std::string>	_ReqBrick;
	std::string					_ReqCharacterAge;
	std::string					_ReqMaxPlayerID;
	std::string					_ReqSeason;
//	bool						_ReqEncycloTasksDone;
	std::string					_ReqEncyclo;
	std::string					_ReqEncycloNeg;
	std::string					_ReqEventFaction;

	/// The list of parent missions
	std::set<std::string>				_ParentMissions;
	/// The list of variable by name
	std::map<std::string, IVar*>		_Variables;
	/// The list of variable in primitive order
	std::vector<IVar*>					_VariablesOrder;
	/// the list of step in execution order
	std::vector<IStep*>					_Steps;
	/// The list of step sorted by step name
	std::map<std::string, IStep *>		_StepsByNames;

	std::set<TJumpInfo>	_JumpPoints;

};

typedef NLMISC::CSmartPtr<CMissionData>	TMissionDataPtr;

/** This class manage the compilation of missions contained under a 
 *	primitive file node.
 */
class CMissionCompiler
{
public:

	/** Generate the dot language script for the missions under the specified node.
	 *	This method is primarily used for world editor mission display.
	 */
//	std::vector<std::string>	generateDotScript(NLLIGO::NLLIGO::IPrimitive *rootPrim);
	bool generateDotScript(NLLIGO::IPrimitive *missionPrim, std::string &dotScript, std::string &log);

	/** compile one mission. The primitive node must be
	 *	a mission tree root node.
	 *	fileName must receive the primitive file name. It is used
	 */
	bool compileMission(NLLIGO::IPrimitive *rootPrim, const std::string &primFileName);

	/** Compile all the missions found under the given primitive node.
	 *	The primitive tree is searched recursively ti find all the 
	 *	mission tree nodes.
	 *	All the compiled missions are stored internaly in precompiled form.
	 */
	bool compileMissions(NLLIGO::IPrimitive *rootPrim, const std::string &primFileName);

	/** Install the generated script into the destination primitive files */
	bool installCompiledMission(NLLIGO::CLigoConfig &ligoConfig, const std::string &primFileName);

	/// Publish the modified to the path parameter
	bool publishFiles(const std::string &serverPathPrim, const std::string &serverPathText, const std::string &localPathText);

	/// Search for text in the file : add it if it's not in
	bool includeText(const std::string filename, const std::string text);
	
	/// Parse the pre requisite node of a mission.
	bool parsePreRequisite(CMissionData &md, NLLIGO::IPrimitive *preReq);

	/// Parse the steps of a missions.
	bool parseSteps(CMissionData &md, NLLIGO::IPrimitive *steps, IStep *parent=NULL);
	bool parseOneStep(CMissionData &md, NLLIGO::IPrimitive *stepToParse, IStep *parent, bool bEndOfBranch);

	/// Helper to retrive a property in a primitive node.
	std::string getProp(NLLIGO::IPrimitive *prim, const std::string &propName);
	/// Helper to retreive the class name of a primitive node.
	std::string getClass(NLLIGO::IPrimitive *prim);
	/// Parse the variable of a missions.
	bool parseVariables(CMissionData &md, NLLIGO::IPrimitive *variables);


	/// Get full paths of files to publish
	uint getFileToPublishCount() { return (uint)_FilesToPublish.size(); }
	std::string getFileToPublish(uint index) { nlassert(index < _FilesToPublish.size()); return _FilesToPublish[index]; }

	
	std::vector <TMissionDataPtr> &getMissions()
	{
		return _CompiledMission;
	}
	uint getMissionsCount()
	{
		return (uint)_CompiledMission.size();
	}
	TMissionDataPtr	getMission(uint index)
	{
		nlassert(index < _CompiledMission.size());
		return _CompiledMission[index];
	}

	
private:

	/// Storage for loaded primitive
	struct TLoadedPrimitive
	{
		NLLIGO::CPrimitives	*PrimDoc;
		std::string			FullFileName;

		TLoadedPrimitive()
			: PrimDoc(NULL)
		{}
		
		TLoadedPrimitive(NLLIGO::CPrimitives *primDoc, const std::string &fullFileName)
			: PrimDoc(primDoc),
			FullFileName(fullFileName)
		{
		}
	};

	/// Storage for precompiled missions.
	std::vector <TMissionDataPtr>	_CompiledMission;

	/// Storage for files to publish
	std::vector<std::string>		_FilesToPublish;
};


// Class to easily handle var and var name pair
struct TCompilerVarName
{
	typedef std::vector <std::vector<struct TCompilerVarName> > TPredefParams;
	// Default parameter name
	std::string			_DefaultName;
	// Type of script parameter 
	STRING_MANAGER::TParamType	_ParamType;
	// the name of the compiler var, like 'the_creature' in $the_creature$ = chab1
	std::string	_VarName;
	// the value of the compiler var line 'chab1' in $the_creature$ = chab1
	std::string	_VarValue;

	void init(const std::string &defaultName, STRING_MANAGER::TParamType type, CMissionData &md, NLLIGO::IPrimitive *prim, const std::string propName);

	void initWithText(const std::string &defaultName, STRING_MANAGER::TParamType type, CMissionData &md, NLLIGO::IPrimitive *prim, const std::string &text);	
	
	CPhrase::TParamInfo getParamInfo() const;

	bool empty() const;

	operator const std::string  () const;

	operator CPhrase::TParamInfo() const;

	//static void getPhrasePredef(const TCompilerVarName::TPredefParams& compilerParams, CPhrase::TPredefParams& params);	
	
	static std::vector<TCompilerVarName> getPropertyArrayWithText(const std::string &defaultName, STRING_MANAGER::TParamType type, CMissionData &md, NLLIGO::IPrimitive *prim, const std::string & arrayProperyName);
	static std::vector<TCompilerVarName> getPropertyArrayWithTextStaticDefaultName(const std::string &defaultName, STRING_MANAGER::TParamType type, CMissionData &md, NLLIGO::IPrimitive *prim, const std::string & arrayProperyName);


};

std::string operator+(const TCompilerVarName& left, const std::string & right);
std::string operator+(const std::string & left, const TCompilerVarName& right);