Files
ortools-clone/ortools/linear_solver/python/model_builder.py
Laurent Perron 622483b682 mostly reformat
2024-04-08 11:34:45 +02:00

2061 lines
74 KiB
Python

# Copyright 2010-2024 Google LLC
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
"""Methods for building and solving model_builder models.
The following two sections describe the main
methods for building and solving those models.
* [`Model`](#model_builder.Model): Methods for creating
models, including variables and constraints.
* [`Solver`](#model_builder.Solver): Methods for solving
a model and evaluating solutions.
Additional methods for solving Model models:
* [`Constraint`](#model_builder.Constraint): A few utility methods for modifying
constraints created by `Model`.
* [`LinearExpr`](#model_builder.LinearExpr): Methods for creating constraints
and the objective from large arrays of coefficients.
Other methods and functions listed are primarily used for developing OR-Tools,
rather than for solving specific optimization problems.
"""
import abc
import dataclasses
import math
import numbers
import typing
from typing import Callable, List, Optional, Sequence, Tuple, Union, cast
import numpy as np
from numpy import typing as npt
import pandas as pd
from ortools.linear_solver import linear_solver_pb2
from ortools.linear_solver.python import model_builder_helper as mbh
from ortools.linear_solver.python import model_builder_numbers as mbn
# Custom types.
NumberT = Union[int, float, numbers.Real, np.number]
IntegerT = Union[int, numbers.Integral, np.integer]
LinearExprT = Union["LinearExpr", NumberT]
ConstraintT = Union["_BoundedLinearExpr", bool]
_IndexOrSeries = Union[pd.Index, pd.Series]
_VariableOrConstraint = Union["LinearConstraint", "Variable"]
# Forward solve statuses.
SolveStatus = mbh.SolveStatus
# pylint: disable=protected-access
class LinearExpr(metaclass=abc.ABCMeta):
"""Holds an linear expression.
A linear expression is built from constants and variables.
For example, `x + 2.0 * (y - z + 1.0)`.
Linear expressions are used in Model models in constraints and in the
objective:
* You can define linear constraints as in:
```
model.add(x + 2 * y <= 5.0)
model.add(sum(array_of_vars) == 5.0)
```
* In Model, the objective is a linear expression:
```
model.minimize(x + 2.0 * y + z)
```
* For large arrays, using the LinearExpr class is faster that using the python
`sum()` function. You can create constraints and the objective from lists of
linear expressions or coefficients as follows:
```
model.minimize(model_builder.LinearExpr.sum(expressions))
model.add(model_builder.LinearExpr.weighted_sum(expressions, coeffs) >= 0)
```
"""
@classmethod
def sum( # pytype: disable=annotation-type-mismatch # numpy-scalars
cls, expressions: Sequence[LinearExprT], *, constant: NumberT = 0.0
) -> LinearExprT:
"""Creates `sum(expressions) + constant`.
It can perform simple simplifications and returns different objects,
including the input.
Args:
expressions: a sequence of linear expressions or constants.
constant: a numerical constant.
Returns:
a LinearExpr instance or a numerical constant.
"""
checked_constant: np.double = mbn.assert_is_a_number(constant)
if not expressions:
return checked_constant
if len(expressions) == 1 and mbn.is_zero(checked_constant):
return expressions[0]
return LinearExpr.weighted_sum(
expressions, np.ones(len(expressions)), constant=checked_constant
)
@classmethod
def weighted_sum( # pytype: disable=annotation-type-mismatch # numpy-scalars
cls,
expressions: Sequence[LinearExprT],
coefficients: Sequence[NumberT],
*,
constant: NumberT = 0.0,
) -> Union[NumberT, "_LinearExpression"]:
"""Creates `sum(expressions[i] * coefficients[i]) + constant`.
It can perform simple simplifications and returns different object,
including the input.
Args:
expressions: a sequence of linear expressions or constants.
coefficients: a sequence of numerical constants.
constant: a numerical constant.
Returns:
a _LinearExpression instance or a numerical constant.
"""
if len(expressions) != len(coefficients):
raise ValueError(
"LinearExpr.weighted_sum: expressions and coefficients have"
" different lengths"
)
checked_constant: np.double = mbn.assert_is_a_number(constant)
if not expressions:
return checked_constant
return _sum_as_flat_linear_expression(
to_process=list(zip(expressions, coefficients)), offset=checked_constant
)
@classmethod
def term( # pytype: disable=annotation-type-mismatch # numpy-scalars
cls,
expression: LinearExprT,
coefficient: NumberT,
*,
constant: NumberT = 0.0,
) -> LinearExprT:
"""Creates `expression * coefficient + constant`.
It can perform simple simplifications and returns different object,
including the input.
Args:
expression: a linear expression or a constant.
coefficient: a numerical constant.
constant: a numerical constant.
Returns:
a LinearExpr instance or a numerical constant.
"""
checked_coefficient: np.double = mbn.assert_is_a_number(coefficient)
checked_constant: np.double = mbn.assert_is_a_number(constant)
if mbn.is_zero(checked_coefficient):
return checked_constant
if mbn.is_one(checked_coefficient) and mbn.is_zero(checked_constant):
return expression
if mbn.is_a_number(expression):
return np.double(expression) * checked_coefficient + checked_constant
if isinstance(expression, LinearExpr):
return _as_flat_linear_expression(
expression * checked_coefficient + checked_constant
)
raise TypeError(f"Unknown expression {expression!r} of type {type(expression)}")
def __hash__(self):
return object.__hash__(self)
def __add__(self, arg: LinearExprT) -> "_Sum":
return _Sum(self, arg)
def __radd__(self, arg: LinearExprT) -> "_Sum":
return self.__add__(arg)
def __sub__(self, arg: LinearExprT) -> "_Sum":
return _Sum(self, -arg)
def __rsub__(self, arg: LinearExprT) -> "_Sum":
return _Sum(-self, arg)
def __mul__(self, arg: NumberT) -> "_Product":
return _Product(self, arg)
def __rmul__(self, arg: NumberT) -> "_Product":
return self.__mul__(arg)
def __truediv__(self, coeff: NumberT) -> "_Product":
return self.__mul__(1.0 / coeff)
def __neg__(self) -> "_Product":
return _Product(self, -1)
def __bool__(self):
raise NotImplementedError(f"Cannot use a LinearExpr {self} as a Boolean value")
def __eq__(self, arg: LinearExprT) -> "BoundedLinearExpression":
return BoundedLinearExpression(self - arg, 0, 0)
def __ge__(self, arg: LinearExprT) -> "BoundedLinearExpression":
return BoundedLinearExpression(
self - arg, 0, math.inf
) # pytype: disable=wrong-arg-types # numpy-scalars
def __le__(self, arg: LinearExprT) -> "BoundedLinearExpression":
return BoundedLinearExpression(
self - arg, -math.inf, 0
) # pytype: disable=wrong-arg-types # numpy-scalars
class Variable(LinearExpr):
"""A variable (continuous or integral).
A Variable is an object that can take on any integer value within defined
ranges. Variables appear in constraint like:
x + y >= 5
Solving a model is equivalent to finding, for each variable, a single value
from the set of initial values (called the initial domain), such that the
model is feasible, or optimal if you provided an objective function.
"""
def __init__(
self,
helper: mbh.ModelBuilderHelper,
lb: NumberT,
ub: Optional[NumberT],
is_integral: Optional[bool],
name: Optional[str],
) -> None:
"""See Model.new_var below."""
LinearExpr.__init__(self)
self.__helper: mbh.ModelBuilderHelper = helper
# Python do not support multiple __init__ methods.
# This method is only called from the Model class.
# We hack the parameter to support the two cases:
# case 1:
# helper is a ModelBuilderHelper, lb is a double value, ub is a double
# value, is_integral is a Boolean value, and name is a string.
# case 2:
# helper is a ModelBuilderHelper, lb is an index (int), ub is None,
# is_integral is None, and name is None.
if mbn.is_integral(lb) and ub is None and is_integral is None:
self.__index: np.int32 = np.int32(lb)
self.__helper: mbh.ModelBuilderHelper = helper
else:
index: np.int32 = helper.add_var()
self.__index: np.int32 = np.int32(index)
self.__helper: mbh.ModelBuilderHelper = helper
helper.set_var_lower_bound(index, lb)
helper.set_var_upper_bound(index, ub)
helper.set_var_integrality(index, is_integral)
if name:
helper.set_var_name(index, name)
@property
def index(self) -> np.int32:
"""Returns the index of the variable in the helper."""
return self.__index
@property
def helper(self) -> mbh.ModelBuilderHelper:
"""Returns the underlying ModelBuilderHelper."""
return self.__helper
def is_equal_to(self, other: LinearExprT) -> bool:
"""Returns true if self == other in the python sense."""
if not isinstance(other, Variable):
return False
return self.index == other.index and self.helper == other.helper
def __str__(self) -> str:
return self.name
def __repr__(self) -> str:
return self.__str__()
@property
def name(self) -> str:
"""Returns the name of the variable."""
var_name = self.__helper.var_name(self.__index)
if var_name:
return var_name
return f"variable#{self.index}"
@name.setter
def name(self, name: str) -> None:
"""Sets the name of the variable."""
self.__helper.set_var_name(self.__index, name)
@property
def lower_bound(self) -> np.double:
"""Returns the lower bound of the variable."""
return self.__helper.var_lower_bound(self.__index)
@lower_bound.setter
def lower_bound(self, bound: NumberT) -> None:
"""Sets the lower bound of the variable."""
self.__helper.set_var_lower_bound(self.__index, bound)
@property
def upper_bound(self) -> np.double:
"""Returns the upper bound of the variable."""
return self.__helper.var_upper_bound(self.__index)
@upper_bound.setter
def upper_bound(self, bound: NumberT) -> None:
"""Sets the upper bound of the variable."""
self.__helper.set_var_upper_bound(self.__index, bound)
@property
def is_integral(self) -> bool:
"""Returns whether the variable is integral."""
return self.__helper.var_is_integral(self.__index)
@is_integral.setter
def integrality(self, is_integral: bool) -> None:
"""Sets the integrality of the variable."""
self.__helper.set_var_integrality(self.__index, is_integral)
@property
def objective_coefficient(self) -> NumberT:
return self.__helper.var_objective_coefficient(self.__index)
@objective_coefficient.setter
def objective_coefficient(self, coeff: NumberT) -> None:
self.__helper.set_var_objective_coefficient(self.__index, coeff)
def __eq__(self, arg: Optional[LinearExprT]) -> ConstraintT:
if arg is None:
return False
if isinstance(arg, Variable):
return VarEqVar(self, arg)
return BoundedLinearExpression(
self - arg, 0.0, 0.0
) # pytype: disable=wrong-arg-types # numpy-scalars
def __hash__(self):
return hash((self.__helper, self.__index))
class _BoundedLinearExpr(metaclass=abc.ABCMeta):
"""Interface for types that can build bounded linear (boolean) expressions.
Classes derived from _BoundedLinearExpr are used to build linear constraints
to be satisfied.
* BoundedLinearExpression: a linear expression with upper and lower bounds.
* VarEqVar: an equality comparison between two variables.
"""
@abc.abstractmethod
def _add_linear_constraint(
self, helper: mbh.ModelBuilderHelper, name: str
) -> "LinearConstraint":
"""Creates a new linear constraint in the helper.
Args:
helper (mbh.ModelBuilderHelper): The helper to create the constraint.
name (str): The name of the linear constraint.
Returns:
LinearConstraint: A reference to the linear constraint in the helper.
"""
@abc.abstractmethod
def _add_enforced_linear_constraint(
self,
helper: mbh.ModelBuilderHelper,
var: Variable,
value: bool,
name: str,
) -> "EnforcedLinearConstraint":
"""Creates a new enforced linear constraint in the helper.
Args:
helper (mbh.ModelBuilderHelper): The helper to create the constraint.
var (Variable): The indicator variable of the constraint.
value (bool): The indicator value of the constraint.
name (str): The name of the linear constraint.
Returns:
Enforced LinearConstraint: A reference to the linear constraint in the
helper.
"""
def _add_linear_constraint_to_helper(
bounded_expr: Union[bool, _BoundedLinearExpr],
helper: mbh.ModelBuilderHelper,
name: Optional[str],
):
"""Creates a new linear constraint in the helper.
It handles boolean values (which might arise in the construction of
BoundedLinearExpressions).
If bounded_expr is a Boolean value, the created constraint is different.
In that case, the constraint will be immutable and marked as under-specified.
It will be always feasible or infeasible whether the value is True or False.
Args:
bounded_expr: The bounded expression used to create the constraint.
helper: The helper to create the constraint.
name: The name of the constraint to be created.
Returns:
LinearConstraint: a constraint in the helper corresponding to the input.
Raises:
TypeError: If constraint is an invalid type.
"""
if isinstance(bounded_expr, bool):
c = LinearConstraint(helper, is_under_specified=True)
if name is not None:
helper.set_constraint_name(c.index, name)
if bounded_expr:
# constraint that is always feasible: 0.0 <= nothing <= 0.0
helper.set_constraint_lower_bound(c.index, 0.0)
helper.set_constraint_upper_bound(c.index, 0.0)
else:
# constraint that is always infeasible: +oo <= nothing <= -oo
helper.set_constraint_lower_bound(c.index, 1)
helper.set_constraint_upper_bound(c.index, -1)
return c
if isinstance(bounded_expr, _BoundedLinearExpr):
# pylint: disable=protected-access
return bounded_expr._add_linear_constraint(helper, name)
raise TypeError("invalid type={}".format(type(bounded_expr)))
def _add_enforced_linear_constraint_to_helper(
bounded_expr: Union[bool, _BoundedLinearExpr],
helper: mbh.ModelBuilderHelper,
var: Variable,
value: bool,
name: Optional[str],
):
"""Creates a new enforced linear constraint in the helper.
It handles boolean values (which might arise in the construction of
BoundedLinearExpressions).
If bounded_expr is a Boolean value, the linear part of the constraint is
different.
In that case, the constraint will be immutable and marked as under-specified.
Its linear part will be always feasible or infeasible whether the value is
True or False.
Args:
bounded_expr: The bounded expression used to create the constraint.
helper: The helper to create the constraint.
var: the variable used in the indicator
value: the value used in the indicator
name: The name of the constraint to be created.
Returns:
EnforcedLinearConstraint: a constraint in the helper corresponding to the
input.
Raises:
TypeError: If constraint is an invalid type.
"""
if isinstance(bounded_expr, bool):
# TODO(user): create indicator variable assignment instead ?
c = EnforcedLinearConstraint(helper, is_under_specified=True)
c.indicator_variable = var
c.indicator_value = value
if name is not None:
helper.set_enforced_constraint_name(c.index, name)
if bounded_expr:
# constraint that is always feasible: 0.0 <= nothing <= 0.0
helper.set_enforced_constraint_lower_bound(c.index, 0.0)
helper.set_enforced_constraint_upper_bound(c.index, 0.0)
else:
# constraint that is always infeasible: +oo <= nothing <= -oo
helper.set_enforced_constraint_lower_bound(c.index, 1)
helper.set_enforced_constraint_upper_bound(c.index, -1)
return c
if isinstance(bounded_expr, _BoundedLinearExpr):
# pylint: disable=protected-access
return bounded_expr._add_enforced_linear_constraint(helper, var, value, name)
raise TypeError("invalid type={}".format(type(bounded_expr)))
@dataclasses.dataclass(repr=False, eq=False, frozen=True)
class VarEqVar(_BoundedLinearExpr):
"""Represents var == var."""
__slots__ = ("left", "right")
left: Variable
right: Variable
def __str__(self):
return f"{self.left} == {self.right}"
def __repr__(self):
return self.__str__()
def __bool__(self) -> bool:
return hash(self.left) == hash(self.right)
def _add_linear_constraint(
self, helper: mbh.ModelBuilderHelper, name: str
) -> "LinearConstraint":
c = LinearConstraint(helper)
helper.set_constraint_lower_bound(c.index, 0.0)
helper.set_constraint_upper_bound(c.index, 0.0)
# pylint: disable=protected-access
helper.add_term_to_constraint(c.index, self.left.index, 1.0)
helper.add_term_to_constraint(c.index, self.right.index, -1.0)
# pylint: enable=protected-access
helper.set_constraint_name(c.index, name)
return c
def _add_enforced_linear_constraint(
self,
helper: mbh.ModelBuilderHelper,
var: Variable,
value: bool,
name: str,
) -> "EnforcedLinearConstraint":
"""Adds an enforced linear constraint to the model."""
c = EnforcedLinearConstraint(helper)
c.indicator_variable = var
c.indicator_value = value
helper.set_enforced_constraint_lower_bound(c.index, 0.0)
helper.set_enforced_constraint_upper_bound(c.index, 0.0)
# pylint: disable=protected-access
helper.add_term_to_enforced_constraint(c.index, self.left.index, 1.0)
helper.add_term_to_enforced_constraint(c.index, self.right.index, -1.0)
# pylint: enable=protected-access
helper.set_enforced_constraint_name(c.index, name)
return c
class BoundedLinearExpression(_BoundedLinearExpr):
"""Represents a linear constraint: `lb <= linear expression <= ub`.
The only use of this class is to be added to the Model through
`Model.add(bounded expression)`, as in:
model.Add(x + 2 * y -1 >= z)
"""
def __init__(self, expr: LinearExprT, lb: NumberT, ub: NumberT) -> None:
self.__expr: LinearExprT = expr
self.__lb: np.double = mbn.assert_is_a_number(lb)
self.__ub: np.double = mbn.assert_is_a_number(ub)
def __str__(self) -> str:
if self.__lb > -math.inf and self.__ub < math.inf:
if self.__lb == self.__ub:
return f"{self.__expr} == {self.__lb}"
else:
return f"{self.__lb} <= {self.__expr} <= {self.__ub}"
elif self.__lb > -math.inf:
return f"{self.__expr} >= {self.__lb}"
elif self.__ub < math.inf:
return f"{self.__expr} <= {self.__ub}"
else:
return f"{self.__expr} free"
def __repr__(self):
return self.__str__()
@property
def expression(self) -> LinearExprT:
return self.__expr
@property
def lower_bound(self) -> np.double:
return self.__lb
@property
def upper_bound(self) -> np.double:
return self.__ub
def __bool__(self) -> bool:
raise NotImplementedError(
f"Cannot use a BoundedLinearExpression {self} as a Boolean value"
)
def _add_linear_constraint(
self, helper: mbh.ModelBuilderHelper, name: Optional[str]
) -> "LinearConstraint":
c = LinearConstraint(helper)
flat_expr = _as_flat_linear_expression(self.__expr)
# pylint: disable=protected-access
helper.add_terms_to_constraint(
c.index, flat_expr._variable_indices, flat_expr._coefficients
)
helper.set_constraint_lower_bound(c.index, self.__lb - flat_expr._offset)
helper.set_constraint_upper_bound(c.index, self.__ub - flat_expr._offset)
# pylint: enable=protected-access
if name is not None:
helper.set_constraint_name(c.index, name)
return c
def _add_enforced_linear_constraint(
self,
helper: mbh.ModelBuilderHelper,
var: Variable,
value: bool,
name: Optional[str],
) -> "EnforcedLinearConstraint":
"""Adds an enforced linear constraint to the model."""
c = EnforcedLinearConstraint(helper)
c.indicator_variable = var
c.indicator_value = value
flat_expr = _as_flat_linear_expression(self.__expr)
# pylint: disable=protected-access
helper.add_terms_to_enforced_constraint(
c.index, flat_expr._variable_indices, flat_expr._coefficients
)
helper.set_enforced_constraint_lower_bound(
c.index, self.__lb - flat_expr._offset
)
helper.set_enforced_constraint_upper_bound(
c.index, self.__ub - flat_expr._offset
)
# pylint: enable=protected-access
if name is not None:
helper.set_enforced_constraint_name(c.index, name)
return c
class LinearConstraint:
"""Stores a linear equation.
Example:
x = model.new_num_var(0, 10, 'x')
y = model.new_num_var(0, 10, 'y')
linear_constraint = model.add(x + 2 * y == 5)
"""
def __init__(
self,
helper: mbh.ModelBuilderHelper,
*,
index: Optional[IntegerT] = None,
is_under_specified: bool = False,
) -> None:
"""LinearConstraint constructor.
Args:
helper: The pybind11 ModelBuilderHelper.
index: If specified, recreates a wrapper to an existing linear constraint.
is_under_specified: indicates if the constraint was created by
model.add(bool).
"""
if index is None:
self.__index = helper.add_linear_constraint()
else:
self.__index = index
self.__helper: mbh.ModelBuilderHelper = helper
self.__is_under_specified = is_under_specified
@property
def index(self) -> IntegerT:
"""Returns the index of the constraint in the helper."""
return self.__index
@property
def helper(self) -> mbh.ModelBuilderHelper:
"""Returns the ModelBuilderHelper instance."""
return self.__helper
@property
def lower_bound(self) -> np.double:
return self.__helper.constraint_lower_bound(self.__index)
@lower_bound.setter
def lower_bound(self, bound: NumberT) -> None:
self.assert_constraint_is_well_defined()
self.__helper.set_constraint_lower_bound(self.__index, bound)
@property
def upper_bound(self) -> np.double:
return self.__helper.constraint_upper_bound(self.__index)
@upper_bound.setter
def upper_bound(self, bound: NumberT) -> None:
self.assert_constraint_is_well_defined()
self.__helper.set_constraint_upper_bound(self.__index, bound)
@property
def name(self) -> str:
constraint_name = self.__helper.constraint_name(self.__index)
if constraint_name:
return constraint_name
return f"linear_constraint#{self.__index}"
@name.setter
def name(self, name: str) -> None:
return self.__helper.set_constraint_name(self.__index, name)
@property
def is_under_specified(self) -> bool:
"""Returns True if the constraint is under specified.
Usually, it means that it was created by model.add(False) or model.add(True)
The effect is that modifying the constraint will raise an exception.
"""
return self.__is_under_specified
def assert_constraint_is_well_defined(self) -> None:
"""Raises an exception if the constraint is under specified."""
if self.__is_under_specified:
raise ValueError(
f"Constraint {self.index} is under specified and cannot be modified"
)
def __str__(self):
return self.name
def __repr__(self):
return (
f"LinearConstraint({self.name}, lb={self.lower_bound},"
f" ub={self.upper_bound},"
f" var_indices={self.helper.constraint_var_indices(self.index)},"
f" coefficients={self.helper.constraint_coefficients(self.index)})"
)
def set_coefficient(self, var: Variable, coeff: NumberT) -> None:
"""Sets the coefficient of the variable in the constraint."""
self.assert_constraint_is_well_defined()
self.__helper.set_constraint_coefficient(self.__index, var.index, coeff)
def add_term(self, var: Variable, coeff: NumberT) -> None:
"""Adds var * coeff to the constraint."""
self.assert_constraint_is_well_defined()
self.__helper.safe_add_term_to_constraint(self.__index, var.index, coeff)
def clear_terms(self) -> None:
"""Clear all terms of the constraint."""
self.assert_constraint_is_well_defined()
self.__helper.clear_constraint_terms(self.__index)
class EnforcedLinearConstraint:
"""Stores an enforced linear equation, also name indicator constraint.
Example:
x = model.new_num_var(0, 10, 'x')
y = model.new_num_var(0, 10, 'y')
z = model.new_bool_var('z')
enforced_linear_constraint = model.add_enforced(x + 2 * y == 5, z, False)
"""
def __init__(
self,
helper: mbh.ModelBuilderHelper,
*,
index: Optional[IntegerT] = None,
is_under_specified: bool = False,
) -> None:
"""EnforcedLinearConstraint constructor.
Args:
helper: The pybind11 ModelBuilderHelper.
index: If specified, recreates a wrapper to an existing linear constraint.
is_under_specified: indicates if the constraint was created by
model.add(bool).
"""
if index is None:
self.__index = helper.add_enforced_linear_constraint()
else:
if not helper.is_enforced_linear_constraint(index):
raise ValueError(
f"the given index {index} does not refer to an enforced linear"
" constraint"
)
self.__index = index
self.__helper: mbh.ModelBuilderHelper = helper
self.__is_under_specified = is_under_specified
@property
def index(self) -> IntegerT:
"""Returns the index of the constraint in the helper."""
return self.__index
@property
def helper(self) -> mbh.ModelBuilderHelper:
"""Returns the ModelBuilderHelper instance."""
return self.__helper
@property
def lower_bound(self) -> np.double:
return self.__helper.enforced_constraint_lower_bound(self.__index)
@lower_bound.setter
def lower_bound(self, bound: NumberT) -> None:
self.assert_constraint_is_well_defined()
self.__helper.set_enforced_constraint_lower_bound(self.__index, bound)
@property
def upper_bound(self) -> np.double:
return self.__helper.enforced_constraint_upper_bound(self.__index)
@upper_bound.setter
def upper_bound(self, bound: NumberT) -> None:
self.assert_constraint_is_well_defined()
self.__helper.set_enforced_constraint_upper_bound(self.__index, bound)
@property
def indicator_variable(self) -> "Variable":
enforcement_var_index = (
self.__helper.enforced_constraint_indicator_variable_index(self.__index)
)
return Variable(self.__helper, enforcement_var_index, None, None, None)
@indicator_variable.setter
def indicator_variable(self, var: "Variable") -> None:
self.__helper.set_enforced_constraint_indicator_variable_index(
self.__index, var.index
)
@property
def indicator_value(self) -> bool:
return self.__helper.enforced_constraint_indicator_value(self.__index)
@indicator_value.setter
def indicator_value(self, value: bool) -> None:
self.__helper.set_enforced_constraint_indicator_value(self.__index, value)
@property
def name(self) -> str:
constraint_name = self.__helper.enforced_constraint_name(self.__index)
if constraint_name:
return constraint_name
return f"enforced_linear_constraint#{self.__index}"
@name.setter
def name(self, name: str) -> None:
return self.__helper.set_enforced_constraint_name(self.__index, name)
@property
def is_under_specified(self) -> bool:
"""Returns True if the constraint is under specified.
Usually, it means that it was created by model.add(False) or model.add(True)
The effect is that modifying the constraint will raise an exception.
"""
return self.__is_under_specified
def assert_constraint_is_well_defined(self) -> None:
"""Raises an exception if the constraint is under specified."""
if self.__is_under_specified:
raise ValueError(
f"Constraint {self.index} is under specified and cannot be modified"
)
def __str__(self):
return self.name
def __repr__(self):
return (
f"EnforcedLinearConstraint({self.name}, lb={self.lower_bound},"
f" ub={self.upper_bound},"
f" var_indices={self.helper.enforced_constraint_var_indices(self.index)},"
f" coefficients={self.helper.enforced_constraint_coefficients(self.index)},"
f" indicator_variable={self.indicator_variable}"
f" indicator_value={self.indicator_value})"
)
def set_coefficient(self, var: Variable, coeff: NumberT) -> None:
"""Sets the coefficient of the variable in the constraint."""
self.assert_constraint_is_well_defined()
self.__helper.set_enforced_constraint_coefficient(
self.__index, var.index, coeff
)
def add_term(self, var: Variable, coeff: NumberT) -> None:
"""Adds var * coeff to the constraint."""
self.assert_constraint_is_well_defined()
self.__helper.safe_add_term_to_enforced_constraint(
self.__index, var.index, coeff
)
def clear_terms(self) -> None:
"""Clear all terms of the constraint."""
self.assert_constraint_is_well_defined()
self.__helper.clear_enforced_constraint_terms(self.__index)
class Model:
"""Methods for building a linear model.
Methods beginning with:
* ```new_``` create integer, boolean, or interval variables.
* ```add_``` create new constraints and add them to the model.
"""
def __init__(self):
self.__helper: mbh.ModelBuilderHelper = mbh.ModelBuilderHelper()
def clone(self) -> "Model":
"""Returns a clone of the current model."""
clone = Model()
clone.helper.overwrite_model(self.helper)
return clone
@typing.overload
def _get_linear_constraints(self, constraints: Optional[pd.Index]) -> pd.Index: ...
@typing.overload
def _get_linear_constraints(self, constraints: pd.Series) -> pd.Series: ...
def _get_linear_constraints(
self, constraints: Optional[_IndexOrSeries] = None
) -> _IndexOrSeries:
if constraints is None:
return self.get_linear_constraints()
return constraints
@typing.overload
def _get_variables(self, variables: Optional[pd.Index]) -> pd.Index: ...
@typing.overload
def _get_variables(self, variables: pd.Series) -> pd.Series: ...
def _get_variables(
self, variables: Optional[_IndexOrSeries] = None
) -> _IndexOrSeries:
if variables is None:
return self.get_variables()
return variables
def get_linear_constraints(self) -> pd.Index:
"""Gets all linear constraints in the model."""
return pd.Index(
[self.linear_constraint_from_index(i) for i in range(self.num_constraints)],
name="linear_constraint",
)
def get_linear_constraint_expressions(
self, constraints: Optional[_IndexOrSeries] = None
) -> pd.Series:
"""Gets the expressions of all linear constraints in the set.
If `constraints` is a `pd.Index`, then the output will be indexed by the
constraints. If `constraints` is a `pd.Series` indexed by the underlying
dimensions, then the output will be indexed by the same underlying
dimensions.
Args:
constraints (Union[pd.Index, pd.Series]): Optional. The set of linear
constraints from which to get the expressions. If unspecified, all
linear constraints will be in scope.
Returns:
pd.Series: The expressions of all linear constraints in the set.
"""
return _attribute_series(
# pylint: disable=g-long-lambda
func=lambda c: _as_flat_linear_expression(
# pylint: disable=g-complex-comprehension
sum(
coeff * Variable(self.__helper, var_id, None, None, None)
for var_id, coeff in zip(
c.helper.constraint_var_indices(c.index),
c.helper.constraint_coefficients(c.index),
)
)
),
values=self._get_linear_constraints(constraints),
)
def get_linear_constraint_lower_bounds(
self, constraints: Optional[_IndexOrSeries] = None
) -> pd.Series:
"""Gets the lower bounds of all linear constraints in the set.
If `constraints` is a `pd.Index`, then the output will be indexed by the
constraints. If `constraints` is a `pd.Series` indexed by the underlying
dimensions, then the output will be indexed by the same underlying
dimensions.
Args:
constraints (Union[pd.Index, pd.Series]): Optional. The set of linear
constraints from which to get the lower bounds. If unspecified, all
linear constraints will be in scope.
Returns:
pd.Series: The lower bounds of all linear constraints in the set.
"""
return _attribute_series(
func=lambda c: c.lower_bound, # pylint: disable=protected-access
values=self._get_linear_constraints(constraints),
)
def get_linear_constraint_upper_bounds(
self, constraints: Optional[_IndexOrSeries] = None
) -> pd.Series:
"""Gets the upper bounds of all linear constraints in the set.
If `constraints` is a `pd.Index`, then the output will be indexed by the
constraints. If `constraints` is a `pd.Series` indexed by the underlying
dimensions, then the output will be indexed by the same underlying
dimensions.
Args:
constraints (Union[pd.Index, pd.Series]): Optional. The set of linear
constraints. If unspecified, all linear constraints will be in scope.
Returns:
pd.Series: The upper bounds of all linear constraints in the set.
"""
return _attribute_series(
func=lambda c: c.upper_bound, # pylint: disable=protected-access
values=self._get_linear_constraints(constraints),
)
def get_variables(self) -> pd.Index:
"""Gets all variables in the model."""
return pd.Index(
[self.var_from_index(i) for i in range(self.num_variables)],
name="variable",
)
def get_variable_lower_bounds(
self, variables: Optional[_IndexOrSeries] = None
) -> pd.Series:
"""Gets the lower bounds of all variables in the set.
If `variables` is a `pd.Index`, then the output will be indexed by the
variables. If `variables` is a `pd.Series` indexed by the underlying
dimensions, then the output will be indexed by the same underlying
dimensions.
Args:
variables (Union[pd.Index, pd.Series]): Optional. The set of variables
from which to get the lower bounds. If unspecified, all variables will
be in scope.
Returns:
pd.Series: The lower bounds of all variables in the set.
"""
return _attribute_series(
func=lambda v: v.lower_bound, # pylint: disable=protected-access
values=self._get_variables(variables),
)
def get_variable_upper_bounds(
self, variables: Optional[_IndexOrSeries] = None
) -> pd.Series:
"""Gets the upper bounds of all variables in the set.
Args:
variables (Union[pd.Index, pd.Series]): Optional. The set of variables
from which to get the upper bounds. If unspecified, all variables will
be in scope.
Returns:
pd.Series: The upper bounds of all variables in the set.
"""
return _attribute_series(
func=lambda v: v.upper_bound, # pylint: disable=protected-access
values=self._get_variables(variables),
)
# Integer variable.
def new_var(
self, lb: NumberT, ub: NumberT, is_integer: bool, name: Optional[str]
) -> Variable:
"""Create an integer variable with domain [lb, ub].
Args:
lb: Lower bound of the variable.
ub: Upper bound of the variable.
is_integer: Indicates if the variable must take integral values.
name: The name of the variable.
Returns:
a variable whose domain is [lb, ub].
"""
return Variable(self.__helper, lb, ub, is_integer, name)
def new_int_var(
self, lb: NumberT, ub: NumberT, name: Optional[str] = None
) -> Variable:
"""Create an integer variable with domain [lb, ub].
Args:
lb: Lower bound of the variable.
ub: Upper bound of the variable.
name: The name of the variable.
Returns:
a variable whose domain is [lb, ub].
"""
return self.new_var(lb, ub, True, name)
def new_num_var(
self, lb: NumberT, ub: NumberT, name: Optional[str] = None
) -> Variable:
"""Create an integer variable with domain [lb, ub].
Args:
lb: Lower bound of the variable.
ub: Upper bound of the variable.
name: The name of the variable.
Returns:
a variable whose domain is [lb, ub].
"""
return self.new_var(lb, ub, False, name)
def new_bool_var(self, name: Optional[str] = None) -> Variable:
"""Creates a 0-1 variable with the given name."""
return self.new_var(
0, 1, True, name
) # pytype: disable=wrong-arg-types # numpy-scalars
def new_constant(self, value: NumberT) -> Variable:
"""Declares a constant variable."""
return self.new_var(value, value, False, None)
def new_var_series(
self,
name: str,
index: pd.Index,
lower_bounds: Union[NumberT, pd.Series] = -math.inf,
upper_bounds: Union[NumberT, pd.Series] = math.inf,
is_integral: Union[bool, pd.Series] = False,
) -> pd.Series:
"""Creates a series of (scalar-valued) variables with the given name.
Args:
name (str): Required. The name of the variable set.
index (pd.Index): Required. The index to use for the variable set.
lower_bounds (Union[int, float, pd.Series]): Optional. A lower bound for
variables in the set. If a `pd.Series` is passed in, it will be based on
the corresponding values of the pd.Series. Defaults to -inf.
upper_bounds (Union[int, float, pd.Series]): Optional. An upper bound for
variables in the set. If a `pd.Series` is passed in, it will be based on
the corresponding values of the pd.Series. Defaults to +inf.
is_integral (bool, pd.Series): Optional. Indicates if the variable can
only take integer values. If a `pd.Series` is passed in, it will be
based on the corresponding values of the pd.Series. Defaults to False.
Returns:
pd.Series: The variable set indexed by its corresponding dimensions.
Raises:
TypeError: if the `index` is invalid (e.g. a `DataFrame`).
ValueError: if the `name` is not a valid identifier or already exists.
ValueError: if the `lowerbound` is greater than the `upperbound`.
ValueError: if the index of `lower_bound`, `upper_bound`, or `is_integer`
does not match the input index.
"""
if not isinstance(index, pd.Index):
raise TypeError("Non-index object is used as index")
if not name.isidentifier():
raise ValueError("name={} is not a valid identifier".format(name))
if (
mbn.is_a_number(lower_bounds)
and mbn.is_a_number(upper_bounds)
and lower_bounds > upper_bounds
):
raise ValueError(
"lower_bound={} is greater than upper_bound={} for variable set={}".format(
lower_bounds, upper_bounds, name
)
)
if (
isinstance(is_integral, bool)
and is_integral
and mbn.is_a_number(lower_bounds)
and mbn.is_a_number(upper_bounds)
and math.isfinite(lower_bounds)
and math.isfinite(upper_bounds)
and math.ceil(lower_bounds) > math.floor(upper_bounds)
):
raise ValueError(
"ceil(lower_bound={})={}".format(lower_bounds, math.ceil(lower_bounds))
+ " is greater than floor("
+ "upper_bound={})={}".format(upper_bounds, math.floor(upper_bounds))
+ " for variable set={}".format(name)
)
lower_bounds = _convert_to_series_and_validate_index(lower_bounds, index)
upper_bounds = _convert_to_series_and_validate_index(upper_bounds, index)
is_integrals = _convert_to_series_and_validate_index(is_integral, index)
return pd.Series(
index=index,
data=[
# pylint: disable=g-complex-comprehension
Variable(
helper=self.__helper,
name=f"{name}[{i}]",
lb=lower_bounds[i],
ub=upper_bounds[i],
is_integral=is_integrals[i],
)
for i in index
],
)
def new_num_var_series(
self,
name: str,
index: pd.Index,
lower_bounds: Union[NumberT, pd.Series] = -math.inf,
upper_bounds: Union[NumberT, pd.Series] = math.inf,
) -> pd.Series:
"""Creates a series of continuous variables with the given name.
Args:
name (str): Required. The name of the variable set.
index (pd.Index): Required. The index to use for the variable set.
lower_bounds (Union[int, float, pd.Series]): Optional. A lower bound for
variables in the set. If a `pd.Series` is passed in, it will be based on
the corresponding values of the pd.Series. Defaults to -inf.
upper_bounds (Union[int, float, pd.Series]): Optional. An upper bound for
variables in the set. If a `pd.Series` is passed in, it will be based on
the corresponding values of the pd.Series. Defaults to +inf.
Returns:
pd.Series: The variable set indexed by its corresponding dimensions.
Raises:
TypeError: if the `index` is invalid (e.g. a `DataFrame`).
ValueError: if the `name` is not a valid identifier or already exists.
ValueError: if the `lowerbound` is greater than the `upperbound`.
ValueError: if the index of `lower_bound`, `upper_bound`, or `is_integer`
does not match the input index.
"""
return self.new_var_series(name, index, lower_bounds, upper_bounds, False)
def new_int_var_series(
self,
name: str,
index: pd.Index,
lower_bounds: Union[NumberT, pd.Series] = -math.inf,
upper_bounds: Union[NumberT, pd.Series] = math.inf,
) -> pd.Series:
"""Creates a series of integer variables with the given name.
Args:
name (str): Required. The name of the variable set.
index (pd.Index): Required. The index to use for the variable set.
lower_bounds (Union[int, float, pd.Series]): Optional. A lower bound for
variables in the set. If a `pd.Series` is passed in, it will be based on
the corresponding values of the pd.Series. Defaults to -inf.
upper_bounds (Union[int, float, pd.Series]): Optional. An upper bound for
variables in the set. If a `pd.Series` is passed in, it will be based on
the corresponding values of the pd.Series. Defaults to +inf.
Returns:
pd.Series: The variable set indexed by its corresponding dimensions.
Raises:
TypeError: if the `index` is invalid (e.g. a `DataFrame`).
ValueError: if the `name` is not a valid identifier or already exists.
ValueError: if the `lowerbound` is greater than the `upperbound`.
ValueError: if the index of `lower_bound`, `upper_bound`, or `is_integer`
does not match the input index.
"""
return self.new_var_series(name, index, lower_bounds, upper_bounds, True)
def new_bool_var_series(
self,
name: str,
index: pd.Index,
) -> pd.Series:
"""Creates a series of Boolean variables with the given name.
Args:
name (str): Required. The name of the variable set.
index (pd.Index): Required. The index to use for the variable set.
Returns:
pd.Series: The variable set indexed by its corresponding dimensions.
Raises:
TypeError: if the `index` is invalid (e.g. a `DataFrame`).
ValueError: if the `name` is not a valid identifier or already exists.
ValueError: if the `lowerbound` is greater than the `upperbound`.
ValueError: if the index of `lower_bound`, `upper_bound`, or `is_integer`
does not match the input index.
"""
return self.new_var_series(name, index, 0, 1, True)
def var_from_index(self, index: IntegerT) -> Variable:
"""Rebuilds a variable object from the model and its index."""
return Variable(self.__helper, index, None, None, None)
# Linear constraints.
def add_linear_constraint( # pytype: disable=annotation-type-mismatch # numpy-scalars
self,
linear_expr: LinearExprT,
lb: NumberT = -math.inf,
ub: NumberT = math.inf,
name: Optional[str] = None,
) -> LinearConstraint:
"""Adds the constraint: `lb <= linear_expr <= ub` with the given name."""
ct = LinearConstraint(self.__helper)
if name:
self.__helper.set_constraint_name(ct.index, name)
if mbn.is_a_number(linear_expr):
self.__helper.set_constraint_lower_bound(ct.index, lb - linear_expr)
self.__helper.set_constraint_upper_bound(ct.index, ub - linear_expr)
elif isinstance(linear_expr, Variable):
self.__helper.set_constraint_lower_bound(ct.index, lb)
self.__helper.set_constraint_upper_bound(ct.index, ub)
self.__helper.add_term_to_constraint(ct.index, linear_expr.index, 1.0)
elif isinstance(linear_expr, LinearExpr):
flat_expr = _as_flat_linear_expression(linear_expr)
# pylint: disable=protected-access
self.__helper.set_constraint_lower_bound(ct.index, lb - flat_expr._offset)
self.__helper.set_constraint_upper_bound(ct.index, ub - flat_expr._offset)
self.__helper.add_terms_to_constraint(
ct.index, flat_expr._variable_indices, flat_expr._coefficients
)
else:
raise TypeError(
f"Not supported: Model.add_linear_constraint({linear_expr})"
f" with type {type(linear_expr)}"
)
return ct
def add(
self, ct: Union[ConstraintT, pd.Series], name: Optional[str] = None
) -> Union[LinearConstraint, pd.Series]:
"""Adds a `BoundedLinearExpression` to the model.
Args:
ct: A [`BoundedLinearExpression`](#boundedlinearexpression).
name: An optional name.
Returns:
An instance of the `Constraint` class.
Note that a special treatment is done when the argument does not contain any
variable, and thus evaluates to True or False.
`model.add(True)` will create a constraint 0 <= empty sum <= 0.
The constraint will be marked as under specified, and cannot be modified
thereafter.
`model.add(False)` will create a constraint inf <= empty sum <= -inf. The
constraint will be marked as under specified, and cannot be modified
thereafter.
you can check the if a constraint is under specified by reading the
`LinearConstraint.is_under_specified` property.
"""
if isinstance(ct, _BoundedLinearExpr):
return ct._add_linear_constraint(self.__helper, name)
elif isinstance(ct, bool):
return _add_linear_constraint_to_helper(ct, self.__helper, name)
elif isinstance(ct, pd.Series):
return pd.Series(
index=ct.index,
data=[
_add_linear_constraint_to_helper(
expr, self.__helper, f"{name}[{i}]"
)
for (i, expr) in zip(ct.index, ct)
],
)
else:
raise TypeError("Not supported: Model.add(" + str(ct) + ")")
def linear_constraint_from_index(self, index: IntegerT) -> LinearConstraint:
"""Rebuilds a linear constraint object from the model and its index."""
return LinearConstraint(self.__helper, index=index)
# EnforcedLinear constraints.
def add_enforced_linear_constraint( # pytype: disable=annotation-type-mismatch # numpy-scalars
self,
linear_expr: LinearExprT,
ivar: "Variable",
ivalue: bool,
lb: NumberT = -math.inf,
ub: NumberT = math.inf,
name: Optional[str] = None,
) -> EnforcedLinearConstraint:
"""Adds the constraint: `ivar == ivalue => lb <= linear_expr <= ub` with the given name."""
ct = EnforcedLinearConstraint(self.__helper)
ct.indicator_variable = ivar
ct.indicator_value = ivalue
if name:
self.__helper.set_constraint_name(ct.index, name)
if mbn.is_a_number(linear_expr):
self.__helper.set_constraint_lower_bound(ct.index, lb - linear_expr)
self.__helper.set_constraint_upper_bound(ct.index, ub - linear_expr)
elif isinstance(linear_expr, Variable):
self.__helper.set_constraint_lower_bound(ct.index, lb)
self.__helper.set_constraint_upper_bound(ct.index, ub)
self.__helper.add_term_to_constraint(ct.index, linear_expr.index, 1.0)
elif isinstance(linear_expr, LinearExpr):
flat_expr = _as_flat_linear_expression(linear_expr)
# pylint: disable=protected-access
self.__helper.set_constraint_lower_bound(ct.index, lb - flat_expr._offset)
self.__helper.set_constraint_upper_bound(ct.index, ub - flat_expr._offset)
self.__helper.add_terms_to_constraint(
ct.index, flat_expr._variable_indices, flat_expr._coefficients
)
else:
raise TypeError(
"Not supported:"
f" Model.add_enforced_linear_constraint({linear_expr}) with"
f" type {type(linear_expr)}"
)
return ct
def add_enforced(
self,
ct: Union[ConstraintT, pd.Series],
var: Union[Variable, pd.Series],
value: Union[bool, pd.Series],
name: Optional[str] = None,
) -> Union[EnforcedLinearConstraint, pd.Series]:
"""Adds a `ivar == ivalue => BoundedLinearExpression` to the model.
Args:
ct: A [`BoundedLinearExpression`](#boundedlinearexpression).
var: The indicator variable
value: the indicator value
name: An optional name.
Returns:
An instance of the `Constraint` class.
Note that a special treatment is done when the argument does not contain any
variable, and thus evaluates to True or False.
model.add_enforced(True, ivar, ivalue) will create a constraint 0 <= empty
sum <= 0
model.add_enforced(False, var, value) will create a constraint inf <=
empty sum <= -inf
you can check the if a constraint is always false (lb=inf, ub=-inf) by
calling EnforcedLinearConstraint.is_always_false()
"""
if isinstance(ct, _BoundedLinearExpr):
return ct._add_enforced_linear_constraint(self.__helper, var, value, name)
elif (
isinstance(ct, bool)
and isinstance(var, Variable)
and isinstance(value, bool)
):
return _add_enforced_linear_constraint_to_helper(
ct, self.__helper, var, value, name
)
elif isinstance(ct, pd.Series):
ivar_series = _convert_to_var_series_and_validate_index(var, ct.index)
ivalue_series = _convert_to_series_and_validate_index(value, ct.index)
return pd.Series(
index=ct.index,
data=[
_add_enforced_linear_constraint_to_helper(
expr,
self.__helper,
ivar_series[i],
ivalue_series[i],
f"{name}[{i}]",
)
for (i, expr) in zip(ct.index, ct)
],
)
else:
raise TypeError("Not supported: Model.add_enforced(" + str(ct) + ")")
def enforced_linear_constraint_from_index(
self, index: IntegerT
) -> EnforcedLinearConstraint:
"""Rebuilds an enforced linear constraint object from the model and its index."""
return EnforcedLinearConstraint(self.__helper, index=index)
# Objective.
def minimize(self, linear_expr: LinearExprT) -> None:
"""Minimizes the given objective."""
self.__optimize(linear_expr, False)
def maximize(self, linear_expr: LinearExprT) -> None:
"""Maximizes the given objective."""
self.__optimize(linear_expr, True)
def __optimize(self, linear_expr: LinearExprT, maximize: bool) -> None:
"""Defines the objective."""
self.helper.clear_objective()
self.__helper.set_maximize(maximize)
if mbn.is_a_number(linear_expr):
self.helper.set_objective_offset(linear_expr)
elif isinstance(linear_expr, Variable):
self.helper.set_var_objective_coefficient(linear_expr.index, 1.0)
elif isinstance(linear_expr, LinearExpr):
flat_expr = _as_flat_linear_expression(linear_expr)
# pylint: disable=protected-access
self.helper.set_objective_offset(flat_expr._offset)
self.helper.set_objective_coefficients(
flat_expr._variable_indices, flat_expr._coefficients
)
else:
raise TypeError(f"Not supported: Model.minimize/maximize({linear_expr})")
@property
def objective_offset(self) -> np.double:
"""Returns the fixed offset of the objective."""
return self.__helper.objective_offset()
@objective_offset.setter
def objective_offset(self, value: NumberT) -> None:
self.__helper.set_objective_offset(value)
def objective_expression(self) -> "_LinearExpression":
"""Returns the expression to optimize."""
return _as_flat_linear_expression(
sum(
variable * self.__helper.var_objective_coefficient(variable.index)
for variable in self.get_variables()
if self.__helper.var_objective_coefficient(variable.index) != 0.0
)
+ self.__helper.objective_offset()
)
# Hints.
def clear_hints(self):
"""Clears all solution hints."""
self.__helper.clear_hints()
def add_hint(self, var: Variable, value: NumberT) -> None:
"""Adds var == value as a hint to the model.
Args:
var: The variable of the hint
value: The value of the hint
Note that variables must not appear more than once in the list of hints.
"""
self.__helper.add_hint(var.index, value)
# Input/Output
def export_to_lp_string(self, obfuscate: bool = False) -> str:
options: mbh.MPModelExportOptions = mbh.MPModelExportOptions()
options.obfuscate = obfuscate
return self.__helper.export_to_lp_string(options)
def export_to_mps_string(self, obfuscate: bool = False) -> str:
options: mbh.MPModelExportOptions = mbh.MPModelExportOptions()
options.obfuscate = obfuscate
return self.__helper.export_to_mps_string(options)
def export_to_proto(self) -> linear_solver_pb2.MPModelProto:
"""Exports the optimization model to a ProtoBuf format."""
return mbh.to_mpmodel_proto(self.__helper)
def import_from_mps_string(self, mps_string: str) -> bool:
"""Reads a model from a MPS string."""
return self.__helper.import_from_mps_string(mps_string)
def import_from_mps_file(self, mps_file: str) -> bool:
"""Reads a model from a .mps file."""
return self.__helper.import_from_mps_file(mps_file)
def import_from_lp_string(self, lp_string: str) -> bool:
"""Reads a model from a LP string."""
return self.__helper.import_from_lp_string(lp_string)
def import_from_lp_file(self, lp_file: str) -> bool:
"""Reads a model from a .lp file."""
return self.__helper.import_from_lp_file(lp_file)
def import_from_proto_file(self, proto_file: str) -> bool:
"""Reads a model from a proto file."""
return self.__helper.read_model_from_proto_file(proto_file)
def export_to_proto_file(self, proto_file: str) -> bool:
"""Writes a model to a proto file."""
return self.__helper.write_model_to_proto_file(proto_file)
# Model getters and Setters
@property
def num_variables(self) -> int:
"""Returns the number of variables in the model."""
return self.__helper.num_variables()
@property
def num_constraints(self) -> int:
"""The number of constraints in the model."""
return self.__helper.num_constraints()
@property
def name(self) -> str:
"""The name of the model."""
return self.__helper.name()
@name.setter
def name(self, name: str):
self.__helper.set_name(name)
@property
def helper(self) -> mbh.ModelBuilderHelper:
"""Returns the model builder helper."""
return self.__helper
class Solver:
"""Main solver class.
The purpose of this class is to search for a solution to the model provided
to the solve() method.
Once solve() is called, this class allows inspecting the solution found
with the value() method, as well as general statistics about the solve
procedure.
"""
def __init__(self, solver_name: str):
self.__solve_helper: mbh.ModelSolverHelper = mbh.ModelSolverHelper(solver_name)
self.log_callback: Optional[Callable[[str], None]] = None
def solver_is_supported(self) -> bool:
"""Checks whether the requested solver backend was found."""
return self.__solve_helper.solver_is_supported()
# Solver backend and parameters.
def set_time_limit_in_seconds(self, limit: NumberT) -> None:
"""Sets a time limit for the solve() call."""
self.__solve_helper.set_time_limit_in_seconds(limit)
def set_solver_specific_parameters(self, parameters: str) -> None:
"""Sets parameters specific to the solver backend."""
self.__solve_helper.set_solver_specific_parameters(parameters)
def enable_output(self, enabled: bool) -> None:
"""Controls the solver backend logs."""
self.__solve_helper.enable_output(enabled)
def solve(self, model: Model) -> SolveStatus:
"""Solves a problem and passes each solution to the callback if not null."""
if self.log_callback is not None:
self.__solve_helper.set_log_callback(self.log_callback)
else:
self.__solve_helper.clear_log_callback()
self.__solve_helper.solve(model.helper)
return SolveStatus(self.__solve_helper.status())
def stop_search(self):
"""Stops the current search asynchronously."""
self.__solve_helper.interrupt_solve()
def value(self, expr: LinearExprT) -> np.double:
"""Returns the value of a linear expression after solve."""
if not self.__solve_helper.has_solution():
return pd.NA
if mbn.is_a_number(expr):
return expr
elif isinstance(expr, Variable):
return self.__solve_helper.var_value(expr.index)
elif isinstance(expr, LinearExpr):
flat_expr = _as_flat_linear_expression(expr)
return self.__solve_helper.expression_value(
flat_expr._variable_indices,
flat_expr._coefficients,
flat_expr._offset,
)
else:
raise TypeError(f"Unknown expression {expr!r} of type {type(expr)}")
def values(self, variables: _IndexOrSeries) -> pd.Series:
"""Returns the values of the input variables.
If `variables` is a `pd.Index`, then the output will be indexed by the
variables. If `variables` is a `pd.Series` indexed by the underlying
dimensions, then the output will be indexed by the same underlying
dimensions.
Args:
variables (Union[pd.Index, pd.Series]): The set of variables from which to
get the values.
Returns:
pd.Series: The values of all variables in the set.
"""
if not self.__solve_helper.has_solution():
return _attribute_series(func=lambda v: pd.NA, values=variables)
return _attribute_series(
func=lambda v: self.__solve_helper.var_value(v.index),
values=variables,
)
def reduced_costs(self, variables: _IndexOrSeries) -> pd.Series:
"""Returns the reduced cost of the input variables.
If `variables` is a `pd.Index`, then the output will be indexed by the
variables. If `variables` is a `pd.Series` indexed by the underlying
dimensions, then the output will be indexed by the same underlying
dimensions.
Args:
variables (Union[pd.Index, pd.Series]): The set of variables from which to
get the values.
Returns:
pd.Series: The reduced cost of all variables in the set.
"""
if not self.__solve_helper.has_solution():
return _attribute_series(func=lambda v: pd.NA, values=variables)
return _attribute_series(
func=lambda v: self.__solve_helper.reduced_cost(v.index),
values=variables,
)
def reduced_cost(self, var: Variable) -> np.double:
"""Returns the reduced cost of a linear expression after solve."""
if not self.__solve_helper.has_solution():
return pd.NA
return self.__solve_helper.reduced_cost(var.index)
def dual_values(self, constraints: _IndexOrSeries) -> pd.Series:
"""Returns the dual values of the input constraints.
If `constraints` is a `pd.Index`, then the output will be indexed by the
constraints. If `constraints` is a `pd.Series` indexed by the underlying
dimensions, then the output will be indexed by the same underlying
dimensions.
Args:
constraints (Union[pd.Index, pd.Series]): The set of constraints from
which to get the dual values.
Returns:
pd.Series: The dual_values of all constraints in the set.
"""
if not self.__solve_helper.has_solution():
return _attribute_series(func=lambda v: pd.NA, values=constraints)
return _attribute_series(
func=lambda v: self.__solve_helper.dual_value(v.index),
values=constraints,
)
def dual_value(self, ct: LinearConstraint) -> np.double:
"""Returns the dual value of a linear constraint after solve."""
if not self.__solve_helper.has_solution():
return pd.NA
return self.__solve_helper.dual_value(ct.index)
def activity(self, ct: LinearConstraint) -> np.double:
"""Returns the activity of a linear constraint after solve."""
if not self.__solve_helper.has_solution():
return pd.NA
return self.__solve_helper.activity(ct.index)
@property
def objective_value(self) -> np.double:
"""Returns the value of the objective after solve."""
if not self.__solve_helper.has_solution():
return pd.NA
return self.__solve_helper.objective_value()
@property
def best_objective_bound(self) -> np.double:
"""Returns the best lower (upper) bound found when min(max)imizing."""
if not self.__solve_helper.has_solution():
return pd.NA
return self.__solve_helper.best_objective_bound()
@property
def status_string(self) -> str:
"""Returns additional information of the last solve.
It can describe why the model is invalid.
"""
return self.__solve_helper.status_string()
@property
def wall_time(self) -> np.double:
return self.__solve_helper.wall_time()
@property
def user_time(self) -> np.double:
return self.__solve_helper.user_time()
# The maximum number of terms to display in a linear expression's repr.
_MAX_LINEAR_EXPRESSION_REPR_TERMS = 5
@dataclasses.dataclass(repr=False, eq=False, frozen=True)
class _LinearExpression(LinearExpr):
"""For variables x, an expression: offset + sum_{i in I} coeff_i * x_i."""
__slots__ = ("_variable_indices", "_coefficients", "_offset", "_helper")
_variable_indices: npt.NDArray[np.int32]
_coefficients: npt.NDArray[np.double]
_offset: float
_helper: Optional[mbh.ModelBuilderHelper]
@property
def variable_indices(self) -> npt.NDArray[np.int32]:
return self._variable_indices
@property
def coefficients(self) -> npt.NDArray[np.double]:
return self._coefficients
@property
def constant(self) -> float:
return self._offset
@property
def helper(self) -> Optional[mbh.ModelBuilderHelper]:
return self._helper
def __repr__(self):
return self.__str__()
def __str__(self):
if self._helper is None:
return str(self._offset)
result = []
for index, coeff in zip(self.variable_indices, self.coefficients):
if len(result) >= _MAX_LINEAR_EXPRESSION_REPR_TERMS:
result.append(" + ...")
break
var_name = Variable(self._helper, index, None, None, None).name
if not result and mbn.is_one(coeff):
result.append(var_name)
elif not result and mbn.is_minus_one(coeff):
result.append(f"-{var_name}")
elif not result:
result.append(f"{coeff} * {var_name}")
elif mbn.is_one(coeff):
result.append(f" + {var_name}")
elif mbn.is_minus_one(coeff):
result.append(f" - {var_name}")
elif coeff > 0.0:
result.append(f" + {coeff} * {var_name}")
elif coeff < 0.0:
result.append(f" - {-coeff} * {var_name}")
if not result:
return f"{self.constant}"
if self.constant > 0:
result.append(f" + {self.constant}")
elif self.constant < 0:
result.append(f" - {-self.constant}")
return "".join(result)
def _sum_as_flat_linear_expression(
to_process: List[Tuple[LinearExprT, float]], offset: float = 0.0
) -> _LinearExpression:
"""Creates a _LinearExpression as the sum of terms."""
indices = []
coeffs = []
helper = None
while to_process: # Flatten AST of LinearTypes.
expr, coeff = to_process.pop()
if isinstance(expr, _Sum):
to_process.append((expr._left, coeff))
to_process.append((expr._right, coeff))
elif isinstance(expr, Variable):
indices.append([expr.index])
coeffs.append([coeff])
if helper is None:
helper = expr.helper
elif mbn.is_a_number(expr):
offset += coeff * cast(NumberT, expr)
elif isinstance(expr, _Product):
to_process.append((expr._expression, coeff * expr._coefficient))
elif isinstance(expr, _LinearExpression):
offset += coeff * expr._offset
if expr._helper is not None:
indices.append(expr.variable_indices)
coeffs.append(np.multiply(expr.coefficients, coeff))
if helper is None:
helper = expr._helper
else:
raise TypeError(
"Unrecognized linear expression: " + str(expr) + f" {type(expr)}"
)
if helper is not None:
all_indices: npt.NDArray[np.int32] = np.concatenate(indices, axis=0)
all_coeffs: npt.NDArray[np.double] = np.concatenate(coeffs, axis=0)
sorted_indices, sorted_coefficients = helper.sort_and_regroup_terms(
all_indices, all_coeffs
)
return _LinearExpression(sorted_indices, sorted_coefficients, offset, helper)
else:
assert not indices
assert not coeffs
return _LinearExpression(
_variable_indices=np.zeros(dtype=np.int32, shape=[0]),
_coefficients=np.zeros(dtype=np.double, shape=[0]),
_offset=offset,
_helper=None,
)
def _as_flat_linear_expression(base_expr: LinearExprT) -> _LinearExpression:
"""Converts floats, ints and Linear objects to a LinearExpression."""
if isinstance(base_expr, _LinearExpression):
return base_expr
return _sum_as_flat_linear_expression(to_process=[(base_expr, 1.0)], offset=0.0)
@dataclasses.dataclass(repr=False, eq=False, frozen=True)
class _Sum(LinearExpr):
"""Represents the (deferred) sum of two expressions."""
__slots__ = ("_left", "_right")
_left: LinearExprT
_right: LinearExprT
def __repr__(self):
return self.__str__()
def __str__(self):
return str(_as_flat_linear_expression(self))
@dataclasses.dataclass(repr=False, eq=False, frozen=True)
class _Product(LinearExpr):
"""Represents the (deferred) product of an expression by a constant."""
__slots__ = ("_expression", "_coefficient")
_expression: LinearExpr
_coefficient: NumberT
def __repr__(self):
return self.__str__()
def __str__(self):
return str(_as_flat_linear_expression(self))
def _get_index(obj: _IndexOrSeries) -> pd.Index:
"""Returns the indices of `obj` as a `pd.Index`."""
if isinstance(obj, pd.Series):
return obj.index
return obj
def _attribute_series(
*,
func: Callable[[_VariableOrConstraint], NumberT],
values: _IndexOrSeries,
) -> pd.Series:
"""Returns the attributes of `values`.
Args:
func: The function to call for getting the attribute data.
values: The values that the function will be applied (element-wise) to.
Returns:
pd.Series: The attribute values.
"""
return pd.Series(
data=[func(v) for v in values],
index=_get_index(values),
)
def _convert_to_series_and_validate_index(
value_or_series: Union[bool, NumberT, pd.Series], index: pd.Index
) -> pd.Series:
"""Returns a pd.Series of the given index with the corresponding values.
Args:
value_or_series: the values to be converted (if applicable).
index: the index of the resulting pd.Series.
Returns:
pd.Series: The set of values with the given index.
Raises:
TypeError: If the type of `value_or_series` is not recognized.
ValueError: If the index does not match.
"""
if mbn.is_a_number(value_or_series) or isinstance(value_or_series, bool):
result = pd.Series(data=value_or_series, index=index)
elif isinstance(value_or_series, pd.Series):
if value_or_series.index.equals(index):
result = value_or_series
else:
raise ValueError("index does not match")
else:
raise TypeError("invalid type={}".format(type(value_or_series)))
return result
def _convert_to_var_series_and_validate_index(
var_or_series: Union["Variable", pd.Series], index: pd.Index
) -> pd.Series:
"""Returns a pd.Series of the given index with the corresponding values.
Args:
var_or_series: the variables to be converted (if applicable).
index: the index of the resulting pd.Series.
Returns:
pd.Series: The set of values with the given index.
Raises:
TypeError: If the type of `value_or_series` is not recognized.
ValueError: If the index does not match.
"""
if isinstance(var_or_series, Variable):
result = pd.Series(data=var_or_series, index=index)
elif isinstance(var_or_series, pd.Series):
if var_or_series.index.equals(index):
result = var_or_series
else:
raise ValueError("index does not match")
else:
raise TypeError("invalid type={}".format(type(var_or_series)))
return result
# Compatibility.
ModelBuilder = Model
ModelSolver = Solver