using JuMP, GLPK, MathOptInterface
const MOI = MathOptInterface
using Random
using BenchmarkTools # Add BenchmarkTools package
using Statistics     # For mean and median in benchmark results

# Global counter for outer iterations (similar to NumStep in Mathematica)
# This global variable will be modified by the simulated_annealing function.
global NumStep = 0

# Define nx for the new problem (as a constant)
const NX_FOR_NEW_PROBLEM = 50

# --- Inner Optimization Function for the New Problem ---
# Solves the LP for x variables, given a fixed t1_val.
# minimize (sum {i in 1..nx} (x[i]/i))
# subject to:
#   (sum {i in 1..nx} (-t1_val^(i-1)*x[i])) + (1/(2-t1_val)) <= 0
#   x[i] >= 0

function solve_inner_new(t1_val::Float64)
    # The problem specifies 0 <= t1_val <= 1, so t1_val will not be 2.0.
    # K = 1/(2-t[1])
    constant_K = 1.0 / (2.0 - t1_val)

    model = Model(GLPK.Optimizer)
    set_silent(model) # Suppress solver output

    # Define x variables: x[i] >= 0
    @variable(model, vars_x[1:NX_FOR_NEW_PROBLEM] >= 0)

    # Define Objective: minimize (sum {i in 1..nx} (x[i]/i))
    @objective(model, Min, sum(vars_x[i] / Float64(i) for i in 1:NX_FOR_NEW_PROBLEM))

    # Define Constraint: (sum {i in 1..nx} (-t1_val^(i-1)*x[i])) + K <= 0
    # Note: In Julia, 0.0^0.0 evaluates to 1.0, which is correct for the i=1 term when t1_val is 0.
    @constraint(model, sum(-(t1_val^(i - 1)) * vars_x[i] for i in 1:NX_FOR_NEW_PROBLEM) + constant_K <= 0)
    
    optimize!(model)
    status = termination_status(model)

    if status == MOI.OPTIMAL
        return objective_value(model)
    elseif status == MOI.INFEASIBLE
        # If the inner problem is infeasible for this t1_val,
        # it means this t1_val is not a valid choice for the outer minimization.
        # Return positive infinity, indicating a very bad objective value for minimization.
        return Inf
    elseif status == MOI.DUAL_INFEASIBLE # Primal LP is unbounded (for minimization, objective -> -Inf)
        # This means the objective can be made arbitrarily small, which is good for minimization.
        return -Inf
    else
        # Other statuses (e.g., numerical error, solver limit)
        # Treat as a failure to find a good minimum, so return a very poor value.
        return Inf 
    end
end

# --- Outer Function "fun" for Simulated Annealing ---
# This function takes the outer variable(s) (params) and returns a value to be maximized by SA.
# For the new problem, params will be a vector containing just t1_val.
function fun_for_new_problem(params::Vector{Float64})
    t1_val = params[1]
    
    # Call the inner solver. It returns min_x (sum x_i/i) for the given t1_val.
    inner_objective_result = solve_inner_new(t1_val)

    # Simulated Annealing (SA) maximizes its objective function.
    # Our overall goal is to MINIMIZE inner_objective_result over t1_val.
    # So, SA should maximize the NEGATIVE of inner_objective_result.
    # If inner_objective_result is +Inf (bad for min), -inner_objective_result is -Inf (SA avoids).
    # If inner_objective_result is -Inf (good for min), -inner_objective_result is +Inf (SA seeks).
    return -inner_objective_result
end

# --- Simulated Annealing Algorithm ---
# This function is identical to the one provided in the original problem.
# It performs maximization of the objective function `obj`.
function simulated_annealing(obj::Function, lower::Vector{Float64}, upper::Vector{Float64};
                             max_iters::Int = 10000, T0::Float64 = 1.0, α::Float64 = 0.995)
    # Start at a random point within the bounds.
    current = lower .+ rand(length(lower)) .* (upper .- lower)
    current_val = obj(current)
    best = current
    best_val = current_val
    T = T0
    
    # Reset the global step counter at the beginning of each SA run.
    # This global counter is part of the original problem's structure.
    global NumStep = 0  

    for iter in 1:max_iters
        # Generate a candidate by perturbing the current solution.
        # Perturbation is scaled by 0.1 of the variable's range.
        candidate = current .+ (rand(length(lower)) .- 0.5) .* (upper .- lower) .* 0.1
        candidate = clamp.(candidate, lower, upper) # Ensure candidate stays within bounds
        
        candidate_val = obj(candidate)
        Δ = candidate_val - current_val # Change in objective

        # Accept candidate if it improves the objective (Δ > 0 for maximization)
        # or with a probability based on Boltzmann distribution if it's worse.
        # This logic handles Inf/-Inf/NaN values correctly due to IEEE 754 behavior.
        if Δ > 0 || (T > 1e-9 && exp(Δ / T) > rand()) # Added T > 1e-9 to avoid exp(big/small_T) issues
            current = candidate
            current_val = candidate_val
            NumStep += 1  # Count each accepted step
            if current_val > best_val # If this accepted step is better than the overall best
                best = current
                best_val = current_val
            end
        end

        T *= α # Cool down the temperature
    end
    return best_val, best # Return best value found and corresponding parameters
end

# --- Wrapper Function for the Entire Optimization Process ---
function run_optimization_for_new_problem()
    # Define bounds for the outer variable t[1]: 0 <= t[1] <= 1
    lower_bounds_sa = [0.0]
    upper_bounds_sa = [1.0]

    # Run the simulated annealing optimization.
    # `simulated_annealing` maximizes `fun_for_new_problem`.
    # `fun_for_new_problem` returns - (actual inner objective minimum).
    # So, `sa_maximized_value` = max_{t1} [- (min_x sum(x_i/i))]
    #                         = - min_{t1} [min_x sum(x_i/i)]
    sa_maximized_value, sa_best_params_vector = simulated_annealing(
                                                    fun_for_new_problem, 
                                                    lower_bounds_sa, 
                                                    upper_bounds_sa;
                                                    max_iters=10000, # Default from original
                                                    T0=1.0,          # Default from original
                                                    α=0.995          # Default from original
                                                )

    # The actual minimum objective value for the new problem is -sa_maximized_value
    actual_min_objective_value = -sa_maximized_value
    optimal_t1_value = sa_best_params_vector[1] # Extract t1 from the vector

    # NumStep is global and reflects the count from the SA run.
    return actual_min_objective_value, optimal_t1_value, NumStep
end

# --- Main Execution Block ---

# Run the optimization once to get results for printing
# NumStep will be set by the call to simulated_annealing within run_optimization_for_new_problem
final_objective_value, final_t1, steps_in_run = run_optimization_for_new_problem()

println("--- New Problem Results ---")
println("Best value (minimum of sum x_i/i): ", final_objective_value)
println("Optimal parameter t1: ", final_t1)
println("Number of accepted steps in the run: ", steps_in_run)

# Run benchmark for the new problem
println("\n--- New Problem Benchmark ---")
# @benchmark runs the function multiple times. Each run will reset NumStep internally.
benchmark_result_new = @benchmark run_optimization_for_new_problem()

println("Median time: ", median(benchmark_result_new.times) / 1_000_000, " ms")
println("Mean time: ", mean(benchmark_result_new.times) / 1_000_000, " ms")
println("Memory: ", benchmark_result_new.memory, " bytes")
println("Allocations: ", benchmark_result_new.allocs)
display(benchmark_result_new)