Skip to main content

pounce_algorithm/conv_check/
opt_error.rs

1//! Optimal-error convergence check — port of
2//! `Algorithm/IpOptErrorConvCheck.{hpp,cpp}`.
3//!
4//! Tolerance state machine over `(nlp_err, iter_count)` plus
5//! per-component infeasibilities pulled directly from
6//! [`IpoptCalculatedQuantities`]. The scalar
7//! [`Self::check_convergence`] entry point only gates on
8//! `nlp_err <= tol` (matching upstream when the per-component
9//! tolerances are at their `+∞` sentinels); the state-aware
10//! [`Self::check_convergence_with_state`] adds the
11//! `dual_inf_tol` / `constr_viol_tol` / `compl_inf_tol` gates that
12//! mirror upstream `OptimalityErrorConvergenceCheck::CheckConvergence`.
13
14use crate::conv_check::r#trait::{ConvCheck, ConvergenceStatus};
15use crate::ipopt_cq::IpoptCqHandle;
16use crate::ipopt_data::IpoptDataHandle;
17use pounce_common::types::{Index, Number};
18
19pub struct OptErrorConvCheck {
20    pub tol: Number,
21    pub dual_inf_tol: Number,
22    pub constr_viol_tol: Number,
23    pub compl_inf_tol: Number,
24    pub acceptable_tol: Number,
25    pub acceptable_dual_inf_tol: Number,
26    pub acceptable_constr_viol_tol: Number,
27    pub acceptable_compl_inf_tol: Number,
28    pub acceptable_obj_change_tol: Number,
29    pub acceptable_iter: Index,
30    pub max_iter: Index,
31    pub max_cpu_time: Number,
32    pub max_wall_time: Number,
33    pub acceptable_count: Index,
34    /// Objective value at the last iterate the main loop stashed via
35    /// `set_curr_acceptable_obj`. Used by the
36    /// `acceptable_obj_change_tol` cross-check. `None` until an
37    /// acceptable point has been recorded.
38    pub last_acceptable_obj: Option<Number>,
39    /// Tolerance on the scaled infeasibility stationarity
40    /// `‖Jᵀc‖/max(1,‖c‖)`. An iterate counts toward the infeasibility
41    /// streak when this ratio is at or below this value while the
42    /// constraint violation stays bounded away from zero. Rapid
43    /// infeasibility detection is disabled when this is non-positive.
44    pub infeas_stationarity_tol: Number,
45    /// Multiple of `constr_viol_tol` the constraint violation must
46    /// exceed before an iterate can count as infeasible-stationary —
47    /// keeps detection from firing on nearly-feasible flat spots.
48    pub infeas_viol_kappa: Number,
49    /// Consecutive infeasible-stationary iterations required before
50    /// terminating with `LocallyInfeasible`. Non-positive disables
51    /// rapid infeasibility detection.
52    pub infeas_max_streak: Index,
53    /// Running count of consecutive infeasible-stationary iterations.
54    pub infeas_streak: Index,
55}
56
57impl Default for OptErrorConvCheck {
58    fn default() -> Self {
59        // Defaults from `IpOptErrorConvCheck.cpp:RegisterOptions`.
60        Self {
61            tol: 1e-8,
62            dual_inf_tol: 1.0,
63            constr_viol_tol: 1e-4,
64            compl_inf_tol: 1e-4,
65            acceptable_tol: 1e-6,
66            acceptable_dual_inf_tol: 1e10,
67            acceptable_constr_viol_tol: 1e-2,
68            acceptable_compl_inf_tol: 1e-2,
69            acceptable_obj_change_tol: 1e20,
70            acceptable_iter: 15,
71            max_iter: 3000,
72            max_cpu_time: 1e6,
73            max_wall_time: 1e6,
74            acceptable_count: 0,
75            last_acceptable_obj: None,
76            infeas_stationarity_tol: 1e-8,
77            infeas_viol_kappa: 1e2,
78            infeas_max_streak: 5,
79            infeas_streak: 0,
80        }
81    }
82}
83
84impl OptErrorConvCheck {
85    pub fn new() -> Self {
86        Self::default()
87    }
88
89    /// Pure helper for the per-component upstream gate. Returns `true`
90    /// iff every supplied residual sits at or below its tolerance.
91    /// Factored out so tests can exercise the gating logic without
92    /// constructing a full `IpoptCq`.
93    fn passes_component_tols(
94        &self,
95        overall: Number,
96        dual_inf: Number,
97        constr_viol: Number,
98        compl_inf: Number,
99    ) -> bool {
100        overall <= self.tol
101            && dual_inf <= self.dual_inf_tol
102            && constr_viol <= self.constr_viol_tol
103            && compl_inf <= self.compl_inf_tol
104    }
105
106    /// Pure helper mirroring upstream
107    /// `OptimalityErrorConvergenceCheck::CurrentIsAcceptable`. Tests
108    /// the per-component `acceptable_*_tol` triplet plus the optional
109    /// `acceptable_obj_change_tol` stability cross-check.
110    fn passes_acceptable_tols(
111        &self,
112        overall: Number,
113        dual_inf: Number,
114        constr_viol: Number,
115        compl_inf: Number,
116        curr_f: Number,
117    ) -> bool {
118        // A point is never acceptable if the scaled error metric or the
119        // objective itself is non-finite. Without the `curr_f` guard a NaN/Inf
120        // objective with otherwise-small infeasibility (e.g. CUTE `himmelbj`,
121        // where f evaluates to NaN at a near-feasible point) would be recorded
122        // as the acceptable rollback point and reported under
123        // `Solved_To_Acceptable_Level` with a `nan` objective.
124        if !overall.is_finite() || !curr_f.is_finite() {
125            return false;
126        }
127        let component_ok = overall <= self.acceptable_tol
128            && dual_inf <= self.acceptable_dual_inf_tol
129            && constr_viol <= self.acceptable_constr_viol_tol
130            && compl_inf <= self.acceptable_compl_inf_tol;
131        if !component_ok {
132            return false;
133        }
134        // Upstream `IpOptErrorConvCheck.cpp:CurrentIsAcceptable` — when
135        // an acceptable point has already been recorded and the user
136        // tightened `acceptable_obj_change_tol` below the 1e20
137        // sentinel, the iterate is only re-acceptable if `f` has moved
138        // by less than `tol * max(1, |f|)` relative to the recorded
139        // value. Skipped when no prior point exists or the cross-check
140        // is disabled.
141        if self.acceptable_obj_change_tol < 1e20 {
142            if let Some(prev) = self.last_acceptable_obj {
143                let denom = curr_f.abs().max(1.0);
144                if (prev - curr_f).abs() >= self.acceptable_obj_change_tol * denom {
145                    return false;
146                }
147            }
148        }
149        true
150    }
151
152    /// Pure predicate for a single infeasible-stationary iterate: the
153    /// constraint violation is bounded away from zero
154    /// (`constr_viol > infeas_viol_kappa · constr_viol_tol`) and the
155    /// scaled infeasibility gradient `‖Jᵀc‖/max(1,‖c‖)` is at or below
156    /// `infeas_stationarity_tol`. Returns `false` when rapid
157    /// infeasibility detection is disabled (either knob non-positive).
158    fn is_infeasible_stationary(&self, constr_viol: Number, stationarity: Number) -> bool {
159        if self.infeas_stationarity_tol <= 0.0 || self.infeas_max_streak <= 0 {
160            return false;
161        }
162        constr_viol > self.infeas_viol_kappa * self.constr_viol_tol
163            && stationarity <= self.infeas_stationarity_tol
164    }
165
166    /// Advance the rapid-infeasibility-detection streak by one
167    /// iteration. An infeasible-stationary iterate (see
168    /// [`Self::is_infeasible_stationary`]) increments the streak; any
169    /// other iterate resets it to zero. Returns `true` once the streak
170    /// reaches `infeas_max_streak`, signalling the caller to terminate
171    /// with `ConvergenceStatus::LocallyInfeasible`. The streak guards
172    /// against firing on a transient flat spot.
173    fn note_infeasible_stationary(&mut self, constr_viol: Number, stationarity: Number) -> bool {
174        if self.is_infeasible_stationary(constr_viol, stationarity) {
175            self.infeas_streak += 1;
176            self.infeas_streak >= self.infeas_max_streak
177        } else {
178            self.infeas_streak = 0;
179            false
180        }
181    }
182}
183
184impl ConvCheck for OptErrorConvCheck {
185    fn check_convergence(&mut self, nlp_err: Number, iter_count: Index) -> ConvergenceStatus {
186        if nlp_err <= self.tol {
187            return ConvergenceStatus::Converged;
188        }
189        if nlp_err <= self.acceptable_tol {
190            self.acceptable_count += 1;
191            if self.acceptable_count >= self.acceptable_iter {
192                return ConvergenceStatus::ConvergedToAcceptable;
193            }
194        } else {
195            self.acceptable_count = 0;
196        }
197        if iter_count >= self.max_iter {
198            return ConvergenceStatus::MaxIterExceeded;
199        }
200        ConvergenceStatus::Continue
201    }
202
203    fn check_convergence_with_state(
204        &mut self,
205        nlp_err: Number,
206        iter_count: Index,
207        data: &IpoptDataHandle,
208        cq: &IpoptCqHandle,
209    ) -> ConvergenceStatus {
210        // Mirror upstream `IpOptErrorConvCheck.cpp::CheckConvergence`:
211        // the scaled scalar `nlp_err` must drop below `tol` AND each
212        // per-component value must sit under its own tolerance. Known
213        // deviation (M1, documented in the code-review notes): upstream
214        // gates the components on the *unscaled* residuals; the CQ layer
215        // exposes no unscaled per-component accessors yet, so these are
216        // the internally *scaled* values. The unscaled expansion is
217        // deferred until the scaling objects are threaded through.
218        let cq_ref = cq.borrow();
219        let dual_inf = cq_ref.curr_dual_infeasibility_max();
220        let constr_viol = cq_ref.curr_primal_infeasibility_max();
221        let compl_inf = cq_ref.curr_complementarity_max();
222        let curr_f = cq_ref.curr_f();
223        drop(cq_ref);
224
225        if self.passes_component_tols(nlp_err, dual_inf, constr_viol, compl_inf) {
226            return ConvergenceStatus::Converged;
227        }
228        if self.passes_acceptable_tols(nlp_err, dual_inf, constr_viol, compl_inf, curr_f) {
229            self.acceptable_count += 1;
230            if self.acceptable_count >= self.acceptable_iter {
231                return ConvergenceStatus::ConvergedToAcceptable;
232            }
233        } else {
234            self.acceptable_count = 0;
235        }
236        if iter_count >= self.max_iter {
237            return ConvergenceStatus::MaxIterExceeded;
238        }
239        // Rapid infeasibility detection — recognise an iterate
240        // converging to a stationary point of the constraint
241        // violation with the violation bounded away from zero, and
242        // exit with `LocallyInfeasible` instead of grinding to
243        // `max_iter` or thrashing restoration. Gated behind an
244        // `infeas_max_streak`-iteration streak to avoid firing on a
245        // transient flat spot. The outer guard skips the two
246        // transpose-products when detection is disabled.
247        if self.infeas_stationarity_tol > 0.0 && self.infeas_max_streak > 0 {
248            let stationarity = cq.borrow().curr_infeasibility_stationarity();
249            if self.note_infeasible_stationary(constr_viol, stationarity) {
250                return ConvergenceStatus::LocallyInfeasible;
251            }
252        }
253        // Time-budget gates. Upstream
254        // `IpOptErrorConvCheck.cpp::CheckConvergence` reads the
255        // application-level start time; pounce piggybacks on
256        // `data.timing.overall_alg`, which `IpoptApplication` starts
257        // at the top of `optimize_constrained`. `live_*` returns the
258        // running elapsed without forcing a `start/end` cycle.
259        let timing = &data.borrow().timing;
260        if timing.overall_alg.live_cpu_time() >= self.max_cpu_time {
261            return ConvergenceStatus::CpuTimeExceeded;
262        }
263        if timing.overall_alg.live_wallclock_time() >= self.max_wall_time {
264            return ConvergenceStatus::WallTimeExceeded;
265        }
266        ConvergenceStatus::Continue
267    }
268
269    fn tol_or_default(&self) -> Number {
270        self.tol
271    }
272
273    fn set_tolerance(&mut self, name: &str, value: Number) -> bool {
274        match name {
275            "tol" => self.tol = value,
276            "dual_inf_tol" => self.dual_inf_tol = value,
277            "constr_viol_tol" => self.constr_viol_tol = value,
278            "compl_inf_tol" => self.compl_inf_tol = value,
279            "acceptable_tol" => self.acceptable_tol = value,
280            "acceptable_dual_inf_tol" => self.acceptable_dual_inf_tol = value,
281            "acceptable_constr_viol_tol" => self.acceptable_constr_viol_tol = value,
282            "acceptable_compl_inf_tol" => self.acceptable_compl_inf_tol = value,
283            "acceptable_obj_change_tol" => self.acceptable_obj_change_tol = value,
284            _ => return false,
285        }
286        true
287    }
288
289    fn current_is_acceptable(&self, nlp_err: Number) -> bool {
290        // Scalar fallback used when the caller has no `IpoptCq` handle
291        // (e.g. unit tests). The state-aware variant
292        // [`Self::current_is_acceptable_with_state`] mirrors upstream
293        // more faithfully by gating on the per-component
294        // `acceptable_*_tol` triplet plus the obj-change cross-check.
295        nlp_err.is_finite() && nlp_err <= self.acceptable_tol
296    }
297
298    fn current_is_acceptable_with_state(
299        &self,
300        nlp_err: Number,
301        _data: &IpoptDataHandle,
302        cq: &IpoptCqHandle,
303    ) -> bool {
304        let cq_ref = cq.borrow();
305        let dual_inf = cq_ref.curr_dual_infeasibility_max();
306        let constr_viol = cq_ref.curr_primal_infeasibility_max();
307        let compl_inf = cq_ref.curr_complementarity_max();
308        let curr_f = cq_ref.curr_f();
309        drop(cq_ref);
310        self.passes_acceptable_tols(nlp_err, dual_inf, constr_viol, compl_inf, curr_f)
311    }
312
313    fn set_curr_acceptable_obj(&mut self, obj: Number) {
314        self.last_acceptable_obj = Some(obj);
315    }
316}
317
318#[cfg(test)]
319mod tests {
320    use super::*;
321
322    #[test]
323    fn converges_at_tol() {
324        let mut c = OptErrorConvCheck::new();
325        assert_eq!(c.check_convergence(1e-9, 0), ConvergenceStatus::Converged);
326    }
327
328    #[test]
329    fn acceptable_iter_count_threshold() {
330        let mut c = OptErrorConvCheck {
331            acceptable_iter: 3,
332            ..Default::default()
333        };
334        // nlp_err between tol (1e-8) and acceptable (1e-6).
335        assert_eq!(c.check_convergence(1e-7, 0), ConvergenceStatus::Continue);
336        assert_eq!(c.check_convergence(1e-7, 1), ConvergenceStatus::Continue);
337        assert_eq!(
338            c.check_convergence(1e-7, 2),
339            ConvergenceStatus::ConvergedToAcceptable
340        );
341    }
342
343    #[test]
344    fn streak_resets_when_above_acceptable() {
345        let mut c = OptErrorConvCheck {
346            acceptable_iter: 3,
347            ..Default::default()
348        };
349        assert_eq!(c.check_convergence(1e-7, 0), ConvergenceStatus::Continue);
350        // Above acceptable resets the counter.
351        assert_eq!(c.check_convergence(1e-3, 1), ConvergenceStatus::Continue);
352        assert_eq!(c.check_convergence(1e-7, 2), ConvergenceStatus::Continue);
353        assert_eq!(c.check_convergence(1e-7, 3), ConvergenceStatus::Continue);
354        assert_eq!(
355            c.check_convergence(1e-7, 4),
356            ConvergenceStatus::ConvergedToAcceptable
357        );
358    }
359
360    #[test]
361    fn passes_acceptable_tols_gates_on_per_component_triplet() {
362        let c = OptErrorConvCheck {
363            acceptable_tol: 1e-6,
364            acceptable_dual_inf_tol: 1e-3,
365            acceptable_constr_viol_tol: 1e-3,
366            acceptable_compl_inf_tol: 1e-3,
367            ..Default::default()
368        };
369        assert!(c.passes_acceptable_tols(1e-7, 1e-4, 1e-4, 1e-4, 0.0));
370        // dual_inf above its acceptable threshold blocks.
371        assert!(!c.passes_acceptable_tols(1e-7, 1.0, 1e-4, 1e-4, 0.0));
372        // overall above acceptable_tol blocks.
373        assert!(!c.passes_acceptable_tols(1e-5, 1e-4, 1e-4, 1e-4, 0.0));
374    }
375
376    #[test]
377    fn passes_acceptable_tols_honors_obj_change_tol() {
378        let mut c = OptErrorConvCheck {
379            acceptable_tol: 1e-6,
380            acceptable_dual_inf_tol: 1.0,
381            acceptable_constr_viol_tol: 1.0,
382            acceptable_compl_inf_tol: 1.0,
383            acceptable_obj_change_tol: 0.1,
384            ..Default::default()
385        };
386        // First call always acceptable (no prior obj).
387        assert!(c.passes_acceptable_tols(1e-7, 0.0, 0.0, 0.0, 10.0));
388        c.set_curr_acceptable_obj(10.0);
389        // Same f → change well under threshold → still acceptable.
390        assert!(c.passes_acceptable_tols(1e-7, 0.0, 0.0, 0.0, 10.0));
391        // f moved by 2.0 with threshold 0.1 * max(1, |11.0|) = 1.1 →
392        // absolute change 1.0 < 1.1: acceptable.
393        assert!(c.passes_acceptable_tols(1e-7, 0.0, 0.0, 0.0, 11.0));
394        // f moved by 5.0 — absolute change 5.0 > 1.5 = 0.1 * 15 →
395        // rejected (the stability cross-check fires).
396        assert!(!c.passes_acceptable_tols(1e-7, 0.0, 0.0, 0.0, 15.0));
397    }
398
399    use crate::conv_check::r#trait::ConvCheck;
400
401    #[test]
402    fn set_curr_acceptable_obj_records_for_cross_check() {
403        let mut c = OptErrorConvCheck::new();
404        assert!(c.last_acceptable_obj.is_none());
405        ConvCheck::set_curr_acceptable_obj(&mut c, 4.2);
406        assert_eq!(c.last_acceptable_obj, Some(4.2));
407    }
408
409    #[test]
410    fn passes_component_tols_requires_all_under_threshold() {
411        let c = OptErrorConvCheck {
412            tol: 1e-8,
413            dual_inf_tol: 1.0,
414            constr_viol_tol: 1e-4,
415            compl_inf_tol: 1e-4,
416            ..Default::default()
417        };
418        // All under threshold → converged.
419        assert!(c.passes_component_tols(1e-9, 0.5, 1e-5, 1e-5));
420        // dual_inf above its tolerance blocks even when nlp_err is tiny.
421        assert!(!c.passes_component_tols(1e-12, 2.0, 1e-5, 1e-5));
422        // compl_inf above its tolerance blocks.
423        assert!(!c.passes_component_tols(1e-12, 0.0, 0.0, 1e-2));
424        // constr_viol above its tolerance blocks.
425        assert!(!c.passes_component_tols(1e-12, 0.0, 1e-2, 0.0));
426    }
427
428    #[test]
429    fn infeasible_stationary_requires_violation_and_flat_gradient() {
430        let c = OptErrorConvCheck {
431            constr_viol_tol: 1e-4,
432            infeas_viol_kappa: 1e2, // violation threshold = 1e-2
433            infeas_stationarity_tol: 1e-8,
434            infeas_max_streak: 5,
435            ..Default::default()
436        };
437        // Violation well above 1e-2 and the infeasibility gradient
438        // essentially zero → counts as infeasible-stationary.
439        assert!(c.is_infeasible_stationary(1e-1, 1e-9));
440        // Violation above threshold but the gradient is not flat →
441        // still making feasibility progress, does not count.
442        assert!(!c.is_infeasible_stationary(1e-1, 1e-3));
443        // Gradient flat but violation below threshold → nearly
444        // feasible, does not count.
445        assert!(!c.is_infeasible_stationary(1e-3, 1e-9));
446    }
447
448    #[test]
449    fn infeasible_stationary_disabled_by_nonpositive_knobs() {
450        let off_tol = OptErrorConvCheck {
451            infeas_stationarity_tol: 0.0,
452            infeas_max_streak: 5,
453            ..Default::default()
454        };
455        assert!(!off_tol.is_infeasible_stationary(1e9, 0.0));
456        let off_streak = OptErrorConvCheck {
457            infeas_stationarity_tol: 1e-8,
458            infeas_max_streak: 0,
459            ..Default::default()
460        };
461        assert!(!off_streak.is_infeasible_stationary(1e9, 0.0));
462    }
463
464    #[test]
465    fn infeasible_stationary_streak_fires_only_after_max_streak() {
466        let mut c = OptErrorConvCheck {
467            constr_viol_tol: 1e-4,
468            infeas_viol_kappa: 1e2, // violation threshold = 1e-2
469            infeas_stationarity_tol: 1e-8,
470            infeas_max_streak: 3,
471            ..Default::default()
472        };
473        // Infeasible-stationary iterate: violation 1e-1 > 1e-2, flat
474        // gradient. Streak accrues but does not fire until the third.
475        assert!(!c.note_infeasible_stationary(1e-1, 1e-9));
476        assert!(!c.note_infeasible_stationary(1e-1, 1e-9));
477        assert!(c.note_infeasible_stationary(1e-1, 1e-9));
478    }
479
480    #[test]
481    fn infeasible_stationary_streak_resets_on_feasibility_progress() {
482        let mut c = OptErrorConvCheck {
483            constr_viol_tol: 1e-4,
484            infeas_viol_kappa: 1e2,
485            infeas_stationarity_tol: 1e-8,
486            infeas_max_streak: 3,
487            ..Default::default()
488        };
489        assert!(!c.note_infeasible_stationary(1e-1, 1e-9));
490        assert!(!c.note_infeasible_stationary(1e-1, 1e-9));
491        // A non-stationary iterate (gradient not flat) resets the streak.
492        assert!(!c.note_infeasible_stationary(1e-1, 1e-3));
493        assert_eq!(c.infeas_streak, 0);
494        // The streak must rebuild from scratch — no carry-over credit.
495        assert!(!c.note_infeasible_stationary(1e-1, 1e-9));
496        assert!(!c.note_infeasible_stationary(1e-1, 1e-9));
497        assert!(c.note_infeasible_stationary(1e-1, 1e-9));
498    }
499
500    #[test]
501    fn infeasible_stationary_streak_never_fires_when_disabled() {
502        let mut c = OptErrorConvCheck {
503            infeas_stationarity_tol: 0.0,
504            infeas_max_streak: 5,
505            ..Default::default()
506        };
507        for _ in 0..20 {
508            assert!(!c.note_infeasible_stationary(1e9, 0.0));
509        }
510        assert_eq!(c.infeas_streak, 0);
511    }
512
513    #[test]
514    fn max_iter_exceeded() {
515        let mut c = OptErrorConvCheck {
516            max_iter: 5,
517            ..Default::default()
518        };
519        assert_eq!(
520            c.check_convergence(1.0, 5),
521            ConvergenceStatus::MaxIterExceeded
522        );
523    }
524}