Skip to content

Commit

Permalink
Merge branch 'master' into hotfix/clippy
Browse files Browse the repository at this point in the history
  • Loading branch information
alphaville committed Oct 27, 2023
2 parents e6968eb + 8d25f34 commit 32fcc40
Show file tree
Hide file tree
Showing 8 changed files with 199 additions and 7 deletions.
2 changes: 1 addition & 1 deletion .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ jobs:
runs-on: ${{ matrix.os }}
strategy:
matrix:
os: [ubuntu-18.04, ubuntu-latest]
os: [ubuntu-20.04, ubuntu-latest]
env:
DO_DOCKER: 0
steps:
Expand Down
22 changes: 22 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,26 @@ and this project adheres to [Semantic Versioning](http://semver.org/).
Note: This is the main Changelog file for the Rust solver. The Changelog file for the Python interface (`opengen`) can be found in [/open-codegen/CHANGELOG.md](open-codegen/CHANGELOG.md)


<!-- ---------------------
v0.8.1
--------------------- -->
## [v0.8.1] - 2023-10-27

### Fixed

- Fix bug in implementation of `ball2.rs` (radius was being ignored for balls centered not at the origin)



<!-- ---------------------
v0.8.0
--------------------- -->
## [v0.8.1] - 2023-10-27

### Added

- New constraint: sphere of Euclidean norm

<!-- ---------------------
v0.7.7
--------------------- -->
Expand Down Expand Up @@ -247,6 +267,8 @@ This is a breaking API change.
--------------------- -->

<!-- Releases -->
[v0.8.1]: https://github.com/alphaville/optimization-engine/compare/v0.8.0...v0.8.1
[v0.8.0]: https://github.com/alphaville/optimization-engine/compare/v0.7.7...v0.8.0
[v0.7.7]: https://github.com/alphaville/optimization-engine/compare/v0.7.6...v0.7.7
[v0.7.6]: https://github.com/alphaville/optimization-engine/compare/v0.7.5...v0.7.6
[v0.7.5]: https://github.com/alphaville/optimization-engine/compare/v0.7.4...v0.7.5
Expand Down
2 changes: 1 addition & 1 deletion Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -42,7 +42,7 @@ homepage = "https://alphaville.github.io/optimization-engine/"
repository = "https://github.com/alphaville/optimization-engine"

# Version of this crate (SemVer)
version = "0.7.7"
version = "0.8.1"

edition = "2018"

Expand Down
2 changes: 1 addition & 1 deletion docs/openrust-alm.md
Original file line number Diff line number Diff line change
Expand Up @@ -150,7 +150,7 @@ pub fn df(u: &[f64], grad: &mut [f64]) -> Result<(), SolverError> {
Suppose we need to impose the constraint $F_1(u) \in C$, where $C=\\{z\in\mathbb{R}^2: \Vert{}z{}\Vert \leq 1\\}$
and $F_1$ is the mapping
$$
F_1(u)=\begin{bmatrix}2u_1 + u_1 + 0.5 \\\\ u_1+3u_2\end{bmatrix}
F_1(u)=\begin{bmatrix}2u_1 + u_3 + 0.5 \\\\ u_1+3u_2\end{bmatrix}
$$

```rust
Expand Down
2 changes: 1 addition & 1 deletion src/constraints/ball2.rs
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,7 @@ impl<'a> Constraint for Ball2<'a> {

if norm_difference > self.radius {
x.iter_mut().zip(center.iter()).for_each(|(x, c)| {
*x = *c + (*x - *c) / norm_difference;
*x = *c + self.radius * (*x - *c) / norm_difference;
});
}
} else {
Expand Down
2 changes: 2 additions & 0 deletions src/constraints/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ mod no_constraints;
mod rectangle;
mod simplex;
mod soc;
mod sphere2;
mod zero;

pub use ball1::Ball1;
Expand All @@ -32,6 +33,7 @@ pub use no_constraints::NoConstraints;
pub use rectangle::Rectangle;
pub use simplex::Simplex;
pub use soc::SecondOrderCone;
pub use sphere2::Sphere2;
pub use zero::Zero;

/// A set which can be used as a constraint
Expand Down
64 changes: 64 additions & 0 deletions src/constraints/sphere2.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,64 @@
use super::Constraint;

#[derive(Copy, Clone)]
/// A Euclidean sphere, that is, a set given by $S_2^r = \\{x \in \mathbb{R}^n {}:{} \Vert{}x{}\Vert = r\\}$
/// or a Euclidean sphere centered at a point $x_c$, that is, $S_2^{x_c, r} = \\{x \in \mathbb{R}^n {}:{} \Vert{}x-x_c{}\Vert = r\\}$
pub struct Sphere2<'a> {
center: Option<&'a [f64]>,
radius: f64,
}

impl<'a> Sphere2<'a> {
/// Construct a new Euclidean sphere with given center and radius
/// If no `center` is given, then it is assumed to be in the origin
pub fn new(center: Option<&'a [f64]>, radius: f64) -> Self {
assert!(radius > 0.0);
Sphere2 { center, radius }
}
}

impl<'a> Constraint for Sphere2<'a> {
/// Projection onto the sphere, $S_{r, c}$ with radius $r$ and center $c$.
/// If $x\neq c$, the projection is uniquely defined by
///
/// $$
/// P_{S_{r, c}}(x) = c + r\frac{x-c}{\Vert{}x-c\Vert_2},
/// $$
///
/// but for $x=c$, the projection is multi-valued. In particular, let
/// $y = P_{S_{r, c}}(c)$. Then $y_1 = c_1 + r$ and $y_i = c_i$ for
/// $i=2,\ldots, n$.
///
/// ## Arguments
///
/// - `x`: The given vector $x$ is updated with the projection on the set
///
fn project(&self, x: &mut [f64]) {
let epsilon = 1e-12;
if let Some(center) = &self.center {
let norm_difference = crate::matrix_operations::norm2_squared_diff(x, center).sqrt();
if norm_difference <= epsilon {
x.copy_from_slice(&center);

Check warning on line 41 in src/constraints/sphere2.rs

View workflow job for this annotation

GitHub Actions / clippy

this expression creates a reference which is immediately dereferenced by the compiler

warning: this expression creates a reference which is immediately dereferenced by the compiler --> src/constraints/sphere2.rs:41:35 | 41 | x.copy_from_slice(&center); | ^^^^^^^ help: change this to: `center` | = help: for further information visit https://rust-lang.github.io/rust-clippy/master/index.html#needless_borrow = note: `#[warn(clippy::needless_borrow)]` on by default
x[0] += self.radius;
return;
}
x.iter_mut().zip(center.iter()).for_each(|(x, c)| {
*x = *c + self.radius * (*x - *c) / norm_difference;
});
} else {
let norm_x = crate::matrix_operations::norm2(x);
if norm_x <= epsilon {
x[0] += self.radius;
return;
}
let norm_over_radius = self.radius / norm_x;
x.iter_mut().for_each(|x_| *x_ *= norm_over_radius);
}
}

/// Returns false (the sphere is not a convex set)
///
fn is_convex(&self) -> bool {
false
}
}
110 changes: 107 additions & 3 deletions src/constraints/tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -226,7 +226,7 @@ fn t_rectangle_only_xmax() {
}

#[test]
fn t_ball_at_origin() {
fn t_ball2_at_origin() {
let radius = 1.0;
let mut x = [1.0, 1.0];
let ball = Ball2::new(None, radius);
Expand All @@ -246,7 +246,47 @@ fn t_ball_at_origin() {
}

#[test]
fn t_ball_elsewhere() {
fn t_ball2_at_origin_different_radius_outside() {
let radius = 0.8;
let mut x = [1.0, 1.0];
let ball = Ball2::new(None, radius);
ball.project(&mut x);
let norm_proj_x = crate::matrix_operations::norm2(&x);
unit_test_utils::assert_nearly_equal(radius, norm_proj_x, 1e-10, 1e-12, "wrong norm");
}

#[test]
fn t_ball2_at_origin_different_radius_inside() {
let radius = 0.8;
let mut x = [-0.2, 0.15];
let ball = Ball2::new(None, radius);
ball.project(&mut x);
unit_test_utils::assert_nearly_equal_array(&x, &[-0.2, 0.15], 1e-10, 1e-12, "wrong");
}

#[test]
fn t_ball2_at_center_different_radius_outside() {
let radius = 1.2;
let mut x = [1.0, 1.0];
let center = [-0.8, -1.1];
let ball = Ball2::new(Some(&center), radius);
ball.project(&mut x);
let norm_x_minus_c = crate::matrix_operations::norm2_squared_diff(&x, &center).sqrt();
unit_test_utils::assert_nearly_equal(radius, norm_x_minus_c, 1e-10, 1e-12, "wrong norm");
}

#[test]
fn t_ball2_at_center_different_radius_inside() {
let radius = 1.2;
let mut x = [-0.9, -0.85];
let center = [-0.8, -1.1];
let ball = Ball2::new(Some(&center), radius);
ball.project(&mut x);
unit_test_utils::assert_nearly_equal_array(&[-0.9, -0.85], &x, 1e-10, 1e-12, "wrong result");
}

#[test]
fn t_ball2_elsewhere() {
let radius = 1.0;
let center = [1.0, 1.0];
let mut x = [2.0, 2.0];
Expand Down Expand Up @@ -465,7 +505,7 @@ fn t_cartesian_product_dimension() {
#[test]
fn t_cartesian_ball_no_constraint() {
let xc = [1., 0., 0.];
let radius = 0.5;
let radius = 1.0;
let ball2 = Ball2::new(Some(&xc), radius);
let no_constraints = NoConstraints::new();
let cartesian = CartesianProduct::new_with_capacity(4)
Expand Down Expand Up @@ -763,6 +803,70 @@ fn t_ball1_random_optimality_conditions_centered() {
}
}

#[test]
fn t_sphere2_no_center() {
let radius = 0.9;
let mut x_out = [1.0, 1.0];
let mut x_in = [-0.3, -0.2];
let unit_sphere = Sphere2::new(None, radius);
unit_sphere.project(&mut x_out);
unit_sphere.project(&mut x_in);
let norm_out = crate::matrix_operations::norm2(&x_out);
let norm_in = crate::matrix_operations::norm2(&x_in);
unit_test_utils::assert_nearly_equal(radius, norm_out, 1e-10, 1e-12, "norm_out is not 1.0");
unit_test_utils::assert_nearly_equal(radius, norm_in, 1e-10, 1e-12, "norm_in is not 1.0");
}

#[test]
fn t_sphere2_no_center_projection_of_zero() {
let radius = 0.9;
let mut x = [0.0, 0.0];
let unit_sphere = Sphere2::new(None, radius);
unit_sphere.project(&mut x);
let norm_result = crate::matrix_operations::norm2(&x);
unit_test_utils::assert_nearly_equal(radius, norm_result, 1e-10, 1e-12, "norm_out is not 1.0");
}

#[test]
fn t_sphere2_center() {
let radius = 1.3;
let center = [-3.0, 5.0];
let mut x = [1.0, 1.0];
let unit_sphere = Sphere2::new(Some(&center), radius);

unit_sphere.project(&mut x);
let mut x_minus_c = [0.0; 2];
x.iter()
.zip(center.iter())
.zip(x_minus_c.iter_mut())
.for_each(|((a, b), c)| {
*c = a - b;
});

let norm_out = crate::matrix_operations::norm2(&x_minus_c);
unit_test_utils::assert_nearly_equal(radius, norm_out, 1e-10, 1e-12, "norm_out is not 1.0");
}

#[test]
fn t_sphere2_center_projection_of_center() {
let radius = 1.3;
let center = [-3.0, 5.0];
let mut x = [-3.0, 5.0];
let unit_sphere = Sphere2::new(Some(&center), radius);

unit_sphere.project(&mut x);
let mut x_minus_c = [0.0; 2];
x.iter()
.zip(center.iter())
.zip(x_minus_c.iter_mut())
.for_each(|((a, b), c)| {
*c = a - b;
});

let norm_out = crate::matrix_operations::norm2(&x_minus_c);
unit_test_utils::assert_nearly_equal(radius, norm_out, 1e-10, 1e-12, "norm_out is not 1.0");
}

#[test]
#[should_panic]
fn t_ball1_alpha_negative() {
Expand Down

0 comments on commit 32fcc40

Please sign in to comment.