import numpy as np
from scipy.stats import rankdata
from scipy.spatial.distance import pdist, euclidean, sqeuclidean, squareform # Added euclidean for get_neighbors_and_pca
from scipy.linalg import eigh # For symmetric matrix eigenvalues/vectors
from scipy.sparse import lil_matrix, csr_matrix
from scipy.sparse.linalg import lsqr, eigsh # Added eigsh for the except block
import mdso # https://github.com/antrec/mdso/tree/master (Recanati et al.)
from scipy.interpolate import interp1d


####################################
# Permutation using Fiedler Vector #
####################################

# basic similarity function using Euclidian distance, can be modified
def gaussian_kernel(x, y, sigma = .25):
    return np.exp(-sqeuclidean(x, y)/(2*sigma**2))

def inverse_distance(x, y):
    return(1/(1 + euclidean(x, y)))

# Find Fiedler vector (eigenvector corresponding to smallest nonzero eigenval)
def fiedler_permutation(X, sigma):
    sim_mat = squareform(pdist(X, gaussian_kernel, sigma = sigma))
    D = np.diag(np.sum(sim_mat, axis=0))
    L = D - sim_mat
    eigenvals, eigenvecs = np.linalg.eig(L)
    fiedler_vec = eigenvecs[:, np.argsort(eigenvals)[2]]
    return(fiedler_vec, rankdata(fiedler_vec))

#######################################
# Spectral Ordering (Recanati et al.) #
#######################################

def spectral_ordering(X, sigma, return_sim_mat=False):
    sim_mat = squareform(pdist(X, gaussian_kernel, sigma = sigma))
    spec_ord = mdso.SpectralOrdering()
    if return_sim_mat:
        return spec_ord.fit_transform(sim_mat), sim_mat

    return spec_ord.fit_transform(sim_mat)

def spectral_ordering_robust(X, sigma):
    sim_mat = squareform(pdist(X, gaussian_kernel, sigma = sigma))
    spec_ord = mdso.SpectralOrdering(merge_if_ccs=True)
    result = spec_ord.fit_transform(sim_mat)
    # Handle case where result is a list of partial orderings
    if isinstance(result, list):
        # Flatten list of lists into a single array
        result = np.concatenate([np.atleast_1d(r) for r in result])
    return result