1use std::{
4 cell::{Ref, RefCell, RefMut},
5 io::{self, Read, Write},
6 pin::Pin,
7 rc::Rc,
8 str::FromStr,
9 task::{Context, Poll},
10};
11
12use actix_codec::{AsyncRead, AsyncWrite, ReadBuf};
13use bytes::{Bytes, BytesMut};
14use http::{header, Method, Uri, Version};
15
16use crate::{
17 header::{HeaderMap, TryIntoHeaderPair},
18 payload::Payload,
19 Request,
20};
21
22pub struct TestRequest(Option<Inner>);
24
25struct Inner {
26 version: Version,
27 method: Method,
28 uri: Uri,
29 headers: HeaderMap,
30 payload: Option<Payload>,
31}
32
33impl Default for TestRequest {
34 fn default() -> TestRequest {
35 TestRequest(Some(Inner {
36 method: Method::GET,
37 uri: Uri::from_str("/").unwrap(),
38 version: Version::HTTP_11,
39 headers: HeaderMap::new(),
40 payload: None,
41 }))
42 }
43}
44
45impl TestRequest {
46 pub fn with_uri(path: &str) -> TestRequest {
48 TestRequest::default().uri(path).take()
49 }
50
51 pub fn version(&mut self, ver: Version) -> &mut Self {
53 parts(&mut self.0).version = ver;
54 self
55 }
56
57 pub fn method(&mut self, meth: Method) -> &mut Self {
59 parts(&mut self.0).method = meth;
60 self
61 }
62
63 pub fn uri(&mut self, path: &str) -> &mut Self {
68 parts(&mut self.0).uri = Uri::from_str(path).unwrap();
69 self
70 }
71
72 pub fn insert_header(&mut self, header: impl TryIntoHeaderPair) -> &mut Self {
74 match header.try_into_pair() {
75 Ok((key, value)) => {
76 parts(&mut self.0).headers.insert(key, value);
77 }
78 Err(err) => {
79 panic!("Error inserting test header: {}.", err.into());
80 }
81 }
82
83 self
84 }
85
86 pub fn append_header(&mut self, header: impl TryIntoHeaderPair) -> &mut Self {
88 match header.try_into_pair() {
89 Ok((key, value)) => {
90 parts(&mut self.0).headers.append(key, value);
91 }
92 Err(err) => {
93 panic!("Error inserting test header: {}.", err.into());
94 }
95 }
96
97 self
98 }
99
100 pub fn set_payload(&mut self, data: impl Into<Bytes>) -> &mut Self {
104 let mut payload = crate::h1::Payload::empty();
105 let bytes = data.into();
106 self.insert_header((header::CONTENT_LENGTH, bytes.len()));
107 payload.unread_data(bytes);
108 parts(&mut self.0).payload = Some(payload.into());
109 self
110 }
111
112 pub fn take(&mut self) -> TestRequest {
113 TestRequest(self.0.take())
114 }
115
116 pub fn finish(&mut self) -> Request {
118 let inner = self.0.take().expect("cannot reuse test request builder");
119
120 let mut req = if let Some(pl) = inner.payload {
121 Request::with_payload(pl)
122 } else {
123 Request::with_payload(crate::h1::Payload::empty().into())
124 };
125
126 let head = req.head_mut();
127 head.uri = inner.uri;
128 head.method = inner.method;
129 head.version = inner.version;
130 head.headers = inner.headers;
131
132 req
133 }
134}
135
136#[inline]
137fn parts(parts: &mut Option<Inner>) -> &mut Inner {
138 parts.as_mut().expect("cannot reuse test request builder")
139}
140
141#[derive(Debug)]
143pub struct TestBuffer {
144 pub read_buf: Rc<RefCell<BytesMut>>,
145 pub write_buf: Rc<RefCell<BytesMut>>,
146 pub err: Option<Rc<io::Error>>,
147}
148
149impl TestBuffer {
150 pub fn new<T>(data: T) -> Self
152 where
153 T: Into<BytesMut>,
154 {
155 Self {
156 read_buf: Rc::new(RefCell::new(data.into())),
157 write_buf: Rc::new(RefCell::new(BytesMut::new())),
158 err: None,
159 }
160 }
161
162 #[allow(dead_code)]
164 pub(crate) fn clone(&self) -> Self {
165 Self {
166 read_buf: Rc::clone(&self.read_buf),
167 write_buf: Rc::clone(&self.write_buf),
168 err: self.err.clone(),
169 }
170 }
171
172 pub fn empty() -> Self {
174 Self::new("")
175 }
176
177 #[allow(dead_code)]
178 pub(crate) fn read_buf_slice(&self) -> Ref<'_, [u8]> {
179 Ref::map(self.read_buf.borrow(), |b| b.as_ref())
180 }
181
182 #[allow(dead_code)]
183 pub(crate) fn read_buf_slice_mut(&self) -> RefMut<'_, [u8]> {
184 RefMut::map(self.read_buf.borrow_mut(), |b| b.as_mut())
185 }
186
187 #[allow(dead_code)]
188 pub(crate) fn write_buf_slice(&self) -> Ref<'_, [u8]> {
189 Ref::map(self.write_buf.borrow(), |b| b.as_ref())
190 }
191
192 #[allow(dead_code)]
193 pub(crate) fn write_buf_slice_mut(&self) -> RefMut<'_, [u8]> {
194 RefMut::map(self.write_buf.borrow_mut(), |b| b.as_mut())
195 }
196
197 #[allow(dead_code)]
198 pub(crate) fn take_write_buf(&self) -> Bytes {
199 self.write_buf.borrow_mut().split().freeze()
200 }
201
202 pub fn extend_read_buf<T: AsRef<[u8]>>(&mut self, data: T) {
204 self.read_buf.borrow_mut().extend_from_slice(data.as_ref())
205 }
206}
207
208impl io::Read for TestBuffer {
209 fn read(&mut self, dst: &mut [u8]) -> Result<usize, io::Error> {
210 if self.read_buf.borrow().is_empty() {
211 if self.err.is_some() {
212 Err(Rc::try_unwrap(self.err.take().unwrap()).unwrap())
213 } else {
214 Err(io::Error::new(io::ErrorKind::WouldBlock, ""))
215 }
216 } else {
217 let size = std::cmp::min(self.read_buf.borrow().len(), dst.len());
218 let b = self.read_buf.borrow_mut().split_to(size);
219 dst[..size].copy_from_slice(&b);
220 Ok(size)
221 }
222 }
223}
224
225impl io::Write for TestBuffer {
226 fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
227 self.write_buf.borrow_mut().extend(buf);
228 Ok(buf.len())
229 }
230
231 fn flush(&mut self) -> io::Result<()> {
232 Ok(())
233 }
234}
235
236impl AsyncRead for TestBuffer {
237 fn poll_read(
238 self: Pin<&mut Self>,
239 _: &mut Context<'_>,
240 buf: &mut ReadBuf<'_>,
241 ) -> Poll<io::Result<()>> {
242 let dst = buf.initialize_unfilled();
243 let res = self.get_mut().read(dst).map(|n| buf.advance(n));
244 Poll::Ready(res)
245 }
246}
247
248impl AsyncWrite for TestBuffer {
249 fn poll_write(
250 self: Pin<&mut Self>,
251 _: &mut Context<'_>,
252 buf: &[u8],
253 ) -> Poll<io::Result<usize>> {
254 Poll::Ready(self.get_mut().write(buf))
255 }
256
257 fn poll_flush(self: Pin<&mut Self>, _: &mut Context<'_>) -> Poll<io::Result<()>> {
258 Poll::Ready(Ok(()))
259 }
260
261 fn poll_shutdown(self: Pin<&mut Self>, _: &mut Context<'_>) -> Poll<io::Result<()>> {
262 Poll::Ready(Ok(()))
263 }
264}
265
266#[derive(Clone)]
268pub struct TestSeqBuffer(Rc<RefCell<TestSeqInner>>);
269
270impl TestSeqBuffer {
271 pub fn new<T>(data: T) -> Self
273 where
274 T: Into<BytesMut>,
275 {
276 Self(Rc::new(RefCell::new(TestSeqInner {
277 read_buf: data.into(),
278 write_buf: BytesMut::new(),
279 err: None,
280 })))
281 }
282
283 pub fn empty() -> Self {
285 Self::new(BytesMut::new())
286 }
287
288 pub fn read_buf(&self) -> Ref<'_, BytesMut> {
289 Ref::map(self.0.borrow(), |inner| &inner.read_buf)
290 }
291
292 pub fn write_buf(&self) -> Ref<'_, BytesMut> {
293 Ref::map(self.0.borrow(), |inner| &inner.write_buf)
294 }
295
296 pub fn err(&self) -> Ref<'_, Option<io::Error>> {
297 Ref::map(self.0.borrow(), |inner| &inner.err)
298 }
299
300 pub fn extend_read_buf<T: AsRef<[u8]>>(&mut self, data: T) {
302 self.0
303 .borrow_mut()
304 .read_buf
305 .extend_from_slice(data.as_ref())
306 }
307}
308
309pub struct TestSeqInner {
310 read_buf: BytesMut,
311 write_buf: BytesMut,
312 err: Option<io::Error>,
313}
314
315impl io::Read for TestSeqBuffer {
316 fn read(&mut self, dst: &mut [u8]) -> Result<usize, io::Error> {
317 if self.0.borrow().read_buf.is_empty() {
318 if self.0.borrow().err.is_some() {
319 Err(self.0.borrow_mut().err.take().unwrap())
320 } else {
321 Err(io::Error::new(io::ErrorKind::WouldBlock, ""))
322 }
323 } else {
324 let size = std::cmp::min(self.0.borrow().read_buf.len(), dst.len());
325 let b = self.0.borrow_mut().read_buf.split_to(size);
326 dst[..size].copy_from_slice(&b);
327 Ok(size)
328 }
329 }
330}
331
332impl io::Write for TestSeqBuffer {
333 fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
334 self.0.borrow_mut().write_buf.extend(buf);
335 Ok(buf.len())
336 }
337
338 fn flush(&mut self) -> io::Result<()> {
339 Ok(())
340 }
341}
342
343impl AsyncRead for TestSeqBuffer {
344 fn poll_read(
345 self: Pin<&mut Self>,
346 _: &mut Context<'_>,
347 buf: &mut ReadBuf<'_>,
348 ) -> Poll<io::Result<()>> {
349 let dst = buf.initialize_unfilled();
350 let r = self.get_mut().read(dst);
351 match r {
352 Ok(n) => {
353 buf.advance(n);
354 Poll::Ready(Ok(()))
355 }
356 Err(err) if err.kind() == io::ErrorKind::WouldBlock => Poll::Pending,
357 Err(err) => Poll::Ready(Err(err)),
358 }
359 }
360}
361
362impl AsyncWrite for TestSeqBuffer {
363 fn poll_write(
364 self: Pin<&mut Self>,
365 _: &mut Context<'_>,
366 buf: &[u8],
367 ) -> Poll<io::Result<usize>> {
368 Poll::Ready(self.get_mut().write(buf))
369 }
370
371 fn poll_flush(self: Pin<&mut Self>, _: &mut Context<'_>) -> Poll<io::Result<()>> {
372 Poll::Ready(Ok(()))
373 }
374
375 fn poll_shutdown(self: Pin<&mut Self>, _: &mut Context<'_>) -> Poll<io::Result<()>> {
376 Poll::Ready(Ok(()))
377 }
378}