from pprint import pprint
class ConfigurationResolver:
    """
    A class to resolve `link` entries in a nested configuration structure.
    Supports navigation through dictionaries, lists, and `children`.
    """

    def __init__(self, config):
        self.config = config

    def resolve_links(self):
        """
        Resolves all `link` entries in the configuration.
        """
        self._recursive_resolve(self.config, self.config)

    def __load_children(self,path):
        """
        Check if explicitly children should be loaded and not parent
        """
        return path.split('.').pop() == "children"
        
    def _replace_in_dict_by_dict(self, dict_origine, old_key, new_dict):
        if old_key in dict_origine:
            # Entferne den alten Key
            old_value = dict_origine.pop(old_key)
            # Füge die neuen Key-Value-Paare hinzu
            dict_origine.update(new_dict)

    def _replace_in_list_by_list(self, list_origine, old_element, new_elements):
        index = list_origine.index(old_element)
        list_origine[index:index+1] = new_elements

    def _replace_element_in_list(self, list_origine, old_element, new_element):
        index = list_origine.index(old_element)
        list_origine[index] = new_element

    def _recursive_resolve(self, current_config, root_config):
        """
        Recursively resolves `link` entries in the configuration.
        """
        if isinstance(current_config, dict):
            for key, value in list(current_config.items()):
                if key == "children":
                    if value is None or not isinstance(value, list):
                        raise ValueError(f"Expected 'children' to be a list, but got {type(value).__name__} instead.")
                    for item in value:
                        if "link" in item:
                            loaded_link = self._find_entry(root_config, item['link'].lower(), False)
                            if isinstance(loaded_link, list):
                                self._replace_in_list_by_list(value,item,loaded_link)
                            else:
                                self._replace_element_in_list(value,item,loaded_link)  
                        else:
                            self._recursive_resolve(value, root_config)            
                elif key == "link":
                    try:
                        loaded = self._find_entry(root_config, value.lower(), True)
                        if isinstance(loaded, list) and len(loaded) > 2:
                            loaded = self._find_entry(root_config, value.lower(), False)  
                        current_config.clear()
                        current_config.update(loaded)
                    except Exception as e: 
                        raise ValueError(
                            f"Error resolving link '{value}': {str(e)}. "
                            f"Current path: {key}, Current config: {current_config}" + (f", Loaded: {loaded}" if 'loaded' in locals() or 'loaded' in globals() else "")
                        )
                else:
                    self._recursive_resolve(value, root_config)
        elif isinstance(current_config, list):
            for item in current_config:
                self._recursive_resolve(item, root_config)

    def _get_children(self,current):
        if isinstance(current, dict) and ("children" in current and current["children"]):
            current = current["children"]
        return current

    def _find_by_name(self,current, part):
        return next(
                    (item for item in current if isinstance(item, dict) and item.get("name", "").lower() == part),
                    None
                )

    def _find_entry(self, config, path, children):
        """
        Finds an entry in the configuration by a dot-separated path.
        Supports both dictionaries and lists with `children` navigation.
        """
        parts = path.split('.')
        current = config
        for part in parts:
            if isinstance(current, list):
                # If children explicit declared just load children
                if part != "children":
                    # Look for a matching name in the list
                    found = self._find_by_name(current,part)
                    if found:
                        current = found
                        print(
                            f"Matching entry for '{part}' in list. Path so far: {' > '.join(parts[:parts.index(part)+1])}. "
                            f"Current list: {current}"
                        )
                    else:
                        raise ValueError(
                            f"No matching entry for '{part}' in list. Path so far: {' > '.join(parts[:parts.index(part)+1])}. "
                            f"Current list: {current}"
                        )
            elif isinstance(current, dict):
                # Case-insensitive dictionary lookup
                key = next((k for k in current if k.lower() == part), None)
                if key is None:
                    current = self._find_by_name(current["children"],part)
                    if not current:
                        raise KeyError(
                            f"Key '{part}' not found in dictionary. Path so far: {' > '.join(parts[:parts.index(part)+1])}. "
                            f"Current dictionary: {current}"
                        )
                else: 
                    current = current[key]
                
            else:
                raise ValueError(
                    f"Invalid path segment '{part}'. Current type: {type(current)}. "
                    f"Path so far: {' > '.join(parts[:parts.index(part)+1])}"
                )
            if children:
                current = self._get_children(current)

        return current

    def get_config(self):
        """
        Returns the resolved configuration.
        """
        return self.config