import matplotlib.pyplot as plt
from kneed import KneeLocator
from sklearn import preprocessing
from sklearn.datasets import make_blobs
from sklearn.cluster import KMeans
from sklearn.metrics import silhouette_score
from sklearn.preprocessing import StandardScaler
from sklearn.utils.extmath import randomized_svd
import random
import numpy as np
import matplotlib.pyplot as plt
import time

#https://realpython.com/k-means-clustering-python/
#https://scikit-learn.org/stable/modules/generated/sklearn.datasets.make_blobs.html

def proj(X):
    return np.transpose(X)@np.linalg.pinv(X@np.transpose(X))@X

def frob_cost_of_proj(A, B):
    U, S, V = np.linalg.svd(B, full_matrices=True)
    Ak = A@proj(V[0:len(S)])
    Atop = A - Ak
    return np.linalg.norm(Atop)**2

def generate_random_matrix(n, d, min_value=0, max_value=100):
    """
    Generate a random n by d integer matrix.

    Parameters:
    - n: Number of rows.
    - d: Number of columns.
    - min_value: Minimum value for the matrix elements (default is 0).
    - max_value: Maximum value for the matrix elements (default is 100).

    Returns:
    - A list of lists representing the random matrix.
    """
    matrix = [[random.randint(min_value, max_value) for _ in range(d)] for _ in range(n)]
    return matrix

#n=len(dataset)
#Too long for SVD
n=3000
##d=len(dataset[0])
##n=10
#d=4
d=100
randA = generate_random_matrix(n, d)
dataset_normalized=np.array(randA)
##
##nums = n
dataset_normalized = dataset_normalized[0:n]
dataset = dataset_normalized

start_time=time.time()
#k=1
k = 20
#c = 10
c_list = [1.1, 2, 5, 10, 100]
#c_list=[10]
count = -1
factors = []
our_costs = []
true_costs = []
ratios=[[],[],[],[],[]]
r = 0
old_r = 0
kcurr = 0
runtimes = []
i = 0
for c in c_list:
    count = -1
    for t in range(n-1):
        At = dataset[0:t+1]
        sq_norm_At = np.linalg.norm(At)**2
        #U, S, V = np.linalg.svd(At, full_matrices=True)
        U, S, V = randomized_svd(At, n_components = k, n_iter=5, random_state=None)
        r = len(V)
        kcurr = min(r,k)
    ##    if(r > old_r and r <= k):
    ##        kcurr = min(r, k)
    ##        count = sq_norm_At
    ##        factors = V[0:kcurr] 
        if(sq_norm_At>c*count):
            count = sq_norm_At
            factors = V[0:kcurr]  
        #old_r = r
        our_cost = frob_cost_of_proj(At, factors)
        if(our_cost < 10**-15):
            #Probably rounding error
            our_cost = 0
        true_cost = frob_cost_of_proj(At, V[0:kcurr])
        if(true_cost < 10**-15):
            #Probably rounding error
            true_cost=0
        our_costs.append(our_cost)
        true_costs.append(true_cost)
        if true_cost==0:
            ratio = 1
        else:
            ratio=our_cost/true_cost
        ratios[i].append(ratio)
        if t%100==0:
            print(t)
        runtimes.append(time.time()-start_time)
    i = i+1

# Generate x-axis values (assuming uniform spacing)
x_values = range(1, len(ratios[0]) + 1)

# Plotting ratios
plt.plot(x_values, ratios[0], linestyle='-', label='c=1.1')
plt.plot(x_values, ratios[1], linestyle='-', label='c=2')
plt.plot(x_values, ratios[2], linestyle='-', label='c=5')
plt.plot(x_values, ratios[3], linestyle='-', label='c=10')
plt.plot(x_values, ratios[4], linestyle='-', label='c=100')

plt.xlim(1,300)

# Adding labels and title for loss
plt.xlabel('Number of rows')
plt.ylabel('Ratio of loss')
plt.title('Accuracy for random dataset')

#Add legend
plt.legend()

# Display the plot
plt.grid(True)
plt.tight_layout()
plt.show()
