# Script to evaluate BBOB on COBYLA

##########
# Imports
import cma
import cocoex
import numpy as np
import os
import time
import torch

from torch.quasirandom import SobolEngine
from torch import Tensor
from botorch.utils.transforms import unnormalize

##########
# Helper function for tracking


def get_best_index_for_batch(n_tr, Y: Tensor, C: Tensor):
    """Return the index for the best point. One for each trust region."""
    is_feas = (C <= 0).all(dim=-1)
    if is_feas.any():  # Choose best feasible candidate
        score = Y.clone()
        score[~is_feas] = -float("inf")
        return torch.topk(score.reshape(-1), k=n_tr).indices
    return torch.topk(C.clamp(min=0).sum(dim=-1), k=n_tr, largest=False).indices # Return smallest violation

def constraint(x): 
    """COBYLA constraints: Must return >= 0 for feasible solutions"""
    x_track.append(x)
    y_track.append(p(x))
    c_track.append(p.constraint(x))
    return p.constraint(x) 

##########
# Main code    

# Define COCO input
suite_name = "bbob-constrained"
suite = cocoex.Suite(suite_name, "", "")

# Load random seeds
seeds = np.load('random_seeds.npy')

# Base directory for saving files
cwd_base = os.path.join(os.getcwd(), 'results')

# General log file
f_gen = open(os.path.join(cwd_base, '00_GeneralLog.txt'), 'w')

for p in suite:
    
    if not ('i01' in p.id or 'i02' in p.id or 'i03' in p.id):
        continue
    if not ('d02' in p.id or
            'd10' in p.id or
            'd40' in p.id):
        continue
    # if not ('f053' in p.id):
    #     # print('Skipped')
    #     continue
    # break

    x_track = []
    y_track = []
    c_track = []
    
    print(p.id)
    print(f"{p.index}) {p.id}:", file=f_gen)
    print("\t Started", file=f_gen)
    
    # Create directory for problem
    cwd_current = os.path.join(cwd_base, p.id)
    if not p.id in os.listdir(cwd_base):
        os.mkdir(cwd_current)
    
    # Log file
    f = open(os.path.join(cwd_current, '00_Log.txt'), 'w')
    print(f"{p.id}", file=f)
    print(f"{time.strftime('%x - %X')}: Start", file=f)
        
    # Start time evaluation
    tic = time.time()
    
    # Perform 30 repetitions
    for i, seed in enumerate(seeds):
        # Optimization start        
        # seed iterative for restarts
        seed_j = 0
        
        # Setting random seed
        np.random.seed(int(seed[seed_j]))
    
        # Defining initial guess for CMAES
        sobol = SobolEngine(dimension=p.dimension, scramble=True, seed=int(seed[seed_j]))
        x0 = unnormalize(sobol.draw(n=int(3 * p.dimension)).cpu().numpy(), [p.lower_bounds, p.upper_bounds])
        Y0 = Tensor([p(x_) for x_ in x0]).unsqueeze(-1)
        C0 = Tensor([np.amax(constraint(x_)) for x_ in x0]).unsqueeze(-1)
        x0 = x0[get_best_index_for_batch(1, Y0, C0)]
    
        # Optimizing with CMAES
        res = cma.fmin_con2(objective_function = p, 
                            x0 = x0, 
                            sigma0 = 1.0, 
                            constraints=constraint, 
                            find_feasible_first=False, 
                            find_feasible_final=False, 
                            kwargs_confit=None, 
                            options = {'seed': int(seed[seed_j]),
                                       'maxfevals': 30 * p.dimension,
                                       'bounds': [p.lower_bounds, p.upper_bounds],
                                       'tolx': 1e-8})
    
    
    # Elaborate results
    X_batch = [x_track[i:i + 30 * p.dimension + res[1].popsize + 1 + 3 * p.dimension] for i in range(0, len(x_track), 30 * p.dimension + res[1].popsize + 1 + 3 * p.dimension)]
    Y_batch = [y_track[i:i + 30 * p.dimension + res[1].popsize + 1 + 3 * p.dimension] for i in range(0, len(y_track), 30 * p.dimension + res[1].popsize + 1 + 3 * p.dimension)]
    C_batch = [np.amax(c_track[i:i + 30 * p.dimension + res[1].popsize + 1 + 3 * p.dimension], axis=1) for i in range(0, len(c_track), 30 * p.dimension + res[1].popsize + 1 + 3 * p.dimension)]
   
# Evaluate objective and constraints for all evaluated samples
# Y_batch = []
# C_batch = []
    
# for X_ in X_batch:
#     yy = []
#     cc = []
#     for xx in X_:
#         yy.append(p(xx))
#         cc.append(np.amax(p.constraint(xx)))
#     Y_batch.append(yy[1:])
#     C_batch.append(cc[1:])
        
    Y_batch = np.array(Y_batch)
    C_batch = np.array(C_batch)
    
    # Retrieve monotonic curves to save
    Y_f = np.copy(Y_batch)
    C_f = np.copy(C_batch)
    Y_f[np.where(C_f > 0)[0], np.where(C_f > 0)[1]] = 0
    Y_f[np.where(C_f > 0)[0], np.where(C_f > 0)[1]] = np.amax(Y_f)

    Y_f_monotonic = []
    for YY in Y_f:
        y_mono = []
        for yy in YY:
            if len(y_mono) == 0:
                y_mono = [yy]
            else:
                if yy < y_mono[-1]:
                    y_mono.append(yy)
                else:
                    y_mono.append(y_mono[-1])

        Y_f_monotonic.append(y_mono)
    
    Y_f_monotonic = np.array(Y_f_monotonic)
    
    # Save interesting curves
    np.save(os.path.join(cwd_current, '01_Y_mono.npy'), Y_f_monotonic)
    np.save(os.path.join(cwd_current,'02_Y_best.npy'), Y_batch)
    np.save(os.path.join(cwd_current,'02_C_best.npy'), C_batch)
    
    
    
        


