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(iroh-net)!: Create our own connection struct #2768

Closed
wants to merge 1 commit into from

Conversation

flub
Copy link
Contributor

@flub flub commented Oct 1, 2024

Description

We used to re-export quinn::Connection, this makes this an owned
connection so we can add our own methods to this. This adds the
.remote_peer_id() method and removes the helper function for this.

Breaking Changes

iroh_net::endpoint::get_remote_node_id has been removed. Use
iroh_net::endpoint::Connection::remote_node_id instead.

Notes & open questions

  • We have talked about wanting to do this for so long, decided to just
    give this a go.

  • Strictly speaking we do not need to remove get_remote_node_id
    right now.

  • This completely removes the ability of iroh-blobs to work with
    upstream Quinn connections. It now only works with iroh. We could
    probably use the h3::quic traits in iroh-blobs to make it generic
    over QUIC implementation.

  • The code just lives in iroh_net::endpoint now. That's probably
    fine, but it's a large file by now. Easy enough to move it later
    though.

  • The public API is at iroh_net::endpoint::Connection. Perhaps it
    should also be exported at iroh_net::Connection just like
    Endpoint is. Though what belongs there? What about all the other
    types? Hence I'm leaving this as the status quo for this PR and
    leave it where it is.

  • There are a bunch of new Quinn structs exported. This probably
    should have happened earlier, we knew we were not exporting enough
    of those but never did a proper check on what is needed. Now maybe
    everything is there, bar possibly some things from quinn-udp still
    missing.

Change checklist

  • Self-review.
  • Documentation updates following the style guide, if relevant.
  • [ ] Tests if relevant.
  • All breaking changes documented.

We used to re-export quinn::Connection, this makes this an owned
connection so we can add our own methods to this.  This adds the
`.remote_peer_id()` method and removes the helper function for this.
@flub flub requested review from dignifiedquire, rklaehn and divagant-martian and removed request for dignifiedquire October 1, 2024 11:36
Copy link

github-actions bot commented Oct 1, 2024

Netsim report & logs for this PR have been generated and is available at: LOGS
This report will remain available for 3 days.

Last updated for commit: 0d3af4d

@divagant-martian
Copy link
Contributor

Seen from iroh_net it makes sense. Is there anything preventing us from implementing Into<quinn::Connection> or AsRef or similar so that it doesn't affect iroh_blobs as much?

@flub
Copy link
Contributor Author

flub commented Oct 2, 2024

Seen from iroh_net it makes sense. Is there anything preventing us from implementing Into<quinn::Connection> or AsRef or similar so that it doesn't affect iroh_blobs as much?

This is a good idea!

@flub
Copy link
Contributor Author

flub commented Oct 2, 2024

Seen from iroh_net it makes sense. Is there anything preventing us from implementing Into<quinn::Connection> or AsRef or similar so that it doesn't affect iroh_blobs as much?

This is a good idea!

Note that there are no code changes since iroh-blobs already uses iroh_net::nedpoint::Connection as import. I'd love to get @rklaehn's opinion on this before changing that back to iroh_quinn::Connection. Because the real breakage was done at the point when we transitioned from quinn to iroh-quinn.

@divagant-martian
Copy link
Contributor

Back to this, is there any reason to not implement Deref and DerefMut so that we don't have to keep updating docs and migrating new functions? Sometimes docs for Dereferencing types is not great, but otherwise it seems appropriate

@flub
Copy link
Contributor Author

flub commented Oct 3, 2024

Back to this, is there any reason to not implement Deref and DerefMut so that we don't have to keep updating docs and migrating new functions? Sometimes docs for Dereferencing types is not great, but otherwise it seems appropriate

I'm not really a fan of Deref for users, it always confuses me as a user. Plus it gives us less flexibility. Yes we need to keep this in sync when we update the dependency, but we also get to benefit from better integrated documentation - I did make changes to the docs here. So I think I'd prefer this approach.

@divagant-martian
Copy link
Contributor

I'm not really a fan of Deref for users

Yeah, I wish rust had a better way to do or at least show this when all we need is delegation.
Agree with the conclussion

@rklaehn
Copy link
Contributor

rklaehn commented Oct 4, 2024

Can you elaborate your future plans with this? If the only thing this allows us to do is to change the get_remote_node_id fn into a member fn, I would say it is not worth it.

It would help with making examples more approachable, but you could do the same (except for an extra import) with an extension method.

@flub
Copy link
Contributor Author

flub commented Oct 4, 2024

Can you elaborate your future plans with this?

This can enable a number of things which we have shied away from in the past simply because taking the step of adding this struct was not yet done:

  • Connection type and changes to it are currently on on the Endpoint, they also belong on the Connection.
  • Connection information, like which paths are used, the latencies of those paths, etc, also belongs on the Connection.
  • Some APIs don't work well for us, e.g. Connection::local_ip or Connection::remote_address are not that meaningful. This allows us to remove those and replace them with more appropriate APIs.
  • It is generally not great to expose types which you have no control over in your public API. This doesn't solve that fully, but it does tackle an important struct.

Probably some more things I'm not remembering right now.

Admittedly this PR doesn't tackle many of these things. But it makes it so much easier to do in the future.

@b5
Copy link
Member

b5 commented Oct 9, 2024

@rklaehn
Copy link
Contributor

rklaehn commented Oct 16, 2024

Given that we have to hide all of quinn for 1.0 anyway, I think this is reasonable to do.

@@ -221,7 +221,7 @@ async fn handle_connection(
) -> anyhow::Result<()> {
let alpn = conn.alpn().await?;
let conn = conn.await?;
let peer_id = iroh_net::endpoint::get_remote_node_id(&conn)?;
let peer_id = conn.remote_node_id()?;
Copy link
Contributor

Choose a reason for hiding this comment

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

Call this node_id while we're at it?

@@ -116,7 +116,7 @@ where
{
let t_start = Instant::now();
let connection = connecting.await.map_err(AcceptError::connect)?;
let peer = get_remote_node_id(&connection).map_err(AcceptError::connect)?;
let peer = connection.remote_node_id().map_err(AcceptError::connect)?;
Copy link
Contributor

Choose a reason for hiding this comment

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

node_id?

@@ -792,7 +792,7 @@ async fn accept(
match connecting.await {
Ok(connection) => {
if n == 0 {
let Ok(remote_peer_id) = endpoint::get_remote_node_id(&connection) else {
let Ok(remote_peer_id) = connection.remote_node_id() else {
Copy link
Contributor

Choose a reason for hiding this comment

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

let Ok(remote_node_id)

///
/// May be cloned to obtain another handle to the same connection.
#[derive(Debug, Clone)]
pub struct Connection {
Copy link
Contributor

Choose a reason for hiding this comment

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

Can we have #[repr(transparent)] on this? Then we could use ref_cast to get a &quinn::Connection from a &Connection as a (possibly feature flagged) compat feature.

We might or might not do this, but the repr(transparent) certainly does not do any harm unless we plan to have additional stuff here...

@divagant-martian
Copy link
Contributor

PR came first checking approved ones (candidates for merging) but this has a long list of conflicts. Since this has essentially passed the approval bar, here's a friendly ping to refresh it @flub

@flub flub marked this pull request as draft November 13, 2024 14:37
@flub
Copy link
Contributor Author

flub commented Nov 13, 2024

Yeah, I was waiting for the dust to settle on all the renames and moving crates before getting back to this. Doing backwards-incompatible changes was very difficult at some point during this churn.

Converted it back to draft for now, will pick it up after the next release probably.

@flub
Copy link
Contributor Author

flub commented Jan 9, 2025

Replaced with #3110

@flub flub closed this Jan 9, 2025
github-merge-queue bot pushed a commit that referenced this pull request Jan 23, 2025
## Description

This makes iroh own the Connection type.  This is desirable because:

- Some: upstream APIs don't work well for us,
  e.g. Connection::remote_address is not that meaningful.  This allows
  us to remove those and replace them with more appropriate APIs.  This
  has not yet been done in this PR however.

- The connection type and changes are currently on the Endpoint, they
  also belong on the Connection.

- Connection information, like which pats are used, the latencies of
  those paths etc, also belong on the Connection.

- It is generally not great to expose types which you have no control
  over in your public API.  For the 1.0 we do not want to have any
  such structs.  This takes an important step in that direction.

As part of this change Connection::remote_node_id replaces the
get_remote_node_id function.  Also Connection::alpn now exists.

## Breaking Changes

### iroh

- `iroh::endpoint::get_remote_node_id` has been removed.  Use
  `iroh::endpoint::Connection::remote_node_id` instead.

## Notes & open questions

This does not yet clean up the existing methods in the Connection,
preserving most of them as-is.  It is easier to handle those
separately as each one involves careful decisions about the API.

Replaces #2768

## Change checklist

- [x] Self-review.
- [x] Documentation updates following the [style
guide](https://rust-lang.github.io/rfcs/1574-more-api-documentation-conventions.html#appendix-a-full-conventions-text),
if relevant.
- [x] Tests if relevant.
- [x] All breaking changes documented.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
Status: ✅ Done
Development

Successfully merging this pull request may close these issues.

4 participants