From 4f3156020017db0585f237c57b51fca732de38f4 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Beno=C3=AEt=20Legat?= Date: Thu, 5 Oct 2023 09:55:29 +0200 Subject: [PATCH 1/7] Add Objective.FunctionConversionBridge --- src/Bridges/Objective/bridges/functionize.jl | 120 ++++++++++++------ src/Bridges/Objective/bridges/quadratize.jl | 74 +---------- .../Objective/bridges/vector_functionize.jl | 6 +- 3 files changed, 88 insertions(+), 112 deletions(-) diff --git a/src/Bridges/Objective/bridges/functionize.jl b/src/Bridges/Objective/bridges/functionize.jl index d30b2f361f..09ce96fbbf 100644 --- a/src/Bridges/Objective/bridges/functionize.jl +++ b/src/Bridges/Objective/bridges/functionize.jl @@ -5,103 +5,139 @@ # in the LICENSE.md file or at https://opensource.org/licenses/MIT. """ - FunctionizeBridge{T} + FunctionConversionBridge{T,F,G} <: AbstractBridge -`FunctionizeBridge` implements the following reformulations: +`FunctionConversionBridge` implements the following reformulations: - * ``\\min \\{x\\}`` into ``\\min\\{1x + 0\\}`` - * ``\\max \\{x\\}`` into ``\\max\\{1x + 0\\}`` + * ``\\min \\{g(x)\\}`` into ``\\min\\{f(x)\\}`` + * ``\\max \\{g(x)\\}`` into ``\\max\\{f(x)\\}`` -where `T` is the coefficient type of `1` and `0`. +for these pairs of functions: + + * [`MOI.ScalarAffineFunction`](@ref)` to [`MOI.ScalarQuadraticFunction`](@ref) + * [`MOI.ScalarQuadraticFunction`](@ref) to [`MOI.ScalarNonlinearFunction`](@ref) + * [`MOI.VectorAffineFunction`](@ref) to [`MOI.VectorQuadraticFunction`](@ref) ## Source node -`FunctionizeBridge` supports: +`FunctionConversionBridge` supports: - * [`MOI.ObjectiveFunction{MOI.VariableIndex}`](@ref) + * [`MOI.ObjectiveFunction{G}`](@ref) ## Target nodes -`FunctionizeBridge` creates: +`FunctionConversionBridge` creates: - * One objective node: [`MOI.ObjectiveFunction{MOI.ScalarAffineFunction{T}}`](@ref) + * One objective node: [`MOI.ObjectiveFunction{F}`](@ref) """ -struct FunctionizeBridge{T} <: AbstractBridge end - -const Functionize{T,OT<:MOI.ModelLike} = - SingleBridgeOptimizer{FunctionizeBridge{T},OT} +struct FunctionConversionBridge{T,F,G} <: AbstractBridge end function bridge_objective( - ::Type{FunctionizeBridge{T}}, + ::Type{FunctionConversionBridge{T,F,G}}, model::MOI.ModelLike, - func::MOI.VariableIndex, -) where {T} - F = MOI.ScalarAffineFunction{T} + func::G, +) where {T,F,G<:MOI.AbstractFunction} MOI.set(model, MOI.ObjectiveFunction{F}(), convert(F, func)) - return FunctionizeBridge{T}() + return FunctionConversionBridge{T,F,G}() end function supports_objective_function( - ::Type{<:FunctionizeBridge}, - ::Type{MOI.VariableIndex}, -) - return true + ::Type{<:FunctionConversionBridge{T,F}}, + ::Type{G}, +) where {T,F,G<:MOI.AbstractFunction} + return isfinite(MOI.Bridges.Constraint.conversion_cost(F, G)) end function MOI.Bridges.added_constrained_variable_types( - ::Type{<:FunctionizeBridge}, + ::Type{<:FunctionConversionBridge}, ) return Tuple{Type}[] end -function MOI.Bridges.added_constraint_types(::Type{<:FunctionizeBridge}) +function MOI.Bridges.added_constraint_types(::Type{<:FunctionConversionBridge}) return Tuple{Type,Type}[] end function MOI.Bridges.set_objective_function_type( - ::Type{FunctionizeBridge{T}}, -) where {T} - return MOI.ScalarAffineFunction{T} + ::Type{<:FunctionConversionBridge{T,F}}, +) where {T,F} + return F +end + +function concrete_bridge_type( + ::Type{<:FunctionConversionBridge{T,F}}, + ::Type{G}, +) where {T,F,G<:MOI.AbstractFunction} + return FunctionConversionBridge{T,F,G} end # Attributes, Bridge acting as a model -MOI.get(::FunctionizeBridge, ::MOI.NumberOfVariables)::Int64 = 0 +MOI.get(::FunctionConversionBridge, ::MOI.NumberOfVariables)::Int64 = 0 -function MOI.get(::FunctionizeBridge, ::MOI.ListOfVariableIndices) +function MOI.get(::FunctionConversionBridge, ::MOI.ListOfVariableIndices) return MOI.VariableIndex[] end # No variables or constraints are created in this bridge so there is nothing to # delete. -MOI.delete(::MOI.ModelLike, ::FunctionizeBridge) = nothing +MOI.delete(::MOI.ModelLike, ::FunctionConversionBridge) = nothing function MOI.set( ::MOI.ModelLike, ::MOI.ObjectiveSense, - ::FunctionizeBridge, + ::FunctionConversionBridge, ::MOI.OptimizationSense, ) - # `FunctionizeBridge` is sense agnostic, therefore, we don't need to change - # anything. + # `FunctionConversionBridge` is sense agnostic, therefore, we don't need to + # change anything. return end function MOI.get( model::MOI.ModelLike, - attr::MOI.Bridges.ObjectiveFunctionValue{MOI.VariableIndex}, - ::FunctionizeBridge{T}, -) where {T} - F = MOI.ScalarAffineFunction{T} + attr::MOI.Bridges.ObjectiveFunctionValue{G}, + ::FunctionConversionBridge{T,F,G}, +) where {T,F,G} attr_f = MOI.Bridges.ObjectiveFunctionValue{F}(attr.result_index) return MOI.get(model, attr_f) end function MOI.get( model::MOI.ModelLike, - ::MOI.ObjectiveFunction{MOI.VariableIndex}, - ::FunctionizeBridge{T}, -) where {T} - F = MOI.ScalarAffineFunction{T} + ::MOI.ObjectiveFunction{G}, + ::FunctionConversionBridge{T,F,G}, +) where {T,F,G<:MOI.AbstractFunction} func = MOI.get(model, MOI.ObjectiveFunction{F}()) - return MOI.Utilities.convert_approx(MOI.VariableIndex, func) + return MOI.Utilities.convert_approx(G, func) end + +""" + FunctionizeBridge{T} + +`FunctionizeBridge` implements the following reformulations: + + * ``\\min \\{x\\}`` into ``\\min\\{1x + 0\\}`` + * ``\\max \\{x\\}`` into ``\\max\\{1x + 0\\}`` + +where `T` is the coefficient type of `1` and `0`. + +## Source node + +`FunctionizeBridge` supports: + + * [`MOI.ObjectiveFunction{MOI.VariableIndex}`](@ref) + +## Target nodes + +`FunctionizeBridge` creates: + + * One objective node: [`MOI.ObjectiveFunction{MOI.ScalarAffineFunction{T}}`](@ref) +""" +const FunctionizeBridge{T,G} = FunctionConversionBridge{ + T, + MOI.ScalarAffineFunction{T}, + G, +} + +const Functionize{T,OT<:MOI.ModelLike} = + SingleBridgeOptimizer{FunctionizeBridge{T},OT} diff --git a/src/Bridges/Objective/bridges/quadratize.jl b/src/Bridges/Objective/bridges/quadratize.jl index 079260483f..84b0921b89 100644 --- a/src/Bridges/Objective/bridges/quadratize.jl +++ b/src/Bridges/Objective/bridges/quadratize.jl @@ -26,75 +26,11 @@ where `T` is the coefficient type of `0`. * One objective node: [`MOI.ObjectiveFunction{MOI.ScalarQuadraticFunction{T}}`](@ref) """ -struct QuadratizeBridge{T} <: AbstractBridge end +const QuadratizeBridge{T,G} = FunctionConversionBridge{ + T, + MOI.ScalarQuadraticFunction{T}, + G, +} const Quadratize{T,OT<:MOI.ModelLike} = SingleBridgeOptimizer{QuadratizeBridge{T},OT} - -function bridge_objective( - ::Type{QuadratizeBridge{T}}, - model::MOI.ModelLike, - func::MOI.ScalarAffineFunction{T}, -) where {T} - f = convert(MOI.ScalarQuadraticFunction{T}, func) - MOI.set(model, MOI.ObjectiveFunction{typeof(f)}(), f) - return QuadratizeBridge{T}() -end - -function supports_objective_function( - ::Type{<:QuadratizeBridge{T}}, - ::Type{MOI.ScalarAffineFunction{T}}, -) where {T} - return true -end - -function MOI.Bridges.added_constrained_variable_types( - ::Type{<:QuadratizeBridge}, -) - return Tuple{Type}[] -end - -function MOI.Bridges.added_constraint_types(::Type{<:QuadratizeBridge}) - return Tuple{Type,Type}[] -end - -function MOI.Bridges.set_objective_function_type( - ::Type{QuadratizeBridge{T}}, -) where {T} - return MOI.ScalarQuadraticFunction{T} -end - -# No variables or constraints are created in this bridge so there is nothing to -# delete. -MOI.delete(::MOI.ModelLike, ::QuadratizeBridge) = nothing - -function MOI.set( - ::MOI.ModelLike, - ::MOI.ObjectiveSense, - ::QuadratizeBridge, - ::MOI.OptimizationSense, -) - # `QuadratizeBridge` is sense agnostic, therefore, we don't need to change - # anything. - return -end - -function MOI.get( - model::MOI.ModelLike, - attr::MOI.Bridges.ObjectiveFunctionValue{MOI.ScalarAffineFunction{T}}, - ::QuadratizeBridge{T}, -) where {T} - F = MOI.ScalarQuadraticFunction{T} - attr_f = MOI.Bridges.ObjectiveFunctionValue{F}(attr.result_index) - return MOI.get(model, attr_f) -end - -function MOI.get( - model::MOI.ModelLike, - ::MOI.ObjectiveFunction{MOI.ScalarAffineFunction{T}}, - ::QuadratizeBridge{T}, -) where {T} - F = MOI.ScalarQuadraticFunction{T} - func = MOI.get(model, MOI.ObjectiveFunction{F}()) - return convert(MOI.ScalarAffineFunction{T}, func) -end diff --git a/src/Bridges/Objective/bridges/vector_functionize.jl b/src/Bridges/Objective/bridges/vector_functionize.jl index 80ea13285f..05baa70c2f 100644 --- a/src/Bridges/Objective/bridges/vector_functionize.jl +++ b/src/Bridges/Objective/bridges/vector_functionize.jl @@ -26,7 +26,11 @@ where `T` is the coefficient type of `1` and `0`. * One objective node: [`MOI.ObjectiveFunction{MOI.VectorAffineFunction{T}}`](@ref) """ -struct VectorFunctionizeBridge{T} <: AbstractBridge end +const VectorFunctionizeBridge{T,G} = FunctionConversionBridge{ + T, + MOI.VectorAffineFunction{T}, + G, +} const VectorFunctionize{T,OT<:MOI.ModelLike} = SingleBridgeOptimizer{VectorFunctionizeBridge{T},OT} From 882bd448e89c58b5dc8e80684d59defc5293e432 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Beno=C3=AEt=20Legat?= Date: Thu, 5 Oct 2023 09:55:55 +0200 Subject: [PATCH 2/7] Rename file --- src/Bridges/Objective/bridges/{functionize.jl => conversion.jl} | 0 1 file changed, 0 insertions(+), 0 deletions(-) rename src/Bridges/Objective/bridges/{functionize.jl => conversion.jl} (100%) diff --git a/src/Bridges/Objective/bridges/functionize.jl b/src/Bridges/Objective/bridges/conversion.jl similarity index 100% rename from src/Bridges/Objective/bridges/functionize.jl rename to src/Bridges/Objective/bridges/conversion.jl From bcb93754745a4ec30beb7ef7c6eb17051fd94372 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Beno=C3=AEt=20Legat?= Date: Thu, 5 Oct 2023 09:57:09 +0200 Subject: [PATCH 3/7] Move everything in conversion.jl --- src/Bridges/Objective/Objective.jl | 4 +- src/Bridges/Objective/bridges/conversion.jl | 66 ++++++++++- src/Bridges/Objective/bridges/quadratize.jl | 36 ------ .../Objective/bridges/vector_functionize.jl | 107 ------------------ 4 files changed, 65 insertions(+), 148 deletions(-) delete mode 100644 src/Bridges/Objective/bridges/quadratize.jl delete mode 100644 src/Bridges/Objective/bridges/vector_functionize.jl diff --git a/src/Bridges/Objective/Objective.jl b/src/Bridges/Objective/Objective.jl index 945ab057fc..ae0c72e5c5 100644 --- a/src/Bridges/Objective/Objective.jl +++ b/src/Bridges/Objective/Objective.jl @@ -12,10 +12,8 @@ include("bridge.jl") include("map.jl") include("single_bridge_optimizer.jl") -include("bridges/functionize.jl") -include("bridges/quadratize.jl") +include("bridges/conversion.jl") include("bridges/slack.jl") -include("bridges/vector_functionize.jl") include("bridges/vector_slack.jl") """ diff --git a/src/Bridges/Objective/bridges/conversion.jl b/src/Bridges/Objective/bridges/conversion.jl index 09ce96fbbf..cda110e3d1 100644 --- a/src/Bridges/Objective/bridges/conversion.jl +++ b/src/Bridges/Objective/bridges/conversion.jl @@ -112,7 +112,7 @@ function MOI.get( end """ - FunctionizeBridge{T} + FunctionizeBridge{T,G} `FunctionizeBridge` implements the following reformulations: @@ -125,7 +125,7 @@ where `T` is the coefficient type of `1` and `0`. `FunctionizeBridge` supports: - * [`MOI.ObjectiveFunction{MOI.VariableIndex}`](@ref) + * [`MOI.ObjectiveFunction{G}`](@ref) ## Target nodes @@ -141,3 +141,65 @@ const FunctionizeBridge{T,G} = FunctionConversionBridge{ const Functionize{T,OT<:MOI.ModelLike} = SingleBridgeOptimizer{FunctionizeBridge{T},OT} + +""" + QuadratizeBridge{T,G} + +`QuadratizeBridge` implements the following reformulations: + + * ``\\min \\{a^\\top x + b\\}`` into ``\\min\\{x^\\top \\mathbf{0} x + a^\\top x + b\\}`` + * ``\\max \\{a^\\top x + b\\}`` into ``\\max\\{x^\\top \\mathbf{0} x + a^\\top x + b\\}`` + +where `T` is the coefficient type of `0`. + +## Source node + +`QuadratizeBridge` supports: + + * [`MOI.ObjectiveFunction{G}`](@ref) + +## Target nodes + +`QuadratizeBridge` creates: + + * One objective node: [`MOI.ObjectiveFunction{MOI.ScalarQuadraticFunction{T}}`](@ref) +""" +const QuadratizeBridge{T,G} = FunctionConversionBridge{ + T, + MOI.ScalarQuadraticFunction{T}, + G, +} + +const Quadratize{T,OT<:MOI.ModelLike} = + SingleBridgeOptimizer{QuadratizeBridge{T},OT} + +""" + VectorFunctionizeBridge{T,G} + +`VectorFunctionizeBridge` implements the following reformulations: + + * ``\\min \\{x\\}`` into ``\\min\\{1x + 0\\}`` + * ``\\max \\{x\\}`` into ``\\max\\{1x + 0\\}`` + +where `T` is the coefficient type of `1` and `0`. + +## Source node + +`VectorFunctionizeBridge` supports: + + * [`MOI.ObjectiveFunction{G}`](@ref) + +## Target nodes + +`VectorFunctionizeBridge` creates: + + * One objective node: [`MOI.ObjectiveFunction{MOI.VectorAffineFunction{T}}`](@ref) +""" +const VectorFunctionizeBridge{T,G} = FunctionConversionBridge{ + T, + MOI.VectorAffineFunction{T}, + G, +} + +const VectorFunctionize{T,OT<:MOI.ModelLike} = + SingleBridgeOptimizer{VectorFunctionizeBridge{T},OT} diff --git a/src/Bridges/Objective/bridges/quadratize.jl b/src/Bridges/Objective/bridges/quadratize.jl deleted file mode 100644 index 84b0921b89..0000000000 --- a/src/Bridges/Objective/bridges/quadratize.jl +++ /dev/null @@ -1,36 +0,0 @@ -# Copyright (c) 2017: Miles Lubin and contributors -# Copyright (c) 2017: Google Inc. -# -# Use of this source code is governed by an MIT-style license that can be found -# in the LICENSE.md file or at https://opensource.org/licenses/MIT. - -""" - QuadratizeBridge{T} - -`QuadratizeBridge` implements the following reformulations: - - * ``\\min \\{a^\\top x + b\\}`` into ``\\min\\{x^\\top \\mathbf{0} x + a^\\top x + b\\}`` - * ``\\max \\{a^\\top x + b\\}`` into ``\\max\\{x^\\top \\mathbf{0} x + a^\\top x + b\\}`` - -where `T` is the coefficient type of `0`. - -## Source node - -`QuadratizeBridge` supports: - - * [`MOI.ObjectiveFunction{MOI.ScalarAffineFunction{T}}`](@ref) - -## Target nodes - -`QuadratizeBridge` creates: - - * One objective node: [`MOI.ObjectiveFunction{MOI.ScalarQuadraticFunction{T}}`](@ref) -""" -const QuadratizeBridge{T,G} = FunctionConversionBridge{ - T, - MOI.ScalarQuadraticFunction{T}, - G, -} - -const Quadratize{T,OT<:MOI.ModelLike} = - SingleBridgeOptimizer{QuadratizeBridge{T},OT} diff --git a/src/Bridges/Objective/bridges/vector_functionize.jl b/src/Bridges/Objective/bridges/vector_functionize.jl deleted file mode 100644 index 05baa70c2f..0000000000 --- a/src/Bridges/Objective/bridges/vector_functionize.jl +++ /dev/null @@ -1,107 +0,0 @@ -# Copyright (c) 2017: Miles Lubin and contributors -# Copyright (c) 2017: Google Inc. -# -# Use of this source code is governed by an MIT-style license that can be found -# in the LICENSE.md file or at https://opensource.org/licenses/MIT. - -""" - VectorFunctionizeBridge{T} - -`VectorFunctionizeBridge` implements the following reformulations: - - * ``\\min \\{x\\}`` into ``\\min\\{1x + 0\\}`` - * ``\\max \\{x\\}`` into ``\\max\\{1x + 0\\}`` - -where `T` is the coefficient type of `1` and `0`. - -## Source node - -`VectorFunctionizeBridge` supports: - - * [`MOI.ObjectiveFunction{MOI.VectorOfVariables}`](@ref) - -## Target nodes - -`VectorFunctionizeBridge` creates: - - * One objective node: [`MOI.ObjectiveFunction{MOI.VectorAffineFunction{T}}`](@ref) -""" -const VectorFunctionizeBridge{T,G} = FunctionConversionBridge{ - T, - MOI.VectorAffineFunction{T}, - G, -} - -const VectorFunctionize{T,OT<:MOI.ModelLike} = - SingleBridgeOptimizer{VectorFunctionizeBridge{T},OT} - -function bridge_objective( - ::Type{VectorFunctionizeBridge{T}}, - model::MOI.ModelLike, - f::MOI.VectorOfVariables, -) where {T} - F = MOI.VectorAffineFunction{T} - MOI.set(model, MOI.ObjectiveFunction{F}(), convert(F, f)) - return VectorFunctionizeBridge{T}() -end - -function supports_objective_function( - ::Type{<:VectorFunctionizeBridge}, - ::Type{MOI.VectorOfVariables}, -) - return true -end - -function MOI.Bridges.added_constrained_variable_types( - ::Type{<:VectorFunctionizeBridge}, -) - return Tuple{Type}[] -end - -function MOI.Bridges.added_constraint_types(::Type{<:VectorFunctionizeBridge}) - return Tuple{Type,Type}[] -end - -function MOI.Bridges.set_objective_function_type( - ::Type{VectorFunctionizeBridge{T}}, -) where {T} - return MOI.VectorAffineFunction{T} -end - -MOI.get(::VectorFunctionizeBridge, ::MOI.NumberOfVariables)::Int64 = 0 - -function MOI.get(::VectorFunctionizeBridge, ::MOI.ListOfVariableIndices) - return MOI.VariableIndex[] -end - -MOI.delete(::MOI.ModelLike, ::VectorFunctionizeBridge) = nothing - -function MOI.set( - ::MOI.ModelLike, - ::MOI.ObjectiveSense, - ::VectorFunctionizeBridge, - ::MOI.OptimizationSense, -) - # `VectorFunctionizeBridge` is sense agnostic, therefore, we don't need to - # change anything. - return -end - -function MOI.get( - model::MOI.ModelLike, - attr::MOI.Bridges.ObjectiveFunctionValue{MOI.VectorOfVariables}, - ::VectorFunctionizeBridge{T}, -) where {T} - F = MOI.VectorAffineFunction{T} - attr_f = MOI.Bridges.ObjectiveFunctionValue{F}(attr.result_index) - return MOI.get(model, attr_f) -end - -function MOI.get( - model::MOI.ModelLike, - ::MOI.ObjectiveFunction{MOI.VectorOfVariables}, - ::VectorFunctionizeBridge{T}, -) where {T} - f = MOI.get(model, MOI.ObjectiveFunction{MOI.VectorAffineFunction{T}}()) - return convert(MOI.VectorOfVariables, f) -end From 12696b0e7980dabb50e91df6b6ab4fc08ddd630b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Beno=C3=AEt=20Legat?= Date: Thu, 5 Oct 2023 10:15:24 +0200 Subject: [PATCH 4/7] Fix format --- src/Bridges/Objective/bridges/conversion.jl | 21 ++++++--------------- 1 file changed, 6 insertions(+), 15 deletions(-) diff --git a/src/Bridges/Objective/bridges/conversion.jl b/src/Bridges/Objective/bridges/conversion.jl index cda110e3d1..5c4b6ddce2 100644 --- a/src/Bridges/Objective/bridges/conversion.jl +++ b/src/Bridges/Objective/bridges/conversion.jl @@ -133,11 +133,8 @@ where `T` is the coefficient type of `1` and `0`. * One objective node: [`MOI.ObjectiveFunction{MOI.ScalarAffineFunction{T}}`](@ref) """ -const FunctionizeBridge{T,G} = FunctionConversionBridge{ - T, - MOI.ScalarAffineFunction{T}, - G, -} +const FunctionizeBridge{T,G} = + FunctionConversionBridge{T,MOI.ScalarAffineFunction{T},G} const Functionize{T,OT<:MOI.ModelLike} = SingleBridgeOptimizer{FunctionizeBridge{T},OT} @@ -164,11 +161,8 @@ where `T` is the coefficient type of `0`. * One objective node: [`MOI.ObjectiveFunction{MOI.ScalarQuadraticFunction{T}}`](@ref) """ -const QuadratizeBridge{T,G} = FunctionConversionBridge{ - T, - MOI.ScalarQuadraticFunction{T}, - G, -} +const QuadratizeBridge{T,G} = + FunctionConversionBridge{T,MOI.ScalarQuadraticFunction{T},G} const Quadratize{T,OT<:MOI.ModelLike} = SingleBridgeOptimizer{QuadratizeBridge{T},OT} @@ -195,11 +189,8 @@ where `T` is the coefficient type of `1` and `0`. * One objective node: [`MOI.ObjectiveFunction{MOI.VectorAffineFunction{T}}`](@ref) """ -const VectorFunctionizeBridge{T,G} = FunctionConversionBridge{ - T, - MOI.VectorAffineFunction{T}, - G, -} +const VectorFunctionizeBridge{T,G} = + FunctionConversionBridge{T,MOI.VectorAffineFunction{T},G} const VectorFunctionize{T,OT<:MOI.ModelLike} = SingleBridgeOptimizer{VectorFunctionizeBridge{T},OT} From 9519b7c00083e3f60a90fe75ab2cee91906e1179 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Beno=C3=AEt=20Legat?= Date: Fri, 6 Oct 2023 08:45:43 +0200 Subject: [PATCH 5/7] Fixes --- docs/src/submodules/Bridges/list_of_bridges.md | 3 ++- src/Bridges/Objective/bridges/conversion.jl | 6 +++--- test/Bridges/lazy_bridge_optimizer.jl | 4 ++-- 3 files changed, 7 insertions(+), 6 deletions(-) diff --git a/docs/src/submodules/Bridges/list_of_bridges.md b/docs/src/submodules/Bridges/list_of_bridges.md index 3a7c8e2c4b..60e3236e65 100644 --- a/docs/src/submodules/Bridges/list_of_bridges.md +++ b/docs/src/submodules/Bridges/list_of_bridges.md @@ -88,8 +88,9 @@ These bridges are subtypes of [`Bridges.Objective.AbstractBridge`](@ref). ```@docs Bridges.Objective.FunctionizeBridge Bridges.Objective.QuadratizeBridge -Bridges.Objective.SlackBridge Bridges.Objective.VectorFunctionizeBridge +Bridges.Objective.FunctionConversionBridge +Bridges.Objective.SlackBridge Bridges.Objective.VectorSlackBridge ``` diff --git a/src/Bridges/Objective/bridges/conversion.jl b/src/Bridges/Objective/bridges/conversion.jl index 5c4b6ddce2..c0e573f551 100644 --- a/src/Bridges/Objective/bridges/conversion.jl +++ b/src/Bridges/Objective/bridges/conversion.jl @@ -112,7 +112,7 @@ function MOI.get( end """ - FunctionizeBridge{T,G} + FunctionizeBridge{T,G} <: FunctionConversionBridge{T,MOI.ScalarAffineFunction{T},G} `FunctionizeBridge` implements the following reformulations: @@ -140,7 +140,7 @@ const Functionize{T,OT<:MOI.ModelLike} = SingleBridgeOptimizer{FunctionizeBridge{T},OT} """ - QuadratizeBridge{T,G} + QuadratizeBridge{T,G} <: FunctionConversionBridge{T,MOI.ScalarQuadraticFunction{T},G} `QuadratizeBridge` implements the following reformulations: @@ -168,7 +168,7 @@ const Quadratize{T,OT<:MOI.ModelLike} = SingleBridgeOptimizer{QuadratizeBridge{T},OT} """ - VectorFunctionizeBridge{T,G} + VectorFunctionizeBridge{T,G} <: FunctionConversionBridge{T,MOI.VectorAffineFunction{T},G} `VectorFunctionizeBridge` implements the following reformulations: diff --git a/test/Bridges/lazy_bridge_optimizer.jl b/test/Bridges/lazy_bridge_optimizer.jl index 4695af4451..8fa70aa3e0 100644 --- a/test/Bridges/lazy_bridge_optimizer.jl +++ b/test/Bridges/lazy_bridge_optimizer.jl @@ -746,7 +746,7 @@ Bridge graph with 1 variable nodes, 3 constraint nodes and 0 objective nodes. MOI.Bridges.add_bridge(bridged, MOI.Bridges.Objective.FunctionizeBridge{T}) @test MOI.supports(bridged, MOI.ObjectiveFunction{MOI.VariableIndex}()) @test MOI.Bridges.bridge_type(bridged, MOI.VariableIndex) == - MOI.Bridges.Objective.FunctionizeBridge{T} + MOI.Bridges.Objective.FunctionizeBridge{T,MOI.VariableIndex} @test MOI.supports(bridged, attr) @test MOI.Bridges.bridge_type(bridged, F) == MOI.Bridges.Objective.SlackBridge{T,F,F} @@ -1085,7 +1085,7 @@ Bridge graph with 2 variable nodes, 5 constraint nodes and 2 objective nodes. (4) `MOI.VectorAffineFunction{$T}`-in-`MOI.Zeros` constraints are bridged (distance 1) by $(MOI.Bridges.Constraint.ScalarizeBridge{T,MOI.ScalarAffineFunction{T},MOI.EqualTo{T}}). (5) `MOI.ScalarQuadraticFunction{$T}`-in-`MOI.LessThan{$T}` constraints are bridged (distance 5) by $(MOI.Bridges.Constraint.QuadtoSOCBridge{T}). |1| objective function of type `MOI.ScalarQuadraticFunction{$T}` is bridged (distance 13) by $(MOI.Bridges.Objective.SlackBridge{T,MOI.ScalarQuadraticFunction{T},MOI.ScalarQuadraticFunction{T}}). - |2| objective function of type `MOI.VariableIndex` is bridged (distance 1) by $(MOI.Bridges.Objective.FunctionizeBridge{T}). + |2| objective function of type `MOI.VariableIndex` is bridged (distance 1) by $(MOI.Bridges.Objective.FunctionizeBridge{T,MOI.VariableIndex}). """, ) end From b67ee5f595f73b68878461134ba5f48c1bd604e1 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Beno=C3=AEt=20Legat?= Date: Mon, 9 Oct 2023 09:49:03 +0200 Subject: [PATCH 6/7] Add test --- test/Bridges/Objective/conversion.jl | 62 ++++++++++++++++++++++++++++ 1 file changed, 62 insertions(+) create mode 100644 test/Bridges/Objective/conversion.jl diff --git a/test/Bridges/Objective/conversion.jl b/test/Bridges/Objective/conversion.jl new file mode 100644 index 0000000000..02504aaf3f --- /dev/null +++ b/test/Bridges/Objective/conversion.jl @@ -0,0 +1,62 @@ +# Copyright (c) 2017: Miles Lubin and contributors +# Copyright (c) 2017: Google Inc. +# +# Use of this source code is governed by an MIT-style license that can be found +# in the LICENSE.md file or at https://opensource.org/licenses/MIT. + +module TestObjectiveConversion + +using Test + +import MathOptInterface as MOI + +function runtests() + for name in names(@__MODULE__; all = true) + if startswith("$(name)", "test_") + @testset "$(name)" begin + getfield(@__MODULE__, name)() + end + end + end + return +end + +include("../utilities.jl") + +struct VariableDifference <: MOI.AbstractScalarFunction + x::MOI.VariableIndex + y::MOI.VariableIndex +end + +function MOI.Bridges.Constraint.conversion_cost( + ::Type{<:MOI.ScalarAffineFunction}, + ::Type{VariableDifference}, +) + return 1.0 +end + +function MOI.convert( + ::Type{MOI.ScalarAffineFunction{T}}, + f::VariableDifference, +) where {T} + return one(T) * f.x - one(T) * f.y +end + +function test_variable_difference(T = Float64) + F = MOI.ScalarAffineFunction{T} + B = MOI.Bridges.Objective.FunctionConversionBridge{T,F} + inner = MOI.Utilities.Model{T}() + model = MOI.Bridges.Objective.SingleBridgeOptimizer{B}(inner) + x = MOI.add_variable(model) + y = MOI.add_variable(model) + g = VariableDifference(x, y) + G = typeof(g) + MOI.set(model, MOI.ObjectiveFunction{G}(), g) + f = one(T) * x - one(T) * y + @test MOI.get(inner, MOI.ObjectiveFunction{F}()) ≈ f + return +end + +end # module + +TestObjectiveConversion.runtests() From 267ce1c872d761d9613968535f554ee6ac92b399 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Beno=C3=AEt=20Legat?= Date: Mon, 9 Oct 2023 10:29:22 +0200 Subject: [PATCH 7/7] Fix --- test/Bridges/debug.jl | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/test/Bridges/debug.jl b/test/Bridges/debug.jl index e310ea3fa0..386116be21 100644 --- a/test/Bridges/debug.jl +++ b/test/Bridges/debug.jl @@ -73,7 +73,7 @@ function test_print_active_bridges() | may introduce: | * Unsupported objective: MOI.VariableIndex | | bridged by: - | | MOIB.Objective.FunctionizeBridge{Float64} + | | MOIB.Objective.FunctionConversionBridge{Float64, MOI.ScalarAffineFunction{Float64}, MOI.VariableIndex} | | may introduce: | | * Supported objective: MOI.ScalarAffineFunction{Float64} | * Unsupported constraint: MOI.ScalarQuadraticFunction{Float64}-in-MOI.GreaterThan{Float64} @@ -188,7 +188,7 @@ function test_print_active_bridges_objective_bridged() """ * Unsupported objective: MOI.VariableIndex | bridged by: - | MOIB.Objective.FunctionizeBridge{Float64} + | MOIB.Objective.FunctionConversionBridge{Float64, MOI.ScalarAffineFunction{Float64}, MOI.VariableIndex} | may introduce: | * Supported objective: MOI.ScalarAffineFunction{Float64} """