import re
import json
from abc import ABC, abstractmethod
from typing import Dict, Any, Tuple, Optional, List


def extract_tool_call_contents(start_token, end_token, text):
    pattern = re.escape(start_token) + r'(.*?)' + re.escape(end_token)
    matches = re.findall(pattern, text, re.DOTALL)
    return matches


class ToolMeta(type):
    def __init__(cls, name, bases, attrs):
        super().__init__(name, bases, attrs)
        if name == 'ToolBase':
            return

        if not hasattr(cls, 'name'):
            raise AttributeError(f"Tool subclass {name} must define a 'name' attribute.")

        if cls.name in ToolBase.registry:
            existing = ToolBase.registry[cls.name]
            print(
                f" [WARNING] Class {cls.__name__} is trying to register '{cls.name}', which has already been registered by {existing.__name__}")
            return

        ToolBase.registry[cls.name] = cls


class ToolBase(metaclass=ToolMeta):
    registry = {}

    def __init__(self, name: str, description: str = '', parameters: Dict = {}, **kwargs):

        self.name = name
        self.description = description
        self.parameters = parameters or {
            "type": "object",
            "properties": {},
            "required": []
        }

    @classmethod
    def create(cls, name, description='', parameters=[], **kwargs):
        tool_cls = cls.registry.get(name, None)
        if not tool_cls:
            raise ValueError(f"No tool registered with name '{name}'")
        return tool_cls(name, description, parameters, **kwargs)

    def get_description(self) -> Dict:
        function_desc = {
            "name": self.name,
            "description": self.description,
            "parameters": self.parameters,
        }
        return {"type": "function", "function": function_desc}

    def get_json_description(self) -> str:
        return json.dumps(self.get_description(), indent=4, ensure_ascii=False)

    @abstractmethod
    def reset(self, *args, **kwargs):
        pass

    @abstractmethod
    def execute(self, *args, **kwargs) -> str:
        pass

    def validate_args(self, args: Dict) -> Tuple[bool, str]:
        if not isinstance(args, dict):
            return False, "Arguments must be a dictionary"

        required_params = self.parameters.get("required", [])
        for param in required_params:
            if param not in args:
                return False, f"Missing required parameter: {param}"

        properties = self.parameters.get("properties", {})
        for param_name, param_value in args.items():
            if param_name in properties:
                param_schema = properties[param_name]

                param_type = param_schema.get("type")
                if param_type and not self._check_type(param_value, param_type):
                    return False, f"Parameter {param_name} has incorrect type, should be {param_type}"

                if "enum" in param_schema and param_value not in param_schema["enum"]:
                    valid_values = ", ".join(map(str, param_schema["enum"]))
                    return False, f"Parameter {param_name} has invalid value, should be one of: {valid_values}"

        return True, "Parameters valid"

    def check_type(self, value: Any, expected_type: str) -> bool:
        if expected_type == "string":
            return isinstance(value, str)
        elif expected_type == "number":
            return isinstance(value, (int, float))
        elif expected_type == "integer":
            return isinstance(value, int)
        elif expected_type == "boolean":
            return isinstance(value, bool)
        elif expected_type == "array":
            return isinstance(value, list)
        elif expected_type == "object":
            return isinstance(value, dict)
        return True