mirror of
https://github.com/uqbar-dao/nectar.git
synced 2025-01-01 13:05:00 +03:00
refactor utils into utils.rs
This commit is contained in:
parent
8b60535b1a
commit
16e9e6898b
@ -1,6 +1,5 @@
|
||||
use alloy::providers::{Provider, ProviderBuilder, RootProvider};
|
||||
use alloy::providers::{Provider, RootProvider};
|
||||
use alloy::pubsub::PubSubFrontend;
|
||||
use alloy::rpc::client::WsConnect;
|
||||
use alloy::rpc::json_rpc::RpcError;
|
||||
use anyhow::Result;
|
||||
use dashmap::DashMap;
|
||||
@ -14,9 +13,10 @@ use std::sync::Arc;
|
||||
use std::time::{Duration, Instant};
|
||||
use tokio::sync::Mutex;
|
||||
use tokio::task::JoinHandle;
|
||||
use url::Url;
|
||||
use utils::*;
|
||||
|
||||
mod subscription;
|
||||
mod utils;
|
||||
|
||||
/// meta-type for all incoming requests we need to handle
|
||||
#[derive(Debug, Serialize, Deserialize)]
|
||||
@ -1073,193 +1073,3 @@ async fn handle_eth_config_action(
|
||||
}
|
||||
EthConfigResponse::Ok
|
||||
}
|
||||
|
||||
async fn activate_url_provider(provider: &mut UrlProvider) -> Result<()> {
|
||||
match Url::parse(&provider.url)?.scheme() {
|
||||
"ws" | "wss" => {
|
||||
let ws = WsConnect::new(provider.url.to_string());
|
||||
|
||||
let client = tokio::time::timeout(
|
||||
std::time::Duration::from_secs(10),
|
||||
ProviderBuilder::new().on_ws(ws),
|
||||
)
|
||||
.await??;
|
||||
provider.pubsub = Some(client);
|
||||
Ok(())
|
||||
}
|
||||
_ => Err(anyhow::anyhow!(
|
||||
"Only `ws://` or `wss://` providers are supported."
|
||||
)),
|
||||
}
|
||||
}
|
||||
|
||||
fn providers_to_saved_configs(providers: &Providers) -> SavedConfigs {
|
||||
providers
|
||||
.iter()
|
||||
.map(|entry| {
|
||||
entry
|
||||
.urls
|
||||
.iter()
|
||||
.map(|url_provider| ProviderConfig {
|
||||
chain_id: *entry.key(),
|
||||
provider: NodeOrRpcUrl::RpcUrl(url_provider.url.clone()),
|
||||
trusted: url_provider.trusted,
|
||||
})
|
||||
.chain(entry.nodes.iter().map(|node_provider| ProviderConfig {
|
||||
chain_id: *entry.key(),
|
||||
provider: NodeOrRpcUrl::Node {
|
||||
kns_update: node_provider.kns_update.clone(),
|
||||
use_as_provider: node_provider.usable,
|
||||
},
|
||||
trusted: node_provider.trusted,
|
||||
}))
|
||||
.collect::<Vec<_>>()
|
||||
})
|
||||
.flatten()
|
||||
.collect()
|
||||
}
|
||||
|
||||
async fn check_for_root_cap(
|
||||
our: &str,
|
||||
process: &ProcessId,
|
||||
caps_oracle: &CapMessageSender,
|
||||
) -> bool {
|
||||
let (send_cap_bool, recv_cap_bool) = tokio::sync::oneshot::channel();
|
||||
caps_oracle
|
||||
.send(CapMessage::Has {
|
||||
on: process.clone(),
|
||||
cap: Capability::new((our, ETH_PROCESS_ID.clone()), "{\"root\":true}"),
|
||||
responder: send_cap_bool,
|
||||
})
|
||||
.await
|
||||
.expect("eth: capability oracle died!");
|
||||
recv_cap_bool.await.unwrap_or(false)
|
||||
}
|
||||
|
||||
async fn verbose_print(print_tx: &PrintSender, content: &str) {
|
||||
let _ = print_tx
|
||||
.send(Printout::new(
|
||||
2,
|
||||
NET_PROCESS_ID.clone(),
|
||||
content.to_string(),
|
||||
))
|
||||
.await;
|
||||
}
|
||||
|
||||
async fn error_message(
|
||||
our: &str,
|
||||
km_id: u64,
|
||||
target: Address,
|
||||
error: EthError,
|
||||
send_to_loop: &MessageSender,
|
||||
) {
|
||||
kernel_message(
|
||||
our,
|
||||
km_id,
|
||||
target,
|
||||
None,
|
||||
false,
|
||||
None,
|
||||
EthResponse::Err(error),
|
||||
send_to_loop,
|
||||
)
|
||||
.await
|
||||
}
|
||||
|
||||
async fn kernel_message<T: Serialize>(
|
||||
our: &str,
|
||||
km_id: u64,
|
||||
target: Address,
|
||||
rsvp: Option<Address>,
|
||||
req: bool,
|
||||
timeout: Option<u64>,
|
||||
body: T,
|
||||
send_to_loop: &MessageSender,
|
||||
) {
|
||||
let Err(e) = send_to_loop.try_send(KernelMessage {
|
||||
id: km_id,
|
||||
source: Address {
|
||||
node: our.to_string(),
|
||||
process: ETH_PROCESS_ID.clone(),
|
||||
},
|
||||
target,
|
||||
rsvp,
|
||||
message: if req {
|
||||
Message::Request(Request {
|
||||
inherit: false,
|
||||
expects_response: timeout,
|
||||
body: serde_json::to_vec(&body).unwrap(),
|
||||
metadata: None,
|
||||
capabilities: vec![],
|
||||
})
|
||||
} else {
|
||||
Message::Response((
|
||||
Response {
|
||||
inherit: false,
|
||||
body: serde_json::to_vec(&body).unwrap(),
|
||||
metadata: None,
|
||||
capabilities: vec![],
|
||||
},
|
||||
None,
|
||||
))
|
||||
},
|
||||
lazy_load_blob: None,
|
||||
}) else {
|
||||
// not Err -> send successful; done here
|
||||
return;
|
||||
};
|
||||
// its an Err: handle
|
||||
match e {
|
||||
tokio::sync::mpsc::error::TrySendError::Closed(_) => {
|
||||
return;
|
||||
}
|
||||
tokio::sync::mpsc::error::TrySendError::Full(_) => {
|
||||
// TODO: implement backpressure
|
||||
panic!("(eth) kernel overloaded with messages: TODO: implement backpressure");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fn find_index(vec: &Vec<&str>, item: &str) -> Option<usize> {
|
||||
vec.iter().enumerate().find_map(
|
||||
|(index, value)| {
|
||||
if *value == item {
|
||||
Some(index)
|
||||
} else {
|
||||
None
|
||||
}
|
||||
},
|
||||
)
|
||||
}
|
||||
|
||||
async fn set_node_unusable(
|
||||
providers: &Providers,
|
||||
chain_id: &u64,
|
||||
node_name: &str,
|
||||
print_tx: &PrintSender,
|
||||
) -> bool {
|
||||
let mut is_replacement_successful = true;
|
||||
providers.entry(chain_id.clone()).and_modify(|aps| {
|
||||
let Some(index) = find_index(
|
||||
&aps.nodes
|
||||
.iter()
|
||||
.map(|n| n.kns_update.name.as_str())
|
||||
.collect(),
|
||||
&node_name,
|
||||
) else {
|
||||
is_replacement_successful = false;
|
||||
return ();
|
||||
};
|
||||
let mut node = aps.nodes.remove(index);
|
||||
node.usable = false;
|
||||
aps.nodes.insert(index, node);
|
||||
});
|
||||
if !is_replacement_successful {
|
||||
verbose_print(
|
||||
print_tx,
|
||||
&format!("eth: unexpectedly couldn't find provider to be modified"),
|
||||
)
|
||||
.await;
|
||||
}
|
||||
is_replacement_successful
|
||||
}
|
||||
|
198
kinode/src/eth/utils.rs
Normal file
198
kinode/src/eth/utils.rs
Normal file
@ -0,0 +1,198 @@
|
||||
use crate::eth::{Providers, UrlProvider};
|
||||
use alloy::providers::ProviderBuilder;
|
||||
use alloy::rpc::client::WsConnect;
|
||||
use anyhow::Result;
|
||||
use lib::types::core::*;
|
||||
use lib::types::eth::*;
|
||||
use serde::Serialize;
|
||||
use url::Url;
|
||||
|
||||
pub async fn activate_url_provider(provider: &mut UrlProvider) -> Result<()> {
|
||||
match Url::parse(&provider.url)?.scheme() {
|
||||
"ws" | "wss" => {
|
||||
let ws = WsConnect::new(provider.url.to_string());
|
||||
|
||||
let client = tokio::time::timeout(
|
||||
std::time::Duration::from_secs(10),
|
||||
ProviderBuilder::new().on_ws(ws),
|
||||
)
|
||||
.await??;
|
||||
provider.pubsub = Some(client);
|
||||
Ok(())
|
||||
}
|
||||
_ => Err(anyhow::anyhow!(
|
||||
"Only `ws://` or `wss://` providers are supported."
|
||||
)),
|
||||
}
|
||||
}
|
||||
|
||||
pub fn providers_to_saved_configs(providers: &Providers) -> SavedConfigs {
|
||||
providers
|
||||
.iter()
|
||||
.map(|entry| {
|
||||
entry
|
||||
.urls
|
||||
.iter()
|
||||
.map(|url_provider| ProviderConfig {
|
||||
chain_id: *entry.key(),
|
||||
provider: NodeOrRpcUrl::RpcUrl(url_provider.url.clone()),
|
||||
trusted: url_provider.trusted,
|
||||
})
|
||||
.chain(entry.nodes.iter().map(|node_provider| ProviderConfig {
|
||||
chain_id: *entry.key(),
|
||||
provider: NodeOrRpcUrl::Node {
|
||||
kns_update: node_provider.kns_update.clone(),
|
||||
use_as_provider: node_provider.usable,
|
||||
},
|
||||
trusted: node_provider.trusted,
|
||||
}))
|
||||
.collect::<Vec<_>>()
|
||||
})
|
||||
.flatten()
|
||||
.collect()
|
||||
}
|
||||
|
||||
pub async fn check_for_root_cap(
|
||||
our: &str,
|
||||
process: &ProcessId,
|
||||
caps_oracle: &CapMessageSender,
|
||||
) -> bool {
|
||||
let (send_cap_bool, recv_cap_bool) = tokio::sync::oneshot::channel();
|
||||
caps_oracle
|
||||
.send(CapMessage::Has {
|
||||
on: process.clone(),
|
||||
cap: Capability::new((our, ETH_PROCESS_ID.clone()), "{\"root\":true}"),
|
||||
responder: send_cap_bool,
|
||||
})
|
||||
.await
|
||||
.expect("eth: capability oracle died!");
|
||||
recv_cap_bool.await.unwrap_or(false)
|
||||
}
|
||||
|
||||
pub async fn verbose_print(print_tx: &PrintSender, content: &str) {
|
||||
let _ = print_tx
|
||||
.send(Printout::new(
|
||||
2,
|
||||
NET_PROCESS_ID.clone(),
|
||||
content.to_string(),
|
||||
))
|
||||
.await;
|
||||
}
|
||||
|
||||
pub async fn error_message(
|
||||
our: &str,
|
||||
km_id: u64,
|
||||
target: Address,
|
||||
error: EthError,
|
||||
send_to_loop: &MessageSender,
|
||||
) {
|
||||
kernel_message(
|
||||
our,
|
||||
km_id,
|
||||
target,
|
||||
None,
|
||||
false,
|
||||
None,
|
||||
EthResponse::Err(error),
|
||||
send_to_loop,
|
||||
)
|
||||
.await
|
||||
}
|
||||
|
||||
pub async fn kernel_message<T: Serialize>(
|
||||
our: &str,
|
||||
km_id: u64,
|
||||
target: Address,
|
||||
rsvp: Option<Address>,
|
||||
req: bool,
|
||||
timeout: Option<u64>,
|
||||
body: T,
|
||||
send_to_loop: &MessageSender,
|
||||
) {
|
||||
let Err(e) = send_to_loop.try_send(KernelMessage {
|
||||
id: km_id,
|
||||
source: Address {
|
||||
node: our.to_string(),
|
||||
process: ETH_PROCESS_ID.clone(),
|
||||
},
|
||||
target,
|
||||
rsvp,
|
||||
message: if req {
|
||||
Message::Request(Request {
|
||||
inherit: false,
|
||||
expects_response: timeout,
|
||||
body: serde_json::to_vec(&body).unwrap(),
|
||||
metadata: None,
|
||||
capabilities: vec![],
|
||||
})
|
||||
} else {
|
||||
Message::Response((
|
||||
Response {
|
||||
inherit: false,
|
||||
body: serde_json::to_vec(&body).unwrap(),
|
||||
metadata: None,
|
||||
capabilities: vec![],
|
||||
},
|
||||
None,
|
||||
))
|
||||
},
|
||||
lazy_load_blob: None,
|
||||
}) else {
|
||||
// not Err -> send successful; done here
|
||||
return;
|
||||
};
|
||||
// its an Err: handle
|
||||
match e {
|
||||
tokio::sync::mpsc::error::TrySendError::Closed(_) => {
|
||||
return;
|
||||
}
|
||||
tokio::sync::mpsc::error::TrySendError::Full(_) => {
|
||||
// TODO: implement backpressure
|
||||
panic!("(eth) kernel overloaded with messages: TODO: implement backpressure");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
pub fn find_index(vec: &Vec<&str>, item: &str) -> Option<usize> {
|
||||
vec.iter().enumerate().find_map(
|
||||
|(index, value)| {
|
||||
if *value == item {
|
||||
Some(index)
|
||||
} else {
|
||||
None
|
||||
}
|
||||
},
|
||||
)
|
||||
}
|
||||
|
||||
pub async fn set_node_unusable(
|
||||
providers: &Providers,
|
||||
chain_id: &u64,
|
||||
node_name: &str,
|
||||
print_tx: &PrintSender,
|
||||
) -> bool {
|
||||
let mut is_replacement_successful = true;
|
||||
providers.entry(chain_id.clone()).and_modify(|aps| {
|
||||
let Some(index) = find_index(
|
||||
&aps.nodes
|
||||
.iter()
|
||||
.map(|n| n.kns_update.name.as_str())
|
||||
.collect(),
|
||||
&node_name,
|
||||
) else {
|
||||
is_replacement_successful = false;
|
||||
return ();
|
||||
};
|
||||
let mut node = aps.nodes.remove(index);
|
||||
node.usable = false;
|
||||
aps.nodes.insert(index, node);
|
||||
});
|
||||
if !is_replacement_successful {
|
||||
verbose_print(
|
||||
print_tx,
|
||||
&format!("eth: unexpectedly couldn't find provider to be modified"),
|
||||
)
|
||||
.await;
|
||||
}
|
||||
is_replacement_successful
|
||||
}
|
Loading…
Reference in New Issue
Block a user