move allowed_lag around

This commit is contained in:
Bryan Stitt 2023-01-04 12:07:53 -08:00
parent c0fc999e02
commit 045065986a
6 changed files with 60 additions and 40 deletions

@ -54,50 +54,50 @@ response_cache_max_bytes = 10_000_000_000
display_name = "Ankr"
url = "https://rpc.ankr.com/eth"
soft_limit = 1_000
weight = 0
tier = 0
[balanced_rpcs.cloudflare]
display_name = "Cloudflare"
url = "https://cloudflare-eth.com"
soft_limit = 1_000
weight = 10
tier = 1
[balanced_rpcs.blastapi]
display_name = "Blast"
url = "https://eth-mainnet.public.blastapi.io"
soft_limit = 1_000
weight = 10
tier = 1
[balanced_rpcs.mycryptoapi]
display_name = "MyCrypto"
disabled = true
url = "https://api.mycryptoapi.com/eth"
soft_limit = 1_000
weight = 25
tier = 2
[balanced_rpcs.pokt-v1]
display_name = "Pokt #1"
url = "https://eth-mainnet.gateway.pokt.network/v1/5f3453978e354ab992c4da79"
soft_limit = 500
weight = 25
tier = 2
[balanced_rpcs.pokt]
display_name = "Pokt #2"
url = "https://eth-rpc.gateway.pokt.network"
soft_limit = 500
weight = 50
[balanced_rpcs.runonflux]
display_name = "Run on Flux (light)"
url = "https://ethereumnodelight.app.runonflux.io"
soft_limit = 1_000
weight = 75
tier = 3
[balanced_rpcs.linkpool]
display_name = "Linkpool"
url = "https://main-rpc.linkpool.io"
soft_limit = 500
weight = 75
tier = 4
[balanced_rpcs.runonflux]
display_name = "Run on Flux (light)"
url = "https://ethereumnodelight.app.runonflux.io"
soft_limit = 1_000
tier = 5
# load balanced light nodes are not very reliable
[balanced_rpcs.linkpool-light]
@ -105,7 +105,7 @@ response_cache_max_bytes = 10_000_000_000
disabled = true
url = "https://main-light.eth.linkpool.io"
soft_limit = 100
weight = 75
tier = 5
[private_rpcs]
@ -116,32 +116,32 @@ response_cache_max_bytes = 10_000_000_000
display_name = "Eden network"
url = "https://api.edennetwork.io/v1/"
soft_limit = 1_805
weight = 0
tier = 0
[private_rpcs.eden_beta]
disabled = true
display_name = "Eden network beta"
url = "https://api.edennetwork.io/v1/beta"
soft_limit = 5_861
weight = 0
tier = 0
[private_rpcs.ethermine]
disabled = true
display_name = "Ethermine"
url = "https://rpc.ethermine.org"
soft_limit = 5_861
weight = 0
tier = 0
[private_rpcs.flashbots]
disabled = true
display_name = "Flashbots Fast"
url = "https://rpc.flashbots.net/fast"
soft_limit = 7_074
weight = 0
tier = 0
[private_rpcs.securerpc]
disabled = true
display_name = "SecureRPC"
url = "https://gibson.securerpc.com/v1"
soft_limit = 4_560
weight = 0
tier = 0

@ -221,6 +221,7 @@ impl Web3ConnectionConfig {
pub async fn spawn(
self,
name: String,
allowed_lag: u64,
db_conn: Option<DatabaseConnection>,
redis_pool: Option<redis_rate_limiter::RedisPool>,
chain_id: u64,
@ -257,6 +258,7 @@ impl Web3ConnectionConfig {
Web3Connection::spawn(
name,
allowed_lag,
self.display_name,
chain_id,
db_conn,

@ -10,6 +10,7 @@ use axum::{response::IntoResponse, Extension, Json};
use axum_client_ip::ClientIp;
use axum_macros::debug_handler;
use itertools::Itertools;
use log::debug;
use std::sync::Arc;
/// POST /rpc -- Public entrypoint for HTTP JSON-RPC requests. Web3 wallets use this.
@ -90,7 +91,7 @@ pub async fn proxy_web3_rpc_with_key(
let headers = response.headers_mut();
// TODO: special string if no rpcs were used (cache hit)?
// TODO: special string if no rpcs were used (cache hit)? or is an empty string fine? maybe the rpc name + "cached"
let rpcs: String = rpcs.into_iter().map(|x| x.name.clone()).join(",");
headers.insert(

@ -29,7 +29,7 @@ pub type BlockHashesCache = Cache<H256, ArcBlock, hashbrown::hash_map::DefaultHa
pub struct SavedBlock {
pub block: ArcBlock,
/// number of seconds this block was behind the current time when received
pub lag: u64,
pub age: u64,
}
impl PartialEq for SavedBlock {
@ -45,11 +45,11 @@ impl PartialEq for SavedBlock {
impl SavedBlock {
pub fn new(block: ArcBlock) -> Self {
let mut x = Self { block, lag: 0 };
let mut x = Self { block, age: 0 };
// no need to recalulate lag every time
// if the head block gets too old, a health check restarts this connection
x.lag = x.lag();
x.age = x.lag();
x
}
@ -81,7 +81,7 @@ impl SavedBlock {
/// When the block was received, this node was still syncing
pub fn syncing(&self, allowed_lag: u64) -> bool {
self.lag > allowed_lag
self.age > allowed_lag
}
}
@ -93,13 +93,7 @@ impl From<ArcBlock> for SavedBlock {
impl Display for SavedBlock {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "{} ({})", self.number(), self.hash())?;
if self.syncing(0) {
write!(f, " (behind by {} seconds)", self.lag)?;
}
Ok(())
write!(f, "{} ({}, {}s old)", self.number(), self.hash(), self.age)
}
}
@ -320,7 +314,7 @@ impl Web3Connections {
// TODO: don't default to 60. different chains are differen
if rpc_head_block.syncing(60) {
if connection_heads.remove(&rpc.name).is_some() {
warn!("{} is behind by {} seconds", &rpc.name, rpc_head_block.lag);
warn!("{} is behind by {} seconds", &rpc.name, rpc_head_block.age);
} else {
// we didn't remove anything and this block is old. exit early
return Ok(());

@ -63,6 +63,7 @@ pub struct Web3Connection {
pub name: String,
pub display_name: Option<String>,
pub db_conn: Option<DatabaseConnection>,
pub(super) allowed_lag: u64,
/// TODO: can we get this from the provider? do we even need it?
pub(super) url: String,
/// Some connections use an http_client. we keep a clone for reconnecting
@ -98,6 +99,7 @@ impl Web3Connection {
#[allow(clippy::too_many_arguments)]
pub async fn spawn(
name: String,
allowed_lag: u64,
display_name: Option<String>,
chain_id: u64,
db_conn: Option<DatabaseConnection>,
@ -135,6 +137,7 @@ impl Web3Connection {
let new_connection = Self {
name,
allowed_lag,
db_conn: db_conn.clone(),
display_name,
http_client,
@ -587,6 +590,8 @@ impl Web3Connection {
reconnect: bool,
tx_id_sender: Option<flume::Sender<(TxHash, Arc<Self>)>>,
) -> anyhow::Result<()> {
let allowed_lag = self.allowed_lag;
loop {
let http_interval_receiver = http_interval_sender.as_ref().map(|x| x.subscribe());
@ -641,10 +646,10 @@ impl Web3Connection {
if let Some(x) = &*conn.head_block.read() {
// if this block is too old, return an error so we reconnect
let current_lag = x.lag();
if current_lag > 0 {
if current_lag > allowed_lag {
let level = if warned == 0 {
log::Level::Warn
} else if current_lag % 1000 == 0 {
} else if warned % 100 == 0 {
log::Level::Debug
} else {
log::Level::Trace
@ -1208,6 +1213,7 @@ mod tests {
let x = Web3Connection {
name: "name".to_string(),
allowed_lag: 10,
db_conn: None,
display_name: None,
url: "ws://example.com".to_string(),
@ -1255,6 +1261,7 @@ mod tests {
// TODO: this is getting long. have a `impl Default`
let x = Web3Connection {
name: "name".to_string(),
allowed_lag: 10,
db_conn: None,
display_name: None,
url: "ws://example.com".to_string(),
@ -1306,6 +1313,7 @@ mod tests {
let x = Web3Connection {
name: "name".to_string(),
allowed_lag: 10,
db_conn: None,
display_name: None,
url: "ws://example.com".to_string(),

@ -88,6 +88,9 @@ impl Web3Connections {
}
};
// TODO: this might be too aggressive. think about this more
let allowed_lag = ((expected_block_time_ms * 3) as f64 / 1000.0).round() as u64;
let http_interval_sender = if http_client.is_some() {
let (sender, receiver) = broadcast::channel(1);
@ -151,6 +154,7 @@ impl Web3Connections {
server_config
.spawn(
server_name,
allowed_lag,
db_conn,
redis_pool,
chain_id,
@ -428,7 +432,7 @@ impl Web3Connections {
match x_head_block {
None => continue,
Some(x_head) => {
let key = (x_head.number(), u64::MAX - x.tier);
let key = (Some(x_head.number()), u64::MAX - x.tier);
m.entry(key).or_insert_with(Vec::new).push(x);
}
@ -450,8 +454,6 @@ impl Web3Connections {
return Ok(OpenRequestResult::NotReady);
}
let head_num = head_block.number();
let mut m = BTreeMap::new();
for x in synced_connections
@ -459,7 +461,7 @@ impl Web3Connections {
.iter()
.filter(|x| !skip.contains(x))
{
let key = (head_num, u64::MAX - x.tier);
let key = (None, u64::MAX - x.tier);
m.entry(key).or_insert_with(Vec::new).push(x.clone());
}
@ -472,6 +474,7 @@ impl Web3Connections {
for usable_rpcs in usable_rpcs_by_head_num_and_weight.into_values().rev() {
// under heavy load, it is possible for even our best server to be negative
let mut minimum = f64::MAX;
let mut maximum = f64::MIN;
// we sort on a combination of values. cache them here so that we don't do this math multiple times.
let mut available_request_map: HashMap<_, f64> = usable_rpcs
@ -487,13 +490,20 @@ impl Web3Connections {
trace!("available requests on {}: {}", rpc, available_requests);
minimum = available_requests.min(minimum);
minimum = minimum.min(available_requests);
maximum = maximum.max(available_requests);
(rpc, available_requests)
})
.collect();
trace!("minimum available requests: {}", minimum);
trace!("maximum available requests: {}", minimum);
if maximum < 0.0 {
// TODO: if maximum < 0 and there are other tiers on the same block, we should include them now
warn!("soft limits overloaded: {} to {}", minimum, maximum)
}
// choose_multiple_weighted can't have negative numbers. shift up if any are negative
// TODO: is this a correct way to shift?
@ -513,7 +523,7 @@ impl Web3Connections {
let sorted_rpcs = {
if usable_rpcs.len() == 1 {
// TODO: return now instead?
// TODO: return now instead? we shouldn't need another alloc
vec![usable_rpcs.get(0).expect("there should be 1")]
} else {
let mut rng = thread_fast_rng::thread_fast_rng();
@ -935,6 +945,8 @@ mod tests {
let head_rpc = Web3Connection {
name: "synced".to_string(),
// TODO: what should this be?
allowed_lag: 10,
db_conn: None,
display_name: None,
url: "ws://example.com/synced".to_string(),
@ -954,6 +966,7 @@ mod tests {
let lagged_rpc = Web3Connection {
name: "lagged".to_string(),
allowed_lag: 10,
db_conn: None,
display_name: None,
url: "ws://example.com/lagged".to_string(),
@ -1157,6 +1170,7 @@ mod tests {
let pruned_rpc = Web3Connection {
name: "pruned".to_string(),
allowed_lag: 10,
db_conn: None,
display_name: None,
url: "ws://example.com/pruned".to_string(),
@ -1176,6 +1190,7 @@ mod tests {
let archive_rpc = Web3Connection {
name: "archive".to_string(),
allowed_lag: 10,
db_conn: None,
display_name: None,
url: "ws://example.com/archive".to_string(),