mirror of
https://github.com/fafhrd91/actix-web
synced 2025-07-03 01:34:32 +02:00
Compare commits
39 Commits
service-v0
...
remove
Author | SHA1 | Date | |
---|---|---|---|
31f0a96c20 | |||
66a7c59aaf | |||
316974616a | |||
b5d84bd980 | |||
e969429e2c | |||
6fe741025f | |||
f8e170fdaf | |||
474fed4dfe | |||
2b8f41e9e4 | |||
3484007e4e | |||
58ba1d8269 | |||
7017bad4bb | |||
112a7b6b1b | |||
48a1c7320c | |||
37d28304c9 | |||
640c39fdc8 | |||
cd5435e5ee | |||
bf9bd97173 | |||
61939c7af2 | |||
e8a1664c15 | |||
d1bfae7414 | |||
5ca00dc798 | |||
fd3e77ea83 | |||
d38eb00793 | |||
3c9d95bd9f | |||
331db2eb47 | |||
de66b5c776 | |||
4adbbad450 | |||
42ec3454d9 | |||
e6daca7995 | |||
d35c87d228 | |||
ba006d95c7 | |||
9577b7bbed | |||
8ad93f4838 | |||
ffb07c8884 | |||
cdd6904aa0 | |||
98a151db4f | |||
227ea15683 | |||
e50be58fdb |
27
.travis.yml
27
.travis.yml
@ -33,7 +33,13 @@ script:
|
|||||||
if [[ "$TRAVIS_RUST_VERSION" != "nightly" ]]; then
|
if [[ "$TRAVIS_RUST_VERSION" != "nightly" ]]; then
|
||||||
cargo clean
|
cargo clean
|
||||||
cargo test --features="ssl,tls,rust-tls" -- --nocapture
|
cargo test --features="ssl,tls,rust-tls" -- --nocapture
|
||||||
cd actix-service && cargo test
|
cd actix-codec && cargo test && cd ..
|
||||||
|
cd actix-service && cargo test && cd ..
|
||||||
|
cd actix-server && cargo test --features="ssl,tls,rust-tls" -- --nocapture && cd ..
|
||||||
|
cd actix-rt && cargo test && cd ..
|
||||||
|
cd actix-connector && cargo test && cd ..
|
||||||
|
cd actix-utils && cargo test && cd ..
|
||||||
|
cd router && cargo test && cd ..
|
||||||
fi
|
fi
|
||||||
- |
|
- |
|
||||||
if [[ "$TRAVIS_RUST_VERSION" == "nightly" ]]; then
|
if [[ "$TRAVIS_RUST_VERSION" == "nightly" ]]; then
|
||||||
@ -41,16 +47,11 @@ script:
|
|||||||
cargo tarpaulin --features="ssl,tls,rust-tls" --out Xml
|
cargo tarpaulin --features="ssl,tls,rust-tls" --out Xml
|
||||||
bash <(curl -s https://codecov.io/bash)
|
bash <(curl -s https://codecov.io/bash)
|
||||||
echo "Uploaded code coverage"
|
echo "Uploaded code coverage"
|
||||||
cd actix-service && cargo tarpaulin --out Xml && bash <(curl -s https://codecov.io/bash)
|
cd actix-service && cargo tarpaulin --out Xml && bash <(curl -s https://codecov.io/bash) && cd ..
|
||||||
fi
|
cd actix-rt && cargo tarpaulin --out Xml && bash <(curl -s https://codecov.io/bash) && cd ..
|
||||||
|
cd actix-connector && cargo tarpaulin --out Xml && bash <(curl -s https://codecov.io/bash) && cd ..
|
||||||
# Upload docs
|
cd actix-codec && cargo tarpaulin --out Xml && bash <(curl -s https://codecov.io/bash) && cd ..
|
||||||
after_success:
|
cd actix-server && cargo tarpaulin --out Xml && bash <(curl -s https://codecov.io/bash) && cd ..
|
||||||
- |
|
cd actix-utils && cargo tarpaulin --out Xml && bash <(curl -s https://codecov.io/bash) && cd ..
|
||||||
if [[ "$TRAVIS_OS_NAME" == "linux" && "$TRAVIS_PULL_REQUEST" = "false" && "$TRAVIS_BRANCH" == "master" && "$TRAVIS_RUST_VERSION" == "beta" ]]; then
|
cd router && cargo tarpaulin --out Xml && bash <(curl -s https://codecov.io/bash) && cd ..
|
||||||
cargo doc --features "ssl,tls,rust-tls" --no-deps &&
|
|
||||||
echo "<meta http-equiv=refresh content=0;url=os_balloon/index.html>" > target/doc/index.html &&
|
|
||||||
git clone https://github.com/davisp/ghp-import.git &&
|
|
||||||
./ghp-import/ghp_import.py -n -p -f -m "Documentation upload" -r https://"$GH_TOKEN"@github.com/"$TRAVIS_REPO_SLUG.git" target/doc &&
|
|
||||||
echo "Uploaded documentation"
|
|
||||||
fi
|
fi
|
||||||
|
82
Cargo.toml
82
Cargo.toml
@ -15,76 +15,22 @@ edition = "2018"
|
|||||||
|
|
||||||
[workspace]
|
[workspace]
|
||||||
members = [
|
members = [
|
||||||
"./",
|
"actix-codec",
|
||||||
|
"actix-connector",
|
||||||
|
"actix-rt",
|
||||||
"actix-service",
|
"actix-service",
|
||||||
|
"actix-server",
|
||||||
|
"actix-test-server",
|
||||||
|
"actix-utils",
|
||||||
|
"router",
|
||||||
]
|
]
|
||||||
|
|
||||||
[package.metadata.docs.rs]
|
|
||||||
features = ["ssl", "tls", "rust-tls"]
|
|
||||||
|
|
||||||
[badges]
|
|
||||||
travis-ci = { repository = "actix/actix-net", branch = "master" }
|
|
||||||
# appveyor = { repository = "fafhrd91/actix-web-hdy9d" }
|
|
||||||
codecov = { repository = "actix/actix-net", branch = "master", service = "github" }
|
|
||||||
|
|
||||||
[lib]
|
|
||||||
name = "actix_net"
|
|
||||||
path = "src/lib.rs"
|
|
||||||
|
|
||||||
[features]
|
|
||||||
default = []
|
|
||||||
|
|
||||||
# tls
|
|
||||||
tls = ["native-tls"]
|
|
||||||
|
|
||||||
# openssl
|
|
||||||
ssl = ["openssl", "tokio-openssl"]
|
|
||||||
|
|
||||||
# rustls
|
|
||||||
rust-tls = ["rustls", "tokio-rustls", "webpki", "webpki-roots"]
|
|
||||||
|
|
||||||
cell = []
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
actix = "0.7.6"
|
|
||||||
actix-service = "0.1.1"
|
|
||||||
|
|
||||||
log = "0.4"
|
|
||||||
num_cpus = "1.0"
|
|
||||||
|
|
||||||
# io
|
|
||||||
mio = "^0.6.13"
|
|
||||||
net2 = "0.2"
|
|
||||||
bytes = "0.4"
|
|
||||||
futures = "0.1"
|
|
||||||
slab = "0.4"
|
|
||||||
tokio = "0.1"
|
|
||||||
tokio-codec = "0.1"
|
|
||||||
tokio-io = "0.1"
|
|
||||||
tokio-tcp = "0.1"
|
|
||||||
tokio-timer = "0.2"
|
|
||||||
tokio-reactor = "0.1"
|
|
||||||
tokio-current-thread = "0.1"
|
|
||||||
trust-dns-proto = "^0.5.0"
|
|
||||||
trust-dns-resolver = "^0.10.0"
|
|
||||||
|
|
||||||
# native-tls
|
|
||||||
native-tls = { version="0.2", optional = true }
|
|
||||||
|
|
||||||
# openssl
|
|
||||||
openssl = { version="0.10", optional = true }
|
|
||||||
tokio-openssl = { version="0.2", optional = true }
|
|
||||||
|
|
||||||
#rustls
|
|
||||||
rustls = { version = "^0.14", optional = true }
|
|
||||||
tokio-rustls = { version = "^0.8", optional = true }
|
|
||||||
webpki = { version = "0.18", optional = true }
|
|
||||||
webpki-roots = { version = "0.15", optional = true }
|
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
|
actix-service = "0.1.1"
|
||||||
|
actix-codec = "0.1.0"
|
||||||
|
actix-rt = { path="actix-rt" }
|
||||||
|
actix-server = { path="actix-server", features=["ssl"] }
|
||||||
env_logger = "0.5"
|
env_logger = "0.5"
|
||||||
|
futures = "0.1.24"
|
||||||
[profile.release]
|
openssl = { version="0.10" }
|
||||||
lto = true
|
tokio-openssl = { version="0.3" }
|
||||||
opt-level = 3
|
|
||||||
codegen-units = 1
|
|
||||||
|
@ -13,7 +13,7 @@ Actix net - framework for composable network services (experimental)
|
|||||||
|
|
||||||
```rust
|
```rust
|
||||||
fn main() {
|
fn main() {
|
||||||
let sys = actix::System::new("test");
|
let sys = actix_rt::System::new("test");
|
||||||
|
|
||||||
// load ssl keys
|
// load ssl keys
|
||||||
let mut builder = SslAcceptor::mozilla_intermediate(SslMethod::tls()).unwrap();
|
let mut builder = SslAcceptor::mozilla_intermediate(SslMethod::tls()).unwrap();
|
||||||
@ -26,7 +26,7 @@ fn main() {
|
|||||||
// bind socket address and start workers. By default server uses number of
|
// bind socket address and start workers. By default server uses number of
|
||||||
// available logical cpu as threads count. actix net start separate
|
// available logical cpu as threads count. actix net start separate
|
||||||
// instances of service pipeline in each worker.
|
// instances of service pipeline in each worker.
|
||||||
Server::default()
|
actix_server::build()
|
||||||
.bind(
|
.bind(
|
||||||
// configure service pipeline
|
// configure service pipeline
|
||||||
"basic", "0.0.0.0:8443",
|
"basic", "0.0.0.0:8443",
|
||||||
|
5
actix-codec/CHANGES.md
Normal file
5
actix-codec/CHANGES.md
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
# Changes
|
||||||
|
|
||||||
|
## [0.1.0] - 2018-12-09
|
||||||
|
|
||||||
|
* Move codec to separate crate
|
25
actix-codec/Cargo.toml
Normal file
25
actix-codec/Cargo.toml
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
[package]
|
||||||
|
name = "actix-codec"
|
||||||
|
version = "0.1.0"
|
||||||
|
authors = ["Nikolay Kim <fafhrd91@gmail.com>"]
|
||||||
|
description = "Utilities for encoding and decoding frames"
|
||||||
|
keywords = ["network", "framework", "async", "futures"]
|
||||||
|
homepage = "https://actix.rs"
|
||||||
|
repository = "https://github.com/actix/actix-net.git"
|
||||||
|
documentation = "https://docs.rs/actix-codec/"
|
||||||
|
categories = ["network-programming", "asynchronous"]
|
||||||
|
license = "MIT/Apache-2.0"
|
||||||
|
exclude = [".gitignore", ".travis.yml", ".cargo/config", "appveyor.yml"]
|
||||||
|
edition = "2018"
|
||||||
|
workspace = "../"
|
||||||
|
|
||||||
|
[lib]
|
||||||
|
name = "actix_codec"
|
||||||
|
path = "src/lib.rs"
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
bytes = "0.4"
|
||||||
|
futures = "0.1.24"
|
||||||
|
tokio-io = "0.1"
|
||||||
|
tokio-codec = "0.1"
|
||||||
|
log = "0.4"
|
@ -10,16 +10,15 @@
|
|||||||
//! [`Stream`]: #
|
//! [`Stream`]: #
|
||||||
//! [transports]: #
|
//! [transports]: #
|
||||||
|
|
||||||
#![deny(missing_docs, missing_debug_implementations, warnings)]
|
|
||||||
|
|
||||||
mod bcodec;
|
mod bcodec;
|
||||||
mod framed;
|
mod framed;
|
||||||
// mod framed2;
|
|
||||||
mod framed_read;
|
mod framed_read;
|
||||||
mod framed_write;
|
mod framed_write;
|
||||||
|
|
||||||
pub use self::bcodec::BytesCodec;
|
pub use self::bcodec::BytesCodec;
|
||||||
pub use self::framed::{Framed, FramedParts};
|
pub use self::framed::{Framed, FramedParts};
|
||||||
// pub use self::framed2::{Framed2, FramedParts2};
|
|
||||||
pub use self::framed_read::FramedRead;
|
pub use self::framed_read::FramedRead;
|
||||||
pub use self::framed_write::FramedWrite;
|
pub use self::framed_write::FramedWrite;
|
||||||
|
|
||||||
|
pub use tokio_codec::{Decoder, Encoder};
|
||||||
|
pub use tokio_io::{AsyncRead, AsyncWrite};
|
10
actix-connector/CHANGES.md
Normal file
10
actix-connector/CHANGES.md
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
# Changes
|
||||||
|
|
||||||
|
## [0.1.1] - 2019-01-13
|
||||||
|
|
||||||
|
* Upgrade trust-dns-proto
|
||||||
|
|
||||||
|
|
||||||
|
## [0.1.0] - 2018-12-09
|
||||||
|
|
||||||
|
* Move server to separate crate
|
40
actix-connector/Cargo.toml
Normal file
40
actix-connector/Cargo.toml
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
[package]
|
||||||
|
name = "actix-connector"
|
||||||
|
version = "0.1.0"
|
||||||
|
authors = ["Nikolay Kim <fafhrd91@gmail.com>"]
|
||||||
|
description = "Actix Connector - tcp connector service"
|
||||||
|
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 = "../"
|
||||||
|
|
||||||
|
[package.metadata.docs.rs]
|
||||||
|
features = ["ssl"]
|
||||||
|
|
||||||
|
[lib]
|
||||||
|
name = "actix_connector"
|
||||||
|
path = "src/lib.rs"
|
||||||
|
|
||||||
|
[features]
|
||||||
|
default = []
|
||||||
|
|
||||||
|
# openssl
|
||||||
|
ssl = ["openssl", "tokio-openssl"]
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
actix-service = "0.1.1"
|
||||||
|
actix-codec = "0.1.0"
|
||||||
|
actix-rt = "0.1.0"
|
||||||
|
futures = "0.1"
|
||||||
|
tokio-tcp = "0.1"
|
||||||
|
trust-dns-proto = "0.6.2"
|
||||||
|
trust-dns-resolver = "0.10.2"
|
||||||
|
|
||||||
|
# openssl
|
||||||
|
openssl = { version="0.10", optional = true }
|
||||||
|
tokio-openssl = { version="0.3", optional = true }
|
16
actix-connector/src/lib.rs
Normal file
16
actix-connector/src/lib.rs
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
//! Actix Connector - tcp connector service
|
||||||
|
//!
|
||||||
|
//! ## Package feature
|
||||||
|
//!
|
||||||
|
//! * `tls` - enables ssl support via `native-tls` crate
|
||||||
|
//! * `ssl` - enables ssl support via `openssl` crate
|
||||||
|
//! * `rust-tls` - enables ssl support via `rustls` crate
|
||||||
|
|
||||||
|
mod connector;
|
||||||
|
mod resolver;
|
||||||
|
pub mod ssl;
|
||||||
|
|
||||||
|
pub use self::connector::{
|
||||||
|
Connect, Connector, ConnectorError, DefaultConnector, RequestPort, TcpConnector,
|
||||||
|
};
|
||||||
|
pub use self::resolver::{RequestHost, Resolver};
|
@ -2,10 +2,8 @@ use std::collections::VecDeque;
|
|||||||
use std::marker::PhantomData;
|
use std::marker::PhantomData;
|
||||||
use std::net::IpAddr;
|
use std::net::IpAddr;
|
||||||
|
|
||||||
use futures::{Async, Future, Poll};
|
|
||||||
|
|
||||||
use actix_service::Service;
|
use actix_service::Service;
|
||||||
use tokio_current_thread::spawn;
|
use futures::{Async, Future, Poll};
|
||||||
use trust_dns_resolver::config::{ResolverConfig, ResolverOpts};
|
use trust_dns_resolver::config::{ResolverConfig, ResolverOpts};
|
||||||
pub use trust_dns_resolver::error::ResolveError;
|
pub use trust_dns_resolver::error::ResolveError;
|
||||||
use trust_dns_resolver::lookup_ip::LookupIpFuture;
|
use trust_dns_resolver::lookup_ip::LookupIpFuture;
|
||||||
@ -44,7 +42,7 @@ impl<T: RequestHost> Resolver<T> {
|
|||||||
/// Create new resolver instance with custom configuration and options.
|
/// Create new resolver instance with custom configuration and options.
|
||||||
pub fn new(cfg: ResolverConfig, opts: ResolverOpts) -> Self {
|
pub fn new(cfg: ResolverConfig, opts: ResolverOpts) -> Self {
|
||||||
let (resolver, bg) = AsyncResolver::new(cfg, opts);
|
let (resolver, bg) = AsyncResolver::new(cfg, opts);
|
||||||
spawn(bg);
|
actix_rt::Arbiter::spawn(bg);
|
||||||
Resolver {
|
Resolver {
|
||||||
resolver,
|
resolver,
|
||||||
req: PhantomData,
|
req: PhantomData,
|
||||||
@ -79,7 +77,13 @@ impl<T: RequestHost> Service<T> for Resolver<T> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn call(&mut self, req: T) -> Self::Future {
|
fn call(&mut self, req: T) -> Self::Future {
|
||||||
ResolverFuture::new(req, &self.resolver)
|
if let Ok(ip) = req.host().parse() {
|
||||||
|
let mut addrs = VecDeque::new();
|
||||||
|
addrs.push_back(ip);
|
||||||
|
ResolverFuture::new(req, &self.resolver, Some(addrs))
|
||||||
|
} else {
|
||||||
|
ResolverFuture::new(req, &self.resolver, None)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -92,13 +96,13 @@ pub struct ResolverFuture<T> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl<T: RequestHost> ResolverFuture<T> {
|
impl<T: RequestHost> ResolverFuture<T> {
|
||||||
pub fn new(addr: T, resolver: &AsyncResolver) -> Self {
|
pub fn new(addr: T, resolver: &AsyncResolver, addrs: Option<VecDeque<IpAddr>>) -> Self {
|
||||||
// we need to do dns resolution
|
// we need to do dns resolution
|
||||||
let lookup = Some(resolver.lookup_ip(addr.host()));
|
let lookup = Some(resolver.lookup_ip(addr.host()));
|
||||||
ResolverFuture {
|
ResolverFuture {
|
||||||
lookup,
|
lookup,
|
||||||
|
addrs,
|
||||||
req: Some(addr),
|
req: Some(addr),
|
||||||
addrs: None,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
6
actix-connector/src/ssl/mod.rs
Normal file
6
actix-connector/src/ssl/mod.rs
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
//! SSL Services
|
||||||
|
|
||||||
|
#[cfg(feature = "ssl")]
|
||||||
|
mod openssl;
|
||||||
|
#[cfg(feature = "ssl")]
|
||||||
|
pub use self::openssl::OpensslConnector;
|
@ -1,104 +1,13 @@
|
|||||||
use std::marker::PhantomData;
|
use std::marker::PhantomData;
|
||||||
|
|
||||||
|
use actix_codec::{AsyncRead, AsyncWrite};
|
||||||
use actix_service::{NewService, Service};
|
use actix_service::{NewService, Service};
|
||||||
use futures::{future::ok, future::FutureResult, Async, Future, Poll};
|
use futures::{future::ok, future::FutureResult, Async, Future, Poll};
|
||||||
use openssl::ssl::{Error, SslAcceptor, SslConnector};
|
use openssl::ssl::{HandshakeError, SslConnector};
|
||||||
use tokio_io::{AsyncRead, AsyncWrite};
|
use tokio_openssl::{ConnectAsync, SslConnectorExt, SslStream};
|
||||||
use tokio_openssl::{AcceptAsync, ConnectAsync, SslAcceptorExt, SslConnectorExt, SslStream};
|
|
||||||
|
|
||||||
use super::MAX_CONN_COUNTER;
|
|
||||||
use crate::counter::{Counter, CounterGuard};
|
|
||||||
use crate::resolver::RequestHost;
|
use crate::resolver::RequestHost;
|
||||||
|
|
||||||
/// Support `SSL` connections via openssl package
|
|
||||||
///
|
|
||||||
/// `ssl` feature enables `OpensslAcceptor` type
|
|
||||||
pub struct OpensslAcceptor<T> {
|
|
||||||
acceptor: SslAcceptor,
|
|
||||||
io: PhantomData<T>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<T> OpensslAcceptor<T> {
|
|
||||||
/// Create default `OpensslAcceptor`
|
|
||||||
pub fn new(acceptor: SslAcceptor) -> Self {
|
|
||||||
OpensslAcceptor {
|
|
||||||
acceptor,
|
|
||||||
io: PhantomData,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<T: AsyncRead + AsyncWrite> Clone for OpensslAcceptor<T> {
|
|
||||||
fn clone(&self) -> Self {
|
|
||||||
Self {
|
|
||||||
acceptor: self.acceptor.clone(),
|
|
||||||
io: PhantomData,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<T: AsyncRead + AsyncWrite> NewService<T> for OpensslAcceptor<T> {
|
|
||||||
type Response = SslStream<T>;
|
|
||||||
type Error = Error;
|
|
||||||
type Service = OpensslAcceptorService<T>;
|
|
||||||
type InitError = ();
|
|
||||||
type Future = FutureResult<Self::Service, Self::InitError>;
|
|
||||||
|
|
||||||
fn new_service(&self) -> Self::Future {
|
|
||||||
MAX_CONN_COUNTER.with(|conns| {
|
|
||||||
ok(OpensslAcceptorService {
|
|
||||||
acceptor: self.acceptor.clone(),
|
|
||||||
conns: conns.clone(),
|
|
||||||
io: PhantomData,
|
|
||||||
})
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct OpensslAcceptorService<T> {
|
|
||||||
acceptor: SslAcceptor,
|
|
||||||
io: PhantomData<T>,
|
|
||||||
conns: Counter,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<T: AsyncRead + AsyncWrite> Service<T> for OpensslAcceptorService<T> {
|
|
||||||
type Response = SslStream<T>;
|
|
||||||
type Error = Error;
|
|
||||||
type Future = OpensslAcceptorServiceFut<T>;
|
|
||||||
|
|
||||||
fn poll_ready(&mut self) -> Poll<(), Self::Error> {
|
|
||||||
if self.conns.available() {
|
|
||||||
Ok(Async::Ready(()))
|
|
||||||
} else {
|
|
||||||
Ok(Async::NotReady)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn call(&mut self, req: T) -> Self::Future {
|
|
||||||
OpensslAcceptorServiceFut {
|
|
||||||
_guard: self.conns.get(),
|
|
||||||
fut: SslAcceptorExt::accept_async(&self.acceptor, req),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct OpensslAcceptorServiceFut<T>
|
|
||||||
where
|
|
||||||
T: AsyncRead + AsyncWrite,
|
|
||||||
{
|
|
||||||
fut: AcceptAsync<T>,
|
|
||||||
_guard: CounterGuard,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<T: AsyncRead + AsyncWrite> Future for OpensslAcceptorServiceFut<T> {
|
|
||||||
type Item = SslStream<T>;
|
|
||||||
type Error = Error;
|
|
||||||
|
|
||||||
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
|
|
||||||
self.fut.poll()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Openssl connector factory
|
/// Openssl connector factory
|
||||||
pub struct OpensslConnector<R, T, E> {
|
pub struct OpensslConnector<R, T, E> {
|
||||||
connector: SslConnector,
|
connector: SslConnector,
|
||||||
@ -117,7 +26,7 @@ impl<R, T, E> OpensslConnector<R, T, E> {
|
|||||||
impl<R: RequestHost, T: AsyncRead + AsyncWrite> OpensslConnector<R, T, ()> {
|
impl<R: RequestHost, T: AsyncRead + AsyncWrite> OpensslConnector<R, T, ()> {
|
||||||
pub fn service(
|
pub fn service(
|
||||||
connector: SslConnector,
|
connector: SslConnector,
|
||||||
) -> impl Service<(R, T), Response = (R, SslStream<T>), Error = Error> {
|
) -> impl Service<(R, T), Response = (R, SslStream<T>), Error = HandshakeError<T>> {
|
||||||
OpensslConnectorService {
|
OpensslConnectorService {
|
||||||
connector: connector,
|
connector: connector,
|
||||||
_t: PhantomData,
|
_t: PhantomData,
|
||||||
@ -138,7 +47,7 @@ impl<R: RequestHost, T: AsyncRead + AsyncWrite, E> NewService<(R, T)>
|
|||||||
for OpensslConnector<R, T, E>
|
for OpensslConnector<R, T, E>
|
||||||
{
|
{
|
||||||
type Response = (R, SslStream<T>);
|
type Response = (R, SslStream<T>);
|
||||||
type Error = Error;
|
type Error = HandshakeError<T>;
|
||||||
type Service = OpensslConnectorService<R, T>;
|
type Service = OpensslConnectorService<R, T>;
|
||||||
type InitError = E;
|
type InitError = E;
|
||||||
type Future = FutureResult<Self::Service, Self::InitError>;
|
type Future = FutureResult<Self::Service, Self::InitError>;
|
||||||
@ -160,7 +69,7 @@ impl<R: RequestHost, T: AsyncRead + AsyncWrite> Service<(R, T)>
|
|||||||
for OpensslConnectorService<R, T>
|
for OpensslConnectorService<R, T>
|
||||||
{
|
{
|
||||||
type Response = (R, SslStream<T>);
|
type Response = (R, SslStream<T>);
|
||||||
type Error = Error;
|
type Error = HandshakeError<T>;
|
||||||
type Future = ConnectAsyncExt<R, T>;
|
type Future = ConnectAsyncExt<R, T>;
|
||||||
|
|
||||||
fn poll_ready(&mut self) -> Poll<(), Self::Error> {
|
fn poll_ready(&mut self) -> Poll<(), Self::Error> {
|
||||||
@ -186,7 +95,7 @@ where
|
|||||||
T: AsyncRead + AsyncWrite,
|
T: AsyncRead + AsyncWrite,
|
||||||
{
|
{
|
||||||
type Item = (R, SslStream<T>);
|
type Item = (R, SslStream<T>);
|
||||||
type Error = Error;
|
type Error = HandshakeError<T>;
|
||||||
|
|
||||||
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
|
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
|
||||||
match self.fut.poll()? {
|
match self.fut.poll()? {
|
5
actix-rt/CHANGES.md
Normal file
5
actix-rt/CHANGES.md
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
# Changes
|
||||||
|
|
||||||
|
## [0.1.0] - 2018-12-09
|
||||||
|
|
||||||
|
* Initial release
|
27
actix-rt/Cargo.toml
Normal file
27
actix-rt/Cargo.toml
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
[package]
|
||||||
|
name = "actix-rt"
|
||||||
|
version = "0.1.0"
|
||||||
|
authors = ["Nikolay Kim <fafhrd91@gmail.com>"]
|
||||||
|
description = "Actix runtime"
|
||||||
|
keywords = ["network", "framework", "async", "futures"]
|
||||||
|
homepage = "https://actix.rs"
|
||||||
|
repository = "https://github.com/actix/actix-net.git"
|
||||||
|
documentation = "https://docs.rs/actix-rt/"
|
||||||
|
categories = ["network-programming", "asynchronous"]
|
||||||
|
license = "MIT/Apache-2.0"
|
||||||
|
exclude = [".gitignore", ".travis.yml", ".cargo/config", "appveyor.yml"]
|
||||||
|
edition = "2018"
|
||||||
|
workspace = "../"
|
||||||
|
|
||||||
|
[lib]
|
||||||
|
name = "actix_rt"
|
||||||
|
path = "src/lib.rs"
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
log = "0.4"
|
||||||
|
bytes = "0.4"
|
||||||
|
futures = "0.1.24"
|
||||||
|
tokio-current-thread = "0.1"
|
||||||
|
tokio-executor = "0.1.5"
|
||||||
|
tokio-reactor = "0.1.7"
|
||||||
|
tokio-timer = "0.2.8"
|
267
actix-rt/src/arbiter.rs
Normal file
267
actix-rt/src/arbiter.rs
Normal file
@ -0,0 +1,267 @@
|
|||||||
|
use std::cell::{Cell, RefCell};
|
||||||
|
use std::collections::HashMap;
|
||||||
|
use std::sync::atomic::{AtomicUsize, Ordering};
|
||||||
|
use std::{fmt, thread};
|
||||||
|
|
||||||
|
use futures::sync::mpsc::{unbounded, UnboundedReceiver, UnboundedSender};
|
||||||
|
use futures::sync::oneshot::{channel, Sender};
|
||||||
|
use futures::{future, Async, Future, IntoFuture, Poll, Stream};
|
||||||
|
use tokio_current_thread::spawn;
|
||||||
|
|
||||||
|
use crate::builder::Builder;
|
||||||
|
use crate::system::System;
|
||||||
|
|
||||||
|
thread_local!(
|
||||||
|
static ADDR: RefCell<Option<Arbiter>> = RefCell::new(None);
|
||||||
|
static RUNNING: Cell<bool> = Cell::new(false);
|
||||||
|
static Q: RefCell<Vec<Box<Future<Item = (), Error = ()>>>> = RefCell::new(Vec::new());
|
||||||
|
);
|
||||||
|
|
||||||
|
pub(crate) static COUNT: AtomicUsize = AtomicUsize::new(0);
|
||||||
|
|
||||||
|
pub(crate) enum ArbiterCommand {
|
||||||
|
Stop,
|
||||||
|
Execute(Box<Future<Item = (), Error = ()> + Send>),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl fmt::Debug for ArbiterCommand {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
|
match self {
|
||||||
|
ArbiterCommand::Stop => write!(f, "ArbiterCommand::Stop"),
|
||||||
|
ArbiterCommand::Execute(_) => write!(f, "ArbiterCommand::Execute"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub struct Arbiter(UnboundedSender<ArbiterCommand>);
|
||||||
|
|
||||||
|
impl Default for Arbiter {
|
||||||
|
fn default() -> Self {
|
||||||
|
Self::new()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Arbiter {
|
||||||
|
pub(crate) fn new_system() -> Self {
|
||||||
|
let (tx, rx) = unbounded();
|
||||||
|
|
||||||
|
let arb = Arbiter(tx);
|
||||||
|
ADDR.with(|cell| *cell.borrow_mut() = Some(arb.clone()));
|
||||||
|
RUNNING.with(|cell| cell.set(false));
|
||||||
|
Arbiter::spawn(ArbiterController { stop: None, rx });
|
||||||
|
|
||||||
|
arb
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns current arbiter's address
|
||||||
|
pub fn current() -> Arbiter {
|
||||||
|
ADDR.with(|cell| match *cell.borrow() {
|
||||||
|
Some(ref addr) => addr.clone(),
|
||||||
|
None => panic!("Arbiter is not running"),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Stop arbiter
|
||||||
|
pub fn stop(&self) {
|
||||||
|
let _ = self.0.unbounded_send(ArbiterCommand::Stop);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Spawn new thread and run event loop in spawned thread.
|
||||||
|
/// Returns address of newly created arbiter.
|
||||||
|
pub fn new() -> Arbiter {
|
||||||
|
let id = COUNT.fetch_add(1, Ordering::Relaxed);
|
||||||
|
let name = format!("actix-rt:worker:{}", id);
|
||||||
|
let sys = System::current();
|
||||||
|
let (arb_tx, arb_rx) = unbounded();
|
||||||
|
let arb_tx2 = arb_tx.clone();
|
||||||
|
|
||||||
|
let _ = thread::Builder::new().name(name.clone()).spawn(move || {
|
||||||
|
let mut rt = Builder::new().build_rt().expect("Can not create Runtime");
|
||||||
|
let arb = Arbiter(arb_tx);
|
||||||
|
|
||||||
|
let (stop, stop_rx) = channel();
|
||||||
|
RUNNING.with(|cell| cell.set(true));
|
||||||
|
|
||||||
|
System::set_current(sys);
|
||||||
|
|
||||||
|
// start arbiter controller
|
||||||
|
rt.spawn(ArbiterController {
|
||||||
|
stop: Some(stop),
|
||||||
|
rx: arb_rx,
|
||||||
|
});
|
||||||
|
ADDR.with(|cell| *cell.borrow_mut() = Some(arb.clone()));
|
||||||
|
|
||||||
|
// register arbiter
|
||||||
|
let _ = System::current()
|
||||||
|
.sys()
|
||||||
|
.unbounded_send(SystemCommand::RegisterArbiter(id, arb.clone()));
|
||||||
|
|
||||||
|
// run loop
|
||||||
|
let _ = match rt.block_on(stop_rx) {
|
||||||
|
Ok(code) => code,
|
||||||
|
Err(_) => 1,
|
||||||
|
};
|
||||||
|
|
||||||
|
// unregister arbiter
|
||||||
|
let _ = System::current()
|
||||||
|
.sys()
|
||||||
|
.unbounded_send(SystemCommand::UnregisterArbiter(id));
|
||||||
|
});
|
||||||
|
|
||||||
|
Arbiter(arb_tx2)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn run_system() {
|
||||||
|
RUNNING.with(|cell| cell.set(true));
|
||||||
|
Q.with(|cell| {
|
||||||
|
let mut v = cell.borrow_mut();
|
||||||
|
for fut in v.drain(..) {
|
||||||
|
spawn(fut);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn stop_system() {
|
||||||
|
RUNNING.with(|cell| cell.set(false));
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Spawn a future on the current thread.
|
||||||
|
pub fn spawn<F>(future: F)
|
||||||
|
where
|
||||||
|
F: Future<Item = (), Error = ()> + 'static,
|
||||||
|
{
|
||||||
|
RUNNING.with(move |cell| {
|
||||||
|
if cell.get() {
|
||||||
|
spawn(Box::new(future));
|
||||||
|
} else {
|
||||||
|
Q.with(move |cell| cell.borrow_mut().push(Box::new(future)));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Executes a future on the current thread.
|
||||||
|
pub fn spawn_fn<F, R>(f: F)
|
||||||
|
where
|
||||||
|
F: FnOnce() -> R + 'static,
|
||||||
|
R: IntoFuture<Item = (), Error = ()> + 'static,
|
||||||
|
{
|
||||||
|
Arbiter::spawn(future::lazy(f))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Send a future on the arbiter's thread and spawn.
|
||||||
|
pub fn send<F>(&self, future: F)
|
||||||
|
where
|
||||||
|
F: Future<Item = (), Error = ()> + Send + 'static,
|
||||||
|
{
|
||||||
|
let _ = self
|
||||||
|
.0
|
||||||
|
.unbounded_send(ArbiterCommand::Execute(Box::new(future)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct ArbiterController {
|
||||||
|
stop: Option<Sender<i32>>,
|
||||||
|
rx: UnboundedReceiver<ArbiterCommand>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Drop for ArbiterController {
|
||||||
|
fn drop(&mut self) {
|
||||||
|
if thread::panicking() {
|
||||||
|
eprintln!("Panic in Arbiter thread, shutting down system.");
|
||||||
|
if System::current().stop_on_panic() {
|
||||||
|
System::current().stop_with_code(1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Future for ArbiterController {
|
||||||
|
type Item = ();
|
||||||
|
type Error = ();
|
||||||
|
|
||||||
|
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
|
||||||
|
loop {
|
||||||
|
match self.rx.poll() {
|
||||||
|
Ok(Async::Ready(None)) | Err(_) => return Ok(Async::Ready(())),
|
||||||
|
Ok(Async::Ready(Some(item))) => match item {
|
||||||
|
ArbiterCommand::Stop => {
|
||||||
|
if let Some(stop) = self.stop.take() {
|
||||||
|
let _ = stop.send(0);
|
||||||
|
};
|
||||||
|
return Ok(Async::Ready(()));
|
||||||
|
}
|
||||||
|
ArbiterCommand::Execute(fut) => {
|
||||||
|
spawn(fut);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Ok(Async::NotReady) => return Ok(Async::NotReady),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub(crate) enum SystemCommand {
|
||||||
|
Exit(i32),
|
||||||
|
RegisterArbiter(usize, Arbiter),
|
||||||
|
UnregisterArbiter(usize),
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub(crate) struct SystemArbiter {
|
||||||
|
stop: Option<Sender<i32>>,
|
||||||
|
commands: UnboundedReceiver<SystemCommand>,
|
||||||
|
arbiters: HashMap<usize, Arbiter>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SystemArbiter {
|
||||||
|
pub(crate) fn new(stop: Sender<i32>, commands: UnboundedReceiver<SystemCommand>) -> Self {
|
||||||
|
SystemArbiter {
|
||||||
|
commands,
|
||||||
|
stop: Some(stop),
|
||||||
|
arbiters: HashMap::new(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Future for SystemArbiter {
|
||||||
|
type Item = ();
|
||||||
|
type Error = ();
|
||||||
|
|
||||||
|
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
|
||||||
|
loop {
|
||||||
|
match self.commands.poll() {
|
||||||
|
Ok(Async::Ready(None)) | Err(_) => return Ok(Async::Ready(())),
|
||||||
|
Ok(Async::Ready(Some(cmd))) => match cmd {
|
||||||
|
SystemCommand::Exit(code) => {
|
||||||
|
// stop arbiters
|
||||||
|
for arb in self.arbiters.values() {
|
||||||
|
arb.stop();
|
||||||
|
}
|
||||||
|
// stop event loop
|
||||||
|
if let Some(stop) = self.stop.take() {
|
||||||
|
let _ = stop.send(code);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
SystemCommand::RegisterArbiter(name, hnd) => {
|
||||||
|
self.arbiters.insert(name, hnd);
|
||||||
|
}
|
||||||
|
SystemCommand::UnregisterArbiter(name) => {
|
||||||
|
self.arbiters.remove(&name);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Ok(Async::NotReady) => return Ok(Async::NotReady),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// /// Execute function in arbiter's thread
|
||||||
|
// impl<I: Send, E: Send> Handler<Execute<I, E>> for SystemArbiter {
|
||||||
|
// type Result = Result<I, E>;
|
||||||
|
|
||||||
|
// fn handle(&mut self, msg: Execute<I, E>, _: &mut Context<Self>) -> Result<I, E> {
|
||||||
|
// msg.exec()
|
||||||
|
// }
|
||||||
|
// }
|
175
actix-rt/src/builder.rs
Normal file
175
actix-rt/src/builder.rs
Normal file
@ -0,0 +1,175 @@
|
|||||||
|
use std::borrow::Cow;
|
||||||
|
use std::io;
|
||||||
|
|
||||||
|
use futures::future::{lazy, Future};
|
||||||
|
use futures::sync::mpsc::unbounded;
|
||||||
|
use futures::sync::oneshot::{channel, Receiver};
|
||||||
|
|
||||||
|
use tokio_current_thread::CurrentThread;
|
||||||
|
use tokio_reactor::Reactor;
|
||||||
|
use tokio_timer::clock::Clock;
|
||||||
|
use tokio_timer::timer::Timer;
|
||||||
|
|
||||||
|
use crate::arbiter::{Arbiter, SystemArbiter};
|
||||||
|
use crate::runtime::Runtime;
|
||||||
|
use crate::system::System;
|
||||||
|
|
||||||
|
/// Builder struct for a actix runtime.
|
||||||
|
///
|
||||||
|
/// Either use `Builder::build` to create a system and start actors.
|
||||||
|
/// Alternatively, use `Builder::run` to start the tokio runtime and
|
||||||
|
/// run a function in its context.
|
||||||
|
pub struct Builder {
|
||||||
|
/// Name of the System. Defaults to "actix" if unset.
|
||||||
|
name: Cow<'static, str>,
|
||||||
|
|
||||||
|
/// The clock to use
|
||||||
|
clock: Clock,
|
||||||
|
|
||||||
|
/// Whether the Arbiter will stop the whole System on uncaught panic. Defaults to false.
|
||||||
|
stop_on_panic: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Builder {
|
||||||
|
pub(crate) fn new() -> Self {
|
||||||
|
Builder {
|
||||||
|
name: Cow::Borrowed("actix"),
|
||||||
|
clock: Clock::new(),
|
||||||
|
stop_on_panic: false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Sets the name of the System.
|
||||||
|
pub fn name<T: Into<String>>(mut self, name: T) -> Self {
|
||||||
|
self.name = Cow::Owned(name.into());
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Set the Clock instance that will be used by this System.
|
||||||
|
///
|
||||||
|
/// Defaults to the system clock.
|
||||||
|
pub fn clock(mut self, clock: Clock) -> Self {
|
||||||
|
self.clock = clock;
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Sets the option 'stop_on_panic' which controls whether the System is stopped when an
|
||||||
|
/// uncaught panic is thrown from a worker thread.
|
||||||
|
///
|
||||||
|
/// Defaults to false.
|
||||||
|
pub fn stop_on_panic(mut self, stop_on_panic: bool) -> Self {
|
||||||
|
self.stop_on_panic = stop_on_panic;
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Create new System.
|
||||||
|
///
|
||||||
|
/// This method panics if it can not create tokio runtime
|
||||||
|
pub fn build(self) -> SystemRunner {
|
||||||
|
self.create_runtime(|| {})
|
||||||
|
}
|
||||||
|
|
||||||
|
/// This function will start tokio runtime and will finish once the
|
||||||
|
/// `System::stop()` message get called.
|
||||||
|
/// Function `f` get called within tokio runtime context.
|
||||||
|
pub fn run<F>(self, f: F) -> i32
|
||||||
|
where
|
||||||
|
F: FnOnce() + 'static,
|
||||||
|
{
|
||||||
|
self.create_runtime(f).run()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn create_runtime<F>(self, f: F) -> SystemRunner
|
||||||
|
where
|
||||||
|
F: FnOnce() + 'static,
|
||||||
|
{
|
||||||
|
let (stop_tx, stop) = channel();
|
||||||
|
let (sys_sender, sys_receiver) = unbounded();
|
||||||
|
|
||||||
|
let arbiter = Arbiter::new_system();
|
||||||
|
let system = System::construct(sys_sender, arbiter.clone(), self.stop_on_panic);
|
||||||
|
|
||||||
|
// system arbiter
|
||||||
|
let arb = SystemArbiter::new(stop_tx, sys_receiver);
|
||||||
|
|
||||||
|
let mut rt = self.build_rt().unwrap();
|
||||||
|
rt.spawn(arb);
|
||||||
|
|
||||||
|
// init system arbiter and run configuration method
|
||||||
|
let _ = rt.block_on(lazy(move || {
|
||||||
|
f();
|
||||||
|
Ok::<_, ()>(())
|
||||||
|
}));
|
||||||
|
|
||||||
|
SystemRunner { rt, stop, system }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn build_rt(&self) -> io::Result<Runtime> {
|
||||||
|
// We need a reactor to receive events about IO objects from kernel
|
||||||
|
let reactor = Reactor::new()?;
|
||||||
|
let reactor_handle = reactor.handle();
|
||||||
|
|
||||||
|
// Place a timer wheel on top of the reactor. If there are no timeouts to fire, it'll let the
|
||||||
|
// reactor pick up some new external events.
|
||||||
|
let timer = Timer::new_with_now(reactor, self.clock.clone());
|
||||||
|
let timer_handle = timer.handle();
|
||||||
|
|
||||||
|
// And now put a single-threaded executor on top of the timer. When there are no futures ready
|
||||||
|
// to do something, it'll let the timer or the reactor to generate some new stimuli for the
|
||||||
|
// futures to continue in their life.
|
||||||
|
let executor = CurrentThread::new_with_park(timer);
|
||||||
|
|
||||||
|
Ok(Runtime::new2(
|
||||||
|
reactor_handle,
|
||||||
|
timer_handle,
|
||||||
|
self.clock.clone(),
|
||||||
|
executor,
|
||||||
|
))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Helper object that runs System's event loop
|
||||||
|
#[must_use = "SystemRunner must be run"]
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub struct SystemRunner {
|
||||||
|
rt: Runtime,
|
||||||
|
stop: Receiver<i32>,
|
||||||
|
system: System,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SystemRunner {
|
||||||
|
/// This function will start event loop and will finish once the
|
||||||
|
/// `System::stop()` function is called.
|
||||||
|
pub fn run(self) -> i32 {
|
||||||
|
let SystemRunner { mut rt, stop, .. } = self;
|
||||||
|
|
||||||
|
// run loop
|
||||||
|
let _ = rt.block_on(lazy(move || {
|
||||||
|
Arbiter::run_system();
|
||||||
|
Ok::<_, ()>(())
|
||||||
|
}));
|
||||||
|
let code = match rt.block_on(stop) {
|
||||||
|
Ok(code) => code,
|
||||||
|
Err(_) => 1,
|
||||||
|
};
|
||||||
|
Arbiter::stop_system();
|
||||||
|
code
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Execute a future and wait for result.
|
||||||
|
pub fn block_on<F, I, E>(&mut self, fut: F) -> Result<I, E>
|
||||||
|
where
|
||||||
|
F: Future<Item = I, Error = E>,
|
||||||
|
{
|
||||||
|
let _ = self.rt.block_on(lazy(move || {
|
||||||
|
Arbiter::run_system();
|
||||||
|
Ok::<_, ()>(())
|
||||||
|
}));
|
||||||
|
let res = self.rt.block_on(fut);
|
||||||
|
let _ = self.rt.block_on(lazy(move || {
|
||||||
|
Arbiter::stop_system();
|
||||||
|
Ok::<_, ()>(())
|
||||||
|
}));
|
||||||
|
res
|
||||||
|
}
|
||||||
|
}
|
27
actix-rt/src/lib.rs
Normal file
27
actix-rt/src/lib.rs
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
//! A runtime implementation that runs everything on the current thread.
|
||||||
|
|
||||||
|
mod arbiter;
|
||||||
|
mod builder;
|
||||||
|
mod runtime;
|
||||||
|
mod system;
|
||||||
|
|
||||||
|
pub use self::arbiter::Arbiter;
|
||||||
|
pub use self::builder::{Builder, SystemRunner};
|
||||||
|
pub use self::runtime::{Handle, Runtime};
|
||||||
|
pub use self::system::System;
|
||||||
|
|
||||||
|
/// Spawns a future on the current arbiter.
|
||||||
|
///
|
||||||
|
/// # Panics
|
||||||
|
///
|
||||||
|
/// This function panics if actix system is not running.
|
||||||
|
pub fn spawn<F>(f: F)
|
||||||
|
where
|
||||||
|
F: futures::Future<Item = (), Error = ()> + 'static,
|
||||||
|
{
|
||||||
|
if !System::is_set() {
|
||||||
|
panic!("System is not running");
|
||||||
|
}
|
||||||
|
|
||||||
|
Arbiter::spawn(f);
|
||||||
|
}
|
92
actix-rt/src/mod.rs
Normal file
92
actix-rt/src/mod.rs
Normal file
@ -0,0 +1,92 @@
|
|||||||
|
//! A runtime implementation that runs everything on the current thread.
|
||||||
|
//!
|
||||||
|
//! [`current_thread::Runtime`][rt] is similar to the primary
|
||||||
|
//! [`Runtime`][concurrent-rt] except that it runs all components on the current
|
||||||
|
//! thread instead of using a thread pool. This means that it is able to spawn
|
||||||
|
//! futures that do not implement `Send`.
|
||||||
|
//!
|
||||||
|
//! Same as the default [`Runtime`][concurrent-rt], the
|
||||||
|
//! [`current_thread::Runtime`][rt] includes:
|
||||||
|
//!
|
||||||
|
//! * A [reactor] to drive I/O resources.
|
||||||
|
//! * An [executor] to execute tasks that use these I/O resources.
|
||||||
|
//! * A [timer] for scheduling work to run after a set period of time.
|
||||||
|
//!
|
||||||
|
//! Note that [`current_thread::Runtime`][rt] does not implement `Send` itself
|
||||||
|
//! and cannot be safely moved to other threads.
|
||||||
|
//!
|
||||||
|
//! # Spawning from other threads
|
||||||
|
//!
|
||||||
|
//! While [`current_thread::Runtime`][rt] does not implement `Send` and cannot
|
||||||
|
//! safely be moved to other threads, it provides a `Handle` that can be sent
|
||||||
|
//! to other threads and allows to spawn new tasks from there.
|
||||||
|
//!
|
||||||
|
//! For example:
|
||||||
|
//!
|
||||||
|
//! ```
|
||||||
|
//! # extern crate tokio;
|
||||||
|
//! # extern crate futures;
|
||||||
|
//! use tokio::runtime::current_thread::Runtime;
|
||||||
|
//! use tokio::prelude::*;
|
||||||
|
//! use std::thread;
|
||||||
|
//!
|
||||||
|
//! # fn main() {
|
||||||
|
//! let mut runtime = Runtime::new().unwrap();
|
||||||
|
//! let handle = runtime.handle();
|
||||||
|
//!
|
||||||
|
//! thread::spawn(move || {
|
||||||
|
//! handle.spawn(future::ok(()));
|
||||||
|
//! }).join().unwrap();
|
||||||
|
//!
|
||||||
|
//! # /*
|
||||||
|
//! runtime.run().unwrap();
|
||||||
|
//! # */
|
||||||
|
//! # }
|
||||||
|
//! ```
|
||||||
|
//!
|
||||||
|
//! # Examples
|
||||||
|
//!
|
||||||
|
//! Creating a new `Runtime` and running a future `f` until its completion and
|
||||||
|
//! returning its result.
|
||||||
|
//!
|
||||||
|
//! ```
|
||||||
|
//! use tokio::runtime::current_thread::Runtime;
|
||||||
|
//! use tokio::prelude::*;
|
||||||
|
//!
|
||||||
|
//! let mut runtime = Runtime::new().unwrap();
|
||||||
|
//!
|
||||||
|
//! // Use the runtime...
|
||||||
|
//! // runtime.block_on(f); // where f is a future
|
||||||
|
//! ```
|
||||||
|
//!
|
||||||
|
//! [rt]: struct.Runtime.html
|
||||||
|
//! [concurrent-rt]: ../struct.Runtime.html
|
||||||
|
//! [chan]: https://docs.rs/futures/0.1/futures/sync/mpsc/fn.channel.html
|
||||||
|
//! [reactor]: ../../reactor/struct.Reactor.html
|
||||||
|
//! [executor]: https://tokio.rs/docs/getting-started/runtime-model/#executors
|
||||||
|
//! [timer]: ../../timer/index.html
|
||||||
|
|
||||||
|
mod builder;
|
||||||
|
mod runtime;
|
||||||
|
|
||||||
|
pub use self::builder::Builder;
|
||||||
|
pub use self::runtime::{Runtime, Handle};
|
||||||
|
pub use tokio_current_thread::spawn;
|
||||||
|
pub use tokio_current_thread::TaskExecutor;
|
||||||
|
|
||||||
|
use futures::Future;
|
||||||
|
|
||||||
|
/// Run the provided future to completion using a runtime running on the current thread.
|
||||||
|
///
|
||||||
|
/// This first creates a new [`Runtime`], and calls [`Runtime::block_on`] with the provided future,
|
||||||
|
/// which blocks the current thread until the provided future completes. It then calls
|
||||||
|
/// [`Runtime::run`] to wait for any other spawned futures to resolve.
|
||||||
|
pub fn block_on_all<F>(future: F) -> Result<F::Item, F::Error>
|
||||||
|
where
|
||||||
|
F: Future,
|
||||||
|
{
|
||||||
|
let mut r = Runtime::new().expect("failed to start runtime on current thread");
|
||||||
|
let v = r.block_on(future)?;
|
||||||
|
r.run().expect("failed to resolve remaining futures");
|
||||||
|
Ok(v)
|
||||||
|
}
|
236
actix-rt/src/runtime.rs
Normal file
236
actix-rt/src/runtime.rs
Normal file
@ -0,0 +1,236 @@
|
|||||||
|
use std::error::Error;
|
||||||
|
use std::fmt;
|
||||||
|
use std::io;
|
||||||
|
|
||||||
|
use futures::{future, Future};
|
||||||
|
use tokio_current_thread::Handle as ExecutorHandle;
|
||||||
|
use tokio_current_thread::{self as current_thread, CurrentThread};
|
||||||
|
use tokio_executor;
|
||||||
|
use tokio_reactor::{self, Reactor};
|
||||||
|
use tokio_timer::clock::{self, Clock};
|
||||||
|
use tokio_timer::timer::{self, Timer};
|
||||||
|
|
||||||
|
use crate::builder::Builder;
|
||||||
|
|
||||||
|
/// Single-threaded runtime provides a way to start reactor
|
||||||
|
/// and executor on the current thread.
|
||||||
|
///
|
||||||
|
/// See [module level][mod] documentation for more details.
|
||||||
|
///
|
||||||
|
/// [mod]: index.html
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub struct Runtime {
|
||||||
|
reactor_handle: tokio_reactor::Handle,
|
||||||
|
timer_handle: timer::Handle,
|
||||||
|
clock: Clock,
|
||||||
|
executor: CurrentThread<Timer<Reactor>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Handle to spawn a future on the corresponding `CurrentThread` runtime instance
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub struct Handle(ExecutorHandle);
|
||||||
|
|
||||||
|
impl Handle {
|
||||||
|
/// Spawn a future onto the `CurrentThread` runtime instance corresponding to this handle
|
||||||
|
///
|
||||||
|
/// # Panics
|
||||||
|
///
|
||||||
|
/// This function panics if the spawn fails. Failure occurs if the `CurrentThread`
|
||||||
|
/// instance of the `Handle` does not exist anymore.
|
||||||
|
pub fn spawn<F>(&self, future: F) -> Result<(), tokio_executor::SpawnError>
|
||||||
|
where
|
||||||
|
F: Future<Item = (), Error = ()> + Send + 'static,
|
||||||
|
{
|
||||||
|
self.0.spawn(future)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Provides a best effort **hint** to whether or not `spawn` will succeed.
|
||||||
|
///
|
||||||
|
/// This function may return both false positives **and** false negatives.
|
||||||
|
/// If `status` returns `Ok`, then a call to `spawn` will *probably*
|
||||||
|
/// succeed, but may fail. If `status` returns `Err`, a call to `spawn` will
|
||||||
|
/// *probably* fail, but may succeed.
|
||||||
|
///
|
||||||
|
/// This allows a caller to avoid creating the task if the call to `spawn`
|
||||||
|
/// has a high likelihood of failing.
|
||||||
|
pub fn status(&self) -> Result<(), tokio_executor::SpawnError> {
|
||||||
|
self.0.status()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> future::Executor<T> for Handle
|
||||||
|
where
|
||||||
|
T: Future<Item = (), Error = ()> + Send + 'static,
|
||||||
|
{
|
||||||
|
fn execute(&self, future: T) -> Result<(), future::ExecuteError<T>> {
|
||||||
|
if let Err(e) = self.status() {
|
||||||
|
let kind = if e.is_at_capacity() {
|
||||||
|
future::ExecuteErrorKind::NoCapacity
|
||||||
|
} else {
|
||||||
|
future::ExecuteErrorKind::Shutdown
|
||||||
|
};
|
||||||
|
|
||||||
|
return Err(future::ExecuteError::new(kind, future));
|
||||||
|
}
|
||||||
|
|
||||||
|
let _ = self.spawn(future);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Error returned by the `run` function.
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub struct RunError {
|
||||||
|
inner: current_thread::RunError,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl fmt::Display for RunError {
|
||||||
|
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
|
||||||
|
write!(fmt, "{}", self.inner)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Error for RunError {
|
||||||
|
fn description(&self) -> &str {
|
||||||
|
self.inner.description()
|
||||||
|
}
|
||||||
|
fn cause(&self) -> Option<&Error> {
|
||||||
|
self.inner.cause()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Runtime {
|
||||||
|
#[allow(clippy::new_ret_no_self)]
|
||||||
|
/// Returns a new runtime initialized with default configuration values.
|
||||||
|
pub fn new() -> io::Result<Runtime> {
|
||||||
|
Builder::new().build_rt()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(super) fn new2(
|
||||||
|
reactor_handle: tokio_reactor::Handle,
|
||||||
|
timer_handle: timer::Handle,
|
||||||
|
clock: Clock,
|
||||||
|
executor: CurrentThread<Timer<Reactor>>,
|
||||||
|
) -> Runtime {
|
||||||
|
Runtime {
|
||||||
|
reactor_handle,
|
||||||
|
timer_handle,
|
||||||
|
clock,
|
||||||
|
executor,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get a new handle to spawn futures on the single-threaded Tokio runtime
|
||||||
|
///
|
||||||
|
/// Different to the runtime itself, the handle can be sent to different
|
||||||
|
/// threads.
|
||||||
|
pub fn handle(&self) -> Handle {
|
||||||
|
Handle(self.executor.handle().clone())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Spawn a future onto the single-threaded Tokio runtime.
|
||||||
|
///
|
||||||
|
/// See [module level][mod] documentation for more details.
|
||||||
|
///
|
||||||
|
/// [mod]: index.html
|
||||||
|
///
|
||||||
|
/// # Examples
|
||||||
|
///
|
||||||
|
/// ```rust
|
||||||
|
/// # use futures::{future, Future, Stream};
|
||||||
|
/// use actix_rt::Runtime;
|
||||||
|
///
|
||||||
|
/// # fn dox() {
|
||||||
|
/// // Create the runtime
|
||||||
|
/// let mut rt = Runtime::new().unwrap();
|
||||||
|
///
|
||||||
|
/// // Spawn a future onto the runtime
|
||||||
|
/// rt.spawn(future::lazy(|| {
|
||||||
|
/// println!("running on the runtime");
|
||||||
|
/// Ok(())
|
||||||
|
/// }));
|
||||||
|
/// # }
|
||||||
|
/// # pub fn main() {}
|
||||||
|
/// ```
|
||||||
|
///
|
||||||
|
/// # Panics
|
||||||
|
///
|
||||||
|
/// This function panics if the spawn fails. Failure occurs if the executor
|
||||||
|
/// is currently at capacity and is unable to spawn a new future.
|
||||||
|
pub fn spawn<F>(&mut self, future: F) -> &mut Self
|
||||||
|
where
|
||||||
|
F: Future<Item = (), Error = ()> + 'static,
|
||||||
|
{
|
||||||
|
self.executor.spawn(future);
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Runs the provided future, blocking the current thread until the future
|
||||||
|
/// completes.
|
||||||
|
///
|
||||||
|
/// This function can be used to synchronously block the current thread
|
||||||
|
/// until the provided `future` has resolved either successfully or with an
|
||||||
|
/// error. The result of the future is then returned from this function
|
||||||
|
/// call.
|
||||||
|
///
|
||||||
|
/// Note that this function will **also** execute any spawned futures on the
|
||||||
|
/// current thread, but will **not** block until these other spawned futures
|
||||||
|
/// have completed. Once the function returns, any uncompleted futures
|
||||||
|
/// remain pending in the `Runtime` instance. These futures will not run
|
||||||
|
/// until `block_on` or `run` is called again.
|
||||||
|
///
|
||||||
|
/// The caller is responsible for ensuring that other spawned futures
|
||||||
|
/// complete execution by calling `block_on` or `run`.
|
||||||
|
pub fn block_on<F>(&mut self, f: F) -> Result<F::Item, F::Error>
|
||||||
|
where
|
||||||
|
F: Future,
|
||||||
|
{
|
||||||
|
self.enter(|executor| {
|
||||||
|
// Run the provided future
|
||||||
|
let ret = executor.block_on(f);
|
||||||
|
ret.map_err(|e| e.into_inner().expect("unexpected execution error"))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Run the executor to completion, blocking the thread until **all**
|
||||||
|
/// spawned futures have completed.
|
||||||
|
pub fn run(&mut self) -> Result<(), RunError> {
|
||||||
|
self.enter(|executor| executor.run())
|
||||||
|
.map_err(|e| RunError { inner: e })
|
||||||
|
}
|
||||||
|
|
||||||
|
fn enter<F, R>(&mut self, f: F) -> R
|
||||||
|
where
|
||||||
|
F: FnOnce(&mut current_thread::Entered<Timer<Reactor>>) -> R,
|
||||||
|
{
|
||||||
|
let Runtime {
|
||||||
|
ref reactor_handle,
|
||||||
|
ref timer_handle,
|
||||||
|
ref clock,
|
||||||
|
ref mut executor,
|
||||||
|
..
|
||||||
|
} = *self;
|
||||||
|
|
||||||
|
// Binds an executor to this thread
|
||||||
|
let mut enter = tokio_executor::enter().expect("Multiple executors at once");
|
||||||
|
|
||||||
|
// This will set the default handle and timer to use inside the closure
|
||||||
|
// and run the future.
|
||||||
|
tokio_reactor::with_default(&reactor_handle, &mut enter, |enter| {
|
||||||
|
clock::with_default(clock, enter, |enter| {
|
||||||
|
timer::with_default(&timer_handle, enter, |enter| {
|
||||||
|
// The TaskExecutor is a fake executor that looks into the
|
||||||
|
// current single-threaded executor when used. This is a trick,
|
||||||
|
// because we need two mutable references to the executor (one
|
||||||
|
// to run the provided future, another to install as the default
|
||||||
|
// one). We use the fake one here as the default one.
|
||||||
|
let mut default_executor = current_thread::TaskExecutor::current();
|
||||||
|
tokio_executor::with_default(&mut default_executor, enter, |enter| {
|
||||||
|
let mut executor = executor.enter(enter);
|
||||||
|
f(&mut executor)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
118
actix-rt/src/system.rs
Normal file
118
actix-rt/src/system.rs
Normal file
@ -0,0 +1,118 @@
|
|||||||
|
use std::cell::RefCell;
|
||||||
|
|
||||||
|
use futures::sync::mpsc::UnboundedSender;
|
||||||
|
|
||||||
|
use crate::arbiter::{Arbiter, SystemCommand};
|
||||||
|
use crate::builder::{Builder, SystemRunner};
|
||||||
|
|
||||||
|
/// System is a runtime manager.
|
||||||
|
#[derive(Clone, Debug)]
|
||||||
|
pub struct System {
|
||||||
|
sys: UnboundedSender<SystemCommand>,
|
||||||
|
arbiter: Arbiter,
|
||||||
|
stop_on_panic: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
thread_local!(
|
||||||
|
static CURRENT: RefCell<Option<System>> = RefCell::new(None);
|
||||||
|
);
|
||||||
|
|
||||||
|
impl System {
|
||||||
|
/// Constructs new system and sets it as current
|
||||||
|
pub(crate) fn construct(
|
||||||
|
sys: UnboundedSender<SystemCommand>,
|
||||||
|
arbiter: Arbiter,
|
||||||
|
stop_on_panic: bool,
|
||||||
|
) -> Self {
|
||||||
|
let sys = System {
|
||||||
|
sys,
|
||||||
|
arbiter,
|
||||||
|
stop_on_panic,
|
||||||
|
};
|
||||||
|
System::set_current(sys.clone());
|
||||||
|
sys
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Build a new system with a customized tokio runtime.
|
||||||
|
///
|
||||||
|
/// This allows to customize the runtime. See struct level docs on
|
||||||
|
/// `Builder` for more information.
|
||||||
|
pub fn builder() -> Builder {
|
||||||
|
Builder::new()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[allow(clippy::new_ret_no_self)]
|
||||||
|
/// Create new system.
|
||||||
|
///
|
||||||
|
/// This method panics if it can not create tokio runtime
|
||||||
|
pub fn new<T: Into<String>>(name: T) -> SystemRunner {
|
||||||
|
Self::builder().name(name).build()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get current running system.
|
||||||
|
pub fn current() -> System {
|
||||||
|
CURRENT.with(|cell| match *cell.borrow() {
|
||||||
|
Some(ref sys) => sys.clone(),
|
||||||
|
None => panic!("System is not running"),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Set current running system.
|
||||||
|
pub(crate) fn is_set() -> bool {
|
||||||
|
CURRENT.with(|cell| cell.borrow().is_some())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Set current running system.
|
||||||
|
#[doc(hidden)]
|
||||||
|
pub fn set_current(sys: System) {
|
||||||
|
CURRENT.with(|s| {
|
||||||
|
*s.borrow_mut() = Some(sys);
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Execute function with system reference.
|
||||||
|
pub fn with_current<F, R>(f: F) -> R
|
||||||
|
where
|
||||||
|
F: FnOnce(&System) -> R,
|
||||||
|
{
|
||||||
|
CURRENT.with(|cell| match *cell.borrow() {
|
||||||
|
Some(ref sys) => f(sys),
|
||||||
|
None => panic!("System is not running"),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Stop the system
|
||||||
|
pub fn stop(&self) {
|
||||||
|
self.stop_with_code(0)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Stop the system with a particular exit code.
|
||||||
|
pub fn stop_with_code(&self, code: i32) {
|
||||||
|
let _ = self.sys.unbounded_send(SystemCommand::Exit(code));
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn sys(&self) -> &UnboundedSender<SystemCommand> {
|
||||||
|
&self.sys
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Return status of 'stop_on_panic' option which controls whether the System is stopped when an
|
||||||
|
/// uncaught panic is thrown from a worker thread.
|
||||||
|
pub fn stop_on_panic(&self) -> bool {
|
||||||
|
self.stop_on_panic
|
||||||
|
}
|
||||||
|
|
||||||
|
/// System arbiter
|
||||||
|
pub fn arbiter(&self) -> &Arbiter {
|
||||||
|
&self.arbiter
|
||||||
|
}
|
||||||
|
|
||||||
|
/// This function will start tokio runtime and will finish once the
|
||||||
|
/// `System::stop()` message get called.
|
||||||
|
/// Function `f` get called within tokio runtime context.
|
||||||
|
pub fn run<F>(f: F) -> i32
|
||||||
|
where
|
||||||
|
F: FnOnce() + 'static,
|
||||||
|
{
|
||||||
|
Self::builder().run(f)
|
||||||
|
}
|
||||||
|
}
|
29
actix-server/CHANGES.md
Normal file
29
actix-server/CHANGES.md
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
# Changes
|
||||||
|
|
||||||
|
## [0.1.3] - 2018-12-21
|
||||||
|
|
||||||
|
## Fixed
|
||||||
|
|
||||||
|
* Fix max concurrent connections handling
|
||||||
|
|
||||||
|
|
||||||
|
## [0.1.2] - 2018-12-12
|
||||||
|
|
||||||
|
## Changed
|
||||||
|
|
||||||
|
* rename ServiceConfig::rt() to ServiceConfig::apply()
|
||||||
|
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
* Fix back-pressure for concurrent ssl handshakes
|
||||||
|
|
||||||
|
|
||||||
|
## [0.1.1] - 2018-12-11
|
||||||
|
|
||||||
|
* Fix signal handling on windows
|
||||||
|
|
||||||
|
|
||||||
|
## [0.1.0] - 2018-12-09
|
||||||
|
|
||||||
|
* Move server to separate crate
|
71
actix-server/Cargo.toml
Normal file
71
actix-server/Cargo.toml
Normal file
@ -0,0 +1,71 @@
|
|||||||
|
[package]
|
||||||
|
name = "actix-server"
|
||||||
|
version = "0.1.3"
|
||||||
|
authors = ["Nikolay Kim <fafhrd91@gmail.com>"]
|
||||||
|
description = "Actix server - General purpose tcp server"
|
||||||
|
keywords = ["network", "framework", "async", "futures"]
|
||||||
|
homepage = "https://actix.rs"
|
||||||
|
repository = "https://github.com/actix/actix-net.git"
|
||||||
|
documentation = "https://docs.rs/actix-server/"
|
||||||
|
categories = ["network-programming", "asynchronous"]
|
||||||
|
license = "MIT/Apache-2.0"
|
||||||
|
exclude = [".gitignore", ".travis.yml", ".cargo/config", "appveyor.yml"]
|
||||||
|
edition = "2018"
|
||||||
|
workspace = "../"
|
||||||
|
|
||||||
|
[package.metadata.docs.rs]
|
||||||
|
features = ["ssl", "tls", "rust-tls"]
|
||||||
|
|
||||||
|
[lib]
|
||||||
|
name = "actix_server"
|
||||||
|
path = "src/lib.rs"
|
||||||
|
|
||||||
|
[features]
|
||||||
|
default = []
|
||||||
|
|
||||||
|
# tls
|
||||||
|
tls = ["native-tls"]
|
||||||
|
|
||||||
|
# openssl
|
||||||
|
ssl = ["openssl", "tokio-openssl"]
|
||||||
|
|
||||||
|
# rustls
|
||||||
|
rust-tls = ["rustls", "tokio-rustls", "webpki", "webpki-roots"]
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
actix-service = "0.1.1"
|
||||||
|
actix-rt = "0.1.0"
|
||||||
|
|
||||||
|
log = "0.4"
|
||||||
|
num_cpus = "1.0"
|
||||||
|
|
||||||
|
# io
|
||||||
|
mio = "^0.6.13"
|
||||||
|
net2 = "0.2"
|
||||||
|
bytes = "0.4"
|
||||||
|
futures = "0.1"
|
||||||
|
slab = "0.4"
|
||||||
|
tokio-io = "0.1"
|
||||||
|
tokio-tcp = "0.1"
|
||||||
|
tokio-timer = "0.2"
|
||||||
|
tokio-reactor = "0.1"
|
||||||
|
tokio-signal = "0.2"
|
||||||
|
|
||||||
|
# native-tls
|
||||||
|
native-tls = { version="0.2", optional = true }
|
||||||
|
|
||||||
|
# openssl
|
||||||
|
openssl = { version="0.10", optional = true }
|
||||||
|
tokio-openssl = { version="0.3", optional = true }
|
||||||
|
|
||||||
|
#rustls
|
||||||
|
rustls = { version = "^0.14", optional = true }
|
||||||
|
tokio-rustls = { version = "^0.8", optional = true }
|
||||||
|
webpki = { version = "0.18", optional = true }
|
||||||
|
webpki-roots = { version = "0.15", optional = true }
|
||||||
|
|
||||||
|
[dev-dependencies]
|
||||||
|
env_logger = "0.5"
|
||||||
|
actix-service = "0.1.1"
|
||||||
|
actix-codec = "0.1.0"
|
||||||
|
actix-rt = "0.1.0"
|
@ -2,15 +2,14 @@ use std::sync::mpsc as sync_mpsc;
|
|||||||
use std::time::{Duration, Instant};
|
use std::time::{Duration, Instant};
|
||||||
use std::{io, net, thread};
|
use std::{io, net, thread};
|
||||||
|
|
||||||
use futures::{sync::mpsc, Future};
|
use actix_rt::System;
|
||||||
|
use futures::future::{lazy, Future};
|
||||||
use log::{error, info};
|
use log::{error, info};
|
||||||
use mio;
|
use mio;
|
||||||
use slab::Slab;
|
use slab::Slab;
|
||||||
use tokio_timer::Delay;
|
use tokio_timer::Delay;
|
||||||
|
|
||||||
use actix::{msgs::Execute, Arbiter, System};
|
use super::server::Server;
|
||||||
|
|
||||||
use super::server::ServerCommand;
|
|
||||||
use super::worker::{Conn, WorkerClient};
|
use super::worker::{Conn, WorkerClient};
|
||||||
use super::Token;
|
use super::Token;
|
||||||
|
|
||||||
@ -54,14 +53,11 @@ pub(crate) struct AcceptLoop {
|
|||||||
notify_ready: mio::SetReadiness,
|
notify_ready: mio::SetReadiness,
|
||||||
tx: sync_mpsc::Sender<Command>,
|
tx: sync_mpsc::Sender<Command>,
|
||||||
rx: Option<sync_mpsc::Receiver<Command>>,
|
rx: Option<sync_mpsc::Receiver<Command>>,
|
||||||
srv: Option<(
|
srv: Option<Server>,
|
||||||
mpsc::UnboundedSender<ServerCommand>,
|
|
||||||
mpsc::UnboundedReceiver<ServerCommand>,
|
|
||||||
)>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AcceptLoop {
|
impl AcceptLoop {
|
||||||
pub fn new() -> AcceptLoop {
|
pub fn new(srv: Server) -> AcceptLoop {
|
||||||
let (tx, rx) = sync_mpsc::channel();
|
let (tx, rx) = sync_mpsc::channel();
|
||||||
let (cmd_reg, cmd_ready) = mio::Registration::new2();
|
let (cmd_reg, cmd_ready) = mio::Registration::new2();
|
||||||
let (notify_reg, notify_ready) = mio::Registration::new2();
|
let (notify_reg, notify_ready) = mio::Registration::new2();
|
||||||
@ -73,7 +69,7 @@ impl AcceptLoop {
|
|||||||
notify_ready,
|
notify_ready,
|
||||||
notify_reg: Some(notify_reg),
|
notify_reg: Some(notify_reg),
|
||||||
rx: Some(rx),
|
rx: Some(rx),
|
||||||
srv: Some(mpsc::unbounded()),
|
srv: Some(srv),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -90,18 +86,17 @@ impl AcceptLoop {
|
|||||||
&mut self,
|
&mut self,
|
||||||
socks: Vec<(Token, net::TcpListener)>,
|
socks: Vec<(Token, net::TcpListener)>,
|
||||||
workers: Vec<WorkerClient>,
|
workers: Vec<WorkerClient>,
|
||||||
) -> mpsc::UnboundedReceiver<ServerCommand> {
|
) {
|
||||||
let (tx, rx) = self.srv.take().expect("Can not re-use AcceptInfo");
|
let srv = self.srv.take().expect("Can not re-use AcceptInfo");
|
||||||
|
|
||||||
Accept::start(
|
Accept::start(
|
||||||
self.rx.take().expect("Can not re-use AcceptInfo"),
|
self.rx.take().expect("Can not re-use AcceptInfo"),
|
||||||
self.cmd_reg.take().expect("Can not re-use AcceptInfo"),
|
self.cmd_reg.take().expect("Can not re-use AcceptInfo"),
|
||||||
self.notify_reg.take().expect("Can not re-use AcceptInfo"),
|
self.notify_reg.take().expect("Can not re-use AcceptInfo"),
|
||||||
socks,
|
socks,
|
||||||
tx,
|
srv,
|
||||||
workers,
|
workers,
|
||||||
);
|
);
|
||||||
rx
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -110,7 +105,7 @@ struct Accept {
|
|||||||
rx: sync_mpsc::Receiver<Command>,
|
rx: sync_mpsc::Receiver<Command>,
|
||||||
sockets: Slab<ServerSocketInfo>,
|
sockets: Slab<ServerSocketInfo>,
|
||||||
workers: Vec<WorkerClient>,
|
workers: Vec<WorkerClient>,
|
||||||
srv: mpsc::UnboundedSender<ServerCommand>,
|
srv: Server,
|
||||||
timer: (mio::Registration, mio::SetReadiness),
|
timer: (mio::Registration, mio::SetReadiness),
|
||||||
next: usize,
|
next: usize,
|
||||||
backpressure: bool,
|
backpressure: bool,
|
||||||
@ -141,14 +136,14 @@ impl Accept {
|
|||||||
cmd_reg: mio::Registration,
|
cmd_reg: mio::Registration,
|
||||||
notify_reg: mio::Registration,
|
notify_reg: mio::Registration,
|
||||||
socks: Vec<(Token, net::TcpListener)>,
|
socks: Vec<(Token, net::TcpListener)>,
|
||||||
srv: mpsc::UnboundedSender<ServerCommand>,
|
srv: Server,
|
||||||
workers: Vec<WorkerClient>,
|
workers: Vec<WorkerClient>,
|
||||||
) {
|
) {
|
||||||
let sys = System::current();
|
let sys = System::current();
|
||||||
|
|
||||||
// start accept thread
|
// start accept thread
|
||||||
let _ = thread::Builder::new()
|
let _ = thread::Builder::new()
|
||||||
.name("actix-web accept loop".to_owned())
|
.name("actix-server accept loop".to_owned())
|
||||||
.spawn(move || {
|
.spawn(move || {
|
||||||
System::set_current(sys);
|
System::set_current(sys);
|
||||||
let mut accept = Accept::new(rx, socks, workers, srv);
|
let mut accept = Accept::new(rx, socks, workers, srv);
|
||||||
@ -181,7 +176,7 @@ impl Accept {
|
|||||||
rx: sync_mpsc::Receiver<Command>,
|
rx: sync_mpsc::Receiver<Command>,
|
||||||
socks: Vec<(Token, net::TcpListener)>,
|
socks: Vec<(Token, net::TcpListener)>,
|
||||||
workers: Vec<WorkerClient>,
|
workers: Vec<WorkerClient>,
|
||||||
srv: mpsc::UnboundedSender<ServerCommand>,
|
srv: Server,
|
||||||
) -> Accept {
|
) -> Accept {
|
||||||
// Create a poll instance
|
// Create a poll instance
|
||||||
let poll = match mio::Poll::new() {
|
let poll = match mio::Poll::new() {
|
||||||
@ -376,9 +371,7 @@ impl Accept {
|
|||||||
match self.workers[self.next].send(msg) {
|
match self.workers[self.next].send(msg) {
|
||||||
Ok(_) => (),
|
Ok(_) => (),
|
||||||
Err(tmp) => {
|
Err(tmp) => {
|
||||||
let _ = self.srv.unbounded_send(ServerCommand::WorkerDied(
|
self.srv.worker_died(self.workers[self.next].idx);
|
||||||
self.workers[self.next].idx,
|
|
||||||
));
|
|
||||||
msg = tmp;
|
msg = tmp;
|
||||||
self.workers.swap_remove(self.next);
|
self.workers.swap_remove(self.next);
|
||||||
if self.workers.is_empty() {
|
if self.workers.is_empty() {
|
||||||
@ -404,9 +397,7 @@ impl Accept {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
Err(tmp) => {
|
Err(tmp) => {
|
||||||
let _ = self.srv.unbounded_send(ServerCommand::WorkerDied(
|
self.srv.worker_died(self.workers[self.next].idx);
|
||||||
self.workers[self.next].idx,
|
|
||||||
));
|
|
||||||
msg = tmp;
|
msg = tmp;
|
||||||
self.workers.swap_remove(self.next);
|
self.workers.swap_remove(self.next);
|
||||||
if self.workers.is_empty() {
|
if self.workers.is_empty() {
|
||||||
@ -449,19 +440,14 @@ impl Accept {
|
|||||||
info.timeout = Some(Instant::now() + Duration::from_millis(500));
|
info.timeout = Some(Instant::now() + Duration::from_millis(500));
|
||||||
|
|
||||||
let r = self.timer.1.clone();
|
let r = self.timer.1.clone();
|
||||||
System::current().arbiter().do_send(Execute::new(
|
System::current().arbiter().send(lazy(move || {
|
||||||
move || -> Result<(), ()> {
|
Delay::new(Instant::now() + Duration::from_millis(510))
|
||||||
Arbiter::spawn(
|
.map_err(|_| ())
|
||||||
Delay::new(Instant::now() + Duration::from_millis(510))
|
.and_then(move |_| {
|
||||||
.map_err(|_| ())
|
let _ = r.set_readiness(mio::Ready::readable());
|
||||||
.and_then(move |_| {
|
Ok(())
|
||||||
let _ = r.set_readiness(mio::Ready::readable());
|
})
|
||||||
Ok(())
|
}));
|
||||||
}),
|
|
||||||
);
|
|
||||||
Ok(())
|
|
||||||
},
|
|
||||||
));
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -1,30 +1,27 @@
|
|||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
use std::{io, mem, net};
|
use std::{io, mem, net};
|
||||||
|
|
||||||
use futures::sync::{mpsc, mpsc::unbounded};
|
use actix_rt::{spawn, Arbiter, System};
|
||||||
use futures::{Future, Sink, Stream};
|
use futures::future::{lazy, ok};
|
||||||
|
use futures::stream::futures_unordered;
|
||||||
|
use futures::sync::mpsc::{unbounded, UnboundedReceiver};
|
||||||
|
use futures::{Async, Future, Poll, Stream};
|
||||||
use log::{error, info};
|
use log::{error, info};
|
||||||
use net2::TcpBuilder;
|
use net2::TcpBuilder;
|
||||||
use num_cpus;
|
use num_cpus;
|
||||||
|
use tokio_timer::sleep;
|
||||||
|
|
||||||
use actix::{
|
use crate::accept::{AcceptLoop, AcceptNotify, Command};
|
||||||
actors::signal, fut, msgs::Execute, Actor, ActorFuture, Addr, Arbiter, AsyncContext,
|
use crate::config::{ConfiguredService, ServiceConfig};
|
||||||
Context, Handler, Response, StreamHandler, System, WrapFuture,
|
use crate::server::{Server, ServerCommand};
|
||||||
};
|
use crate::services::{InternalServiceFactory, StreamNewService, StreamServiceFactory};
|
||||||
|
use crate::services::{ServiceFactory, ServiceNewService};
|
||||||
|
use crate::signals::{Signal, Signals};
|
||||||
|
use crate::worker::{self, Worker, WorkerAvailability, WorkerClient};
|
||||||
|
use crate::Token;
|
||||||
|
|
||||||
use super::accept::{AcceptLoop, AcceptNotify, Command};
|
/// Server builder
|
||||||
use super::config::{ConfiguredService, ServiceConfig};
|
pub struct ServerBuilder {
|
||||||
use super::services::{InternalServiceFactory, StreamNewService, StreamServiceFactory};
|
|
||||||
use super::services::{ServiceFactory, ServiceNewService};
|
|
||||||
use super::worker::{self, Worker, WorkerAvailability, WorkerClient};
|
|
||||||
use super::{PauseServer, ResumeServer, StopServer, Token};
|
|
||||||
|
|
||||||
pub(crate) enum ServerCommand {
|
|
||||||
WorkerDied(usize),
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Server
|
|
||||||
pub struct Server {
|
|
||||||
threads: usize,
|
threads: usize,
|
||||||
token: Token,
|
token: Token,
|
||||||
workers: Vec<(usize, WorkerClient)>,
|
workers: Vec<(usize, WorkerClient)>,
|
||||||
@ -33,36 +30,41 @@ pub struct Server {
|
|||||||
accept: AcceptLoop,
|
accept: AcceptLoop,
|
||||||
exit: bool,
|
exit: bool,
|
||||||
shutdown_timeout: Duration,
|
shutdown_timeout: Duration,
|
||||||
signals: Option<Addr<signal::ProcessSignals>>,
|
|
||||||
no_signals: bool,
|
no_signals: bool,
|
||||||
|
cmd: UnboundedReceiver<ServerCommand>,
|
||||||
|
server: Server,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for Server {
|
impl Default for ServerBuilder {
|
||||||
fn default() -> Self {
|
fn default() -> Self {
|
||||||
Self::new()
|
Self::new()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Server {
|
impl ServerBuilder {
|
||||||
/// Create new Server instance
|
/// Create new Server builder instance
|
||||||
pub fn new() -> Server {
|
pub fn new() -> ServerBuilder {
|
||||||
Server {
|
let (tx, rx) = unbounded();
|
||||||
|
let server = Server::new(tx);
|
||||||
|
|
||||||
|
ServerBuilder {
|
||||||
threads: num_cpus::get(),
|
threads: num_cpus::get(),
|
||||||
token: Token(0),
|
token: Token(0),
|
||||||
workers: Vec::new(),
|
workers: Vec::new(),
|
||||||
services: Vec::new(),
|
services: Vec::new(),
|
||||||
sockets: Vec::new(),
|
sockets: Vec::new(),
|
||||||
accept: AcceptLoop::new(),
|
accept: AcceptLoop::new(server.clone()),
|
||||||
exit: false,
|
exit: false,
|
||||||
shutdown_timeout: Duration::from_secs(30),
|
shutdown_timeout: Duration::from_secs(30),
|
||||||
signals: None,
|
|
||||||
no_signals: false,
|
no_signals: false,
|
||||||
|
cmd: rx,
|
||||||
|
server,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Set number of workers to start.
|
/// Set number of workers to start.
|
||||||
///
|
///
|
||||||
/// By default server uses number of available logical cpu as threads
|
/// By default server uses number of available logical cpu as workers
|
||||||
/// count.
|
/// count.
|
||||||
pub fn workers(mut self, num: usize) -> Self {
|
pub fn workers(mut self, num: usize) -> Self {
|
||||||
self.threads = num;
|
self.threads = num;
|
||||||
@ -88,13 +90,6 @@ impl Server {
|
|||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
#[doc(hidden)]
|
|
||||||
/// Set alternative address for `ProcessSignals` actor.
|
|
||||||
pub fn signals(mut self, addr: Addr<signal::ProcessSignals>) -> Self {
|
|
||||||
self.signals = Some(addr);
|
|
||||||
self
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Disable signal handling
|
/// Disable signal handling
|
||||||
pub fn disable_signals(mut self) -> Self {
|
pub fn disable_signals(mut self) -> Self {
|
||||||
self.no_signals = true;
|
self.no_signals = true;
|
||||||
@ -113,31 +108,34 @@ impl Server {
|
|||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Run external configuration as part of the server building
|
/// Execute external configuration as part of the server building
|
||||||
/// process
|
/// process.
|
||||||
///
|
///
|
||||||
/// This function is useful for moving parts of configuration to a
|
/// This function is useful for moving parts of configuration to a
|
||||||
/// different module or even library.
|
/// different module or even library.
|
||||||
pub fn configure<F>(mut self, f: F) -> io::Result<Server>
|
pub fn configure<F>(mut self, f: F) -> io::Result<ServerBuilder>
|
||||||
where
|
where
|
||||||
F: Fn(&mut ServiceConfig) -> io::Result<()>,
|
F: Fn(&mut ServiceConfig) -> io::Result<()>,
|
||||||
{
|
{
|
||||||
let mut cfg = ServiceConfig::new();
|
let mut cfg = ServiceConfig::new(self.threads);
|
||||||
|
|
||||||
f(&mut cfg)?;
|
f(&mut cfg)?;
|
||||||
|
|
||||||
let mut srv = ConfiguredService::new(cfg.rt);
|
if let Some(apply) = cfg.apply {
|
||||||
for (name, lst) in cfg.services {
|
let mut srv = ConfiguredService::new(apply);
|
||||||
let token = self.token.next();
|
for (name, lst) in cfg.services {
|
||||||
srv.stream(token, name);
|
let token = self.token.next();
|
||||||
self.sockets.push((token, lst));
|
srv.stream(token, name);
|
||||||
|
self.sockets.push((token, lst));
|
||||||
|
}
|
||||||
|
self.services.push(Box::new(srv));
|
||||||
}
|
}
|
||||||
self.services.push(Box::new(srv));
|
self.threads = cfg.threads;
|
||||||
|
|
||||||
Ok(self)
|
Ok(self)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Add new service to 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: StreamServiceFactory,
|
||||||
@ -158,7 +156,7 @@ impl Server {
|
|||||||
Ok(self)
|
Ok(self)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Add new service to server
|
/// Add new service to the server.
|
||||||
pub fn listen<F, N: AsRef<str>>(
|
pub fn listen<F, N: AsRef<str>>(
|
||||||
mut self,
|
mut self,
|
||||||
name: N,
|
name: N,
|
||||||
@ -178,7 +176,7 @@ impl Server {
|
|||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Add new service to server
|
/// Add new service to the server.
|
||||||
pub fn listen2<F, N: AsRef<str>>(
|
pub fn listen2<F, N: AsRef<str>>(
|
||||||
mut self,
|
mut self,
|
||||||
name: N,
|
name: N,
|
||||||
@ -226,10 +224,10 @@ impl Server {
|
|||||||
sys.run();
|
sys.run();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Starts Server Actor and returns its address
|
/// Starts processing incoming connections and return server controller.
|
||||||
pub fn start(mut self) -> Addr<Server> {
|
pub fn start(mut self) -> Server {
|
||||||
if self.sockets.is_empty() {
|
if self.sockets.is_empty() {
|
||||||
panic!("Service should have at least one bound socket");
|
panic!("Server should have at least one bound socket");
|
||||||
} else {
|
} else {
|
||||||
info!("Starting {} workers", self.threads);
|
info!("Starting {} workers", self.threads);
|
||||||
|
|
||||||
@ -245,33 +243,18 @@ impl Server {
|
|||||||
for sock in &self.sockets {
|
for sock in &self.sockets {
|
||||||
info!("Starting server on {}", sock.1.local_addr().ok().unwrap());
|
info!("Starting server on {}", sock.1.local_addr().ok().unwrap());
|
||||||
}
|
}
|
||||||
let rx = self
|
self.accept
|
||||||
.accept
|
|
||||||
.start(mem::replace(&mut self.sockets, Vec::new()), workers);
|
.start(mem::replace(&mut self.sockets, Vec::new()), workers);
|
||||||
|
|
||||||
// start http server actor
|
// handle signals
|
||||||
let signals = self.subscribe_to_signals();
|
if !self.no_signals {
|
||||||
let addr = Actor::create(move |ctx| {
|
Signals::start(self.server.clone());
|
||||||
ctx.add_stream(rx);
|
|
||||||
self
|
|
||||||
});
|
|
||||||
if let Some(signals) = signals {
|
|
||||||
signals.do_send(signal::Subscribe(addr.clone().recipient()))
|
|
||||||
}
|
}
|
||||||
addr
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// subscribe to os signals
|
// start http server actor
|
||||||
fn subscribe_to_signals(&self) -> Option<Addr<signal::ProcessSignals>> {
|
let server = self.server.clone();
|
||||||
if !self.no_signals {
|
spawn(self);
|
||||||
if let Some(ref signals) = self.signals {
|
server
|
||||||
Some(signals.clone())
|
|
||||||
} else {
|
|
||||||
Some(System::current().registry().get::<signal::ProcessSignals>())
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -284,118 +267,99 @@ impl Server {
|
|||||||
let services: Vec<Box<InternalServiceFactory>> =
|
let services: Vec<Box<InternalServiceFactory>> =
|
||||||
self.services.iter().map(|v| v.clone_factory()).collect();
|
self.services.iter().map(|v| v.clone_factory()).collect();
|
||||||
|
|
||||||
Arbiter::new(format!("actix-net-worker-{}", idx)).do_send(Execute::new(move || {
|
Arbiter::new().send(lazy(move || {
|
||||||
Worker::start(rx1, rx2, services, avail, timeout);
|
Worker::start(rx1, rx2, services, avail, timeout);
|
||||||
Ok::<_, ()>(())
|
Ok::<_, ()>(())
|
||||||
}));
|
}));
|
||||||
|
|
||||||
worker
|
worker
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
impl Actor for Server {
|
fn handle_cmd(&mut self, item: ServerCommand) {
|
||||||
type Context = Context<Self>;
|
match item {
|
||||||
}
|
ServerCommand::Pause(tx) => {
|
||||||
|
self.accept.send(Command::Pause);
|
||||||
/// Signals support
|
let _ = tx.send(());
|
||||||
/// Handle `SIGINT`, `SIGTERM`, `SIGQUIT` signals and stop actix system
|
|
||||||
/// message to `System` actor.
|
|
||||||
impl Handler<signal::Signal> for Server {
|
|
||||||
type Result = ();
|
|
||||||
|
|
||||||
fn handle(&mut self, msg: signal::Signal, ctx: &mut Context<Self>) {
|
|
||||||
match msg.0 {
|
|
||||||
signal::SignalType::Int => {
|
|
||||||
info!("SIGINT received, exiting");
|
|
||||||
self.exit = true;
|
|
||||||
Handler::<StopServer>::handle(self, StopServer { graceful: false }, ctx);
|
|
||||||
}
|
}
|
||||||
signal::SignalType::Term => {
|
ServerCommand::Resume(tx) => {
|
||||||
info!("SIGTERM received, stopping");
|
self.accept.send(Command::Resume);
|
||||||
self.exit = true;
|
let _ = tx.send(());
|
||||||
Handler::<StopServer>::handle(self, StopServer { graceful: true }, ctx);
|
|
||||||
}
|
}
|
||||||
signal::SignalType::Quit => {
|
ServerCommand::Signal(sig) => {
|
||||||
info!("SIGQUIT received, exiting");
|
// Signals support
|
||||||
self.exit = true;
|
// Handle `SIGINT`, `SIGTERM`, `SIGQUIT` signals and stop actix system
|
||||||
Handler::<StopServer>::handle(self, StopServer { graceful: false }, ctx);
|
match sig {
|
||||||
|
Signal::Int => {
|
||||||
|
info!("SIGINT received, exiting");
|
||||||
|
self.exit = true;
|
||||||
|
self.handle_cmd(ServerCommand::Stop {
|
||||||
|
graceful: false,
|
||||||
|
completion: None,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
Signal::Term => {
|
||||||
|
info!("SIGTERM received, stopping");
|
||||||
|
self.exit = true;
|
||||||
|
self.handle_cmd(ServerCommand::Stop {
|
||||||
|
graceful: true,
|
||||||
|
completion: None,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
Signal::Quit => {
|
||||||
|
info!("SIGQUIT received, exiting");
|
||||||
|
self.exit = true;
|
||||||
|
self.handle_cmd(ServerCommand::Stop {
|
||||||
|
graceful: false,
|
||||||
|
completion: None,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
_ => (),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
_ => (),
|
ServerCommand::Stop {
|
||||||
}
|
graceful,
|
||||||
}
|
completion,
|
||||||
}
|
} => {
|
||||||
|
let exit = self.exit;
|
||||||
|
|
||||||
impl Handler<PauseServer> for Server {
|
// stop accept thread
|
||||||
type Result = ();
|
self.accept.send(Command::Stop);
|
||||||
|
|
||||||
fn handle(&mut self, _: PauseServer, _: &mut Context<Self>) {
|
// stop workers
|
||||||
self.accept.send(Command::Pause);
|
if !self.workers.is_empty() {
|
||||||
}
|
spawn(
|
||||||
}
|
futures_unordered(
|
||||||
|
self.workers
|
||||||
impl Handler<ResumeServer> for Server {
|
.iter()
|
||||||
type Result = ();
|
.map(move |worker| worker.1.stop(graceful)),
|
||||||
|
)
|
||||||
fn handle(&mut self, _: ResumeServer, _: &mut Context<Self>) {
|
.collect()
|
||||||
self.accept.send(Command::Resume);
|
.then(move |_| {
|
||||||
}
|
if let Some(tx) = completion {
|
||||||
}
|
let _ = tx.send(());
|
||||||
|
|
||||||
impl Handler<StopServer> for Server {
|
|
||||||
type Result = Response<(), ()>;
|
|
||||||
|
|
||||||
fn handle(&mut self, msg: StopServer, ctx: &mut Context<Self>) -> Self::Result {
|
|
||||||
// stop accept thread
|
|
||||||
self.accept.send(Command::Stop);
|
|
||||||
|
|
||||||
// stop workers
|
|
||||||
let (tx, rx) = mpsc::channel(1);
|
|
||||||
|
|
||||||
for worker in &self.workers {
|
|
||||||
let tx2 = tx.clone();
|
|
||||||
ctx.spawn(
|
|
||||||
worker
|
|
||||||
.1
|
|
||||||
.stop(msg.graceful)
|
|
||||||
.into_actor(self)
|
|
||||||
.then(move |_, slf, ctx| {
|
|
||||||
slf.workers.pop();
|
|
||||||
if slf.workers.is_empty() {
|
|
||||||
let _ = tx2.send(());
|
|
||||||
|
|
||||||
// we need to stop system if server was spawned
|
|
||||||
if slf.exit {
|
|
||||||
ctx.run_later(Duration::from_millis(300), |_, _| {
|
|
||||||
System::current().stop();
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
}
|
if exit {
|
||||||
|
spawn(sleep(Duration::from_millis(300)).then(|_| {
|
||||||
fut::ok(())
|
System::current().stop();
|
||||||
}),
|
ok(())
|
||||||
);
|
}));
|
||||||
}
|
}
|
||||||
|
ok(())
|
||||||
if !self.workers.is_empty() {
|
}),
|
||||||
Response::r#async(rx.into_future().map(|_| ()).map_err(|_| ()))
|
)
|
||||||
} else {
|
} else {
|
||||||
// we need to stop system if server was spawned
|
// we need to stop system if server was spawned
|
||||||
if self.exit {
|
if self.exit {
|
||||||
ctx.run_later(Duration::from_millis(300), |_, _| {
|
spawn(sleep(Duration::from_millis(300)).then(|_| {
|
||||||
System::current().stop();
|
System::current().stop();
|
||||||
});
|
ok(())
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
if let Some(tx) = completion {
|
||||||
|
let _ = tx.send(());
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
Response::reply(Ok(()))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Commands from accept threads
|
|
||||||
impl StreamHandler<ServerCommand, ()> for Server {
|
|
||||||
fn finished(&mut self, _: &mut Context<Self>) {}
|
|
||||||
|
|
||||||
fn handle(&mut self, msg: ServerCommand, _: &mut Context<Self>) {
|
|
||||||
match msg {
|
|
||||||
ServerCommand::WorkerDied(idx) => {
|
ServerCommand::WorkerDied(idx) => {
|
||||||
let mut found = false;
|
let mut found = false;
|
||||||
for i in 0..self.workers.len() {
|
for i in 0..self.workers.len() {
|
||||||
@ -429,6 +393,21 @@ impl StreamHandler<ServerCommand, ()> for Server {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl Future for ServerBuilder {
|
||||||
|
type Item = ();
|
||||||
|
type Error = ();
|
||||||
|
|
||||||
|
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
|
||||||
|
loop {
|
||||||
|
match self.cmd.poll() {
|
||||||
|
Ok(Async::Ready(None)) | Err(_) => return Ok(Async::Ready(())),
|
||||||
|
Ok(Async::NotReady) => return Ok(Async::NotReady),
|
||||||
|
Ok(Async::Ready(Some(item))) => self.handle_cmd(item),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub(super) fn bind_addr<S: net::ToSocketAddrs>(addr: S) -> io::Result<Vec<net::TcpListener>> {
|
pub(super) fn bind_addr<S: net::ToSocketAddrs>(addr: S) -> io::Result<Vec<net::TcpListener>> {
|
||||||
let mut err = None;
|
let mut err = None;
|
||||||
let mut succ = false;
|
let mut succ = false;
|
@ -8,25 +8,35 @@ use tokio_tcp::TcpStream;
|
|||||||
|
|
||||||
use crate::counter::CounterGuard;
|
use crate::counter::CounterGuard;
|
||||||
|
|
||||||
use super::server::bind_addr;
|
use super::builder::bind_addr;
|
||||||
use super::services::{
|
use super::services::{
|
||||||
BoxedServerService, InternalServiceFactory, ServerMessage, StreamService,
|
BoxedServerService, InternalServiceFactory, ServerMessage, StreamService,
|
||||||
};
|
};
|
||||||
use super::Token;
|
use super::Token;
|
||||||
|
|
||||||
pub struct ServiceConfig {
|
pub struct ServiceConfig {
|
||||||
pub(super) services: Vec<(String, net::TcpListener)>,
|
pub(crate) services: Vec<(String, net::TcpListener)>,
|
||||||
pub(super) rt: Box<ServiceRuntimeConfiguration>,
|
pub(crate) apply: Option<Box<ServiceRuntimeConfiguration>>,
|
||||||
|
pub(crate) threads: usize,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ServiceConfig {
|
impl ServiceConfig {
|
||||||
pub(super) fn new() -> ServiceConfig {
|
pub(super) fn new(threads: usize) -> ServiceConfig {
|
||||||
ServiceConfig {
|
ServiceConfig {
|
||||||
|
threads,
|
||||||
services: Vec::new(),
|
services: Vec::new(),
|
||||||
rt: Box::new(not_configured),
|
apply: None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Set number of workers to start.
|
||||||
|
///
|
||||||
|
/// By default server uses number of available logical cpu as workers
|
||||||
|
/// count.
|
||||||
|
pub fn workers(&mut self, num: usize) {
|
||||||
|
self.threads = num;
|
||||||
|
}
|
||||||
|
|
||||||
/// Add new service to server
|
/// Add new service to server
|
||||||
pub fn bind<U, N: AsRef<str>>(&mut self, name: N, addr: U) -> io::Result<&mut Self>
|
pub fn bind<U, N: AsRef<str>>(&mut self, name: N, addr: U) -> io::Result<&mut Self>
|
||||||
where
|
where
|
||||||
@ -43,16 +53,20 @@ impl ServiceConfig {
|
|||||||
|
|
||||||
/// Add new service to server
|
/// Add new service to server
|
||||||
pub fn listen<N: AsRef<str>>(&mut self, name: N, lst: net::TcpListener) -> &mut Self {
|
pub fn listen<N: AsRef<str>>(&mut self, name: N, lst: net::TcpListener) -> &mut Self {
|
||||||
|
if self.apply.is_none() {
|
||||||
|
self.apply = Some(Box::new(not_configured));
|
||||||
|
}
|
||||||
self.services.push((name.as_ref().to_string(), lst));
|
self.services.push((name.as_ref().to_string(), lst));
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Register service configuration function
|
/// Register service configuration function. This function get called
|
||||||
pub fn rt<F>(&mut self, f: F) -> io::Result<()>
|
/// during worker runtime configuration. It get executed in worker thread.
|
||||||
|
pub fn apply<F>(&mut self, f: F) -> io::Result<()>
|
||||||
where
|
where
|
||||||
F: Fn(&mut ServiceRuntime) + Send + Clone + 'static,
|
F: Fn(&mut ServiceRuntime) + Send + Clone + 'static,
|
||||||
{
|
{
|
||||||
self.rt = Box::new(f);
|
self.apply = Some(Box::new(f));
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
80
actix-server/src/counter.rs
Normal file
80
actix-server/src/counter.rs
Normal file
@ -0,0 +1,80 @@
|
|||||||
|
use std::cell::Cell;
|
||||||
|
use std::rc::Rc;
|
||||||
|
|
||||||
|
use futures::task::AtomicTask;
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
/// Simple counter with ability to notify task on reaching specific number
|
||||||
|
///
|
||||||
|
/// Counter could be cloned, total ncount is shared across all clones.
|
||||||
|
pub struct Counter(Rc<CounterInner>);
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
struct CounterInner {
|
||||||
|
count: Cell<usize>,
|
||||||
|
capacity: usize,
|
||||||
|
task: AtomicTask,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Counter {
|
||||||
|
/// Create `Counter` instance and set max value.
|
||||||
|
pub fn new(capacity: usize) -> Self {
|
||||||
|
Counter(Rc::new(CounterInner {
|
||||||
|
capacity,
|
||||||
|
count: Cell::new(0),
|
||||||
|
task: AtomicTask::new(),
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get(&self) -> CounterGuard {
|
||||||
|
CounterGuard::new(self.0.clone())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Check if counter is not at capacity
|
||||||
|
pub fn available(&self) -> bool {
|
||||||
|
self.0.available()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get total number of acquired counts
|
||||||
|
pub fn total(&self) -> usize {
|
||||||
|
self.0.count.get()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub struct CounterGuard(Rc<CounterInner>);
|
||||||
|
|
||||||
|
impl CounterGuard {
|
||||||
|
fn new(inner: Rc<CounterInner>) -> Self {
|
||||||
|
inner.inc();
|
||||||
|
CounterGuard(inner)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Drop for CounterGuard {
|
||||||
|
fn drop(&mut self) {
|
||||||
|
self.0.dec();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl CounterInner {
|
||||||
|
fn inc(&self) {
|
||||||
|
self.count.set(self.count.get() + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn dec(&self) {
|
||||||
|
let num = self.count.get();
|
||||||
|
self.count.set(num - 1);
|
||||||
|
if num == self.capacity {
|
||||||
|
self.task.notify();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn available(&self) -> bool {
|
||||||
|
let avail = self.count.get() < self.capacity;
|
||||||
|
if !avail {
|
||||||
|
self.task.register();
|
||||||
|
}
|
||||||
|
avail
|
||||||
|
}
|
||||||
|
}
|
33
actix-server/src/lib.rs
Normal file
33
actix-server/src/lib.rs
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
//! General purpose tcp server
|
||||||
|
|
||||||
|
mod accept;
|
||||||
|
mod builder;
|
||||||
|
mod config;
|
||||||
|
mod counter;
|
||||||
|
mod server;
|
||||||
|
mod services;
|
||||||
|
mod signals;
|
||||||
|
pub mod ssl;
|
||||||
|
mod worker;
|
||||||
|
|
||||||
|
pub use self::builder::ServerBuilder;
|
||||||
|
pub use self::config::{ServiceConfig, ServiceRuntime};
|
||||||
|
pub use self::server::Server;
|
||||||
|
pub use self::services::{ServerMessage, ServiceFactory, StreamServiceFactory};
|
||||||
|
|
||||||
|
/// Socket id token
|
||||||
|
#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)]
|
||||||
|
pub(crate) struct Token(usize);
|
||||||
|
|
||||||
|
impl Token {
|
||||||
|
pub(crate) fn next(&mut self) -> Token {
|
||||||
|
let token = Token(self.0 + 1);
|
||||||
|
self.0 += 1;
|
||||||
|
token
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Start server building process
|
||||||
|
pub fn build() -> ServerBuilder {
|
||||||
|
ServerBuilder::default()
|
||||||
|
}
|
69
actix-server/src/server.rs
Normal file
69
actix-server/src/server.rs
Normal file
@ -0,0 +1,69 @@
|
|||||||
|
use futures::sync::mpsc::UnboundedSender;
|
||||||
|
use futures::sync::oneshot;
|
||||||
|
use futures::Future;
|
||||||
|
|
||||||
|
use crate::builder::ServerBuilder;
|
||||||
|
use crate::signals::Signal;
|
||||||
|
|
||||||
|
pub(crate) enum ServerCommand {
|
||||||
|
WorkerDied(usize),
|
||||||
|
Pause(oneshot::Sender<()>),
|
||||||
|
Resume(oneshot::Sender<()>),
|
||||||
|
Signal(Signal),
|
||||||
|
/// Whether to try and shut down gracefully
|
||||||
|
Stop {
|
||||||
|
graceful: bool,
|
||||||
|
completion: Option<oneshot::Sender<()>>,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct Server(UnboundedSender<ServerCommand>);
|
||||||
|
|
||||||
|
impl Server {
|
||||||
|
pub(crate) fn new(tx: UnboundedSender<ServerCommand>) -> Self {
|
||||||
|
Server(tx)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Start server building process
|
||||||
|
pub fn build() -> ServerBuilder {
|
||||||
|
ServerBuilder::default()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn signal(&self, sig: Signal) {
|
||||||
|
let _ = self.0.unbounded_send(ServerCommand::Signal(sig));
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn worker_died(&self, idx: usize) {
|
||||||
|
let _ = self.0.unbounded_send(ServerCommand::WorkerDied(idx));
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Pause accepting incoming connections
|
||||||
|
///
|
||||||
|
/// If socket contains some pending connection, they might be dropped.
|
||||||
|
/// All opened connection remains active.
|
||||||
|
pub fn pause(&self) -> impl Future<Item = (), Error = ()> {
|
||||||
|
let (tx, rx) = oneshot::channel();
|
||||||
|
let _ = self.0.unbounded_send(ServerCommand::Pause(tx));
|
||||||
|
rx.map_err(|_| ())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Resume accepting incoming connections
|
||||||
|
pub fn resume(&self) -> impl Future<Item = (), Error = ()> {
|
||||||
|
let (tx, rx) = oneshot::channel();
|
||||||
|
let _ = self.0.unbounded_send(ServerCommand::Resume(tx));
|
||||||
|
rx.map_err(|_| ())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Stop incoming connection processing, stop all workers and exit.
|
||||||
|
///
|
||||||
|
/// If server starts with `spawn()` method, then spawned thread get terminated.
|
||||||
|
pub fn stop(&self, graceful: bool) -> impl Future<Item = (), Error = ()> {
|
||||||
|
let (tx, rx) = oneshot::channel();
|
||||||
|
let _ = self.0.unbounded_send(ServerCommand::Stop {
|
||||||
|
graceful,
|
||||||
|
completion: Some(tx),
|
||||||
|
});
|
||||||
|
rx.map_err(|_| ())
|
||||||
|
}
|
||||||
|
}
|
@ -1,11 +1,11 @@
|
|||||||
use std::net;
|
use std::net;
|
||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
|
|
||||||
|
use actix_rt::spawn;
|
||||||
use actix_service::{NewService, Service};
|
use actix_service::{NewService, Service};
|
||||||
use futures::future::{err, ok, FutureResult};
|
use futures::future::{err, ok, FutureResult};
|
||||||
use futures::{Future, Poll};
|
use futures::{Future, Poll};
|
||||||
use log::error;
|
use log::error;
|
||||||
use tokio_current_thread::spawn;
|
|
||||||
use tokio_reactor::Handle;
|
use tokio_reactor::Handle;
|
||||||
use tokio_tcp::TcpStream;
|
use tokio_tcp::TcpStream;
|
||||||
|
|
||||||
@ -83,9 +83,9 @@ where
|
|||||||
});
|
});
|
||||||
|
|
||||||
if let Ok(stream) = stream {
|
if let Ok(stream) = stream {
|
||||||
spawn(self.service.call(stream).map_err(|_| ()).map(move |val| {
|
spawn(self.service.call(stream).then(move |res| {
|
||||||
drop(guard);
|
drop(guard);
|
||||||
val
|
res.map_err(|_| ())
|
||||||
}));
|
}));
|
||||||
ok(())
|
ok(())
|
||||||
} else {
|
} else {
|
||||||
@ -122,9 +122,9 @@ where
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn call(&mut self, (guard, req): (Option<CounterGuard>, ServerMessage)) -> Self::Future {
|
fn call(&mut self, (guard, req): (Option<CounterGuard>, ServerMessage)) -> Self::Future {
|
||||||
spawn(self.service.call(req).map_err(|_| ()).map(move |val| {
|
spawn(self.service.call(req).then(move |res| {
|
||||||
drop(guard);
|
drop(guard);
|
||||||
val
|
res.map_err(|_| ())
|
||||||
}));
|
}));
|
||||||
ok(())
|
ok(())
|
||||||
}
|
}
|
118
actix-server/src/signals.rs
Normal file
118
actix-server/src/signals.rs
Normal file
@ -0,0 +1,118 @@
|
|||||||
|
use std::io;
|
||||||
|
|
||||||
|
use actix_rt::spawn;
|
||||||
|
use futures::stream::futures_unordered;
|
||||||
|
use futures::{Async, Future, Poll, Stream};
|
||||||
|
|
||||||
|
use crate::server::Server;
|
||||||
|
|
||||||
|
/// Different types of process signals
|
||||||
|
#[derive(PartialEq, Clone, Copy, Debug)]
|
||||||
|
pub(crate) enum Signal {
|
||||||
|
/// SIGHUP
|
||||||
|
Hup,
|
||||||
|
/// SIGINT
|
||||||
|
Int,
|
||||||
|
/// SIGTERM
|
||||||
|
Term,
|
||||||
|
/// SIGQUIT
|
||||||
|
Quit,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) struct Signals {
|
||||||
|
srv: Server,
|
||||||
|
#[cfg(not(unix))]
|
||||||
|
stream: SigStream,
|
||||||
|
#[cfg(unix)]
|
||||||
|
streams: Vec<SigStream>,
|
||||||
|
}
|
||||||
|
|
||||||
|
type SigStream = Box<Stream<Item = Signal, Error = io::Error>>;
|
||||||
|
|
||||||
|
impl Signals {
|
||||||
|
pub(crate) fn start(srv: Server) {
|
||||||
|
let fut = {
|
||||||
|
#[cfg(not(unix))]
|
||||||
|
{
|
||||||
|
tokio_signal::ctrl_c()
|
||||||
|
.map_err(|_| ())
|
||||||
|
.and_then(move |stream| Signals {
|
||||||
|
srv,
|
||||||
|
stream: Box::new(stream.map(|_| Signal::Int)),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(unix)]
|
||||||
|
{
|
||||||
|
use tokio_signal::unix;
|
||||||
|
|
||||||
|
let mut sigs: Vec<Box<Future<Item = SigStream, Error = io::Error>>> =
|
||||||
|
Vec::new();
|
||||||
|
sigs.push(Box::new(
|
||||||
|
tokio_signal::unix::Signal::new(tokio_signal::unix::SIGINT).map(|stream| {
|
||||||
|
let s: SigStream = Box::new(stream.map(|_| Signal::Int));
|
||||||
|
s
|
||||||
|
}),
|
||||||
|
));
|
||||||
|
sigs.push(Box::new(
|
||||||
|
tokio_signal::unix::Signal::new(tokio_signal::unix::SIGHUP).map(
|
||||||
|
|stream: unix::Signal| {
|
||||||
|
let s: SigStream = Box::new(stream.map(|_| Signal::Hup));
|
||||||
|
s
|
||||||
|
},
|
||||||
|
),
|
||||||
|
));
|
||||||
|
sigs.push(Box::new(
|
||||||
|
tokio_signal::unix::Signal::new(tokio_signal::unix::SIGTERM).map(
|
||||||
|
|stream| {
|
||||||
|
let s: SigStream = Box::new(stream.map(|_| Signal::Term));
|
||||||
|
s
|
||||||
|
},
|
||||||
|
),
|
||||||
|
));
|
||||||
|
sigs.push(Box::new(
|
||||||
|
tokio_signal::unix::Signal::new(tokio_signal::unix::SIGQUIT).map(
|
||||||
|
|stream| {
|
||||||
|
let s: SigStream = Box::new(stream.map(|_| Signal::Quit));
|
||||||
|
s
|
||||||
|
},
|
||||||
|
),
|
||||||
|
));
|
||||||
|
futures_unordered(sigs)
|
||||||
|
.collect()
|
||||||
|
.map_err(|_| ())
|
||||||
|
.and_then(move |streams| Signals { srv, streams })
|
||||||
|
}
|
||||||
|
};
|
||||||
|
spawn(fut);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Future for Signals {
|
||||||
|
type Item = ();
|
||||||
|
type Error = ();
|
||||||
|
|
||||||
|
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
|
||||||
|
#[cfg(not(unix))]
|
||||||
|
loop {
|
||||||
|
match self.stream.poll() {
|
||||||
|
Ok(Async::Ready(None)) | Err(_) => return Ok(Async::Ready(())),
|
||||||
|
Ok(Async::Ready(Some(sig))) => self.srv.signal(sig),
|
||||||
|
Ok(Async::NotReady) => return Ok(Async::NotReady),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#[cfg(unix)]
|
||||||
|
{
|
||||||
|
for s in &mut self.streams {
|
||||||
|
loop {
|
||||||
|
match s.poll() {
|
||||||
|
Ok(Async::Ready(None)) | Err(_) => return Ok(Async::Ready(())),
|
||||||
|
Ok(Async::NotReady) => break,
|
||||||
|
Ok(Async::Ready(Some(sig))) => self.srv.signal(sig),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(Async::NotReady)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,12 +1,12 @@
|
|||||||
//! SSL Services
|
//! SSL Services
|
||||||
use std::sync::atomic::{AtomicUsize, Ordering};
|
use std::sync::atomic::{AtomicUsize, Ordering};
|
||||||
|
|
||||||
use super::counter::Counter;
|
use crate::counter::Counter;
|
||||||
|
|
||||||
#[cfg(feature = "ssl")]
|
#[cfg(feature = "ssl")]
|
||||||
mod openssl;
|
mod openssl;
|
||||||
#[cfg(feature = "ssl")]
|
#[cfg(feature = "ssl")]
|
||||||
pub use self::openssl::{OpensslAcceptor, OpensslConnector};
|
pub use self::openssl::OpensslAcceptor;
|
||||||
|
|
||||||
#[cfg(feature = "tls")]
|
#[cfg(feature = "tls")]
|
||||||
mod nativetls;
|
mod nativetls;
|
||||||
@ -28,7 +28,7 @@ pub fn max_concurrent_ssl_connect(num: usize) {
|
|||||||
MAX_CONN.store(num, Ordering::Relaxed);
|
MAX_CONN.store(num, Ordering::Relaxed);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) const MAX_CONN: AtomicUsize = AtomicUsize::new(256);
|
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));
|
@ -21,7 +21,7 @@ impl<T: AsyncRead + AsyncWrite> NativeTlsAcceptor<T> {
|
|||||||
/// Create `NativeTlsAcceptor` instance
|
/// Create `NativeTlsAcceptor` instance
|
||||||
pub fn new(acceptor: TlsAcceptor) -> Self {
|
pub fn new(acceptor: TlsAcceptor) -> Self {
|
||||||
NativeTlsAcceptor {
|
NativeTlsAcceptor {
|
||||||
acceptor: acceptor.into(),
|
acceptor,
|
||||||
io: PhantomData,
|
io: PhantomData,
|
||||||
}
|
}
|
||||||
}
|
}
|
99
actix-server/src/ssl/openssl.rs
Normal file
99
actix-server/src/ssl/openssl.rs
Normal file
@ -0,0 +1,99 @@
|
|||||||
|
use std::marker::PhantomData;
|
||||||
|
|
||||||
|
use actix_service::{NewService, Service};
|
||||||
|
use futures::{future::ok, future::FutureResult, Async, Future, Poll};
|
||||||
|
use openssl::ssl::{HandshakeError, SslAcceptor};
|
||||||
|
use tokio_io::{AsyncRead, AsyncWrite};
|
||||||
|
use tokio_openssl::{AcceptAsync, SslAcceptorExt, SslStream};
|
||||||
|
|
||||||
|
use super::MAX_CONN_COUNTER;
|
||||||
|
use crate::counter::{Counter, CounterGuard};
|
||||||
|
|
||||||
|
/// Support `SSL` connections via openssl package
|
||||||
|
///
|
||||||
|
/// `ssl` feature enables `OpensslAcceptor` type
|
||||||
|
pub struct OpensslAcceptor<T> {
|
||||||
|
acceptor: SslAcceptor,
|
||||||
|
io: PhantomData<T>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> OpensslAcceptor<T> {
|
||||||
|
/// Create default `OpensslAcceptor`
|
||||||
|
pub fn new(acceptor: SslAcceptor) -> Self {
|
||||||
|
OpensslAcceptor {
|
||||||
|
acceptor,
|
||||||
|
io: PhantomData,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: AsyncRead + AsyncWrite> Clone for OpensslAcceptor<T> {
|
||||||
|
fn clone(&self) -> Self {
|
||||||
|
Self {
|
||||||
|
acceptor: self.acceptor.clone(),
|
||||||
|
io: PhantomData,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: AsyncRead + AsyncWrite> NewService<T> for OpensslAcceptor<T> {
|
||||||
|
type Response = SslStream<T>;
|
||||||
|
type Error = HandshakeError<T>;
|
||||||
|
type Service = OpensslAcceptorService<T>;
|
||||||
|
type InitError = ();
|
||||||
|
type Future = FutureResult<Self::Service, Self::InitError>;
|
||||||
|
|
||||||
|
fn new_service(&self) -> Self::Future {
|
||||||
|
MAX_CONN_COUNTER.with(|conns| {
|
||||||
|
ok(OpensslAcceptorService {
|
||||||
|
acceptor: self.acceptor.clone(),
|
||||||
|
conns: conns.clone(),
|
||||||
|
io: PhantomData,
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct OpensslAcceptorService<T> {
|
||||||
|
acceptor: SslAcceptor,
|
||||||
|
io: PhantomData<T>,
|
||||||
|
conns: Counter,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: AsyncRead + AsyncWrite> Service<T> for OpensslAcceptorService<T> {
|
||||||
|
type Response = SslStream<T>;
|
||||||
|
type Error = HandshakeError<T>;
|
||||||
|
type Future = OpensslAcceptorServiceFut<T>;
|
||||||
|
|
||||||
|
fn poll_ready(&mut self) -> Poll<(), Self::Error> {
|
||||||
|
if self.conns.available() {
|
||||||
|
Ok(Async::Ready(()))
|
||||||
|
} else {
|
||||||
|
Ok(Async::NotReady)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn call(&mut self, req: T) -> Self::Future {
|
||||||
|
OpensslAcceptorServiceFut {
|
||||||
|
_guard: self.conns.get(),
|
||||||
|
fut: SslAcceptorExt::accept_async(&self.acceptor, req),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct OpensslAcceptorServiceFut<T>
|
||||||
|
where
|
||||||
|
T: AsyncRead + AsyncWrite,
|
||||||
|
{
|
||||||
|
fut: AcceptAsync<T>,
|
||||||
|
_guard: CounterGuard,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: AsyncRead + AsyncWrite> Future for OpensslAcceptorServiceFut<T> {
|
||||||
|
type Item = SslStream<T>;
|
||||||
|
type Error = HandshakeError<T>;
|
||||||
|
|
||||||
|
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
|
||||||
|
self.fut.poll()
|
||||||
|
}
|
||||||
|
}
|
@ -2,20 +2,17 @@ use std::sync::atomic::{AtomicBool, AtomicUsize, Ordering};
|
|||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use std::{mem, net, time};
|
use std::{mem, net, time};
|
||||||
|
|
||||||
|
use actix_rt::{spawn, Arbiter};
|
||||||
use futures::sync::mpsc::{UnboundedReceiver, UnboundedSender};
|
use futures::sync::mpsc::{UnboundedReceiver, UnboundedSender};
|
||||||
use futures::sync::oneshot;
|
use futures::sync::oneshot;
|
||||||
use futures::{future, Async, Future, Poll, Stream};
|
use futures::{future, Async, Future, Poll, Stream};
|
||||||
use log::{error, info, trace};
|
use log::{error, info, trace};
|
||||||
use tokio_current_thread::spawn;
|
|
||||||
use tokio_timer::{sleep, Delay};
|
use tokio_timer::{sleep, Delay};
|
||||||
|
|
||||||
use actix::msgs::StopArbiter;
|
use crate::accept::AcceptNotify;
|
||||||
use actix::{Arbiter, Message};
|
|
||||||
|
|
||||||
use super::accept::AcceptNotify;
|
|
||||||
use super::services::{BoxedServerService, InternalServiceFactory, ServerMessage};
|
|
||||||
use super::Token;
|
|
||||||
use crate::counter::Counter;
|
use crate::counter::Counter;
|
||||||
|
use crate::services::{BoxedServerService, InternalServiceFactory, ServerMessage};
|
||||||
|
use crate::Token;
|
||||||
|
|
||||||
pub(crate) struct WorkerCommand(Conn);
|
pub(crate) struct WorkerCommand(Conn);
|
||||||
|
|
||||||
@ -26,14 +23,14 @@ pub(crate) struct StopCommand {
|
|||||||
result: oneshot::Sender<bool>,
|
result: oneshot::Sender<bool>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Message)]
|
#[derive(Debug)]
|
||||||
pub(crate) struct Conn {
|
pub(crate) struct Conn {
|
||||||
pub io: net::TcpStream,
|
pub io: net::TcpStream,
|
||||||
pub token: Token,
|
pub token: Token,
|
||||||
pub peer: Option<net::SocketAddr>,
|
pub peer: Option<net::SocketAddr>,
|
||||||
}
|
}
|
||||||
|
|
||||||
const MAX_CONNS: AtomicUsize = AtomicUsize::new(25600);
|
static MAX_CONNS: AtomicUsize = AtomicUsize::new(25600);
|
||||||
|
|
||||||
/// Sets the maximum per-worker number of concurrent connections.
|
/// Sets the maximum per-worker number of concurrent connections.
|
||||||
///
|
///
|
||||||
@ -167,7 +164,7 @@ impl Worker {
|
|||||||
future::join_all(fut)
|
future::join_all(fut)
|
||||||
.map_err(|e| {
|
.map_err(|e| {
|
||||||
error!("Can not start worker: {:?}", e);
|
error!("Can not start worker: {:?}", e);
|
||||||
Arbiter::current().do_send(StopArbiter(0));
|
Arbiter::current().stop();
|
||||||
})
|
})
|
||||||
.and_then(move |services| {
|
.and_then(move |services| {
|
||||||
for item in services {
|
for item in services {
|
||||||
@ -365,7 +362,7 @@ impl Future for Worker {
|
|||||||
let num = num_connections();
|
let num = num_connections();
|
||||||
if num == 0 {
|
if num == 0 {
|
||||||
let _ = tx.send(true);
|
let _ = tx.send(true);
|
||||||
Arbiter::current().do_send(StopArbiter(0));
|
Arbiter::current().stop();
|
||||||
return Ok(Async::Ready(()));
|
return Ok(Async::Ready(()));
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -375,7 +372,7 @@ impl Future for Worker {
|
|||||||
Async::Ready(_) => {
|
Async::Ready(_) => {
|
||||||
self.shutdown(true);
|
self.shutdown(true);
|
||||||
let _ = tx.send(false);
|
let _ = tx.send(false);
|
||||||
Arbiter::current().do_send(StopArbiter(0));
|
Arbiter::current().stop();
|
||||||
return Ok(Async::Ready(()));
|
return Ok(Async::Ready(()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -401,7 +398,7 @@ impl Future for Worker {
|
|||||||
let guard = self.conns.get();
|
let guard = self.conns.get();
|
||||||
let _ = self.services[msg.token.0]
|
let _ = self.services[msg.token.0]
|
||||||
.as_mut()
|
.as_mut()
|
||||||
.expect("actix net bug")
|
.expect("actix-server bug")
|
||||||
.1
|
.1
|
||||||
.call((Some(guard), ServerMessage::Connect(msg.io)));
|
.call((Some(guard), ServerMessage::Connect(msg.io)));
|
||||||
continue;
|
continue;
|
@ -1,5 +1,26 @@
|
|||||||
# Changes
|
# Changes
|
||||||
|
|
||||||
|
## [0.1.4] - 2019-01-11
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
* Use `FnMut` instead of `Fn` for `FnService`
|
||||||
|
|
||||||
|
|
||||||
|
## [0.1.3] - 2018-12-12
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
* Split service combinators to separate trait
|
||||||
|
|
||||||
|
|
||||||
|
## [0.1.2] - 2018-12-12
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
* Release future early for `.and_then()` and `.then()` combinators
|
||||||
|
|
||||||
|
|
||||||
## [0.1.1] - 2018-12-09
|
## [0.1.1] - 2018-12-09
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "actix-service"
|
name = "actix-service"
|
||||||
version = "0.1.1"
|
version = "0.1.4"
|
||||||
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"]
|
||||||
|
@ -61,7 +61,7 @@ where
|
|||||||
{
|
{
|
||||||
b: Cell<B>,
|
b: Cell<B>,
|
||||||
fut_b: Option<B::Future>,
|
fut_b: Option<B::Future>,
|
||||||
fut_a: A::Future,
|
fut_a: Option<A::Future>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<A, B, Request> AndThenFuture<A, B, Request>
|
impl<A, B, Request> AndThenFuture<A, B, Request>
|
||||||
@ -69,10 +69,10 @@ where
|
|||||||
A: Service<Request>,
|
A: Service<Request>,
|
||||||
B: Service<A::Response, Error = A::Error>,
|
B: Service<A::Response, Error = A::Error>,
|
||||||
{
|
{
|
||||||
fn new(fut_a: A::Future, b: Cell<B>) -> Self {
|
fn new(a: A::Future, b: Cell<B>) -> Self {
|
||||||
AndThenFuture {
|
AndThenFuture {
|
||||||
b,
|
b,
|
||||||
fut_a,
|
fut_a: Some(a),
|
||||||
fut_b: None,
|
fut_b: None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -91,8 +91,9 @@ where
|
|||||||
return fut.poll();
|
return fut.poll();
|
||||||
}
|
}
|
||||||
|
|
||||||
match self.fut_a.poll() {
|
match self.fut_a.as_mut().expect("Bug in actix-service").poll() {
|
||||||
Ok(Async::Ready(resp)) => {
|
Ok(Async::Ready(resp)) => {
|
||||||
|
let _ = self.fut_a.take();
|
||||||
self.fut_b = Some(self.b.get_mut().call(resp));
|
self.fut_b = Some(self.b.get_mut().call(resp));
|
||||||
self.poll()
|
self.poll()
|
||||||
}
|
}
|
||||||
@ -218,7 +219,7 @@ mod tests {
|
|||||||
use std::rc::Rc;
|
use std::rc::Rc;
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::{NewService, Service};
|
use crate::{NewService, Service, ServiceExt};
|
||||||
|
|
||||||
struct Srv1(Rc<Cell<usize>>);
|
struct Srv1(Rc<Cell<usize>>);
|
||||||
impl Service<&'static str> for Srv1 {
|
impl Service<&'static str> for Srv1 {
|
||||||
|
@ -171,7 +171,7 @@ 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};
|
use crate::{IntoNewService, IntoService, NewService, Service, ServiceExt};
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
struct Srv;
|
struct Srv;
|
||||||
|
@ -9,7 +9,7 @@ use super::{IntoNewService, IntoService, NewService, Service};
|
|||||||
|
|
||||||
pub struct FnService<F, Req, Resp, E, Fut>
|
pub struct FnService<F, Req, Resp, E, Fut>
|
||||||
where
|
where
|
||||||
F: Fn(Req) -> Fut,
|
F: FnMut(Req) -> Fut,
|
||||||
Fut: IntoFuture<Item = Resp, Error = E>,
|
Fut: IntoFuture<Item = Resp, Error = E>,
|
||||||
{
|
{
|
||||||
f: F,
|
f: F,
|
||||||
@ -18,7 +18,7 @@ where
|
|||||||
|
|
||||||
impl<F, Req, Resp, E, Fut> FnService<F, Req, Resp, E, Fut>
|
impl<F, Req, Resp, E, Fut> FnService<F, Req, Resp, E, Fut>
|
||||||
where
|
where
|
||||||
F: Fn(Req) -> Fut,
|
F: FnMut(Req) -> Fut,
|
||||||
Fut: IntoFuture<Item = Resp, Error = E>,
|
Fut: IntoFuture<Item = Resp, Error = E>,
|
||||||
{
|
{
|
||||||
pub fn new(f: F) -> Self {
|
pub fn new(f: F) -> Self {
|
||||||
@ -31,7 +31,7 @@ where
|
|||||||
|
|
||||||
impl<F, Req, Resp, E, Fut> Clone for FnService<F, Req, Resp, E, Fut>
|
impl<F, Req, Resp, E, Fut> Clone for FnService<F, Req, Resp, E, Fut>
|
||||||
where
|
where
|
||||||
F: Fn(Req) -> Fut + Clone,
|
F: FnMut(Req) -> Fut + Clone,
|
||||||
Fut: IntoFuture<Item = Resp, Error = E>,
|
Fut: IntoFuture<Item = Resp, Error = E>,
|
||||||
{
|
{
|
||||||
fn clone(&self) -> Self {
|
fn clone(&self) -> Self {
|
||||||
@ -44,7 +44,7 @@ where
|
|||||||
|
|
||||||
impl<F, Req, Resp, E, Fut> Service<Req> for FnService<F, Req, Resp, E, Fut>
|
impl<F, Req, Resp, E, Fut> Service<Req> for FnService<F, Req, Resp, E, Fut>
|
||||||
where
|
where
|
||||||
F: Fn(Req) -> Fut,
|
F: FnMut(Req) -> Fut,
|
||||||
Fut: IntoFuture<Item = Resp, Error = E>,
|
Fut: IntoFuture<Item = Resp, Error = E>,
|
||||||
{
|
{
|
||||||
type Response = Resp;
|
type Response = Resp;
|
||||||
@ -62,7 +62,7 @@ where
|
|||||||
|
|
||||||
impl<F, Req, Resp, Err, Fut> IntoService<FnService<F, Req, Resp, Err, Fut>, Req> for F
|
impl<F, Req, Resp, Err, Fut> IntoService<FnService<F, Req, Resp, Err, Fut>, Req> for F
|
||||||
where
|
where
|
||||||
F: Fn(Req) -> Fut + 'static,
|
F: FnMut(Req) -> Fut + 'static,
|
||||||
Fut: IntoFuture<Item = Resp, Error = Err>,
|
Fut: IntoFuture<Item = Resp, Error = Err>,
|
||||||
{
|
{
|
||||||
fn into_service(self) -> FnService<F, Req, Resp, Err, Fut> {
|
fn into_service(self) -> FnService<F, Req, Resp, Err, Fut> {
|
||||||
@ -72,7 +72,7 @@ where
|
|||||||
|
|
||||||
pub struct FnNewService<F, Req, Resp, Err, Fut>
|
pub struct FnNewService<F, Req, Resp, Err, Fut>
|
||||||
where
|
where
|
||||||
F: Fn(Req) -> Fut,
|
F: FnMut(Req) -> Fut,
|
||||||
Fut: IntoFuture<Item = Resp, Error = Err>,
|
Fut: IntoFuture<Item = Resp, Error = Err>,
|
||||||
{
|
{
|
||||||
f: F,
|
f: F,
|
||||||
@ -81,7 +81,7 @@ where
|
|||||||
|
|
||||||
impl<F, Req, Resp, Err, Fut> FnNewService<F, Req, Resp, Err, Fut>
|
impl<F, Req, Resp, Err, Fut> FnNewService<F, Req, Resp, Err, Fut>
|
||||||
where
|
where
|
||||||
F: Fn(Req) -> Fut + Clone,
|
F: FnMut(Req) -> Fut + Clone,
|
||||||
Fut: IntoFuture<Item = Resp, Error = Err>,
|
Fut: IntoFuture<Item = Resp, Error = Err>,
|
||||||
{
|
{
|
||||||
pub fn new(f: F) -> Self {
|
pub fn new(f: F) -> Self {
|
||||||
@ -94,7 +94,7 @@ where
|
|||||||
|
|
||||||
impl<F, Req, Resp, Err, Fut> NewService<Req> for FnNewService<F, Req, Resp, Err, Fut>
|
impl<F, Req, Resp, Err, Fut> NewService<Req> for FnNewService<F, Req, Resp, Err, Fut>
|
||||||
where
|
where
|
||||||
F: Fn(Req) -> Fut + Clone,
|
F: FnMut(Req) -> Fut + Clone,
|
||||||
Fut: IntoFuture<Item = Resp, Error = Err>,
|
Fut: IntoFuture<Item = Resp, Error = Err>,
|
||||||
{
|
{
|
||||||
type Response = Resp;
|
type Response = Resp;
|
||||||
@ -110,7 +110,7 @@ where
|
|||||||
|
|
||||||
impl<F, Req, Resp, Err, Fut> IntoNewService<FnNewService<F, Req, Resp, Err, Fut>, Req> for F
|
impl<F, Req, Resp, Err, Fut> IntoNewService<FnNewService<F, Req, Resp, Err, Fut>, Req> for F
|
||||||
where
|
where
|
||||||
F: Fn(Req) -> Fut + Clone + 'static,
|
F: FnMut(Req) -> Fut + Clone + 'static,
|
||||||
Fut: IntoFuture<Item = Resp, Error = Err>,
|
Fut: IntoFuture<Item = Resp, Error = Err>,
|
||||||
{
|
{
|
||||||
fn into_new_service(self) -> FnNewService<F, Req, Resp, Err, Fut> {
|
fn into_new_service(self) -> FnNewService<F, Req, Resp, Err, Fut> {
|
||||||
@ -120,7 +120,7 @@ where
|
|||||||
|
|
||||||
impl<F, Req, Resp, Err, Fut> Clone for FnNewService<F, Req, Resp, Err, Fut>
|
impl<F, Req, Resp, Err, Fut> Clone for FnNewService<F, Req, Resp, Err, Fut>
|
||||||
where
|
where
|
||||||
F: Fn(Req) -> Fut + Clone,
|
F: FnMut(Req) -> Fut + Clone,
|
||||||
Fut: IntoFuture<Item = Resp, Error = Err>,
|
Fut: IntoFuture<Item = Resp, Error = Err>,
|
||||||
{
|
{
|
||||||
fn clone(&self) -> Self {
|
fn clone(&self) -> Self {
|
||||||
|
@ -159,7 +159,7 @@ mod tests {
|
|||||||
use futures::future::{err, FutureResult};
|
use futures::future::{err, FutureResult};
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::{IntoNewService, NewService, Service};
|
use crate::{IntoNewService, NewService, Service, ServiceExt};
|
||||||
|
|
||||||
struct Srv;
|
struct Srv;
|
||||||
impl Service<()> for Srv {
|
impl Service<()> for Srv {
|
||||||
|
@ -50,7 +50,11 @@ pub trait Service<Request> {
|
|||||||
/// Calling `call` without calling `poll_ready` is permitted. The
|
/// Calling `call` without calling `poll_ready` is permitted. The
|
||||||
/// implementation must be resilient to this fact.
|
/// implementation must be resilient to this fact.
|
||||||
fn call(&mut self, req: Request) -> Self::Future;
|
fn call(&mut self, req: Request) -> Self::Future;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// An extension trait for `Service`s that provides a variety of convenient
|
||||||
|
/// adapters
|
||||||
|
pub trait ServiceExt<Request>: Service<Request> {
|
||||||
/// 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<T, I, F, Out, Req>(
|
fn apply<T, I, F, Out, Req>(
|
||||||
@ -146,6 +150,8 @@ pub trait Service<Request> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl<T: ?Sized, Request> ServiceExt<Request> for T where T: Service<Request> {}
|
||||||
|
|
||||||
/// Creates new `Service` values.
|
/// Creates new `Service` values.
|
||||||
///
|
///
|
||||||
/// Acts as a service factory. This is useful for cases where new `Service`
|
/// Acts as a service factory. This is useful for cases where new `Service`
|
||||||
|
@ -189,7 +189,7 @@ mod tests {
|
|||||||
use futures::future::{ok, FutureResult};
|
use futures::future::{ok, FutureResult};
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::{IntoNewService, Service};
|
use crate::{IntoNewService, Service, ServiceExt};
|
||||||
|
|
||||||
struct Srv;
|
struct Srv;
|
||||||
impl Service<()> for Srv {
|
impl Service<()> for Srv {
|
||||||
|
@ -190,7 +190,7 @@ mod tests {
|
|||||||
use futures::future::{err, FutureResult};
|
use futures::future::{err, FutureResult};
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::{IntoNewService, NewService, Service};
|
use crate::{IntoNewService, NewService, Service, ServiceExt};
|
||||||
|
|
||||||
struct Srv;
|
struct Srv;
|
||||||
|
|
||||||
|
@ -61,7 +61,7 @@ where
|
|||||||
{
|
{
|
||||||
b: Cell<B>,
|
b: Cell<B>,
|
||||||
fut_b: Option<B::Future>,
|
fut_b: Option<B::Future>,
|
||||||
fut_a: A::Future,
|
fut_a: Option<A::Future>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<A, B, Request> ThenFuture<A, B, Request>
|
impl<A, B, Request> ThenFuture<A, B, Request>
|
||||||
@ -69,10 +69,10 @@ where
|
|||||||
A: Service<Request>,
|
A: Service<Request>,
|
||||||
B: Service<Result<A::Response, A::Error>>,
|
B: Service<Result<A::Response, A::Error>>,
|
||||||
{
|
{
|
||||||
fn new(fut_a: A::Future, b: Cell<B>) -> Self {
|
fn new(a: A::Future, b: Cell<B>) -> Self {
|
||||||
ThenFuture {
|
ThenFuture {
|
||||||
b,
|
b,
|
||||||
fut_a,
|
fut_a: Some(a),
|
||||||
fut_b: None,
|
fut_b: None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -91,12 +91,14 @@ where
|
|||||||
return fut.poll();
|
return fut.poll();
|
||||||
}
|
}
|
||||||
|
|
||||||
match self.fut_a.poll() {
|
match self.fut_a.as_mut().expect("bug in actix-service").poll() {
|
||||||
Ok(Async::Ready(resp)) => {
|
Ok(Async::Ready(resp)) => {
|
||||||
|
let _ = self.fut_a.take();
|
||||||
self.fut_b = Some(self.b.get_mut().call(Ok(resp)));
|
self.fut_b = Some(self.b.get_mut().call(Ok(resp)));
|
||||||
self.poll()
|
self.poll()
|
||||||
}
|
}
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
|
let _ = self.fut_a.take();
|
||||||
self.fut_b = Some(self.b.get_mut().call(Err(err)));
|
self.fut_b = Some(self.b.get_mut().call(Err(err)));
|
||||||
self.poll()
|
self.poll()
|
||||||
}
|
}
|
||||||
@ -221,11 +223,11 @@ where
|
|||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use futures::future::{err, ok, FutureResult};
|
use futures::future::{err, ok, FutureResult};
|
||||||
use futures::{Async, Poll};
|
use futures::{Async, Future, Poll};
|
||||||
use std::cell::Cell;
|
use std::cell::Cell;
|
||||||
use std::rc::Rc;
|
use std::rc::Rc;
|
||||||
|
|
||||||
use super::*;
|
use crate::{IntoNewService, NewService, Service, ServiceExt};
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
struct Srv1(Rc<Cell<usize>>);
|
struct Srv1(Rc<Cell<usize>>);
|
||||||
|
57
actix-test-server/Cargo.toml
Normal file
57
actix-test-server/Cargo.toml
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
[package]
|
||||||
|
name = "actix-test-server"
|
||||||
|
version = "0.1.0"
|
||||||
|
authors = ["Nikolay Kim <fafhrd91@gmail.com>"]
|
||||||
|
description = "Actix test server"
|
||||||
|
keywords = ["network", "framework", "async", "futures"]
|
||||||
|
homepage = "https://actix.rs"
|
||||||
|
repository = "https://github.com/actix/actix-net.git"
|
||||||
|
documentation = "https://docs.rs/actix-test-server/"
|
||||||
|
categories = ["network-programming", "asynchronous"]
|
||||||
|
license = "MIT/Apache-2.0"
|
||||||
|
exclude = [".gitignore", ".travis.yml", ".cargo/config", "appveyor.yml"]
|
||||||
|
edition = "2018"
|
||||||
|
workspace = "../"
|
||||||
|
|
||||||
|
[package.metadata.docs.rs]
|
||||||
|
features = ["ssl", "tls", "rust-tls"]
|
||||||
|
|
||||||
|
[lib]
|
||||||
|
name = "actix_test_server"
|
||||||
|
path = "src/lib.rs"
|
||||||
|
|
||||||
|
[features]
|
||||||
|
default = []
|
||||||
|
|
||||||
|
# tls
|
||||||
|
tls = ["native-tls", "actix-server/tls"]
|
||||||
|
|
||||||
|
# openssl
|
||||||
|
ssl = ["openssl", "actix-server/ssl"]
|
||||||
|
|
||||||
|
# rustls
|
||||||
|
rust-tls = ["rustls", "tokio-rustls", "webpki", "webpki-roots"]
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
actix-rt = "0.1.0"
|
||||||
|
actix-server = "0.1.0"
|
||||||
|
|
||||||
|
log = "0.4"
|
||||||
|
|
||||||
|
# io
|
||||||
|
net2 = "0.2"
|
||||||
|
futures = "0.1"
|
||||||
|
tokio-tcp = "0.1"
|
||||||
|
tokio-reactor = "0.1"
|
||||||
|
|
||||||
|
# native-tls
|
||||||
|
native-tls = { version="0.2", optional = true }
|
||||||
|
|
||||||
|
# openssl
|
||||||
|
openssl = { version="0.10", optional = true }
|
||||||
|
|
||||||
|
#rustls
|
||||||
|
rustls = { version = "^0.14", optional = true }
|
||||||
|
tokio-rustls = { version = "^0.8", optional = true }
|
||||||
|
webpki = { version = "0.18", optional = true }
|
||||||
|
webpki-roots = { version = "0.15", optional = true }
|
142
actix-test-server/src/lib.rs
Normal file
142
actix-test-server/src/lib.rs
Normal file
@ -0,0 +1,142 @@
|
|||||||
|
//! Various helpers for Actix applications to use during testing.
|
||||||
|
use std::sync::mpsc;
|
||||||
|
use std::{net, thread};
|
||||||
|
|
||||||
|
use actix_rt::{Runtime, System};
|
||||||
|
use actix_server::{Server, StreamServiceFactory};
|
||||||
|
|
||||||
|
use futures::Future;
|
||||||
|
use net2::TcpBuilder;
|
||||||
|
use tokio_reactor::Handle;
|
||||||
|
use tokio_tcp::TcpStream;
|
||||||
|
|
||||||
|
/// The `TestServer` type.
|
||||||
|
///
|
||||||
|
/// `TestServer` is very simple test server that simplify process of writing
|
||||||
|
/// integration tests cases for actix applications.
|
||||||
|
///
|
||||||
|
/// # Examples
|
||||||
|
///
|
||||||
|
/// ```rust
|
||||||
|
/// # extern crate actix_test_server;
|
||||||
|
/// # use actix_web::*;
|
||||||
|
/// #
|
||||||
|
/// # fn my_handler(req: &HttpRequest) -> HttpResponse {
|
||||||
|
/// # HttpResponse::Ok().into()
|
||||||
|
/// # }
|
||||||
|
/// #
|
||||||
|
/// # fn main() {
|
||||||
|
/// use actix_test_server::TestServer;
|
||||||
|
///
|
||||||
|
/// let mut srv = TestServer::new(|app| app.handler(my_handler));
|
||||||
|
///
|
||||||
|
/// let req = srv.get().finish().unwrap();
|
||||||
|
/// let response = srv.execute(req.send()).unwrap();
|
||||||
|
/// assert!(response.status().is_success());
|
||||||
|
/// # }
|
||||||
|
/// ```
|
||||||
|
pub struct TestServer;
|
||||||
|
|
||||||
|
/// Test server runstime
|
||||||
|
pub struct TestServerRuntime {
|
||||||
|
addr: net::SocketAddr,
|
||||||
|
host: String,
|
||||||
|
port: u16,
|
||||||
|
rt: Runtime,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl TestServer {
|
||||||
|
/// Start new test server with application factory
|
||||||
|
pub fn with<F: StreamServiceFactory>(factory: F) -> TestServerRuntime {
|
||||||
|
let (tx, rx) = mpsc::channel();
|
||||||
|
|
||||||
|
// run server in separate thread
|
||||||
|
thread::spawn(move || {
|
||||||
|
let sys = System::new("actix-test-server");
|
||||||
|
let tcp = net::TcpListener::bind("127.0.0.1:0").unwrap();
|
||||||
|
let local_addr = tcp.local_addr().unwrap();
|
||||||
|
|
||||||
|
Server::build()
|
||||||
|
.listen("test", tcp, factory)
|
||||||
|
.workers(1)
|
||||||
|
.disable_signals()
|
||||||
|
.start();
|
||||||
|
|
||||||
|
tx.send((System::current(), local_addr)).unwrap();
|
||||||
|
sys.run();
|
||||||
|
});
|
||||||
|
|
||||||
|
let (system, addr) = rx.recv().unwrap();
|
||||||
|
System::set_current(system);
|
||||||
|
|
||||||
|
let rt = Runtime::new().unwrap();
|
||||||
|
let host = format!("{}", addr.ip());
|
||||||
|
let port = addr.port();
|
||||||
|
|
||||||
|
TestServerRuntime {
|
||||||
|
addr,
|
||||||
|
rt,
|
||||||
|
host,
|
||||||
|
port,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get firat available unused local address
|
||||||
|
pub fn unused_addr() -> net::SocketAddr {
|
||||||
|
let addr: net::SocketAddr = "127.0.0.1:0".parse().unwrap();
|
||||||
|
let socket = TcpBuilder::new_v4().unwrap();
|
||||||
|
socket.bind(&addr).unwrap();
|
||||||
|
socket.reuse_address(true).unwrap();
|
||||||
|
let tcp = socket.to_tcp_listener().unwrap();
|
||||||
|
tcp.local_addr().unwrap()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl TestServerRuntime {
|
||||||
|
/// Execute future on current runtime
|
||||||
|
pub fn block_on<F, I, E>(&mut self, fut: F) -> Result<I, E>
|
||||||
|
where
|
||||||
|
F: Future<Item = I, Error = E>,
|
||||||
|
{
|
||||||
|
self.rt.block_on(fut)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Spawn future to the current runtime
|
||||||
|
pub fn spawn<F>(&mut self, fut: F)
|
||||||
|
where
|
||||||
|
F: Future<Item = (), Error = ()> + 'static,
|
||||||
|
{
|
||||||
|
self.rt.spawn(fut);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Test server host
|
||||||
|
pub fn host(&self) -> &str {
|
||||||
|
&self.host
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Test server port
|
||||||
|
pub fn port(&self) -> u16 {
|
||||||
|
self.port
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get test server address
|
||||||
|
pub fn addr(&self) -> net::SocketAddr {
|
||||||
|
self.addr
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Stop http server
|
||||||
|
fn stop(&mut self) {
|
||||||
|
System::current().stop();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Connect to server, return tokio TcpStream
|
||||||
|
pub fn connect(&self) -> std::io::Result<TcpStream> {
|
||||||
|
TcpStream::from_std(net::TcpStream::connect(self.addr)?, &Handle::default())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Drop for TestServerRuntime {
|
||||||
|
fn drop(&mut self) {
|
||||||
|
self.stop()
|
||||||
|
}
|
||||||
|
}
|
10
actix-utils/CHANGES.md
Normal file
10
actix-utils/CHANGES.md
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
# Changes
|
||||||
|
|
||||||
|
## [0.1.1] - 2018-xx-xx
|
||||||
|
|
||||||
|
* Fix framed transport error handling
|
||||||
|
|
||||||
|
|
||||||
|
## [0.1.0] - 2018-12-09
|
||||||
|
|
||||||
|
* Move utils services to separate crate
|
27
actix-utils/Cargo.toml
Normal file
27
actix-utils/Cargo.toml
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
[package]
|
||||||
|
name = "actix-utils"
|
||||||
|
version = "0.1.1"
|
||||||
|
authors = ["Nikolay Kim <fafhrd91@gmail.com>"]
|
||||||
|
description = "Actix utils - various actix net related services"
|
||||||
|
keywords = ["network", "framework", "async", "futures"]
|
||||||
|
homepage = "https://actix.rs"
|
||||||
|
repository = "https://github.com/actix/actix-net.git"
|
||||||
|
documentation = "https://docs.rs/actix-utils/"
|
||||||
|
categories = ["network-programming", "asynchronous"]
|
||||||
|
license = "MIT/Apache-2.0"
|
||||||
|
exclude = [".gitignore", ".travis.yml", ".cargo/config", "appveyor.yml"]
|
||||||
|
edition = "2018"
|
||||||
|
workspace = "../"
|
||||||
|
|
||||||
|
[lib]
|
||||||
|
name = "actix_utils"
|
||||||
|
path = "src/lib.rs"
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
actix-service = "0.1.2"
|
||||||
|
actix-codec = "0.1.0"
|
||||||
|
actix-rt = "0.1.0"
|
||||||
|
bytes = "0.4"
|
||||||
|
futures = "0.1"
|
||||||
|
tokio-timer = "0.2.8"
|
||||||
|
log = "0.4"
|
@ -2,15 +2,13 @@
|
|||||||
use std::marker::PhantomData;
|
use std::marker::PhantomData;
|
||||||
use std::mem;
|
use std::mem;
|
||||||
|
|
||||||
use actix;
|
use actix_codec::{AsyncRead, AsyncWrite, Decoder, Encoder, Framed};
|
||||||
|
use actix_rt::Arbiter;
|
||||||
use actix_service::{IntoNewService, IntoService, NewService, Service};
|
use actix_service::{IntoNewService, IntoService, NewService, Service};
|
||||||
use futures::future::{ok, FutureResult};
|
use futures::future::{ok, FutureResult};
|
||||||
use futures::unsync::mpsc;
|
use futures::unsync::mpsc;
|
||||||
use futures::{Async, AsyncSink, Future, Poll, Sink, Stream};
|
use futures::{Async, Future, Poll, Sink, Stream};
|
||||||
use tokio_codec::{Decoder, Encoder};
|
use log::debug;
|
||||||
use tokio_io::{AsyncRead, AsyncWrite};
|
|
||||||
|
|
||||||
use crate::codec::Framed;
|
|
||||||
|
|
||||||
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;
|
||||||
@ -182,16 +180,14 @@ where
|
|||||||
state: TransportState<S, U>,
|
state: TransportState<S, U>,
|
||||||
framed: Framed<T, U>,
|
framed: Framed<T, U>,
|
||||||
request: Option<Request<U>>,
|
request: Option<Request<U>>,
|
||||||
response: Option<Response<U>>,
|
|
||||||
write_rx: mpsc::Receiver<Result<Response<U>, S::Error>>,
|
write_rx: mpsc::Receiver<Result<Response<U>, S::Error>>,
|
||||||
write_tx: mpsc::Sender<Result<Response<U>, S::Error>>,
|
write_tx: mpsc::Sender<Result<Response<U>, S::Error>>,
|
||||||
flushed: bool,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
enum TransportState<S: Service<Request<U>>, U: Encoder + Decoder> {
|
enum TransportState<S: Service<Request<U>>, U: Encoder + Decoder> {
|
||||||
Processing,
|
Processing,
|
||||||
Error(FramedTransportError<S::Error, U>),
|
Error(FramedTransportError<S::Error, U>),
|
||||||
EncoderError(FramedTransportError<S::Error, U>),
|
FramedError(FramedTransportError<S::Error, U>),
|
||||||
Stopping,
|
Stopping,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -213,8 +209,6 @@ where
|
|||||||
service: service.into_service(),
|
service: service.into_service(),
|
||||||
state: TransportState::Processing,
|
state: TransportState::Processing,
|
||||||
request: None,
|
request: None,
|
||||||
response: None,
|
|
||||||
flushed: true,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -250,14 +244,14 @@ where
|
|||||||
S::Future: 'static,
|
S::Future: 'static,
|
||||||
S::Error: 'static,
|
S::Error: 'static,
|
||||||
<U as Encoder>::Item: 'static,
|
<U as Encoder>::Item: 'static,
|
||||||
<U as Encoder>::Error: 'static,
|
<U as Encoder>::Error: std::fmt::Debug + 'static,
|
||||||
{
|
{
|
||||||
fn poll_service(&mut self) -> bool {
|
fn poll_service(&mut self) -> bool {
|
||||||
match self.service.poll_ready() {
|
match self.service.poll_ready() {
|
||||||
Ok(Async::Ready(_)) => {
|
Ok(Async::Ready(_)) => {
|
||||||
if let Some(item) = self.request.take() {
|
if let Some(item) = self.request.take() {
|
||||||
let sender = self.write_tx.clone();
|
let sender = self.write_tx.clone();
|
||||||
actix::Arbiter::spawn(
|
Arbiter::spawn(
|
||||||
self.service
|
self.service
|
||||||
.call(item)
|
.call(item)
|
||||||
.then(|item| sender.send(item).map(|_| ()).map_err(|_| ())),
|
.then(|item| sender.send(item).map(|_| ()).map_err(|_| ())),
|
||||||
@ -269,7 +263,7 @@ where
|
|||||||
Ok(Async::Ready(Some(el))) => el,
|
Ok(Async::Ready(Some(el))) => el,
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
self.state =
|
self.state =
|
||||||
TransportState::Error(FramedTransportError::Decoder(err));
|
TransportState::FramedError(FramedTransportError::Decoder(err));
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
Ok(Async::NotReady) => return false,
|
Ok(Async::NotReady) => return false,
|
||||||
@ -282,7 +276,7 @@ where
|
|||||||
match self.service.poll_ready() {
|
match self.service.poll_ready() {
|
||||||
Ok(Async::Ready(_)) => {
|
Ok(Async::Ready(_)) => {
|
||||||
let sender = self.write_tx.clone();
|
let sender = self.write_tx.clone();
|
||||||
actix::Arbiter::spawn(
|
Arbiter::spawn(
|
||||||
self.service
|
self.service
|
||||||
.call(item)
|
.call(item)
|
||||||
.then(|item| sender.send(item).map(|_| ()).map_err(|_| ())),
|
.then(|item| sender.send(item).map(|_| ()).map_err(|_| ())),
|
||||||
@ -310,59 +304,42 @@ where
|
|||||||
|
|
||||||
/// write to sink
|
/// write to sink
|
||||||
fn poll_response(&mut self) -> bool {
|
fn poll_response(&mut self) -> bool {
|
||||||
let mut item = self.response.take();
|
|
||||||
loop {
|
loop {
|
||||||
item = if let Some(msg) = item {
|
while !self.framed.is_write_buf_full() {
|
||||||
self.flushed = false;
|
match self.write_rx.poll() {
|
||||||
match self.framed.start_send(msg) {
|
Ok(Async::Ready(Some(msg))) => match msg {
|
||||||
Ok(AsyncSink::Ready) => None,
|
Ok(msg) => {
|
||||||
Ok(AsyncSink::NotReady(item)) => Some(item),
|
if let Err(err) = self.framed.force_send(msg) {
|
||||||
Err(err) => {
|
self.state = TransportState::FramedError(
|
||||||
self.state =
|
FramedTransportError::Encoder(err),
|
||||||
TransportState::EncoderError(FramedTransportError::Encoder(err));
|
);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
Err(err) => {
|
||||||
None
|
self.state =
|
||||||
};
|
TransportState::Error(FramedTransportError::Service(err));
|
||||||
|
return true;
|
||||||
// flush sink
|
}
|
||||||
if !self.flushed {
|
},
|
||||||
match self.framed.poll_complete() {
|
|
||||||
Ok(Async::Ready(_)) => {
|
|
||||||
self.flushed = true;
|
|
||||||
}
|
|
||||||
Ok(Async::NotReady) => break,
|
Ok(Async::NotReady) => break,
|
||||||
Err(err) => {
|
Err(_) => panic!("Bug in actix-net code"),
|
||||||
self.state =
|
Ok(Async::Ready(None)) => panic!("Bug in actix-net code"),
|
||||||
TransportState::EncoderError(FramedTransportError::Encoder(err));
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// check channel
|
if !self.framed.is_write_buf_empty() {
|
||||||
if self.flushed {
|
match self.framed.poll_complete() {
|
||||||
if item.is_none() {
|
Ok(Async::NotReady) => break,
|
||||||
match self.write_rx.poll() {
|
Err(err) => {
|
||||||
Ok(Async::Ready(Some(msg))) => match msg {
|
debug!("Error sending data: {:?}", err);
|
||||||
Ok(msg) => item = Some(msg),
|
self.state =
|
||||||
Err(err) => {
|
TransportState::FramedError(FramedTransportError::Encoder(err));
|
||||||
self.state =
|
return true;
|
||||||
TransportState::Error(FramedTransportError::Service(err));
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
Ok(Async::NotReady) => break,
|
|
||||||
Err(_) => panic!("Bug in gw code"),
|
|
||||||
Ok(Async::Ready(None)) => panic!("Bug in gw code"),
|
|
||||||
}
|
}
|
||||||
} else {
|
Ok(Async::Ready(_)) => (),
|
||||||
continue;
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
self.response = item;
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -379,7 +356,7 @@ where
|
|||||||
S::Future: 'static,
|
S::Future: 'static,
|
||||||
S::Error: 'static,
|
S::Error: 'static,
|
||||||
<U as Encoder>::Item: 'static,
|
<U as Encoder>::Item: 'static,
|
||||||
<U as Encoder>::Error: 'static,
|
<U as Encoder>::Error: std::fmt::Debug + 'static,
|
||||||
{
|
{
|
||||||
type Item = ();
|
type Item = ();
|
||||||
type Error = FramedTransportError<S::Error, U>;
|
type Error = FramedTransportError<S::Error, U>;
|
||||||
@ -394,14 +371,16 @@ where
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
TransportState::Error(err) => {
|
TransportState::Error(err) => {
|
||||||
if self.poll_response() || self.flushed {
|
if self.framed.is_write_buf_empty()
|
||||||
|
|| (self.poll_response() || self.framed.is_write_buf_empty())
|
||||||
|
{
|
||||||
Err(err)
|
Err(err)
|
||||||
} else {
|
} else {
|
||||||
self.state = TransportState::Error(err);
|
self.state = TransportState::Error(err);
|
||||||
Ok(Async::NotReady)
|
Ok(Async::NotReady)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
TransportState::EncoderError(err) => Err(err),
|
TransportState::FramedError(err) => Err(err),
|
||||||
TransportState::Stopping => Ok(Async::Ready(())),
|
TransportState::Stopping => Ok(Async::Ready(())),
|
||||||
}
|
}
|
||||||
}
|
}
|
14
actix-utils/src/lib.rs
Normal file
14
actix-utils/src/lib.rs
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
//! Actix utils - various helper services
|
||||||
|
mod cell;
|
||||||
|
pub mod cloneable;
|
||||||
|
pub mod counter;
|
||||||
|
pub mod either;
|
||||||
|
pub mod framed;
|
||||||
|
pub mod inflight;
|
||||||
|
pub mod keepalive;
|
||||||
|
pub mod stream;
|
||||||
|
pub mod time;
|
||||||
|
pub mod timeout;
|
||||||
|
|
||||||
|
#[derive(Copy, Clone, Debug)]
|
||||||
|
pub enum Never {}
|
@ -1,22 +1,26 @@
|
|||||||
use std::marker::PhantomData;
|
use std::marker::PhantomData;
|
||||||
|
|
||||||
|
use actix_rt::spawn;
|
||||||
use actix_service::{IntoService, NewService, Service};
|
use actix_service::{IntoService, NewService, Service};
|
||||||
use futures::unsync::mpsc;
|
use futures::unsync::mpsc;
|
||||||
use futures::{future, Async, Future, Poll, Stream};
|
use futures::{future, Async, Future, Poll, Stream};
|
||||||
use tokio_current_thread::spawn;
|
|
||||||
|
|
||||||
pub struct StreamDispatcher<S: Stream, T> {
|
pub struct StreamDispatcher<S, T>
|
||||||
|
where
|
||||||
|
S: Stream,
|
||||||
|
T: Service<Result<S::Item, S::Error>>,
|
||||||
|
{
|
||||||
stream: S,
|
stream: S,
|
||||||
service: T,
|
service: T,
|
||||||
item: Option<Result<S::Item, S::Error>>,
|
item: Option<Result<S::Item, S::Error>>,
|
||||||
stop_rx: mpsc::UnboundedReceiver<()>,
|
stop_rx: mpsc::UnboundedReceiver<T::Error>,
|
||||||
stop_tx: mpsc::UnboundedSender<()>,
|
stop_tx: mpsc::UnboundedSender<T::Error>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<S, T> StreamDispatcher<S, T>
|
impl<S, T> StreamDispatcher<S, T>
|
||||||
where
|
where
|
||||||
S: Stream,
|
S: Stream,
|
||||||
T: Service<Result<S::Item, S::Error>, Response = (), Error = ()>,
|
T: Service<Result<S::Item, S::Error>, Response = ()>,
|
||||||
T::Future: 'static,
|
T::Future: 'static,
|
||||||
{
|
{
|
||||||
pub fn new<F>(stream: S, service: F) -> Self
|
pub fn new<F>(stream: S, service: F) -> Self
|
||||||
@ -37,15 +41,15 @@ where
|
|||||||
impl<S, T> Future for StreamDispatcher<S, T>
|
impl<S, T> Future for StreamDispatcher<S, T>
|
||||||
where
|
where
|
||||||
S: Stream,
|
S: Stream,
|
||||||
T: Service<Result<S::Item, S::Error>, Response = (), Error = ()>,
|
T: Service<Result<S::Item, S::Error>, Response = ()>,
|
||||||
T::Future: 'static,
|
T::Future: 'static,
|
||||||
{
|
{
|
||||||
type Item = ();
|
type Item = ();
|
||||||
type Error = ();
|
type Error = T::Error;
|
||||||
|
|
||||||
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
|
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
|
||||||
if let Ok(Async::Ready(Some(_))) = self.stop_rx.poll() {
|
if let Ok(Async::Ready(Some(e))) = self.stop_rx.poll() {
|
||||||
return Ok(Async::Ready(()));
|
return Err(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut item = self.item.take();
|
let mut item = self.item.take();
|
||||||
@ -74,7 +78,7 @@ where
|
|||||||
|
|
||||||
struct StreamDispatcherService<F: Future> {
|
struct StreamDispatcherService<F: Future> {
|
||||||
fut: F,
|
fut: F,
|
||||||
stop: mpsc::UnboundedSender<()>,
|
stop: mpsc::UnboundedSender<F::Error>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<F: Future> Future for StreamDispatcherService<F> {
|
impl<F: Future> Future for StreamDispatcherService<F> {
|
||||||
@ -85,8 +89,8 @@ impl<F: Future> Future for StreamDispatcherService<F> {
|
|||||||
match self.fut.poll() {
|
match self.fut.poll() {
|
||||||
Ok(Async::Ready(_)) => Ok(Async::Ready(())),
|
Ok(Async::Ready(_)) => Ok(Async::Ready(())),
|
||||||
Ok(Async::NotReady) => Ok(Async::NotReady),
|
Ok(Async::NotReady) => Ok(Async::NotReady),
|
||||||
Err(_) => {
|
Err(e) => {
|
||||||
let _ = self.stop.unbounded_send(());
|
let _ = self.stop.unbounded_send(e);
|
||||||
Ok(Async::Ready(()))
|
Ok(Async::Ready(()))
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -1,9 +1,9 @@
|
|||||||
use std::time::{Duration, Instant};
|
use std::time::{Duration, Instant};
|
||||||
|
|
||||||
|
use actix_rt::spawn;
|
||||||
use actix_service::{NewService, Service};
|
use actix_service::{NewService, Service};
|
||||||
use futures::future::{ok, FutureResult};
|
use futures::future::{ok, FutureResult};
|
||||||
use futures::{Async, Future, Poll};
|
use futures::{Async, Future, Poll};
|
||||||
use tokio_current_thread::spawn;
|
|
||||||
use tokio_timer::sleep;
|
use tokio_timer::sleep;
|
||||||
|
|
||||||
use super::cell::Cell;
|
use super::cell::Cell;
|
@ -1,29 +1,20 @@
|
|||||||
//! simple composite service
|
//! simple composite service
|
||||||
//! build: cargo run --example basic --features "ssl"
|
//! build: cargo run --example basic --features "ssl"
|
||||||
//! to test: curl https://127.0.0.1:8443/ -k
|
//! to test: curl https://127.0.0.1:8443/ -k
|
||||||
extern crate actix;
|
|
||||||
extern crate actix_net;
|
|
||||||
extern crate env_logger;
|
|
||||||
extern crate futures;
|
|
||||||
extern crate openssl;
|
|
||||||
extern crate tokio_io;
|
|
||||||
extern crate tokio_openssl;
|
|
||||||
extern crate tokio_tcp;
|
|
||||||
|
|
||||||
use std::sync::{
|
use std::sync::{
|
||||||
atomic::{AtomicUsize, Ordering},
|
atomic::{AtomicUsize, Ordering},
|
||||||
Arc,
|
Arc,
|
||||||
};
|
};
|
||||||
use std::{env, fmt};
|
use std::{env, fmt};
|
||||||
|
|
||||||
|
use actix_codec::{AsyncRead, AsyncWrite};
|
||||||
|
use actix_rt::System;
|
||||||
|
use actix_server::Server;
|
||||||
|
use actix_service::{IntoNewService, NewService};
|
||||||
use futures::{future, Future};
|
use futures::{future, Future};
|
||||||
use openssl::ssl::{SslAcceptor, SslFiletype, SslMethod};
|
use openssl::ssl::{SslAcceptor, SslFiletype, SslMethod};
|
||||||
use tokio_io::{AsyncRead, AsyncWrite};
|
|
||||||
use tokio_openssl::SslAcceptorExt;
|
use tokio_openssl::SslAcceptorExt;
|
||||||
|
|
||||||
use actix_net::server::Server;
|
|
||||||
use actix_net::service::{IntoNewService, NewServiceExt};
|
|
||||||
|
|
||||||
/// Simple logger service, it just prints fact of the new connections
|
/// Simple logger service, it just prints fact of the new connections
|
||||||
fn logger<T: AsyncRead + AsyncWrite + fmt::Debug>(
|
fn logger<T: AsyncRead + AsyncWrite + fmt::Debug>(
|
||||||
stream: T,
|
stream: T,
|
||||||
@ -36,7 +27,7 @@ fn main() {
|
|||||||
env::set_var("RUST_LOG", "actix_net=trace");
|
env::set_var("RUST_LOG", "actix_net=trace");
|
||||||
env_logger::init();
|
env_logger::init();
|
||||||
|
|
||||||
let sys = actix::System::new("test");
|
let sys = System::new("test");
|
||||||
|
|
||||||
// load ssl keys
|
// load ssl keys
|
||||||
let mut builder = SslAcceptor::mozilla_intermediate(SslMethod::tls()).unwrap();
|
let mut builder = SslAcceptor::mozilla_intermediate(SslMethod::tls()).unwrap();
|
||||||
@ -53,7 +44,7 @@ fn main() {
|
|||||||
// bind socket address and start workers. By default server uses number of
|
// bind socket address and start workers. By default server uses number of
|
||||||
// available logical cpu as threads count. actix net start separate
|
// available logical cpu as threads count. actix net start separate
|
||||||
// instances of service pipeline in each worker.
|
// instances of service pipeline in each worker.
|
||||||
Server::default()
|
Server::build()
|
||||||
.bind(
|
.bind(
|
||||||
// configure service pipeline
|
// configure service pipeline
|
||||||
"basic",
|
"basic",
|
||||||
|
@ -1,22 +1,14 @@
|
|||||||
extern crate actix;
|
|
||||||
extern crate actix_net;
|
|
||||||
extern crate futures;
|
|
||||||
extern crate openssl;
|
|
||||||
extern crate tokio_io;
|
|
||||||
extern crate tokio_tcp;
|
|
||||||
|
|
||||||
use std::sync::{
|
use std::sync::{
|
||||||
atomic::{AtomicUsize, Ordering},
|
atomic::{AtomicUsize, Ordering},
|
||||||
Arc,
|
Arc,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
use actix_codec::{AsyncRead, AsyncWrite};
|
||||||
|
use actix_rt::System;
|
||||||
|
use actix_server::{ssl, Server};
|
||||||
|
use actix_service::NewService;
|
||||||
use futures::{future, Future};
|
use futures::{future, Future};
|
||||||
use openssl::ssl::{SslAcceptor, SslFiletype, SslMethod};
|
use openssl::ssl::{SslAcceptor, SslFiletype, SslMethod};
|
||||||
use tokio_io::{AsyncRead, AsyncWrite};
|
|
||||||
|
|
||||||
use actix_net::server::Server;
|
|
||||||
use actix_net::service::NewServiceExt;
|
|
||||||
use actix_net::ssl;
|
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
struct ServiceState {
|
struct ServiceState {
|
||||||
@ -33,7 +25,7 @@ fn service<T: AsyncRead + AsyncWrite>(
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
let sys = actix::System::new("test");
|
let sys = System::new("test");
|
||||||
|
|
||||||
// load ssl keys
|
// load ssl keys
|
||||||
let mut builder = SslAcceptor::mozilla_intermediate(SslMethod::tls()).unwrap();
|
let mut builder = SslAcceptor::mozilla_intermediate(SslMethod::tls()).unwrap();
|
||||||
@ -48,7 +40,7 @@ fn main() {
|
|||||||
let openssl = ssl::OpensslAcceptor::new(builder.build());
|
let openssl = ssl::OpensslAcceptor::new(builder.build());
|
||||||
|
|
||||||
// server start mutiple workers, it runs supplied `Fn` in each worker.
|
// server start mutiple workers, it runs supplied `Fn` in each worker.
|
||||||
Server::default()
|
Server::build()
|
||||||
.bind("test-ssl", "0.0.0.0:8443", move || {
|
.bind("test-ssl", "0.0.0.0:8443", move || {
|
||||||
let num = num.clone();
|
let num = num.clone();
|
||||||
|
|
||||||
|
30
router/Cargo.toml
Normal file
30
router/Cargo.toml
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
[package]
|
||||||
|
name = "actix-router"
|
||||||
|
version = "0.1.0"
|
||||||
|
authors = ["Nikolay Kim <fafhrd91@gmail.com>"]
|
||||||
|
description = "Path router"
|
||||||
|
keywords = ["actix"]
|
||||||
|
homepage = "https://actix.rs"
|
||||||
|
repository = "https://github.com/actix/actix-net.git"
|
||||||
|
documentation = "https://actix.rs/api/actix-net/stable/actix_router/"
|
||||||
|
license = "MIT/Apache-2.0"
|
||||||
|
exclude = [".gitignore", ".travis.yml", ".cargo/config", "appveyor.yml"]
|
||||||
|
edition = "2018"
|
||||||
|
workspace = "../"
|
||||||
|
|
||||||
|
[lib]
|
||||||
|
name = "actix_router"
|
||||||
|
path = "src/lib.rs"
|
||||||
|
|
||||||
|
[features]
|
||||||
|
default = ["http"]
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
bytes = "0.4"
|
||||||
|
regex = "1.0"
|
||||||
|
serde = "1.0.80"
|
||||||
|
string = "0.1.3"
|
||||||
|
http = { version="0.1.14", optional=true }
|
||||||
|
|
||||||
|
[dev-dependencies]
|
||||||
|
serde_derive = "1.0"
|
709
router/src/de.rs
Normal file
709
router/src/de.rs
Normal file
@ -0,0 +1,709 @@
|
|||||||
|
use serde::de::{self, Deserializer, Error as DeError, Visitor};
|
||||||
|
use serde::forward_to_deserialize_any;
|
||||||
|
|
||||||
|
use crate::path::{Path, PathIter};
|
||||||
|
use crate::RequestPath;
|
||||||
|
|
||||||
|
macro_rules! unsupported_type {
|
||||||
|
($trait_fn:ident, $name:expr) => {
|
||||||
|
fn $trait_fn<V>(self, _: V) -> Result<V::Value, Self::Error>
|
||||||
|
where V: Visitor<'de>
|
||||||
|
{
|
||||||
|
Err(de::value::Error::custom(concat!("unsupported type: ", $name)))
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
macro_rules! parse_single_value {
|
||||||
|
($trait_fn:ident, $visit_fn:ident, $tp:tt) => {
|
||||||
|
fn $trait_fn<V>(self, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where V: Visitor<'de>
|
||||||
|
{
|
||||||
|
if self.path.len() != 1 {
|
||||||
|
Err(de::value::Error::custom(
|
||||||
|
format!("wrong number of parameters: {} expected 1",
|
||||||
|
self.path.len()).as_str()))
|
||||||
|
} else {
|
||||||
|
let v = self.path[0].parse().map_err(
|
||||||
|
|_| de::value::Error::custom(
|
||||||
|
format!("can not parse {:?} to a {}", &self.path[0], $tp)))?;
|
||||||
|
visitor.$visit_fn(v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct PathDeserializer<'de, T: RequestPath + 'de> {
|
||||||
|
path: &'de Path<T>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'de, T: RequestPath + 'de> PathDeserializer<'de, T> {
|
||||||
|
pub fn new(path: &'de Path<T>) -> Self {
|
||||||
|
PathDeserializer { path }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'de, T: RequestPath + 'de> Deserializer<'de> for PathDeserializer<'de, T> {
|
||||||
|
type Error = de::value::Error;
|
||||||
|
|
||||||
|
fn deserialize_map<V>(self, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
visitor.visit_map(ParamsDeserializer {
|
||||||
|
params: self.path.iter(),
|
||||||
|
current: None,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deserialize_struct<V>(
|
||||||
|
self,
|
||||||
|
_: &'static str,
|
||||||
|
_: &'static [&'static str],
|
||||||
|
visitor: V,
|
||||||
|
) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
self.deserialize_map(visitor)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deserialize_unit<V>(self, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
visitor.visit_unit()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deserialize_unit_struct<V>(
|
||||||
|
self,
|
||||||
|
_: &'static str,
|
||||||
|
visitor: V,
|
||||||
|
) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
self.deserialize_unit(visitor)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deserialize_newtype_struct<V>(
|
||||||
|
self,
|
||||||
|
_: &'static str,
|
||||||
|
visitor: V,
|
||||||
|
) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
visitor.visit_newtype_struct(self)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deserialize_tuple<V>(self, len: usize, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
if self.path.len() < len {
|
||||||
|
Err(de::value::Error::custom(
|
||||||
|
format!(
|
||||||
|
"wrong number of parameters: {} expected {}",
|
||||||
|
self.path.len(),
|
||||||
|
len
|
||||||
|
)
|
||||||
|
.as_str(),
|
||||||
|
))
|
||||||
|
} else {
|
||||||
|
visitor.visit_seq(ParamsSeq {
|
||||||
|
params: self.path.iter(),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deserialize_tuple_struct<V>(
|
||||||
|
self,
|
||||||
|
_: &'static str,
|
||||||
|
len: usize,
|
||||||
|
visitor: V,
|
||||||
|
) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
if self.path.len() < len {
|
||||||
|
Err(de::value::Error::custom(
|
||||||
|
format!(
|
||||||
|
"wrong number of parameters: {} expected {}",
|
||||||
|
self.path.len(),
|
||||||
|
len
|
||||||
|
)
|
||||||
|
.as_str(),
|
||||||
|
))
|
||||||
|
} else {
|
||||||
|
visitor.visit_seq(ParamsSeq {
|
||||||
|
params: self.path.iter(),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deserialize_enum<V>(
|
||||||
|
self,
|
||||||
|
_: &'static str,
|
||||||
|
_: &'static [&'static str],
|
||||||
|
visitor: V,
|
||||||
|
) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
if self.path.len() < 1 {
|
||||||
|
Err(de::value::Error::custom(
|
||||||
|
"expeceted at least one parameters",
|
||||||
|
))
|
||||||
|
} else {
|
||||||
|
visitor.visit_enum(ValueEnum {
|
||||||
|
value: &self.path[0],
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deserialize_str<V>(self, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
if self.path.len() != 1 {
|
||||||
|
Err(de::value::Error::custom(
|
||||||
|
format!("wrong number of parameters: {} expected 1", self.path.len()).as_str(),
|
||||||
|
))
|
||||||
|
} else {
|
||||||
|
visitor.visit_str(&self.path[0])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deserialize_seq<V>(self, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
visitor.visit_seq(ParamsSeq {
|
||||||
|
params: self.path.iter(),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
unsupported_type!(deserialize_any, "'any'");
|
||||||
|
unsupported_type!(deserialize_bytes, "bytes");
|
||||||
|
unsupported_type!(deserialize_option, "Option<T>");
|
||||||
|
unsupported_type!(deserialize_identifier, "identifier");
|
||||||
|
unsupported_type!(deserialize_ignored_any, "ignored_any");
|
||||||
|
|
||||||
|
parse_single_value!(deserialize_bool, visit_bool, "bool");
|
||||||
|
parse_single_value!(deserialize_i8, visit_i8, "i8");
|
||||||
|
parse_single_value!(deserialize_i16, visit_i16, "i16");
|
||||||
|
parse_single_value!(deserialize_i32, visit_i32, "i32");
|
||||||
|
parse_single_value!(deserialize_i64, visit_i64, "i64");
|
||||||
|
parse_single_value!(deserialize_u8, visit_u8, "u8");
|
||||||
|
parse_single_value!(deserialize_u16, visit_u16, "u16");
|
||||||
|
parse_single_value!(deserialize_u32, visit_u32, "u32");
|
||||||
|
parse_single_value!(deserialize_u64, visit_u64, "u64");
|
||||||
|
parse_single_value!(deserialize_f32, visit_f32, "f32");
|
||||||
|
parse_single_value!(deserialize_f64, visit_f64, "f64");
|
||||||
|
parse_single_value!(deserialize_string, visit_string, "String");
|
||||||
|
parse_single_value!(deserialize_byte_buf, visit_string, "String");
|
||||||
|
parse_single_value!(deserialize_char, visit_char, "char");
|
||||||
|
}
|
||||||
|
|
||||||
|
struct ParamsDeserializer<'de, T: RequestPath> {
|
||||||
|
params: PathIter<'de, T>,
|
||||||
|
current: Option<(&'de str, &'de str)>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'de, T: RequestPath> de::MapAccess<'de> for ParamsDeserializer<'de, T> {
|
||||||
|
type Error = de::value::Error;
|
||||||
|
|
||||||
|
fn next_key_seed<K>(&mut self, seed: K) -> Result<Option<K::Value>, Self::Error>
|
||||||
|
where
|
||||||
|
K: de::DeserializeSeed<'de>,
|
||||||
|
{
|
||||||
|
self.current = self.params.next().map(|ref item| (item.0, item.1));
|
||||||
|
match self.current {
|
||||||
|
Some((key, _)) => Ok(Some(seed.deserialize(Key { key })?)),
|
||||||
|
None => Ok(None),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn next_value_seed<V>(&mut self, seed: V) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: de::DeserializeSeed<'de>,
|
||||||
|
{
|
||||||
|
if let Some((_, value)) = self.current.take() {
|
||||||
|
seed.deserialize(Value { value })
|
||||||
|
} else {
|
||||||
|
Err(de::value::Error::custom("unexpected item"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct Key<'de> {
|
||||||
|
key: &'de str,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'de> Deserializer<'de> for Key<'de> {
|
||||||
|
type Error = de::value::Error;
|
||||||
|
|
||||||
|
fn deserialize_identifier<V>(self, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
visitor.visit_str(self.key)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deserialize_any<V>(self, _visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
Err(de::value::Error::custom("Unexpected"))
|
||||||
|
}
|
||||||
|
|
||||||
|
forward_to_deserialize_any! {
|
||||||
|
bool i8 i16 i32 i64 u8 u16 u32 u64 f32 f64 char str string bytes
|
||||||
|
byte_buf option unit unit_struct newtype_struct seq tuple
|
||||||
|
tuple_struct map struct enum ignored_any
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
macro_rules! parse_value {
|
||||||
|
($trait_fn:ident, $visit_fn:ident, $tp:tt) => {
|
||||||
|
fn $trait_fn<V>(self, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where V: Visitor<'de>
|
||||||
|
{
|
||||||
|
let v = self.value.parse().map_err(
|
||||||
|
|_| de::value::Error::custom(
|
||||||
|
format!("can not parse {:?} to a {}", self.value, $tp)))?;
|
||||||
|
visitor.$visit_fn(v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct Value<'de> {
|
||||||
|
value: &'de str,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'de> Deserializer<'de> for Value<'de> {
|
||||||
|
type Error = de::value::Error;
|
||||||
|
|
||||||
|
parse_value!(deserialize_bool, visit_bool, "bool");
|
||||||
|
parse_value!(deserialize_i8, visit_i8, "i8");
|
||||||
|
parse_value!(deserialize_i16, visit_i16, "i16");
|
||||||
|
parse_value!(deserialize_i32, visit_i32, "i16");
|
||||||
|
parse_value!(deserialize_i64, visit_i64, "i64");
|
||||||
|
parse_value!(deserialize_u8, visit_u8, "u8");
|
||||||
|
parse_value!(deserialize_u16, visit_u16, "u16");
|
||||||
|
parse_value!(deserialize_u32, visit_u32, "u32");
|
||||||
|
parse_value!(deserialize_u64, visit_u64, "u64");
|
||||||
|
parse_value!(deserialize_f32, visit_f32, "f32");
|
||||||
|
parse_value!(deserialize_f64, visit_f64, "f64");
|
||||||
|
parse_value!(deserialize_string, visit_string, "String");
|
||||||
|
parse_value!(deserialize_byte_buf, visit_string, "String");
|
||||||
|
parse_value!(deserialize_char, visit_char, "char");
|
||||||
|
|
||||||
|
fn deserialize_ignored_any<V>(self, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
visitor.visit_unit()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deserialize_unit<V>(self, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
visitor.visit_unit()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deserialize_unit_struct<V>(
|
||||||
|
self,
|
||||||
|
_: &'static str,
|
||||||
|
visitor: V,
|
||||||
|
) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
visitor.visit_unit()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deserialize_bytes<V>(self, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
visitor.visit_borrowed_bytes(self.value.as_bytes())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deserialize_str<V>(self, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
visitor.visit_borrowed_str(self.value)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deserialize_option<V>(self, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
visitor.visit_some(self)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deserialize_enum<V>(
|
||||||
|
self,
|
||||||
|
_: &'static str,
|
||||||
|
_: &'static [&'static str],
|
||||||
|
visitor: V,
|
||||||
|
) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
visitor.visit_enum(ValueEnum { value: self.value })
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deserialize_newtype_struct<V>(
|
||||||
|
self,
|
||||||
|
_: &'static str,
|
||||||
|
visitor: V,
|
||||||
|
) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
visitor.visit_newtype_struct(self)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deserialize_tuple<V>(self, _: usize, _: V) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
Err(de::value::Error::custom("unsupported type: tuple"))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deserialize_struct<V>(
|
||||||
|
self,
|
||||||
|
_: &'static str,
|
||||||
|
_: &'static [&'static str],
|
||||||
|
_: V,
|
||||||
|
) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
Err(de::value::Error::custom("unsupported type: struct"))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deserialize_tuple_struct<V>(
|
||||||
|
self,
|
||||||
|
_: &'static str,
|
||||||
|
_: usize,
|
||||||
|
_: V,
|
||||||
|
) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
Err(de::value::Error::custom("unsupported type: tuple struct"))
|
||||||
|
}
|
||||||
|
|
||||||
|
unsupported_type!(deserialize_any, "any");
|
||||||
|
unsupported_type!(deserialize_seq, "seq");
|
||||||
|
unsupported_type!(deserialize_map, "map");
|
||||||
|
unsupported_type!(deserialize_identifier, "identifier");
|
||||||
|
}
|
||||||
|
|
||||||
|
struct ParamsSeq<'de, T: RequestPath> {
|
||||||
|
params: PathIter<'de, T>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'de, T: RequestPath> de::SeqAccess<'de> for ParamsSeq<'de, T> {
|
||||||
|
type Error = de::value::Error;
|
||||||
|
|
||||||
|
fn next_element_seed<U>(&mut self, seed: U) -> Result<Option<U::Value>, Self::Error>
|
||||||
|
where
|
||||||
|
U: de::DeserializeSeed<'de>,
|
||||||
|
{
|
||||||
|
match self.params.next() {
|
||||||
|
Some(item) => Ok(Some(seed.deserialize(Value { value: item.1 })?)),
|
||||||
|
None => Ok(None),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct ValueEnum<'de> {
|
||||||
|
value: &'de str,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'de> de::EnumAccess<'de> for ValueEnum<'de> {
|
||||||
|
type Error = de::value::Error;
|
||||||
|
type Variant = UnitVariant;
|
||||||
|
|
||||||
|
fn variant_seed<V>(self, seed: V) -> Result<(V::Value, Self::Variant), Self::Error>
|
||||||
|
where
|
||||||
|
V: de::DeserializeSeed<'de>,
|
||||||
|
{
|
||||||
|
Ok((seed.deserialize(Key { key: self.value })?, UnitVariant))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct UnitVariant;
|
||||||
|
|
||||||
|
impl<'de> de::VariantAccess<'de> for UnitVariant {
|
||||||
|
type Error = de::value::Error;
|
||||||
|
|
||||||
|
fn unit_variant(self) -> Result<(), Self::Error> {
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn newtype_variant_seed<T>(self, _seed: T) -> Result<T::Value, Self::Error>
|
||||||
|
where
|
||||||
|
T: de::DeserializeSeed<'de>,
|
||||||
|
{
|
||||||
|
Err(de::value::Error::custom("not supported"))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn tuple_variant<V>(self, _len: usize, _visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
Err(de::value::Error::custom("not supported"))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn struct_variant<V>(
|
||||||
|
self,
|
||||||
|
_: &'static [&'static str],
|
||||||
|
_: V,
|
||||||
|
) -> Result<V::Value, Self::Error>
|
||||||
|
where
|
||||||
|
V: Visitor<'de>,
|
||||||
|
{
|
||||||
|
Err(de::value::Error::custom("not supported"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use serde::de;
|
||||||
|
use serde_derive::Deserialize;
|
||||||
|
|
||||||
|
use super::*;
|
||||||
|
use crate::path::Path;
|
||||||
|
use crate::router::Router;
|
||||||
|
|
||||||
|
#[derive(Deserialize)]
|
||||||
|
struct MyStruct {
|
||||||
|
key: String,
|
||||||
|
value: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Deserialize)]
|
||||||
|
struct Id {
|
||||||
|
id: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Deserialize)]
|
||||||
|
struct Test1(String, u32);
|
||||||
|
|
||||||
|
#[derive(Debug, Deserialize)]
|
||||||
|
struct Test2 {
|
||||||
|
key: String,
|
||||||
|
value: u32,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Deserialize, PartialEq)]
|
||||||
|
#[serde(rename_all = "lowercase")]
|
||||||
|
enum TestEnum {
|
||||||
|
Val1,
|
||||||
|
Val2,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Deserialize)]
|
||||||
|
struct Test3 {
|
||||||
|
val: TestEnum,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_request_extract() {
|
||||||
|
let mut router = Router::<()>::build();
|
||||||
|
router.path("/{key}/{value}/", ());
|
||||||
|
let router = router.finish();
|
||||||
|
|
||||||
|
let mut path = Path::new("/name/user1/");
|
||||||
|
assert!(router.recognize(&mut path).is_some());
|
||||||
|
|
||||||
|
let s: MyStruct = de::Deserialize::deserialize(PathDeserializer::new(&path)).unwrap();
|
||||||
|
assert_eq!(s.key, "name");
|
||||||
|
assert_eq!(s.value, "user1");
|
||||||
|
|
||||||
|
let s: (String, String) =
|
||||||
|
de::Deserialize::deserialize(PathDeserializer::new(&path)).unwrap();
|
||||||
|
assert_eq!(s.0, "name");
|
||||||
|
assert_eq!(s.1, "user1");
|
||||||
|
|
||||||
|
let mut router = Router::<()>::build();
|
||||||
|
router.path("/{key}/{value}/", ());
|
||||||
|
let router = router.finish();
|
||||||
|
|
||||||
|
let mut path = Path::new("/name/32/");
|
||||||
|
assert!(router.recognize(&mut path).is_some());
|
||||||
|
|
||||||
|
let s: Test1 = de::Deserialize::deserialize(PathDeserializer::new(&path)).unwrap();
|
||||||
|
assert_eq!(s.0, "name");
|
||||||
|
assert_eq!(s.1, 32);
|
||||||
|
|
||||||
|
let s: Test2 = de::Deserialize::deserialize(PathDeserializer::new(&path)).unwrap();
|
||||||
|
assert_eq!(s.key, "name");
|
||||||
|
assert_eq!(s.value, 32);
|
||||||
|
|
||||||
|
let s: (String, u8) =
|
||||||
|
de::Deserialize::deserialize(PathDeserializer::new(&path)).unwrap();
|
||||||
|
assert_eq!(s.0, "name");
|
||||||
|
assert_eq!(s.1, 32);
|
||||||
|
|
||||||
|
let res: Vec<String> =
|
||||||
|
de::Deserialize::deserialize(PathDeserializer::new(&path)).unwrap();
|
||||||
|
assert_eq!(res[0], "name".to_owned());
|
||||||
|
assert_eq!(res[1], "32".to_owned());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_extract_path_single() {
|
||||||
|
let mut router = Router::<()>::build();
|
||||||
|
router.path("/{value}/", ());
|
||||||
|
let router = router.finish();
|
||||||
|
|
||||||
|
let mut path = Path::new("/32/");
|
||||||
|
assert!(router.recognize(&mut path).is_some());
|
||||||
|
let i: i8 = de::Deserialize::deserialize(PathDeserializer::new(&path)).unwrap();
|
||||||
|
assert_eq!(i, 32);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_extract_enum() {
|
||||||
|
let mut router = Router::<()>::build();
|
||||||
|
router.path("/{val}/", ());
|
||||||
|
let router = router.finish();
|
||||||
|
|
||||||
|
let mut path = Path::new("/val1/");
|
||||||
|
assert!(router.recognize(&mut path).is_some());
|
||||||
|
let i: TestEnum = de::Deserialize::deserialize(PathDeserializer::new(&path)).unwrap();
|
||||||
|
assert_eq!(i, TestEnum::Val1);
|
||||||
|
|
||||||
|
let mut router = Router::<()>::build();
|
||||||
|
router.path("/{val1}/{val2}/", ());
|
||||||
|
let router = router.finish();
|
||||||
|
|
||||||
|
let mut path = Path::new("/val1/val2/");
|
||||||
|
assert!(router.recognize(&mut path).is_some());
|
||||||
|
let i: (TestEnum, TestEnum) =
|
||||||
|
de::Deserialize::deserialize(PathDeserializer::new(&path)).unwrap();
|
||||||
|
assert_eq!(i, (TestEnum::Val1, TestEnum::Val2));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_extract_enum_value() {
|
||||||
|
let mut router = Router::<()>::build();
|
||||||
|
router.path("/{val}/", ());
|
||||||
|
let router = router.finish();
|
||||||
|
|
||||||
|
let mut path = Path::new("/val1/");
|
||||||
|
assert!(router.recognize(&mut path).is_some());
|
||||||
|
let i: Test3 = de::Deserialize::deserialize(PathDeserializer::new(&path)).unwrap();
|
||||||
|
assert_eq!(i.val, TestEnum::Val1);
|
||||||
|
|
||||||
|
let mut path = Path::new("/val3/");
|
||||||
|
assert!(router.recognize(&mut path).is_some());
|
||||||
|
let i: Result<Test3, de::value::Error> =
|
||||||
|
de::Deserialize::deserialize(PathDeserializer::new(&path));
|
||||||
|
assert!(i.is_err());
|
||||||
|
assert!(format!("{:?}", i).contains("unknown variant"));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_extract_errors() {
|
||||||
|
let mut router = Router::<()>::build();
|
||||||
|
router.path("/{value}/", ());
|
||||||
|
let router = router.finish();
|
||||||
|
|
||||||
|
let mut path = Path::new("/name/");
|
||||||
|
assert!(router.recognize(&mut path).is_some());
|
||||||
|
|
||||||
|
let s: Result<Test1, de::value::Error> =
|
||||||
|
de::Deserialize::deserialize(PathDeserializer::new(&path));
|
||||||
|
assert!(s.is_err());
|
||||||
|
assert!(format!("{:?}", s).contains("wrong number of parameters"));
|
||||||
|
|
||||||
|
let s: Result<Test2, de::value::Error> =
|
||||||
|
de::Deserialize::deserialize(PathDeserializer::new(&path));
|
||||||
|
assert!(s.is_err());
|
||||||
|
assert!(format!("{:?}", s).contains("can not parse"));
|
||||||
|
|
||||||
|
let s: Result<(String, String), de::value::Error> =
|
||||||
|
de::Deserialize::deserialize(PathDeserializer::new(&path));
|
||||||
|
assert!(s.is_err());
|
||||||
|
assert!(format!("{:?}", s).contains("wrong number of parameters"));
|
||||||
|
|
||||||
|
let s: Result<u32, de::value::Error> =
|
||||||
|
de::Deserialize::deserialize(PathDeserializer::new(&path));
|
||||||
|
assert!(s.is_err());
|
||||||
|
assert!(format!("{:?}", s).contains("can not parse"));
|
||||||
|
}
|
||||||
|
|
||||||
|
// #[test]
|
||||||
|
// fn test_extract_path_decode() {
|
||||||
|
// let mut router = Router::<()>::default();
|
||||||
|
// router.register_resource(Resource::new(ResourceDef::new("/{value}/")));
|
||||||
|
|
||||||
|
// macro_rules! test_single_value {
|
||||||
|
// ($value:expr, $expected:expr) => {{
|
||||||
|
// let req = TestRequest::with_uri($value).finish();
|
||||||
|
// let info = router.recognize(&req, &(), 0);
|
||||||
|
// let req = req.with_route_info(info);
|
||||||
|
// assert_eq!(
|
||||||
|
// *Path::<String>::from_request(&req, &PathConfig::default()).unwrap(),
|
||||||
|
// $expected
|
||||||
|
// );
|
||||||
|
// }};
|
||||||
|
// }
|
||||||
|
|
||||||
|
// test_single_value!("/%25/", "%");
|
||||||
|
// test_single_value!("/%40%C2%A3%24%25%5E%26%2B%3D/", "@£$%^&+=");
|
||||||
|
// test_single_value!("/%2B/", "+");
|
||||||
|
// test_single_value!("/%252B/", "%2B");
|
||||||
|
// test_single_value!("/%2F/", "/");
|
||||||
|
// test_single_value!("/%252F/", "%2F");
|
||||||
|
// test_single_value!(
|
||||||
|
// "/http%3A%2F%2Flocalhost%3A80%2Ffoo/",
|
||||||
|
// "http://localhost:80/foo"
|
||||||
|
// );
|
||||||
|
// test_single_value!("/%2Fvar%2Flog%2Fsyslog/", "/var/log/syslog");
|
||||||
|
// test_single_value!(
|
||||||
|
// "/http%3A%2F%2Flocalhost%3A80%2Ffile%2F%252Fvar%252Flog%252Fsyslog/",
|
||||||
|
// "http://localhost:80/file/%2Fvar%2Flog%2Fsyslog"
|
||||||
|
// );
|
||||||
|
|
||||||
|
// let req = TestRequest::with_uri("/%25/7/?id=test").finish();
|
||||||
|
|
||||||
|
// let mut router = Router::<()>::default();
|
||||||
|
// router.register_resource(Resource::new(ResourceDef::new("/{key}/{value}/")));
|
||||||
|
// let info = router.recognize(&req, &(), 0);
|
||||||
|
// let req = req.with_route_info(info);
|
||||||
|
|
||||||
|
// let s = Path::<Test2>::from_request(&req, &PathConfig::default()).unwrap();
|
||||||
|
// assert_eq!(s.key, "%");
|
||||||
|
// assert_eq!(s.value, 7);
|
||||||
|
|
||||||
|
// let s = Path::<(String, String)>::from_request(&req, &PathConfig::default()).unwrap();
|
||||||
|
// assert_eq!(s.0, "%");
|
||||||
|
// assert_eq!(s.1, "7");
|
||||||
|
// }
|
||||||
|
|
||||||
|
// #[test]
|
||||||
|
// fn test_extract_path_no_decode() {
|
||||||
|
// let mut router = Router::<()>::default();
|
||||||
|
// router.register_resource(Resource::new(ResourceDef::new("/{value}/")));
|
||||||
|
|
||||||
|
// let req = TestRequest::with_uri("/%25/").finish();
|
||||||
|
// let info = router.recognize(&req, &(), 0);
|
||||||
|
// let req = req.with_route_info(info);
|
||||||
|
// assert_eq!(
|
||||||
|
// *Path::<String>::from_request(&req, &&PathConfig::default().disable_decoding())
|
||||||
|
// .unwrap(),
|
||||||
|
// "%25"
|
||||||
|
// );
|
||||||
|
// }
|
||||||
|
}
|
44
router/src/lib.rs
Normal file
44
router/src/lib.rs
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
//! Resource path matching library.
|
||||||
|
mod de;
|
||||||
|
mod path;
|
||||||
|
mod pattern;
|
||||||
|
mod router;
|
||||||
|
|
||||||
|
pub use self::de::PathDeserializer;
|
||||||
|
pub use self::path::Path;
|
||||||
|
pub use self::pattern::Pattern;
|
||||||
|
pub use self::router::{ResourceInfo, Router, RouterBuilder};
|
||||||
|
|
||||||
|
pub trait RequestPath {
|
||||||
|
fn path(&self) -> &str;
|
||||||
|
}
|
||||||
|
|
||||||
|
impl RequestPath for String {
|
||||||
|
fn path(&self) -> &str {
|
||||||
|
self.as_str()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> RequestPath for &'a str {
|
||||||
|
fn path(&self) -> &str {
|
||||||
|
self
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: AsRef<[u8]>> RequestPath for string::String<T> {
|
||||||
|
fn path(&self) -> &str {
|
||||||
|
&*self
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(feature = "http")]
|
||||||
|
mod http_support {
|
||||||
|
use super::RequestPath;
|
||||||
|
use http::Uri;
|
||||||
|
|
||||||
|
impl RequestPath for Uri {
|
||||||
|
fn path(&self) -> &str {
|
||||||
|
self.path()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
196
router/src/path.rs
Normal file
196
router/src/path.rs
Normal file
@ -0,0 +1,196 @@
|
|||||||
|
use std::ops::Index;
|
||||||
|
use std::rc::Rc;
|
||||||
|
|
||||||
|
use crate::RequestPath;
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, Copy)]
|
||||||
|
pub(crate) enum PathItem {
|
||||||
|
Static(&'static str),
|
||||||
|
Segment(u16, u16),
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Resource path match information
|
||||||
|
///
|
||||||
|
/// If resource path contains variable patterns, `Path` stores them.
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub struct Path<T> {
|
||||||
|
path: T,
|
||||||
|
pub(crate) skip: u16,
|
||||||
|
pub(crate) segments: Vec<(Rc<String>, PathItem)>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: Default> Default for Path<T> {
|
||||||
|
fn default() -> Self {
|
||||||
|
Path {
|
||||||
|
path: T::default(),
|
||||||
|
skip: 0,
|
||||||
|
segments: Vec::new(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: Clone> Clone for Path<T> {
|
||||||
|
fn clone(&self) -> Self {
|
||||||
|
Path {
|
||||||
|
path: self.path.clone(),
|
||||||
|
skip: self.skip,
|
||||||
|
segments: self.segments.clone(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: RequestPath> Path<T> {
|
||||||
|
pub fn new(path: T) -> Path<T> {
|
||||||
|
Path {
|
||||||
|
path,
|
||||||
|
skip: 0,
|
||||||
|
segments: Vec::new(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get reference to inner path instance
|
||||||
|
pub fn get_ref(&self) -> &T {
|
||||||
|
&self.path
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get mutable reference to inner path instance
|
||||||
|
pub fn get_mut(&mut self) -> &mut T {
|
||||||
|
&mut self.path
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Path
|
||||||
|
pub fn path(&self) -> &str {
|
||||||
|
let skip = self.skip as usize;
|
||||||
|
let path = self.path.path();
|
||||||
|
if skip <= path.len() {
|
||||||
|
&path[skip..]
|
||||||
|
} else {
|
||||||
|
""
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Reset inner path
|
||||||
|
pub fn set(&mut self, path: T) {
|
||||||
|
self.skip = 0;
|
||||||
|
self.path = path;
|
||||||
|
self.segments.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Skip first `n` chars in path
|
||||||
|
pub fn skip(&mut self, n: u16) {
|
||||||
|
self.skip = self.skip + n;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn add(&mut self, name: Rc<String>, value: PathItem) {
|
||||||
|
match value {
|
||||||
|
PathItem::Static(s) => self.segments.push((name, PathItem::Static(s))),
|
||||||
|
PathItem::Segment(begin, end) => self
|
||||||
|
.segments
|
||||||
|
.push((name, PathItem::Segment(self.skip + begin, self.skip + end))),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[doc(hidden)]
|
||||||
|
pub fn add_static(&mut self, name: &str, value: &'static str) {
|
||||||
|
self.segments
|
||||||
|
.push((Rc::new(name.to_string()), PathItem::Static(value)));
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Check if there are any matched patterns
|
||||||
|
pub fn is_empty(&self) -> bool {
|
||||||
|
self.segments.is_empty()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Check number of extracted parameters
|
||||||
|
pub fn len(&self) -> usize {
|
||||||
|
self.segments.len()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get matched parameter by name without type conversion
|
||||||
|
pub fn get(&self, key: &str) -> Option<&str> {
|
||||||
|
for item in self.segments.iter() {
|
||||||
|
if key == item.0.as_str() {
|
||||||
|
return match item.1 {
|
||||||
|
PathItem::Static(ref s) => Some(&s),
|
||||||
|
PathItem::Segment(s, e) => {
|
||||||
|
Some(&self.path.path()[(s as usize)..(e as usize)])
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if key == "tail" {
|
||||||
|
Some(&self.path.path()[(self.skip as usize)..])
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get unprocessed part of the path
|
||||||
|
pub fn unprocessed(&self) -> &str {
|
||||||
|
&self.path.path()[(self.skip as usize)..]
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get matched parameter by name.
|
||||||
|
///
|
||||||
|
/// If keyed parameter is not available empty string is used as default
|
||||||
|
/// value.
|
||||||
|
pub fn query(&self, key: &str) -> &str {
|
||||||
|
if let Some(s) = self.get(key) {
|
||||||
|
s
|
||||||
|
} else {
|
||||||
|
""
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Return iterator to items in parameter container
|
||||||
|
pub fn iter(&self) -> PathIter<T> {
|
||||||
|
PathIter {
|
||||||
|
idx: 0,
|
||||||
|
params: self,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub struct PathIter<'a, T> {
|
||||||
|
idx: usize,
|
||||||
|
params: &'a Path<T>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, T: RequestPath> Iterator for PathIter<'a, T> {
|
||||||
|
type Item = (&'a str, &'a str);
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn next(&mut self) -> Option<(&'a str, &'a str)> {
|
||||||
|
if self.idx < self.params.len() {
|
||||||
|
let idx = self.idx;
|
||||||
|
let res = match self.params.segments[idx].1 {
|
||||||
|
PathItem::Static(ref s) => &s,
|
||||||
|
PathItem::Segment(s, e) => &self.params.path.path()[(s as usize)..(e as usize)],
|
||||||
|
};
|
||||||
|
self.idx += 1;
|
||||||
|
return Some((&self.params.segments[idx].0, res));
|
||||||
|
}
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, T: RequestPath> Index<&'a str> for Path<T> {
|
||||||
|
type Output = str;
|
||||||
|
|
||||||
|
fn index(&self, name: &'a str) -> &str {
|
||||||
|
self.get(name)
|
||||||
|
.expect("Value for parameter is not available")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: RequestPath> Index<usize> for Path<T> {
|
||||||
|
type Output = str;
|
||||||
|
|
||||||
|
fn index(&self, idx: usize) -> &str {
|
||||||
|
match self.segments[idx].1 {
|
||||||
|
PathItem::Static(ref s) => &s,
|
||||||
|
PathItem::Segment(s, e) => &self.path.path()[(s as usize)..(e as usize)],
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
373
router/src/pattern.rs
Normal file
373
router/src/pattern.rs
Normal file
@ -0,0 +1,373 @@
|
|||||||
|
use std::cmp::min;
|
||||||
|
use std::hash::{Hash, Hasher};
|
||||||
|
use std::rc::Rc;
|
||||||
|
|
||||||
|
use regex::{escape, Regex};
|
||||||
|
|
||||||
|
use crate::path::{Path, PathItem};
|
||||||
|
use crate::RequestPath;
|
||||||
|
|
||||||
|
const MAX_DYNAMIC_SEGMENTS: usize = 16;
|
||||||
|
|
||||||
|
/// Resource type describes an entry in resources table
|
||||||
|
///
|
||||||
|
/// Resource pattern can contain only 16 dynamic segments
|
||||||
|
#[derive(Clone, Debug)]
|
||||||
|
pub struct Pattern {
|
||||||
|
tp: PatternType,
|
||||||
|
pattern: String,
|
||||||
|
elements: Vec<PatternElement>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, PartialEq)]
|
||||||
|
enum PatternElement {
|
||||||
|
Str(String),
|
||||||
|
Var(String),
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Debug)]
|
||||||
|
enum PatternType {
|
||||||
|
Static(String),
|
||||||
|
Prefix(String),
|
||||||
|
Dynamic(Regex, Vec<Rc<String>>, usize),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Pattern {
|
||||||
|
/// Parse path pattern and create new `Pattern` instance.
|
||||||
|
///
|
||||||
|
/// Panics if path pattern is wrong.
|
||||||
|
pub fn new(path: &str) -> Self {
|
||||||
|
Pattern::with_prefix(path, false)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Parse path pattern and create new `Pattern` instance.
|
||||||
|
///
|
||||||
|
/// Use `prefix` type instead of `static`.
|
||||||
|
///
|
||||||
|
/// Panics if path regex pattern is wrong.
|
||||||
|
pub fn prefix(path: &str) -> Self {
|
||||||
|
Pattern::with_prefix(path, true)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Parse path pattern and create new `Pattern` instance with custom prefix
|
||||||
|
fn with_prefix(path: &str, for_prefix: bool) -> Self {
|
||||||
|
let path = path.to_owned();
|
||||||
|
let (pattern, elements, is_dynamic, len) = Pattern::parse(&path, for_prefix);
|
||||||
|
|
||||||
|
let tp = if is_dynamic {
|
||||||
|
let re = match Regex::new(&pattern) {
|
||||||
|
Ok(re) => re,
|
||||||
|
Err(err) => panic!("Wrong path pattern: \"{}\" {}", path, err),
|
||||||
|
};
|
||||||
|
// actix creates one router per thread
|
||||||
|
let names = re
|
||||||
|
.capture_names()
|
||||||
|
.filter_map(|name| name.map(|name| Rc::new(name.to_owned())))
|
||||||
|
.collect();
|
||||||
|
PatternType::Dynamic(re, names, len)
|
||||||
|
} else if for_prefix {
|
||||||
|
PatternType::Prefix(pattern.clone())
|
||||||
|
} else {
|
||||||
|
PatternType::Static(pattern.clone())
|
||||||
|
};
|
||||||
|
|
||||||
|
Pattern {
|
||||||
|
tp,
|
||||||
|
elements,
|
||||||
|
pattern: path.to_owned(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Path pattern of the resource
|
||||||
|
pub fn pattern(&self) -> &str {
|
||||||
|
&self.pattern
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Check if path matchs this pattern?
|
||||||
|
pub fn is_match(&self, path: &str) -> bool {
|
||||||
|
match self.tp {
|
||||||
|
PatternType::Static(ref s) => s == path,
|
||||||
|
PatternType::Dynamic(ref re, _, _) => re.is_match(path),
|
||||||
|
PatternType::Prefix(ref s) => path.starts_with(s),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Is the given path and parameters a match against this pattern?
|
||||||
|
pub fn match_path<T: RequestPath>(&self, path: &mut Path<T>) -> bool {
|
||||||
|
match self.tp {
|
||||||
|
PatternType::Static(ref s) => {
|
||||||
|
if s == path.path() {
|
||||||
|
path.skip(path.len() as u16);
|
||||||
|
true
|
||||||
|
} else {
|
||||||
|
false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
PatternType::Dynamic(ref re, ref names, len) => {
|
||||||
|
let mut idx = 0;
|
||||||
|
let mut pos = 0;
|
||||||
|
let mut segments: [PathItem; MAX_DYNAMIC_SEGMENTS] =
|
||||||
|
[PathItem::Static(""); MAX_DYNAMIC_SEGMENTS];
|
||||||
|
|
||||||
|
if let Some(captures) = re.captures(path.path()) {
|
||||||
|
let mut passed = false;
|
||||||
|
|
||||||
|
for capture in captures.iter() {
|
||||||
|
if let Some(ref m) = capture {
|
||||||
|
if !passed {
|
||||||
|
passed = true;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
segments[idx] = PathItem::Segment(m.start() as u16, m.end() as u16);
|
||||||
|
idx += 1;
|
||||||
|
pos = m.end();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
for idx in 0..idx {
|
||||||
|
path.add(names[idx].clone(), segments[idx]);
|
||||||
|
}
|
||||||
|
path.skip((pos + len) as u16);
|
||||||
|
true
|
||||||
|
}
|
||||||
|
PatternType::Prefix(ref s) => {
|
||||||
|
let rpath = path.path();
|
||||||
|
let len = if s == rpath {
|
||||||
|
s.len()
|
||||||
|
} else if rpath.starts_with(s)
|
||||||
|
&& (s.ends_with('/') || rpath.split_at(s.len()).1.starts_with('/'))
|
||||||
|
{
|
||||||
|
if s.ends_with('/') {
|
||||||
|
s.len() - 1
|
||||||
|
} else {
|
||||||
|
s.len()
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
path.skip(min(rpath.len(), len) as u16);
|
||||||
|
true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// /// Build resource path.
|
||||||
|
// pub fn resource_path<U, I>(
|
||||||
|
// &self, path: &mut String, elements: &mut U,
|
||||||
|
// ) -> Result<(), UrlGenerationError>
|
||||||
|
// where
|
||||||
|
// U: Iterator<Item = I>,
|
||||||
|
// I: AsRef<str>,
|
||||||
|
// {
|
||||||
|
// match self.tp {
|
||||||
|
// PatternType::Prefix(ref p) => path.push_str(p),
|
||||||
|
// PatternType::Static(ref p) => path.push_str(p),
|
||||||
|
// PatternType::Dynamic(..) => {
|
||||||
|
// for el in &self.elements {
|
||||||
|
// match *el {
|
||||||
|
// PatternElement::Str(ref s) => path.push_str(s),
|
||||||
|
// PatternElement::Var(_) => {
|
||||||
|
// if let Some(val) = elements.next() {
|
||||||
|
// path.push_str(val.as_ref())
|
||||||
|
// } else {
|
||||||
|
// return Err(UrlGenerationError::NotEnoughElements);
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// };
|
||||||
|
// Ok(())
|
||||||
|
// }
|
||||||
|
|
||||||
|
fn parse_param(pattern: &str) -> (PatternElement, String, &str) {
|
||||||
|
const DEFAULT_PATTERN: &str = "[^/]+";
|
||||||
|
let mut params_nesting = 0usize;
|
||||||
|
let close_idx = pattern
|
||||||
|
.find(|c| match c {
|
||||||
|
'{' => {
|
||||||
|
params_nesting += 1;
|
||||||
|
false
|
||||||
|
}
|
||||||
|
'}' => {
|
||||||
|
params_nesting -= 1;
|
||||||
|
params_nesting == 0
|
||||||
|
}
|
||||||
|
_ => false,
|
||||||
|
})
|
||||||
|
.expect("malformed dynamic segment");
|
||||||
|
let (mut param, rem) = pattern.split_at(close_idx + 1);
|
||||||
|
param = ¶m[1..param.len() - 1]; // Remove outer brackets
|
||||||
|
let (name, pattern) = match param.find(':') {
|
||||||
|
Some(idx) => {
|
||||||
|
let (name, pattern) = param.split_at(idx);
|
||||||
|
(name, &pattern[1..])
|
||||||
|
}
|
||||||
|
None => (param, DEFAULT_PATTERN),
|
||||||
|
};
|
||||||
|
(
|
||||||
|
PatternElement::Var(name.to_string()),
|
||||||
|
format!(r"(?P<{}>{})", &name, &pattern),
|
||||||
|
rem,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn parse(
|
||||||
|
mut pattern: &str,
|
||||||
|
for_prefix: bool,
|
||||||
|
) -> (String, Vec<PatternElement>, bool, usize) {
|
||||||
|
if pattern.find('{').is_none() {
|
||||||
|
return (
|
||||||
|
String::from(pattern),
|
||||||
|
vec![PatternElement::Str(String::from(pattern))],
|
||||||
|
false,
|
||||||
|
pattern.chars().count(),
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut elems = Vec::new();
|
||||||
|
let mut re = String::from("^");
|
||||||
|
let mut dyn_elems = 0;
|
||||||
|
|
||||||
|
while let Some(idx) = pattern.find('{') {
|
||||||
|
let (prefix, rem) = pattern.split_at(idx);
|
||||||
|
elems.push(PatternElement::Str(String::from(prefix)));
|
||||||
|
re.push_str(&escape(prefix));
|
||||||
|
let (param_pattern, re_part, rem) = Self::parse_param(rem);
|
||||||
|
elems.push(param_pattern);
|
||||||
|
re.push_str(&re_part);
|
||||||
|
pattern = rem;
|
||||||
|
dyn_elems += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
elems.push(PatternElement::Str(String::from(pattern)));
|
||||||
|
re.push_str(&escape(pattern));
|
||||||
|
|
||||||
|
if dyn_elems > MAX_DYNAMIC_SEGMENTS {
|
||||||
|
panic!(
|
||||||
|
"Only {} dynanic segments are allowed, provided: {}",
|
||||||
|
MAX_DYNAMIC_SEGMENTS, dyn_elems
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if !for_prefix {
|
||||||
|
re.push_str("$");
|
||||||
|
}
|
||||||
|
|
||||||
|
(re, elems, true, pattern.chars().count())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl PartialEq for Pattern {
|
||||||
|
fn eq(&self, other: &Pattern) -> bool {
|
||||||
|
self.pattern == other.pattern
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Eq for Pattern {}
|
||||||
|
|
||||||
|
impl Hash for Pattern {
|
||||||
|
fn hash<H: Hasher>(&self, state: &mut H) {
|
||||||
|
self.pattern.hash(state);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use super::*;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_parse_static() {
|
||||||
|
let re = Pattern::new("/");
|
||||||
|
assert!(re.is_match("/"));
|
||||||
|
assert!(!re.is_match("/a"));
|
||||||
|
|
||||||
|
let re = Pattern::new("/name");
|
||||||
|
assert!(re.is_match("/name"));
|
||||||
|
assert!(!re.is_match("/name1"));
|
||||||
|
assert!(!re.is_match("/name/"));
|
||||||
|
assert!(!re.is_match("/name~"));
|
||||||
|
|
||||||
|
let re = Pattern::new("/name/");
|
||||||
|
assert!(re.is_match("/name/"));
|
||||||
|
assert!(!re.is_match("/name"));
|
||||||
|
assert!(!re.is_match("/name/gs"));
|
||||||
|
|
||||||
|
let re = Pattern::new("/user/profile");
|
||||||
|
assert!(re.is_match("/user/profile"));
|
||||||
|
assert!(!re.is_match("/user/profile/profile"));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_parse_param() {
|
||||||
|
let re = Pattern::new("/user/{id}");
|
||||||
|
assert!(re.is_match("/user/profile"));
|
||||||
|
assert!(re.is_match("/user/2345"));
|
||||||
|
assert!(!re.is_match("/user/2345/"));
|
||||||
|
assert!(!re.is_match("/user/2345/sdg"));
|
||||||
|
|
||||||
|
let mut path = Path::new("/user/profile");
|
||||||
|
assert!(re.match_path(&mut path));
|
||||||
|
assert_eq!(path.get("id").unwrap(), "profile");
|
||||||
|
|
||||||
|
let mut path = Path::new("/user/1245125");
|
||||||
|
assert!(re.match_path(&mut path));
|
||||||
|
assert_eq!(path.get("id").unwrap(), "1245125");
|
||||||
|
|
||||||
|
let re = Pattern::new("/v{version}/resource/{id}");
|
||||||
|
assert!(re.is_match("/v1/resource/320120"));
|
||||||
|
assert!(!re.is_match("/v/resource/1"));
|
||||||
|
assert!(!re.is_match("/resource"));
|
||||||
|
|
||||||
|
let mut path = Path::new("/v151/resource/adahg32");
|
||||||
|
assert!(re.match_path(&mut path));
|
||||||
|
assert_eq!(path.get("version").unwrap(), "151");
|
||||||
|
assert_eq!(path.get("id").unwrap(), "adahg32");
|
||||||
|
|
||||||
|
let re = Pattern::new("/{id:[[:digit:]]{6}}");
|
||||||
|
assert!(re.is_match("/012345"));
|
||||||
|
assert!(!re.is_match("/012"));
|
||||||
|
assert!(!re.is_match("/01234567"));
|
||||||
|
assert!(!re.is_match("/XXXXXX"));
|
||||||
|
|
||||||
|
let mut path = Path::new("/012345");
|
||||||
|
assert!(re.match_path(&mut path));
|
||||||
|
assert_eq!(path.get("id").unwrap(), "012345");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_resource_prefix() {
|
||||||
|
let re = Pattern::prefix("/name");
|
||||||
|
assert!(re.is_match("/name"));
|
||||||
|
assert!(re.is_match("/name/"));
|
||||||
|
assert!(re.is_match("/name/test/test"));
|
||||||
|
assert!(re.is_match("/name1"));
|
||||||
|
assert!(re.is_match("/name~"));
|
||||||
|
|
||||||
|
let re = Pattern::prefix("/name/");
|
||||||
|
assert!(re.is_match("/name/"));
|
||||||
|
assert!(re.is_match("/name/gs"));
|
||||||
|
assert!(!re.is_match("/name"));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_reousrce_prefix_dynamic() {
|
||||||
|
let re = Pattern::prefix("/{name}/");
|
||||||
|
assert!(re.is_match("/name/"));
|
||||||
|
assert!(re.is_match("/name/gs"));
|
||||||
|
assert!(!re.is_match("/name"));
|
||||||
|
|
||||||
|
let mut path = Path::new("/test2/");
|
||||||
|
assert!(re.match_path(&mut path));
|
||||||
|
assert_eq!(&path["name"], "test2");
|
||||||
|
assert_eq!(&path[0], "test2");
|
||||||
|
|
||||||
|
let mut path = Path::new("/test2/subpath1/subpath2/index.html");
|
||||||
|
assert!(re.match_path(&mut path));
|
||||||
|
assert_eq!(&path["name"], "test2");
|
||||||
|
assert_eq!(&path[0], "test2");
|
||||||
|
}
|
||||||
|
}
|
405
router/src/router.rs
Normal file
405
router/src/router.rs
Normal file
@ -0,0 +1,405 @@
|
|||||||
|
use std::collections::HashMap;
|
||||||
|
use std::rc::Rc;
|
||||||
|
|
||||||
|
use crate::path::Path;
|
||||||
|
use crate::pattern::Pattern;
|
||||||
|
use crate::RequestPath;
|
||||||
|
|
||||||
|
#[derive(Debug, Copy, Clone, PartialEq)]
|
||||||
|
pub(crate) enum ResourceId {
|
||||||
|
Default,
|
||||||
|
Normal(u16),
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Information about current resource
|
||||||
|
#[derive(Clone, Debug)]
|
||||||
|
pub struct ResourceInfo {
|
||||||
|
rmap: Rc<ResourceMap>,
|
||||||
|
resource: ResourceId,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Default, Debug)]
|
||||||
|
pub(crate) struct ResourceMap {
|
||||||
|
root: Option<Pattern>,
|
||||||
|
named: HashMap<String, Pattern>,
|
||||||
|
patterns: Vec<Pattern>,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Resource router.
|
||||||
|
pub struct Router<T> {
|
||||||
|
rmap: Rc<ResourceMap>,
|
||||||
|
named: HashMap<String, Pattern>,
|
||||||
|
resources: Vec<T>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> Router<T> {
|
||||||
|
pub fn build() -> RouterBuilder<T> {
|
||||||
|
RouterBuilder {
|
||||||
|
rmap: ResourceMap::default(),
|
||||||
|
named: HashMap::new(),
|
||||||
|
resources: Vec::new(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn recognize<U: RequestPath>(&self, path: &mut Path<U>) -> Option<(&T, ResourceInfo)> {
|
||||||
|
if !path.path().is_empty() {
|
||||||
|
for (idx, resource) in self.rmap.patterns.iter().enumerate() {
|
||||||
|
if resource.match_path(path) {
|
||||||
|
let info = ResourceInfo {
|
||||||
|
rmap: self.rmap.clone(),
|
||||||
|
resource: ResourceId::Normal(idx as u16),
|
||||||
|
};
|
||||||
|
return Some((&self.resources[idx], info));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
None
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn recognize_mut<U: RequestPath>(
|
||||||
|
&mut self,
|
||||||
|
path: &mut Path<U>,
|
||||||
|
) -> Option<(&mut T, ResourceInfo)> {
|
||||||
|
if !path.path().is_empty() {
|
||||||
|
for (idx, resource) in self.rmap.patterns.iter().enumerate() {
|
||||||
|
if resource.match_path(path) {
|
||||||
|
let info = ResourceInfo {
|
||||||
|
rmap: self.rmap.clone(),
|
||||||
|
resource: ResourceId::Normal(idx as u16),
|
||||||
|
};
|
||||||
|
return Some((&mut self.resources[idx], info));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, T> IntoIterator for &'a Router<T> {
|
||||||
|
type Item = &'a T;
|
||||||
|
type IntoIter = std::slice::Iter<'a, T>;
|
||||||
|
|
||||||
|
fn into_iter(self) -> Self::IntoIter {
|
||||||
|
self.resources.iter()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, T> IntoIterator for &'a mut Router<T> {
|
||||||
|
type Item = &'a mut T;
|
||||||
|
type IntoIter = std::slice::IterMut<'a, T>;
|
||||||
|
|
||||||
|
fn into_iter(self) -> Self::IntoIter {
|
||||||
|
self.resources.iter_mut()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ResourceMap {
|
||||||
|
fn register(&mut self, pattern: Pattern) {
|
||||||
|
self.patterns.push(pattern);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn register_named(&mut self, name: String, pattern: Pattern) {
|
||||||
|
self.patterns.push(pattern.clone());
|
||||||
|
self.named.insert(name, pattern);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn has_resource(&self, path: &str) -> bool {
|
||||||
|
unimplemented!()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct RouterBuilder<T> {
|
||||||
|
rmap: ResourceMap,
|
||||||
|
named: HashMap<String, Pattern>,
|
||||||
|
resources: Vec<T>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> RouterBuilder<T> {
|
||||||
|
pub fn path(&mut self, path: &str, resource: T) {
|
||||||
|
self.rmap.register(Pattern::new(path));
|
||||||
|
self.resources.push(resource);
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn prefix(&mut self, prefix: &str, resource: T) {
|
||||||
|
self.rmap.register(Pattern::prefix(prefix));
|
||||||
|
self.resources.push(resource);
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn finish(self) -> Router<T> {
|
||||||
|
Router {
|
||||||
|
rmap: Rc::new(self.rmap),
|
||||||
|
named: self.named,
|
||||||
|
resources: self.resources,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use crate::path::Path;
|
||||||
|
use crate::router::{ResourceId, Router};
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_recognizer_1() {
|
||||||
|
let mut router = Router::<usize>::build();
|
||||||
|
router.path("/name", 10);
|
||||||
|
router.path("/name/{val}", 11);
|
||||||
|
router.path("/name/{val}/index.html", 12);
|
||||||
|
router.path("/file/{file}.{ext}", 13);
|
||||||
|
router.path("/v{val}/{val2}/index.html", 14);
|
||||||
|
router.path("/v/{tail:.*}", 15);
|
||||||
|
router.path("/test2/{test}.html", 16);
|
||||||
|
router.path("/{test}/index.html", 17);
|
||||||
|
let mut router = router.finish();
|
||||||
|
|
||||||
|
let mut path = Path::new("/unknown");
|
||||||
|
assert!(router.recognize_mut(&mut path).is_none());
|
||||||
|
|
||||||
|
let mut path = Path::new("/name");
|
||||||
|
let (h, info) = router.recognize_mut(&mut path).unwrap();
|
||||||
|
assert_eq!(*h, 10);
|
||||||
|
assert_eq!(info.resource, ResourceId::Normal(0));
|
||||||
|
assert!(path.is_empty());
|
||||||
|
|
||||||
|
let mut path = Path::new("/name/value");
|
||||||
|
let (h, info) = router.recognize_mut(&mut path).unwrap();
|
||||||
|
assert_eq!(*h, 11);
|
||||||
|
assert_eq!(info.resource, ResourceId::Normal(1));
|
||||||
|
assert_eq!(path.get("val").unwrap(), "value");
|
||||||
|
assert_eq!(&path["val"], "value");
|
||||||
|
|
||||||
|
let mut path = Path::new("/name/value2/index.html");
|
||||||
|
let (h, info) = router.recognize_mut(&mut path).unwrap();
|
||||||
|
assert_eq!(*h, 12);
|
||||||
|
assert_eq!(info.resource, ResourceId::Normal(2));
|
||||||
|
assert_eq!(path.get("val").unwrap(), "value2");
|
||||||
|
|
||||||
|
let mut path = Path::new("/file/file.gz");
|
||||||
|
let (h, info) = router.recognize_mut(&mut path).unwrap();
|
||||||
|
assert_eq!(*h, 13);
|
||||||
|
assert_eq!(info.resource, ResourceId::Normal(3));
|
||||||
|
assert_eq!(path.get("file").unwrap(), "file");
|
||||||
|
assert_eq!(path.get("ext").unwrap(), "gz");
|
||||||
|
|
||||||
|
let mut path = Path::new("/vtest/ttt/index.html");
|
||||||
|
let (h, info) = router.recognize_mut(&mut path).unwrap();
|
||||||
|
assert_eq!(*h, 14);
|
||||||
|
assert_eq!(info.resource, ResourceId::Normal(4));
|
||||||
|
assert_eq!(path.get("val").unwrap(), "test");
|
||||||
|
assert_eq!(path.get("val2").unwrap(), "ttt");
|
||||||
|
|
||||||
|
let mut path = Path::new("/v/blah-blah/index.html");
|
||||||
|
let (h, info) = router.recognize_mut(&mut path).unwrap();
|
||||||
|
assert_eq!(*h, 15);
|
||||||
|
assert_eq!(info.resource, ResourceId::Normal(5));
|
||||||
|
assert_eq!(path.get("tail").unwrap(), "blah-blah/index.html");
|
||||||
|
|
||||||
|
let mut path = Path::new("/test2/index.html");
|
||||||
|
let (h, info) = router.recognize_mut(&mut path).unwrap();
|
||||||
|
assert_eq!(*h, 16);
|
||||||
|
assert_eq!(info.resource, ResourceId::Normal(6));
|
||||||
|
assert_eq!(path.get("test").unwrap(), "index");
|
||||||
|
|
||||||
|
let mut path = Path::new("/bbb/index.html");
|
||||||
|
let (h, info) = router.recognize_mut(&mut path).unwrap();
|
||||||
|
assert_eq!(*h, 17);
|
||||||
|
assert_eq!(info.resource, ResourceId::Normal(7));
|
||||||
|
assert_eq!(path.get("test").unwrap(), "bbb");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_recognizer_2() {
|
||||||
|
let mut router = Router::<usize>::build();
|
||||||
|
router.path("/index.json", 10);
|
||||||
|
router.path("/{source}.json", 11);
|
||||||
|
let mut router = router.finish();
|
||||||
|
|
||||||
|
let mut path = Path::new("/index.json");
|
||||||
|
let (h, _) = router.recognize_mut(&mut path).unwrap();
|
||||||
|
assert_eq!(*h, 10);
|
||||||
|
|
||||||
|
let mut path = Path::new("/test.json");
|
||||||
|
let (h, _) = router.recognize_mut(&mut path).unwrap();
|
||||||
|
assert_eq!(*h, 11);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_recognizer_with_prefix() {
|
||||||
|
let mut router = Router::<usize>::build();
|
||||||
|
router.path("/name", 10);
|
||||||
|
router.path("/name/{val}", 11);
|
||||||
|
let mut router = router.finish();
|
||||||
|
|
||||||
|
let mut path = Path::new("/name");
|
||||||
|
path.skip(5);
|
||||||
|
assert!(router.recognize_mut(&mut path).is_none());
|
||||||
|
|
||||||
|
let mut path = Path::new("/test/name");
|
||||||
|
path.skip(5);
|
||||||
|
let (h, _) = router.recognize_mut(&mut path).unwrap();
|
||||||
|
assert_eq!(*h, 10);
|
||||||
|
|
||||||
|
let mut path = Path::new("/test/name/value");
|
||||||
|
path.skip(5);
|
||||||
|
let (h, info) = router.recognize_mut(&mut path).unwrap();
|
||||||
|
assert_eq!(*h, 11);
|
||||||
|
assert_eq!(info.resource, ResourceId::Normal(1));
|
||||||
|
assert_eq!(path.get("val").unwrap(), "value");
|
||||||
|
assert_eq!(&path["val"], "value");
|
||||||
|
|
||||||
|
// same patterns
|
||||||
|
let mut router = Router::<usize>::build();
|
||||||
|
router.path("/name", 10);
|
||||||
|
router.path("/name/{val}", 11);
|
||||||
|
let mut router = router.finish();
|
||||||
|
|
||||||
|
let mut path = Path::new("/name");
|
||||||
|
path.skip(6);
|
||||||
|
assert!(router.recognize_mut(&mut path).is_none());
|
||||||
|
|
||||||
|
let mut path = Path::new("/test2/name");
|
||||||
|
path.skip(6);
|
||||||
|
let (h, _) = router.recognize_mut(&mut path).unwrap();
|
||||||
|
assert_eq!(*h, 10);
|
||||||
|
|
||||||
|
let mut path = Path::new("/test2/name-test");
|
||||||
|
path.skip(6);
|
||||||
|
assert!(router.recognize_mut(&mut path).is_none());
|
||||||
|
|
||||||
|
let mut path = Path::new("/test2/name/ttt");
|
||||||
|
path.skip(6);
|
||||||
|
let (h, _) = router.recognize_mut(&mut path).unwrap();
|
||||||
|
assert_eq!(*h, 11);
|
||||||
|
assert_eq!(&path["val"], "ttt");
|
||||||
|
}
|
||||||
|
|
||||||
|
// #[test]
|
||||||
|
// fn test_request_resource() {
|
||||||
|
// let mut router = Router::<()>::default();
|
||||||
|
// let mut resource = Resource::new(ResourcePattern::new("/index.json"));
|
||||||
|
// resource.name("r1");
|
||||||
|
// router.register_resource(resource);
|
||||||
|
// let mut resource = Resource::new(ResourcePattern::new("/test.json"));
|
||||||
|
// resource.name("r2");
|
||||||
|
// router.register_resource(resource);
|
||||||
|
|
||||||
|
// let req = TestRequest::with_uri("/index.json").finish();
|
||||||
|
// let info = router.recognize(&req, &(), 0);
|
||||||
|
// assert_eq!(info.resource, ResourceId::Normal(0));
|
||||||
|
|
||||||
|
// assert_eq!(info.name(), "r1");
|
||||||
|
|
||||||
|
// let req = TestRequest::with_uri("/test.json").finish();
|
||||||
|
// let info = router.recognize(&req, &(), 0);
|
||||||
|
// assert_eq!(info.resource, ResourceId::Normal(1));
|
||||||
|
// assert_eq!(info.name(), "r2");
|
||||||
|
// }
|
||||||
|
|
||||||
|
// #[test]
|
||||||
|
// fn test_has_resource() {
|
||||||
|
// let mut router = Router::<()>::default();
|
||||||
|
// let scope = Scope::new("/test").resource("/name", |_| "done");
|
||||||
|
// router.register_scope(scope);
|
||||||
|
|
||||||
|
// {
|
||||||
|
// let info = router.default_route_info();
|
||||||
|
// assert!(!info.has_resource("/test"));
|
||||||
|
// assert!(info.has_resource("/test/name"));
|
||||||
|
// }
|
||||||
|
|
||||||
|
// let scope = Scope::new("/test2").nested("/test10", |s| s.resource("/name", |_| "done"));
|
||||||
|
// router.register_scope(scope);
|
||||||
|
|
||||||
|
// let info = router.default_route_info();
|
||||||
|
// assert!(info.has_resource("/test2/test10/name"));
|
||||||
|
// }
|
||||||
|
|
||||||
|
// #[test]
|
||||||
|
// fn test_url_for() {
|
||||||
|
// let mut router = Router::<()>::new(ResourcePattern::prefix(""));
|
||||||
|
|
||||||
|
// let mut resource = Resource::new(ResourcePattern::new("/tttt"));
|
||||||
|
// resource.name("r0");
|
||||||
|
// router.register_resource(resource);
|
||||||
|
|
||||||
|
// let scope = Scope::new("/test").resource("/name", |r| {
|
||||||
|
// r.name("r1");
|
||||||
|
// });
|
||||||
|
// router.register_scope(scope);
|
||||||
|
|
||||||
|
// let scope =
|
||||||
|
// Scope::new("/test2").nested("/test10", |s| s.resource("/name", |r| r.name("r2")));
|
||||||
|
// router.register_scope(scope);
|
||||||
|
// router.finish();
|
||||||
|
|
||||||
|
// let req = TestRequest::with_uri("/test").request();
|
||||||
|
// {
|
||||||
|
// let info = router.default_route_info();
|
||||||
|
|
||||||
|
// let res = info
|
||||||
|
// .url_for(&req, "r0", Vec::<&'static str>::new())
|
||||||
|
// .unwrap();
|
||||||
|
// assert_eq!(res.as_str(), "http://localhost:8080/tttt");
|
||||||
|
|
||||||
|
// let res = info
|
||||||
|
// .url_for(&req, "r1", Vec::<&'static str>::new())
|
||||||
|
// .unwrap();
|
||||||
|
// assert_eq!(res.as_str(), "http://localhost:8080/test/name");
|
||||||
|
|
||||||
|
// let res = info
|
||||||
|
// .url_for(&req, "r2", Vec::<&'static str>::new())
|
||||||
|
// .unwrap();
|
||||||
|
// assert_eq!(res.as_str(), "http://localhost:8080/test2/test10/name");
|
||||||
|
// }
|
||||||
|
|
||||||
|
// let req = TestRequest::with_uri("/test/name").request();
|
||||||
|
// let info = router.recognize(&req, &(), 0);
|
||||||
|
// assert_eq!(info.resource, ResourceId::Normal(1));
|
||||||
|
|
||||||
|
// let res = info
|
||||||
|
// .url_for(&req, "r0", Vec::<&'static str>::new())
|
||||||
|
// .unwrap();
|
||||||
|
// assert_eq!(res.as_str(), "http://localhost:8080/tttt");
|
||||||
|
|
||||||
|
// let res = info
|
||||||
|
// .url_for(&req, "r1", Vec::<&'static str>::new())
|
||||||
|
// .unwrap();
|
||||||
|
// assert_eq!(res.as_str(), "http://localhost:8080/test/name");
|
||||||
|
|
||||||
|
// let res = info
|
||||||
|
// .url_for(&req, "r2", Vec::<&'static str>::new())
|
||||||
|
// .unwrap();
|
||||||
|
// assert_eq!(res.as_str(), "http://localhost:8080/test2/test10/name");
|
||||||
|
// }
|
||||||
|
|
||||||
|
// #[test]
|
||||||
|
// fn test_url_for_dynamic() {
|
||||||
|
// let mut router = Router::<()>::new(ResourcePattern::prefix(""));
|
||||||
|
|
||||||
|
// let mut resource = Resource::new(ResourcePattern::new("/{name}/test/index.{ext}"));
|
||||||
|
// resource.name("r0");
|
||||||
|
// router.register_resource(resource);
|
||||||
|
|
||||||
|
// let scope = Scope::new("/{name1}").nested("/{name2}", |s| {
|
||||||
|
// s.resource("/{name3}/test/index.{ext}", |r| r.name("r2"))
|
||||||
|
// });
|
||||||
|
// router.register_scope(scope);
|
||||||
|
// router.finish();
|
||||||
|
|
||||||
|
// let req = TestRequest::with_uri("/test").request();
|
||||||
|
// {
|
||||||
|
// let info = router.default_route_info();
|
||||||
|
|
||||||
|
// let res = info.url_for(&req, "r0", vec!["sec1", "html"]).unwrap();
|
||||||
|
// assert_eq!(res.as_str(), "http://localhost:8080/sec1/test/index.html");
|
||||||
|
|
||||||
|
// let res = info
|
||||||
|
// .url_for(&req, "r2", vec!["sec1", "sec2", "sec3", "html"])
|
||||||
|
// .unwrap();
|
||||||
|
// assert_eq!(
|
||||||
|
// res.as_str(),
|
||||||
|
// "http://localhost:8080/sec1/sec2/sec3/test/index.html"
|
||||||
|
// );
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
}
|
@ -1,313 +0,0 @@
|
|||||||
#![allow(deprecated)]
|
|
||||||
|
|
||||||
use std::fmt;
|
|
||||||
use std::io::{self, Read, Write};
|
|
||||||
|
|
||||||
use bytes::BytesMut;
|
|
||||||
use futures::{Poll, Sink, StartSend, Stream};
|
|
||||||
use tokio_codec::{Decoder, Encoder};
|
|
||||||
use tokio_io::{AsyncRead, AsyncWrite};
|
|
||||||
|
|
||||||
use super::framed_read::{framed_read2, framed_read2_with_buffer, FramedRead2};
|
|
||||||
use super::framed_write::{framed_write2, framed_write2_with_buffer, FramedWrite2};
|
|
||||||
|
|
||||||
/// A unified `Stream` and `Sink` interface to an underlying I/O object, using
|
|
||||||
/// the `Encoder` and `Decoder` traits to encode and decode frames.
|
|
||||||
///
|
|
||||||
/// You can create a `Framed` instance by using the `AsyncRead::framed` adapter.
|
|
||||||
pub struct Framed2<T, D, E> {
|
|
||||||
inner: FramedRead2<FramedWrite2<Fuse2<T, D, E>>>,
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct Fuse2<T, D, E>(pub T, pub D, pub E);
|
|
||||||
|
|
||||||
impl<T, D, E> Framed2<T, D, E>
|
|
||||||
where
|
|
||||||
T: AsyncRead + AsyncWrite,
|
|
||||||
D: Decoder,
|
|
||||||
E: Encoder,
|
|
||||||
{
|
|
||||||
/// Provides a `Stream` and `Sink` interface for reading and writing to this
|
|
||||||
/// `Io` object, using `Decode` and `Encode` to read and write the raw data.
|
|
||||||
///
|
|
||||||
/// Raw I/O objects work with byte sequences, but higher-level code usually
|
|
||||||
/// wants to batch these into meaningful chunks, called "frames". This
|
|
||||||
/// method layers framing on top of an I/O object, by using the `Codec`
|
|
||||||
/// traits to handle encoding and decoding of messages frames. Note that
|
|
||||||
/// the incoming and outgoing frame types may be distinct.
|
|
||||||
///
|
|
||||||
/// This function returns a *single* object that is both `Stream` and
|
|
||||||
/// `Sink`; grouping this into a single object is often useful for layering
|
|
||||||
/// things like gzip or TLS, which require both read and write access to the
|
|
||||||
/// underlying object.
|
|
||||||
///
|
|
||||||
/// If you want to work more directly with the streams and sink, consider
|
|
||||||
/// calling `split` on the `Framed` returned by this method, which will
|
|
||||||
/// break them into separate objects, allowing them to interact more easily.
|
|
||||||
pub fn new(inner: T, decoder: D, encoder: E) -> Framed2<T, D, E> {
|
|
||||||
Framed2 {
|
|
||||||
inner: framed_read2(framed_write2(Fuse2(inner, decoder, encoder))),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<T, D, E> Framed2<T, D, E> {
|
|
||||||
/// Provides a `Stream` and `Sink` interface for reading and writing to this
|
|
||||||
/// `Io` object, using `Decode` and `Encode` to read and write the raw data.
|
|
||||||
///
|
|
||||||
/// Raw I/O objects work with byte sequences, but higher-level code usually
|
|
||||||
/// wants to batch these into meaningful chunks, called "frames". This
|
|
||||||
/// method layers framing on top of an I/O object, by using the `Codec`
|
|
||||||
/// traits to handle encoding and decoding of messages frames. Note that
|
|
||||||
/// the incoming and outgoing frame types may be distinct.
|
|
||||||
///
|
|
||||||
/// This function returns a *single* object that is both `Stream` and
|
|
||||||
/// `Sink`; grouping this into a single object is often useful for layering
|
|
||||||
/// things like gzip or TLS, which require both read and write access to the
|
|
||||||
/// underlying object.
|
|
||||||
///
|
|
||||||
/// This objects takes a stream and a readbuffer and a writebuffer. These
|
|
||||||
/// field can be obtained from an existing `Framed` with the
|
|
||||||
/// `into_parts` method.
|
|
||||||
///
|
|
||||||
/// If you want to work more directly with the streams and sink, consider
|
|
||||||
/// calling `split` on the `Framed` returned by this method, which will
|
|
||||||
/// break them into separate objects, allowing them to interact more easily.
|
|
||||||
pub fn from_parts(parts: FramedParts2<T, D, E>) -> Framed2<T, D, E> {
|
|
||||||
Framed2 {
|
|
||||||
inner: framed_read2_with_buffer(
|
|
||||||
framed_write2_with_buffer(
|
|
||||||
Fuse2(parts.io, parts.decoder, parts.encoder),
|
|
||||||
parts.write_buf,
|
|
||||||
),
|
|
||||||
parts.read_buf,
|
|
||||||
),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Returns a reference to the underlying I/O stream wrapped by
|
|
||||||
/// `Frame`.
|
|
||||||
///
|
|
||||||
/// Note that care should be taken to not tamper with the underlying stream
|
|
||||||
/// of data coming in as it may corrupt the stream of frames otherwise
|
|
||||||
/// being worked with.
|
|
||||||
pub fn get_ref(&self) -> &T {
|
|
||||||
&self.inner.get_ref().get_ref().0
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Returns a mutable reference to the underlying I/O stream wrapped by
|
|
||||||
/// `Frame`.
|
|
||||||
///
|
|
||||||
/// Note that care should be taken to not tamper with the underlying stream
|
|
||||||
/// of data coming in as it may corrupt the stream of frames otherwise
|
|
||||||
/// being worked with.
|
|
||||||
pub fn get_mut(&mut self) -> &mut T {
|
|
||||||
&mut self.inner.get_mut().get_mut().0
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Returns a reference to the underlying decoder.
|
|
||||||
pub fn decocer(&self) -> &D {
|
|
||||||
&self.inner.get_ref().get_ref().1
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Returns a mutable reference to the underlying decoder.
|
|
||||||
pub fn decoder_mut(&mut self) -> &mut D {
|
|
||||||
&mut self.inner.get_mut().get_mut().1
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Returns a reference to the underlying encoder.
|
|
||||||
pub fn encoder(&self) -> &E {
|
|
||||||
&self.inner.get_ref().get_ref().2
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Returns a mutable reference to the underlying codec.
|
|
||||||
pub fn encoder_mut(&mut self) -> &mut E {
|
|
||||||
&mut self.inner.get_mut().get_mut().2
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Consumes the `Frame`, returning its underlying I/O stream.
|
|
||||||
///
|
|
||||||
/// Note that care should be taken to not tamper with the underlying stream
|
|
||||||
/// of data coming in as it may corrupt the stream of frames otherwise
|
|
||||||
/// being worked with.
|
|
||||||
pub fn into_inner(self) -> T {
|
|
||||||
self.inner.into_inner().into_inner().0
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Consume the `Frame`, returning `Frame` with different codec.
|
|
||||||
pub fn switch_encoder<E2>(self, encoder: E2) -> Framed2<T, D, E2> {
|
|
||||||
let (inner, read_buf) = self.inner.into_parts();
|
|
||||||
let (inner, write_buf) = inner.into_parts();
|
|
||||||
|
|
||||||
Framed2 {
|
|
||||||
inner: framed_read2_with_buffer(
|
|
||||||
framed_write2_with_buffer(Fuse2(inner.0, inner.1, encoder), write_buf),
|
|
||||||
read_buf,
|
|
||||||
),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Consumes the `Frame`, returning its underlying I/O stream, the buffer
|
|
||||||
/// with unprocessed data, and the codec.
|
|
||||||
///
|
|
||||||
/// Note that care should be taken to not tamper with the underlying stream
|
|
||||||
/// of data coming in as it may corrupt the stream of frames otherwise
|
|
||||||
/// being worked with.
|
|
||||||
pub fn into_parts(self) -> FramedParts2<T, D, E> {
|
|
||||||
let (inner, read_buf) = self.inner.into_parts();
|
|
||||||
let (inner, write_buf) = inner.into_parts();
|
|
||||||
|
|
||||||
FramedParts2 {
|
|
||||||
io: inner.0,
|
|
||||||
decoder: inner.1,
|
|
||||||
encoder: inner.2,
|
|
||||||
read_buf: read_buf,
|
|
||||||
write_buf: write_buf,
|
|
||||||
_priv: (),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<T, D, E> Stream for Framed2<T, D, E>
|
|
||||||
where
|
|
||||||
T: AsyncRead,
|
|
||||||
D: Decoder,
|
|
||||||
{
|
|
||||||
type Item = D::Item;
|
|
||||||
type Error = D::Error;
|
|
||||||
|
|
||||||
fn poll(&mut self) -> Poll<Option<Self::Item>, Self::Error> {
|
|
||||||
self.inner.poll()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<T, D, E> Sink for Framed2<T, D, E>
|
|
||||||
where
|
|
||||||
T: AsyncWrite,
|
|
||||||
E: Encoder,
|
|
||||||
E::Error: From<io::Error>,
|
|
||||||
{
|
|
||||||
type SinkItem = E::Item;
|
|
||||||
type SinkError = E::Error;
|
|
||||||
|
|
||||||
fn start_send(
|
|
||||||
&mut self,
|
|
||||||
item: Self::SinkItem,
|
|
||||||
) -> StartSend<Self::SinkItem, Self::SinkError> {
|
|
||||||
self.inner.get_mut().start_send(item)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn poll_complete(&mut self) -> Poll<(), Self::SinkError> {
|
|
||||||
self.inner.get_mut().poll_complete()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn close(&mut self) -> Poll<(), Self::SinkError> {
|
|
||||||
self.inner.get_mut().close()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<T, D, E> fmt::Debug for Framed2<T, D, E>
|
|
||||||
where
|
|
||||||
T: fmt::Debug,
|
|
||||||
D: fmt::Debug,
|
|
||||||
E: fmt::Debug,
|
|
||||||
{
|
|
||||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
||||||
f.debug_struct("Framed2")
|
|
||||||
.field("io", &self.inner.get_ref().get_ref().0)
|
|
||||||
.field("decoder", &self.inner.get_ref().get_ref().1)
|
|
||||||
.field("encoder", &self.inner.get_ref().get_ref().2)
|
|
||||||
.finish()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// ===== impl Fuse2 =====
|
|
||||||
|
|
||||||
impl<T: Read, D, E> Read for Fuse2<T, D, E> {
|
|
||||||
fn read(&mut self, dst: &mut [u8]) -> io::Result<usize> {
|
|
||||||
self.0.read(dst)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<T: AsyncRead, D, E> AsyncRead for Fuse2<T, D, E> {
|
|
||||||
unsafe fn prepare_uninitialized_buffer(&self, buf: &mut [u8]) -> bool {
|
|
||||||
self.0.prepare_uninitialized_buffer(buf)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<T: Write, D, E> Write for Fuse2<T, D, E> {
|
|
||||||
fn write(&mut self, src: &[u8]) -> io::Result<usize> {
|
|
||||||
self.0.write(src)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn flush(&mut self) -> io::Result<()> {
|
|
||||||
self.0.flush()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<T: AsyncWrite, D, E> AsyncWrite for Fuse2<T, D, E> {
|
|
||||||
fn shutdown(&mut self) -> Poll<(), io::Error> {
|
|
||||||
self.0.shutdown()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<T, D: Decoder, E> Decoder for Fuse2<T, D, E> {
|
|
||||||
type Item = D::Item;
|
|
||||||
type Error = D::Error;
|
|
||||||
|
|
||||||
fn decode(&mut self, buffer: &mut BytesMut) -> Result<Option<Self::Item>, Self::Error> {
|
|
||||||
self.1.decode(buffer)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn decode_eof(&mut self, buffer: &mut BytesMut) -> Result<Option<Self::Item>, Self::Error> {
|
|
||||||
self.1.decode_eof(buffer)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<T, D, E: Encoder> Encoder for Fuse2<T, D, E> {
|
|
||||||
type Item = E::Item;
|
|
||||||
type Error = E::Error;
|
|
||||||
|
|
||||||
fn encode(&mut self, item: Self::Item, dst: &mut BytesMut) -> Result<(), Self::Error> {
|
|
||||||
self.2.encode(item, dst)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// `FramedParts` contains an export of the data of a Framed transport.
|
|
||||||
/// It can be used to construct a new `Framed` with a different codec.
|
|
||||||
/// It contains all current buffers and the inner transport.
|
|
||||||
#[derive(Debug)]
|
|
||||||
pub struct FramedParts2<T, D, E> {
|
|
||||||
/// The inner transport used to read bytes to and write bytes to
|
|
||||||
pub io: T,
|
|
||||||
|
|
||||||
/// The decoder
|
|
||||||
pub decoder: D,
|
|
||||||
|
|
||||||
/// The encoder
|
|
||||||
pub encoder: E,
|
|
||||||
|
|
||||||
/// The buffer with read but unprocessed data.
|
|
||||||
pub read_buf: BytesMut,
|
|
||||||
|
|
||||||
/// A buffer with unprocessed data which are not written yet.
|
|
||||||
pub write_buf: BytesMut,
|
|
||||||
|
|
||||||
/// This private field allows us to add additional fields in the future in a
|
|
||||||
/// backwards compatible way.
|
|
||||||
_priv: (),
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<T, D, E> FramedParts2<T, D, E> {
|
|
||||||
/// Create a new, default, `FramedParts`
|
|
||||||
pub fn new(io: T, decoder: D, encoder: E) -> FramedParts2<T, D, E> {
|
|
||||||
FramedParts2 {
|
|
||||||
io,
|
|
||||||
decoder,
|
|
||||||
encoder,
|
|
||||||
read_buf: BytesMut::new(),
|
|
||||||
write_buf: BytesMut::new(),
|
|
||||||
_priv: (),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
32
src/lib.rs
32
src/lib.rs
@ -1,32 +0,0 @@
|
|||||||
//! Actix net - framework for the compisible network services for Rust.
|
|
||||||
//!
|
|
||||||
//! ## Package feature
|
|
||||||
//!
|
|
||||||
//! * `tls` - enables ssl support via `native-tls` crate
|
|
||||||
//! * `ssl` - enables ssl support via `openssl` crate
|
|
||||||
//! * `rust-tls` - enables ssl support via `rustls` crate
|
|
||||||
// #![warn(missing_docs)]
|
|
||||||
|
|
||||||
#![allow(
|
|
||||||
clippy::declare_interior_mutable_const,
|
|
||||||
clippy::borrow_interior_mutable_const
|
|
||||||
)]
|
|
||||||
|
|
||||||
mod cell;
|
|
||||||
pub mod cloneable;
|
|
||||||
pub mod codec;
|
|
||||||
pub mod connector;
|
|
||||||
pub mod counter;
|
|
||||||
pub mod either;
|
|
||||||
pub mod framed;
|
|
||||||
pub mod inflight;
|
|
||||||
pub mod keepalive;
|
|
||||||
pub mod resolver;
|
|
||||||
pub mod server;
|
|
||||||
pub mod ssl;
|
|
||||||
pub mod stream;
|
|
||||||
pub mod time;
|
|
||||||
pub mod timeout;
|
|
||||||
|
|
||||||
#[derive(Copy, Clone, Debug)]
|
|
||||||
pub enum Never {}
|
|
@ -1,48 +0,0 @@
|
|||||||
//! General purpose networking server
|
|
||||||
|
|
||||||
use actix::Message;
|
|
||||||
|
|
||||||
mod accept;
|
|
||||||
mod config;
|
|
||||||
mod server;
|
|
||||||
mod services;
|
|
||||||
mod worker;
|
|
||||||
|
|
||||||
pub use self::config::{ServiceConfig, ServiceRuntime};
|
|
||||||
pub use self::server::Server;
|
|
||||||
pub use self::services::{ServerMessage, ServiceFactory, StreamServiceFactory};
|
|
||||||
|
|
||||||
/// Pause accepting incoming connections
|
|
||||||
///
|
|
||||||
/// If socket contains some pending connection, they might be dropped.
|
|
||||||
/// All opened connection remains active.
|
|
||||||
#[derive(Message)]
|
|
||||||
pub struct PauseServer;
|
|
||||||
|
|
||||||
/// Resume accepting incoming connections
|
|
||||||
#[derive(Message)]
|
|
||||||
pub struct ResumeServer;
|
|
||||||
|
|
||||||
/// Stop incoming connection processing, stop all workers and exit.
|
|
||||||
///
|
|
||||||
/// If server starts with `spawn()` method, then spawned thread get terminated.
|
|
||||||
pub struct StopServer {
|
|
||||||
/// Whether to try and shut down gracefully
|
|
||||||
pub graceful: bool,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Message for StopServer {
|
|
||||||
type Result = Result<(), ()>;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Socket id token
|
|
||||||
#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)]
|
|
||||||
pub(crate) struct Token(usize);
|
|
||||||
|
|
||||||
impl Token {
|
|
||||||
pub(crate) fn next(&mut self) -> Token {
|
|
||||||
let token = Token(self.0 + 1);
|
|
||||||
self.0 += 1;
|
|
||||||
token
|
|
||||||
}
|
|
||||||
}
|
|
Reference in New Issue
Block a user