package analyseplans;

import static org.codelutin.i18n.I18n._;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;

import scripts.ResultName;

import java.io.*;
import java.io.File;
import java.io.FileReader;
import java.io.Writer;
import java.util.*;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Properties;

import org.codelutin.math.matrix.*;
import org.codelutin.util.*;
import org.codelutin.topia.*;

import fr.ifremer.isisfish.*;
import fr.ifremer.isisfish.types.*;
import fr.ifremer.isisfish.rule.Rule;
import fr.ifremer.isisfish.rule.RuleHelper;
import fr.ifremer.isisfish.simulator.SimulationContext;
import fr.ifremer.isisfish.types.Date;
import fr.ifremer.isisfish.entities.*;
import fr.ifremer.isisfish.simulator.AnalysePlan;
import fr.ifremer.isisfish.simulator.AnalysePlanContext;
import fr.ifremer.isisfish.simulator.SimulationParameter;
import fr.ifremer.isisfish.datastore.RegionStorage;
import fr.ifremer.isisfish.datastore.RuleStorage;
import fr.ifremer.isisfish.datastore.SimulationStorage;
import fr.ifremer.isisfish.datastore.ResultStorage;

/**
 * Sensibility.java
 *
 * Created: 21 décembre 2008
 *
 * @author HUSSEIN chirine <m475chi@yahoo.fr>
 *
 */
public class Sensibility implements AnalysePlan {

    /** to use log facility, just put in your code: log.info("..."); */
    static private Log log = LogFactory.getLog(Sensibility.class);

    static private final String MATRIX = "matrix"; /// Le fichier est donc matrix.txt
    static private final String VBGF = "growth";
    static private final String SELECTIVITY = "selectivity";
	static private final String SELECTIVITYFiletMer = "selectivityFiletMer";
	static private final String SELECTIVITYFiletEtang = "selectivityFiletEtang";
	static private final String SELECTIVITYLigncote = "selectivityLignecote";
	static private final String SELECTIVITYLigneemb = "selectivityLigneemb";
	static private final String SELECTIVITYChalut = "selectivityChalut";
	static private final String SELECTIVITYHarpon = "selectivityHarpon";
    static private final String CATCHABILITY = "capturabilite";
    static private final String MORTALITY = "mortalite";
    static private final String STANDARDISATION = "standardisation";
	//static private final String AMP = "amp";
    //static private final String AMP1 = "ampRPrepro_Lignecote";
	//static private final String AMP2 = "ampRPrepro_FiletMer"; 
	///////////* completer ici avec tous les fichiers .txt des amp
	
public int param_parameterNumber = 5; /// ce paramètre n'est normalement pas modifié
    public int param_first = 0; /// on utilise ce paramètre pour indiquer à quelle ligne 
		/// de la matrice on doit commencer à faire tourner les simulations au cas où on ait dû interompre le plan
    public int param_simulationNumber = 32;/// ce paramètre n'est normalement pas modifié 
    public String param_directory = "C:/Documents and Settings/hussein/Bureau/isis-fish-3.1.3/isis-fish-3.1.3/fichier paramètres//"; /// Il s'agit du chemin vers le dossier 
	///contenant les fichier de matrice et paramètres, par défaut le repertoire du go.bat 
	/// donc dans ce cas c'est le dossier Exemple_repertory qui est dans le dossier contenant le go.bat
  
/// Déclaration de la matrice d'expérience 

 private MatrixND matrix = null;

/// Non utilisé pour ce script mais à ne pas effacer
    public String [] necessaryResult = {
    // put here all necessary result for this rule
    // example: 
    // ResultName.MATRIX_BIOMASS,
    // ResultName.MATRIX_NET_VALUE_OF_LANDINGS_PER_STRATEGY_MET,
    };

    public String[] getNecessaryResult() {
        return this.necessaryResult;
    }


    /**
     * Permet d'afficher a l'utilisateur une aide sur le plan.
     * @return L'aide ou la description du plan
     */
    public String getDescription() throws Exception {
       
        return _("réalise les expériences d'un plan complet pour 5 parametres à 2 modalités");
    }
 
    /**
     * Appel? au d?marrage de la simulation, cette m?thode permet d'initialiser
     * des valeurs
     * @param simulation La simulation pour lequel on utilise cette regle
     */
    public void init(AnalysePlanContext context) throws Exception {
/// Création de la matrice d'expérience et chargement :
File dir = new File(param_directory);

File csvFile = new File(dir, MATRIX + ".txt");
log.info("CSV file is : " + csvFile.getAbsolutePath());
if(!csvFile.exists()) {
log.warn("File doesn't exists");
}

matrix = MatrixFactory.getInstance().create(new int[]{param_simulationNumber, param_parameterNumber});
matrix.importCSV(new FileReader(csvFile), new int[]{0,0});
matrix.setSemantics(1, Arrays.asList(new String[]{VBGF,CATCHABILITY,MORTALITY,SELECTIVITY, STANDARDISATION}));
System.out.println(matrix);
//System.out.println("amp 0 : "+ matrix.getValue(0,AMP));
/// ajout des regles qui sont modifiées par le plan
//context.getParam().addExtraRules("Cantonnement","Cantonnement","Cantonnement","Cantonnement","Cantonnement","Cantonnement");
		////////////* tu dois ecrire cantonnement 11 fois si tu as 11 regles cantonnement
}

         /// Création des méthodes qui réccupère dans la matrice la modalité du paramètre pour l'expérience en cours 
		/// Les arguments des méthodes sont le nom du fichier et le numéro de la simulation
		/// (On détermine l'expérience en cours et donc la ligne de la matrice en sommant le numéro de la simulation et le param_first)
		/// pour un double  

 private double getDouble(String name, int simulation) throws Exception {
        File dir = new File(param_directory);
        Properties prop = new Properties();
        prop.load(new BufferedReader(new FileReader(new File(dir, name + ".txt"))));        
        int ligne = simulation + param_first;
		int mod = (int)matrix.getValue(ligne, name);
System.out.println("mod et ligne : "+ mod+" "+ligne);
        double result = Double.parseDouble(prop.getProperty(""+mod));
System.out.println("result : "+ result);
        return result;
    }
	
		/// pour une liste de doubles

 /**
     * @param name le nom de l'element a recuperer
     * @param simulation le numero de la simulation
     * @return
     */   
  private double [] getList(String name, int simulation) throws Exception {
        File dir = new File(param_directory);
        Properties prop = new Properties();
        prop.load(new BufferedReader(new FileReader(new File(dir, name + ".txt"))));
		int ligne = simulation + param_first;
        int mod = (int)matrix.getValue(ligne, name);
        double [] result = StringUtil.toArrayDouble(prop.getProperty(""+mod).split(";")); // ""+ ca construit une chaine de caractere
        return result;
    }
	
		/// pour une chaine de characteres (equation)
    /**
     * @param name le nom de l'element a recuperer
     * @param simulation le numero de la simulation
     * @return
     */
 private String getString(String name,String col, int simulation) throws Exception {
        File dir = new File(param_directory);
        Properties prop = new Properties();
        prop.load(new BufferedReader(new FileReader(new File(dir, name + ".txt"))));
		int ligne = simulation + param_first;
        int mod = (int)matrix.getValue(ligne , col);
        String result = prop.getProperty(""+mod);
        return result;
    }
		/// pour les paramètres d'une règle de gestion
	  /**
     * @param name le nom de l'element a recuperer
     * @param simulation le numero de la simulation
     * @return
     */	
   private Rule getRule(RegionStorage regionStorage, String name, String col, int simulation) throws Exception {
        File dir = new File(param_directory);
        Properties prop = new Properties();
        prop.load(new BufferedReader(new FileReader(new File(dir, name+".txt"))));
		int ligne = simulation + param_first;
        int mod = (int)matrix.getValue(ligne , col);
        String ruleName = prop.getProperty(""+mod);
        RuleStorage ruleStorage = RuleStorage.getRule(ruleName);
        Rule rule = ruleStorage.getNewRuleInstance();
        RuleHelper.populateRule(mod, regionStorage, rule, prop); 
        return rule;
    }
    
	
	
    /**
     * Call before each simulation
     * @param context plan context
     * @param nextSimulation storage used for next simulation
     * @return true if we must do next simulation, false to stop plan
     * @throws Exception
     */
    public boolean beforeSimulation(AnalysePlanContext context, SimulationStorage nextSimulation) throws Exception {
        
		int simNum = context.getNumber()+ param_first; 
        
		if (simNum < param_simulationNumber+1) { /// On vérifie qu'il reste des expériences à faire

			/// On utilise les méthodes crées pour récupérer les valeurs des modalités pour l'expérience en cours
            double [] croissance    = getList (VBGF , simNum);
            double catchability     = getDouble (CATCHABILITY, simNum);
            double [] standardisation    = getList (STANDARDISATION , simNum);
			String selectivity      = getString (SELECTIVITY, simNum);
			String selectivityFiletMer      = getString (SELECTIVITYFiletMer,SELECTIVITY, simNum);
			String selectivityFiletEtang      = getString (SELECTIVITYFiletEtang,SELECTIVITY, simNum);
			String selectivityLignecote      = getString (SELECTIVITYLignecote,SELECTIVITY, simNum);
			String selectivityLigneemb      = getString (SELECTIVITYFLigneemb,SELECTIVITY, simNum);
			String selectivityChalut      = getString (SELECTIVITYChalut,SELECTIVITY, simNum);
			String selectivityHarpon      = getString (SELECTIVITYHarpon,SELECTIVITY, simNum);
            String mortality      = getString (MORTALITY,MORTALITY, simNum);
			//Rule amp1 = getRule(context.getParam().getRegion(), AMP1, AMP, simNum);
			//Rule amp2 = getRule(context.getParam().getRegion(), AMP2, AMP, simNum);
			///////////////* à completer avec le reste des amp*/
			
			/// On peut afficher des informations dans les logs
            nextSimulation.getInformation().addInformation("Growth ("+context.getNumber()+")= " + croissance);
            nextSimulation.getInformation().addInformation("Selectivity ("+context.getNumber()+")= " + selectivity);


			/// Il faut maintenant modifier la valeur des paramètres dans la base de données et les paramètres des règle de la simulation
				/// modif les parametres des regles de gestion :
            
			// on enleve la regle de la simulation passé (elle est stoquée sous le nom LastAMP) 
            //List<Rule> paramRules = nextSimulation.getParameter().getRules();
            
			//paramRules.remove(context.getValue("lastAMP1"));
            // on nomme la regle que l'on va ajouter LastAMP (pour pouvoir la retrouver a la simulation suivante)
			//context.setValue("lastAMP1", amp1);
            // on ajoute la regle à la simulation
           // paramRules.add(amp1);
			
			//paramRules.remove(context.getValue("lastAMP2"));
			//context.setValue("lastAMP2", amp2);
            //paramRules.add(amp2);
			/////////////////* à completer avec les autres amp */
            
			
				/// modif les parametres dans la base de données :
            TopiaContext tx = nextSimulation.getStorage().beginTransaction();
            /// On récupère les objets ISIS a modifier dans la base de données
			PopulationDAO popDAO = IsisFishDAOHelper.getPopulationDAO(tx);			
			Population pop = popDAO.findByName("CSar");
            List<PopulationGroup> groups = pop.getPopulationGroup();
			GearDAO gearDAO = IsisFishDAOHelper.getGearDAO(tx);
            List<Gear> gears = gearDAO.findAll();

		



					///modif borne des classes de taille
					/// On remplace les bornes par les éléments de la liste
            for (PopulationGroup group : groups) {
                group.setMinLength(croissance[group.getId()]) ;
                group.setMaxLength(croissance[group.getId()+1]) ;
            }
            
					
					// modif la capturabilité
					/// On modifie la valeur de la base en la multipliant par la valeur contenue dans la modalité
		    MatrixND c = pop.getCapturability();
		    for (MatrixIterator i = c.iterator(); i.hasNext();){
			    i.next();
        		i.setValue(catchability);
            }
					 

					 // modif la mortalité naturelle
					/// On modifie la valeur de la base en la remplacant par la valeur contenue dans la modalité
		  
		    for (PopulationGroup group : groups) {
              Equation eq = pop.getNaturalDeathRate();
              eq.setContent(mortality);
            }

		
    
                     // modif les facteurs de standardisation
					/// On modifie la valeur de la base en la remplacant par la valeur contenue dans la modalité

		    for (Gear gear : gears) {
                 
                 if ("FiletMer".equals(gear.getName())) {
					gear.setStandardisationFactor(standardisation[0]);
				 }else if("Lignecote".equals(gear.getName())) {
					gear.setStandardisationFactor(standardisation[1]);
				}else if("Harpon".equals(gear.getName())) {
					gear.setStandardisationFactor(standardisation[2]);
				}else if("chalut".equals(gear.getName())) {
					gear.setStandardisationFactor(standardisation[3]);
				}else if("Ligneemb".equals(gear.getName())) {
					gear.setStandardisationFactor(standardisation[4]);
				}else if("FiletEtang".equals(gear.getName())) {
					gear.setStandardisationFactor(standardisation[5]);
/////////////*a completer
				}
		}	


					/// modif selectivité
					/// On remplace l'equation existante par une autre equation
          for (Gear gear : gears) {

				 if ("FiletMer".equals(gear.getNam())) {
                Selectivity sel = gear.getPopulationSelectivity(pop);
                Equation eq = sel.getEquation();                
				eq.setContent(selectivityFiletMer);
			} else  if ("FiletEtang".equals(gear.getNam())) {
               Selectivity sel = gear.getPopulationSelectivity(pop);
                Equation eq = sel.getEquation();                
				eq.setContent(selectivityFiletEtang);
			}else  if ("Lignecote".equals(gear.getNam())) {
                Selectivity sel = gear.getPopulationSelectivity(pop);
                Equation eq = sel.getEquation();                
				eq.setContent(selectivityLignecote);
             }else  if ("Ligneemb".equals(gear.getNam())) {
                Selectivity sel = gear.getPopulationSelectivity(pop);
                Equation eq = sel.getEquation();                
				eq.setContent(selectivityLigneemb);
			}else  if ("Chalut".equals(gear.getNam())) {
                Selectivity sel = gear.getPopulationSelectivity(pop);
                Equation eq = sel.getEquation();                
				eq.setContent(selectivityChalut);
			}else  if ("Harpon".equals(gear.getNam())) {
                Selectivity sel = gear.getPopulationSelectivity(pop); 
                Equation eq = sel.getEquation();                
				eq.setContent(selectivityHarpon);
         }   
			tx.commitTransaction();
            return true;

        } else {
            return false;
        }
}


 
    /**
     * Call before each simulation
     * @param context plan context
     * @param nextSimulation storage used for next simulation
     * @return true if we must do next simulation, false to stop plan
     * @throws Exception
     */
 	
	 public boolean afterSimulation(AnalysePlanContext context, SimulationStorage lastSimulation) throws Exception {
        return true;
    }

}
