import math
import random
import itertools
import ipdb
import json
# from entry.parameter import *
from itertools import product

# Random selection from itertools.combinations(iterable, r)
def random_combination(iterable, r):
    pool = tuple(iterable)
    indices = sorted(random.sample(range(len(pool)), r))
    return tuple(pool[i] for i in indices)

# Generate a COP instance with forced satisfaction
def generate_forced_optimization_instance(k, n, alpha, r, p, file_path, min_cost=1, max_cost=5):
    d = math.ceil((pow(n, alpha)))
    m = math.ceil(r * n * math.log(n))
    nb = math.ceil((1 - p) * pow(d, k))

    var_pairs = [(n1, n2) for n1 in range(n) for n2 in range(n) if n1 != n2]
    weights = {}
    with open(file_path + 'weight.txt', 'w') as file:
        for n1, n2 in var_pairs:
            weights[(n1, n2)] = {}
            weights[(n2, n1)] = {}
            for val_pair in product(range(d), range(d)):
                if val_pair[0] == val_pair[1]:
                    weights[(n2, n1)][(val_pair[1], val_pair[0])] =  max_cost
                weight = random.randint(min_cost, max_cost)
                # weight = round(weight, 1)
                weights[(n1, n2)][val_pair] = weight
                weights[(n2, n1)][(val_pair[1], val_pair[0])] = weight
                
            file.write(f"{(n1, n2)}|{weights[(n1, n2)]}\n")
            file.write(f"{(n2, n1)}|{weights[(n2, n1)]}\n")

    with open(file_path, 'w') as file:
        file.write(f"n\td\tm\tk\tnb\talpha\tr\tp\n")
        file.write(f"{n}\t{d}\t{m}\t{k}\t{nb}\t{alpha}\t{r}\t{p}\n")

        rand_sol = [random.randint(0, d - 1) for _ in range(n)]
        file.write(f"{rand_sol}\n")

        for _ in range(m):
            scope = random_combination(range(n), k)
            support = []
            
            while True:
                choose_tuple = (random.randint(0, d), random.randint(0, d))
                if not (rand_sol[scope[0]] == choose_tuple[0] and rand_sol[scope[1]] == choose_tuple[1]):
                    support.append(choose_tuple)
                if len(support) >= nb:
                    break
            contr_scope = (scope[1], scope[0])
            contr_support = [(x2, x1) for x1, x2 in support]
            file.write(f"{scope}|{support}\n")
            file.write(f"{contr_scope}|{contr_support}\n")

def generate_instance_files(k, n, alpha, r, p, file_path, num_instances):
    for i in range(num_instances):
        instance_file_path = f"{file_path}{i}.txt"
        generate_forced_optimization_instance(k, n, alpha, r, p, instance_file_path)

if __name__=='__main__':
    for i in [20, 30, 40]:
        generate_instance_files(2, i, 0.7, 3, 0.21, f"data/COP{i}/", 100)
