Allow multiple grpc sources

This commit is contained in:
Christian Kamm 2021-11-08 15:48:50 +01:00
parent f43aaf0f55
commit 029cbd2fb5
5 changed files with 35 additions and 28 deletions

View File

@ -49,10 +49,10 @@ Solana ---------------> Connector -----------> PostgreSQL
``` ```
For reliability it is recommended to feed data from multiple Solana nodes into For reliability it is recommended to feed data from multiple Solana nodes into
the Connector service. (That's not yet fully supported) the Connector service.
It is also allowed to run multiple Connector services that target the same It is also allowed to run multiple Connector services that target the same
PostgeSQL target database. (That's not yet fully supported) PostgeSQL target database. (but has not been tested)
The Connector service is stateless (except for some caches). Restarting it is The Connector service is stateless (except for some caches). Restarting it is
always safe. always safe.

View File

@ -1,6 +1,6 @@
rpc_ws_url = "" rpc_ws_url = ""
[grpc_source] [[grpc_sources]]
connection_string = "http://[::1]:10000" connection_string = "http://[::1]:10000"
retry_connection_sleep_secs = 30 retry_connection_sleep_secs = 30

View File

@ -1,6 +1,6 @@
rpc_ws_url = "" rpc_ws_url = ""
[grpc_source] [[grpc_sources]]
connection_string = "http://[::1]:10000" connection_string = "http://[::1]:10000"
retry_connection_sleep_secs = 30 retry_connection_sleep_secs = 30

View File

@ -18,7 +18,10 @@ pub mod accountsdb_proto {
} }
use accountsdb_proto::accounts_db_client::AccountsDbClient; use accountsdb_proto::accounts_db_client::AccountsDbClient;
use crate::{AccountWrite, AnyhowWrap, Config, SlotStatus, SlotUpdate}; use crate::{
AccountWrite, AnyhowWrap, Config, GrpcSourceConfig, SlotStatus, SlotUpdate,
SnapshotSourceConfig,
};
type SnapshotData = Response<Vec<RpcKeyedAccount>>; type SnapshotData = Response<Vec<RpcKeyedAccount>>;
@ -71,14 +74,14 @@ async fn get_snapshot(
} }
async fn feed_data_accountsdb( async fn feed_data_accountsdb(
config: &Config, grpc_config: &GrpcSourceConfig,
snapshot_config: &SnapshotSourceConfig,
sender: async_channel::Sender<Message>, sender: async_channel::Sender<Message>,
) -> Result<(), anyhow::Error> { ) -> Result<(), anyhow::Error> {
let program_id = Pubkey::from_str(&config.snapshot_source.program_id)?; let program_id = Pubkey::from_str(&snapshot_config.program_id)?;
let mut client = let mut client =
AccountsDbClient::connect(Endpoint::from_str(&config.grpc_source.connection_string)?) AccountsDbClient::connect(Endpoint::from_str(&grpc_config.connection_string)?).await?;
.await?;
let mut update_stream = client let mut update_stream = client
.subscribe(accountsdb_proto::SubscribeRequest {}) .subscribe(accountsdb_proto::SubscribeRequest {})
@ -113,7 +116,7 @@ async fn feed_data_accountsdb(
if trigger_snapshot_slot_counter > 1 { if trigger_snapshot_slot_counter > 1 {
trigger_snapshot_slot_counter -= 1; trigger_snapshot_slot_counter -= 1;
} else if trigger_snapshot_slot_counter == 1 { } else if trigger_snapshot_slot_counter == 1 {
snapshot_future = tokio::spawn(get_snapshot(config.snapshot_source.rpc_http_url.clone(), program_id, slot_update.slot - trigger_snapshot_after_slots + 1)).fuse(); snapshot_future = tokio::spawn(get_snapshot(snapshot_config.rpc_http_url.clone(), program_id, slot_update.slot - trigger_snapshot_after_slots + 1)).fuse();
trigger_snapshot_slot_counter = 0; trigger_snapshot_slot_counter = 0;
} }
} }
@ -145,24 +148,28 @@ pub async fn process_events(
) { ) {
// Subscribe to accountsdb // Subscribe to accountsdb
let (msg_sender, msg_receiver) = async_channel::unbounded::<Message>(); let (msg_sender, msg_receiver) = async_channel::unbounded::<Message>();
tokio::spawn(async move { for grpc_source in config.grpc_sources {
// Continuously reconnect on failure let msg_sender = msg_sender.clone();
loop { let snapshot_source = config.snapshot_source.clone();
let out = feed_data_accountsdb(&config, msg_sender.clone()); tokio::spawn(async move {
let result = out.await; // Continuously reconnect on failure
assert!(result.is_err()); loop {
if let Err(err) = result { let out = feed_data_accountsdb(&grpc_source, &snapshot_source, msg_sender.clone());
warn!( let result = out.await;
"error during communication with the accountsdb plugin. retrying. {:?}", assert!(result.is_err());
err if let Err(err) = result {
); warn!(
"error during communication with the accountsdb plugin. retrying. {:?}",
err
);
}
tokio::time::sleep(std::time::Duration::from_secs(
grpc_source.retry_connection_sleep_secs,
))
.await;
} }
tokio::time::sleep(std::time::Duration::from_secs( });
config.grpc_source.retry_connection_sleep_secs, }
))
.await;
}
});
let mut latest_write = HashMap::<Vec<u8>, (u64, u64)>::new(); let mut latest_write = HashMap::<Vec<u8>, (u64, u64)>::new();

View File

@ -93,7 +93,7 @@ pub struct SnapshotSourceConfig {
#[derive(Clone, Debug, Deserialize)] #[derive(Clone, Debug, Deserialize)]
pub struct Config { pub struct Config {
pub postgres_target: PostgresConfig, pub postgres_target: PostgresConfig,
pub grpc_source: GrpcSourceConfig, pub grpc_sources: Vec<GrpcSourceConfig>,
pub snapshot_source: SnapshotSourceConfig, pub snapshot_source: SnapshotSourceConfig,
pub rpc_ws_url: String, pub rpc_ws_url: String,
} }