-
Notifications
You must be signed in to change notification settings - Fork 128
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
3 changed files
with
291 additions
and
277 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,284 @@ | ||
use std::time::Duration; | ||
|
||
use crate::settings::{Settings, StartConditionSettings, ToggleableThreshold}; | ||
|
||
use anyhow::Context; | ||
use blockscout_service_launcher::launcher::ConfigSettings; | ||
use reqwest::StatusCode; | ||
use tokio::time::sleep; | ||
use tracing::{info, warn}; | ||
|
||
fn is_retryable_code(status_code: &reqwest::StatusCode) -> bool { | ||
matches!( | ||
*status_code, | ||
StatusCode::INTERNAL_SERVER_ERROR | ||
| StatusCode::SERVICE_UNAVAILABLE | ||
| StatusCode::GATEWAY_TIMEOUT | ||
| StatusCode::TOO_MANY_REQUESTS | ||
| StatusCode::IM_A_TEAPOT | ||
) | ||
} | ||
|
||
fn is_threshold_passed( | ||
threshold: &ToggleableThreshold, | ||
float_value: Option<String>, | ||
value_name: &str, | ||
) -> Result<bool, anyhow::Error> { | ||
let threshold = if threshold.enabled { | ||
threshold.threshold | ||
} else { | ||
return Ok(true); | ||
}; | ||
let value = float_value | ||
.map(|s| s.parse::<f64>()) | ||
.transpose() | ||
.context(format!("Parsing `{value_name}`"))?; | ||
let Some(value) = value else { | ||
anyhow::bail!("Received `null` value of `{value_name}`. Can't determine indexing status.",); | ||
}; | ||
if value < threshold { | ||
info!( | ||
threshold = threshold, | ||
current_value = value, | ||
"Threshold for `{value_name}` is not satisfied" | ||
); | ||
Ok(false) | ||
} else { | ||
info!( | ||
threshold = threshold, | ||
current_value = value, | ||
"Threshold for `{value_name}` is satisfied" | ||
); | ||
Ok(true) | ||
} | ||
} | ||
|
||
pub async fn wait_for_blockscout_indexing( | ||
api_config: blockscout_client::Configuration, | ||
wait_config: StartConditionSettings, | ||
) -> Result<(), anyhow::Error> { | ||
loop { | ||
match blockscout_client::apis::main_page_api::get_indexing_status(&api_config).await { | ||
Ok(result) | ||
if is_threshold_passed( | ||
&wait_config.blocks_ratio, | ||
result.indexed_blocks_ratio.clone(), | ||
"indexed_blocks_ratio", | ||
) | ||
.context("check index block ratio")? | ||
&& is_threshold_passed( | ||
&wait_config.internal_transactions_ratio, | ||
result.indexed_internal_transactions_ratio.clone(), | ||
"indexed_internal_transactions_ratio", | ||
)? => | ||
{ | ||
info!("Blockscout indexing threshold passed"); | ||
return Ok(()); | ||
} | ||
Ok(_) => {} | ||
Err(blockscout_client::Error::ResponseError(r)) if is_retryable_code(&r.status) => { | ||
warn!("Error from indexing status endpoint: {r:?}"); | ||
} | ||
Err(e) => { | ||
return Err(e).context("Requesting indexing status"); | ||
} | ||
} | ||
info!( | ||
"Blockscout is not indexed enough. Checking again in {} secs", | ||
wait_config.check_period_secs | ||
); | ||
sleep(Duration::from_secs(wait_config.check_period_secs.into())).await; | ||
} | ||
} | ||
|
||
pub async fn init_blockscout_api_client( | ||
settings: &Settings, | ||
) -> anyhow::Result<Option<blockscout_client::Configuration>> { | ||
match (settings.ignore_blockscout_api_absence, &settings.blockscout_api_url) { | ||
(_, Some(blockscout_api_url)) => Ok(Some(blockscout_client::Configuration::new(blockscout_api_url.clone()))), | ||
(true, None) => { | ||
info!( | ||
"Blockscout API URL has not been provided and `IGNORE_BLOCKSCOUT_API_ABSENCE` setting is \ | ||
set to `true`. Disabling API-related functionality." | ||
); | ||
Ok(None) | ||
} | ||
(false, None) => anyhow::bail!( | ||
"Blockscout API URL has not been provided. Please specify it with corresponding \ | ||
env variable (`{0}__BLOCKSCOUT_API_URL`) or set `{0}__IGNORE_BLOCKSCOUT_API_ABSENCE=true` to disable \ | ||
functionality depending on the API.", | ||
Settings::SERVICE_NAME | ||
), | ||
} | ||
} | ||
|
||
#[cfg(test)] | ||
mod tests { | ||
use std::str::FromStr; | ||
|
||
use rstest::*; | ||
use std::time::Duration; | ||
use tokio::{task::JoinSet, time::error::Elapsed}; | ||
use url::Url; | ||
use wiremock::{ | ||
matchers::{method, path}, | ||
Mock, MockServer, ResponseTemplate, | ||
}; | ||
|
||
use super::*; | ||
|
||
async fn mock_indexing_status(response: ResponseTemplate) -> MockServer { | ||
let mock_server = MockServer::start().await; | ||
Mock::given(method("GET")) | ||
.and(path("/api/v2/main-page/indexing-status")) | ||
.respond_with(response) | ||
.mount(&mock_server) | ||
.await; | ||
mock_server | ||
} | ||
|
||
async fn test_wait_indexing( | ||
wait_config: StartConditionSettings, | ||
response: ResponseTemplate, | ||
) -> Result<Result<(), anyhow::Error>, Elapsed> { | ||
let server = mock_indexing_status(response).await; | ||
tokio::time::timeout( | ||
Duration::from_millis(500), | ||
wait_for_blockscout_indexing( | ||
blockscout_client::Configuration::new(Url::from_str(&server.uri()).unwrap()), | ||
wait_config, | ||
), | ||
) | ||
.await | ||
} | ||
|
||
#[fixture] | ||
fn wait_config( | ||
#[default(0.9)] blocks: f64, | ||
#[default(0.9)] internal_transactions: f64, | ||
) -> StartConditionSettings { | ||
StartConditionSettings { | ||
blocks_ratio: ToggleableThreshold::enabled(blocks), | ||
internal_transactions_ratio: ToggleableThreshold::enabled(internal_transactions), | ||
check_period_secs: 0, | ||
} | ||
} | ||
|
||
#[rstest] | ||
#[tokio::test] | ||
async fn wait_for_blockscout_indexing_works_with_200_response( | ||
wait_config: StartConditionSettings, | ||
) { | ||
test_wait_indexing( | ||
wait_config.clone(), | ||
ResponseTemplate::new(200).set_body_string( | ||
r#"{ | ||
"finished_indexing": true, | ||
"finished_indexing_blocks": true, | ||
"indexed_blocks_ratio": "1.00", | ||
"indexed_internal_transactions_ratio": "1" | ||
}"#, | ||
), | ||
) | ||
.await | ||
.expect("must not timeout") | ||
.expect("must not error"); | ||
|
||
test_wait_indexing( | ||
wait_config, | ||
ResponseTemplate::new(200).set_body_string( | ||
r#"{ | ||
"finished_indexing": false, | ||
"finished_indexing_blocks": false, | ||
"indexed_blocks_ratio": "0.80", | ||
"indexed_internal_transactions_ratio": "0.80" | ||
}"#, | ||
), | ||
) | ||
.await | ||
.expect_err("must time out"); | ||
} | ||
|
||
#[rstest] | ||
#[tokio::test] | ||
async fn wait_for_blockscout_indexing_works_with_slow_response( | ||
wait_config: StartConditionSettings, | ||
) { | ||
test_wait_indexing( | ||
wait_config, | ||
ResponseTemplate::new(200) | ||
.set_body_string( | ||
r#"{ | ||
"finished_indexing": false, | ||
"finished_indexing_blocks": false, | ||
"indexed_blocks_ratio": "1.0", | ||
"indexed_internal_transactions_ratio": "1.0" | ||
}"#, | ||
) | ||
.set_delay(Duration::from_millis(100)), | ||
) | ||
.await | ||
.expect("must not timeout") | ||
.expect("must not error") | ||
} | ||
|
||
#[rstest] | ||
#[tokio::test] | ||
async fn wait_for_blockscout_indexing_works_with_infinite_timeout( | ||
wait_config: StartConditionSettings, | ||
) { | ||
test_wait_indexing( | ||
wait_config, | ||
ResponseTemplate::new(200) | ||
.set_body_string( | ||
r#"{ | ||
"finished_indexing": false, | ||
"finished_indexing_blocks": false, | ||
"indexed_blocks_ratio": "0.80", | ||
"indexed_internal_transactions_ratio": "0.80" | ||
}"#, | ||
) | ||
.set_delay(Duration::MAX), | ||
) | ||
.await | ||
.expect_err("must time out"); | ||
} | ||
|
||
#[rstest] | ||
#[tokio::test] | ||
async fn wait_for_blockscout_indexing_retries_with_error_codes( | ||
wait_config: StartConditionSettings, | ||
) { | ||
let mut error_servers = JoinSet::from_iter([ | ||
test_wait_indexing(wait_config.clone(), ResponseTemplate::new(429)), | ||
test_wait_indexing(wait_config.clone(), ResponseTemplate::new(500)), | ||
test_wait_indexing(wait_config.clone(), ResponseTemplate::new(503)), | ||
test_wait_indexing(wait_config.clone(), ResponseTemplate::new(504)), | ||
]); | ||
#[allow(for_loops_over_fallibles)] | ||
for server in error_servers.join_next().await { | ||
let test_result = server.unwrap(); | ||
test_result.expect_err("must time out"); | ||
} | ||
} | ||
|
||
#[rstest] | ||
#[tokio::test] | ||
async fn wait_for_blockscout_indexing_fails_with_error_codes( | ||
wait_config: StartConditionSettings, | ||
) { | ||
let mut error_servers = JoinSet::from_iter([ | ||
test_wait_indexing(wait_config.clone(), ResponseTemplate::new(400)), | ||
test_wait_indexing(wait_config.clone(), ResponseTemplate::new(403)), | ||
test_wait_indexing(wait_config.clone(), ResponseTemplate::new(404)), | ||
test_wait_indexing(wait_config.clone(), ResponseTemplate::new(405)), | ||
]); | ||
#[allow(for_loops_over_fallibles)] | ||
for server in error_servers.join_next().await { | ||
let test_result = server.unwrap(); | ||
test_result | ||
.expect("must fail immediately") | ||
.expect_err("must report error"); | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,3 +1,4 @@ | ||
pub mod blockscout_waiter; | ||
mod config; | ||
mod health; | ||
mod read_service; | ||
|
Oops, something went wrong.