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
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
use futures_io::{self as io, AsyncBufRead, AsyncRead, AsyncWrite};
use pin_utils::{unsafe_pinned, unsafe_unpinned};
use std::{
cmp,
pin::Pin,
task::{Context, Poll},
};
#[derive(Debug)]
pub struct Limited<Io> {
io: Io,
limit: usize,
}
impl<Io: Unpin> Unpin for Limited<Io> {}
impl<Io> Limited<Io> {
unsafe_pinned!(io: Io);
unsafe_unpinned!(limit: usize);
pub(crate) fn new(io: Io, limit: usize) -> Limited<Io> {
Limited { io, limit }
}
pub fn get_ref(&self) -> &Io {
&self.io
}
pub fn get_mut(&mut self) -> &mut Io {
&mut self.io
}
pub fn get_pin_mut<'a>(self: Pin<&'a mut Self>) -> Pin<&'a mut Io> {
self.io()
}
pub fn into_inner(self) -> Io {
self.io
}
}
impl<W: AsyncWrite> AsyncWrite for Limited<W> {
fn poll_write(
mut self: Pin<&mut Self>,
cx: &mut Context<'_>,
buf: &[u8],
) -> Poll<io::Result<usize>> {
let limit = *self.as_mut().limit();
self.io().poll_write(cx, &buf[..cmp::min(limit, buf.len())])
}
fn poll_flush(
self: Pin<&mut Self>,
cx: &mut Context<'_>,
) -> Poll<io::Result<()>> {
self.io().poll_flush(cx)
}
fn poll_close(
self: Pin<&mut Self>,
cx: &mut Context<'_>,
) -> Poll<io::Result<()>> {
self.io().poll_close(cx)
}
}
impl<R: AsyncRead> AsyncRead for Limited<R> {
fn poll_read(
mut self: Pin<&mut Self>,
cx: &mut Context<'_>,
buf: &mut [u8],
) -> Poll<io::Result<usize>> {
let limit = cmp::min(*self.as_mut().limit(), buf.len());
self.io().poll_read(cx, &mut buf[..limit])
}
}
impl<R: AsyncBufRead> AsyncBufRead for Limited<R> {
fn poll_fill_buf<'a>(
self: Pin<&'a mut Self>,
cx: &mut Context<'_>,
) -> Poll<io::Result<&'a [u8]>> {
self.io().poll_fill_buf(cx)
}
fn consume(self: Pin<&mut Self>, amount: usize) {
self.io().consume(amount)
}
}