diff --git a/CMakeLists.txt b/CMakeLists.txt index 34b37a73..9cde1046 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -206,7 +206,7 @@ if(NOT AMPLSOLVER) message(WARNING "Optional library amplsolver (ASL) was not found.") else() message(STATUS "Library amplsolver was found.") - add_executable(uno_ampl bindings/AMPL/AMPLModel.cpp bindings/AMPL/uno_ampl.cpp) + add_executable(uno_ampl bindings/AMPL/AMPLModel.cpp bindings/AMPL/AMPLUserCallbacks.cpp bindings/AMPL/uno_ampl.cpp) target_link_libraries(uno_ampl PUBLIC uno ${AMPLSOLVER} ${CMAKE_DL_LIBS}) add_definitions("-D HAS_AMPLSOLVER") diff --git a/bindings/AMPL/AMPLUserCallbacks.cpp b/bindings/AMPL/AMPLUserCallbacks.cpp new file mode 100644 index 00000000..16c5e8f4 --- /dev/null +++ b/bindings/AMPL/AMPLUserCallbacks.cpp @@ -0,0 +1,20 @@ +// Copyright (c) 2024 Charlie Vanaret +// Licensed under the MIT license. See LICENSE file in the project directory for details. + +#include "AMPLUserCallbacks.hpp" +#include "linear_algebra/Vector.hpp" +#include "optimization/Multipliers.hpp" + +namespace uno { + AMPLUserCallbacks::AMPLUserCallbacks(): UserCallbacks() { } + + void AMPLUserCallbacks::notify_acceptable_iterate(const Vector& /*primals*/, const Multipliers& /*multipliers*/, + double /*objective_multiplier*/) { + } + + void AMPLUserCallbacks::notify_new_primals(const Vector& /*primals*/) { + } + + void AMPLUserCallbacks::notify_new_multipliers(const Multipliers& /*multipliers*/) { + } +} // namespace \ No newline at end of file diff --git a/bindings/AMPL/AMPLUserCallbacks.hpp b/bindings/AMPL/AMPLUserCallbacks.hpp new file mode 100644 index 00000000..cdc30dde --- /dev/null +++ b/bindings/AMPL/AMPLUserCallbacks.hpp @@ -0,0 +1,20 @@ +// Copyright (c) 2024 Charlie Vanaret +// Licensed under the MIT license. See LICENSE file in the project directory for details. + +#ifndef UNO_AMPLUSERCALLBACKS_H +#define UNO_AMPLUSERCALLBACKS_H + +#include "tools/UserCallbacks.hpp" + +namespace uno { + class AMPLUserCallbacks: public UserCallbacks { + public: + AMPLUserCallbacks(); + + void notify_acceptable_iterate(const Vector& primals, const Multipliers& multipliers, double objective_multiplier) override; + void notify_new_primals(const Vector& primals) override; + void notify_new_multipliers(const Multipliers& multipliers) override; + }; +} // namespace + +#endif //UNO_AMPLUSERCALLBACKS_H \ No newline at end of file diff --git a/bindings/AMPL/uno_ampl.cpp b/bindings/AMPL/uno_ampl.cpp index 1b054961..368d0305 100644 --- a/bindings/AMPL/uno_ampl.cpp +++ b/bindings/AMPL/uno_ampl.cpp @@ -8,6 +8,7 @@ #include "ingredients/constraint_relaxation_strategies/ConstraintRelaxationStrategy.hpp" #include "ingredients/constraint_relaxation_strategies/ConstraintRelaxationStrategyFactory.hpp" #include "AMPLModel.hpp" +#include "AMPLUserCallbacks.hpp" #include "Uno.hpp" #include "model/ModelFactory.hpp" #include "options/DefaultOptions.hpp" @@ -50,8 +51,11 @@ namespace uno { auto globalization_mechanism = GlobalizationMechanismFactory::create(*constraint_relaxation_strategy, options); Uno uno = Uno(*globalization_mechanism, options); + // create the user callbacks + AMPLUserCallbacks user_callbacks{}; + // solve the instance - uno.solve(*model, initial_iterate, options); + uno.solve(*model, initial_iterate, options, user_callbacks); // std::cout << "memory_allocation_amount = " << memory_allocation_amount << '\n'; } catch (std::exception& exception) { diff --git a/uno/Uno.cpp b/uno/Uno.cpp index c0419b07..0f0a2897 100644 --- a/uno/Uno.cpp +++ b/uno/Uno.cpp @@ -19,6 +19,7 @@ #include "options/Options.hpp" #include "tools/Statistics.hpp" #include "tools/Timer.hpp" +#include "tools/UserCallbacks.hpp" namespace uno { Uno::Uno(GlobalizationMechanism& globalization_mechanism, const Options& options) : @@ -30,7 +31,15 @@ namespace uno { Level Logger::level = INFO; + // solve without user callbacks void Uno::solve(const Model& model, Iterate& current_iterate, const Options& options) { + // pass user callbacks that do nothing + NoUserCallbacks user_callbacks{}; + this->solve(model, current_iterate, options, user_callbacks); + } + + // solve with user callbacks + void Uno::solve(const Model& model, Iterate& current_iterate, const Options& options, UserCallbacks& user_callbacks) { Timer timer{}; Statistics statistics = Uno::create_statistics(model, options); WarmstartInformation warmstart_information{}; @@ -54,8 +63,11 @@ namespace uno { // compute an acceptable iterate by solving a subproblem at the current point warmstart_information.iterate_changed(); - this->globalization_mechanism.compute_next_iterate(statistics, model, current_iterate, trial_iterate, warmstart_information); + this->globalization_mechanism.compute_next_iterate(statistics, model, current_iterate, trial_iterate, warmstart_information, user_callbacks); termination = this->termination_criteria(trial_iterate.status, major_iterations, timer.get_duration()); + user_callbacks.notify_new_primals(trial_iterate.primals); + user_callbacks.notify_new_multipliers(trial_iterate.multipliers); + // the trial iterate becomes the current iterate for the next iteration std::swap(current_iterate, trial_iterate); } diff --git a/uno/Uno.hpp b/uno/Uno.hpp index e9a10369..716c0c43 100644 --- a/uno/Uno.hpp +++ b/uno/Uno.hpp @@ -14,12 +14,15 @@ namespace uno { class Options; class Statistics; class Timer; + class UserCallbacks; class Uno { public: Uno(GlobalizationMechanism& globalization_mechanism, const Options& options); + // solve with or without user callbacks void solve(const Model& model, Iterate& initial_iterate, const Options& options); + void solve(const Model& model, Iterate& initial_iterate, const Options& options, UserCallbacks& user_callbacks); static std::string current_version(); static void print_available_strategies(); diff --git a/uno/ingredients/constraint_relaxation_strategies/ConstraintRelaxationStrategy.hpp b/uno/ingredients/constraint_relaxation_strategies/ConstraintRelaxationStrategy.hpp index 3c2d1eb5..a80af245 100644 --- a/uno/ingredients/constraint_relaxation_strategies/ConstraintRelaxationStrategy.hpp +++ b/uno/ingredients/constraint_relaxation_strategies/ConstraintRelaxationStrategy.hpp @@ -24,6 +24,7 @@ namespace uno { class Subproblem; template class SymmetricMatrix; + class UserCallbacks; template class Vector; struct WarmstartInformation; @@ -50,7 +51,7 @@ namespace uno { // trial iterate acceptance [[nodiscard]] virtual bool is_iterate_acceptable(Statistics& statistics, Iterate& current_iterate, Iterate& trial_iterate, const Direction& direction, - double step_length, WarmstartInformation& warmstart_information) = 0; + double step_length, WarmstartInformation& warmstart_information, UserCallbacks& user_callbacks) = 0; [[nodiscard]] TerminationStatus check_termination(Iterate& iterate); // primal-dual residuals diff --git a/uno/ingredients/constraint_relaxation_strategies/FeasibilityRestoration.cpp b/uno/ingredients/constraint_relaxation_strategies/FeasibilityRestoration.cpp index 9068138a..85730f0e 100644 --- a/uno/ingredients/constraint_relaxation_strategies/FeasibilityRestoration.cpp +++ b/uno/ingredients/constraint_relaxation_strategies/FeasibilityRestoration.cpp @@ -10,8 +10,9 @@ #include "model/Model.hpp" #include "optimization/Iterate.hpp" #include "optimization/WarmstartInformation.hpp" -#include "symbolic/VectorView.hpp" #include "options/Options.hpp" +#include "symbolic/VectorView.hpp" +#include "tools/UserCallbacks.hpp" namespace uno { FeasibilityRestoration::FeasibilityRestoration(const Model& model, const Options& options) : @@ -148,7 +149,7 @@ namespace uno { } bool FeasibilityRestoration::is_iterate_acceptable(Statistics& statistics, Iterate& current_iterate, Iterate& trial_iterate, const Direction& direction, - double step_length, WarmstartInformation& warmstart_information) { + double step_length, WarmstartInformation& warmstart_information, UserCallbacks& user_callbacks) { // TODO pick right multipliers this->subproblem->postprocess_iterate(this->current_problem(), trial_iterate); this->compute_progress_measures(current_iterate, trial_iterate); @@ -176,6 +177,11 @@ namespace uno { predicted_reduction, this->current_problem().get_objective_multiplier()); } ConstraintRelaxationStrategy::set_progress_statistics(statistics, trial_iterate); + if (accept_iterate) { + user_callbacks.notify_acceptable_iterate(trial_iterate.primals, + this->current_phase == Phase::OPTIMALITY ? trial_iterate.multipliers : trial_iterate.feasibility_multipliers, + this->current_problem().get_objective_multiplier()); + } return accept_iterate; } diff --git a/uno/ingredients/constraint_relaxation_strategies/FeasibilityRestoration.hpp b/uno/ingredients/constraint_relaxation_strategies/FeasibilityRestoration.hpp index f268ced6..26810bc3 100644 --- a/uno/ingredients/constraint_relaxation_strategies/FeasibilityRestoration.hpp +++ b/uno/ingredients/constraint_relaxation_strategies/FeasibilityRestoration.hpp @@ -29,7 +29,7 @@ namespace uno { // trial iterate acceptance [[nodiscard]] bool is_iterate_acceptable(Statistics& statistics, Iterate& current_iterate, Iterate& trial_iterate, const Direction& direction, - double step_length, WarmstartInformation& warmstart_information) override; + double step_length, WarmstartInformation& warmstart_information, UserCallbacks& user_callbacks) override; // primal-dual residuals void compute_primal_dual_residuals(Iterate& iterate) override; diff --git a/uno/ingredients/constraint_relaxation_strategies/l1Relaxation.cpp b/uno/ingredients/constraint_relaxation_strategies/l1Relaxation.cpp index 2e600b52..64254dbb 100644 --- a/uno/ingredients/constraint_relaxation_strategies/l1Relaxation.cpp +++ b/uno/ingredients/constraint_relaxation_strategies/l1Relaxation.cpp @@ -11,6 +11,7 @@ #include "symbolic/VectorView.hpp" #include "options/Options.hpp" #include "tools/Statistics.hpp" +#include "tools/UserCallbacks.hpp" /* * Infeasibility detection and SQP methods for nonlinear optimization @@ -233,7 +234,7 @@ namespace uno { } bool l1Relaxation::is_iterate_acceptable(Statistics& statistics, Iterate& current_iterate, Iterate& trial_iterate, const Direction& direction, - double step_length, WarmstartInformation& /*warmstart_information*/) { + double step_length, WarmstartInformation& /*warmstart_information*/, UserCallbacks& user_callbacks) { this->subproblem->postprocess_iterate(this->l1_relaxed_problem, trial_iterate); this->compute_progress_measures(current_iterate, trial_iterate); trial_iterate.objective_multiplier = this->l1_relaxed_problem.get_objective_multiplier(); @@ -254,6 +255,7 @@ namespace uno { if (accept_iterate) { this->check_exact_relaxation(trial_iterate); // this->set_dual_residuals_statistics(statistics, trial_iterate); + user_callbacks.notify_acceptable_iterate(trial_iterate.primals, trial_iterate.multipliers, this->penalty_parameter); } this->set_progress_statistics(statistics, trial_iterate); return accept_iterate; diff --git a/uno/ingredients/constraint_relaxation_strategies/l1Relaxation.hpp b/uno/ingredients/constraint_relaxation_strategies/l1Relaxation.hpp index 805a6bbc..e3635d0c 100644 --- a/uno/ingredients/constraint_relaxation_strategies/l1Relaxation.hpp +++ b/uno/ingredients/constraint_relaxation_strategies/l1Relaxation.hpp @@ -36,7 +36,7 @@ namespace uno { // trial iterate acceptance [[nodiscard]] bool is_iterate_acceptable(Statistics& statistics, Iterate& current_iterate, Iterate& trial_iterate, const Direction& direction, - double step_length, WarmstartInformation& warmstart_information) override; + double step_length, WarmstartInformation& warmstart_information, UserCallbacks& user_callbacks) override; // primal-dual residuals void compute_primal_dual_residuals(Iterate& iterate) override; diff --git a/uno/ingredients/globalization_mechanisms/BacktrackingLineSearch.cpp b/uno/ingredients/globalization_mechanisms/BacktrackingLineSearch.cpp index 22b98793..5f2f722c 100644 --- a/uno/ingredients/globalization_mechanisms/BacktrackingLineSearch.cpp +++ b/uno/ingredients/globalization_mechanisms/BacktrackingLineSearch.cpp @@ -31,17 +31,17 @@ namespace uno { } void BacktrackingLineSearch::compute_next_iterate(Statistics& statistics, const Model& model, Iterate& current_iterate, Iterate& trial_iterate, - WarmstartInformation& warmstart_information) { + WarmstartInformation& warmstart_information, UserCallbacks& user_callbacks) { DEBUG2 << "Current iterate\n" << current_iterate << '\n'; this->constraint_relaxation_strategy.compute_feasible_direction(statistics, current_iterate, this->direction, warmstart_information); BacktrackingLineSearch::check_unboundedness(this->direction); - this->backtrack_along_direction(statistics, model, current_iterate, trial_iterate, warmstart_information); + this->backtrack_along_direction(statistics, model, current_iterate, trial_iterate, warmstart_information, user_callbacks); } // go a fraction along the direction by finding an acceptable step length void BacktrackingLineSearch::backtrack_along_direction(Statistics& statistics, const Model& model, Iterate& current_iterate, - Iterate& trial_iterate, WarmstartInformation& warmstart_information) { + Iterate& trial_iterate, WarmstartInformation& warmstart_information, UserCallbacks& user_callbacks) { double step_length = 1.; bool termination = false; size_t number_iterations = 0; @@ -59,7 +59,7 @@ namespace uno { this->scale_duals_with_step_length ? step_length : 1.); is_acceptable = this->constraint_relaxation_strategy.is_iterate_acceptable(statistics, current_iterate, trial_iterate, this->direction, - step_length, warmstart_information); + step_length, warmstart_information, user_callbacks); this->set_statistics(statistics, trial_iterate, this->direction, step_length, number_iterations); } catch (const EvaluationError& e) { diff --git a/uno/ingredients/globalization_mechanisms/BacktrackingLineSearch.hpp b/uno/ingredients/globalization_mechanisms/BacktrackingLineSearch.hpp index 21884c02..864bb13f 100644 --- a/uno/ingredients/globalization_mechanisms/BacktrackingLineSearch.hpp +++ b/uno/ingredients/globalization_mechanisms/BacktrackingLineSearch.hpp @@ -16,7 +16,7 @@ namespace uno { void initialize(Statistics& statistics, Iterate& initial_iterate, const Options& options) override; void compute_next_iterate(Statistics& statistics, const Model& model, Iterate& current_iterate, Iterate& trial_iterate, - WarmstartInformation& warmstart_information) override; + WarmstartInformation& warmstart_information, UserCallbacks& user_callbacks) override; private: const double backtracking_ratio; @@ -24,7 +24,7 @@ namespace uno { const bool scale_duals_with_step_length; void backtrack_along_direction(Statistics& statistics, const Model& model, Iterate& current_iterate, Iterate& trial_iterate, - WarmstartInformation& warmstart_information); + WarmstartInformation& warmstart_information, UserCallbacks& user_callbacks); [[nodiscard]] bool terminate_with_small_step_length(Statistics& statistics, Iterate& trial_iterate); [[nodiscard]] double decrease_step_length(double step_length) const; static void check_unboundedness(const Direction& direction); diff --git a/uno/ingredients/globalization_mechanisms/GlobalizationMechanism.hpp b/uno/ingredients/globalization_mechanisms/GlobalizationMechanism.hpp index 4a1d1c97..3ccb6ae6 100644 --- a/uno/ingredients/globalization_mechanisms/GlobalizationMechanism.hpp +++ b/uno/ingredients/globalization_mechanisms/GlobalizationMechanism.hpp @@ -13,6 +13,7 @@ namespace uno { class Model; class Options; class Statistics; + class UserCallbacks; struct WarmstartInformation; class GlobalizationMechanism { @@ -22,7 +23,7 @@ namespace uno { virtual void initialize(Statistics& statistics, Iterate& initial_iterate, const Options& options) = 0; virtual void compute_next_iterate(Statistics& statistics, const Model& model, Iterate& current_iterate, Iterate& trial_iterate, - WarmstartInformation& warmstart_information) = 0; + WarmstartInformation& warmstart_information, UserCallbacks& user_callbacks) = 0; [[nodiscard]] size_t get_hessian_evaluation_count() const; [[nodiscard]] size_t get_number_subproblems_solved() const; diff --git a/uno/ingredients/globalization_mechanisms/TrustRegionStrategy.cpp b/uno/ingredients/globalization_mechanisms/TrustRegionStrategy.cpp index 26179e1e..0b72b35d 100644 --- a/uno/ingredients/globalization_mechanisms/TrustRegionStrategy.cpp +++ b/uno/ingredients/globalization_mechanisms/TrustRegionStrategy.cpp @@ -39,7 +39,7 @@ namespace uno { } void TrustRegionStrategy::compute_next_iterate(Statistics& statistics, const Model& model, Iterate& current_iterate, Iterate& trial_iterate, - WarmstartInformation& warmstart_information) { + WarmstartInformation& warmstart_information, UserCallbacks& user_callbacks) { DEBUG2 << "Current iterate\n" << current_iterate << '\n'; size_t number_iterations = 0; @@ -77,7 +77,8 @@ namespace uno { GlobalizationMechanism::assemble_trial_iterate(model, current_iterate, trial_iterate, this->direction, 1., 1.); this->reset_active_trust_region_multipliers(model, this->direction, trial_iterate); - is_acceptable = this->is_iterate_acceptable(statistics, current_iterate, trial_iterate, this->direction, warmstart_information); + is_acceptable = this->is_iterate_acceptable(statistics, current_iterate, trial_iterate, this->direction, warmstart_information, + user_callbacks); if (is_acceptable) { this->constraint_relaxation_strategy.set_dual_residuals_statistics(statistics, trial_iterate); this->reset_radius(); @@ -122,9 +123,9 @@ namespace uno { // the trial iterate is accepted by the constraint relaxation strategy or if the step is small and we cannot switch to solving the feasibility problem bool TrustRegionStrategy::is_iterate_acceptable(Statistics& statistics, Iterate& current_iterate, Iterate& trial_iterate, - const Direction& direction, WarmstartInformation& warmstart_information) { + const Direction& direction, WarmstartInformation& warmstart_information, UserCallbacks& user_callbacks) { bool accept_iterate = this->constraint_relaxation_strategy.is_iterate_acceptable(statistics, current_iterate, trial_iterate, direction, 1., - warmstart_information); + warmstart_information, user_callbacks); this->set_statistics(statistics, trial_iterate, direction); if (accept_iterate) { trial_iterate.status = this->constraint_relaxation_strategy.check_termination(trial_iterate); diff --git a/uno/ingredients/globalization_mechanisms/TrustRegionStrategy.hpp b/uno/ingredients/globalization_mechanisms/TrustRegionStrategy.hpp index 378408be..a5845ef6 100644 --- a/uno/ingredients/globalization_mechanisms/TrustRegionStrategy.hpp +++ b/uno/ingredients/globalization_mechanisms/TrustRegionStrategy.hpp @@ -13,7 +13,7 @@ namespace uno { void initialize(Statistics& statistics, Iterate& initial_iterate, const Options& options) override; void compute_next_iterate(Statistics& statistics, const Model& model, Iterate& current_iterate, Iterate& trial_iterate, - WarmstartInformation& warmstart_information) override; + WarmstartInformation& warmstart_information, UserCallbacks& user_callbacks) override; private: double radius; /*!< Current trust region radius */ @@ -26,7 +26,7 @@ namespace uno { const double tolerance; bool is_iterate_acceptable(Statistics& statistics, Iterate& current_iterate, Iterate& trial_iterate, const Direction& direction, - WarmstartInformation& warmstart_information); + WarmstartInformation& warmstart_information, UserCallbacks& user_callbacks); void possibly_increase_radius(double step_norm); void decrease_radius(double step_norm); void decrease_radius(); diff --git a/uno/tools/UserCallbacks.hpp b/uno/tools/UserCallbacks.hpp new file mode 100644 index 00000000..d1285fec --- /dev/null +++ b/uno/tools/UserCallbacks.hpp @@ -0,0 +1,33 @@ +// Copyright (c) 2024 Charlie Vanaret +// Licensed under the MIT license. See LICENSE file in the project directory for details. + +#ifndef UNO_USERCALLBACKS_H +#define UNO_USERCALLBACKS_H + +namespace uno { + // forward declarations + struct Multipliers; + template + class Vector; + + class UserCallbacks { + public: + UserCallbacks() = default; + virtual ~UserCallbacks() = default; + + virtual void notify_acceptable_iterate(const Vector& primals, const Multipliers& multipliers, double objective_multiplier) = 0; + virtual void notify_new_primals(const Vector& primals) = 0; + virtual void notify_new_multipliers(const Multipliers& multipliers) = 0; + }; + + class NoUserCallbacks: public UserCallbacks { + public: + NoUserCallbacks(): UserCallbacks() { } + + void notify_acceptable_iterate(const Vector& /*primals*/, const Multipliers& /*multipliers*/, double /*objective_multiplier*/) override { } + void notify_new_primals(const Vector& /*primals*/) override { } + void notify_new_multipliers(const Multipliers& /*multipliers*/) override { } + }; +} // namespace + +#endif //UNO_USERCALLBACKS_H \ No newline at end of file