2018-08-21 04:34:47 +00:00
|
|
|
//! simple composite service
|
2018-08-21 05:21:23 +00:00
|
|
|
//! build: cargo run --example basic --features "ssl"
|
2018-08-21 04:34:47 +00:00
|
|
|
//! to test: curl https://127.0.0.1:8443/ -k
|
|
|
|
use std::sync::{
|
|
|
|
atomic::{AtomicUsize, Ordering},
|
|
|
|
Arc,
|
|
|
|
};
|
2018-09-18 03:19:48 +00:00
|
|
|
use std::{env, fmt};
|
2018-08-21 04:34:47 +00:00
|
|
|
|
2018-12-11 16:20:19 +00:00
|
|
|
use actix_codec::{AsyncRead, AsyncWrite};
|
|
|
|
use actix_rt::System;
|
|
|
|
use actix_server::Server;
|
|
|
|
use actix_service::{IntoNewService, NewService};
|
2018-08-21 04:34:47 +00:00
|
|
|
use futures::{future, Future};
|
|
|
|
use openssl::ssl::{SslAcceptor, SslFiletype, SslMethod};
|
|
|
|
use tokio_openssl::SslAcceptorExt;
|
|
|
|
|
|
|
|
/// Simple logger service, it just prints fact of the new connections
|
|
|
|
fn logger<T: AsyncRead + AsyncWrite + fmt::Debug>(
|
|
|
|
stream: T,
|
2018-09-08 21:50:16 +00:00
|
|
|
) -> impl Future<Item = T, Error = ()> {
|
2018-08-21 04:34:47 +00:00
|
|
|
println!("New connection: {:?}", stream);
|
|
|
|
future::ok(stream)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn main() {
|
2018-09-18 03:19:48 +00:00
|
|
|
env::set_var("RUST_LOG", "actix_net=trace");
|
|
|
|
env_logger::init();
|
|
|
|
|
2018-12-10 05:51:35 +00:00
|
|
|
let sys = System::new("test");
|
2018-08-21 04:34:47 +00:00
|
|
|
|
|
|
|
// load ssl keys
|
|
|
|
let mut builder = SslAcceptor::mozilla_intermediate(SslMethod::tls()).unwrap();
|
|
|
|
builder
|
|
|
|
.set_private_key_file("./examples/key.pem", SslFiletype::PEM)
|
|
|
|
.unwrap();
|
|
|
|
builder
|
|
|
|
.set_certificate_chain_file("./examples/cert.pem")
|
|
|
|
.unwrap();
|
|
|
|
let acceptor = builder.build();
|
|
|
|
|
|
|
|
let num = Arc::new(AtomicUsize::new(0));
|
|
|
|
|
2018-08-22 04:11:16 +00:00
|
|
|
// bind socket address and start workers. By default server uses number of
|
|
|
|
// available logical cpu as threads count. actix net start separate
|
|
|
|
// instances of service pipeline in each worker.
|
2018-12-10 05:51:35 +00:00
|
|
|
Server::build()
|
2018-08-28 23:24:36 +00:00
|
|
|
.bind(
|
|
|
|
// configure service pipeline
|
2018-09-18 03:19:48 +00:00
|
|
|
"basic",
|
2018-08-28 23:24:36 +00:00
|
|
|
"0.0.0.0:8443",
|
|
|
|
move || {
|
|
|
|
let num = num.clone();
|
|
|
|
let acceptor = acceptor.clone();
|
2018-08-23 20:39:13 +00:00
|
|
|
|
2018-08-28 23:24:36 +00:00
|
|
|
// service for converting incoming TcpStream to a SslStream<TcpStream>
|
|
|
|
(move |stream| {
|
2018-09-11 18:28:13 +00:00
|
|
|
SslAcceptorExt::accept_async(&acceptor, stream)
|
|
|
|
.map_err(|e| println!("Openssl error: {}", e))
|
|
|
|
})
|
|
|
|
// convert closure to a `NewService`
|
|
|
|
.into_new_service()
|
|
|
|
// .and_then() combinator uses other service to convert incoming `Request` to a
|
|
|
|
// `Response` and then uses that response as an input for next
|
|
|
|
// service. in this case, on success we use `logger` service
|
|
|
|
.and_then(logger)
|
2018-09-19 15:04:31 +00:00
|
|
|
// Next service counts number of connections
|
|
|
|
.and_then(move |_| {
|
|
|
|
let num = num.fetch_add(1, Ordering::Relaxed);
|
|
|
|
println!("got ssl connection {:?}", num);
|
|
|
|
future::ok(())
|
|
|
|
})
|
2018-08-28 23:24:36 +00:00
|
|
|
},
|
2018-12-06 22:04:42 +00:00
|
|
|
)
|
|
|
|
.unwrap()
|
2018-08-28 23:24:36 +00:00
|
|
|
.start();
|
2018-08-21 04:34:47 +00:00
|
|
|
|
|
|
|
sys.run();
|
|
|
|
}
|