Skip to content

Commit

Permalink
feat(jstzd): use AsyncDropper in JstzdServer
Browse files Browse the repository at this point in the history
  • Loading branch information
huancheng-trili committed Nov 19, 2024
1 parent 3eaef9d commit bf6b52b
Show file tree
Hide file tree
Showing 2 changed files with 52 additions and 23 deletions.
69 changes: 46 additions & 23 deletions crates/jstzd/src/task/jstzd.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
use super::{octez_baker::OctezBaker, octez_node::OctezNode, utils::retry, Task};
use anyhow::Result;
use async_dropper_simple::AsyncDrop;
use async_dropper_simple::{AsyncDrop, AsyncDropper};
use async_trait::async_trait;
use axum::{
extract::{Path, State},
Expand Down Expand Up @@ -167,70 +167,91 @@ impl Jstzd {
}
}

#[derive(Clone)]
pub struct JstzdServer {
jstzd_server_port: u16,
#[derive(Clone, Default)]
pub struct JstzdServerInner {
state: Arc<RwLock<ServerState>>,
}

#[derive(Default)]
struct ServerState {
jstzd_config: JstzdConfig,
jstzd_config: Option<JstzdConfig>,
jstzd_config_json: serde_json::Map<String, serde_json::Value>,
jstzd: Option<Jstzd>,
server_handle: Option<JoinHandle<()>>,
}

#[async_trait]
impl AsyncDrop for JstzdServer {
impl AsyncDrop for JstzdServerInner {
async fn async_drop(&mut self) {
let mut lock = self.state.write().await;
let _ = shutdown(&mut lock).await;
}
}

pub struct JstzdServer {
port: u16,
inner: Arc<AsyncDropper<JstzdServerInner>>,
}

impl JstzdServer {
pub fn new(config: JstzdConfig, port: u16) -> Self {
Self {
jstzd_server_port: port,
state: Arc::new(RwLock::new(ServerState {
jstzd_config_json: serde_json::to_value(&config)
.unwrap()
.as_object()
.unwrap()
.to_owned(),
jstzd_config: config,
jstzd: None,
server_handle: None,
port,
inner: Arc::new(AsyncDropper::new(JstzdServerInner {
state: Arc::new(RwLock::new(ServerState {
jstzd_config_json: serde_json::to_value(&config)
.unwrap()
.as_object()
.unwrap()
.to_owned(),
jstzd_config: Some(config),
jstzd: None,
server_handle: None,
})),
})),
}
}

pub async fn health_check(&self) -> bool {
let lock = self.state.read().await;
let lock = self.inner.state.read().await;
health_check(&lock).await
}

pub async fn stop(&mut self) -> Result<()> {
let mut lock = self.state.write().await;
let mut lock = self.inner.state.write().await;
shutdown(&mut lock).await
}

pub async fn run(&mut self) -> Result<()> {
let jstzd = Jstzd::spawn(self.state.read().await.jstzd_config.clone()).await?;
self.state.write().await.jstzd.replace(jstzd);
let jstzd = Jstzd::spawn(
self.inner
.state
.read()
.await
.jstzd_config
.as_ref()
.ok_or(anyhow::anyhow!(
// shouldn't really reach this branch since jstzd config is required at instantiation
// unless someone calls `run` after calling `stop`
"cannot run jstzd server without jstzd config"
))?
.clone(),
)
.await?;
self.inner.state.write().await.jstzd.replace(jstzd);

let router = Router::new()
.route("/health", get(health_check_handler))
.route("/shutdown", put(shutdown_handler))
.route("/config/:config_type", get(config_handler))
.route("/config/", get(all_config_handler))
.with_state(self.state.clone());
let listener = TcpListener::bind(("0.0.0.0", self.jstzd_server_port)).await?;
.with_state(self.inner.state.clone());
let listener = TcpListener::bind(("0.0.0.0", self.port)).await?;

let handle = tokio::spawn(async {
axum::serve(listener, router).await.unwrap();
});
self.state.write().await.server_handle.replace(handle);
self.inner.state.write().await.server_handle.replace(handle);
Ok(())
}
}
Expand Down Expand Up @@ -259,6 +280,8 @@ async fn shutdown(state: &mut ServerState) -> Result<()> {
if let Some(server) = state.server_handle.take() {
server.abort();
}
state.jstzd_config.take();
state.jstzd_config_json.clear();
Ok(())
}

Expand Down
6 changes: 6 additions & 0 deletions crates/jstzd/tests/jstzd_test.rs
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,12 @@ async fn jstzd_test() {
.unwrap();

ensure_jstzd_components_are_down(&rpc_endpoint, jstzd_port).await;

// calling `run` after calling `stop` should fail because all states should have been cleared
assert_eq!(
jstzd.run().await.unwrap_err().to_string(),
"cannot run jstzd server without jstzd config"
);
}

async fn create_jstzd_server(
Expand Down

0 comments on commit bf6b52b

Please sign in to comment.