1
0
mirror of https://github.com/fafhrd91/actix-net synced 2025-08-14 10:00:31 +02:00

Compare commits

...

51 Commits

Author SHA1 Message Date
Nikolay Kim
ed14e6b8ea change to IntoFuture 2019-03-04 20:29:35 -08:00
Nikolay Kim
700abc997e prepare actix-utils release 2019-03-04 19:45:17 -08:00
Nikolay Kim
8c48bf4de7 simplify transform trait 2019-03-04 19:38:11 -08:00
Nikolay Kim
9bc492cf6c add SslError 2019-03-04 16:16:39 -08:00
Nikolay Kim
d2a223e69e update changes 2019-03-04 15:42:25 -08:00
Nikolay Kim
e9657a399a add maxconnrate 2019-03-04 15:41:16 -08:00
Nikolay Kim
9f25fdf929 rename StreamServiceFactory to ServiceFactory 2019-03-04 14:31:46 -08:00
Nikolay Kim
82930de8e7 use default type for RouterBuilder 2019-03-04 14:03:46 -08:00
Nikolay Kim
04a3e59bd5 update tests 2019-03-04 12:41:39 -08:00
Nikolay Kim
0ff0daa795 allow custom checks for resource selection 2019-03-04 11:47:03 -08:00
Nikolay Kim
fb43940824 allow empty pattern 2019-03-03 21:00:58 -08:00
Nikolay Kim
0410f59cf5 prep release 2019-03-02 14:55:22 -08:00
Nikolay Kim
400023a07b prepare actix-connector release 2019-03-02 14:47:52 -08:00
Nikolay Kim
2e4c84dbb6 prepare actix-server release 2019-03-02 14:42:31 -08:00
Nikolay Kim
672c3936a6 prepare actix-utils release 2019-03-02 14:30:32 -08:00
Nikolay Kim
fbf4444b04 prep release 2019-03-02 14:22:03 -08:00
Nikolay Kim
b5b3168b34 do not use void for now 2019-03-02 13:49:21 -08:00
Nikolay Kim
668e4f9ac4 update utils; add NewTransform::map_init_err 2019-03-02 13:18:01 -08:00
Nikolay Kim
d0b8b6940c add configuration parameter to transform factory 2019-03-02 12:16:30 -08:00
Nikolay Kim
f1bc9d0deb update tests 2019-02-22 18:47:29 -08:00
Nikolay Kim
6ed020565c update service crate 2019-02-22 18:31:25 -08:00
Nikolay Kim
7ee33efdfd moved boxed service and new service to actix-service 2019-02-22 18:20:54 -08:00
Nikolay Kim
83f51b28d7 allow to customize cfg parameter for FnNewService 2019-02-22 17:12:26 -08:00
Nikolay Kim
83a19e9cb3 add IntoConfigurableNewService 2019-02-22 14:30:00 -08:00
Nikolay Kim
6b4010892d add IntoNewService for FnNewService 2019-02-22 14:19:43 -08:00
Nikolay Kim
d2bd9134aa add fn service helpers 2019-02-22 14:13:48 -08:00
Nikolay Kim
a0e2d926e6 add fn_nservice 2019-02-22 13:20:52 -08:00
Nikolay Kim
43d2dd473f update tests 2019-02-22 13:08:31 -08:00
Nikolay Kim
862be49e30 add Config argument to NewService 2019-02-22 12:44:37 -08:00
Nikolay Kim
6ea128fac5 Custom BoxedNewService implementation 2019-02-21 11:19:28 -08:00
Nikolay Kim
a97d7f0ccf add BoxedNewService and BoxedService 2019-02-21 10:41:39 -08:00
Nikolay Kim
3d7daabdd7 add NewService impls for Rc<S> and Arc<S> 2019-02-19 11:31:54 -08:00
Nikolay Kim
32f4718880 Add Display impl for InOrderError 2019-02-11 08:39:28 -08:00
Nikolay Kim
b8f9bf4bc8 Add Display impl for TimeoutError 2019-02-11 08:34:57 -08:00
Nikolay Kim
e354c6df92 Drop service response 2019-02-09 21:39:17 -08:00
Nikolay Kim
a53f06a1a4 allow register router resource for ResourceDef 2019-02-09 20:43:39 -08:00
Nikolay Kim
9979bfb3ef rename Pattern to ResourceDef 2019-02-09 07:24:35 -08:00
Nikolay Kim
17d0f84f63 prep actix-utils 2019-02-06 11:40:22 -08:00
Nikolay Kim
08bc328826 clippy warnings 2019-02-04 11:04:10 -08:00
Nikolay Kim
7dca264546 move transform map_err combinator to separate module 2019-02-04 10:55:39 -08:00
Nikolay Kim
3bddba5da5 helper method 2019-02-03 14:12:15 -08:00
Nikolay Kim
4be025926c add InOrder service 2019-02-03 14:05:13 -08:00
Nikolay Kim
0063a26aab prepare actix-service release 2019-02-03 13:32:51 -08:00
Nikolay Kim
bcc466f6ab update tests 2019-02-03 11:48:11 -08:00
Nikolay Kim
663ae53954 fix Clone impl for Timeout 2019-02-03 11:37:34 -08:00
Nikolay Kim
406088524e depend on git repo 2019-02-03 11:33:26 -08:00
Nikolay Kim
5b8446105f depend on repo 2019-02-03 11:16:24 -08:00
Nikolay Kim
429ad453d3 change Apply::new_fn to old args order 2019-02-03 10:52:44 -08:00
Nikolay Kim
bd977373bc generalize apply combinator with transform trait 2019-02-03 10:42:27 -08:00
Nikolay Kim
d45fb9521f update deps 2019-02-01 20:11:30 -08:00
Nikolay Kim
94a0da3659 prepare actix-connector release 2019-02-01 20:06:53 -08:00
51 changed files with 2347 additions and 974 deletions

View File

@@ -1,18 +1,3 @@
[package]
name = "actix-net"
version = "0.3.0"
authors = ["Nikolay Kim <fafhrd91@gmail.com>"]
description = "Actix net - framework for the compisible network services for Rust (experimental)"
readme = "README.md"
keywords = ["network", "framework", "async", "futures"]
homepage = "https://actix.rs"
repository = "https://github.com/actix/actix-net.git"
documentation = "https://docs.rs/actix-net/"
categories = ["network-programming", "asynchronous"]
license = "MIT/Apache-2.0"
exclude = [".gitignore", ".travis.yml", ".cargo/config", "appveyor.yml"]
edition = "2018"
[workspace] [workspace]
members = [ members = [
"actix-codec", "actix-codec",
@@ -24,13 +9,3 @@ members = [
"actix-utils", "actix-utils",
"router", "router",
] ]
[dev-dependencies]
actix-service = "0.2.0"
actix-codec = "0.1.0"
actix-rt = { path="actix-rt" }
actix-server = { path="actix-server", features=["ssl"] }
env_logger = "0.5"
futures = "0.1.24"
openssl = { version="0.10" }
tokio-openssl = { version="0.3" }

View File

@@ -1,14 +1,21 @@
# Changes # Changes
## [0.2.0] - 2019-01-xx ## [0.3.0] - 2019-03-02
### Changes ### Changed
* Migrate to actix-service 0.3
## [0.2.0] - 2019-02-01
### Changed
* Migrate to actix-service 0.2 * Migrate to actix-service 0.2
* Upgrade trust-dns-resolver * Upgrade trust-dns-resolver
* Use tokio-current-thread instead of diract actix-rt dipendency * Use tokio-current-thread instead of direct actix-rt dipendency
## [0.1.1] - 2019-01-13 ## [0.1.1] - 2019-01-13

View File

@@ -1,6 +1,6 @@
[package] [package]
name = "actix-connector" name = "actix-connector"
version = "0.2.0" version = "0.3.0"
authors = ["Nikolay Kim <fafhrd91@gmail.com>"] authors = ["Nikolay Kim <fafhrd91@gmail.com>"]
description = "Actix Connector - tcp connector service" description = "Actix Connector - tcp connector service"
keywords = ["network", "framework", "async", "futures"] keywords = ["network", "framework", "async", "futures"]
@@ -11,7 +11,7 @@ categories = ["network-programming", "asynchronous"]
license = "MIT/Apache-2.0" license = "MIT/Apache-2.0"
exclude = [".gitignore", ".travis.yml", ".cargo/config", "appveyor.yml"] exclude = [".gitignore", ".travis.yml", ".cargo/config", "appveyor.yml"]
edition = "2018" edition = "2018"
workspace = "../" workspace = ".."
[package.metadata.docs.rs] [package.metadata.docs.rs]
features = ["ssl"] features = ["ssl"]
@@ -27,8 +27,7 @@ default = []
ssl = ["openssl", "tokio-openssl"] ssl = ["openssl", "tokio-openssl"]
[dependencies] [dependencies]
#actix-service = "0.1.6" actix-service = "0.3.0"
actix-service = { path="../actix-service" }
actix-codec = "0.1.0" actix-codec = "0.1.0"
futures = "0.1" futures = "0.1"
tokio-tcp = "0.1" tokio-tcp = "0.1"

View File

@@ -4,8 +4,8 @@ use std::net::{IpAddr, SocketAddr};
use std::time::Duration; use std::time::Duration;
use std::{fmt, io}; use std::{fmt, io};
use actix_service::{NewService, Service}; use actix_service::{fn_factory, NewService, Service};
use futures::future::{ok, Either, FutureResult}; use futures::future::{ok, Either};
use futures::{try_ready, Async, Future, Poll}; use futures::{try_ready, Async, Future, Poll};
use tokio_tcp::{ConnectFuture, TcpStream}; use tokio_tcp::{ConnectFuture, TcpStream};
use trust_dns_resolver::config::{ResolverConfig, ResolverOpts}; use trust_dns_resolver::config::{ResolverConfig, ResolverOpts};
@@ -177,12 +177,13 @@ impl Connector {
cfg: ResolverConfig, cfg: ResolverConfig,
opts: ResolverOpts, opts: ResolverOpts,
) -> impl NewService< ) -> impl NewService<
(),
Request = Connect, Request = Connect,
Response = (Connect, TcpStream), Response = (Connect, TcpStream),
Error = ConnectorError, Error = ConnectorError,
InitError = E, InitError = E,
> + Clone { > + Clone {
move || -> FutureResult<Connector, E> { ok(Connector::new(cfg.clone(), opts)) } fn_factory(move || ok(Connector::new(cfg.clone(), opts)))
} }
} }

View File

@@ -44,7 +44,9 @@ impl<R, T, E> Clone for OpensslConnector<R, T, E> {
} }
} }
impl<R: RequestHost, T: AsyncRead + AsyncWrite, E> NewService for OpensslConnector<R, T, E> { impl<R: RequestHost, T: AsyncRead + AsyncWrite, E> NewService<()>
for OpensslConnector<R, T, E>
{
type Request = (R, T); type Request = (R, T);
type Response = (R, SslStream<T>); type Response = (R, SslStream<T>);
type Error = HandshakeError<T>; type Error = HandshakeError<T>;
@@ -52,7 +54,7 @@ impl<R: RequestHost, T: AsyncRead + AsyncWrite, E> NewService for OpensslConnect
type InitError = E; type InitError = E;
type Future = FutureResult<Self::Service, Self::InitError>; type Future = FutureResult<Self::Service, Self::InitError>;
fn new_service(&self) -> Self::Future { fn new_service(&self, _: &()) -> Self::Future {
ok(OpensslConnectorService { ok(OpensslConnectorService {
connector: self.connector.clone(), connector: self.connector.clone(),
_t: PhantomData, _t: PhantomData,

View File

@@ -1,8 +1,38 @@
# Changes # Changes
## [0.3.1] - 2019-03-xx
### Added
* Add `ServerBuilder::maxconnrate` sets the maximum per-worker number of concurrent connections
* Add helper ssl error `SslError`
### Changed
* Rename `StreamServiceFactory` to `ServiceFactory`
* Deprecate `StreamServiceFactory`
## [0.3.0] - 2019-03-02
### Changed
* Use new `NewService` trait
## [0.2.1] - 2019-02-09
### Changed
* Drop service response
## [0.2.0] - 2019-02-01 ## [0.2.0] - 2019-02-01
## Changes ### Changed
* Migrate to actix-service 0.2 * Migrate to actix-service 0.2
@@ -11,14 +41,14 @@
## [0.1.3] - 2018-12-21 ## [0.1.3] - 2018-12-21
## Fixed ### Fixed
* Fix max concurrent connections handling * Fix max concurrent connections handling
## [0.1.2] - 2018-12-12 ## [0.1.2] - 2018-12-12
## Changed ### Changed
* rename ServiceConfig::rt() to ServiceConfig::apply() * rename ServiceConfig::rt() to ServiceConfig::apply()

View File

@@ -1,6 +1,6 @@
[package] [package]
name = "actix-server" name = "actix-server"
version = "0.2.0" version = "0.3.0"
authors = ["Nikolay Kim <fafhrd91@gmail.com>"] authors = ["Nikolay Kim <fafhrd91@gmail.com>"]
description = "Actix server - General purpose tcp server" description = "Actix server - General purpose tcp server"
keywords = ["network", "framework", "async", "futures"] keywords = ["network", "framework", "async", "futures"]
@@ -11,7 +11,7 @@ categories = ["network-programming", "asynchronous"]
license = "MIT/Apache-2.0" license = "MIT/Apache-2.0"
exclude = [".gitignore", ".travis.yml", ".cargo/config", "appveyor.yml"] exclude = [".gitignore", ".travis.yml", ".cargo/config", "appveyor.yml"]
edition = "2018" edition = "2018"
workspace = "../" workspace = ".."
[package.metadata.docs.rs] [package.metadata.docs.rs]
features = ["ssl", "tls", "rust-tls"] features = ["ssl", "tls", "rust-tls"]
@@ -33,16 +33,14 @@ ssl = ["openssl", "tokio-openssl"]
rust-tls = ["rustls", "tokio-rustls", "webpki", "webpki-roots"] rust-tls = ["rustls", "tokio-rustls", "webpki", "webpki-roots"]
[dependencies] [dependencies]
actix-service = "0.2.0" actix-service = "0.3.0"
actix-rt = "0.1.0" actix-rt = "0.1.0"
log = "0.4" log = "0.4"
num_cpus = "1.0" num_cpus = "1.0"
# io
mio = "^0.6.13" mio = "^0.6.13"
net2 = "0.2" net2 = "0.2"
bytes = "0.4"
futures = "0.1" futures = "0.1"
slab = "0.4" slab = "0.4"
tokio-io = "0.1" tokio-io = "0.1"

View File

@@ -14,11 +14,10 @@ use tokio_timer::sleep;
use crate::accept::{AcceptLoop, AcceptNotify, Command}; use crate::accept::{AcceptLoop, AcceptNotify, Command};
use crate::config::{ConfiguredService, ServiceConfig}; use crate::config::{ConfiguredService, ServiceConfig};
use crate::server::{Server, ServerCommand}; use crate::server::{Server, ServerCommand};
use crate::services::{InternalServiceFactory, StreamNewService, StreamServiceFactory}; use crate::services::{InternalServiceFactory, ServiceFactory, StreamNewService};
use crate::services::{ServiceFactory, ServiceNewService};
use crate::signals::{Signal, Signals}; use crate::signals::{Signal, Signals};
use crate::worker::{self, Worker, WorkerAvailability, WorkerClient}; use crate::worker::{self, Worker, WorkerAvailability, WorkerClient};
use crate::Token; use crate::{ssl, Token};
/// Server builder /// Server builder
pub struct ServerBuilder { pub struct ServerBuilder {
@@ -82,9 +81,18 @@ impl ServerBuilder {
self self
} }
/// Stop actix system. /// Sets the maximum per-worker concurrent connection establish process.
/// ///
/// `SystemExit` message stops currently running system. /// All listeners will stop accepting connections when this limit is reached. It
/// can be used to limit the global SSL CPU usage.
///
/// By default max connections is set to a 256.
pub fn maxconnrate(self, num: usize) -> Self {
ssl::max_concurrent_ssl_connect(num);
self
}
/// Stop actix system.
pub fn system_exit(mut self) -> Self { pub fn system_exit(mut self) -> Self {
self.exit = true; self.exit = true;
self self
@@ -138,7 +146,7 @@ impl ServerBuilder {
/// Add new service to the server. /// Add new service to the server.
pub fn bind<F, U, N: AsRef<str>>(mut self, name: N, addr: U, factory: F) -> io::Result<Self> pub fn bind<F, U, N: AsRef<str>>(mut self, name: N, addr: U, factory: F) -> io::Result<Self>
where where
F: StreamServiceFactory, F: ServiceFactory,
U: net::ToSocketAddrs, U: net::ToSocketAddrs,
{ {
let sockets = bind_addr(addr)?; let sockets = bind_addr(addr)?;
@@ -163,31 +171,11 @@ impl ServerBuilder {
lst: net::TcpListener, lst: net::TcpListener,
factory: F, factory: F,
) -> Self ) -> Self
where
F: StreamServiceFactory,
{
let token = self.token.next();
self.services.push(StreamNewService::create(
name.as_ref().to_string(),
token,
factory,
));
self.sockets.push((token, lst));
self
}
/// Add new service to the server.
pub fn listen2<F, N: AsRef<str>>(
mut self,
name: N,
lst: net::TcpListener,
factory: F,
) -> Self
where where
F: ServiceFactory, F: ServiceFactory,
{ {
let token = self.token.next(); let token = self.token.next();
self.services.push(ServiceNewService::create( self.services.push(StreamNewService::create(
name.as_ref().to_string(), name.as_ref().to_string(),
token, token,
factory, factory,
@@ -204,9 +192,6 @@ impl ServerBuilder {
/// This methods panics if no socket addresses get bound. /// This methods panics if no socket addresses get bound.
/// ///
/// ```rust,ignore /// ```rust,ignore
/// # extern crate futures;
/// # extern crate actix_web;
/// # use futures::Future;
/// use actix_web::*; /// use actix_web::*;
/// ///
/// fn main() { /// fn main() {

View File

@@ -114,7 +114,7 @@ impl InternalServiceFactory for ConfiguredService {
// construct services // construct services
let mut fut = Vec::new(); let mut fut = Vec::new();
for (token, ns) in rt.services { for (token, ns) in rt.services {
fut.push(ns.new_service().map(move |service| (token, service))); fut.push(ns.new_service(&()).map(move |service| (token, service)));
} }
Box::new(join_all(fut).map_err(|e| { Box::new(join_all(fut).map_err(|e| {
@@ -219,8 +219,8 @@ where
type Service = BoxedServerService; type Service = BoxedServerService;
type Future = Box<Future<Item = BoxedServerService, Error = ()>>; type Future = Box<Future<Item = BoxedServerService, Error = ()>>;
fn new_service(&self) -> Self::Future { fn new_service(&self, _: &()) -> Self::Future {
Box::new(self.inner.new_service().map_err(|_| ()).map(|s| { Box::new(self.inner.new_service(&()).map_err(|_| ()).map(|s| {
let service: BoxedServerService = Box::new(StreamService::new(s)); let service: BoxedServerService = Box::new(StreamService::new(s));
service service
})) }))

View File

@@ -13,7 +13,10 @@ mod worker;
pub use self::builder::ServerBuilder; pub use self::builder::ServerBuilder;
pub use self::config::{ServiceConfig, ServiceRuntime}; pub use self::config::{ServiceConfig, ServiceRuntime};
pub use self::server::Server; pub use self::server::Server;
pub use self::services::{ServerMessage, ServiceFactory, StreamServiceFactory}; pub use self::services::ServiceFactory;
#[doc(hidden)]
pub use self::services::ServiceFactory as StreamServiceFactory;
/// Socket id token /// Socket id token
#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)] #[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)]
@@ -28,6 +31,6 @@ impl Token {
} }
/// Start server building process /// Start server building process
pub fn build() -> ServerBuilder { pub fn new() -> ServerBuilder {
ServerBuilder::default() ServerBuilder::default()
} }

View File

@@ -13,7 +13,7 @@ use super::Token;
use crate::counter::CounterGuard; use crate::counter::CounterGuard;
/// Server message /// Server message
pub enum ServerMessage { pub(crate) enum ServerMessage {
/// New stream /// New stream
Connect(net::TcpStream), Connect(net::TcpStream),
/// Gracefull shutdown /// Gracefull shutdown
@@ -22,14 +22,8 @@ pub enum ServerMessage {
ForceShutdown, ForceShutdown,
} }
pub trait StreamServiceFactory: Send + Clone + 'static {
type NewService: NewService<Request = TcpStream, Response = ()>;
fn create(&self) -> Self::NewService;
}
pub trait ServiceFactory: Send + Clone + 'static { pub trait ServiceFactory: Send + Clone + 'static {
type NewService: NewService<Request = ServerMessage, Response = ()>; type NewService: NewService<Request = TcpStream>;
fn create(&self) -> Self::NewService; fn create(&self) -> Self::NewService;
} }
@@ -63,7 +57,7 @@ impl<T> StreamService<T> {
impl<T> Service for StreamService<T> impl<T> Service for StreamService<T>
where where
T: Service<Request = TcpStream, Response = ()>, T: Service<Request = TcpStream>,
T::Future: 'static, T::Future: 'static,
T::Error: 'static, T::Error: 'static,
{ {
@@ -86,7 +80,7 @@ where
if let Ok(stream) = stream { if let Ok(stream) = stream {
spawn(self.service.call(stream).then(move |res| { spawn(self.service.call(stream).then(move |res| {
drop(guard); drop(guard);
res.map_err(|_| ()) res.map_err(|_| ()).map(|_| ())
})); }));
ok(()) ok(())
} else { } else {
@@ -98,87 +92,7 @@ where
} }
} }
pub(crate) struct ServerService<T> { pub(crate) struct StreamNewService<F: ServiceFactory> {
service: T,
}
impl<T> ServerService<T> {
fn new(service: T) -> Self {
ServerService { service }
}
}
impl<T> Service for ServerService<T>
where
T: Service<Request = ServerMessage, Response = ()>,
T::Future: 'static,
T::Error: 'static,
{
type Request = (Option<CounterGuard>, ServerMessage);
type Response = ();
type Error = ();
type Future = FutureResult<(), ()>;
fn poll_ready(&mut self) -> Poll<(), Self::Error> {
self.service.poll_ready().map_err(|_| ())
}
fn call(&mut self, (guard, req): (Option<CounterGuard>, ServerMessage)) -> Self::Future {
spawn(self.service.call(req).then(move |res| {
drop(guard);
res.map_err(|_| ())
}));
ok(())
}
}
pub(crate) struct ServiceNewService<F: ServiceFactory> {
name: String,
inner: F,
token: Token,
}
impl<F> ServiceNewService<F>
where
F: ServiceFactory,
{
pub(crate) fn create(name: String, token: Token, inner: F) -> Box<InternalServiceFactory> {
Box::new(Self { name, inner, token })
}
}
impl<F> InternalServiceFactory for ServiceNewService<F>
where
F: ServiceFactory,
{
fn name(&self, _: Token) -> &str {
&self.name
}
fn clone_factory(&self) -> Box<InternalServiceFactory> {
Box::new(Self {
name: self.name.clone(),
inner: self.inner.clone(),
token: self.token,
})
}
fn create(&self) -> Box<Future<Item = Vec<(Token, BoxedServerService)>, Error = ()>> {
let token = self.token;
Box::new(
self.inner
.create()
.new_service()
.map_err(|_| ())
.map(move |inner| {
let service: BoxedServerService = Box::new(ServerService::new(inner));
vec![(token, service)]
}),
)
}
}
pub(crate) struct StreamNewService<F: StreamServiceFactory> {
name: String, name: String,
inner: F, inner: F,
token: Token, token: Token,
@@ -186,7 +100,7 @@ pub(crate) struct StreamNewService<F: StreamServiceFactory> {
impl<F> StreamNewService<F> impl<F> StreamNewService<F>
where where
F: StreamServiceFactory, F: ServiceFactory,
{ {
pub(crate) fn create(name: String, token: Token, inner: F) -> Box<InternalServiceFactory> { pub(crate) fn create(name: String, token: Token, inner: F) -> Box<InternalServiceFactory> {
Box::new(Self { name, token, inner }) Box::new(Self { name, token, inner })
@@ -195,7 +109,7 @@ where
impl<F> InternalServiceFactory for StreamNewService<F> impl<F> InternalServiceFactory for StreamNewService<F>
where where
F: StreamServiceFactory, F: ServiceFactory,
{ {
fn name(&self, _: Token) -> &str { fn name(&self, _: Token) -> &str {
&self.name &self.name
@@ -214,7 +128,7 @@ where
Box::new( Box::new(
self.inner self.inner
.create() .create()
.new_service() .new_service(&())
.map_err(|_| ()) .map_err(|_| ())
.map(move |inner| { .map(move |inner| {
let service: BoxedServerService = Box::new(StreamService::new(inner)); let service: BoxedServerService = Box::new(StreamService::new(inner));
@@ -241,19 +155,7 @@ impl InternalServiceFactory for Box<InternalServiceFactory> {
impl<F, T> ServiceFactory for F impl<F, T> ServiceFactory for F
where where
F: Fn() -> T + Send + Clone + 'static, F: Fn() -> T + Send + Clone + 'static,
T: NewService<Request = ServerMessage, Response = ()>, T: NewService<Request = TcpStream>,
{
type NewService = T;
fn create(&self) -> T {
(self)()
}
}
impl<F, T> StreamServiceFactory for F
where
F: Fn() -> T + Send + Clone + 'static,
T: NewService<Request = TcpStream, Response = ()>,
{ {
type NewService = T; type NewService = T;

View File

@@ -33,3 +33,9 @@ pub(crate) static MAX_CONN: AtomicUsize = AtomicUsize::new(256);
thread_local! { thread_local! {
static MAX_CONN_COUNTER: Counter = Counter::new(MAX_CONN.load(Ordering::Relaxed)); static MAX_CONN_COUNTER: Counter = Counter::new(MAX_CONN.load(Ordering::Relaxed));
} }
/// Ssl error combinded with service error.
pub enum SslError<E1, E2> {
Ssl(E1),
Service(E2),
}

View File

@@ -44,7 +44,7 @@ impl<T: AsyncRead + AsyncWrite> NewService for NativeTlsAcceptor<T> {
type InitError = (); type InitError = ();
type Future = FutureResult<Self::Service, Self::InitError>; type Future = FutureResult<Self::Service, Self::InitError>;
fn new_service(&self) -> Self::Future { fn new_service(&self, _: &()) -> Self::Future {
MAX_CONN_COUNTER.with(|conns| { MAX_CONN_COUNTER.with(|conns| {
ok(NativeTlsAcceptorService { ok(NativeTlsAcceptorService {
acceptor: self.acceptor.clone(), acceptor: self.acceptor.clone(),

View File

@@ -44,7 +44,7 @@ impl<T: AsyncRead + AsyncWrite> NewService for OpensslAcceptor<T> {
type InitError = (); type InitError = ();
type Future = FutureResult<Self::Service, Self::InitError>; type Future = FutureResult<Self::Service, Self::InitError>;
fn new_service(&self) -> Self::Future { fn new_service(&self, _: &()) -> Self::Future {
MAX_CONN_COUNTER.with(|conns| { MAX_CONN_COUNTER.with(|conns| {
ok(OpensslAcceptorService { ok(OpensslAcceptorService {
acceptor: self.acceptor.clone(), acceptor: self.acceptor.clone(),

View File

@@ -46,7 +46,7 @@ impl<T: AsyncRead + AsyncWrite> NewService for RustlsAcceptor<T> {
type InitError = (); type InitError = ();
type Future = FutureResult<Self::Service, Self::InitError>; type Future = FutureResult<Self::Service, Self::InitError>;
fn new_service(&self) -> Self::Future { fn new_service(&self, _: &()) -> Self::Future {
MAX_CONN_COUNTER.with(|conns| { MAX_CONN_COUNTER.with(|conns| {
ok(RustlsAcceptorService { ok(RustlsAcceptorService {
acceptor: self.config.clone().into(), acceptor: self.config.clone().into(),

View File

@@ -1,5 +1,50 @@
# Changes # Changes
## [0.3.2] - 2019-03-04
### Changed
* Change `NewService::Future` and `Transform::Future` to the `IntoFuture` trait.
* Export `AndThenTransform` type
## [0.3.1] - 2019-03-04
### Changed
* Simplify Transform trait
## [0.3.0] - 2019-03-02
## Added
* Added boxed NewService and Service.
## Changed
* Added `Config` parameter to `NewService` trait.
* Added `Config` parameter to `NewTransform` trait.
## [0.2.2] - 2019-02-19
### Added
* Added `NewService` impl for `Rc<S> where S: NewService`
* Added `NewService` impl for `Arc<S> where S: NewService`
## [0.2.1] - 2019-02-03
### Changed
* Generalize `.apply` combinator with Transform trait
## [0.2.0] - 2019-02-01 ## [0.2.0] - 2019-02-01
### Changed ### Changed

View File

@@ -1,6 +1,6 @@
[package] [package]
name = "actix-service" name = "actix-service"
version = "0.2.0" version = "0.3.2"
authors = ["Nikolay Kim <fafhrd91@gmail.com>"] authors = ["Nikolay Kim <fafhrd91@gmail.com>"]
description = "Actix Service" description = "Actix Service"
keywords = ["network", "framework", "async", "futures"] keywords = ["network", "framework", "async", "futures"]
@@ -11,7 +11,7 @@ categories = ["network-programming", "asynchronous"]
license = "MIT/Apache-2.0" license = "MIT/Apache-2.0"
exclude = [".gitignore", ".travis.yml", ".cargo/config", "appveyor.yml"] exclude = [".gitignore", ".travis.yml", ".cargo/config", "appveyor.yml"]
edition = "2018" edition = "2018"
workspace = "../" workspace = ".."
[badges] [badges]
travis-ci = { repository = "actix/actix-service", branch = "master" } travis-ci = { repository = "actix/actix-service", branch = "master" }
@@ -24,3 +24,4 @@ path = "src/lib.rs"
[dependencies] [dependencies]
futures = "0.1.24" futures = "0.1.24"
void = "1.0.2"

View File

@@ -1,4 +1,6 @@
use futures::{try_ready, Async, Future, Poll}; use std::marker::PhantomData;
use futures::{try_ready, Async, Future, IntoFuture, Poll};
use super::{IntoNewService, NewService, Service}; use super::{IntoNewService, NewService, Service};
use crate::cell::Cell; use crate::cell::Cell;
@@ -105,29 +107,31 @@ where
} }
/// `AndThenNewService` new service combinator /// `AndThenNewService` new service combinator
pub struct AndThenNewService<A, B> { pub struct AndThenNewService<A, B, C> {
a: A, a: A,
b: B, b: B,
_t: PhantomData<C>,
} }
impl<A, B> AndThenNewService<A, B> { impl<A, B, C> AndThenNewService<A, B, C> {
/// Create new `AndThen` combinator /// Create new `AndThen` combinator
pub fn new<F: IntoNewService<B>>(a: A, f: F) -> Self pub fn new<F: IntoNewService<B, C>>(a: A, f: F) -> Self
where where
A: NewService, A: NewService<C>,
B: NewService<Request = A::Response, Error = A::Error, InitError = A::InitError>, B: NewService<C, Request = A::Response, Error = A::Error, InitError = A::InitError>,
{ {
Self { Self {
a, a,
b: f.into_new_service(), b: f.into_new_service(),
_t: PhantomData,
} }
} }
} }
impl<A, B> NewService for AndThenNewService<A, B> impl<A, B, C> NewService<C> for AndThenNewService<A, B, C>
where where
A: NewService, A: NewService<C>,
B: NewService<Request = A::Response, Error = A::Error, InitError = A::InitError>, B: NewService<C, Request = A::Response, Error = A::Error, InitError = A::InitError>,
{ {
type Request = A::Request; type Request = A::Request;
type Response = B::Response; type Response = B::Response;
@@ -135,14 +139,17 @@ where
type Service = AndThen<A::Service, B::Service>; type Service = AndThen<A::Service, B::Service>;
type InitError = A::InitError; type InitError = A::InitError;
type Future = AndThenNewServiceFuture<A, B>; type Future = AndThenNewServiceFuture<A, B, C>;
fn new_service(&self) -> Self::Future { fn new_service(&self, cfg: &C) -> Self::Future {
AndThenNewServiceFuture::new(self.a.new_service(), self.b.new_service()) AndThenNewServiceFuture::new(
self.a.new_service(cfg).into_future(),
self.b.new_service(cfg).into_future(),
)
} }
} }
impl<A, B> Clone for AndThenNewService<A, B> impl<A, B, C> Clone for AndThenNewService<A, B, C>
where where
A: Clone, A: Clone,
B: Clone, B: Clone,
@@ -151,27 +158,31 @@ where
Self { Self {
a: self.a.clone(), a: self.a.clone(),
b: self.b.clone(), b: self.b.clone(),
_t: PhantomData,
} }
} }
} }
pub struct AndThenNewServiceFuture<A, B> pub struct AndThenNewServiceFuture<A, B, C>
where where
A: NewService, A: NewService<C>,
B: NewService<Request = A::Response>, B: NewService<C, Request = A::Response>,
{ {
fut_b: B::Future, fut_b: <B::Future as IntoFuture>::Future,
fut_a: A::Future, fut_a: <A::Future as IntoFuture>::Future,
a: Option<A::Service>, a: Option<A::Service>,
b: Option<B::Service>, b: Option<B::Service>,
} }
impl<A, B> AndThenNewServiceFuture<A, B> impl<A, B, C> AndThenNewServiceFuture<A, B, C>
where where
A: NewService, A: NewService<C>,
B: NewService<Request = A::Response>, B: NewService<C, Request = A::Response>,
{ {
fn new(fut_a: A::Future, fut_b: B::Future) -> Self { fn new(
fut_a: <A::Future as IntoFuture>::Future,
fut_b: <B::Future as IntoFuture>::Future,
) -> Self {
AndThenNewServiceFuture { AndThenNewServiceFuture {
fut_a, fut_a,
fut_b, fut_b,
@@ -181,10 +192,10 @@ where
} }
} }
impl<A, B> Future for AndThenNewServiceFuture<A, B> impl<A, B, C> Future for AndThenNewServiceFuture<A, B, C>
where where
A: NewService, A: NewService<C>,
B: NewService<Request = A::Response, Error = A::Error, InitError = A::InitError>, B: NewService<C, Request = A::Response, Error = A::Error, InitError = A::InitError>,
{ {
type Item = AndThen<A::Service, B::Service>; type Item = AndThen<A::Service, B::Service>;
type Error = A::InitError; type Error = A::InitError;
@@ -286,7 +297,7 @@ mod tests {
let new_srv = blank let new_srv = blank
.into_new_service() .into_new_service()
.and_then(move || Ok(Srv2(cnt.clone()))); .and_then(move || Ok(Srv2(cnt.clone())));
if let Async::Ready(mut srv) = new_srv.new_service().poll().unwrap() { if let Async::Ready(mut srv) = new_srv.new_service(&()).poll().unwrap() {
let res = srv.call("srv1").poll(); let res = srv.call("srv1").poll();
assert!(res.is_ok()); assert!(res.is_ok());
assert_eq!(res.unwrap(), Async::Ready(("srv1", "srv2"))); assert_eq!(res.unwrap(), Async::Ready(("srv1", "srv2")));

View File

@@ -1,161 +1,38 @@
use std::marker::PhantomData; use std::rc::Rc;
use futures::{try_ready, Async, Future, IntoFuture, Poll}; use futures::{Async, Future, IntoFuture, Poll};
use super::{IntoNewService, IntoService, NewService, Service}; use crate::and_then::AndThen;
use crate::cell::Cell; use crate::from_err::FromErr;
use crate::{NewService, Transform};
/// `Apply` service combinator /// `Apply` new service combinator
pub struct AndThenApply<A, B, F, Out> pub struct AndThenTransform<T, A, B, C> {
where
A: Service,
B: Service<Error = A::Error>,
F: FnMut(A::Response, &mut B) -> Out,
Out: IntoFuture,
Out::Error: Into<A::Error>,
{
a: A,
b: Cell<B>,
f: Cell<F>,
r: PhantomData<(Out,)>,
}
impl<A, B, F, Out> AndThenApply<A, B, F, Out>
where
A: Service,
B: Service<Error = A::Error>,
F: FnMut(A::Response, &mut B) -> Out,
Out: IntoFuture,
Out::Error: Into<A::Error>,
{
/// Create new `Apply` combinator
pub fn new<A1: IntoService<A>, B1: IntoService<B>>(a: A1, b: B1, f: F) -> Self {
Self {
f: Cell::new(f),
a: a.into_service(),
b: Cell::new(b.into_service()),
r: PhantomData,
}
}
}
impl<A, B, F, Out> Clone for AndThenApply<A, B, F, Out>
where
A: Service + Clone,
B: Service<Error = A::Error>,
F: FnMut(A::Response, &mut B) -> Out,
Out: IntoFuture,
Out::Error: Into<A::Error>,
{
fn clone(&self) -> Self {
AndThenApply {
a: self.a.clone(),
b: self.b.clone(),
f: self.f.clone(),
r: PhantomData,
}
}
}
impl<A, B, F, Out> Service for AndThenApply<A, B, F, Out>
where
A: Service,
B: Service<Error = A::Error>,
F: FnMut(A::Response, &mut B) -> Out,
Out: IntoFuture,
Out::Error: Into<A::Error>,
{
type Request = A::Request;
type Response = Out::Item;
type Error = A::Error;
type Future = AndThenApplyFuture<A, B, F, Out>;
fn poll_ready(&mut self) -> Poll<(), Self::Error> {
try_ready!(self.a.poll_ready());
self.b.get_mut().poll_ready().map_err(|e| e.into())
}
fn call(&mut self, req: A::Request) -> Self::Future {
AndThenApplyFuture {
b: self.b.clone(),
f: self.f.clone(),
fut_b: None,
fut_a: Some(self.a.call(req)),
}
}
}
pub struct AndThenApplyFuture<A, B, F, Out>
where
A: Service,
B: Service<Error = A::Error>,
F: FnMut(A::Response, &mut B) -> Out,
Out: IntoFuture,
Out::Error: Into<A::Error>,
{
b: Cell<B>,
f: Cell<F>,
fut_a: Option<A::Future>,
fut_b: Option<Out::Future>,
}
impl<A, B, F, Out> Future for AndThenApplyFuture<A, B, F, Out>
where
A: Service,
B: Service<Error = A::Error>,
F: FnMut(A::Response, &mut B) -> Out,
Out: IntoFuture,
Out::Error: Into<A::Error>,
{
type Item = Out::Item;
type Error = A::Error;
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
if let Some(ref mut fut) = self.fut_b {
return fut.poll().map_err(|e| e.into());
}
match self.fut_a.as_mut().expect("Bug in actix-service").poll() {
Ok(Async::Ready(resp)) => {
let _ = self.fut_a.take();
self.fut_b =
Some((&mut *self.f.get_mut())(resp, self.b.get_mut()).into_future());
self.poll()
}
Ok(Async::NotReady) => Ok(Async::NotReady),
Err(err) => Err(err.into()),
}
}
}
/// `ApplyNewService` new service combinator
pub struct AndThenApplyNewService<A, B, F, Out> {
a: A, a: A,
b: B, b: B,
f: Cell<F>, t: Rc<T>,
r: PhantomData<(Out)>, _t: std::marker::PhantomData<C>,
} }
impl<A, B, F, Out> AndThenApplyNewService<A, B, F, Out> impl<T, A, B, C> AndThenTransform<T, A, B, C>
where where
A: NewService, A: NewService<C>,
B: NewService<Error = A::Error, InitError = A::InitError>, B: NewService<C, InitError = A::InitError>,
F: FnMut(A::Response, &mut B::Service) -> Out, T: Transform<B::Service, Request = A::Response, InitError = A::InitError>,
Out: IntoFuture, T::Error: From<A::Error>,
Out::Error: Into<A::Error>,
{ {
/// Create new `ApplyNewService` new service instance /// Create new `ApplyNewService` new service instance
pub fn new<A1: IntoNewService<A>, B1: IntoNewService<B>>(a: A1, b: B1, f: F) -> Self { pub fn new(t: T, a: A, b: B) -> Self {
Self { Self {
f: Cell::new(f), a,
a: a.into_new_service(), b,
b: b.into_new_service(), t: Rc::new(t),
r: PhantomData, _t: std::marker::PhantomData,
} }
} }
} }
impl<A, B, F, Out> Clone for AndThenApplyNewService<A, B, F, Out> impl<T, A, B, C> Clone for AndThenTransform<T, A, B, C>
where where
A: Clone, A: Clone,
B: Clone, B: Clone,
@@ -164,85 +41,88 @@ where
Self { Self {
a: self.a.clone(), a: self.a.clone(),
b: self.b.clone(), b: self.b.clone(),
f: self.f.clone(), t: self.t.clone(),
r: PhantomData, _t: std::marker::PhantomData,
} }
} }
} }
impl<A, B, F, Out> NewService for AndThenApplyNewService<A, B, F, Out> impl<T, A, B, C> NewService<C> for AndThenTransform<T, A, B, C>
where where
A: NewService, A: NewService<C>,
B: NewService<Error = A::Error, InitError = A::InitError>, B: NewService<C, InitError = A::InitError>,
F: FnMut(A::Response, &mut B::Service) -> Out, T: Transform<B::Service, Request = A::Response, InitError = A::InitError>,
Out: IntoFuture, T::Error: From<A::Error>,
Out::Error: Into<A::Error>,
{ {
type Request = A::Request; type Request = A::Request;
type Response = Out::Item; type Response = T::Response;
type Error = A::Error; type Error = T::Error;
type InitError = A::InitError; type InitError = T::InitError;
type Service = AndThenApply<A::Service, B::Service, F, Out>; type Service = AndThen<FromErr<A::Service, T::Error>, T::Transform>;
type Future = AndThenApplyNewServiceFuture<A, B, F, Out>; type Future = AndThenTransformFuture<T, A, B, C>;
fn new_service(&self) -> Self::Future { fn new_service(&self, cfg: &C) -> Self::Future {
AndThenApplyNewServiceFuture { AndThenTransformFuture {
a: None, a: None,
b: None, t: None,
f: self.f.clone(), t_cell: self.t.clone(),
fut_a: self.a.new_service(), fut_a: self.a.new_service(cfg).into_future(),
fut_b: self.b.new_service(), fut_b: self.b.new_service(cfg).into_future(),
fut_t: None,
} }
} }
} }
pub struct AndThenApplyNewServiceFuture<A, B, F, Out> pub struct AndThenTransformFuture<T, A, B, C>
where where
A: NewService, A: NewService<C>,
B: NewService<Error = A::Error, InitError = A::InitError>, B: NewService<C, InitError = A::InitError>,
F: FnMut(A::Response, &mut B::Service) -> Out, T: Transform<B::Service, Request = A::Response, InitError = A::InitError>,
Out: IntoFuture, T::Error: From<A::Error>,
Out::Error: Into<A::Error>,
{ {
fut_b: B::Future, fut_a: <A::Future as IntoFuture>::Future,
fut_a: A::Future, fut_b: <B::Future as IntoFuture>::Future,
f: Cell<F>, fut_t: Option<<T::Future as IntoFuture>::Future>,
a: Option<A::Service>, a: Option<A::Service>,
b: Option<B::Service>, t: Option<T::Transform>,
t_cell: Rc<T>,
} }
impl<A, B, F, Out> Future for AndThenApplyNewServiceFuture<A, B, F, Out> impl<T, A, B, C> Future for AndThenTransformFuture<T, A, B, C>
where where
A: NewService, A: NewService<C>,
B: NewService<Error = A::Error, InitError = A::InitError>, B: NewService<C, InitError = A::InitError>,
F: FnMut(A::Response, &mut B::Service) -> Out, T: Transform<B::Service, Request = A::Response, InitError = A::InitError>,
Out: IntoFuture, T::Error: From<A::Error>,
Out::Error: Into<A::Error>,
{ {
type Item = AndThenApply<A::Service, B::Service, F, Out>; type Item = AndThen<FromErr<A::Service, T::Error>, T::Transform>;
type Error = A::InitError; type Error = T::InitError;
fn poll(&mut self) -> Poll<Self::Item, Self::Error> { fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
if self.fut_t.is_none() {
if let Async::Ready(service) = self.fut_b.poll()? {
self.fut_t = Some(self.t_cell.new_transform(service).into_future());
}
}
if self.a.is_none() { if self.a.is_none() {
if let Async::Ready(service) = self.fut_a.poll()? { if let Async::Ready(service) = self.fut_a.poll()? {
self.a = Some(service); self.a = Some(service);
} }
} }
if self.b.is_none() { if let Some(ref mut fut) = self.fut_t {
if let Async::Ready(service) = self.fut_b.poll()? { if let Async::Ready(transform) = fut.poll()? {
self.b = Some(service); self.t = Some(transform);
} }
} }
if self.a.is_some() && self.b.is_some() { if self.a.is_some() && self.t.is_some() {
Ok(Async::Ready(AndThenApply { Ok(Async::Ready(AndThen::new(
f: self.f.clone(), FromErr::new(self.a.take().unwrap()),
a: self.a.take().unwrap(), self.t.take().unwrap(),
b: Cell::new(self.b.take().unwrap()), )))
r: PhantomData,
}))
} else { } else {
Ok(Async::NotReady) Ok(Async::NotReady)
} }
@@ -274,12 +154,14 @@ mod tests {
} }
#[test] #[test]
fn test_call() { fn test_apply() {
let blank = |req| Ok(req); let blank = |req| Ok(req);
let mut srv = blank.into_service().apply(Srv, |req: &'static str, srv| { let mut srv = blank
srv.call(()).map(move |res| (req, res)) .into_service()
}); .apply_fn(Srv, |req: &'static str, srv: &mut Srv| {
srv.call(()).map(move |res| (req, res))
});
assert!(srv.poll_ready().is_ok()); assert!(srv.poll_ready().is_ok());
let res = srv.call("srv").poll(); let res = srv.call("srv").poll();
assert!(res.is_ok()); assert!(res.is_ok());
@@ -291,10 +173,10 @@ mod tests {
let blank = || Ok::<_, ()>((|req| Ok(req)).into_service()); let blank = || Ok::<_, ()>((|req| Ok(req)).into_service());
let new_srv = blank.into_new_service().apply( let new_srv = blank.into_new_service().apply(
|req: &'static str, srv: &mut Srv| srv.call(()).map(move |res| (req, res)),
|| Ok(Srv), || Ok(Srv),
|req: &'static str, srv| srv.call(()).map(move |res| (req, res)),
); );
if let Async::Ready(mut srv) = new_srv.new_service().poll().unwrap() { if let Async::Ready(mut srv) = new_srv.new_service(&()).poll().unwrap() {
assert!(srv.poll_ready().is_ok()); assert!(srv.poll_ready().is_ok());
let res = srv.call("srv").poll(); let res = srv.call("srv").poll();
assert!(res.is_ok()); assert!(res.is_ok());

View File

@@ -0,0 +1,307 @@
use std::marker::PhantomData;
use futures::{try_ready, Async, Future, IntoFuture, Poll};
use super::{IntoNewService, IntoService, NewService, Service};
use crate::cell::Cell;
/// `Apply` service combinator
pub struct AndThenApply<A, B, F, Out>
where
A: Service,
B: Service<Error = A::Error>,
F: FnMut(A::Response, &mut B) -> Out,
Out: IntoFuture,
Out::Error: Into<A::Error>,
{
a: A,
b: Cell<B>,
f: Cell<F>,
r: PhantomData<(Out,)>,
}
impl<A, B, F, Out> AndThenApply<A, B, F, Out>
where
A: Service,
B: Service<Error = A::Error>,
F: FnMut(A::Response, &mut B) -> Out,
Out: IntoFuture,
Out::Error: Into<A::Error>,
{
/// Create new `Apply` combinator
pub fn new<A1: IntoService<A>, B1: IntoService<B>>(a: A1, b: B1, f: F) -> Self {
Self {
f: Cell::new(f),
a: a.into_service(),
b: Cell::new(b.into_service()),
r: PhantomData,
}
}
}
impl<A, B, F, Out> Clone for AndThenApply<A, B, F, Out>
where
A: Service + Clone,
B: Service<Error = A::Error>,
F: FnMut(A::Response, &mut B) -> Out,
Out: IntoFuture,
Out::Error: Into<A::Error>,
{
fn clone(&self) -> Self {
AndThenApply {
a: self.a.clone(),
b: self.b.clone(),
f: self.f.clone(),
r: PhantomData,
}
}
}
impl<A, B, F, Out> Service for AndThenApply<A, B, F, Out>
where
A: Service,
B: Service<Error = A::Error>,
F: FnMut(A::Response, &mut B) -> Out,
Out: IntoFuture,
Out::Error: Into<A::Error>,
{
type Request = A::Request;
type Response = Out::Item;
type Error = A::Error;
type Future = AndThenApplyFuture<A, B, F, Out>;
fn poll_ready(&mut self) -> Poll<(), Self::Error> {
try_ready!(self.a.poll_ready());
self.b.get_mut().poll_ready()
}
fn call(&mut self, req: A::Request) -> Self::Future {
AndThenApplyFuture {
b: self.b.clone(),
f: self.f.clone(),
fut_b: None,
fut_a: Some(self.a.call(req)),
}
}
}
pub struct AndThenApplyFuture<A, B, F, Out>
where
A: Service,
B: Service<Error = A::Error>,
F: FnMut(A::Response, &mut B) -> Out,
Out: IntoFuture,
Out::Error: Into<A::Error>,
{
b: Cell<B>,
f: Cell<F>,
fut_a: Option<A::Future>,
fut_b: Option<Out::Future>,
}
impl<A, B, F, Out> Future for AndThenApplyFuture<A, B, F, Out>
where
A: Service,
B: Service<Error = A::Error>,
F: FnMut(A::Response, &mut B) -> Out,
Out: IntoFuture,
Out::Error: Into<A::Error>,
{
type Item = Out::Item;
type Error = A::Error;
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
if let Some(ref mut fut) = self.fut_b {
return fut.poll().map_err(|e| e.into());
}
match self.fut_a.as_mut().expect("Bug in actix-service").poll() {
Ok(Async::Ready(resp)) => {
let _ = self.fut_a.take();
self.fut_b =
Some((&mut *self.f.get_mut())(resp, self.b.get_mut()).into_future());
self.poll()
}
Ok(Async::NotReady) => Ok(Async::NotReady),
Err(err) => Err(err),
}
}
}
/// `ApplyNewService` new service combinator
pub struct AndThenApplyNewService<A, B, F, Out, Cfg> {
a: A,
b: B,
f: Cell<F>,
r: PhantomData<(Out, Cfg)>,
}
impl<A, B, F, Out, Cfg> AndThenApplyNewService<A, B, F, Out, Cfg>
where
A: NewService<Cfg>,
B: NewService<Cfg, Error = A::Error, InitError = A::InitError>,
F: FnMut(A::Response, &mut B::Service) -> Out,
Out: IntoFuture,
Out::Error: Into<A::Error>,
{
/// Create new `ApplyNewService` new service instance
pub fn new<A1: IntoNewService<A, Cfg>, B1: IntoNewService<B, Cfg>>(
a: A1,
b: B1,
f: F,
) -> Self {
Self {
f: Cell::new(f),
a: a.into_new_service(),
b: b.into_new_service(),
r: PhantomData,
}
}
}
impl<A, B, F, Out, Cfg> Clone for AndThenApplyNewService<A, B, F, Out, Cfg>
where
A: Clone,
B: Clone,
{
fn clone(&self) -> Self {
Self {
a: self.a.clone(),
b: self.b.clone(),
f: self.f.clone(),
r: PhantomData,
}
}
}
impl<A, B, F, Out, Cfg> NewService<Cfg> for AndThenApplyNewService<A, B, F, Out, Cfg>
where
A: NewService<Cfg>,
B: NewService<Cfg, Error = A::Error, InitError = A::InitError>,
F: FnMut(A::Response, &mut B::Service) -> Out,
Out: IntoFuture,
Out::Error: Into<A::Error>,
{
type Request = A::Request;
type Response = Out::Item;
type Error = A::Error;
type Service = AndThenApply<A::Service, B::Service, F, Out>;
type InitError = A::InitError;
type Future = AndThenApplyNewServiceFuture<A, B, F, Out, Cfg>;
fn new_service(&self, cfg: &Cfg) -> Self::Future {
AndThenApplyNewServiceFuture {
a: None,
b: None,
f: self.f.clone(),
fut_a: self.a.new_service(cfg).into_future(),
fut_b: self.b.new_service(cfg).into_future(),
}
}
}
pub struct AndThenApplyNewServiceFuture<A, B, F, Out, Cfg>
where
A: NewService<Cfg>,
B: NewService<Cfg, Error = A::Error, InitError = A::InitError>,
F: FnMut(A::Response, &mut B::Service) -> Out,
Out: IntoFuture,
Out::Error: Into<A::Error>,
{
fut_b: <B::Future as IntoFuture>::Future,
fut_a: <A::Future as IntoFuture>::Future,
f: Cell<F>,
a: Option<A::Service>,
b: Option<B::Service>,
}
impl<A, B, F, Out, Cfg> Future for AndThenApplyNewServiceFuture<A, B, F, Out, Cfg>
where
A: NewService<Cfg>,
B: NewService<Cfg, Error = A::Error, InitError = A::InitError>,
F: FnMut(A::Response, &mut B::Service) -> Out,
Out: IntoFuture,
Out::Error: Into<A::Error>,
{
type Item = AndThenApply<A::Service, B::Service, F, Out>;
type Error = A::InitError;
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
if self.a.is_none() {
if let Async::Ready(service) = self.fut_a.poll()? {
self.a = Some(service);
}
}
if self.b.is_none() {
if let Async::Ready(service) = self.fut_b.poll()? {
self.b = Some(service);
}
}
if self.a.is_some() && self.b.is_some() {
Ok(Async::Ready(AndThenApply {
f: self.f.clone(),
a: self.a.take().unwrap(),
b: Cell::new(self.b.take().unwrap()),
r: PhantomData,
}))
} else {
Ok(Async::NotReady)
}
}
}
#[cfg(test)]
mod tests {
use futures::future::{ok, FutureResult};
use futures::{Async, Future, Poll};
use crate::blank::{Blank, BlankNewService};
use crate::{NewService, Service, ServiceExt};
#[derive(Clone)]
struct Srv;
impl Service for Srv {
type Request = ();
type Response = ();
type Error = ();
type Future = FutureResult<(), ()>;
fn poll_ready(&mut self) -> Poll<(), Self::Error> {
Ok(Async::Ready(()))
}
fn call(&mut self, _: ()) -> Self::Future {
ok(())
}
}
#[test]
fn test_call() {
let mut srv = Blank::new().apply_fn(Srv, |req: &'static str, srv| {
srv.call(()).map(move |res| (req, res))
});
assert!(srv.poll_ready().is_ok());
let res = srv.call("srv").poll();
assert!(res.is_ok());
assert_eq!(res.unwrap(), Async::Ready(("srv", ())));
}
#[test]
fn test_new_service() {
let new_srv = BlankNewService::new_unit().apply_fn(
|| Ok(Srv),
|req: &'static str, srv| srv.call(()).map(move |res| (req, res)),
);
if let Async::Ready(mut srv) = new_srv.new_service(&()).poll().unwrap() {
assert!(srv.poll_ready().is_ok());
let res = srv.call("srv").poll();
assert!(res.is_ok());
assert_eq!(res.unwrap(), Async::Ready(("srv", ())));
} else {
panic!()
}
}
}

View File

@@ -67,24 +67,24 @@ where
} }
/// `ApplyNewService` new service combinator /// `ApplyNewService` new service combinator
pub struct ApplyNewService<T, F, In, Out> pub struct ApplyNewService<T, F, In, Out, Cfg>
where where
T: NewService, T: NewService<Cfg>,
{ {
service: T, service: T,
f: F, f: F,
r: PhantomData<(In, Out)>, r: PhantomData<(In, Out, Cfg)>,
} }
impl<T, F, In, Out> ApplyNewService<T, F, In, Out> impl<T, F, In, Out, Cfg> ApplyNewService<T, F, In, Out, Cfg>
where where
T: NewService, T: NewService<Cfg>,
F: FnMut(In, &mut T::Service) -> Out + Clone, F: FnMut(In, &mut T::Service) -> Out + Clone,
Out: IntoFuture, Out: IntoFuture,
Out::Error: From<T::Error>, Out::Error: From<T::Error>,
{ {
/// Create new `ApplyNewService` new service instance /// Create new `ApplyNewService` new service instance
pub fn new<F1: IntoNewService<T>>(service: F1, f: F) -> Self { pub fn new<F1: IntoNewService<T, Cfg>>(service: F1, f: F) -> Self {
Self { Self {
f, f,
service: service.into_new_service(), service: service.into_new_service(),
@@ -93,9 +93,9 @@ where
} }
} }
impl<T, F, In, Out> Clone for ApplyNewService<T, F, In, Out> impl<T, F, In, Out, Cfg> Clone for ApplyNewService<T, F, In, Out, Cfg>
where where
T: NewService + Clone, T: NewService<Cfg> + Clone,
F: FnMut(In, &mut T::Service) -> Out + Clone, F: FnMut(In, &mut T::Service) -> Out + Clone,
Out: IntoFuture, Out: IntoFuture,
{ {
@@ -108,9 +108,9 @@ where
} }
} }
impl<T, F, In, Out> NewService for ApplyNewService<T, F, In, Out> impl<T, F, In, Out, Cfg> NewService<Cfg> for ApplyNewService<T, F, In, Out, Cfg>
where where
T: NewService, T: NewService<Cfg>,
F: FnMut(In, &mut T::Service) -> Out + Clone, F: FnMut(In, &mut T::Service) -> Out + Clone,
Out: IntoFuture, Out: IntoFuture,
Out::Error: From<T::Error>, Out::Error: From<T::Error>,
@@ -121,31 +121,31 @@ where
type Service = Apply<T::Service, F, In, Out>; type Service = Apply<T::Service, F, In, Out>;
type InitError = T::InitError; type InitError = T::InitError;
type Future = ApplyNewServiceFuture<T, F, In, Out>; type Future = ApplyNewServiceFuture<T, F, In, Out, Cfg>;
fn new_service(&self) -> Self::Future { fn new_service(&self, cfg: &Cfg) -> Self::Future {
ApplyNewServiceFuture::new(self.service.new_service(), self.f.clone()) ApplyNewServiceFuture::new(self.service.new_service(cfg).into_future(), self.f.clone())
} }
} }
pub struct ApplyNewServiceFuture<T, F, In, Out> pub struct ApplyNewServiceFuture<T, F, In, Out, Cfg>
where where
T: NewService, T: NewService<Cfg>,
F: FnMut(In, &mut T::Service) -> Out + Clone, F: FnMut(In, &mut T::Service) -> Out + Clone,
Out: IntoFuture, Out: IntoFuture,
{ {
fut: T::Future, fut: <T::Future as IntoFuture>::Future,
f: Option<F>, f: Option<F>,
r: PhantomData<(In, Out)>, r: PhantomData<(In, Out)>,
} }
impl<T, F, In, Out> ApplyNewServiceFuture<T, F, In, Out> impl<T, F, In, Out, Cfg> ApplyNewServiceFuture<T, F, In, Out, Cfg>
where where
T: NewService, T: NewService<Cfg>,
F: FnMut(In, &mut T::Service) -> Out + Clone, F: FnMut(In, &mut T::Service) -> Out + Clone,
Out: IntoFuture, Out: IntoFuture,
{ {
fn new(fut: T::Future, f: F) -> Self { fn new(fut: <T::Future as IntoFuture>::Future, f: F) -> Self {
ApplyNewServiceFuture { ApplyNewServiceFuture {
f: Some(f), f: Some(f),
fut, fut,
@@ -154,9 +154,9 @@ where
} }
} }
impl<T, F, In, Out> Future for ApplyNewServiceFuture<T, F, In, Out> impl<T, F, In, Out, Cfg> Future for ApplyNewServiceFuture<T, F, In, Out, Cfg>
where where
T: NewService, T: NewService<Cfg>,
F: FnMut(In, &mut T::Service) -> Out + Clone, F: FnMut(In, &mut T::Service) -> Out + Clone,
Out: IntoFuture, Out: IntoFuture,
Out::Error: From<T::Error>, Out::Error: From<T::Error>,
@@ -178,7 +178,8 @@ mod tests {
use futures::future::{ok, FutureResult}; use futures::future::{ok, FutureResult};
use futures::{Async, Future, Poll}; use futures::{Async, Future, Poll};
use crate::{IntoNewService, IntoService, NewService, Service, ServiceExt}; use super::*;
use crate::{IntoService, NewService, Service, ServiceExt};
#[derive(Clone)] #[derive(Clone)]
struct Srv; struct Srv;
@@ -201,9 +202,11 @@ mod tests {
fn test_call() { fn test_call() {
let blank = |req| Ok(req); let blank = |req| Ok(req);
let mut srv = blank.into_service().apply(Srv, |req: &'static str, srv| { let mut srv = blank
srv.call(()).map(move |res| (req, res)) .into_service()
}); .apply_fn(Srv, |req: &'static str, srv| {
srv.call(()).map(move |res| (req, res))
});
assert!(srv.poll_ready().is_ok()); assert!(srv.poll_ready().is_ok());
let res = srv.call("srv").poll(); let res = srv.call("srv").poll();
assert!(res.is_ok()); assert!(res.is_ok());
@@ -212,13 +215,11 @@ mod tests {
#[test] #[test]
fn test_new_service() { fn test_new_service() {
let blank = || Ok::<_, ()>((|req| Ok(req)).into_service()); let new_srv = ApplyNewService::new(
|| Ok::<_, ()>(Srv),
let new_srv = blank.into_new_service().apply(
|| Ok(Srv),
|req: &'static str, srv| srv.call(()).map(move |res| (req, res)), |req: &'static str, srv| srv.call(()).map(move |res| (req, res)),
); );
if let Async::Ready(mut srv) = new_srv.new_service().poll().unwrap() { if let Async::Ready(mut srv) = new_srv.new_service(&()).poll().unwrap() {
assert!(srv.poll_ready().is_ok()); assert!(srv.poll_ready().is_ok());
let res = srv.call("srv").poll(); let res = srv.call("srv").poll();
assert!(res.is_ok()); assert!(res.is_ok());

View File

@@ -0,0 +1,83 @@
use std::marker::PhantomData;
use futures::future::{ok, FutureResult};
use futures::{Async, Poll};
use super::{NewService, Service};
/// Empty service
#[derive(Clone)]
pub struct Blank<R, E> {
_t: PhantomData<(R, E)>,
}
impl<R, E> Blank<R, E> {
pub fn err<E1>(self) -> Blank<R, E1> {
Blank { _t: PhantomData }
}
}
impl<R> Blank<R, ()> {
#[allow(clippy::new_ret_no_self)]
pub fn new<E>() -> Blank<R, E> {
Blank { _t: PhantomData }
}
}
impl<R, E> Default for Blank<R, E> {
fn default() -> Blank<R, E> {
Blank { _t: PhantomData }
}
}
impl<R, E> Service for Blank<R, E> {
type Request = R;
type Response = R;
type Error = E;
type Future = FutureResult<R, E>;
fn poll_ready(&mut self) -> Poll<(), Self::Error> {
Ok(Async::Ready(()))
}
fn call(&mut self, req: R) -> Self::Future {
ok(req)
}
}
/// Empty service factory
pub struct BlankNewService<R, E1, E2 = ()> {
_t: PhantomData<(R, E1, E2)>,
}
impl<R, E1, E2> BlankNewService<R, E1, E2> {
pub fn new() -> BlankNewService<R, E1, E2> {
BlankNewService { _t: PhantomData }
}
}
impl<R, E1> BlankNewService<R, E1, ()> {
pub fn new_unit() -> BlankNewService<R, E1, ()> {
BlankNewService { _t: PhantomData }
}
}
impl<R, E1, E2> Default for BlankNewService<R, E1, E2> {
fn default() -> BlankNewService<R, E1, E2> {
Self::new()
}
}
impl<R, E1, E2> NewService<()> for BlankNewService<R, E1, E2> {
type Request = R;
type Response = R;
type Error = E1;
type Service = Blank<R, E1>;
type InitError = E2;
type Future = FutureResult<Self::Service, Self::InitError>;
fn new_service(&self, _: &()) -> Self::Future {
ok(Blank::default())
}
}

137
actix-service/src/boxed.rs Normal file
View File

@@ -0,0 +1,137 @@
use crate::{NewService, Service};
use futures::{Future, IntoFuture, Poll};
pub type BoxedService<Req, Res, Err> = Box<
Service<
Request = Req,
Response = Res,
Error = Err,
Future = Box<Future<Item = Res, Error = Err>>,
>,
>;
/// Create boxed new service
pub fn new_service<T, C>(
service: T,
) -> BoxedNewService<C, T::Request, T::Response, T::Error, T::InitError>
where
C: 'static,
T: NewService<C> + 'static,
T::Request: 'static,
T::Response: 'static,
T::Service: 'static,
T::Future: 'static,
T::Error: 'static,
T::InitError: 'static,
{
BoxedNewService(Box::new(NewServiceWrapper {
service,
_t: std::marker::PhantomData,
}))
}
/// Create boxed service
pub fn service<T>(service: T) -> BoxedService<T::Request, T::Response, T::Error>
where
T: Service + 'static,
T::Future: 'static,
{
Box::new(ServiceWrapper(service))
}
type Inner<C, Req, Res, Err, InitErr> = Box<
NewService<
C,
Request = Req,
Response = Res,
Error = Err,
InitError = InitErr,
Service = BoxedService<Req, Res, Err>,
Future = Box<Future<Item = BoxedService<Req, Res, Err>, Error = InitErr>>,
>,
>;
pub struct BoxedNewService<C, Req, Res, Err, InitErr>(Inner<C, Req, Res, Err, InitErr>);
impl<C, Req, Res, Err, InitErr> NewService<C> for BoxedNewService<C, Req, Res, Err, InitErr>
where
Req: 'static,
Res: 'static,
Err: 'static,
InitErr: 'static,
{
type Request = Req;
type Response = Res;
type Error = Err;
type InitError = InitErr;
type Service = BoxedService<Req, Res, Err>;
type Future = Box<Future<Item = Self::Service, Error = Self::InitError>>;
fn new_service(&self, cfg: &C) -> Self::Future {
self.0.new_service(cfg)
}
}
struct NewServiceWrapper<C, T: NewService<C>> {
service: T,
_t: std::marker::PhantomData<C>,
}
impl<C, T, Req, Res, Err, InitErr> NewService<C> for NewServiceWrapper<C, T>
where
Req: 'static,
Res: 'static,
Err: 'static,
InitErr: 'static,
T: NewService<C, Request = Req, Response = Res, Error = Err, InitError = InitErr>,
T::Future: 'static,
T::Service: 'static,
<T::Service as Service>::Future: 'static,
{
type Request = Req;
type Response = Res;
type Error = Err;
type InitError = InitErr;
type Service = BoxedService<Req, Res, Err>;
type Future = Box<Future<Item = Self::Service, Error = Self::InitError>>;
fn new_service(&self, cfg: &C) -> Self::Future {
Box::new(
self.service
.new_service(cfg)
.into_future()
.map(ServiceWrapper::boxed),
)
}
}
struct ServiceWrapper<T: Service>(T);
impl<T> ServiceWrapper<T>
where
T: Service + 'static,
T::Future: 'static,
{
fn boxed(service: T) -> BoxedService<T::Request, T::Response, T::Error> {
Box::new(ServiceWrapper(service))
}
}
impl<T, Req, Res, Err> Service for ServiceWrapper<T>
where
T: Service<Request = Req, Response = Res, Error = Err>,
T::Future: 'static,
{
type Request = Req;
type Response = Res;
type Error = Err;
type Future = Box<Future<Item = Self::Response, Error = Self::Error>>;
fn poll_ready(&mut self) -> Poll<(), Self::Error> {
self.0.poll_ready()
}
fn call(&mut self, req: Self::Request) -> Self::Future {
Box::new(self.0.call(req))
}
}

View File

@@ -1,56 +1,80 @@
use std::marker; use std::marker::PhantomData;
use futures::{ use futures::future::{ok, FutureResult};
future::{ok, FutureResult}, use futures::{Async, IntoFuture, Poll};
Async, IntoFuture, Poll,
};
use super::{IntoNewService, IntoService, NewService, Service}; use crate::{IntoConfigurableNewService, IntoNewService, IntoService, NewService, Service};
pub struct FnService<F, Req, Resp, E, Fut> /// Create `NewService` for function that can act as Service
pub fn fn_service<F, Req, Out, Cfg>(f: F) -> FnNewService<F, Req, Out, Cfg>
where where
F: FnMut(Req) -> Fut, F: FnMut(Req) -> Out + Clone,
Fut: IntoFuture<Item = Resp, Error = E>, Out: IntoFuture,
{ {
f: F, FnNewService::new(f)
_t: marker::PhantomData<(Req, Resp, E)>,
} }
impl<F, Req, Resp, E, Fut> FnService<F, Req, Resp, E, Fut> /// Create `NewService` for function that can produce services
pub fn fn_factory<F, R, S, E>(f: F) -> FnNewServiceNoConfig<F, R, S, E>
where where
F: FnMut(Req) -> Fut, F: Fn() -> R,
Fut: IntoFuture<Item = Resp, Error = E>, R: IntoFuture<Item = S, Error = E>,
S: Service,
{
FnNewServiceNoConfig::new(f)
}
/// Create `NewService` for function that can produce services with configuration
pub fn fn_cfg_factory<F, C, R, S, E>(f: F) -> FnNewServiceConfig<F, C, R, S, E>
where
F: Fn(&C) -> R,
R: IntoFuture<Item = S, Error = E>,
S: Service,
{
FnNewServiceConfig::new(f)
}
pub struct FnService<F, Req, Out>
where
F: FnMut(Req) -> Out,
Out: IntoFuture,
{
f: F,
_t: PhantomData<(Req,)>,
}
impl<F, Req, Out> FnService<F, Req, Out>
where
F: FnMut(Req) -> Out,
Out: IntoFuture,
{ {
pub fn new(f: F) -> Self { pub fn new(f: F) -> Self {
FnService { FnService { f, _t: PhantomData }
f,
_t: marker::PhantomData,
}
} }
} }
impl<F, Req, Resp, E, Fut> Clone for FnService<F, Req, Resp, E, Fut> impl<F, Req, Out> Clone for FnService<F, Req, Out>
where where
F: FnMut(Req) -> Fut + Clone, F: FnMut(Req) -> Out + Clone,
Fut: IntoFuture<Item = Resp, Error = E>, Out: IntoFuture,
{ {
fn clone(&self) -> Self { fn clone(&self) -> Self {
FnService { FnService {
f: self.f.clone(), f: self.f.clone(),
_t: marker::PhantomData, _t: PhantomData,
} }
} }
} }
impl<F, Req, Resp, E, Fut> Service for FnService<F, Req, Resp, E, Fut> impl<F, Req, Out> Service for FnService<F, Req, Out>
where where
F: FnMut(Req) -> Fut, F: FnMut(Req) -> Out,
Fut: IntoFuture<Item = Resp, Error = E>, Out: IntoFuture,
{ {
type Request = Req; type Request = Req;
type Response = Resp; type Response = Out::Item;
type Error = E; type Error = Out::Error;
type Future = Fut::Future; type Future = Out::Future;
fn poll_ready(&mut self) -> Poll<(), Self::Error> { fn poll_ready(&mut self) -> Poll<(), Self::Error> {
Ok(Async::Ready(())) Ok(Async::Ready(()))
@@ -61,71 +85,194 @@ where
} }
} }
impl<F, Req, Resp, Err, Fut> IntoService<FnService<F, Req, Resp, Err, Fut>> for F impl<F, Req, Out> IntoService<FnService<F, Req, Out>> for F
where where
F: FnMut(Req) -> Fut + 'static, F: FnMut(Req) -> Out + 'static,
Fut: IntoFuture<Item = Resp, Error = Err>, Out: IntoFuture,
{ {
fn into_service(self) -> FnService<F, Req, Resp, Err, Fut> { fn into_service(self) -> FnService<F, Req, Out> {
FnService::new(self) FnService::new(self)
} }
} }
pub struct FnNewService<F, Req, Resp, Err, Fut> pub struct FnNewService<F, Req, Out, Cfg>
where where
F: FnMut(Req) -> Fut, F: FnMut(Req) -> Out,
Fut: IntoFuture<Item = Resp, Error = Err>, Out: IntoFuture,
{ {
f: F, f: F,
_t: marker::PhantomData<(Req, Resp, Err)>, _t: PhantomData<(Req, Cfg)>,
} }
impl<F, Req, Resp, Err, Fut> FnNewService<F, Req, Resp, Err, Fut> impl<F, Req, Out, Cfg> FnNewService<F, Req, Out, Cfg>
where where
F: FnMut(Req) -> Fut + Clone, F: FnMut(Req) -> Out + Clone,
Fut: IntoFuture<Item = Resp, Error = Err>, Out: IntoFuture,
{ {
pub fn new(f: F) -> Self { pub fn new(f: F) -> Self {
FnNewService { FnNewService { f, _t: PhantomData }
f,
_t: marker::PhantomData,
}
} }
} }
impl<F, Req, Resp, Err, Fut> NewService for FnNewService<F, Req, Resp, Err, Fut> impl<F, Req, Out, Cfg> NewService<Cfg> for FnNewService<F, Req, Out, Cfg>
where where
F: FnMut(Req) -> Fut + Clone, F: FnMut(Req) -> Out + Clone,
Fut: IntoFuture<Item = Resp, Error = Err>, Out: IntoFuture,
{ {
type Request = Req; type Request = Req;
type Response = Resp; type Response = Out::Item;
type Error = Err; type Error = Out::Error;
type Service = FnService<F, Req, Resp, Err, Fut>; type Service = FnService<F, Req, Out>;
type InitError = (); type InitError = ();
type Future = FutureResult<Self::Service, Self::InitError>; type Future = FutureResult<Self::Service, Self::InitError>;
fn new_service(&self) -> Self::Future { fn new_service(&self, _: &Cfg) -> Self::Future {
ok(FnService::new(self.f.clone())) ok(FnService::new(self.f.clone()))
} }
} }
impl<F, Req, Resp, Err, Fut> IntoNewService<FnNewService<F, Req, Resp, Err, Fut>> for F impl<F, Req, Out, Cfg> Clone for FnNewService<F, Req, Out, Cfg>
where where
F: FnMut(Req) -> Fut + Clone + 'static, F: FnMut(Req) -> Out + Clone,
Fut: IntoFuture<Item = Resp, Error = Err>, Out: IntoFuture,
{
fn into_new_service(self) -> FnNewService<F, Req, Resp, Err, Fut> {
FnNewService::new(self)
}
}
impl<F, Req, Resp, Err, Fut> Clone for FnNewService<F, Req, Resp, Err, Fut>
where
F: FnMut(Req) -> Fut + Clone,
Fut: IntoFuture<Item = Resp, Error = Err>,
{ {
fn clone(&self) -> Self { fn clone(&self) -> Self {
Self::new(self.f.clone()) Self::new(self.f.clone())
} }
} }
impl<F, Req, Out, Cfg> IntoNewService<FnNewService<F, Req, Out, Cfg>, Cfg> for F
where
F: Fn(Req) -> Out + Clone,
Out: IntoFuture,
{
fn into_new_service(self) -> FnNewService<F, Req, Out, Cfg> {
FnNewService::new(self)
}
}
/// Converter for `Fn() -> Future<Service>` fn
pub struct FnNewServiceNoConfig<F, R, S, E>
where
F: Fn() -> R,
R: IntoFuture<Item = S, Error = E>,
S: Service,
{
f: F,
}
impl<F, R, S, E> FnNewServiceNoConfig<F, R, S, E>
where
F: Fn() -> R,
R: IntoFuture<Item = S, Error = E>,
S: Service,
{
pub fn new(f: F) -> Self {
FnNewServiceNoConfig { f }
}
}
impl<F, R, S, E> NewService<()> for FnNewServiceNoConfig<F, R, S, E>
where
F: Fn() -> R,
R: IntoFuture<Item = S, Error = E>,
S: Service,
{
type Request = S::Request;
type Response = S::Response;
type Error = S::Error;
type Service = S;
type InitError = E;
type Future = R::Future;
fn new_service(&self, _: &()) -> Self::Future {
(self.f)().into_future()
}
}
impl<F, R, S, E> Clone for FnNewServiceNoConfig<F, R, S, E>
where
F: Fn() -> R + Clone,
R: IntoFuture<Item = S, Error = E>,
S: Service,
{
fn clone(&self) -> Self {
Self::new(self.f.clone())
}
}
impl<F, R, S, E> IntoNewService<FnNewServiceNoConfig<F, R, S, E>, ()> for F
where
F: Fn() -> R,
R: IntoFuture<Item = S, Error = E>,
S: Service,
{
fn into_new_service(self) -> FnNewServiceNoConfig<F, R, S, E> {
FnNewServiceNoConfig::new(self)
}
}
/// Convert `Fn(&Config) -> Future<Service>` fn to NewService
pub struct FnNewServiceConfig<F, C, R, S, E>
where
F: Fn(&C) -> R,
R: IntoFuture<Item = S, Error = E>,
S: Service,
{
f: F,
_t: PhantomData<(C, R, S, E)>,
}
impl<F, C, R, S, E> FnNewServiceConfig<F, C, R, S, E>
where
F: Fn(&C) -> R,
R: IntoFuture<Item = S, Error = E>,
S: Service,
{
pub fn new(f: F) -> Self {
FnNewServiceConfig { f, _t: PhantomData }
}
}
impl<F, C, R, S, E> NewService<C> for FnNewServiceConfig<F, C, R, S, E>
where
F: Fn(&C) -> R,
R: IntoFuture<Item = S, Error = E>,
S: Service,
{
type Request = S::Request;
type Response = S::Response;
type Error = S::Error;
type Service = S;
type InitError = E;
type Future = R::Future;
fn new_service(&self, cfg: &C) -> Self::Future {
(self.f)(cfg).into_future()
}
}
impl<F, C, R, S, E> Clone for FnNewServiceConfig<F, C, R, S, E>
where
F: Fn(&C) -> R + Clone,
R: IntoFuture<Item = S, Error = E>,
S: Service,
{
fn clone(&self) -> Self {
Self::new(self.f.clone())
}
}
impl<F, C, R, S, E> IntoConfigurableNewService<FnNewServiceConfig<F, C, R, S, E>, C> for F
where
F: Fn(&C) -> R,
R: IntoFuture<Item = S, Error = E>,
S: Service,
{
fn into_new_service(self) -> FnNewServiceConfig<F, C, R, S, E> {
FnNewServiceConfig::new(self)
}
}

View File

@@ -0,0 +1,66 @@
use std::marker::PhantomData;
use futures::future::{ok, FutureResult};
use futures::IntoFuture;
use crate::{Apply, IntoTransform, Service, Transform};
pub struct FnTransform<F, S, In, Out, Err>
where
F: FnMut(In, &mut S) -> Out + Clone,
Out: IntoFuture,
{
f: F,
_t: PhantomData<(S, In, Out, Err)>,
}
impl<F, S, In, Out, Err> FnTransform<F, S, In, Out, Err>
where
F: FnMut(In, &mut S) -> Out + Clone,
Out: IntoFuture,
{
pub fn new(f: F) -> Self {
FnTransform { f, _t: PhantomData }
}
}
impl<F, S, In, Out, Err> Transform<S> for FnTransform<F, S, In, Out, Err>
where
S: Service,
F: FnMut(In, &mut S) -> Out + Clone,
Out: IntoFuture,
Out::Error: From<S::Error>,
{
type Request = In;
type Response = Out::Item;
type Error = Out::Error;
type Transform = Apply<S, F, In, Out>;
type InitError = Err;
type Future = FutureResult<Self::Transform, Self::InitError>;
fn new_transform(&self, service: S) -> Self::Future {
ok(Apply::new(service, self.f.clone()))
}
}
impl<F, S, In, Out, Err> IntoTransform<FnTransform<F, S, In, Out, Err>, S> for F
where
S: Service,
F: FnMut(In, &mut S) -> Out + Clone,
Out: IntoFuture,
Out::Error: From<S::Error>,
{
fn into_transform(self) -> FnTransform<F, S, In, Out, Err> {
FnTransform::new(self)
}
}
impl<F, S, In, Out, Err> Clone for FnTransform<F, S, In, Out, Err>
where
F: FnMut(In, &mut S) -> Out + Clone,
Out: IntoFuture,
{
fn clone(&self) -> Self {
Self::new(self.f.clone())
}
}

View File

@@ -1,6 +1,6 @@
use std::marker::PhantomData; use std::marker::PhantomData;
use futures::{Async, Future, Poll}; use futures::{Async, Future, IntoFuture, Poll};
use super::{NewService, Service}; use super::{NewService, Service};
@@ -81,23 +81,23 @@ where
/// service's error. /// service's error.
/// ///
/// This is created by the `NewServiceExt::from_err` method. /// This is created by the `NewServiceExt::from_err` method.
pub struct FromErrNewService<A, E> { pub struct FromErrNewService<A, E, C> {
a: A, a: A,
e: PhantomData<E>, e: PhantomData<(E, C)>,
} }
impl<A, E> FromErrNewService<A, E> { impl<A, E, C> FromErrNewService<A, E, C> {
/// Create new `FromErr` new service instance /// Create new `FromErr` new service instance
pub fn new(a: A) -> Self pub fn new(a: A) -> Self
where where
A: NewService, A: NewService<C>,
E: From<A::Error>, E: From<A::Error>,
{ {
Self { a, e: PhantomData } Self { a, e: PhantomData }
} }
} }
impl<A, E> Clone for FromErrNewService<A, E> impl<A, E, C> Clone for FromErrNewService<A, E, C>
where where
A: Clone, A: Clone,
{ {
@@ -109,9 +109,9 @@ where
} }
} }
impl<A, E> NewService for FromErrNewService<A, E> impl<A, E, C> NewService<C> for FromErrNewService<A, E, C>
where where
A: NewService, A: NewService<C>,
E: From<A::Error>, E: From<A::Error>,
{ {
type Request = A::Request; type Request = A::Request;
@@ -120,28 +120,28 @@ where
type Service = FromErr<A::Service, E>; type Service = FromErr<A::Service, E>;
type InitError = A::InitError; type InitError = A::InitError;
type Future = FromErrNewServiceFuture<A, E>; type Future = FromErrNewServiceFuture<A, E, C>;
fn new_service(&self) -> Self::Future { fn new_service(&self, cfg: &C) -> Self::Future {
FromErrNewServiceFuture { FromErrNewServiceFuture {
fut: self.a.new_service(), fut: self.a.new_service(cfg).into_future(),
e: PhantomData, e: PhantomData,
} }
} }
} }
pub struct FromErrNewServiceFuture<A, E> pub struct FromErrNewServiceFuture<A, E, C>
where where
A: NewService, A: NewService<C>,
E: From<A::Error>, E: From<A::Error>,
{ {
fut: A::Future, fut: <A::Future as IntoFuture>::Future,
e: PhantomData<E>, e: PhantomData<E>,
} }
impl<A, E> Future for FromErrNewServiceFuture<A, E> impl<A, E, C> Future for FromErrNewServiceFuture<A, E, C>
where where
A: NewService, A: NewService<C>,
E: From<A::Error>, E: From<A::Error>,
{ {
type Item = FromErr<A::Service, E>; type Item = FromErr<A::Service, E>;
@@ -208,7 +208,7 @@ mod tests {
fn test_new_service() { fn test_new_service() {
let blank = || Ok::<_, ()>(Srv); let blank = || Ok::<_, ()>(Srv);
let new_srv = blank.into_new_service().from_err::<Error>(); let new_srv = blank.into_new_service().from_err::<Error>();
if let Async::Ready(mut srv) = new_srv.new_service().poll().unwrap() { if let Async::Ready(mut srv) = new_srv.new_service(&()).poll().unwrap() {
let res = srv.call(()).poll(); let res = srv.call(()).poll();
assert!(res.is_err()); assert!(res.is_err());
assert_eq!(res.err().unwrap(), Error); assert_eq!(res.err().unwrap(), Error);

View File

@@ -1,25 +1,39 @@
use std::rc::Rc;
use std::sync::Arc;
use futures::{Future, IntoFuture, Poll}; use futures::{Future, IntoFuture, Poll};
pub use void::Void;
mod and_then; mod and_then;
mod and_then_apply; mod and_then_apply;
mod and_then_apply_fn;
mod apply; mod apply;
pub mod blank;
pub mod boxed;
mod cell; mod cell;
mod fn_service; mod fn_service;
mod fn_transform;
mod from_err; mod from_err;
mod map; mod map;
mod map_err; mod map_err;
mod map_init_err; mod map_init_err;
mod then; mod then;
mod transform;
mod transform_map_init_err;
pub use self::and_then::{AndThen, AndThenNewService}; pub use self::and_then::{AndThen, AndThenNewService};
pub use self::and_then_apply::{AndThenApply, AndThenApplyNewService}; pub use self::and_then_apply::AndThenTransform;
use self::and_then_apply_fn::{AndThenApply, AndThenApplyNewService};
pub use self::apply::{Apply, ApplyNewService}; pub use self::apply::{Apply, ApplyNewService};
pub use self::fn_service::{FnNewService, FnService}; pub use self::fn_service::{fn_cfg_factory, fn_factory, fn_service, FnService};
pub use self::fn_transform::FnTransform;
pub use self::from_err::{FromErr, FromErrNewService}; pub use self::from_err::{FromErr, FromErrNewService};
pub use self::map::{Map, MapNewService}; pub use self::map::{Map, MapNewService};
pub use self::map_err::{MapErr, MapErrNewService}; pub use self::map_err::{MapErr, MapErrNewService};
pub use self::map_init_err::MapInitErr; pub use self::map_init_err::MapInitErr;
pub use self::then::{Then, ThenNewService}; pub use self::then::{Then, ThenNewService};
pub use self::transform::{IntoTransform, Transform};
/// An asynchronous function from `Request` to a `Response`. /// An asynchronous function from `Request` to a `Response`.
pub trait Service { pub trait Service {
@@ -63,14 +77,14 @@ pub trait Service {
pub trait ServiceExt: Service { pub trait ServiceExt: Service {
/// Apply function to specified service and use it as a next service in /// Apply function to specified service and use it as a next service in
/// chain. /// chain.
fn apply<B, I, F, Out, Req>(self, service: I, f: F) -> AndThenApply<Self, B, F, Out> fn apply_fn<F, B, B1, Out>(self, service: B1, f: F) -> AndThenApply<Self, B, F, Out>
where where
Self: Sized, Self: Sized,
B: Service<Request = Req, Error = Self::Error>,
I: IntoService<B>,
F: FnMut(Self::Response, &mut B) -> Out, F: FnMut(Self::Response, &mut B) -> Out,
Out: IntoFuture, Out: IntoFuture,
Out::Error: Into<Self::Error>, Out::Error: Into<Self::Error>,
B: Service<Error = Self::Error>,
B1: IntoService<B>,
{ {
AndThenApply::new(self, service, f) AndThenApply::new(self, service, f)
} }
@@ -162,7 +176,9 @@ impl<T: ?Sized> ServiceExt for T where T: Service {}
/// accepts new TCP streams, obtains a new `Service` value using the /// accepts new TCP streams, obtains a new `Service` value using the
/// `NewService` trait, and uses that new `Service` value to process inbound /// `NewService` trait, and uses that new `Service` value to process inbound
/// requests on that new TCP stream. /// requests on that new TCP stream.
pub trait NewService { ///
/// `Config` is a service factory configuration type.
pub trait NewService<Config = ()> {
/// Requests handled by the service. /// Requests handled by the service.
type Request; type Request;
@@ -183,22 +199,40 @@ pub trait NewService {
type InitError; type InitError;
/// The future of the `Service` instance. /// The future of the `Service` instance.
type Future: Future<Item = Self::Service, Error = Self::InitError>; type Future: IntoFuture<Item = Self::Service, Error = Self::InitError>;
/// Create and return a new service value asynchronously. /// Create and return a new service value asynchronously.
fn new_service(&self) -> Self::Future; fn new_service(&self, cfg: &Config) -> Self::Future;
/// Apply function to specified service and use it as a next service in /// Apply function to specified service and use it as a next service in
/// chain. /// chain.
fn apply<B, I, F, Out, Req>( fn apply<T, T1, B, B1>(
self,
transform: T1,
service: B1,
) -> AndThenTransform<T, Self, B, Config>
where
Self: Sized,
T: Transform<B::Service, Request = Self::Response, InitError = Self::InitError>,
T::Error: From<Self::Error>,
T1: IntoTransform<T, B::Service>,
B: NewService<Config, InitError = Self::InitError>,
B1: IntoNewService<B, Config>,
{
AndThenTransform::new(transform.into_transform(), self, service.into_new_service())
}
/// Apply function to specified service and use it as a next service in
/// chain.
fn apply_fn<B, I, F, Out>(
self, self,
service: I, service: I,
f: F, f: F,
) -> AndThenApplyNewService<Self, B, F, Out> ) -> AndThenApplyNewService<Self, B, F, Out, Config>
where where
Self: Sized, Self: Sized,
B: NewService<Request = Req, Error = Self::Error, InitError = Self::InitError>, B: NewService<Config, Error = Self::Error, InitError = Self::InitError>,
I: IntoNewService<B>, I: IntoNewService<B, Config>,
F: FnMut(Self::Response, &mut B::Service) -> Out, F: FnMut(Self::Response, &mut B::Service) -> Out,
Out: IntoFuture, Out: IntoFuture,
Out::Error: Into<Self::Error>, Out::Error: Into<Self::Error>,
@@ -207,11 +241,12 @@ pub trait NewService {
} }
/// Call another service after call to this one has resolved successfully. /// Call another service after call to this one has resolved successfully.
fn and_then<F, B>(self, new_service: F) -> AndThenNewService<Self, B> fn and_then<F, B>(self, new_service: F) -> AndThenNewService<Self, B, Config>
where where
Self: Sized, Self: Sized,
F: IntoNewService<B>, F: IntoNewService<B, Config>,
B: NewService< B: NewService<
Config,
Request = Self::Response, Request = Self::Response,
Error = Self::Error, Error = Self::Error,
InitError = Self::InitError, InitError = Self::InitError,
@@ -226,7 +261,7 @@ pub trait NewService {
/// ///
/// Note that this function consumes the receiving new service and returns a /// Note that this function consumes the receiving new service and returns a
/// wrapped version of it. /// wrapped version of it.
fn from_err<E>(self) -> FromErrNewService<Self, E> fn from_err<E>(self) -> FromErrNewService<Self, E, Config>
where where
Self: Sized, Self: Sized,
E: From<Self::Error>, E: From<Self::Error>,
@@ -240,11 +275,12 @@ pub trait NewService {
/// ///
/// Note that this function consumes the receiving future and returns a /// Note that this function consumes the receiving future and returns a
/// wrapped version of it. /// wrapped version of it.
fn then<F, B>(self, new_service: F) -> ThenNewService<Self, B> fn then<F, B>(self, new_service: F) -> ThenNewService<Self, B, Config>
where where
Self: Sized, Self: Sized,
F: IntoNewService<B>, F: IntoNewService<B, Config>,
B: NewService< B: NewService<
Config,
Request = Result<Self::Response, Self::Error>, Request = Result<Self::Response, Self::Error>,
Error = Self::Error, Error = Self::Error,
InitError = Self::InitError, InitError = Self::InitError,
@@ -255,7 +291,7 @@ pub trait NewService {
/// Map this service's output to a different type, returning a new service /// Map this service's output to a different type, returning a new service
/// of the resulting type. /// of the resulting type.
fn map<F, R>(self, f: F) -> MapNewService<Self, F, R> fn map<F, R>(self, f: F) -> MapNewService<Self, F, R, Config>
where where
Self: Sized, Self: Sized,
F: FnMut(Self::Response) -> R, F: FnMut(Self::Response) -> R,
@@ -264,7 +300,7 @@ pub trait NewService {
} }
/// Map this service's error to a different error, returning a new service. /// Map this service's error to a different error, returning a new service.
fn map_err<F, E>(self, f: F) -> MapErrNewService<Self, F, E> fn map_err<F, E>(self, f: F) -> MapErrNewService<Self, F, E, Config>
where where
Self: Sized, Self: Sized,
F: Fn(Self::Error) -> E, F: Fn(Self::Error) -> E,
@@ -273,7 +309,7 @@ pub trait NewService {
} }
/// Map this service's init error to a different error, returning a new service. /// Map this service's init error to a different error, returning a new service.
fn map_init_err<F, E>(self, f: F) -> MapInitErr<Self, F, E> fn map_init_err<F, E>(self, f: F) -> MapInitErr<Self, F, E, Config>
where where
Self: Sized, Self: Sized,
F: Fn(Self::InitError) -> E, F: Fn(Self::InitError) -> E,
@@ -318,21 +354,35 @@ where
} }
} }
impl<F, R, E, S> NewService for F impl<S, C> NewService<C> for Rc<S>
where where
F: Fn() -> R, S: NewService<C>,
R: IntoFuture<Item = S, Error = E>,
S: Service,
{ {
type Request = S::Request; type Request = S::Request;
type Response = S::Response; type Response = S::Response;
type Error = S::Error; type Error = S::Error;
type Service = S; type Service = S::Service;
type InitError = E; type InitError = S::InitError;
type Future = R::Future; type Future = S::Future;
fn new_service(&self) -> Self::Future { fn new_service(&self, cfg: &C) -> S::Future {
(*self)().into_future() self.as_ref().new_service(cfg)
}
}
impl<S, C> NewService<C> for Arc<S>
where
S: NewService<C>,
{
type Request = S::Request;
type Response = S::Response;
type Error = S::Error;
type Service = S::Service;
type InitError = S::InitError;
type Future = S::Future;
fn new_service(&self, cfg: &C) -> S::Future {
self.as_ref().new_service(cfg)
} }
} }
@@ -345,10 +395,10 @@ where
fn into_service(self) -> T; fn into_service(self) -> T;
} }
/// Trait for types that can be converted to a Service /// Trait for types that can be converted to a `NewService`
pub trait IntoNewService<T> pub trait IntoNewService<T, C = ()>
where where
T: NewService, T: NewService<C>,
{ {
/// Convert to an `NewService` /// Convert to an `NewService`
fn into_new_service(self) -> T; fn into_new_service(self) -> T;
@@ -363,9 +413,27 @@ where
} }
} }
impl<T> IntoNewService<T> for T impl<T, C> IntoNewService<T, C> for T
where where
T: NewService, T: NewService<C>,
{
fn into_new_service(self) -> T {
self
}
}
/// Trait for types that can be converted to a configurable `NewService`
pub trait IntoConfigurableNewService<T, C>
where
T: NewService<C>,
{
/// Convert to an `NewService`
fn into_new_service(self) -> T;
}
impl<T, C> IntoConfigurableNewService<T, C> for T
where
T: NewService<C>,
{ {
fn into_new_service(self) -> T { fn into_new_service(self) -> T {
self self

View File

@@ -1,6 +1,6 @@
use std::marker::PhantomData; use std::marker::PhantomData;
use futures::{Async, Future, Poll}; use futures::{Async, Future, IntoFuture, Poll};
use super::{NewService, Service}; use super::{NewService, Service};
@@ -97,18 +97,18 @@ where
} }
/// `MapNewService` new service combinator /// `MapNewService` new service combinator
pub struct MapNewService<A, F, Response> { pub struct MapNewService<A, F, Res, Cfg> {
a: A, a: A,
f: F, f: F,
r: PhantomData<Response>, r: PhantomData<(Res, Cfg)>,
} }
impl<A, F, Response> MapNewService<A, F, Response> { impl<A, F, Res, Cfg> MapNewService<A, F, Res, Cfg> {
/// Create new `Map` new service instance /// Create new `Map` new service instance
pub fn new(a: A, f: F) -> Self pub fn new(a: A, f: F) -> Self
where where
A: NewService, A: NewService<Cfg>,
F: FnMut(A::Response) -> Response, F: FnMut(A::Response) -> Res,
{ {
Self { Self {
a, a,
@@ -118,7 +118,7 @@ impl<A, F, Response> MapNewService<A, F, Response> {
} }
} }
impl<A, F, Response> Clone for MapNewService<A, F, Response> impl<A, F, Res, Cfg> Clone for MapNewService<A, F, Res, Cfg>
where where
A: Clone, A: Clone,
F: Clone, F: Clone,
@@ -132,49 +132,49 @@ where
} }
} }
impl<A, F, Response> NewService for MapNewService<A, F, Response> impl<A, F, Res, Cfg> NewService<Cfg> for MapNewService<A, F, Res, Cfg>
where where
A: NewService, A: NewService<Cfg>,
F: FnMut(A::Response) -> Response + Clone, F: FnMut(A::Response) -> Res + Clone,
{ {
type Request = A::Request; type Request = A::Request;
type Response = Response; type Response = Res;
type Error = A::Error; type Error = A::Error;
type Service = Map<A::Service, F, Response>; type Service = Map<A::Service, F, Res>;
type InitError = A::InitError; type InitError = A::InitError;
type Future = MapNewServiceFuture<A, F, Response>; type Future = MapNewServiceFuture<A, F, Res, Cfg>;
fn new_service(&self) -> Self::Future { fn new_service(&self, cfg: &Cfg) -> Self::Future {
MapNewServiceFuture::new(self.a.new_service(), self.f.clone()) MapNewServiceFuture::new(self.a.new_service(cfg).into_future(), self.f.clone())
} }
} }
pub struct MapNewServiceFuture<A, F, Response> pub struct MapNewServiceFuture<A, F, Res, Cfg>
where where
A: NewService, A: NewService<Cfg>,
F: FnMut(A::Response) -> Response, F: FnMut(A::Response) -> Res,
{ {
fut: A::Future, fut: <A::Future as IntoFuture>::Future,
f: Option<F>, f: Option<F>,
} }
impl<A, F, Response> MapNewServiceFuture<A, F, Response> impl<A, F, Res, Cfg> MapNewServiceFuture<A, F, Res, Cfg>
where where
A: NewService, A: NewService<Cfg>,
F: FnMut(A::Response) -> Response, F: FnMut(A::Response) -> Res,
{ {
fn new(fut: A::Future, f: F) -> Self { fn new(fut: <A::Future as IntoFuture>::Future, f: F) -> Self {
MapNewServiceFuture { f: Some(f), fut } MapNewServiceFuture { f: Some(f), fut }
} }
} }
impl<A, F, Response> Future for MapNewServiceFuture<A, F, Response> impl<A, F, Res, Cfg> Future for MapNewServiceFuture<A, F, Res, Cfg>
where where
A: NewService, A: NewService<Cfg>,
F: FnMut(A::Response) -> Response, F: FnMut(A::Response) -> Res,
{ {
type Item = Map<A::Service, F, Response>; type Item = Map<A::Service, F, Res>;
type Error = A::InitError; type Error = A::InitError;
fn poll(&mut self) -> Poll<Self::Item, Self::Error> { fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
@@ -229,7 +229,7 @@ mod tests {
fn test_new_service() { fn test_new_service() {
let blank = || Ok::<_, ()>(Srv); let blank = || Ok::<_, ()>(Srv);
let new_srv = blank.into_new_service().map(|_| "ok"); let new_srv = blank.into_new_service().map(|_| "ok");
if let Async::Ready(mut srv) = new_srv.new_service().poll().unwrap() { if let Async::Ready(mut srv) = new_srv.new_service(&()).poll().unwrap() {
let res = srv.call(()).poll(); let res = srv.call(()).poll();
assert!(res.is_ok()); assert!(res.is_ok());
assert_eq!(res.unwrap(), Async::Ready("ok")); assert_eq!(res.unwrap(), Async::Ready("ok"));

View File

@@ -1,6 +1,6 @@
use std::marker::PhantomData; use std::marker::PhantomData;
use futures::{Async, Future, Poll}; use futures::{Async, Future, IntoFuture, Poll};
use super::{NewService, Service}; use super::{NewService, Service};
@@ -98,17 +98,17 @@ where
/// service's error. /// service's error.
/// ///
/// This is created by the `NewServiceExt::map_err` method. /// This is created by the `NewServiceExt::map_err` method.
pub struct MapErrNewService<A, F, E> { pub struct MapErrNewService<A, F, E, C> {
a: A, a: A,
f: F, f: F,
e: PhantomData<E>, e: PhantomData<(E, C)>,
} }
impl<A, F, E> MapErrNewService<A, F, E> { impl<A, F, E, C> MapErrNewService<A, F, E, C> {
/// Create new `MapErr` new service instance /// Create new `MapErr` new service instance
pub fn new(a: A, f: F) -> Self pub fn new(a: A, f: F) -> Self
where where
A: NewService, A: NewService<C>,
F: Fn(A::Error) -> E, F: Fn(A::Error) -> E,
{ {
Self { Self {
@@ -119,7 +119,7 @@ impl<A, F, E> MapErrNewService<A, F, E> {
} }
} }
impl<A, F, E> Clone for MapErrNewService<A, F, E> impl<A, F, E, C> Clone for MapErrNewService<A, F, E, C>
where where
A: Clone, A: Clone,
F: Clone, F: Clone,
@@ -133,9 +133,9 @@ where
} }
} }
impl<A, F, E> NewService for MapErrNewService<A, F, E> impl<A, F, E, C> NewService<C> for MapErrNewService<A, F, E, C>
where where
A: NewService, A: NewService<C>,
F: Fn(A::Error) -> E + Clone, F: Fn(A::Error) -> E + Clone,
{ {
type Request = A::Request; type Request = A::Request;
@@ -144,35 +144,35 @@ where
type Service = MapErr<A::Service, F, E>; type Service = MapErr<A::Service, F, E>;
type InitError = A::InitError; type InitError = A::InitError;
type Future = MapErrNewServiceFuture<A, F, E>; type Future = MapErrNewServiceFuture<A, F, E, C>;
fn new_service(&self) -> Self::Future { fn new_service(&self, cfg: &C) -> Self::Future {
MapErrNewServiceFuture::new(self.a.new_service(), self.f.clone()) MapErrNewServiceFuture::new(self.a.new_service(cfg).into_future(), self.f.clone())
} }
} }
pub struct MapErrNewServiceFuture<A, F, E> pub struct MapErrNewServiceFuture<A, F, E, C>
where where
A: NewService, A: NewService<C>,
F: Fn(A::Error) -> E, F: Fn(A::Error) -> E,
{ {
fut: A::Future, fut: <A::Future as IntoFuture>::Future,
f: F, f: F,
} }
impl<A, F, E> MapErrNewServiceFuture<A, F, E> impl<A, F, E, C> MapErrNewServiceFuture<A, F, E, C>
where where
A: NewService, A: NewService<C>,
F: Fn(A::Error) -> E, F: Fn(A::Error) -> E,
{ {
fn new(fut: A::Future, f: F) -> Self { fn new(fut: <A::Future as IntoFuture>::Future, f: F) -> Self {
MapErrNewServiceFuture { f, fut } MapErrNewServiceFuture { f, fut }
} }
} }
impl<A, F, E> Future for MapErrNewServiceFuture<A, F, E> impl<A, F, E, C> Future for MapErrNewServiceFuture<A, F, E, C>
where where
A: NewService, A: NewService<C>,
F: Fn(A::Error) -> E + Clone, F: Fn(A::Error) -> E + Clone,
{ {
type Item = MapErr<A::Service, F, E>; type Item = MapErr<A::Service, F, E>;
@@ -231,7 +231,7 @@ mod tests {
fn test_new_service() { fn test_new_service() {
let blank = || Ok::<_, ()>(Srv); let blank = || Ok::<_, ()>(Srv);
let new_srv = blank.into_new_service().map_err(|_| "error"); let new_srv = blank.into_new_service().map_err(|_| "error");
if let Async::Ready(mut srv) = new_srv.new_service().poll().unwrap() { if let Async::Ready(mut srv) = new_srv.new_service(&()).poll().unwrap() {
let res = srv.call(()).poll(); let res = srv.call(()).poll();
assert!(res.is_err()); assert!(res.is_err());
assert_eq!(res.err().unwrap(), "error"); assert_eq!(res.err().unwrap(), "error");

View File

@@ -1,21 +1,21 @@
use std::marker::PhantomData; use std::marker::PhantomData;
use futures::{Future, Poll}; use futures::{Future, IntoFuture, Poll};
use super::NewService; use super::NewService;
/// `MapInitErr` service combinator /// `MapInitErr` service combinator
pub struct MapInitErr<A, F, E> { pub struct MapInitErr<A, F, E, C> {
a: A, a: A,
f: F, f: F,
e: PhantomData<E>, e: PhantomData<(E, C)>,
} }
impl<A, F, E> MapInitErr<A, F, E> { impl<A, F, E, C> MapInitErr<A, F, E, C> {
/// Create new `MapInitErr` combinator /// Create new `MapInitErr` combinator
pub fn new(a: A, f: F) -> Self pub fn new(a: A, f: F) -> Self
where where
A: NewService, A: NewService<C>,
F: Fn(A::InitError) -> E, F: Fn(A::InitError) -> E,
{ {
Self { Self {
@@ -26,7 +26,7 @@ impl<A, F, E> MapInitErr<A, F, E> {
} }
} }
impl<A, F, E> Clone for MapInitErr<A, F, E> impl<A, F, E, C> Clone for MapInitErr<A, F, E, C>
where where
A: Clone, A: Clone,
F: Clone, F: Clone,
@@ -40,9 +40,9 @@ where
} }
} }
impl<A, F, E> NewService for MapInitErr<A, F, E> impl<A, F, E, C> NewService<C> for MapInitErr<A, F, E, C>
where where
A: NewService, A: NewService<C>,
F: Fn(A::InitError) -> E + Clone, F: Fn(A::InitError) -> E + Clone,
{ {
type Request = A::Request; type Request = A::Request;
@@ -51,35 +51,35 @@ where
type Service = A::Service; type Service = A::Service;
type InitError = E; type InitError = E;
type Future = MapInitErrFuture<A, F, E>; type Future = MapInitErrFuture<A, F, E, C>;
fn new_service(&self) -> Self::Future { fn new_service(&self, cfg: &C) -> Self::Future {
MapInitErrFuture::new(self.a.new_service(), self.f.clone()) MapInitErrFuture::new(self.a.new_service(cfg).into_future(), self.f.clone())
} }
} }
pub struct MapInitErrFuture<A, F, E> pub struct MapInitErrFuture<A, F, E, C>
where where
A: NewService, A: NewService<C>,
F: Fn(A::InitError) -> E, F: Fn(A::InitError) -> E,
{ {
f: F, f: F,
fut: A::Future, fut: <A::Future as IntoFuture>::Future,
} }
impl<A, F, E> MapInitErrFuture<A, F, E> impl<A, F, E, C> MapInitErrFuture<A, F, E, C>
where where
A: NewService, A: NewService<C>,
F: Fn(A::InitError) -> E, F: Fn(A::InitError) -> E,
{ {
fn new(fut: A::Future, f: F) -> Self { fn new(fut: <A::Future as IntoFuture>::Future, f: F) -> Self {
MapInitErrFuture { f, fut } MapInitErrFuture { f, fut }
} }
} }
impl<A, F, E> Future for MapInitErrFuture<A, F, E> impl<A, F, E, C> Future for MapInitErrFuture<A, F, E, C>
where where
A: NewService, A: NewService<C>,
F: Fn(A::InitError) -> E, F: Fn(A::InitError) -> E,
{ {
type Item = A::Service; type Item = A::Service;

View File

@@ -1,4 +1,6 @@
use futures::{try_ready, Async, Future, Poll}; use std::marker::PhantomData;
use futures::{try_ready, Async, Future, IntoFuture, Poll};
use super::{IntoNewService, NewService, Service}; use super::{IntoNewService, NewService, Service};
use crate::cell::Cell; use crate::cell::Cell;
@@ -109,34 +111,38 @@ where
} }
/// `ThenNewService` new service combinator /// `ThenNewService` new service combinator
pub struct ThenNewService<A, B> { pub struct ThenNewService<A, B, C> {
a: A, a: A,
b: B, b: B,
_t: PhantomData<C>,
} }
impl<A, B> ThenNewService<A, B> { impl<A, B, C> ThenNewService<A, B, C> {
/// Create new `AndThen` combinator /// Create new `AndThen` combinator
pub fn new<F>(a: A, f: F) -> Self pub fn new<F>(a: A, f: F) -> Self
where where
A: NewService, A: NewService<C>,
B: NewService< B: NewService<
C,
Request = Result<A::Response, A::Error>, Request = Result<A::Response, A::Error>,
Error = A::Error, Error = A::Error,
InitError = A::InitError, InitError = A::InitError,
>, >,
F: IntoNewService<B>, F: IntoNewService<B, C>,
{ {
Self { Self {
a, a,
b: f.into_new_service(), b: f.into_new_service(),
_t: PhantomData,
} }
} }
} }
impl<A, B> NewService for ThenNewService<A, B> impl<A, B, C> NewService<C> for ThenNewService<A, B, C>
where where
A: NewService, A: NewService<C>,
B: NewService< B: NewService<
C,
Request = Result<A::Response, A::Error>, Request = Result<A::Response, A::Error>,
Error = A::Error, Error = A::Error,
InitError = A::InitError, InitError = A::InitError,
@@ -148,14 +154,17 @@ where
type Service = Then<A::Service, B::Service>; type Service = Then<A::Service, B::Service>;
type InitError = A::InitError; type InitError = A::InitError;
type Future = ThenNewServiceFuture<A, B>; type Future = ThenNewServiceFuture<A, B, C>;
fn new_service(&self) -> Self::Future { fn new_service(&self, cfg: &C) -> Self::Future {
ThenNewServiceFuture::new(self.a.new_service(), self.b.new_service()) ThenNewServiceFuture::new(
self.a.new_service(cfg).into_future(),
self.b.new_service(cfg).into_future(),
)
} }
} }
impl<A, B> Clone for ThenNewService<A, B> impl<A, B, C> Clone for ThenNewService<A, B, C>
where where
A: Clone, A: Clone,
B: Clone, B: Clone,
@@ -164,35 +173,41 @@ where
Self { Self {
a: self.a.clone(), a: self.a.clone(),
b: self.b.clone(), b: self.b.clone(),
_t: PhantomData,
} }
} }
} }
pub struct ThenNewServiceFuture<A, B> pub struct ThenNewServiceFuture<A, B, C>
where where
A: NewService, A: NewService<C>,
B: NewService< B: NewService<
C,
Request = Result<A::Response, A::Error>, Request = Result<A::Response, A::Error>,
Error = A::Error, Error = A::Error,
InitError = A::InitError, InitError = A::InitError,
>, >,
{ {
fut_b: B::Future, fut_b: <B::Future as IntoFuture>::Future,
fut_a: A::Future, fut_a: <A::Future as IntoFuture>::Future,
a: Option<A::Service>, a: Option<A::Service>,
b: Option<B::Service>, b: Option<B::Service>,
} }
impl<A, B> ThenNewServiceFuture<A, B> impl<A, B, C> ThenNewServiceFuture<A, B, C>
where where
A: NewService, A: NewService<C>,
B: NewService< B: NewService<
C,
Request = Result<A::Response, A::Error>, Request = Result<A::Response, A::Error>,
Error = A::Error, Error = A::Error,
InitError = A::InitError, InitError = A::InitError,
>, >,
{ {
fn new(fut_a: A::Future, fut_b: B::Future) -> Self { fn new(
fut_a: <A::Future as IntoFuture>::Future,
fut_b: <B::Future as IntoFuture>::Future,
) -> Self {
ThenNewServiceFuture { ThenNewServiceFuture {
fut_a, fut_a,
fut_b, fut_b,
@@ -202,10 +217,11 @@ where
} }
} }
impl<A, B> Future for ThenNewServiceFuture<A, B> impl<A, B, C> Future for ThenNewServiceFuture<A, B, C>
where where
A: NewService, A: NewService<C>,
B: NewService< B: NewService<
C,
Request = Result<A::Response, A::Error>, Request = Result<A::Response, A::Error>,
Error = A::Error, Error = A::Error,
InitError = A::InitError, InitError = A::InitError,
@@ -319,7 +335,7 @@ mod tests {
let cnt2 = cnt.clone(); let cnt2 = cnt.clone();
let blank = move || Ok::<_, ()>(Srv1(cnt2.clone())); let blank = move || Ok::<_, ()>(Srv1(cnt2.clone()));
let new_srv = blank.into_new_service().then(move || Ok(Srv2(cnt.clone()))); let new_srv = blank.into_new_service().then(move || Ok(Srv2(cnt.clone())));
if let Async::Ready(mut srv) = new_srv.clone().new_service().poll().unwrap() { if let Async::Ready(mut srv) = new_srv.clone().new_service(&()).poll().unwrap() {
let res = srv.call(Ok("srv1")).poll(); let res = srv.call(Ok("srv1")).poll();
assert!(res.is_ok()); assert!(res.is_ok());
assert_eq!(res.unwrap(), Async::Ready(("srv1", "ok"))); assert_eq!(res.unwrap(), Async::Ready(("srv1", "ok")));

View File

@@ -0,0 +1,98 @@
use std::rc::Rc;
use std::sync::Arc;
use futures::IntoFuture;
use crate::transform_map_init_err::TransformMapInitErr;
use crate::Service;
/// `Transform` service factory.
///
/// Transform factory creates service that wraps other services.
/// `Config` is a service factory configuration type.
pub trait Transform<S> {
/// Requests handled by the service.
type Request;
/// Responses given by the service.
type Response;
/// Errors produced by the service.
type Error;
/// The `TransformService` value created by this factory
type Transform: Service<
Request = Self::Request,
Response = Self::Response,
Error = Self::Error,
>;
/// Errors produced while building a service.
type InitError;
/// The future response value.
type Future: IntoFuture<Item = Self::Transform, Error = Self::InitError>;
/// Create and return a new service value asynchronously.
fn new_transform(&self, service: S) -> Self::Future;
/// Map this service's factory init error to a different error,
/// returning a new transform service factory.
fn map_init_err<F, E>(self, f: F) -> TransformMapInitErr<Self, S, F, E>
where
Self: Sized,
F: Fn(Self::InitError) -> E,
{
TransformMapInitErr::new(self, f)
}
}
impl<T, S> Transform<S> for Rc<T>
where
T: Transform<S>,
{
type Request = T::Request;
type Response = T::Response;
type Error = T::Error;
type InitError = T::InitError;
type Transform = T::Transform;
type Future = T::Future;
fn new_transform(&self, service: S) -> T::Future {
self.as_ref().new_transform(service)
}
}
impl<T, S> Transform<S> for Arc<T>
where
T: Transform<S>,
{
type Request = T::Request;
type Response = T::Response;
type Error = T::Error;
type InitError = T::InitError;
type Transform = T::Transform;
type Future = T::Future;
fn new_transform(&self, service: S) -> T::Future {
self.as_ref().new_transform(service)
}
}
/// Trait for types that can be converted to a *transform service*
pub trait IntoTransform<T, S>
where
T: Transform<S>,
{
/// Convert to a `TransformService`
fn into_transform(self) -> T;
}
impl<T, S> IntoTransform<T, S> for T
where
T: Transform<S>,
{
fn into_transform(self) -> T {
self
}
}

View File

@@ -0,0 +1,97 @@
use std::marker::PhantomData;
use futures::{Future, IntoFuture, Poll};
use super::Transform;
/// NewTransform for the `map_init_err` combinator, changing the type of a new
/// transform's error.
///
/// This is created by the `NewTransform::map_init_err` method.
pub struct TransformMapInitErr<T, S, F, E> {
t: T,
f: F,
e: PhantomData<(S, E)>,
}
impl<T, S, F, E> TransformMapInitErr<T, S, F, E> {
/// Create new `MapInitErr` new transform instance
pub fn new(t: T, f: F) -> Self
where
T: Transform<S>,
F: Fn(T::InitError) -> E,
{
Self {
t,
f,
e: PhantomData,
}
}
}
impl<T, S, F, E> Clone for TransformMapInitErr<T, S, F, E>
where
T: Clone,
F: Clone,
{
fn clone(&self) -> Self {
Self {
t: self.t.clone(),
f: self.f.clone(),
e: PhantomData,
}
}
}
impl<T, S, F, E> Transform<S> for TransformMapInitErr<T, S, F, E>
where
T: Transform<S>,
F: Fn(T::InitError) -> E + Clone,
{
type Request = T::Request;
type Response = T::Response;
type Error = T::Error;
type Transform = T::Transform;
type InitError = E;
type Future = TransformMapInitErrFuture<T, S, F, E>;
fn new_transform(&self, service: S) -> Self::Future {
TransformMapInitErrFuture::new(
self.t.new_transform(service).into_future(),
self.f.clone(),
)
}
}
pub struct TransformMapInitErrFuture<T, S, F, E>
where
T: Transform<S>,
F: Fn(T::InitError) -> E,
{
fut: <T::Future as IntoFuture>::Future,
f: F,
}
impl<T, S, F, E> TransformMapInitErrFuture<T, S, F, E>
where
T: Transform<S>,
F: Fn(T::InitError) -> E,
{
fn new(fut: <T::Future as IntoFuture>::Future, f: F) -> Self {
TransformMapInitErrFuture { f, fut }
}
}
impl<T, S, F, E> Future for TransformMapInitErrFuture<T, S, F, E>
where
T: Transform<S>,
F: Fn(T::InitError) -> E + Clone,
{
type Item = T::Transform;
type Error = E;
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
self.fut.poll().map_err(&self.f)
}
}

View File

@@ -34,12 +34,9 @@ rust-tls = ["rustls", "tokio-rustls", "webpki", "webpki-roots"]
[dependencies] [dependencies]
actix-rt = "0.1.0" actix-rt = "0.1.0"
#actix-server = "0.1.0" actix-server = "0.3.0"
actix-server = { path="../actix-server" }
log = "0.4" log = "0.4"
# io
net2 = "0.2" net2 = "0.2"
futures = "0.1" futures = "0.1"
tokio-tcp = "0.1" tokio-tcp = "0.1"

View File

@@ -1,5 +1,56 @@
# Changes # Changes
## [0.3.1] - 2019-03-04
### Changed
* Use new type of transform trait
## [0.3.0] - 2019-03-02
### Changed
* Use new `NewService` trait
* BoxedNewService` and `BoxedService` types moved to actix-service crate.
## [0.2.4] - 2019-02-21
### Changed
* Custom `BoxedNewService` implementation.
## [0.2.3] - 2019-02-21
### Added
* Add `BoxedNewService` and `BoxedService`
## [0.2.2] - 2019-02-11
### Added
* Add `Display` impl for `TimeoutError`
* Add `Display` impl for `InOrderError`
## [0.2.1] - 2019-02-06
### Added
* Add `InOrder` service. the service yields responses as they become available,
in the order that their originating requests were submitted to the service.
### Changed
* Convert `Timeout` and `InFlight` services to a transforms
## [0.2.0] - 2019-02-01 ## [0.2.0] - 2019-02-01
* Fix framed transport error handling * Fix framed transport error handling

View File

@@ -1,6 +1,6 @@
[package] [package]
name = "actix-utils" name = "actix-utils"
version = "0.2.0" version = "0.3.1"
authors = ["Nikolay Kim <fafhrd91@gmail.com>"] authors = ["Nikolay Kim <fafhrd91@gmail.com>"]
description = "Actix utils - various actix net related services" description = "Actix utils - various actix net related services"
keywords = ["network", "framework", "async", "futures"] keywords = ["network", "framework", "async", "futures"]
@@ -11,20 +11,20 @@ categories = ["network-programming", "asynchronous"]
license = "MIT/Apache-2.0" license = "MIT/Apache-2.0"
exclude = [".gitignore", ".travis.yml", ".cargo/config", "appveyor.yml"] exclude = [".gitignore", ".travis.yml", ".cargo/config", "appveyor.yml"]
edition = "2018" edition = "2018"
workspace = "../" workspace = ".."
[lib] [lib]
name = "actix_utils" name = "actix_utils"
path = "src/lib.rs" path = "src/lib.rs"
[dependencies] [dependencies]
actix-service = "0.2.0" actix-service = "0.3.1"
actix-codec = "0.1.0" actix-codec = "0.1.0"
bytes = "0.4" bytes = "0.4"
futures = "0.1" futures = "0.1.24"
tokio-timer = "0.2.8" tokio-timer = "0.2.8"
tokio-current-thread = "0.1" tokio-current-thread = "0.1.4"
log = "0.4" log = "0.4"
[dev-dependencies] [dev-dependencies]
actix-rt = "0.1" actix-rt = "0.1"

View File

@@ -53,10 +53,11 @@ pub enum Either<A, B> {
} }
impl<A, B> Either<A, B> { impl<A, B> Either<A, B> {
pub fn new_a(srv: A) -> Self pub fn new_a<C>(srv: A) -> Self
where where
A: NewService, A: NewService<C>,
B: NewService< B: NewService<
C,
Request = A::Request, Request = A::Request,
Response = A::Response, Response = A::Response,
Error = A::Error, Error = A::Error,
@@ -66,10 +67,11 @@ impl<A, B> Either<A, B> {
Either::A(srv) Either::A(srv)
} }
pub fn new_b(srv: B) -> Self pub fn new_b<C>(srv: B) -> Self
where where
A: NewService, A: NewService<C>,
B: NewService< B: NewService<
C,
Request = A::Request, Request = A::Request,
Response = A::Response, Response = A::Response,
Error = A::Error, Error = A::Error,
@@ -80,10 +82,11 @@ impl<A, B> Either<A, B> {
} }
} }
impl<A, B> NewService for Either<A, B> impl<A, B, C> NewService<C> for Either<A, B>
where where
A: NewService, A: NewService<C>,
B: NewService< B: NewService<
C,
Request = A::Request, Request = A::Request,
Response = A::Response, Response = A::Response,
Error = A::Error, Error = A::Error,
@@ -95,12 +98,12 @@ where
type Error = A::Error; type Error = A::Error;
type InitError = A::InitError; type InitError = A::InitError;
type Service = EitherService<A::Service, B::Service>; type Service = EitherService<A::Service, B::Service>;
type Future = EitherNewService<A, B>; type Future = EitherNewService<A, B, C>;
fn new_service(&self) -> Self::Future { fn new_service(&self, cfg: &C) -> Self::Future {
match self { match self {
Either::A(ref inner) => EitherNewService::A(inner.new_service()), Either::A(ref inner) => EitherNewService::A(inner.new_service(cfg)),
Either::B(ref inner) => EitherNewService::B(inner.new_service()), Either::B(ref inner) => EitherNewService::B(inner.new_service(cfg)),
} }
} }
} }
@@ -115,15 +118,16 @@ impl<A: Clone, B: Clone> Clone for Either<A, B> {
} }
#[doc(hidden)] #[doc(hidden)]
pub enum EitherNewService<A: NewService, B: NewService> { pub enum EitherNewService<A: NewService<C>, B: NewService<C>, C> {
A(A::Future), A(A::Future),
B(B::Future), B(B::Future),
} }
impl<A, B> Future for EitherNewService<A, B> impl<A, B, C> Future for EitherNewService<A, B, C>
where where
A: NewService, A: NewService<C>,
B: NewService< B: NewService<
C,
Request = A::Request, Request = A::Request,
Response = A::Response, Response = A::Response,
Error = A::Error, Error = A::Error,

View File

@@ -15,14 +15,15 @@ use crate::cell::Cell;
type Request<U> = <U as Decoder>::Item; type Request<U> = <U as Decoder>::Item;
type Response<U> = <U as Encoder>::Item; type Response<U> = <U as Encoder>::Item;
pub struct FramedNewService<S, T, U> { pub struct FramedNewService<S, T, U, C> {
factory: S, factory: S,
_t: PhantomData<(T, U)>, _t: PhantomData<(T, U, C)>,
} }
impl<S, T, U> FramedNewService<S, T, U> impl<S, T, U, C> FramedNewService<S, T, U, C>
where where
S: NewService<Request = Request<U>, Response = Response<U>>, C: Clone,
S: NewService<C, Request = Request<U>, Response = Response<U>>,
S::Error: 'static, S::Error: 'static,
<S::Service as Service>::Future: 'static, <S::Service as Service>::Future: 'static,
T: AsyncRead + AsyncWrite, T: AsyncRead + AsyncWrite,
@@ -30,7 +31,7 @@ where
<U as Encoder>::Item: 'static, <U as Encoder>::Item: 'static,
<U as Encoder>::Error: std::fmt::Debug, <U as Encoder>::Error: std::fmt::Debug,
{ {
pub fn new<F1: IntoNewService<S>>(factory: F1) -> Self { pub fn new<F1: IntoNewService<S, C>>(factory: F1) -> Self {
Self { Self {
factory: factory.into_new_service(), factory: factory.into_new_service(),
_t: PhantomData, _t: PhantomData,
@@ -38,7 +39,7 @@ where
} }
} }
impl<S, T, U> Clone for FramedNewService<S, T, U> impl<S, T, U, C> Clone for FramedNewService<S, T, U, C>
where where
S: Clone, S: Clone,
{ {
@@ -50,9 +51,10 @@ where
} }
} }
impl<S, T, U> NewService for FramedNewService<S, T, U> impl<S, T, U, C> NewService<C> for FramedNewService<S, T, U, C>
where where
S: NewService<Request = Request<U>, Response = Response<U>> + Clone, C: Clone,
S: NewService<C, Request = Request<U>, Response = Response<U>> + Clone,
S::Error: 'static, S::Error: 'static,
<S::Service as Service>::Future: 'static, <S::Service as Service>::Future: 'static,
T: AsyncRead + AsyncWrite, T: AsyncRead + AsyncWrite,
@@ -64,48 +66,53 @@ where
type Response = FramedTransport<S::Service, T, U>; type Response = FramedTransport<S::Service, T, U>;
type Error = S::InitError; type Error = S::InitError;
type InitError = S::InitError; type InitError = S::InitError;
type Service = FramedService<S, T, U>; type Service = FramedService<S, T, U, C>;
type Future = FutureResult<Self::Service, Self::InitError>; type Future = FutureResult<Self::Service, Self::InitError>;
fn new_service(&self) -> Self::Future { fn new_service(&self, cfg: &C) -> Self::Future {
ok(FramedService { ok(FramedService {
factory: self.factory.clone(), factory: self.factory.clone(),
config: cfg.clone(),
_t: PhantomData, _t: PhantomData,
}) })
} }
} }
pub struct FramedService<S, T, U> { pub struct FramedService<S, T, U, C> {
factory: S, factory: S,
config: C,
_t: PhantomData<(T, U)>, _t: PhantomData<(T, U)>,
} }
impl<S, T, U> Clone for FramedService<S, T, U> impl<S, T, U, C> Clone for FramedService<S, T, U, C>
where where
S: Clone, S: Clone,
C: Clone,
{ {
fn clone(&self) -> Self { fn clone(&self) -> Self {
Self { Self {
factory: self.factory.clone(), factory: self.factory.clone(),
config: self.config.clone(),
_t: PhantomData, _t: PhantomData,
} }
} }
} }
impl<S, T, U> Service for FramedService<S, T, U> impl<S, T, U, C> Service for FramedService<S, T, U, C>
where where
S: NewService<Request = Request<U>, Response = Response<U>>, S: NewService<C, Request = Request<U>, Response = Response<U>>,
S::Error: 'static, S::Error: 'static,
<S::Service as Service>::Future: 'static, <S::Service as Service>::Future: 'static,
T: AsyncRead + AsyncWrite, T: AsyncRead + AsyncWrite,
U: Decoder + Encoder, U: Decoder + Encoder,
<U as Encoder>::Item: 'static, <U as Encoder>::Item: 'static,
<U as Encoder>::Error: std::fmt::Debug, <U as Encoder>::Error: std::fmt::Debug,
C: Clone,
{ {
type Request = Framed<T, U>; type Request = Framed<T, U>;
type Response = FramedTransport<S::Service, T, U>; type Response = FramedTransport<S::Service, T, U>;
type Error = S::InitError; type Error = S::InitError;
type Future = FramedServiceResponseFuture<S, T, U>; type Future = FramedServiceResponseFuture<S, T, U, C>;
fn poll_ready(&mut self) -> Poll<(), Self::Error> { fn poll_ready(&mut self) -> Poll<(), Self::Error> {
Ok(Async::Ready(())) Ok(Async::Ready(()))
@@ -113,17 +120,16 @@ where
fn call(&mut self, req: Framed<T, U>) -> Self::Future { fn call(&mut self, req: Framed<T, U>) -> Self::Future {
FramedServiceResponseFuture { FramedServiceResponseFuture {
fut: self.factory.new_service(), fut: self.factory.new_service(&self.config),
framed: Some(req), framed: Some(req),
} }
} }
} }
#[doc(hidden)] #[doc(hidden)]
pub struct FramedServiceResponseFuture<S, T, U> pub struct FramedServiceResponseFuture<S, T, U, C>
where where
S: NewService<Request = Request<U>, Response = Response<U>>, S: NewService<C, Request = Request<U>, Response = Response<U>>,
S::Error: 'static, S::Error: 'static,
<S::Service as Service>::Future: 'static, <S::Service as Service>::Future: 'static,
T: AsyncRead + AsyncWrite, T: AsyncRead + AsyncWrite,
@@ -135,9 +141,9 @@ where
framed: Option<Framed<T, U>>, framed: Option<Framed<T, U>>,
} }
impl<S, T, U> Future for FramedServiceResponseFuture<S, T, U> impl<S, T, U, C> Future for FramedServiceResponseFuture<S, T, U, C>
where where
S: NewService<Request = Request<U>, Response = Response<U>>, S: NewService<C, Request = Request<U>, Response = Response<U>>,
S::Error: 'static, S::Error: 'static,
<S::Service as Service>::Future: 'static, <S::Service as Service>::Future: 'static,
T: AsyncRead + AsyncWrite, T: AsyncRead + AsyncWrite,
@@ -402,7 +408,7 @@ where
} }
} }
impl<T, U, F> NewService for IntoFramed<T, U, F> impl<T, U, F> NewService<()> for IntoFramed<T, U, F>
where where
T: AsyncRead + AsyncWrite, T: AsyncRead + AsyncWrite,
F: Fn() -> U + Send + Clone + 'static, F: Fn() -> U + Send + Clone + 'static,
@@ -415,7 +421,7 @@ where
type Service = IntoFramedService<T, U, F>; type Service = IntoFramedService<T, U, F>;
type Future = FutureResult<Self::Service, Self::InitError>; type Future = FutureResult<Self::Service, Self::InitError>;
fn new_service(&self) -> Self::Future { fn new_service(&self, _: &()) -> Self::Future {
ok(IntoFramedService { ok(IntoFramedService {
factory: self.factory.clone(), factory: self.factory.clone(),
_t: PhantomData, _t: PhantomData,

View File

@@ -1,5 +1,6 @@
use actix_service::{IntoNewService, IntoService, NewService, Service}; use actix_service::{Service, Transform, Void};
use futures::{try_ready, Async, Future, Poll}; use futures::future::{ok, FutureResult};
use futures::{Async, Future, Poll};
use super::counter::{Counter, CounterGuard}; use super::counter::{Counter, CounterGuard};
@@ -7,92 +8,44 @@ use super::counter::{Counter, CounterGuard};
/// async requests. /// async requests.
/// ///
/// Default number of in-flight requests is 15 /// Default number of in-flight requests is 15
pub struct InFlight<T> { pub struct InFlight {
factory: T,
max_inflight: usize, max_inflight: usize,
} }
impl<T> InFlight<T> { impl InFlight {
pub fn new<F>(factory: F) -> Self pub fn new(max: usize) -> Self {
where Self { max_inflight: max }
T: NewService,
F: IntoNewService<T>,
{
Self {
factory: factory.into_new_service(),
max_inflight: 15,
}
}
/// Set max number of in-flight requests.
///
/// By default max in-flight requests is 15.
pub fn max_inflight(mut self, max: usize) -> Self {
self.max_inflight = max;
self
} }
} }
impl<T> NewService for InFlight<T> impl Default for InFlight {
where fn default() -> Self {
T: NewService, Self::new(15)
{
type Request = T::Request;
type Response = T::Response;
type Error = T::Error;
type InitError = T::InitError;
type Service = InFlightService<T::Service>;
type Future = InFlightResponseFuture<T>;
fn new_service(&self) -> Self::Future {
InFlightResponseFuture {
fut: self.factory.new_service(),
max_inflight: self.max_inflight,
}
} }
} }
pub struct InFlightResponseFuture<T: NewService> { impl<S: Service> Transform<S> for InFlight {
fut: T::Future, type Request = S::Request;
max_inflight: usize, type Response = S::Response;
} type Error = S::Error;
type InitError = Void;
type Transform = InFlightService<S>;
type Future = FutureResult<Self::Transform, Self::InitError>;
impl<T: NewService> Future for InFlightResponseFuture<T> { fn new_transform(&self, service: S) -> Self::Future {
type Item = InFlightService<T::Service>; ok(InFlightService::new(self.max_inflight, service))
type Error = T::InitError;
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
Ok(Async::Ready(InFlightService::with_max_inflight(
self.max_inflight,
try_ready!(self.fut.poll()),
)))
} }
} }
pub struct InFlightService<T> { pub struct InFlightService<S> {
service: T,
count: Counter, count: Counter,
service: S,
} }
impl<T> InFlightService<T> { impl<S> InFlightService<S> {
pub fn new<F>(service: F) -> Self pub fn new(max: usize, service: S) -> Self {
where
T: Service,
F: IntoService<T>,
{
Self { Self {
service: service.into_service(), service,
count: Counter::new(15),
}
}
pub fn with_max_inflight<F>(max: usize, service: F) -> Self
where
T: Service,
F: IntoService<T>,
{
Self {
service: service.into_service(),
count: Counter::new(max), count: Counter::new(max),
} }
} }
@@ -108,12 +61,14 @@ where
type Future = InFlightServiceResponse<T>; type Future = InFlightServiceResponse<T>;
fn poll_ready(&mut self) -> Poll<(), Self::Error> { fn poll_ready(&mut self) -> Poll<(), Self::Error> {
let res = self.service.poll_ready()?; self.service.poll_ready()?;
if res.is_ready() && !self.count.available() {
if !self.count.available() {
log::trace!("InFlight limit exceeded"); log::trace!("InFlight limit exceeded");
return Ok(Async::NotReady); Ok(Async::NotReady)
} else {
Ok(Async::Ready(()))
} }
Ok(res)
} }
fn call(&mut self, req: T::Request) -> Self::Future { fn call(&mut self, req: T::Request) -> Self::Future {
@@ -138,3 +93,75 @@ impl<T: Service> Future for InFlightServiceResponse<T> {
self.fut.poll() self.fut.poll()
} }
} }
#[cfg(test)]
mod tests {
use futures::future::lazy;
use futures::{Async, Poll};
use std::time::Duration;
use super::*;
use actix_service::blank::{Blank, BlankNewService};
use actix_service::{NewService, Service, ServiceExt};
struct SleepService(Duration);
impl Service for SleepService {
type Request = ();
type Response = ();
type Error = ();
type Future = Box<Future<Item = (), Error = ()>>;
fn poll_ready(&mut self) -> Poll<(), Self::Error> {
Ok(Async::Ready(()))
}
fn call(&mut self, _: ()) -> Self::Future {
Box::new(tokio_timer::sleep(self.0).map_err(|_| ()))
}
}
#[test]
fn test_transform() {
let wait_time = Duration::from_millis(50);
let _ = actix_rt::System::new("test").block_on(lazy(|| {
let mut srv =
Blank::new().and_then(InFlightService::new(1, SleepService(wait_time)));
assert_eq!(srv.poll_ready(), Ok(Async::Ready(())));
let mut res = srv.call(());
let _ = res.poll();
assert_eq!(srv.poll_ready(), Ok(Async::NotReady));
drop(res);
assert_eq!(srv.poll_ready(), Ok(Async::Ready(())));
Ok::<_, ()>(())
}));
}
#[test]
fn test_newtransform() {
let wait_time = Duration::from_millis(50);
let _ = actix_rt::System::new("test").block_on(lazy(|| {
let srv =
BlankNewService::new().apply(InFlight::new(1), || Ok(SleepService(wait_time)));
if let Async::Ready(mut srv) = srv.new_service(&()).poll().unwrap() {
assert_eq!(srv.poll_ready(), Ok(Async::Ready(())));
let mut res = srv.call(());
let _ = res.poll();
assert_eq!(srv.poll_ready(), Ok(Async::NotReady));
drop(res);
assert_eq!(srv.poll_ready(), Ok(Async::Ready(())));
} else {
panic!()
}
Ok::<_, ()>(())
}));
}
}

View File

@@ -1,13 +1,12 @@
use std::marker::PhantomData; use std::marker::PhantomData;
use std::time::{Duration, Instant}; use std::time::{Duration, Instant};
use actix_service::{NewService, Service}; use actix_service::{NewService, Service, Void};
use futures::future::{ok, FutureResult}; use futures::future::{ok, FutureResult};
use futures::{Async, Future, Poll}; use futures::{Async, Future, Poll};
use tokio_timer::Delay; use tokio_timer::Delay;
use super::time::{LowResTime, LowResTimeService}; use super::time::{LowResTime, LowResTimeService};
use super::Never;
pub struct KeepAlive<R, E, F> { pub struct KeepAlive<R, E, F> {
f: F, f: F,
@@ -44,18 +43,18 @@ where
} }
} }
impl<R, E, F> NewService for KeepAlive<R, E, F> impl<R, E, F> NewService<()> for KeepAlive<R, E, F>
where where
F: Fn() -> E + Clone, F: Fn() -> E + Clone,
{ {
type Request = R; type Request = R;
type Response = R; type Response = R;
type Error = E; type Error = E;
type InitError = Never; type InitError = Void;
type Service = KeepAliveService<R, E, F>; type Service = KeepAliveService<R, E, F>;
type Future = FutureResult<Self::Service, Self::InitError>; type Future = FutureResult<Self::Service, Self::InitError>;
fn new_service(&self) -> Self::Future { fn new_service(&self, _: &()) -> Self::Future {
ok(KeepAliveService::new( ok(KeepAliveService::new(
self.ka, self.ka,
self.time.timer(), self.time.timer(),
@@ -112,7 +111,7 @@ where
} }
} }
Ok(Async::NotReady) => Ok(Async::Ready(())), Ok(Async::NotReady) => Ok(Async::Ready(())),
Err(_) => panic!(), Err(_e) => panic!(),
} }
} }

View File

@@ -6,9 +6,7 @@ pub mod either;
pub mod framed; pub mod framed;
pub mod inflight; pub mod inflight;
pub mod keepalive; pub mod keepalive;
pub mod order;
pub mod stream; pub mod stream;
pub mod time; pub mod time;
pub mod timeout; pub mod timeout;
#[derive(Copy, Clone, Debug)]
pub enum Never {}

273
actix-utils/src/order.rs Normal file
View File

@@ -0,0 +1,273 @@
use std::collections::VecDeque;
use std::fmt;
use std::marker::PhantomData;
use std::rc::Rc;
use actix_service::{Service, Transform, Void};
use futures::future::{ok, FutureResult};
use futures::task::AtomicTask;
use futures::unsync::oneshot;
use futures::{Async, Future, Poll};
struct Record<I, E> {
rx: oneshot::Receiver<Result<I, E>>,
tx: oneshot::Sender<Result<I, E>>,
}
/// Timeout error
pub enum InOrderError<E> {
/// Service error
Service(E),
/// Service call dropped
Disconnected,
}
impl<E> From<E> for InOrderError<E> {
fn from(err: E) -> Self {
InOrderError::Service(err)
}
}
impl<E: fmt::Debug> fmt::Debug for InOrderError<E> {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
match self {
InOrderError::Service(e) => write!(f, "InOrderError::Service({:?})", e),
InOrderError::Disconnected => write!(f, "InOrderError::Disconnected"),
}
}
}
impl<E: fmt::Display> fmt::Display for InOrderError<E> {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
match self {
InOrderError::Service(e) => e.fmt(f),
InOrderError::Disconnected => write!(f, "InOrder service disconnected"),
}
}
}
/// InOrder - The service will yield responses as they become available,
/// in the order that their originating requests were submitted to the service.
pub struct InOrder<S> {
_t: PhantomData<S>,
}
impl<S> InOrder<S>
where
S: Service,
S::Response: 'static,
S::Future: 'static,
S::Error: 'static,
{
pub fn new() -> Self {
Self { _t: PhantomData }
}
pub fn service(service: S) -> InOrderService<S> {
InOrderService::new(service)
}
}
impl<S> Default for InOrder<S>
where
S: Service,
S::Response: 'static,
S::Future: 'static,
S::Error: 'static,
{
fn default() -> Self {
Self::new()
}
}
impl<S> Transform<S> for InOrder<S>
where
S: Service,
S::Response: 'static,
S::Future: 'static,
S::Error: 'static,
{
type Request = S::Request;
type Response = S::Response;
type Error = InOrderError<S::Error>;
type InitError = Void;
type Transform = InOrderService<S>;
type Future = FutureResult<Self::Transform, Self::InitError>;
fn new_transform(&self, service: S) -> Self::Future {
ok(InOrderService::new(service))
}
}
pub struct InOrderService<S: Service> {
service: S,
task: Rc<AtomicTask>,
acks: VecDeque<Record<S::Response, S::Error>>,
}
impl<S> InOrderService<S>
where
S: Service,
S::Response: 'static,
S::Future: 'static,
S::Error: 'static,
{
pub fn new(service: S) -> Self {
Self {
service,
acks: VecDeque::new(),
task: Rc::new(AtomicTask::new()),
}
}
}
impl<S> Service for InOrderService<S>
where
S: Service,
S::Response: 'static,
S::Future: 'static,
S::Error: 'static,
{
type Request = S::Request;
type Response = S::Response;
type Error = InOrderError<S::Error>;
type Future = InOrderServiceResponse<S>;
fn poll_ready(&mut self) -> Poll<(), Self::Error> {
// poll_ready could be called from different task
self.task.register();
// check nested service
self.service.poll_ready().map_err(InOrderError::Service)?;
// check acks
while !self.acks.is_empty() {
let rec = self.acks.front_mut().unwrap();
match rec.rx.poll() {
Ok(Async::Ready(res)) => {
let rec = self.acks.pop_front().unwrap();
let _ = rec.tx.send(res);
}
Ok(Async::NotReady) => break,
Err(oneshot::Canceled) => return Err(InOrderError::Disconnected),
}
}
Ok(Async::Ready(()))
}
fn call(&mut self, request: S::Request) -> Self::Future {
let (tx1, rx1) = oneshot::channel();
let (tx2, rx2) = oneshot::channel();
self.acks.push_back(Record { rx: rx1, tx: tx2 });
let task = self.task.clone();
tokio_current_thread::spawn(self.service.call(request).then(move |res| {
task.notify();
let _ = tx1.send(res);
Ok(())
}));
InOrderServiceResponse { rx: rx2 }
}
}
#[doc(hidden)]
pub struct InOrderServiceResponse<S: Service> {
rx: oneshot::Receiver<Result<S::Response, S::Error>>,
}
impl<S: Service> Future for InOrderServiceResponse<S> {
type Item = S::Response;
type Error = InOrderError<S::Error>;
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
match self.rx.poll() {
Ok(Async::NotReady) => Ok(Async::NotReady),
Ok(Async::Ready(Ok(res))) => Ok(Async::Ready(res)),
Ok(Async::Ready(Err(e))) => Err(e.into()),
Err(oneshot::Canceled) => Err(InOrderError::Disconnected),
}
}
}
#[cfg(test)]
mod tests {
use futures::future::{lazy, Future};
use futures::{stream::futures_unordered, sync::oneshot, Async, Poll, Stream};
use std::time::Duration;
use super::*;
use actix_service::blank::Blank;
use actix_service::{Service, ServiceExt};
struct Srv;
impl Service for Srv {
type Request = oneshot::Receiver<usize>;
type Response = usize;
type Error = ();
type Future = Box<Future<Item = usize, Error = ()>>;
fn poll_ready(&mut self) -> Poll<(), Self::Error> {
Ok(Async::Ready(()))
}
fn call(&mut self, req: oneshot::Receiver<usize>) -> Self::Future {
Box::new(req.map_err(|_| ()))
}
}
struct SrvPoll<S: Service> {
s: S,
}
impl<S: Service> Future for SrvPoll<S> {
type Item = ();
type Error = ();
fn poll(&mut self) -> Poll<(), ()> {
let _ = self.s.poll_ready();
Ok(Async::NotReady)
}
}
#[test]
fn test_inorder() {
let (tx1, rx1) = oneshot::channel();
let (tx2, rx2) = oneshot::channel();
let (tx3, rx3) = oneshot::channel();
let (tx_stop, rx_stop) = oneshot::channel();
std::thread::spawn(move || {
let rx1 = rx1;
let rx2 = rx2;
let rx3 = rx3;
let tx_stop = tx_stop;
let _ = actix_rt::System::new("test").block_on(lazy(move || {
let mut srv = Blank::new().and_then(InOrderService::new(Srv));
let res1 = srv.call(rx1);
let res2 = srv.call(rx2);
let res3 = srv.call(rx3);
tokio_current_thread::spawn(SrvPoll { s: srv });
futures_unordered(vec![res1, res2, res3])
.collect()
.and_then(move |res: Vec<_>| {
assert_eq!(res, vec![1, 2, 3]);
let _ = tx_stop.send(());
Ok(())
})
}));
});
let _ = tx3.send(3);
std::thread::sleep(Duration::from_millis(50));
let _ = tx2.send(2);
let _ = tx1.send(1);
let _ = rx_stop.wait();
}
}

View File

@@ -29,20 +29,21 @@ where
} }
} }
pub struct StreamNewService<S, T, E> { pub struct StreamNewService<S, T, E, C> {
factory: Rc<T>, factory: Rc<T>,
_t: PhantomData<(S, E)>, _t: PhantomData<(S, E, C)>,
} }
impl<S, T, E> StreamNewService<S, T, E> impl<S, T, E, C> StreamNewService<S, T, E, C>
where where
C: Clone,
S: IntoStream, S: IntoStream,
T: NewService<Request = Request<S>, Response = (), Error = E, InitError = E>, T: NewService<C, Request = Request<S>, Response = (), Error = E, InitError = E>,
T::Future: 'static, T::Future: 'static,
T::Service: 'static, T::Service: 'static,
<T::Service as Service>::Future: 'static, <T::Service as Service>::Future: 'static,
{ {
pub fn new<F: IntoNewService<T>>(factory: F) -> Self { pub fn new<F: IntoNewService<T, C>>(factory: F) -> Self {
Self { Self {
factory: Rc::new(factory.into_new_service()), factory: Rc::new(factory.into_new_service()),
_t: PhantomData, _t: PhantomData,
@@ -50,7 +51,7 @@ where
} }
} }
impl<S, T, E> Clone for StreamNewService<S, T, E> { impl<S, T, E, C> Clone for StreamNewService<S, T, E, C> {
fn clone(&self) -> Self { fn clone(&self) -> Self {
Self { Self {
factory: self.factory.clone(), factory: self.factory.clone(),
@@ -59,10 +60,11 @@ impl<S, T, E> Clone for StreamNewService<S, T, E> {
} }
} }
impl<S, T, E> NewService for StreamNewService<S, T, E> impl<S, T, E, C> NewService<C> for StreamNewService<S, T, E, C>
where where
C: Clone,
S: IntoStream + 'static, S: IntoStream + 'static,
T: NewService<Request = Request<S>, Response = (), Error = E, InitError = E>, T: NewService<C, Request = Request<S>, Response = (), Error = E, InitError = E>,
T::Future: 'static, T::Future: 'static,
T::Service: 'static, T::Service: 'static,
<T::Service as Service>::Future: 'static, <T::Service as Service>::Future: 'static,
@@ -71,29 +73,32 @@ where
type Response = (); type Response = ();
type Error = E; type Error = E;
type InitError = E; type InitError = E;
type Service = StreamService<S, T, E>; type Service = StreamService<S, T, E, C>;
type Future = FutureResult<Self::Service, E>; type Future = FutureResult<Self::Service, E>;
fn new_service(&self) -> Self::Future { fn new_service(&self, cfg: &C) -> Self::Future {
ok(StreamService { ok(StreamService {
factory: self.factory.clone(), factory: self.factory.clone(),
config: cfg.clone(),
_t: PhantomData, _t: PhantomData,
}) })
} }
} }
pub struct StreamService<S, T, E> { pub struct StreamService<S, T, E, C = ()> {
factory: Rc<T>, factory: Rc<T>,
config: C,
_t: PhantomData<(S, E)>, _t: PhantomData<(S, E)>,
} }
impl<S, T, E> Service for StreamService<S, T, E> impl<S, T, E, C> Service for StreamService<S, T, E, C>
where where
S: IntoStream + 'static, S: IntoStream + 'static,
T: NewService<Request = Request<S>, Response = (), Error = E, InitError = E>, T: NewService<C, Request = Request<S>, Response = (), Error = E, InitError = E>,
T::Future: 'static, T::Future: 'static,
T::Service: 'static, T::Service: 'static,
<T::Service as Service>::Future: 'static, <T::Service as Service>::Future: 'static,
C: Clone,
{ {
type Request = S; type Request = S;
type Response = (); type Response = ();
@@ -107,7 +112,7 @@ where
fn call(&mut self, req: S) -> Self::Future { fn call(&mut self, req: S) -> Self::Future {
Box::new( Box::new(
self.factory self.factory
.new_service() .new_service(&self.config)
.and_then(move |srv| StreamDispatcher::new(req, srv)), .and_then(move |srv| StreamDispatcher::new(req, srv)),
) )
} }
@@ -227,7 +232,7 @@ impl<T> Clone for TakeItem<T> {
} }
} }
impl<T: Stream> NewService for TakeItem<T> { impl<T: Stream> NewService<()> for TakeItem<T> {
type Request = T; type Request = T;
type Response = (Option<T::Item>, T); type Response = (Option<T::Item>, T);
type Error = T::Error; type Error = T::Error;
@@ -235,7 +240,7 @@ impl<T: Stream> NewService for TakeItem<T> {
type Service = TakeItemService<T>; type Service = TakeItemService<T>;
type Future = FutureResult<Self::Service, Self::InitError>; type Future = FutureResult<Self::Service, Self::InitError>;
fn new_service(&self) -> Self::Future { fn new_service(&self, _: &()) -> Self::Future {
ok(TakeItemService { _t: PhantomData }) ok(TakeItemService { _t: PhantomData })
} }
} }

View File

@@ -1,12 +1,11 @@
use std::time::{self, Duration, Instant}; use std::time::{self, Duration, Instant};
use actix_service::{NewService, Service}; use actix_service::{NewService, Service, Void};
use futures::future::{ok, FutureResult}; use futures::future::{ok, FutureResult};
use futures::{Async, Future, Poll}; use futures::{Async, Future, Poll};
use tokio_timer::sleep; use tokio_timer::sleep;
use super::cell::Cell; use super::cell::Cell;
use super::Never;
#[derive(Clone, Debug)] #[derive(Clone, Debug)]
pub struct LowResTime(Cell<Inner>); pub struct LowResTime(Cell<Inner>);
@@ -42,15 +41,15 @@ impl Default for LowResTime {
} }
} }
impl NewService for LowResTime { impl NewService<()> for LowResTime {
type Request = (); type Request = ();
type Response = Instant; type Response = Instant;
type Error = Never; type Error = Void;
type InitError = Never; type InitError = Void;
type Service = LowResTimeService; type Service = LowResTimeService;
type Future = FutureResult<Self::Service, Self::InitError>; type Future = FutureResult<Self::Service, Self::InitError>;
fn new_service(&self) -> Self::Future { fn new_service(&self, _: &()) -> Self::Future {
ok(self.timer()) ok(self.timer())
} }
} }
@@ -92,7 +91,7 @@ impl LowResTimeService {
impl Service for LowResTimeService { impl Service for LowResTimeService {
type Request = (); type Request = ();
type Response = Instant; type Response = Instant;
type Error = Never; type Error = Void;
type Future = FutureResult<Self::Response, Self::Error>; type Future = FutureResult<Self::Response, Self::Error>;
fn poll_ready(&mut self) -> Poll<(), Self::Error> { fn poll_ready(&mut self) -> Poll<(), Self::Error> {
@@ -133,7 +132,7 @@ impl SystemTimeService {
/// Get current time. This function has to be called from /// Get current time. This function has to be called from
/// future's poll method, otherwise it panics. /// future's poll method, otherwise it panics.
pub fn now(&self) -> time::SystemTime { pub fn now(&self) -> time::SystemTime {
let cur = self.0.get_ref().current.clone(); let cur = self.0.get_ref().current;
if let Some(cur) = cur { if let Some(cur) = cur {
cur cur
} else { } else {

View File

@@ -3,18 +3,19 @@
//! If the response does not complete within the specified timeout, the response //! If the response does not complete within the specified timeout, the response
//! will be aborted. //! will be aborted.
use std::fmt; use std::fmt;
use std::marker::PhantomData;
use std::time::Duration; use std::time::Duration;
use actix_service::{NewService, Service}; use actix_service::{Service, Transform};
use futures::try_ready; use futures::future::{ok, FutureResult};
use futures::{Async, Future, Poll}; use futures::{Async, Future, Poll};
use tokio_timer::{clock, Delay}; use tokio_timer::{clock, Delay};
/// Applies a timeout to requests. /// Applies a timeout to requests.
#[derive(Debug)] #[derive(Debug)]
pub struct Timeout<T> { pub struct Timeout<E = ()> {
inner: T,
timeout: Duration, timeout: Duration,
_t: PhantomData<E>,
} }
/// Timeout error /// Timeout error
@@ -25,6 +26,12 @@ pub enum TimeoutError<E> {
Timeout, Timeout,
} }
impl<E> From<E> for TimeoutError<E> {
fn from(err: E) -> Self {
TimeoutError::Service(err)
}
}
impl<E: fmt::Debug> fmt::Debug for TimeoutError<E> { impl<E: fmt::Debug> fmt::Debug for TimeoutError<E> {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
match self { match self {
@@ -34,107 +41,93 @@ impl<E: fmt::Debug> fmt::Debug for TimeoutError<E> {
} }
} }
impl<T> Timeout<T> { impl<E: fmt::Display> fmt::Display for TimeoutError<E> {
pub fn new(timeout: Duration, inner: T) -> Self fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
where match self {
T: NewService + Clone, TimeoutError::Service(e) => e.fmt(f),
{ TimeoutError::Timeout => write!(f, "Service call timeout"),
Timeout { inner, timeout } }
} }
} }
impl<T> Clone for Timeout<T> impl<E: PartialEq> PartialEq for TimeoutError<E> {
where fn eq(&self, other: &TimeoutError<E>) -> bool {
T: Clone, match self {
{ TimeoutError::Service(e1) => match other {
fn clone(&self) -> Self { TimeoutError::Service(e2) => e1 == e2,
TimeoutError::Timeout => false,
},
TimeoutError::Timeout => match other {
TimeoutError::Service(_) => false,
TimeoutError::Timeout => true,
},
}
}
}
impl<E> Timeout<E> {
pub fn new(timeout: Duration) -> Self {
Timeout { Timeout {
inner: self.inner.clone(), timeout,
timeout: self.timeout, _t: PhantomData,
} }
} }
} }
impl<T> NewService for Timeout<T> impl<E> Clone for Timeout<E> {
where fn clone(&self) -> Self {
T: NewService + Clone, Timeout::new(self.timeout)
{
type Request = T::Request;
type Response = T::Response;
type Error = TimeoutError<T::Error>;
type InitError = T::InitError;
type Service = TimeoutService<T::Service>;
type Future = TimeoutFut<T>;
fn new_service(&self) -> Self::Future {
TimeoutFut {
fut: self.inner.new_service(),
timeout: self.timeout,
}
} }
} }
/// `Timeout` response future impl<S, E> Transform<S> for Timeout<E>
#[derive(Debug)]
pub struct TimeoutFut<T: NewService> {
fut: T::Future,
timeout: Duration,
}
impl<T> Future for TimeoutFut<T>
where where
T: NewService, S: Service,
{ {
type Item = TimeoutService<T::Service>; type Request = S::Request;
type Error = T::InitError; type Response = S::Response;
type Error = TimeoutError<S::Error>;
type InitError = E;
type Transform = TimeoutService<S>;
type Future = FutureResult<Self::Transform, Self::InitError>;
fn poll(&mut self) -> Poll<Self::Item, Self::Error> { fn new_transform(&self, service: S) -> Self::Future {
let service = try_ready!(self.fut.poll()); ok(TimeoutService {
Ok(Async::Ready(TimeoutService::new(self.timeout, service))) service,
timeout: self.timeout,
})
} }
} }
/// Applies a timeout to requests. /// Applies a timeout to requests.
#[derive(Debug)] #[derive(Debug, Clone)]
pub struct TimeoutService<T> { pub struct TimeoutService<S> {
inner: T, service: S,
timeout: Duration, timeout: Duration,
} }
impl<T> TimeoutService<T> { impl<S> TimeoutService<S> {
pub fn new(timeout: Duration, inner: T) -> Self pub fn new(timeout: Duration, service: S) -> Self {
where TimeoutService { service, timeout }
T: Service,
{
TimeoutService { inner, timeout }
} }
} }
impl<T: Clone> Clone for TimeoutService<T> { impl<S> Service for TimeoutService<S>
fn clone(&self) -> Self {
TimeoutService {
inner: self.inner.clone(),
timeout: self.timeout,
}
}
}
impl<T> Service for TimeoutService<T>
where where
T: Service, S: Service,
{ {
type Request = T::Request; type Request = S::Request;
type Response = T::Response; type Response = S::Response;
type Error = TimeoutError<T::Error>; type Error = TimeoutError<S::Error>;
type Future = TimeoutServiceResponse<T>; type Future = TimeoutServiceResponse<S>;
fn poll_ready(&mut self) -> Poll<(), Self::Error> { fn poll_ready(&mut self) -> Poll<(), Self::Error> {
self.inner.poll_ready().map_err(TimeoutError::Service) self.service.poll_ready().map_err(TimeoutError::Service)
} }
fn call(&mut self, request: T::Request) -> Self::Future { fn call(&mut self, request: S::Request) -> Self::Future {
TimeoutServiceResponse { TimeoutServiceResponse {
fut: self.inner.call(request), fut: self.service.call(request),
sleep: Delay::new(clock::now() + self.timeout), sleep: Delay::new(clock::now() + self.timeout),
} }
} }
@@ -170,3 +163,75 @@ where
} }
} }
} }
#[cfg(test)]
mod tests {
use futures::future::lazy;
use futures::{Async, Poll};
use std::time::Duration;
use super::*;
use actix_service::blank::{Blank, BlankNewService};
use actix_service::{NewService, Service, ServiceExt};
struct SleepService(Duration);
impl Service for SleepService {
type Request = ();
type Response = ();
type Error = ();
type Future = Box<Future<Item = (), Error = ()>>;
fn poll_ready(&mut self) -> Poll<(), Self::Error> {
Ok(Async::Ready(()))
}
fn call(&mut self, _: ()) -> Self::Future {
Box::new(tokio_timer::sleep(self.0).map_err(|_| ()))
}
}
#[test]
fn test_success() {
let resolution = Duration::from_millis(100);
let wait_time = Duration::from_millis(50);
let res = actix_rt::System::new("test").block_on(lazy(|| {
let mut timeout = Blank::default()
.and_then(TimeoutService::new(resolution, SleepService(wait_time)));
timeout.call(())
}));
assert_eq!(res, Ok(()));
}
#[test]
fn test_timeout() {
let resolution = Duration::from_millis(100);
let wait_time = Duration::from_millis(150);
let res = actix_rt::System::new("test").block_on(lazy(|| {
let mut timeout = Blank::default()
.and_then(TimeoutService::new(resolution, SleepService(wait_time)));
timeout.call(())
}));
assert_eq!(res, Err(TimeoutError::Timeout));
}
#[test]
fn test_timeout_newservice() {
let resolution = Duration::from_millis(100);
let wait_time = Duration::from_millis(150);
let res = actix_rt::System::new("test").block_on(lazy(|| {
let timeout = BlankNewService::<(), (), ()>::default()
.apply(Timeout::new(resolution), || Ok(SleepService(wait_time)));
if let Async::Ready(mut to) = timeout.new_service(&()).poll().unwrap() {
to.call(())
} else {
panic!()
}
}));
assert_eq!(res, Err(TimeoutError::Timeout));
}
}

View File

@@ -2,7 +2,7 @@ use serde::de::{self, Deserializer, Error as DeError, Visitor};
use serde::forward_to_deserialize_any; use serde::forward_to_deserialize_any;
use crate::path::{Path, PathIter}; use crate::path::{Path, PathIter};
use crate::RequestPath; use crate::ResourcePath;
macro_rules! unsupported_type { macro_rules! unsupported_type {
($trait_fn:ident, $name:expr) => { ($trait_fn:ident, $name:expr) => {
@@ -33,17 +33,17 @@ macro_rules! parse_single_value {
} }
} }
pub struct PathDeserializer<'de, T: RequestPath + 'de> { pub struct PathDeserializer<'de, T: ResourcePath + 'de> {
path: &'de Path<T>, path: &'de Path<T>,
} }
impl<'de, T: RequestPath + 'de> PathDeserializer<'de, T> { impl<'de, T: ResourcePath + 'de> PathDeserializer<'de, T> {
pub fn new(path: &'de Path<T>) -> Self { pub fn new(path: &'de Path<T>) -> Self {
PathDeserializer { path } PathDeserializer { path }
} }
} }
impl<'de, T: RequestPath + 'de> Deserializer<'de> for PathDeserializer<'de, T> { impl<'de, T: ResourcePath + 'de> Deserializer<'de> for PathDeserializer<'de, T> {
type Error = de::value::Error; type Error = de::value::Error;
fn deserialize_map<V>(self, visitor: V) -> Result<V::Value, Self::Error> fn deserialize_map<V>(self, visitor: V) -> Result<V::Value, Self::Error>
@@ -206,12 +206,12 @@ impl<'de, T: RequestPath + 'de> Deserializer<'de> for PathDeserializer<'de, T> {
parse_single_value!(deserialize_char, visit_char, "char"); parse_single_value!(deserialize_char, visit_char, "char");
} }
struct ParamsDeserializer<'de, T: RequestPath> { struct ParamsDeserializer<'de, T: ResourcePath> {
params: PathIter<'de, T>, params: PathIter<'de, T>,
current: Option<(&'de str, &'de str)>, current: Option<(&'de str, &'de str)>,
} }
impl<'de, T: RequestPath> de::MapAccess<'de> for ParamsDeserializer<'de, T> { impl<'de, T: ResourcePath> de::MapAccess<'de> for ParamsDeserializer<'de, T> {
type Error = de::value::Error; type Error = de::value::Error;
fn next_key_seed<K>(&mut self, seed: K) -> Result<Option<K::Value>, Self::Error> fn next_key_seed<K>(&mut self, seed: K) -> Result<Option<K::Value>, Self::Error>
@@ -406,11 +406,11 @@ impl<'de> Deserializer<'de> for Value<'de> {
unsupported_type!(deserialize_identifier, "identifier"); unsupported_type!(deserialize_identifier, "identifier");
} }
struct ParamsSeq<'de, T: RequestPath> { struct ParamsSeq<'de, T: ResourcePath> {
params: PathIter<'de, T>, params: PathIter<'de, T>,
} }
impl<'de, T: RequestPath> de::SeqAccess<'de> for ParamsSeq<'de, T> { impl<'de, T: ResourcePath> de::SeqAccess<'de> for ParamsSeq<'de, T> {
type Error = de::value::Error; type Error = de::value::Error;
fn next_element_seed<U>(&mut self, seed: U) -> Result<Option<U::Value>, Self::Error> fn next_element_seed<U>(&mut self, seed: U) -> Result<Option<U::Value>, Self::Error>

View File

@@ -1,31 +1,35 @@
//! Resource path matching library. //! Resource path matching library.
mod de; mod de;
mod path; mod path;
mod pattern; mod resource;
mod router; mod router;
pub use self::de::PathDeserializer; pub use self::de::PathDeserializer;
pub use self::path::Path; pub use self::path::Path;
pub use self::pattern::Pattern; pub use self::resource::ResourceDef;
pub use self::router::{ResourceInfo, Router, RouterBuilder}; pub use self::router::{ResourceInfo, Router, RouterBuilder};
pub trait RequestPath { pub trait Resource<T: ResourcePath> {
fn resource_path(&mut self) -> &mut Path<T>;
}
pub trait ResourcePath {
fn path(&self) -> &str; fn path(&self) -> &str;
} }
impl RequestPath for String { impl ResourcePath for String {
fn path(&self) -> &str { fn path(&self) -> &str {
self.as_str() self.as_str()
} }
} }
impl<'a> RequestPath for &'a str { impl<'a> ResourcePath for &'a str {
fn path(&self) -> &str { fn path(&self) -> &str {
self self
} }
} }
impl<T: AsRef<[u8]>> RequestPath for string::String<T> { impl<T: AsRef<[u8]>> ResourcePath for string::String<T> {
fn path(&self) -> &str { fn path(&self) -> &str {
&*self &*self
} }
@@ -39,10 +43,10 @@ pub use self::url::Url;
#[cfg(feature = "http")] #[cfg(feature = "http")]
mod http_support { mod http_support {
use super::RequestPath; use super::ResourcePath;
use http::Uri; use http::Uri;
impl RequestPath for Uri { impl ResourcePath for Uri {
fn path(&self) -> &str { fn path(&self) -> &str {
self.path() self.path()
} }

View File

@@ -4,7 +4,7 @@ use std::rc::Rc;
use serde::de; use serde::de;
use crate::de::PathDeserializer; use crate::de::PathDeserializer;
use crate::RequestPath; use crate::{Resource, ResourcePath};
#[derive(Debug, Clone, Copy)] #[derive(Debug, Clone, Copy)]
pub(crate) enum PathItem { pub(crate) enum PathItem {
@@ -42,7 +42,7 @@ impl<T: Clone> Clone for Path<T> {
} }
} }
impl<T: RequestPath> Path<T> { impl<T: ResourcePath> Path<T> {
pub fn new(path: T) -> Path<T> { pub fn new(path: T) -> Path<T> {
Path { Path {
path, path,
@@ -165,7 +165,7 @@ pub struct PathIter<'a, T> {
params: &'a Path<T>, params: &'a Path<T>,
} }
impl<'a, T: RequestPath> Iterator for PathIter<'a, T> { impl<'a, T: ResourcePath> Iterator for PathIter<'a, T> {
type Item = (&'a str, &'a str); type Item = (&'a str, &'a str);
#[inline] #[inline]
@@ -183,7 +183,7 @@ impl<'a, T: RequestPath> Iterator for PathIter<'a, T> {
} }
} }
impl<'a, T: RequestPath> Index<&'a str> for Path<T> { impl<'a, T: ResourcePath> Index<&'a str> for Path<T> {
type Output = str; type Output = str;
fn index(&self, name: &'a str) -> &str { fn index(&self, name: &'a str) -> &str {
@@ -192,7 +192,7 @@ impl<'a, T: RequestPath> Index<&'a str> for Path<T> {
} }
} }
impl<T: RequestPath> Index<usize> for Path<T> { impl<T: ResourcePath> Index<usize> for Path<T> {
type Output = str; type Output = str;
fn index(&self, idx: usize) -> &str { fn index(&self, idx: usize) -> &str {
@@ -202,3 +202,9 @@ impl<T: RequestPath> Index<usize> for Path<T> {
} }
} }
} }
impl<T: ResourcePath> Resource<T> for Path<T> {
fn resource_path(&mut self) -> &mut Self {
self
}
}

View File

@@ -5,20 +5,35 @@ use std::rc::Rc;
use regex::{escape, Regex}; use regex::{escape, Regex};
use crate::path::{Path, PathItem}; use crate::path::{Path, PathItem};
use crate::RequestPath; use crate::ResourcePath;
const MAX_DYNAMIC_SEGMENTS: usize = 16; const MAX_DYNAMIC_SEGMENTS: usize = 16;
/// Resource type describes an entry in resources table /// ResourceDef describes an entry in resources table
/// ///
/// Resource pattern can contain only 16 dynamic segments /// Resource definition can contain only 16 dynamic segments
#[derive(Clone, Debug)] #[derive(Clone, Debug)]
pub struct Pattern { pub struct ResourceDef {
tp: PatternType, tp: PatternType,
rtp: ResourceType,
name: String,
pattern: String, pattern: String,
elements: Vec<PatternElement>, elements: Vec<PatternElement>,
} }
#[derive(Debug, Copy, Clone, PartialEq)]
/// Resource type
pub enum ResourceType {
/// Normal resource
Normal,
/// Resource for application default handler
Default,
/// External resource
External,
/// Unknown resource type
Unset,
}
#[derive(Debug, Clone, PartialEq)] #[derive(Debug, Clone, PartialEq)]
enum PatternElement { enum PatternElement {
Str(String), Str(String),
@@ -32,12 +47,12 @@ enum PatternType {
Dynamic(Regex, Vec<Rc<String>>, usize), Dynamic(Regex, Vec<Rc<String>>, usize),
} }
impl Pattern { impl ResourceDef {
/// Parse path pattern and create new `Pattern` instance. /// Parse path pattern and create new `Pattern` instance.
/// ///
/// Panics if path pattern is wrong. /// Panics if path pattern is wrong.
pub fn new(path: &str) -> Self { pub fn new(path: &str) -> Self {
Pattern::with_prefix(path, false) ResourceDef::with_prefix(path, false)
} }
/// Parse path pattern and create new `Pattern` instance. /// Parse path pattern and create new `Pattern` instance.
@@ -46,13 +61,22 @@ impl Pattern {
/// ///
/// Panics if path regex pattern is wrong. /// Panics if path regex pattern is wrong.
pub fn prefix(path: &str) -> Self { pub fn prefix(path: &str) -> Self {
Pattern::with_prefix(path, true) ResourceDef::with_prefix(path, true)
}
/// Construct external resource def
///
/// Panics if path pattern is malformed.
pub fn external(path: &str) -> Self {
let mut resource = ResourceDef::with_prefix(path, false);
resource.rtp = ResourceType::External;
resource
} }
/// Parse path pattern and create new `Pattern` instance with custom prefix /// Parse path pattern and create new `Pattern` instance with custom prefix
fn with_prefix(path: &str, for_prefix: bool) -> Self { fn with_prefix(path: &str, for_prefix: bool) -> Self {
let path = path.to_owned(); let path = path.to_owned();
let (pattern, elements, is_dynamic, len) = Pattern::parse(&path, for_prefix); let (pattern, elements, is_dynamic, len) = ResourceDef::parse(&path, for_prefix);
let tp = if is_dynamic { let tp = if is_dynamic {
let re = match Regex::new(&pattern) { let re = match Regex::new(&pattern) {
@@ -71,9 +95,11 @@ impl Pattern {
PatternType::Static(pattern.clone()) PatternType::Static(pattern.clone())
}; };
Pattern { ResourceDef {
tp, tp,
elements, elements,
name: String::new(),
rtp: ResourceType::Normal,
pattern: path.to_owned(), pattern: path.to_owned(),
} }
} }
@@ -93,7 +119,7 @@ impl Pattern {
} }
/// Is the given path and parameters a match against this pattern? /// Is the given path and parameters a match against this pattern?
pub fn match_path<T: RequestPath>(&self, path: &mut Path<T>) -> bool { pub fn match_path<T: ResourcePath>(&self, path: &mut Path<T>) -> bool {
match self.tp { match self.tp {
PatternType::Static(ref s) => { PatternType::Static(ref s) => {
if s == path.path() { if s == path.path() {
@@ -261,20 +287,32 @@ impl Pattern {
} }
} }
impl PartialEq for Pattern { impl Eq for ResourceDef {}
fn eq(&self, other: &Pattern) -> bool {
impl PartialEq for ResourceDef {
fn eq(&self, other: &ResourceDef) -> bool {
self.pattern == other.pattern self.pattern == other.pattern
} }
} }
impl Eq for Pattern {} impl Hash for ResourceDef {
impl Hash for Pattern {
fn hash<H: Hasher>(&self, state: &mut H) { fn hash<H: Hasher>(&self, state: &mut H) {
self.pattern.hash(state); self.pattern.hash(state);
} }
} }
impl<'a> From<&'a str> for ResourceDef {
fn from(path: &'a str) -> ResourceDef {
ResourceDef::new(path)
}
}
impl From<String> for ResourceDef {
fn from(path: String) -> ResourceDef {
ResourceDef::new(&path)
}
}
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use super::*; use super::*;
@@ -282,29 +320,29 @@ mod tests {
#[test] #[test]
fn test_parse_static() { fn test_parse_static() {
let re = Pattern::new("/"); let re = ResourceDef::new("/");
assert!(re.is_match("/")); assert!(re.is_match("/"));
assert!(!re.is_match("/a")); assert!(!re.is_match("/a"));
let re = Pattern::new("/name"); let re = ResourceDef::new("/name");
assert!(re.is_match("/name")); assert!(re.is_match("/name"));
assert!(!re.is_match("/name1")); assert!(!re.is_match("/name1"));
assert!(!re.is_match("/name/")); assert!(!re.is_match("/name/"));
assert!(!re.is_match("/name~")); assert!(!re.is_match("/name~"));
let re = Pattern::new("/name/"); let re = ResourceDef::new("/name/");
assert!(re.is_match("/name/")); assert!(re.is_match("/name/"));
assert!(!re.is_match("/name")); assert!(!re.is_match("/name"));
assert!(!re.is_match("/name/gs")); assert!(!re.is_match("/name/gs"));
let re = Pattern::new("/user/profile"); let re = ResourceDef::new("/user/profile");
assert!(re.is_match("/user/profile")); assert!(re.is_match("/user/profile"));
assert!(!re.is_match("/user/profile/profile")); assert!(!re.is_match("/user/profile/profile"));
} }
#[test] #[test]
fn test_parse_param() { fn test_parse_param() {
let re = Pattern::new("/user/{id}"); let re = ResourceDef::new("/user/{id}");
assert!(re.is_match("/user/profile")); assert!(re.is_match("/user/profile"));
assert!(re.is_match("/user/2345")); assert!(re.is_match("/user/2345"));
assert!(!re.is_match("/user/2345/")); assert!(!re.is_match("/user/2345/"));
@@ -318,7 +356,7 @@ mod tests {
assert!(re.match_path(&mut path)); assert!(re.match_path(&mut path));
assert_eq!(path.get("id").unwrap(), "1245125"); assert_eq!(path.get("id").unwrap(), "1245125");
let re = Pattern::new("/v{version}/resource/{id}"); let re = ResourceDef::new("/v{version}/resource/{id}");
assert!(re.is_match("/v1/resource/320120")); assert!(re.is_match("/v1/resource/320120"));
assert!(!re.is_match("/v/resource/1")); assert!(!re.is_match("/v/resource/1"));
assert!(!re.is_match("/resource")); assert!(!re.is_match("/resource"));
@@ -328,7 +366,7 @@ mod tests {
assert_eq!(path.get("version").unwrap(), "151"); assert_eq!(path.get("version").unwrap(), "151");
assert_eq!(path.get("id").unwrap(), "adahg32"); assert_eq!(path.get("id").unwrap(), "adahg32");
let re = Pattern::new("/{id:[[:digit:]]{6}}"); let re = ResourceDef::new("/{id:[[:digit:]]{6}}");
assert!(re.is_match("/012345")); assert!(re.is_match("/012345"));
assert!(!re.is_match("/012")); assert!(!re.is_match("/012"));
assert!(!re.is_match("/01234567")); assert!(!re.is_match("/01234567"));
@@ -341,7 +379,7 @@ mod tests {
#[test] #[test]
fn test_parse_urlencoded_param() { fn test_parse_urlencoded_param() {
let re = Pattern::new("/user/{id}/test"); let re = ResourceDef::new("/user/{id}/test");
let mut path = Path::new("/user/2345/test"); let mut path = Path::new("/user/2345/test");
assert!(re.match_path(&mut path)); assert!(re.match_path(&mut path));
@@ -359,14 +397,14 @@ mod tests {
#[test] #[test]
fn test_resource_prefix() { fn test_resource_prefix() {
let re = Pattern::prefix("/name"); let re = ResourceDef::prefix("/name");
assert!(re.is_match("/name")); assert!(re.is_match("/name"));
assert!(re.is_match("/name/")); assert!(re.is_match("/name/"));
assert!(re.is_match("/name/test/test")); assert!(re.is_match("/name/test/test"));
assert!(re.is_match("/name1")); assert!(re.is_match("/name1"));
assert!(re.is_match("/name~")); assert!(re.is_match("/name~"));
let re = Pattern::prefix("/name/"); let re = ResourceDef::prefix("/name/");
assert!(re.is_match("/name/")); assert!(re.is_match("/name/"));
assert!(re.is_match("/name/gs")); assert!(re.is_match("/name/gs"));
assert!(!re.is_match("/name")); assert!(!re.is_match("/name"));
@@ -374,7 +412,7 @@ mod tests {
#[test] #[test]
fn test_reousrce_prefix_dynamic() { fn test_reousrce_prefix_dynamic() {
let re = Pattern::prefix("/{name}/"); let re = ResourceDef::prefix("/{name}/");
assert!(re.is_match("/name/")); assert!(re.is_match("/name/"));
assert!(re.is_match("/name/gs")); assert!(re.is_match("/name/gs"));
assert!(!re.is_match("/name")); assert!(!re.is_match("/name"));

View File

@@ -1,9 +1,8 @@
use std::collections::HashMap; use std::collections::HashMap;
use std::rc::Rc; use std::rc::Rc;
use crate::path::Path; use crate::resource::ResourceDef;
use crate::pattern::Pattern; use crate::{Resource, ResourcePath};
use crate::RequestPath;
#[derive(Debug, Copy, Clone, PartialEq)] #[derive(Debug, Copy, Clone, PartialEq)]
pub(crate) enum ResourceId { pub(crate) enum ResourceId {
@@ -20,20 +19,20 @@ pub struct ResourceInfo {
#[derive(Default, Debug)] #[derive(Default, Debug)]
pub(crate) struct ResourceMap { pub(crate) struct ResourceMap {
root: Option<Pattern>, root: Option<ResourceDef>,
named: HashMap<String, Pattern>, named: HashMap<String, ResourceDef>,
patterns: Vec<Pattern>, patterns: Vec<ResourceDef>,
} }
/// Resource router. /// Resource router.
pub struct Router<T> { pub struct Router<T, U = ()> {
rmap: Rc<ResourceMap>, rmap: Rc<ResourceMap>,
named: HashMap<String, Pattern>, named: HashMap<String, ResourceDef>,
resources: Vec<T>, resources: Vec<(T, Option<U>)>,
} }
impl<T> Router<T> { impl<T, U> Router<T, U> {
pub fn build() -> RouterBuilder<T> { pub fn build() -> RouterBuilder<T, U> {
RouterBuilder { RouterBuilder {
rmap: ResourceMap::default(), rmap: ResourceMap::default(),
named: HashMap::new(), named: HashMap::new(),
@@ -41,52 +40,71 @@ impl<T> Router<T> {
} }
} }
pub fn recognize<U: RequestPath>(&self, path: &mut Path<U>) -> Option<(&T, ResourceInfo)> { pub fn recognize<R: Resource<P>, P: ResourcePath>(
if !path.path().is_empty() { &self,
for (idx, resource) in self.rmap.patterns.iter().enumerate() { res: &mut R,
if resource.match_path(path) { ) -> Option<(&T, ResourceInfo)> {
let info = ResourceInfo { for (idx, resource) in self.rmap.patterns.iter().enumerate() {
rmap: self.rmap.clone(), if resource.match_path(res.resource_path()) {
resource: ResourceId::Normal(idx as u16), let info = ResourceInfo {
}; rmap: self.rmap.clone(),
return Some((&self.resources[idx], info)); resource: ResourceId::Normal(idx as u16),
} };
return Some((&self.resources[idx].0, info));
} }
} }
None None
} }
pub fn recognize_mut<U: RequestPath>( pub fn recognize_mut<R: Resource<P>, P: ResourcePath>(
&mut self, &mut self,
path: &mut Path<U>, res: &mut R,
) -> Option<(&mut T, ResourceInfo)> { ) -> Option<(&mut T, ResourceInfo)> {
if !path.path().is_empty() { for (idx, resource) in self.rmap.patterns.iter().enumerate() {
for (idx, resource) in self.rmap.patterns.iter().enumerate() { if resource.match_path(res.resource_path()) {
if resource.match_path(path) { let info = ResourceInfo {
let info = ResourceInfo { rmap: self.rmap.clone(),
rmap: self.rmap.clone(), resource: ResourceId::Normal(idx as u16),
resource: ResourceId::Normal(idx as u16), };
}; return Some((&mut self.resources[idx].0, info));
return Some((&mut self.resources[idx], info)); }
} }
None
}
pub fn recognize_mut_checked<R: Resource<P>, P: ResourcePath, F>(
&mut self,
res: &mut R,
check: F,
) -> Option<(&mut T, ResourceInfo)>
where
F: Fn(&R, &Option<U>) -> bool,
{
for (idx, resource) in self.rmap.patterns.iter().enumerate() {
if resource.match_path(res.resource_path()) && check(res, &self.resources[idx].1) {
let info = ResourceInfo {
rmap: self.rmap.clone(),
resource: ResourceId::Normal(idx as u16),
};
return Some((&mut self.resources[idx].0, info));
} }
} }
None None
} }
} }
impl<'a, T> IntoIterator for &'a Router<T> { impl<'a, T, U> IntoIterator for &'a Router<T, U> {
type Item = &'a T; type Item = &'a (T, Option<U>);
type IntoIter = std::slice::Iter<'a, T>; type IntoIter = std::slice::Iter<'a, (T, Option<U>)>;
fn into_iter(self) -> Self::IntoIter { fn into_iter(self) -> Self::IntoIter {
self.resources.iter() self.resources.iter()
} }
} }
impl<'a, T> IntoIterator for &'a mut Router<T> { impl<'a, T, U> IntoIterator for &'a mut Router<T, U> {
type Item = &'a mut T; type Item = &'a mut (T, Option<U>);
type IntoIter = std::slice::IterMut<'a, T>; type IntoIter = std::slice::IterMut<'a, (T, Option<U>)>;
fn into_iter(self) -> Self::IntoIter { fn into_iter(self) -> Self::IntoIter {
self.resources.iter_mut() self.resources.iter_mut()
@@ -94,11 +112,11 @@ impl<'a, T> IntoIterator for &'a mut Router<T> {
} }
impl ResourceMap { impl ResourceMap {
fn register(&mut self, pattern: Pattern) { fn register(&mut self, pattern: ResourceDef) {
self.patterns.push(pattern); self.patterns.push(pattern);
} }
fn register_named(&mut self, name: String, pattern: Pattern) { fn register_named(&mut self, name: String, pattern: ResourceDef) {
self.patterns.push(pattern.clone()); self.patterns.push(pattern.clone());
self.named.insert(name, pattern); self.named.insert(name, pattern);
} }
@@ -108,24 +126,40 @@ impl ResourceMap {
} }
} }
pub struct RouterBuilder<T> { pub struct RouterBuilder<T, U = ()> {
rmap: ResourceMap, rmap: ResourceMap,
named: HashMap<String, Pattern>, named: HashMap<String, ResourceDef>,
resources: Vec<T>, resources: Vec<(T, Option<U>)>,
} }
impl<T> RouterBuilder<T> { impl<T, U> RouterBuilder<T, U> {
/// Register resource for specified path.
pub fn path(&mut self, path: &str, resource: T) { pub fn path(&mut self, path: &str, resource: T) {
self.rmap.register(Pattern::new(path)); self.rmap.register(ResourceDef::new(path));
self.resources.push(resource); self.resources.push((resource, None));
} }
/// Register resource for specified path prefix.
pub fn prefix(&mut self, prefix: &str, resource: T) { pub fn prefix(&mut self, prefix: &str, resource: T) {
self.rmap.register(Pattern::prefix(prefix)); self.rmap.register(ResourceDef::prefix(prefix));
self.resources.push(resource); self.resources.push((resource, None));
} }
pub fn finish(self) -> Router<T> { /// Register resource for ResourceDef
pub fn rdef(&mut self, rdef: ResourceDef, resource: T) {
self.rmap.register(rdef);
self.resources.push((resource, None));
}
/// Method attachs user data to lastly added resource.
///
/// This panics if no resources were added.
pub fn set_user_data(&mut self, userdata: Option<U>) {
self.resources.last_mut().unwrap().1 = userdata;
}
/// Finish configuration and create router instance.
pub fn finish(self) -> Router<T, U> {
Router { Router {
rmap: Rc::new(self.rmap), rmap: Rc::new(self.rmap),
named: self.named, named: self.named,

View File

@@ -1,6 +1,6 @@
use std::rc::Rc; use std::rc::Rc;
use crate::RequestPath; use crate::ResourcePath;
#[allow(dead_code)] #[allow(dead_code)]
const GEN_DELIMS: &[u8] = b":/?#[]@"; const GEN_DELIMS: &[u8] = b":/?#[]@";
@@ -67,7 +67,7 @@ impl Url {
} }
} }
impl RequestPath for Url { impl ResourcePath for Url {
fn path(&self) -> &str { fn path(&self) -> &str {
self.path() self.path()
} }
@@ -190,11 +190,11 @@ mod tests {
use http::{HttpTryFrom, Uri}; use http::{HttpTryFrom, Uri};
use super::*; use super::*;
use crate::{Path, Pattern}; use crate::{Path, ResourceDef};
#[test] #[test]
fn test_parse_url() { fn test_parse_url() {
let re = Pattern::new("/user/{id}/test"); let re = ResourceDef::new("/user/{id}/test");
let url = Uri::try_from("/user/2345/test").unwrap(); let url = Uri::try_from("/user/2345/test").unwrap();
let mut path = Path::new(Url::new(url)); let mut path = Path::new(Url::new(url));