import random
from typing import List, Dict, Any


def normalize_to_pddl_name(human_name: str) -> str:
    """
    Converts a human-readable object name into a PDDL-compatible format.
    e.g., "Personal journal" -> "personal_journal.n.01"
    """
    if not human_name:
        return ""
    normalized = human_name.strip().lower().replace(" ", "_")

    if not normalized.endswith(".n.01"):
        return f"{normalized}.n.01"
    return normalized


def parse_json_to_pddl_objects_config(
    json_data: List[Dict[str, Any]],
) -> Dict[str, Dict[str, Any]]:
    """
    Parses a list of dictionaries from the benchmark JSON format to create
    an objects_config dictionary compatible with the PDDL generator.

    It primarily processes the 'physical objects' field.
    """
    parsed_config: Dict[str, Dict[str, Any]] = {}
    for entry in json_data:
        physical_objects_str = entry.get("physical objects", "")
        if not physical_objects_str:
            continue

        first_object_raw = physical_objects_str.split(",")[0].strip()
        if not first_object_raw:
            continue

        pddl_object_name = normalize_to_pddl_name(first_object_raw)
        if not pddl_object_name:
            continue

        if pddl_object_name not in parsed_config:
            parsed_config[pddl_object_name] = {
                "count": 1,
                "is_furniture": False,
                "ontop_sources": ["table.n.02", "desk.n.01", "shelf.n.01"],
                "inside_sources": ["cabinet.n.01", "cupboard.n.01", "dresser.n.01"],
            }
        else:

            parsed_config[pddl_object_name]["count"] += 1
    return parsed_config


def merge_objects_configs(
    base_config: Dict[str, Any], additional_config: Dict[str, Any]
) -> Dict[str, Any]:
    """
    Merges two objects_config dictionaries.

    It starts with a copy of the base_config and then adds or updates
    entries from the additional_config. If an object type exists in both,
    their counts are summed.
    """
    merged_config = {k: v.copy() for k, v in base_config.items()}
    for obj_type, properties in additional_config.items():
        if obj_type in merged_config:

            merged_config[obj_type]["count"] = merged_config[obj_type].get(
                "count", 0
            ) + properties.get("count", 1)
        else:

            merged_config[obj_type] = properties
    return merged_config


def create_reasoning_prompt(
    original_prompt: str, model_answer: str
) -> list[dict[str, str]]:
    """
    Creates a conversational prompt to ask for the reasoning behind a model's answer.
    """
    return [
        {"role": "user", "content": original_prompt},
        {"role": "assistant", "content": model_answer},
        {
            "role": "user",
            "content": "Please explain your reasoning for this answer. Provide a step-by-step thinking process that led to your decision.",
        },
    ]


def find_tript(action_ratings: List[Dict[str, Any]]) -> List[List[str]]:
    """
    Finds unique combinations (tripts) of actions based on a tiered priority system.
    A tript is only considered valid if its highest-rated action has a rating of 4 or 5.

    Priority 1: Hard Positive (5), Neutral (3), Hard Negative (1)
    Priority 2: Hard Positive (5), Intermediate (2 or 4), Hard Negative (1)


    Args:
        action_ratings: A list of dictionaries, where each dictionary
                        contains 'action' (str) and 'expected_rating' (int).

    Returns:
        A list of lists, where each inner list is a shuffled tript.
        Returns an empty list if no valid tript can be formed.
    """

    actions_by_rating = {i: [] for i in range(1, 6)}
    for a in action_ratings:
        rating = a.get("expected_rating")
        if rating in actions_by_rating:
            actions_by_rating[rating].append(a["action"])

    tripts = _generate_tript_combinations(
        actions_by_rating[5], actions_by_rating[3], actions_by_rating[1]
    )
    if tripts:
        return tripts

    intermediate_actions = actions_by_rating[2] + actions_by_rating[4]
    tripts = _generate_tript_combinations(
        actions_by_rating[5], intermediate_actions, actions_by_rating[1]
    )
    if tripts:
        return tripts

    return []


def _generate_tript_combinations(
    list_a: List[str], list_b: List[str], list_c: List[str]
) -> List[List[str]]:
    """Helper to generate unique, shuffled tripts from three lists of actions."""
    if not all([list_a, list_b, list_c]):
        return []

    tripts = set()
    for a in list_a:
        for b in list_b:
            for c in list_c:

                if a != b and a != c and b != c:
                    tript_list = [a, b, c]
                    random.shuffle(tript_list)
                    tripts.add(frozenset(tript_list))

    return [list(t) for t in tripts]
