import torch
import numpy as np


def invert_regularized(J: np.ndarray, lam: float = 1e-4) -> np.ndarray:
    """
    Compute the regularized inverse of a square matrix J.

    Args:
        J (np.ndarray): Square matrix to invert (d x d).
        lam (float, optional): Regularization parameter (lambda). Default is 1e-4.

    Returns:
        np.ndarray: Regularized inverse of J (d x d).
    """
    d, _ = J.shape
    I = np.eye(d)
    return np.linalg.inv(J + lam * I)


import numpy as np



def j_inv_linear_system(H1: np.ndarray, H2: np.ndarray, *, tol: float = 1e-8):
    """
    Model: H_i = A^{-T} D_i A^{-1}, i=1,2 with diagonal invertible D_i and invertible A.
    Returns A_inv_hat ≈ A^{-1} up to left diagonal scaling and permutation.

    Parameters
    ----------
    H1, H2 : (d,d) symmetric arrays
    tol    : tolerance for eigenvalue separation / imaginary parts

    Returns
    -------
    A_inv_hat : (d,d) ndarray
        Estimate of A^{-1}, identifiable up to left diag scaling and permutation.
    lambdas   : (d,) ndarray
        Generalized eigenvalues of (H2, H1), i.e., diag(D1^{-1} D2) up to ordering.
    info      : dict
        Diagnostics.
    """
    H1 = np.asarray(H1, dtype=float)
    H2 = np.asarray(H2, dtype=float)
    d = H1.shape[0]
    assert H1.shape == (d, d) and H2.shape == (d, d)

    # # Symmetrize to tame numerical noise
    # H1 = 0.5 * (H1 + H1.T)
    # H2 = 0.5 * (H2 + H2.T)

    # Form M = H1^{-1} H2 without explicitly inverting H1
    M = np.linalg.solve(H1, H2)

    # Eigendecompose M = V diag(lambdas) V^{-1}  with V ≈ A (up to scale/perm)
    w, V = np.linalg.eig(M)

    # Reality checks (ideal model => real)
    if np.max(np.abs(np.imag(w))) > tol or np.max(np.abs(np.imag(V))) > tol:
        if max(np.max(np.abs(np.imag(w))), np.max(np.abs(np.imag(V)))) < 1e-4:
            w = np.real(w); V = np.real(V)
        else:
            print("Eigenpairs are significantly complex; assumptions likely violated.")
            return None 
    else:
        w = np.real(w); V = np.real(V)

    # Identifiability requires distinct generalized eigenvalues
    gaps = np.diff(np.sort(w))
    if gaps.size and np.min(np.abs(gaps)) < tol:
        print("Some generalized eigenvalues coincide; A^{-1} not fully identifiable.")
        return None

    # Since V ≈ A (up to column scaling/permutation), A^{-1} ≈ V^{-1}
    A_inv_hat = np.linalg.inv(V)
    return A_inv_hat
