-
Notifications
You must be signed in to change notification settings - Fork 4
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
feat!: Boundary order checking #164
Conversation
Codecov ReportAttention: Patch coverage is
Additional details and impacted files@@ Coverage Diff @@
## main #164 +/- ##
==========================================
+ Coverage 82.29% 83.23% +0.93%
==========================================
Files 23 24 +1
Lines 5830 6091 +261
Branches 5830 6091 +261
==========================================
+ Hits 4798 5070 +272
+ Misses 956 945 -11
Partials 76 76 ☔ View full report in Codecov by Sentry. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Very cool, thanks Agustin, mostly minor comments. The only part that confuses me is what arguments exactly fn port_ordering
expects. See my comment below.
src/algorithms/boundary.rs
Outdated
/// Computes a partial order between the ports of a boundary. | ||
/// | ||
/// Returns a map indicating for each input port in the boundary, the set of | ||
/// output ports that can be reached from it in the graph. | ||
/// | ||
/// The `graph` parameter must be a valid graph for this boundary. When | ||
/// multiple nodes not reachable from the boundary are present, consider | ||
/// using a [`crate::view::Subgraph`] to restrict the traversal. | ||
/// | ||
/// # Complexity | ||
/// | ||
/// The complexity of this operation is `O(e log(n) + k*n)`, where `e` is | ||
/// the number of links in the `graph`, is the number of nodes, and `k` is | ||
/// the number of ports in the boundary. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I am confused here whether the argument graph: &impl LinkView
needs to be a graph of the bounded subgraph exactly, or it can be a larger graph that included the subgraph of interest.
The fact that on line 176 you filter for nodes with 0 incoming neighbours makes me think that this will only work if the bounded graph is the entire graph -- but in that case, isn't the boundary already specified by the set of disconnected ports?
From the user's perspective, I would find passing a graph for which the region of interest is only a subgraph a simpler interface to use -- otherwise I'd expect that in most cases the user will have to construct a Subgraph
struct before being able to call this function.
If the intention is to let user pass large graphs of which only a subgraph is of interest, then doing a toposort of the entire graph on line 180 seems wasteful.
Let me know if I misunderstood something!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There was a bug in my code, but the main problem here are nodes that cannot be reached from the boundary inputs.
For example, consider the boundary input: [0-->1], output: [2-->3]
in the following graph (
flowchart LR
0["0"] --> 1["1"]
1 --> 2["2"]
2 --> 3["3"]
dangling["dangling"] --> 2
style 1 stroke:#D50000
style 2 stroke:#D50000
style dangling stroke:#D50000
linkStyle 0 stroke:#2962FF,fill:none
linkStyle 1 stroke:#D50000,fill:none
linkStyle 2 stroke:#2962FF,fill:none
linkStyle 3 stroke:#D50000
The code started the toposort using only 1
as initial node to avoid the full graph traversal, and as result never processed 2
.
The solution here is to add dangling
to the initial set too, but doing so eagerly requires either computing the subgraph first (O(n)
space) or doing a full graph traversal.
In my last commit I added the full graph traversal option, but only when we realize we are stuck.
I think avoiding the O(n)
memory overhead is nice since it lets the user pass their pre-computed subgraphs if they have them, but if you think that's not needed we can compute the subgraph instead.
(there's a new test case in boundary.rs
for this)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Uhm, giving it a bit more though I think you have a point in that I should be doing a better job at filtering nodes/edges not in the boundary-induced subgraph.
In this case:
flowchart LR
0["0"] --> 1["1"]
2["3"] --> 3["4"]
1 --> n1["2 (outside boundary)"]
n1 --> 2
style 1 stroke:#D50000
style 2 stroke:#D50000
linkStyle 0 stroke:#2962FF,fill:none
linkStyle 1 stroke:#2962FF,fill:none
linkStyle 2 stroke:#2962FF
linkStyle 3 stroke:#2962FF,fill:none
Where input boundary ports = [0->1, 2->3]
and output boundary ports = [1->2, 3->4]
, There is no order dependency between both sections of the boundary but the current algorithm will detect one from traversing 2
.
It should be fixable using toposort's port/node filters, and probably computing the induced subgraph on-the-fly if needed if there are dangling nodes.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fixed. I moved collect_nodes
from subgraph.rs
to a private Boundary
method and added a test for this.
I'm afraid this PR is becoming a bit noisy :/
e3fbdd2
to
3cdb260
Compare
a57afab
to
e7cd68e
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Very nice!
Co-authored-by: Luca Mondada <[email protected]>
4ee64a7
to
c35aeb5
Compare
## 🤖 New release * `portgraph`: 0.12.3 -> 0.13.0 (⚠️ API breaking changes) ###⚠️ `portgraph` breaking changes ``` --- failure auto_trait_impl_removed: auto trait no longer implemented --- Description: A public type has stopped implementing one or more auto traits. This can break downstream code that depends on the traits being implemented. ref: https://doc.rust-lang.org/reference/special-types-and-traits.html#auto-traits impl: https://github.com/obi1kenobi/cargo-semver-checks/tree/v0.38.0/src/lints/auto_trait_impl_removed.ron Failed in: type EdgeRefs is no longer Send, in /tmp/.tmpncG4Dv/portgraph/src/view/petgraph.rs:326 type EdgeRefs is no longer Sync, in /tmp/.tmpncG4Dv/portgraph/src/view/petgraph.rs:326 type EdgeRefs is no longer UnwindSafe, in /tmp/.tmpncG4Dv/portgraph/src/view/petgraph.rs:326 type EdgeRefs is no longer RefUnwindSafe, in /tmp/.tmpncG4Dv/portgraph/src/view/petgraph.rs:326 type NodeEdgeRefs is no longer Send, in /tmp/.tmpncG4Dv/portgraph/src/view/petgraph.rs:396 type NodeEdgeRefs is no longer Sync, in /tmp/.tmpncG4Dv/portgraph/src/view/petgraph.rs:396 type NodeEdgeRefs is no longer UnwindSafe, in /tmp/.tmpncG4Dv/portgraph/src/view/petgraph.rs:396 type NodeEdgeRefs is no longer RefUnwindSafe, in /tmp/.tmpncG4Dv/portgraph/src/view/petgraph.rs:396 --- failure inherent_method_missing: pub method removed or renamed --- Description: A publicly-visible method or associated fn is no longer available under its prior name. It may have been renamed or removed entirely. ref: https://doc.rust-lang.org/cargo/reference/semver.html#item-remove impl: https://github.com/obi1kenobi/cargo-semver-checks/tree/v0.38.0/src/lints/inherent_method_missing.ron Failed in: FilteredGraph::new_region, previously in file /tmp/.tmpJ9kcgB/portgraph/src/view/region.rs:26 FilteredGraph::new_flat_region, previously in file /tmp/.tmpJ9kcgB/portgraph/src/view/region.rs:105 FilteredGraph::new_subgraph, previously in file /tmp/.tmpJ9kcgB/portgraph/src/view/subgraph.rs:73 FilteredGraph::is_convex, previously in file /tmp/.tmpJ9kcgB/portgraph/src/view/subgraph.rs:100 FilteredGraph::is_convex_with_checker, previously in file /tmp/.tmpJ9kcgB/portgraph/src/view/subgraph.rs:106 FilteredGraph::new_region, previously in file /tmp/.tmpJ9kcgB/portgraph/src/view/region.rs:26 FilteredGraph::new_flat_region, previously in file /tmp/.tmpJ9kcgB/portgraph/src/view/region.rs:105 FilteredGraph::new_subgraph, previously in file /tmp/.tmpJ9kcgB/portgraph/src/view/subgraph.rs:73 FilteredGraph::is_convex, previously in file /tmp/.tmpJ9kcgB/portgraph/src/view/subgraph.rs:100 FilteredGraph::is_convex_with_checker, previously in file /tmp/.tmpJ9kcgB/portgraph/src/view/subgraph.rs:106 --- failure module_missing: pub module removed or renamed --- Description: A publicly-visible module cannot be imported by its prior path. A `pub use` may have been removed, or the module may have been renamed, removed, or made non-public. ref: https://doc.rust-lang.org/cargo/reference/semver.html#item-remove impl: https://github.com/obi1kenobi/cargo-semver-checks/tree/v0.38.0/src/lints/module_missing.ron Failed in: mod portgraph::py_graph, previously in file /tmp/.tmpJ9kcgB/portgraph/src/py_graph.rs:1 mod portgraph::view::subgraph, previously in file /tmp/.tmpJ9kcgB/portgraph/src/view/subgraph.rs:1 mod portgraph::view::region, previously in file /tmp/.tmpJ9kcgB/portgraph/src/view/region.rs:1 --- failure struct_missing: pub struct removed or renamed --- Description: A publicly-visible struct cannot be imported by its prior path. A `pub use` may have been removed, or the struct itself may have been renamed or removed entirely. ref: https://doc.rust-lang.org/cargo/reference/semver.html#item-remove impl: https://github.com/obi1kenobi/cargo-semver-checks/tree/v0.38.0/src/lints/struct_missing.ron Failed in: struct portgraph::view::region::RegionContext, previously in file /tmp/.tmpJ9kcgB/portgraph/src/view/region.rs:35 struct portgraph::view::filter::FilteredGraphCtx, previously in file /tmp/.tmpJ9kcgB/portgraph/src/view/filter.rs:131 struct portgraph::view::subgraph::SubgraphContext, previously in file /tmp/.tmpJ9kcgB/portgraph/src/view/subgraph.rs:53 --- failure trait_removed_associated_type: trait's associated type was removed --- Description: A public trait's associated type was removed or renamed. ref: https://doc.rust-lang.org/cargo/reference/semver.html#item-remove impl: https://github.com/obi1kenobi/cargo-semver-checks/tree/v0.38.0/src/lints/trait_removed_associated_type.ron Failed in: associated type LinkView::Neighbours, previously at /tmp/.tmpJ9kcgB/portgraph/src/view.rs:291 associated type LinkView::NodeConnections, previously at /tmp/.tmpJ9kcgB/portgraph/src/view.rs:296 associated type LinkView::NodeLinks, previously at /tmp/.tmpJ9kcgB/portgraph/src/view.rs:302 associated type LinkView::PortLinks, previously at /tmp/.tmpJ9kcgB/portgraph/src/view.rs:308 associated type LinkView::Neighbours, previously at /tmp/.tmpJ9kcgB/portgraph/src/view.rs:291 associated type LinkView::NodeConnections, previously at /tmp/.tmpJ9kcgB/portgraph/src/view.rs:296 associated type LinkView::NodeLinks, previously at /tmp/.tmpJ9kcgB/portgraph/src/view.rs:302 associated type LinkView::PortLinks, previously at /tmp/.tmpJ9kcgB/portgraph/src/view.rs:308 associated type MultiView::NodeSubports, previously at /tmp/.tmpJ9kcgB/portgraph/src/view.rs:589 associated type MultiView::NodeSubports, previously at /tmp/.tmpJ9kcgB/portgraph/src/view.rs:589 associated type PortView::Nodes, previously at /tmp/.tmpJ9kcgB/portgraph/src/view.rs:22 associated type PortView::Ports, previously at /tmp/.tmpJ9kcgB/portgraph/src/view.rs:27 associated type PortView::NodePorts, previously at /tmp/.tmpJ9kcgB/portgraph/src/view.rs:32 associated type PortView::NodePortOffsets, previously at /tmp/.tmpJ9kcgB/portgraph/src/view.rs:37 associated type PortView::Nodes, previously at /tmp/.tmpJ9kcgB/portgraph/src/view.rs:22 associated type PortView::Ports, previously at /tmp/.tmpJ9kcgB/portgraph/src/view.rs:27 associated type PortView::NodePorts, previously at /tmp/.tmpJ9kcgB/portgraph/src/view.rs:32 associated type PortView::NodePortOffsets, previously at /tmp/.tmpJ9kcgB/portgraph/src/view.rs:37 ``` <details><summary><i><b>Changelog</b></i></summary><p> <blockquote> ## [0.13.0](v0.12.3...v0.13.0) - 2025-01-17 ### New Features - [**breaking**] Use RPITIT for graph traits (#156) - [**breaking**] Boundary order checking (#164) ### Performance - [**breaking**] Fix Subgraph O(n) complexity (#157) - [**breaking**] Manual impl of `Region`/`FlatRegion` for improved perf (#162) </blockquote> </p></details> --- This PR was generated with [release-plz](https://github.com/release-plz/release-plz/). --------- Co-authored-by: Agustín Borgna <[email protected]>
Implements #161 as described in CQCL/hugr#1656.
Once this is merged, we should be able to optimise the convex checker to only test the edges returned by
PortOrdering::missing_pairs
. When the original order is stronger than the other in a rewrite, convex checking can be skipped altogether.BREAKING CHANGE:
Subgraph::new_subgraph
now takes aBoundary
as argument.