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

# Problem dimensions
const nx = 3
const nt = 2

# Inner minimization: for a given t = [t1, t2], solve
#   min 2 x1 + 4 x2 + x3
#   s.t. nonlinear “if‐then” constraint
#        0 ≤ xi ≤ 1
function solve_inner(t::Vector{Float64})
    model = Model(Ipopt.Optimizer)
    set_silent(model)

    @variable(model, 0 <= x[1:nx] <= 1)
    @objective(model, Min, 2*x[1] + 4*x[2] + x[3])

    @NLconstraint(model,
        (1 - x[1]) * ifelse(t[1] > 0,
            (2 / t[1]) * exp(- (1 + (t[2] - 1)^2) / t[1]),
            0
        ) +
        (1 - x[2]) * ifelse(t[1] > 0,
            (1 / (2*t[1])) * exp(- (4 + t[2]^2) / (4*t[1])),
            0
        ) +
        (1 - x[3]) * ifelse(t[1] > 2,
            (1 / (t[1] - 2)) * exp(- (1 + (t[2] + 1)^2) / (t[1] - 2)),
            0
        )
        - 0.5 <= 0
    )

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

# Wrapper so fun(t) returns the inner‐min result
fun(t::Vector{Float64}) = solve_inner(t)

# Simulated annealing for MAXIMIZATION over t ∈ [lower,upper]
function simulated_annealing(obj, lower::Vector{Float64}, upper::Vector{Float64};
                             max_iters::Int=10_000, 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 iter in 1:max_iters
        # small perturbation
        cand = current .+ (rand(length(lower)) .- 0.5) .* (upper .- lower) .* 0.1
        cand = clamp.(cand, lower, upper)         # ← use clamp. instead of clamp!
        cand_val = obj(cand)
        Δ = cand_val - current_val

        # acceptance criterion
        if Δ > 0 || exp(Δ/T) > rand()
            current, current_val = cand, cand_val
            if current_val > best_val
                best, best_val = copy(current), current_val
            end
        end

        T *= α
    end

    return best_val, best
end

# Define t‐bounds and run
lower_bounds = fill(-1.0, nt)
upper_bounds = fill(4.0,  nt)

# Benchmark (optional)
bench = @benchmark simulated_annealing(fun, $lower_bounds, $upper_bounds;
                                      max_iters=10_000, T0=1.0, α=0.995)
println("Median SA time: ", median(bench.times)/1e6, " ms")

# Run once to get the solution
best_val, best_t = simulated_annealing(fun, lower_bounds, upper_bounds;
                                       max_iters=10_000, T0=1.0, α=0.995)
println("Maximized inner‐minimum = ", best_val)
println("Optimal t = ", best_t)
