from pathlib import Path
import json
import os
from pandas import DataFrame, read_csv

# from util.interface import RulesDict

RulesDict = dict[str, list[dict[str, bool]]]

class Scenario:
    @classmethod
    def get_index(cls, scenario: str) -> int:
        """
        Get the index of scenario stored in "../database/scenario.csv".
        Ralated files of this scenario is stored in f"../database/{index}".
        """
        if scenario is None:
            raise ValueError("scenario can not be None.")
        database_path = Path(__file__).resolve().parent.parent / 'database'
        database_path.mkdir(parents=True, exist_ok=True)
        file_path = database_path / 'scenario.csv'
        index_df = read_csv(file_path, index_col=0) if file_path.exists() else \
                        DataFrame(columns=["scenario_id", "scenario"])
        scenario_index = -1
        if scenario in index_df["scenario"].values:
            row_index = list(index_df["scenario"].values).index(scenario)
            scenario_index = index_df.loc[row_index]["scenario_id"]
        else:
            n = len(index_df)
            index_df.loc[n] = {"scenario": scenario, "scenario_id": n}
            index_df.to_csv(file_path)
            scenario_index = n
        return scenario_index
    
    @classmethod
    def get_scenario(cls, scenario_index: int) -> str:
        """
        Get scenario from its id.
        """
        file_path = Path(__file__).resolve().parent.parent / 'database' / 'scenario.csv'
        index_df = read_csv(file_path, index_col=0) if file_path.exists() else \
                        DataFrame(columns=["scenario_id", "scenario"])
        if len(index_df) <= scenario_index:
            raise ValueError(f"Scenario does not exist for id {scenario_index}.")
        return index_df.loc[scenario_index]["scenario"]
    
    @classmethod
    def get_all_scenarios(cls) -> list[str]:
        """
        Get all scenarios from "../database/scenario.csv".
        Return a list of scenario strings.
        """
        file_path = Path(__file__).resolve().parent.parent / 'database' / 'scenario.csv'
        index_df = read_csv(file_path, index_col=0) if file_path.exists() else \
                        DataFrame(columns=["scenario_id", "scenario"])
        return list(index_df["scenario"].values)
    
    @classmethod
    def set_basic_data(cls, 
        scenario: str | int,
        roots: list[str],
        non_roots: list[str], 
        rules: RulesDict, 
        llm_name="sample"):
        """
        Write the information of roots, non-roots, and rules to 
            f"../database/{index}/basic_info.json", where index refers
            to the index of scenario, stored in ../database/scenario.csv.
        The scenario parameter can be both string or integer. If it is a
            string, it represents the scenario; if it is an integer, it
            represents the index of the scenario.
        """
        index = cls.get_index(scenario) if isinstance(scenario, str) else scenario
        folder_path = Path(__file__).resolve().parent.parent / 'database' / llm_name / str(index)
        folder_path.mkdir(parents=True, exist_ok=True)
        os.makedirs(folder_path, exist_ok=True)
        file_path = folder_path / 'basic_info.json'
        data = {
            "roots": roots,
            "non_roots": non_roots,
            "rules": rules
        }
        with open(file_path, "w") as basic_info_file:
            json.dump(data, basic_info_file)
    
    @classmethod
    def get_basic_data(cls, scenario: str | int, llm_name="sample") -> tuple[list[str], list[str], RulesDict]:
        """
        Read the information of roots, non-roots, and rules from 
            f"../database/{index}/basic_info.json", where index refers
            to the index of scenario, stored in ../database/scenario.csv.
        The scenario parameter can be both string or integer. If it is a
            string, it represents the scenario; if it is an integer, it
            represents the index of the scenario.
        Return a tuple of (roots, non_roots, rules).
        Usage example:
        scenario = "test scenario"
        roots, non_roots, rules = Scenario.get_basic_data(scenario)
        """
        index = cls.get_index(scenario) if isinstance(scenario, str) else scenario
        folder_path = Path(__file__).resolve().parent.parent / 'database' / llm_name / str(index)
        folder_path.mkdir(parents=True, exist_ok=True)
        file_path = folder_path / 'basic_info.json'
        with open(file_path, "r") as basic_info_file:
            data = json.load(basic_info_file)
        rules = data["rules"]
        for non_root, rule_term in rules.items():
            rules[non_root] = tuple(rule_term)
        return data["roots"], data["non_roots"], rules
    
    @classmethod
    def add_scenario_from_json_v0(cls, json_file_path: str):
        with open(json_file_path, "r") as json_file:
            data = json.load(json_file)
        scenario = data["//"]
        roots, non_roots = [], []
        rules = {}
        all_vars = []
        for name, rule_s in data.items():
            if name == "//":
                continue
            non_roots.append(name)
            rule = []
            for rule_term_s in rule_s:
                rule_term = {}
                for describe in rule_term_s:
                    if describe[:5] == "<neg>":
                        rule_term[describe[5:]] = False
                    else:
                        rule_term[describe] = True
                all_vars.extend(rule_term.keys())
                rule.append(rule_term)
            rules[name] = tuple(rule)
        for name in list(set(all_vars)):
            if name not in non_roots:
                roots.append(name)
        cls.set_basic_data(scenario=scenario, roots=roots, non_roots=non_roots, rules=rules)

    @classmethod
    def add_scenario_from_json(cls, json_file_path: str):
        stem = Path(json_file_path).stem
        with open(json_file_path, "r") as json_file:
            data = json.load(json_file)
        scenario = stem + "%" + data["scenario"]
        roots = data["roots"]
        non_roots = data["non_roots"]
        rules = data["rules"]
        cls.set_basic_data(scenario=scenario, roots=roots, non_roots=non_roots, rules=rules)
                    
