| 1 | use crate::fns::FnOnce1; |
| 2 | use crate::stream::{Fuse, StreamExt}; |
| 3 | use core::fmt; |
| 4 | use core::marker::PhantomData; |
| 5 | use core::pin::Pin; |
| 6 | use futures_core::future::{FusedFuture, Future}; |
| 7 | use futures_core::ready; |
| 8 | use futures_core::stream::{FusedStream, Stream}; |
| 9 | use futures_core::task::{Context, Poll}; |
| 10 | #[cfg (feature = "sink" )] |
| 11 | use futures_sink::Sink; |
| 12 | use pin_project_lite::pin_project; |
| 13 | |
| 14 | pin_project! { |
| 15 | /// A `Stream` that implements a `peek` method. |
| 16 | /// |
| 17 | /// The `peek` method can be used to retrieve a reference |
| 18 | /// to the next `Stream::Item` if available. A subsequent |
| 19 | /// call to `poll` will return the owned item. |
| 20 | #[derive (Debug)] |
| 21 | #[must_use = "streams do nothing unless polled" ] |
| 22 | pub struct Peekable<St: Stream> { |
| 23 | #[pin] |
| 24 | stream: Fuse<St>, |
| 25 | peeked: Option<St::Item>, |
| 26 | } |
| 27 | } |
| 28 | |
| 29 | impl<St: Stream> Peekable<St> { |
| 30 | pub(super) fn new(stream: St) -> Self { |
| 31 | Self { stream: stream.fuse(), peeked: None } |
| 32 | } |
| 33 | |
| 34 | delegate_access_inner!(stream, St, (.)); |
| 35 | |
| 36 | /// Produces a future which retrieves a reference to the next item |
| 37 | /// in the stream, or `None` if the underlying stream terminates. |
| 38 | pub fn peek(self: Pin<&mut Self>) -> Peek<'_, St> { |
| 39 | Peek { inner: Some(self) } |
| 40 | } |
| 41 | |
| 42 | /// Peek retrieves a reference to the next item in the stream. |
| 43 | /// |
| 44 | /// This method polls the underlying stream and return either a reference |
| 45 | /// to the next item if the stream is ready or passes through any errors. |
| 46 | pub fn poll_peek(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<&St::Item>> { |
| 47 | let mut this = self.project(); |
| 48 | |
| 49 | Poll::Ready(loop { |
| 50 | if this.peeked.is_some() { |
| 51 | break this.peeked.as_ref(); |
| 52 | } else if let Some(item) = ready!(this.stream.as_mut().poll_next(cx)) { |
| 53 | *this.peeked = Some(item); |
| 54 | } else { |
| 55 | break None; |
| 56 | } |
| 57 | }) |
| 58 | } |
| 59 | |
| 60 | /// Produces a future which retrieves a mutable reference to the next item |
| 61 | /// in the stream, or `None` if the underlying stream terminates. |
| 62 | /// |
| 63 | /// # Examples |
| 64 | /// |
| 65 | /// ``` |
| 66 | /// # futures::executor::block_on(async { |
| 67 | /// use futures::stream::{self, StreamExt}; |
| 68 | /// use futures::pin_mut; |
| 69 | /// |
| 70 | /// let stream = stream::iter(vec![1, 2, 3]).peekable(); |
| 71 | /// pin_mut!(stream); |
| 72 | /// |
| 73 | /// assert_eq!(stream.as_mut().peek_mut().await, Some(&mut 1)); |
| 74 | /// assert_eq!(stream.as_mut().next().await, Some(1)); |
| 75 | /// |
| 76 | /// // Peek into the stream and modify the value which will be returned next |
| 77 | /// if let Some(p) = stream.as_mut().peek_mut().await { |
| 78 | /// if *p == 2 { |
| 79 | /// *p = 5; |
| 80 | /// } |
| 81 | /// } |
| 82 | /// |
| 83 | /// assert_eq!(stream.collect::<Vec<_>>().await, vec![5, 3]); |
| 84 | /// # }); |
| 85 | /// ``` |
| 86 | pub fn peek_mut(self: Pin<&mut Self>) -> PeekMut<'_, St> { |
| 87 | PeekMut { inner: Some(self) } |
| 88 | } |
| 89 | |
| 90 | /// Peek retrieves a mutable reference to the next item in the stream. |
| 91 | pub fn poll_peek_mut( |
| 92 | self: Pin<&mut Self>, |
| 93 | cx: &mut Context<'_>, |
| 94 | ) -> Poll<Option<&mut St::Item>> { |
| 95 | let mut this = self.project(); |
| 96 | |
| 97 | Poll::Ready(loop { |
| 98 | if this.peeked.is_some() { |
| 99 | break this.peeked.as_mut(); |
| 100 | } else if let Some(item) = ready!(this.stream.as_mut().poll_next(cx)) { |
| 101 | *this.peeked = Some(item); |
| 102 | } else { |
| 103 | break None; |
| 104 | } |
| 105 | }) |
| 106 | } |
| 107 | |
| 108 | /// Creates a future which will consume and return the next value of this |
| 109 | /// stream if a condition is true. |
| 110 | /// |
| 111 | /// If `func` returns `true` for the next value of this stream, consume and |
| 112 | /// return it. Otherwise, return `None`. |
| 113 | /// |
| 114 | /// # Examples |
| 115 | /// |
| 116 | /// Consume a number if it's equal to 0. |
| 117 | /// |
| 118 | /// ``` |
| 119 | /// # futures::executor::block_on(async { |
| 120 | /// use futures::stream::{self, StreamExt}; |
| 121 | /// use futures::pin_mut; |
| 122 | /// |
| 123 | /// let stream = stream::iter(0..5).peekable(); |
| 124 | /// pin_mut!(stream); |
| 125 | /// // The first item of the stream is 0; consume it. |
| 126 | /// assert_eq!(stream.as_mut().next_if(|&x| x == 0).await, Some(0)); |
| 127 | /// // The next item returned is now 1, so `consume` will return `false`. |
| 128 | /// assert_eq!(stream.as_mut().next_if(|&x| x == 0).await, None); |
| 129 | /// // `next_if` saves the value of the next item if it was not equal to `expected`. |
| 130 | /// assert_eq!(stream.next().await, Some(1)); |
| 131 | /// # }); |
| 132 | /// ``` |
| 133 | /// |
| 134 | /// Consume any number less than 10. |
| 135 | /// |
| 136 | /// ``` |
| 137 | /// # futures::executor::block_on(async { |
| 138 | /// use futures::stream::{self, StreamExt}; |
| 139 | /// use futures::pin_mut; |
| 140 | /// |
| 141 | /// let stream = stream::iter(1..20).peekable(); |
| 142 | /// pin_mut!(stream); |
| 143 | /// // Consume all numbers less than 10 |
| 144 | /// while stream.as_mut().next_if(|&x| x < 10).await.is_some() {} |
| 145 | /// // The next value returned will be 10 |
| 146 | /// assert_eq!(stream.next().await, Some(10)); |
| 147 | /// # }); |
| 148 | /// ``` |
| 149 | pub fn next_if<F>(self: Pin<&mut Self>, func: F) -> NextIf<'_, St, F> |
| 150 | where |
| 151 | F: FnOnce(&St::Item) -> bool, |
| 152 | { |
| 153 | NextIf { inner: Some((self, func)) } |
| 154 | } |
| 155 | |
| 156 | /// Creates a future which will consume and return the next item if it is |
| 157 | /// equal to `expected`. |
| 158 | /// |
| 159 | /// # Example |
| 160 | /// |
| 161 | /// Consume a number if it's equal to 0. |
| 162 | /// |
| 163 | /// ``` |
| 164 | /// # futures::executor::block_on(async { |
| 165 | /// use futures::stream::{self, StreamExt}; |
| 166 | /// use futures::pin_mut; |
| 167 | /// |
| 168 | /// let stream = stream::iter(0..5).peekable(); |
| 169 | /// pin_mut!(stream); |
| 170 | /// // The first item of the stream is 0; consume it. |
| 171 | /// assert_eq!(stream.as_mut().next_if_eq(&0).await, Some(0)); |
| 172 | /// // The next item returned is now 1, so `consume` will return `false`. |
| 173 | /// assert_eq!(stream.as_mut().next_if_eq(&0).await, None); |
| 174 | /// // `next_if_eq` saves the value of the next item if it was not equal to `expected`. |
| 175 | /// assert_eq!(stream.next().await, Some(1)); |
| 176 | /// # }); |
| 177 | /// ``` |
| 178 | pub fn next_if_eq<'a, T>(self: Pin<&'a mut Self>, expected: &'a T) -> NextIfEq<'a, St, T> |
| 179 | where |
| 180 | T: ?Sized, |
| 181 | St::Item: PartialEq<T>, |
| 182 | { |
| 183 | NextIfEq { |
| 184 | inner: NextIf { inner: Some((self, NextIfEqFn { expected, _next: PhantomData })) }, |
| 185 | } |
| 186 | } |
| 187 | } |
| 188 | |
| 189 | impl<St: Stream> FusedStream for Peekable<St> { |
| 190 | fn is_terminated(&self) -> bool { |
| 191 | self.peeked.is_none() && self.stream.is_terminated() |
| 192 | } |
| 193 | } |
| 194 | |
| 195 | impl<S: Stream> Stream for Peekable<S> { |
| 196 | type Item = S::Item; |
| 197 | |
| 198 | fn poll_next(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<Self::Item>> { |
| 199 | let this: Projection<'_, S> = self.project(); |
| 200 | if let Some(item: ::Item) = this.peeked.take() { |
| 201 | return Poll::Ready(Some(item)); |
| 202 | } |
| 203 | this.stream.poll_next(cx) |
| 204 | } |
| 205 | |
| 206 | fn size_hint(&self) -> (usize, Option<usize>) { |
| 207 | let peek_len: usize = usize::from(self.peeked.is_some()); |
| 208 | let (lower: usize, upper: Option) = self.stream.size_hint(); |
| 209 | let lower: usize = lower.saturating_add(peek_len); |
| 210 | let upper: Option = match upper { |
| 211 | Some(x: usize) => x.checked_add(peek_len), |
| 212 | None => None, |
| 213 | }; |
| 214 | (lower, upper) |
| 215 | } |
| 216 | } |
| 217 | |
| 218 | // Forwarding impl of Sink from the underlying stream |
| 219 | #[cfg (feature = "sink" )] |
| 220 | impl<S, Item> Sink<Item> for Peekable<S> |
| 221 | where |
| 222 | S: Sink<Item> + Stream, |
| 223 | { |
| 224 | type Error = S::Error; |
| 225 | |
| 226 | delegate_sink!(stream, Item); |
| 227 | } |
| 228 | |
| 229 | pin_project! { |
| 230 | /// Future for the [`Peekable::peek`](self::Peekable::peek) method. |
| 231 | #[must_use = "futures do nothing unless polled" ] |
| 232 | pub struct Peek<'a, St: Stream> { |
| 233 | inner: Option<Pin<&'a mut Peekable<St>>>, |
| 234 | } |
| 235 | } |
| 236 | |
| 237 | impl<St> fmt::Debug for Peek<'_, St> |
| 238 | where |
| 239 | St: Stream + fmt::Debug, |
| 240 | St::Item: fmt::Debug, |
| 241 | { |
| 242 | fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { |
| 243 | f.debug_struct("Peek" ).field(name:"inner" , &self.inner).finish() |
| 244 | } |
| 245 | } |
| 246 | |
| 247 | impl<St: Stream> FusedFuture for Peek<'_, St> { |
| 248 | fn is_terminated(&self) -> bool { |
| 249 | self.inner.is_none() |
| 250 | } |
| 251 | } |
| 252 | |
| 253 | impl<'a, St> Future for Peek<'a, St> |
| 254 | where |
| 255 | St: Stream, |
| 256 | { |
| 257 | type Output = Option<&'a St::Item>; |
| 258 | |
| 259 | fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> { |
| 260 | let inner: &mut Option>> = self.project().inner; |
| 261 | if let Some(peekable: &mut Pin<&'a mut Peekable>) = inner { |
| 262 | ready!(peekable.as_mut().poll_peek(cx)); |
| 263 | |
| 264 | inner.take().unwrap().poll_peek(cx) |
| 265 | } else { |
| 266 | panic!("Peek polled after completion" ) |
| 267 | } |
| 268 | } |
| 269 | } |
| 270 | |
| 271 | pin_project! { |
| 272 | /// Future for the [`Peekable::peek_mut`](self::Peekable::peek_mut) method. |
| 273 | #[must_use = "futures do nothing unless polled" ] |
| 274 | pub struct PeekMut<'a, St: Stream> { |
| 275 | inner: Option<Pin<&'a mut Peekable<St>>>, |
| 276 | } |
| 277 | } |
| 278 | |
| 279 | impl<St> fmt::Debug for PeekMut<'_, St> |
| 280 | where |
| 281 | St: Stream + fmt::Debug, |
| 282 | St::Item: fmt::Debug, |
| 283 | { |
| 284 | fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { |
| 285 | f.debug_struct("PeekMut" ).field(name:"inner" , &self.inner).finish() |
| 286 | } |
| 287 | } |
| 288 | |
| 289 | impl<St: Stream> FusedFuture for PeekMut<'_, St> { |
| 290 | fn is_terminated(&self) -> bool { |
| 291 | self.inner.is_none() |
| 292 | } |
| 293 | } |
| 294 | |
| 295 | impl<'a, St> Future for PeekMut<'a, St> |
| 296 | where |
| 297 | St: Stream, |
| 298 | { |
| 299 | type Output = Option<&'a mut St::Item>; |
| 300 | |
| 301 | fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> { |
| 302 | let inner: &mut Option>> = self.project().inner; |
| 303 | if let Some(peekable: &mut Pin<&'a mut Peekable>) = inner { |
| 304 | ready!(peekable.as_mut().poll_peek_mut(cx)); |
| 305 | |
| 306 | inner.take().unwrap().poll_peek_mut(cx) |
| 307 | } else { |
| 308 | panic!("PeekMut polled after completion" ) |
| 309 | } |
| 310 | } |
| 311 | } |
| 312 | |
| 313 | pin_project! { |
| 314 | /// Future for the [`Peekable::next_if`](self::Peekable::next_if) method. |
| 315 | #[must_use = "futures do nothing unless polled" ] |
| 316 | pub struct NextIf<'a, St: Stream, F> { |
| 317 | inner: Option<(Pin<&'a mut Peekable<St>>, F)>, |
| 318 | } |
| 319 | } |
| 320 | |
| 321 | impl<St, F> fmt::Debug for NextIf<'_, St, F> |
| 322 | where |
| 323 | St: Stream + fmt::Debug, |
| 324 | St::Item: fmt::Debug, |
| 325 | { |
| 326 | fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { |
| 327 | f.debug_struct("NextIf" ).field(name:"inner" , &self.inner.as_ref().map(|(s: &Pin<&mut Peekable>, _f: &F)| s)).finish() |
| 328 | } |
| 329 | } |
| 330 | |
| 331 | #[allow (single_use_lifetimes)] // https://github.com/rust-lang/rust/issues/55058 |
| 332 | impl<St, F> FusedFuture for NextIf<'_, St, F> |
| 333 | where |
| 334 | St: Stream, |
| 335 | F: for<'a> FnOnce1<&'a St::Item, Output = bool>, |
| 336 | { |
| 337 | fn is_terminated(&self) -> bool { |
| 338 | self.inner.is_none() |
| 339 | } |
| 340 | } |
| 341 | |
| 342 | #[allow (single_use_lifetimes)] // https://github.com/rust-lang/rust/issues/55058 |
| 343 | impl<St, F> Future for NextIf<'_, St, F> |
| 344 | where |
| 345 | St: Stream, |
| 346 | F: for<'a> FnOnce1<&'a St::Item, Output = bool>, |
| 347 | { |
| 348 | type Output = Option<St::Item>; |
| 349 | |
| 350 | fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> { |
| 351 | let inner: &mut Option<(Pin<&mut Peekable>, …)> = self.project().inner; |
| 352 | if let Some((peekable: &mut Pin<&mut Peekable>, _)) = inner { |
| 353 | let res: Option<::Item> = ready!(peekable.as_mut().poll_next(cx)); |
| 354 | |
| 355 | let (peekable: Pin<&mut Peekable>, func: F) = inner.take().unwrap(); |
| 356 | match res { |
| 357 | Some(ref matched: &::Item) if func.call_once(arg:matched) => Poll::Ready(res), |
| 358 | other: Option<::Item> => { |
| 359 | let peekable: Projection<'_, St> = peekable.project(); |
| 360 | // Since we called `self.next()`, we consumed `self.peeked`. |
| 361 | assert!(peekable.peeked.is_none()); |
| 362 | *peekable.peeked = other; |
| 363 | Poll::Ready(None) |
| 364 | } |
| 365 | } |
| 366 | } else { |
| 367 | panic!("NextIf polled after completion" ) |
| 368 | } |
| 369 | } |
| 370 | } |
| 371 | |
| 372 | pin_project! { |
| 373 | /// Future for the [`Peekable::next_if_eq`](self::Peekable::next_if_eq) method. |
| 374 | #[must_use = "futures do nothing unless polled" ] |
| 375 | pub struct NextIfEq<'a, St: Stream, T: ?Sized> { |
| 376 | #[pin] |
| 377 | inner: NextIf<'a, St, NextIfEqFn<'a, T, St::Item>>, |
| 378 | } |
| 379 | } |
| 380 | |
| 381 | impl<St, T> fmt::Debug for NextIfEq<'_, St, T> |
| 382 | where |
| 383 | St: Stream + fmt::Debug, |
| 384 | St::Item: fmt::Debug, |
| 385 | T: ?Sized, |
| 386 | { |
| 387 | fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { |
| 388 | f&mut DebugStruct<'_, '_>.debug_struct("NextIfEq" ) |
| 389 | .field(name:"inner" , &self.inner.inner.as_ref().map(|(s: &Pin<&mut Peekable>, _f: &NextIfEqFn<'_, T, impl Debug>)| s)) |
| 390 | .finish() |
| 391 | } |
| 392 | } |
| 393 | |
| 394 | impl<St, T> FusedFuture for NextIfEq<'_, St, T> |
| 395 | where |
| 396 | St: Stream, |
| 397 | T: ?Sized, |
| 398 | St::Item: PartialEq<T>, |
| 399 | { |
| 400 | fn is_terminated(&self) -> bool { |
| 401 | self.inner.is_terminated() |
| 402 | } |
| 403 | } |
| 404 | |
| 405 | impl<St, T> Future for NextIfEq<'_, St, T> |
| 406 | where |
| 407 | St: Stream, |
| 408 | T: ?Sized, |
| 409 | St::Item: PartialEq<T>, |
| 410 | { |
| 411 | type Output = Option<St::Item>; |
| 412 | |
| 413 | fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> { |
| 414 | self.project().inner.poll(cx) |
| 415 | } |
| 416 | } |
| 417 | |
| 418 | struct NextIfEqFn<'a, T: ?Sized, Item> { |
| 419 | expected: &'a T, |
| 420 | _next: PhantomData<Item>, |
| 421 | } |
| 422 | |
| 423 | impl<T, Item> FnOnce1<&Item> for NextIfEqFn<'_, T, Item> |
| 424 | where |
| 425 | T: ?Sized, |
| 426 | Item: PartialEq<T>, |
| 427 | { |
| 428 | type Output = bool; |
| 429 | |
| 430 | fn call_once(self, next: &Item) -> Self::Output { |
| 431 | next == self.expected |
| 432 | } |
| 433 | } |
| 434 | |