using JuMP, Ipopt, MathOptInterface
const MOI = MathOptInterface
using Random, BenchmarkTools

# Count accepted outer steps
global NumStep = 0

# Inner solve: for a given x, find max_t { exp(t1^2+t2^2) - P(x,t) }
function solve_inner(x::Vector{Float64})
    x1,x2,x3,x4,x5,x6 = x
    model = Model(Ipopt.Optimizer)
    set_silent(model)

    @variable(model, 0 <= t1 <= 1)
    @variable(model, 0 <= t2 <= 1)

    # nonlinear max objective
    @NLobjective(model, Min,
        exp(t1^2 + t2^2)
      - (x1 + x2*t1 + x3*t2 + x4*t1^2 + x5*t1*t2 + x6*t2^2)
    )

    optimize!(model)
    status = termination_status(model)
    return status == MOI.OPTIMAL ? objective_value(model) : Inf
end

# Outer objective: original f(x) + big penalty if constraint violated
function fun(x::Vector{Float64})
    # f(x)
    fval = x[1] + 0.5*x[2] + 0.5*x[3] + (1/3)*x[4] + 0.25*x[5] + (1/3)*x[6]
    # worst-case violation
    gmax = solve_inner(x)
    # penalty factor (tune as needed)
    penalty = max(0.0, gmax)
    return fval + 1e3 * penalty
end

# Simulated annealing for minimization over x ∈ [−50,50]^6
function simulated_annealing(obj, lower::Vector{Float64}, upper::Vector{Float64};
                             max_iters::Int=10_000, T0::Float64=1.0, α::Float64=0.995)
    current = lower .+ rand(length(lower)).*(upper .- lower)
    current_val = obj(current)
    best, best_val = copy(current), current_val
    T = T0
    global NumStep = 0

    for iter in 1:max_iters
        # propose
        candidate = current .+ (rand(length(lower)).-0.5).*(upper .- lower).*0.1
        candidate = clamp.(candidate, lower, upper)
        cand_val = obj(candidate)
        Δ = current_val - cand_val
        if Δ > 0 || exp(Δ/T) > rand()
            current, current_val = candidate, cand_val
            NumStep += 1
            if current_val < best_val
                best, best_val = copy(current), current_val
            end
        end
        T *= α
    end

    return best_val, best
end

# bounds for x
lower_bounds = fill(-50.0, 6)
upper_bounds = fill( 50.0, 6)

function run_optimization()
    simulated_annealing(fun, lower_bounds, upper_bounds;
                        max_iters=10_000, T0=1.0, α=0.995)
end

# benchmark + one run
benchmark_result = @benchmark run_optimization()
res_val, res_x = run_optimization()

println("Best penalized objective: ", res_val)
println("Optimal x: ", res_x)
println("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:      ", first(benchmark_result.memory), " bytes")
println("  Allocs:      ", first(benchmark_result.allocs))
display(benchmark_result)
