26#include "absl/memory/memory.h"
27#include "absl/status/status.h"
28#include "absl/status/statusor.h"
29#include "absl/strings/match.h"
30#include "absl/strings/str_cat.h"
31#include "absl/strings/str_join.h"
32#include "absl/strings/str_split.h"
33#include "absl/time/clock.h"
34#include "absl/time/time.h"
38#include "ortools/linear_solver/linear_solver.pb.h"
40#include "ortools/math_opt/callback.pb.h"
46#include "ortools/math_opt/model.pb.h"
47#include "ortools/math_opt/model_parameters.pb.h"
48#include "ortools/math_opt/model_update.pb.h"
49#include "ortools/math_opt/parameters.pb.h"
50#include "ortools/math_opt/result.pb.h"
51#include "ortools/math_opt/solution.pb.h"
52#include "ortools/math_opt/sparse_containers.pb.h"
55#include "ortools/sat/sat_parameters.pb.h"
62constexpr double kInf = std::numeric_limits<double>::infinity();
65bool ApplyCutoff(
const double cutoff, MPModelProto*
model) {
68 if (
model->has_quadratic_objective()) {
74 MPConstraintProto*
const cutoff_constraint =
model->add_constraint();
75 for (
int i = 0; i <
model->variable_size(); ++i) {
76 const double obj_coef =
model->variable(i).objective_coefficient();
78 cutoff_constraint->add_var_index(i);
79 cutoff_constraint->add_coefficient(obj_coef);
82 const double cutoff_minus_offset = cutoff -
model->objective_offset();
83 if (
model->maximize()) {
85 cutoff_constraint->set_lower_bound(cutoff_minus_offset);
88 cutoff_constraint->set_upper_bound(cutoff_minus_offset);
95std::vector<std::string> SetSolveParameters(
96 const SolveParametersProto&
parameters,
const bool has_message_callback,
97 MPModelRequest& request) {
98 std::vector<std::string> warnings;
100 request.set_solver_time_limit_seconds(absl::ToDoubleSeconds(
104 warnings.push_back(
"The node_limit parameter is not supported for CP-SAT.");
114 sat::SatParameters sat_parameters;
118 sat_parameters.set_catch_sigint_signal(
false);
121 sat_parameters.set_random_seed(
parameters.random_seed());
124 sat_parameters.set_num_search_workers(
parameters.threads());
126 if (
parameters.has_relative_gap_tolerance()) {
127 sat_parameters.set_relative_gap_limit(
parameters.relative_gap_tolerance());
130 if (
parameters.has_absolute_gap_tolerance()) {
131 sat_parameters.set_absolute_gap_limit(
parameters.absolute_gap_tolerance());
136 "The best_bound_limit parameter is not supported for CP-SAT.");
140 "The objective_limit parameter is not supported for CP-SAT.");
144 sat_parameters.set_stop_after_first_solution(
true);
146 warnings.push_back(absl::StrCat(
147 "The CP-SAT solver only supports value 1 for solution_limit, found: ",
152 if (
parameters.lp_algorithm() != LP_ALGORITHM_UNSPECIFIED) {
154 absl::StrCat(
"Setting the LP Algorithm (was set to ",
156 ") is not supported for CP_SAT solver"));
158 if (
parameters.presolve() != EMPHASIS_UNSPECIFIED) {
161 sat_parameters.set_cp_model_presolve(
false);
164 case EMPHASIS_MEDIUM:
166 case EMPHASIS_VERY_HIGH:
167 sat_parameters.set_cp_model_presolve(
true);
172 <<
" unknown, error setting CP-SAT parameters";
175 if (
parameters.scaling() != EMPHASIS_UNSPECIFIED) {
176 warnings.push_back(absl::StrCat(
"Setting the scaling (was set to ",
178 ") is not supported for CP_SAT solver"));
180 if (
parameters.cuts() != EMPHASIS_UNSPECIFIED) {
185 sat_parameters.set_add_cg_cuts(
false);
186 sat_parameters.set_add_mir_cuts(
false);
187 sat_parameters.set_add_zero_half_cuts(
false);
188 sat_parameters.set_add_clique_cuts(
false);
189 sat_parameters.set_max_all_diff_cut_size(0);
190 sat_parameters.set_add_lin_max_cuts(
false);
193 case EMPHASIS_MEDIUM:
195 case EMPHASIS_VERY_HIGH:
199 <<
" unknown, error setting CP-SAT parameters";
202 if (
parameters.heuristics() != EMPHASIS_UNSPECIFIED) {
203 warnings.push_back(absl::StrCat(
"Setting the heuristics (was set to ",
205 ") is not supported for CP_SAT solver"));
207 sat_parameters.MergeFrom(
parameters.cp_sat());
212 if (has_message_callback) {
215 sat_parameters.set_log_search_progress(
true);
219 sat_parameters.set_log_to_stdout(
false);
223 request.set_enable_internal_solver_output(
parameters.enable_output());
226 request.set_solver_specific_parameters(
231absl::StatusOr<std::pair<SolveStatsProto, TerminationProto>>
232GetTerminationAndStats(
const bool is_interrupted,
const bool maximize,
233 const bool used_cutoff,
234 const MPSolutionResponse&
response) {
235 SolveStatsProto solve_stats;
236 TerminationProto termination;
239 solve_stats.mutable_problem_status()->set_primal_status(
240 FEASIBILITY_STATUS_UNDETERMINED);
241 solve_stats.set_best_primal_bound(maximize ? -
kInf :
kInf);
242 solve_stats.mutable_problem_status()->set_dual_status(
243 FEASIBILITY_STATUS_UNDETERMINED);
244 solve_stats.set_best_dual_bound(maximize ?
kInf : -
kInf);
248 case MPSOLVER_OPTIMAL:
251 solve_stats.mutable_problem_status()->set_primal_status(
252 FEASIBILITY_STATUS_FEASIBLE);
253 solve_stats.set_best_primal_bound(
response.objective_value());
254 solve_stats.mutable_problem_status()->set_dual_status(
255 FEASIBILITY_STATUS_FEASIBLE);
256 solve_stats.set_best_dual_bound(
response.best_objective_bound());
258 case MPSOLVER_INFEASIBLE:
265 solve_stats.mutable_problem_status()->set_primal_status(
266 FEASIBILITY_STATUS_INFEASIBLE);
269 case MPSOLVER_UNKNOWN_STATUS:
283 if (absl::StrContains(
response.status_str(),
"infeasible or unbounded")) {
285 TERMINATION_REASON_INFEASIBLE_OR_UNBOUNDED,
response.status_str());
286 solve_stats.mutable_problem_status()->set_primal_or_dual_infeasible(
293 case MPSOLVER_FEASIBLE:
295 is_interrupted ? LIMIT_INTERRUPTED : LIMIT_UNDETERMINED,
297 solve_stats.mutable_problem_status()->set_primal_status(
298 FEASIBILITY_STATUS_FEASIBLE);
299 solve_stats.set_best_primal_bound(
response.objective_value());
300 solve_stats.set_best_dual_bound(
response.best_objective_bound());
301 if (std::isfinite(
response.best_objective_bound())) {
302 solve_stats.mutable_problem_status()->set_dual_status(
303 FEASIBILITY_STATUS_FEASIBLE);
306 case MPSOLVER_NOT_SOLVED:
308 is_interrupted ? LIMIT_INTERRUPTED : LIMIT_UNDETERMINED,
311 case MPSOLVER_MODEL_INVALID:
312 return absl::InternalError(
313 absl::StrCat(
"cp-sat solver returned MODEL_INVALID, details: ",
316 return absl::InternalError(
317 absl::StrCat(
"unexpected solve status: ",
response.status()));
319 return std::make_pair(std::move(solve_stats), std::move(termination));
329 model.variables().ids().end());
330 std::vector linear_constraint_ids(
model.linear_constraints().ids().begin(),
331 model.linear_constraints().ids().end());
333 if (!
model.objective().quadratic_coefficients().row_ids().empty()) {
334 return absl::InvalidArgumentError(
335 "MathOpt does not currently support CP-SAT models with quadratic "
339 std::move(cp_sat_model),
341 std::move(linear_constraint_ids)));
346 const ModelSolveParametersProto& model_parameters,
348 const CallbackRegistrationProto& callback_registration,
const Callback cb,
350 const absl::Time
start = absl::Now();
353 callback_registration,
354 {CALLBACK_EVENT_MIP_SOLUTION}));
355 if (callback_registration.add_lazy_constraints()) {
356 return absl::InvalidArgumentError(
357 "CallbackRegistrationProto.add_lazy_constraints=true is not supported "
363 SolveResultProto result;
367 *req.mutable_model() = cp_sat_model_;
369 req.set_solver_type(MPModelRequest::SAT_INTEGER_PROGRAMMING);
370 bool used_cutoff =
false;
372 std::vector<std::string> param_warnings =
374 message_cb !=
nullptr, req);
376 used_cutoff = ApplyCutoff(
parameters.cutoff_limit(), req.mutable_model());
378 param_warnings.push_back(
379 "The cutoff_limit parameter not supported for quadratic objectives "
383 if (!param_warnings.empty()) {
384 return absl::InvalidArgumentError(absl::StrJoin(param_warnings,
"; "));
388 if (!model_parameters.solution_hints().empty()) {
390 for (
const auto [
id, val] :
391 MakeView(model_parameters.solution_hints(0).variable_values())) {
392 while (variable_ids_[i] <
id) {
395 req.mutable_model()->mutable_solution_hint()->add_var_index(i);
396 req.mutable_model()->mutable_solution_hint()->add_var_value(val);
404 std::atomic<bool> interrupt_solve =
false;
409 interrupter, [&]() { local_interrupter.
Interrupt(); });
411 std::function<void(
const std::string&)> logging_callback;
412 if (message_cb !=
nullptr) {
413 logging_callback = [&](
const std::string&
message) {
414 message_cb(absl::StrSplit(
message,
'\n'));
418 const absl::flat_hash_set<CallbackEventProto> events =
420 std::function<void(
const MPSolution&)> solution_callback;
421 absl::Status callback_error = absl::OkStatus();
422 if (events.contains(CALLBACK_EVENT_MIP_SOLUTION)) {
423 solution_callback = [
this, &cb, &callback_error, &local_interrupter,
424 &model_parameters](
const MPSolution& mp_solution) {
425 if (!callback_error.ok()) {
429 CallbackDataProto cb_data;
430 cb_data.set_event(CALLBACK_EVENT_MIP_SOLUTION);
431 *cb_data.mutable_primal_solution_vector() =
432 ExtractSolution(mp_solution.variable_value(), model_parameters);
433 const absl::StatusOr<CallbackResultProto> cb_result = cb(cb_data);
434 if (!cb_result.ok()) {
435 callback_error = cb_result.status();
438 local_interrupter.Interrupt();
439 }
else if (cb_result->terminate()) {
440 local_interrupter.Interrupt();
452 logging_callback, solution_callback));
455 (
auto [solve_stats, termination]),
457 cp_sat_model_.maximize(),
459 *result.mutable_solve_stats() = std::move(solve_stats);
460 *result.mutable_termination() = std::move(termination);
461 if (
response.status() == MPSOLVER_OPTIMAL ||
462 response.status() == MPSOLVER_FEASIBLE) {
463 PrimalSolutionProto& solution =
464 *result.add_solutions()->mutable_primal_solution();
465 *solution.mutable_variable_values() =
466 ExtractSolution(
response.variable_value(), model_parameters);
467 solution.set_objective_value(
response.objective_value());
468 solution.set_feasibility_status(SOLUTION_STATUS_FEASIBLE);
472 absl::Now() -
start, result.mutable_solve_stats()->mutable_solve_time()));
484 return absl::InternalError(
"CP-SAT solver does not support incrementalism");
487CpSatSolver::CpSatSolver(MPModelProto cp_sat_model,
489 std::vector<int64_t> linear_constraint_ids)
490 : cp_sat_model_(
std::move(cp_sat_model)),
492 linear_constraint_ids_(
std::move(linear_constraint_ids)) {}
494SparseDoubleVectorProto CpSatSolver::ExtractSolution(
495 const absl::Span<const double> cp_sat_variable_values,
496 const ModelSolveParametersProto& model_parameters)
const {
499 CHECK_EQ(cp_sat_variable_values.size(), variable_ids_.size());
501 SparseVectorFilterPredicate predicate(
502 model_parameters.variable_values_filter());
503 SparseDoubleVectorProto result;
504 for (
int i = 0; i < variable_ids_.size(); ++i) {
505 const int64_t
id = variable_ids_[i];
506 const double value = cp_sat_variable_values[i];
507 if (predicate.AcceptsAndUpdate(
id,
value)) {
509 result.add_values(
value);
515InvertedBounds CpSatSolver::ListInvertedBounds()
const {
516 InvertedBounds inverted_bounds;
517 for (
int v = 0; v < cp_sat_model_.variable_size(); ++v) {
518 const MPVariableProto&
var = cp_sat_model_.variable(v);
519 if (
var.lower_bound() >
var.upper_bound()) {
520 inverted_bounds.variables.push_back(variable_ids_[v]);
523 for (
int c = 0; c < cp_sat_model_.constraint_size(); ++c) {
524 const MPConstraintProto& cstr = cp_sat_model_.constraint(c);
525 if (cstr.lower_bound() > cstr.upper_bound()) {
526 inverted_bounds.linear_constraints.push_back(linear_constraint_ids_[c]);
530 return inverted_bounds;
#define CHECK_EQ(val1, val2)
#define ASSIGN_OR_RETURN(lhs, rexpr)
#define RETURN_IF_ERROR(expr)
bool CanUpdate(const ModelUpdateProto &model_update) override
absl::Status Update(const ModelUpdateProto &model_update) override
static absl::StatusOr< std::unique_ptr< SolverInterface > > New(const ModelProto &model, const InitArgs &init_args)
absl::StatusOr< SolveResultProto > Solve(const SolveParametersProto ¶meters, const ModelSolveParametersProto &model_parameters, MessageCallback message_cb, const CallbackRegistrationProto &callback_registration, Callback cb, SolveInterrupter *interrupter) override
bool IsInterrupted() const
CallbackId AddInterruptionCallback(Callback callback)
std::function< void(const std::vector< std::string > &)> MessageCallback
std::function< absl::StatusOr< CallbackResultProto >(const CallbackDataProto &)> Callback
SharedResponseManager * response
absl::Span< const int64_t > variable_ids
TerminationProto FeasibleTermination(const LimitProto limit, const absl::string_view detail)
absl::Status CheckRegisteredCallbackEvents(const CallbackRegistrationProto ®istration, const absl::flat_hash_set< CallbackEventProto > &supported_events)
MATH_OPT_REGISTER_SOLVER(SOLVER_TYPE_CP_SAT, CpSatSolver::New)
absl::StatusOr<::operations_research::MPModelProto > MathOptModelToMPModelProto(const ::operations_research::math_opt::ModelProto &model)
SparseVectorView< T > MakeView(absl::Span< const int64_t > ids, const Collection &values)
TerminationProto NoSolutionFoundTermination(const LimitProto limit, const absl::string_view detail)
TerminationProto TerminateForReason(const TerminationReasonProto reason, const absl::string_view detail)
absl::flat_hash_set< CallbackEventProto > EventSet(const CallbackRegistrationProto &callback_registration)
Collection of objects used to extend the Constraint Solver library.
std::string ProtoEnumToString(ProtoEnumType enum_value)
absl::StatusOr< MPSolutionResponse > SatSolveProto(MPModelRequest request, std::atomic< bool > *interrupt_solve, std::function< void(const std::string &)> logging_callback, std::function< void(const MPSolution &)> solution_callback)
std::string EncodeSatParametersAsString(const sat::SatParameters ¶meters)
inline ::absl::StatusOr< absl::Duration > DecodeGoogleApiProto(const google::protobuf::Duration &proto)
inline ::absl::StatusOr< google::protobuf::Duration > EncodeGoogleApiProto(absl::Duration d)