using JuMP, GLPK, MathOptInterface
const MOI = MathOptInterface
using Random
using BenchmarkTools # Add BenchmarkTools package
using Statistics # For mean and median

# Global counter for outer iterations (similar to NumStep in Mathematica)
global NumStep = 0

# Define the inner optimization function for the new problem.
# Given three pairs of (t1, t2) values, solve for p1, p2, and d.
# Minimize d subject to:
# d >= abs( (ti^2+tj^2) - (p1*sqrt(ti^2+tj^2) + p2*exp(sqrt(ti^2+tj^2))) )
# for each of the three (ti, tj) pairs.
# This is equivalent to:
# d + p1*R + p2*ExpR >= S
# d - p1*R - p2*ExpR >= -S
# where R = sqrt(ti^2+tj^2), S = ti^2+tj^2, ExpR = exp(R)

function solve_inner_new(t_params_list::Vector{Tuple{Float64, Float64}})
    # Create a model with GLPK (a linear programming solver)
    model = Model(GLPK.Optimizer)
    set_silent(model)

    # Define inner variables (p1, p2, d)
    # Using similar bounds as the example for x, y, z
    @variable(model, -50 <= var_p1 <= 50)
    @variable(model, -50 <= var_p2 <= 50)
    @variable(model, -50 <= var_d_objective <= 50) # Renamed to avoid conflict if d was a param name

    # Objective: minimize var_d_objective
    @objective(model, Min, var_d_objective)

    for (ti, tj) in t_params_list
        # Ensure t values are positive as per bounds 1e-10
        # This check is mostly a safeguard; SA should respect bounds.
        if ti <= 0 || tj <= 0
            # This case should ideally not happen if SA bounds are correct
            # Return a very poor value for minimization
            return Inf
        end

        R_val = sqrt(ti^2 + tj^2)
        S_val = ti^2 + tj^2
        ExpR_val = exp(R_val)

        # Constraints:
        # var_d_objective >= S_val - (var_p1*R_val + var_p2*ExpR_val)
        # var_d_objective >= -(S_val - (var_p1*R_val + var_p2*ExpR_val))
        # Rewritten as:
        @constraint(model, var_d_objective + var_p1*R_val + var_p2*ExpR_val >= S_val)
        @constraint(model, var_d_objective - var_p1*R_val - var_p2*ExpR_val >= -S_val)
    end

    optimize!(model)
    status = termination_status(model)
    if status == MOI.OPTIMAL || status == MOI.LOCALLY_SOLVED
        return objective_value(model)
    else
        # If the solver fails, return a very poor value for minimization.
        return Inf
    end
end

# Define the outer function "fun_new_problem" that depends on six t parameters.
# It returns the inner minimization result (the value of d).
function fun_new_problem(params::Vector{Float64})
    t1a, t2a, t1b, t2b, t1c, t2c = params[1], params[2], params[3], params[4], params[5], params[6]
    
    t_params_list = [
        (t1a, t2a),
        (t1b, t2b),
        (t1c, t2c)
    ]
    return solve_inner_new(t_params_list)
end

# Simulated annealing algorithm modified for MINIMIZATION.
# It searches over a 6-dimensional box defined by lower and upper bounds.
function simulated_annealing_minimize(obj_func, lower::Vector{Float64}, upper::Vector{Float64};
                                      max_iters::Int = 10000, T0::Float64 = 1.0, α::Float64 = 0.995)
    # Start at a random point within the bounds.
    current_sol = lower .+ rand(length(lower)) .* (upper .- lower)
    current_val = obj_func(current_sol)
    
    best_sol = copy(current_sol)
    best_val = current_val
    
    T = T0
    global NumStep = 0  # Reset counter at start

    for iter in 1:max_iters
        # Generate a candidate by perturbing the current solution.
        # Perturbation scaled by 0.1 of the range (same as example)
        candidate_sol = current_sol .+ (rand(length(lower)) .- 0.5) .* (upper .- lower) .* 0.1
        candidate_sol = clamp.(candidate_sol, lower, upper) # Ensure within bounds
        
        candidate_val = obj_func(candidate_sol)
        
        # For MINIMIZATION:
        # Δ > 0 if candidate_val is better (smaller) than current_val
        Δ = current_val - candidate_val 

        # Accept candidate if it improves the objective or with a probability.
        # Added T > 1e-9 check to prevent issues with exp(Δ / T) when T is very small
        if Δ > 0 || (T > 1e-9 && exp(Δ / T) > rand())
            current_sol = candidate_sol
            current_val = candidate_val
            NumStep += 1  # Count each accepted step

            if current_val < best_val # For minimization, better is smaller
                best_sol = copy(current_sol) # copy() is important for arrays
                best_val = current_val
            end
        end

        # Cool down the temperature.
        T *= α
    end
    return best_val, best_sol
end

# Set up the outer optimization bounds for t parameters: each in [1e-10, sqrt(2)].
# There are 6 such parameters (3 pairs of t1, t2).
lower_bounds_new = fill(1e-10, 6)
upper_bounds_new = fill(sqrt(2.0), 6)

# Define a function that wraps the whole optimization process
function run_optimization_new()
    # Run the simulated annealing optimization.
    # Using fun_new_problem and simulated_annealing_minimize
    res_val, res_params = simulated_annealing_minimize(fun_new_problem, 
                                                       lower_bounds_new, 
                                                       upper_bounds_new; 
                                                       max_iters=10000, T0=1.0, α=0.995)
    return res_val, res_params
end

# --- Execution and Benchmarking ---

# Run benchmark
# Note: Benchmarking involves multiple runs, so NumStep will be from the last run in the benchmark.
println("Running benchmark... (NumStep will be from the last benchmark trial)")
benchmark_result = @benchmark run_optimization_new()

# Run once normally to get the results for printing (and reset NumStep for this specific run)
println("\nRunning a single optimization for results...")
res_val_final, res_params_final = run_optimization_new()

# Print results
println("\n--- Optimization Results ---")
println("Best value (minimum of d): ", res_val_final)
println("Optimal t parameters (t1a, t2a, t1b, t2b, t1c, t2c): ", res_params_final)
println("Number of accepted steps in the final run: ", NumStep)

# Print benchmark results
println("\n--- Benchmark Results ---")
println("Median time: ", median(benchmark_result.times) / 1_000_000, " ms")
println("Mean time: ", mean(benchmark_result.times) / 1_000_000, " ms")
println("Memory: ", benchmark_result.memory, " bytes")
println("Allocations: ", benchmark_result.allocs)
println("Full benchmark object:")
display(benchmark_result)

end_message = """

Note on Interpretation:
The problem "minimize dp: d" subject to constraints involving p and t variables has been interpreted
as a bilevel optimization problem, following the structure of the example code provided.
- The SA (outer loop) searches for optimal 't' parameters (three pairs: (t1a, t2a), (t1b, t2b), (t1c, t2c)).
- For each set of 't' parameters, an inner LP solves for 'p1', 'p2', and 'd' to minimize 'd'
  subject to constraints derived from the problem for those 't' values.
- The SA's objective is to find the 't' parameters that lead to the overall minimum 'd'.
The bounds for 'p1', 'p2', and 'd' in the inner LP are set to [-50, 50] analogous to 'x,y,z' in the example.
The bounds for 't' parameters in SA are [1e-10, sqrt(2)].
"""
println(end_message)