stream.rs 3.16 KB
Newer Older
Ryan Olson's avatar
Ryan Olson committed
1
2
3
4
5
6
7
8
9
10
// SPDX-FileCopyrightText: Copyright (c) 2024-2025 NVIDIA CORPORATION & AFFILIATES. All rights reserved.
// SPDX-License-Identifier: Apache-2.0

use futures::stream::{Stream, StreamExt};
use std::{
    future::Future,
    pin::Pin,
    task::{Context, Poll},
};

11
use tokio::time::{self, Duration, Instant, Sleep, sleep_until};
Ryan Olson's avatar
Ryan Olson committed
12
13
14

pub struct DeadlineStream<S> {
    stream: S,
15
    sleep: Pin<Box<Sleep>>,
Ryan Olson's avatar
Ryan Olson committed
16
17
18
19
20
21
}

impl<S: Stream + Unpin> Stream for DeadlineStream<S> {
    type Item = S::Item;

    fn poll_next(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<Self::Item>> {
22
        // Check if we've passed the deadline
23
        if Pin::new(&mut self.sleep).poll(cx).is_ready() {
Ryan Olson's avatar
Ryan Olson committed
24
25
26
27
28
            // The deadline expired; end the stream now
            return Poll::Ready(None);
        }

        // Otherwise, poll the underlying stream
29
30
31
32
33
34
35
36
        let val = self.as_mut().stream.poll_next_unpin(cx);
        // Log the poll result and return it
        match &val {
            Poll::Ready(Some(_)) => tracing::trace!("DeadlineStream: received item"),
            Poll::Ready(None) => tracing::trace!("DeadlineStream: underlying stream ended"),
            Poll::Pending => tracing::trace!("DeadlineStream: waiting for next item"),
        }
        val
Ryan Olson's avatar
Ryan Olson committed
37
38
39
40
    }
}

pub fn until_deadline<S: Stream + Unpin>(stream: S, deadline: Instant) -> DeadlineStream<S> {
41
42
43
44
45
    DeadlineStream {
        stream,
        // Set an async task that sleeps until deadline and wakes up to cancel the stream
        sleep: Box::pin(sleep_until(deadline)),
    }
46
47
48
49
50
51
52
53
54
}

#[cfg(test)]
mod tests {
    use futures::stream::{self, Stream, StreamExt};
    use tokio::pin;

    use super::*;

55
56
57
    // Helper function to run the deadline stream test with given parameters
    async fn run_deadline_test(sleep_times_ms: Vec<u64>, deadline_ms: u64) -> Vec<u64> {
        let stream = stream::iter(sleep_times_ms);
58
59
60
61
62
63
64
        let stream = stream.then(|x| {
            let sleep = time::sleep(Duration::from_millis(x));
            async move {
                sleep.await;
                x
            }
        });
65
66

        let deadline = Instant::now() + Duration::from_millis(deadline_ms);
67
        let mut result = Vec::new();
68

69
70
        pin!(stream);
        let mut stream = until_deadline(stream, deadline);
71

72
73
74
        while let Some(x) = stream.next().await {
            result.push(x);
        }
75
76
77
78
79
80
81
82
83
84
85
86

        result
    }

    #[tokio::test]
    async fn test_deadline_exceeded() {
        // The sum of the sleep times should exceed the deadline
        let sleep_times_ms = vec![100, 100, 200, 50];
        let deadline_ms = 300;

        let result = run_deadline_test(sleep_times_ms, deadline_ms).await;
        // Since deadline is exceeded, only the items before deadline should be returned
87
        assert_eq!(result, vec![100, 100]);
Ryan Olson's avatar
Ryan Olson committed
88
    }
89
90
91
92
93
94
95
96
97
98
99

    #[tokio::test]
    async fn test_complete_before_deadline() {
        // The sum of the sleep times should be less than the deadline
        let sleep_times_ms = vec![100, 50, 50];
        let deadline_ms = 300;

        let result = run_deadline_test(sleep_times_ms, deadline_ms).await;
        // Since deadline is not exceeded, all items should be returned from stream
        assert_eq!(result, vec![100, 50, 50]);
    }
Ryan Olson's avatar
Ryan Olson committed
100
}