//@ edition: 2024 //@ run-pass #![feature(gen_blocks, async_iterator)] // make sure that a ridiculously simple async gen fn works as an iterator. async fn pause() { // this doesn't actually do anything, lol } async fn one() -> i32 { 1 } async fn two() -> i32 { 2 } async gen fn foo() -> i32 { yield one().await; pause().await; yield two().await; pause().await; yield 3; pause().await; } async fn async_main() { let mut iter = std::pin::pin!(foo()); assert_eq!(iter.next().await, Some(1)); assert_eq!(iter.as_mut().next().await, Some(2)); assert_eq!(iter.as_mut().next().await, Some(3)); assert_eq!(iter.as_mut().next().await, None); // Test that the iterator is fused and does not panic assert_eq!(iter.as_mut().next().await, None); assert_eq!(iter.as_mut().next().await, None); } // ------------------------------------------------------------------------- // // Implementation Details Below... use std::pin::{Pin, pin}; use std::task::*; use std::async_iter::AsyncIterator; trait AsyncIterExt { fn next(&mut self) -> Next<'_, Self>; } impl AsyncIterExt for T { fn next(&mut self) -> Next<'_, Self> { Next { s: self } } } struct Next<'s, S: ?Sized> { s: &'s mut S, } impl<'s, S: AsyncIterator> Future for Next<'s, S> where S: Unpin { type Output = Option; fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll> { Pin::new(&mut *self.s).poll_next(cx) } } fn main() { let mut fut = pin!(async_main()); // Poll loop, just to test the future... let ctx = &mut Context::from_waker(Waker::noop()); loop { match fut.as_mut().poll(ctx) { Poll::Pending => {} Poll::Ready(()) => break, } } }