23#include "absl/container/flat_hash_map.h"
24#include "absl/status/status.h"
25#include "absl/status/statusor.h"
27#include "ortools/linear_solver/linear_solver.pb.h"
31#include "ortools/math_opt/model.pb.h"
32#include "ortools/math_opt/sparse_containers.pb.h"
39absl::Status IsSupported(
const MPModelProto&
model) {
41 if (validity_string.length() > 0) {
42 return absl::InvalidArgumentError(validity_string);
44 if (
model.general_constraint_size() > 0) {
45 return absl::InvalidArgumentError(
"General constraints are not supported");
47 if (
model.solution_hint().var_index_size() > 0) {
48 return absl::InvalidArgumentError(
"Solution Hint not supported");
50 return absl::OkStatus();
53absl::Status IsSupported(
const math_opt::ModelProto&
model) {
57bool AnyVarNamed(
const MPModelProto&
model) {
58 for (
const MPVariableProto&
var :
model.variable()) {
59 if (
var.name().length() > 0) {
66bool AnyConstraintNamed(
const MPModelProto&
model) {
67 for (
const MPConstraintProto& constraint :
model.constraint()) {
68 if (constraint.name().length() > 0) {
77absl::StatusOr<::operations_research::math_opt::ModelProto>
82 output.set_name(
model.name());
84 math_opt::VariablesProto*
const vars = output.mutable_variables();
85 int linear_objective_non_zeros = 0;
86 const int num_vars =
model.variable_size();
87 const bool vars_have_name = AnyVarNamed(
model);
88 vars->mutable_lower_bounds()->Reserve(num_vars);
89 vars->mutable_upper_bounds()->Reserve(num_vars);
90 vars->mutable_integers()->Reserve(num_vars);
92 vars->mutable_names()->Reserve(num_vars);
94 for (
int i = 0; i <
model.variable_size(); ++i) {
95 const MPVariableProto&
var =
model.variable(i);
96 if (
var.objective_coefficient() != 0.0) {
97 ++linear_objective_non_zeros;
100 vars->add_lower_bounds(
var.lower_bound());
101 vars->add_upper_bounds(
var.upper_bound());
102 vars->add_integers(
var.is_integer());
103 if (vars_have_name) {
104 vars->add_names(
var.name());
108 math_opt::ObjectiveProto*
const objective = output.mutable_objective();
109 if (linear_objective_non_zeros > 0) {
110 objective->mutable_linear_coefficients()->mutable_ids()->Reserve(
111 linear_objective_non_zeros);
112 objective->mutable_linear_coefficients()->mutable_values()->Reserve(
113 linear_objective_non_zeros);
114 for (
int j = 0; j < num_vars; ++j) {
115 const double value =
model.variable(j).objective_coefficient();
116 if (
value == 0.0)
continue;
117 objective->mutable_linear_coefficients()->add_ids(j);
118 objective->mutable_linear_coefficients()->add_values(
value);
121 const MPQuadraticObjective& origin_qp_terms =
model.quadratic_objective();
122 const int num_qp_terms = origin_qp_terms.coefficient().size();
123 if (num_qp_terms > 0) {
129 std::vector<std::pair<std::pair<int, int>,
double>> qp_terms_in_order;
130 for (
int k = 0; k < num_qp_terms; ++k) {
131 int first_index = origin_qp_terms.qvar1_index(k);
132 int second_index = origin_qp_terms.qvar2_index(k);
133 if (first_index > second_index) {
136 qp_terms_in_order.emplace_back(std::make_pair(first_index, second_index),
137 origin_qp_terms.coefficient(k));
139 std::sort(qp_terms_in_order.begin(), qp_terms_in_order.end());
140 SparseDoubleMatrixProto& destination_qp_terms =
141 *objective->mutable_quadratic_coefficients();
142 std::pair<int, int> previous = {-1, -1};
143 for (
const auto& [indices,
coeff] : qp_terms_in_order) {
144 if (indices == previous) {
145 *destination_qp_terms.mutable_coefficients()->rbegin() +=
coeff;
147 destination_qp_terms.add_row_ids(indices.first);
148 destination_qp_terms.add_column_ids(indices.second);
149 destination_qp_terms.add_coefficients(
coeff);
154 objective->set_maximize(
model.maximize());
155 objective->set_offset(
model.objective_offset());
157 math_opt::LinearConstraintsProto*
const constraints =
158 output.mutable_linear_constraints();
159 const int num_constraints =
model.constraint_size();
160 const bool constraints_have_name = AnyConstraintNamed(
model);
161 int num_non_zeros = 0;
162 constraints->mutable_lower_bounds()->Reserve(num_constraints);
163 constraints->mutable_upper_bounds()->Reserve(num_constraints);
164 if (constraints_have_name) {
165 constraints->mutable_names()->Reserve(num_constraints);
167 for (
int i = 0; i < num_constraints; ++i) {
168 const MPConstraintProto& constraint =
model.constraint(i);
169 constraints->add_ids(i);
170 constraints->add_lower_bounds(constraint.lower_bound());
171 constraints->add_upper_bounds(constraint.upper_bound());
172 if (constraints_have_name) {
173 constraints->add_names(constraint.name());
175 num_non_zeros += constraint.var_index_size();
178 SparseDoubleMatrixProto*
const matrix =
179 output.mutable_linear_constraint_matrix();
180 matrix->mutable_column_ids()->Reserve(num_non_zeros);
181 matrix->mutable_row_ids()->Reserve(num_non_zeros);
182 matrix->mutable_coefficients()->Reserve(num_non_zeros);
184 std::vector<std::pair<int, double>> terms_in_order;
185 for (
int i = 0; i < num_constraints; ++i) {
186 const MPConstraintProto& constraint =
model.constraint(i);
187 const int constraint_non_zeros = constraint.var_index_size();
188 for (
int k = 0; k < constraint_non_zeros; ++k) {
189 const double coefficient = constraint.coefficient(k);
193 terms_in_order.emplace_back(constraint.var_index(k),
coefficient);
195 std::sort(terms_in_order.begin(), terms_in_order.end());
196 for (
const auto& term : terms_in_order) {
197 matrix->add_row_ids(i);
198 matrix->add_column_ids(term.first);
199 matrix->add_coefficients(term.second);
201 terms_in_order.clear();
207 const ::operations_research::math_opt::ModelProto&
model) {
210 const bool vars_have_name =
model.variables().names_size() > 0;
211 const bool constraints_have_name =
212 model.linear_constraints().names_size() > 0;
213 absl::flat_hash_map<int64_t, int> variable_id_to_mp_position;
214 absl::flat_hash_map<int64_t, MPConstraintProto*>
215 constraint_id_to_mp_constraint;
218 output.set_name(
model.name());
221 output.mutable_variable()->Reserve(num_vars);
222 for (
int j = 0; j < num_vars; ++j) {
223 MPVariableProto*
const variable = output.add_variable();
224 variable_id_to_mp_position.emplace(
model.variables().ids(j), j);
225 variable->set_lower_bound(
model.variables().lower_bounds(j));
226 variable->set_upper_bound(
model.variables().upper_bounds(j));
227 variable->set_is_integer(
model.variables().integers(j));
228 if (vars_have_name) {
229 variable->set_name(
model.variables().names(j));
234 output.mutable_constraint()->Reserve(num_constraints);
235 for (
int i = 0; i < num_constraints; ++i) {
236 MPConstraintProto*
const constraint = output.add_constraint();
237 constraint_id_to_mp_constraint.emplace(
model.linear_constraints().ids(i),
239 constraint->set_lower_bound(
model.linear_constraints().lower_bounds(i));
240 constraint->set_upper_bound(
model.linear_constraints().upper_bounds(i));
241 if (constraints_have_name) {
242 constraint->set_name(
model.linear_constraints().names(i));
246 output.set_maximize(
model.objective().maximize());
247 output.set_objective_offset(
model.objective().offset());
250 const int var_position = variable_id_to_mp_position[
var];
251 MPVariableProto*
const variable = output.mutable_variable(var_position);
252 variable->set_objective_coefficient(
coef);
254 const SparseDoubleMatrixProto& origin_qp_terms =
255 model.objective().quadratic_coefficients();
256 if (!origin_qp_terms.coefficients().empty()) {
257 MPQuadraticObjective& destination_qp_terms =
258 *output.mutable_quadratic_objective();
259 for (
int k = 0; k < origin_qp_terms.coefficients().size(); ++k) {
260 destination_qp_terms.add_qvar1_index(
261 variable_id_to_mp_position[origin_qp_terms.row_ids(k)]);
262 destination_qp_terms.add_qvar2_index(
263 variable_id_to_mp_position[origin_qp_terms.column_ids(k)]);
264 destination_qp_terms.add_coefficient(origin_qp_terms.coefficients(k));
269 const int constraint_non_zeros =
270 model.linear_constraint_matrix().coefficients_size();
271 for (
int k = 0; k < constraint_non_zeros; ++k) {
272 const int64_t constraint_id =
model.linear_constraint_matrix().row_ids(k);
273 MPConstraintProto*
const constraint =
274 constraint_id_to_mp_constraint[constraint_id];
275 const int64_t variable_id =
model.linear_constraint_matrix().column_ids(k);
276 const int variable_position = variable_id_to_mp_position[variable_id];
277 constraint->add_var_index(variable_position);
278 const double value =
model.linear_constraint_matrix().coefficients(k);
279 constraint->add_coefficient(
value);
#define RETURN_IF_ERROR(expr)
int NumVariables(const VariablesProto &variables)
absl::StatusOr<::operations_research::MPModelProto > MathOptModelToMPModelProto(const ::operations_research::math_opt::ModelProto &model)
void swap(IdMap< K, V > &a, IdMap< K, V > &b)
SparseVectorView< T > MakeView(absl::Span< const int64_t > ids, const Collection &values)
int NumConstraints(const LinearConstraintsProto &linear_constraints)
absl::StatusOr<::operations_research::math_opt::ModelProto > MPModelProtoToMathOptModel(const ::operations_research::MPModelProto &model)
absl::Status ValidateModel(const ModelProto &model, const bool check_names)
Collection of objects used to extend the Constraint Solver library.
std::string FindErrorInMPModelProto(const MPModelProto &model, double abs_value_threshold, const bool accept_trivially_infeasible_bounds)
Returns an empty string iff the model is valid and not trivially infeasible.