-
Notifications
You must be signed in to change notification settings - Fork 13k
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
Delay stabilizing async closures to consider if they should return impl IntoFuture
instead of impl Future
#135664
Comments
(edited to be a bit more coherent and also express my feelings more clearly) I think it's frankly ridiculous that this is being brought up so late in the process, and it's incredibly disappointing and stressful to go through stabilization whiplash like this. I hope T-lang is aware of the burden this type of shit has on contributors who spend significant time on a feature to just have it be churned after stabilization like this as soon as someone comes up with a possible blocker, regardless of how remotely relevant or even feasible it is to change at that point. It's also pretty disappointing to read from the issue that apparently two T-lang members were consulted about this, but I had to hear about this at 7:55 PM on a Friday with no prior knowledge; I'd like to believe I'm pretty accessible, and on one hand I know there's no obligation to ever keep any one person in the loop1, but on the other hand, this really erodes my confidence in Rust's stabilization process and makes me really second guess whether my insight is valued at all here. I'm probably not going to participate in any of this discussion out of consideration for my own mental health. I'd encourage anyone who does participate to be acutely aware of the technical limitations and burden on users that changing this design has, and the implications that last-minute blocking this feature has on delivering async closures to Rust users. Footnotes
|
This is a really neat idea, and we should definitely explore it in the future! In the meantime, I don't think it makes sense to block the stabilization of async closures. As @compiler-errors pointed out, an absolutely massive amount of work has been done in order to prepare this feature for stabilization and to exercise the details of the interface and (as mentioned above) ensure that it is compatible with existing APIs that accept The current behavior is also consistent with the current semantics of P.S.: we'll need to investigate some ergonomic way to spell the bounds for these types-- |
From a technical perspective, unlike what was discussed in rust-lang/rfcs#3668 (comment), in order to mitigate this "
...would be breaking, as the type of Async closures would also no longer be compatible with any existing APIs that take a "old-style"/async-closure-like
Using async closures with existing future combinators like futures-rs's I've taken a lot of care to ensure that async closures are compatible with the existing ecosystem which has had to rely on I think that this churn will be far more prevalent compared to the relatively niche case of wanting |
as I mentioned:
It seems to me that this was in fact effectively discussed, as it is isomorphic to the concern raised by programmerjake, then resolved. Then the FCP commenced (i.e. all T-lang members decided and resolved their concerns) after. Then the RFC was accepted. So we already looked at something almost identical to this problem and said "nah, not worth" for fundamentally the same reasons to make the same decision again. I think it would be surprising for us to about-face now, and I would like to see the explanation that materially distinguishes these two cases in any significant way. In other words, why does @tmandry's answer not still hold?
|
This was explicitly brought up as a concern as part of the FCP no? a concern was registered here rust-lang/rfcs#3668 (comment) and then resolved here rust-lang/rfcs#3668 (comment). I can see atleast @tmandry @traviscross and @eholk were aware of this at time of RFC and the nebuluous set of people "in the triage meeting" talked about this. I don't see any reason why lang should be re-litigating something this late in process that has already been taken into account during the RFC process |
Hey everyone. I think the points raised on this thread have been good ones. While I think the motivation provided on the issue is more compelling than any we considered for |
I would like to apologize for the way in which I raised this issue because it caused unnecessary stress, particularly on @compiler-errors. It was sloppy, and as a result, hurtful. This is an exceptional request, and I did not do the due diligence required in making a request like this. As a bit of background, here's how this got posted. I say this not as a defense, but so I can use that to show concrete ways I could have done better. I was having a discussion with some folks about a design for generators, which have a lot in common with async Rust due to both being built around the coroutine transform. The question of whether We believed this amounted to new information that could have bearing on the decision. Part of why we have a beta period is to allow this to happen, so I think raising issues like this with actual new information is appropriate. But it should be done with care and diligence that I did not show. Given the urgency, this afternoon I drafted up an issue and chose to post it so that it could be considered at the next Lang Team meeting. The result was that I posted the issue at around 5pm on a Friday evening. This meant @compiler-errors got to go into his weekend with me trying to roll back work that he has put a ton energy into getting across the finish line. With that background, what could and should I have done differently? A simple thing that I wish I had done is to hold off and post the issue after the weekend. That would have avoided the timing and also given me more time to think through a proposal of this magnitude. Had I done so, I would have either decided not to post it, or would have posted something that was stronger. While I felt there was some urgency, posting on Monday or Tuesday would have still given enough time to discuss it in the next Lang Team meeting and make a decision before the beta is promoted to stable. Second, I did not consult with @compiler-errors before posting this issue. This was inconsiderate. Had I asked @compiler-errors, it would have meant this issue didn't come as a public surprise and would have given him the chance to respond without feeling the stress and urgency of rebutting something on GitHub. After asking him, had I gone forward with proposing the revert, I would have likely done so with his support (and if I didn't have his support, he probably would have also convinced me this change was a bad idea). There's a principle I've heard discussed somewhere of respecting what came before. The idea is that carefully thought-out decisions that have stood the test of time and/or faced a lot of scrutiny are likely correct. Obviously, that's not always the case, but changing these decisions needs to be done with care and understanding. I could have and should have done better to live up to this principle. I intend to learn from this and show more care in the future. I apologize to @compiler-errors for the stress and harm I have caused. |
In case it wasn't clear, lang hadn't and hasn't made any decision here -- not about a beta revert and not about changing the behavior -- neither had any lang members, and as best as I understand @eholk's position at the time, nor had he either really. The most any of us thought for sure is that it was worth discussing with the team, and that we were curious what others might make of it. As everyone here knows, I think, nothing is or was going to happen until Wednesday at the earliest. As above, @eholk has started in on beating himself up a bit, so I want to be clear that I don't think he's at any fault. In light of @yoshuawuyts' piece, and some reasonable extrapolation, he put forward a reasonable write-up of something we on lang should reasonably consider, in the spirit of double checking things, as we try to do. He clearly put it forward in good faith and in trying to do the best thing for Rust. He CCed the likely interested parties as a necessary and appropriate courtesy, as we do. He's certainly not responsible for what we on lang might or might not do with this. We check, double check, and yes -- sometimes second-guess ourselves. That's all this is. It normally amounts to us still doing what we planned to do. At the same time, having certainly been on all sides of that myself, I understand how even the prospect of that can be concerning and frustrating. I get it. Certainly concern and frustration isn't what we want. That's a hard problem, and minimizing that is of course something to which we will continue to give thought and attention. |
That doesn't make it better. |
Async closures are currently in beta for stabilization in 1.85. I'm very excited to see this feature stabilized!
That said, I would like the Lang team to consider delaying their stabilization to consider a new design detail. Delaying stabilization would amount to putting a PR in nightly that undoes the stabilization, and then doing a beta backport.
Note that this doesn't necessarily mean the lang team would be committing to making this change, but if the lang team feels there's a non-trivial chance they would prefer the
IntoFuture
semantics, then we should delay stabilization so there is time to consider this.I talked with @tmandry and @traviscross about this and they did not remember discussing this particular detail during the stabilization process. If we are mistaken and this was discussed, then we should let the decision stand and not change anything.
Why should T-lang consider this change?
@yoshuawuyts recently published a blog post about generators and auto trait leakage, which argues that
gen {}
should returnimpl IntoIterator
instead ofimpl Iterator
. The key example in the post is (adapted slightly):If
gen {}
returnsimpl Iterator
, as it does on nightly today, this example fails to compile because theRc
is notSend
. However, this is overly restrictive because theRc
is not created until the spawned thread callsnext()
the first time.Instead, having
gen {}
returnimpl IntoIterator
allows this to work, because theIntoIterator
part could beSend
, and once it becomes anIterator
it would no longer beSend
.This has a similar shape to annoyances we've seen with async Rust, where it's not uncommon for people to want to write futures that are
Send
until they are polled for the first time. If we had madeasync {}
blocks returnimpl IntoFuture
instead, this would be straightforward to do for the same reasons.So then the next logical question is what should generator closures or async closures do? For much the same reason, it would be nice for async closures to return futures that are
Send
until first polled.At this point, changing the return type for
async {}
would require an edition change. However, we have a brief window where we can still make this change for async closures.Why should T-lang keep the status quo?
A lot of care has gone into the current design by @compiler-errors and others to come up with something that balances a lot of competing goals. For example, with the current design, the following example works, but it would break with this proposed change:
This is a desirable property, since a lot of existing ecosystem code is written using
Fn() -> impl Future
as an approximation of an async closure. If async closures returnedIntoFuture
then they would not work until existing code is updated to support async closures natively.I bring this up mainly to show that it's not obvious that we actually should make this change. There are many factors in tension and there's a good chance we already have the optimal balance. We'd want to thoroughly consider any changes, so this issue is mostly about whether we want to take the time to consider those changes.
Mitigating Factors
This would still be changeable over an edition if we were to decide it's worthwhile but don't want to churn things now. If we were to make this change, we'd likely want to change the behavior of
async {}
. Doing both over an edition at the same time would keep the overall language more consistent.The text was updated successfully, but these errors were encountered: