import math
from dataclasses import dataclass
from typing import Callable, Dict, List, Tuple

import torch


@dataclass(frozen=True)
class TaskSpec:
    name: str
    dim: int
    bounds: object
    objective: Callable[[torch.Tensor], torch.Tensor]

    def evaluate(self, X: torch.Tensor, noise_std: float = 0.0) -> torch.Tensor:
        if X.dim() == 1:
            X_in = X.unsqueeze(0)
        else:
            X_in = X
        y = self.objective(X_in).view(-1, 1)
        if noise_std > 0:
            y = y + noise_std * torch.randn_like(y)
        return y


def _base_target_implementation(X: torch.Tensor) -> torch.Tensor:
    """
    The ground truth function implementation.
    Based on the previous 'targetfunction' logic with a fixed seed (2025).
    """
    dim = int(X.shape[-1])
    bounds = (0.0, 1.0)

    g = torch.Generator()
    g.manual_seed(2025)

    w1 = torch.randn(dim, generator=g)
    w2 = torch.randn(dim, generator=g)
    w1 = w1 / (w1.norm() + 1e-12)
    w2 = w2 / (w2.norm() + 1e-12)

    a = float(torch.rand((), generator=g) * 1.5 + 0.5)
    b = float(torch.rand((), generator=g) * 1.0 + 0.2)
    c = float(torch.rand((), generator=g) * 0.3 + 0.05)

    X_clamped = X.clamp(bounds[0], bounds[1])
    w1_local = w1.to(device=X.device, dtype=X.dtype)
    w2_local = w2.to(device=X.device, dtype=X.dtype)

    z1 = 2.0 * math.pi * (X_clamped @ w1_local)
    z2 = 2.0 * math.pi * (X_clamped @ w2_local)
    bowl = ((X_clamped - 0.5) ** 2).sum(dim=1)

    y = a * torch.sin(z1) + b * torch.cos(z2) - c * bowl

    if dim >= 2:
        x0 = X_clamped[:, 0]
        x1 = X_clamped[:, 1]
        y = y + 0.15 * torch.sin(3.0 * math.pi * (x0 - 0.3)) * torch.cos(2.0 * math.pi * (x1 + 0.1))

    return y


def targetfunction(X: torch.Tensor) -> torch.Tensor:
    return _base_target_implementation(X)


def history_function_1(X: torch.Tensor) -> torch.Tensor:
    y = _base_target_implementation(X)
    return y + 0.5


def history_function_2(X: torch.Tensor) -> torch.Tensor:
    y = _base_target_implementation(X)
    return y - 0.5


def build_history_tasks(dim: int = 6, bounds: Tuple[float, float] = (0.0, 1.0)) -> List[TaskSpec]:
    return [
        TaskSpec(name="history_1", dim=dim, bounds=bounds, objective=history_function_1),
        TaskSpec(name="history_2", dim=dim, bounds=bounds, objective=history_function_2),
    ]


def build_real_task(dim: int = 6, bounds: Tuple[float, float] = (0.0, 1.0)) -> TaskSpec:
    return TaskSpec(name="target_real", dim=dim, bounds=bounds, objective=targetfunction)


def as_task_dict(tasks: List[TaskSpec]) -> Dict[str, TaskSpec]:
    return {t.name: t for t in tasks}

