from typing import Any, Callable, Dict, List, Optional

import json_repair


class OutputStringParser:
    @staticmethod
    def parse_dict(
        content: str,
        expected_keys: List[str] = None,
        value_converter: Callable[[Any], Any] | dict[str, Callable] = None,
        expected_value_types: Dict[str, type] = None,
        validator: Callable[[Dict[str, Any]], None] = None,
        fallback: Any = {},
    ) -> tuple[Dict[str, Any], Any]:
        """
        Parses a string content into a dictionary with optional validation and conversion.

        check expected keys -> convert values -> check expected types -> custom validation

        Args:
            content (str): The string content to parse as JSON.
            expected_keys (List[str], optional): List of keys that must be present in the parsed dictionary.
            value_converter (Callable[[Any], Any] | dict[str, Callable], optional): Function or mapping to apply to all values in the dictionary.
            expected_value_types (Dict[str, type], optional): Dictionary mapping keys to their expected types.
            validator (Callable[[Dict[str, Any]], None], optional): Custom validation function for the parsed dictionary.
            fallback (Any, optional): Value to return if parsing or validation fails. Defaults to {}.

        Returns:
            tuple: (parsed dictionary or fallback value, fail_reason or None)
        """
        fail_reason = None
        try:
            solution = json_repair.loads(content)
            assert isinstance(solution, dict), (
                f"Expected parsed content to be a dictionary, got {type(solution)}"
            )

            # Check for required keys
            if expected_keys:
                for key in expected_keys:
                    assert key in solution, (
                        f"Required key '{key}' not found in parsed dictionary"
                    )

            # Apply value conversion to all values
            if value_converter:
                if callable(value_converter):
                    solution = {k: value_converter(v) for k, v in solution.items()}
                elif isinstance(value_converter, dict):
                    for k, converter in value_converter.items():
                        if k in solution:
                            solution[k] = converter(solution[k])

            # Check expected types
            if expected_value_types:
                for key, expected_type in expected_value_types.items():
                    assert isinstance(solution[key], expected_type), (
                        f"Expected key '{key}' to be of type {expected_type}, got {type(solution[key])}"
                    )

            # Apply custom validation
            if validator:
                validator(solution)

            return solution, fail_reason
        except Exception as e:
            fail_reason = str(e)
            return fallback, fail_reason

    @staticmethod
    def parse_list(
        content: str,
        expected_length: Optional[int] = None,
        item_converter: Optional[Callable[[Any], Any]] = None,
        expected_item_type: Optional[type] = None,
        validator: Optional[Callable[[List[Any]], None]] = None,
        fallback: Any = [],
    ) -> tuple[List[Any], Any]:
        """
        Parses a string content into a list with optional validation and conversion.

        Args:
            content (str): The string content to parse as JSON.
            expected_length (int, optional): Expected length of the parsed list.
            item_converter (Callable[[Any], Any], optional): Function to apply to all items in the list.
            expected_item_type (type, optional): Expected type for all items in the list.
            validator (Callable[[List[Any]], None], optional): Custom validation function for the parsed list.
            fallback (Any, optional): Value to return if parsing or validation fails. Defaults to [].

        Returns:
            tuple: (parsed list or fallback value, fail_reason or None)
        """
        fail_reason = None
        try:
            solution = json_repair.loads(content)
            assert isinstance(solution, list), (
                f"Expected parsed content to be a list, got {type(solution)}"
            )

            # Check expected length
            if expected_length is not None:
                assert len(solution) == expected_length, (
                    f"Expected list length to be {expected_length}, got {len(solution)}"
                )

            # Apply item conversion to all items
            if item_converter:
                solution = [item_converter(item) for item in solution]

            # Check expected item types
            if expected_item_type:
                for item in solution:
                    assert isinstance(item, expected_item_type), (
                        f"Expected list item to be of type {expected_item_type}, got {type(item)}"
                    )

            # Apply custom validation
            if validator:
                validator(solution)

            return solution, fail_reason
        except Exception as e:
            fail_reason = str(e)
            return fallback, fail_reason
