Skip to content

Commit b53f315

Browse files
committed
Switch to MOI.modify
1 parent 54c6291 commit b53f315

File tree

3 files changed

+37
-42
lines changed

3 files changed

+37
-42
lines changed

docs/src/submodules/Utilities/overview.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -330,7 +330,7 @@ julia> MOI.set(model, MOI.VariableName(), x, "x")
330330
331331
julia> c = MOI.add_constraint(model, 1.0 * x, MOI.LessThan(2.0));
332332
333-
julia> MOI.set(model, MOI.Utilities.FeasibilityRelaxation(Dict(c => 2.0)))
333+
julia> MOI.modify(model, MOI.Utilities.FeasibilityRelaxation(Dict(c => 2.0)))
334334
335335
julia> print(model)
336336
Minimize ScalarAffineFunction{Float64}:

src/Utilities/feasibility_relaxation.jl

+34-39
Original file line numberDiff line numberDiff line change
@@ -7,15 +7,15 @@
77
"""
88
FeasibilityRelaxation(
99
penalties = Dict{MOI.ConstraintIndex,Float64}(),
10-
) <: MOI.AbstractModelAttribute
10+
) <: MOI.AbstractFunctionModification
1111
12-
A model attribute that, when set, destructively modifies the model in-place to
13-
create a feasibility relaxation.
12+
A problem modifier that, when passed to [`MOI.modify`](@ref), destructively
13+
modifies the model in-place to create a feasibility relaxation.
1414
1515
!!! warning
1616
This is a destructive routine that modifies the model in-place. If you don't
17-
want to modify the original model, use `copy_model` to create a copy before
18-
setting this attribute.
17+
want to modify the original model, use `JuMP.copy_model` to create a copy
18+
before setting this attribute.
1919
2020
## Reformulation
2121
@@ -25,13 +25,18 @@ term into the objective of ``a \\times (y + z)`` (if minimizing, else ``-a``),
2525
where `a` is the value in the `penalties` dictionary associated with the
2626
constraint that is being relaxed. If no value exists, the default is `1.0`.
2727
28-
The feasibility relaxation is limited to modifying constraint types for which
29-
`MOI.supports(model, ::FeasibilityRelaxation, MOI.ConstraintIndex{F,S})` is
30-
`true`. By default, this is only true for [`MOI.ScalarAffineFunction`](@ref) and
31-
[`MOI.ScalarQuadraticFunction`](@ref) constraints in the linear sets
32-
[`MOI.LessThan`](@ref), [`MOI.GreaterThan`](@ref), [`MOI.EqualTo`](@ref) and
33-
[`MOI.Interval`](@ref). It does not include variable bound or integrality
34-
constraints, because these cannot be modified in-place.
28+
When `S` is [`MOI.LessThan`](@ref) or [`MOI.GreaterThan`](@ref), we omit `y` or
29+
`z` respectively as a performance optimization.
30+
31+
## Supported constraint types
32+
33+
The feasibility relaxation is currently limited to modifying
34+
[`MOI.ScalarAffineFunction`](@ref) and [`MOI.ScalarQuadraticFunction`](@ref)
35+
constraints in the linear sets [`MOI.LessThan`](@ref), [`MOI.GreaterThan`](@ref),
36+
[`MOI.EqualTo`](@ref) and [`MOI.Interval`](@ref).
37+
38+
It does not include variable bound or integrality constraints, because these
39+
cannot be modified in-place.
3540
3641
## Example
3742
@@ -42,7 +47,7 @@ julia> x = MOI.add_variable(model);
4247
4348
julia> c = MOI.add_constraint(model, 1.0 * x, MOI.LessThan(2.0));
4449
45-
julia> MOI.set(model, MOI.Utilities.FeasibilityRelaxation(Dict(c => 2.0)))
50+
julia> MOI.modify(model, MOI.Utilities.FeasibilityRelaxation(Dict(c => 2.0)))
4651
4752
julia> print(model)
4853
Minimize ScalarAffineFunction{Float64}:
@@ -57,7 +62,7 @@ VariableIndex-in-GreaterThan{Float64}
5762
v[2] >= 0.0
5863
```
5964
"""
60-
mutable struct FeasibilityRelaxation{T} <: MOI.AbstractModelAttribute
65+
mutable struct FeasibilityRelaxation{T} <: MOI.AbstractFunctionModification
6166
penalties::Dict{MOI.ConstraintIndex,T}
6267
scale::T
6368
function FeasibilityRelaxation(p::Dict{MOI.ConstraintIndex,T}) where {T}
@@ -73,7 +78,7 @@ function FeasibilityRelaxation(d::Dict{<:MOI.ConstraintIndex,T}) where {T}
7378
return FeasibilityRelaxation(convert(Dict{MOI.ConstraintIndex,T}, d))
7479
end
7580

76-
function MOI.set(
81+
function MOI.modify(
7782
model::MOI.ModelLike,
7883
relax::FeasibilityRelaxation{T},
7984
) where {T}
@@ -89,45 +94,35 @@ function MOI.set(
8994
relax.scale = -one(T)
9095
end
9196
for (F, S) in MOI.get(model, MOI.ListOfConstraintTypesPresent())
92-
MOI.set(model, relax, F, S)
97+
_modify_feasibility_relaxation(model, relax, F, S)
9398
end
9499
return
95100
end
96101

97-
function MOI.set(
102+
function _modify_feasibility_relaxation(
98103
model::MOI.ModelLike,
99104
relax::FeasibilityRelaxation,
100105
::Type{F},
101106
::Type{S},
102107
) where {F,S}
103-
if MOI.supports(model, relax, MOI.ConstraintIndex{F,S})
104-
for ci in MOI.get(model, MOI.ListOfConstraintIndices{F,S}())
105-
MOI.set(model, relax, ci)
106-
end
108+
for ci in MOI.get(model, MOI.ListOfConstraintIndices{F,S}())
109+
MOI.modify(model, ci, relax)
107110
end
108111
return
109112
end
110113

111-
function MOI.supports(
114+
function MOI.modify(
112115
::MOI.ModelLike,
116+
::MOI.ConstraintIndex,
113117
::FeasibilityRelaxation,
114-
::Type{<:MOI.ConstraintIndex{F,<:MOI.AbstractScalarSet}},
115-
) where {T,F<:Union{MOI.ScalarAffineFunction{T},MOI.ScalarQuadraticFunction{T}}}
116-
return true
118+
)
119+
return # Silently skip modifying other constraint types.
117120
end
118121

119-
function MOI.supports_fallback(
120-
::MOI.ModelLike,
121-
::FeasibilityRelaxation,
122-
::Type{MOI.ConstraintIndex{F,S}},
123-
) where {F,S}
124-
return false
125-
end
126-
127-
function MOI.set(
122+
function MOI.modify(
128123
model::MOI.ModelLike,
129-
relax::FeasibilityRelaxation,
130124
ci::MOI.ConstraintIndex{F,<:MOI.AbstractScalarSet},
125+
relax::FeasibilityRelaxation,
131126
) where {T,F<:Union{MOI.ScalarAffineFunction{T},MOI.ScalarQuadraticFunction{T}}}
132127
y = MOI.add_variable(model)
133128
z = MOI.add_variable(model)
@@ -143,10 +138,10 @@ function MOI.set(
143138
return
144139
end
145140

146-
function MOI.set(
141+
function MOI.modify(
147142
model::MOI.ModelLike,
148-
relax::FeasibilityRelaxation,
149143
ci::MOI.ConstraintIndex{F,MOI.GreaterThan{T}},
144+
relax::FeasibilityRelaxation,
150145
) where {T,F<:Union{MOI.ScalarAffineFunction{T},MOI.ScalarQuadraticFunction{T}}}
151146
# Performance optimization: we don't need the z relaxation variable.
152147
y = MOI.add_variable(model)
@@ -159,10 +154,10 @@ function MOI.set(
159154
return
160155
end
161156

162-
function MOI.set(
157+
function MOI.modify(
163158
model::MOI.ModelLike,
164-
relax::FeasibilityRelaxation,
165159
ci::MOI.ConstraintIndex{F,MOI.LessThan{T}},
160+
relax::FeasibilityRelaxation,
166161
) where {T,F<:Union{MOI.ScalarAffineFunction{T},MOI.ScalarQuadraticFunction{T}}}
167162
# Performance optimization: we don't need the y relaxation variable.
168163
z = MOI.add_variable(model)

test/Utilities/feasibility_relaxation.jl

+2-2
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,7 @@ end
2525
function _test_roundtrip(src_str, relaxed_str)
2626
model = MOI.Utilities.Model{Float64}()
2727
MOI.Utilities.loadfromstring!(model, src_str)
28-
MOI.set(model, MOI.Utilities.FeasibilityRelaxation())
28+
MOI.modify(model, MOI.Utilities.FeasibilityRelaxation())
2929
dest = MOI.Utilities.Model{Float64}()
3030
MOI.Utilities.loadfromstring!(dest, relaxed_str)
3131
MOI.Bridges._test_structural_identical(model, dest)
@@ -212,7 +212,7 @@ function test_penalties()
212212
model = MOI.Utilities.Model{Float64}()
213213
x = MOI.add_variable(model)
214214
c = MOI.add_constraint(model, 1.0 * x, MOI.EqualTo(2.0))
215-
MOI.set(model, MOI.Utilities.FeasibilityRelaxation(Dict(c => 2.0)))
215+
MOI.modify(model, MOI.Utilities.FeasibilityRelaxation(Dict(c => 2.0)))
216216
@test sprint(print, model) === """
217217
Minimize ScalarAffineFunction{Float64}:
218218
0.0 + 2.0 v[2] + 2.0 v[3]

0 commit comments

Comments
 (0)