Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Indicator constraint support #709

Merged
merged 35 commits into from
Apr 30, 2019
Merged
Show file tree
Hide file tree
Changes from 28 commits
Commits
Show all changes
35 commits
Select commit Hold shift + click to select a range
049e4cb
first draft indicator cons
matbesancon Apr 11, 2019
a550983
fixed docstring
matbesancon Apr 11, 2019
d17ba8e
fixed LessThan
matbesancon Apr 11, 2019
3bce12e
fix import
matbesancon Apr 11, 2019
2ca54d9
last SmallerThan
matbesancon Apr 12, 2019
c3c2fa6
copy correct impl
matbesancon Apr 12, 2019
6e57763
generic vec
matbesancon Apr 12, 2019
9db213a
Revert "generic vec"
matbesancon Apr 12, 2019
9c0e8b6
changed dim and doc
matbesancon Apr 12, 2019
255c409
ActiveCond type parameter
matbesancon Apr 13, 2019
d741b61
enums + constraint
matbesancon Apr 14, 2019
232d5bf
capital enum values
matbesancon Apr 15, 2019
f940ea8
relax type constraint on S
matbesancon Apr 16, 2019
6ce7a65
latex escaping, naming
matbesancon Apr 17, 2019
18e9168
explicit set name
matbesancon Apr 18, 2019
8621773
test 1 indicator
matbesancon Apr 18, 2019
c10e1b9
penalized version
matbesancon Apr 18, 2019
e7bb25b
add reference, fixed doc, tests
matbesancon Apr 19, 2019
4d889be
Merge branch 'master' into indicator-cons
matbesancon Apr 19, 2019
ab42f21
fix test formatting and errors
matbesancon Apr 23, 2019
8da75e0
fix conflict
matbesancon Apr 23, 2019
d15fd9a
test in main, example usage
matbesancon Apr 23, 2019
4bc4d9e
split tests
matbesancon Apr 23, 2019
24af034
fix test 2
matbesancon Apr 23, 2019
f60c3c8
added test
matbesancon Apr 23, 2019
1560400
remove unsupported tests, names
matbesancon Apr 24, 2019
74836af
test passing
matbesancon Apr 24, 2019
e89d734
fix support
matbesancon Apr 24, 2019
d9565d6
fix style
matbesancon Apr 24, 2019
3c0155e
copy implementation and test
matbesancon Apr 24, 2019
c7f926e
fix comments, added test for ACTIVE_ON_ZERO
matbesancon Apr 25, 2019
9d85fee
test support indicator
matbesancon Apr 25, 2019
82d3d56
explicit comment
matbesancon Apr 25, 2019
01f8690
fix conflict
matbesancon Apr 28, 2019
caeb4c4
add mutable test
matbesancon Apr 29, 2019
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 2 additions & 2 deletions Project.toml
Original file line number Diff line number Diff line change
Expand Up @@ -8,9 +8,9 @@ BenchmarkTools = "6e4b80f9-dd63-53aa-95a3-0cdb28fa8baf"
LinearAlgebra = "37e2e46d-f89d-539d-b4ee-838fcccc9c8e"
SparseArrays = "2f01184e-e22b-5df5-ae63-d93ebab69eaf"
Unicode = "4ec0a83e-493e-50e2-b9ac-8f72acf5a8f5"
Test = "8dfed614-e22c-5e08-85e1-65c5234f0b40"

[extras]
Test = "8dfed614-e22c-5e08-85e1-65c5234f0b40"

[targets]
test = ["Test"]
test = []
1 change: 1 addition & 0 deletions docs/src/apireference.md
Original file line number Diff line number Diff line change
Expand Up @@ -251,6 +251,7 @@ Semicontinuous
Semiinteger
SOS1
SOS2
IndicatorSet
```

Functions for getting and setting properties of sets.
Expand Down
157 changes: 156 additions & 1 deletion src/Test/intlinear.jl
Original file line number Diff line number Diff line change
Expand Up @@ -356,9 +356,164 @@ function knapsacktest(model::MOI.ModelLike, config::TestConfig)
end
end

function indicator1_test(model::MOI.ModelLike, config::TestConfig)
atol = config.atol
rtol = config.rtol
# linear problem with indicator constraint
# max 2x1 + 3x2
# s.t. x1 + x2 <= 10
# z1 ==> x2 <= 8
# z2 ==> x2 + x1/5 <= 9
# z1 + z2 >= 1

MOI.empty!(model)
@test MOI.is_empty(model)

@test MOI.supports(model, MOI.ObjectiveFunction{MOI.ScalarAffineFunction{Float64}}())
@test MOI.supports(model, MOI.ObjectiveSense())
@test MOI.supports_constraint(model, MOI.SingleVariable, MOI.ZeroOne)
@test MOI.supports_constraint(model, MOI.SingleVariable, MOI.Interval{Float64})
@test MOI.supports_constraint(model, MOI.ScalarAffineFunction{Float64}, MOI.Interval{Float64})
@test MOI.supports_constraint(model, MOI.VectorAffineFunction{Float64}, MOI.IndicatorSet{MOI.ACTIVATE_ON_ONE, MOI.LessThan{Float64}})
x1 = MOI.add_variable(model)
x2 = MOI.add_variable(model)
z1 = MOI.add_variable(model)
z2 = MOI.add_variable(model)
MOI.add_constraint(model, z1, MOI.ZeroOne())
MOI.add_constraint(model, z2, MOI.ZeroOne())
f1 = MOI.VectorAffineFunction(
[MOI.VectorAffineTerm(1, MOI.ScalarAffineTerm(1.0, z1)),
MOI.VectorAffineTerm(2, MOI.ScalarAffineTerm(1.0, x2)),
],
[0., 0.]
)
iset1 = MOI.IndicatorSet{MOI.ACTIVATE_ON_ONE}(MOI.LessThan(8.))
MOI.add_constraint(model, f1, iset1)

f2 = MOI.VectorAffineFunction(
[MOI.VectorAffineTerm(1, MOI.ScalarAffineTerm(1.0, z2)),
MOI.VectorAffineTerm(2, MOI.ScalarAffineTerm(0.2, x1)),
MOI.VectorAffineTerm(2, MOI.ScalarAffineTerm(1.0, x2)),
],
[0., 0.],
)
iset2 = MOI.IndicatorSet{MOI.ACTIVATE_ON_ONE}(MOI.LessThan(9.))

MOI.add_constraint(model, f2, iset2)

# additional regular constraint
blegat marked this conversation as resolved.
Show resolved Hide resolved
MOI.add_constraint(model,
MOI.ScalarAffineFunction([MOI.ScalarAffineTerm(1.0, x1), MOI.ScalarAffineTerm(1.0, x2)], 0.0),
MOI.LessThan(10.0),
)

# disjunction z1 ⋁ z2
MOI.add_constraint(model,
MOI.ScalarAffineFunction([MOI.ScalarAffineTerm(1.0, z1), MOI.ScalarAffineTerm(1.0, z2)], 0.0),
MOI.GreaterThan(1.),
)

MOI.set(model, MOI.ObjectiveFunction{MOI.ScalarAffineFunction{Float64}}(),
MOI.ScalarAffineFunction(MOI.ScalarAffineTerm.([2., 3.], [x1, x2]), 0.)
)
MOI.set(model, MOI.ObjectiveSense(), MOI.MAX_SENSE)

if config.solve
@test MOI.get(model, MOI.TerminationStatus()) == MOI.OPTIMIZE_NOT_CALLED

MOI.optimize!(model)

@test MOI.get(model, MOI.TerminationStatus()) == MOI.OPTIMAL
@test MOI.get(model, MOI.PrimalStatus()) == MOI.FEASIBLE_POINT
@test MOI.get(model, MOI.ObjectiveValue()) ≈ 28.75 atol=atol rtol=rtol
@test MOI.get(model, MOI.VariablePrimal(), x1) ≈ 1.25 atol=atol rtol=rtol
@test MOI.get(model, MOI.VariablePrimal(), x2) ≈ 8.75 atol=atol rtol=rtol
@test MOI.get(model, MOI.VariablePrimal(), z1) ≈ 0.0 atol=atol rtol=rtol
@test MOI.get(model, MOI.VariablePrimal(), z2) ≈ 1.0 atol=atol rtol=rtol
end
end

function indicator2_test(model::MOI.ModelLike, config::TestConfig)
atol = config.atol
rtol = config.rtol
# linear problem with indicator constraint
# max 2x1 + 3x2 - 30 z2
# s.t. x1 + x2 <= 10
# z1 ==> x2 <= 8
# z2 ==> x2 + x1/5 <= 9
# z1 + z2 >= 1

MOI.empty!(model)
matbesancon marked this conversation as resolved.
Show resolved Hide resolved
@test MOI.is_empty(model)

# same model as indicator_test1 with penalty
# on z2 switches active constraint to z1
blegat marked this conversation as resolved.
Show resolved Hide resolved

x1 = MOI.add_variable(model)
x2 = MOI.add_variable(model)
z1 = MOI.add_variable(model)
z2 = MOI.add_variable(model)
MOI.add_constraint(model, z1, MOI.ZeroOne())
MOI.add_constraint(model, z2, MOI.ZeroOne())
f1 = MOI.VectorAffineFunction(
[MOI.VectorAffineTerm(1, MOI.ScalarAffineTerm(1.0, z1)),
MOI.VectorAffineTerm(2, MOI.ScalarAffineTerm(1.0, x2)),
],
[0., 0.]
)
iset1 = MOI.IndicatorSet{MOI.ACTIVATE_ON_ONE}(MOI.LessThan(8.))
MOI.add_constraint(model, f1, iset1)

f2 = MOI.VectorAffineFunction(
[MOI.VectorAffineTerm(1, MOI.ScalarAffineTerm(1.0, z2)),
MOI.VectorAffineTerm(2, MOI.ScalarAffineTerm(0.2, x1)),
MOI.VectorAffineTerm(2, MOI.ScalarAffineTerm(1.0, x2)),
],
[0., 0.],
)
iset2 = MOI.IndicatorSet{MOI.ACTIVATE_ON_ONE}(MOI.LessThan(9.))

MOI.add_constraint(model, f2, iset2)

# additional regular constraint
MOI.add_constraint(model,
MOI.ScalarAffineFunction([MOI.ScalarAffineTerm(1.0, x1), MOI.ScalarAffineTerm(1.0, x2)], 0.0),
MOI.LessThan(10.0),
)

# disjunction z1 ⋁ z2
MOI.add_constraint(model,
MOI.ScalarAffineFunction([MOI.ScalarAffineTerm(1.0, z1), MOI.ScalarAffineTerm(1.0, z2)], 0.0),
MOI.GreaterThan(1.),
)

# objective penalized on z2
MOI.set(model, MOI.ObjectiveFunction{MOI.ScalarAffineFunction{Float64}}(),
MOI.ScalarAffineFunction(MOI.ScalarAffineTerm.([2., 3., -30.], [x1, x2, z2]), 0.)
)
MOI.set(model, MOI.ObjectiveSense(), MOI.MAX_SENSE)

if config.solve
@test MOI.get(model, MOI.TerminationStatus()) == MOI.OPTIMIZE_NOT_CALLED

MOI.optimize!(model)

@test MOI.get(model, MOI.TerminationStatus()) == MOI.OPTIMAL
@test MOI.get(model, MOI.PrimalStatus()) == MOI.FEASIBLE_POINT
@test MOI.get(model, MOI.ObjectiveValue()) ≈ 28.0 atol=atol rtol=rtol
@test MOI.get(model, MOI.VariablePrimal(), x1) ≈ 2.0 atol=atol rtol=rtol
@test MOI.get(model, MOI.VariablePrimal(), x2) ≈ 8.0 atol=atol rtol=rtol
@test MOI.get(model, MOI.VariablePrimal(), z1) ≈ 1.0 atol=atol rtol=rtol
@test MOI.get(model, MOI.VariablePrimal(), z2) ≈ 0.0 atol=atol rtol=rtol
end
end
blegat marked this conversation as resolved.
Show resolved Hide resolved

const intlineartests = Dict("knapsack" => knapsacktest,
"int1" => int1test,
"int2" => int2test,
"int3" => int3test)
"int3" => int3test,
"indicator1" => indicator1_test,
"indicator2" => indicator2_test,
)

@moitestset intlinear
49 changes: 48 additions & 1 deletion src/sets.jl
Original file line number Diff line number Diff line change
Expand Up @@ -435,6 +435,53 @@ Base.isapprox(a::T, b::T; kwargs...) where {T <: Union{SOS1, SOS2}} = isapprox(a

dimension(s::Union{SOS1, SOS2}) = length(s.weights)

"""
ActivationCond

Activation condition for an indicator constraint.
Used as first type parameter of `IndicatorSet{ActiveCond,S}`.
"""
@enum ActivationCond begin
matbesancon marked this conversation as resolved.
Show resolved Hide resolved
ACTIVATE_ON_ZERO
ACTIVATE_ON_ONE
end

"""
IndicatorSet{A, S <: AbstractScalarSet}(set::S)

``\\{((y, x) \\in \\{0, 1\\} \\times \\mathbb{R}^n : y = 0 \\implies x \\in set\\}``
when `A` is `ACTIVATE_ON_ZERO` and
``\\{((y, x) \\in \\{0, 1\\} \\times \\mathbb{R}^n : y = 1 \\implies x \\in set\\}``
when `A` is `ACTIVATE_ON_ONE`.

`S` has to be a sub-type of `AbstractScalarSet`.
`A` is one of the value of the `ActivationCond` enum.
`IndicatorSet` is used with a `VectorAffineFunction` holding
the indicator variable first.
matbesancon marked this conversation as resolved.
Show resolved Hide resolved

Example: ``\\{(y, x) \\in \\{0, 1\\} \\times \\mathbb{R}^2 : y = 1 \\implies x_1 + x_2 \\leq 9 \\} ``

```julia
f = MOI.VectorAffineFunction(
[MOI.VectorAffineTerm(1, MOI.ScalarAffineTerm(1.0, z)),
MOI.VectorAffineTerm(2, MOI.ScalarAffineTerm(0.2, x1)),
MOI.VectorAffineTerm(2, MOI.ScalarAffineTerm(1.0, x2)),
],
[0., 0.],
)

indicator_set = MOI.IndicatorSet{MOI.ACTIVATE_ON_ONE}(MOI.LessThan(9.))
blegat marked this conversation as resolved.
Show resolved Hide resolved

MOI.add_constraint(model, f, indicator_set)
```
"""
struct IndicatorSet{A, S <: AbstractScalarSet} <: AbstractVectorSet
set::S
IndicatorSet{A}(set::S) where {A, S <: AbstractScalarSet} = new{A,S}(set)
end
odow marked this conversation as resolved.
Show resolved Hide resolved
blegat marked this conversation as resolved.
Show resolved Hide resolved

dimension(::IndicatorSet) = 2

# isbits types, nothing to copy
function Base.copy(set::Union{Reals, Zeros, Nonnegatives, Nonpositives,
GreaterThan, LessThan, EqualTo, Interval,
Expand All @@ -445,7 +492,7 @@ function Base.copy(set::Union{Reals, Zeros, Nonnegatives, Nonpositives,
PositiveSemidefiniteConeSquare,
LogDetConeTriangle, LogDetConeSquare,
RootDetConeTriangle, RootDetConeSquare,
Integer, ZeroOne, Semicontinuous, Semiinteger})
Integer, ZeroOne, Semicontinuous, Semiinteger, IndicatorSet})
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It is not an isbits type since it contains a vector. The copy should probably copy this vector too

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You're right, I'll re-implement copy. Should we make the vector type generic?

return set
end
Base.copy(set::S) where {S <: Union{SOS1, SOS2}} = S(copy(set.weights))
8 changes: 8 additions & 0 deletions test/Test/intlinear.jl
Original file line number Diff line number Diff line change
Expand Up @@ -21,4 +21,12 @@
MOIU.set_mock_optimize!(mock,
(mock::MOIU.MockOptimizer) -> MOIU.mock_optimize!(mock, [1, 0, 0, 1, 1]))
MOIT.knapsacktest(mock, config)
MOIU.set_mock_optimize!(mock,
(mock::MOIU.MockOptimizer) -> MOIU.mock_optimize!(mock, [1.25, 8.75, 0., 1.])
)
MOIT.indicator1_test(mock, config)
MOIU.set_mock_optimize!(mock,
(mock::MOIU.MockOptimizer) -> MOIU.mock_optimize!(mock, [2.0, 8.0, 1., 0.])
)
MOIT.indicator2_test(mock, config)
end
4 changes: 3 additions & 1 deletion test/runtests.jl
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,8 @@ include("dummy.jl")
include("attributes.jl")
end

const LessThanIndicatorSet{T} = MOI.IndicatorSet{MOI.ACTIVATE_ON_ONE, MOI.LessThan{T}}

# Needed by test spread over several files, defining it here make it easier to comment out tests
# Model supporting every MOI functions and sets
MOIU.@model(Model,
Expand All @@ -33,7 +35,7 @@ MOIU.@model(Model,
MOI.PositiveSemidefiniteConeTriangle, MOI.PositiveSemidefiniteConeSquare,
MOI.RootDetConeTriangle, MOI.RootDetConeSquare, MOI.LogDetConeTriangle,
MOI.LogDetConeSquare),
(MOI.PowerCone, MOI.DualPowerCone, MOI.SOS1, MOI.SOS2),
(MOI.PowerCone, MOI.DualPowerCone, MOI.SOS1, MOI.SOS2, LessThanIndicatorSet),
blegat marked this conversation as resolved.
Show resolved Hide resolved
(MOI.SingleVariable,),
(MOI.ScalarAffineFunction, MOI.ScalarQuadraticFunction),
(MOI.VectorOfVariables,),
Expand Down