Skip to content
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

Merged
merged 12 commits into from
Jan 17, 2025
Merged

feat!: Boundary order checking #164

merged 12 commits into from
Jan 17, 2025

Conversation

aborgna-q
Copy link
Collaborator

@aborgna-q aborgna-q commented Nov 21, 2024

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 a Boundary as argument.

@aborgna-q aborgna-q linked an issue Nov 21, 2024 that may be closed by this pull request
Copy link

codecov bot commented Nov 21, 2024

Codecov Report

Attention: Patch coverage is 94.76744% with 18 lines in your changes missing coverage. Please review.

Project coverage is 83.23%. Comparing base (f358376) to head (c35aeb5).
Report is 1 commits behind head on main.

Files with missing lines Patch % Lines
src/boundary.rs 95.98% 10 Missing and 1 partial ⚠️
src/algorithms/toposort.rs 69.56% 6 Missing and 1 partial ⚠️
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.
📢 Have feedback on the report? Share it here.

@aborgna-q aborgna-q marked this pull request as ready for review November 24, 2024 14:50
@aborgna-q aborgna-q requested a review from lmondada November 25, 2024 09:45
Copy link
Contributor

@lmondada lmondada left a 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.

Comment on lines 139 to 152
/// 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.
Copy link
Contributor

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!

Copy link
Collaborator Author

@aborgna-q aborgna-q Jan 15, 2025

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 ($\color{red}\text{red}$ = inside boundary, $\color{blue}\text{blue}$ = boundary edge).

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
Loading

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)

Copy link
Collaborator Author

@aborgna-q aborgna-q Jan 16, 2025

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
Loading

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.

Copy link
Collaborator Author

@aborgna-q aborgna-q Jan 16, 2025

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 :/

@aborgna-q aborgna-q requested a review from lmondada January 15, 2025 15:33
@aborgna-q aborgna-q changed the title feat: Boundary order checking feat!: Boundary order checking Jan 15, 2025
@aborgna-q aborgna-q removed the request for review from lmondada January 16, 2025 13:29
@aborgna-q aborgna-q requested a review from lmondada January 16, 2025 15:49
Copy link
Contributor

@lmondada lmondada left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Very nice!

@aborgna-q aborgna-q added this pull request to the merge queue Jan 17, 2025
Merged via the queue into main with commit 55b951f Jan 17, 2025
12 checks passed
@aborgna-q aborgna-q deleted the ab/boundary-order branch January 17, 2025 11:42
@hugrbot hugrbot mentioned this pull request Jan 13, 2025
github-merge-queue bot pushed a commit that referenced this pull request Jan 17, 2025
## 🤖 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]>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Compare boundary partial orders
2 participants