From 8902d94fd7ee4481d9804f24b426b91a1b4d4919 Mon Sep 17 00:00:00 2001 From: A-Walrus Date: Thu, 16 Mar 2023 08:49:35 +0200 Subject: [PATCH 1/7] Update flake.lock Nix flake was out-of-date since rust-toolchain was raised to 1.68, but flake.lock was older than that. Ran `nix flake update`. In the future when raising rust-toolchain version we need to make sure that we update the flake as well --- nix/flake.lock | 24 ++++++++++++------------ 1 file changed, 12 insertions(+), 12 deletions(-) diff --git a/nix/flake.lock b/nix/flake.lock index 0f39d58323..7bbf7adbec 100644 --- a/nix/flake.lock +++ b/nix/flake.lock @@ -16,11 +16,11 @@ ] }, "locked": { - "lastModified": 1675295673, - "narHash": "sha256-+u4cR9rtwOHpUSQG+2IgnEBzryCnC/Fl7KIkMNY1Xa8=", + "lastModified": 1678152261, + "narHash": "sha256-cPRDxwygVMleiSEGELrvAiq9vYAN4c3KK/K4UEO13vU=", "owner": "ipetkov", "repo": "crane", - "rev": "013614dd509e6a4fd127f096192c553ef589f8db", + "rev": "5291dd0aa7a52d607fc952763ef60714e4c881d4", "type": "github" }, "original": { @@ -47,11 +47,11 @@ }, "flake-utils": { "locked": { - "lastModified": 1667395993, - "narHash": "sha256-nuEHfE/LcWyuSWnS8t12N1wc105Qtau+/OdUAjtQ0rA=", + "lastModified": 1678901627, + "narHash": "sha256-U02riOqrKKzwjsxc/400XnElV+UtPUQWpANPlyazjH0=", "owner": "numtide", "repo": "flake-utils", - "rev": "5aed5285a952e0b949eb3ba02c12fa4fcfef535f", + "rev": "93a2b84fc4b70d9e089d029deacc3583435c2ed6", "type": "github" }, "original": { @@ -62,11 +62,11 @@ }, "nixpkgs": { "locked": { - "lastModified": 1675386474, - "narHash": "sha256-wRetHHKPNd//fMVkNPxKZ+ggKGzN8TTgN5x/nVxe81k=", + "lastModified": 1678949004, + "narHash": "sha256-8c0hnhTjmLRxtYp7wLr2nN4sRa61quL9PBSaGuYaBL0=", "owner": "NixOS", "repo": "nixpkgs", - "rev": "3cf2d77f64b9eeafd2f5ea302594b04621aac6f2", + "rev": "67712ddbdcfa2790dc0d58c56b83eb980ab43213", "type": "github" }, "original": { @@ -94,11 +94,11 @@ ] }, "locked": { - "lastModified": 1675304959, - "narHash": "sha256-SQNtt71uT++7F3+kHOZ9PEThswk0MnEt6zO2xJdUo/o=", + "lastModified": 1678933473, + "narHash": "sha256-UY19R278O9bwneLWC7ady8VMoQ+UlAWy8SkUsfDZvQs=", "owner": "oxalica", "repo": "rust-overlay", - "rev": "61ec735acfb1f549237bc525da355bd0b9cc70a3", + "rev": "5c1af9b9d618e02a87cdd30a3022aec0b78cd9aa", "type": "github" }, "original": { From 334652474ce09ee52005447ab89fa51d5a6e8486 Mon Sep 17 00:00:00 2001 From: A-Walrus Date: Wed, 15 Mar 2023 21:36:42 +0200 Subject: [PATCH 2/7] Basic cycle validation --- crates/fj-kernel/src/objects/full/edge.rs | 10 ++ crates/fj-kernel/src/validate/cycle.rs | 121 +++++++++++++++++++++- crates/fj-kernel/src/validate/mod.rs | 5 + 3 files changed, 134 insertions(+), 2 deletions(-) diff --git a/crates/fj-kernel/src/objects/full/edge.rs b/crates/fj-kernel/src/objects/full/edge.rs index 65216649bc..646ea688bf 100644 --- a/crates/fj-kernel/src/objects/full/edge.rs +++ b/crates/fj-kernel/src/objects/full/edge.rs @@ -86,6 +86,16 @@ impl HalfEdge { self.curve.point_from_path_coords(start) } + /// Compute the surface position where the half-edge ends + pub fn end_position(&self) -> Point<2> { + // Computing the surface position from the curve position is fine. + // `HalfEdge` "owns" its end position. There is no competing code that + // could compute the surface position from slightly different data. + + let [_, end] = self.boundary; + self.curve.point_from_path_coords(end) + } + /// Access the vertex from where this half-edge starts pub fn start_vertex(&self) -> &Handle { &self.start_vertex diff --git a/crates/fj-kernel/src/validate/cycle.rs b/crates/fj-kernel/src/validate/cycle.rs index 16cc826f50..c3e8c43967 100644 --- a/crates/fj-kernel/src/validate/cycle.rs +++ b/crates/fj-kernel/src/validate/cycle.rs @@ -1,12 +1,129 @@ use crate::objects::Cycle; +use crate::objects::HalfEdge; +use fj_math::Point; +use fj_math::Scalar; +use itertools::Itertools; use super::{Validate, ValidationConfig, ValidationError}; impl Validate for Cycle { fn validate_with_config( &self, - _: &ValidationConfig, - _: &mut Vec, + config: &ValidationConfig, + errors: &mut Vec, ) { + CycleValidationError::check_half_edges_disconnected( + self, config, errors, + ) + } +} + +/// [`Cycle`] validation failed +#[derive(Clone, Debug, thiserror::Error)] +pub enum CycleValidationError { + /// [`Cycle`]'s half-edges are not connected + #[error( + "Adjacent `HalfEdge`s are distinct\n\ + - End position of first `HalfEdge`: {end_of_first:?}\n\ + - Start position of second `HalfEdge`: {start_of_second:?}\n\ + - `HalfEdge`s: {half_edges:#?}" + )] + HalfEdgesDisconnected { + /// The end position of the first [`HalfEdge`] + end_of_first: Point<2>, + + /// The start position of the second [`HalfEdge`] + start_of_second: Point<2>, + + /// The distance between the two vertices + distance: Scalar, + + /// The half-edge + half_edges: (HalfEdge, HalfEdge), + }, + #[error("Expected at least one `HalfEdge`\n")] + NotEnoughHalfEdges, +} + +impl CycleValidationError { + fn check_half_edges_disconnected( + cycle: &Cycle, + config: &ValidationConfig, + errors: &mut Vec, + ) { + for (first, second) in cycle + .half_edges() + .chain(std::iter::once(cycle.half_edges().next().unwrap())) + .tuple_windows() + { + let end_of_first = first.end_position(); + let start_of_second = second.start_position(); + + let distance = (end_of_first - start_of_second).magnitude(); + + if distance > config.identical_max_distance { + errors.push( + Self::HalfEdgesDisconnected { + end_of_first, + start_of_second, + distance, + half_edges: ( + first.clone_object(), + second.clone_object(), + ), + } + .into(), + ); + } + } + } +} + +#[cfg(test)] +mod tests { + use fj_math::Point; + + use crate::{ + builder::{CycleBuilder, HalfEdgeBuilder}, + objects::{Cycle, HalfEdge}, + services::Services, + validate::{ + cycle::CycleValidationError, HalfEdgeValidationError, Validate, + ValidationError, + }, + }; + + #[test] + fn half_edges_connected() -> anyhow::Result<()> { + let mut services = Services::new(); + + let valid = Cycle::new([]) + .update_as_polygon_from_points( + [[0.0, 0.0], [1.0, 0.0], [1.0, 1.0]], + &mut services.objects, + ) + .0; + + valid.validate_and_return_first_error()?; + + let first = HalfEdgeBuilder::line_segment([[0., 0.], [1., 0.]], None) + .build(&mut services.objects); + let second = HalfEdgeBuilder::line_segment([[0., 0.], [1., 0.]], None) + .build(&mut services.objects); + + let invalid = Cycle::new([]) + .add_half_edge(first, &mut services.objects) + .0 + .add_half_edge(second, &mut services.objects) + .0; + + assert!(matches!( + invalid.validate_and_return_first_error(), + Err(ValidationError::Cycle( + CycleValidationError::HalfEdgesDisconnected { .. } + )) + )); + + Ok(()) } } diff --git a/crates/fj-kernel/src/validate/mod.rs b/crates/fj-kernel/src/validate/mod.rs index f45bfbd5b9..5a0dcfc843 100644 --- a/crates/fj-kernel/src/validate/mod.rs +++ b/crates/fj-kernel/src/validate/mod.rs @@ -9,6 +9,7 @@ mod solid; mod surface; mod vertex; +use self::cycle::CycleValidationError; pub use self::{edge::HalfEdgeValidationError, face::FaceValidationError}; use std::convert::Infallible; @@ -87,6 +88,10 @@ pub enum ValidationError { /// `HalfEdge` validation error #[error("`HalfEdge` validation error:\n{0}")] HalfEdge(#[from] HalfEdgeValidationError), + + /// `Cycle` validation error + #[error("`Cycle` validation error:\n{0}")] + Cycle(#[from] CycleValidationError), } impl From for ValidationError { From 3e0efd9b6ab6d45040a49efdf15eafd2c277a8ca Mon Sep 17 00:00:00 2001 From: A-Walrus Date: Wed, 15 Mar 2023 21:40:46 +0200 Subject: [PATCH 3/7] Make clippy happy --- crates/fj-kernel/src/validate/cycle.rs | 14 +++++--------- 1 file changed, 5 insertions(+), 9 deletions(-) diff --git a/crates/fj-kernel/src/validate/cycle.rs b/crates/fj-kernel/src/validate/cycle.rs index c3e8c43967..a591063027 100644 --- a/crates/fj-kernel/src/validate/cycle.rs +++ b/crates/fj-kernel/src/validate/cycle.rs @@ -39,7 +39,7 @@ pub enum CycleValidationError { distance: Scalar, /// The half-edge - half_edges: (HalfEdge, HalfEdge), + half_edges: Box<(HalfEdge, HalfEdge)>, }, #[error("Expected at least one `HalfEdge`\n")] NotEnoughHalfEdges, @@ -67,10 +67,10 @@ impl CycleValidationError { end_of_first, start_of_second, distance, - half_edges: ( + half_edges: Box::new(( first.clone_object(), second.clone_object(), - ), + )), } .into(), ); @@ -81,16 +81,12 @@ impl CycleValidationError { #[cfg(test)] mod tests { - use fj_math::Point; use crate::{ builder::{CycleBuilder, HalfEdgeBuilder}, - objects::{Cycle, HalfEdge}, + objects::Cycle, services::Services, - validate::{ - cycle::CycleValidationError, HalfEdgeValidationError, Validate, - ValidationError, - }, + validate::{cycle::CycleValidationError, Validate, ValidationError}, }; #[test] From 7eb19cdc94163cc3d692d8bca6b3f8d741e6b747 Mon Sep 17 00:00:00 2001 From: A-Walrus Date: Wed, 15 Mar 2023 21:51:19 +0200 Subject: [PATCH 4/7] Cleanup unwrap --- crates/fj-kernel/src/validate/cycle.rs | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/crates/fj-kernel/src/validate/cycle.rs b/crates/fj-kernel/src/validate/cycle.rs index a591063027..60f206d26f 100644 --- a/crates/fj-kernel/src/validate/cycle.rs +++ b/crates/fj-kernel/src/validate/cycle.rs @@ -51,8 +51,15 @@ impl CycleValidationError { config: &ValidationConfig, errors: &mut Vec, ) { + // If there are less than two half edges + if cycle.half_edges().nth(1).is_none() { + errors.push(Self::NotEnoughHalfEdges.into()); + return; + } for (first, second) in cycle .half_edges() + // Chain the first half_edge so that we make sure that the last connects to the first. + // This unwrap will never fail because we checked before that there are enough half_edges. .chain(std::iter::once(cycle.half_edges().next().unwrap())) .tuple_windows() { From fc1f51aca6d7e4bbd24ff3a3191786f38ecff8c0 Mon Sep 17 00:00:00 2001 From: A-Walrus Date: Thu, 16 Mar 2023 08:42:00 +0200 Subject: [PATCH 5/7] Change minimum half_edge count to 1 Forgot about the fact that a single circle counts as a cycle --- crates/fj-kernel/src/validate/cycle.rs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/crates/fj-kernel/src/validate/cycle.rs b/crates/fj-kernel/src/validate/cycle.rs index 60f206d26f..a7259c9566 100644 --- a/crates/fj-kernel/src/validate/cycle.rs +++ b/crates/fj-kernel/src/validate/cycle.rs @@ -51,8 +51,8 @@ impl CycleValidationError { config: &ValidationConfig, errors: &mut Vec, ) { - // If there are less than two half edges - if cycle.half_edges().nth(1).is_none() { + // If there are no half edges + if cycle.half_edges().next().is_none() { errors.push(Self::NotEnoughHalfEdges.into()); return; } From 94948895af332996533ca00aa5891f1f8f1b95c1 Mon Sep 17 00:00:00 2001 From: A-Walrus Date: Thu, 16 Mar 2023 13:01:01 +0200 Subject: [PATCH 6/7] Make requested changes - Inlined the `end_position` function to not expose potentially error- prone behaviour. - Seperate empty check to new method. - Switch to `circular_tuple_windows` - Add test for empty cycle --- crates/fj-kernel/src/validate/cycle.rs | 62 ++++++++++++++++---------- 1 file changed, 39 insertions(+), 23 deletions(-) diff --git a/crates/fj-kernel/src/validate/cycle.rs b/crates/fj-kernel/src/validate/cycle.rs index a7259c9566..c5a8c37283 100644 --- a/crates/fj-kernel/src/validate/cycle.rs +++ b/crates/fj-kernel/src/validate/cycle.rs @@ -14,7 +14,8 @@ impl Validate for Cycle { ) { CycleValidationError::check_half_edges_disconnected( self, config, errors, - ) + ); + CycleValidationError::check_enough_half_edges(self, config, errors); } } @@ -46,9 +47,9 @@ pub enum CycleValidationError { } impl CycleValidationError { - fn check_half_edges_disconnected( + fn check_enough_half_edges( cycle: &Cycle, - config: &ValidationConfig, + _config: &ValidationConfig, errors: &mut Vec, ) { // If there are no half edges @@ -56,14 +57,18 @@ impl CycleValidationError { errors.push(Self::NotEnoughHalfEdges.into()); return; } - for (first, second) in cycle - .half_edges() - // Chain the first half_edge so that we make sure that the last connects to the first. - // This unwrap will never fail because we checked before that there are enough half_edges. - .chain(std::iter::once(cycle.half_edges().next().unwrap())) - .tuple_windows() - { - let end_of_first = first.end_position(); + } + + fn check_half_edges_disconnected( + cycle: &Cycle, + config: &ValidationConfig, + errors: &mut Vec, + ) { + for (first, second) in cycle.half_edges().circular_tuple_windows() { + let end_of_first = { + let [_, end] = first.boundary(); + first.curve().point_from_path_coords(end) + }; let start_of_second = second.start_position(); let distance = (end_of_first - start_of_second).magnitude(); @@ -109,24 +114,35 @@ mod tests { valid.validate_and_return_first_error()?; - let first = HalfEdgeBuilder::line_segment([[0., 0.], [1., 0.]], None) - .build(&mut services.objects); - let second = HalfEdgeBuilder::line_segment([[0., 0.], [1., 0.]], None) - .build(&mut services.objects); - - let invalid = Cycle::new([]) - .add_half_edge(first, &mut services.objects) - .0 - .add_half_edge(second, &mut services.objects) - .0; - + let disconnected = { + let first = + HalfEdgeBuilder::line_segment([[0., 0.], [1., 0.]], None) + .build(&mut services.objects); + let second = + HalfEdgeBuilder::line_segment([[0., 0.], [1., 0.]], None) + .build(&mut services.objects); + + Cycle::new([]) + .add_half_edge(first, &mut services.objects) + .0 + .add_half_edge(second, &mut services.objects) + .0 + }; assert!(matches!( - invalid.validate_and_return_first_error(), + disconnected.validate_and_return_first_error(), Err(ValidationError::Cycle( CycleValidationError::HalfEdgesDisconnected { .. } )) )); + let empty = Cycle::new([]); + assert!(matches!( + empty.validate_and_return_first_error(), + Err(ValidationError::Cycle( + CycleValidationError::NotEnoughHalfEdges + )) + )); + Ok(()) } } From deade38df14740313bf7801f2fc6387f0daa58bd Mon Sep 17 00:00:00 2001 From: A-Walrus Date: Thu, 16 Mar 2023 13:08:54 +0200 Subject: [PATCH 7/7] Make clippy happy --- crates/fj-kernel/src/validate/cycle.rs | 1 - 1 file changed, 1 deletion(-) diff --git a/crates/fj-kernel/src/validate/cycle.rs b/crates/fj-kernel/src/validate/cycle.rs index c5a8c37283..0c0442038b 100644 --- a/crates/fj-kernel/src/validate/cycle.rs +++ b/crates/fj-kernel/src/validate/cycle.rs @@ -55,7 +55,6 @@ impl CycleValidationError { // If there are no half edges if cycle.half_edges().next().is_none() { errors.push(Self::NotEnoughHalfEdges.into()); - return; } }