use std::sync::Arc;
use axum::extract::ws::{Message, WebSocket, WebSocketUpgrade};
use axum::response::IntoResponse;
use futures_util::{SinkExt, StreamExt};
use once_cell::sync::Lazy;
use tokio::sync::broadcast;
static DEV_TX: Lazy<broadcast::Sender<String>> = Lazy::new(|| {
let (tx, _) = broadcast::channel(64);
tx
});
pub fn dev_mode_enabled() -> bool {
matches!(
std::env::var("RESUMA_DEV").as_deref(),
Ok("1") | Ok("true") | Ok("TRUE")
)
}
pub fn broadcast_dev_event(event: impl Into<String>) {
if dev_mode_enabled() {
let _ = DEV_TX.send(event.into());
}
}
pub async fn dev_ws_handler(ws: WebSocketUpgrade) -> impl IntoResponse {
ws.on_upgrade(handle_socket)
}
async fn handle_socket(socket: WebSocket) {
let (mut sender, mut receiver) = socket.split();
let mut rx = DEV_TX.subscribe();
let mut send_task = tokio::spawn(async move {
while let Ok(msg) = rx.recv().await {
if sender.send(Message::Text(msg)).await.is_err() {
break;
}
}
});
let mut recv_task = tokio::spawn(async move {
while let Some(Ok(msg)) = receiver.next().await {
if matches!(msg, Message::Close(_)) {
break;
}
}
});
tokio::select! {
_ = &mut send_task => recv_task.abort(),
_ = &mut recv_task => send_task.abort(),
};
}
#[doc(hidden)]
pub fn dev_broadcast_sender() -> Arc<broadcast::Sender<String>> {
Arc::new(DEV_TX.clone())
}