From bfa6139c9f8692d23ea5680b3ed80c587558ab09 Mon Sep 17 00:00:00 2001 From: Oscar Dowson Date: Mon, 2 Dec 2024 09:23:56 +1300 Subject: [PATCH] Add Env(params::Dict) for setting parameters before GRBstartenv (#596) --- README.md | 19 +++++++- src/MOI_wrapper/MOI_wrapper.jl | 85 +++++++++++++++++++++++----------- test/MOI/MOI_wrapper.jl | 19 ++++++++ 3 files changed, 94 insertions(+), 29 deletions(-) diff --git a/README.md b/README.md index 17d7010d..ab832cf4 100644 --- a/README.md +++ b/README.md @@ -285,7 +285,22 @@ end # Note the need for GRB_ENV_REF[] not GRB_ENV_REF create_optimizer() = Gurobi.Optimizer(GRB_ENV_REF[]) -end +end # MyModule +``` + +## Pass parameters to an Environment + +To set parameters in an environment before the environment is started, pass a +`Dict{String,Any}` that maps parameter names to values: + +```julia +env = Gurobi.Env( + Dict{String,Any}( + "CSAppName" => "some name", + "CSManager" => "some url", + "CSPriority" => 10, + ), +) ``` ## Accessing Gurobi-specific attributes @@ -368,7 +383,7 @@ optimize!(model) See the [Gurobi documentation](https://www.gurobi.com/documentation/current/refman/cb_codes.html) for other information that can be queried with `GRBcbget`. -### Common Performance Pitfall with JuMP +## Common Performance Pitfall with JuMP Gurobi's API works differently than most solvers. Any changes to the model are not applied immediately, but instead go sit in a internal buffer (making any diff --git a/src/MOI_wrapper/MOI_wrapper.jl b/src/MOI_wrapper/MOI_wrapper.jl index 09c85fe1..bb668313 100644 --- a/src/MOI_wrapper/MOI_wrapper.jl +++ b/src/MOI_wrapper/MOI_wrapper.jl @@ -102,6 +102,36 @@ mutable struct _NLConstraintInfo end end +""" + Env( + params::Dict{String,Any} = Dict{String,Any}(); + started::Bool = true, + ) + +Create a new Gurobi environment object. + +Optionally pass a `params` dictionary which sets parameters for the created +environment before starting. + +## kwargs + +The extra keyword argument `started` delays starting the environment if set to +`false`. + +## Example + +```julia +using JuMP, Gurobi +const GRB_ENV = Gurobi.Env( + Dict( + "ComputeServer" => "localhost:61000", + "OutputFlag" => 0, + ); + started = true, +) +model = Model(() -> Gurobi.Optimizer(GRB_ENV)) +``` +""" mutable struct Env ptr_env::Ptr{Cvoid} # These fields keep track of how many models the `Env` is used for to help @@ -110,24 +140,17 @@ mutable struct Env finalize_called::Bool attached_models::Int - function Env(; + function Env( + params::Union{Nothing,Dict{String,Any}} = nothing; + started::Bool = true, + # These kwargs are provided for legacy backwards compatibility output_flag::Int = 1, memory_limit::Union{Nothing,Real} = nothing, - started::Bool = true, ) a = Ref{Ptr{Cvoid}}() ret = GRBemptyenv(a) env = new(a[], false, 0) _check_ret(env, ret) - ret = GRBsetintparam(env.ptr_env, GRB_INT_PAR_OUTPUTFLAG, output_flag) - _check_ret(env, ret) - if _GUROBI_VERSION >= v"9.5.0" && memory_limit !== nothing - ret = GRBsetdblparam(env, GRB_DBL_PAR_MEMLIMIT, memory_limit) - _check_ret(env, ret) - end - if started - ret = GRBstartenv(env.ptr_env) - end finalizer(env) do e e.finalize_called = true if e.attached_models == 0 @@ -135,9 +158,23 @@ mutable struct Env GRBfreeenv(e.ptr_env) e.ptr_env = C_NULL end + return + end + if params === nothing + # These two parameters are provided for backwards compability + _set_param(env.ptr_env, GRB_INT_PAR_OUTPUTFLAG, 1) + if _GUROBI_VERSION >= v"9.5.0" && memory_limit !== nothing + _set_param(env.ptr_env, GRB_DBL_PAR_MEMLIMIT, memory_limit) + end + else + for (param_name, value) in params + _set_param(env.ptr_env, param_name, value) + end + end + if started + ret = GRBstartenv(env.ptr_env) + _check_ret(env, ret) end - # Even if the loadenv fails, the pointer is still valid. - _check_ret(env, ret) return env end end @@ -170,22 +207,11 @@ function Env( server_password::Union{String,Nothing} = nothing; started::Bool = true, ) - env = Env(; started = false) - ret = GRBsetstrparam(env.ptr_env, GRB_STR_PAR_COMPUTESERVER, server_address) - _check_ret(env, ret) + params = Dict{String,Any}(GRB_STR_PAR_COMPUTESERVER => server_address) if server_password !== nothing - ret = GRBsetstrparam( - env.ptr_env, - GRB_STR_PAR_SERVERPASSWORD, - server_password, - ) - _check_ret(env, ret) + params[GRB_STR_PAR_SERVERPASSWORD] = server_password end - if started - ret = GRBstartenv(env.ptr_env) - _check_ret(env, ret) - end - return env + return Env(params; started) end Base.cconvert(::Type{Ptr{Cvoid}}, x::Env) = x @@ -685,6 +711,11 @@ function MOI.set(model::Optimizer, raw::MOI.RawOptimizerAttribute, value) env = GRBgetenv(model) param = raw.name model.params[param] = value + _set_param(env, param, value) + return +end + +function _set_param(env, param::String, value) param_type = GRBgetparamtype(env, param) ret = if param_type == -1 throw(MOI.UnsupportedAttribute(MOI.RawOptimizerAttribute(param))) diff --git a/test/MOI/MOI_wrapper.jl b/test/MOI/MOI_wrapper.jl index 361931ac..5a4f3412 100644 --- a/test/MOI/MOI_wrapper.jl +++ b/test/MOI/MOI_wrapper.jl @@ -1508,6 +1508,25 @@ function test_multiple_solution_nonlinear_objective() return end +function test_Env() + function test_err(f) + try + f() + @assert false + catch err + @test occursin("Gurobi Error 10022:", err.msg) + end + end + test_err(() -> Gurobi.Env("localhost:1234")) + test_err(() -> Gurobi.Env("localhost:1234", "password")) + test_err(() -> Gurobi.Env("localhost:1234", "password"; started = true)) + env = Gurobi.Env(; output_flag = 2, memory_limit = 1) + p = Ref{Cdouble}() + @test GRBgetdblparam(env, "MemLimit", p) == 0 + @test p[] == 1.0 + return +end + end # TestMOIWrapper TestMOIWrapper.runtests()