2019-05-12 06:03:50 -07:00
|
|
|
use std::convert::Infallible;
|
2019-11-14 18:38:24 +06:00
|
|
|
use std::future::Future;
|
|
|
|
use std::pin::Pin;
|
|
|
|
use std::task::{Context, Poll};
|
2019-05-12 06:03:50 -07:00
|
|
|
|
|
|
|
use actix_service::{IntoService, Service, Transform};
|
2019-11-14 18:38:24 +06:00
|
|
|
use futures::future::{ok, Ready};
|
2018-09-14 13:30:29 -07:00
|
|
|
|
|
|
|
use super::counter::{Counter, CounterGuard};
|
|
|
|
|
|
|
|
/// InFlight - new service for service that can limit number of in-flight
|
|
|
|
/// async requests.
|
|
|
|
///
|
|
|
|
/// Default number of in-flight requests is 15
|
2019-02-03 10:42:27 -08:00
|
|
|
pub struct InFlight {
|
2018-09-14 13:30:29 -07:00
|
|
|
max_inflight: usize,
|
|
|
|
}
|
|
|
|
|
2019-02-03 10:42:27 -08:00
|
|
|
impl InFlight {
|
|
|
|
pub fn new(max: usize) -> Self {
|
|
|
|
Self { max_inflight: max }
|
2018-09-14 13:30:29 -07:00
|
|
|
}
|
2019-02-03 10:42:27 -08:00
|
|
|
}
|
2018-09-14 13:30:29 -07:00
|
|
|
|
2019-02-03 10:42:27 -08:00
|
|
|
impl Default for InFlight {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self::new(15)
|
2018-09-14 13:30:29 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-18 18:28:54 +06:00
|
|
|
impl<S> Transform<S> for InFlight
|
|
|
|
where
|
|
|
|
S: Service,
|
|
|
|
{
|
2019-03-09 07:27:35 -08:00
|
|
|
type Request = S::Request;
|
2019-03-04 19:38:11 -08:00
|
|
|
type Response = S::Response;
|
|
|
|
type Error = S::Error;
|
2019-05-12 06:03:50 -07:00
|
|
|
type InitError = Infallible;
|
2019-03-04 19:38:11 -08:00
|
|
|
type Transform = InFlightService<S>;
|
2019-11-14 18:38:24 +06:00
|
|
|
type Future = Ready<Result<Self::Transform, Self::InitError>>;
|
2018-09-14 13:30:29 -07:00
|
|
|
|
2019-03-04 19:38:11 -08:00
|
|
|
fn new_transform(&self, service: S) -> Self::Future {
|
|
|
|
ok(InFlightService::new(self.max_inflight, service))
|
2018-09-14 13:30:29 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-04 19:38:11 -08:00
|
|
|
pub struct InFlightService<S> {
|
2018-09-14 13:30:29 -07:00
|
|
|
count: Counter,
|
2019-03-04 19:38:11 -08:00
|
|
|
service: S,
|
2018-09-14 13:30:29 -07:00
|
|
|
}
|
|
|
|
|
2019-03-12 13:39:04 -07:00
|
|
|
impl<S> InFlightService<S>
|
|
|
|
where
|
|
|
|
S: Service,
|
|
|
|
{
|
|
|
|
pub fn new<U>(max: usize, service: U) -> Self
|
|
|
|
where
|
|
|
|
U: IntoService<S>,
|
|
|
|
{
|
2018-09-14 13:30:29 -07:00
|
|
|
Self {
|
|
|
|
count: Counter::new(max),
|
2019-03-12 13:39:04 -07:00
|
|
|
service: service.into_service(),
|
2018-09-14 13:30:29 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-09 07:27:35 -08:00
|
|
|
impl<T> Service for InFlightService<T>
|
2018-11-29 16:56:15 -10:00
|
|
|
where
|
2019-03-09 07:27:35 -08:00
|
|
|
T: Service,
|
2018-11-29 16:56:15 -10:00
|
|
|
{
|
2019-03-09 07:27:35 -08:00
|
|
|
type Request = T::Request;
|
2018-09-14 13:30:29 -07:00
|
|
|
type Response = T::Response;
|
|
|
|
type Error = T::Error;
|
2019-03-09 07:27:35 -08:00
|
|
|
type Future = InFlightServiceResponse<T>;
|
2018-09-14 13:30:29 -07:00
|
|
|
|
2019-12-02 22:30:09 +06:00
|
|
|
fn poll_ready(&mut self, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
|
2019-11-14 18:38:24 +06:00
|
|
|
if let Poll::Pending = self.service.poll_ready(cx)? {
|
|
|
|
Poll::Pending
|
|
|
|
} else if !self.count.available(cx) {
|
2019-02-01 15:15:53 -08:00
|
|
|
log::trace!("InFlight limit exceeded");
|
2019-11-14 18:38:24 +06:00
|
|
|
Poll::Pending
|
2019-02-03 10:42:27 -08:00
|
|
|
} else {
|
2019-11-14 18:38:24 +06:00
|
|
|
Poll::Ready(Ok(()))
|
2018-09-14 13:30:29 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-09 07:27:35 -08:00
|
|
|
fn call(&mut self, req: T::Request) -> Self::Future {
|
2018-09-14 13:30:29 -07:00
|
|
|
InFlightServiceResponse {
|
2019-03-04 19:38:11 -08:00
|
|
|
fut: self.service.call(req),
|
2018-11-29 16:56:15 -10:00
|
|
|
_guard: self.count.get(),
|
2018-09-14 13:30:29 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[doc(hidden)]
|
2019-11-19 14:51:40 +06:00
|
|
|
#[pin_project::pin_project]
|
2019-03-09 07:27:35 -08:00
|
|
|
pub struct InFlightServiceResponse<T: Service> {
|
2019-11-19 14:51:40 +06:00
|
|
|
#[pin]
|
2018-09-14 13:30:29 -07:00
|
|
|
fut: T::Future,
|
2018-11-29 16:56:15 -10:00
|
|
|
_guard: CounterGuard,
|
2018-09-14 13:30:29 -07:00
|
|
|
}
|
|
|
|
|
2019-11-19 14:51:40 +06:00
|
|
|
impl<T: Service> Future for InFlightServiceResponse<T> {
|
2019-11-14 18:38:24 +06:00
|
|
|
type Output = Result<T::Response, T::Error>;
|
2018-09-14 13:30:29 -07:00
|
|
|
|
2019-12-02 22:30:09 +06:00
|
|
|
fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
|
2019-11-19 14:51:40 +06:00
|
|
|
self.project().fut.poll(cx)
|
2018-09-14 13:30:29 -07:00
|
|
|
}
|
|
|
|
}
|
2019-02-03 10:42:27 -08:00
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
|
2019-11-14 18:38:24 +06:00
|
|
|
use std::task::{Context, Poll};
|
2019-02-03 10:42:27 -08:00
|
|
|
use std::time::Duration;
|
|
|
|
|
|
|
|
use super::*;
|
2019-11-14 18:38:24 +06:00
|
|
|
use actix_service::{apply, factory_fn, Service, ServiceFactory};
|
|
|
|
use futures::future::{lazy, ok, FutureExt, LocalBoxFuture};
|
2019-02-03 10:42:27 -08:00
|
|
|
|
|
|
|
struct SleepService(Duration);
|
|
|
|
|
2019-03-09 07:27:35 -08:00
|
|
|
impl Service for SleepService {
|
|
|
|
type Request = ();
|
2019-02-03 10:42:27 -08:00
|
|
|
type Response = ();
|
|
|
|
type Error = ();
|
2019-11-14 18:38:24 +06:00
|
|
|
type Future = LocalBoxFuture<'static, Result<(), ()>>;
|
2019-02-03 10:42:27 -08:00
|
|
|
|
2019-12-02 22:30:09 +06:00
|
|
|
fn poll_ready(&mut self, _: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
|
2019-11-14 18:38:24 +06:00
|
|
|
Poll::Ready(Ok(()))
|
2019-02-03 10:42:27 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
fn call(&mut self, _: ()) -> Self::Future {
|
2019-11-26 08:26:22 +06:00
|
|
|
actix_rt::time::delay_for(self.0)
|
2019-11-14 18:38:24 +06:00
|
|
|
.then(|_| ok::<_, ()>(()))
|
|
|
|
.boxed_local()
|
2019-02-03 10:42:27 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-25 21:49:11 +06:00
|
|
|
#[actix_rt::test]
|
|
|
|
async fn test_transform() {
|
2019-02-03 10:42:27 -08:00
|
|
|
let wait_time = Duration::from_millis(50);
|
|
|
|
|
2019-11-25 21:49:11 +06:00
|
|
|
let mut srv = InFlightService::new(1, SleepService(wait_time));
|
|
|
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
2019-02-03 10:42:27 -08:00
|
|
|
|
2019-11-25 21:49:11 +06:00
|
|
|
let res = srv.call(());
|
|
|
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Pending);
|
|
|
|
|
|
|
|
let _ = res.await;
|
|
|
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
2019-02-03 10:42:27 -08:00
|
|
|
}
|
|
|
|
|
2019-11-25 21:49:11 +06:00
|
|
|
#[actix_rt::test]
|
|
|
|
async fn test_newtransform() {
|
2019-02-03 10:42:27 -08:00
|
|
|
let wait_time = Duration::from_millis(50);
|
|
|
|
|
2019-11-25 21:49:11 +06:00
|
|
|
let srv = apply(InFlight::new(1), factory_fn(|| ok(SleepService(wait_time))));
|
2019-02-03 10:42:27 -08:00
|
|
|
|
2019-11-25 21:49:11 +06:00
|
|
|
let mut srv = srv.new_service(&()).await.unwrap();
|
|
|
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
2019-02-03 10:42:27 -08:00
|
|
|
|
2019-11-25 21:49:11 +06:00
|
|
|
let res = srv.call(());
|
|
|
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Pending);
|
2019-02-03 10:42:27 -08:00
|
|
|
|
2019-11-25 21:49:11 +06:00
|
|
|
let _ = res.await;
|
|
|
|
assert_eq!(lazy(|cx| srv.poll_ready(cx)).await, Poll::Ready(Ok(())));
|
2019-02-03 10:42:27 -08:00
|
|
|
}
|
|
|
|
}
|