from itertools import combinations
from constants import *
import pandas as pd
import numpy as np
from numba import njit

# start from configuration {0, ..., k-1}
start_config_set = frozenset(range(0, k))
start_config_array = np.array(list(range(0, k)))
start_config_id = 0

global config_array, num_config, configBinary, configBinaryInverse
config_array = np.array(list(combinations(range(n), k)))  # representation of configurations as arrays
num_config = len(config_array)  # number of configurations
configBinary = np.zeros(num_config, dtype=np.int32)  # representation of configurations in binary
configBinaryInverse = np.full((1 << n), -1, dtype=np.int32)  # inverse map of configBinary

for i in range(num_config):
    configBinary[i] = sum([(1 << j) for j in config_array[i]])
    configBinaryInverse[configBinary[i]] = i

# update a work function wf with request r
@njit
def update_wf(wf, r):
    new_wf = np.full(num_config, 1e9, dtype=np.int32)

    assert 0 <= r < n

    for i in range(num_config):
        if (configBinary[i] & (1 << r)) > 0:  # request is already part of configuration
            new_wf[i] = wf[i]
        else:
            min_val = 1e9

            for p in config_array[i]:
                new_config_id = configBinaryInverse[configBinary[i] ^ (1 << r) ^ (1 << p)]  # id of configuration obtained by swapping p with r
                min_val = min(min_val, wf[new_config_id] + distances[p, r])

            new_wf[i] = min_val

    return new_wf