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

# Problem constants
const π    = 4*atan(1)
const N    = 241
const R    = (N-1) ÷ 2           # = 120
const U1_1 = 0.9
const U2_1 = 1.1
const U1_2 = -0.0001
const U2_2 =  0.0001
const D1   = 0.5*(U2_1 + U1_1)
const d1   = 0.5*(U2_1 - U1_1)
const D2   = 0.5*(U2_2 + U1_2)
const d2   = 0.5*(U2_2 - U1_2)

# Counter for accepted outer steps
global NumStep = 0

"""
    solve_inner(t1, t2)

Given two frequency points `t1`, `t2`, builds and solves the LP
  minimize x[R+1]
subject to the four Chebyshev‐type constraints at t1 and t2.
Returns the optimal objective (minimum x[R+1]) or +Inf on failure.
"""
function solve_inner(t1, t2)
    model = Model(GLPK.Optimizer)
    set_silent(model)

    # x[0], x[1], …, x[R+1]
    @variable(model, x[0:R+1])

    # (D1 – A1) – d1 – x[R+1] ≤ 0
    @constraint(model,
        sum(-cos(t1*i)*x[i] for i in 0:R) 
        - x[R+1] + (D1 - d1) ≤ 0)

    # –(D1 – A1) – d1 – x[R+1] ≤ 0
    @constraint(model,
        sum( cos(t1*i)*x[i] for i in 0:R) 
        - x[R+1] + (-D1 - d1) ≤ 0)

    # (D2 – A2) – d2 – x[R+1] ≤ 0
    @constraint(model,
        sum(-cos(t2*i)*x[i] for i in 0:R) 
        - x[R+1] + (D2 - d2) ≤ 0)

    # –(D2 – A2) – d2 – x[R+1] ≤ 0
    @constraint(model,
        sum( cos(t2*i)*x[i] for i in 0:R) 
        - x[R+1] + (-D2 - d2) ≤ 0)

    @objective(model, Min, x[R+1])

    optimize!(model)
    status = termination_status(model)
    if status in (MOI.OPTIMAL, MOI.LOCALLY_SOLVED)
        return objective_value(model)
    else
        return Inf
    end
end

# Wrapper so that 'fun([t1, t2])' returns solve_inner(t1, t2)
fun(params::Vector{Float64}) = solve_inner(params[1], params[2])

"""
Generic simulated annealing to *maximize* obj(v) over box [lower,upper].
Tracks number of accepted moves in global NumStep.
"""
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
        # perturb by up to ±10% of the box
        candidate = current .+(rand(length(lower)) .- 0.5) .* (upper .- lower) .* 0.1
        candidate = clamp.(candidate, lower, upper)
        cand_val = obj(candidate)
        Δ = cand_val - current_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 t[1] ∈ [0, 0.2π], t[2] ∈ [0.24π, π]
lower_bounds = [0.0,    0.24*π]
upper_bounds = [0.2*π,  π]

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

# Benchmark
benchmark_result = @benchmark run_optimization()

# Single run for printing
res_val, res_params = run_optimization()

println("Best (max of min x[R+1]): ", res_val)
println("Optimal t values [t1, t2]: ", res_params)
println("Number of accepted outer 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)
display(benchmark_result)
