Source code for pyomo.mpec.plugins.mpec1
# ___________________________________________________________________________
#
# Pyomo: Python Optimization Modeling Objects
# Copyright (c) 2008-2024
# National Technology and Engineering Solutions of Sandia, LLC
# Under the terms of Contract DE-NA0003525 with National Technology and
# Engineering Solutions of Sandia, LLC, the U.S. Government retains certain
# rights in this software.
# This software is distributed under the 3-clause BSD License.
# ___________________________________________________________________________
import logging
from pyomo.core.base import (
Transformation,
TransformationFactory,
Constraint,
Block,
Param,
SortComponents,
ComponentUID,
)
from pyomo.mpec.complementarity import Complementarity
from pyomo.gdp import Disjunct
logger = logging.getLogger('pyomo.core')
#
# This transformation reworks each Complementarity block to
# add a constraint that ensures the complementarity condition.
# Specifically,
#
# x1 >= 0 OR x2 >= 0
#
# becomes
#
# x1 >= 0
# x2 >= 0
# x1*x2 <= 0
#
[docs]
@TransformationFactory.register(
'mpec.simple_nonlinear',
doc="Nonlinear transformations of complementarity conditions when all variables are non-negative",
)
class MPEC1_Transformation(Transformation):
def _apply_to(self, instance, **kwds):
options = kwds.pop('options', {})
bound = kwds.pop('mpec_bound', 0.0)
#
# Create a mutable parameter that defines the value of the upper bound
# on the constraints
#
bound = options.get('mpec_bound', bound)
instance.mpec_bound = Param(mutable=True, initialize=bound)
#
# Setup transformation data
#
tdata = instance._transformation_data['mpec.simple_nonlinear']
tdata.compl_cuids = []
#
# Iterate over the model finding Complementarity components
#
for complementarity in instance.component_objects(
Complementarity,
active=True,
descend_into=(Block, Disjunct),
sort=SortComponents.deterministic,
):
block = complementarity.parent_block()
for index in sorted(complementarity.keys()):
_data = complementarity[index]
if not _data.active:
continue
_data.to_standard_form()
#
_type = getattr(_data.c, "_complementarity_type", 0)
if _type == 1:
#
# Constraint expression is bounded below, so we can replace
# constraint c with a constraint that ensures that either
# constraint c is active or variable v is at its lower bound.
#
_data.ccon = Constraint(
expr=(_data.c.body - _data.c.lower) * _data.v
<= instance.mpec_bound
)
del _data.c._complementarity_type
elif _type == 3:
#
# Variable v is bounded above and below. We can define
#
_data.ccon_l = Constraint(
expr=(_data.v - _data.v.bounds[0]) * _data.c.body
<= instance.mpec_bound
)
_data.ccon_u = Constraint(
expr=(_data.v - _data.v.bounds[1]) * _data.c.body
<= instance.mpec_bound
)
del _data.c._complementarity_type
elif _type == 2: # pragma:nocover
raise ValueError(
"to_standard_form does not generate _type 2 expressions"
)
tdata.compl_cuids.append(ComponentUID(complementarity))
block.reclassify_component_type(complementarity, Block)