stream.rs 3.35 KB
Newer Older
Ryan Olson's avatar
Ryan Olson committed
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
// SPDX-FileCopyrightText: Copyright (c) 2024-2025 NVIDIA CORPORATION & AFFILIATES. All rights reserved.
// SPDX-License-Identifier: Apache-2.0
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

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

use tokio::time::{self, sleep_until, Duration, Instant, Sleep};

pub struct DeadlineStream<S> {
    stream: S,
27
    sleep: Pin<Box<Sleep>>,
Ryan Olson's avatar
Ryan Olson committed
28
29
30
31
32
33
}

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>> {
34
        // Check if we've passed the deadline
35
        if Pin::new(&mut self.sleep).poll(cx).is_ready() {
Ryan Olson's avatar
Ryan Olson committed
36
37
38
39
40
41
42
43
44
45
            // The deadline expired; end the stream now
            return Poll::Ready(None);
        }

        // Otherwise, poll the underlying stream
        self.as_mut().stream.poll_next_unpin(cx)
    }
}

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

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

    use super::*;

60
61
62
    // 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);
63
64
65
66
67
68
69
        let stream = stream.then(|x| {
            let sleep = time::sleep(Duration::from_millis(x));
            async move {
                sleep.await;
                x
            }
        });
70
71

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

74
75
        pin!(stream);
        let mut stream = until_deadline(stream, deadline);
76

77
78
79
        while let Some(x) = stream.next().await {
            result.push(x);
        }
80
81
82
83
84
85
86
87
88
89
90
91

        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
92
        assert_eq!(result, vec![100, 100]);
Ryan Olson's avatar
Ryan Olson committed
93
    }
94
95
96
97
98
99
100
101
102
103
104

    #[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
105
}