2022-07-11 02:44:46 +02:00
|
|
|
//! Simple echo websocket server.
|
|
|
|
//!
|
|
|
|
//! Open `http://localhost:8080/` in browser to test.
|
|
|
|
|
|
|
|
use actix_files::NamedFile;
|
|
|
|
use actix_web::{
|
|
|
|
middleware, rt, web, App, Error, HttpRequest, HttpResponse, HttpServer, Responder,
|
|
|
|
};
|
2023-08-30 16:36:21 +02:00
|
|
|
use tokio::sync::broadcast;
|
2022-07-11 02:44:46 +02:00
|
|
|
|
|
|
|
mod handler;
|
|
|
|
|
|
|
|
async fn index() -> impl Responder {
|
|
|
|
NamedFile::open_async("./static/index.html").await.unwrap()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Handshake and start WebSocket handler with heartbeats.
|
|
|
|
async fn echo_heartbeat_ws(req: HttpRequest, stream: web::Payload) -> Result<HttpResponse, Error> {
|
|
|
|
let (res, session, msg_stream) = actix_ws::handle(&req, stream)?;
|
|
|
|
|
|
|
|
// spawn websocket handler (and don't await it) so that the response is returned immediately
|
|
|
|
rt::spawn(handler::echo_heartbeat_ws(session, msg_stream));
|
|
|
|
|
|
|
|
Ok(res)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Handshake and start basic WebSocket handler.
|
|
|
|
///
|
|
|
|
/// This example is just for demonstration of simplicity. In reality, you likely want to include
|
|
|
|
/// some handling of heartbeats for connection health tracking to free up server resources when
|
|
|
|
/// connections die or network issues arise.
|
|
|
|
async fn echo_ws(req: HttpRequest, stream: web::Payload) -> Result<HttpResponse, Error> {
|
|
|
|
let (res, session, msg_stream) = actix_ws::handle(&req, stream)?;
|
|
|
|
|
|
|
|
// spawn websocket handler (and don't await it) so that the response is returned immediately
|
|
|
|
rt::spawn(handler::echo_ws(session, msg_stream));
|
|
|
|
|
|
|
|
Ok(res)
|
|
|
|
}
|
|
|
|
|
2023-08-30 16:40:53 +02:00
|
|
|
/// Send message to clients connected to broadcast WebSocket.
|
|
|
|
async fn send_to_broadcast_ws(
|
2023-08-30 16:36:21 +02:00
|
|
|
body: web::Bytes,
|
|
|
|
tx: web::Data<broadcast::Sender<web::Bytes>>,
|
|
|
|
) -> Result<impl Responder, Error> {
|
|
|
|
tx.send(body)
|
|
|
|
.map_err(actix_web::error::ErrorInternalServerError)?;
|
|
|
|
|
|
|
|
Ok(HttpResponse::NoContent())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Handshake and start broadcast WebSocket handler with heartbeats.
|
|
|
|
async fn broadcast_ws(
|
|
|
|
req: HttpRequest,
|
|
|
|
stream: web::Payload,
|
|
|
|
tx: web::Data<broadcast::Sender<web::Bytes>>,
|
|
|
|
) -> Result<HttpResponse, Error> {
|
|
|
|
let (res, session, msg_stream) = actix_ws::handle(&req, stream)?;
|
|
|
|
|
|
|
|
// spawn websocket handler (and don't await it) so that the response is returned immediately
|
|
|
|
rt::spawn(handler::broadcast_ws(session, msg_stream, tx.subscribe()));
|
|
|
|
|
|
|
|
Ok(res)
|
|
|
|
}
|
|
|
|
|
2022-07-12 02:08:15 +02:00
|
|
|
// note that the `actix` based WebSocket handling would NOT work under `tokio::main`
|
|
|
|
#[tokio::main(flavor = "current_thread")]
|
2022-07-11 02:44:46 +02:00
|
|
|
async fn main() -> std::io::Result<()> {
|
|
|
|
env_logger::init_from_env(env_logger::Env::new().default_filter_or("info"));
|
|
|
|
|
|
|
|
log::info!("starting HTTP server at http://localhost:8080");
|
|
|
|
|
2023-08-30 16:36:21 +02:00
|
|
|
let (tx, _) = broadcast::channel::<web::Bytes>(128);
|
|
|
|
|
|
|
|
HttpServer::new(move || {
|
2022-07-11 02:44:46 +02:00
|
|
|
App::new()
|
|
|
|
// WebSocket UI HTML file
|
|
|
|
.service(web::resource("/").to(index))
|
|
|
|
// websocket routes
|
|
|
|
.service(web::resource("/ws").route(web::get().to(echo_heartbeat_ws)))
|
|
|
|
.service(web::resource("/ws-basic").route(web::get().to(echo_ws)))
|
2023-08-30 16:36:21 +02:00
|
|
|
.app_data(web::Data::new(tx.clone()))
|
|
|
|
.service(web::resource("/ws-broadcast").route(web::get().to(broadcast_ws)))
|
2023-08-30 16:40:53 +02:00
|
|
|
.service(web::resource("/send").route(web::post().to(send_to_broadcast_ws)))
|
2022-07-11 02:44:46 +02:00
|
|
|
// enable logger
|
|
|
|
.wrap(middleware::Logger::default())
|
|
|
|
})
|
|
|
|
.workers(2)
|
|
|
|
.bind(("127.0.0.1", 8080))?
|
|
|
|
.run()
|
|
|
|
.await
|
|
|
|
}
|