using JuMP, GLPK
using MathOptInterface; const MOI = MathOptInterface
using Random, BenchmarkTools

# problem dimensions
const d = 2    # polynomial degree
const nt = 3   # number of t‐components

# inner solve: for a given t, find the best x‐coeffs minimizing "dist"
function solve_inner(t::Vector{Float64})
    t1, t2, t3 = t
    # target value at t
    fval = cos(t3) * (1 + t1)^t2

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

    # decision vars: x[i,j,k] and dist
    @variable(model, x[0:d, 0:d, 0:d])
    @variable(model, dist)

    # minimize maximum absolute error
    @objective(model, Min, dist)

    # two linear constraints enforcing |poly − fval| ≤ dist
    @constraint(model,
        sum(x[i,j,k] * t1^i * t2^j * t3^k
            for i in 0:d, j in 0:(d-i), k in 0:(d-i-j))
        - fval - dist <= 0
    )
    @constraint(model,
        -(sum(x[i,j,k] * t1^i * t2^j * t3^k
              for i in 0:d, j in 0:(d-i), k in 0:(d-i-j))
          - fval) - dist <= 0
    )

    optimize!(model)
    stat = termination_status(model)
    if stat == MOI.OPTIMAL || stat == MOI.LOCALLY_SOLVED
        return objective_value(model)
    else
        return -Inf
    end
end

# generic SA routine (maximize obj)
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 = copy(current), current_val
    T = T0

    for _ in 1:max_iters
        # propose neighbor
        cand = current .+ (rand(length(lower)) .- 0.5) .* (upper .- lower) .* 0.1
        cand = clamp.(cand, lower, upper)

        val = obj(cand)
        Δ = val - current_val

        if Δ > 0 || exp(Δ/T) > rand()
            current, current_val = cand, val
            if current_val > best_val
                best, best_val = copy(current), current_val
            end
        end
        T *= α
    end

    return best_val, best
end

# bounds for t: t[1] ∈ [0,1], t[2] ∈ [1,2], t[3] ∈ [0,1]
lower_bounds = [0.0, 1.0, 0.0]
upper_bounds = [1.0, 2.0, 1.0]

function run_maximization()
    simulated_annealing(solve_inner, lower_bounds, upper_bounds;
                        max_iters=10000, T0=1.0, α=0.995)
end

# run once to see output
max_dist, worst_t = run_maximization()
println("Maximized dist = ", max_dist)
println("Worst‐case t = ", worst_t)

# benchmark performance
bm = @benchmark run_maximization()
println("\nBenchmark:")
println("  Median time: ", median(bm.times)/1e6, " ms")
println("  Mean time:   ", mean(bm.times)/1e6, " ms")
println("  Memory:      ", bm.memory, " bytes")
println("  Allocations: ", bm.allocs)
