persistent websocket

This commit is contained in:
2023-06-13 11:27:00 +02:00
parent 70d99b0c6b
commit 78999174ab
18 changed files with 334 additions and 309 deletions

View File

@@ -16,6 +16,7 @@ use common::CreateMilestone;
use common::DeleteAchievement;
use common::DeleteMilestone;
use common::Milestone;
use common::RestResponse;
use common::ToggleAchievement;
use serde::Deserialize;
use serde::Serialize;
@@ -37,6 +38,20 @@ struct SharedStateParts {
watcher_tx: tokio::sync::watch::Sender<common::State>,
}
type Response<T> = Result<(StatusCode, Json<RestResponse<T>>), HandlerError>;
// TODO: still needed?
#[derive(Debug, thiserror::Error)]
enum HandlerError {
// #[error("Failed to lock state")]
// LockAppStateError,
}
impl IntoResponse for HandlerError {
fn into_response(self) -> axum::response::Response {
let error_message = format!("{self}");
(StatusCode::INTERNAL_SERVER_ERROR, error_message).into_response()
}
}
#[tokio::main]
async fn main() {
tracing_subscriber::fmt::init();
@@ -168,7 +183,7 @@ async fn handle_socket(mut socket: WebSocket, state_watch_rx: tokio::sync::watch
async fn create_achievement(
Extension(app_state): Extension<SharedState>,
Json(create_achievement): Json<CreateAchievement>,
) -> Result<(StatusCode, ()), HandlerError> {
) -> Response<()> {
tracing::debug!("Creating achievement: {create_achievement:?}.");
let achievement = Achievement {
goal: create_achievement.goal,
@@ -180,17 +195,21 @@ async fn create_achievement(
lock.watcher_tx
.send(lock.app_state.state.clone())
.expect("watch channel is closed, every receiver was dropped.");
Ok((StatusCode::CREATED, ()))
Ok((StatusCode::CREATED, Json(Ok(()))))
}
async fn create_milestone(
Extension(app_state): Extension<SharedState>,
Json(create_milestone): Json<CreateMilestone>,
) -> Result<(StatusCode, ()), HandlerError> {
) -> Response<()> {
tracing::debug!("Creating milestone: {create_milestone:?}.");
if create_milestone.goal > 100 {
return Ok((StatusCode::BAD_REQUEST, ()));
let goal_max = 100;
if create_milestone.goal > goal_max {
return Ok((
StatusCode::BAD_REQUEST,
Json(Err(format!("Max goal allowed: {goal_max}.").into())),
));
}
let milestone = Milestone {
@@ -202,13 +221,13 @@ async fn create_milestone(
lock.watcher_tx
.send(lock.app_state.state.clone())
.expect("watch channel is closed, every receiver was dropped.");
Ok((StatusCode::CREATED, ()))
Ok((StatusCode::CREATED, Json(Ok(()))))
}
async fn delete_milestone(
Extension(app_state): Extension<SharedState>,
Json(delete_milestone): Json<DeleteMilestone>,
) -> Result<(StatusCode, ()), HandlerError> {
) -> Response<()> {
tracing::debug!("Deleting milestone: {delete_milestone:?}.");
let mut lock = app_state.write().await;
if let Some(pos) = lock
@@ -223,13 +242,13 @@ async fn delete_milestone(
.send(lock.app_state.state.clone())
.expect("watch channel is closed, every receiver was dropped.");
}
Ok((StatusCode::OK, ()))
Ok((StatusCode::OK, Json(Ok(()))))
}
async fn toggle_achievement(
Extension(app_state): Extension<SharedState>,
Json(toggle_achievement): Json<ToggleAchievement>,
) -> Result<(StatusCode, ()), HandlerError> {
) -> Response<()> {
tracing::debug!("Toggling achievement: {toggle_achievement:?}.");
let mut lock = app_state.write().await;
if let Some(achievement) = lock
@@ -244,13 +263,13 @@ async fn toggle_achievement(
.send(lock.app_state.state.clone())
.expect("watch channel is closed, every receiver was dropped.");
}
Ok((StatusCode::OK, ()))
Ok((StatusCode::OK, Json(Ok(()))))
}
async fn delete_achievement(
Extension(app_state): Extension<SharedState>,
Json(delete_achievement): Json<DeleteAchievement>,
) -> Result<(StatusCode, ()), HandlerError> {
) -> Response<()> {
tracing::debug!("Deleting achievement: {delete_achievement:?}.");
let mut lock = app_state.write().await;
if let Some(pos) = lock
@@ -265,7 +284,7 @@ async fn delete_achievement(
.send(lock.app_state.state.clone())
.expect("watch channel is closed, every receiver was dropped.");
}
Ok((StatusCode::OK, ()))
Ok((StatusCode::OK, Json(Ok(()))))
}
async fn shutdown_signal() {
@@ -335,17 +354,3 @@ pub enum AppStateWriteError {
#[error("Failed to serialize the state")]
SerializationError(#[from] serde_json::Error),
}
// TODO: still needed?
#[derive(Debug, thiserror::Error)]
enum HandlerError {
// #[error("Failed to lock state")]
// LockAppStateError,
}
impl IntoResponse for HandlerError {
fn into_response(self) -> axum::response::Response {
let error_message = format!("{self}");
(StatusCode::INTERNAL_SERVER_ERROR, error_message).into_response()
}
}