use std::marker::PhantomData; use std::net::SocketAddr; use std::task::{Context, Poll}; use actix_service::{Service, ServiceFactory as BaseServiceFactory}; use actix_utils::{ counter::CounterGuard, future::{ready, Ready}, }; use futures_core::future::LocalBoxFuture; use log::error; use crate::socket::{FromStream, MioStream}; use crate::Token; pub trait ServiceFactory: Send + Clone + 'static { type Factory: BaseServiceFactory; fn create(&self) -> Self::Factory; } pub(crate) trait InternalServiceFactory: Send { fn name(&self, token: Token) -> &str; fn clone_factory(&self) -> Box; fn create(&self) -> LocalBoxFuture<'static, Result<(Token, BoxedServerService), ()>>; } pub(crate) type BoxedServerService = Box< dyn Service< (CounterGuard, MioStream), Response = (), Error = (), Future = Ready>, >, >; pub(crate) struct StreamService { service: S, _phantom: PhantomData, } impl StreamService { pub(crate) fn new(service: S) -> Self { StreamService { service, _phantom: PhantomData, } } } impl Service<(CounterGuard, MioStream)> for StreamService where S: Service, S::Future: 'static, S::Error: 'static, I: FromStream, { type Response = (); type Error = (); type Future = Ready>; fn poll_ready(&self, ctx: &mut Context<'_>) -> Poll> { self.service.poll_ready(ctx).map_err(|_| ()) } fn call(&self, (guard, req): (CounterGuard, MioStream)) -> Self::Future { ready(match FromStream::from_mio(req) { Ok(stream) => { let f = self.service.call(stream); actix_rt::spawn(async move { let _ = f.await; drop(guard); }); Ok(()) } Err(e) => { error!("Can not convert to an async tcp stream: {}", e); Err(()) } }) } } pub(crate) struct StreamNewService, Io: FromStream> { name: String, inner: F, token: Token, addr: SocketAddr, _t: PhantomData, } impl StreamNewService where F: ServiceFactory, Io: FromStream + Send + 'static, { pub(crate) fn create( name: String, token: Token, inner: F, addr: SocketAddr, ) -> Box { Box::new(Self { name, token, inner, addr, _t: PhantomData, }) } } impl InternalServiceFactory for StreamNewService where F: ServiceFactory, Io: FromStream + Send + 'static, { fn name(&self, _: Token) -> &str { &self.name } fn clone_factory(&self) -> Box { Box::new(Self { name: self.name.clone(), inner: self.inner.clone(), token: self.token, addr: self.addr, _t: PhantomData, }) } fn create(&self) -> LocalBoxFuture<'static, Result<(Token, BoxedServerService), ()>> { let token = self.token; let fut = self.inner.create().new_service(()); Box::pin(async move { match fut.await { Ok(inner) => { let service = Box::new(StreamService::new(inner)) as _; Ok((token, service)) } Err(_) => Err(()), } }) } } impl ServiceFactory for F where F: Fn() -> T + Send + Clone + 'static, T: BaseServiceFactory, I: FromStream, { type Factory = T; fn create(&self) -> T { (self)() } }