using JuMP, GLPK, MathOptInterface
const MOI = MathOptInterface
using Random
using BenchmarkTools
using Statistics  # for mean and median

# Global counter for accepted steps
global NumStep = 0

"""
    solve_inner(t1, t2)

Given parameters t1, t2, solves the inner minimization:
    minimize x[16]
    subject to
      exp(t2*log(t1)) - poly(x, t1, t2) - x[16] <= 0
     -exp(t2*log(t1)) + poly(x, t1, t2) - x[16] <= 0
    with x[i] ∈ [-50, 50] for i = 1..16.
Returns the optimal objective (x[16]) or -Inf if the solver fails.
"""
function solve_inner(t1, t2)
    # Compute t1^t2
    v = exp(t2 * log(t1))

    # Build the model
    model = Model(GLPK.Optimizer)
    set_silent(model)

    # x[1] through x[16], each in [-50, 50]
    @variable(model, -50 <= x[1:16] <= 50)

    # Objective: minimize x[16]
    @objective(model, Min, x[16])

    # Define the polynomial in x and t
    poly = x[1] +
           x[2]*t2 + x[3]*t2^2 + x[4]*t2^3 + x[5]*t2^4 +
           t1*(x[6]  + x[7]*t2 + x[8]*t2^2 + x[9]*t2^3) +
           t1^2*(x[10] + x[11]*t2 + x[12]*t2^2) +
           t1^3*(x[13] + x[14]*t2) +
           x[15]*t1^4

    # Constraints
    @constraint(model, v     - poly - x[16] <= 0)
    @constraint(model, -v + poly - x[16] <= 0)

    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 (we're maximizing outer)
        return -Inf
    end
end

"""
    fun(params)

Wrapper that takes a length-2 vector [t1, t2] and returns the inner optimization result.
"""
function fun(params::Vector{Float64})
    t1, t2 = params
    return solve_inner(t1, t2)
end

"""
    simulated_annealing(obj, lower, upper; max_iters, T0, α)

Basic simulated annealing to maximize `obj` over the box defined by `lower` and `upper`.
Returns `(best_value, best_params)` and updates global `NumStep` with the count of accepted steps.
"""
function simulated_annealing(obj, lower::Vector{Float64}, upper::Vector{Float64};
                             max_iters::Int = 10000, T0::Float64 = 1.0, α::Float64 = 0.995)
    # Initialize
    current     = lower .+ rand(length(lower)) .* (upper .- lower)
    current_val = obj(current)
    best, best_val = current, current_val
    T = T0

    global NumStep = 0

    for iter in 1:max_iters
        # Propose a new candidate
        candidate     = current .+ (rand(length(lower)) .- 0.5) .* (upper .- lower) .* 0.1
        candidate     = clamp.(candidate, lower, upper)
        candidate_val = obj(candidate)
        Δ = candidate_val - current_val

        # Accept if better or by Metropolis criterion
        if Δ > 0 || exp(Δ / T) > rand()
            current     = candidate
            current_val = candidate_val
            NumStep += 1
            if current_val > best_val
                best, best_val = current, current_val
            end
        end

        # Cool down
        T *= α
    end

    return best_val, best
end

# Bounds for t1 and t2
lower_bounds = fill(1.0, 2)
upper_bounds = fill(2.0, 2)

"""
    run_optimization()

Runs the outer simulated annealing and returns `(best_value, best_params)`.
"""
function run_optimization()
    return simulated_annealing(fun, lower_bounds, upper_bounds; max_iters=10000, T0=1.0, α=0.995)
end

# Benchmark the full run
benchmark_result = @benchmark run_optimization()

# Execute once for reporting
res_val, res_params = run_optimization()

# Display the results
println("Best value (maximum of fun): ", res_val)
println("Optimal parameters (t1, t2): ", res_params)
println("Number of accepted steps: ", NumStep)

println("\nBenchmark 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)

# Show the detailed benchmark table
display(benchmark_result)
