using JuMP, Ipopt, MathOptInterface
const MOI = MathOptInterface
using Random
using BenchmarkTools  # add BenchmarkTools package

# Global counter for accepted outer steps
global NumStep = 0

"""
    solve_inner(coeffs::Vector{Float64})

Given a vector `coeffs = [a₁, a₂, a₃, a₄, a₅, a₆]`, solves

    max_{t ∈ [-3,3]} z

subject to
    poly(t; coeffs) - (1 - exp(-t^2)) ≤ z
    (1 - exp(-t^2)) - poly(t; coeffs) ≤ z
    z ≥ 0

where
    poly(t) = a₁ + a₂·t + a₃·(t²-1) + a₄·(4t³-3t)
              + a₅·(8t⁴-8t²+1) + a₆·(16t⁵-20t³+5t).

Returns the maximum absolute error z.
"""
function solve_inner(coeffs::Vector{Float64})
    a1, a2, a3, a4, a5, a6 = coeffs

    model = Model(Ipopt.Optimizer)
    set_silent(model)

    @variable(model, -3 <= t <= 3)
    @variable(model, z >= 0)

    @NLexpression(model, poly, a1 +
        a2 * t +
        a3 * (t^2 - 1) +
        a4 * (4*t^3 - 3*t) +
        a5 * (8*t^4 - 8*t^2 + 1) +
        a6 * (16*t^5 - 20*t^3 + 5*t)
    )
    @NLexpression(model, ft, 1 - exp(-t^2))

    @NLconstraint(model, poly - ft <= z)
    @NLconstraint(model, ft - poly <= z)

    @NLobjective(model, Max, z)

    optimize!(model)
    status = termination_status(model)
    if status == MOI.OPTIMAL || status == MOI.LOCALLY_SOLVED
        return objective_value(model)
    else
        return -Inf  # indicate failure
    end
end

# Wrap to turn this into a *minimization* outer objective
fun(params::Vector{Float64}) = -solve_inner(params)

"""
    simulated_annealing(obj, lower, upper; max_iters=10_000, T0=1.0, α=0.995)

Simple SA for maximizing `obj`.  `lower`/`upper` define a box in ℝᵈ.
Returns (best_val, best_point).
"""
function simulated_annealing(obj, lower::Vector{Float64}, upper::Vector{Float64};
                             max_iters::Int=10_000, T0::Float64=1.0, α::Float64=0.995)
    d = length(lower)
    current = lower .+ rand(d) .* (upper .- lower)
    current_val = obj(current)
    best, best_val = current, current_val
    T = T0
    global NumStep = 0

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

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

        T *= α
    end

    return best_val, best
end

"""
    run_optimization()

Runs SA to find the six Chebyshev–like coefficients that minimize the max‐error.
"""
function run_optimization()
    # Coefficient bounds—tweak as needed
    lower_bounds = fill(-5.0, 6)
    upper_bounds = fill( 5.0, 6)

    res_val, res_params = simulated_annealing(fun, lower_bounds, upper_bounds;
                                             max_iters=10_000, T0=1.0, α=0.995)
    return res_val, res_params
end

# Benchmark
benchmark_result = @benchmark run_optimization()

# Run once for reporting
res_val, res_params = run_optimization()
println("Minimum max‐error ≈ ", -res_val)
println("Optimal coeffs [a₁…a₆]: ", res_params)
println("Number of accepted outer steps: ", NumStep)

println("\nBenchmark:")
println("  Median time: ", median(benchmark_result.times) / 1e6, " ms")
println("  Mean time:   ", mean(benchmark_result.times) / 1e6, " ms")
println("  Memory:      ", benchmark_result.memory, " bytes")
println("  Allocations: ", benchmark_result.allocs)
display(benchmark_result)
