from typing import List, Dict, Any
import torch

def list_of_dict_to_dict_of_list(list_of_dicts: List[Dict[str, Any]]):
    keys = list(list_of_dicts[0].keys())
    new_dict = {}
    for key in keys:
        new_dict[key] = []
        for sub_dict in list_of_dicts:
            new_dict[key].append(sub_dict[key])
    return new_dict


def cuda_to_cpu(obj):
    """
    Recursively converts all PyTorch tensors from CUDA to CPU in a nested structure
    of dictionaries, lists, tuples and other iterables.
    
    Args:
        obj: The object to convert (can be a tensor, dict, list, tuple or other type)
        
    Returns:
        The same structure with all tensors moved to CPU
    """
    
    if isinstance(obj, torch.Tensor):
        # Move tensor to CPU if it's on CUDA
        return obj.cpu() if obj.is_cuda else obj
    elif isinstance(obj, dict):
        # Process each key-value pair in the dictionary
        return {k: cuda_to_cpu(v) for k, v in obj.items()}
    elif isinstance(obj, list):
        # Process each item in the list
        return [cuda_to_cpu(item) for item in obj]
    elif isinstance(obj, tuple):
        # Process each item in the tuple
        return tuple(cuda_to_cpu(item) for item in obj)
    elif hasattr(obj, '__iter__') and not isinstance(obj, (str, bytes)):
        # Handle other iterable types
        try:
            return type(obj)(cuda_to_cpu(item) for item in obj)
        except:
            # If conversion fails, return the object as is
            return obj
    else:
        # Return non-tensor, non-container objects unchanged
        return obj