diff --git a/crates/jstzd/src/task/jstzd.rs b/crates/jstzd/src/task/jstzd.rs index 6240d9eb3..73554eab4 100644 --- a/crates/jstzd/src/task/jstzd.rs +++ b/crates/jstzd/src/task/jstzd.rs @@ -3,7 +3,8 @@ use anyhow::Result; use async_dropper_simple::AsyncDrop; use async_trait::async_trait; use axum::{ - extract::State, + extract::{Path, State}, + response::IntoResponse, routing::{get, put}, Router, }; @@ -13,20 +14,24 @@ use octez::r#async::{ node_config::OctezNodeConfig, protocol::ProtocolParameter, }; +use serde::Serialize; use std::sync::Arc; use tokio::{net::TcpListener, sync::RwLock, task::JoinHandle}; -#[derive(Clone)] struct Jstzd { octez_node: Arc>, baker: Arc>, } -#[derive(Clone)] +#[derive(Clone, Serialize)] pub struct JstzdConfig { + #[serde(rename(serialize = "octez-node"))] octez_node_config: OctezNodeConfig, + #[serde(rename(serialize = "octez-baker"))] baker_config: OctezBakerConfig, + #[serde(rename(serialize = "octez-client"))] octez_client_config: OctezClientConfig, + #[serde(skip_serializing)] protocol_params: ProtocolParameter, } @@ -44,6 +49,18 @@ impl JstzdConfig { protocol_params, } } + + pub fn octez_node_config(&self) -> &OctezNodeConfig { + &self.octez_node_config + } + + pub fn octez_client_config(&self) -> &OctezClientConfig { + &self.octez_client_config + } + + pub fn baker_config(&self) -> &OctezBakerConfig { + &self.baker_config + } } #[async_trait] @@ -150,13 +167,15 @@ impl Jstzd { } } +#[derive(Clone)] pub struct JstzdServer { - jstzd_config: JstzdConfig, jstzd_server_port: u16, state: Arc>, } struct ServerState { + jstzd_config: JstzdConfig, + jstzd_config_json: serde_json::Map, jstzd: Option, server_handle: Option>, } @@ -172,9 +191,14 @@ impl AsyncDrop for JstzdServer { impl JstzdServer { pub fn new(config: JstzdConfig, port: u16) -> Self { Self { - jstzd_config: config, 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, })), @@ -192,12 +216,14 @@ impl JstzdServer { } pub async fn run(&mut self) -> Result<()> { - let jstzd = Jstzd::spawn(self.jstzd_config.clone()).await?; + let jstzd = Jstzd::spawn(self.state.read().await.jstzd_config.clone()).await?; self.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?; @@ -261,3 +287,23 @@ async fn shutdown_handler(state: State>>) -> http::Statu }; http::StatusCode::NO_CONTENT } + +async fn all_config_handler(state: State>>) -> impl IntoResponse { + let config = &state.read().await.jstzd_config_json; + serde_json::to_string(config).unwrap().into_response() +} + +async fn config_handler( + state: State>>, + Path(config_type): Path, +) -> impl IntoResponse { + let config = &state.read().await.jstzd_config_json; + match config.get(&config_type) { + Some(v) => match serde_json::to_string(v) { + Ok(s) => s.into_response(), + // TODO: log this error + Err(_) => http::StatusCode::INTERNAL_SERVER_ERROR.into_response(), + }, + None => http::StatusCode::NOT_FOUND.into_response(), + } +} diff --git a/crates/jstzd/tests/jstzd_test.rs b/crates/jstzd/tests/jstzd_test.rs index 2260fcac7..a2011b7a3 100644 --- a/crates/jstzd/tests/jstzd_test.rs +++ b/crates/jstzd/tests/jstzd_test.rs @@ -1,6 +1,7 @@ mod utils; use std::path::PathBuf; +use jstzd::task::jstzd::{JstzdConfig, JstzdServer}; use jstzd::task::utils::retry; use jstzd::{EXCHANGER_ADDRESS, JSTZ_NATIVE_BRIDGE_ADDRESS}; use octez::r#async::baker::{BakerBinaryPath, OctezBakerConfigBuilder}; @@ -21,13 +22,36 @@ const CONTRACT_NAMES: [(&str, &str); 2] = [ #[tokio::test(flavor = "multi_thread")] async fn jstzd_test() { let rpc_endpoint = Endpoint::localhost(unused_port()); + let jstzd_port = unused_port(); + let (mut jstzd, config) = create_jstzd_server(&rpc_endpoint, jstzd_port).await; + + jstzd.run().await.unwrap(); + ensure_jstzd_components_are_up(&jstzd, &rpc_endpoint, jstzd_port).await; + let octez_client = OctezClient::new(config.octez_client_config().clone()); + check_bootstrap_contracts(&octez_client).await; + + fetch_config_test(config, jstzd_port).await; + + reqwest::Client::new() + .put(&format!("http://localhost:{}/shutdown", jstzd_port)) + .send() + .await + .unwrap(); + + ensure_jstzd_components_are_down(&rpc_endpoint, jstzd_port).await; +} + +async fn create_jstzd_server( + octez_node_rpc_endpoint: &Endpoint, + jstzd_port: u16, +) -> (JstzdServer, JstzdConfig) { let run_options = OctezNodeRunOptionsBuilder::new() .set_synchronisation_threshold(0) .set_network("sandbox") .build(); let octez_node_config = OctezNodeConfigBuilder::new() .set_network("sandbox") - .set_rpc_endpoint(&rpc_endpoint) + .set_rpc_endpoint(octez_node_rpc_endpoint) .set_run_options(&run_options) .build() .unwrap(); @@ -59,18 +83,24 @@ async fn jstzd_test() { .build() .expect("Failed to build baker config"); - let config = jstzd::task::jstzd::JstzdConfig::new( + let config = JstzdConfig::new( octez_node_config, baker_config, octez_client_config.clone(), protocol_params, ); - let jstzd_port = unused_port(); - let mut jstzd = jstzd::task::jstzd::JstzdServer::new(config, jstzd_port); - jstzd.run().await.unwrap(); + (JstzdServer::new(config.clone(), jstzd_port), config) +} +async fn ensure_jstzd_components_are_up( + jstzd: &JstzdServer, + octez_node_rpc_endpoint: &Endpoint, + jstzd_port: u16, +) { let jstz_health_check_endpoint = format!("http://localhost:{}/health", jstzd_port); - let octez_node_health_check_endpoint = format!("{}/health/ready", rpc_endpoint); + let octez_node_health_check_endpoint = + format!("{}/health/ready", octez_node_rpc_endpoint); + let jstzd_running = retry(30, 1000, || async { let res = reqwest::get(&jstz_health_check_endpoint).await; Ok(res.is_ok()) @@ -78,24 +108,23 @@ async fn jstzd_test() { .await; assert!(jstzd_running); - let node_running = retry(30, 1000, || async { - let res = reqwest::get(&octez_node_health_check_endpoint).await; - Ok(res.is_ok()) - }) - .await; - assert!(node_running); + // check if individual components are up / jstzd health check indeed covers all components + assert!(reqwest::get(&octez_node_health_check_endpoint) + .await + .is_ok()); assert!(jstzd.baker_healthy().await); assert!(jstzd.health_check().await); +} - let octez_client = OctezClient::new(octez_client_config); - check_bootstrap_contracts(&octez_client).await; - - reqwest::Client::new() - .put(&format!("http://localhost:{}/shutdown", jstzd_port)) - .send() - .await - .unwrap(); +async fn ensure_jstzd_components_are_down( + jstzd: &JstzdServer, + octez_node_rpc_endpoint: &Endpoint, + jstzd_port: u16, +) { + let jstz_health_check_endpoint = format!("http://localhost:{}/health", jstzd_port); + let octez_node_health_check_endpoint = + format!("{}/health/ready", octez_node_rpc_endpoint); let jstzd_stopped = retry(30, 1000, || async { let res = reqwest::get(&jstz_health_check_endpoint).await; @@ -107,6 +136,8 @@ async fn jstzd_test() { .await; assert!(jstzd_stopped); + // check if individual components are terminated + // and jstzd indeed tears down all components before it shuts down let node_destroyed = retry(30, 1000, || async { let res = reqwest::get(&octez_node_health_check_endpoint).await; // Should get an error since the node should have been terminated @@ -120,9 +151,59 @@ async fn jstzd_test() { assert!(!jstzd.baker_healthy().await); assert!(!jstzd.health_check().await); +} - // stop should be idempotent and thus should be okay after jstzd is already stopped - jstzd.stop().await.unwrap(); +async fn fetch_config_test(jstzd_config: JstzdConfig, jstzd_port: u16) { + let mut full_config = serde_json::json!({}); + for (key, expected_json) in [ + ( + "octez-node", + serde_json::to_value(jstzd_config.octez_node_config()).unwrap(), + ), + ( + "octez-client", + serde_json::to_value(jstzd_config.octez_client_config()).unwrap(), + ), + ( + "octez-baker", + serde_json::to_value(jstzd_config.baker_config()).unwrap(), + ), + ] { + let res = + reqwest::get(&format!("http://localhost:{}/config/{}", jstzd_port, key)) + .await + .unwrap(); + assert_eq!( + expected_json, + serde_json::from_str::(&res.text().await.unwrap()) + .unwrap(), + "config mismatch at /config/{}", + key + ); + full_config + .as_object_mut() + .unwrap() + .insert(key.to_owned(), expected_json); + } + + // invalid config type + assert_eq!( + reqwest::get(&format!("http://localhost:{}/config/foobar", jstzd_port)) + .await + .unwrap() + .status(), + reqwest::StatusCode::NOT_FOUND + ); + + // all configs + let res = reqwest::get(&format!("http://localhost:{}/config/", jstzd_port)) + .await + .unwrap(); + assert_eq!( + full_config, + serde_json::from_str::(&res.text().await.unwrap()).unwrap(), + "config mismatch at /config/", + ); } async fn read_bootstrap_contracts() -> Vec {