import bisect
from E01_ICE import E01_ICE
import random
import time
import torch

def add_cnfg(cnfgs, new_config, size):

    
    # Extract the objective values for the sorting process
    objective_values = [cnfg[1] for cnfg in cnfgs]

    # If the list has fewer than 10 items or the new config has a better (smaller) objective value
    if len(cnfgs) < size or new_config[1] < objective_values[-1]:
        # Find the position to insert based on the objective value
        idx = bisect.bisect_left(objective_values, new_config[1])
        
        # Insert the new configuration at the correct position
        cnfgs.insert(idx, new_config)

        if len(cnfgs) > size:
            cnfgs.pop(-1)

    return cnfgs

def divide_into_blocks(my_list, block_size):
    return [my_list[i:i + block_size] for i in range(0, len(my_list), block_size)]

def flatten(xs):
   return [a for x in xs for a in x]

def unique(xs):
    x = flatten(xs)
    return list(set(x))

def divide_into_blocks_weights(coreset, inds, blocksize):
    coreset_tensor = torch.tensor(coreset)
    inds_tensor = torch.tensor(inds)
    
    # Create a mask for elements in coreset that are in inds
    mask = torch.isin(coreset_tensor, inds_tensor)
    
    _, indices = torch.sort(-mask.to(torch.int), stable=True)
    coreset = coreset_tensor[indices].tolist()

    blocks = [coreset[i:i + blocksize] for i in range(0, len(coreset), blocksize)]
    blocks = [random.sample(block, len(block)) for block in blocks] 
    return blocks


def coreset_while(paras_coresets, paras_ice):
    C, inds_svm, blocksize, max_unchanged, Nremain, device, timelimit, verbose = paras_coresets
    X,y, size_cs, device = paras_ice

    start_time = time.time()
    elapsed_time = 0

    N,D = X.shape
    
    
    coreset = [i for i in range(N) ]
    last_coreset_size = N


    container = []
    best_candidates = []
    r=0
    


    counter = 0  # Counter to track unchanged value occurrences

    while((C>0) and (len(coreset) > Nremain) and elapsed_time< timelimit ):
             
        container = container[:C]
        r=r+1

        start = time.perf_counter()
        if verbose == True:
            print(f'this is {r}th shuffle')

        if len(inds_svm) < N:
            shuflled_blocks = divide_into_blocks_weights(coreset, inds_svm, blocksize)
        else:
            random.shuffle(coreset)
            shuflled_blocks = divide_into_blocks(coreset, blocksize)

        i = 0

        for block in shuflled_blocks:
            i = i+1
            start = time.perf_counter()
            # res = E01_ICE(block, X_hom, t)
            res = E01_ICE(block, X, y, 1,  size_cs, device, verbose=False)
            end = time.perf_counter()
            if verbose == True:
                print(f'the total time of {i}th block is {end-start}')

            best_candidates = add_cnfg(best_candidates, res, 1000)
            res_block = (block,res[1])
            container = add_cnfg(container, res_block, C)
            # print(container)
            if container[0][1] == 0:
                return best_candidates
        end = time.perf_counter()
        coreset = []
        coreset = [(coreset + i[0]) for i in container]
        coreset = unique(coreset)
        if verbose == True:
           print(f'the total time for {r}th shuffle is {end-start}')
           print(f'the coresetsize in last shuffle is {last_coreset_size}')

    

        # if len(coreset) == last_coreset_size :
        if len(coreset)<= 100:
            threshold = 2
        elif len(coreset)<=200:
            threshold = 2
            # blocksize =30
        elif len(coreset)<=300:
 
            threshold = 3 
        elif len(coreset)<=400:
            # blocksize =3
            # block_size_ncs_rand = int(50)
            threshold = 2
            # blocksize = 28
        elif len(coreset)<=700:

            threshold = 5       
        elif len(coreset)<=1000:
            
            threshold = 8
        
        elif len(coreset)<=2500:
            threshold = 10
        elif len(coreset)<=2500:
            threshold = 15
        elif len(coreset)<=5000:
            threshold = 20
        elif len(coreset)<=7500:
            threshold = 25
        else:
            threshold = 4

        if last_coreset_size - len(coreset) < threshold :
            counter += 1
        else:
            counter = 0 
        if verbose == True: 
            print(f'the maximal unchange is {max_unchanged}')
            print(f'the counter  in this shuffle is {counter}')

        if counter >= max_unchanged:
            if len(coreset)<=60:
                C = int(C*0.95)
                max_unchanged = 30
            elif len(coreset)<=80:
                C = int(C*0.95)
                max_unchanged = 30
            elif len(coreset)<=100:
                C = int(C*0.9)
                max_unchanged = 25
            elif len(coreset)<=150:
                # blocksize=35
                C = int(C*0.95)
                max_unchanged = 20
            elif len(coreset)<=300:
                C = int(C*0.95)
                max_unchanged = 15
            elif len(coreset)<=500:
                C = int(C*0.95)
                max_unchanged = 10
            elif len(coreset)<=1000:
                C = int(C*0.9)
                max_unchanged = 8
            elif len(coreset)<=1500:
                C = int(C*0.7)
                max_unchanged = 5
            elif len(coreset)<=2000:
                C = int(C*0.7)
                max_unchanged = 5
            else:
                C = int(C*0.8)
 
            counter = 0  # Reset counter after action

        last_coreset_size = len(coreset)
        elapsed_time = time.time() - start_time
        remaining_time = timelimit - elapsed_time
        print(f"    Remaining time: {remaining_time:.2f} seconds")
        print(f'    the best configuration in this shuffle{best_candidates[0]}')
        print(f'    the number of remaining data {len(coreset)}')
        print(f'    the new container has size {C}')
        
        print('\n')
    
    if len(coreset) >= Nremain:
        res = E01_ICE(coreset[:Nremain], X, y, 1,  size_cs, device, verbose=True)
    else:
        res = E01_ICE(coreset, X, y, 1,  size_cs, device, verbose=True)
    # res = E01_ICE_print(coreset, X_hom, t)
    best_candidates = add_cnfg(best_candidates, res, 1000)
    return best_candidates
