from __future__ import annotations

from typing import Any, Dict, Mapping, Union

import numpy as np


AffineTransform = Dict[str, Union[float, str]]


def identity_affine() -> AffineTransform:
    return {"kind": "affine", "scale": 1.0, "shift": 0.0}


def _to_float(value: Any, field_name: str) -> float:
    try:
        number = float(value)
    except (TypeError, ValueError) as exc:
        raise ValueError(f"{field_name} must be numeric.") from exc
    if not np.isfinite(number):
        raise ValueError(f"{field_name} must be finite.")
    return number


def parse_affine_transform(value: Any, *, field_name: str = "transform") -> AffineTransform:
    if value is None:
        return identity_affine()
    if not isinstance(value, Mapping):
        raise ValueError(f"{field_name} must be an object.")

    kind = str(value.get("kind", "affine")).strip().lower()
    if kind not in ("affine", "identity"):
        raise ValueError(f"{field_name}.kind must be 'affine' or 'identity'.")

    scale = _to_float(value.get("scale", 1.0), f"{field_name}.scale")
    shift = _to_float(value.get("shift", 0.0), f"{field_name}.shift")
    return {"kind": "affine", "scale": scale, "shift": shift}


def parse_io_transform(value: Any) -> Dict[str, AffineTransform]:
    if value is None:
        return {"input": identity_affine(), "output": identity_affine()}
    if not isinstance(value, Mapping):
        raise ValueError("transform must be an object.")

    # Shorthand: {"kind":"affine","scale":...,"shift":...} means input transform only.
    if "input" not in value and "output" not in value:
        return {
            "input": parse_affine_transform(value, field_name="transform"),
            "output": identity_affine(),
        }

    return {
        "input": parse_affine_transform(value.get("input"), field_name="transform.input"),
        "output": parse_affine_transform(value.get("output"), field_name="transform.output"),
    }


def inverse_affine(transform: Mapping[str, Any], *, field_name: str = "transform") -> AffineTransform:
    scale = _to_float(transform.get("scale", 1.0), f"{field_name}.scale")
    shift = _to_float(transform.get("shift", 0.0), f"{field_name}.shift")
    if scale == 0.0:
        raise ValueError(f"{field_name}.scale must be non-zero for inversion.")
    return {"kind": "affine", "scale": 1.0 / scale, "shift": -shift / scale}


def compose_affine(
    first: Mapping[str, Any],
    second: Mapping[str, Any],
) -> AffineTransform:
    """Return second(first(x))."""
    s1 = _to_float(first.get("scale", 1.0), "first.scale")
    b1 = _to_float(first.get("shift", 0.0), "first.shift")
    s2 = _to_float(second.get("scale", 1.0), "second.scale")
    b2 = _to_float(second.get("shift", 0.0), "second.shift")
    return {"kind": "affine", "scale": s2 * s1, "shift": s2 * b1 + b2}


def apply_affine(values: "Union[np.ndarray, float]", transform: Mapping[str, Any]) -> np.ndarray:
    scale = _to_float(transform.get("scale", 1.0), "transform.scale")
    shift = _to_float(transform.get("shift", 0.0), "transform.shift")
    array = np.asarray(values, dtype=np.float64)
    return scale * array + shift
