import json
import os
import pickle
import re
import shutil
import random
from time import time
import gurobipy as gp
import torch
import numpy as np
from gurobipy import GRB

import config
import utils
from helper import map_model_to_filtered_indices, get_a_new2, get_pattern, get_bigraph
import pyscipopt as scp

DEVICE = config.DEVICE
random.seed(0)
torch.manual_seed(0)
torch.cuda.manual_seed(0)


def search(m, scores, delta):
    """

    :param m: GUROBI model
    :param scores:
    :param delta:
    :return:
    """
    instance_variabels = m.getVars()
    instance_variabels.sort(key=lambda v: v.VarName)
    variabels_map = {}
    for v in instance_variabels:  # get a dict (variable map), varname:var clasee
        variabels_map[v.VarName] = v
    alphas = []
    for i in range(len(scores)):
        tar_var = variabels_map[scores[i][1]]  # target variable <-- variable map
        x_star = scores[i][3]  # 1,0,-1, decide whether need to fix
        if x_star < 0:
            continue
        # tmp_var = m1.addVar(f'alp_{tar_var}', 'C')
        tmp_var = m.addVar(name=f'alp_{tar_var}', vtype=GRB.CONTINUOUS)
        alphas.append(tmp_var)
        m.addConstr(tmp_var >= tar_var - x_star, name=f'alpha_up_{i}')
        m.addConstr(tmp_var >= x_star - tar_var, name=f'alpha_dowm_{i}')
    all_tmp = 0
    for tmp in alphas:
        all_tmp += tmp
    m.addConstr(all_tmp <= delta, name="sum_alpha")
    return m


def search_SCIP(m1, scores, delta):
    """

    :param m1:  SCIP MODEL
    :param scores:
    :param delta:
    :return:
    """
    m1_vars = m1.getVars()
    var_map1 = {}
    for v in m1_vars:  # get a dict (variable map), varname:var clasee
        var_map1[v.name] = v
    alphas = []
    for i in range(len(scores)):
        tar_var = var_map1[scores[i][1]]  # target variable <-- variable map
        x_star = scores[i][3]  # 1,0,-1, decide whether to fix
        if x_star < 0:
            continue
        tmp_var = m1.addVar(f'alp_{tar_var}_{i}', 'C')
        alphas.append(tmp_var)
        m1.addCons(tmp_var >= tar_var - x_star, f'alpha_up_{i}')
        m1.addCons(tmp_var >= x_star - tar_var, f'alpha_down_{i}')
    m1.addCons(scp.quicksum(ap for ap in alphas) <= delta, 'sum_alpha')
    return m1


def get_gp_best_objective(log_folder):
    best_objectives = []
    pattern = r"Best objective ([\d.e+-]+)"  #  "Best objective "
    filenames = sorted(os.listdir(log_folder))

    for filename in filenames:
        filepath = os.path.join(log_folder, filename)

        if os.path.isfile(filepath):
            try:
                with open(filepath, 'r') as f:
                    for line in f:
                        match = re.search(pattern, line)
                        if match:
                            best_objectives.append(float(match.group(1)))
                            break 
            except Exception as e:
                print(f"Error reading file {filename}: {e}")

    return best_objectives

