From b6d8ec1a4298eb055842e3fc3179f56a318b0c3f Mon Sep 17 00:00:00 2001 From: junderw Date: Sun, 1 Oct 2023 19:14:03 -0700 Subject: [PATCH 1/2] REST API blocking async: Solution B, spawn_blocking --- src/rest.rs | 39 +++++++++++++++++++++++++++------------ 1 file changed, 27 insertions(+), 12 deletions(-) diff --git a/src/rest.rs b/src/rest.rs index 56567393..61cdc586 100644 --- a/src/rest.rs +++ b/src/rest.rs @@ -570,19 +570,34 @@ async fn run_server(config: Arc, query: Arc, rx: oneshot::Receive let uri = req.uri().clone(); let body = hyper::body::to_bytes(req.into_body()).await?; - let mut resp = handle_request(method, uri, body, &query, &config) - .unwrap_or_else(|err| { - warn!("{:?}", err); - Response::builder() - .status(err.0) - .header("Content-Type", "text/plain") - .header("X-Powered-By", &**VERSION_STRING) - .body(Body::from(err.1)) - .unwrap() - }); - if let Some(ref origins) = config.cors { + let cors = config + .cors + .as_ref() + .map(|c| c.parse::().unwrap()); + + let mut resp = tokio::task::spawn_blocking(move || { + handle_request(method, uri, body, &query, &config) + }) + .await + .unwrap_or_else(|err| { + warn!("JoinHandle error: {:?}", err); + Err(HttpError( + StatusCode::INTERNAL_SERVER_ERROR, + String::from("Internal Server Error: JoinHandle"), + )) + }) + .unwrap_or_else(|err| { + warn!("{:?}", err); + Response::builder() + .status(err.0) + .header("Content-Type", "text/plain") + .header("X-Powered-By", &**VERSION_STRING) + .body(Body::from(err.1)) + .unwrap() + }); + if let Some(origins) = cors { resp.headers_mut() - .insert("Access-Control-Allow-Origin", origins.parse().unwrap()); + .insert("Access-Control-Allow-Origin", origins); } Ok::<_, hyper::Error>(resp) } From cb85da72e658f094859f93b9b123c9de10501b43 Mon Sep 17 00:00:00 2001 From: junderw Date: Mon, 2 Oct 2023 21:18:06 -0700 Subject: [PATCH 2/2] Add metrics for REST response times --- src/bin/electrs.rs | 2 +- src/rest.rs | 20 +++++++++++++++++--- 2 files changed, 18 insertions(+), 4 deletions(-) diff --git a/src/bin/electrs.rs b/src/bin/electrs.rs index 76373d2b..9a301be4 100644 --- a/src/bin/electrs.rs +++ b/src/bin/electrs.rs @@ -99,7 +99,7 @@ fn run_server(config: Arc) -> Result<()> { )); // TODO: configuration for which servers to start - let rest_server = rest::start(Arc::clone(&config), Arc::clone(&query)); + let rest_server = rest::start(Arc::clone(&config), Arc::clone(&query), &metrics); let electrum_server = ElectrumRPC::start(Arc::clone(&config), Arc::clone(&query), &metrics); loop { diff --git a/src/rest.rs b/src/rest.rs index 61cdc586..13fc0c6f 100644 --- a/src/rest.rs +++ b/src/rest.rs @@ -1,6 +1,7 @@ use crate::chain::{address, BlockHash, Network, OutPoint, Script, Transaction, TxIn, TxOut, Txid}; use crate::config::{Config, VERSION_STRING}; use crate::errors; +use crate::metrics::Metrics; use crate::new_index::{compute_script_hash, Query, SpendingInput, Utxo}; use crate::util::{ create_socket, electrum_merkle, extract_tx_prevouts, full_hash, get_innerscripts, get_tx_fee, @@ -17,6 +18,7 @@ use bitcoin::hashes::Error as HashError; use hex::{self, FromHexError}; use hyper::service::{make_service_fn, service_fn}; use hyper::{Body, Method, Response, Server, StatusCode}; +use prometheus::{HistogramOpts, HistogramVec}; use tokio::sync::oneshot; use hyperlocal::UnixServerExt; @@ -549,7 +551,12 @@ fn prepare_txs( } #[tokio::main] -async fn run_server(config: Arc, query: Arc, rx: oneshot::Receiver<()>) { +async fn run_server( + config: Arc, + query: Arc, + rx: oneshot::Receiver<()>, + metric: HistogramVec, +) { let addr = &config.http_addr; let socket_file = &config.http_socket_file; @@ -559,11 +566,13 @@ async fn run_server(config: Arc, query: Arc, rx: oneshot::Receive let make_service_fn_inn = || { let query = Arc::clone(&query); let config = Arc::clone(&config); + let metric = metric.clone(); async move { Ok::<_, hyper::Error>(service_fn(move |req| { let query = Arc::clone(&query); let config = Arc::clone(&config); + let timer = metric.with_label_values(&["all_methods"]).start_timer(); async move { let method = req.method().clone(); @@ -599,6 +608,7 @@ async fn run_server(config: Arc, query: Arc, rx: oneshot::Receive resp.headers_mut() .insert("Access-Control-Allow-Origin", origins); } + timer.observe_duration(); Ok::<_, hyper::Error>(resp) } })) @@ -645,13 +655,17 @@ async fn run_server(config: Arc, query: Arc, rx: oneshot::Receive } } -pub fn start(config: Arc, query: Arc) -> Handle { +pub fn start(config: Arc, query: Arc, metrics: &Metrics) -> Handle { let (tx, rx) = oneshot::channel::<()>(); + let response_timer = metrics.histogram_vec( + HistogramOpts::new("electrs_rest_api", "Electrs REST API response timings"), + &["method"], + ); Handle { tx, thread: crate::util::spawn_thread("rest-server", move || { - run_server(config, query, rx); + run_server(config, query, rx, response_timer); }), } }