2019-03-16 20:23:09 -07:00
|
|
|
//! Application may have multiple data objects that are shared across
|
2020-01-12 17:45:59 +00:00
|
|
|
//! all handlers within same Application.
|
2018-05-02 06:20:43 -07:00
|
|
|
//!
|
2022-01-29 15:23:32 +01:00
|
|
|
//! For global shared state, we wrap our state in a [`actix_web::web::Data`] and move it into
|
2020-01-12 17:45:59 +00:00
|
|
|
//! the factory closure. The closure is called once-per-thread, and we clone our state
|
2022-01-29 15:23:32 +01:00
|
|
|
//! and attach to each instance of the [`App`] with `.app_data(state.clone())`.
|
2020-01-12 17:45:59 +00:00
|
|
|
//!
|
|
|
|
//! For thread-local state, we construct our state within the factory closure and attach to
|
2022-01-29 15:23:32 +01:00
|
|
|
//! the app with `.app_data(Data::new(state))`.
|
2020-01-12 17:45:59 +00:00
|
|
|
//!
|
|
|
|
//! We retrieve our app state within our handlers with a `state: Data<...>` argument.
|
|
|
|
//!
|
2022-02-06 08:13:24 +00:00
|
|
|
//! By default, Actix Web runs one [`App`] per logical cpu core.
|
2022-01-29 15:23:32 +01:00
|
|
|
//! When running on `<N>` cores, we see that the example will increment `counter_mutex` (global state via
|
|
|
|
//! Mutex) and `counter_atomic` (global state via Atomic variable) each time the endpoint is called,
|
|
|
|
//! but only appear to increment `counter_cell` every Nth time on average (thread-local state). This
|
2020-01-28 01:41:45 +01:00
|
|
|
//! is because the workload is being shared equally among cores.
|
2018-05-02 06:20:43 -07:00
|
|
|
//!
|
2019-11-15 05:16:48 +09:00
|
|
|
//! Check [user guide](https://actix.rs/docs/application/#state) for more info.
|
2018-04-13 09:18:42 +08:00
|
|
|
|
2022-08-28 18:39:28 +01:00
|
|
|
use std::{
|
|
|
|
cell::Cell,
|
|
|
|
io,
|
|
|
|
sync::{
|
|
|
|
atomic::{AtomicUsize, Ordering},
|
|
|
|
Mutex,
|
|
|
|
},
|
|
|
|
};
|
2018-04-13 09:18:42 +08:00
|
|
|
|
2022-01-29 15:23:32 +01:00
|
|
|
use actix_web::{
|
|
|
|
middleware,
|
|
|
|
web::{self, Data},
|
|
|
|
App, HttpRequest, HttpResponse, HttpServer,
|
|
|
|
};
|
2018-04-13 09:18:42 +08:00
|
|
|
|
|
|
|
/// simple handle
|
2020-01-12 17:45:59 +00:00
|
|
|
async fn index(
|
2022-01-29 15:23:32 +01:00
|
|
|
counter_mutex: Data<Mutex<usize>>,
|
|
|
|
counter_cell: Data<Cell<u32>>,
|
|
|
|
counter_atomic: Data<AtomicUsize>,
|
2020-01-12 17:45:59 +00:00
|
|
|
req: HttpRequest,
|
|
|
|
) -> HttpResponse {
|
2022-06-07 22:53:38 -04:00
|
|
|
println!("{req:?}");
|
2018-04-13 09:18:42 +08:00
|
|
|
|
2020-01-12 17:45:59 +00:00
|
|
|
// Increment the counters
|
2022-01-29 15:23:32 +01:00
|
|
|
*counter_mutex.lock().unwrap() += 1;
|
|
|
|
counter_cell.set(counter_cell.get() + 1);
|
|
|
|
counter_atomic.fetch_add(1, Ordering::SeqCst);
|
2020-01-12 17:45:59 +00:00
|
|
|
|
|
|
|
let body = format!(
|
2020-01-28 01:41:45 +01:00
|
|
|
"global mutex counter: {}, local counter: {}, global atomic counter: {}",
|
2022-01-29 15:23:32 +01:00
|
|
|
*counter_mutex.lock().unwrap(),
|
|
|
|
counter_cell.get(),
|
|
|
|
counter_atomic.load(Ordering::SeqCst),
|
2020-01-12 17:45:59 +00:00
|
|
|
);
|
|
|
|
HttpResponse::Ok().body(body)
|
2018-04-13 09:18:42 +08:00
|
|
|
}
|
|
|
|
|
2020-09-12 16:49:45 +01:00
|
|
|
#[actix_web::main]
|
2019-12-07 23:59:24 +06:00
|
|
|
async fn main() -> io::Result<()> {
|
2023-03-14 03:11:49 +00:00
|
|
|
env_logger::init_from_env(env_logger::Env::new().default_filter_or("info"));
|
2018-04-13 09:18:42 +08:00
|
|
|
|
2020-01-12 17:45:59 +00:00
|
|
|
// Create some global state prior to building the server
|
2020-04-03 16:14:30 +09:00
|
|
|
#[allow(clippy::mutex_atomic)] // it's intentional.
|
2022-01-29 15:23:32 +01:00
|
|
|
let counter_mutex = Data::new(Mutex::new(0usize));
|
|
|
|
let counter_atomic = Data::new(AtomicUsize::new(0usize));
|
2019-03-07 14:50:29 -08:00
|
|
|
|
2023-03-14 03:11:49 +00:00
|
|
|
log::info!("starting HTTP server at http://localhost:8080");
|
|
|
|
|
2020-01-12 17:45:59 +00:00
|
|
|
// move is necessary to give closure below ownership of counter1
|
2019-03-07 14:50:29 -08:00
|
|
|
HttpServer::new(move || {
|
2020-01-12 17:45:59 +00:00
|
|
|
// Create some thread-local state
|
2022-01-29 15:23:32 +01:00
|
|
|
let counter_cell = Cell::new(0u32);
|
2020-01-12 17:45:59 +00:00
|
|
|
|
2019-03-07 14:50:29 -08:00
|
|
|
App::new()
|
2022-01-29 15:23:32 +01:00
|
|
|
.app_data(counter_mutex.clone()) // add shared state
|
|
|
|
.app_data(counter_atomic.clone()) // add shared state
|
|
|
|
.app_data(Data::new(counter_cell)) // add thread-local state
|
2018-04-13 09:18:42 +08:00
|
|
|
// enable logger
|
2019-03-26 04:29:00 +01:00
|
|
|
.wrap(middleware::Logger::default())
|
2020-01-12 17:45:59 +00:00
|
|
|
// register simple handler
|
2019-03-07 14:50:29 -08:00
|
|
|
.service(web::resource("/").to(index))
|
|
|
|
})
|
2022-02-17 20:22:36 +00:00
|
|
|
.bind(("127.0.0.1", 8080))?
|
2019-12-25 20:48:33 +04:00
|
|
|
.run()
|
2019-12-07 23:59:24 +06:00
|
|
|
.await
|
2018-04-13 09:18:42 +08:00
|
|
|
}
|