heartbeat

This commit is contained in:
Asger Juul Brunshøj 2023-06-14 12:18:41 +02:00
parent 18cd7a4b5d
commit 7e49d9f700
9 changed files with 132 additions and 48 deletions

1
Cargo.lock generated
View File

@ -134,6 +134,7 @@ dependencies = [
"axum",
"clap",
"common",
"futures-util",
"serde",
"serde_json",
"thiserror",

View File

@ -18,3 +18,4 @@ uuid.workspace = true
tower-http = { version = "0.4.0", features = ["fs", "trace", "cors"] }
tokio-stream = { version = "0.1.14", features = ["sync"] }
clap = { version = "4.3.3", features = ["derive"] }
futures-util = "0.3.28"

View File

@ -20,13 +20,15 @@ use common::DeleteMilestone;
use common::Milestone;
use common::RestResponse;
use common::ToggleAchievement;
use futures_util::SinkExt;
use futures_util::StreamExt;
use serde::Deserialize;
use serde::Serialize;
use std::net::SocketAddr;
use std::sync::Arc;
use std::time::Duration;
use tokio::fs;
use tokio_stream::StreamExt;
use tokio::select;
use tower::ServiceBuilder;
use tower_http::cors::CorsLayer;
use tower_http::trace::DefaultMakeSpan;
@ -85,7 +87,7 @@ async fn main() {
);
AppState::default()
}
Err(e) => panic!("Unexpected error: {:?}", e),
Err(e) => panic!("Failed to read app state. Unexpected error: {:?}", e),
};
let (app_state_watch_tx, app_state_watch_rx) = tokio::sync::watch::channel(init_app_state.state.clone());
let app_state: SharedState = Arc::new(tokio::sync::RwLock::new(SharedStateParts {
@ -184,14 +186,54 @@ async fn ws_handler(
}
/// Websocket statemachine (one will be spawned per connection)
async fn handle_socket(mut socket: WebSocket, state_watch_rx: tokio::sync::watch::Receiver<common::State>) {
let mut stream = tokio_stream::wrappers::WatchStream::new(state_watch_rx);
while let Some(state) = stream.next().await {
let state: common::WebSocketMessage = state;
let serialized = serde_json::to_string(&state).expect("Failed to serialize app state to JSON");
if socket.send(Message::Text(serialized)).await.is_err() {
tracing::debug!("Websocket client disconnected");
break;
async fn handle_socket(socket: WebSocket, state_watch_rx: tokio::sync::watch::Receiver<common::State>) {
let (mut send, mut recv) = socket.split();
let mut outgoing = tokio_stream::wrappers::WatchStream::new(state_watch_rx).map(|state| {
let msg = common::WebSocketMessageServerToApp::State(state);
let msg = serde_json::to_string(&msg).expect("Failed to serialize app state to JSON");
Message::Text(msg)
});
let incoming = &mut recv;
loop {
select! {
out = outgoing.next() => match out {
Some(msg) => {
if send.send(msg).await.is_err() {
tracing::debug!("Websocket client disconnected");
break;
}
},
None => break,
},
inc = incoming.next() => match inc {
Some(Ok(Message::Text(msg))) => {
match serde_json::from_str::<common::WebSocketMessageAppToServer>(&msg) {
Ok(msg) => match msg {
common::WebSocketMessageAppToServer::HeartBeat => {
tracing::info!("ws recv heartbeat");
},
},
Err(err) => {
tracing::error!("ws deserialization error: {err}");
},
}
},
Some(Ok(Message::Binary(_))) => {},
Some(Ok(Message::Ping(_))) => {
tracing::info!("ws recv ping");
},
Some(Ok(Message::Pong(_))) => {
tracing::info!("ws recv pong");
},
Some(Ok(Message::Close(_))) => {},
Some(Err(err)) => {
tracing::error!("ws error: {err}");
},
None => break,
}
}
}

View File

@ -1,7 +1,21 @@
use serde::Deserialize;
use serde::Serialize;
pub type WebSocketMessage = State;
#[derive(Clone, Debug, Serialize, Deserialize, PartialEq, Eq)]
pub enum WebSocketMessageServerToApp {
/// State (server to app).
State(State),
/// Keep-alive message.
HeartBeat,
}
#[derive(Clone, Debug, Serialize, Deserialize, PartialEq, Eq)]
pub enum WebSocketMessageAppToServer {
/// Keep-alive message.
HeartBeat,
}
pub type RestResponse<T> = Result<T, RestError>;
#[derive(thiserror::Error, Default, Clone, Debug, Serialize, Deserialize, PartialEq, Eq)]

View File

@ -2,5 +2,5 @@ use frontend::event_bus::EventBus;
use yew_agent::PublicWorker;
fn main() {
EventBus::<common::WebSocketMessage>::register();
EventBus::<common::State>::register();
}

View File

@ -48,7 +48,7 @@ struct AppStateInner {
state: common::State,
}
impl Reducible for AppStateInner {
type Action = common::WebSocketMessage;
type Action = common::State;
fn reduce(self: Rc<Self>, action: Self::Action) -> Rc<Self> {
Rc::new(Self { state: action })
}
@ -63,7 +63,7 @@ pub fn App() -> Html {
let _event_bus = use_memo(
|_| {
log::info!("Creating event bus bridge.");
EventBus::<common::WebSocketMessage>::bridge(Rc::new(move |ws_msg: common::WebSocketMessage| {
EventBus::<common::State>::bridge(Rc::new(move |ws_msg: common::State| {
log::debug!("dispatching websocket msg to reducer");
app_state_dispatcher.dispatch(ws_msg);
}))

View File

@ -1,16 +1,17 @@
use crate::event_bus::EventBus;
use crate::event_bus::EventBusInput;
use futures::channel::mpsc::Sender;
use futures::SinkExt;
use futures::StreamExt;
use reqwasm::websocket::futures::WebSocket;
use reqwasm::websocket::Message;
use std::time::Duration;
use wasm_bindgen_futures::spawn_local;
use yew_agent::Dispatched;
#[derive(Debug)]
pub struct WebsocketService {
// No messages sent from app to server on websocket at the moment.
pub tx: Sender<()>,
pub tx: Sender<common::WebSocketMessageAppToServer>,
}
impl WebsocketService {
@ -34,16 +35,34 @@ impl WebsocketService {
let ws = WebSocket::open(&ws_url).unwrap();
log::info!("Opened websocket connection to {ws_url}");
let (_write, mut read) = ws.split();
let (write, mut read) = ws.split();
let (in_tx, mut in_rx) = futures::channel::mpsc::channel::<()>(1000);
let mut event_bus = EventBus::<common::WebSocketMessage>::dispatcher();
let (in_tx, in_rx) = futures::channel::mpsc::channel::<common::WebSocketMessageAppToServer>(1000);
let mut event_bus = EventBus::<common::State>::dispatcher();
// Generate regular heartbeat messages from app to server
let (mut heartbeat_tx, heartbeat_rx) = futures::channel::mpsc::channel(1);
spawn_local(async move {
let heartbeat_interval = yew::platform::time::interval(Duration::from_secs(20));
futures::pin_mut!(heartbeat_interval);
while heartbeat_interval.next().await.is_some() {
let msg = common::WebSocketMessageAppToServer::HeartBeat;
heartbeat_tx.send(msg).await.unwrap();
}
});
// App to Server
let all_tx = futures::stream::select(in_rx, heartbeat_rx);
spawn_local(async move {
while let Some(()) = in_rx.next().await {
// write.send(Message::Text(s)).await.unwrap();
}
// Serialize as JSON and map to websocket string message
let all_tx = all_tx
.map(|msg| serde_json::to_string(&msg).expect("Serialization error"))
.map(Message::Text)
.map(Ok);
all_tx
.forward(write)
.await
.expect("Forward to websocket write half failed");
});
// Server to App
@ -51,11 +70,14 @@ impl WebsocketService {
while let Some(msg) = read.next().await {
match msg {
Ok(Message::Text(data)) => {
match serde_json::from_str::<common::WebSocketMessage>(&data) {
Ok(ws_msg) => {
log::debug!("Received ws message. Dispatching to event bus.");
event_bus.send(EventBusInput::EventBusMsg(ws_msg));
}
match serde_json::from_str::<common::WebSocketMessageServerToApp>(&data) {
Ok(ws_msg) => match ws_msg {
common::WebSocketMessageServerToApp::State(state) => {
log::debug!("Received ws message. Dispatching to event bus.");
event_bus.send(EventBusInput::EventBusMsg(state));
}
common::WebSocketMessageServerToApp::HeartBeat => {}
},
Err(err) => {
log::error!("{err:?}");
}

View File

@ -47,6 +47,14 @@ deploy-backend:
ssh root@ajb.dk -- systemctl restart achievements-backend.service
just logs-prod
# Copy the nginx conf to the server and reload nginx
deploy-nginx-conf:
rsync -avz nginx.conf root@ajb.dk:/etc/nginx/sites-available/achievements.conf
ssh plul@ajb.dk cat /etc/nginx/sites-available/achievements.conf
ssh root@ajb.dk nginx -T
ssh root@ajb.dk systemctl reload nginx
# List logs with journalctl
logs-prod:
ssh plul@ajb.dk journalctl -u achievements-backend

View File

@ -1,30 +1,26 @@
# Backend
server {
listen 4000 ssl http2;
listen [::]:4000 ssl http2;
server_name achievements.ajb.dk;
ssl_certificate /etc/letsencrypt/live/ajb.dk/fullchain.pem;
ssl_certificate_key /etc/letsencrypt/live/ajb.dk/privkey.pem;
location / {
proxy_pass http://localhost:4001/;
proxy_http_version 1.1;
proxy_set_header Upgrade $http_upgrade;
proxy_set_header Connection "Upgrade";
proxy_set_header Host $host;
}
}
# Frontend (serve yew app static files)
server {
listen 443 ssl http2;
listen [::]:443 ssl http2;
server_name achievements.ajb.dk;
ssl_certificate /etc/letsencrypt/live/ajb.dk/fullchain.pem;
ssl_certificate_key /etc/letsencrypt/live/ajb.dk/privkey.pem;
ssl_certificate /etc/letsencrypt/live/git.ajb.dk/fullchain.pem; # managed by Certbot
ssl_certificate_key /etc/letsencrypt/live/git.ajb.dk/privkey.pem; # managed by Certbot
# Backend
location /api {
proxy_pass http://localhost:4000;
proxy_http_version 1.1;
proxy_set_header Upgrade $http_upgrade;
proxy_set_header Connection "Upgrade";
proxy_set_header Host $host;
# Prevent websocket connections timing out after 1 minute.
# This sets the timeout to 24 hours:
proxy_read_timeout 86400;
proxy_send_timeout 86400;
}
# Frontend (serve yew app static files)
location / {
alias /var/www/achievements/;
try_files $uri $uri/ /index.html;