1
0
Fork 0
mirror of https://github.com/actix/actix-web.git synced 2025-03-10 21:41:07 +00:00
actix-web/src/server_service.rs

132 lines
3.4 KiB
Rust
Raw Normal View History

2018-08-21 05:21:23 +00:00
use std::sync::{
atomic::{AtomicUsize, Ordering},
Arc,
};
2018-08-19 17:47:04 +00:00
use std::{fmt, io, net};
use futures::{future, Future, Poll};
use tokio_reactor::Handle;
use tokio_tcp::TcpStream;
use tower_service::{NewService, Service};
pub(crate) type BoxedServerService = Box<
Service<
Request = net::TcpStream,
Response = (),
Error = (),
Future = Box<Future<Item = (), Error = ()>>,
>,
>;
pub(crate) struct ServerService<T> {
inner: T,
2018-08-21 05:21:23 +00:00
counter: Arc<AtomicUsize>,
2018-08-19 17:47:04 +00:00
}
impl<T> Service for ServerService<T>
where
T: Service<Request = TcpStream, Response = ()>,
T::Future: 'static,
T::Error: fmt::Display + 'static,
{
type Request = net::TcpStream;
type Response = ();
type Error = ();
type Future = Box<Future<Item = (), Error = ()>>;
fn poll_ready(&mut self) -> Poll<(), Self::Error> {
self.inner.poll_ready().map_err(|_| ())
}
fn call(&mut self, stream: net::TcpStream) -> Self::Future {
let stream = TcpStream::from_std(stream, &Handle::default()).map_err(|e| {
error!("Can not convert to an async tcp stream: {}", e);
});
if let Ok(stream) = stream {
2018-08-21 05:21:23 +00:00
let counter = self.counter.clone();
let _ = counter.fetch_add(1, Ordering::Relaxed);
Box::new(self.inner.call(stream).map_err(|_| ()).map(move |_| {
let _ = counter.fetch_sub(1, Ordering::Relaxed);
}))
2018-08-19 17:47:04 +00:00
} else {
Box::new(future::err(()))
}
}
}
pub(crate) struct ServerNewService<T> {
inner: T,
2018-08-21 05:21:23 +00:00
counter: Arc<AtomicUsize>,
2018-08-19 17:47:04 +00:00
}
impl<T> ServerNewService<T>
where
T: NewService<Request = TcpStream, Response = (), InitError = io::Error>
+ Clone
+ Send
+ 'static,
T::Service: 'static,
T::Future: 'static,
T::Error: fmt::Display,
{
pub(crate) fn create(inner: T) -> Box<ServerServiceFactory + Send> {
2018-08-21 05:21:23 +00:00
Box::new(Self {
inner,
counter: Arc::new(AtomicUsize::new(0)),
})
2018-08-19 17:47:04 +00:00
}
}
pub trait ServerServiceFactory {
2018-08-21 05:21:23 +00:00
fn counter(&self) -> Arc<AtomicUsize>;
2018-08-19 17:47:04 +00:00
fn clone_factory(&self) -> Box<ServerServiceFactory + Send>;
fn create(&self) -> Box<Future<Item = BoxedServerService, Error = ()>>;
}
impl<T> ServerServiceFactory for ServerNewService<T>
where
T: NewService<Request = TcpStream, Response = (), InitError = io::Error>
+ Clone
+ Send
+ 'static,
T::Service: 'static,
T::Future: 'static,
T::Error: fmt::Display,
{
2018-08-21 05:21:23 +00:00
fn counter(&self) -> Arc<AtomicUsize> {
self.counter.clone()
}
2018-08-19 17:47:04 +00:00
fn clone_factory(&self) -> Box<ServerServiceFactory + Send> {
Box::new(Self {
inner: self.inner.clone(),
2018-08-21 05:21:23 +00:00
counter: Arc::new(AtomicUsize::new(0)),
2018-08-19 17:47:04 +00:00
})
}
fn create(&self) -> Box<Future<Item = BoxedServerService, Error = ()>> {
2018-08-21 05:21:23 +00:00
let counter = self.counter.clone();
Box::new(self.inner.new_service().map_err(|_| ()).map(move |inner| {
let service: BoxedServerService = Box::new(ServerService { inner, counter });
2018-08-19 17:47:04 +00:00
service
}))
}
}
impl ServerServiceFactory for Box<ServerServiceFactory> {
2018-08-21 05:21:23 +00:00
fn counter(&self) -> Arc<AtomicUsize> {
self.as_ref().counter()
}
2018-08-19 17:47:04 +00:00
fn clone_factory(&self) -> Box<ServerServiceFactory + Send> {
self.as_ref().clone_factory()
}
fn create(&self) -> Box<Future<Item = BoxedServerService, Error = ()>> {
self.as_ref().create()
}
}