WIP:
This commit is contained in:
parent
071bcc1fcd
commit
4fbef582e3
|
@ -1,3 +0,0 @@
|
||||||
total_time_elapsed_sec,txs_sent,time_to_send_txs,txs_confirmed,txs_un_confirmed,tps
|
|
||||||
8.96211512,20000,6.054988221,14321,5679,1597.9486770975554
|
|
||||||
8.96211512,20000,6.054988221,14321,5679,1597.9486770975554
|
|
|
|
@ -0,0 +1,22 @@
|
||||||
|
CREATE TABLE lite_rpc.Txs (
|
||||||
|
id SERIAL NOT NULL PRIMARY KEY,
|
||||||
|
signature BINARY(64) NOT NULL,
|
||||||
|
recent_slot BIGINT NOT NULL,
|
||||||
|
forwarded_slot BIGINT NOT NULL,
|
||||||
|
processed_slot BIGINT,
|
||||||
|
cu_consumed BIGINT,
|
||||||
|
cu_requested BIGINT,
|
||||||
|
quic_response CHAR
|
||||||
|
);
|
||||||
|
|
||||||
|
|
||||||
|
CREATE TABLE lite_rpc.Blocks (
|
||||||
|
slot BIGINT NOT NULL PRIMARY KEY,
|
||||||
|
leader_id BIGINT NOT NULL,
|
||||||
|
parent_slot BIGINT NOT NULL
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE TABLE lite_rpc.AccountAddrs (
|
||||||
|
id SERIAL PRIMARY KEY,
|
||||||
|
addr VARCHAR(45) NOT NULL
|
||||||
|
);
|
|
@ -0,0 +1,3 @@
|
||||||
|
DROP TABLE lite_rpc.Txs;
|
||||||
|
DROP TABLE lite_rpc.Blocks;
|
||||||
|
DROP TABLE lite_rpc.AccountAddrs;
|
|
@ -1,9 +1,10 @@
|
||||||
use crate::{
|
use crate::{
|
||||||
configs::{IsBlockHashValidConfig, SendTransactionConfig},
|
configs::{IsBlockHashValidConfig, SendTransactionConfig},
|
||||||
encoding::BinaryEncoding,
|
encoding::BinaryEncoding,
|
||||||
|
postgres::Postgres,
|
||||||
rpc::LiteRpcServer,
|
rpc::LiteRpcServer,
|
||||||
tpu_manager::TpuManager,
|
tpu_manager::TpuManager,
|
||||||
workers::{BlockListener, Cleaner, Metrics, MetricsCapture, MetricsPostgres, TxSender},
|
workers::{BlockListener, Cleaner, MetricsCapture, TxSender},
|
||||||
};
|
};
|
||||||
|
|
||||||
use std::{ops::Deref, str::FromStr, sync::Arc, time::Duration};
|
use std::{ops::Deref, str::FromStr, sync::Arc, time::Duration};
|
||||||
|
@ -29,14 +30,12 @@ use solana_transaction_status::TransactionStatus;
|
||||||
use tokio::{net::ToSocketAddrs, task::JoinHandle};
|
use tokio::{net::ToSocketAddrs, task::JoinHandle};
|
||||||
|
|
||||||
/// A bridge between clients and tpu
|
/// A bridge between clients and tpu
|
||||||
#[derive(Clone)]
|
|
||||||
pub struct LiteBridge {
|
pub struct LiteBridge {
|
||||||
pub rpc_client: Arc<RpcClient>,
|
pub rpc_client: Arc<RpcClient>,
|
||||||
pub tpu_manager: Arc<TpuManager>,
|
pub tpu_manager: Arc<TpuManager>,
|
||||||
pub tx_sender: TxSender,
|
pub tx_sender: TxSender,
|
||||||
pub finalized_block_listenser: BlockListener,
|
pub finalized_block_listenser: BlockListener,
|
||||||
pub confirmed_block_listenser: BlockListener,
|
pub confirmed_block_listenser: BlockListener,
|
||||||
pub metrics_capture: MetricsCapture,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl LiteBridge {
|
impl LiteBridge {
|
||||||
|
@ -66,15 +65,12 @@ impl LiteBridge {
|
||||||
)
|
)
|
||||||
.await?;
|
.await?;
|
||||||
|
|
||||||
let metrics_capture = MetricsCapture::new(tx_sender.clone());
|
|
||||||
|
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
rpc_client,
|
rpc_client,
|
||||||
tpu_manager,
|
tpu_manager,
|
||||||
tx_sender,
|
tx_sender,
|
||||||
finalized_block_listenser,
|
finalized_block_listenser,
|
||||||
confirmed_block_listenser,
|
confirmed_block_listenser,
|
||||||
metrics_capture,
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -96,43 +92,23 @@ impl LiteBridge {
|
||||||
clean_interval: Duration,
|
clean_interval: Duration,
|
||||||
postgres_config: &str,
|
postgres_config: &str,
|
||||||
) -> anyhow::Result<[JoinHandle<anyhow::Result<()>>; 8]> {
|
) -> anyhow::Result<[JoinHandle<anyhow::Result<()>>; 8]> {
|
||||||
let finalized_block_listenser = self.finalized_block_listenser.clone().listen();
|
|
||||||
|
|
||||||
let confirmed_block_listenser = self.confirmed_block_listenser.clone().listen();
|
|
||||||
|
|
||||||
let tx_sender = self
|
let tx_sender = self
|
||||||
.tx_sender
|
.tx_sender
|
||||||
.clone()
|
.clone()
|
||||||
.execute(tx_batch_size, tx_send_interval);
|
.execute(tx_batch_size, tx_send_interval);
|
||||||
|
|
||||||
let ws_server_handle = ServerBuilder::default()
|
let metrics_capture = MetricsCapture::new(self.tx_sender.clone());
|
||||||
.ws_only()
|
let (postgres_connection, postgres) = Postgres::new(postgres_config).await?;
|
||||||
.build(ws_addr.clone())
|
|
||||||
.await?
|
|
||||||
.start(self.clone().into_rpc())?;
|
|
||||||
|
|
||||||
let http_server_handle = ServerBuilder::default()
|
let finalized_block_listenser = self
|
||||||
.http_only()
|
.finalized_block_listenser
|
||||||
.build(http_addr.clone())
|
.clone()
|
||||||
.await?
|
.listen(Some(postgres.clone()));
|
||||||
.start(self.clone().into_rpc())?;
|
|
||||||
|
|
||||||
let ws_server = tokio::spawn(async move {
|
let confirmed_block_listenser = self
|
||||||
info!("Websocket Server started at {ws_addr:?}");
|
.confirmed_block_listenser
|
||||||
ws_server_handle.stopped().await;
|
.clone()
|
||||||
bail!("Websocket server stopped");
|
.listen(Some(postgres.clone()));
|
||||||
});
|
|
||||||
|
|
||||||
let http_server = tokio::spawn(async move {
|
|
||||||
info!("HTTP Server started at {http_addr:?}");
|
|
||||||
http_server_handle.stopped().await;
|
|
||||||
bail!("HTTP server stopped");
|
|
||||||
});
|
|
||||||
|
|
||||||
let metrics_capture = self.metrics_capture.clone().capture();
|
|
||||||
let metrics_postgres = MetricsPostgres::new(self.metrics_capture, postgres_config)
|
|
||||||
.await?
|
|
||||||
.sync();
|
|
||||||
|
|
||||||
let cleaner = Cleaner::new(
|
let cleaner = Cleaner::new(
|
||||||
self.tx_sender.clone(),
|
self.tx_sender.clone(),
|
||||||
|
@ -143,14 +119,44 @@ impl LiteBridge {
|
||||||
)
|
)
|
||||||
.start(clean_interval);
|
.start(clean_interval);
|
||||||
|
|
||||||
|
let rpc = self.into_rpc();
|
||||||
|
|
||||||
|
let (ws_server, http_server) = {
|
||||||
|
let ws_server_handle = ServerBuilder::default()
|
||||||
|
.ws_only()
|
||||||
|
.build(ws_addr.clone())
|
||||||
|
.await?
|
||||||
|
.start(rpc.clone())?;
|
||||||
|
|
||||||
|
let http_server_handle = ServerBuilder::default()
|
||||||
|
.http_only()
|
||||||
|
.build(http_addr.clone())
|
||||||
|
.await?
|
||||||
|
.start(rpc)?;
|
||||||
|
|
||||||
|
let ws_server = tokio::spawn(async move {
|
||||||
|
info!("Websocket Server started at {ws_addr:?}");
|
||||||
|
ws_server_handle.stopped().await;
|
||||||
|
bail!("Websocket server stopped");
|
||||||
|
});
|
||||||
|
|
||||||
|
let http_server = tokio::spawn(async move {
|
||||||
|
info!("HTTP Server started at {http_addr:?}");
|
||||||
|
http_server_handle.stopped().await;
|
||||||
|
bail!("HTTP server stopped");
|
||||||
|
});
|
||||||
|
|
||||||
|
(ws_server, http_server)
|
||||||
|
};
|
||||||
|
|
||||||
Ok([
|
Ok([
|
||||||
ws_server,
|
ws_server,
|
||||||
http_server,
|
http_server,
|
||||||
tx_sender,
|
tx_sender,
|
||||||
finalized_block_listenser,
|
finalized_block_listenser,
|
||||||
confirmed_block_listenser,
|
confirmed_block_listenser,
|
||||||
metrics_capture,
|
postgres_connection,
|
||||||
metrics_postgres,
|
metrics_capture.capture(Some(postgres)),
|
||||||
cleaner,
|
cleaner,
|
||||||
])
|
])
|
||||||
}
|
}
|
||||||
|
@ -317,10 +323,6 @@ impl LiteRpcServer for LiteBridge {
|
||||||
Ok(airdrop_sig)
|
Ok(airdrop_sig)
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn get_metrics(&self) -> crate::rpc::Result<Metrics> {
|
|
||||||
return Ok(self.metrics_capture.get_metrics().await);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn signature_subscribe(
|
fn signature_subscribe(
|
||||||
&self,
|
&self,
|
||||||
mut sink: SubscriptionSink,
|
mut sink: SubscriptionSink,
|
||||||
|
|
|
@ -6,6 +6,7 @@ pub mod cli;
|
||||||
pub mod configs;
|
pub mod configs;
|
||||||
pub mod encoding;
|
pub mod encoding;
|
||||||
pub mod errors;
|
pub mod errors;
|
||||||
|
pub mod postgres;
|
||||||
pub mod rpc;
|
pub mod rpc;
|
||||||
pub mod tpu_manager;
|
pub mod tpu_manager;
|
||||||
pub mod workers;
|
pub mod workers;
|
||||||
|
|
|
@ -0,0 +1,112 @@
|
||||||
|
use std::sync::Arc;
|
||||||
|
|
||||||
|
use anyhow::Context;
|
||||||
|
use log::info;
|
||||||
|
use postgres_native_tls::MakeTlsConnector;
|
||||||
|
use tokio::fs;
|
||||||
|
use tokio::task::JoinHandle;
|
||||||
|
use tokio_postgres::Client;
|
||||||
|
|
||||||
|
use native_tls::{Certificate, Identity, TlsConnector};
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct Postgres {
|
||||||
|
client: Arc<Client>,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct PostgresTx<'a> {
|
||||||
|
pub signature: &'a [u8],
|
||||||
|
pub recent_slot: i64,
|
||||||
|
pub forwarded_slot: i64,
|
||||||
|
pub processed_slot: Option<i64>,
|
||||||
|
pub cu_consumed: Option<i64>,
|
||||||
|
pub cu_requested: Option<i64>,
|
||||||
|
pub quic_response: u32,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct PostgresBlock {
|
||||||
|
pub slot: i64,
|
||||||
|
pub leader_id: i64,
|
||||||
|
pub parent_slot: i64,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct PostgreAccountAddr {
|
||||||
|
pub id: u32,
|
||||||
|
pub addr: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Postgres {
|
||||||
|
/// # Return
|
||||||
|
/// (connection join handle, Self)
|
||||||
|
///
|
||||||
|
/// returned join handle is required to be polled
|
||||||
|
pub async fn new(
|
||||||
|
porstgres_config: &str,
|
||||||
|
) -> anyhow::Result<(JoinHandle<anyhow::Result<()>>, Self)> {
|
||||||
|
let connector = TlsConnector::builder()
|
||||||
|
.add_root_certificate(Certificate::from_pem(&fs::read("ca.pem").await?)?)
|
||||||
|
.identity(
|
||||||
|
Identity::from_pkcs12(&fs::read("client.pks").await?, "p").context("Identity")?,
|
||||||
|
)
|
||||||
|
.danger_accept_invalid_hostnames(true)
|
||||||
|
.danger_accept_invalid_certs(true)
|
||||||
|
.build()?;
|
||||||
|
|
||||||
|
info!("making tls config");
|
||||||
|
|
||||||
|
let connector = MakeTlsConnector::new(connector);
|
||||||
|
let (client, connection) = tokio_postgres::connect(porstgres_config, connector).await?;
|
||||||
|
let client = Arc::new(client);
|
||||||
|
|
||||||
|
Ok((
|
||||||
|
tokio::spawn(async move { Ok(connection.await?) }),
|
||||||
|
Self { client },
|
||||||
|
))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn send_block(&self, block: PostgresBlock) -> anyhow::Result<()> {
|
||||||
|
let PostgresBlock {
|
||||||
|
slot,
|
||||||
|
leader_id,
|
||||||
|
parent_slot,
|
||||||
|
} = block;
|
||||||
|
|
||||||
|
self.client
|
||||||
|
.execute(
|
||||||
|
r#"
|
||||||
|
INSERT INTO lite_rpc.Blocks
|
||||||
|
(slot, leader_id, parent_slot)
|
||||||
|
VALUES
|
||||||
|
($1, $2, $3)
|
||||||
|
"#,
|
||||||
|
&[&slot, &leader_id, &parent_slot],
|
||||||
|
)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn send_tx<'a>(&self, tx: PostgresTx<'a>) -> anyhow::Result<()> {
|
||||||
|
let PostgresTx {
|
||||||
|
signature,
|
||||||
|
recent_slot,
|
||||||
|
forwarded_slot,
|
||||||
|
processed_slot,
|
||||||
|
cu_consumed,
|
||||||
|
cu_requested,
|
||||||
|
quic_response,
|
||||||
|
} = tx;
|
||||||
|
|
||||||
|
self.client.execute(
|
||||||
|
r#"
|
||||||
|
INSERT INTO lite_rpc.Txs
|
||||||
|
(signature, recent_slot, forwarded_slot, processed_slot, cu_consumed, cu_requested, quic_response)
|
||||||
|
VALUES
|
||||||
|
($1, $2, $3, $4, $5, $6)
|
||||||
|
"#,
|
||||||
|
&[&signature, &recent_slot, &forwarded_slot, &processed_slot, &cu_consumed, &cu_requested, &quic_response],
|
||||||
|
).await?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
|
@ -6,10 +6,7 @@ use solana_client::rpc_response::{Response as RpcResponse, RpcBlockhash, RpcVers
|
||||||
use solana_sdk::commitment_config::CommitmentConfig;
|
use solana_sdk::commitment_config::CommitmentConfig;
|
||||||
use solana_transaction_status::TransactionStatus;
|
use solana_transaction_status::TransactionStatus;
|
||||||
|
|
||||||
use crate::{
|
use crate::configs::{IsBlockHashValidConfig, SendTransactionConfig};
|
||||||
configs::{IsBlockHashValidConfig, SendTransactionConfig},
|
|
||||||
workers::Metrics,
|
|
||||||
};
|
|
||||||
|
|
||||||
pub type Result<T> = std::result::Result<T, jsonrpsee::core::Error>;
|
pub type Result<T> = std::result::Result<T, jsonrpsee::core::Error>;
|
||||||
|
|
||||||
|
@ -53,9 +50,6 @@ pub trait LiteRpc {
|
||||||
config: Option<RpcRequestAirdropConfig>,
|
config: Option<RpcRequestAirdropConfig>,
|
||||||
) -> Result<String>;
|
) -> Result<String>;
|
||||||
|
|
||||||
#[method(name = "getMetrics")]
|
|
||||||
async fn get_metrics(&self) -> Result<Metrics>;
|
|
||||||
|
|
||||||
#[subscription(name = "signatureSubscribe" => "signatureNotification", unsubscribe="signatureUnsubscribe", item=RpcResponse<serde_json::Value>)]
|
#[subscription(name = "signatureSubscribe" => "signatureNotification", unsubscribe="signatureUnsubscribe", item=RpcResponse<serde_json::Value>)]
|
||||||
fn signature_subscribe(&self, signature: String, commitment_config: CommitmentConfig);
|
fn signature_subscribe(&self, signature: String, commitment_config: CommitmentConfig);
|
||||||
}
|
}
|
||||||
|
|
|
@ -14,11 +14,16 @@ use solana_client::{
|
||||||
use solana_sdk::commitment_config::{CommitmentConfig, CommitmentLevel};
|
use solana_sdk::commitment_config::{CommitmentConfig, CommitmentLevel};
|
||||||
|
|
||||||
use solana_transaction_status::{
|
use solana_transaction_status::{
|
||||||
TransactionConfirmationStatus, TransactionStatus, UiTransactionStatusMeta,
|
TransactionConfirmationStatus, TransactionStatus, UiConfirmedBlock, UiTransactionStatusMeta,
|
||||||
|
};
|
||||||
|
use tokio::{
|
||||||
|
sync::{mpsc::Sender, RwLock},
|
||||||
|
task::JoinHandle,
|
||||||
};
|
};
|
||||||
use tokio::{sync::RwLock, task::JoinHandle};
|
|
||||||
|
|
||||||
use super::TxSender;
|
use crate::postgres::{Postgres, PostgresBlock, PostgresTx};
|
||||||
|
|
||||||
|
use super::{TxProps, TxSender};
|
||||||
|
|
||||||
/// Background worker which listen's to new blocks
|
/// Background worker which listen's to new blocks
|
||||||
/// and keeps a track of confirmed txs
|
/// and keeps a track of confirmed txs
|
||||||
|
@ -37,6 +42,11 @@ struct BlockInformation {
|
||||||
pub block_height: u64,
|
pub block_height: u64,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub struct BlockListnerNotificatons {
|
||||||
|
pub block: Sender<UiConfirmedBlock>,
|
||||||
|
pub tx: Sender<TxProps>,
|
||||||
|
}
|
||||||
|
|
||||||
impl BlockListener {
|
impl BlockListener {
|
||||||
pub async fn new(
|
pub async fn new(
|
||||||
pub_sub_client: Arc<PubsubClient>,
|
pub_sub_client: Arc<PubsubClient>,
|
||||||
|
@ -89,7 +99,7 @@ impl BlockListener {
|
||||||
self.signature_subscribers.remove(&signature);
|
self.signature_subscribers.remove(&signature);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn listen(self) -> JoinHandle<anyhow::Result<()>> {
|
pub fn listen(self, postgres: Option<Postgres>) -> JoinHandle<anyhow::Result<()>> {
|
||||||
tokio::spawn(async move {
|
tokio::spawn(async move {
|
||||||
info!("Subscribing to blocks");
|
info!("Subscribing to blocks");
|
||||||
|
|
||||||
|
@ -142,6 +152,14 @@ impl BlockListener {
|
||||||
block_height,
|
block_height,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if let Some(postgres) = &postgres {
|
||||||
|
postgres.send_block(PostgresBlock {
|
||||||
|
slot: slot as i64,
|
||||||
|
leader_id: 0, //FIX:
|
||||||
|
parent_slot: 0, //FIX:
|
||||||
|
}).await?;
|
||||||
|
}
|
||||||
|
|
||||||
for tx in transactions {
|
for tx in transactions {
|
||||||
let Some(UiTransactionStatusMeta { err, status, .. }) = tx.meta else {
|
let Some(UiTransactionStatusMeta { err, status, .. }) = tx.meta else {
|
||||||
info!("tx with no meta");
|
info!("tx with no meta");
|
||||||
|
@ -156,6 +174,18 @@ impl BlockListener {
|
||||||
let sig = tx.get_signature().to_string();
|
let sig = tx.get_signature().to_string();
|
||||||
|
|
||||||
if let Some(mut tx_status) = self.tx_sender.txs_sent.get_mut(&sig) {
|
if let Some(mut tx_status) = self.tx_sender.txs_sent.get_mut(&sig) {
|
||||||
|
if let Some(postgres) = &postgres {
|
||||||
|
postgres.send_tx(PostgresTx {
|
||||||
|
signature: tx.get_signature().as_ref(),
|
||||||
|
recent_slot: slot as i64,
|
||||||
|
forwarded_slot: 0,
|
||||||
|
processed_slot: None,
|
||||||
|
cu_consumed: None,
|
||||||
|
cu_requested: None,
|
||||||
|
quic_response: 0,
|
||||||
|
}).await?;
|
||||||
|
}
|
||||||
|
|
||||||
tx_status.value_mut().status = Some(TransactionStatus {
|
tx_status.value_mut().status = Some(TransactionStatus {
|
||||||
slot,
|
slot,
|
||||||
confirmations: None, //TODO: talk about this
|
confirmations: None, //TODO: talk about this
|
||||||
|
@ -167,7 +197,6 @@ impl BlockListener {
|
||||||
|
|
||||||
// subscribers
|
// subscribers
|
||||||
if let Some((_sig, mut sink)) = self.signature_subscribers.remove(&sig) {
|
if let Some((_sig, mut sink)) = self.signature_subscribers.remove(&sig) {
|
||||||
// info!("notification {}", sig);
|
|
||||||
// none if transaction succeeded
|
// none if transaction succeeded
|
||||||
sink.send(&RpcResponse {
|
sink.send(&RpcResponse {
|
||||||
context: RpcResponseContext {
|
context: RpcResponseContext {
|
||||||
|
|
|
@ -1,17 +1,16 @@
|
||||||
use std::sync::Arc;
|
|
||||||
|
|
||||||
use log::{info, warn};
|
use log::{info, warn};
|
||||||
use solana_transaction_status::TransactionConfirmationStatus;
|
use solana_transaction_status::TransactionConfirmationStatus;
|
||||||
use tokio::{sync::RwLock, task::JoinHandle};
|
use tokio::{sync::RwLock, task::JoinHandle};
|
||||||
|
|
||||||
|
use crate::postgres::Postgres;
|
||||||
|
|
||||||
use super::TxSender;
|
use super::TxSender;
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
/// Background worker which captures metrics
|
/// Background worker which captures metrics
|
||||||
#[derive(Clone)]
|
|
||||||
pub struct MetricsCapture {
|
pub struct MetricsCapture {
|
||||||
tx_sender: TxSender,
|
tx_sender: TxSender,
|
||||||
metrics: Arc<RwLock<Metrics>>,
|
metrics: RwLock<Metrics>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, Default, Debug, Serialize, Deserialize)]
|
#[derive(Clone, Default, Debug, Serialize, Deserialize)]
|
||||||
|
@ -37,7 +36,7 @@ impl MetricsCapture {
|
||||||
self.metrics.read().await.to_owned()
|
self.metrics.read().await.to_owned()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn capture(self) -> JoinHandle<anyhow::Result<()>> {
|
pub fn capture(self, postgres: Option<Postgres>) -> JoinHandle<anyhow::Result<()>> {
|
||||||
let mut one_second = tokio::time::interval(std::time::Duration::from_secs(1));
|
let mut one_second = tokio::time::interval(std::time::Duration::from_secs(1));
|
||||||
|
|
||||||
tokio::spawn(async move {
|
tokio::spawn(async move {
|
||||||
|
@ -84,6 +83,10 @@ impl MetricsCapture {
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if let Some(_postgres) = &postgres {
|
||||||
|
// postgres.send_metrics(metrics.clone()).await?;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,94 +0,0 @@
|
||||||
use anyhow::{bail, Context};
|
|
||||||
use log::info;
|
|
||||||
use postgres_native_tls::MakeTlsConnector;
|
|
||||||
use tokio::fs;
|
|
||||||
use tokio::task::JoinHandle;
|
|
||||||
use tokio_postgres::Client;
|
|
||||||
|
|
||||||
use native_tls::{Certificate, Identity, TlsConnector};
|
|
||||||
|
|
||||||
use super::{Metrics, MetricsCapture};
|
|
||||||
|
|
||||||
pub struct MetricsPostgres {
|
|
||||||
connection: JoinHandle<Result<(), tokio_postgres::Error>>,
|
|
||||||
client: Client,
|
|
||||||
metrics_capture: MetricsCapture,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl MetricsPostgres {
|
|
||||||
pub async fn new(
|
|
||||||
metrics_capture: MetricsCapture,
|
|
||||||
porstgres_config: &str,
|
|
||||||
) -> anyhow::Result<Self> {
|
|
||||||
let connector = TlsConnector::builder()
|
|
||||||
.add_root_certificate(Certificate::from_pem(&fs::read("ca.pem").await?)?)
|
|
||||||
.identity(
|
|
||||||
Identity::from_pkcs12(&fs::read("client.pks").await?, "p").context("Identity")?,
|
|
||||||
)
|
|
||||||
.danger_accept_invalid_hostnames(true)
|
|
||||||
.danger_accept_invalid_certs(true)
|
|
||||||
.build()?;
|
|
||||||
|
|
||||||
info!("making tls config");
|
|
||||||
|
|
||||||
let connector = MakeTlsConnector::new(connector);
|
|
||||||
let (client, connection) = tokio_postgres::connect(porstgres_config, connector).await?;
|
|
||||||
|
|
||||||
Ok(Self {
|
|
||||||
connection: tokio::spawn(connection),
|
|
||||||
client,
|
|
||||||
metrics_capture,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn sync(self) -> JoinHandle<anyhow::Result<()>> {
|
|
||||||
let mut one_second = tokio::time::interval(std::time::Duration::from_secs(1));
|
|
||||||
let Self {
|
|
||||||
connection,
|
|
||||||
client,
|
|
||||||
metrics_capture,
|
|
||||||
} = self;
|
|
||||||
|
|
||||||
let metrics_send: JoinHandle<anyhow::Result<()>> = tokio::spawn(async move {
|
|
||||||
info!("Sending Metrics To Postgres");
|
|
||||||
|
|
||||||
loop {
|
|
||||||
let Metrics {
|
|
||||||
txs_sent,
|
|
||||||
txs_confirmed,
|
|
||||||
txs_finalized,
|
|
||||||
txs_ps,
|
|
||||||
txs_confirmed_ps,
|
|
||||||
txs_finalized_ps,
|
|
||||||
mem_used,
|
|
||||||
} = metrics_capture.get_metrics().await;
|
|
||||||
|
|
||||||
client.execute(
|
|
||||||
r#"INSERT INTO LiteRpcMetrics
|
|
||||||
(txs_sent, txs_confirmed, txs_finalized, transactions_per_second, confirmations_per_second, finalized_per_second, memory_used)
|
|
||||||
VALUES
|
|
||||||
($1, $2, $3, $4, $5, $6, $7)
|
|
||||||
"#,
|
|
||||||
&[&(txs_sent as i64), &(txs_confirmed as i64), &(txs_finalized as i64), &(txs_ps as i64), &(txs_confirmed_ps as i64), &(txs_finalized_ps as i64), &(mem_used.unwrap_or_default() as i64)],
|
|
||||||
)
|
|
||||||
.await.unwrap();
|
|
||||||
|
|
||||||
one_second.tick().await;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
#[allow(unreachable_code)]
|
|
||||||
tokio::spawn(async move {
|
|
||||||
tokio::select! {
|
|
||||||
r = metrics_send => {
|
|
||||||
bail!("Postgres metrics send thread stopped {r:?}")
|
|
||||||
}
|
|
||||||
r = connection => {
|
|
||||||
bail!("Postgres connection poll stopped {r:?}")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,11 +1,9 @@
|
||||||
mod block_listenser;
|
mod block_listenser;
|
||||||
mod cleaner;
|
mod cleaner;
|
||||||
mod metrics_capture;
|
mod metrics_capture;
|
||||||
mod metrics_postgres;
|
|
||||||
mod tx_sender;
|
mod tx_sender;
|
||||||
|
|
||||||
pub use block_listenser::*;
|
pub use block_listenser::*;
|
||||||
pub use cleaner::*;
|
pub use cleaner::*;
|
||||||
pub use metrics_capture::*;
|
pub use metrics_capture::*;
|
||||||
pub use metrics_postgres::*;
|
|
||||||
pub use tx_sender::*;
|
pub use tx_sender::*;
|
||||||
|
|
|
@ -44,7 +44,7 @@ async fn send_and_confirm_txs() {
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
let services = try_join_all(vec![
|
let services = try_join_all(vec![
|
||||||
block_listener.clone().listen(),
|
block_listener.clone().listen(None),
|
||||||
tx_sender.clone().execute(
|
tx_sender.clone().execute(
|
||||||
DEFAULT_TX_BATCH_SIZE,
|
DEFAULT_TX_BATCH_SIZE,
|
||||||
Duration::from_millis(DEFAULT_TX_BATCH_INTERVAL_MS),
|
Duration::from_millis(DEFAULT_TX_BATCH_INTERVAL_MS),
|
||||||
|
|
Loading…
Reference in New Issue