| //@ 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<T> 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<S::Item>; |
| |
| fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<S::Item>> { |
| 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, |
| } |
| } |
| } |